Your IP : 13.59.19.39


Current Path : /var/www/www-root/data/www/www.monolith-realty.ru/bitrix/components/bitrix/main.post.list/
Upload File :
Current File : /var/www/www-root/data/www/www.monolith-realty.ru/bitrix/components/bitrix/main.post.list/class.php

<?php

if (!defined('B_PROLOG_INCLUDED') || B_PROLOG_INCLUDED !== true)
{
	die();
}

use Bitrix\Disk\Uf\Integration\DiskUploaderController;
use Bitrix\Main\Localization\Loc;
use Bitrix\Main\Security\Sign\Signer;
use Bitrix\Main\Web\Json;
use Bitrix\Main\Loader;
use Bitrix\Main\ModuleManager;
use Bitrix\Main\Web\Uri;

Loc::loadMessages(__FILE__);

final class MainPostList extends CBitrixComponent
{
	const STATUS_SCOPE_MOBILE = 'mobile';
	const STATUS_SCOPE_WEB = 'web';

	private $scope = null;
	private $sign;
	static $users = array();
	public $exemplarId;

	public function __construct($component = null)
	{
		parent::__construct($component);

		if(is_object($component) && ($component instanceof cbitrixcomponent) && method_exists($component, "isweb"))
		{
			$this->scope = $component->isWeb() ? self::STATUS_SCOPE_WEB : self::STATUS_SCOPE_MOBILE;
		}

		$this->sign = (new Signer());
		if ($this->request->get('EXEMPLAR_ID'))
		{
			$this->exemplarId = preg_replace('/[^a-z0-9_\-]/i', '', $this->request->get('EXEMPLAR_ID'));
		}
		else if (
			$this->request->isPost() ||
			$this->request->get("sessid") !== null ||
			$this->request->get("logajax") !== null/*socialnetwork*/
		)
			$this->exemplarId = implode("_", [$this->getUserId(), randString(6)]);
		else
			$this->exemplarId = implode("_", [$this->getUserId(), $this->randString(6)]);
	}

	protected function isWeb()
	{
		return ($this->scope == self::STATUS_SCOPE_WEB);
	}

	protected function isAjax()
	{
		return (
			$this->request->getQuery("AJAX_POST") == "Y" && $this->request->getQuery("ENTITY_XML_ID") == $this->arParams["ENTITY_XML_ID"] ||
			$this->request->getPost("AJAX_POST") == "Y" && $this->request->getPost("ENTITY_XML_ID") == $this->arParams["ENTITY_XML_ID"]
		);
	}

	protected function getMode()
	{
		$viewMode = "plain";
		if ($this->isAjax())
		{
			$viewMode = mb_strtoupper($this->request->getPost("MODE")?: $this->request->getQuery("MODE"));
		}
		return $viewMode;
	}

	protected function joinToPull()
	{
		$text = "";
		if ($this->getUser() && $this->getUser()->isAuthorized()
			&& Loader::includeModule("pull")
			&& \CPullOptions::GetNginxStatus()
		)
		{
			if ($this->isWeb())
			{
				if ($this->arParams["RIGHTS"]["MODERATE"] == "Y" || $this->arParams["RIGHTS"]["MODERATE"] == "ALL")
				{
					\CPullWatch::Add($this->getUserId(), 'UNICOMMENTSEXTENDED'.$this->arParams["ENTITY_XML_ID"]);
					$text = <<<HTML
						<script>
							BX.ready(function(){if (BX.PULL) { BX.PULL.extendWatch("UNICOMMENTSEXTENDED{$this->arParams["ENTITY_XML_ID"]}"); }});
						</script>
HTML;
				}
				else
				{
					\CPullWatch::Add($this->getUserId(), 'UNICOMMENTS'.$this->arParams["ENTITY_XML_ID"]);
					$text = <<<HTML
						<script>
							BX.ready(function(){if (BX.PULL) { BX.PULL.extendWatch("UNICOMMENTS{$this->arParams["ENTITY_XML_ID"]}"); }});
						</script>
HTML;
				}
			}
			else
			{
				if ($this->arParams["RIGHTS"]["MODERATE"] == "Y" || $this->arParams["RIGHTS"]["MODERATE"] == "ALL")
				{
					\CPullWatch::Add($this->getUserId(), 'UNICOMMENTSMOBILEEXTENDED'.$this->arParams["ENTITY_XML_ID"]);
					$text .= <<<HTML
						<script>
							BXMobileApp.onCustomEvent('onPullExtendWatch', {'id': "UNICOMMENTSMOBILEEXTENDED{$this->arParams["ENTITY_XML_ID"]}"}, true);
						</script>
HTML;
				}
				else
				{
					\CPullWatch::Add($this->getUserId(), 'UNICOMMENTSMOBILE'.$this->arParams["ENTITY_XML_ID"]);
					$text = <<<HTML
						<script>
							BXMobileApp.onCustomEvent('onPullExtendWatch', {'id': "UNICOMMENTSMOBILE{$this->arParams["ENTITY_XML_ID"]}"}, true);
						</script>
HTML;
				}
				\CPullWatch::DeferredSql();
			}
		}
		return preg_replace("/\\s+/", "", $text);
	}

	protected function sendIntoPull(array &$arParams, array &$arResult)
	{
		if (((
				check_bitrix_sessid()
				|| (
					isset($arParams["PUSH&PULL"])
					&& isset($arParams["PUSH&PULL"]["AUTHOR_ID"])
					&& intval($arParams["PUSH&PULL"]["AUTHOR_ID"]) > 0
				)
			) &&
			($this->request->getPost("ENTITY_XML_ID") == $arParams["ENTITY_XML_ID"] ||
				$this->request->getQuery("ENTITY_XML_ID") == $arParams["ENTITY_XML_ID"]) || $arParams["MODE"] == "PULL_MESSAGE") &&
			is_array($arParams["PUSH&PULL"]) && $arParams["PUSH&PULL"]["ID"] > 0 &&
			Loader::includeModule("pull") && \CPullOptions::GetNginxStatus())
		{
			if ($arParams["PUSH&PULL"]["ACTION"] != "DELETE")
			{
				if (($res = $arParams["RECORDS"][$arParams["PUSH&PULL"]["ID"]]))
				{
					$comment = array_merge($res, $res["WEB"]);
					unset($comment["WEB"]);
					unset($comment["MOBILE"]);
					$comment["ACTION"] = $arParams["PUSH&PULL"]["ACTION"];
					$comment["POST_CONTENT_TYPE_ID"] = (!empty($arParams["POST_CONTENT_TYPE_ID"]) ? $arParams["POST_CONTENT_TYPE_ID"] : '');
					$comment["COMMENT_CONTENT_TYPE_ID"] = (!empty($arParams["COMMENT_CONTENT_TYPE_ID"]) ? $arParams["COMMENT_CONTENT_TYPE_ID"] : '');
					$comment["USER_ID"] = (isset($arParams["PUSH&PULL"]["AUTHOR_ID"]) && intval($arParams["PUSH&PULL"]["AUTHOR_ID"]) > 0 ? intval($arParams["PUSH&PULL"]["AUTHOR_ID"]) : $this->getUserId());
					$comment["EXEMPLAR_ID"] = $this->exemplarId;
					$comment["OPERATION_ID"] = $this->request->get("OPERATION_ID") ?: $this->exemplarId;
					if ($this->request->getPost("COMMENT_EXEMPLAR_ID") !== null)
						$comment["COMMENT_EXEMPLAR_ID"] = $this->request->getPost("COMMENT_EXEMPLAR_ID");

					\CPullWatch::AddToStack('UNICOMMENTSEXTENDED'.$arParams["ENTITY_XML_ID"],
						array(
							'module_id' => 'unicomments',
							'command' => 'comment',
							'params' => $comment
						)
					);
					if ($comment["APPROVED"] == "Y")
					{
						\CPullWatch::AddToStack('UNICOMMENTS'.$arParams["ENTITY_XML_ID"],
							array(
								'module_id' => 'unicomments',
								'command' => 'comment',
								'params' => $comment
							)
						);
					}
					else if ($comment["ACTION"] == "MODERATE" || $comment["ACTION"] == "HIDE")
					{
						\CPullWatch::AddToStack('UNICOMMENTS'.$arParams["ENTITY_XML_ID"],
							array(
								'module_id' => 'unicomments',
								'command' => 'comment',
								'params' => array(
									"ID" => $comment["ID"],
									"ENTITY_XML_ID" => $comment["ENTITY_XML_ID"],
									"APPROVED" => "N",
									"ACTION" => "HIDE",
									"USER_ID" => $comment["USER_ID"]
								)
							)
						);
					}
					if (IsModuleInstalled("mobile"))
					{
						$comment = array_merge($comment, $res["MOBILE"]);
						\CPullWatch::AddToStack('UNICOMMENTSMOBILEEXTENDED'.$arParams["ENTITY_XML_ID"],
							Array(
								'module_id' => 'unicomments',
								'command' => 'comment_mobile',
								'params' => $comment
							)
						);
						if ($comment["APPROVED"] == "Y")
						{
							\CPullWatch::AddToStack('UNICOMMENTSMOBILE'.$arParams["ENTITY_XML_ID"],
								Array(
									'module_id' => 'unicomments',
									'command' => 'comment_mobile',
									'params' => $comment
								)
							);
						}
						else if ($comment["ACTION"] == "MODERATE" || $comment["ACTION"] == "HIDE")
						{
							\CPullWatch::AddToStack('UNICOMMENTSMOBILE'.$arParams["ENTITY_XML_ID"],
								Array(
									'module_id' => 'unicomments',
									'command' => 'comment_mobile',
									'params' => array(
										"ID" => $comment["ID"],
										"ENTITY_XML_ID" => $comment["ENTITY_XML_ID"],
										"APPROVED" => "N",
										"ACTION" => "HIDE",
										"USER_ID" => $comment["USER_ID"]
									)
								)
							);
						}
					}
				}
			}
			else
			{
				\CPullWatch::AddToStack('UNICOMMENTS'.$arParams["ENTITY_XML_ID"],
					array(
						'module_id' => 'unicomments',
						'command' => 'comment',
						'params' => array(
							"ID" => $arParams["PUSH&PULL"]["ID"],
							"ENTITY_XML_ID" => $arParams["ENTITY_XML_ID"],
							"EXEMPLAR_ID" => $this->exemplarId,
							"OPERATION_ID" => $this->request->get("OPERATION_ID") ?: $this->exemplarId,
							"ACTION" => "DELETE",
							"USER_ID" => $this->getUserId()
						)
					)
				);
				\CPullWatch::AddToStack('UNICOMMENTSEXTENDED'.$arParams["ENTITY_XML_ID"],
					array(
						'module_id' => 'unicomments',
						'command' => 'comment',
						'params' => array(
							"ID" => $arParams["PUSH&PULL"]["ID"],
							"ENTITY_XML_ID" => $arParams["ENTITY_XML_ID"],
							"EXEMPLAR_ID" => $this->exemplarId,
							"OPERATION_ID" => $this->request->get("OPERATION_ID") ?: $this->exemplarId,
							"ACTION" => "DELETE",
							"USER_ID" => $this->getUserId()
						)
					)
				);

				if (IsModuleInstalled("mobile"))
				{
					\CPullWatch::AddToStack('UNICOMMENTSMOBILEEXTENDED'.$arParams["ENTITY_XML_ID"],
						Array(
							'module_id' => 'unicomments',
							'command' => 'comment_mobile',
							'params' => array(
								"ID" => $arParams["PUSH&PULL"]["ID"],
								"ENTITY_XML_ID" => $arParams["ENTITY_XML_ID"],
								"EXEMPLAR_ID" => $this->exemplarId,
								"OPERATION_ID" => $this->request->get("OPERATION_ID") ?: $this->exemplarId,
								"ACTION" => "DELETE",
								"USER_ID" => $this->getUserId()
							)
						)
					);
					\CPullWatch::AddToStack('UNICOMMENTSMOBILE'.$arParams["ENTITY_XML_ID"],
						Array(
							'module_id' => 'unicomments',
							'command' => 'comment_mobile',
							'params' => array(
								"ID" => $arParams["PUSH&PULL"]["ID"],
								"ENTITY_XML_ID" => $arParams["ENTITY_XML_ID"],
								"EXEMPLAR_ID" => $this->exemplarId,
								"OPERATION_ID" => $this->request->get("OPERATION_ID") ?: $this->exemplarId,
								"ACTION" => "DELETE",
								"USER_ID" => $this->getUserId()
							)
						)
					);
				}
			}
		}
	}

	protected function buildUser($id)
	{
		static $extranetUserIdList = false;

		if (
			$extranetUserIdList === false
			&& Loader::includeModule('socialnetwork')
		)
		{
			$extranetUserIdList = \Bitrix\Socialnetwork\ComponentHelper::getExtranetUserIdList();
		}

		$res = $id;
		if (!is_array($res))
		{
			if (!array_key_exists($res, static::$users))
			{
				$res = \CUser::GetById($res)->Fetch();
				$avatarId = intval($res["PERSONAL_PHOTO"]);
				$res = array(
					"ID" => $res["ID"],
					"NAME" => $res["NAME"],
					"LAST_NAME" => $res["LAST_NAME"],
					"SECOND_NAME" => $res["SECOND_NAME"],
					"AVATAR" => null,
					"EXTERNAL_AUTH_ID" => $res["EXTERNAL_AUTH_ID"]
				);
				if (
						$avatarId > 0 &&
						(
							$avatar = \CFile::ResizeImageGet(
								$avatarId,
								["width" => 42, "height" => 42],
								BX_RESIZE_IMAGE_EXACT,
								false)
						)
					)
				{
					$res["AVATAR"] = $avatar["src"];
				}
				static::$users[$id] = $res;
			}
			$res = static::$users[$id];
		}

		$res["NAME"] = htmlspecialcharsbx($res["NAME"]);
		$res["LAST_NAME"] = htmlspecialcharsbx($res["LAST_NAME"]);
		$res["SECOND_NAME"] = htmlspecialcharsbx($res["SECOND_NAME"]);
		$res["IS_EXTRANET"] = is_array($extranetUserIdList) && in_array($res["ID"], $extranetUserIdList) ? "Y" : "N";
		if (!isset($res["TYPE"]))
		{
			if (!empty($res["UF_USER_CRM_ENTITY"]))
			{
				$res["TYPE"] = "EMAILCRM";
			}
			elseif (
				isset($res["EXTERNAL_AUTH_ID"])
				&& $res["EXTERNAL_AUTH_ID"] == 'email'
			)
			{
				$res["TYPE"] = "EMAIL";
			}
			elseif ($res["IS_EXTRANET"] == 'Y')
			{
				$res["TYPE"] = "EXTRANET";
			}
			else
			{
				$res["TYPE"] = false;
			}
		}
		return $res;
	}

	protected function buildComment($res, array $specifiedTemplates = ['WEB', 'MOBILE'])
	{
		$arParams = $this->arParams;
		$templateId = implode('_', array($arParams["TEMPLATE_ID"], 'ID', $res['ID'], ''));

		$result = array(
			"ID" => $res["ID"], // integer
			"ENTITY_XML_ID" => $arParams["ENTITY_XML_ID"], // string
			"FULL_ID" => array($arParams["ENTITY_XML_ID"], $res["ID"]),
			"NEW" => $res["NEW"], //"Y" | "N"
			"COLLAPSED" => isset($res["COLLAPSED"]) && $res["COLLAPSED"] === "Y" ? "Y" : "N",
			"AUX" => $res["AUX"] ?? '',
			"AUX_LIVE_PARAMS" => $res["AUX_LIVE_PARAMS"] ?? [],
			"CAN_DELETE" => $res["CAN_DELETE"] ?? 'Y',
			"APPROVED" => $res["APPROVED"], //"Y" | "N"
			"POST_TIMESTAMP" => ($res["POST_TIMESTAMP"] - CTimeZone::GetOffset()),
			"~POST_MESSAGE_TEXT" => $res["~POST_MESSAGE_TEXT"],
			"AUTHOR" => $this->buildUser(isset($res["AUTHOR_ID"]) && $res["AUTHOR_ID"] ? $res["AUTHOR_ID"] : $res["AUTHOR"]),
			"RATING" => array_key_exists("RATING", $res) ? $res["RATING"] : false,
			"CLASSNAME" => '',
			"SHOW_MOBILE_HINTS" => $res['SHOW_MOBILE_HINTS'] ?? 'N',
		);

		//region Special data for web and mobile views. General data
		$specifiedTemplates = array_intersect(['WEB', 'MOBILE'], array_merge($specifiedTemplates, [$this->isWeb() ? 'WEB' : 'MOBILE']));
		//1. General
		foreach ($specifiedTemplates as $templateCode)
		{
			$val = $res[$templateCode] ?? $res;

			$defaultDateTime = \CComponentUtil::getDateTimeFormatted(array(
				'TIMESTAMP' => $res["POST_TIMESTAMP"],
				'DATETIME_FORMAT' => $arParams["DATE_TIME_FORMAT"],
				'DATETIME_FORMAT_WITHOUT_YEAR' => ($arParams["DATE_TIME_FORMAT_WITHOUT_YEAR"] ?? false),
				'TZ_OFFSET' => CTimeZone::GetOffset(),
				'HIDE_TODAY' => true
			));

			$classNameList = [
				$val["CLASSNAME"]
			];

			if (!empty($res['AUX']))
			{
				$classNameList[] = 'mpl-comment-aux';
				$classNameList[] = 'mpl-comment-aux-'.mb_strtolower($res['AUX']);
			}

			$result[$templateCode] = array(
				"POST_TIME" => ($val["POST_TIME"] ?? $defaultDateTime),
				"POST_DATE" => ($val["POST_DATE"] ?? $defaultDateTime),
				"POST_DATE_AGO" => FormatDate(array(
					"s" => "sshort",
					"i" => "ishort",
					"H" => "Hshort",
					"d" => "dshort",
					"m" => "mshort",
					"Y" => "Yshort"
				), $res["POST_TIMESTAMP"], time() + CTimeZone::getOffset()),
				"POST_DATE_AGO_FULL" => FormatDate(array(
					"s" => "sago",
					"i" => "iago",
					"H" => "Hago",
					"d" => "dago",
					"m" => "mago",
					"Y" => "Yago"
				), $res["POST_TIMESTAMP"], time() + CTimeZone::getOffset()),
				"CLASSNAME" => implode(' ', $classNameList),
				"POST_MESSAGE_TEXT" => $val["POST_MESSAGE_TEXT"] ?? '',
				"BEFORE_HEADER" => ($val["BEFORE_HEADER"] ?? '').$this->getApplication()->GetViewContent($templateId.'BEFORE_HEADER'),
				"BEFORE_ACTIONS" => ($val["BEFORE_ACTIONS"] ?? '').$this->getApplication()->GetViewContent($templateId.'BEFORE_ACTIONS'),
				"AFTER_ACTIONS" => ($val["AFTER_ACTIONS"] ?? '').$this->getApplication()->GetViewContent($templateId.'AFTER_ACTIONS'),
				"AFTER_HEADER" => ($val["AFTER_HEADER"] ?? '').$this->getApplication()->GetViewContent($templateId.'AFTER_HEADER'),
				"BEFORE" => ($val["BEFORE"] ?? '').$this->getApplication()->GetViewContent($templateId.'BEFORE'),
				"AFTER" => ($val["AFTER"] ?? '').$this->getApplication()->GetViewContent($templateId.'AFTER'),
				"BEFORE_RECORD" => ($val["BEFORE_RECORD"] ?? '').$this->getApplication()->GetViewContent($templateId.'BEFORE_RECORD'),
				"AFTER_RECORD" => ($val["AFTER_RECORD"] ?? '').$this->getApplication()->GetViewContent($templateId.'AFTER_RECORD'),
				"LIKE_REACT" => ($val["LIKE_REACT"] ?? '') . $this->getApplication()->GetViewContent($templateId.'LIKE_REACT'),
			);
		}
		//2. Rating
		$userHasVoted = (
			(
				isset($res["RATING_USER_HAS_VOTED"])
				&& $res["RATING_USER_HAS_VOTED"] == "Y"
			)
			|| (
				isset($this->arParams["RATING_RESULTS"][$result["ID"]]["USER_HAS_VOTED"])
				&& $this->arParams["RATING_RESULTS"][$result["ID"]]["USER_HAS_VOTED"] == 'Y'
			)
		);

		if (
			!empty($res["RATING_VOTE_ID"])
			&& ModuleManager::isModuleInstalled('intranet')
		)
		{
			if (!empty($res["RATING_USER_REACTION"]))
			{
				$emotion = mb_strtoupper($res["RATING_USER_REACTION"]);
			}
			else
			{
				$emotion = (!empty($this->arParams["RATING_RESULTS"][$result["ID"]]["USER_REACTION"])? mb_strtoupper($this->arParams["RATING_RESULTS"][$result["ID"]]["USER_REACTION"]) : 'LIKE');
			}

			$buttonText = \CRatingsComponentsMain::getRatingLikeMessage($emotion);

			if (isset($result['WEB']))
			{
				ob_start();
				?><span id="bx-ilike-button-<?=htmlspecialcharsbx($res["RATING_VOTE_ID"])?>" class="feed-inform-ilike feed-new-like">
					<span class="bx-ilike-left-wrap<?=($userHasVoted ? ' bx-you-like-button' : '')?>"><a href="#like" class="bx-ilike-text"><?=$buttonText?></a></span>
				</span><?
				$result["WEB"]["BEFORE_ACTIONS"] .= ob_get_clean();
			}
			if (isset($result['MOBILE']))
			{
				ob_start();
				?><span id="bx-ilike-button-<?=htmlspecialcharsbx($res["RATING_VOTE_ID"])?>" class="post-comment-control-item post-comment-control-item-like bx-ilike-text" data-rating-vote-id="<?=htmlspecialcharsbx($res["RATING_VOTE_ID"])?>"><?
					?><span class="bx-ilike-left-wrap<?=($userHasVoted ? ' bx-you-like-button' : '')?>"><?
						?><span class="bx-ilike-text"><?=$buttonText?></span><?
					?></span><?
				?></span><?
				$result["MOBILE"]["BEFORE_ACTIONS"] .= ob_get_clean();
			}
		}

		if (
			$result["RATING"] === false
			&& array_key_exists("RATING_RESULTS", $this->arParams)
		)
		{
			$ratingValues = (
				!empty($this->arParams["RATING_RESULTS"][$result["ID"]])
					? array(
						"USER_VOTE" => $this->arParams["RATING_RESULTS"][$result["ID"]]["USER_VOTE"],
						"USER_HAS_VOTED" => $this->arParams["RATING_RESULTS"][$result["ID"]]["USER_HAS_VOTED"],
						"TOTAL_VOTES" => $this->arParams["RATING_RESULTS"][$result["ID"]]["TOTAL_VOTES"],
						"TOTAL_POSITIVE_VOTES" => $this->arParams["RATING_RESULTS"][$result["ID"]]["TOTAL_POSITIVE_VOTES"],
						"TOTAL_NEGATIVE_VOTES" => $this->arParams["RATING_RESULTS"][$result["ID"]]["TOTAL_NEGATIVE_VOTES"],
						"TOTAL_VALUE" => $this->arParams["RATING_RESULTS"][$result["ID"]]["TOTAL_VALUE"],
						"USER_REACTION" => $this->arParams["RATING_RESULTS"][$result["ID"]]["USER_REACTION"],
						"REACTIONS_LIST" => $this->arParams["RATING_RESULTS"][$result["ID"]]["REACTIONS_LIST"],
					)
					: array()
			);

			if (isset($result['WEB']))
			{
				ob_start();
				$result["WEB"]["RATING"] = $this->getApplication()->includeComponent(
					"bitrix:rating.vote",
					(!empty($res["RATING_VOTE_ID"]) && ModuleManager::isModuleInstalled('intranet') ? "like_react" : "like"),
					array(
						"COMMENT" => "Y",
						"ENTITY_TYPE_ID" => $this->arParams["RATING_TYPE_ID"],
						"ENTITY_ID" => $result["ID"],
						"OWNER_ID" => $result["AUTHOR"]["ID"],
						"PATH_TO_USER_PROFILE" => $this->arParams["AUTHOR_URL"],
						"VOTE_ID" => (!empty($res["RATING_VOTE_ID"]) ? $res["RATING_VOTE_ID"] : ""),
						'CURRENT_USER_ID' => (isset($this->arParams['CURRENT_USER_ID']) ? (int)$this->arParams['CURRENT_USER_ID'] : 0),
					) + $ratingValues,
					$this,
					array("HIDE_ICONS" => "Y")
				);

				$result["WEB"][(!empty($res["RATING_VOTE_ID"]) && ModuleManager::isModuleInstalled('intranet') ? "LIKE_REACT" : "BEFORE_ACTIONS")] .= ob_get_clean();
			}

			if (isset($result['MOBILE']))
			{
				ob_start();
				$result["MOBILE"]["RATING"] = $this->getApplication()->includeComponent(
					"bitrix:rating.vote",
					"like_react",
					array(
						"MOBILE" => "Y",
						"COMMENT" => "Y",
						"ENTITY_TYPE_ID" => $this->arParams["RATING_TYPE_ID"],
						"ENTITY_ID" => $result["ID"],
						"OWNER_ID" => $result["AUTHOR"]["ID"],
						"PATH_TO_USER_PROFILE" => $this->arParams["AUTHOR_URL"],
						"VOTE_ID" => (!empty($res["RATING_VOTE_ID"]) ? $res["RATING_VOTE_ID"] : "")
					) + $ratingValues,
					$this,
					array("HIDE_ICONS" => "Y")
				);
				$result["MOBILE"]["LIKE_REACT"] .= ob_get_clean();
			}
		}
		//3. Files
		if (isset($res["FILES"]) && is_array($res["FILES"]))
		{
			$images = array();
			$files = array();
			foreach ($res["FILES"] as $file)
			{
				if (is_array($file) &&
					($file = array_change_key_case($file, CASE_UPPER)) &&
					array_key_exists("SRC", $file))
				{
					if (CFile::IsImage($file["ORIGINAL_NAME"], $file["CONTENT_TYPE"]))
					{
						$images[] = $file;
					}
					else
					{
						$files[] = $file;
					}
				}
			}
			if (!empty($images) && isset($result['WEB']))
			{
				ob_start();
				?><div class="feed-com-files">
					<div class="feed-com-files-title"><?=GetMessage("MPL_PHOTO")?></div>
					<div class="feed-com-files-cont"><?
				foreach ($images as $file)
				{
					$thumbnail = ($file["THUMBNAIL"] ?: $file["SRC"]);
					?><span class="feed-com-files-photo">
						<img src="<?=$thumbnail?>" data-bx-src="<?=$file["SRC"]?>" <?
							?>border="0" data-bx-viewer="image" <?
							if (!empty($file["RESIZED_WIDTH"]))
							{
								?>width="<?= (int)$file["RESIZED_WIDTH"] ?>" <?
							}
							if (!empty($file["RESIZED_HEIGHT"]))
							{
								?>height="<?= (int)$file["RESIZED_HEIGHT"] ?>" <?
							 }
							?>data-bx-width="<?=$file["WIDTH"]?>" <?
							?>data-bx-height="<?=$file["HEIGHT"]?>" <?
							?>data-bx-title="<?=($file["FILE_NAME"])?>" <?
							?>data-bx-size="<?=$file["FILE_SIZE"]?>"/></span><?
				}
					?></div>
				</div><?
				$result["WEB"]["AFTER"] = preg_replace("/[\n\t]/", "", ob_get_clean()).$result["WEB"]["AFTER"];
			}
			if (!empty($images) && isset($result['MOBILE']))
			{
				ob_start();
				?><div class="post-item-attached-img-wrap"><?
					$ids = array();
					foreach($images as $file)
					{
						$id = "mpl-".$arParams["ENTITY_XML_ID"]."-".mb_strtolower(randString(5));
						$ids[] = $id;
						$thumbnail = ($file["THUMBNAIL"] ?: $file["SRC"]);
						?><div class="post-item-attached-img-block" onclick="<?
							?>app.loadPageBlank({ url: '<?=$file["SRC"]?>' }); <?
							?>event.stopPropagation();"><img class="post-item-attached-img" <?
							?>id="<?=$id?>" src="" <?
							?>data-src="<?=$thumbnail?>" border="0"></div><?
					}
				?><script>BitrixMobile.LazyLoad.registerImages(<?=CUtil::PhpToJSObject($ids)?>, oMSL.checkVisibility);</script><?
				?></div><?
				$result["MOBILE"]["AFTER"] = preg_replace("/[\n\t]/", "", ob_get_clean()).$result["MOBILE"]["AFTER"];
			}
			if (!empty($files) && isset($result['WEB']))
			{
				ob_start();
				?><div class="feed-com-files feed-com-basic-files-entity">
					<div class="feed-com-files-title"><?=GetMessage("MPL_FILES")?></div>
					<div class="feed-com-files-cont"><?
				foreach ($files as $file)
				{
					$url = $file["URL"] ?: $file["SRC"];
					$size = CFile::FormatSize($file["FILE_SIZE"]);
					$ext = GetFileExtension($file["FILE_NAME"]);
					?><div class="feed-com-file-wrap">
						<span class="feed-con-file-icon feed-file-icon-<?=$ext?>"></span>
						<span class="feed-com-file-name-wrap">
							<a href="<?=$url?>" <?
								?>class="feed-com-file-name" <?
								?>data-bx-viewer="unknown" <?
								?>data-bx-src="<?=$url?>" <?
								?>data-bx-title="<?=($file["FILE_NAME"])?>" <?
								?>data-bx-size="<?=$size?>" <?
								?>data-bx-owner="" <?
								?>data-bx-dateModify="" <?
								?>title="<?=($file["FILE_NAME"])?>" <?
								?>target="_blank" ><?=($file["FILE_NAME"])?></a>
							<span class="feed-com-file-size"><?=$size?></span>
						</span>
					</div><?
				}
					?></div>
				</div><?
				$result["WEB"]["AFTER"] = preg_replace("/[\n\t]/", "", ob_get_clean()).$result["WEB"]["AFTER"];
			}
			if (!empty($files) && isset($result['MOBILE']))
			{
				ob_start();
				?><ul class="post-item-attached-file-wrap"><?
					foreach($files as $file)
					{
						?><li><?=$file["FILE_NAME"]?></li><?
					}
				?></ul><?
				$result["MOBILE"]["AFTER"] .= ob_get_clean();
			}
		}
		//4. UFs
		if (is_array($res['UF']) && isset($result['WEB']))
		{
			ob_start();
			$uf = ($res["WEB"]['UF'] ?? $res['UF']);
			foreach ($uf as $arPostField)
			{
				if (!empty($arPostField["VALUE"]))
				{
					$this->getApplication()->IncludeComponent(
						"bitrix:system.field.view",
						$arPostField["USER_TYPE"]["USER_TYPE_ID"],
						[
							"arUserField" => $arPostField,
							"TEMPLATE" => $this->getTemplateName(),
							"LAZYLOAD" => (isset($arParams["LAZYLOAD"]) && $arParams["LAZYLOAD"] == "Y" ? "Y" : "N"),
							"DISABLE_LOCAL_EDIT" => (isset($arParams["bPublicPage"]) && $arParams["bPublicPage"])
						] + $arParams,
						null,
						["HIDE_ICONS" => "Y"]
					);
				}
			}
			$result["WEB"]["AFTER"] = ob_get_clean() . $result["WEB"]["AFTER"];
		}

		if (is_array($res["UF"]) && isset($result['MOBILE']))
		{
			ob_start();
			$uf = ($res["MOBILE"]['UF'] ?? $res['UF']);
			foreach ($uf as $arPostField)
			{
				if(!empty($arPostField["VALUE"]))
				{
					$this->getApplication()->IncludeComponent(
						"bitrix:system.field.view",
						$arPostField["USER_TYPE"]["USER_TYPE_ID"],
						array(
							"arUserField" => $arPostField,
							"TEMPLATE" => $this->getTemplateName(),
							"LAZYLOAD" => (isset($arParams["LAZYLOAD"]) && $arParams["LAZYLOAD"] == "Y" ? "Y" : "N"),
							"MOBILE" => "Y"
						) + $arParams,
						null,
						array("HIDE_ICONS"=>"Y")
					);
				}
			}
			$html = ob_get_clean();
			if (!empty($html))
			{
				$result["MOBILE"]["AFTER"] = '<div class="post-item-attached-file-wrap" id="record-'.$arParams["ENTITY_XML_ID"].'-'.$res["ID"].'-uf">'.$html.'</div>'.$result["MOBILE"]["AFTER"];
			}
			$result["CLASSNAME"] .= " feed-com-block-uf";
		}
		$result = array_merge($result, ($this->isWeb() ? $result['WEB'] : $result['MOBILE']));
		//endregion

		return $result;
	}

	public function parseTemplate(array $res, array $arParams, $template)
	{
		global $USER;
		static $extranetSiteId = null;

		$todayString = ConvertTimeStamp();

		if ($extranetSiteId === null)
		{
			$extranetSiteId = (Loader::includeModule('extranet') ? CExtranet::GetExtranetSiteID() : false);
		}

		$authorUrl = (
			$res["AUTHOR"]["ID"]
				? str_replace(
					array("#ID#", "#id#", "#USER_ID#", "#user_id#"),
					array($res["ID"], $res["ID"], $res["AUTHOR"]["ID"], $res["AUTHOR"]["ID"]),
					$arParams["AUTHOR_URL"]
				)
				: "javascript:void();"
		);

		$authorStyle = '';
		$authorTooltipParams = array();

		if (!empty($res["AUTHOR"]["TYPE"]))
		{
			if ($res["AUTHOR"]["TYPE"] == 'EMAILCRM')
			{
				$authorStyle = ' feed-com-name-emailcrm';
			}
			if ($res["AUTHOR"]["TYPE"] == 'EMAIL')
			{
				$authorStyle = ' feed-com-name-email';
			}
			else if ($res["AUTHOR"]["TYPE"] == 'EXTRANET')
			{
				$authorStyle = ' feed-com-name-extranet';
			}
		}
		else if ($res["AUTHOR"]["IS_EXTRANET"] == "Y")
		{
			$authorStyle = ' feed-com-name-extranet';
		}

		if (
			!empty($arParams["AUTHOR_URL_PARAMS"]) && is_array($arParams["AUTHOR_URL_PARAMS"])
			&& (
				(isset($arParams["bPublicPage"]) && $arParams["bPublicPage"])
				|| SITE_ID == $extranetSiteId
				|| (!empty($res["AUTHOR"]["TYPE"]) && in_array($res["AUTHOR"]["TYPE"], array('EMAIL', 'EMAILCRM', 'EXTRANET')))
			)
		)
		{
			$authorTooltipParams = $arParams["AUTHOR_URL_PARAMS"];
			if (
				!isset($arParams["bPublicPage"])
				|| !$arParams["bPublicPage"])
			{
				$strParams = '';
				$i = 0;
				foreach ($arParams["AUTHOR_URL_PARAMS"] as $key => $value)
				{
					$strParams .= ($i > 0 ? '&' : '').urlencode($key).'='.urlencode($value);
					$i++;
				}
				$authorUrl .= (strpos($authorUrl, '?') === false ? '?' : '&').$strParams;
			}
		}

		$viewUri = new \Bitrix\Main\Web\Uri(htmlspecialcharsback(str_replace(array("#ID#", "#id#"), $res["ID"], $arParams["VIEW_URL"])));
		$viewUri->deleteParams(['b24statAction']);

		$contentId = (
			!empty($arParams["RATING_TYPE_ID"])
				? $arParams["RATING_TYPE_ID"] . "-" . $res["ID"]
				: (
					!empty($arParams["CONTENT_TYPE_ID"])
						? $arParams["CONTENT_TYPE_ID"] . "-" . $res["ID"]
						: ""
				)
		);

		$replacement = array(
			"#ID#" =>
				$res["ID"],
			"#EXEMPLAR_ID#" =>
				$this->exemplarId,
			"#FULL_ID#" =>
				$arParams["ENTITY_XML_ID"]."-".$res["ID"],
			"#CONTENT_ID#" =>
				$contentId,
			"#ENTITY_XML_ID#" =>
				$arParams["ENTITY_XML_ID"],
			"#NEW#" =>
				($res["NEW"] == "Y" ? "new" : "old"),
			"#APPROVED#" =>
				($res["APPROVED"] != "Y" ? "hidden" : "approved"),
			"#DATE#" => (ConvertTimeStamp(($res["POST_TIMESTAMP"] + CTimeZone::GetOffset()), "SHORT") == $todayString ? $res["POST_TIME"] : $res["POST_DATE"]),
//			"#DATE#" => $res["POST_DATE_AGO"],
//			"#DATE_FULL#" => $res["POST_DATE_AGO_FULL"],
			"#TEXT#" => str_replace(array("\001", "#"), array("", "\001"), $res["POST_MESSAGE_TEXT"]),
			"#CLASSNAME#" =>
				(isset($res["CLASSNAME"]) ? " ".$res["CLASSNAME"] : ""),
			"#VOTE_ID#" =>
				(is_array($res["RATING"]) ? $res["RATING"]["VOTE_ID"] : ""),
			"#VIEW_URL#" =>
				$viewUri->getUri(),
			"#VIEW_SHOW#" =>
				($arParams["VIEW_URL"] == "" ? "N" : "Y"),
			"#EDIT_URL#" =>
				str_replace(array("#ID#", "#id#"), $res["ID"], $arParams["EDIT_URL"]),
			"#EDIT_SHOW#" => (
				empty($res["AUX"])
				&& (
					$arParams["RIGHTS"]["EDIT"] == "Y"
					|| $arParams["RIGHTS"]["EDIT"] == "ALL"
					|| ($arParams["RIGHTS"]["EDIT"] == "OWN" && $this->getUserId() == intval($res["AUTHOR"]["ID"]))
				)
					? "Y"
					: "N"
			),
			"#MODERATE_URL#" =>
				str_replace(array("#ID#", "#id#"), $res["ID"], $arParams["MODERATE_URL"]),
			"#MODERATE_SHOW#" =>(
				empty($res["AUX"])
				&& (
					$arParams["RIGHTS"]["MODERATE"] == "Y"
					|| $arParams["RIGHTS"]["MODERATE"] == "ALL"
					|| $arParams["RIGHTS"]["MODERATE"] == "OWN" && $this->getUserId() == intval($res["AUTHOR"]["ID"])
				)
					? "Y"
					: "N"
			),
			"#DELETE_URL#" =>
				str_replace(array("#ID#", "#id#"), $res["ID"], $arParams["DELETE_URL"]),
			"#DELETE_SHOW#" => (
				(empty($res["CAN_DELETE"]) || $res["CAN_DELETE"] == 'Y')
				&& (
					$arParams["RIGHTS"]["DELETE"] == "Y"
					|| $arParams["RIGHTS"]["DELETE"] == "ALL"
					|| ($arParams["RIGHTS"]["DELETE"] == "OWN" && $this->getUserId() == intval($res["AUTHOR"]["ID"]))
				)
					? "Y"
					: "N"
			),
			"#CREATETASK_SHOW#" => (
				empty($res["AUX"])
				&& $arParams["RIGHTS"]["CREATETASK"] == "Y"
					? "Y"
					: "N"
			),
			"#CREATESUBTASK_SHOW#" => (
				empty($res['AUX'])
				&& isset($arParams['RIGHTS']['CREATESUBTASK'])
				&& $arParams['RIGHTS']['CREATESUBTASK'] === 'Y'
					? 'Y'
					: 'N'
			),
			"#POST_ENTITY_TYPE#" => (!empty($arParams["POST_CONTENT_TYPE_ID"]) ? $arParams["POST_CONTENT_TYPE_ID"] : ''),
			"#COMMENT_ENTITY_TYPE#" => (!empty($arParams["COMMENT_CONTENT_TYPE_ID"]) ? $arParams["COMMENT_CONTENT_TYPE_ID"] : ''),
			"#BEFORE_HEADER#" => $res["BEFORE_HEADER"],
			"#BEFORE_ACTIONS#" => $res["BEFORE_ACTIONS"],
			"#AFTER_ACTIONS#" => $res["AFTER_ACTIONS"],
			"#AFTER_HEADER#" => $res["AFTER_HEADER"],
			"#BEFORE#" => $res["BEFORE"],
			"#AFTER#" => $res["AFTER"],
			"#BEFORE_RECORD#" => $res["BEFORE_RECORD"],
			"#AFTER_RECORD#" => $res["AFTER_RECORD"],
			"#LIKE_REACT#" => (!empty($res["LIKE_REACT"]) ? $res["LIKE_REACT"] : ''),
			"#AUTHOR_ID#" =>
				$res["AUTHOR"]["ID"],
			"#AUTHOR_AVATAR_IS#" =>
				(empty($res["AUTHOR"]["AVATAR"]) ? "N" : "Y"),
			"#AUTHOR_AVATAR#" => (
				!empty($res["AUTHOR"]["AVATAR"])
					? Uri::urnEncode($res['AUTHOR']['AVATAR'])
					: (
						!empty($arParams["AVATAR_DEFAULT"])
							? Uri::urnEncode($arParams["AVATAR_DEFAULT"])
							: ""
					)
			),
			"#AUTHOR_AVATAR_BG#" => (
				!empty($res["AUTHOR"]["AVATAR"])
					? "background-image:url('" . Uri::urnEncode($res["AUTHOR"]["AVATAR"]) . "')"
					: (
						!empty($arParams["AVATAR_DEFAULT"])
							? "background-image:url('" . Uri::urnEncode($arParams["AVATAR_DEFAULT"]) . "')"
							: ""
					)
				),
			"#AUTHOR_URL#" => $authorUrl,
			"#AUTHOR_NAME#" =>
				CUser::FormatName(
				$arParams["NAME_TEMPLATE"],
				array(
					"NAME" => $res["AUTHOR"]["NAME"],
					"LAST_NAME" => $res["AUTHOR"]["LAST_NAME"],
					"SECOND_NAME" => $res["AUTHOR"]["SECOND_NAME"],
					"LOGIN" => $res["AUTHOR"]["LOGIN"],
					"NAME_LIST_FORMATTED" => ""
				),
				($arParams["SHOW_LOGIN"] != "N"),
				false),
			"#AUTHOR_PERSONAL_GENDER#" => !empty($res["AUTHOR"]["PERSONAL_GENDER"]) ?
				$res["AUTHOR"]["PERSONAL_GENDER"] : "",
			"#AUTHOR_TOOLTIP_PARAMS#" => htmlspecialcharsbx(\Bitrix\Main\Web\Json::encode($authorTooltipParams)),
			"#SHOW_POST_FORM#" =>
				$arParams["SHOW_POST_FORM"],
			"#SHOW_MENU#" =>
				(!isset($arParams["SHOW_MENU"]) || !!$arParams["SHOW_MENU"] ? 'Y' : 'N'),
			"#AUTHOR_EXTRANET_STYLE#" =>
				$authorStyle,
			"#RATING_NONEMPTY_CLASS#" =>
				(!empty($res['RATING']) && !empty($res['RATING']['TOTAL_VOTES']) && $res['RATING']['TOTAL_VOTES'] > 0 ? 'comment-block-rating-nonempty' : ''),
			"background:url('') no-repeat center;" =>
				"",
			"#MOBILE_HINTS#" => (isset($res['SHOW_MOBILE_HINTS']) && $res['SHOW_MOBILE_HINTS'] === 'Y')
				? '<span class="feed__mobile_btn"></span>'
				: '',
		);

		return str_replace(array_merge(array_keys($replacement), array("\001")), array_merge(array_values($replacement), array("#")), $template);
	}

	protected function prepareParams(array &$arParams, array &$arResult)
	{
		global $USER;

		static $currentExtranetUser = null;
		static $availableUsersList = null;

		// Action params
		/*@param string $arParams["mfi"] contains hash of something to add new uploaded file into session array */
		$arParams["mfi"] = trim($arParams["mfi"]);
		// List params
		$arParams["EXEMPLAR_ID"] = trim($this->exemplarId);
		/*@param string $arParams["ENTITY_XML_ID"] main param that means ID */
		$arParams["ENTITY_XML_ID"] = trim($arParams["ENTITY_XML_ID"]);
		/*@param array $arParams["RECORDS"] contains data to view */
		$arParams["RECORDS"] = (is_array($arParams["RECORDS"]) ? $arParams["RECORDS"] : array());
		$arParams["~RECORDS"] = $arParams["RECORDS"];
		$arParams["NAV_STRING"] = (!!$arParams["NAV_STRING"] && is_string($arParams["NAV_STRING"]) ? $arParams["NAV_STRING"] : "");
		//$arParams["NAV_RESULT"] = (!!$arParams["NAV_STRING"] && is_object($arParams["NAV_RESULT"]) ? $arParams["NAV_RESULT"] : false);
		$arParams["PREORDER"] = ($arParams["PREORDER"] == "Y" ? "Y" : "N");
		$arParams["RIGHTS"] = (is_array($arParams["RIGHTS"]) ? $arParams["RIGHTS"] : array());
		foreach (array("MODERATE", "EDIT", "DELETE", "CREATETASK") as $act)
			$arParams["RIGHTS"][$act] = in_array(mb_strtoupper($arParams["RIGHTS"][$act]), array("Y", "ALL", "OWN", "OWNLAST")) ? $arParams["RIGHTS"][$act] : "N";
		$arParams["LAST_RECORD"] = array();
		// Answer params
		/*@param int $arParams["RESULT"] contains id of new record for cutting out and sending back*/
		$arParams["RESULT"] = intval($arParams["RESULT"] ?: $this->request->getPost("MID"));
		/*@param array $arParams["PUSH&PULL"] contains record id to pushing other clients */
		$arParams["PUSH&PULL"] = ($arParams["~PUSH&PULL"] ?? $arParams["PUSH&PULL"]);
		$arParams["MODE"] = (is_array($arParams["PUSH&PULL"]) && $arParams["PUSH&PULL"]["ID"] > 0 && isset($arParams["MODE"]) && $arParams["MODE"] == "PULL_MESSAGE" ? "PULL_MESSAGE" : "PLAIN");

		/*@param string $arParams["NOTIFY_TAG"] params for bottom notifier */
		$arParams["NOTIFY_TAG"] = trim($arParams["NOTIFY_TAG"]);
		$arParams["NOTIFY_TEXT"] = trim($arParams["NOTIFY_TEXT"]);
		$arParams["ERROR_MESSAGE"] = trim($arParams["ERROR_MESSAGE"]);
		$arParams["OK_MESSAGE"] = trim($arParams["OK_MESSAGE"]);
		// Template params
		$arParams["VISIBLE_RECORDS_COUNT"] = (!!$arParams["NAV_RESULT"] ? intval($arParams["VISIBLE_RECORDS_COUNT"]) : 0);
		$arParams["TEMPLATE_ID"] = (!!$arParams["TEMPLATE_ID"] ? $arParams["TEMPLATE_ID"] : 'COMMENT_'.$arParams["ENTITY_XML_ID"].'_');
		$arParams["AVATAR_SIZE"] = ($arParams["AVATAR_SIZE"] > 0 ? $arParams["AVATAR_SIZE"] : 100);
		//$arParams["IMAGE_SIZE"] = ($arParams["IMAGE_SIZE"] > 0 ? $arParams["IMAGE_SIZE"] : 30);
		$arParams['SHOW_MINIMIZED'] = ($arParams['SHOW_MINIMIZED'] == "Y" ? "Y" : "N");

		$arParams["NAME_TEMPLATE"] = (isset($_REQUEST["NAME_TEMPLATE"]) && $_REQUEST["NAME_TEMPLATE"] ? $_REQUEST["NAME_TEMPLATE"] : (isset ($arParams["NAME_TEMPLATE"]) && $arParams["NAME_TEMPLATE"] ? $arParams["NAME_TEMPLATE"] : \CSite::GetNameFormat()));
		$arParams["SHOW_LOGIN"] = (isset($_REQUEST["SHOW_LOGIN"]) && $_REQUEST["SHOW_LOGIN"] == "Y" ? "Y" : (isset($arParams["SHOW_LOGIN"]) && $arParams["SHOW_LOGIN"] == "Y" ? "Y" : "N"));
		$arParams["DATE_TIME_FORMAT"] = trim($arParams["DATE_TIME_FORMAT"]);
		$arParams["FORM_ID"] = trim($arParams["FORM_ID"]);
		$arParams["SHOW_POST_FORM"] = ($arParams["SHOW_POST_FORM"] == "Y" || $arParams["FORM_ID"] <> '' ? "Y" : "N");
		$arParams["BIND_VIEWER"] = (isset($arParams["BIND_VIEWER"]) && $arParams["BIND_VIEWER"] == "N" ? "N" : "Y");
		$arParams["SIGN"] = $this->sign->sign($arParams["ENTITY_XML_ID"], "main.post.list");

		$arParams["VIEW_URL"] = trim($arParams["VIEW_URL"] ?? '');
		$arParams["EDIT_URL"] = trim($arParams["EDIT_URL"] ?? '');
		$arParams["MODERATE_URL"] = trim($arParams["MODERATE_URL"] ?? '');
		$arParams["DELETE_URL"] = trim($arParams["DELETE_URL"] ?? '');
		$arParams["AUTHOR_URL"] = trim(($arParams["PATH_TO_USER"] ?? '') ?: $arParams["AUTHOR_URL"]);

		if ($arParams["VISIBLE_RECORDS_COUNT"] > 0)
		{
			if ($arParams["NAV_RESULT"]->bShowAll)
			{
				$arParams["VISIBLE_RECORDS_COUNT"] = 0;
			}
			elseif (array_key_exists($arParams["RESULT"], $arParams["RECORDS"]))
			{
				$arParams["VISIBLE_RECORDS_COUNT"] = count($arParams["RECORDS"]);
			}
			elseif (isset($_REQUEST["PAGEN_".$arParams["NAV_RESULT"]->NavNum]) ||
				isset($_REQUEST["FILTER"]) && $arParams["ENTITY_XML_ID"] == $_REQUEST["ENTITY_XML_ID"])
			{
				$arParams["VISIBLE_RECORDS_COUNT"] = 0;
			}
			elseif (0 < $arParams["NAV_RESULT"]->NavRecordCount &&
				$arParams["NAV_RESULT"]->NavRecordCount <= $arParams["VISIBLE_RECORDS_COUNT"])
			{
				$arParams["VISIBLE_RECORDS_COUNT"] = count($arParams["RECORDS"]);
			}
			if (!!$arParams["NAV_STRING"])
			{
				$path = "PAGEN_".$arParams["NAV_RESULT"]->NavNum."=";
				if ($arParams["VISIBLE_RECORDS_COUNT"] > 0)
					$path .= $arParams["NAV_RESULT"]->NavPageNomer;
				else if ($arParams["NAV_RESULT"]->bDescPageNumbering)
					$path .= ($arParams["NAV_RESULT"]->NavPageNomer - 1);
				else
					$path .= ($arParams["NAV_RESULT"]->NavPageNomer + 1);
				$arParams["NAV_STRING"] .= (strpos($arParams["NAV_STRING"], "?") === false ? "?" : "&").$path;
			}
		}
		if (!empty($arParams["RECORDS"]))
		{
			if ($arParams["VISIBLE_RECORDS_COUNT"] > 0)
			{
				$arParams["RECORDS"] = array_slice($arParams["RECORDS"], 0, $arParams["VISIBLE_RECORDS_COUNT"], true);
			}

			$arParams["LAST_RECORD"] = end($arParams["RECORDS"]);
			reset($arParams["RECORDS"]);

			if ($arParams["PREORDER"] === "N")
				$arParams["RECORDS"] = array_reverse($arParams["RECORDS"], true);

			if (!empty($arParams["RATING_TYPE_ID"]))
				$arParams["RATING_RESULTS"] = CRatings::GetRatingVoteResult($arParams["RATING_TYPE_ID"], array_keys($arParams["RECORDS"]));

			$arParams["~RECORDS"] = $arParams["RECORDS"];
			foreach ($arParams["~RECORDS"] as $key => $res)
			{
				$arParams["RECORDS"][$key] = $this->buildComment($res, !empty($arParams["PUSH&PULL"]["ID"])
					? ['WEB', 'MOBILE'] : [$this->isWeb() ? 'WEB' : 'MOBILE'])
				;
			}
		}

		if ($this->getUserId() > 0)
		{
			$arResult["AUTHOR"] = array(
				"ID" => $this->getUserId(),
				"NAME" => CUser::FormatName(
					$arParams["NAME_TEMPLATE"],
					array(
						"NAME" => $this->getUser()->getFirstName(),
						"LAST_NAME" => $this->getUser()->getLastName(),
						"SECOND_NAME" => $this->getUser()->getSecondName(),
						"LOGIN" => $this->getUser()->getLogin(),
						"NAME_LIST_FORMATTED" => "",
					),
					($arParams["SHOW_LOGIN"] != "N"),
					false),
				"AVATAR" => \CFile::ResizeImageGet(
					$USER->GetParam("PERSONAL_PHOTO"),
					array(
						"width" => $arParams["AVATAR_SIZE"],
						"height" => $arParams["AVATAR_SIZE"]
					),
					BX_RESIZE_IMAGE_EXACT
				)
			);
		}
		else
		{
			$arResult["AUTHOR"] = [
				"ID" => 0,
				"NAME" => "Guest",
				"AVATAR" => ""
			];
		}

		$arResult["NAV_STRING_COUNT_MORE"] = 0;
		if ($arParams["NAV_STRING"] && $arParams["NAV_RESULT"])
		{
			$arResult["NAV_STRING_COUNT_MORE"] =
				$arParams["NAV_RESULT"]->NavRecordCount - (
					$arParams["VISIBLE_RECORDS_COUNT"] > 0
						? $arParams["VISIBLE_RECORDS_COUNT"]
						: $arParams["NAV_RESULT"]->NavPageNomer * $arParams["NAV_RESULT"]->NavPageSize
				);
		}

		if (
			$currentExtranetUser === null
			&& $availableUsersList === null
		)
		{
			$currentExtranetUser = (
				Loader::includeModule('socialnetwork')
				&& !CSocNetUser::isCurrentUserModuleAdmin(SITE_ID, false)
				&& Loader::includeModule('extranet')
				&& !CExtranet::isIntranetUser()
			);

			$availableUsersList = ($currentExtranetUser ? \CExtranet::getMyGroupsUsers(SITE_ID) : array());
		}

		$arResult["currentExtranetUser"] = $currentExtranetUser;
		$arResult["availableUsersList"] = $availableUsersList;
	}

	public function executeComponent()
	{
		if ($this->scope === null)
		{
			if ($this->isAjax() && $this->request->getPost("scope") !== null)
			{
				$this->scope = $this->request->getPost("scope");
			}
			else
			{
				$this->scope = self::STATUS_SCOPE_WEB;
				if (is_callable(array('\Bitrix\MobileApp\Mobile', 'getApiVersion')) && \Bitrix\MobileApp\Mobile::getApiVersion() >= 1 &&
					defined("BX_MOBILE") && BX_MOBILE === true)
					$this->scope = self::STATUS_SCOPE_MOBILE;
			}
		}

		$templateName = $this->getTemplateName();

		if ((empty($templateName) || $templateName == ".default" || $templateName == "bitrix24"))
		{
			if ($this->isWeb())
				$this->setTemplateName(".default");
			else
				$this->setTemplateName("mobile_app");
		}

		try
		{
			$this->prepareParams($this->arParams, $this->arResult);
			ob_start();

			$this->includeComponentTemplate();

			$output = ob_get_clean();
			$json = false;

			foreach (GetModuleEvents('main.post.list', 'OnCommentsDisplayTemplate', true) as $arEvent)
			{
				ExecuteModuleEventEx($arEvent, array(&$output, &$this->arParams, &$this->arResult));
			}
			$this->sendIntoPull($this->arParams, $this->arResult);

			if (
				(!$this->isWeb())
				&& mb_strtolower($this->getMode()) == 'plain'
				&& is_array($this->arParams['RECORDS'])
				&& !empty($this->arParams['RECORDS'])
				&& !empty($this->arParams['IS_POSTS_LIST'])
				&& $this->arParams['IS_POSTS_LIST'] == "N"
				&& Loader::includeModule('socialnetwork')
			)
			{
				$contentEntityType = (
					!empty($this->arParams["RATING_TYPE_ID"])
						? $this->arParams["RATING_TYPE_ID"]
						: (
							!empty($this->arParams["CONTENT_TYPE_ID"])
								? $this->arParams["CONTENT_TYPE_ID"]
								: ''
							)
				);

				foreach($this->arParams['RECORDS'] as $key => $record)
				{
					if (!empty($contentEntityType))
					{
						$provider = \Bitrix\Socialnetwork\Livefeed\Provider::init([
							'ENTITY_TYPE' => $contentEntityType,
							'ENTITY_ID' => $record["ID"],
						]);
						if ($provider)
						{
							$provider->setContentView(array(
								'save' => false
							));
						}
					}
				}
			}

			if ($this->arParams["MODE"] == "PULL_MESSAGE")
			{
				$json = $this->parseHTML($output, "RECORD");
			}
			else if (in_array($this->getMode(), ["RECORD", "RECORDS", "LIST"]))
			{
				$json = $this->parseHTML($output, $this->getMode());
				$this->sendJsonResponse($json);
			}

			$output .= $this->joinToPull();
			return array(
				"HTML" => $output,
				"JSON" => $json,
				"DATA" => $this->arResult
			);
		}
		catch (\Exception $e)
		{
			$this->sendJsonResponse(array(
				"status" => "error",
				"message" => $e->getMessage()
			));
		}
	}

	protected function sendJsonResponse($response)
	{
		$this->getApplication()->restartBuffer();
		while (ob_end_clean());
		header('Content-Type:application/json; charset=UTF-8');
		/** @noinspection PhpUndefinedClassInspection */
		\CMain::finalActions(Json::encode($response));
	}

	private function parseHTML($response, $mode = "RECORD")
	{
		include_once(__DIR__."/html_parser.php");
		$JSResult = array(
			"exemplarId" => $this->exemplarId,
			"operationId" => $this->request->get("OPERATION_ID")
		);
		$FHParser = new MPLSimpleHTMLParser($response);
		$SHParser = new MPLSimpleHTMLParser($this->getApplication()->GetHeadStrings());
		$arParams = &$this->arParams;

		if ($mode == "LIST")
		{
			$messageList = $SHParser->getInnerHTML('<!--LOAD_SCRIPT-->', '<!--END_LOAD_SCRIPT-->').
				$FHParser->getInnerHTML('<!--RCRDLIST_'.$arParams["ENTITY_XML_ID"].'-->', '<!--RCRDLIST_END_'.$arParams["ENTITY_XML_ID"].'-->');

			$messageNavigation = $FHParser->getTagHTML(
					(!$this->isWeb())
						? 'a[class=post-comments-link]'
						: 'a[class=feed-com-all]'
			);

			$JSResult += array(
				'status' => "success",
				'messageList' => $messageList,
				'navigation' => $messageNavigation
			);
		}
		else if ($mode == "RECORD" || $mode == "RECORDS")
		{
			$recordIds = [];
			if ($arParams["RESULT"] > 0)
			{
				$recordIds[] = $arParams["RESULT"];
			}
			elseif (($filter = $this->request->get("FILTER"))
				&& is_array($filter)
				&& array_key_exists("ID", $filter))
			{
				if ($mode == "RECORD")
				{
					$recordIds[] = $filter["ID"];
				}
				else
				{
					$recordIds = $filter["ID"];
				}
			}
			$arParams["RECORDS"] = array_intersect_key($arParams["RECORDS"], $arParams["~RECORDS"], array_flip($recordIds));

			$records = [];
			foreach ($arParams["RECORDS"]  as $recordId => $res)
			{
				$message = $FHParser->getInnerHTML(
					'<!--RCRD_'.$arParams["ENTITY_XML_ID"]."-".$recordId.'-->',
					'<!--RCRD_END_'.$arParams["ENTITY_XML_ID"]."-".$recordId.'-->');
				$res = array_merge($arParams["~RECORDS"][$recordId], $res, ($this->isWeb() ? $res["WEB"] : $res["MOBILE"]));
				unset($res["WEB"]);
				unset($res["MOBILE"]);

				if (isset($res["FILES"]) && is_array($res["FILES"]) && !empty($res["FILES"]) && (
						$this->arParams["RIGHTS"]["EDIT"] == "ALL" ||
						$this->arParams["RIGHTS"]["EDIT"] == "Y" ||
						$this->arParams["RIGHTS"]["EDIT"] == "OWN" && $res["AUTHOR"]["ID"] == $this->getUserId()
					))
				{
					if (!array_key_exists("MFI_UPLOADED_FILES_".$arParams["mfi"], $_SESSION))
					{
						$_SESSION["MFI_UPLOADED_FILES_".$arParams["mfi"]] = [];
					}
					foreach($res["FILES"] as $key => $arFile)
					{
						$_SESSION["MFI_UPLOADED_FILES_".$arParams["mfi"]][] = $key;
						if (CFile::IsImage($arFile["FILE_NAME"], $arFile["CONTENT_TYPE"]))
						{
							$aImgNew = CFile::ResizeImageGet(
								$key,
								array("width" => 90, "height" => 90),
								BX_RESIZE_IMAGE_EXACT,
								true
							);
							$res["FILES"][$key]["THUMBNAIL"] = $aImgNew["src"];
							$aImgNew = CFile::ResizeImageGet(
								$key,
								array("width" => $arParams["IMAGE_SIZE"], "height" => $arParams["IMAGE_SIZE"]),
								BX_RESIZE_IMAGE_PROPORTIONAL,
								true
							);
							$res["FILES"][$key]["SRC"] = $aImgNew["src"];
						}
					}
				}


				if ($res['UF'] && is_array($res['UF']))
				{
					foreach ($res['UF'] as $fieldName => $userField)
					{
						if (
							$userField['USER_TYPE_ID'] === 'disk_file'
							&& isset($userField['VALUE'])
							&& is_array($userField['VALUE'])
							&& Loader::includeModule('disk')
						)
						{
							$res['UF'][$fieldName]['FILES'] = DiskUploaderController::getFileInfo($userField['VALUE']);
						}
					}
				}

				$records[$recordId] = [
					'message' => $SHParser->getInnerHTML('<!--LOAD_SCRIPT-->', '<!--END_LOAD_SCRIPT-->').$message,
					'messageBBCode' => $arParams["~RECORDS"][$recordId]["~POST_MESSAGE_TEXT"],
					'messageId' => array($arParams["ENTITY_XML_ID"], $recordId),
					'messageFields' => $res
				];
			}

			$JSResult += array(
				'warningCode' => ($arParams["WARNING_CODE"] ?? ''),
				'warningMessage' => ($arParams["~WARNING_MESSAGE"] ?? ''),
				'errorMessage' => ($arParams["~ERROR_MESSAGE"] ?? ($arParams["ERROR_MESSAGE"] ?? '')),
				'okMessage' => ($arParams["~OK_MESSAGE"] ?? ($arParams["OK_MESSAGE"] ?? '')),
				'status' => "success",
			);
			if ($mode === "RECORDS")
			{
				$JSResult["messageList"] = $records;
			}
			elseif (!empty($records))
			{
				$JSResult += reset($records);
			}
		}
		return $JSResult;
	}

	protected function getApplication()
	{
		global $APPLICATION;
		return $APPLICATION;
	}

	protected function getUser()
	{
		global $USER;
		return $USER;
	}

	protected function getUserId()
	{
		static $userId = null;
		if (is_null($userId))
		{
			$userId = 0;

			global $USER;
			if (($USER instanceof \CUser) && $USER->IsAuthorized())
			{
				$userId = $USER->GetID();
			}
		}
		return $userId;
	}

	public function getDateTimeFormatted($timestamp, $arFormatParams)
	{
		return \CComponentUtil::getDateTimeFormatted(array(
			'TIMESTAMP' => $timestamp,
			'DATETIME_FORMAT' => ($arFormatParams["DATE_TIME_FORMAT"] ?? false),
			'DATETIME_FORMAT_WITHOUT_YEAR' => ($arFormatParams["DATE_TIME_FORMAT_WITHOUT_YEAR"] ?? false)
		));
	}
}