mirror of
https://github.com/danog/telegram-tt.git
synced 2025-01-22 05:11:55 +01:00
1 line
423 KiB
Plaintext
1 line
423 KiB
Plaintext
{"version":3,"sources":["webpack:///./src/components/ui/ListItem.tsx","webpack:///./src/hooks/useIntersectionObserver.ts","webpack:///./src/hooks/useMedia.ts","webpack:///./src/components/common/PrivateChatInfo.tsx","webpack:///./src/components/ui/Modal.tsx","webpack:///./src/util/trapFocus.ts","webpack:///./src/components/ui/Transition.tsx","webpack:///./src/components/common/Avatar.tsx","webpack:///./src/components/ui/InfiniteScroll.tsx","webpack:///./src/components/ui/FloatingActionButton.tsx","webpack:///./src/hooks/useTransitionForMedia.ts","webpack:///./src/components/common/helpers/mediaDimensions.ts","webpack:///./src/components/common/AnimatedSticker.tsx","webpack:///./src/components/ui/ConfirmDialog.tsx","webpack:///./src/components/common/NothingFound.tsx","webpack:///./src/components/right/hooks/useAsyncRendering.ts","webpack:///./src/hooks/useMediaWithDownloadProgress.ts","webpack:///./src/components/ui/ProgressSpinner.tsx","webpack:///./src/components/common/StickerButton.tsx","webpack:///./src/components/ui/Radio.tsx","webpack:///./src/components/ui/RadioGroup.tsx","webpack:///./src/hooks/useBackgroundMode.ts","webpack:///./src/components/ui/Portal.ts","webpack:///./src/components/common/AnimatedEmoji.tsx","webpack:///./src/components/common/GroupChatInfo.tsx","webpack:///./src/util/fastSmoothScroll.ts","webpack:///./src/hooks/useInfiniteScroll.ts","webpack:///./src/util/cycleRestrict.ts","webpack:///./src/util/focusEditableElement.ts","webpack:///./src/util/captureEvents.ts","webpack:///./src/components/common/VerifiedIcon.tsx","webpack:///./src/hooks/useCurrentOrPrev.ts","webpack:///./src/util/safePlay.ts","webpack:///./src/hooks/useBuffering.ts","webpack:///./src/components/middle/composer/helpers/parseMessageInput.ts","webpack:///./src/components/middle/composer/DropTarget.tsx","webpack:///./src/components/middle/composer/DropArea.tsx","webpack:///./src/hooks/reducers/useFoldersReducer.ts","webpack:///./src/components/common/PickerSelectedItem.tsx","webpack:///./src/util/fastSmoothScrollHorizontal.ts","webpack:///./src/hooks/useHorizontalScroll.ts","webpack:///./src/hooks/useEnsureMessage.ts","webpack:///./src/hooks/useContextMenuHandlers.ts","webpack:///./src/components/common/helpers/renderMessageText.tsx","webpack:///./src/components/common/StickerSetModal.async.tsx","webpack:///./src/hooks/useVideoCleanup.ts","webpack:///./src/hooks/useMouseInside.ts","webpack:///./src/hooks/useReducer.ts","webpack:///./src/components/ui/SearchInput.tsx","webpack:///./src/util/resetScroll.ts","webpack:///./src/components/common/TypingStatus.tsx","webpack:///./src/components/common/DeleteChatModal.tsx","webpack:///./src/components/ui/ShowTransition.tsx","webpack:///./src/util/patchSafariProgressiveAudio.ts","webpack:///./src/hooks/useLayoutEffectWithPrevDeps.ts","webpack:///./src/components/common/Document.tsx","webpack:///./src/hooks/useCanvasBlur.ts","webpack:///./src/lib/fastBlur.js","webpack:///./src/components/middle/composer/TextFormatter.tsx","webpack:///./src/util/clipboard.ts","webpack:///./src/components/common/Media.tsx","webpack:///./src/components/common/Picker.tsx","webpack:///./src/components/common/Audio.tsx","webpack:///./src/util/waveform.ts","webpack:///./src/components/common/helpers/waveform.ts","webpack:///./src/components/ui/Tab.tsx","webpack:///./src/components/ui/TabList.tsx","webpack:///./src/util/getReadableErrorText.ts","webpack:///./src/util/cssAnimationEndListeners.ts","webpack:///./src/hooks/useChatContextActions.ts","webpack:///./src/components/common/LastMessageMeta.tsx","webpack:///./src/components/common/MessageOutgoingStatus.tsx","webpack:///./src/hooks/useContextMenuPosition.ts","webpack:///./src/hooks/usePrevDuringAnimation.ts","webpack:///./src/hooks/useAudioPlayer.ts","webpack:///./src/util/audioPlayer.ts","webpack:///./src/util/download.ts","webpack:///./src/components/common/File.tsx","webpack:///./src/hooks/useMediaDownload.ts","webpack:///./src/components/ui/Notification.tsx","webpack:///./src/components/middle/composer/helpers/selection.ts","webpack:///./src/components/middle/composer/hooks/useEmojiTooltip.ts","webpack:///./src/components/middle/composer/EmojiTooltip.async.tsx","webpack:///./src/components/middle/composer/MessageInput.tsx","webpack:///./src/components/common/WebLink.tsx","webpack:///./src/components/left/main/hooks/useChatAnimationType.ts","webpack:///./src/components/left/main/Badge.tsx","webpack:///./src/components/left/main/Chat.tsx","webpack:///./src/components/left/main/EmptyFolder.tsx","webpack:///./src/components/left/main/ChatList.tsx","webpack:///./src/components/middle/composer/helpers/searchUserName.ts","webpack:///./src/components/middle/composer/hooks/useMentionTooltip.ts","webpack:///./src/hooks/useBlur.ts","webpack:///./src/hooks/useBlurSync.ts"],"names":["ListItem","props","ref","buttonRef","icon","className","style","children","disabled","ripple","narrow","inactive","focus","destructive","multiline","isStatic","contextActions","onClick","containerRef","useRef","isTouched","markIsTouched","unmarkIsTouched","useFlag","isContextMenuOpen","contextMenuPosition","handleBeforeContextMenu","handleContextMenu","handleContextMenuClose","handleContextMenuHide","useContextMenuHandlers","getTriggerElement","useCallback","current","getRootElement","closest","getMenuElement","querySelector","positionX","positionY","menuStyle","useContextMenuPosition","handleClick","e","IS_TOUCH_ENV","fastRaf","handleMouseDown","button","lang","useLang","fullClassName","buildClassName","dir","isRtl","undefined","role","tabIndex","onMouseDown","onContextMenu","isOpen","autoClose","onClose","onCloseAnimationEnd","map","action","key","title","handler","useIntersectionObserver","rootRef","throttleMs","debounceMs","shouldSkipFirst","margin","threshold","isDisabled","rootCallback","controllerRef","rootCallbackRef","freezeFlagsRef","onUnfreezeRef","freeze","unfreeze","useHeavyAnimationCheck","useEffect","observer","disconnect","callbacks","clear","observe","target","targetCallback","Map","entriesAccumulator","observerCallbackSync","entries","Array","from","values","forEach","entry","callback","get","scheduler","throttle","debounce","observerCallback","IntersectionObserver","set","root","rootMargin","initController","controller","delete","unobserve","useOnIntersect","targetRef","useIsIntersecting","isIntersecting","setIsIntersecting","useState","mediaHash","noLoad","mediaFormat","ApiMediaFormat","BlobUrl","cacheBuster","delay","mediaData","mediaLoader","forceUpdate","useForceUpdate","startedAt","Date","now","then","spentTime","setTimeout","memo","withGlobal","global","userId","forceShowSelf","lastSyncTime","serverTimeOffset","user","selectUser","isSavedMessages","isSelf","areMessagesLoaded","Boolean","selectChatMessages","setGlobal","actions","pick","typingStatus","avatarSize","status","withMediaViewer","withUsername","withFullInfo","withUpdatingStatus","noStatusOrTyping","noRtl","loadFullUser","openMediaViewer","id","fullName","getUserFullName","handleAvatarViewerOpen","hasPhoto","stopPropagation","avatarOwnerId","origin","MediaViewerOrigin","ProfileAvatar","MiddleHeaderAvatar","size","renderText","isVerified","isUserOnline","username","getUserStatus","Modal","header","hasCloseButton","noBackdrop","onEnter","shouldSkipHistoryAnimations","shouldRender","transitionClassNames","useShowTransition","modalRef","captureKeyboardListeners","onEsc","element","handleKeyDown","preventDefault","focusableElements","querySelectorAll","length","currentFocusedIndex","findIndex","em","isSameNode","document","activeElement","newFocusedIndex","shiftKey","addEventListener","removeEventListener","trapFocus","forceClose","useHistoryBack","useEffectWithPrevDeps","prevIsOpen","body","classList","toggle","dispatchHeavyAnimationEvent","remove","Portal","Button","round","color","ariaLabel","CLEANED_UP","Symbol","Transition","activeKey","name","direction","renderCount","shouldRestoreHeight","shouldCleanup","cleanupExceptionKey","onStart","onStop","animationLevel","getGlobal","settings","byKey","rendersRef","prevActiveKey","usePrevious","activeKeyChanged","useLayoutEffect","cleanup","container","childElements","add","childNodes","isBackwards","keys","Object","Number","prevActiveIndex","indexOf","activeIndex","node","i","HTMLElement","dispatchHeavyAnimationStop","requestAnimationFrame","onAnimationEnd","height","clientHeight","toNode","firstChild","waitForAnimationEnd","flexBasis","renders","contents","fill","render","chat","text","withOnlineStatus","isDeleted","isDeletedUser","imageHash","getChatAvatarHash","dataUri","useMedia","DataUri","shouldRenderFullMedia","useTransitionForMedia","content","src","alt","decoding","userFullName","getFirstLetters","getChatTitle","isChatPrivate","isOnline","getUserColorKey","senderId","data-test-sender-id","IS_TEST","InfiniteScroll","onLoadMore","onScroll","onKeyDown","items","itemSelector","preloadBackwards","sensitiveArea","noScrollRestore","noScrollRestoreOnTop","noFastList","stateRef","loadMoreBackwards","loadMoreForwards","useMemo","noScroll","LoadMoreDirection","Backwards","Forwards","scrollHeight","state","newScrollTop","listItemElements","currentAnchor","includes","scrollTop","getBoundingClientRect","top","currentAnchorTop","nextAnchor","resetScroll","isScrollTopJustUpdated","handleScroll","listLength","offsetHeight","isNearTop","offsetTop","isNearBottom","isUpdated","nextAnchorTop","newAnchorTop","offsetParent","teactFastList","FloatingActionButton","isShown","buttonClassName","SPEED","fast","slow","speed","noAnimate","isMediaLoaded","willAnimate","shouldRenderThumb","setShouldRenderThumb","MEDIA_VIEWER_MEDIA_QUERY","REM","parseInt","getComputedStyle","documentElement","fontSize","ROUND_VIDEO_DIMENSIONS","AVATAR_FULL_DIMENSIONS","width","LIKE_STICKER_ID","DEFAULT_MEDIA_DIMENSIONS","cachedMaxWidthOwn","cachedMaxWidth","cachedMaxWidthNoAvatar","getAvailableWidth","fromOwnMessage","isForwarded","isWebPagePhoto","noAvatars","extraPaddingRem","regularMaxWidth","IS_SINGLE_COLUMN_LAYOUT","windowWidth","windowSize","Math","min","getMaxMessageWidthRem","calculateDimensionsForMessageMedia","isGif","aspectRatio","calculateDimensions","getAvailableHeight","getMediaViewerAvailableDimensions","withFooter","isVideo","mql","window","matchMedia","windowHeight","occupiedHeightRem","matches","calculateInlineImageDimensions","photo","getPhotoInlineDimensions","calculateVideoDimensions","video","getVideoDimensions","getPictogramDimensions","getDocumentThumbnailDimensions","smaller","getStickerDimensions","sticker","baseWidth","STICKER_SIZE_INLINE_MOBILE_FACTOR","STICKER_SIZE_INLINE_DESKTOP_FACTOR","calculatedHeight","calculateMediaViewerDimensions","availableWidth","availableHeight","mediaWidth","mediaHeight","calculatedWidth","lottiePromise","RLottie","async","ensureLottie","default","animationData","play","playSegment","noLoop","quality","isLowPriority","onLoad","animation","setAnimation","wasPlaying","isFrozen","playRef","playSegmentRef","exec","newAnimation","setSpeed","destroy","playAnimation","shouldRestart","goToAndPlay","pauseAnimation","pause","freezeAnimation","isPlaying","unfreezeAnimation","unfreezeAnimationOnRaf","useBackgroundMode","textParts","confirmLabel","confirmHandler","confirmIsDestructive","isButtonsInOneRow","split","textPart","isText","description","useAsyncRendering","dependencies","shouldRenderRef","timeoutRef","useOnChange","clearTimeout","isStreaming","Stream","IS_PROGRESSIVE_SUPPORTED","Progressive","downloadProgress","setDownloadProgress","startedAtRef","handleProgress","progress","RADIUSES","s","m","l","xl","square","transparent","noCross","radius","circleRadius","STROKE_WIDTH","borderRadius","circumference","PI","svg","firstElementChild","strokeDashOffset","max","setAttribute","toString","innerHTML","observeIntersection","clickArg","onUnfaveClick","localMediaHash","stickerSelector","thumbDataUri","thumbnail","previewBlobUrl","shouldPlay","lottieData","isAnimated","Lottie","isAnimationLoaded","markLoaded","unmarkLoaded","canAnimatedPlay","shouldRenderPreview","previewTransitionClassNames","emoji","data-sticker-id","label","subLabel","value","checked","hidden","isLoading","onChange","type","Spinner","options","selected","loadingOption","handleChange","event","currentTarget","option","onBlur","onFocus","hasFocus","containerId","elementRef","createElement","appendChild","TeactDOM","removeChild","WIDTH","large","medium","small","AnimatedEmoji","forceLoadPreview","markAnimationLoaded","playKey","setPlayKey","String","random","chatId","selectChat","onlineCount","selectChatOnlineCount","withChatType","loadFullChat","isSuperGroup","isChatSuperGroup","isMin","isRestricted","getChatTypeString","handle","groupStatus","chatTypeString","membersCount","getGroupStatus","onlineStatus","renderStatusOrTyping","isAnimating","fastSmoothScroll","position","maxDistance","FAST_SMOOTH_MAX_DISTANCE","forceDirection","forceDuration","forceCurrentContainerHeight","scrollFrom","elementTop","offset","FocusDirection","Up","Down","calculateScrollFrom","Static","ANIMATION_LEVEL_MIN","scrollWithJs","isAnimatingScroll","elementHeight","currentScrollTop","containerHeight","targetContainerHeight","dataset","normalHeight","path","remainingPath","absPath","abs","transition","FAST_SMOOTH_SHORT_TRANSITION_MAX_DISTANCE","shortTransition","longTransition","duration","FAST_SMOOTH_MIN_DURATION","FAST_SMOOTH_MAX_DURATION","startAt","onHeavyAnimationStop","animateSingle","t","currentPath","getViewportSlice","sourceIds","offsetId","listSlice","index","indexForDirection","to","newViewportIds","slice","areSomeLocal","areAllLocal","listIds","forceFullPreload","lastParamsRef","viewportIdsRef","prevListIds","prevIsDisabled","areSortedArraysEqual","viewportIds","getMore","cycleRestrict","floor","focusEditableElement","force","selection","getSelection","range","createRange","lastChild","nodeValue","selectNodeContents","collapse","removeAllRanges","addRange","SwipeDirection","captureEvents","captureEvent","currentSwipeAxis","hasMoved","onCapture","excludedClosestSelector","onMove","onRelease","pageX","touches","pageY","withCursor","dragOffsetX","dragOffsetY","onDrag","onSwipe","xAbs","yAbs","Left","Right","processSwipe","passive","VerifiedIcon","useCurrentOrPrev","shouldSkipUndefined","shouldForceCurrent","prev","mediaEl","catch","err","DEBUG","console","warn","noInitiallyBuffered","isBuffered","setIsBuffered","bufferedProgress","setBufferedProgress","setIsBufferedDebounced","handleBuffering","media","isSafariPatchInProgress","buffered","end","readyState","currentTime","bufferingHandlers","onLoadedData","onPlaying","onLoadStart","onPause","onTimeUpdate","onProgress","checkBuffering","ENTITY_CLASS_BY_NODE_NAME","B","ApiMessageEntityTypes","Bold","STRONG","I","Italic","EM","U","Underline","S","Strike","STRIKE","DEL","CODE","Code","PRE","Pre","BLOCKQUOTE","Blockquote","parseMessageInput","html","fragment","parsedHtml","IS_EMOJI_SUPPORTED","replace","parseMarkdown","innerText","trim","textIndex","recursionDeepness","entities","addEntity","entity","rawText","nodeName","anchor","entityType","MentionName","Url","href","startsWith","Email","Phone","textContent","TextUrl","getEntityTypeFromNode","rawIndex","substring","url","getEntityDataFromNode","push","hasChildNodes","isQuick","onFileSelect","isHovered","markHovered","unmarkHovered","onDrop","onDragEnter","onDragLeave","relatedTarget","toTarget","DropAreaState","withQuick","onHide","hideTimeoutRef","prevWithQuick","captureEscKeyListener","handleFilesDrop","dataTransfer","dt","files","handleQuickFilesDrop","handleDragLeave","fromTarget","onDragOver","INCLUDED_CHAT_TYPES","EXCLUDED_CHAT_TYPES","INCLUDE_FILTER_FIELDS","EXCLUDE_FILTER_FIELDS","selectChatFilters","mode","selectTemp","selectedChatIds","selectedChatTypes","includedChatIds","includeFilters","folder","filter","excludedChatIds","excludeFilters","getSuggestedFolderName","filters","bots","groups","channels","contacts","nonContacts","INITIAL_STATE","chatFilter","foldersReducer","payload","omit","folderId","error","useReducer","chatOrUserId","isMinimized","canClose","iconElement","titleText","getUserFirstOrLastName","fastSmoothScrollHorizontal","left","IS_IOS","scrollLeft","offsetWidth","containerWidth","scrollWidth","animate","scrollTo","behavior","deltaX","deltaY","messageId","message","replyOriginForId","loadMessage","getDispatch","loadMessageThrottled","throttled","contextMenuCounter","isMenuDisabled","shouldDisableOnLink","shouldDisableOnLongTap","setIsContextMenuOpen","setContextMenuPosition","x","clientX","y","clientY","timer","clearLongPressTimer","startLongPressTimer","contains","originalEvent","cancelClickOnce","stopImmediatePropagation","emulateContextMenuEvent","renderMessageText","highlight","shouldRenderHqEmoji","formattedText","getMessageText","renderTextWithEntities","organizeEntities","organizedEntityIndexes","Set","organizedEntities","has","organizedEntity","organizeEntity","organizedIndexes","nestedEntities","parsedEntity","isChanged","organizedIndex","shouldRenderAsHtml","renderMessagePart","result","deleteLineBreakAfterPre","renderEntity","textPartStart","textPartEnd","isLastEntity","renderResult","textBefore","textBeforeLength","substr","entityStartIndex","entityEndIndex","entityContent","nestedEntityContent","nestedIndex","nestedEntity","nestedEntityIndex","nestedResult","nestedEntityEndIndex","newEntity","rawEntityText","renderedContent","join","getLinkUrl","processEntityAsHtml","entityText","BotCommand","handleBotCommandClick","Hashtag","Cashtag","handleHashtagClick","rel","Mention","processEntity","isArray","textAfter","arrayIndex","c","emojiFilter","sendBotCommand","command","setLocalTextSearchQuery","query","searchTextMessagesLocal","StickerSetModal","useModuleLoader","Bundles","Extra","useVideoCleanup","videoRef","videoEl","load","closeTimeout","useMouseInside","menuCloseTimeout","isMouseInside","reducer","initialState","reducerRef","setState","currentState","parentContainerClassName","inputId","focused","placeholder","autoComplete","onReset","inputRef","isInputFocused","markInputFocused","unmarkInputFocused","blur","overflow","typingUser","typingUserName","isPrivateChat","isChatWithSelf","selectIsChatWithSelf","getPrivateChatUserId","isBot","isUserBot","isSupport","canDeleteForAll","contactName","isChannel","isChatChannel","isBasicGroup","isChatBasicGroup","currentUserId","leaveChannel","deleteHistory","deleteChannel","deleteChatUser","blockContact","chatTitle","handleDeleteMessageForAll","shouldDeleteForAll","handleDeleteAndStop","contactId","accessHash","handleDeleteChat","isCreator","ShowTransition","isCustom","prevChildren","fromChildrenRef","patchSafariProgressiveAudio","audioEl","patchedForSafari","patchForSafariInProgress","paused","once","cb","debugKey","prevDeps","uploadProgress","withDate","datetime","sender","isSelected","isSelectable","onCancelUpload","onMediaClick","onDateClick","extension","getDocumentExtension","fileName","timestamp","mediaType","isDownloadAllowed","setIsDownloadAllowed","useMediaWithDownloadProgress","getMessageMediaHash","isUploading","isTransferring","transferProgress","getMediaTransferState","hasPreview","getDocumentHasPreview","getMessageMediaThumbDataUri","localBlobUrl","previewData","isAllowed","handleDateClick","download","thumbnailDataUri","actionIcon","isMessageDocumentVideo","useCanvasBlur","withRaf","canvasRef","canvas","img","Image","processBlur","ctx","getContext","alpha","IS_CANVAS_FILTER_SUPPORTED","drawImage","RADIUS","fastBlur","onload","mul_table","shg_table","boxBlurCanvasRGB","context","top_x","top_y","iterations","isNaN","imageData","getImageData","pixels","data","rsum","gsum","bsum","p","p1","p2","yp","yi","yw","wm","hm","rad1","r","g","b","mul_sum","shg_sum","vmin","vmax","putImageData","TEXT_FORMAT_BY_TAG_NAME","fragmentEl","anchorPosition","selectedRange","linkUrlInputRef","isLinkControlOpen","openLinkControl","closeLinkControl","linkUrl","setLinkUrl","isEditingLink","setIsEditingLink","inputClassName","setInputClassName","selectedTextFormats","setSelectedTextFormats","useVirtualBackdrop","selectedFormats","parentElement","commonAncestorContainer","EDITABLE_INPUT_ID","textFormat","tagName","getSelectedText","getSelectedElement","updateInputStyles","input","getFormatButtonClassName","some","fKey","monospace","strikethrough","handleBoldText","execCommand","bold","handleItalicText","italic","handleUnderlineText","underline","handleStrikethroughText","replaceWith","handleMonospaceText","handleLinkUrlConfirm","formattedLinkUrl","encodeURI","restoreSelection","KeyK","KeyB","KeyU","KeyI","KeyM","KeyS","code","altKey","ctrlKey","metaKey","linkUrlConfirmClassName","inputMode","CLIPBOARD_ITEM_SUPPORTED","navigator","clipboard","ClipboardItem","textCopyEl","copyTextToClipboard","str","rangeToRestore","rangeCount","getRangeAt","select","copyImageToClipboard","imageUrl","imageEl","toBlob","copyBlobToClipboard","pngBlob","write","idPrefix","mediaBlobUrl","getMessageVideo","formatMediaDuration","itemIds","selectedIds","filterValue","filterPlaceholder","notFoundText","searchInputId","onSelectedIdsChange","onFilterChange","shouldMinimize","handleItemClick","newSelectedIds","splice","handleFilterChange","useInfiniteScroll","MIN_SPIKES","MAX_SPIKES","renderSeekline","playProgress","handleStartSeek","handleSeek","handleStopSeek","onMouseMove","onMouseUp","theme","senderTitle","date","onPlay","onReadMedia","audio","voice","isMediaUnread","isVoice","isSeeking","isActivated","setIsActivated","shouldDownload","getMessageMediaFormat","useBuffering","playPause","setCurrentTime","useAudioPlayer","getMessageKey","getMediaDuration","isMessageLocal","isDownloadStarted","directDownloadProgress","handleDownloadClick","useMediaDownload","getMessageAudioCaption","isLoadingForPlaying","shouldRenderSpinner","spinnerClassNames","handleButtonClick","seekBar","seekHandlers","isOwn","isOwnMessage","renderedWaveform","waveform","fillColor","fillOwnColor","progressFillColor","progressFillOwnColor","durationFactor","spikesCount","decodedWaveform","encoded5bit","bitsCount","valuesCount","bitsData","byteIndex","bitShift","lastByteIndex","lastBitShift","lastValue","decodeWaveform","Uint8Array","spikes","peak","fitCount","newData","springFactor","leftFiller","rightFiller","idx","val","interpolateArray","fillStyle","progressFillStyle","scale","item","globalAlpha","spikeHeight","beginPath","moveTo","arcTo","closePath","roundedRectangle","toDataURL","renderWaveformToDataUri","draggable","renderWaveform","buttonClassNames","showSeekline","contentClassName","ProgressSpinner","getFirstLine","Link","formatPastTimeShort","performer","getSecondLine","formatMediaDateTime","renderAudio","renderVoice","active","badgeCount","isBadgeActive","previousActiveTab","tabRef","tab","indicator","prevTab","currentIndicator","shiftLeft","offsetLeft","scaleFactor","clientWidth","transform","tabs","activeTab","big","onSwitchTab","useHorizontalScroll","activeTabElement","newLeft","READABLE_ERROR_MESSAGES","CHAT_RESTRICTED","CHAT_WRITE_FORBIDDEN","CHAT_SEND_POLL_FORBIDDEN","CHAT_SEND_STICKERS_FORBIDDEN","CHAT_SEND_GIFS_FORBIDDEN","CHAT_SEND_MEDIA_FORBIDDEN","CHAT_LINK_EXISTS","SLOWMODE_WAIT_X","USER_BANNED_IN_CHANNEL","USER_IS_BLOCKED","YOU_BLOCKED_USER","IMAGE_PROCESS_FAILED","MEDIA_EMPTY","MEDIA_INVALID","PHOTO_EXT_INVALID","PHOTO_INVALID_DIMENSIONS","PHOTO_SAVE_FILE_INVALID","MESSAGE_DELETE_FORBIDDEN","MESSAGE_POLL_CLOSED","MESSAGE_EDIT_TIME_EXPIRED","CHAT_ADMIN_REQUIRED","PINNED_DIALOGS_TOO_MUCH","DIALOG_FILTERS_TOO_MUCH","CHANNEL_PRIVATE","MEDIA_CAPTION_TOO_LONG","ADDRESS_STREET_LINE1_INVALID","ADDRESS_STREET_LINE2_INVALID","ADDRESS_CITY_INVALID","ADDRESS_COUNTRY_INVALID","ADDRESS_POSTCODE_INVALID","ADDRESS_STATE_INVALID","REQ_INFO_NAME_INVALID","REQ_INFO_PHONE_INVALID","REQ_INFO_EMAIL_INVALID","LINK_NOT_MODIFIED","SERVICE_WORKER_DISABLED","CAPTION_TOO_LONG_PLEASE_REMOVE_CHARACTERS","FRESH_RESET_AUTHORISATION_FORBIDDEN","BOTS_TOO_MUCH","BOT_GROUPS_BLOCKED","USERS_TOO_MUCH","USER_CHANNELS_TOO_MUCH","USER_KICKED","USER_NOT_MUTUAL_CONTACT","USER_PRIVACY_RESTRICTED","INVITE_HASH_EMPTY","INVITE_HASH_EXPIRED","INVITE_HASH_INVALID","CHANNELS_TOO_MUCH","USER_ALREADY_PARTICIPANT","SCHEDULE_DATE_INVALID","WALLPAPER_DIMENSIONS_INVALID","getReadableErrorText","isSlowMode","textParams","extraPartIndex","errorMessage","reduce","acc","waitForTransitionEnd","propertyName","waitForEndEvent","animationName","eventType","detailedName","isHandled","handleAnimationEnd","TransitionEvent","AnimationEvent","privateChatUser","handleDelete","isPinned","isMuted","toggleChatPinned","updateChatMutedState","toggleChatArchived","toggleChatUnread","actionUnreadMark","unreadCount","hasUnreadMark","actionPin","actionMute","actionArchive","isChatArchived","getCanDeleteChat","outgoingStatus","Keys","extraPaddingX","extraTopPadding","setPositionX","setPositionY","setStyle","triggerEl","emptyRect","menuEl","rootEl","triggerRect","menuRect","rootRect","horizontalPostition","usePrevDuringAnimation","isCurrentPresent","isPrevPresent","trackId","originalDuration","handlers","onInit","onForcePlay","noPlaylist","setIsPlaying","isPlayingSync","setPlayProgress","register","eventName","proxy","isFinite","prevShouldPlay","prevSrc","audioProxy","tracks","currentTrackId","queue","stopCurrentAudio","currentTrack","Audio","handleEvent","nextTrackId","openAudioPlayer","parseMessageKey","nextTrack","safePlay","Proxy","createAudio","preload","IS_SAFARI","time","shouldRemoveFromQueue","track","h","filename","link","click","shouldSpinnerRender","getColorFromExtension","sizeString","getFileSizeString","setIsDownloadStarted","onDismiss","setIsOpen","timerRef","closeAndDismiss","ANIMATION_END_DELAY","handleMouseEnter","handleMouseLeave","onMouseEnter","onMouseLeave","isSelectionInsideInput","selectionRange","parentNode","emojiDataPromise","emojiRawData","emojiData","RE_EMOJI_SEARCH","RE_BR","RE_SPACE","RE_CLEAN_HTML","RegExp","useEmojiTooltip","recentEmojiIds","onUpdateHtml","baseEmojiKeywords","emojiKeywords","markIsOpen","unmarkIsOpen","byId","setById","keywords","setKeywords","byKeyword","setByKeyword","names","setNames","byName","setByName","shouldForceInsertEmoji","setShouldForceInsertEmoji","filteredEmojis","setFilteredEmojis","MEMO_EMPTY_ARRAY","recentEmojis","pickTruthy","emojis","uncompressEmoji","ensureEmojiData","byNative","buildCollectionByKey","baseEmojisByKeyword","mapValues","natives","emojisByKeyword","emojisByName","match","getEmojiCode","forceSend","endsWith","matched","matchedKeywords","keyword","sort","concat","flatten","matchedNames","unique","insertEmoji","textEmoji","isForce","atIndex","lastIndexOf","messageInput","getElementById","native","isEmojiTooltipOpen","closeEmojiTooltip","EmojiTooltip","MAX_INPUT_HEIGHT","currentChatId","threadId","selectCurrentMessageList","messageSendKeyCombo","replyingToId","selectReplyingToId","noTabCapture","isPollModalOpen","payment","isPaymentModalOpen","isAttachmentModalInput","editableInputId","forcedPlaceholder","shouldSetFocus","shouldSuppressFocus","shouldSuppressTextFormatter","onUpdate","onSuppressedFocus","onSend","editLastMessage","replyToNextMessage","cloneRef","isContextMenuOpenRef","isTextFormatterOpen","openTextFormatter","closeTextFormatter","textFormatterAnchorPosition","setTextFormatterAnchorPosition","setSelectedRange","updateInputHeight","useLayoutEffectWithPrevDeps","prevHtml","focusInput","handleCloseTextFormatter","empty","clearSelection","checkSelection","selectedText","parseEmojiOnlyString","START_TO_END","selectionRect","inputRect","willSend","clone","currentHeight","newHeight","transitionDuration","log","captureFirstTab","onTab","suppressFocus","contentEditable","targetIndexDelta","handleKeyUp","IS_ANDROID","closeContextMenuMouseListener","closeContextMenuKeyListener","handleMouseUp","onTouchCancel","onMessageClick","linkData","getMessageWebPage","getFirstLinkInMessage","domain","messageText","getMessageSummaryText","siteName","handleMessageClick","displayUrl","truncatedDescription","trimText","data-initial","ChatAnimationTypes","renderContent","unreadMentionsCount","formatIntegerCompact","lastMessage","replyToMessageId","isOutgoing","lastMessageSender","lastMessageAction","getMessageAction","actionTargetMessage","selectChatMessage","targetUserIds","actionTargetUserIds","targetChatId","actionTargetChatId","privateChatUserId","usersById","users","currentThreadId","messageListType","MAIN_THREAD_ID","selectIsChatMuted","selectNotifySettings","selectNotifyExceptions","draft","selectDraft","canScrollDown","lastMessageOutgoingStatus","selectOutgoingStatus","orderDiff","animationType","openChat","focusLastMessage","isDeleteModalOpen","openDeleteModal","closeDeleteModal","shouldRenderDeleteModal","markRenderDeleteModal","unmarkRenderDeleteModal","isAction","isActionMessage","useEnsureMessage","mediaThumbnail","getMessageSticker","isRoundVideo","getMessageRoundVideo","actionTargetUsers","Opacity","opacity","Move","shouldReplaceHistory","useChatContextActions","Avatar","LastMessageMeta","TypingStatus","actionOrigin","renderActionMessageText","asPlain","senderName","getMessageSenderName","blobUrl","renderMessageSummary","renderLastMessageOrTyping","DeleteChatModal","FolderTypeToListType","folderType","chatFolder","selectChatFolder","animatedEmoji","selectAnimatedEmoji","foldersDispatch","onScreenSelect","handleEditFolder","SettingsScreens","FoldersEditFolderFromChatList","fluid","pill","chats","chatsById","orderedPinnedIds","listType","notifySettings","notifyExceptions","isActive","loadMoreChats","preloadTopChatMessages","openNextChat","currentListIds","currentPinnedIds","prepareFolderListIds","orderById","orderedIds","newChatArrays","prepareChatList","singleList","pinnedChats","otherChats","newOrderedIds","prevOrderById","orderDiffById","order","Infinity","loadMoreOfType","CHAT_LIST_SLICE","ALL_CHATS_PRELOAD_DISABLED","chatArrays","getAnimationType","movesUp","movesDown","orderDiffIds","numberOfUp","numberOfDown","None","useChatAnimationType","IS_PWA","IS_MAC_OS","digit","viewportOffset","pinnedOffset","CHAT_HEIGHT_PX","teactOrderKey","getChatOrder","renderChats","Loading","searchUserName","usernameLowered","toLowerCase","fullNameLowered","filterLowered","searchWords","runThrottled","RE_USERNAME_SEARCH","useMentionTooltip","canSuggestMembers","groupChatMembers","topInlineBotIds","currentFilter","setCurrentFilter","usersToMention","setUsersToMention","topInlineBots","getFilteredUsers","withInlineBots","inlineBots","inlineBot","chatMembers","usernameFilter","getUsernameFilter","canSuggestInlineBots","insertMention","forceFocus","insertedHtml","isMentionTooltipOpen","mentionFilter","closeMentionTooltip","mentionFilteredUsers","cache","useBlurSync","blurredRef","imgToCanvas","blurredAsync","blurredDataUri","useBlur"],"mappings":"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AK,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,IAAK,IAAK,IAAK,GAAI,GAAI,GAAI,GAAI,GAAI,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,GAAI,IAAK,GAAI,IAAK,EAAG,GAAI,IAAK,GAAI,GAAI,IAAK,IAAK,IAAK,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,IAAK,IAAK,IAAK,IAAK,IAAK,GAAI,GAAI,IAAK,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,GAAI,IAAK,IAAK,GAAI,GAAI,IAAK,IAAK,IAAK,GAAI,GAAI,GAAI,GAAI,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,GAAI,GAAI,GAAI,GAAI,GAAI,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,IAAK,GAAI,GAAI,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,GAAI,IAAK,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,GAAI,IAAK,GAAI,GAAI,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,IAAK,GAAI,GAAI,IAAK,IAAK,GAAI,IAAK,IAAK,EAAG,GAAI,IAAK,GAAI,IAAK,IAAK,GAAI,GAAI,IAAK,GAAI,IAAK,GAAI,IAAK,IAAK,GAAI,IAAK,IAAK,IAAK,IAAK,GAAI,IAAK,IAAK,IAAK,GAAI,IAAK,GAAI,IAAK,GAE5qCC,EAAY,CAAC,EAAG,EAAG,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI,GAAI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","file":"0.5b30c68c9d873a9c7445.js","sourcesContent":["import { RefObject } from 'react';\nimport React, { FC, useRef, useCallback } from '../../lib/teact/teact';\n\nimport { IS_TOUCH_ENV } from '../../util/environment';\nimport { fastRaf } from '../../util/schedulers';\nimport buildClassName from '../../util/buildClassName';\nimport useContextMenuHandlers from '../../hooks/useContextMenuHandlers';\nimport useContextMenuPosition from '../../hooks/useContextMenuPosition';\nimport useFlag from '../../hooks/useFlag';\nimport useLang from '../../hooks/useLang';\n\nimport RippleEffect from './RippleEffect';\nimport Menu from './Menu';\nimport MenuItem from './MenuItem';\n\nimport './ListItem.scss';\n\ntype OnClickHandler = (e: React.MouseEvent<HTMLDivElement>) => void;\n\ntype MenuItemContextAction = {\n title: string;\n icon: string;\n destructive?: boolean;\n handler?: () => void;\n};\n\ntype OwnProps = {\n ref?: RefObject<HTMLDivElement>;\n buttonRef?: RefObject<HTMLDivElement>;\n icon?: string;\n className?: string;\n style?: string;\n children: any;\n disabled?: boolean;\n ripple?: boolean;\n narrow?: boolean;\n inactive?: boolean;\n focus?: boolean;\n destructive?: boolean;\n multiline?: boolean;\n isStatic?: boolean;\n contextActions?: MenuItemContextAction[];\n onClick?: OnClickHandler;\n};\n\nconst ListItem: FC<OwnProps> = (props) => {\n const {\n ref,\n buttonRef,\n icon,\n className,\n style,\n children,\n disabled,\n ripple,\n narrow,\n inactive,\n focus,\n destructive,\n multiline,\n isStatic,\n contextActions,\n onClick,\n } = props;\n\n // eslint-disable-next-line no-null/no-null\n let containerRef = useRef<HTMLDivElement>(null);\n if (ref) {\n containerRef = ref;\n }\n const [isTouched, markIsTouched, unmarkIsTouched] = useFlag();\n\n const {\n isContextMenuOpen, contextMenuPosition,\n handleBeforeContextMenu, handleContextMenu,\n handleContextMenuClose, handleContextMenuHide,\n } = useContextMenuHandlers(containerRef, !contextActions);\n\n const getTriggerElement = useCallback(() => containerRef.current, []);\n\n const getRootElement = useCallback(\n () => containerRef.current!.closest('.custom-scroll'),\n [],\n );\n\n const getMenuElement = useCallback(\n () => containerRef.current!.querySelector('.ListItem-context-menu .bubble'),\n [],\n );\n\n const { positionX, positionY, style: menuStyle } = useContextMenuPosition(\n contextMenuPosition,\n getTriggerElement,\n getRootElement,\n getMenuElement,\n );\n\n const handleClick = useCallback((e: React.MouseEvent<HTMLDivElement, MouseEvent>) => {\n if (disabled || !onClick) {\n return;\n }\n onClick(e);\n\n if (IS_TOUCH_ENV && !ripple) {\n markIsTouched();\n fastRaf(unmarkIsTouched);\n }\n }, [disabled, markIsTouched, onClick, ripple, unmarkIsTouched]);\n\n const handleMouseDown = useCallback((e: React.MouseEvent<HTMLDivElement, MouseEvent>) => {\n if (inactive || IS_TOUCH_ENV) {\n return;\n }\n if (contextActions && (e.button === 2 || !onClick)) {\n handleBeforeContextMenu(e);\n }\n if (e.button === 0) {\n if (!onClick) {\n handleContextMenu(e);\n } else {\n handleClick(e);\n }\n }\n }, [inactive, contextActions, onClick, handleBeforeContextMenu, handleContextMenu, handleClick]);\n\n const lang = useLang();\n\n const fullClassName = buildClassName(\n 'ListItem',\n className,\n !isStatic && 'no-selection',\n ripple && 'has-ripple',\n narrow && 'narrow',\n disabled && 'disabled',\n inactive && 'inactive',\n contextMenuPosition && 'has-menu-open',\n focus && 'focus',\n destructive && 'destructive',\n multiline && 'multiline',\n isStatic && 'is-static',\n );\n\n return (\n <div\n ref={containerRef}\n className={fullClassName}\n dir={lang.isRtl ? 'rtl' : undefined}\n // @ts-ignore\n style={style}\n >\n <div\n className={buildClassName('ListItem-button', isTouched && 'active')}\n role=\"button\"\n ref={buttonRef}\n tabIndex={0}\n onClick={!inactive && IS_TOUCH_ENV ? handleClick : undefined}\n onMouseDown={handleMouseDown}\n onContextMenu={!inactive && contextActions ? handleContextMenu : undefined}\n >\n {icon && (\n <i className={`icon-${icon}`} />\n )}\n {multiline && (<div className=\"multiline-item\">{children}</div>)}\n {!multiline && children}\n {!disabled && !inactive && ripple && (\n <RippleEffect />\n )}\n </div>\n {contextActions && contextMenuPosition !== undefined && (\n <Menu\n isOpen={isContextMenuOpen}\n positionX={positionX}\n positionY={positionY}\n style={menuStyle}\n className=\"ListItem-context-menu\"\n autoClose\n onClose={handleContextMenuClose}\n onCloseAnimationEnd={handleContextMenuHide}\n >\n {contextActions.map((action) => (\n <MenuItem\n key={action.title}\n icon={action.icon}\n destructive={action.destructive}\n disabled={!action.handler}\n onClick={action.handler}\n >\n {action.title}\n </MenuItem>\n ))}\n </Menu>\n )}\n </div>\n );\n};\n\nexport default ListItem;\n","import { RefObject } from 'react';\nimport {\n useEffect, useRef, useCallback, useState,\n} from '../lib/teact/teact';\n\nimport { throttle, debounce } from '../util/schedulers';\nimport useHeavyAnimationCheck from './useHeavyAnimationCheck';\n\ntype TargetCallback = (entry: IntersectionObserverEntry) => void;\ntype RootCallback = (entries: IntersectionObserverEntry[]) => void;\ntype ObserveCleanup = NoneToVoidFunction;\nexport type ObserveFn = (target: HTMLElement, targetCallback?: TargetCallback) => ObserveCleanup;\n\ninterface IntersectionController {\n observer: IntersectionObserver;\n callbacks: Map<HTMLElement, TargetCallback>;\n}\n\ninterface Response {\n observe: ObserveFn;\n freeze: NoneToVoidFunction;\n unfreeze: NoneToVoidFunction;\n}\n\nexport function useIntersectionObserver({\n rootRef,\n throttleMs,\n debounceMs,\n shouldSkipFirst,\n margin,\n threshold,\n isDisabled,\n}: {\n rootRef: RefObject<HTMLDivElement>;\n throttleMs?: number;\n debounceMs?: number;\n shouldSkipFirst?: boolean;\n margin?: number;\n threshold?: number | number[];\n isDisabled?: boolean;\n}, rootCallback?: RootCallback): Response {\n const controllerRef = useRef<IntersectionController>();\n const rootCallbackRef = useRef<RootCallback>();\n const freezeFlagsRef = useRef(0);\n const onUnfreezeRef = useRef<NoneToVoidFunction>();\n\n rootCallbackRef.current = rootCallback;\n\n const freeze = useCallback(() => {\n freezeFlagsRef.current++;\n }, []);\n\n const unfreeze = useCallback(() => {\n if (!freezeFlagsRef.current) {\n return;\n }\n\n freezeFlagsRef.current--;\n\n if (!freezeFlagsRef.current && onUnfreezeRef.current) {\n onUnfreezeRef.current();\n onUnfreezeRef.current = undefined;\n }\n }, []);\n\n useHeavyAnimationCheck(freeze, unfreeze);\n\n useEffect(() => {\n if (isDisabled) {\n return undefined;\n }\n\n return () => {\n if (controllerRef.current) {\n controllerRef.current.observer.disconnect();\n controllerRef.current.callbacks.clear();\n controllerRef.current = undefined;\n }\n };\n }, [isDisabled]);\n\n function initController() {\n const callbacks = new Map();\n const entriesAccumulator = new Map<Element, IntersectionObserverEntry>();\n const observerCallbackSync = () => {\n const entries = Array.from(entriesAccumulator.values());\n\n entries.forEach((entry: IntersectionObserverEntry) => {\n const callback = callbacks.get(entry.target);\n if (callback) {\n callback!(entry, entries);\n }\n });\n\n if (rootCallbackRef.current) {\n rootCallbackRef.current(entries);\n }\n\n entriesAccumulator.clear();\n };\n const scheduler = throttleMs ? throttle : debounceMs ? debounce : undefined;\n const observerCallback = scheduler\n ? scheduler(observerCallbackSync, (throttleMs || debounceMs)!, !shouldSkipFirst)\n : observerCallbackSync;\n const observer = new IntersectionObserver(\n (entries) => {\n entries.forEach((entry) => {\n entriesAccumulator.set(entry.target, entry);\n });\n\n if (freezeFlagsRef.current) {\n onUnfreezeRef.current = () => {\n observerCallback();\n };\n } else {\n observerCallback();\n }\n },\n {\n root: rootRef.current,\n rootMargin: margin ? `${margin}px` : undefined,\n threshold,\n },\n );\n\n controllerRef.current = { observer, callbacks };\n }\n\n const observe = useCallback((target, targetCallback) => {\n if (!controllerRef.current) {\n initController();\n }\n\n const controller = controllerRef.current!;\n controller.observer.observe(target);\n\n if (targetCallback) {\n controller.callbacks.set(target, targetCallback);\n }\n\n return () => {\n if (targetCallback) {\n controller.callbacks.delete(target);\n }\n\n controller.observer.unobserve(target);\n };\n // Arguments should never change\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [isDisabled]);\n\n return { observe, freeze, unfreeze };\n}\n\nexport function useOnIntersect(\n targetRef: RefObject<HTMLDivElement>, observe?: ObserveFn, callback?: TargetCallback,\n) {\n useEffect(() => {\n return observe ? observe(targetRef.current!, callback) : undefined;\n // Arguments should never change\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n}\n\nexport function useIsIntersecting(\n targetRef: RefObject<HTMLDivElement>, observe?: ObserveFn, callback?: TargetCallback,\n) {\n const [isIntersecting, setIsIntersecting] = useState(!observe);\n\n useOnIntersect(targetRef, observe, (entry) => {\n setIsIntersecting(entry.isIntersecting);\n\n if (callback) {\n callback(entry);\n }\n });\n\n return isIntersecting;\n}\n","import { useEffect } from '../lib/teact/teact';\n\nimport { ApiMediaFormat } from '../api/types';\n\nimport * as mediaLoader from '../util/mediaLoader';\nimport useForceUpdate from './useForceUpdate';\n\nexport default <T extends ApiMediaFormat = ApiMediaFormat.BlobUrl>(\n mediaHash: string | false | undefined,\n noLoad = false,\n // @ts-ignore (workaround for \"could be instantiated with a different subtype\" issue)\n mediaFormat: T = ApiMediaFormat.BlobUrl,\n cacheBuster?: number,\n delay?: number | false,\n) => {\n const mediaData = mediaHash ? mediaLoader.getFromMemory<T>(mediaHash) : undefined;\n const forceUpdate = useForceUpdate();\n\n useEffect(() => {\n if (!noLoad && mediaHash && !mediaData) {\n const startedAt = Date.now();\n\n mediaLoader.fetch(mediaHash, mediaFormat).then(() => {\n const spentTime = Date.now() - startedAt;\n if (!delay || spentTime >= delay) {\n forceUpdate();\n } else {\n setTimeout(forceUpdate, delay - spentTime);\n }\n });\n }\n }, [noLoad, mediaHash, mediaData, mediaFormat, cacheBuster, forceUpdate, delay]);\n\n return mediaData;\n};\n","import { MouseEvent as ReactMouseEvent } from 'react';\nimport React, {\n FC, useEffect, useCallback, memo,\n} from '../../lib/teact/teact';\nimport { withGlobal } from '../../lib/teact/teactn';\n\nimport { ApiUser, ApiTypingStatus } from '../../api/types';\nimport { GlobalActions, GlobalState } from '../../global/types';\nimport { MediaViewerOrigin } from '../../types';\n\nimport { selectChatMessages, selectUser } from '../../modules/selectors';\nimport { getUserFullName, getUserStatus, isUserOnline } from '../../modules/helpers';\nimport renderText from './helpers/renderText';\nimport { pick } from '../../util/iteratees';\nimport useLang from '../../hooks/useLang';\n\nimport Avatar from './Avatar';\nimport VerifiedIcon from './VerifiedIcon';\nimport TypingStatus from './TypingStatus';\n\ntype OwnProps = {\n userId: number;\n typingStatus?: ApiTypingStatus;\n avatarSize?: 'tiny' | 'small' | 'medium' | 'large' | 'jumbo';\n forceShowSelf?: boolean;\n status?: string;\n withMediaViewer?: boolean;\n withUsername?: boolean;\n withFullInfo?: boolean;\n withUpdatingStatus?: boolean;\n noStatusOrTyping?: boolean;\n noRtl?: boolean;\n};\n\ntype StateProps = {\n user?: ApiUser;\n isSavedMessages?: boolean;\n areMessagesLoaded: boolean;\n serverTimeOffset: number;\n} & Pick<GlobalState, 'lastSyncTime'>;\n\ntype DispatchProps = Pick<GlobalActions, 'loadFullUser' | 'openMediaViewer'>;\n\nconst PrivateChatInfo: FC<OwnProps & StateProps & DispatchProps> = ({\n typingStatus,\n avatarSize = 'medium',\n status,\n withMediaViewer,\n withUsername,\n withFullInfo,\n withUpdatingStatus,\n noStatusOrTyping,\n noRtl,\n user,\n isSavedMessages,\n areMessagesLoaded,\n lastSyncTime,\n loadFullUser,\n openMediaViewer,\n serverTimeOffset,\n}) => {\n const { id: userId } = user || {};\n const fullName = getUserFullName(user);\n\n useEffect(() => {\n if (withFullInfo && lastSyncTime && userId) {\n loadFullUser({ userId });\n }\n }, [userId, loadFullUser, lastSyncTime, withFullInfo]);\n\n const handleAvatarViewerOpen = useCallback((e: ReactMouseEvent<HTMLDivElement, MouseEvent>, hasPhoto: boolean) => {\n if (user && hasPhoto) {\n e.stopPropagation();\n openMediaViewer({\n avatarOwnerId: user.id,\n origin: avatarSize === 'jumbo' ? MediaViewerOrigin.ProfileAvatar : MediaViewerOrigin.MiddleHeaderAvatar,\n });\n }\n }, [user, avatarSize, openMediaViewer]);\n\n const lang = useLang();\n\n if (!user) {\n return undefined;\n }\n\n function renderStatusOrTyping() {\n if (status) {\n return (\n <span className=\"status\" dir=\"auto\">{status}</span>\n );\n }\n\n if (withUpdatingStatus && !areMessagesLoaded) {\n return (\n <span className=\"status\" dir=\"auto\">{lang('Updating')}</span>\n );\n }\n\n if (!user) {\n return undefined;\n }\n\n if (typingStatus) {\n return <TypingStatus typingStatus={typingStatus} />;\n }\n\n return (\n <div className={`status ${isUserOnline(user) ? 'online' : ''}`}>\n {withUsername && user.username && <span className=\"handle\">{user.username}</span>}\n <span className=\"user-status\" dir=\"auto\">{getUserStatus(lang, user, serverTimeOffset)}</span>\n </div>\n );\n }\n\n return (\n <div className=\"ChatInfo\" dir={!noRtl && lang.isRtl ? 'rtl' : undefined}>\n <Avatar\n key={user.id}\n size={avatarSize}\n user={user}\n isSavedMessages={isSavedMessages}\n onClick={withMediaViewer ? handleAvatarViewerOpen : undefined}\n />\n <div className=\"info\">\n {isSavedMessages ? (\n <div className=\"title\">\n <h3>{lang('SavedMessages')}</h3>\n </div>\n ) : (\n <div className=\"title\">\n <h3 dir=\"auto\">{fullName && renderText(fullName)}</h3>\n {user && user.isVerified && <VerifiedIcon />}\n </div>\n )}\n {(status || (!isSavedMessages && !noStatusOrTyping)) && renderStatusOrTyping()}\n </div>\n </div>\n );\n};\n\nexport default memo(withGlobal<OwnProps>(\n (global, { userId, forceShowSelf }): StateProps => {\n const { lastSyncTime, serverTimeOffset } = global;\n const user = selectUser(global, userId);\n const isSavedMessages = !forceShowSelf && user && user.isSelf;\n const areMessagesLoaded = Boolean(selectChatMessages(global, userId));\n\n return {\n lastSyncTime, user, isSavedMessages, areMessagesLoaded, serverTimeOffset,\n };\n },\n (setGlobal, actions): DispatchProps => pick(actions, ['loadFullUser', 'openMediaViewer']),\n)(PrivateChatInfo));\n","import React, {\n FC, useEffect, useRef,\n} from '../../lib/teact/teact';\n\nimport { TextPart } from '../common/helpers/renderMessageText';\n\nimport captureKeyboardListeners from '../../util/captureKeyboardListeners';\nimport trapFocus from '../../util/trapFocus';\nimport buildClassName from '../../util/buildClassName';\nimport { dispatchHeavyAnimationEvent } from '../../hooks/useHeavyAnimationCheck';\nimport useShowTransition from '../../hooks/useShowTransition';\nimport useEffectWithPrevDeps from '../../hooks/useEffectWithPrevDeps';\nimport useLang from '../../hooks/useLang';\nimport useHistoryBack from '../../hooks/useHistoryBack';\n\nimport Button from './Button';\nimport Portal from './Portal';\n\nimport './Modal.scss';\n\nconst ANIMATION_DURATION = 200;\n\ntype OwnProps = {\n title?: string | TextPart[];\n className?: string;\n isOpen?: boolean;\n header?: any;\n hasCloseButton?: boolean;\n noBackdrop?: boolean;\n children: any;\n onClose: () => void;\n onCloseAnimationEnd?: () => void;\n onEnter?: () => void;\n};\n\ntype StateProps = {\n shouldSkipHistoryAnimations?: boolean;\n};\n\nconst Modal: FC<OwnProps & StateProps> = ({\n title,\n className,\n isOpen,\n header,\n hasCloseButton,\n noBackdrop,\n children,\n onClose,\n onCloseAnimationEnd,\n onEnter,\n shouldSkipHistoryAnimations,\n}) => {\n const {\n shouldRender,\n transitionClassNames,\n } = useShowTransition(\n isOpen, onCloseAnimationEnd, shouldSkipHistoryAnimations, undefined, shouldSkipHistoryAnimations,\n );\n // eslint-disable-next-line no-null/no-null\n const modalRef = useRef<HTMLDivElement>(null);\n\n useEffect(() => (isOpen\n ? captureKeyboardListeners({ onEsc: onClose, onEnter })\n : undefined), [isOpen, onClose, onEnter]);\n useEffect(() => (isOpen && modalRef.current ? trapFocus(modalRef.current) : undefined), [isOpen]);\n\n const { forceClose } = useHistoryBack(isOpen, onClose);\n\n // For modals that are closed by unmounting without changing `isOpen` to `false`\n useEffect(() => {\n return () => {\n if (isOpen) {\n forceClose();\n }\n };\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useEffectWithPrevDeps(([prevIsOpen]) => {\n document.body.classList.toggle('has-open-dialog', isOpen);\n\n if (isOpen || (!isOpen && prevIsOpen !== undefined)) {\n dispatchHeavyAnimationEvent(ANIMATION_DURATION);\n }\n\n return () => {\n document.body.classList.remove('has-open-dialog');\n };\n }, [isOpen]);\n\n const lang = useLang();\n\n if (!shouldRender) {\n return undefined;\n }\n\n function renderHeader() {\n if (header) {\n return header;\n }\n\n if (!title) {\n return undefined;\n }\n\n return (\n <div className=\"modal-header\">\n {hasCloseButton && (\n <Button\n round\n color=\"translucent\"\n size=\"smaller\"\n ariaLabel={lang('Close')}\n onClick={onClose}\n >\n <i className=\"icon-close\" />\n </Button>\n )}\n <div className=\"modal-title\">{title}</div>\n </div>\n );\n }\n\n const fullClassName = buildClassName(\n 'Modal',\n className,\n transitionClassNames,\n noBackdrop && 'transparent-backdrop',\n );\n\n return (\n <Portal>\n <div\n ref={modalRef}\n className={fullClassName}\n tabIndex={-1}\n role=\"dialog\"\n >\n <div className=\"modal-container\">\n <div className=\"modal-backdrop\" onClick={onClose} />\n <div className=\"modal-dialog\">\n {renderHeader()}\n <div className=\"modal-content custom-scroll\">\n {children}\n </div>\n </div>\n </div>\n </div>\n </Portal>\n );\n};\n\nexport default Modal;\n","export default function trapFocus(element: HTMLElement) {\n function handleKeyDown(e: KeyboardEvent) {\n if (e.key !== 'Tab') {\n return;\n }\n\n e.preventDefault();\n e.stopPropagation();\n\n const focusableElements = Array.from(\n element.querySelectorAll('button, [href], input, select, textarea, [tabindex]:not([tabindex=\"-1\"])'),\n ) as HTMLElement[];\n if (!focusableElements.length) {\n return;\n }\n\n const currentFocusedIndex = focusableElements.findIndex((em) => em.isSameNode(document.activeElement));\n let newFocusedIndex = 0;\n if (currentFocusedIndex >= 0) {\n if (e.shiftKey) {\n newFocusedIndex = currentFocusedIndex > 0\n ? currentFocusedIndex - 1\n : focusableElements.length - 1;\n } else {\n newFocusedIndex = currentFocusedIndex < focusableElements.length - 1\n ? currentFocusedIndex + 1\n : 0;\n }\n }\n\n focusableElements[newFocusedIndex].focus();\n }\n\n document.addEventListener('keydown', handleKeyDown, false);\n\n return () => {\n document.removeEventListener('keydown', handleKeyDown, false);\n };\n}\n","import { RefObject } from 'react';\nimport React, {\n FC, useLayoutEffect, useRef,\n} from '../../lib/teact/teact';\nimport { getGlobal } from '../../lib/teact/teactn';\n\nimport useForceUpdate from '../../hooks/useForceUpdate';\nimport usePrevious from '../../hooks/usePrevious';\nimport buildClassName from '../../util/buildClassName';\nimport { dispatchHeavyAnimationEvent } from '../../hooks/useHeavyAnimationCheck';\nimport { waitForAnimationEnd } from '../../util/cssAnimationEndListeners';\n\nimport './Transition.scss';\n\ntype ChildrenFn = (isActive: boolean, isFrom: boolean, currentKey: number) => any;\ntype OwnProps = {\n ref?: RefObject<HTMLDivElement>;\n activeKey: number;\n name: (\n 'none' | 'slide' | 'slide-reversed' | 'mv-slide' | 'slide-fade' | 'zoom-fade' | 'slide-layers'\n | 'fade' | 'push-slide' | 'reveal'\n );\n direction?: 'auto' | 'inverse' | 1 | -1;\n renderCount?: number;\n shouldRestoreHeight?: boolean;\n shouldCleanup?: boolean;\n cleanupExceptionKey?: number;\n id?: string;\n className?: string;\n onStart?: () => void;\n onStop?: () => void;\n children: ChildrenFn;\n};\n\nconst CLEANED_UP = Symbol('CLEANED_UP');\n\nconst Transition: FC<OwnProps> = ({\n ref,\n activeKey,\n name,\n direction = 'auto',\n renderCount,\n shouldRestoreHeight,\n shouldCleanup,\n cleanupExceptionKey,\n id,\n className,\n onStart,\n onStop,\n children,\n}) => {\n // No need for a container to update on change\n const { animationLevel } = getGlobal().settings.byKey;\n\n // eslint-disable-next-line no-null/no-null\n let containerRef = useRef<HTMLDivElement>(null);\n if (ref) {\n containerRef = ref;\n }\n\n const rendersRef = useRef<Record<number, ChildrenFn | typeof CLEANED_UP>>({});\n const prevActiveKey = usePrevious<any>(activeKey);\n const forceUpdate = useForceUpdate();\n\n const activeKeyChanged = prevActiveKey !== undefined && activeKey !== prevActiveKey;\n\n if (!renderCount && activeKeyChanged) {\n rendersRef.current = { [prevActiveKey]: rendersRef.current[prevActiveKey] };\n }\n\n rendersRef.current[activeKey] = children;\n\n useLayoutEffect(() => {\n function cleanup() {\n if (!shouldCleanup || (cleanupExceptionKey !== undefined && cleanupExceptionKey === prevActiveKey)) {\n return;\n }\n\n rendersRef.current = { [prevActiveKey]: CLEANED_UP };\n forceUpdate();\n }\n\n const container = containerRef.current!;\n\n const childElements = container.children;\n if (childElements.length === 1 && !activeKeyChanged) {\n childElements[0].classList.add('active');\n\n return;\n }\n\n const childNodes = Array.from(container.childNodes);\n\n if (!activeKeyChanged || !childNodes.length) {\n return;\n }\n\n const isBackwards = (\n direction === -1\n || (direction === 'auto' && prevActiveKey > activeKey)\n || (direction === 'inverse' && prevActiveKey < activeKey)\n );\n\n container.classList.remove('animating');\n container.classList.toggle('backwards', isBackwards);\n\n const keys = Object.keys(rendersRef.current).map(Number);\n const prevActiveIndex = renderCount ? prevActiveKey : keys.indexOf(prevActiveKey);\n const activeIndex = renderCount ? activeKey : keys.indexOf(activeKey);\n\n if (name === 'none' || animationLevel === 0) {\n childNodes.forEach((node, i) => {\n if (node instanceof HTMLElement) {\n node.classList.remove('from', 'through', 'to');\n node.classList.toggle('active', i === activeIndex);\n }\n });\n\n cleanup();\n\n return;\n }\n\n childNodes.forEach((node, i) => {\n if (node instanceof HTMLElement) {\n node.classList.remove('active');\n node.classList.toggle('from', i === prevActiveIndex);\n node.classList.toggle('through', (\n (i > prevActiveIndex && i < activeIndex) || (i < prevActiveIndex && i > activeIndex)\n ));\n node.classList.toggle('to', i === activeIndex);\n }\n });\n\n let dispatchHeavyAnimationStop: NoneToVoidFunction;\n if (animationLevel > 0) {\n dispatchHeavyAnimationStop = dispatchHeavyAnimationEvent();\n }\n\n requestAnimationFrame(() => {\n container.classList.add('animating');\n\n if (onStart) {\n onStart();\n }\n\n function onAnimationEnd() {\n requestAnimationFrame(() => {\n container.classList.remove('animating', 'backwards');\n\n childNodes.forEach((node, i) => {\n if (node instanceof HTMLElement) {\n node.classList.remove('from', 'through', 'to');\n node.classList.toggle('active', i === activeIndex);\n }\n });\n\n if (shouldRestoreHeight) {\n const activeElement = container.querySelector<HTMLDivElement>('.active');\n\n if (activeElement) {\n activeElement.style.height = 'auto';\n container.style.height = `${activeElement.clientHeight}px`;\n }\n }\n\n if (dispatchHeavyAnimationStop) {\n dispatchHeavyAnimationStop();\n }\n\n cleanup();\n\n if (onStop) {\n onStop();\n }\n });\n }\n\n const toNode = name === 'mv-slide'\n ? childNodes[activeIndex] && childNodes[activeIndex].firstChild\n : childNodes[activeIndex];\n\n if (animationLevel > 0 && toNode) {\n waitForAnimationEnd(toNode, onAnimationEnd);\n } else {\n onAnimationEnd();\n }\n });\n }, [\n activeKey,\n prevActiveKey,\n activeKeyChanged,\n direction,\n name,\n onStart,\n onStop,\n renderCount,\n shouldRestoreHeight,\n shouldCleanup,\n cleanupExceptionKey,\n animationLevel,\n forceUpdate,\n ]);\n\n useLayoutEffect(() => {\n if (shouldRestoreHeight) {\n const container = containerRef.current!;\n const activeElement = container.querySelector<HTMLDivElement>('.active')\n || container.querySelector<HTMLDivElement>('.from');\n\n if (activeElement) {\n activeElement.style.height = 'auto';\n container.style.height = `${activeElement.clientHeight}px`;\n container.style.flexBasis = `${activeElement.clientHeight}px`;\n }\n }\n }, [shouldRestoreHeight, children]);\n\n const renders = rendersRef.current;\n const collection = Object.keys(renderCount ? new Array(renderCount).fill(undefined) : renders).map(Number);\n const contents = collection.map((key) => {\n const render = renders[key];\n\n return (\n typeof render === 'function'\n ? <div key={key}>{render(key === activeKey, key === prevActiveKey, activeKey)}</div>\n : undefined\n );\n });\n\n const fullClassName = buildClassName(\n 'Transition',\n className,\n name,\n );\n\n return (\n <div ref={containerRef} id={id} className={fullClassName}>\n {contents}\n </div>\n );\n};\n\nexport default Transition;\n","import { MouseEvent as ReactMouseEvent } from 'react';\nimport React, { FC, useCallback, memo } from '../../lib/teact/teact';\n\nimport { ApiUser, ApiChat, ApiMediaFormat } from '../../api/types';\n\nimport { IS_TEST } from '../../config';\nimport {\n getChatAvatarHash, getChatTitle, isChatPrivate,\n getUserFullName, isUserOnline, isDeletedUser, getUserColorKey,\n} from '../../modules/helpers';\nimport { getFirstLetters } from '../../util/textFormat';\nimport buildClassName from '../../util/buildClassName';\nimport renderText from './helpers/renderText';\nimport useMedia from '../../hooks/useMedia';\nimport useTransitionForMedia from '../../hooks/useTransitionForMedia';\nimport useLang from '../../hooks/useLang';\n\nimport './Avatar.scss';\n\ntype OwnProps = {\n className?: string;\n size?: 'micro' | 'tiny' | 'small' | 'medium' | 'large' | 'jumbo';\n withOnlineStatus?: boolean;\n chat?: ApiChat;\n user?: ApiUser;\n text?: string;\n isSavedMessages?: boolean;\n lastSyncTime?: number;\n onClick?: (e: ReactMouseEvent<HTMLDivElement, MouseEvent>, hasPhoto: boolean) => void;\n};\n\nconst Avatar: FC<OwnProps> = ({\n className,\n size = 'large',\n chat,\n user,\n text,\n withOnlineStatus,\n isSavedMessages,\n lastSyncTime,\n onClick,\n}) => {\n const isDeleted = user && isDeletedUser(user);\n let imageHash: string | undefined;\n\n if (!isSavedMessages && !isDeleted) {\n if (user) {\n imageHash = getChatAvatarHash(user);\n } else if (chat) {\n imageHash = getChatAvatarHash(chat);\n }\n }\n\n const dataUri = useMedia(imageHash, false, ApiMediaFormat.DataUri, lastSyncTime);\n const { shouldRenderFullMedia, transitionClassNames } = useTransitionForMedia(dataUri, 'slow');\n\n const lang = useLang();\n\n let content: string | undefined = '';\n\n if (isSavedMessages) {\n content = <i className=\"icon-avatar-saved-messages\" />;\n } else if (isDeleted) {\n content = <i className=\"icon-avatar-deleted-account\" />;\n } else if (shouldRenderFullMedia) {\n content = <img src={dataUri} className={`${transitionClassNames} avatar-media`} alt=\"\" decoding=\"async\" />;\n } else if (user) {\n const userFullName = getUserFullName(user);\n content = userFullName ? getFirstLetters(userFullName, 2) : undefined;\n } else if (chat) {\n const title = getChatTitle(lang, chat);\n content = title && getFirstLetters(title, isChatPrivate(chat.id) ? 2 : 1);\n } else if (text) {\n content = getFirstLetters(text, 2);\n }\n\n const isOnline = !isSavedMessages && user && isUserOnline(user);\n const fullClassName = buildClassName(\n `Avatar size-${size}`,\n className,\n `color-bg-${getUserColorKey(user || chat)}`,\n isSavedMessages && 'saved-messages',\n isDeleted && 'deleted-account',\n withOnlineStatus && isOnline && 'online',\n onClick && 'interactive',\n (!isSavedMessages && !shouldRenderFullMedia) && 'no-photo',\n );\n\n const handleClick = useCallback((e: ReactMouseEvent<HTMLDivElement, MouseEvent>) => {\n if (onClick) {\n onClick(e, isSavedMessages || shouldRenderFullMedia);\n }\n }, [onClick, isSavedMessages, shouldRenderFullMedia]);\n\n const senderId = (user || chat) && (user || chat)!.id;\n\n return (\n <div className={fullClassName} onClick={handleClick} data-test-sender-id={IS_TEST ? senderId : undefined}>\n {typeof content === 'string' ? renderText(content, [size === 'jumbo' ? 'hq_emoji' : 'emoji']) : content}\n </div>\n );\n};\n\nexport default memo(Avatar);\n","import { RefObject, UIEvent } from 'react';\nimport { LoadMoreDirection } from '../../types';\n\nimport React, {\n FC, useCallback, useEffect, useLayoutEffect, useMemo, useRef,\n} from '../../lib/teact/teact';\n\nimport { debounce } from '../../util/schedulers';\nimport resetScroll from '../../util/resetScroll';\n\ntype OwnProps = {\n ref?: RefObject<HTMLDivElement>;\n className?: string;\n onLoadMore?: ({ direction }: { direction: LoadMoreDirection; noScroll?: boolean }) => void;\n onScroll?: (e: UIEvent<HTMLDivElement>) => void;\n onKeyDown?: (e: React.KeyboardEvent<any>) => void;\n items?: any[];\n itemSelector?: string;\n preloadBackwards?: number;\n sensitiveArea?: number;\n noScrollRestore?: boolean;\n noScrollRestoreOnTop?: boolean;\n noFastList?: boolean;\n cacheBuster?: any;\n children: any;\n};\n\nconst DEFAULT_LIST_SELECTOR = '.ListItem';\nconst DEFAULT_PRELOAD_BACKWARDS = 20;\nconst DEFAULT_SENSITIVE_AREA = 800;\n\nconst InfiniteScroll: FC<OwnProps> = ({\n ref,\n className,\n onLoadMore,\n onScroll,\n onKeyDown,\n items,\n itemSelector = DEFAULT_LIST_SELECTOR,\n preloadBackwards = DEFAULT_PRELOAD_BACKWARDS,\n sensitiveArea = DEFAULT_SENSITIVE_AREA,\n // Used to turn off restoring scroll position (e.g. for frequently re-ordered chat or user lists)\n noScrollRestore = false,\n noScrollRestoreOnTop = false,\n noFastList,\n // Used to re-query `listItemElements` if rendering is delayed by transition\n cacheBuster,\n children,\n}: OwnProps) => {\n // eslint-disable-next-line no-null/no-null\n let containerRef = useRef<HTMLDivElement>(null);\n if (ref) {\n containerRef = ref;\n }\n\n const stateRef = useRef<{\n listItemElements: NodeListOf<HTMLDivElement>;\n isScrollTopJustUpdated: boolean;\n currentAnchor: HTMLDivElement | undefined;\n currentAnchorTop: number;\n }>({} as any);\n\n const [loadMoreBackwards, loadMoreForwards] = useMemo(() => {\n if (!onLoadMore) {\n return [];\n }\n\n return [\n debounce((noScroll = false) => {\n onLoadMore({ direction: LoadMoreDirection.Backwards, noScroll });\n }, 1000, true, false),\n debounce(() => {\n onLoadMore({ direction: LoadMoreDirection.Forwards });\n }, 1000, true, false),\n ];\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [onLoadMore, items]);\n\n // Initial preload\n useEffect(() => {\n if (!loadMoreBackwards) {\n return;\n }\n\n if (preloadBackwards > 0 && (!items || items.length < preloadBackwards)) {\n loadMoreBackwards(true);\n return;\n }\n\n const { scrollHeight, clientHeight } = containerRef.current!;\n if (clientHeight && scrollHeight <= clientHeight) {\n loadMoreBackwards();\n }\n }, [items, loadMoreBackwards, preloadBackwards]);\n\n // Restore `scrollTop` after adding items\n useLayoutEffect(() => {\n const container = containerRef.current!;\n const state = stateRef.current;\n\n state.listItemElements = container.querySelectorAll<HTMLDivElement>(itemSelector);\n\n let newScrollTop;\n\n if (state.currentAnchor && Array.from(state.listItemElements).includes(state.currentAnchor)) {\n const { scrollTop } = container;\n const newAnchorTop = state.currentAnchor.getBoundingClientRect().top;\n newScrollTop = scrollTop + (newAnchorTop - state.currentAnchorTop);\n } else {\n const nextAnchor = state.listItemElements[0];\n if (nextAnchor) {\n state.currentAnchor = nextAnchor;\n state.currentAnchorTop = nextAnchor.getBoundingClientRect().top;\n }\n }\n\n if (noScrollRestore) {\n return;\n }\n\n if (noScrollRestoreOnTop && container.scrollTop === 0) {\n return;\n }\n\n resetScroll(container, newScrollTop);\n\n state.isScrollTopJustUpdated = true;\n }, [items, itemSelector, noScrollRestore, noScrollRestoreOnTop, cacheBuster]);\n\n const handleScroll = useCallback((e: UIEvent<HTMLDivElement>) => {\n if (loadMoreForwards && loadMoreBackwards) {\n const {\n listItemElements, isScrollTopJustUpdated, currentAnchor, currentAnchorTop,\n } = stateRef.current;\n\n if (isScrollTopJustUpdated) {\n stateRef.current.isScrollTopJustUpdated = false;\n return;\n }\n\n const listLength = listItemElements.length;\n const container = containerRef.current!;\n const { scrollTop, scrollHeight, offsetHeight } = container;\n const top = listLength ? listItemElements[0].offsetTop : 0;\n const isNearTop = scrollTop <= top + sensitiveArea;\n const bottom = listLength\n ? listItemElements[listLength - 1].offsetTop + listItemElements[listLength - 1].offsetHeight\n : scrollHeight;\n const isNearBottom = bottom - (scrollTop + offsetHeight) <= sensitiveArea;\n let isUpdated = false;\n\n if (isNearTop) {\n const nextAnchor = listItemElements[0];\n if (nextAnchor) {\n const nextAnchorTop = nextAnchor.getBoundingClientRect().top;\n const newAnchorTop = currentAnchor && currentAnchor.offsetParent && currentAnchor !== nextAnchor\n ? currentAnchor.getBoundingClientRect().top\n : nextAnchorTop;\n const isMovingUp = (\n currentAnchor && currentAnchorTop !== undefined && newAnchorTop > currentAnchorTop\n );\n\n if (isMovingUp) {\n stateRef.current.currentAnchor = nextAnchor;\n stateRef.current.currentAnchorTop = nextAnchorTop;\n isUpdated = true;\n resetScroll(container);\n loadMoreForwards();\n }\n }\n }\n\n if (isNearBottom) {\n const nextAnchor = listItemElements[listLength - 1];\n if (nextAnchor) {\n const nextAnchorTop = nextAnchor.getBoundingClientRect().top;\n const newAnchorTop = currentAnchor && currentAnchor.offsetParent && currentAnchor !== nextAnchor\n ? currentAnchor.getBoundingClientRect().top\n : nextAnchorTop;\n const isMovingDown = (\n currentAnchor && currentAnchorTop !== undefined && newAnchorTop < currentAnchorTop\n );\n\n if (isMovingDown) {\n stateRef.current.currentAnchor = nextAnchor;\n stateRef.current.currentAnchorTop = nextAnchorTop;\n isUpdated = true;\n resetScroll(container);\n loadMoreBackwards();\n }\n }\n }\n\n if (!isUpdated) {\n if (currentAnchor && currentAnchor.offsetParent) {\n stateRef.current.currentAnchorTop = currentAnchor.getBoundingClientRect().top;\n } else {\n const nextAnchor = listItemElements[0];\n\n if (nextAnchor) {\n stateRef.current.currentAnchor = nextAnchor;\n stateRef.current.currentAnchorTop = nextAnchor.getBoundingClientRect().top;\n }\n }\n }\n }\n\n if (onScroll) {\n onScroll(e);\n }\n }, [loadMoreBackwards, loadMoreForwards, onScroll, sensitiveArea]);\n\n return (\n <div\n ref={containerRef}\n className={className}\n onScroll={handleScroll}\n teactFastList={!noFastList}\n onKeyDown={onKeyDown}\n >\n {children}\n </div>\n );\n};\n\nexport default InfiniteScroll;\n","import React, { FC } from '../../lib/teact/teact';\n\nimport buildClassName from '../../util/buildClassName';\n\nimport Button, { OwnProps as ButtonProps } from './Button';\n\nimport './FloatingActionButton.scss';\n\ntype OwnProps = {\n isShown: boolean;\n className?: string;\n color?: ButtonProps['color'];\n ariaLabel?: ButtonProps['ariaLabel'];\n disabled?: boolean;\n onClick: () => void;\n children: any;\n};\n\nconst FloatingActionButton: FC<OwnProps> = ({\n isShown,\n className,\n color = 'primary',\n ariaLabel,\n disabled,\n onClick,\n children,\n}) => {\n const buttonClassName = buildClassName(\n 'FloatingActionButton',\n isShown && 'revealed',\n className,\n );\n\n return (\n <Button\n className={buttonClassName}\n color={color}\n round\n disabled={disabled}\n onClick={isShown && !disabled ? onClick : undefined}\n ariaLabel={ariaLabel}\n tabIndex={-1}\n >\n {children}\n </Button>\n );\n};\n\nexport default FloatingActionButton;\n","import { useEffect, useRef, useState } from '../lib/teact/teact';\n\nimport useShowTransition from './useShowTransition';\n\nconst SPEED = {\n fast: 200,\n slow: 350,\n};\n\nexport default (mediaData?: any, speed: keyof typeof SPEED = 'fast', noAnimate = false) => {\n const isMediaLoaded = Boolean(mediaData);\n const willAnimate = !useRef(isMediaLoaded).current && !noAnimate;\n const [shouldRenderThumb, setShouldRenderThumb] = useState(!isMediaLoaded);\n\n const {\n shouldRender: shouldRenderFullMedia,\n transitionClassNames,\n } = useShowTransition(isMediaLoaded, undefined, !willAnimate, speed);\n\n useEffect(() => {\n if (shouldRenderFullMedia) {\n if (willAnimate) {\n setTimeout(() => {\n setShouldRenderThumb(false);\n }, SPEED[speed]);\n } else {\n setShouldRenderThumb(false);\n }\n }\n }, [willAnimate, shouldRenderFullMedia, speed]);\n\n return {\n shouldRenderThumb,\n shouldRenderFullMedia,\n transitionClassNames,\n };\n};\n","import {\n ApiPhoto, ApiVideo, ApiSticker, ApiDimensions,\n} from '../../../api/types';\n\nimport { STICKER_SIZE_INLINE_DESKTOP_FACTOR, STICKER_SIZE_INLINE_MOBILE_FACTOR } from '../../../config';\nimport { IS_SINGLE_COLUMN_LAYOUT } from '../../../util/environment';\nimport windowSize from '../../../util/windowSize';\nimport { getPhotoInlineDimensions, getVideoDimensions } from '../../../modules/helpers';\n\nexport const MEDIA_VIEWER_MEDIA_QUERY = '(max-height: 640px)';\nexport const REM = parseInt(getComputedStyle(document.documentElement).fontSize, 10);\nexport const ROUND_VIDEO_DIMENSIONS = 200;\nexport const AVATAR_FULL_DIMENSIONS = { width: 640, height: 640 };\nexport const LIKE_STICKER_ID = '1258816259753933';\n\nconst DEFAULT_MEDIA_DIMENSIONS: ApiDimensions = { width: 100, height: 100 };\nconst MOBILE_SCREEN_NO_AVATARS_MESSAGE_EXTRA_WIDTH_REM = 4.5;\nconst MOBILE_SCREEN_MESSAGE_EXTRA_WIDTH_REM = 7;\nconst MESSAGE_MAX_WIDTH_REM = 29;\nconst MESSAGE_OWN_MAX_WIDTH_REM = 30;\n\nlet cachedMaxWidthOwn: number | undefined;\nlet cachedMaxWidth: number | undefined;\nlet cachedMaxWidthNoAvatar: number | undefined;\n\nfunction getMaxMessageWidthRem(fromOwnMessage: boolean, noAvatars?: boolean) {\n const regularMaxWidth = fromOwnMessage ? MESSAGE_OWN_MAX_WIDTH_REM : MESSAGE_MAX_WIDTH_REM;\n if (!IS_SINGLE_COLUMN_LAYOUT) {\n return regularMaxWidth;\n }\n\n const { width: windowWidth } = windowSize.get();\n\n // @optimization Limitation: changing device screen width not supported\n if (!cachedMaxWidthOwn) {\n cachedMaxWidthOwn = Math.min(\n MESSAGE_OWN_MAX_WIDTH_REM,\n windowWidth / REM - MOBILE_SCREEN_NO_AVATARS_MESSAGE_EXTRA_WIDTH_REM,\n );\n }\n if (!cachedMaxWidth) {\n cachedMaxWidth = Math.min(\n MESSAGE_MAX_WIDTH_REM,\n windowWidth / REM - MOBILE_SCREEN_MESSAGE_EXTRA_WIDTH_REM,\n );\n }\n if (!cachedMaxWidthNoAvatar) {\n cachedMaxWidthNoAvatar = Math.min(\n MESSAGE_MAX_WIDTH_REM,\n windowWidth / REM - MOBILE_SCREEN_NO_AVATARS_MESSAGE_EXTRA_WIDTH_REM,\n );\n }\n\n return fromOwnMessage\n ? cachedMaxWidthOwn\n : (noAvatars ? cachedMaxWidthNoAvatar : cachedMaxWidth);\n}\n\nexport function getAvailableWidth(\n fromOwnMessage: boolean,\n isForwarded?: boolean,\n isWebPagePhoto?: boolean,\n noAvatars?: boolean,\n) {\n const extraPaddingRem = isForwarded || isWebPagePhoto ? 1.625 : 0;\n const availableWidthRem = getMaxMessageWidthRem(fromOwnMessage, noAvatars) - extraPaddingRem;\n\n return availableWidthRem * REM;\n}\n\nfunction getAvailableHeight(isGif?: boolean, aspectRatio?: number) {\n if (\n isGif && aspectRatio\n && aspectRatio >= 0.75 && aspectRatio <= 1.25\n ) {\n return 20 * REM;\n }\n\n return 27 * REM;\n}\n\nfunction calculateDimensionsForMessageMedia({\n width,\n height,\n fromOwnMessage,\n isForwarded,\n isWebPagePhoto,\n isGif,\n noAvatars,\n}: {\n width: number;\n height: number;\n fromOwnMessage: boolean;\n isForwarded?: boolean;\n isWebPagePhoto?: boolean;\n isGif?: boolean;\n noAvatars?: boolean;\n}): ApiDimensions {\n const aspectRatio = height / width;\n const availableWidth = getAvailableWidth(fromOwnMessage, isForwarded, isWebPagePhoto, noAvatars);\n const availableHeight = getAvailableHeight(isGif, aspectRatio);\n\n return calculateDimensions(availableWidth, availableHeight, width, height);\n}\n\nexport function getMediaViewerAvailableDimensions(withFooter: boolean, isVideo: boolean): ApiDimensions {\n const mql = window.matchMedia(MEDIA_VIEWER_MEDIA_QUERY);\n const { width: windowWidth, height: windowHeight } = windowSize.get();\n let occupiedHeightRem = isVideo && mql.matches ? 10 : 8.25;\n if (withFooter) {\n occupiedHeightRem = mql.matches ? 10 : 15;\n }\n\n return {\n width: windowWidth,\n height: windowHeight - occupiedHeightRem * REM,\n };\n}\n\nexport function calculateInlineImageDimensions(\n photo: ApiPhoto,\n fromOwnMessage: boolean,\n isForwarded?: boolean,\n isWebPagePhoto?: boolean,\n noAvatars?: boolean,\n) {\n const { width, height } = getPhotoInlineDimensions(photo) || DEFAULT_MEDIA_DIMENSIONS;\n\n return calculateDimensionsForMessageMedia({\n width,\n height,\n fromOwnMessage,\n isForwarded,\n isWebPagePhoto,\n noAvatars,\n });\n}\n\nexport function calculateVideoDimensions(\n video: ApiVideo,\n fromOwnMessage: boolean,\n isForwarded?: boolean,\n noAvatars?: boolean,\n) {\n const { width, height } = getVideoDimensions(video) || DEFAULT_MEDIA_DIMENSIONS;\n\n return calculateDimensionsForMessageMedia({\n width,\n height,\n fromOwnMessage,\n isForwarded,\n isGif: video.isGif,\n noAvatars,\n });\n}\n\nexport function getPictogramDimensions(): ApiDimensions {\n return {\n width: 2 * REM,\n height: 2 * REM,\n };\n}\n\nexport function getDocumentThumbnailDimensions(smaller?: boolean): ApiDimensions {\n if (smaller) {\n return {\n width: 3 * REM,\n height: 3 * REM,\n };\n }\n\n return {\n width: 3.375 * REM,\n height: 3.375 * REM,\n };\n}\n\nexport function getStickerDimensions(sticker: ApiSticker): ApiDimensions {\n const { width } = sticker;\n let { height } = sticker;\n\n // For some reason this sticker has some weird `height` value\n if (sticker.id === LIKE_STICKER_ID) {\n height = width;\n }\n\n const aspectRatio = (height && width) && height / width;\n const baseWidth = REM * (\n IS_SINGLE_COLUMN_LAYOUT\n ? STICKER_SIZE_INLINE_MOBILE_FACTOR\n : STICKER_SIZE_INLINE_DESKTOP_FACTOR\n );\n const calculatedHeight = aspectRatio ? baseWidth * aspectRatio : baseWidth;\n\n if (aspectRatio && calculatedHeight > baseWidth) {\n return {\n width: Math.round(baseWidth / aspectRatio),\n height: baseWidth,\n };\n }\n\n return {\n width: baseWidth,\n height: calculatedHeight,\n };\n}\n\nexport function calculateMediaViewerDimensions(\n { width, height }: ApiDimensions, withFooter: boolean, isVideo: boolean = false,\n): ApiDimensions {\n const { width: availableWidth, height: availableHeight } = getMediaViewerAvailableDimensions(withFooter, isVideo);\n\n return calculateDimensions(availableWidth, availableHeight, width, height);\n}\n\nexport function calculateDimensions(\n availableWidth: number,\n availableHeight: number,\n mediaWidth: number,\n mediaHeight: number,\n): ApiDimensions {\n const aspectRatio = mediaHeight / mediaWidth;\n const calculatedWidth = Math.min(mediaWidth, availableWidth);\n const calculatedHeight = Math.round(calculatedWidth * aspectRatio);\n\n if (calculatedHeight > availableHeight) {\n return {\n width: Math.round(availableHeight / aspectRatio),\n height: availableHeight,\n };\n }\n\n return {\n width: calculatedWidth,\n height: Math.round(calculatedWidth * aspectRatio),\n };\n}\n","import React, {\n FC, useEffect, useRef, memo, useCallback, useState,\n} from '../../lib/teact/teact';\n\nimport { fastRaf } from '../../util/schedulers';\nimport buildClassName from '../../util/buildClassName';\nimport useHeavyAnimationCheck from '../../hooks/useHeavyAnimationCheck';\nimport useBackgroundMode from '../../hooks/useBackgroundMode';\n\ntype OwnProps = {\n className?: string;\n id: string;\n animationData: AnyLiteral;\n play?: boolean | string;\n playSegment?: [number, number];\n speed?: number;\n noLoop?: boolean;\n size: number;\n quality?: number;\n isLowPriority?: boolean;\n onLoad?: NoneToVoidFunction;\n};\n\ntype RLottieClass = typeof import('../../lib/rlottie/RLottie').default;\ntype RLottieInstance = import('../../lib/rlottie/RLottie').default;\nlet lottiePromise: Promise<RLottieClass>;\nlet RLottie: RLottieClass;\n\n// Time supposed for judges to measure \"Transferred Size\" in Dev Tools\nconst LOTTIE_LOAD_DELAY = 5000;\n\nasync function ensureLottie() {\n if (!lottiePromise) {\n lottiePromise = import('../../lib/rlottie/RLottie') as unknown as Promise<RLottieClass>;\n RLottie = (await lottiePromise as any).default;\n }\n\n return lottiePromise;\n}\n\nsetTimeout(ensureLottie, LOTTIE_LOAD_DELAY);\n\nconst AnimatedSticker: FC<OwnProps> = ({\n className,\n id,\n animationData,\n play,\n playSegment,\n speed,\n noLoop,\n size,\n quality,\n isLowPriority,\n onLoad,\n}) => {\n const [animation, setAnimation] = useState<RLottieInstance>();\n // eslint-disable-next-line no-null/no-null\n const container = useRef<HTMLDivElement>(null);\n const wasPlaying = useRef(false);\n const isFrozen = useRef(false);\n\n const playRef = useRef();\n playRef.current = play;\n const playSegmentRef = useRef<[number, number]>();\n playSegmentRef.current = playSegment;\n\n useEffect(() => {\n if (animation || !animationData) {\n return;\n }\n\n const exec = () => {\n if (!container.current) {\n return;\n }\n\n const newAnimation = new RLottie(\n id,\n container.current,\n animationData,\n {\n noLoop,\n size,\n quality,\n isLowPriority,\n },\n onLoad,\n );\n\n if (speed) {\n newAnimation.setSpeed(speed);\n }\n\n setAnimation(newAnimation);\n };\n\n if (RLottie) {\n exec();\n } else {\n ensureLottie().then(() => {\n fastRaf(() => {\n if (container.current) {\n exec();\n }\n });\n });\n }\n }, [animation, animationData, id, isLowPriority, noLoop, onLoad, quality, size, speed]);\n\n useEffect(() => {\n return () => {\n if (animation) {\n animation.destroy();\n }\n };\n }, [animation]);\n\n const playAnimation = useCallback((shouldRestart = false) => {\n if (animation && (playRef.current || playSegmentRef.current)) {\n if (playSegmentRef.current) {\n animation.playSegment(playSegmentRef.current);\n } else if (shouldRestart) {\n animation.goToAndPlay(0);\n } else {\n animation.play();\n }\n }\n }, [animation]);\n\n const pauseAnimation = useCallback(() => {\n if (!animation) {\n return;\n }\n\n animation.pause();\n }, [animation]);\n\n const freezeAnimation = useCallback(() => {\n isFrozen.current = true;\n\n if (!animation) {\n return;\n }\n\n if (!wasPlaying.current) {\n wasPlaying.current = animation.isPlaying();\n }\n\n pauseAnimation();\n }, [animation, pauseAnimation]);\n\n const unfreezeAnimation = useCallback(() => {\n if (wasPlaying.current) {\n playAnimation();\n }\n\n wasPlaying.current = false;\n isFrozen.current = false;\n }, [playAnimation]);\n\n const unfreezeAnimationOnRaf = useCallback(() => {\n fastRaf(unfreezeAnimation);\n }, [unfreezeAnimation]);\n\n useEffect(() => {\n if (!animation) {\n return;\n }\n\n if (play || playSegment) {\n if (isFrozen.current) {\n wasPlaying.current = true;\n } else {\n playAnimation(noLoop);\n }\n } else {\n // eslint-disable-next-line no-lonely-if\n if (isFrozen.current) {\n wasPlaying.current = false;\n } else {\n pauseAnimation();\n }\n }\n }, [animation, play, playSegment, noLoop, playAnimation, pauseAnimation]);\n\n useHeavyAnimationCheck(freezeAnimation, unfreezeAnimation);\n // Pausing frame may not happen in background\n // so we need to make sure it happens right after focusing,\n // then we can play again.\n useBackgroundMode(freezeAnimation, unfreezeAnimationOnRaf);\n\n const fullClassName = buildClassName('AnimatedSticker', className);\n\n const style = size ? `width: ${size}px; height: ${size}px;` : undefined;\n\n return (\n <div\n ref={container}\n className={fullClassName}\n // @ts-ignore\n style={style}\n />\n );\n};\n\nexport default memo(AnimatedSticker);\n","import React, { FC, memo } from '../../lib/teact/teact';\n\nimport useLang from '../../hooks/useLang';\nimport { TextPart } from '../common/helpers/renderMessageText';\n\nimport Modal from './Modal';\nimport Button from './Button';\n\ntype OwnProps = {\n isOpen: boolean;\n onClose: () => void;\n onCloseAnimationEnd?: () => void;\n title?: string;\n header?: FC;\n textParts?: TextPart[];\n text?: string;\n confirmLabel?: string;\n confirmHandler: () => void;\n confirmIsDestructive?: boolean;\n isButtonsInOneRow?: boolean;\n};\n\nconst ConfirmDialog: FC<OwnProps> = ({\n isOpen,\n onClose,\n onCloseAnimationEnd,\n title,\n header,\n text,\n textParts,\n confirmLabel = 'Confirm',\n confirmHandler,\n confirmIsDestructive,\n isButtonsInOneRow,\n}) => {\n const lang = useLang();\n\n return (\n <Modal\n className=\"confirm\"\n title={title}\n header={header}\n isOpen={isOpen}\n onClose={onClose}\n onCloseAnimationEnd={onCloseAnimationEnd}\n onEnter={confirmHandler}\n >\n {text && text.split('\\\\n').map((textPart) => (\n <p>{textPart}</p>\n ))}\n {textParts}\n <div className={isButtonsInOneRow ? 'dialog-buttons mt-2' : ''}>\n <Button\n className=\"confirm-dialog-button\"\n isText\n onClick={confirmHandler}\n color={confirmIsDestructive ? 'danger' : 'primary'}\n >\n {confirmLabel}\n </Button>\n <Button className=\"confirm-dialog-button\" isText onClick={onClose}>{lang('Cancel')}</Button>\n </div>\n </Modal>\n );\n};\n\nexport default memo(ConfirmDialog);\n","import React, { FC, memo } from '../../lib/teact/teact';\n\nimport buildClassName from '../../util/buildClassName';\nimport useShowTransition from '../../hooks/useShowTransition';\nimport renderText from './helpers/renderText';\nimport useLang from '../../hooks/useLang';\n\nimport './NothingFound.scss';\n\ninterface OwnProps {\n text?: string;\n description?: string;\n}\n\nconst DEFAULT_TEXT = 'Nothing found.';\n\nconst NothingFound: FC<OwnProps> = ({ text = DEFAULT_TEXT, description }) => {\n const lang = useLang();\n const { transitionClassNames } = useShowTransition(true);\n\n return (\n <div className={buildClassName('NothingFound', transitionClassNames, description && 'with-description')}>\n {text}\n {description && <p className=\"description\">{renderText(lang(description), ['br'])}</p>}\n </div>\n );\n};\n\nexport default memo(NothingFound);\n","import { useEffect, useRef } from '../../../lib/teact/teact';\n\nimport useOnChange from '../../../hooks/useOnChange';\nimport useForceUpdate from '../../../hooks/useForceUpdate';\n\nexport default function useAsyncRendering<T extends any[]>(dependencies: T, delay?: number) {\n const isDisabled = delay === undefined;\n const shouldRenderRef = useRef(isDisabled);\n const timeoutRef = useRef<number>();\n const forceUpdate = useForceUpdate();\n\n useOnChange(() => {\n if (isDisabled) {\n return;\n }\n\n shouldRenderRef.current = false;\n\n if (timeoutRef.current) {\n clearTimeout(timeoutRef.current);\n timeoutRef.current = undefined;\n }\n }, dependencies);\n\n useEffect(() => {\n if (isDisabled || shouldRenderRef.current) {\n return;\n }\n\n const exec = () => {\n shouldRenderRef.current = true;\n forceUpdate();\n };\n\n if (delay! > 0) {\n timeoutRef.current = window.setTimeout(exec, delay);\n } else {\n exec();\n }\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, dependencies);\n\n return shouldRenderRef.current;\n}\n","import {\n useEffect, useMemo, useRef, useState,\n} from '../lib/teact/teact';\n\nimport { IS_PROGRESSIVE_SUPPORTED } from '../util/environment';\nimport { ApiMediaFormat } from '../api/types';\nimport { throttle } from '../util/schedulers';\nimport * as mediaLoader from '../util/mediaLoader';\nimport useForceUpdate from './useForceUpdate';\n\nconst STREAMING_PROGRESS = 0.75;\nconst STREAMING_TIMEOUT = 1500;\nconst PROGRESS_THROTTLE = 500;\n\nexport default <T extends ApiMediaFormat = ApiMediaFormat.BlobUrl>(\n mediaHash: string | undefined,\n noLoad = false,\n // @ts-ignore (workaround for \"could be instantiated with a different subtype\" issue)\n mediaFormat: T = ApiMediaFormat.BlobUrl,\n cacheBuster?: number,\n delay?: number | false,\n) => {\n const mediaData = mediaHash ? mediaLoader.getFromMemory<T>(mediaHash) : undefined;\n const isStreaming = mediaFormat === ApiMediaFormat.Stream || (\n IS_PROGRESSIVE_SUPPORTED && mediaFormat === ApiMediaFormat.Progressive\n );\n const forceUpdate = useForceUpdate();\n const [downloadProgress, setDownloadProgress] = useState(mediaData && !isStreaming ? 1 : 0);\n const startedAtRef = useRef<number>();\n\n const handleProgress = useMemo(() => {\n return throttle((progress: number) => {\n if (!delay || (Date.now() - startedAtRef.current! > delay)) {\n setDownloadProgress(progress);\n }\n }, PROGRESS_THROTTLE, true);\n }, [delay]);\n\n useEffect(() => {\n if (!noLoad && mediaHash) {\n if (!mediaData) {\n setDownloadProgress(0);\n\n if (startedAtRef.current) {\n mediaLoader.cancelProgress(handleProgress);\n }\n\n startedAtRef.current = Date.now();\n\n mediaLoader.fetch(mediaHash, mediaFormat, handleProgress).then(() => {\n const spentTime = Date.now() - startedAtRef.current!;\n startedAtRef.current = undefined;\n\n if (!delay || spentTime >= delay) {\n forceUpdate();\n } else {\n setTimeout(forceUpdate, delay - spentTime);\n }\n });\n } else if (isStreaming) {\n setTimeout(() => {\n setDownloadProgress(STREAMING_PROGRESS);\n }, STREAMING_TIMEOUT);\n }\n }\n }, [noLoad, mediaHash, mediaData, mediaFormat, cacheBuster, forceUpdate, isStreaming, delay, handleProgress]);\n\n useEffect(() => {\n if (noLoad && startedAtRef.current) {\n mediaLoader.cancelProgress(handleProgress);\n setDownloadProgress(0);\n }\n }, [handleProgress, noLoad]);\n\n return { mediaData, downloadProgress };\n};\n","import React, {\n FC, useEffect, useRef, memo,\n} from '../../lib/teact/teact';\n\nimport buildClassName from '../../util/buildClassName';\n\nimport './ProgressSpinner.scss';\n\nconst RADIUSES = {\n s: 22, m: 25, l: 28, xl: 20,\n};\nconst STROKE_WIDTH = 2;\nconst MIN_PROGRESS = 0.05;\nconst MAX_PROGRESS = 1;\n\nconst ProgressSpinner: FC<{\n progress?: number;\n size?: 's' | 'm' | 'l' | 'xl';\n square?: boolean;\n transparent?: boolean;\n noCross?: boolean;\n onClick?: (e: React.MouseEvent<HTMLElement, MouseEvent>) => void;\n}> = ({\n progress = 0,\n size = 'l',\n square,\n transparent,\n noCross,\n onClick,\n}) => {\n const radius = RADIUSES[size];\n const circleRadius = radius - STROKE_WIDTH * 2;\n const borderRadius = radius - 1;\n const circumference = circleRadius * 2 * Math.PI;\n // eslint-disable-next-line no-null/no-null\n const container = useRef<HTMLDivElement>(null);\n\n useEffect(() => {\n if (!container.current) {\n return;\n }\n\n const svg = container.current.firstElementChild;\n const strokeDashOffset = circumference - Math.min(Math.max(MIN_PROGRESS, progress), MAX_PROGRESS) * circumference;\n\n if (!svg) {\n container.current.innerHTML = `<svg\n viewBox=\"0 0 ${borderRadius * 2} ${borderRadius * 2}\"\n height=\"${borderRadius * 2}\"\n width=\"${borderRadius * 2}\"\n >\n <circle\n stroke=\"white\"\n fill=\"transparent\"\n stroke-width=${STROKE_WIDTH}\n stroke-dasharray=\"${circumference} ${circumference}\"}\n stroke-dashoffset=\"${strokeDashOffset}\"\n stroke-linecap=\"round\"\n r=${circleRadius}\n cx=${borderRadius}\n cy=${borderRadius}\n />\n </svg>`;\n } else {\n (svg.firstElementChild as SVGElement).setAttribute('stroke-dashoffset', strokeDashOffset.toString());\n }\n }, [container, circumference, borderRadius, circleRadius, progress]);\n\n const className = buildClassName(\n `ProgressSpinner size-${size}`,\n transparent && 'transparent',\n square && 'square',\n noCross && 'no-cross',\n );\n\n return (\n <div\n ref={container}\n className={className}\n onClick={onClick}\n />\n );\n};\n\nexport default memo(ProgressSpinner);\n","import { MouseEvent as ReactMouseEvent } from 'react';\nimport React, {\n FC, memo, useEffect, useRef,\n} from '../../lib/teact/teact';\n\nimport { ApiMediaFormat, ApiSticker } from '../../api/types';\n\nimport { useIsIntersecting, ObserveFn } from '../../hooks/useIntersectionObserver';\nimport useMedia from '../../hooks/useMedia';\nimport useTransitionForMedia from '../../hooks/useTransitionForMedia';\nimport useFlag from '../../hooks/useFlag';\nimport buildClassName from '../../util/buildClassName';\n\nimport AnimatedSticker from './AnimatedSticker';\nimport Button from '../ui/Button';\n\nimport './StickerButton.scss';\n\ntype OwnProps = {\n sticker: ApiSticker;\n size: number;\n observeIntersection: ObserveFn;\n noAnimate?: boolean;\n title?: string;\n className?: string;\n onClick?: (arg: any) => void;\n clickArg?: any;\n onUnfaveClick?: (sticker: ApiSticker) => void;\n};\n\nconst StickerButton: FC<OwnProps> = ({\n sticker, size, observeIntersection, noAnimate, title, className, onClick, clickArg, onUnfaveClick,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const ref = useRef<HTMLDivElement>(null);\n\n const localMediaHash = `sticker${sticker.id}`;\n const stickerSelector = `sticker-button-${sticker.id}`;\n\n const isIntersecting = useIsIntersecting(ref, observeIntersection);\n\n const thumbDataUri = sticker.thumbnail ? sticker.thumbnail.dataUri : undefined;\n const previewBlobUrl = useMedia(`${localMediaHash}?size=m`, !isIntersecting, ApiMediaFormat.BlobUrl);\n\n const shouldPlay = isIntersecting && !noAnimate;\n const lottieData = useMedia(sticker.isAnimated && localMediaHash, !shouldPlay, ApiMediaFormat.Lottie);\n const [isAnimationLoaded, markLoaded, unmarkLoaded] = useFlag(Boolean(lottieData));\n const canAnimatedPlay = isAnimationLoaded && shouldPlay;\n\n const {\n shouldRenderThumb,\n shouldRenderFullMedia: shouldRenderPreview,\n transitionClassNames: previewTransitionClassNames,\n } = useTransitionForMedia(previewBlobUrl || canAnimatedPlay, 'slow');\n\n // To avoid flickering\n useEffect(() => {\n if (!shouldPlay) {\n unmarkLoaded();\n }\n }, [unmarkLoaded, shouldPlay]);\n\n function handleClick() {\n if (onClick) {\n onClick(clickArg);\n }\n }\n\n function handleUnfaveClick(e: ReactMouseEvent<HTMLButtonElement, MouseEvent>) {\n e.stopPropagation();\n e.preventDefault();\n\n onUnfaveClick!(sticker);\n }\n\n const fullClassName = buildClassName(\n 'StickerButton',\n sticker.isAnimated && 'animated',\n stickerSelector,\n className,\n );\n\n const style = shouldRenderThumb && thumbDataUri ? `background-image: url('${thumbDataUri}');` : '';\n\n return (\n <div\n ref={ref}\n className={fullClassName}\n title={title || (sticker && sticker.emoji)}\n // @ts-ignore\n style={style}\n data-sticker-id={sticker.id}\n onClick={handleClick}\n >\n {shouldRenderPreview && !canAnimatedPlay && (\n // eslint-disable-next-line jsx-a11y/alt-text\n <img src={previewBlobUrl} className={previewTransitionClassNames} />\n )}\n {shouldPlay && lottieData && (\n <AnimatedSticker\n id={localMediaHash}\n animationData={lottieData}\n play\n size={size}\n isLowPriority\n onLoad={markLoaded}\n />\n )}\n {onUnfaveClick && (\n <Button\n className=\"sticker-unfave-button\"\n color=\"dark\"\n round\n onClick={handleUnfaveClick}\n >\n <i className=\"icon-close\" />\n </Button>\n )}\n </div>\n );\n};\n\nexport default memo(StickerButton);\n","import { ChangeEvent } from 'react';\nimport React, { FC, memo } from '../../lib/teact/teact';\n\nimport buildClassName from '../../util/buildClassName';\nimport useLang from '../../hooks/useLang';\n\nimport Spinner from './Spinner';\n\nimport './Radio.scss';\n\ntype OwnProps = {\n id?: string;\n name: string;\n label: string;\n subLabel?: string;\n value: string;\n checked: boolean;\n disabled?: boolean;\n hidden?: boolean;\n isLoading?: boolean;\n onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n};\n\nconst Radio: FC<OwnProps> = ({\n id,\n label,\n subLabel,\n value,\n name,\n checked,\n disabled,\n hidden,\n isLoading,\n onChange,\n}) => {\n const lang = useLang();\n const className = buildClassName(\n 'Radio',\n disabled && 'disabled',\n hidden && 'hidden-widget',\n isLoading && 'loading',\n );\n\n return (\n <label className={className} dir={lang.isRtl ? 'rtl' : undefined}>\n <input\n type=\"radio\"\n name={name}\n value={value}\n id={id}\n checked={checked}\n onChange={onChange}\n disabled={disabled || hidden}\n />\n <div className=\"Radio-main\">\n <span className=\"label\" dir={lang.isRtl ? 'auto' : undefined}>{label}</span>\n {subLabel && <span className=\"subLabel\" dir={lang.isRtl ? 'auto' : undefined}>{subLabel}</span>}\n </div>\n {isLoading && <Spinner />}\n </label>\n );\n};\n\nexport default memo(Radio);\n","import { ChangeEvent } from 'react';\nimport React, { FC, useCallback, memo } from '../../lib/teact/teact';\n\nimport Radio from './Radio';\n\ntype IRadioOption = {\n label: string;\n subLabel?: string;\n value: string;\n hidden?: boolean;\n};\n\ntype OwnProps = {\n id?: string;\n name: string;\n options: IRadioOption[];\n selected?: string;\n disabled?: boolean;\n loadingOption?: string;\n onChange: (value: string) => void;\n};\n\nconst RadioGroup: FC<OwnProps> = ({\n id,\n name,\n options,\n selected,\n disabled,\n loadingOption,\n onChange,\n}) => {\n const handleChange = useCallback((event: ChangeEvent<HTMLInputElement>) => {\n const { value } = event.currentTarget;\n onChange(value);\n }, [onChange]);\n\n return (\n <div id={id} className=\"radio-group\">\n {options.map((option) => (\n <Radio\n name={name}\n label={option.label}\n subLabel={option.subLabel}\n value={option.value}\n checked={option.value === selected}\n hidden={option.hidden}\n disabled={disabled}\n isLoading={loadingOption ? loadingOption === option.value : undefined}\n onChange={handleChange}\n />\n ))}\n </div>\n );\n};\n\nexport default memo(RadioGroup);\n","import { useEffect } from '../lib/teact/teact';\n\nexport default function useBackgroundMode(\n onBlur?: AnyToVoidFunction,\n onFocus?: AnyToVoidFunction,\n) {\n useEffect(() => {\n if (onBlur && !document.hasFocus()) {\n onBlur();\n }\n\n if (onBlur) {\n window.addEventListener('blur', onBlur);\n }\n\n if (onFocus) {\n window.addEventListener('focus', onFocus);\n }\n\n return () => {\n if (onFocus) {\n window.removeEventListener('focus', onFocus);\n }\n\n if (onBlur) {\n window.removeEventListener('blur', onBlur);\n }\n };\n }, [onBlur, onFocus]);\n}\n","import { FC, useRef, useLayoutEffect } from '../../lib/teact/teact';\nimport TeactDOM from '../../lib/teact/teact-dom';\n\ntype OwnProps = {\n containerId?: string;\n className?: string;\n children: any;\n};\n\nconst Portal: FC<OwnProps> = ({ containerId, className, children }) => {\n const elementRef = useRef(document.createElement('div'));\n\n useLayoutEffect(() => {\n const container = document.querySelector<HTMLDivElement>(containerId || '#portals');\n if (!container) {\n return undefined;\n }\n\n const element = elementRef.current;\n if (className) {\n element.classList.add(className);\n }\n\n container.appendChild(element);\n\n return () => {\n TeactDOM.render(undefined, element);\n container.removeChild(element);\n };\n }, [className, containerId]);\n\n return TeactDOM.render(children, elementRef.current);\n};\n\nexport default Portal;\n","import React, {\n FC, useCallback, useRef, useState,\n} from '../../lib/teact/teact';\n\nimport { ApiMediaFormat, ApiSticker } from '../../api/types';\n\nimport { LIKE_STICKER_ID } from './helpers/mediaDimensions';\nimport { ObserveFn, useIsIntersecting } from '../../hooks/useIntersectionObserver';\nimport useMedia from '../../hooks/useMedia';\nimport useTransitionForMedia from '../../hooks/useTransitionForMedia';\nimport useFlag from '../../hooks/useFlag';\n\nimport AnimatedSticker from './AnimatedSticker';\n\nimport './AnimatedEmoji.scss';\n\ntype OwnProps = {\n sticker: ApiSticker;\n observeIntersection?: ObserveFn;\n size?: 'large' | 'medium' | 'small';\n lastSyncTime?: number;\n forceLoadPreview?: boolean;\n};\n\nconst QUALITY = 1;\nconst WIDTH = {\n large: 160,\n medium: 128,\n small: 104,\n};\n\nconst AnimatedEmoji: FC<OwnProps> = ({\n sticker,\n size = 'medium',\n observeIntersection,\n lastSyncTime,\n forceLoadPreview,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const ref = useRef<HTMLDivElement>(null);\n\n const [isAnimationLoaded, markAnimationLoaded] = useFlag();\n const localMediaHash = `sticker${sticker.id}`;\n\n const isIntersecting = useIsIntersecting(ref, observeIntersection);\n\n const thumbDataUri = sticker.thumbnail && sticker.thumbnail.dataUri;\n const previewBlobUrl = useMedia(\n `${localMediaHash}?size=m`,\n !isIntersecting && !forceLoadPreview,\n ApiMediaFormat.BlobUrl,\n lastSyncTime,\n );\n const { transitionClassNames } = useTransitionForMedia(previewBlobUrl, 'slow');\n\n const mediaData = useMedia(localMediaHash, !isIntersecting, ApiMediaFormat.Lottie, lastSyncTime);\n const isMediaLoaded = Boolean(mediaData);\n\n const [playKey, setPlayKey] = useState(String(Math.random()));\n const handleClick = useCallback(() => {\n setPlayKey(String(Math.random()));\n }, []);\n\n const width = WIDTH[size];\n const style = `width: ${width}px; height: ${width}px;`;\n\n return (\n <div\n ref={ref}\n className=\"AnimatedEmoji media-inner\"\n // @ts-ignore\n style={style}\n onClick={handleClick}\n >\n {!isAnimationLoaded && thumbDataUri && (\n <img src={thumbDataUri} className={sticker.id === LIKE_STICKER_ID ? 'like-sticker-thumb' : undefined} alt=\"\" />\n )}\n {!isAnimationLoaded && previewBlobUrl && (\n <img src={previewBlobUrl} className={transitionClassNames} alt=\"\" />\n )}\n {isMediaLoaded && (\n <AnimatedSticker\n key={localMediaHash}\n id={localMediaHash}\n animationData={mediaData as AnyLiteral}\n size={width}\n quality={QUALITY}\n play={isIntersecting && playKey}\n noLoop\n onLoad={markAnimationLoaded}\n />\n )}\n </div>\n );\n};\n\nexport default AnimatedEmoji;\n","import { MouseEvent as ReactMouseEvent } from 'react';\nimport React, {\n FC, useEffect, useCallback, memo,\n} from '../../lib/teact/teact';\nimport { withGlobal } from '../../lib/teact/teactn';\n\nimport { ApiChat, ApiTypingStatus } from '../../api/types';\nimport { GlobalActions, GlobalState } from '../../global/types';\nimport { MediaViewerOrigin } from '../../types';\n\nimport {\n getChatTypeString,\n getChatTitle,\n isChatSuperGroup,\n} from '../../modules/helpers';\nimport { selectChat, selectChatMessages, selectChatOnlineCount } from '../../modules/selectors';\nimport renderText from './helpers/renderText';\nimport { pick } from '../../util/iteratees';\nimport useLang, { LangFn } from '../../hooks/useLang';\n\nimport Avatar from './Avatar';\nimport VerifiedIcon from './VerifiedIcon';\nimport TypingStatus from './TypingStatus';\n\ntype OwnProps = {\n chatId: number;\n typingStatus?: ApiTypingStatus;\n avatarSize?: 'small' | 'medium' | 'large' | 'jumbo';\n withMediaViewer?: boolean;\n withUsername?: boolean;\n withFullInfo?: boolean;\n withUpdatingStatus?: boolean;\n withChatType?: boolean;\n noRtl?: boolean;\n};\n\ntype StateProps = {\n chat?: ApiChat;\n onlineCount?: number;\n areMessagesLoaded: boolean;\n} & Pick<GlobalState, 'lastSyncTime'>;\n\ntype DispatchProps = Pick<GlobalActions, 'loadFullChat' | 'openMediaViewer'>;\n\nconst GroupChatInfo: FC<OwnProps & StateProps & DispatchProps> = ({\n typingStatus,\n avatarSize = 'medium',\n withMediaViewer,\n withUsername,\n withFullInfo,\n withUpdatingStatus,\n withChatType,\n noRtl,\n chat,\n onlineCount,\n areMessagesLoaded,\n lastSyncTime,\n loadFullChat,\n openMediaViewer,\n}) => {\n const isSuperGroup = chat && isChatSuperGroup(chat);\n const { id: chatId, isMin, isRestricted } = chat || {};\n\n useEffect(() => {\n if (chatId && !isMin && withFullInfo && lastSyncTime) {\n loadFullChat({ chatId });\n }\n }, [chatId, isMin, lastSyncTime, withFullInfo, loadFullChat, isSuperGroup]);\n\n const handleAvatarViewerOpen = useCallback((e: ReactMouseEvent<HTMLDivElement, MouseEvent>, hasPhoto: boolean) => {\n if (chat && hasPhoto) {\n e.stopPropagation();\n openMediaViewer({\n avatarOwnerId: chat.id,\n origin: avatarSize === 'jumbo' ? MediaViewerOrigin.ProfileAvatar : MediaViewerOrigin.MiddleHeaderAvatar,\n });\n }\n }, [chat, avatarSize, openMediaViewer]);\n\n const lang = useLang();\n\n if (!chat) {\n return undefined;\n }\n\n function renderStatusOrTyping() {\n if (withUpdatingStatus && !areMessagesLoaded && !isRestricted) {\n return (\n <span className=\"status\" dir=\"auto\">{lang('Updating')}</span>\n );\n }\n\n if (!chat) {\n return undefined;\n }\n\n if (typingStatus) {\n return <TypingStatus typingStatus={typingStatus} />;\n }\n\n if (withChatType) {\n return (\n <div className=\"status\" dir=\"auto\">{lang(getChatTypeString(chat))}</div>\n );\n }\n\n const handle = withUsername ? chat.username : undefined;\n const groupStatus = getGroupStatus(lang, chat);\n const onlineStatus = onlineCount ? `, ${lang('OnlineCount', onlineCount, 'i')}` : undefined;\n\n return (\n <div className=\"status\">\n {handle && <span className=\"handle\">{handle}</span>}\n <span className=\"group-status\">{groupStatus}</span>\n {onlineStatus && <span className=\"online-status\">{onlineStatus}</span>}\n </div>\n );\n }\n\n return (\n <div className=\"ChatInfo\" dir={!noRtl && lang.isRtl ? 'rtl' : undefined}>\n <Avatar\n key={chat.id}\n size={avatarSize}\n chat={chat}\n onClick={withMediaViewer ? handleAvatarViewerOpen : undefined}\n />\n <div className=\"info\">\n <div className=\"title\">\n <h3 dir=\"auto\">{renderText(getChatTitle(lang, chat))}</h3>\n {chat.isVerified && <VerifiedIcon />}\n </div>\n {renderStatusOrTyping()}\n </div>\n </div>\n );\n};\n\nfunction getGroupStatus(lang: LangFn, chat: ApiChat) {\n const chatTypeString = lang(getChatTypeString(chat));\n const { membersCount } = chat;\n\n if (chat.isRestricted) {\n return chatTypeString === 'Channel' ? 'channel is inaccessible' : 'group is inaccessible';\n }\n\n if (!membersCount) {\n return chatTypeString;\n }\n\n return chatTypeString === 'Channel'\n ? lang('Subscribers', membersCount, 'i')\n : lang('Members', membersCount, 'i');\n}\n\nexport default memo(withGlobal<OwnProps>(\n (global, { chatId }): StateProps => {\n const { lastSyncTime } = global;\n const chat = selectChat(global, chatId);\n const onlineCount = chat ? selectChatOnlineCount(global, chat) : undefined;\n const areMessagesLoaded = Boolean(selectChatMessages(global, chatId));\n\n return {\n lastSyncTime, chat, onlineCount, areMessagesLoaded,\n };\n },\n (setGlobal, actions): DispatchProps => pick(actions, ['loadFullChat', 'openMediaViewer']),\n)(GroupChatInfo));\n","import { getGlobal } from '../lib/teact/teactn';\n\nimport { FocusDirection } from '../types';\n\nimport {\n ANIMATION_LEVEL_MIN,\n FAST_SMOOTH_MAX_DISTANCE, FAST_SMOOTH_MAX_DURATION, FAST_SMOOTH_MIN_DURATION,\n FAST_SMOOTH_SHORT_TRANSITION_MAX_DISTANCE,\n} from '../config';\nimport { dispatchHeavyAnimationEvent } from '../hooks/useHeavyAnimationCheck';\nimport { animateSingle } from './animation';\n\nlet isAnimating = false;\n\nexport default function fastSmoothScroll(\n container: HTMLElement,\n element: HTMLElement,\n position: ScrollLogicalPosition | 'centerOrTop',\n margin = 0,\n maxDistance = FAST_SMOOTH_MAX_DISTANCE,\n forceDirection?: FocusDirection,\n forceDuration?: number,\n forceCurrentContainerHeight?: boolean,\n) {\n const scrollFrom = calculateScrollFrom(container, element, maxDistance, forceDirection);\n\n if (forceDirection === FocusDirection.Static) {\n scrollWithJs(container, element, scrollFrom, position, margin, 0);\n return;\n }\n\n\n if (getGlobal().settings.byKey.animationLevel === ANIMATION_LEVEL_MIN) {\n forceDuration = 0;\n }\n\n scrollWithJs(container, element, scrollFrom, position, margin, forceDuration, forceCurrentContainerHeight);\n}\n\nexport function isAnimatingScroll() {\n return isAnimating;\n}\n\nfunction calculateScrollFrom(\n container: HTMLElement,\n element: HTMLElement,\n maxDistance = FAST_SMOOTH_MAX_DISTANCE,\n forceDirection?: FocusDirection,\n) {\n const { offsetTop: elementTop } = element;\n const { scrollTop } = container;\n\n if (forceDirection === undefined) {\n const offset = elementTop - container.scrollTop;\n\n if (offset < -maxDistance) {\n return scrollTop + (offset + maxDistance);\n } else if (offset > maxDistance) {\n return scrollTop + (offset - maxDistance);\n }\n } else if (forceDirection === FocusDirection.Up) {\n return elementTop + maxDistance;\n } else if (forceDirection === FocusDirection.Down) {\n return Math.max(0, elementTop - maxDistance);\n }\n\n return scrollTop;\n}\n\nfunction scrollWithJs(\n container: HTMLElement,\n element: HTMLElement,\n scrollFrom: number,\n position: ScrollLogicalPosition | 'centerOrTop',\n margin = 0,\n forceDuration?: number,\n forceCurrentContainerHeight?: boolean,\n) {\n const { offsetTop: elementTop, offsetHeight: elementHeight } = element;\n const { scrollTop: currentScrollTop, offsetHeight: containerHeight, scrollHeight } = container;\n const targetContainerHeight = !forceCurrentContainerHeight && container.dataset.normalHeight\n ? Number(container.dataset.normalHeight)\n : containerHeight;\n\n if (currentScrollTop !== scrollFrom) {\n container.scrollTop = scrollFrom;\n }\n\n let path!: number;\n\n switch (position) {\n case 'start':\n path = (elementTop - margin) - scrollFrom;\n break;\n case 'end':\n path = (elementTop + elementHeight + margin) - (scrollFrom + targetContainerHeight);\n break;\n // 'nearest' is not supported yet\n case 'nearest':\n case 'center':\n case 'centerOrTop':\n path = elementHeight < targetContainerHeight\n ? (elementTop + elementHeight / 2) - (scrollFrom + targetContainerHeight / 2)\n : (elementTop - margin) - scrollFrom;\n break;\n }\n\n if (path < 0) {\n const remainingPath = -scrollFrom;\n path = Math.max(path, remainingPath);\n } else if (path > 0) {\n const remainingPath = scrollHeight - (scrollFrom + targetContainerHeight);\n path = Math.min(path, remainingPath);\n }\n\n if (path === 0) {\n return;\n }\n\n const target = scrollFrom + path;\n\n if (forceDuration === 0) {\n container.scrollTop = target;\n return;\n }\n\n isAnimating = true;\n\n const absPath = Math.abs(path);\n const transition = absPath < FAST_SMOOTH_SHORT_TRANSITION_MAX_DISTANCE ? shortTransition : longTransition;\n const duration = forceDuration || (\n FAST_SMOOTH_MIN_DURATION\n + (absPath / FAST_SMOOTH_MAX_DISTANCE) * (FAST_SMOOTH_MAX_DURATION - FAST_SMOOTH_MIN_DURATION)\n );\n const startAt = Date.now();\n const onHeavyAnimationStop = dispatchHeavyAnimationEvent();\n\n animateSingle(() => {\n const t = Math.min((Date.now() - startAt) / duration, 1);\n const currentPath = path * (1 - transition(t));\n\n container.scrollTop = Math.round(target - currentPath);\n\n isAnimating = t < 1;\n\n if (!isAnimating) {\n onHeavyAnimationStop();\n }\n\n return isAnimating;\n });\n}\n\nfunction longTransition(t: number) {\n return 1 - ((1 - t) ** 5);\n}\n\nfunction shortTransition(t: number) {\n return 1 - ((1 - t) ** 3.5);\n}\n","import { useCallback, useEffect, useRef } from '../lib/teact/teact';\nimport { LoadMoreDirection } from '../types';\n\nimport { areSortedArraysEqual } from '../util/iteratees';\nimport useForceUpdate from './useForceUpdate';\nimport usePrevious from './usePrevious';\n\ntype GetMore = (args: { direction: LoadMoreDirection }) => void;\ntype LoadMoreBackwards = (args: { offsetId?: number }) => void;\n\nconst DEFAULT_LIST_SLICE = 30;\n\nexport default (\n loadMoreBackwards?: LoadMoreBackwards,\n listIds?: number[],\n isDisabled = false,\n listSlice = DEFAULT_LIST_SLICE,\n forceFullPreload = false,\n): [number[]?, GetMore?] => {\n const lastParamsRef = useRef<{\n direction?: LoadMoreDirection;\n offsetId?: number;\n }>();\n\n const viewportIdsRef = useRef<number[] | undefined>((() => {\n // Only run once to initialize\n if (!listIds || lastParamsRef.current) {\n return undefined;\n }\n\n const { newViewportIds } = getViewportSlice(listIds, listIds[0], LoadMoreDirection.Forwards, listSlice);\n return newViewportIds;\n })());\n\n const forceUpdate = useForceUpdate();\n\n const prevListIds = usePrevious(listIds);\n const prevIsDisabled = usePrevious(isDisabled);\n if (listIds && !isDisabled && (listIds !== prevListIds || isDisabled !== prevIsDisabled)) {\n const { offsetId = listIds[0], direction = LoadMoreDirection.Forwards } = lastParamsRef.current || {};\n const { newViewportIds } = getViewportSlice(listIds, offsetId, direction, listSlice);\n\n if (!viewportIdsRef.current || !areSortedArraysEqual(viewportIdsRef.current, newViewportIds)) {\n viewportIdsRef.current = newViewportIds;\n }\n }\n\n useEffect(() => {\n if (listIds && !isDisabled && loadMoreBackwards && forceFullPreload) {\n const viewportIds = viewportIdsRef.current!;\n loadMoreBackwards({ offsetId: viewportIds[viewportIds.length - 1] });\n }\n }, [listIds, isDisabled, loadMoreBackwards, forceFullPreload]);\n\n const getMore: GetMore = useCallback(({\n direction,\n noScroll,\n }: { direction: LoadMoreDirection; noScroll?: boolean }) => {\n const viewportIds = viewportIdsRef.current;\n\n const offsetId = viewportIds\n ? direction === LoadMoreDirection.Backwards ? viewportIds[viewportIds.length - 1] : viewportIds[0]\n : undefined;\n\n if (!listIds) {\n if (loadMoreBackwards) {\n loadMoreBackwards({ offsetId });\n }\n\n return;\n }\n\n if (!noScroll) {\n lastParamsRef.current = { ...lastParamsRef.current, direction, offsetId };\n }\n\n const {\n newViewportIds, areSomeLocal, areAllLocal,\n } = getViewportSlice(listIds, offsetId, direction, listSlice);\n\n if (areSomeLocal && !(viewportIds && areSortedArraysEqual(viewportIds, newViewportIds))) {\n viewportIdsRef.current = newViewportIds;\n forceUpdate();\n }\n\n if (!areAllLocal && loadMoreBackwards) {\n loadMoreBackwards({ offsetId });\n }\n }, [listIds, listSlice, loadMoreBackwards, forceUpdate]);\n\n return isDisabled ? [listIds] : [viewportIdsRef.current, getMore];\n};\n\nfunction getViewportSlice(\n sourceIds: number[],\n offsetId = 0,\n direction: LoadMoreDirection,\n listSlice: number,\n) {\n const { length } = sourceIds;\n const index = sourceIds.indexOf(offsetId);\n const isForwards = direction === LoadMoreDirection.Forwards;\n const indexForDirection = isForwards ? index : (index + 1) || length;\n const from = Math.max(0, indexForDirection - listSlice);\n const to = indexForDirection + listSlice - 1;\n const newViewportIds = sourceIds.slice(Math.max(0, from), to + 1);\n\n let areSomeLocal;\n let areAllLocal;\n switch (direction) {\n case LoadMoreDirection.Forwards:\n areSomeLocal = indexForDirection > 0;\n areAllLocal = from >= 0;\n break;\n case LoadMoreDirection.Backwards:\n areSomeLocal = indexForDirection < length;\n areAllLocal = to <= length - 1;\n break;\n }\n\n return { newViewportIds, areSomeLocal, areAllLocal };\n}\n","export default function cycleRestrict(length: number, index: number) {\n return index - Math.floor(index / length) * length;\n}\n","import { IS_TOUCH_ENV } from './environment';\n\nexport default function focusEditableElement(element: HTMLElement, force?: boolean) {\n if (!force && element === document.activeElement) {\n return;\n }\n\n const selection = window.getSelection()!;\n const range = document.createRange();\n const lastChild = element.lastChild || element;\n\n if (!IS_TOUCH_ENV && (!lastChild || !lastChild.nodeValue)) {\n element.focus();\n return;\n }\n\n range.selectNodeContents(lastChild);\n // `false` means collapse to the end rather than the start\n range.collapse(false);\n selection.removeAllRanges();\n selection.addRange(range);\n}\n","export enum SwipeDirection {\n Up,\n Down,\n Left,\n Right,\n}\n\ninterface CaptureOptions {\n onCapture?: (e: MouseEvent | TouchEvent) => void;\n onRelease?: (e: MouseEvent | TouchEvent) => void;\n onDrag?: (\n e: MouseEvent | TouchEvent,\n captureEvent: MouseEvent | TouchEvent,\n params: {\n dragOffsetX: number;\n dragOffsetY: number;\n },\n ) => void;\n onSwipe?: (e: Event, direction: SwipeDirection) => void;\n onClick?: (e: MouseEvent | TouchEvent) => void;\n excludedClosestSelector?: string;\n withCursor?: boolean;\n}\n\n// https://stackoverflow.com/questions/11287877/how-can-i-get-e-offsetx-on-mobile-ipad\n// Android does not have this value, and iOS has it but as read-only\nexport interface RealTouchEvent extends TouchEvent {\n pageX?: number;\n pageY?: number;\n}\n\ntype TSwipeAxis = 'x' | 'y' | undefined;\n\nconst MOVED_THRESHOLD = 15;\nconst SWIPE_THRESHOLD = 50;\n\nexport function captureEvents(element: HTMLElement, options: CaptureOptions) {\n let captureEvent: MouseEvent | RealTouchEvent | undefined;\n let hasMoved = false;\n let currentSwipeAxis: TSwipeAxis;\n\n function onCapture(e: MouseEvent | RealTouchEvent) {\n if (options.excludedClosestSelector && (\n (e.target as HTMLElement).matches(options.excludedClosestSelector)\n || (e.target as HTMLElement).closest(options.excludedClosestSelector)\n )) {\n return;\n }\n\n captureEvent = e;\n\n if (e.type === 'mousedown') {\n document.addEventListener('mousemove', onMove);\n document.addEventListener('mouseup', onRelease);\n } else if (e.type === 'touchstart') {\n document.addEventListener('touchmove', onMove);\n document.addEventListener('touchend', onRelease);\n document.addEventListener('touchcancel', onRelease);\n\n if ('touches' in e) {\n if (e.pageX === undefined) {\n e.pageX = e.touches[0].pageX;\n }\n\n if (e.pageY === undefined) {\n e.pageY = e.touches[0].pageY;\n }\n }\n }\n\n document.body.classList.add('no-selection');\n if (options.withCursor) {\n document.body.classList.add('cursor-grabbing');\n }\n\n if (options.onCapture) {\n options.onCapture(e);\n }\n }\n\n function onRelease(e: MouseEvent | TouchEvent) {\n if (captureEvent) {\n if (options.withCursor) {\n document.body.classList.remove('cursor-grabbing');\n }\n document.body.classList.remove('no-selection');\n\n document.removeEventListener('mouseup', onRelease);\n document.removeEventListener('mousemove', onMove);\n document.removeEventListener('touchcancel', onRelease);\n document.removeEventListener('touchend', onRelease);\n document.removeEventListener('touchmove', onMove);\n\n captureEvent = undefined;\n\n if (hasMoved) {\n if (options.onRelease) {\n options.onRelease(e);\n }\n } else if (options.onClick && (!('button' in e) || e.button === 0)) {\n options.onClick(e);\n }\n }\n\n hasMoved = false;\n currentSwipeAxis = undefined;\n }\n\n function onMove(e: MouseEvent | RealTouchEvent) {\n if (captureEvent) {\n if (e.type === 'touchmove' && ('touches' in e)) {\n if (e.pageX === undefined) {\n e.pageX = e.touches[0].pageX;\n }\n\n if (e.pageY === undefined) {\n e.pageY = e.touches[0].pageY;\n }\n }\n\n const dragOffsetX = e.pageX! - captureEvent.pageX!;\n const dragOffsetY = e.pageY! - captureEvent.pageY!;\n\n if (Math.abs(dragOffsetX) >= MOVED_THRESHOLD || Math.abs(dragOffsetY) >= MOVED_THRESHOLD) {\n hasMoved = true;\n }\n\n if (options.onDrag) {\n e.preventDefault();\n options.onDrag(e, captureEvent, { dragOffsetX, dragOffsetY });\n }\n\n if (options.onSwipe) {\n onSwipe(e, dragOffsetX, dragOffsetY);\n }\n }\n }\n\n function onSwipe(e: Event, dragOffsetX: number, dragOffsetY: number) {\n if (!currentSwipeAxis) {\n const xAbs = Math.abs(dragOffsetX);\n const yAbs = Math.abs(dragOffsetY);\n\n if (dragOffsetX && dragOffsetY) {\n const ratio = Math.max(xAbs, yAbs) / Math.min(xAbs, yAbs);\n // Diagonal swipe\n if (ratio < 2) {\n return;\n }\n }\n\n if (xAbs >= SWIPE_THRESHOLD) {\n currentSwipeAxis = 'x';\n } else if (yAbs >= SWIPE_THRESHOLD) {\n currentSwipeAxis = 'y';\n }\n }\n\n processSwipe(e, currentSwipeAxis, dragOffsetX, dragOffsetY, options.onSwipe!);\n }\n\n element.addEventListener('mousedown', onCapture);\n element.addEventListener('touchstart', onCapture, { passive: true });\n\n return () => {\n element.removeEventListener('mousedown', onCapture);\n element.removeEventListener('touchstart', onCapture);\n };\n}\n\nfunction processSwipe(\n e: Event,\n currentSwipeAxis:TSwipeAxis,\n dragOffsetX: number,\n dragOffsetY: number,\n onSwipe: (e: Event, direction: SwipeDirection) => void,\n) {\n if (currentSwipeAxis === 'x') {\n if (dragOffsetX < 0) {\n onSwipe(e, SwipeDirection.Left);\n } else {\n onSwipe(e, SwipeDirection.Right);\n }\n } else if (currentSwipeAxis === 'y') {\n if (dragOffsetY < 0) {\n onSwipe(e, SwipeDirection.Up);\n } else {\n onSwipe(e, SwipeDirection.Down);\n }\n }\n}\n","import React, { FC } from '../../lib/teact/teact';\n\nimport './VerifiedIcon.scss';\n\nconst VerifiedIcon: FC = () => {\n return (\n <span className=\"VerifiedIcon\" />\n );\n};\n\nexport default VerifiedIcon;\n","import usePrevious from './usePrevious';\n\nexport default function useCurrentOrPrev<T extends any>(\n current: T, shouldSkipUndefined = false, shouldForceCurrent = false,\n): T | undefined {\n const prev = usePrevious(current, shouldSkipUndefined);\n\n // eslint-disable-next-line no-null/no-null\n return shouldForceCurrent || (current !== null && current !== undefined) ? current : prev;\n}\n","import { DEBUG } from '../config';\n\nexport default (mediaEl: HTMLMediaElement) => {\n mediaEl.play().catch((err) => {\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.warn(err);\n }\n });\n};\n","import React, { useCallback, useMemo, useState } from '../lib/teact/teact';\nimport { debounce } from '../util/schedulers';\nimport { isSafariPatchInProgress } from '../util/patchSafariProgressiveAudio';\n\ntype BufferingEvent = (e: Event | React.SyntheticEvent<HTMLMediaElement>) => void;\n\nconst MIN_READY_STATE = 3;\n// Avoid flickering when re-mounting previously buffered video\nconst DEBOUNCE = 200;\n\nexport default (noInitiallyBuffered = false) => {\n const [isBuffered, setIsBuffered] = useState(!noInitiallyBuffered);\n const [bufferedProgress, setBufferedProgress] = useState(0);\n\n const setIsBufferedDebounced = useMemo(() => {\n return debounce(setIsBuffered, DEBOUNCE, false, true);\n }, []);\n\n const handleBuffering = useCallback<BufferingEvent>((e) => {\n const media = e.currentTarget as HTMLMediaElement;\n\n if (!isSafariPatchInProgress(media)) {\n if (media.buffered.length) {\n setBufferedProgress(media.buffered.end(0) / media.duration);\n }\n\n setIsBufferedDebounced(media.readyState >= MIN_READY_STATE || media.currentTime > 0);\n }\n }, [setIsBufferedDebounced]);\n\n const bufferingHandlers = {\n onLoadedData: handleBuffering,\n onPlaying: handleBuffering,\n onLoadStart: handleBuffering, // Needed for Safari to start\n onPause: handleBuffering, // Needed for Chrome when seeking\n onTimeUpdate: handleBuffering, // Needed for audio buffering progress\n onProgress: handleBuffering, // Needed for video buffering progress\n };\n\n return {\n isBuffered,\n bufferedProgress,\n bufferingHandlers,\n checkBuffering(element: HTMLMediaElement) {\n setIsBufferedDebounced(element.readyState >= MIN_READY_STATE);\n },\n };\n};\n","import { ApiMessageEntity, ApiMessageEntityTypes, ApiFormattedText } from '../../../../api/types';\nimport { IS_EMOJI_SUPPORTED } from '../../../../util/environment';\n\nconst ENTITY_CLASS_BY_NODE_NAME: Record<string, string> = {\n B: ApiMessageEntityTypes.Bold,\n STRONG: ApiMessageEntityTypes.Bold,\n I: ApiMessageEntityTypes.Italic,\n EM: ApiMessageEntityTypes.Italic,\n U: ApiMessageEntityTypes.Underline,\n S: ApiMessageEntityTypes.Strike,\n STRIKE: ApiMessageEntityTypes.Strike,\n DEL: ApiMessageEntityTypes.Strike,\n CODE: ApiMessageEntityTypes.Code,\n PRE: ApiMessageEntityTypes.Pre,\n BLOCKQUOTE: ApiMessageEntityTypes.Blockquote,\n};\n\nconst MAX_TAG_DEEPNESS = 3;\nconst MAX_MESSAGE_LENGTH = 4096;\n\nexport default function parseMessageInput(html: string): ApiFormattedText {\n const fragment = document.createElement('div');\n fragment.innerHTML = parseMarkdown(html);\n const text = fragment.innerText.trim().replace(/\\u200b+/g, '').slice(0, MAX_MESSAGE_LENGTH);\n let textIndex = 0;\n let recursionDeepness = 0;\n const entities: ApiMessageEntity[] = [];\n\n function addEntity(node: ChildNode) {\n const { index, entity } = getEntityDataFromNode(node, text, textIndex);\n\n if (entity) {\n textIndex = index;\n entities.push(entity);\n } else if (node.textContent) {\n textIndex += node.textContent.length;\n }\n\n if (node.hasChildNodes() && recursionDeepness <= MAX_TAG_DEEPNESS) {\n recursionDeepness += 1;\n Array.from(node.childNodes).forEach(addEntity);\n }\n }\n\n Array.from(fragment.childNodes).forEach((node) => {\n recursionDeepness = 1;\n addEntity(node);\n });\n\n return {\n text,\n entities: entities.length ? entities : undefined,\n };\n}\n\nfunction parseMarkdown(html: string) {\n let parsedHtml = html.slice(0);\n\n if (!IS_EMOJI_SUPPORTED) {\n // Emojis\n parsedHtml = parsedHtml.replace(/<img[^>]+alt=\"([^\"]+)\"[^>]*>/gm, '$1');\n }\n\n // Strip redundant <span> tags\n parsedHtml = parsedHtml.replace(/<\\/?span([^>]*)?>/g, '');\n\n // Strip redundant nbsp's\n parsedHtml = parsedHtml.replace(/ /g, ' ');\n\n // Replace <div><br></div> with newline (new line in Safari)\n parsedHtml = parsedHtml.replace(/<div><br([^>]*)?><\\/div>/g, '\\n');\n // Replace <br> with newline\n parsedHtml = parsedHtml.replace(/<br([^>]*)?>/g, '\\n');\n\n // Strip redundant <div> tags\n parsedHtml = parsedHtml.replace(/<\\/div>(\\s*)<div>/g, '\\n');\n parsedHtml = parsedHtml.replace(/<div>/g, '\\n');\n parsedHtml = parsedHtml.replace(/<\\/div>/g, '');\n\n // Pre\n parsedHtml = parsedHtml.replace(/^`{3}(.*[\\n\\r][^]*?^)`{3}/gm, '<pre>$1</pre>');\n parsedHtml = parsedHtml.replace(/[`]{3}([^`]+)[`]{3}/g, '<pre>$1</pre>');\n\n // Code\n parsedHtml = parsedHtml.replace(/[`]{1}([^`\\n]+)[`]{1}/g, '<code>$1</code>');\n\n // Other simple markdown\n parsedHtml = parsedHtml.replace(/[*]{2}([^*\\n]+)[*]{2}/g, '<b>$1</b>');\n parsedHtml = parsedHtml.replace(/[_]{2}([^*\\n]+)[_]{2}/g, '<i>$1</i>');\n parsedHtml = parsedHtml.replace(/[~]{2}([^~\\n]+)[~]{2}/g, '<s>$1</s>');\n\n return parsedHtml;\n}\n\nfunction getEntityDataFromNode(\n node: ChildNode,\n rawText: string,\n textIndex: number,\n): { index: number; entity?: ApiMessageEntity } {\n const type = getEntityTypeFromNode(node);\n if (!type || !node.textContent) {\n return {\n index: textIndex,\n entity: undefined,\n };\n }\n\n const rawIndex = rawText.indexOf(node.textContent, textIndex);\n // In some cases, last text entity ends with a newline (which gets trimmed from `rawText`).\n // In this case, `rawIndex` would return `-1`, so we use `textIndex` instead.\n const index = rawIndex >= 0 ? rawIndex : textIndex;\n const offset = rawText.substring(0, index).length;\n const { length } = rawText.substring(index, index + node.textContent.length);\n\n let url: string | undefined;\n let userId: number | undefined;\n if (type === ApiMessageEntityTypes.TextUrl) {\n url = (node as HTMLAnchorElement).href;\n }\n if (type === ApiMessageEntityTypes.MentionName) {\n userId = Number((node as HTMLAnchorElement).dataset.userId);\n }\n\n return {\n index,\n entity: {\n type,\n offset,\n length,\n ...(url && { url }),\n ...(userId && { userId }),\n },\n };\n}\n\nfunction getEntityTypeFromNode(node: ChildNode) {\n if (ENTITY_CLASS_BY_NODE_NAME[node.nodeName]) {\n return ENTITY_CLASS_BY_NODE_NAME[node.nodeName];\n }\n\n if (node.nodeName === 'A') {\n const anchor = node as HTMLAnchorElement;\n if (anchor.dataset.entityType === ApiMessageEntityTypes.MentionName) {\n return ApiMessageEntityTypes.MentionName;\n }\n if (anchor.dataset.entityType === ApiMessageEntityTypes.Url) {\n return ApiMessageEntityTypes.Url;\n }\n if (anchor.href.startsWith('mailto:')) {\n return ApiMessageEntityTypes.Email;\n }\n if (anchor.href.startsWith('tel:')) {\n return ApiMessageEntityTypes.Phone;\n }\n if (anchor.href !== anchor.textContent) {\n return ApiMessageEntityTypes.TextUrl;\n }\n\n return ApiMessageEntityTypes.Url;\n }\n\n if (node.nodeName === 'SPAN') {\n return (node as HTMLElement).dataset.entityType;\n }\n\n return undefined;\n}\n","import React, { FC, memo } from '../../../lib/teact/teact';\n\nimport useFlag from '../../../hooks/useFlag';\nimport buildClassName from '../../../util/buildClassName';\n\nimport './DropTarget.scss';\n\nexport type OwnProps = {\n isQuick?: boolean;\n onFileSelect: (e: React.DragEvent<HTMLDivElement>) => void;\n};\n\nconst DropTarget: FC<OwnProps> = ({ isQuick, onFileSelect }) => {\n const [isHovered, markHovered, unmarkHovered] = useFlag();\n\n const handleDragEnter = () => { markHovered(); };\n const handleDragLeave = (e: React.DragEvent<HTMLDivElement>) => {\n const { relatedTarget: toTarget } = e;\n\n if (toTarget) {\n e.stopPropagation();\n }\n\n unmarkHovered();\n };\n\n const className = buildClassName(\n 'DropTarget',\n isHovered && 'hovered',\n );\n\n return (\n <div\n className={className}\n onDrop={onFileSelect}\n onDragEnter={handleDragEnter}\n onDragLeave={handleDragLeave}\n >\n <div className=\"target-content\">\n <div className={`icon icon-${isQuick ? 'photo' : 'document'}`} />\n <div className=\"title\">Drop files here to send them</div>\n <div className=\"description\">{isQuick ? 'in a quick way' : 'without compression'}</div>\n </div>\n </div>\n );\n};\n\nexport default memo(DropTarget);\n","import React, {\n FC, memo, useCallback, useEffect, useRef,\n} from '../../../lib/teact/teact';\n\nimport useShowTransition from '../../../hooks/useShowTransition';\nimport buildClassName from '../../../util/buildClassName';\n\nimport captureEscKeyListener from '../../../util/captureEscKeyListener';\nimport usePrevious from '../../../hooks/usePrevious';\n\nimport DropTarget from './DropTarget';\n\nimport './DropArea.scss';\n\nexport type OwnProps = {\n isOpen: boolean;\n withQuick?: boolean;\n onHide: NoneToVoidFunction;\n onFileSelect: (files: File[], isQuick: boolean) => void;\n};\n\nexport enum DropAreaState {\n None = 'none',\n Document = 'document',\n QuickFile = 'quick_file',\n}\n\nconst DROP_LEAVE_TIMEOUT_MS = 150;\n\nconst DropArea: FC<OwnProps> = ({\n isOpen, withQuick, onHide, onFileSelect,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const hideTimeoutRef = useRef<number>(null);\n const prevWithQuick = usePrevious(withQuick);\n const { shouldRender, transitionClassNames } = useShowTransition(isOpen);\n\n useEffect(() => (isOpen ? captureEscKeyListener(onHide) : undefined), [isOpen, onHide]);\n\n const handleFilesDrop = useCallback((e: React.DragEvent<HTMLDivElement>) => {\n const { dataTransfer: dt } = e;\n\n if (dt.files && dt.files.length > 0) {\n onHide();\n onFileSelect(Array.from(dt.files), false);\n }\n }, [onFileSelect, onHide]);\n\n const handleQuickFilesDrop = useCallback((e: React.DragEvent<HTMLDivElement>) => {\n const { dataTransfer: dt } = e;\n\n if (dt.files && dt.files.length > 0) {\n onHide();\n onFileSelect(Array.from(dt.files), true);\n }\n }, [onFileSelect, onHide]);\n\n const handleDragLeave = useCallback((e: React.DragEvent<HTMLDivElement>) => {\n e.stopPropagation();\n\n const { target: fromTarget, relatedTarget: toTarget } = e;\n\n // Esc button pressed during drag event\n if ((fromTarget as HTMLDivElement).matches('.DropTarget, .DropArea') && !toTarget) {\n hideTimeoutRef.current = window.setTimeout(() => {\n onHide();\n }, DROP_LEAVE_TIMEOUT_MS);\n }\n }, [onHide]);\n\n const handleDragOver = () => {\n if (hideTimeoutRef.current) {\n window.clearTimeout(hideTimeoutRef.current);\n }\n };\n\n if (!shouldRender) {\n return undefined;\n }\n\n const className = buildClassName(\n 'DropArea',\n transitionClassNames,\n );\n\n return (\n <div className={className} onDragLeave={handleDragLeave} onDragOver={handleDragOver} onDrop={onHide}>\n <DropTarget onFileSelect={handleFilesDrop} />\n {(withQuick || prevWithQuick) && <DropTarget onFileSelect={handleQuickFilesDrop} isQuick />}\n </div>\n );\n};\n\nexport default memo(DropArea);\n","import useReducer, { StateReducer, Dispatch } from '../useReducer';\nimport { ApiChatFolder } from '../../api/types';\nimport { pick, omit } from '../../util/iteratees';\n\nexport type FolderChatType = {\n icon: string;\n title: string;\n key: keyof Pick<ApiChatFolder, (\n 'contacts' | 'nonContacts' | 'groups' | 'channels' | 'bots' |\n 'excludeMuted' | 'excludeArchived' | 'excludeRead'\n )>;\n};\n\nexport const INCLUDED_CHAT_TYPES: FolderChatType[] = [\n { icon: 'user', title: 'FilterContacts', key: 'contacts' },\n { icon: 'non-contacts', title: 'FilterNonContacts', key: 'nonContacts' },\n { icon: 'group', title: 'FilterGroups', key: 'groups' },\n { icon: 'channel', title: 'FilterChannels', key: 'channels' },\n { icon: 'bots', title: 'FilterBots', key: 'bots' },\n];\n\nexport const EXCLUDED_CHAT_TYPES: FolderChatType[] = [\n { icon: 'mute', title: 'FilterMuted', key: 'excludeMuted' },\n { icon: 'archive', title: 'FilterArchived', key: 'excludeArchived' },\n { icon: 'readchats', title: 'FilterRead', key: 'excludeRead' },\n];\n\nconst INCLUDE_FILTER_FIELDS: Array<keyof FolderIncludeFilters> = [\n 'includedChatIds', 'bots', 'channels', 'groups', 'contacts', 'nonContacts',\n];\nconst EXCLUDE_FILTER_FIELDS: Array<keyof FolderExcludeFilters> = [\n 'excludedChatIds', 'excludeArchived', 'excludeMuted', 'excludeRead',\n];\n\nexport function selectChatFilters(state: FoldersState, mode: 'included' | 'excluded', selectTemp?: boolean) {\n let selectedChatIds: number[] = [];\n let selectedChatTypes: FolderChatType['key'][] = [];\n\n if (mode === 'included') {\n const {\n includedChatIds,\n ...includeFilters\n } = selectTemp\n ? state.includeFilters || {}\n : pick(\n state.folder,\n INCLUDE_FILTER_FIELDS,\n );\n\n selectedChatIds = includedChatIds || [];\n selectedChatTypes = (Object.keys(includeFilters) as Array<keyof typeof includeFilters>)\n .filter((key) => Boolean(includeFilters[key]));\n } else {\n const {\n excludedChatIds,\n ...excludeFilters\n } = selectTemp\n ? state.excludeFilters || {}\n : pick(\n state.folder,\n EXCLUDE_FILTER_FIELDS,\n );\n\n selectedChatIds = excludedChatIds || [];\n selectedChatTypes = (Object.keys(excludeFilters) as Array<keyof typeof excludeFilters>)\n .filter((key) => Boolean(excludeFilters[key]));\n }\n\n return {\n selectedChatIds,\n selectedChatTypes,\n };\n}\n\nfunction getSuggestedFolderName(includeFilters?: FolderIncludeFilters) {\n if (includeFilters) {\n const {\n includedChatIds,\n ...filters\n } = includeFilters;\n\n if (\n Object.values(filters).filter(Boolean).length > 1\n || (includedChatIds && includedChatIds.length)\n ) {\n return '';\n }\n\n if (filters.bots) {\n return 'Bots';\n } else if (filters.groups) {\n return 'Groups';\n } else if (filters.channels) {\n return 'Channels';\n } else if (filters.contacts) {\n return 'Contacts';\n } else if (filters.nonContacts) {\n return 'Non-Contacts';\n }\n }\n\n return '';\n}\n\ntype FolderIncludeFilters = Pick<ApiChatFolder, (\n 'includedChatIds' | 'bots' | 'channels' | 'groups' | 'contacts' | 'nonContacts'\n)>;\ntype FolderExcludeFilters = Pick<ApiChatFolder, 'excludedChatIds' | 'excludeArchived' | 'excludeMuted' | 'excludeRead'>;\n\nexport type FoldersState = {\n mode: 'create' | 'edit';\n isLoading?: boolean;\n isTouched?: boolean;\n error?: string;\n folderId?: number;\n chatFilter: string;\n folder: Omit<ApiChatFolder, 'id' | 'description' | 'emoticon'>;\n includeFilters?: FolderIncludeFilters;\n excludeFilters?: FolderExcludeFilters;\n};\nexport type FoldersActions = (\n 'setTitle' | 'saveFilters' | 'editFolder' | 'reset' | 'setChatFilter' | 'setIsLoading' | 'setError' |\n 'editIncludeFilters' | 'editExcludeFilters' | 'setIncludeFilters' | 'setExcludeFilters'\n);\nexport type FolderEditDispatch = Dispatch<FoldersActions>;\n\nconst INITIAL_STATE: FoldersState = {\n mode: 'create',\n chatFilter: '',\n folder: {\n title: '',\n includedChatIds: [],\n excludedChatIds: [],\n },\n};\n\nconst foldersReducer: StateReducer<FoldersState, FoldersActions> = (\n state,\n action,\n) => {\n switch (action.type) {\n case 'setTitle':\n return {\n ...state,\n folder: {\n ...state.folder,\n title: action.payload,\n },\n isTouched: true,\n };\n case 'editIncludeFilters':\n return {\n ...state,\n includeFilters: pick(\n state.folder,\n INCLUDE_FILTER_FIELDS,\n ),\n };\n case 'editExcludeFilters':\n return {\n ...state,\n excludeFilters: pick(\n state.folder,\n EXCLUDE_FILTER_FIELDS,\n ),\n };\n case 'setIncludeFilters':\n return {\n ...state,\n includeFilters: action.payload,\n chatFilter: '',\n };\n case 'setExcludeFilters':\n return {\n ...state,\n excludeFilters: action.payload,\n chatFilter: '',\n };\n case 'saveFilters':\n if (state.includeFilters) {\n return {\n ...state,\n folder: {\n ...omit(state.folder, INCLUDE_FILTER_FIELDS),\n title: state.folder.title ? state.folder.title : getSuggestedFolderName(state.includeFilters),\n ...state.includeFilters,\n },\n includeFilters: undefined,\n chatFilter: '',\n isTouched: true,\n };\n } else if (state.excludeFilters) {\n return {\n ...state,\n folder: {\n ...omit(state.folder, EXCLUDE_FILTER_FIELDS),\n ...state.excludeFilters,\n },\n excludeFilters: undefined,\n chatFilter: '',\n isTouched: true,\n };\n } else {\n return state;\n }\n case 'editFolder': {\n const { id: folderId, description, ...folder } = action.payload;\n\n return {\n mode: 'edit',\n folderId,\n folder,\n chatFilter: '',\n };\n }\n case 'setChatFilter': {\n return {\n ...state,\n chatFilter: action.payload,\n };\n }\n case 'setIsLoading': {\n return {\n ...state,\n isLoading: action.payload,\n };\n }\n case 'setError': {\n return {\n ...state,\n error: action.payload,\n };\n }\n case 'reset':\n return INITIAL_STATE;\n default:\n return state;\n }\n};\n\nexport default () => {\n return useReducer(foldersReducer, INITIAL_STATE);\n};\n","import React, { FC, memo } from '../../lib/teact/teact';\nimport { withGlobal } from '../../lib/teact/teactn';\n\nimport { ApiChat, ApiUser } from '../../api/types';\n\nimport { selectChat, selectUser } from '../../modules/selectors';\nimport { getChatTitle, getUserFirstOrLastName, isChatPrivate } from '../../modules/helpers';\nimport renderText from './helpers/renderText';\nimport buildClassName from '../../util/buildClassName';\nimport useLang from '../../hooks/useLang';\n\nimport Avatar from './Avatar';\n\nimport './PickerSelectedItem.scss';\n\ntype OwnProps = {\n chatOrUserId?: number;\n icon?: string;\n title?: string;\n isMinimized?: boolean;\n canClose?: boolean;\n onClick: (arg: any) => void;\n clickArg: any;\n className?: string;\n};\n\ntype StateProps = {\n chat?: ApiChat;\n user?: ApiUser;\n};\n\nconst PickerSelectedItem: FC<OwnProps & StateProps> = ({\n icon,\n title,\n isMinimized,\n canClose,\n onClick,\n clickArg,\n chat,\n user,\n className,\n}) => {\n const lang = useLang();\n\n let iconElement: any;\n let titleText: any;\n\n if (icon && title) {\n iconElement = (\n <div className=\"item-icon\">\n <i className={`icon-${icon}`} />\n </div>\n );\n\n titleText = title;\n } else if (chat || user) {\n iconElement = (\n <Avatar\n chat={chat}\n user={user}\n size=\"small\"\n isSavedMessages={user && user.isSelf}\n />\n );\n\n const name = !chat || (user && !user.isSelf)\n ? getUserFirstOrLastName(user)\n : getChatTitle(lang, chat, user);\n\n titleText = name ? renderText(name) : undefined;\n }\n\n const fullClassName = buildClassName(\n 'PickerSelectedItem',\n className,\n isMinimized && 'minimized',\n canClose && 'closeable',\n );\n\n return (\n <div\n className={fullClassName}\n onClick={() => onClick(clickArg)}\n title={isMinimized ? titleText : undefined}\n dir={lang.isRtl ? 'rtl' : undefined}\n >\n {iconElement}\n {!isMinimized && (\n <div className=\"item-name\" dir=\"auto\">\n {titleText}\n </div>\n )}\n {canClose && (\n <div className=\"item-remove\">\n <i className=\"icon-close\" />\n </div>\n )}\n </div>\n );\n};\n\nexport default memo(withGlobal<OwnProps>(\n (global, { chatOrUserId }): StateProps => {\n if (!chatOrUserId) {\n return {};\n }\n\n const chat = chatOrUserId ? selectChat(global, chatOrUserId) : undefined;\n const user = isChatPrivate(chatOrUserId) ? selectUser(global, chatOrUserId) : undefined;\n\n return {\n chat,\n user,\n };\n },\n)(PickerSelectedItem));\n","import { getGlobal } from '../lib/teact/teactn';\n\nimport { ANIMATION_LEVEL_MIN } from '../config';\nimport { IS_IOS } from './environment';\nimport { animate } from './animation';\n\nconst DEFAULT_DURATION = 300;\n\nexport default function fastSmoothScrollHorizontal(container: HTMLElement, left: number, duration = DEFAULT_DURATION) {\n if (getGlobal().settings.byKey.animationLevel === ANIMATION_LEVEL_MIN) {\n duration = 0;\n }\n\n // Native way seems to be smoother in Chrome\n if (!IS_IOS) {\n container.scrollTo({\n left,\n ...(duration && { behavior: 'smooth' }),\n });\n } else {\n scrollWithJs(container, left, duration);\n }\n}\n\nfunction scrollWithJs(container: HTMLElement, left: number, duration: number) {\n const { scrollLeft, offsetWidth: containerWidth, scrollWidth } = container;\n let path = left - scrollLeft;\n\n if (path < 0) {\n const remainingPath = -scrollLeft;\n path = Math.max(path, remainingPath);\n } else if (path > 0) {\n const remainingPath = scrollWidth - (scrollLeft + containerWidth);\n path = Math.min(path, remainingPath);\n }\n\n if (path === 0) {\n return;\n }\n\n const target = scrollLeft + path;\n\n if (duration === 0) {\n container.scrollLeft = target;\n return;\n }\n\n const startAt = Date.now();\n\n animate(() => {\n const t = Math.min((Date.now() - startAt) / duration, 1);\n\n const currentPath = path * (1 - transition(t));\n container.scrollLeft = Math.round(target - currentPath);\n\n return t < 1;\n });\n}\n\nfunction transition(t: number) {\n return 1 - ((1 - t) ** 3.5);\n}\n","import { useEffect } from '../lib/teact/teact';\n\nexport default (container: HTMLElement | null, isDisabled?: boolean) => {\n useEffect(() => {\n if (!container) {\n return undefined;\n }\n\n function handleScroll(e: WheelEvent) {\n // Ignore horizontal scroll and let it work natively (e.g. on touchpad)\n if (!e.deltaX) {\n container!.scrollLeft += e.deltaY / 4;\n }\n }\n\n container.addEventListener('wheel', handleScroll, { passive: true });\n\n return () => {\n container.removeEventListener('wheel', handleScroll);\n };\n }, [container, isDisabled]);\n};\n","import { useEffect, useMemo } from '../lib/teact/teact';\nimport { getDispatch } from '../lib/teact/teactn';\n\nimport { ApiMessage } from '../api/types';\n\nimport { throttle } from '../util/schedulers';\n\nexport default (\n chatId: number,\n messageId?: number,\n message?: ApiMessage,\n replyOriginForId?: number,\n) => {\n const { loadMessage } = getDispatch();\n const loadMessageThrottled = useMemo(() => {\n const throttled = throttle(loadMessage, 500, true);\n return () => {\n throttled({ chatId, messageId, replyOriginForId });\n };\n }, [loadMessage, chatId, messageId, replyOriginForId]);\n\n useEffect(() => {\n if (messageId && !message) {\n loadMessageThrottled();\n }\n });\n};\n","import { RefObject } from 'react';\nimport { useState, useEffect, useCallback } from '../lib/teact/teact';\n\nimport { IAnchorPosition } from '../types';\nimport { IS_TOUCH_ENV, IS_SINGLE_COLUMN_LAYOUT } from '../util/environment';\n\nconst LONG_TAP_DURATION_MS = 250;\nconst SELECTION_ANIMATION_DURATION_MS = 200;\n\nlet contextMenuCounter = 0;\n\nfunction checkIsDisabledForMobile() {\n return IS_SINGLE_COLUMN_LAYOUT\n && window.document.body.classList.contains('enable-symbol-menu-transforms');\n}\n\nexport default (\n elementRef: RefObject<HTMLElement>,\n isMenuDisabled?: boolean,\n shouldDisableOnLink?: boolean,\n shouldDisableOnLongTap?: boolean,\n) => {\n const [isContextMenuOpen, setIsContextMenuOpen] = useState(false);\n const [contextMenuPosition, setContextMenuPosition] = useState<IAnchorPosition | undefined>(undefined);\n\n const handleBeforeContextMenu = useCallback((e: React.MouseEvent) => {\n if (!isMenuDisabled && e.button === 2) {\n document.body.classList.add('no-selection');\n }\n }, [isMenuDisabled]);\n\n const handleContextMenu = useCallback((e: React.MouseEvent) => {\n document.body.classList.remove('no-selection');\n\n if (isMenuDisabled || (shouldDisableOnLink && (e.target as HTMLElement).matches('a.text-entity-link[href]'))) {\n return;\n }\n e.preventDefault();\n\n if (contextMenuPosition) {\n return;\n }\n document.body.classList.remove('no-selection');\n if (contextMenuCounter === 0) {\n document.body.classList.add('has-context-menu');\n }\n contextMenuCounter++;\n\n setIsContextMenuOpen(true);\n setContextMenuPosition({ x: e.clientX, y: e.clientY });\n }, [isMenuDisabled, shouldDisableOnLink, contextMenuPosition]);\n\n const handleContextMenuClose = useCallback(() => {\n setIsContextMenuOpen(false);\n }, []);\n\n const handleContextMenuHide = useCallback(() => {\n setContextMenuPosition(undefined);\n document.body.classList.remove('no-selection');\n\n setTimeout(() => {\n contextMenuCounter--;\n if (contextMenuCounter === 0) {\n document.body.classList.remove('has-context-menu');\n }\n }, SELECTION_ANIMATION_DURATION_MS);\n }, []);\n\n // Support context menu on touch-devices\n useEffect(() => {\n if (isMenuDisabled || !IS_TOUCH_ENV || shouldDisableOnLongTap) {\n return undefined;\n }\n\n const element = elementRef.current;\n if (!element) {\n return undefined;\n }\n\n let timer: number | undefined;\n\n const clearLongPressTimer = () => {\n if (timer) {\n clearTimeout(timer);\n timer = undefined;\n }\n };\n\n const emulateContextMenuEvent = (originalEvent: TouchEvent) => {\n clearLongPressTimer();\n\n const { clientX, clientY, target } = originalEvent.touches[0];\n\n if (contextMenuPosition || (shouldDisableOnLink && (target as HTMLElement).matches('a.text-entity-link[href]'))) {\n return;\n }\n\n // temporarily intercept and clear the next click\n element.addEventListener('touchend', function cancelClickOnce(e) {\n element.removeEventListener('touchend', cancelClickOnce, true);\n e.stopImmediatePropagation();\n e.preventDefault();\n e.stopPropagation();\n }, true);\n\n document.body.classList.add('no-selection');\n setIsContextMenuOpen(true);\n setContextMenuPosition({ x: clientX, y: clientY });\n };\n\n const startLongPressTimer = (e: TouchEvent) => {\n if (isMenuDisabled || checkIsDisabledForMobile()) {\n return;\n }\n clearLongPressTimer();\n\n timer = window.setTimeout(() => emulateContextMenuEvent(e), LONG_TAP_DURATION_MS);\n };\n\n // @perf Consider event delegation\n element.addEventListener('touchstart', startLongPressTimer, { passive: true });\n element.addEventListener('touchcancel', clearLongPressTimer, true);\n element.addEventListener('touchend', clearLongPressTimer, true);\n element.addEventListener('touchmove', clearLongPressTimer, { passive: true });\n\n return () => {\n clearLongPressTimer();\n element.removeEventListener('touchstart', startLongPressTimer);\n element.removeEventListener('touchcancel', clearLongPressTimer, true);\n element.removeEventListener('touchend', clearLongPressTimer, true);\n element.removeEventListener('touchmove', clearLongPressTimer);\n };\n }, [contextMenuPosition, isMenuDisabled, shouldDisableOnLongTap, elementRef, shouldDisableOnLink]);\n\n return {\n isContextMenuOpen,\n contextMenuPosition,\n handleBeforeContextMenu,\n handleContextMenu,\n handleContextMenuClose,\n handleContextMenuHide,\n };\n};\n","import { MouseEvent } from 'react';\nimport React from '../../../lib/teact/teact';\nimport { getDispatch } from '../../../lib/teact/teactn';\n\nimport { ApiMessageEntity, ApiMessageEntityTypes, ApiMessage } from '../../../api/types';\n\nimport { getMessageText } from '../../../modules/helpers';\nimport renderText from './renderText';\n\nimport MentionLink from '../../middle/message/MentionLink';\nimport SafeLink from '../SafeLink';\n\nexport type TextPart = string | Element;\n\nexport function renderMessageText(message: ApiMessage, highlight?: string, shouldRenderHqEmoji?: boolean) {\n const formattedText = message.content.text;\n\n if (!formattedText || !formattedText.text) {\n const rawText = getMessageText(message);\n return rawText ? [rawText] : undefined;\n }\n const { text, entities } = formattedText;\n\n return renderTextWithEntities(text, entities, highlight, shouldRenderHqEmoji);\n}\n\ninterface IOrganizedEntity {\n entity: ApiMessageEntity;\n organizedIndexes: Set<number>;\n nestedEntities: IOrganizedEntity[];\n}\n\nfunction organizeEntity(\n entity: ApiMessageEntity,\n index: number,\n entities: ApiMessageEntity[],\n organizedEntityIndexes: Set<number>,\n): IOrganizedEntity | undefined {\n const { offset, length } = entity;\n const organizedIndexes = new Set([index]);\n\n if (organizedEntityIndexes.has(index)) {\n return undefined;\n }\n\n // Determine any nested entities inside current entity\n const nestedEntities: IOrganizedEntity[] = [];\n const parsedNestedEntities = entities\n .filter((e, i) => i > index && e.offset >= offset && e.offset < offset + length)\n .map((e) => organizeEntity(e, entities.indexOf(e), entities, organizedEntityIndexes))\n .filter<IOrganizedEntity>(Boolean as any);\n\n parsedNestedEntities.forEach((parsedEntity) => {\n let isChanged = false;\n\n parsedEntity.organizedIndexes.forEach((organizedIndex) => {\n if (!isChanged && !organizedIndexes.has(organizedIndex)) {\n isChanged = true;\n }\n\n organizedIndexes.add(organizedIndex);\n });\n\n if (isChanged) {\n nestedEntities.push(parsedEntity);\n }\n });\n\n return {\n entity,\n organizedIndexes,\n nestedEntities,\n };\n}\n\n// Organize entities in a tree-like structure to better represent how the text will be displayed\nfunction organizeEntities(entities: ApiMessageEntity[]) {\n const organizedEntityIndexes: Set<number> = new Set();\n const organizedEntities: IOrganizedEntity[] = [];\n\n entities.forEach((entity, index) => {\n if (organizedEntityIndexes.has(index)) {\n return;\n }\n\n const organizedEntity = organizeEntity(entity, index, entities, organizedEntityIndexes);\n if (organizedEntity) {\n organizedEntity.organizedIndexes.forEach((organizedIndex) => {\n organizedEntityIndexes.add(organizedIndex);\n });\n\n organizedEntities.push(organizedEntity);\n }\n });\n\n return organizedEntities;\n}\n\nexport function renderTextWithEntities(\n text: string,\n entities?: ApiMessageEntity[],\n highlight?: string,\n shouldRenderHqEmoji?: boolean,\n shouldRenderAsHtml?: boolean,\n) {\n if (!entities || !entities.length) {\n return renderMessagePart(text, highlight, shouldRenderHqEmoji, shouldRenderAsHtml);\n }\n\n const result: TextPart[] = [];\n let deleteLineBreakAfterPre = false;\n\n const organizedEntities = organizeEntities(entities);\n\n // Recursive function to render regular and nested entities\n function renderEntity(\n textPartStart: number,\n textPartEnd: number,\n organizedEntity: IOrganizedEntity,\n isLastEntity: boolean,\n ) {\n const renderResult: TextPart[] = [];\n const { entity, nestedEntities } = organizedEntity;\n const { offset, length, type } = entity;\n\n // Render text before the entity\n let textBefore = text.substring(textPartStart, offset);\n const textBeforeLength = textBefore.length;\n if (textBefore) {\n if (deleteLineBreakAfterPre && textBefore.length > 0 && textBefore[0] === '\\n') {\n textBefore = textBefore.substr(1);\n deleteLineBreakAfterPre = false;\n }\n if (textBefore) {\n renderResult.push(...renderMessagePart(\n textBefore, highlight, shouldRenderHqEmoji, shouldRenderAsHtml,\n ) as TextPart[]);\n }\n }\n\n const entityStartIndex = textPartStart + textBeforeLength;\n const entityEndIndex = entityStartIndex + length;\n\n let entityContent: TextPart = text.substring(offset, offset + length);\n const nestedEntityContent: TextPart[] = [];\n\n if (deleteLineBreakAfterPre && entityContent.length > 0 && entityContent[0] === '\\n') {\n entityContent = entityContent.substr(1);\n deleteLineBreakAfterPre = false;\n }\n\n if (type === ApiMessageEntityTypes.Pre) {\n deleteLineBreakAfterPre = true;\n }\n\n // Render nested entities, if any\n if (nestedEntities.length) {\n let nestedIndex = entityStartIndex;\n\n nestedEntities.forEach((nestedEntity, nestedEntityIndex) => {\n const {\n renderResult: nestedResult,\n entityEndIndex: nestedEntityEndIndex,\n } = renderEntity(\n nestedIndex,\n entityEndIndex,\n nestedEntity,\n nestedEntityIndex === nestedEntities.length - 1,\n );\n\n nestedEntityContent.push(...nestedResult);\n nestedIndex = nestedEntityEndIndex;\n });\n }\n\n // Render the entity itself\n const newEntity = shouldRenderAsHtml\n ? processEntityAsHtml(entity, entityContent, nestedEntityContent)\n : processEntity(entity, entityContent, nestedEntityContent);\n\n if (Array.isArray(newEntity)) {\n renderResult.push(...newEntity);\n } else {\n renderResult.push(newEntity);\n }\n\n // Render text after the entity, if it is the last entity in the text,\n // or the last nested entity inside of another entity\n if (isLastEntity && entityEndIndex < textPartEnd) {\n let textAfter = text.substring(entityEndIndex, textPartEnd);\n if (deleteLineBreakAfterPre && textAfter.length > 0 && textAfter[0] === '\\n') {\n textAfter = textAfter.substring(1);\n }\n if (textAfter) {\n renderResult.push(...renderMessagePart(\n textAfter, highlight, shouldRenderHqEmoji, shouldRenderAsHtml,\n ) as TextPart[]);\n }\n }\n\n return {\n renderResult,\n entityEndIndex,\n };\n }\n\n // Process highest-level entities\n let index = 0;\n\n organizedEntities.forEach((entity, arrayIndex) => {\n const { renderResult, entityEndIndex } = renderEntity(\n index,\n text.length,\n entity,\n arrayIndex === organizedEntities.length - 1,\n );\n\n result.push(...renderResult);\n index = entityEndIndex;\n });\n\n return result;\n}\n\nfunction processEntity(\n entity: ApiMessageEntity,\n entityContent: TextPart,\n nestedEntityContent: TextPart[],\n) {\n const entityText = typeof entityContent === 'string' && entityContent;\n const renderedContent = nestedEntityContent.length ? nestedEntityContent : entityContent;\n\n if (!entityText) {\n return renderMessagePart(renderedContent);\n }\n\n switch (entity.type) {\n case ApiMessageEntityTypes.Bold:\n return <strong>{renderMessagePart(renderedContent)}</strong>;\n case ApiMessageEntityTypes.Blockquote:\n return <blockquote>{renderMessagePart(renderedContent)}</blockquote>;\n case ApiMessageEntityTypes.BotCommand:\n return (\n <a\n onClick={handleBotCommandClick}\n className=\"text-entity-link\"\n dir=\"auto\"\n >\n {renderMessagePart(renderedContent)}\n </a>\n );\n case ApiMessageEntityTypes.Hashtag:\n return (\n <a\n onClick={handleHashtagClick}\n className=\"text-entity-link\"\n dir=\"auto\"\n >\n {renderMessagePart(renderedContent)}\n </a>\n );\n case ApiMessageEntityTypes.Cashtag:\n return (\n <a\n onClick={handleHashtagClick}\n className=\"text-entity-link\"\n dir=\"auto\"\n >\n {renderMessagePart(renderedContent)}\n </a>\n );\n case ApiMessageEntityTypes.Code:\n return <code className=\"text-entity-code\">{renderMessagePart(renderedContent)}</code>;\n case ApiMessageEntityTypes.Email:\n return (\n <a\n href={`mailto:${entityText}`}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n className=\"text-entity-link\"\n dir=\"auto\"\n >\n {renderMessagePart(renderedContent)}\n </a>\n );\n case ApiMessageEntityTypes.Italic:\n return <em>{renderMessagePart(renderedContent)}</em>;\n case ApiMessageEntityTypes.MentionName:\n return (\n <MentionLink userId={entity.userId}>\n {renderMessagePart(renderedContent)}\n </MentionLink>\n );\n case ApiMessageEntityTypes.Mention:\n return (\n <MentionLink username={entityText}>\n {renderMessagePart(renderedContent)}\n </MentionLink>\n );\n case ApiMessageEntityTypes.Phone:\n return (\n <a\n href={`tel:${entityText}`}\n className=\"text-entity-link\"\n dir=\"auto\"\n >\n {renderMessagePart(renderedContent)}\n </a>\n );\n case ApiMessageEntityTypes.Pre:\n return <pre className=\"text-entity-pre\">{renderMessagePart(renderedContent)}</pre>;\n case ApiMessageEntityTypes.Strike:\n return <del>{renderMessagePart(renderedContent)}</del>;\n case ApiMessageEntityTypes.TextUrl:\n case ApiMessageEntityTypes.Url:\n return (\n <SafeLink\n url={getLinkUrl(entityText, entity)}\n text={entityText}\n >\n {renderMessagePart(renderedContent)}\n </SafeLink>\n );\n case ApiMessageEntityTypes.Underline:\n return <ins>{renderMessagePart(renderedContent)}</ins>;\n default:\n return renderMessagePart(renderedContent);\n }\n}\n\nfunction renderMessagePart(\n content: TextPart | TextPart[],\n highlight?: string,\n shouldRenderHqEmoji?: boolean,\n shouldRenderAsHtml?: boolean,\n) {\n if (Array.isArray(content)) {\n const result: TextPart[] = [];\n\n content.forEach((c) => {\n result.push(...renderMessagePart(c, highlight, shouldRenderHqEmoji, shouldRenderAsHtml));\n });\n\n return result;\n }\n\n if (shouldRenderAsHtml) {\n return renderText(content, ['escape_html', 'emoji_html', 'br_html']);\n }\n\n const emojiFilter = shouldRenderHqEmoji ? 'hq_emoji' : 'emoji';\n\n if (highlight) {\n return renderText(content, [emojiFilter, 'br', 'highlight'], { highlight });\n } else {\n return renderText(content, [emojiFilter, 'br']);\n }\n}\n\nfunction getLinkUrl(entityContent: string, entity: ApiMessageEntity) {\n const { type, url } = entity;\n return type === ApiMessageEntityTypes.TextUrl && url ? url : entityContent;\n}\n\nfunction handleBotCommandClick(e: MouseEvent<HTMLAnchorElement>) {\n getDispatch().sendBotCommand({ command: e.currentTarget.innerText });\n}\n\nfunction handleHashtagClick(e: MouseEvent<HTMLAnchorElement>) {\n getDispatch().setLocalTextSearchQuery({ query: e.currentTarget.innerText });\n getDispatch().searchTextMessagesLocal();\n}\n\nfunction processEntityAsHtml(\n entity: ApiMessageEntity,\n entityContent: TextPart,\n nestedEntityContent: TextPart[],\n) {\n const rawEntityText = typeof entityContent === 'string' && entityContent;\n\n const renderedContent = nestedEntityContent.length\n ? nestedEntityContent.join('')\n : renderText(entityContent, ['escape_html', 'emoji_html', 'br_html']).join('');\n\n if (!rawEntityText) {\n return renderedContent;\n }\n\n switch (entity.type) {\n case ApiMessageEntityTypes.Bold:\n return `<b>${renderedContent}</b>`;\n case ApiMessageEntityTypes.Italic:\n return `<i>${renderedContent}</i>`;\n case ApiMessageEntityTypes.Underline:\n return `<u>${renderedContent}</u>`;\n case ApiMessageEntityTypes.Code:\n return `<code class=\"text-entity-code\">${renderedContent}</code>`;\n case ApiMessageEntityTypes.Pre:\n return `\\`\\`\\`<br/>${renderedContent}<br/>\\`\\`\\``;\n case ApiMessageEntityTypes.Strike:\n return `<del>${renderedContent}</del>`;\n case ApiMessageEntityTypes.MentionName:\n return `<a\n class=\"text-entity-link\"\n data-entity-type=\"${ApiMessageEntityTypes.MentionName}\"\n data-user-id=\"${entity.userId}\"\n contenteditable=\"false\"\n dir=\"auto\"\n >${renderedContent}</a>`;\n case ApiMessageEntityTypes.Url:\n case ApiMessageEntityTypes.TextUrl:\n return `<a\n class=\"text-entity-link\"\n href=${getLinkUrl(rawEntityText, entity)}\n data-entity-type=\"${entity.type}\"\n dir=\"auto\"\n >${renderedContent}</a>`;\n default:\n return renderedContent;\n }\n}\n","import React, { FC, memo } from '../../lib/teact/teact';\nimport { OwnProps } from './StickerSetModal';\nimport { Bundles } from '../../util/moduleLoader';\n\nimport useModuleLoader from '../../hooks/useModuleLoader';\n\nconst StickerSetModalAsync: FC<OwnProps> = (props) => {\n const { isOpen } = props;\n const StickerSetModal = useModuleLoader(Bundles.Extra, 'StickerSetModal', !isOpen);\n\n // eslint-disable-next-line react/jsx-props-no-spreading\n return StickerSetModal ? <StickerSetModal {...props} /> : undefined;\n};\n\nexport default memo(StickerSetModalAsync);\n","import { RefObject } from 'react';\nimport { useEffect } from '../lib/teact/teact';\nimport { fastRaf } from '../util/schedulers';\n\n// Fix for memory leak when unmounting video element\nexport default function useVideoCleanup(videoRef: RefObject<HTMLVideoElement>, dependencies: any[]) {\n useEffect(() => {\n const videoEl = videoRef.current;\n\n return () => {\n if (videoEl) {\n fastRaf(() => {\n videoEl.pause();\n videoEl.src = '';\n videoEl.load();\n });\n }\n };\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, dependencies);\n}\n","import { useCallback, useEffect, useRef } from '../lib/teact/teact';\n\nimport { IS_TOUCH_ENV } from '../util/environment';\n\nconst MENU_CLOSE_TIMEOUT = 250;\nlet closeTimeout: number | undefined;\n\nexport default function useMouseInside(\n isOpen: boolean, onClose: NoneToVoidFunction, menuCloseTimeout = MENU_CLOSE_TIMEOUT, isDisabled = false,\n) {\n const isMouseInside = useRef(false);\n\n useEffect(() => {\n if (closeTimeout) {\n clearTimeout(closeTimeout);\n closeTimeout = undefined;\n }\n\n if (isOpen && !IS_TOUCH_ENV && !isDisabled) {\n closeTimeout = window.setTimeout(() => {\n if (!isMouseInside.current) {\n onClose();\n }\n }, menuCloseTimeout * 2);\n }\n }, [isDisabled, isOpen, menuCloseTimeout, onClose]);\n\n const handleMouseEnter = useCallback(() => {\n isMouseInside.current = true;\n }, []);\n\n const handleMouseLeave = useCallback(() => {\n isMouseInside.current = false;\n\n if (closeTimeout) {\n clearTimeout(closeTimeout);\n closeTimeout = undefined;\n }\n\n closeTimeout = window.setTimeout(() => {\n if (!isMouseInside.current) {\n onClose();\n }\n }, menuCloseTimeout);\n }, [menuCloseTimeout, onClose]);\n\n return [handleMouseEnter, handleMouseLeave];\n}\n","import { useState, useCallback, useRef } from '../lib/teact/teact';\n\nexport type ReducerAction<Actions> = { type: Actions; payload?: any };\nexport type StateReducer<State, Actions> = (state: State, action: ReducerAction<Actions>) => State;\nexport type Dispatch<Actions> = (action: ReducerAction<Actions>) => void;\n\nexport default function useReducer<State, Actions>(\n reducer: StateReducer<State, Actions>,\n initialState: State,\n) {\n const reducerRef = useRef(reducer);\n const [state, setState] = useState<State>(initialState);\n\n const dispatch = useCallback((action: ReducerAction<Actions>) => {\n setState((currentState) => reducerRef.current(currentState, action));\n }, []);\n\n return [\n state,\n dispatch,\n ] as [State, Dispatch<Actions>];\n}\n","import { RefObject } from 'react';\nimport React, {\n FC, useRef, useEffect, memo, useCallback,\n} from '../../lib/teact/teact';\n\nimport buildClassName from '../../util/buildClassName';\nimport useFlag from '../../hooks/useFlag';\nimport useLang from '../../hooks/useLang';\n\nimport Loading from './Loading';\nimport Button from './Button';\n\nimport './SearchInput.scss';\n\ntype OwnProps = {\n ref?: RefObject<HTMLInputElement>;\n children?: any;\n parentContainerClassName?: string;\n className?: string;\n inputId?: string;\n value?: string;\n focused?: boolean;\n isLoading?: boolean;\n placeholder?: string;\n disabled?: boolean;\n autoComplete?: string;\n canClose?: boolean;\n onChange: (value: string) => void;\n onReset?: NoneToVoidFunction;\n onFocus?: NoneToVoidFunction;\n onBlur?: NoneToVoidFunction;\n};\n\nconst SearchInput: FC<OwnProps> = ({\n ref,\n children,\n parentContainerClassName,\n value,\n inputId,\n className,\n focused,\n isLoading,\n placeholder,\n disabled,\n autoComplete,\n canClose,\n onChange,\n onReset,\n onFocus,\n onBlur,\n}) => {\n // eslint-disable-next-line no-null/no-null\n let inputRef = useRef<HTMLInputElement>(null);\n if (ref) {\n inputRef = ref;\n }\n\n const [isInputFocused, markInputFocused, unmarkInputFocused] = useFlag(focused);\n\n useEffect(() => {\n if (!inputRef.current) {\n return;\n }\n\n if (focused) {\n inputRef.current.focus();\n } else {\n inputRef.current.blur();\n }\n }, [focused, placeholder]); // Trick for setting focus when selecting a contact to search for\n\n const lang = useLang();\n\n function handleChange(event: React.ChangeEvent<HTMLInputElement>) {\n const { currentTarget } = event;\n onChange(currentTarget.value);\n }\n\n function handleFocus() {\n markInputFocused();\n if (onFocus) {\n onFocus();\n }\n }\n\n function handleBlur() {\n unmarkInputFocused();\n if (onBlur) {\n onBlur();\n }\n }\n\n const handleKeyDown = useCallback((e: React.KeyboardEvent) => {\n if (e.key === 'ArrowDown' || e.key === 'Enter') {\n const element = document.querySelector(`.${parentContainerClassName} .ListItem-button`) as HTMLElement;\n if (element) {\n element.focus();\n }\n }\n }, [parentContainerClassName]);\n\n return (\n <div\n className={buildClassName('SearchInput', className, isInputFocused && 'has-focus')}\n dir={lang.isRtl ? 'rtl' : undefined}\n >\n {children}\n <input\n ref={inputRef}\n id={inputId}\n type=\"text\"\n dir=\"auto\"\n placeholder={placeholder || lang('Search')}\n className=\"form-control\"\n value={value}\n disabled={disabled}\n autoComplete={autoComplete}\n onChange={handleChange}\n onFocus={handleFocus}\n onBlur={handleBlur}\n onKeyDown={handleKeyDown}\n />\n <i className=\"icon-search\" />\n {isLoading && (\n <Loading />\n )}\n {!isLoading && (value || canClose) && onReset && (\n <Button\n round\n size=\"tiny\"\n color=\"translucent\"\n onClick={onReset}\n >\n <span className=\"icon-close\" />\n </Button>\n )}\n </div>\n );\n};\n\nexport default memo(SearchInput);\n","import { IS_IOS } from './environment';\n\nexport default (container: HTMLDivElement, scrollTop?: number) => {\n if (IS_IOS) {\n container.style.overflow = 'hidden';\n }\n\n if (scrollTop !== undefined) {\n container.scrollTop = scrollTop;\n }\n\n if (IS_IOS) {\n container.style.overflow = '';\n }\n};\n","import React, { FC, memo } from '../../lib/teact/teact';\nimport { withGlobal } from '../../lib/teact/teactn';\n\nimport { ApiUser, ApiTypingStatus } from '../../api/types';\n\nimport { selectUser } from '../../modules/selectors';\nimport { getUserFirstOrLastName } from '../../modules/helpers';\nimport renderText from './helpers/renderText';\nimport useLang from '../../hooks/useLang';\n\nimport './TypingStatus.scss';\n\ntype OwnProps = {\n typingStatus: ApiTypingStatus;\n};\n\ntype StateProps = {\n typingUser?: ApiUser;\n};\n\nconst TypingStatus: FC<OwnProps & StateProps> = ({ typingStatus, typingUser }) => {\n const lang = useLang();\n const typingUserName = typingUser && !typingUser.isSelf && getUserFirstOrLastName(typingUser);\n\n return (\n <p className=\"typing-status\" dir={lang.isRtl ? 'rtl' : 'auto'}>\n {typingUserName && (\n <span className=\"sender-name\" dir=\"auto\">{renderText(typingUserName)}</span>\n )}\n {/* fix for translation \"username _is_ typing\" */}\n {lang(typingStatus.action).replace('{user}', '').trim()}\n <span className=\"ellipsis\" />\n </p>\n );\n};\n\nexport default memo(withGlobal<OwnProps>(\n (global, { typingStatus }): StateProps => {\n if (!typingStatus.userId) {\n return {};\n }\n\n const typingUser = selectUser(global, typingStatus.userId);\n\n return { typingUser };\n },\n)(TypingStatus));\n","import React, { FC, useCallback, memo } from '../../lib/teact/teact';\nimport { withGlobal } from '../../lib/teact/teactn';\n\nimport { ApiChat } from '../../api/types';\nimport { GlobalActions } from '../../global/types';\n\nimport { selectIsChatWithSelf, selectUser } from '../../modules/selectors';\nimport {\n isChatPrivate,\n isUserBot,\n getUserFirstOrLastName,\n getPrivateChatUserId,\n isChatBasicGroup,\n isChatSuperGroup,\n isChatChannel,\n getChatTitle,\n} from '../../modules/helpers';\nimport { pick } from '../../util/iteratees';\nimport useLang from '../../hooks/useLang';\nimport renderText from './helpers/renderText';\n\nimport Avatar from './Avatar';\nimport Modal from '../ui/Modal';\nimport Button from '../ui/Button';\n\nimport './DeleteChatModal.scss';\n\nexport type OwnProps = {\n isOpen: boolean;\n chat: ApiChat;\n onClose: () => void;\n onCloseAnimationEnd?: () => void;\n};\n\ntype StateProps = {\n isChannel: boolean;\n isChatWithSelf?: boolean;\n isBot?: boolean;\n isPrivateChat: boolean;\n isBasicGroup: boolean;\n isSuperGroup: boolean;\n currentUserId: number | undefined;\n canDeleteForAll?: boolean;\n contactName?: string;\n};\n\ntype DispatchProps = Pick<GlobalActions, (\n 'leaveChannel' | 'deleteHistory' | 'deleteChannel' | 'deleteChatUser' | 'blockContact'\n)>;\n\nconst DeleteChatModal: FC<OwnProps & StateProps & DispatchProps> = ({\n isOpen,\n chat,\n isChannel,\n isPrivateChat,\n isChatWithSelf,\n isBot,\n isBasicGroup,\n isSuperGroup,\n currentUserId,\n canDeleteForAll,\n contactName,\n onClose,\n onCloseAnimationEnd,\n leaveChannel,\n deleteHistory,\n deleteChannel,\n deleteChatUser,\n blockContact,\n}) => {\n const lang = useLang();\n const chatTitle = getChatTitle(lang, chat);\n\n const handleDeleteMessageForAll = useCallback(() => {\n deleteHistory({ chatId: chat.id, shouldDeleteForAll: true });\n\n onClose();\n }, [deleteHistory, chat.id, onClose]);\n\n const handleDeleteAndStop = useCallback(() => {\n deleteHistory({ chatId: chat.id, shouldDeleteForAll: true });\n blockContact({ contactId: chat.id, accessHash: chat.accessHash });\n\n onClose();\n }, [deleteHistory, chat.id, chat.accessHash, blockContact, onClose]);\n\n const handleDeleteChat = useCallback(() => {\n if (isPrivateChat) {\n deleteHistory({ chatId: chat.id, shouldDeleteForAll: false });\n } else if (isBasicGroup) {\n deleteChatUser({ chatId: chat.id, userId: currentUserId });\n deleteHistory({ chatId: chat.id, shouldDeleteForAll: false });\n } else if ((isChannel || isSuperGroup) && !chat.isCreator) {\n leaveChannel({ chatId: chat.id });\n } else if ((isChannel || isSuperGroup) && chat.isCreator) {\n deleteChannel({ chatId: chat.id });\n }\n onClose();\n }, [\n isPrivateChat,\n isBasicGroup,\n isChannel,\n isSuperGroup,\n currentUserId,\n chat.isCreator,\n chat.id,\n onClose,\n deleteHistory,\n deleteChatUser,\n leaveChannel,\n deleteChannel,\n ]);\n\n function renderHeader() {\n return (\n <div className=\"modal-header\" dir={lang.isRtl ? 'rtl' : undefined}>\n <Avatar\n size=\"tiny\"\n chat={chat}\n isSavedMessages={isChatWithSelf}\n />\n <h3 className=\"modal-title\">{lang(renderTitle())}</h3>\n </div>\n );\n }\n\n function renderTitle() {\n if (isChannel && !chat.isCreator) {\n return 'LeaveChannel';\n }\n\n if (isChannel && chat.isCreator) {\n return 'ChannelDelete';\n }\n\n if (isBasicGroup || isSuperGroup) {\n return 'Group.LeaveGroup';\n }\n\n return 'DeleteChatUser';\n }\n\n function renderMessage() {\n if (isChannel && chat.isCreator) {\n return <p>{renderText(lang('ChatList.DeleteAndLeaveGroupConfirmation', chatTitle), ['simple_markdown'])}</p>;\n }\n\n if ((isChannel && !chat.isCreator) || isBasicGroup || isSuperGroup) {\n return <p>{renderText(lang('ChannelLeaveAlertWithName', chatTitle), ['simple_markdown'])}</p>;\n }\n\n return <p>{renderText(lang('ChatList.DeleteChatConfirmation', contactName), ['simple_markdown'])}</p>;\n }\n\n function renderActionText() {\n if (isChannel && !chat.isCreator) {\n return 'LeaveChannel';\n }\n if (isChannel && chat.isCreator) {\n return 'Chat.Input.Delete';\n }\n\n if (isBasicGroup || isSuperGroup) {\n return 'Group.LeaveGroup';\n }\n\n return canDeleteForAll ? 'ChatList.DeleteForCurrentUser' : 'Delete';\n }\n\n return (\n <Modal\n isOpen={isOpen}\n className=\"DeleteChatModal\"\n header={renderHeader()}\n onClose={onClose}\n onCloseAnimationEnd={onCloseAnimationEnd}\n >\n {renderMessage()}\n {isBot && (\n <Button color=\"danger\" className=\"confirm-dialog-button\" isText onClick={handleDeleteAndStop}>\n {lang('DeleteAndStop')}\n </Button>\n )}\n {canDeleteForAll && (\n <Button color=\"danger\" className=\"confirm-dialog-button\" isText onClick={handleDeleteMessageForAll}>\n {contactName ? renderText(lang('ChatList.DeleteForEveryone', contactName)) : lang('DeleteForAll')}\n </Button>\n )}\n <Button color=\"danger\" className=\"confirm-dialog-button\" isText onClick={handleDeleteChat}>\n {lang(renderActionText())}\n </Button>\n <Button className=\"confirm-dialog-button\" isText onClick={onClose}>{lang('Cancel')}</Button>\n </Modal>\n );\n};\n\nexport default memo(withGlobal<OwnProps>(\n (global, { chat }): StateProps => {\n const isPrivateChat = isChatPrivate(chat.id);\n const isChatWithSelf = selectIsChatWithSelf(global, chat.id);\n const user = isPrivateChat && selectUser(global, getPrivateChatUserId(chat)!);\n const isBot = user && isUserBot(user) && !chat.isSupport;\n const canDeleteForAll = (isPrivateChat && !isChatWithSelf && !isBot);\n const contactName = isPrivateChat\n ? getUserFirstOrLastName(selectUser(global, getPrivateChatUserId(chat)!))\n : undefined;\n\n return {\n isPrivateChat,\n isChatWithSelf,\n isBot,\n isChannel: isChatChannel(chat),\n isBasicGroup: isChatBasicGroup(chat),\n isSuperGroup: isChatSuperGroup(chat),\n currentUserId: global.currentUserId,\n canDeleteForAll,\n contactName,\n };\n },\n (setGlobal, actions): DispatchProps => pick(actions,\n ['leaveChannel', 'deleteHistory', 'deleteChannel', 'deleteChatUser', 'blockContact']),\n)(DeleteChatModal));\n","import React, { FC, useRef } from '../../lib/teact/teact';\n\nimport useShowTransition from '../../hooks/useShowTransition';\nimport usePrevious from '../../hooks/usePrevious';\nimport buildClassName from '../../util/buildClassName';\n\ntype ChildrenFn = () => any;\n\ntype OwnProps = {\n isOpen: boolean;\n isCustom?: boolean;\n id?: string;\n className?: string;\n onClick?: (e: React.MouseEvent<HTMLElement, MouseEvent>) => void;\n children: ChildrenFn;\n};\n\nconst ShowTransition: FC<OwnProps> = ({\n isOpen, isCustom, id, className, onClick, children,\n}) => {\n const { shouldRender, transitionClassNames } = useShowTransition(\n isOpen, undefined, undefined, isCustom ? false : undefined,\n );\n const prevIsOpen = usePrevious(isOpen);\n const prevChildren = usePrevious(children);\n const fromChildrenRef = useRef<ChildrenFn>();\n\n if (prevIsOpen && !isOpen) {\n fromChildrenRef.current = prevChildren;\n }\n\n return (\n shouldRender && (\n <div id={id} className={buildClassName(className, transitionClassNames)} onClick={onClick}>\n {isOpen ? children() : fromChildrenRef.current!()}\n </div>\n )\n );\n};\n\nexport default ShowTransition;\n","/*\n * Thanks to Ace Monkey for this mind-blowing patch.\n */\n\nexport function patchSafariProgressiveAudio(audioEl: HTMLAudioElement) {\n if (audioEl.dataset.patchedForSafari) {\n return;\n }\n\n audioEl.addEventListener('play', () => {\n const t = audioEl.currentTime;\n\n function onProgress() {\n if (!audioEl.buffered.length) {\n return;\n }\n\n audioEl.dataset.patchForSafariInProgress = 'true';\n audioEl.currentTime = audioEl.duration - 1;\n audioEl.addEventListener('progress', () => {\n delete audioEl.dataset.patchForSafariInProgress;\n audioEl.currentTime = t;\n if (audioEl.paused) {\n audioEl.play();\n }\n }, { once: true });\n\n audioEl.removeEventListener('progress', onProgress);\n }\n\n audioEl.addEventListener('progress', onProgress);\n }, { once: true });\n\n audioEl.dataset.patchedForSafari = 'true';\n}\n\nexport function isSafariPatchInProgress(audioEl: HTMLAudioElement) {\n return Boolean(audioEl.dataset.patchForSafariInProgress);\n}\n","import { useLayoutEffect } from '../lib/teact/teact';\nimport usePrevious from './usePrevious';\n\nexport default <T extends any[], PT = T>(cb: (args: PT) => void, dependencies: T, debugKey?: string) => {\n const prevDeps = usePrevious<T>(dependencies);\n return useLayoutEffect(() => {\n // @ts-ignore (workaround for \"could be instantiated with a different subtype\" issue)\n return cb(prevDeps || []);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, dependencies, debugKey);\n};\n","import React, {\n FC, useCallback, useEffect, useState, memo, useRef,\n} from '../../lib/teact/teact';\n\nimport { ApiMessage } from '../../api/types';\n\nimport { getDocumentExtension, getDocumentHasPreview } from './helpers/documentInfo';\nimport {\n getMediaTransferState,\n getMessageMediaHash,\n getMessageMediaThumbDataUri,\n isMessageDocumentVideo,\n} from '../../modules/helpers';\nimport { ObserveFn, useIsIntersecting } from '../../hooks/useIntersectionObserver';\nimport useMediaWithDownloadProgress from '../../hooks/useMediaWithDownloadProgress';\nimport useMedia from '../../hooks/useMedia';\nimport download from '../../util/download';\n\nimport File from './File';\n\ntype OwnProps = {\n message: ApiMessage;\n observeIntersection?: ObserveFn;\n smaller?: boolean;\n isSelected?: boolean;\n isSelectable?: boolean;\n uploadProgress?: number;\n withDate?: boolean;\n datetime?: number;\n className?: string;\n sender?: string;\n onCancelUpload?: () => void;\n onMediaClick?: () => void;\n onDateClick?: (messageId: number, chatId: number) => void;\n};\n\nconst Document: FC<OwnProps> = ({\n message,\n observeIntersection,\n smaller,\n uploadProgress,\n withDate,\n datetime,\n className,\n sender,\n isSelected,\n isSelectable,\n onCancelUpload,\n onMediaClick,\n onDateClick,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const ref = useRef<HTMLDivElement>(null);\n\n const document = message.content.document!;\n const extension = getDocumentExtension(document) || '';\n const { fileName, size, timestamp } = document;\n const withMediaViewer = onMediaClick && Boolean(document.mediaType);\n\n const isIntersecting = useIsIntersecting(ref, observeIntersection);\n\n const [isDownloadAllowed, setIsDownloadAllowed] = useState(false);\n const {\n mediaData, downloadProgress,\n } = useMediaWithDownloadProgress(getMessageMediaHash(message, 'download'), !isDownloadAllowed);\n const {\n isUploading, isTransferring, transferProgress,\n } = getMediaTransferState(message, uploadProgress || downloadProgress, isDownloadAllowed);\n\n const hasPreview = getDocumentHasPreview(document);\n const thumbDataUri = hasPreview ? getMessageMediaThumbDataUri(message) : undefined;\n const localBlobUrl = hasPreview ? document.previewBlobUrl : undefined;\n const previewData = useMedia(getMessageMediaHash(message, 'pictogram'), !isIntersecting);\n\n const handleClick = useCallback(() => {\n if (withMediaViewer) {\n onMediaClick!();\n } else if (isUploading) {\n if (onCancelUpload) {\n onCancelUpload();\n }\n } else {\n setIsDownloadAllowed((isAllowed) => !isAllowed);\n }\n }, [withMediaViewer, isUploading, onCancelUpload, onMediaClick]);\n\n const handleDateClick = useCallback(() => {\n onDateClick!(message.id, message.chatId);\n }, [onDateClick, message.id, message.chatId]);\n\n useEffect(() => {\n if (isDownloadAllowed && mediaData) {\n download(mediaData, fileName);\n setIsDownloadAllowed(false);\n }\n }, [fileName, mediaData, isDownloadAllowed]);\n\n return (\n <File\n ref={ref}\n name={fileName}\n extension={extension}\n size={size}\n timestamp={withDate ? datetime || timestamp : undefined}\n thumbnailDataUri={thumbDataUri}\n previewData={localBlobUrl || previewData}\n smaller={smaller}\n isTransferring={isTransferring}\n isUploading={isUploading}\n transferProgress={transferProgress}\n className={className}\n sender={sender}\n isSelectable={isSelectable}\n isSelected={isSelected}\n actionIcon={withMediaViewer ? (isMessageDocumentVideo(message) ? 'icon-play' : 'icon-eye') : 'icon-download'}\n onClick={handleClick}\n onDateClick={onDateClick ? handleDateClick : undefined}\n />\n );\n};\n\nexport default memo(Document);\n","import { useEffect, useRef } from '../lib/teact/teact';\n\nimport fastBlur from '../lib/fastBlur';\nimport useForceUpdate from './useForceUpdate';\nimport { IS_CANVAS_FILTER_SUPPORTED } from '../util/environment';\n\nconst RADIUS = 2;\nconst ITERATIONS = 2;\n\nexport default function useCanvasBlur(dataUri?: string, isDisabled = false, withRaf?: boolean) {\n // eslint-disable-next-line no-null/no-null\n const canvasRef = useRef<HTMLCanvasElement>(null);\n const forceUpdate = useForceUpdate();\n\n useEffect(() => {\n const canvas = canvasRef.current;\n\n if (!dataUri || !canvas || isDisabled) {\n return;\n }\n\n const img = new Image();\n\n const processBlur = () => {\n canvas.width = img.width;\n canvas.height = img.height;\n\n const ctx = canvas.getContext('2d', { alpha: false })!;\n\n if (IS_CANVAS_FILTER_SUPPORTED) {\n ctx.filter = `blur(${RADIUS}px)`;\n }\n\n ctx.drawImage(img, -RADIUS * 2, -RADIUS * 2, canvas.width + RADIUS * 4, canvas.height + RADIUS * 4);\n\n if (!IS_CANVAS_FILTER_SUPPORTED) {\n fastBlur(ctx, 0, 0, canvas.width, canvas.height, RADIUS, ITERATIONS);\n }\n };\n\n img.onload = () => {\n if (withRaf) {\n requestAnimationFrame(processBlur);\n } else {\n processBlur();\n }\n };\n\n img.src = dataUri;\n }, [canvasRef, dataUri, forceUpdate, isDisabled, withRaf]);\n\n return canvasRef;\n}\n","/* eslint-disable no-nested-ternary */\n/* eslint-disable no-bitwise */\n/* eslint-disable no-multi-assign */\n/* eslint-disable no-cond-assign */\n/* eslint-disable no-param-reassign */\n/* eslint-disable prefer-const */\n/* eslint-disable eqeqeq */\n\n/*\nSuperfast Blur - a fast Box Blur For Canvas\n\nVersion: 0.5\nAuthor: Mario Klingemann\nContact: mario@quasimondo.com\nWebsite: http://www.quasimondo.com/BoxBlurForCanvas\nTwitter: @quasimondo\n\nIn case you find this class useful - especially in commercial projects -\nI am not totally unhappy for a small donation to my PayPal account\nmario@quasimondo.de\n\nOr support me on flattr:\nhttps://flattr.com/thing/140066/Superfast-Blur-a-pretty-fast-Box-Blur-Effect-for-CanvasJavascript\n\nCopyright (c) 2011 Mario Klingemann\n\nPermission is hereby granted, free of charge, to any person\nobtaining a copy of this software and associated documentation\nfiles (the \"Software\"), to deal in the Software without\nrestriction, including without limitation the rights to use,\ncopy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the\nSoftware is furnished to do so, subject to the following\nconditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES\nOF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND\nNONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT\nHOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,\nWHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\nFROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR\nOTHER DEALINGS IN THE SOFTWARE.\n*/\n\n// eslint-disable-next-line max-len\nconst mul_table = [1, 57, 41, 21, 203, 34, 97, 73, 227, 91, 149, 62, 105, 45, 39, 137, 241, 107, 3, 173, 39, 71, 65, 238, 219, 101, 187, 87, 81, 151, 141, 133, 249, 117, 221, 209, 197, 187, 177, 169, 5, 153, 73, 139, 133, 127, 243, 233, 223, 107, 103, 99, 191, 23, 177, 171, 165, 159, 77, 149, 9, 139, 135, 131, 253, 245, 119, 231, 224, 109, 211, 103, 25, 195, 189, 23, 45, 175, 171, 83, 81, 79, 155, 151, 147, 9, 141, 137, 67, 131, 129, 251, 123, 30, 235, 115, 113, 221, 217, 53, 13, 51, 50, 49, 193, 189, 185, 91, 179, 175, 43, 169, 83, 163, 5, 79, 155, 19, 75, 147, 145, 143, 35, 69, 17, 67, 33, 65, 255, 251, 247, 243, 239, 59, 29, 229, 113, 111, 219, 27, 213, 105, 207, 51, 201, 199, 49, 193, 191, 47, 93, 183, 181, 179, 11, 87, 43, 85, 167, 165, 163, 161, 159, 157, 155, 77, 19, 75, 37, 73, 145, 143, 141, 35, 138, 137, 135, 67, 33, 131, 129, 255, 63, 250, 247, 61, 121, 239, 237, 117, 29, 229, 227, 225, 111, 55, 109, 216, 213, 211, 209, 207, 205, 203, 201, 199, 197, 195, 193, 48, 190, 47, 93, 185, 183, 181, 179, 178, 176, 175, 173, 171, 85, 21, 167, 165, 41, 163, 161, 5, 79, 157, 78, 154, 153, 19, 75, 149, 74, 147, 73, 144, 143, 71, 141, 140, 139, 137, 17, 135, 134, 133, 66, 131, 65, 129, 1];\n// eslint-disable-next-line max-len\nconst shg_table = [0, 9, 10, 10, 14, 12, 14, 14, 16, 15, 16, 15, 16, 15, 15, 17, 18, 17, 12, 18, 16, 17, 17, 19, 19, 18, 19, 18, 18, 19, 19, 19, 20, 19, 20, 20, 20, 20, 20, 20, 15, 20, 19, 20, 20, 20, 21, 21, 21, 20, 20, 20, 21, 18, 21, 21, 21, 21, 20, 21, 17, 21, 21, 21, 22, 22, 21, 22, 22, 21, 22, 21, 19, 22, 22, 19, 20, 22, 22, 21, 21, 21, 22, 22, 22, 18, 22, 22, 21, 22, 22, 23, 22, 20, 23, 22, 22, 23, 23, 21, 19, 21, 21, 21, 23, 23, 23, 22, 23, 23, 21, 23, 22, 23, 18, 22, 23, 20, 22, 23, 23, 23, 21, 22, 20, 22, 21, 22, 24, 24, 24, 24, 24, 22, 21, 24, 23, 23, 24, 21, 24, 23, 24, 22, 24, 24, 22, 24, 24, 22, 23, 24, 24, 24, 20, 23, 22, 23, 24, 24, 24, 24, 24, 24, 24, 23, 21, 23, 22, 23, 24, 24, 24, 22, 24, 24, 24, 23, 22, 24, 24, 25, 23, 25, 25, 23, 24, 25, 25, 24, 22, 25, 25, 25, 24, 23, 24, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 23, 25, 23, 24, 25, 25, 25, 25, 25, 25, 25, 25, 25, 24, 22, 25, 25, 23, 25, 25, 20, 24, 25, 24, 25, 25, 22, 24, 25, 24, 25, 24, 25, 25, 24, 25, 25, 25, 25, 22, 25, 25, 25, 24, 25, 24, 25, 18];\n\nexport default function boxBlurCanvasRGB(context, top_x, top_y, width, height, radius, iterations) {\n if (Number.isNaN(radius) || radius < 1) return;\n\n radius |= 0;\n\n if (Number.isNaN(iterations)) iterations = 1;\n iterations |= 0;\n if (iterations > 3) iterations = 3;\n if (iterations < 1) iterations = 1;\n\n const imageData = context.getImageData(top_x, top_y, width, height);\n\n const pixels = imageData.data;\n\n let rsum;\n let gsum;\n let bsum;\n let x;\n let y;\n let i;\n let p;\n let p1;\n let p2;\n let yp;\n let yi;\n let yw;\n let wm = width - 1;\n let hm = height - 1;\n let rad1 = radius + 1;\n\n let r = [];\n let g = [];\n let b = [];\n\n let mul_sum = mul_table[radius];\n let shg_sum = shg_table[radius];\n\n let vmin = [];\n let vmax = [];\n\n while (iterations-- > 0) {\n yw = yi = 0;\n\n for (y = 0; y < height; y++) {\n rsum = pixels[yw] * rad1;\n gsum = pixels[yw + 1] * rad1;\n bsum = pixels[yw + 2] * rad1;\n\n for (i = 1; i <= radius; i++) {\n p = yw + (((i > wm ? wm : i)) << 2);\n rsum += pixels[p++];\n gsum += pixels[p++];\n bsum += pixels[p++];\n }\n\n for (x = 0; x < width; x++) {\n r[yi] = rsum;\n g[yi] = gsum;\n b[yi] = bsum;\n\n if (y == 0) {\n vmin[x] = ((p = x + rad1) < wm ? p : wm) << 2;\n vmax[x] = ((p = x - radius) > 0 ? p << 2 : 0);\n }\n\n p1 = yw + vmin[x];\n p2 = yw + vmax[x];\n\n rsum += pixels[p1++] - pixels[p2++];\n gsum += pixels[p1++] - pixels[p2++];\n bsum += pixels[p1++] - pixels[p2++];\n\n yi++;\n }\n yw += (width << 2);\n }\n\n for (x = 0; x < width; x++) {\n yp = x;\n rsum = r[yp] * rad1;\n gsum = g[yp] * rad1;\n bsum = b[yp] * rad1;\n\n for (i = 1; i <= radius; i++) {\n yp += (i > hm ? 0 : width);\n rsum += r[yp];\n gsum += g[yp];\n bsum += b[yp];\n }\n\n yi = x << 2;\n for (y = 0; y < height; y++) {\n pixels[yi] = (rsum * mul_sum) >>> shg_sum;\n pixels[yi + 1] = (gsum * mul_sum) >>> shg_sum;\n pixels[yi + 2] = (bsum * mul_sum) >>> shg_sum;\n\n if (x == 0) {\n vmin[y] = ((p = y + rad1) < hm ? p : hm) * width;\n vmax[y] = ((p = y - radius) > 0 ? p * width : 0);\n }\n\n p1 = x + vmin[y];\n p2 = x + vmax[y];\n\n rsum += r[p1] - r[p2];\n gsum += g[p1] - g[p2];\n bsum += b[p1] - b[p2];\n\n yi += width << 2;\n }\n }\n }\n\n context.putImageData(imageData, top_x, top_y);\n}\n","import React, {\n FC, memo, useCallback, useEffect, useRef, useState,\n} from '../../../lib/teact/teact';\n\nimport { IAnchorPosition } from '../../../types';\n\nimport { EDITABLE_INPUT_ID } from '../../../config';\nimport buildClassName from '../../../util/buildClassName';\nimport captureEscKeyListener from '../../../util/captureEscKeyListener';\nimport useShowTransition from '../../../hooks/useShowTransition';\nimport useVirtualBackdrop from '../../../hooks/useVirtualBackdrop';\nimport useFlag from '../../../hooks/useFlag';\nimport useLang from '../../../hooks/useLang';\n\nimport Button from '../../ui/Button';\n\nimport './TextFormatter.scss';\n\nexport type OwnProps = {\n isOpen: boolean;\n anchorPosition?: IAnchorPosition;\n selectedRange?: Range;\n onClose: () => void;\n};\n\ninterface ISelectedTextFormats {\n bold?: boolean;\n italic?: boolean;\n underline?: boolean;\n strikethrough?: boolean;\n monospace?: boolean;\n}\n\nconst TEXT_FORMAT_BY_TAG_NAME: Record<string, keyof ISelectedTextFormats> = {\n B: 'bold',\n STRONG: 'bold',\n I: 'italic',\n EM: 'italic',\n U: 'underline',\n DEL: 'strikethrough',\n CODE: 'monospace',\n};\nconst fragmentEl = document.createElement('div');\n\nconst TextFormatter: FC<OwnProps> = ({\n isOpen,\n anchorPosition,\n selectedRange,\n onClose,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const containerRef = useRef<HTMLDivElement>(null);\n // eslint-disable-next-line no-null/no-null\n const linkUrlInputRef = useRef<HTMLInputElement>(null);\n const { shouldRender, transitionClassNames } = useShowTransition(isOpen);\n const [isLinkControlOpen, openLinkControl, closeLinkControl] = useFlag();\n const [linkUrl, setLinkUrl] = useState('');\n const [isEditingLink, setIsEditingLink] = useState(false);\n const [inputClassName, setInputClassName] = useState<string | undefined>();\n const [selectedTextFormats, setSelectedTextFormats] = useState<ISelectedTextFormats>({});\n\n useEffect(() => (isOpen ? captureEscKeyListener(onClose) : undefined), [isOpen, onClose]);\n useVirtualBackdrop(\n isOpen,\n containerRef,\n onClose,\n );\n\n useEffect(() => {\n if (isLinkControlOpen) {\n linkUrlInputRef.current!.focus();\n } else {\n setLinkUrl('');\n setIsEditingLink(false);\n }\n }, [isLinkControlOpen]);\n\n useEffect(() => {\n if (!shouldRender) {\n closeLinkControl();\n setSelectedTextFormats({});\n setInputClassName(undefined);\n }\n }, [closeLinkControl, shouldRender]);\n\n useEffect(() => {\n if (!isOpen || !selectedRange) {\n return;\n }\n\n const selectedFormats: ISelectedTextFormats = {};\n let { parentElement } = selectedRange.commonAncestorContainer;\n while (parentElement && parentElement.id !== EDITABLE_INPUT_ID) {\n const textFormat = TEXT_FORMAT_BY_TAG_NAME[parentElement.tagName];\n if (textFormat) {\n selectedFormats[textFormat] = true;\n }\n\n parentElement = parentElement.parentElement;\n }\n\n setSelectedTextFormats(selectedFormats);\n }, [isOpen, selectedRange, openLinkControl]);\n\n function restoreSelection() {\n if (!selectedRange) {\n return;\n }\n\n const selection = window.getSelection();\n if (selection) {\n selection.removeAllRanges();\n selection.addRange(selectedRange);\n }\n }\n\n const getSelectedText = useCallback(() => {\n if (!selectedRange) {\n return undefined;\n }\n fragmentEl.innerText = selectedRange.toString();\n\n return fragmentEl.innerHTML;\n }, [selectedRange]);\n\n const getSelectedElement = useCallback(() => {\n if (!selectedRange) {\n return undefined;\n }\n\n return selectedRange.commonAncestorContainer.parentElement;\n }, [selectedRange]);\n\n function updateInputStyles() {\n const input = linkUrlInputRef.current;\n if (!input) {\n return;\n }\n\n const { offsetWidth, scrollWidth, scrollLeft } = input;\n if (scrollWidth <= offsetWidth) {\n setInputClassName(undefined);\n return;\n }\n\n let className = '';\n if (scrollLeft < scrollWidth - offsetWidth) {\n className = 'mask-right';\n }\n if (scrollLeft > 0) {\n className += ' mask-left';\n }\n\n setInputClassName(className);\n }\n\n function handleLinkUrlChange(e: React.ChangeEvent<HTMLInputElement>) {\n setLinkUrl(e.target.value);\n updateInputStyles();\n }\n\n function getFormatButtonClassName(key: keyof ISelectedTextFormats) {\n if (selectedTextFormats[key]) {\n return 'active';\n }\n\n if (key === 'monospace' || key === 'strikethrough') {\n if (Object.keys(selectedTextFormats).some(\n (fKey) => fKey !== key && !!selectedTextFormats[fKey as keyof ISelectedTextFormats],\n )) {\n return 'disabled';\n }\n } else if (selectedTextFormats.monospace || selectedTextFormats.strikethrough) {\n return 'disabled';\n }\n\n return undefined;\n }\n\n const handleBoldText = useCallback(() => {\n setSelectedTextFormats((selectedFormats) => {\n // Somehow re-applying 'bold' command to already bold text doesn't work\n document.execCommand(selectedFormats.bold ? 'removeFormat' : 'bold');\n Object.keys(selectedFormats).forEach((key) => {\n if ((key === 'italic' || key === 'underline') && !!selectedFormats[key]) {\n document.execCommand(key);\n }\n });\n\n return {\n ...selectedFormats,\n bold: !selectedFormats.bold,\n };\n });\n }, []);\n\n const handleItalicText = useCallback(() => {\n document.execCommand('italic');\n setSelectedTextFormats((selectedFormats) => ({\n ...selectedFormats,\n italic: !selectedFormats.italic,\n }));\n }, []);\n\n const handleUnderlineText = useCallback(() => {\n document.execCommand('underline');\n setSelectedTextFormats((selectedFormats) => ({\n ...selectedFormats,\n underline: !selectedFormats.underline,\n }));\n }, []);\n\n const handleStrikethroughText = useCallback(() => {\n if (selectedTextFormats.strikethrough) {\n const element = getSelectedElement();\n if (\n !selectedRange\n || !element\n || element.tagName !== 'DEL'\n || !element.textContent\n ) {\n return;\n }\n\n element.replaceWith(element.textContent);\n setSelectedTextFormats((selectedFormats) => ({\n ...selectedFormats,\n strikethrough: false,\n }));\n\n return;\n }\n\n const text = getSelectedText();\n document.execCommand('insertHTML', false, `<del>${text}</del>`);\n onClose();\n }, [\n getSelectedElement, getSelectedText, onClose,\n selectedRange, selectedTextFormats.strikethrough,\n ]);\n\n const handleMonospaceText = useCallback(() => {\n if (selectedTextFormats.monospace) {\n const element = getSelectedElement();\n if (\n !selectedRange\n || !element\n || element.tagName !== 'CODE'\n || !element.textContent\n ) {\n return;\n }\n\n element.replaceWith(element.textContent);\n setSelectedTextFormats((selectedFormats) => ({\n ...selectedFormats,\n monospace: false,\n }));\n return;\n }\n\n const text = getSelectedText();\n document.execCommand('insertHTML', false, `<code class=\"text-entity-code\" dir=\"auto\">${text}</code>`);\n onClose();\n }, [\n getSelectedElement, getSelectedText, onClose,\n selectedRange, selectedTextFormats.monospace,\n ]);\n\n function handleLinkUrlConfirm() {\n const formattedLinkUrl = encodeURI(linkUrl.includes('://') ? linkUrl : `http://${linkUrl}`);\n\n if (isEditingLink) {\n const element = getSelectedElement();\n if (!element || element.tagName !== 'A') {\n return;\n }\n\n (element as HTMLAnchorElement).href = formattedLinkUrl;\n\n onClose();\n return;\n }\n\n const text = getSelectedText();\n restoreSelection();\n document.execCommand(\n 'insertHTML',\n false,\n `<a href=${formattedLinkUrl} class=\"text-entity-link\" dir=\"auto\">${text}</a>`,\n );\n onClose();\n }\n\n const handleKeyDown = useCallback((e: KeyboardEvent) => {\n const HANDLERS_BY_KEY_CODE: Record<string, AnyToVoidFunction> = {\n KeyK: openLinkControl,\n KeyB: handleBoldText,\n KeyU: handleUnderlineText,\n KeyI: handleItalicText,\n KeyM: handleMonospaceText,\n KeyS: handleStrikethroughText,\n };\n\n const handler = HANDLERS_BY_KEY_CODE[e.code];\n\n if (\n e.altKey\n || !(e.ctrlKey || e.metaKey)\n || !handler\n ) {\n return;\n }\n\n e.preventDefault();\n e.stopPropagation();\n handler();\n }, [\n handleBoldText, handleItalicText, handleUnderlineText,\n handleMonospaceText, handleStrikethroughText,\n openLinkControl,\n ]);\n\n useEffect(() => {\n if (isOpen) {\n document.addEventListener('keydown', handleKeyDown);\n }\n\n return () => document.removeEventListener('keydown', handleKeyDown);\n }, [isOpen, handleKeyDown]);\n\n const lang = useLang();\n\n function handleContainerKeyDown(e: React.KeyboardEvent<HTMLDivElement>) {\n if (e.key === 'Enter' && isLinkControlOpen) {\n handleLinkUrlConfirm();\n e.preventDefault();\n }\n }\n\n if (!shouldRender) {\n return undefined;\n }\n\n const className = buildClassName(\n 'TextFormatter',\n transitionClassNames,\n isLinkControlOpen && 'link-control-shown',\n );\n\n const linkUrlConfirmClassName = buildClassName(\n 'TextFormatter-link-url-confirm',\n !!linkUrl.length && 'shown',\n );\n\n const style = anchorPosition\n ? `left: ${anchorPosition.x}px; top: ${anchorPosition.y}px;--text-formatter-left: ${anchorPosition.x}px;`\n : '';\n\n return (\n <div\n ref={containerRef}\n className={className}\n // @ts-ignore Teact feature\n style={style}\n onKeyDown={handleContainerKeyDown}\n >\n <div className=\"TextFormatter-buttons\">\n <Button\n color=\"translucent\"\n ariaLabel=\"Bold text\"\n className={getFormatButtonClassName('bold')}\n onClick={handleBoldText}\n >\n <i className=\"icon-bold\" />\n </Button>\n <Button\n color=\"translucent\"\n ariaLabel=\"Italic text\"\n className={getFormatButtonClassName('italic')}\n onClick={handleItalicText}\n >\n <i className=\"icon-italic\" />\n </Button>\n <Button\n color=\"translucent\"\n ariaLabel=\"Underlined text\"\n className={getFormatButtonClassName('underline')}\n onClick={handleUnderlineText}\n >\n <i className=\"icon-underlined\" />\n </Button>\n <Button\n color=\"translucent\"\n ariaLabel=\"Strikethrough text\"\n className={getFormatButtonClassName('strikethrough')}\n onClick={handleStrikethroughText}\n >\n <i className=\"icon-strikethrough\" />\n </Button>\n <Button\n color=\"translucent\"\n ariaLabel=\"Monospace text\"\n className={getFormatButtonClassName('monospace')}\n onClick={handleMonospaceText}\n >\n <i className=\"icon-monospace\" />\n </Button>\n <div className=\"TextFormatter-divider\" />\n <Button color=\"translucent\" ariaLabel={lang('TextFormat.AddLinkTitle')} onClick={openLinkControl}>\n <i className=\"icon-link\" />\n </Button>\n </div>\n\n <div className=\"TextFormatter-link-control\">\n <div className=\"TextFormatter-buttons\">\n <Button color=\"translucent\" ariaLabel={lang('Cancel')} onClick={closeLinkControl}>\n <i className=\"icon-arrow-left\" />\n </Button>\n <div className=\"TextFormatter-divider\" />\n\n <div\n className={buildClassName('TextFormatter-link-url-input-wrapper', inputClassName)}\n >\n <input\n ref={linkUrlInputRef}\n className=\"TextFormatter-link-url-input\"\n type=\"text\"\n value={linkUrl}\n placeholder=\"Enter URL...\"\n autoComplete=\"off\"\n inputMode=\"url\"\n dir=\"auto\"\n onChange={handleLinkUrlChange}\n onScroll={updateInputStyles}\n />\n </div>\n\n <div className={linkUrlConfirmClassName}>\n <div className=\"TextFormatter-divider\" />\n <Button\n color=\"translucent\"\n ariaLabel={lang('Save')}\n className=\"color-primary\"\n onClick={handleLinkUrlConfirm}\n >\n <i className=\"icon-check\" />\n </Button>\n </div>\n </div>\n </div>\n </div>\n );\n};\n\nexport default memo(TextFormatter);\n","import { DEBUG } from '../config';\n\nexport const CLIPBOARD_ITEM_SUPPORTED = navigator.clipboard && window.ClipboardItem;\n\nconst textCopyEl = document.createElement('textarea');\ntextCopyEl.setAttribute('readonly', '');\ntextCopyEl.tabIndex = -1;\ntextCopyEl.className = 'visually-hidden';\n\nexport const copyTextToClipboard = (str: string): void => {\n textCopyEl.value = str;\n document.body.appendChild(textCopyEl);\n const selection = document.getSelection();\n\n if (selection) {\n // Store previous selection\n const rangeToRestore = selection.rangeCount > 0 && selection.getRangeAt(0);\n textCopyEl.select();\n document.execCommand('copy');\n // Restore the original selection\n if (rangeToRestore) {\n selection.removeAllRanges();\n selection.addRange(rangeToRestore);\n }\n }\n\n document.body.removeChild(textCopyEl);\n};\n\nexport const copyImageToClipboard = (imageUrl?: string) => {\n if (!imageUrl) return;\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n const imageEl = new Image();\n imageEl.onload = (e: Event) => {\n if (ctx && e.currentTarget) {\n const img = e.currentTarget as HTMLImageElement;\n canvas.width = img.width;\n canvas.height = img.height;\n ctx.drawImage(img, 0, 0, img.width, img.height);\n canvas.toBlob(copyBlobToClipboard, 'image/png', 1);\n }\n };\n\n imageEl.src = imageUrl;\n};\n\nasync function copyBlobToClipboard(pngBlob: Blob | null) {\n if (!pngBlob || !CLIPBOARD_ITEM_SUPPORTED) {\n return;\n }\n\n try {\n await navigator.clipboard.write([\n new window.ClipboardItem({\n [pngBlob.type]: pngBlob,\n }),\n ]);\n } catch (error) {\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.error(error);\n }\n }\n}\n","import React, { FC, memo, useCallback } from '../../lib/teact/teact';\n\nimport { ApiMessage } from '../../api/types';\n\nimport { formatMediaDuration } from '../../util/dateFormat';\nimport {\n getMessageMediaHash,\n getMessageMediaThumbDataUri,\n getMessageVideo,\n} from '../../modules/helpers';\nimport useMedia from '../../hooks/useMedia';\nimport useTransitionForMedia from '../../hooks/useTransitionForMedia';\n\nimport './Media.scss';\n\ntype OwnProps = {\n message: ApiMessage;\n idPrefix?: string;\n onClick?: (messageId: number, chatId: number) => void;\n};\n\nconst Media: FC<OwnProps> = ({ message, idPrefix = 'shared-media', onClick }) => {\n const handleClick = useCallback(() => {\n onClick!(message.id, message.chatId);\n }, [message.id, message.chatId, onClick]);\n\n const thumbDataUri = getMessageMediaThumbDataUri(message);\n const mediaBlobUrl = useMedia(getMessageMediaHash(message, 'pictogram'));\n const {\n shouldRenderThumb, shouldRenderFullMedia, transitionClassNames,\n } = useTransitionForMedia(mediaBlobUrl, 'slow');\n\n const video = getMessageVideo(message);\n\n return (\n <div id={`${idPrefix}${message.id}`} className=\"Media scroll-item\" onClick={onClick ? handleClick : undefined}>\n {shouldRenderThumb && (\n <img src={thumbDataUri} alt=\"\" />\n )}\n {shouldRenderFullMedia && (\n <img src={mediaBlobUrl} className={`${transitionClassNames} full-media`} alt=\"\" />\n )}\n {video && <span className=\"video-duration\">{video.isGif ? 'GIF' : formatMediaDuration(video.duration)}</span>}\n </div>\n );\n};\n\nexport default memo(Media);\n","import React, {\n FC, useCallback, useRef, useEffect, memo,\n} from '../../lib/teact/teact';\n\nimport { isChatPrivate } from '../../modules/helpers';\n\nimport InfiniteScroll from '../ui/InfiniteScroll';\nimport Checkbox from '../ui/Checkbox';\nimport InputText from '../ui/InputText';\nimport ListItem from '../ui/ListItem';\nimport PrivateChatInfo from './PrivateChatInfo';\nimport GroupChatInfo from './GroupChatInfo';\nimport PickerSelectedItem from './PickerSelectedItem';\nimport useInfiniteScroll from '../../hooks/useInfiniteScroll';\nimport useLang from '../../hooks/useLang';\n\nimport Loading from '../ui/Loading';\n\nimport './Picker.scss';\n\ntype OwnProps = {\n itemIds: number[];\n selectedIds: number[];\n filterValue?: string;\n filterPlaceholder?: string;\n notFoundText?: string;\n searchInputId?: string;\n isLoading?: boolean;\n noScrollRestore?: boolean;\n onSelectedIdsChange: (ids: number[]) => void;\n onFilterChange: (value: string) => void;\n onLoadMore?: () => void;\n};\n\n// Focus slows down animation, also it breaks transition layout in Chrome\nconst FOCUS_DELAY_MS = 500;\n\nconst MAX_FULL_ITEMS = 10;\nconst ALWAYS_FULL_ITEMS_COUNT = 5;\n\nconst Picker: FC<OwnProps> = ({\n itemIds,\n selectedIds,\n filterValue,\n filterPlaceholder,\n notFoundText,\n searchInputId,\n isLoading,\n noScrollRestore,\n onSelectedIdsChange,\n onFilterChange,\n onLoadMore,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const inputRef = useRef<HTMLInputElement>(null);\n const shouldMinimize = selectedIds.length > MAX_FULL_ITEMS;\n\n useEffect(() => {\n setTimeout(() => {\n requestAnimationFrame(() => {\n inputRef.current!.focus();\n });\n }, FOCUS_DELAY_MS);\n }, []);\n\n const handleItemClick = useCallback((id: number) => {\n const newSelectedIds = [...selectedIds];\n if (newSelectedIds.includes(id)) {\n newSelectedIds.splice(newSelectedIds.indexOf(id), 1);\n } else {\n newSelectedIds.push(id);\n }\n onSelectedIdsChange(newSelectedIds);\n onFilterChange('');\n }, [selectedIds, onSelectedIdsChange, onFilterChange]);\n\n const handleFilterChange = useCallback((e: React.ChangeEvent<HTMLInputElement>) => {\n const { value } = e.currentTarget;\n onFilterChange(value);\n }, [onFilterChange]);\n\n const [viewportIds, getMore] = useInfiniteScroll(onLoadMore, itemIds, Boolean(filterValue));\n\n const lang = useLang();\n\n return (\n <div className=\"Picker\">\n <div className=\"picker-header custom-scroll\" dir={lang.isRtl ? 'rtl' : undefined}>\n {selectedIds.map((id, i) => (\n <PickerSelectedItem\n chatOrUserId={id}\n isMinimized={shouldMinimize && i < selectedIds.length - ALWAYS_FULL_ITEMS_COUNT}\n canClose\n onClick={handleItemClick}\n clickArg={id}\n />\n ))}\n <InputText\n id={searchInputId}\n ref={inputRef}\n value={filterValue}\n onChange={handleFilterChange}\n placeholder={filterPlaceholder || lang('SelectChat')}\n />\n </div>\n\n {viewportIds && viewportIds.length ? (\n <InfiniteScroll\n className=\"picker-list custom-scroll\"\n items={viewportIds}\n onLoadMore={getMore}\n noScrollRestore={noScrollRestore}\n >\n {viewportIds.map((id) => (\n <ListItem\n key={id}\n className=\"chat-item-clickable picker-list-item\"\n onClick={() => handleItemClick(id)}\n ripple\n >\n <Checkbox label=\"\" checked={selectedIds.includes(id)} />\n {isChatPrivate(id) ? (\n <PrivateChatInfo userId={id} />\n ) : (\n <GroupChatInfo chatId={id} />\n )}\n </ListItem>\n ))}\n </InfiniteScroll>\n ) : !isLoading && viewportIds && !viewportIds.length ? (\n <p className=\"no-results\">{notFoundText || 'Sorry, nothing found.'}</p>\n ) : (\n <Loading />\n )}\n </div>\n );\n};\n\nexport default memo(Picker);\n","import React, {\n FC, memo, useCallback, useEffect, useMemo, useRef, useState,\n} from '../../lib/teact/teact';\n\nimport {\n ApiAudio, ApiMessage, ApiVoice,\n} from '../../api/types';\nimport { ISettings } from '../../types';\n\nimport { IS_SINGLE_COLUMN_LAYOUT } from '../../util/environment';\nimport { formatMediaDateTime, formatMediaDuration, formatPastTimeShort } from '../../util/dateFormat';\nimport {\n getMediaDuration,\n getMediaTransferState,\n getMessageAudioCaption,\n getMessageKey,\n getMessageMediaFormat,\n getMessageMediaHash,\n isMessageLocal,\n isOwnMessage,\n} from '../../modules/helpers';\nimport { renderWaveformToDataUri } from './helpers/waveform';\nimport buildClassName from '../../util/buildClassName';\nimport renderText from './helpers/renderText';\nimport { decodeWaveform, interpolateArray } from '../../util/waveform';\nimport useMediaWithDownloadProgress from '../../hooks/useMediaWithDownloadProgress';\nimport useShowTransition from '../../hooks/useShowTransition';\nimport useBuffering from '../../hooks/useBuffering';\nimport useAudioPlayer from '../../hooks/useAudioPlayer';\nimport useMediaDownload from '../../hooks/useMediaDownload';\nimport useLang, { LangFn } from '../../hooks/useLang';\n\nimport Button from '../ui/Button';\nimport ProgressSpinner from '../ui/ProgressSpinner';\nimport Link from '../ui/Link';\n\nimport './Audio.scss';\n\ntype OwnProps = {\n theme: ISettings['theme'];\n message: ApiMessage;\n senderTitle?: string;\n uploadProgress?: number;\n target?: 'searchResult' | 'sharedMedia';\n date?: number;\n lastSyncTime?: number;\n className?: string;\n isSelectable?: boolean;\n isSelected?: boolean;\n onPlay: (messageId: number, chatId: number) => void;\n onReadMedia?: () => void;\n onCancelUpload?: () => void;\n onDateClick?: (messageId: number, chatId: number) => void;\n};\n\ninterface ISeekMethods {\n handleStartSeek: (e: React.MouseEvent<HTMLElement>) => void;\n handleSeek: (e: React.MouseEvent<HTMLElement>) => void;\n handleStopSeek: () => void;\n}\n\nconst AVG_VOICE_DURATION = 30;\nconst MIN_SPIKES = IS_SINGLE_COLUMN_LAYOUT ? 20 : 25;\nconst MAX_SPIKES = IS_SINGLE_COLUMN_LAYOUT ? 50 : 75;\n// This is needed for browsers requiring user interaction before playing.\nconst PRELOAD = true;\n\nconst Audio: FC<OwnProps> = ({\n theme,\n message,\n senderTitle,\n uploadProgress,\n target,\n date,\n lastSyncTime,\n className,\n isSelectable,\n isSelected,\n onPlay,\n onReadMedia,\n onCancelUpload,\n onDateClick,\n}) => {\n const { content: { audio, voice }, isMediaUnread } = message;\n const isVoice = Boolean(voice);\n const isSeeking = useRef<boolean>(false);\n const lang = useLang();\n\n const [isActivated, setIsActivated] = useState(false);\n const shouldDownload = (isActivated || PRELOAD) && lastSyncTime;\n\n const { mediaData, downloadProgress } = useMediaWithDownloadProgress(\n getMessageMediaHash(message, 'inline'),\n !shouldDownload,\n getMessageMediaFormat(message, 'inline'),\n );\n\n function handleForcePlay() {\n setIsActivated(true);\n onPlay(message.id, message.chatId);\n }\n\n const {\n isBuffered, bufferedProgress, bufferingHandlers, checkBuffering,\n } = useBuffering();\n\n const {\n isPlaying, playProgress, playPause, setCurrentTime, duration,\n } = useAudioPlayer(\n getMessageKey(message),\n getMediaDuration(message)!,\n mediaData,\n bufferingHandlers,\n checkBuffering,\n isActivated,\n handleForcePlay,\n isMessageLocal(message),\n );\n\n useEffect(() => {\n setIsActivated(isPlaying);\n }, [isPlaying]);\n\n const {\n isDownloadStarted,\n downloadProgress: directDownloadProgress,\n handleDownloadClick,\n } = useMediaDownload(getMessageMediaHash(message, 'download'), getMessageAudioCaption(message));\n\n const isLoadingForPlaying = isActivated && !isBuffered;\n\n const {\n isUploading, isTransferring, transferProgress,\n } = getMediaTransferState(\n message,\n isDownloadStarted ? directDownloadProgress : (uploadProgress || downloadProgress),\n isLoadingForPlaying || isDownloadStarted,\n );\n\n const {\n shouldRender: shouldRenderSpinner,\n transitionClassNames: spinnerClassNames,\n } = useShowTransition(isTransferring);\n\n const handleButtonClick = useCallback(() => {\n if (isUploading) {\n if (onCancelUpload) {\n onCancelUpload();\n }\n\n return;\n }\n\n if (!isPlaying) {\n onPlay(message.id, message.chatId);\n }\n\n setIsActivated(!isActivated);\n playPause();\n }, [isPlaying, isUploading, message.id, message.chatId, onCancelUpload, onPlay, playPause, isActivated]);\n\n useEffect(() => {\n if (isPlaying && onReadMedia && isMediaUnread) {\n onReadMedia();\n }\n }, [isPlaying, isMediaUnread, onReadMedia]);\n\n const handleSeek = useCallback((e: React.MouseEvent<HTMLElement>) => {\n if (isSeeking.current) {\n const seekBar = e.currentTarget.closest('.seekline,.waveform');\n if (seekBar) {\n const { width, left } = seekBar.getBoundingClientRect();\n setCurrentTime(duration * ((e.clientX - left) / width));\n }\n }\n }, [duration, setCurrentTime]);\n\n const handleStartSeek = useCallback((e: React.MouseEvent<HTMLElement>) => {\n isSeeking.current = true;\n handleSeek(e);\n }, [handleSeek]);\n\n const handleStopSeek = useCallback(() => {\n isSeeking.current = false;\n }, []);\n\n const handleDateClick = useCallback(() => {\n onDateClick!(message.id, message.chatId);\n }, [onDateClick, message.id, message.chatId]);\n\n function getFirstLine() {\n if (isVoice) {\n return senderTitle || 'Voice';\n }\n\n const { title, fileName } = audio!;\n\n return title || fileName;\n }\n\n function getSecondLine() {\n if (isVoice) {\n return formatMediaDuration(voice!.duration);\n }\n\n const { performer } = audio!;\n\n return (\n <>\n {performer && renderText(performer)}\n {performer && senderTitle && <span>•</span>}\n {senderTitle && renderText(senderTitle)}\n </>\n );\n }\n\n const seekHandlers = { handleStartSeek, handleSeek, handleStopSeek };\n const isOwn = isOwnMessage(message);\n const renderedWaveform = useMemo(\n () => voice && renderWaveform(voice, playProgress, isOwn, seekHandlers, theme),\n [voice, playProgress, isOwn, seekHandlers, theme],\n );\n\n const fullClassName = buildClassName(\n 'Audio media-inner',\n className,\n isOwn && !target && 'own',\n target && 'bigger',\n isSelected && 'audio-is-selected',\n );\n\n const buttonClassNames = ['toggle-play'];\n if (isLoadingForPlaying) {\n buttonClassNames.push('loading');\n } else if (isPlaying) {\n buttonClassNames.push('pause');\n } else if (!isPlaying) {\n buttonClassNames.push('play');\n }\n\n const showSeekline = isPlaying || (playProgress > 0 && playProgress < 1);\n const contentClassName = buildClassName('content', showSeekline && 'with-seekline');\n\n function renderSearchResult() {\n return (\n <>\n <div className={contentClassName}>\n <div className=\"content-row\">\n <p className=\"title\" dir=\"auto\">{renderText(getFirstLine())}</p>\n\n <div className=\"message-date\">\n {date && (\n <Link\n className=\"date\"\n onClick={handleDateClick}\n >\n {formatPastTimeShort(lang, date * 1000)}\n </Link>\n )}\n </div>\n </div>\n\n {showSeekline && renderSeekline(playProgress, bufferedProgress, seekHandlers)}\n {!showSeekline && (\n <p className=\"duration\" dir=\"auto\">\n {playProgress > 0 ? `${formatMediaDuration(duration * playProgress)} / ` : undefined}\n {getSecondLine()}\n </p>\n )}\n </div>\n </>\n );\n }\n\n return (\n <div className={fullClassName} dir={lang.isRtl ? 'rtl' : undefined}>\n {isSelectable && (\n <div className=\"message-select-control\">\n {isSelected && <i className=\"icon-select\" />}\n </div>\n )}\n <Button\n round\n ripple={!IS_SINGLE_COLUMN_LAYOUT}\n size={target ? 'smaller' : 'tiny'}\n className={buttonClassNames.join(' ')}\n ariaLabel={isPlaying ? 'Pause audio' : 'Play audio'}\n onClick={handleButtonClick}\n isRtl={lang.isRtl}\n >\n <i className=\"icon-play\" />\n <i className=\"icon-pause\" />\n </Button>\n {shouldRenderSpinner && (\n <div className={buildClassName('media-loading', spinnerClassNames, isLoadingForPlaying && 'interactive')}>\n <ProgressSpinner\n progress={transferProgress}\n transparent\n size={target ? 'm' : 's'}\n onClick={isLoadingForPlaying ? handleButtonClick : undefined}\n noCross={!isLoadingForPlaying}\n />\n </div>\n )}\n {audio && (\n <Button\n round\n size=\"tiny\"\n className=\"download-button\"\n ariaLabel={isDownloadStarted ? 'Cancel download' : 'Download'}\n onClick={handleDownloadClick}\n >\n <i className={isDownloadStarted ? 'icon-close' : 'icon-arrow-down'} />\n </Button>\n )}\n {target === 'searchResult' && renderSearchResult()}\n {target !== 'searchResult' && audio && renderAudio(\n lang, audio, isPlaying, playProgress, bufferedProgress, seekHandlers, date,\n onDateClick ? handleDateClick : undefined,\n )}\n {target !== 'searchResult' && voice && renderVoice(voice, renderedWaveform, isMediaUnread)}\n </div>\n );\n};\n\nfunction renderAudio(\n lang: LangFn,\n audio: ApiAudio,\n isPlaying: boolean,\n playProgress: number,\n bufferedProgress: number,\n seekHandlers: ISeekMethods,\n date?: number,\n handleDateClick?: NoneToVoidFunction,\n) {\n const {\n title, performer, duration, fileName,\n } = audio;\n const showSeekline = isPlaying || (playProgress > 0 && playProgress < 1);\n\n return (\n <div className=\"content\">\n <p className=\"title\" dir=\"auto\">{renderText(title || fileName)}</p>\n {showSeekline && renderSeekline(playProgress, bufferedProgress, seekHandlers)}\n {!showSeekline && (\n <div className=\"meta\" dir=\"auto\">\n <span className=\"performer\">{renderText(performer || 'Unknown')}</span>\n {date && (\n <>\n {' '}\n •\n {' '}\n <Link className=\"date\" onClick={handleDateClick}>{formatMediaDateTime(lang, date * 1000)}</Link>\n </>\n )}\n </div>\n )}\n <p className=\"duration\" dir=\"auto\">\n {playProgress > 0 ? `${formatMediaDuration(duration * playProgress)} / ` : undefined}\n {formatMediaDuration(duration)}\n </p>\n </div>\n );\n}\n\nfunction renderVoice(voice: ApiVoice, renderedWaveform: any, isMediaUnread?: boolean) {\n return (\n <div className=\"content\">\n {renderedWaveform}\n <p className=\"voice-duration\" dir=\"auto\">\n {formatMediaDuration(voice.duration)}\n {isMediaUnread && <span>•</span>}\n </p>\n </div>\n );\n}\n\nfunction renderWaveform(\n voice: ApiVoice,\n playProgress = 0,\n isOwn = false,\n { handleStartSeek, handleSeek, handleStopSeek }: ISeekMethods,\n theme: ISettings['theme'],\n) {\n const { waveform, duration } = voice;\n\n if (!waveform) {\n return undefined;\n }\n\n const fillColor = theme === 'dark' ? '#494B75' : '#CBCBCB';\n const fillOwnColor = theme === 'dark' ? '#C0BBED' : '#B0DEA6';\n const progressFillColor = theme === 'dark' ? '#868DF5' : '#54a3e6';\n const progressFillOwnColor = theme === 'dark' ? '#FFFFFF' : '#53ad53';\n const durationFactor = Math.min(duration / AVG_VOICE_DURATION, 1);\n const spikesCount = Math.round(MIN_SPIKES + (MAX_SPIKES - MIN_SPIKES) * durationFactor);\n const decodedWaveform = decodeWaveform(new Uint8Array(waveform));\n const { data: spikes, peak } = interpolateArray(decodedWaveform, spikesCount);\n const { src, width, height } = renderWaveformToDataUri(spikes, playProgress, {\n peak,\n fillStyle: isOwn ? fillOwnColor : fillColor,\n progressFillStyle: isOwn ? progressFillOwnColor : progressFillColor,\n });\n\n return (\n // eslint-disable-next-line jsx-a11y/no-noninteractive-element-interactions\n <img\n src={src}\n alt=\"\"\n width={width}\n height={height}\n className=\"waveform\"\n draggable={false}\n onMouseDown={handleStartSeek}\n onMouseMove={handleSeek}\n onMouseUp={handleStopSeek}\n />\n );\n}\n\nfunction renderSeekline(\n playProgress: number,\n bufferedProgress: number,\n { handleStartSeek, handleSeek, handleStopSeek }: ISeekMethods,\n) {\n return (\n <div\n className=\"seekline no-selection\"\n onMouseDown={handleStartSeek}\n onMouseMove={handleSeek}\n onMouseUp={handleStopSeek}\n >\n <span className=\"seekline-buffered-progress\">\n <i\n // @ts-ignore\n style={`transform: translateX(${bufferedProgress * 100}%)`}\n />\n </span>\n <span className=\"seekline-play-progress\">\n <i\n // @ts-ignore\n style={`transform: translateX(${playProgress * 100}%)`}\n />\n </span>\n <span className=\"seekline-thumb\">\n <i\n // @ts-ignore\n style={`transform: translateX(${playProgress * 100}%)`}\n />\n </span>\n </div>\n );\n}\n\nexport default memo(Audio);\n","/* eslint-disable no-bitwise */\n\n// eslint-disable-next-line max-len\n// Ref: https://github.com/telegramdesktop/tdesktop/blob/0743e71ab6b928d2ee5bae1aed991849b1e2b291/Telegram/SourceFiles/data/data_document.cpp#L1018\nexport function decodeWaveform(encoded5bit: Uint8Array) {\n const bitsCount = encoded5bit.length * 8;\n const valuesCount = Math.floor(bitsCount / 5);\n if (!valuesCount) {\n return [];\n }\n\n // Read each 5 bit of encoded5bit as 0-31 unsigned char.\n // We count the index of the byte in which the desired 5-bit sequence starts.\n // And then we read a uint16 starting from that byte to guarantee to get all of those 5 bits.\n //\n // BUT! if it is the last byte we have, we're not allowed to read a uint16 starting with it.\n // Because it will be an overflow (we'll access one byte after the available memory).\n // We see, that only the last 5 bits could start in the last available byte and be problematic.\n // So we read in a general way all the entries in a general way except the last one.\n const result = Array(valuesCount);\n const bitsData = encoded5bit;\n for (let i = 0, l = valuesCount - 1; i !== l; ++i) {\n const byteIndex = Math.floor((i * 5) / 8);\n const bitShift = Math.floor((i * 5) % 8);\n const value = bitsData[byteIndex] + (bitsData[byteIndex + 1] << 8);\n result[i] = ((value >> bitShift) & 0x1F);\n }\n const lastByteIndex = Math.floor(((valuesCount - 1) * 5) / 8);\n const lastBitShift = Math.floor(((valuesCount - 1) * 5) % 8);\n const lastValue = bitsData[lastByteIndex] + (bitsData[lastByteIndex + 1] << 8);\n result[valuesCount - 1] = (lastValue >> lastBitShift) & 0x1F;\n\n return result;\n}\n\nexport function interpolateArray(data: number[], fitCount: number) {\n let peak = 0;\n const newData = new Array(fitCount);\n const springFactor = data.length / fitCount;\n const leftFiller = data[0];\n const rightFiller = data[data.length - 1];\n for (let i = 0; i < fitCount; i++) {\n const idx = Math.floor(i * springFactor);\n const val = ((data[idx - 1] ?? leftFiller) + (data[idx] ?? leftFiller) + (data[idx + 1] ?? rightFiller)) / 3;\n newData[i] = val;\n if (peak < val) {\n peak = val;\n }\n }\n return { data: newData, peak };\n}\n","type IWaveformProps = {\n peak: number;\n fillStyle: string;\n progressFillStyle: string;\n};\n\nconst SPIKE_WIDTH = 2;\nconst SPIKE_STEP = 4;\nconst SPIKE_RADIUS = 1;\nconst HEIGHT = 23;\n\nexport function renderWaveformToDataUri(\n spikes: number[],\n progress: number,\n {\n peak, fillStyle, progressFillStyle,\n }: IWaveformProps,\n) {\n const width = spikes.length * SPIKE_STEP;\n const height = HEIGHT;\n\n const canvas = document.createElement('canvas');\n canvas.width = width * 2;\n canvas.height = height * 2;\n canvas.style.width = `${width}px`;\n canvas.style.height = `${height}px`;\n\n const ctx = canvas.getContext('2d')!;\n ctx.scale(2, 2);\n\n spikes.forEach((item, i) => {\n ctx.globalAlpha = (i / spikes.length >= progress) ? 0.5 : 1;\n ctx.fillStyle = progress > i / spikes.length ? progressFillStyle : fillStyle;\n const spikeHeight = Math.max(2, HEIGHT * (item / Math.max(1, peak)));\n roundedRectangle(ctx, i * SPIKE_STEP, height, SPIKE_WIDTH, spikeHeight, SPIKE_RADIUS);\n ctx.fill();\n });\n\n return {\n src: canvas.toDataURL(),\n width,\n height,\n };\n}\n\nfunction roundedRectangle(\n ctx: CanvasRenderingContext2D, x: number, y: number, width: number, height: number, radius: number,\n) {\n if (width < 2 * radius) {\n radius = width / 2;\n }\n if (height < 2 * radius) {\n radius = height / 2;\n }\n\n ctx.beginPath();\n ctx.moveTo(x + radius, y);\n ctx.arcTo(x + width, y, x + width, y - height, radius);\n ctx.arcTo(x + width, y - height, x, y - height, radius);\n ctx.arcTo(x, y - height, x, y, radius);\n ctx.arcTo(x, y, x + width, y, radius);\n ctx.closePath();\n}\n","import React, {\n FC, useLayoutEffect, useRef, memo,\n} from '../../lib/teact/teact';\n\nimport buildClassName from '../../util/buildClassName';\nimport renderText from '../common/helpers/renderText';\n\nimport './Tab.scss';\n\ntype OwnProps = {\n className?: string;\n title: string;\n active?: boolean;\n badgeCount?: number;\n isBadgeActive?: boolean;\n previousActiveTab?: number;\n onClick: (arg: number) => void;\n clickArg: number;\n};\n\nconst Tab: FC<OwnProps> = ({\n className,\n title,\n active,\n badgeCount,\n isBadgeActive,\n previousActiveTab,\n onClick,\n clickArg,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const tabRef = useRef<HTMLButtonElement>(null);\n\n useLayoutEffect(() => {\n if (!active || previousActiveTab === undefined) {\n return;\n }\n\n const tab = tabRef.current!;\n const indicator = tab.querySelector('i')!;\n const prevTab = tab.parentElement!.children[previousActiveTab];\n if (!prevTab) {\n return;\n }\n const currentIndicator = prevTab.querySelector('i')!;\n\n currentIndicator.classList.remove('animate');\n indicator.classList.remove('animate');\n\n // We move and resize our indicator so it repeats the position and size of the previous one.\n const shiftLeft = currentIndicator.parentElement!.offsetLeft - indicator.parentElement!.offsetLeft;\n const scaleFactor = currentIndicator.clientWidth / indicator.clientWidth;\n indicator.style.transform = `translate3d(${shiftLeft}px, 0, 0) scale3d(${scaleFactor}, 1, 1)`;\n\n // 3 AFs needed here to synchronize animations with Transition component\n requestAnimationFrame(() => {\n requestAnimationFrame(() => {\n requestAnimationFrame(() => {\n // Now we remove the transform to let it animate to its own position and size.\n indicator.classList.add('animate');\n indicator.style.transform = 'none';\n });\n });\n });\n }, [active, previousActiveTab]);\n\n return (\n <button\n type=\"button\"\n className={buildClassName('Tab', className, active && 'active')}\n onClick={() => onClick(clickArg)}\n ref={tabRef}\n >\n <span>\n {renderText(title)}\n {!!badgeCount && (\n <span className={buildClassName('badge', isBadgeActive && 'active')}>{badgeCount}</span>\n )}\n <i />\n </span>\n </button>\n );\n};\n\nexport default memo(Tab);\n","import React, {\n FC, memo, useRef, useEffect,\n} from '../../lib/teact/teact';\n\nimport fastSmoothScrollHorizontal from '../../util/fastSmoothScrollHorizontal';\nimport usePrevious from '../../hooks/usePrevious';\nimport useHorizontalScroll from '../../hooks/useHorizontalScroll';\nimport useLang from '../../hooks/useLang';\n\nimport Tab from './Tab';\n\nimport './TabList.scss';\n\nexport type TabWithProperties = {\n title: string;\n badgeCount?: number;\n isBadgeActive?: boolean;\n};\n\ntype OwnProps = {\n tabs: readonly TabWithProperties[];\n activeTab: number;\n big?: boolean;\n onSwitchTab: (index: number) => void;\n};\n\nconst TAB_SCROLL_THRESHOLD_PX = 16;\n\nconst TabList: FC<OwnProps> = ({\n tabs, activeTab, big, onSwitchTab,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const containerRef = useRef<HTMLDivElement>(null);\n const previousActiveTab = usePrevious(activeTab);\n\n useHorizontalScroll(containerRef.current);\n\n // Scroll container to place active tab in the center\n useEffect(() => {\n const container = containerRef.current!;\n if (container.scrollWidth <= container.offsetWidth) {\n return;\n }\n\n const activeTabElement = container.querySelector('.Tab.active') as HTMLElement | null;\n if (activeTabElement) {\n const newLeft = activeTabElement.offsetLeft - (container.offsetWidth / 2) + (activeTabElement.offsetWidth / 2);\n\n // Prevent scrolling by only a couple of pixels, which doesn't look smooth\n if (Math.abs(newLeft - container.scrollLeft) < TAB_SCROLL_THRESHOLD_PX) {\n return;\n }\n\n fastSmoothScrollHorizontal(container, newLeft);\n }\n }, [activeTab]);\n\n const lang = useLang();\n\n return (\n <div\n className={`TabList no-selection no-scrollbar ${big ? 'big' : ''}`}\n ref={containerRef}\n dir={lang.isRtl ? 'rtl' : undefined}\n >\n {tabs.map((tab, i) => (\n <Tab\n key={tab.title}\n title={lang(tab.title)}\n active={i === activeTab}\n badgeCount={tab.badgeCount}\n isBadgeActive={tab.isBadgeActive}\n previousActiveTab={previousActiveTab}\n onClick={onSwitchTab}\n clickArg={i}\n />\n ))}\n </div>\n );\n};\n\nexport default memo(TabList);\n","import { ApiError } from '../api/types';\n\nconst READABLE_ERROR_MESSAGES: Record<string, string> = {\n CHAT_RESTRICTED: 'You can\\'t send messages in this chat, you were restricted',\n CHAT_WRITE_FORBIDDEN: 'You can\\'t write in this chat',\n CHAT_SEND_POLL_FORBIDDEN: 'You can\\'t create polls in this chat',\n CHAT_SEND_STICKERS_FORBIDDEN: 'You can\\'t send stickers in this chat',\n CHAT_SEND_GIFS_FORBIDDEN: 'You can\\'t send gifs in this chat',\n CHAT_SEND_MEDIA_FORBIDDEN: 'You can\\'t send media in this chat',\n CHAT_LINK_EXISTS: 'The chat is public, you can\\'t hide the history to new users',\n // eslint-disable-next-line max-len\n SLOWMODE_WAIT_X: 'Slowmode is enabled in this chat: you must wait for the specified number of seconds before sending another message to the chat.',\n USER_BANNED_IN_CHANNEL: 'You\\'re banned from sending messages in supergroups / channels',\n USER_IS_BLOCKED: 'You were blocked by this user',\n YOU_BLOCKED_USER: 'You blocked this user',\n IMAGE_PROCESS_FAILED: 'Failure while processing image',\n MEDIA_EMPTY: 'The provided media object is invalid',\n MEDIA_INVALID: 'Media invalid',\n PHOTO_EXT_INVALID: 'The extension of the photo is invalid',\n PHOTO_INVALID_DIMENSIONS: 'The photo dimensions are invalid',\n PHOTO_SAVE_FILE_INVALID: 'Internal issues, try again later',\n // eslint-disable-next-line max-len\n MESSAGE_DELETE_FORBIDDEN: 'You can\\'t delete one of the messages you tried to delete, most likely because it is a service message.',\n MESSAGE_POLL_CLOSED: 'Poll closed',\n MESSAGE_EDIT_TIME_EXPIRED: 'You can\\'t edit this message anymore.',\n CHAT_ADMIN_REQUIRED: 'You must be an admin in this chat to do this',\n PINNED_DIALOGS_TOO_MUCH: 'Sorry, you can only pin 5 chats to the top',\n DIALOG_FILTERS_TOO_MUCH: 'Sorry, you can\\'t have more than 10 folders',\n CHANNEL_PRIVATE: 'This channel is private',\n MEDIA_CAPTION_TOO_LONG: 'The provided caption is too long',\n ADDRESS_STREET_LINE1_INVALID: 'The address you provided is not valid',\n ADDRESS_STREET_LINE2_INVALID: 'The address you provided is not valid',\n ADDRESS_CITY_INVALID: 'The city you provided is not valid',\n ADDRESS_COUNTRY_INVALID: 'The country you provided is not valid',\n ADDRESS_POSTCODE_INVALID: 'The postcode you provided is not valid',\n ADDRESS_STATE_INVALID: 'The state you provided is not valid',\n REQ_INFO_NAME_INVALID: 'The name you provided is not valid',\n REQ_INFO_PHONE_INVALID: 'The phone you provided is not valid',\n REQ_INFO_EMAIL_INVALID: 'The email you provided is not valid',\n // TODO Bring back after fixing the weird bug\n // CHANNEL_INVALID: 'An error occurred. Please try again later',\n LINK_NOT_MODIFIED: 'This discussion is already linked to the channel',\n\n // Non-API errors\n SERVICE_WORKER_DISABLED: 'Service Worker is disabled. Please reload the page without holding <Shift> key.',\n // eslint-disable-next-line max-len\n CAPTION_TOO_LONG_PLEASE_REMOVE_CHARACTERS: 'The provided caption is too long. Please remove {EXTRA_CHARS_COUNT} character{PLURAL_S}.',\n // eslint-disable-next-line max-len\n FRESH_RESET_AUTHORISATION_FORBIDDEN: 'You can’t logout other sessions if less than 24 hours have passed since you logged on the current session',\n\n BOTS_TOO_MUCH: 'There are too many bots in this chat/channel',\n BOT_GROUPS_BLOCKED: 'This bot can\\'t be added to groups',\n USERS_TOO_MUCH: 'The maximum number of users has been exceeded',\n USER_CHANNELS_TOO_MUCH: 'One of the users you tried to add is already in too many channels/supergroups',\n USER_KICKED: 'This user was kicked from this supergroup/channel',\n USER_NOT_MUTUAL_CONTACT: 'The provided user is not a mutual contact',\n USER_PRIVACY_RESTRICTED: 'The user\\'s privacy settings do not allow you to do this',\n INVITE_HASH_EMPTY: 'The invite hash is empty',\n INVITE_HASH_EXPIRED: 'The invite link has expired',\n INVITE_HASH_INVALID: 'The invite hash is invalid',\n CHANNELS_TOO_MUCH: 'You have joined too many channels/supergroups',\n USER_ALREADY_PARTICIPANT: 'You already in the group',\n SCHEDULE_DATE_INVALID: 'Invalid schedule date provided',\n WALLPAPER_DIMENSIONS_INVALID: 'The wallpaper dimensions are invalid, please select another file',\n};\n\nexport default function getReadableErrorText(error: ApiError) {\n const { message, isSlowMode, textParams } = error;\n // Currently, Telegram API doesn't return `SLOWMODE_WAIT_X` error as described in the docs\n if (isSlowMode) {\n const extraPartIndex = message.indexOf(' (caused by');\n return extraPartIndex > 0 ? message.substring(0, extraPartIndex) : message;\n }\n let errorMessage = READABLE_ERROR_MESSAGES[message];\n if (errorMessage && textParams) {\n errorMessage = Object.keys(textParams).reduce((acc, current) => {\n return acc.replace(current, textParams[current]);\n }, errorMessage as string);\n }\n return errorMessage;\n}\n","// Sometimes event is fired earlier than animation completes\nconst ANIMATION_END_DELAY = 50;\n\nexport function waitForTransitionEnd(node: Node, handler: NoneToVoidFunction, propertyName?: string) {\n waitForEndEvent('transitionend', node, handler, propertyName);\n}\n\nexport function waitForAnimationEnd(node: Node, handler: NoneToVoidFunction, animationName?: string) {\n waitForEndEvent('animationend', node, handler, animationName);\n}\n\nfunction waitForEndEvent(\n eventType: 'transitionend' | 'animationend',\n node: Node,\n handler: NoneToVoidFunction,\n detailedName?: string,\n) {\n let isHandled = false;\n\n node.addEventListener(eventType, function handleAnimationEnd(e: TransitionEvent | AnimationEvent) {\n if (isHandled || e.target !== e.currentTarget) {\n return;\n }\n\n if (detailedName && (\n (e instanceof TransitionEvent && e.propertyName === detailedName)\n || (e instanceof AnimationEvent && e.animationName === detailedName)\n )) {\n return;\n }\n\n isHandled = true;\n\n node.removeEventListener(eventType, handleAnimationEnd as EventListener);\n\n setTimeout(() => {\n handler();\n }, ANIMATION_END_DELAY);\n } as EventListener);\n}\n","import { useMemo } from '../lib/teact/teact';\nimport { getDispatch } from '../lib/teact/teactn';\n\nimport { ApiChat, ApiUser } from '../api/types';\n\nimport {\n isChatArchived, getCanDeleteChat, isChatPrivate, isChatChannel,\n} from '../modules/helpers';\nimport useLang from './useLang';\n\nexport default ({\n chat,\n privateChatUser,\n handleDelete,\n folderId,\n isPinned,\n isMuted,\n}: {\n chat: ApiChat | undefined;\n privateChatUser: ApiUser | undefined;\n handleDelete: () => void;\n folderId?: number;\n isPinned?: boolean;\n isMuted?: boolean;\n}) => {\n const lang = useLang();\n\n const {\n toggleChatPinned,\n updateChatMutedState,\n toggleChatArchived,\n toggleChatUnread,\n } = getDispatch();\n\n return useMemo(() => {\n if (!chat) {\n return undefined;\n }\n\n const isChatWithSelf = privateChatUser && privateChatUser.isSelf;\n\n const actionUnreadMark = chat.unreadCount || chat.hasUnreadMark\n ? { title: lang('MarkAsRead'), icon: 'readchats', handler: () => toggleChatUnread({ id: chat.id }) }\n : { title: lang('MarkAsUnread'), icon: 'unread', handler: () => toggleChatUnread({ id: chat.id }) };\n\n const actionPin = isPinned\n ? {\n title: lang('UnpinFromTop'),\n icon: 'unpin',\n handler: () => toggleChatPinned({ id: chat.id, folderId }),\n }\n : { title: lang('PinToTop'), icon: 'pin', handler: () => toggleChatPinned({ id: chat.id, folderId }) };\n\n const actionMute = isMuted\n ? {\n title: lang('ChatList.Unmute'),\n icon: 'unmute',\n handler: () => updateChatMutedState({ chatId: chat.id, isMuted: false }),\n }\n : {\n title: lang('ChatList.Mute'),\n icon: 'mute',\n handler: () => updateChatMutedState({ chatId: chat.id, isMuted: true }),\n };\n\n const actionArchive = isChatArchived(chat)\n ? { title: lang('Unarchive'), icon: 'unarchive', handler: () => toggleChatArchived({ id: chat.id }) }\n : { title: lang('Archive'), icon: 'archive', handler: () => toggleChatArchived({ id: chat.id }) };\n\n const actionDelete = {\n title: isChatPrivate(chat.id)\n ? lang('Delete')\n : lang(getCanDeleteChat(chat)\n ? 'DeleteChat'\n : (isChatChannel(chat) ? 'LeaveChannel' : 'Group.LeaveGroup')),\n icon: 'delete',\n destructive: true,\n handler: handleDelete,\n };\n\n return [\n actionUnreadMark,\n actionPin,\n ...(!isChatWithSelf ? [\n actionMute,\n actionArchive,\n ] : []),\n actionDelete,\n ];\n }, [\n chat, privateChatUser, lang, isPinned, handleDelete, toggleChatUnread, toggleChatPinned, folderId,\n updateChatMutedState, toggleChatArchived, isMuted,\n ]);\n};\n","import React, { FC, memo } from '../../lib/teact/teact';\n\nimport { ApiMessage, ApiMessageOutgoingStatus } from '../../api/types';\n\nimport { formatPastTimeShort } from '../../util/dateFormat';\nimport useLang from '../../hooks/useLang';\n\nimport MessageOutgoingStatus from './MessageOutgoingStatus';\n\nimport './LastMessageMeta.scss';\n\ntype OwnProps = {\n message: ApiMessage;\n outgoingStatus?: ApiMessageOutgoingStatus;\n};\n\nconst LastMessageMeta: FC<OwnProps> = ({ message, outgoingStatus }) => {\n const lang = useLang();\n return (\n <div className=\"LastMessageMeta\">\n {outgoingStatus && (\n <MessageOutgoingStatus status={outgoingStatus} />\n )}\n <span className=\"time\">{formatPastTimeShort(lang, message.date * 1000)}</span>\n </div>\n );\n};\n\nexport default memo(LastMessageMeta);\n","import React, { FC, memo } from '../../lib/teact/teact';\n\nimport { ApiMessageOutgoingStatus } from '../../api/types';\n\nimport Transition from '../ui/Transition';\n\nimport './MessageOutgoingStatus.scss';\n\ntype OwnProps = {\n status: ApiMessageOutgoingStatus;\n};\n\nenum Keys {\n failed, pending, succeeded, read,\n}\n\nconst MessageOutgoingStatus: FC<OwnProps> = ({ status }) => {\n return (\n <div className=\"MessageOutgoingStatus\">\n <Transition name=\"reveal\" activeKey={Keys[status]}>\n {() => (\n <i className={`icon-message-${status}`} />\n )}\n </Transition>\n </div>\n );\n};\n\nexport default memo(MessageOutgoingStatus);\n","import { useState, useEffect } from '../lib/teact/teact';\nimport { IAnchorPosition } from '../types';\n\nconst MENU_POSITION_VISUAL_COMFORT_SPACE_PX = 16;\n\nexport default (\n anchor: IAnchorPosition | undefined,\n getTriggerElement: () => HTMLElement | null,\n getRootElement: () => HTMLElement | null,\n getMenuElement: () => HTMLElement | null,\n extraPaddingX = 0,\n extraTopPadding = 0,\n) => {\n const [positionX, setPositionX] = useState<'right' | 'left'>('right');\n const [positionY, setPositionY] = useState<'top' | 'bottom'>('bottom');\n const [style, setStyle] = useState('');\n\n useEffect(() => {\n const triggerEl = getTriggerElement();\n if (!anchor || !triggerEl) {\n return;\n }\n\n let { x, y } = anchor;\n const emptyRect = {\n width: 0, left: 0, height: 0, top: 0,\n };\n\n const menuEl = getMenuElement();\n const rootEl = getRootElement();\n\n const triggerRect = triggerEl.getBoundingClientRect();\n const menuRect = menuEl ? { width: menuEl.offsetWidth, height: menuEl.offsetHeight } : emptyRect;\n const rootRect = rootEl ? rootEl.getBoundingClientRect() : emptyRect;\n\n let horizontalPostition: 'left' | 'right';\n if (x + menuRect.width + extraPaddingX < rootRect.width + rootRect.left) {\n x += 3;\n horizontalPostition = 'left';\n } else if (x - menuRect.width > 0) {\n horizontalPostition = 'right';\n x -= 3;\n } else {\n horizontalPostition = 'left';\n x = 16;\n }\n setPositionX(horizontalPostition);\n\n if (y + menuRect.height < rootRect.height + rootRect.top) {\n setPositionY('top');\n } else {\n setPositionY('bottom');\n\n if (y - menuRect.height < rootRect.top + extraTopPadding) {\n y = rootRect.top + extraTopPadding + menuRect.height;\n }\n }\n\n const left = horizontalPostition === 'left'\n ? Math.min(x - triggerRect.left, rootRect.width - menuRect.width - MENU_POSITION_VISUAL_COMFORT_SPACE_PX)\n : Math.max((x - triggerRect.left), menuRect.width + MENU_POSITION_VISUAL_COMFORT_SPACE_PX);\n\n setStyle(`left: ${left}px; top: ${y - triggerRect.top}px;`);\n }, [\n anchor, extraPaddingX, extraTopPadding,\n getMenuElement, getRootElement, getTriggerElement,\n ]);\n\n return {\n positionX,\n positionY,\n style,\n };\n};\n","import { useRef } from '../lib/teact/teact';\n\nimport usePrevious from './usePrevious';\nimport useForceUpdate from './useForceUpdate';\nimport useOnChange from './useOnChange';\n\nexport default function usePrevDuringAnimation(current: any, duration?: number) {\n const prev = usePrevious(current, true);\n const timeoutRef = useRef<number>();\n const forceUpdate = useForceUpdate();\n // eslint-disable-next-line no-null/no-null\n const isCurrentPresent = current !== undefined && current !== null;\n // eslint-disable-next-line no-null/no-null\n const isPrevPresent = prev !== undefined && prev !== null;\n\n if (isCurrentPresent && timeoutRef.current) {\n clearTimeout(timeoutRef.current);\n timeoutRef.current = undefined;\n }\n\n useOnChange(() => {\n // When `current` becomes empty\n if (duration && !isCurrentPresent && isPrevPresent && !timeoutRef.current) {\n timeoutRef.current = window.setTimeout(() => {\n timeoutRef.current = undefined;\n forceUpdate();\n }, duration);\n }\n }, [current]);\n\n return !timeoutRef.current || !duration || isCurrentPresent ? current : prev;\n}\n","import {\n useCallback, useEffect, useRef, useState,\n} from '../lib/teact/teact';\n\nimport { register } from '../util/audioPlayer';\nimport useEffectWithPrevDeps from './useEffectWithPrevDeps';\nimport { isSafariPatchInProgress } from '../util/patchSafariProgressiveAudio';\nimport useOnChange from './useOnChange';\n\ntype Handler = (e: Event) => void;\n\nexport default (\n trackId: string,\n originalDuration: number, // Sometimes incorrect for voice messages\n src?: string,\n handlers?: Record<string, Handler>,\n onInit?: (element: HTMLAudioElement) => void,\n shouldPlay = false,\n onForcePlay?: NoneToVoidFunction,\n noPlaylist = false,\n) => {\n // eslint-disable-next-line no-null/no-null\n const controllerRef = useRef<ReturnType<typeof register>>(null);\n\n const [isPlaying, setIsPlaying] = useState(false);\n let isPlayingSync = isPlaying;\n\n const [playProgress, setPlayProgress] = useState<number>(0);\n\n useOnChange(() => {\n controllerRef.current = register(trackId, (eventName, e) => {\n switch (eventName) {\n case 'onPlay':\n setIsPlaying(true);\n break;\n case 'onPause':\n setIsPlaying(false);\n break;\n case 'onTimeUpdate': {\n const { proxy } = controllerRef.current!;\n const duration = proxy.duration && Number.isFinite(proxy.duration) ? proxy.duration : originalDuration;\n setPlayProgress(proxy.currentTime / duration);\n break;\n }\n }\n\n if (handlers && handlers[eventName]) {\n handlers[eventName](e);\n }\n }, onForcePlay);\n\n const { proxy } = controllerRef.current!;\n\n if (!isPlaying && !proxy.paused) {\n setIsPlaying(true);\n isPlayingSync = true;\n }\n\n if (onInit) {\n onInit(proxy);\n }\n }, [trackId]);\n\n const {\n play, pause, setCurrentTime, proxy, destroy,\n } = controllerRef.current!;\n const duration = proxy.duration && Number.isFinite(proxy.duration) ? proxy.duration : originalDuration;\n\n // RAF progress\n useEffect(() => {\n if (duration && !isSafariPatchInProgress(proxy)) {\n setPlayProgress(proxy.currentTime / duration);\n }\n }, [duration, playProgress, proxy]);\n\n // Cleanup\n useEffect(() => () => {\n destroy(noPlaylist);\n }, [destroy, noPlaylist]);\n\n // Autoplay once `src` is present\n useEffectWithPrevDeps(([prevShouldPlay, prevSrc]) => {\n if (prevShouldPlay === shouldPlay && src === prevSrc) {\n return;\n }\n\n // When paused by another player\n if (proxy.src && proxy.paused) {\n return;\n }\n\n if (shouldPlay && src && !isPlaying) {\n play(src);\n }\n }, [shouldPlay, src, isPlaying, play, proxy.src, proxy.paused]);\n\n const playPause = useCallback(() => {\n if (isPlaying) {\n pause();\n } else if (src) {\n play(src);\n }\n }, [src, pause, play, isPlaying]);\n\n return {\n isPlaying: isPlayingSync,\n playProgress,\n playPause,\n setCurrentTime,\n audioProxy: proxy,\n duration,\n };\n};\n","import { IS_SAFARI } from './environment';\nimport safePlay from './safePlay';\nimport { patchSafariProgressiveAudio, isSafariPatchInProgress } from './patchSafariProgressiveAudio';\nimport { getDispatch } from '../lib/teact/teactn';\nimport { parseMessageKey } from '../modules/helpers';\n\ntype Handler = (eventName: string, e: Event) => void;\n\ninterface Track {\n audio: HTMLAudioElement;\n proxy: HTMLAudioElement;\n handlers: Handler[];\n onForcePlay?: NoneToVoidFunction;\n}\n\nconst tracks = new Map<string, Track>();\nlet queue: string[] = [];\n\nlet currentTrackId: string | undefined;\n\nfunction createAudio(trackId: string, onForcePlay?: NoneToVoidFunction) {\n const audio = new Audio();\n\n function handleEvent(eventName: string) {\n return (e: Event) => {\n if (!tracks.has(trackId)) {\n return;\n }\n\n if (isSafariPatchInProgress(audio)) {\n return;\n }\n\n tracks.get(trackId)!.handlers.forEach((handler) => {\n handler(eventName, e);\n });\n };\n }\n\n audio.addEventListener('timeupdate', handleEvent('onTimeUpdate'));\n audio.addEventListener('play', handleEvent('onPlay'));\n audio.addEventListener('pause', handleEvent('onPause'));\n audio.addEventListener('loadstart', handleEvent('onLoadStart'));\n audio.addEventListener('loadeddata', handleEvent('onLoadedData'));\n audio.addEventListener('playing', handleEvent('onPlaying'));\n audio.addEventListener('ended', () => {\n if (isSafariPatchInProgress(audio)) {\n return;\n }\n\n const nextTrackId = queue[queue.indexOf(trackId) + 1];\n if (!nextTrackId) {\n return;\n }\n\n if (!tracks.has(nextTrackId)) {\n // A bit hacky way to continue playlist when switching chat\n getDispatch().openAudioPlayer(parseMessageKey(nextTrackId));\n\n return;\n }\n\n const nextTrack = tracks.get(nextTrackId)!;\n\n if (nextTrack.onForcePlay) {\n nextTrack.onForcePlay();\n }\n\n currentTrackId = nextTrackId;\n\n if (nextTrack.audio.src) {\n safePlay(nextTrack.audio);\n }\n });\n\n return {\n audio,\n proxy: new Proxy(audio, {\n get: (origin, key: keyof HTMLAudioElement) => origin[key],\n }),\n handlers: [],\n onForcePlay,\n };\n}\n\nexport function stopCurrentAudio() {\n const currentTrack = currentTrackId && tracks.get(currentTrackId);\n if (currentTrack) {\n currentTrack.audio.pause();\n }\n}\n\nexport function register(trackId: string, handler: Handler, onForcePlay?: NoneToVoidFunction) {\n if (!tracks.has(trackId)) {\n tracks.set(trackId, createAudio(trackId, onForcePlay));\n\n if (!queue.includes(trackId)) {\n queue.push(trackId);\n }\n }\n\n const { audio, proxy, handlers } = tracks.get(trackId)!;\n\n handlers.push(handler);\n\n return {\n play(src: string) {\n if (currentTrackId && currentTrackId !== trackId) {\n tracks.get(currentTrackId)!.audio.pause();\n }\n\n currentTrackId = trackId;\n\n if (!audio.src) {\n audio.src = src;\n audio.preload = 'auto';\n\n if (src.includes('/progressive/') && IS_SAFARI) {\n patchSafariProgressiveAudio(audio);\n }\n }\n\n safePlay(audio);\n },\n\n pause() {\n if (currentTrackId === trackId) {\n audio.pause();\n }\n },\n\n setCurrentTime(time: number) {\n if (currentTrackId === trackId) {\n audio.currentTime = time;\n }\n },\n\n proxy,\n\n destroy(shouldRemoveFromQueue = false) {\n const track = tracks.get(trackId);\n if (!track) {\n return;\n }\n\n track.handlers = track.handlers.filter((h) => h !== handler);\n\n if (!track.handlers.length) {\n track.audio.pause();\n tracks.delete(trackId);\n\n if (shouldRemoveFromQueue) {\n queue = queue.filter((id) => id !== trackId);\n }\n\n if (trackId === currentTrackId) {\n currentTrackId = undefined;\n }\n }\n },\n };\n}\n","export default function download(url: string, filename: string) {\n const link = document.createElement('a');\n link.href = url;\n link.download = filename;\n link.click();\n}\n","import { RefObject } from 'react';\nimport React, { FC, memo, useRef } from '../../lib/teact/teact';\n\nimport useShowTransition from '../../hooks/useShowTransition';\nimport useTransitionForMedia from '../../hooks/useTransitionForMedia';\nimport buildClassName from '../../util/buildClassName';\nimport { formatMediaDateTime, formatPastTimeShort } from '../../util/dateFormat';\nimport { getColorFromExtension, getFileSizeString } from './helpers/documentInfo';\nimport { getDocumentThumbnailDimensions } from './helpers/mediaDimensions';\nimport renderText from './helpers/renderText';\nimport useLang from '../../hooks/useLang';\n\nimport ProgressSpinner from '../ui/ProgressSpinner';\nimport Link from '../ui/Link';\n\nimport './File.scss';\n\ntype OwnProps = {\n ref?: RefObject<HTMLDivElement>;\n name: string;\n extension?: string;\n size: number;\n timestamp?: number;\n sender?: string;\n thumbnailDataUri?: string;\n previewData?: string;\n className?: string;\n smaller?: boolean;\n isTransferring?: boolean;\n isUploading?: boolean;\n isSelectable?: boolean;\n isSelected?: boolean;\n transferProgress?: number;\n actionIcon?: string;\n onClick?: () => void;\n onDateClick?: (e: React.MouseEvent<HTMLAnchorElement>) => void;\n};\n\nconst File: FC<OwnProps> = ({\n ref,\n name,\n size,\n extension = '',\n timestamp,\n sender,\n thumbnailDataUri,\n previewData,\n className,\n smaller,\n isTransferring,\n isUploading,\n isSelectable,\n isSelected,\n transferProgress,\n actionIcon,\n onClick,\n onDateClick,\n}) => {\n const lang = useLang();\n // eslint-disable-next-line no-null/no-null\n let elementRef = useRef<HTMLDivElement>(null);\n if (ref) {\n elementRef = ref;\n }\n\n const {\n shouldRender: shouldSpinnerRender,\n transitionClassNames: spinnerClassNames,\n } = useShowTransition(isTransferring, undefined, true);\n const color = getColorFromExtension(extension);\n const sizeString = getFileSizeString(size);\n\n const {\n shouldRenderThumb, shouldRenderFullMedia, transitionClassNames,\n } = useTransitionForMedia(previewData, 'slow');\n const { width, height } = getDocumentThumbnailDimensions(smaller);\n\n const fullClassName = buildClassName(\n 'File',\n className,\n smaller && 'smaller',\n onClick && !isUploading && 'interactive',\n isSelected && 'file-is-selected',\n );\n\n return (\n <div ref={elementRef} className={fullClassName} dir={lang.isRtl ? 'rtl' : undefined}>\n {isSelectable && (\n <div className=\"message-select-control\">\n {isSelected && <i className=\"icon-select\" />}\n </div>\n )}\n <div className=\"file-icon-container\" onClick={isUploading ? undefined : onClick}>\n {thumbnailDataUri || previewData ? (\n <div className=\"file-preview media-inner\">\n {shouldRenderThumb && (\n <img\n src={thumbnailDataUri}\n width={width}\n height={height}\n className=\"thumbnail\"\n alt=\"\"\n />\n )}\n {shouldRenderFullMedia && (\n <img\n src={previewData}\n className={`full-media ${transitionClassNames}`}\n width={width}\n height={height}\n alt=\"\"\n />\n )}\n </div>\n ) : (\n <div className={`file-icon ${color}`}>\n {extension.length <= 4 && (\n <span className=\"file-ext\" dir=\"auto\">{extension}</span>\n )}\n </div>\n )}\n {shouldSpinnerRender && (\n <div className={buildClassName('file-progress', color, spinnerClassNames)}>\n <ProgressSpinner\n progress={transferProgress}\n size={smaller ? 's' : 'm'}\n onClick={isUploading ? onClick : undefined}\n />\n </div>\n )}\n {onClick && (\n <i\n className={buildClassName(\n 'action-icon',\n actionIcon || 'icon-download',\n shouldSpinnerRender && 'hidden',\n )}\n />\n )}\n </div>\n <div className=\"file-info\">\n <div className=\"file-title\" dir=\"auto\">{renderText(name)}</div>\n <div className=\"file-subtitle\" dir=\"auto\">\n <span>\n {isTransferring && transferProgress ? `${Math.round(transferProgress * 100)}%` : sizeString}\n </span>\n {sender && <span className=\"file-sender\">{renderText(sender)}</span>}\n {!sender && timestamp && (\n <>\n {' '}\n <Link onClick={onDateClick}>{formatMediaDateTime(lang, timestamp * 1000)}</Link>\n </>\n )}\n </div>\n </div>\n {sender && timestamp && (\n <Link onClick={onDateClick}>{formatPastTimeShort(lang, timestamp * 1000)}</Link>\n )}\n </div>\n );\n};\n\nexport default memo(File);\n","import React, { useCallback, useEffect, useState } from '../lib/teact/teact';\n\nimport useMediaWithDownloadProgress from './useMediaWithDownloadProgress';\nimport download from '../util/download';\n\nexport default function useMediaDownload(\n mediaHash?: string,\n fileName?: string,\n) {\n const [isDownloadStarted, setIsDownloadStarted] = useState(false);\n\n const { mediaData, downloadProgress } = useMediaWithDownloadProgress(mediaHash, !isDownloadStarted);\n\n // Download with browser when fully loaded\n useEffect(() => {\n if (isDownloadStarted && mediaData) {\n download(mediaData, fileName!);\n setIsDownloadStarted(false);\n }\n }, [fileName, mediaData, isDownloadStarted]);\n\n // Cancel download on source change\n useEffect(() => {\n setIsDownloadStarted(false);\n }, [mediaHash]);\n\n const handleDownloadClick = useCallback((e: React.SyntheticEvent<HTMLElement>) => {\n e.stopPropagation();\n setIsDownloadStarted((isAllowed) => !isAllowed);\n }, []);\n\n return {\n isDownloadStarted,\n downloadProgress,\n handleDownloadClick,\n };\n}\n","import React, {\n FC,\n memo,\n useCallback,\n useEffect,\n useRef,\n useState,\n} from '../../lib/teact/teact';\n\nimport { ANIMATION_END_DELAY } from '../../config';\nimport useShowTransition from '../../hooks/useShowTransition';\nimport { TextPart } from '../common/helpers/renderMessageText';\nimport buildClassName from '../../util/buildClassName';\nimport captureEscKeyListener from '../../util/captureEscKeyListener';\n\nimport Portal from './Portal';\n\nimport './Notification.scss';\n\ntype OwnProps = {\n containerId?: string;\n message: TextPart[];\n duration?: number;\n onDismiss: () => void;\n};\n\nconst DEFAULT_DURATION = 3000;\nconst ANIMATION_DURATION = 150;\n\nconst Notification: FC<OwnProps> = ({\n message, duration = DEFAULT_DURATION, containerId, onDismiss,\n}) => {\n const [isOpen, setIsOpen] = useState(true);\n // eslint-disable-next-line no-null/no-null\n const timerRef = useRef<number | undefined>(null);\n\n const { transitionClassNames } = useShowTransition(isOpen);\n\n const closeAndDismiss = useCallback(() => {\n setIsOpen(false);\n setTimeout(onDismiss, ANIMATION_DURATION + ANIMATION_END_DELAY);\n }, [onDismiss]);\n\n useEffect(() => (isOpen ? captureEscKeyListener(closeAndDismiss) : undefined), [isOpen, closeAndDismiss]);\n\n useEffect(() => {\n timerRef.current = window.setTimeout(closeAndDismiss, duration);\n\n return () => {\n if (timerRef.current) {\n clearTimeout(timerRef.current);\n timerRef.current = undefined;\n }\n };\n }, [duration, closeAndDismiss]);\n\n const handleMouseEnter = useCallback(() => {\n if (timerRef.current) {\n clearTimeout(timerRef.current);\n timerRef.current = undefined;\n }\n }, []);\n\n const handleMouseLeave = useCallback(() => {\n timerRef.current = window.setTimeout(closeAndDismiss, duration);\n }, [duration, closeAndDismiss]);\n\n return (\n <Portal className=\"Notification-container\" containerId={containerId}>\n <div\n className={buildClassName('Notification', transitionClassNames)}\n onClick={closeAndDismiss}\n onMouseEnter={handleMouseEnter}\n onMouseLeave={handleMouseLeave}\n >\n <div className=\"content\">\n {message}\n </div>\n </div>\n </Portal>\n );\n};\n\nexport default memo(Notification);\n","const MAX_NESTING_PARENTS = 5;\n\nexport function isSelectionInsideInput(selectionRange: Range, inputId: string) {\n const { commonAncestorContainer } = selectionRange;\n let parentNode: HTMLElement | null = commonAncestorContainer as HTMLElement;\n let iterations = 1;\n while (parentNode && parentNode.id !== inputId && iterations < MAX_NESTING_PARENTS) {\n parentNode = parentNode.parentElement;\n iterations++;\n }\n\n return Boolean(parentNode && parentNode.id === inputId);\n}\n","import {\n useCallback, useEffect, useMemo, useState,\n} from '../../../../lib/teact/teact';\n\nimport { EDITABLE_INPUT_ID } from '../../../../config';\nimport { MEMO_EMPTY_ARRAY } from '../../../../util/memo';\nimport {\n EmojiData, EmojiModule, EmojiRawData, uncompressEmoji,\n} from '../../../../util/emoji';\nimport focusEditableElement from '../../../../util/focusEditableElement';\nimport {\n buildCollectionByKey, flatten, mapValues, pickTruthy, unique,\n} from '../../../../util/iteratees';\nimport useFlag from '../../../../hooks/useFlag';\n\nlet emojiDataPromise: Promise<EmojiModule>;\nlet emojiRawData: EmojiRawData;\nlet emojiData: EmojiData;\n\nlet RE_EMOJI_SEARCH: RegExp;\nconst EMOJIS_LIMIT = 36;\nconst FILTER_MIN_LENGTH = 2;\nconst RE_BR = /(<br>|<br\\s?\\/>)/g;\nconst RE_SPACE = / /g;\nconst RE_CLEAN_HTML = /(<div>|<\\/div>)/gi;\n\ntry {\n RE_EMOJI_SEARCH = new RegExp('(^|\\\\s):[-+_:\\\\p{L}\\\\p{N}]*$', 'gui');\n} catch (e) {\n // Support for older versions of firefox\n RE_EMOJI_SEARCH = new RegExp('(^|\\\\s):[-+_:\\\\d\\\\wа-яё]*$', 'gi');\n}\n\nexport default function useEmojiTooltip(\n isAllowed: boolean,\n html: string,\n recentEmojiIds: string[],\n inputId = EDITABLE_INPUT_ID,\n onUpdateHtml: (html: string) => void,\n baseEmojiKeywords?: Record<string, string[]>,\n emojiKeywords?: Record<string, string[]>,\n isDisabled = false,\n) {\n const [isOpen, markIsOpen, unmarkIsOpen] = useFlag();\n\n const [byId, setById] = useState<Record<string, Emoji> | undefined>();\n const [keywords, setKeywords] = useState<string[]>();\n const [byKeyword, setByKeyword] = useState<Record<string, Emoji[]>>({});\n const [names, setNames] = useState<string[]>();\n const [byName, setByName] = useState<Record<string, Emoji[]>>({});\n const [shouldForceInsertEmoji, setShouldForceInsertEmoji] = useState(false);\n\n const [filteredEmojis, setFilteredEmojis] = useState<Emoji[]>(MEMO_EMPTY_ARRAY);\n\n const recentEmojis = useMemo(\n () => {\n if (!byId || !recentEmojiIds.length) {\n return [];\n }\n\n return Object.values(pickTruthy(byId, recentEmojiIds));\n },\n [byId, recentEmojiIds],\n );\n\n // Initialize data on first render.\n useEffect(() => {\n if (isDisabled) return;\n const exec = () => {\n setById(emojiData.emojis);\n };\n\n if (emojiData) {\n exec();\n } else {\n ensureEmojiData()\n .then(exec);\n }\n }, [isDisabled]);\n\n useEffect(() => {\n if (!byId || isDisabled) {\n return;\n }\n\n const emojis = Object.values(byId);\n\n const byNative = buildCollectionByKey(emojis, 'native');\n const baseEmojisByKeyword = baseEmojiKeywords\n ? mapValues(baseEmojiKeywords, (natives) => {\n return Object.values(pickTruthy(byNative, natives));\n })\n : {};\n const emojisByKeyword = emojiKeywords\n ? mapValues(emojiKeywords, (natives) => {\n return Object.values(pickTruthy(byNative, natives));\n })\n : {};\n\n setByKeyword({ ...baseEmojisByKeyword, ...emojisByKeyword });\n setKeywords([...Object.keys(baseEmojisByKeyword), ...Object.keys(emojisByKeyword)]);\n\n const emojisByName = emojis.reduce((result, emoji) => {\n emoji.names.forEach((name) => {\n if (!result[name]) {\n result[name] = [];\n }\n\n result[name].push(emoji);\n });\n\n return result;\n }, {} as Record<string, Emoji[]>);\n setByName(emojisByName);\n setNames(Object.keys(emojisByName));\n }, [isDisabled, baseEmojiKeywords, byId, emojiKeywords]);\n\n useEffect(() => {\n if (!isAllowed || !html || !byId || !keywords || !keywords.length) {\n unmarkIsOpen();\n return;\n }\n\n const code = html.includes(':') && getEmojiCode(html);\n if (!code) {\n setFilteredEmojis(MEMO_EMPTY_ARRAY);\n unmarkIsOpen();\n return;\n }\n\n const forceSend = code.length > 2 && code.endsWith(':');\n const filter = code.substr(1, forceSend ? code.length - 2 : undefined);\n let matched: Emoji[] = [];\n\n setShouldForceInsertEmoji(forceSend);\n\n if (!filter) {\n matched = recentEmojis;\n } else if (filter.length >= FILTER_MIN_LENGTH) {\n const matchedKeywords = keywords.filter((keyword) => keyword.startsWith(filter)).sort();\n matched = matched.concat(flatten(Object.values(pickTruthy(byKeyword, matchedKeywords))));\n\n // Also search by names, which is useful for non-English languages\n const matchedNames = names.filter((name) => name.startsWith(filter));\n matched = matched.concat(flatten(Object.values(pickTruthy(byName, matchedNames))));\n\n matched = unique(matched);\n }\n\n if (matched.length) {\n if (!forceSend) {\n markIsOpen();\n }\n setFilteredEmojis(matched.slice(0, EMOJIS_LIMIT));\n } else {\n unmarkIsOpen();\n }\n }, [\n byId, byKeyword, keywords, byName, names, html, isAllowed, markIsOpen,\n recentEmojis, unmarkIsOpen, setShouldForceInsertEmoji,\n ]);\n\n const insertEmoji = useCallback((textEmoji: string, isForce?: boolean) => {\n const atIndex = html.lastIndexOf(':', isForce ? html.lastIndexOf(':') - 1 : undefined);\n if (atIndex !== -1) {\n onUpdateHtml(`${html.substr(0, atIndex)}${textEmoji}`);\n const messageInput = document.getElementById(inputId)!;\n requestAnimationFrame(() => {\n focusEditableElement(messageInput, true);\n });\n }\n\n unmarkIsOpen();\n }, [html, inputId, onUpdateHtml, unmarkIsOpen]);\n\n useEffect(() => {\n if (isOpen && shouldForceInsertEmoji && filteredEmojis.length) {\n insertEmoji(filteredEmojis[0].native, true);\n }\n }, [filteredEmojis, insertEmoji, isOpen, shouldForceInsertEmoji]);\n\n return {\n isEmojiTooltipOpen: isOpen,\n closeEmojiTooltip: unmarkIsOpen,\n filteredEmojis,\n insertEmoji,\n };\n}\n\nfunction getEmojiCode(html: string) {\n const emojis = html\n .replace(RE_SPACE, ' ')\n .replace(RE_BR, '\\n')\n .replace(/\\n$/i, '')\n .replace(RE_CLEAN_HTML, '')\n .match(RE_EMOJI_SEARCH);\n\n return emojis ? emojis[0].trim() : undefined;\n}\n\nasync function ensureEmojiData() {\n if (!emojiDataPromise) {\n emojiDataPromise = import('emoji-data-ios/emoji-data.json') as unknown as Promise<EmojiModule>;\n emojiRawData = (await emojiDataPromise).default;\n\n emojiData = uncompressEmoji(emojiRawData);\n }\n\n return emojiDataPromise;\n}\n","import React, { FC, memo } from '../../../lib/teact/teact';\nimport { OwnProps } from './EmojiTooltip';\nimport { Bundles } from '../../../util/moduleLoader';\n\nimport useModuleLoader from '../../../hooks/useModuleLoader';\n\nconst EmojiTooltipAsync: FC<OwnProps> = (props) => {\n const { isOpen } = props;\n const EmojiTooltip = useModuleLoader(Bundles.Extra, 'EmojiTooltip', !isOpen);\n\n // eslint-disable-next-line react/jsx-props-no-spreading\n return EmojiTooltip ? <EmojiTooltip {...props} /> : undefined;\n};\n\nexport default memo(EmojiTooltipAsync);\n","import { ChangeEvent } from 'react';\nimport React, {\n FC, useEffect, useRef, memo, useState, useCallback,\n} from '../../../lib/teact/teact';\nimport { withGlobal } from '../../../lib/teact/teactn';\n\nimport { GlobalActions } from '../../../global/types';\nimport { IAnchorPosition, ISettings } from '../../../types';\n\nimport { EDITABLE_INPUT_ID } from '../../../config';\nimport { selectCurrentMessageList, selectReplyingToId } from '../../../modules/selectors';\nimport { debounce } from '../../../util/schedulers';\nimport focusEditableElement from '../../../util/focusEditableElement';\nimport buildClassName from '../../../util/buildClassName';\nimport { pick } from '../../../util/iteratees';\nimport {\n IS_ANDROID, IS_IOS, IS_SINGLE_COLUMN_LAYOUT, IS_TOUCH_ENV,\n} from '../../../util/environment';\nimport captureKeyboardListeners from '../../../util/captureKeyboardListeners';\nimport useLayoutEffectWithPrevDeps from '../../../hooks/useLayoutEffectWithPrevDeps';\nimport useFlag from '../../../hooks/useFlag';\nimport parseEmojiOnlyString from '../../common/helpers/parseEmojiOnlyString';\nimport { isSelectionInsideInput } from './helpers/selection';\nimport useLang from '../../../hooks/useLang';\nimport renderText from '../../common/helpers/renderText';\n\nimport TextFormatter from './TextFormatter';\n\nconst CONTEXT_MENU_CLOSE_DELAY_MS = 100;\n// Focus slows down animation, also it breaks transition layout in Chrome\nconst FOCUS_DELAY_MS = 350;\nconst TRANSITION_DURATION_FACTOR = 50;\n\ntype OwnProps = {\n id: string;\n isAttachmentModalInput?: boolean;\n editableInputId?: string;\n html: string;\n placeholder: string;\n forcedPlaceholder?: string;\n shouldSetFocus: boolean;\n shouldSuppressFocus?: boolean;\n shouldSuppressTextFormatter?: boolean;\n onUpdate: (html: string) => void;\n onSuppressedFocus?: () => void;\n onSend: () => void;\n};\n\ntype StateProps = {\n currentChatId?: number;\n replyingToId?: number;\n noTabCapture?: boolean;\n messageSendKeyCombo?: ISettings['messageSendKeyCombo'];\n};\n\ntype DispatchProps = Pick<GlobalActions, 'editLastMessage' | 'replyToNextMessage'>;\n\nconst MAX_INPUT_HEIGHT = IS_SINGLE_COLUMN_LAYOUT ? 256 : 416;\nconst TAB_INDEX_PRIORITY_TIMEOUT = 2000;\nconst TEXT_FORMATTER_SAFE_AREA_PX = 90;\n// For some reason Safari inserts `<br>` after user removes text from input\nconst SAFARI_BR = '<br>';\n\nfunction clearSelection() {\n const selection = window.getSelection();\n if (!selection) {\n return;\n }\n\n if (selection.removeAllRanges) {\n selection.removeAllRanges();\n } else if (selection.empty) {\n selection.empty();\n }\n}\n\nconst MessageInput: FC<OwnProps & StateProps & DispatchProps> = ({\n id,\n isAttachmentModalInput,\n editableInputId,\n html,\n placeholder,\n forcedPlaceholder,\n shouldSetFocus,\n shouldSuppressFocus,\n shouldSuppressTextFormatter,\n onUpdate,\n onSuppressedFocus,\n onSend,\n currentChatId,\n replyingToId,\n noTabCapture,\n messageSendKeyCombo,\n editLastMessage,\n replyToNextMessage,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const inputRef = useRef<HTMLDivElement>(null);\n // eslint-disable-next-line no-null/no-null\n const cloneRef = useRef<HTMLDivElement>(null);\n\n const lang = useLang();\n const isContextMenuOpenRef = useRef(false);\n const [isTextFormatterOpen, openTextFormatter, closeTextFormatter] = useFlag();\n const [textFormatterAnchorPosition, setTextFormatterAnchorPosition] = useState<IAnchorPosition>();\n const [selectedRange, setSelectedRange] = useState<Range>();\n\n useEffect(() => {\n if (!isAttachmentModalInput) return;\n updateInputHeight(false);\n }, [isAttachmentModalInput]);\n\n useLayoutEffectWithPrevDeps(([prevHtml]) => {\n if (html !== inputRef.current!.innerHTML) {\n inputRef.current!.innerHTML = html;\n }\n\n if (html !== cloneRef.current!.innerHTML) {\n cloneRef.current!.innerHTML = html;\n }\n\n if (prevHtml !== undefined && prevHtml !== html) {\n updateInputHeight(!html.length);\n }\n }, [html]);\n\n const focusInput = useCallback(() => {\n // Avoid focusing during animation\n if (inputRef.current!.closest('.from, .to')) {\n setTimeout(focusInput, FOCUS_DELAY_MS);\n return;\n }\n\n focusEditableElement(inputRef.current!);\n }, []);\n\n const handleCloseTextFormatter = useCallback(() => {\n closeTextFormatter();\n clearSelection();\n }, [closeTextFormatter]);\n\n function checkSelection() {\n // Disable the formatter on iOS devices for now.\n if (IS_IOS) {\n return;\n }\n\n const selection = window.getSelection();\n if (!selection || !selection.rangeCount || isContextMenuOpenRef.current) {\n closeTextFormatter();\n return;\n }\n\n const selectionRange = selection.getRangeAt(0);\n const selectedText = selectionRange.toString().trim();\n if (\n shouldSuppressTextFormatter\n || !isSelectionInsideInput(selectionRange, editableInputId || EDITABLE_INPUT_ID)\n || !selectedText\n || parseEmojiOnlyString(selectedText)\n || !selectionRange.START_TO_END\n ) {\n closeTextFormatter();\n return;\n }\n\n const selectionRect = selectionRange.getBoundingClientRect();\n const inputRect = inputRef.current!.getBoundingClientRect();\n\n let x = (selectionRect.left + selectionRect.width / 2) - inputRect.left;\n\n if (x < TEXT_FORMATTER_SAFE_AREA_PX) {\n x = TEXT_FORMATTER_SAFE_AREA_PX;\n } else if (x > inputRect.width - TEXT_FORMATTER_SAFE_AREA_PX) {\n x = inputRect.width - TEXT_FORMATTER_SAFE_AREA_PX;\n }\n\n setTextFormatterAnchorPosition({\n x,\n y: selectionRect.top - inputRect.top,\n });\n\n setSelectedRange(selectionRange);\n openTextFormatter();\n }\n\n function handleMouseDown(event: React.MouseEvent<HTMLDivElement, MouseEvent>) {\n function handleMouseUp() {\n checkSelection();\n\n event.target.removeEventListener('mouseup', handleMouseUp);\n }\n\n if (event.button !== 2) {\n event.target.addEventListener('mouseup', handleMouseUp);\n return;\n }\n\n if (isContextMenuOpenRef.current === true) {\n return;\n }\n\n isContextMenuOpenRef.current = true;\n\n function closeContextMenuMouseListener() {\n setTimeout(() => {\n isContextMenuOpenRef.current = false;\n }, CONTEXT_MENU_CLOSE_DELAY_MS);\n\n window.removeEventListener('mouseup', closeContextMenuMouseListener);\n }\n\n function closeContextMenuKeyListener(e: KeyboardEvent) {\n if (e.key !== 'Esc' && e.key !== 'Escape') {\n return;\n }\n\n setTimeout(() => {\n isContextMenuOpenRef.current = false;\n }, CONTEXT_MENU_CLOSE_DELAY_MS);\n\n window.removeEventListener('keydown', closeContextMenuKeyListener);\n }\n\n document.addEventListener('mousedown', closeContextMenuMouseListener);\n document.addEventListener('keydown', closeContextMenuKeyListener);\n }\n\n function handleKeyDown(e: React.KeyboardEvent<HTMLDivElement>) {\n function handleKeyUp() {\n checkSelection();\n\n e.target.removeEventListener('keyup', handleKeyUp);\n }\n\n if (e.metaKey && !html.length) {\n const targetIndexDelta = e.key === 'ArrowDown' ? 1 : e.key === 'ArrowUp' ? -1 : undefined;\n if (targetIndexDelta) {\n e.preventDefault();\n\n replyToNextMessage({ targetIndexDelta });\n return;\n }\n }\n\n if (e.key === 'Enter' && !e.shiftKey) {\n if (\n !(IS_IOS || IS_ANDROID)\n && (\n (messageSendKeyCombo === 'enter' && !e.shiftKey)\n || (messageSendKeyCombo === 'ctrl-enter' && (e.ctrlKey || e.metaKey))\n )\n ) {\n e.preventDefault();\n\n closeTextFormatter();\n onSend();\n }\n } else if (e.key === 'ArrowUp' && !html.length && !e.metaKey) {\n e.preventDefault();\n editLastMessage();\n } else {\n e.target.addEventListener('keyup', handleKeyUp);\n }\n }\n\n function handleTouchSelection() {\n if (!IS_ANDROID) {\n return;\n }\n\n checkSelection();\n }\n\n function handleChange(e: ChangeEvent<HTMLDivElement>) {\n const { innerHTML, textContent } = e.currentTarget;\n\n onUpdate(innerHTML === SAFARI_BR ? '' : innerHTML);\n\n // Reset focus on the input to remove any active styling when input is cleared\n if (!IS_TOUCH_ENV && (!textContent || !textContent.length)) {\n const selection = window.getSelection()!;\n if (selection) {\n inputRef.current!.blur();\n selection.removeAllRanges();\n focusEditableElement(inputRef.current!, true);\n }\n }\n }\n\n function stopEvent(e: React.MouseEvent<HTMLDivElement, MouseEvent>) {\n if (!IS_ANDROID) {\n return;\n }\n\n e.preventDefault();\n e.stopPropagation();\n }\n\n function updateInputHeight(willSend = false) {\n const input = inputRef.current!;\n const clone = cloneRef.current!;\n const currentHeight = Number(input.style.height.replace('px', ''));\n const newHeight = Math.min(clone.scrollHeight, MAX_INPUT_HEIGHT);\n if (newHeight === currentHeight) {\n return;\n }\n\n const transitionDuration = Math.round(\n TRANSITION_DURATION_FACTOR * Math.log(Math.abs(newHeight - currentHeight)),\n );\n\n const exec = () => {\n input.style.height = `${newHeight}px`;\n input.style.transitionDuration = `${transitionDuration}ms`;\n input.classList.toggle('overflown', clone.scrollHeight > MAX_INPUT_HEIGHT);\n };\n\n if (willSend) {\n // Sync with sending animation\n requestAnimationFrame(exec);\n } else {\n exec();\n }\n }\n\n useEffect(() => {\n if (IS_TOUCH_ENV) {\n return;\n }\n\n focusInput();\n }, [currentChatId, focusInput, replyingToId, shouldSetFocus]);\n\n useEffect(() => {\n if (noTabCapture) {\n return undefined;\n }\n\n const captureFirstTab = debounce((e: KeyboardEvent) => {\n if (e.key === 'Tab') {\n e.preventDefault();\n requestAnimationFrame(focusInput);\n }\n }, TAB_INDEX_PRIORITY_TIMEOUT, true, false);\n\n return captureKeyboardListeners({ onTab: captureFirstTab });\n }, [focusInput, noTabCapture]);\n\n useEffect(() => {\n const input = inputRef.current!;\n\n function suppressFocus() {\n input.blur();\n }\n\n if (shouldSuppressFocus) {\n input.addEventListener('focus', suppressFocus);\n }\n\n return () => {\n input.removeEventListener('focus', suppressFocus);\n };\n }, [shouldSuppressFocus]);\n\n const className = buildClassName(\n 'form-control custom-scroll',\n html.length > 0 && 'touched',\n shouldSuppressFocus && 'focus-disabled',\n );\n\n return (\n <div id={id} onClick={shouldSuppressFocus ? onSuppressedFocus : undefined} dir={lang.isRtl ? 'rtl' : undefined}>\n <div\n ref={inputRef}\n id={editableInputId || EDITABLE_INPUT_ID}\n className={className}\n contentEditable\n dir=\"auto\"\n onClick={focusInput}\n onChange={handleChange}\n onKeyDown={handleKeyDown}\n onMouseDown={handleMouseDown}\n onContextMenu={stopEvent}\n onTouchCancel={handleTouchSelection}\n />\n <div ref={cloneRef} className={buildClassName(className, 'clone')} dir=\"auto\" />\n {!forcedPlaceholder && <span className=\"placeholder-text\" dir=\"auto\">{placeholder}</span>}\n <TextFormatter\n isOpen={isTextFormatterOpen}\n anchorPosition={textFormatterAnchorPosition}\n selectedRange={selectedRange}\n onClose={handleCloseTextFormatter}\n />\n {forcedPlaceholder && <span className=\"forced-placeholder\">{renderText(forcedPlaceholder!)}</span>}\n </div>\n );\n};\n\nexport default memo(withGlobal<OwnProps>(\n (global): StateProps => {\n const { chatId: currentChatId, threadId } = selectCurrentMessageList(global) || {};\n const { messageSendKeyCombo } = global.settings.byKey;\n\n return {\n currentChatId,\n messageSendKeyCombo,\n replyingToId: currentChatId && threadId ? selectReplyingToId(global, currentChatId, threadId) : undefined,\n noTabCapture: global.isPollModalOpen || global.payment.isPaymentModalOpen,\n };\n },\n (setGlobal, actions): DispatchProps => pick(actions, ['editLastMessage', 'replyToNextMessage']),\n)(MessageInput));\n","import React, { FC, memo, useCallback } from '../../lib/teact/teact';\n\nimport { ApiMessage, ApiWebPage } from '../../api/types';\n\nimport { getFirstLinkInMessage, getMessageSummaryText, getMessageWebPage } from '../../modules/helpers';\nimport buildClassName from '../../util/buildClassName';\nimport trimText from '../../util/trimText';\nimport renderText from './helpers/renderText';\nimport { formatPastTimeShort } from '../../util/dateFormat';\nimport useLang from '../../hooks/useLang';\n\nimport Media from './Media';\nimport Link from '../ui/Link';\nimport SafeLink from './SafeLink';\n\nimport './WebLink.scss';\n\nconst MAX_TEXT_LENGTH = 170; // symbols\n\ntype OwnProps = {\n message: ApiMessage;\n senderTitle?: string;\n onMessageClick: (messageId: number, chatId: number) => void;\n};\n\nconst WebLink: FC<OwnProps> = ({ message, senderTitle, onMessageClick }) => {\n const lang = useLang();\n\n let linkData: ApiWebPage | undefined = getMessageWebPage(message);\n\n if (!linkData) {\n const link = getFirstLinkInMessage(message);\n if (link) {\n const { url, domain } = link;\n const messageText = getMessageSummaryText(lang, message);\n\n linkData = {\n siteName: domain.replace(/^www./, ''),\n url: url.includes('://') ? url : url.includes('@') ? `mailto:${url}` : `http://${url}`,\n description: messageText !== url ? messageText : undefined,\n } as ApiWebPage;\n }\n }\n\n const handleMessageClick = useCallback(() => {\n onMessageClick(message.id, message.chatId);\n }, [onMessageClick, message.id, message.chatId]);\n\n if (!linkData) {\n return undefined;\n }\n\n const {\n siteName,\n url,\n displayUrl,\n title,\n description,\n photo,\n video,\n } = linkData;\n\n const truncatedDescription = !senderTitle && trimText(description, MAX_TEXT_LENGTH);\n\n const className = buildClassName(\n 'WebLink scroll-item',\n (!photo && !video) && 'without-media',\n );\n\n return (\n <div\n className={className}\n data-initial={(siteName || displayUrl)[0]}\n dir={lang.isRtl ? 'rtl' : undefined}\n >\n {photo && (\n <Media message={message} />\n )}\n <div className=\"content\">\n <Link isRtl={lang.isRtl} className=\"site-title\" onClick={handleMessageClick}>\n {renderText(title || siteName || displayUrl)}\n </Link>\n {truncatedDescription && (\n <Link isRtl={lang.isRtl} className=\"site-description\" onClick={handleMessageClick}>\n {renderText(truncatedDescription)}\n </Link>\n )}\n <SafeLink\n url={url}\n className=\"site-name\"\n text=\"\"\n isRtl={lang.isRtl}\n >\n {url.replace('mailto:', '') || displayUrl}\n </SafeLink>\n {senderTitle && <div className=\"sender-name\">{renderText(senderTitle)}</div>}\n </div>\n {senderTitle && (\n <div className=\"message-date\">\n <Link\n className=\"date\"\n onClick={handleMessageClick}\n isRtl={lang.isRtl}\n >\n {formatPastTimeShort(lang, message.date * 1000)}\n </Link>\n </div>\n )}\n </div>\n );\n};\n\nexport default memo(WebLink);\n","import { useCallback } from '../../../../lib/teact/teact';\n\nexport enum ChatAnimationTypes {\n Move,\n Opacity,\n None,\n}\n\nexport function useChatAnimationType(orderDiffById: Record<number, number>) {\n const movesUp = useCallback((id: number) => orderDiffById[id] < 0, [orderDiffById]);\n const movesDown = useCallback((id: number) => orderDiffById[id] > 0, [orderDiffById]);\n\n const orderDiffIds = Object.keys(orderDiffById).map(Number);\n const numberOfUp = orderDiffIds.filter(movesUp).length;\n const numberOfDown = orderDiffIds.filter(movesDown).length;\n\n return useCallback((chatId: number): ChatAnimationTypes => {\n const orderDiff = orderDiffById[chatId];\n\n if (orderDiff === 0) {\n return ChatAnimationTypes.None;\n }\n\n if (\n orderDiff === Infinity\n || orderDiff === -Infinity\n || (movesUp(chatId) && numberOfUp <= numberOfDown)\n || (movesDown(chatId) && numberOfDown < numberOfUp)\n ) {\n return ChatAnimationTypes.Opacity;\n }\n\n return ChatAnimationTypes.Move;\n }, [movesDown, movesUp, numberOfDown, numberOfUp, orderDiffById]);\n}\n","import React, { FC, memo } from '../../../lib/teact/teact';\n\nimport { ApiChat } from '../../../api/types';\n\nimport { formatIntegerCompact } from '../../../util/textFormat';\nimport buildClassName from '../../../util/buildClassName';\n\nimport ShowTransition from '../../ui/ShowTransition';\n\nimport './Badge.scss';\n\ntype OwnProps = {\n chat: ApiChat;\n isPinned?: boolean;\n isMuted?: boolean;\n};\n\nconst Badge: FC<OwnProps> = ({ chat, isPinned, isMuted }) => {\n const isShown = Boolean(chat.unreadCount || chat.hasUnreadMark || isPinned);\n const className = buildClassName(\n 'Badge',\n isMuted && 'muted',\n isPinned && 'pinned',\n Boolean(chat.unreadCount || chat.hasUnreadMark) && 'unread',\n );\n\n function renderContent() {\n if (chat.unreadCount) {\n if (chat.unreadMentionsCount) {\n return (\n <div className=\"Badge-wrapper\">\n <div className=\"Badge mention\">\n <i className=\"icon-mention\" />\n </div>\n <div className={className}>\n {formatIntegerCompact(chat.unreadCount)}\n </div>\n </div>\n );\n }\n\n return (\n <div className={className}>\n {formatIntegerCompact(chat.unreadCount)}\n </div>\n );\n } else if (chat.hasUnreadMark) {\n return (\n <div className={className} />\n );\n } else if (isPinned) {\n return (\n <div className={className}>\n <i className=\"icon-pinned-chat\" />\n </div>\n );\n }\n\n return undefined;\n }\n\n return (\n <ShowTransition isCustom className=\"Badge-transition\" isOpen={isShown}>\n {renderContent}\n </ShowTransition>\n );\n};\n\nexport default memo(Badge);\n","import React, {\n FC, memo, useCallback, useLayoutEffect, useMemo, useRef,\n} from '../../../lib/teact/teact';\nimport { withGlobal } from '../../../lib/teact/teactn';\n\nimport useLang, { LangFn } from '../../../hooks/useLang';\n\nimport { GlobalActions } from '../../../global/types';\nimport {\n ApiChat, ApiUser, ApiMessage, ApiMessageOutgoingStatus, ApiFormattedText, MAIN_THREAD_ID,\n} from '../../../api/types';\n\nimport { ANIMATION_END_DELAY } from '../../../config';\nimport { IS_SINGLE_COLUMN_LAYOUT } from '../../../util/environment';\nimport {\n getChatTitle,\n isChatPrivate,\n isActionMessage,\n getPrivateChatUserId,\n getMessageAction,\n getMessageSenderName,\n isChatChannel,\n getMessageMediaHash,\n getMessageSummaryText,\n getMessageMediaThumbDataUri,\n getMessageVideo,\n getMessageSticker,\n selectIsChatMuted,\n getMessageRoundVideo,\n} from '../../../modules/helpers';\nimport {\n selectChat, selectUser, selectChatMessage, selectOutgoingStatus, selectDraft, selectCurrentMessageList,\n selectNotifySettings, selectNotifyExceptions,\n} from '../../../modules/selectors';\nimport { renderActionMessageText } from '../../common/helpers/renderActionMessageText';\nimport renderText from '../../common/helpers/renderText';\nimport { fastRaf } from '../../../util/schedulers';\nimport buildClassName from '../../../util/buildClassName';\nimport { pick } from '../../../util/iteratees';\nimport useEnsureMessage from '../../../hooks/useEnsureMessage';\nimport useChatContextActions from '../../../hooks/useChatContextActions';\nimport useFlag from '../../../hooks/useFlag';\nimport useMedia from '../../../hooks/useMedia';\nimport { ChatAnimationTypes } from './hooks';\n\nimport Avatar from '../../common/Avatar';\nimport VerifiedIcon from '../../common/VerifiedIcon';\nimport TypingStatus from '../../common/TypingStatus';\nimport LastMessageMeta from '../../common/LastMessageMeta';\nimport DeleteChatModal from '../../common/DeleteChatModal';\nimport ListItem from '../../ui/ListItem';\nimport Badge from './Badge';\n\nimport './Chat.scss';\n\ntype OwnProps = {\n style?: string;\n chatId: number;\n folderId?: number;\n orderDiff: number;\n animationType: ChatAnimationTypes;\n isPinned?: boolean;\n};\n\ntype StateProps = {\n chat?: ApiChat;\n isMuted?: boolean;\n privateChatUser?: ApiUser;\n actionTargetUserIds?: number[];\n usersById?: Record<number, ApiUser>;\n actionTargetMessage?: ApiMessage;\n actionTargetChatId?: number;\n lastMessageSender?: ApiUser;\n lastMessageOutgoingStatus?: ApiMessageOutgoingStatus;\n draft?: ApiFormattedText;\n animationLevel?: number;\n isSelected?: boolean;\n canScrollDown?: boolean;\n lastSyncTime?: number;\n};\n\ntype DispatchProps = Pick<GlobalActions, 'openChat' | 'focusLastMessage'>;\n\nconst ANIMATION_DURATION = 200;\n\nconst Chat: FC<OwnProps & StateProps & DispatchProps> = ({\n style,\n chatId,\n folderId,\n orderDiff,\n animationType,\n isPinned,\n chat,\n isMuted,\n privateChatUser,\n actionTargetUserIds,\n usersById,\n lastMessageSender,\n lastMessageOutgoingStatus,\n actionTargetMessage,\n actionTargetChatId,\n draft,\n animationLevel,\n isSelected,\n canScrollDown,\n lastSyncTime,\n openChat,\n focusLastMessage,\n}) => {\n // eslint-disable-next-line no-null/no-null\n const ref = useRef<HTMLDivElement>(null);\n\n const [isDeleteModalOpen, openDeleteModal, closeDeleteModal] = useFlag();\n const [shouldRenderDeleteModal, markRenderDeleteModal, unmarkRenderDeleteModal] = useFlag();\n\n const { lastMessage, typingStatus } = chat || {};\n const isAction = lastMessage && isActionMessage(lastMessage);\n\n useEnsureMessage(chatId, isAction ? lastMessage!.replyToMessageId : undefined, actionTargetMessage);\n\n const mediaThumbnail = lastMessage && !getMessageSticker(lastMessage)\n ? getMessageMediaThumbDataUri(lastMessage)\n : undefined;\n const mediaBlobUrl = useMedia(lastMessage ? getMessageMediaHash(lastMessage, 'micro') : undefined);\n const isRoundVideo = Boolean(lastMessage && getMessageRoundVideo(lastMessage));\n\n const actionTargetUsers = useMemo(() => {\n return actionTargetUserIds\n ? actionTargetUserIds.map((userId) => usersById && usersById[userId]).filter<ApiUser>(Boolean as any)\n : undefined;\n }, [actionTargetUserIds, usersById]);\n\n // Sets animation excess values when `orderDiff` changes and then resets excess values to animate.\n useLayoutEffect(() => {\n const element = ref.current;\n\n if (animationLevel === 0 || !element) {\n return;\n }\n\n // TODO Refactor animation: create `useListAnimation` that owns `orderDiff` and `animationType`\n if (animationType === ChatAnimationTypes.Opacity) {\n element.style.opacity = '0';\n\n fastRaf(() => {\n element.classList.add('animate-opacity');\n element.style.opacity = '1';\n });\n } else if (animationType === ChatAnimationTypes.Move) {\n element.style.transform = `translate3d(0, ${-orderDiff * 100}%, 0)`;\n\n fastRaf(() => {\n element.classList.add('animate-transform');\n element.style.transform = '';\n });\n } else {\n return;\n }\n\n setTimeout(() => {\n fastRaf(() => {\n element.classList.remove('animate-opacity', 'animate-transform');\n element.style.opacity = '';\n element.style.transform = '';\n });\n }, ANIMATION_DURATION + ANIMATION_END_DELAY);\n }, [animationLevel, orderDiff, animationType]);\n\n const handleClick = useCallback(() => {\n openChat({ id: chatId, shouldReplaceHistory: true });\n\n if (isSelected && canScrollDown) {\n focusLastMessage();\n }\n }, [\n isSelected,\n canScrollDown,\n openChat,\n chatId,\n focusLastMessage,\n ]);\n\n function handleDelete() {\n markRenderDeleteModal();\n openDeleteModal();\n }\n\n const contextActions = useChatContextActions({\n chat,\n privateChatUser,\n handleDelete,\n folderId,\n isPinned,\n isMuted,\n });\n\n const lang = useLang();\n\n if (!chat) {\n return undefined;\n }\n\n function renderLastMessageOrTyping() {\n if (typingStatus && lastMessage && typingStatus.timestamp > lastMessage.date * 1000) {\n return <TypingStatus typingStatus={typingStatus} />;\n }\n\n if (draft && draft.text.length) {\n return (\n <p className=\"last-message\" dir={lang.isRtl ? 'auto' : 'ltr'}>\n <span className=\"draft\">{lang('Draft')}</span>\n {renderText(draft.text)}\n </p>\n );\n }\n\n if (!lastMessage) {\n return undefined;\n }\n\n if (isAction) {\n const actionOrigin = chat && (isChatChannel(chat) || lastMessage.senderId === lastMessage.chatId)\n ? chat\n : lastMessageSender;\n\n return (\n <p className=\"last-message\" dir={lang.isRtl ? 'auto' : 'ltr'}>\n {renderText(renderActionMessageText(\n lang,\n lastMessage,\n actionOrigin,\n actionTargetUsers,\n actionTargetMessage,\n actionTargetChatId,\n { asPlain: true },\n ) as string)}\n </p>\n );\n }\n\n const senderName = getMessageSenderName(lang, chatId, lastMessageSender);\n\n return (\n <p className=\"last-message\" dir={lang.isRtl ? 'auto' : 'ltr'}>\n {senderName && (\n <>\n <span className=\"sender-name\">{renderText(senderName)}</span>\n <span className=\"colon\">:</span>\n </>\n )}\n {renderMessageSummary(lang, lastMessage!, mediaBlobUrl || mediaThumbnail, isRoundVideo)}\n </p>\n );\n }\n\n const className = buildClassName(\n 'Chat chat-item-clickable',\n isChatPrivate(chatId) ? 'private' : 'group',\n isSelected && 'selected',\n );\n\n return (\n <ListItem\n ref={ref}\n className={className}\n style={style}\n ripple={!IS_SINGLE_COLUMN_LAYOUT}\n contextActions={contextActions}\n onClick={handleClick}\n >\n <div className=\"status\">\n <Avatar\n chat={chat}\n user={privateChatUser}\n withOnlineStatus\n isSavedMessages={privateChatUser && privateChatUser.isSelf}\n lastSyncTime={lastSyncTime}\n />\n </div>\n <div className=\"info\">\n <div className=\"title\">\n <h3>{renderText(getChatTitle(lang, chat, privateChatUser))}</h3>\n {chat.isVerified && <VerifiedIcon />}\n {isMuted && <i className=\"icon-muted-chat\" />}\n {chat.lastMessage && (\n <LastMessageMeta message={chat.lastMessage} outgoingStatus={lastMessageOutgoingStatus} />\n )}\n </div>\n <div className=\"subtitle\">\n {renderLastMessageOrTyping()}\n <Badge chat={chat} isPinned={isPinned} isMuted={isMuted} />\n </div>\n </div>\n {shouldRenderDeleteModal && (\n <DeleteChatModal\n isOpen={isDeleteModalOpen}\n onClose={closeDeleteModal}\n onCloseAnimationEnd={unmarkRenderDeleteModal}\n chat={chat}\n />\n )}\n </ListItem>\n );\n};\n\nfunction renderMessageSummary(lang: LangFn, message: ApiMessage, blobUrl?: string, isRoundVideo?: boolean) {\n if (!blobUrl) {\n return renderText(getMessageSummaryText(lang, message));\n }\n\n return (\n <span className=\"media-preview\">\n <img src={blobUrl} alt=\"\" className={isRoundVideo ? 'round' : undefined} />\n {getMessageVideo(message) && <i className=\"icon-play\" />}\n {renderText(getMessageSummaryText(lang, message, true))}\n </span>\n );\n}\n\nexport default memo(withGlobal<OwnProps>(\n (global, { chatId }): StateProps => {\n const chat = selectChat(global, chatId);\n if (!chat || !chat.lastMessage) {\n return {};\n }\n\n const { senderId, replyToMessageId, isOutgoing } = chat.lastMessage;\n const lastMessageSender = senderId ? selectUser(global, senderId) : undefined;\n const lastMessageAction = getMessageAction(chat.lastMessage);\n const actionTargetMessage = lastMessageAction && replyToMessageId\n ? selectChatMessage(global, chat.id, replyToMessageId)\n : undefined;\n const { targetUserIds: actionTargetUserIds, targetChatId: actionTargetChatId } = lastMessageAction || {};\n const privateChatUserId = getPrivateChatUserId(chat);\n const { byId: usersById } = global.users;\n const {\n chatId: currentChatId,\n threadId: currentThreadId,\n type: messageListType,\n } = selectCurrentMessageList(global) || {};\n const isSelected = chatId === currentChatId && currentThreadId === MAIN_THREAD_ID;\n\n return {\n chat,\n isMuted: selectIsChatMuted(chat, selectNotifySettings(global), selectNotifyExceptions(global)),\n lastMessageSender,\n actionTargetUserIds,\n actionTargetChatId,\n actionTargetMessage,\n draft: selectDraft(global, chatId, MAIN_THREAD_ID),\n animationLevel: global.settings.byKey.animationLevel,\n isSelected,\n canScrollDown: isSelected && messageListType === 'thread',\n lastSyncTime: global.lastSyncTime,\n ...(isOutgoing && { lastMessageOutgoingStatus: selectOutgoingStatus(global, chat.lastMessage) }),\n ...(privateChatUserId && { privateChatUser: selectUser(global, privateChatUserId) }),\n ...(actionTargetUserIds && { usersById }),\n };\n },\n (setGlobal, actions): DispatchProps => pick(actions, [\n 'openChat',\n 'focusLastMessage',\n ]),\n)(Chat));\n","import React, { FC, memo, useCallback } from '../../../lib/teact/teact';\nimport { withGlobal } from '../../../lib/teact/teactn';\n\nimport { ApiChatFolder, ApiSticker } from '../../../api/types';\nimport { SettingsScreens } from '../../../types';\nimport { FolderEditDispatch } from '../../../hooks/reducers/useFoldersReducer';\n\nimport { IS_SINGLE_COLUMN_LAYOUT } from '../../../util/environment';\nimport { selectAnimatedEmoji, selectChatFolder } from '../../../modules/selectors';\nimport useLang from '../../../hooks/useLang';\n\nimport Button from '../../ui/Button';\nimport AnimatedEmoji from '../../common/AnimatedEmoji';\n\nimport './EmptyFolder.scss';\n\ntype OwnProps = {\n folderId?: number;\n folderType: 'all' | 'archived' | 'folder';\n foldersDispatch?: FolderEditDispatch;\n onScreenSelect?: (screen: SettingsScreens) => void;\n};\n\ntype StateProps = {\n chatFolder?: ApiChatFolder;\n animatedEmoji?: ApiSticker;\n};\n\nconst EmptyFolder: FC<OwnProps & StateProps> = ({\n chatFolder, animatedEmoji, foldersDispatch, onScreenSelect,\n}) => {\n const lang = useLang();\n\n const handleEditFolder = useCallback(() => {\n foldersDispatch!({ type: 'editFolder', payload: chatFolder });\n onScreenSelect!(SettingsScreens.FoldersEditFolderFromChatList);\n }, [chatFolder, foldersDispatch, onScreenSelect]);\n\n return (\n <div className=\"EmptyFolder\">\n <div className=\"sticker\">{animatedEmoji && <AnimatedEmoji sticker={animatedEmoji} />}</div>\n <h3 className=\"title\" dir=\"auto\">{lang('FilterNoChatsToDisplay')}</h3>\n <p className=\"description\" dir=\"auto\">\n {lang(chatFolder ? 'ChatList.EmptyChatListFilterText' : 'Chat.EmptyChat')}\n </p>\n {chatFolder && foldersDispatch && onScreenSelect && (\n <Button\n ripple={!IS_SINGLE_COLUMN_LAYOUT}\n fluid\n pill\n onClick={handleEditFolder}\n size=\"smaller\"\n isRtl={lang.isRtl}\n >\n <i className=\"icon-settings\" />\n {lang('ChatList.EmptyChatListEditFilter')}\n </Button>\n )}\n </div>\n );\n};\n\nexport default memo(withGlobal<OwnProps>((global, { folderId, folderType }): StateProps => {\n const chatFolder = folderId && folderType === 'folder' ? selectChatFolder(global, folderId) : undefined;\n\n return {\n chatFolder,\n animatedEmoji: selectAnimatedEmoji(global, '📂'),\n };\n})(EmptyFolder));\n","import React, {\n FC, memo, useMemo, useCallback, useEffect,\n} from '../../../lib/teact/teact';\nimport { withGlobal } from '../../../lib/teact/teactn';\n\nimport { GlobalActions } from '../../../global/types';\nimport {\n ApiChat, ApiChatFolder, ApiUser,\n} from '../../../api/types';\nimport { NotifyException, NotifySettings, SettingsScreens } from '../../../types';\nimport { FolderEditDispatch } from '../../../hooks/reducers/useFoldersReducer';\n\nimport { ALL_CHATS_PRELOAD_DISABLED, CHAT_HEIGHT_PX, CHAT_LIST_SLICE } from '../../../config';\nimport { IS_ANDROID, IS_MAC_OS, IS_PWA } from '../../../util/environment';\nimport usePrevious from '../../../hooks/usePrevious';\nimport { mapValues, pick } from '../../../util/iteratees';\nimport { getChatOrder, prepareChatList, prepareFolderListIds } from '../../../modules/helpers';\nimport {\n selectChatFolder, selectNotifyExceptions, selectNotifySettings,\n} from '../../../modules/selectors';\nimport useInfiniteScroll from '../../../hooks/useInfiniteScroll';\nimport { useChatAnimationType } from './hooks';\n\nimport InfiniteScroll from '../../ui/InfiniteScroll';\nimport Loading from '../../ui/Loading';\nimport Chat from './Chat';\nimport EmptyFolder from './EmptyFolder';\n\ntype OwnProps = {\n folderType: 'all' | 'archived' | 'folder';\n folderId?: number;\n isActive: boolean;\n onScreenSelect?: (screen: SettingsScreens) => void;\n foldersDispatch?: FolderEditDispatch;\n};\n\ntype StateProps = {\n chatsById: Record<number, ApiChat>;\n usersById: Record<number, ApiUser>;\n chatFolder?: ApiChatFolder;\n listIds?: number[];\n orderedPinnedIds?: number[];\n lastSyncTime?: number;\n notifySettings: NotifySettings;\n notifyExceptions?: Record<number, NotifyException>;\n};\n\ntype DispatchProps = Pick<GlobalActions, 'loadMoreChats' | 'preloadTopChatMessages' | 'openChat' | 'openNextChat'>;\n\nenum FolderTypeToListType {\n 'all' = 'active',\n 'archived' = 'archived'\n}\n\nconst ChatList: FC<OwnProps & StateProps & DispatchProps> = ({\n folderType,\n folderId,\n isActive,\n chatFolder,\n chatsById,\n usersById,\n listIds,\n orderedPinnedIds,\n lastSyncTime,\n foldersDispatch,\n notifySettings,\n notifyExceptions,\n onScreenSelect,\n loadMoreChats,\n preloadTopChatMessages,\n openChat,\n openNextChat,\n}) => {\n const [currentListIds, currentPinnedIds] = useMemo(() => {\n return folderType === 'folder' && chatFolder\n ? prepareFolderListIds(chatsById, usersById, chatFolder, notifySettings, notifyExceptions)\n : [listIds, orderedPinnedIds];\n }, [folderType, chatFolder, chatsById, usersById, notifySettings, notifyExceptions, listIds, orderedPinnedIds]);\n\n const [orderById, orderedIds] = useMemo(() => {\n if (!currentListIds || (folderType === 'folder' && !chatFolder)) {\n return [];\n }\n const newChatArrays = prepareChatList(chatsById, currentListIds, currentPinnedIds, folderType);\n const singleList = [...newChatArrays.pinnedChats, ...newChatArrays.otherChats];\n const newOrderedIds = singleList.map(({ id }) => id);\n const newOrderById = singleList.reduce((acc, chat, i) => {\n acc[chat.id] = i;\n return acc;\n }, {} as Record<string, number>);\n\n return [newOrderById, newOrderedIds];\n }, [currentListIds, currentPinnedIds, folderType, chatFolder, chatsById]);\n\n const prevOrderById = usePrevious(orderById);\n\n const orderDiffById = useMemo(() => {\n if (!orderById || !prevOrderById) {\n return {};\n }\n\n return mapValues(orderById, (order, id) => {\n return order - (prevOrderById[id] !== undefined ? prevOrderById[id] : Infinity);\n });\n }, [orderById, prevOrderById]);\n\n const loadMoreOfType = useCallback(() => {\n loadMoreChats({ listType: folderType === 'archived' ? 'archived' : 'active' });\n }, [loadMoreChats, folderType]);\n\n const [viewportIds, getMore] = useInfiniteScroll(\n lastSyncTime ? loadMoreOfType : undefined,\n orderedIds,\n undefined,\n CHAT_LIST_SLICE,\n folderType === 'all' && !ALL_CHATS_PRELOAD_DISABLED,\n );\n\n // TODO Refactor to not call `prepareChatList` twice\n const chatArrays = viewportIds && prepareChatList(chatsById, viewportIds, currentPinnedIds, folderType);\n\n useEffect(() => {\n if (lastSyncTime && folderType === 'all') {\n preloadTopChatMessages();\n }\n }, [lastSyncTime, folderType, preloadTopChatMessages]);\n\n const getAnimationType = useChatAnimationType(orderDiffById);\n\n function renderChats() {\n const viewportOffset = orderedIds!.indexOf(viewportIds![0]);\n const pinnedOffset = viewportOffset + chatArrays!.pinnedChats.length;\n\n return (\n <div\n className=\"scroll-container\"\n // @ts-ignore\n style={IS_ANDROID ? `height: ${orderedIds!.length * CHAT_HEIGHT_PX}px` : undefined}\n teactFastList\n >\n {chatArrays!.pinnedChats.map(({ id }, i) => (\n <Chat\n key={id}\n teactOrderKey={i}\n chatId={id}\n isPinned\n folderId={folderId}\n animationType={getAnimationType(id)}\n orderDiff={orderDiffById[id]}\n // @ts-ignore\n style={`top: ${(viewportOffset + i) * CHAT_HEIGHT_PX}px;`}\n />\n ))}\n {chatArrays!.otherChats.map((chat, i) => (\n <Chat\n key={chat.id}\n teactOrderKey={getChatOrder(chat)}\n chatId={chat.id}\n folderId={folderId}\n animationType={getAnimationType(chat.id)}\n orderDiff={orderDiffById[chat.id]}\n // @ts-ignore\n style={`top: ${(pinnedOffset + i) * CHAT_HEIGHT_PX}px;`}\n />\n ))}\n </div>\n );\n }\n\n useEffect(() => {\n const handleKeyDown = (e: KeyboardEvent) => {\n if (isActive && orderedIds) {\n if (IS_PWA && ((IS_MAC_OS && e.metaKey) || (!IS_MAC_OS && e.ctrlKey)) && e.code.startsWith('Digit')) {\n const [, digit] = e.code.match(/Digit(\\d)/) || [];\n if (!digit) return;\n\n const position = Number(digit) - 1;\n if (position > orderedIds.length - 1) return;\n\n openChat({ id: orderedIds[position], shouldReplaceHistory: true });\n }\n\n if (e.altKey) {\n const targetIndexDelta = e.key === 'ArrowDown' ? 1 : e.key === 'ArrowUp' ? -1 : undefined;\n if (!targetIndexDelta) return;\n\n e.preventDefault();\n openNextChat({ targetIndexDelta, orderedIds });\n }\n }\n };\n\n document.addEventListener('keydown', handleKeyDown, false);\n\n return () => {\n document.removeEventListener('keydown', handleKeyDown, false);\n };\n });\n\n return (\n <InfiniteScroll\n className=\"chat-list custom-scroll\"\n items={viewportIds}\n onLoadMore={getMore}\n preloadBackwards={CHAT_LIST_SLICE}\n noFastList\n noScrollRestore\n >\n {viewportIds && viewportIds.length && chatArrays ? (\n renderChats()\n ) : viewportIds && !viewportIds.length ? (\n (\n <EmptyFolder\n folderId={folderId}\n folderType={folderType}\n foldersDispatch={foldersDispatch}\n onScreenSelect={onScreenSelect}\n />\n )\n ) : (\n <Loading key=\"loading\" />\n )}\n </InfiniteScroll>\n );\n};\n\nexport default memo(withGlobal<OwnProps>(\n (global, { folderType, folderId }): StateProps => {\n const {\n chats: {\n listIds,\n byId: chatsById,\n orderedPinnedIds,\n },\n users: { byId: usersById },\n lastSyncTime,\n } = global;\n const listType = folderType !== 'folder' ? FolderTypeToListType[folderType] : undefined;\n const chatFolder = folderId ? selectChatFolder(global, folderId) : undefined;\n\n return {\n chatsById,\n usersById,\n lastSyncTime,\n notifySettings: selectNotifySettings(global),\n notifyExceptions: selectNotifyExceptions(global),\n ...(listType ? {\n listIds: listIds[listType],\n orderedPinnedIds: orderedPinnedIds[listType],\n } : {\n chatFolder,\n }),\n };\n },\n (setGlobal, actions): DispatchProps => pick(actions, [\n 'loadMoreChats',\n 'preloadTopChatMessages',\n 'openChat',\n 'openNextChat',\n ]),\n)(ChatList));\n","import { ApiUser } from '../../../../api/types';\nimport { getUserFullName } from '../../../../modules/helpers';\nimport searchWords from '../../../../util/searchWords';\n\n// TODO: Support cyrillic translit search\nexport default function searchUserName(filter: string, user: ApiUser) {\n const usernameLowered = user.username.toLowerCase();\n const fullName = getUserFullName(user);\n const fullNameLowered = fullName && fullName.toLowerCase();\n const filterLowered = filter.toLowerCase();\n\n return usernameLowered.startsWith(filterLowered) || (\n fullNameLowered && searchWords(fullNameLowered, filterLowered)\n );\n}\n","import {\n useCallback, useEffect, useState, useMemo,\n} from '../../../../lib/teact/teact';\n\nimport { ApiMessageEntityTypes, ApiChatMember, ApiUser } from '../../../../api/types';\nimport { EDITABLE_INPUT_ID } from '../../../../config';\nimport { getUserFirstOrLastName } from '../../../../modules/helpers';\nimport searchUserName from '../helpers/searchUserName';\nimport focusEditableElement from '../../../../util/focusEditableElement';\nimport useFlag from '../../../../hooks/useFlag';\nimport { unique } from '../../../../util/iteratees';\nimport { throttle } from '../../../../util/schedulers';\n\nconst runThrottled = throttle((cb) => cb(), 500, true);\nconst RE_BR = /(<br>|<br\\s?\\/>)/g;\nconst RE_SPACE = / /g;\nconst RE_CLEAN_HTML = /(<div>|<\\/div>)/gi;\nlet RE_USERNAME_SEARCH: RegExp;\n\ntry {\n RE_USERNAME_SEARCH = new RegExp('(^|\\\\s)@[-_\\\\p{L}\\\\p{M}\\\\p{N}]*$', 'gui');\n} catch (e) {\n // Support for older versions of firefox\n RE_USERNAME_SEARCH = new RegExp('(^|\\\\s)@[-_\\\\d\\\\wа-яё]*$', 'gi');\n}\n\nexport default function useMentionTooltip(\n canSuggestMembers: boolean | undefined,\n html: string,\n onUpdateHtml: (html: string) => void,\n inputId: string = EDITABLE_INPUT_ID,\n groupChatMembers?: ApiChatMember[],\n topInlineBotIds?: number[],\n currentUserId?: number,\n usersById?: Record<number, ApiUser>,\n) {\n const [isOpen, markIsOpen, unmarkIsOpen] = useFlag();\n const [currentFilter, setCurrentFilter] = useState('');\n const [usersToMention, setUsersToMention] = useState<ApiUser[] | undefined>();\n\n const topInlineBots = useMemo(() => {\n return (topInlineBotIds || []).map((id) => usersById && usersById[id]).filter<ApiUser>(Boolean as any);\n }, [topInlineBotIds, usersById]);\n\n const getFilteredUsers = useCallback((filter, withInlineBots: boolean) => {\n if (!(groupChatMembers || topInlineBotIds) || !usersById) {\n setUsersToMention(undefined);\n\n return;\n }\n runThrottled(() => {\n const inlineBots = (withInlineBots ? topInlineBots : []).filter((inlineBot) => {\n return !filter || searchUserName(filter, inlineBot);\n });\n\n const chatMembers = (groupChatMembers || [])\n .map(({ userId }) => usersById[userId])\n .filter((user) => {\n if (!user || user.id === currentUserId) {\n return false;\n }\n\n return !filter || searchUserName(filter, user);\n });\n\n setUsersToMention(unique(inlineBots.concat(chatMembers)));\n });\n }, [currentUserId, groupChatMembers, topInlineBotIds, topInlineBots, usersById]);\n\n useEffect(() => {\n if (!canSuggestMembers || !html.length) {\n unmarkIsOpen();\n return;\n }\n\n const usernameFilter = html.includes('@') && getUsernameFilter(html);\n\n if (usernameFilter) {\n const filter = usernameFilter ? usernameFilter.substr(1) : '';\n setCurrentFilter(filter);\n getFilteredUsers(filter, canSuggestInlineBots(html));\n } else {\n unmarkIsOpen();\n }\n }, [canSuggestMembers, html, getFilteredUsers, markIsOpen, unmarkIsOpen]);\n\n useEffect(() => {\n if (usersToMention && usersToMention.length) {\n markIsOpen();\n } else {\n unmarkIsOpen();\n }\n }, [markIsOpen, unmarkIsOpen, usersToMention]);\n\n const insertMention = useCallback((user: ApiUser, forceFocus = false) => {\n if (!user.username && !getUserFirstOrLastName(user)) {\n return;\n }\n\n const insertedHtml = user.username\n ? `@${user.username}`\n : `<a\n class=\"text-entity-link\"\n data-entity-type=\"${ApiMessageEntityTypes.MentionName}\"\n data-user-id=\"${user.id}\"\n contenteditable=\"false\"\n dir=\"auto\"\n >${getUserFirstOrLastName(user)}</a>`;\n\n const atIndex = html.lastIndexOf('@');\n if (atIndex !== -1) {\n onUpdateHtml(`${html.substr(0, atIndex)}${insertedHtml} `);\n const messageInput = document.getElementById(inputId)!;\n requestAnimationFrame(() => {\n focusEditableElement(messageInput, forceFocus);\n });\n }\n\n unmarkIsOpen();\n }, [html, inputId, onUpdateHtml, unmarkIsOpen]);\n\n return {\n isMentionTooltipOpen: isOpen,\n mentionFilter: currentFilter,\n closeMentionTooltip: unmarkIsOpen,\n insertMention,\n mentionFilteredUsers: usersToMention,\n };\n}\n\nfunction getUsernameFilter(html: string) {\n const username = html\n .replace(RE_SPACE, ' ')\n .replace(RE_BR, '\\n')\n .replace(RE_CLEAN_HTML, '')\n .replace(/\\n$/i, '')\n .match(RE_USERNAME_SEARCH);\n\n return username ? username[0].trim() : undefined;\n}\n\nfunction canSuggestInlineBots(html: string) {\n return html.startsWith('@');\n}\n","import { useEffect, useRef } from '../lib/teact/teact';\n\nimport fastBlur from '../lib/fastBlur';\nimport { imgToCanvas } from '../util/files';\nimport useForceUpdate from './useForceUpdate';\n\nconst RADIUS = 2;\nconst ITERATIONS = 2;\nconst MAX_CACHE_SIZE = 1000;\n\nconst cache = new Map<string, string>();\n\nexport default function useBlur(dataUri?: string, isDisabled = false, delay?: number) {\n const blurredRef = useRef<string | undefined>(dataUri ? cache.get(dataUri) : undefined);\n const timeoutRef = useRef<number>();\n const forceUpdate = useForceUpdate();\n\n if (timeoutRef.current && isDisabled) {\n clearTimeout(timeoutRef.current);\n timeoutRef.current = undefined;\n }\n\n useEffect(() => {\n if (!dataUri || blurredRef.current || isDisabled) {\n return;\n }\n\n const img = new Image();\n\n img.onload = () => {\n const canvas = imgToCanvas(img);\n fastBlur(canvas.getContext('2d'), 0, 0, canvas.width, canvas.height, RADIUS, ITERATIONS);\n const blurredDataUri = canvas.toDataURL();\n\n blurredRef.current = blurredDataUri;\n forceUpdate();\n\n if (cache.size >= MAX_CACHE_SIZE) {\n cache.clear();\n }\n cache.set(dataUri, blurredDataUri);\n };\n\n if (delay) {\n timeoutRef.current = window.setTimeout(() => {\n img.src = dataUri;\n }, delay);\n } else {\n img.src = dataUri;\n }\n }, [dataUri, delay, forceUpdate, isDisabled]);\n\n return blurredRef.current;\n}\n","import { useRef } from '../lib/teact/teact';\n\nimport fastBlur from '../lib/fastBlur';\nimport useOnChange from './useOnChange';\nimport useBlur from './useBlur';\nimport { imgToCanvas } from '../util/files';\n\nconst RADIUS = 2;\nconst ITERATIONS = 2;\n\nexport default function useBlurSync(dataUri: string | false | undefined) {\n const blurredRef = useRef<string>();\n\n let isChanged = false;\n\n useOnChange(() => {\n isChanged = true;\n\n blurredRef.current = undefined;\n\n if (!dataUri) {\n return;\n }\n\n const img = new Image();\n img.src = dataUri;\n if (!img.width) {\n return;\n }\n\n const canvas = imgToCanvas(img);\n fastBlur(canvas.getContext('2d'), 0, 0, canvas.width, canvas.height, RADIUS, ITERATIONS);\n\n blurredRef.current = canvas.toDataURL();\n }, [dataUri]);\n\n // Sometimes `Image` do not manage to load synchronously,\n // so we fall back the non-blurred variant and prepare the async one at least for the next time\n const blurredAsync = useBlur(dataUri || undefined, Boolean(blurredRef.current));\n\n return blurredRef.current || (!isChanged && blurredAsync) || dataUri || undefined;\n}\n"],"sourceRoot":""} |