1 line
244 KiB
Plaintext
1 line
244 KiB
Plaintext
{"version":3,"mappings":"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","names":["ansiRegex","onlyFirst","pattern","regex","stripAnsi","string","segmenter","stringLength","countAnsiEscapeCodes","length","_","urlRegexObj","RegExp","urlRegex","source","flags","isMobileSafari","test","navigator","userAgent","CameraCaptureInput","hidden","disabled","supportedMimeTypes","setMediaAttachments","filteredSupportedMimeTypes","filter","mimeType","_jsx","type","accept","join","capture","onChange","e","files","target","mediaFile","Array","from","attachments","file","size","url","URL","createObjectURL","id","description","value","supportsCameraCapture","document","createElement","CharCountMeter","maxCharacters","charCount","useSnapshot","states","composerCharacterCount","leftChars","class","_jsxs","title","style","children","max","Combobox","input","list","tabInsertsSuggestions","defaultFirstOption","event","keyboardBindings","trackComposition","commitWithElement","_a","visible","indexDiff","focusEl","els","focusIndex","indexOfItem","newIndex","el","scrollTo","combobox","commit","fireCommitEvent","container","inViewport","element","scrollTop","containerBottom","top","bottom","boundary","query","text","key","cursor","multiWord","lookBackIndex","lastMatchPosition","keyIndex","pre","InputStyleCloneUpdateEvent","CloneRegistry","InputStyleClone","#styleObserver","#updateStyles","#resizeObserver","#requestUpdateLayout","#inputRef","#container","#cloneElement","clone","#updateText","#onDocumentScrollOrResize","#onInput","#input","#usingInput","fn","#xOffset","#yOffset","#updateLayout","inputStyle","inputRect","cloneRect","#isLayoutUpdating","prop","propertiesToCopy","InputRange","#inputElement","#startOffset","#endOffset","startOffset","endOffset","selectionStart","selectionEnd","offset","#clampOffset","toStart","#createCloneRange","#styleClone","range","textNode","TextExpander$2","expander","match","menu","_b","position","caretRect","targetPosition","currentPosition","delta","currentStyle","item","beginning","remaining","detail","suffix","found","providers","provide","result","changeEvent","x","TextExpanderElement","keysAttr","keys","multiWordAttr","globalMultiWord","state","TextExpander","_getCustomEmojis","instance","masto","visibleEmojis","v1","customEmojis","visibleInPicker","searcher","Fuse","findAllMatches","getCustomEmojis","pmem","matchesArg","cacheKeyArg","keyArg","maxAge","role","className","windowMargin","observer","IntersectionObserver","entries","forEach","entry","isIntersecting","left","width","boundingClientRect","innerWidth","window","insetInlineStart","isRTL","observe","encodeHTML","str","replace","char","charCodeAt","onTrigger","props","ref","t","_useLingui","textExpanderRef","useRef","api","searcherRef","textExpanderTextRef","hasTextExpanderRef","useImperativeHandle","setStyle","current","Object","assign","activated","useEffect","then","catch","console","textExpander","showMore","html","results","emoji","shortcode","matched","resolve","searchResults","q","response","avatarStatic","name","displayName","username","acct","emojis","history","roles","displayNameWithEmoji","accountInstance","total","acc","cur","_t","error","handleValue","more","dataset","setTimeout","handleCommited","Event","handleActivate","handleChange","addEventListener","handleDeactivate","removeEventListener","expiryOptions","i18nDuration","ComposePoll","lang","poll","onInput","maxOptions","maxExpiration","minExpiration","maxCharactersPerOption","options","expiresIn","multiple","map","option","i","required","maxlength","placeholder","values","0","spellCheck","autocomplete","dir","onClick","splice","push","checked","_Trans","label","USERNAME_RE","MENTION_RE","HASHTAG_RE","SHORTCODE_RE_FRAGMENT","SCAN_RE","Intl","Segmenter","highlightText","Infinity","withinLimitHTML","exceedLimitHTML","htmlSegments","segment","index","escapeHTML","autoResizeTextarea","textarea","offsetHeight","scrollHeight","clientHeight","innerHeight","height","detectLangs","langDetector","langs","detect","slice","detectedLanguage","detectAll","__vitePreload","Textarea","forwardRef","setText","useState","textareaProps","ResizeObserver","getBoundingClientRect","minHeight","slowHighlightPerf","composeHighlightRef","throttleHighlightText","useThrottledCallback","innerHTML","cancel","start","end","Date","now","debouncedAutoDetectLanguage","useDebouncedCallback","dom","mark","cloneNode","text2","innerText","autoCapitalize","autoComplete","autoCorrect","rows","cols","hasTextExpander","lastLine","number","bullet","newBullet","preSpaces","postSpaces","pos","setRangeText","setSelectionRange","CUSTOM_EMOJIS_COUNT","EMOJI_SIZE_MIN","EMOJI_SIZE_MAX","EMOJI_SIZE_STEP","CustomEmojiButton","memo","showCode","addEdges","parent","currentTarget","closest","rect","selfRect","targetClassList","classList","buffer","add","remove","right","undefined","onPointerEnter","onFocus","staticUrl","srcSet","media","src","alt","loading","decoding","_Fragment","CustomEmojisList","onSelect","i18n","setMax","CustomEmojisModal","onClose","defaultSearchTerm","uiState","setUIState","customEmojisList","setCustomEmojis","recentlyUsedCustomEmojis","useMemo","store","account","get","log","customEmojisCatList","emojisCat","othersCat","category","setMatches","scrollableRef","onEmojiSizeDecrease","useCallback","emojiSize","setEmojiSize","Math","onEmojiSizeIncrease","newSize","onFind","onSelectEmoji","recentlyUsedCustomEmojis2","queueMicrotask","findIndex","unshift","emoji2","inputRef","Icon","Loader","matches","autocorrect","autocapitalize","icon","FilePickerInput","maxMediaAttachments","mediaAttachments","mediaFiles","poweredByGiphyURL","PHANPY_GIPHY_API_KEY","GIPHY_API_KEY","import","GIFS_PER_PAGE","GIFPickerModal","setResults","formRef","qRef","currentOffset","fetchGIFs","behavior","api_key","rating","limit","r","fetch","debouncedOnInput","onSubmit","abrupt","gif","alt_text","images","original","fixed_height_small","fixed_height_downsampled","fixed_height","webp","urlObj","strippedWebP","parse","strippedURL","origin","pathname","webpObj","url2","theURL","mp4","srcset","referrerpolicy","onLoad","pagination","PHANPY_IMG_ALT_API_URL","IMG_ALT_API_URL","scaleDimension","matrix","matrixLimit","scalingFactor","sqrt","newWidth","floor","newHeight","MediaAttachment","attachment","descriptionLimit","onDescriptionChange","onRemove","supportsEdit","supports","checkMaxError","imageSizeLimit","imageMatrixLimit","videoSizeLimit","videoMatrixLimit","videoFrameRateLimit","maxError","imageMatrix","setImageMatrix","videoMatrix","setVideoMatrix","setDescription","suffixType","subtype","debouncedOnDescriptionChange","setShowModal","timer","textareaRef","showModal","clearTimeout","descTextarea","toastRef","hideToast","maxErrorText","err","maxErrorToast","details","imageSize","imageSizeLimit2","prettyBytes","1","imageMatrix2","imageMatrixLimit2","3","videoSize","videoSizeLimit2","videoMatrix2","videoMatrixLimit2","naturalHeight","naturalWidth","playsinline","muted","onLoadedMetadata","videoHeight","videoWidth","showToast","Modal","controls","Menu2","portal","align","menuButton","MenuItem","duration","body","append","message","params","method","components","MentionModal","accounts","setAccounts","relationshipsMap","setRelationshipsMap","selectedIndex","setSelectedIndex","loadRelationships","relationships","fetchRelationships","loadAccounts","term","search","debouncedLoadAccounts","socialAddress","useHotkeys","selectedAccount","selectAccount","preventDefault","enableOnFormTags","useKey","ignoreEventWhen","listRef","selectedItem","block","relationship","avatarSize","showStats","supportedLanguagesMap","supportedLanguages","reduce","l","code","common","native","expirySeconds","oneDay","expiresInFromExpiresAt","expiresAt","find","s","DEFAULT_LANG","localeMatch","DateTimeFormat","resolvedOptions","locale","languages","usernameRegex","urlPlaceholder","countableText","inputText","LF","mem","ListFormat","ADD_LABELS","camera","customEmoji","sensitive","scheduledPost","DEFAULT_SCHEDULED_AT","MIN_SCHEDULED_AT","Compose","replyToStatus","editStatus","draftStatus","standalone","hasOpener","rtf","RTF","lf","UID","uid","currentAccount","currentAccountInfo","charactersReservedPerUrl","supportedImagesVideosTypes","spoilerTextRef","visibility","prevLanguage","language","sensitiveMedia","setLanguage","prefs","oninputTextarea","scheduledAt","getPreferences","focusTextarea","dispatchEvent","insertTextAtCursor","targetElement","textAfterInsert","textBeforeInsert","spaceAfterInsert","lastFocusedFieldRef","lastFocusedEmojiFieldRef","focusLastFocusedField","composeContainerRef","handleFocus","isFormElement","composeContainer","spoilerText","visibility2","language2","sensitive2","m","allMentions","setSensitive","poll2","statusSource","mediaAttachments2","setVisibility","composablePoll","sensitiveMedia2","scheduledAt2","status","o","setPoll","setScheduledAt","setSensitiveMedia","beforeUnloadCopy","snapStates","canClose","composerState","hasValue","hasMediaAttachments","isSelf","hasOnlyAcct","confirmClose","hasIDMediaAttachments","handleBeforeUnload","sameWithSource","getCharCount","updateCharCount","supportsCloseWatcher","yes","escDownRef","modals","hasOnlyComposer","count","draftKey","enabled","backgroundDraft","metaKey","hasModal","useCloseWatcher","prevBackgroundDraft","db","saveUnsavedDraft","handleItems","unsupportedFiles","i2","items","_i18n","drafts","set","updatedAt","f","useInterval","clipboardData","alert","allowedFiles","2","handleDragover","setShowMentionPicker","showEmoji2Picker","showGIFPicker","contentTranslationHideLanguages","autoDetectedLanguages","topLanguages","restLanguages","commonB","codeA","commonA","codeB","onMinimize","mediaButtonDisabled","showMentionPicker","topSupportedLanguages","settings","showAddButton","BUTTON_WIDTH","addSubToolbarRef","sort","truncated","scrollWidth","setShowAddButton","overTruncated","showScheduledAt","onScheduledAtClick","showPollButton","pollButtonDisabled","querySelector","highlightLanguageField","confirmText","passData","__STATES__","Status","opener","replyToStatusMonthsAgo","formData","opacity","params2","res","mediaPromises","hasNoDescriptions","removeNullUndefined","allSettled","newStatus","media_attributes","in_reply_to_id","saveStatus","requestInit","publishing","action","fileID","newAttachments","_2","j","setAutoDetectedLanguages","open","cwButtonDisabled","onCWButtonClick","MenuDivider","setShowEmoji2Picker","setShowGIFPicker","scheduledAtButtonDisabled","onPollButtonClick","composerGIFPicker","commonText","localeCode2Text","restSupportedLanguages","visibilityText","fallback","showCommon","emojiShortcode","theToast","blob","newMediaAttachments","obj"],"ignoreList":[0,1,2,6,7],"sources":["../../node_modules/ansi-regex/index.js","../../node_modules/strip-ansi/index.js","../../node_modules/string-length/index.js","../../src/utils/url-regex.js","../../src/components/camera-capture-input.jsx","../../src/components/char-count-meter.jsx","../../node_modules/@github/combobox-nav/dist/index.js","../../node_modules/@github/text-expander-element/dist/index.js","../../src/utils/custom-emojis.js","../../src/components/text-expander.jsx","../../src/components/compose-poll.jsx","../../src/components/compose-textarea.jsx","../../src/components/custom-emojis-modal.jsx","../../src/components/file-picker-input.jsx","../../src/assets/powered-by-giphy.svg","../../src/components/gif-picker-modal.jsx","../../src/components/media-attachment.jsx","../../src/components/mention-modal.jsx","../../src/components/compose.jsx"],"sourcesContent":["export default function ansiRegex({onlyFirst = false} = {}) {\n\tconst pattern = [\n\t '[\\\\u001B\\\\u009B][[\\\\]()#;?]*(?:(?:(?:(?:;[-a-zA-Z\\\\d\\\\/#&.:=?%@~_]+)*|[a-zA-Z\\\\d]+(?:;[-a-zA-Z\\\\d\\\\/#&.:=?%@~_]*)*)?\\\\u0007)',\n\t\t'(?:(?:\\\\d{1,4}(?:;\\\\d{0,4})*)?[\\\\dA-PR-TZcf-ntqry=><~]))'\n\t].join('|');\n\n\treturn new RegExp(pattern, onlyFirst ? undefined : 'g');\n}\n","import ansiRegex from 'ansi-regex';\n\nconst regex = ansiRegex();\n\nexport default function stripAnsi(string) {\n\tif (typeof string !== 'string') {\n\t\tthrow new TypeError(`Expected a \\`string\\`, got \\`${typeof string}\\``);\n\t}\n\n\t// Even though the regex is global, we don't need to reset the `.lastIndex`\n\t// because unlike `.exec()` and `.test()`, `.replace()` does it automatically\n\t// and doing it manually has a performance penalty.\n\treturn string.replace(regex, '');\n}\n","import stripAnsi from 'strip-ansi';\n\nconst segmenter = new Intl.Segmenter();\n\nexport default function stringLength(string, {countAnsiEscapeCodes = false} = {}) {\n\tif (string === '') {\n\t\treturn 0;\n\t}\n\n\tif (!countAnsiEscapeCodes) {\n\t\tstring = stripAnsi(string);\n\t}\n\n\tif (string === '') {\n\t\treturn 0;\n\t}\n\n\tlet length = 0;\n\n\tfor (const _ of segmenter.segment(string)) { // eslint-disable-line no-unused-vars\n\t\tlength++;\n\t}\n\n\treturn length;\n}\n","import urlRegex from '../data/url-regex';\n\nconst urlRegexObj = new RegExp(urlRegex.source, urlRegex.flags);\n\nexport default urlRegexObj;\n","const isMobileSafari =\n /iPad|iPhone|iPod/.test(navigator.userAgent) &&\n /^((?!chrome|android).)*safari/i.test(navigator.userAgent);\n\nfunction CameraCaptureInput({\n hidden,\n disabled = false,\n supportedMimeTypes,\n setMediaAttachments,\n}) {\n // If not Mobile Safari, only apply image/*\n // Chrome Android doesn't show the camera if image and video combined\n // It also can't switch between photo and video mode like iOS/Safari\n const filteredSupportedMimeTypes = isMobileSafari\n ? supportedMimeTypes\n : supportedMimeTypes?.filter((mimeType) => !/^image\\//i.test(mimeType));\n\n return (\n <input\n type=\"file\"\n hidden={hidden}\n accept={filteredSupportedMimeTypes?.join(',')}\n capture=\"environment\"\n disabled={disabled}\n onChange={(e) => {\n const files = e.target.files;\n if (!files) return;\n const mediaFile = Array.from(files)[0];\n if (!mediaFile) return;\n setMediaAttachments((attachments) => [\n ...attachments,\n {\n file: mediaFile,\n type: mediaFile.type,\n size: mediaFile.size,\n url: URL.createObjectURL(mediaFile),\n id: null, // indicate uploaded state\n description: null,\n },\n ]);\n e.target.value = null;\n }}\n />\n );\n}\n\nexport const supportsCameraCapture = (() => {\n const input = document.createElement('input');\n return 'capture' in input;\n})();\n\nexport default CameraCaptureInput;\n","import { useSnapshot } from 'valtio';\n\nimport states from '../utils/states';\n\nfunction CharCountMeter({ maxCharacters = 500, hidden }) {\n const snapStates = useSnapshot(states);\n const charCount = snapStates.composerCharacterCount;\n const leftChars = maxCharacters - charCount;\n if (hidden) {\n return <span class=\"char-counter\" hidden />;\n }\n return (\n <span\n class=\"char-counter\"\n title={`${leftChars}/${maxCharacters}`}\n style={{\n '--percentage': (charCount / maxCharacters) * 100,\n }}\n >\n <meter\n class={`${\n leftChars <= -10\n ? 'explode'\n : leftChars <= 0\n ? 'danger'\n : leftChars <= 20\n ? 'warning'\n : ''\n }`}\n value={charCount}\n max={maxCharacters}\n />\n <span class=\"counter\">{leftChars}</span>\n </span>\n );\n}\n\nexport default CharCountMeter;\n","export default class Combobox {\n constructor(input, list, { tabInsertsSuggestions, defaultFirstOption } = {}) {\n this.input = input;\n this.list = list;\n this.tabInsertsSuggestions = tabInsertsSuggestions !== null && tabInsertsSuggestions !== void 0 ? tabInsertsSuggestions : true;\n this.defaultFirstOption = defaultFirstOption !== null && defaultFirstOption !== void 0 ? defaultFirstOption : false;\n this.isComposing = false;\n if (!list.id) {\n list.id = `combobox-${Math.random().toString().slice(2, 6)}`;\n }\n this.ctrlBindings = !!navigator.userAgent.match(/Macintosh/);\n this.keyboardEventHandler = event => keyboardBindings(event, this);\n this.compositionEventHandler = event => trackComposition(event, this);\n this.inputHandler = this.clearSelection.bind(this);\n input.setAttribute('role', 'combobox');\n input.setAttribute('aria-controls', list.id);\n input.setAttribute('aria-expanded', 'false');\n input.setAttribute('aria-autocomplete', 'list');\n input.setAttribute('aria-haspopup', 'listbox');\n }\n destroy() {\n this.clearSelection();\n this.stop();\n this.input.removeAttribute('role');\n this.input.removeAttribute('aria-controls');\n this.input.removeAttribute('aria-expanded');\n this.input.removeAttribute('aria-autocomplete');\n this.input.removeAttribute('aria-haspopup');\n }\n start() {\n this.input.setAttribute('aria-expanded', 'true');\n this.input.addEventListener('compositionstart', this.compositionEventHandler);\n this.input.addEventListener('compositionend', this.compositionEventHandler);\n this.input.addEventListener('input', this.inputHandler);\n this.input.addEventListener('keydown', this.keyboardEventHandler);\n this.list.addEventListener('click', commitWithElement);\n this.indicateDefaultOption();\n }\n stop() {\n this.clearSelection();\n this.input.setAttribute('aria-expanded', 'false');\n this.input.removeEventListener('compositionstart', this.compositionEventHandler);\n this.input.removeEventListener('compositionend', this.compositionEventHandler);\n this.input.removeEventListener('input', this.inputHandler);\n this.input.removeEventListener('keydown', this.keyboardEventHandler);\n this.list.removeEventListener('click', commitWithElement);\n }\n indicateDefaultOption() {\n var _a;\n if (this.defaultFirstOption) {\n (_a = Array.from(this.list.querySelectorAll('[role=\"option\"]:not([aria-disabled=\"true\"])'))\n .filter(visible)[0]) === null || _a === void 0 ? void 0 : _a.setAttribute('data-combobox-option-default', 'true');\n }\n }\n navigate(indexDiff = 1) {\n const focusEl = Array.from(this.list.querySelectorAll('[aria-selected=\"true\"]')).filter(visible)[0];\n const els = Array.from(this.list.querySelectorAll('[role=\"option\"]')).filter(visible);\n const focusIndex = els.indexOf(focusEl);\n if ((focusIndex === els.length - 1 && indexDiff === 1) || (focusIndex === 0 && indexDiff === -1)) {\n this.clearSelection();\n this.input.focus();\n return;\n }\n let indexOfItem = indexDiff === 1 ? 0 : els.length - 1;\n if (focusEl && focusIndex >= 0) {\n const newIndex = focusIndex + indexDiff;\n if (newIndex >= 0 && newIndex < els.length)\n indexOfItem = newIndex;\n }\n const target = els[indexOfItem];\n if (!target)\n return;\n for (const el of els) {\n el.removeAttribute('data-combobox-option-default');\n if (target === el) {\n this.input.setAttribute('aria-activedescendant', target.id);\n target.setAttribute('aria-selected', 'true');\n scrollTo(this.list, target);\n }\n else {\n el.removeAttribute('aria-selected');\n }\n }\n }\n clearSelection() {\n this.input.removeAttribute('aria-activedescendant');\n for (const el of this.list.querySelectorAll('[aria-selected=\"true\"]')) {\n el.removeAttribute('aria-selected');\n }\n this.indicateDefaultOption();\n }\n}\nfunction keyboardBindings(event, combobox) {\n if (event.shiftKey || event.metaKey || event.altKey)\n return;\n if (!combobox.ctrlBindings && event.ctrlKey)\n return;\n if (combobox.isComposing)\n return;\n switch (event.key) {\n case 'Enter':\n if (commit(combobox.input, combobox.list)) {\n event.preventDefault();\n }\n break;\n case 'Tab':\n if (combobox.tabInsertsSuggestions && commit(combobox.input, combobox.list)) {\n event.preventDefault();\n }\n break;\n case 'Escape':\n combobox.clearSelection();\n break;\n case 'ArrowDown':\n combobox.navigate(1);\n event.preventDefault();\n break;\n case 'ArrowUp':\n combobox.navigate(-1);\n event.preventDefault();\n break;\n case 'n':\n if (combobox.ctrlBindings && event.ctrlKey) {\n combobox.navigate(1);\n event.preventDefault();\n }\n break;\n case 'p':\n if (combobox.ctrlBindings && event.ctrlKey) {\n combobox.navigate(-1);\n event.preventDefault();\n }\n break;\n default:\n if (event.ctrlKey)\n break;\n combobox.clearSelection();\n }\n}\nfunction commitWithElement(event) {\n if (!(event.target instanceof Element))\n return;\n const target = event.target.closest('[role=\"option\"]');\n if (!target)\n return;\n if (target.getAttribute('aria-disabled') === 'true')\n return;\n fireCommitEvent(target);\n}\nfunction commit(input, list) {\n const target = list.querySelector('[aria-selected=\"true\"], [data-combobox-option-default=\"true\"]');\n if (!target)\n return false;\n if (target.getAttribute('aria-disabled') === 'true')\n return true;\n target.click();\n return true;\n}\nfunction fireCommitEvent(target) {\n target.dispatchEvent(new CustomEvent('combobox-commit', { bubbles: true }));\n}\nfunction visible(el) {\n return (!el.hidden &&\n !(el instanceof HTMLInputElement && el.type === 'hidden') &&\n (el.offsetWidth > 0 || el.offsetHeight > 0));\n}\nfunction trackComposition(event, combobox) {\n combobox.isComposing = event.type === 'compositionstart';\n const list = document.getElementById(combobox.input.getAttribute('aria-controls') || '');\n if (!list)\n return;\n combobox.clearSelection();\n}\nfunction scrollTo(container, target) {\n if (!inViewport(container, target)) {\n container.scrollTop = target.offsetTop;\n }\n}\nfunction inViewport(container, element) {\n const scrollTop = container.scrollTop;\n const containerBottom = scrollTop + container.clientHeight;\n const top = element.offsetTop;\n const bottom = top + element.clientHeight;\n return top >= scrollTop && bottom <= containerBottom;\n}\n","import Combobox from '@github/combobox-nav';\n\nconst boundary = /\\s|\\(|\\[/;\nfunction query(text, key, cursor, { multiWord, lookBackIndex, lastMatchPosition } = {\n multiWord: false,\n lookBackIndex: 0,\n lastMatchPosition: null\n}) {\n let keyIndex = text.lastIndexOf(key, cursor - 1);\n if (keyIndex === -1)\n return;\n if (keyIndex < lookBackIndex)\n return;\n if (multiWord) {\n if (lastMatchPosition != null) {\n if (lastMatchPosition === keyIndex)\n return;\n keyIndex = lastMatchPosition - key.length;\n }\n const charAfterKey = text[keyIndex + 1];\n if (charAfterKey === ' ' && cursor >= keyIndex + key.length + 1)\n return;\n const newLineIndex = text.lastIndexOf('\\n', cursor - 1);\n if (newLineIndex > keyIndex)\n return;\n const dotIndex = text.lastIndexOf('.', cursor - 1);\n if (dotIndex > keyIndex)\n return;\n }\n else {\n const spaceIndex = text.lastIndexOf(' ', cursor - 1);\n if (spaceIndex > keyIndex)\n return;\n }\n const pre = text[keyIndex - 1];\n if (pre && !boundary.test(pre))\n return;\n const queryString = text.substring(keyIndex + key.length, cursor);\n return {\n text: queryString,\n position: keyIndex + key.length\n };\n}\n\nclass InputStyleCloneUpdateEvent extends Event {\n constructor() {\n super(\"update\");\n }\n}\nconst CloneRegistry = new WeakMap();\n/**\n * Creates an element that exactly matches an input pixel-for-pixel and automatically stays in sync with it. This\n * is a non-interactive overlay on to the input and can be used to affect the visual appearance of the input\n * without modifying its behavior. The clone element is hidden by default.\n *\n * This lower level API powers the `InputRange` but provides more advanced functionality including event updates.\n *\n * Emits `update` events whenever anything is recalculated: when the layout changes, when the user scrolls, when the\n * input is updated, etc. This event may be emitted more than once per change.\n *\n * @note There may be cases in which the clone cannot observe changes to the input and fails to automatically update.\n * For example, if the `value` property on the input is written to directly, no `input` event is emitted by the input\n * and the clone does not automatically update. In these cases, `forceUpdate` can be used to manually trigger an update.\n */\n// PRIOR ART: This approach was adapted from the following MIT-licensed sources:\n// - primer/react (Copyright (c) 2018 GitHub, Inc.): https://github.com/primer/react/blob/a0db832302702b869aa22b0c4049ad9305ef631f/src/drafts/utils/character-coordinates.ts\n// - component/textarea-caret-position (Copyright (c) 2015 Jonathan Ong me@jongleberry.com): https://github.com/component/textarea-caret-position/blob/b5db7a7e47dd149c2a66276183c69234e4dabe30/index.js\n// - koddsson/textarea-caret-position (Copyright (c) 2015 Jonathan Ong me@jongleberry.com): https://github.com/koddsson/textarea-caret-position/blob/eba40ec8488eed4d77815f109af22e1d9c0751d3/index.js\nclass InputStyleClone extends EventTarget {\n #styleObserver = new MutationObserver(() => this.#updateStyles());\n #resizeObserver = new ResizeObserver(() => this.#requestUpdateLayout());\n // This class is unique in that it will prevent itself from getting garbage collected because of the subscribed\n // observers (if never detached). Because of this, we want to avoid preventing the existence of this class from also\n // preventing the garbage collection of the associated input. This also allows us to automatically detach if the\n // input gets collected.\n #inputRef;\n #container = document.createElement(\"div\");\n #cloneElement = document.createElement(\"div\");\n /**\n * Get the clone for an input, reusing an existing one if available. This avoids creating unecessary clones, which\n * have a performance cost due to their high-frequency event-based updates. Because these elements are shared, they\n * should be mutated with caution. If you're planning to mutate the clone, consider constructing a new one instead.\n *\n * Upon initial creation the clone element will automatically be inserted into the DOM and begin observing the\n * linked input.\n * @param input The target input to clone.\n */\n static for(input) {\n let clone = CloneRegistry.get(input);\n if (!clone) {\n clone = new InputStyleClone(input);\n CloneRegistry.set(input, clone);\n }\n return clone;\n }\n /**\n * Connect this instance to a target input element and insert this instance into the DOM in the correct location.\n *\n * NOTE: calling the static `for` method is usually preferable as it will reuse an existing clone if available.\n * However, if reusing clones is problematic (ie, if the clone needs to be mutated), a clone can be constructed\n * directly with `new InputStyleClone(target)`.\n */\n constructor(input) {\n super();\n this.#inputRef = new WeakRef(input);\n // We want position:absolute so it doesn't take space in the layout, but that doesn't work with display:table-cell\n // used in the HTMLInputElement approach. So we need a wrapper.\n this.#container.style.position = \"absolute\";\n this.#container.style.pointerEvents = \"none\";\n this.#container.setAttribute(\"aria-hidden\", \"true\");\n this.#container.appendChild(this.#cloneElement);\n this.#cloneElement.style.pointerEvents = \"none\";\n this.#cloneElement.style.userSelect = \"none\";\n this.#cloneElement.style.overflow = \"hidden\";\n this.#cloneElement.style.display = \"block\";\n // Important not to use display:none which would not render the content at all\n this.#cloneElement.style.visibility = \"hidden\";\n if (input instanceof HTMLTextAreaElement) {\n this.#cloneElement.style.whiteSpace = \"pre-wrap\";\n this.#cloneElement.style.wordWrap = \"break-word\";\n }\n else {\n this.#cloneElement.style.whiteSpace = \"nowrap\";\n // text in single-line inputs is vertically centered\n this.#cloneElement.style.display = \"table-cell\";\n this.#cloneElement.style.verticalAlign = \"middle\";\n }\n input.after(this.#container);\n this.#updateStyles();\n this.#updateText();\n this.#styleObserver.observe(input, {\n attributeFilter: [\n \"style\",\n \"dir\", // users can right-click in some browsers to change the text direction dynamically\n ],\n });\n this.#resizeObserver.observe(input);\n document.addEventListener(\"scroll\", this.#onDocumentScrollOrResize, { capture: true });\n window.addEventListener(\"resize\", this.#onDocumentScrollOrResize, { capture: true });\n // capture so this happens first, so other things can respond to `input` events after this data updates\n input.addEventListener(\"input\", this.#onInput, { capture: true });\n }\n /** Get the clone element. */\n get element() {\n return this.#cloneElement;\n }\n /**\n * Force a recalculation. Will emit an `update` event. This is typically not needed unless the input has changed in\n * an unobservable way, eg by directly writing to the `value` property.\n */\n forceUpdate() {\n this.#updateStyles();\n this.#updateText();\n }\n disconnect() {\n this.#container?.remove();\n this.#styleObserver.disconnect();\n this.#resizeObserver.disconnect();\n document.removeEventListener(\"scroll\", this.#onDocumentScrollOrResize, { capture: true });\n window.removeEventListener(\"resize\", this.#onDocumentScrollOrResize, { capture: true });\n // Can't use `usingInput` here since that could infinitely recurse\n const input = this.#input;\n if (input) {\n input.removeEventListener(\"input\", this.#onInput, { capture: true });\n CloneRegistry.delete(input);\n }\n }\n // --- private ---\n get #input() {\n return this.#inputRef?.deref();\n }\n /** Perform `fn` using the `input` if it is still available. If not, clean up the clone instead. */\n #usingInput(fn) {\n const input = this.#input;\n if (!input)\n return this.disconnect();\n return fn(input);\n }\n /** Current relative x-adjustment in pixels, executed via CSS transform. */\n #xOffset = 0;\n /** Current relative y-adjustment in pixels, executed via CSS transform. */\n #yOffset = 0;\n /**\n * Update only geometric properties without recalculating styles. Typically call `#requestUpdateLayout` instead to\n * only update once per animation frame.\n */\n #updateLayout() {\n // This runs often, so keep it as fast as possible! Avoid all unecessary updates.\n this.#usingInput((input) => {\n const inputStyle = window.getComputedStyle(input);\n this.#cloneElement.style.height = inputStyle.height;\n this.#cloneElement.style.width = inputStyle.width;\n // Immediately re-adjust for browser inconsistencies in scrollbar handling, if necessary\n if (input.clientHeight !== this.#cloneElement.clientHeight)\n this.#cloneElement.style.height = `calc(${inputStyle.height} + ${input.clientHeight - this.#cloneElement.clientHeight}px)`;\n if (input.clientWidth !== this.#cloneElement.clientWidth)\n this.#cloneElement.style.width = `calc(${inputStyle.width} + ${input.clientWidth - this.#cloneElement.clientWidth}px)`;\n // Position on top of the input\n const inputRect = input.getBoundingClientRect();\n const cloneRect = this.#cloneElement.getBoundingClientRect();\n this.#xOffset = this.#xOffset + inputRect.left - cloneRect.left;\n this.#yOffset = this.#yOffset + inputRect.top - cloneRect.top;\n this.#cloneElement.style.transform = `translate(${this.#xOffset}px, ${this.#yOffset}px)`;\n this.#cloneElement.scrollTop = input.scrollTop;\n this.#cloneElement.scrollLeft = input.scrollLeft;\n this.dispatchEvent(new InputStyleCloneUpdateEvent());\n });\n }\n #isLayoutUpdating = false;\n /** Request a layout update. Will only happen once per animation frame, to avoid unecessary updates. */\n #requestUpdateLayout() {\n if (this.#isLayoutUpdating)\n return;\n this.#isLayoutUpdating = true;\n requestAnimationFrame(() => {\n this.#updateLayout();\n this.#isLayoutUpdating = false;\n });\n }\n /** Update the styles of the clone based on the styles of the input, then request a layout update. */\n #updateStyles() {\n this.#usingInput((input) => {\n const inputStyle = window.getComputedStyle(input);\n for (const prop of propertiesToCopy)\n this.#cloneElement.style[prop] = inputStyle[prop];\n this.#requestUpdateLayout();\n });\n }\n /**\n * Update the text content of the clone based on the text content of the input. Triggers a layout update in case the\n * text update caused scrolling.\n */\n #updateText() {\n this.#usingInput((input) => {\n this.#cloneElement.textContent = input.value;\n // This is often unecessary on a pure text update, but text updates could potentially cause layout updates like\n // scrolling or resizing. And we run the update on _every frame_ when scrolling, so this isn't that expensive.\n // We don't requestUpdateLayout here because this one should happen synchronously, so that clients can react\n // within their own `input` event handlers.\n this.#updateLayout();\n });\n }\n #onInput = () => this.#updateText();\n #onDocumentScrollOrResize = (event) => {\n this.#usingInput((input) => {\n if (event.target === document ||\n event.target === window ||\n (event.target instanceof Node && event.target.contains(input)))\n this.#requestUpdateLayout();\n });\n };\n}\n// Note that some browsers, such as Firefox, do not concatenate properties\n// into their shorthand (e.g. padding-top, padding-bottom etc. -> padding),\n// so we have to list every single property explicitly.\nconst propertiesToCopy = [\n // RTL / vertical writing modes support:\n \"direction\",\n \"writingMode\",\n \"unicodeBidi\",\n \"textOrientation\",\n \"boxSizing\",\n \"borderTopWidth\",\n \"borderRightWidth\",\n \"borderBottomWidth\",\n \"borderLeftWidth\",\n \"borderStyle\",\n \"paddingTop\",\n \"paddingRight\",\n \"paddingBottom\",\n \"paddingLeft\",\n // https://developer.mozilla.org/en-US/docs/Web/CSS/font\n \"fontStyle\",\n \"fontVariant\",\n \"fontWeight\",\n \"fontStretch\",\n \"fontSize\",\n \"fontSizeAdjust\",\n \"lineHeight\",\n \"fontFamily\",\n \"textAlign\",\n \"textTransform\",\n \"textIndent\",\n \"textDecoration\",\n \"letterSpacing\",\n \"wordSpacing\",\n \"tabSize\",\n \"MozTabSize\",\n];\n\nclass InputRange {\n #inputElement;\n #startOffset;\n #endOffset;\n /**\n * Construct a new `InputRange`.\n * @param element The target input element that contains the content for the range.\n * @param startOffset The inclusive 0-based start index for the range. Will be adjusted to fit in the input contents.\n * @param endOffset The exclusive 0-based end index for the range. Will be adjusted to fit in the input contents.\n */\n constructor(element, startOffset = 0, endOffset = startOffset) {\n this.#inputElement = element;\n this.#startOffset = startOffset;\n this.#endOffset = endOffset;\n }\n /**\n * Create a new range from the current user selection. If the input is not focused, the range will just be the start\n * of the input (offsets `0` to `0`).\n *\n * This can be used to get the caret coordinates: if the resulting range is `collapsed`, the location of the\n * `getBoundingClientRect` will be the location of the caret caret (note, however, that the width will be `0` in\n * this case).\n */\n static fromSelection(input) {\n const { selectionStart, selectionEnd } = input;\n return new InputRange(input, selectionStart ?? undefined, selectionEnd ?? undefined);\n }\n /** Returns true if the start is equal to the end of this range. */\n get collapsed() {\n return this.startOffset === this.endOffset;\n }\n /** Always returns the containing input element. */\n get commonAncestorContainer() {\n return this.#inputElement;\n }\n /** Always returns the containing input element. */\n get endContainer() {\n return this.#inputElement;\n }\n /** Always returns the containing input element. */\n get startContainer() {\n return this.#inputElement;\n }\n get startOffset() {\n return this.#startOffset;\n }\n get endOffset() {\n return this.#endOffset;\n }\n /** Update the inclusive start offset. Will be adjusted to fit within the content size. */\n setStartOffset(offset) {\n this.#startOffset = this.#clampOffset(offset);\n }\n /** Update the exclusive end offset. Will be adjusted to fit within the content size. */\n setEndOffset(offset) {\n this.#endOffset = this.#clampOffset(offset);\n }\n /**\n * Collapse this range to one side.\n * @param toStart If `true`, will collapse to the start side. Otherwise, will collapse to the end.\n */\n collapse(toStart = false) {\n if (toStart)\n this.setEndOffset(this.startOffset);\n else\n this.setStartOffset(this.endOffset);\n }\n /** Returns a `DocumentFragment` containing a new `Text` node containing the content in the range. */\n cloneContents() {\n return this.#createCloneRange().cloneContents();\n }\n /** Create a copy of this range. */\n cloneRange() {\n return new InputRange(this.#inputElement, this.startOffset, this.endOffset);\n }\n /**\n * Obtain one rect that contains the entire contents of the range. If the range spans multiple lines, this box will\n * contain all pieces of the range but may also contain some space outside the range.\n * @see https://iansan5653.github.io/dom-input-range/demos/playground/\n */\n getBoundingClientRect() {\n return this.#createCloneRange().getBoundingClientRect();\n }\n /**\n * Obtain the rects that contain contents of this range. If the range spans multiple lines, there will be multiple\n * bounding boxes. These boxes can be used, for example, to draw a highlight over the range.\n * @see https://iansan5653.github.io/dom-input-range/demos/playground/\n */\n getClientRects() {\n return this.#createCloneRange().getClientRects();\n }\n /** Get the contents of the range as a string. */\n toString() {\n return this.#createCloneRange().toString();\n }\n /**\n * Get the underlying `InputStyleClone` instance powering these calculations. This can be used to listen for\n * updates to trigger layout recalculation.\n */\n getStyleClone() {\n return this.#styleClone;\n }\n // --- private ---\n get #styleClone() {\n return InputStyleClone.for(this.#inputElement);\n }\n get #cloneElement() {\n return this.#styleClone;\n }\n #clampOffset(offset) {\n return Math.max(0, Math.min(offset, this.#inputElement.value.length));\n }\n #createCloneRange() {\n // It's tempting to create a single Range and reuse it across the lifetime of the class. However, this wouldn't be\n // accurate because the contents of the input can change and the contents of the range would become stale. So we\n // must create a new range every time we need it.\n const range = document.createRange();\n const textNode = this.#cloneElement.element.childNodes[0];\n if (textNode) {\n range.setStart(textNode, this.startOffset);\n range.setEnd(textNode, this.endOffset);\n }\n return range;\n }\n}\n\nconst states = new WeakMap();\nclass TextExpander {\n constructor(expander, input) {\n this.expander = expander;\n this.input = input;\n this.combobox = null;\n this.menu = null;\n this.match = null;\n this.justPasted = false;\n this.lookBackIndex = 0;\n this.oninput = this.onInput.bind(this);\n this.onpaste = this.onPaste.bind(this);\n this.onkeydown = this.onKeydown.bind(this);\n this.oncommit = this.onCommit.bind(this);\n this.onmousedown = this.onMousedown.bind(this);\n this.onblur = this.onBlur.bind(this);\n this.interactingWithList = false;\n input.addEventListener('paste', this.onpaste);\n input.addEventListener('input', this.oninput);\n input.addEventListener('keydown', this.onkeydown);\n input.addEventListener('blur', this.onblur);\n }\n destroy() {\n this.input.removeEventListener('paste', this.onpaste);\n this.input.removeEventListener('input', this.oninput);\n this.input.removeEventListener('keydown', this.onkeydown);\n this.input.removeEventListener('blur', this.onblur);\n }\n dismissMenu() {\n if (this.deactivate()) {\n this.lookBackIndex = this.input.selectionEnd || this.lookBackIndex;\n }\n }\n activate(match, menu) {\n var _a, _b;\n if (this.input !== document.activeElement && this.input !== ((_b = (_a = document.activeElement) === null || _a === void 0 ? void 0 : _a.shadowRoot) === null || _b === void 0 ? void 0 : _b.activeElement)) {\n return;\n }\n this.deactivate();\n this.menu = menu;\n if (!menu.id)\n menu.id = `text-expander-${Math.floor(Math.random() * 100000).toString()}`;\n this.expander.append(menu);\n this.combobox = new Combobox(this.input, menu);\n this.expander.dispatchEvent(new Event('text-expander-activate'));\n this.positionMenu(menu, match.position);\n this.combobox.start();\n menu.addEventListener('combobox-commit', this.oncommit);\n menu.addEventListener('mousedown', this.onmousedown);\n this.combobox.navigate(1);\n }\n positionMenu(menu, position) {\n const caretRect = new InputRange(this.input, position).getBoundingClientRect();\n const targetPosition = { left: caretRect.left, top: caretRect.top + caretRect.height };\n const currentPosition = menu.getBoundingClientRect();\n const delta = {\n left: targetPosition.left - currentPosition.left,\n top: targetPosition.top - currentPosition.top\n };\n if (delta.left !== 0 || delta.top !== 0) {\n const currentStyle = getComputedStyle(menu);\n menu.style.left = currentStyle.left ? `calc(${currentStyle.left} + ${delta.left}px)` : `${delta.left}px`;\n menu.style.top = currentStyle.top ? `calc(${currentStyle.top} + ${delta.top}px)` : `${delta.top}px`;\n }\n }\n deactivate() {\n const menu = this.menu;\n if (!menu || !this.combobox)\n return false;\n this.expander.dispatchEvent(new Event('text-expander-deactivate'));\n this.menu = null;\n menu.removeEventListener('combobox-commit', this.oncommit);\n menu.removeEventListener('mousedown', this.onmousedown);\n this.combobox.destroy();\n this.combobox = null;\n menu.remove();\n return true;\n }\n onCommit({ target }) {\n var _a;\n const item = target;\n if (!(item instanceof HTMLElement))\n return;\n if (!this.combobox)\n return;\n const match = this.match;\n if (!match)\n return;\n const beginning = this.input.value.substring(0, match.position - match.key.length);\n const remaining = this.input.value.substring(match.position + match.text.length);\n const detail = { item, key: match.key, value: null, continue: false };\n const canceled = !this.expander.dispatchEvent(new CustomEvent('text-expander-value', { cancelable: true, detail }));\n if (canceled)\n return;\n if (!detail.value)\n return;\n let suffix = (_a = this.expander.getAttribute('suffix')) !== null && _a !== void 0 ? _a : ' ';\n if (detail.continue) {\n suffix = '';\n }\n const value = `${detail.value}${suffix}`;\n this.input.value = beginning + value + remaining;\n const cursor = beginning.length + value.length;\n this.deactivate();\n this.input.focus({\n preventScroll: true\n });\n this.input.selectionStart = cursor;\n this.input.selectionEnd = cursor;\n if (!detail.continue) {\n this.lookBackIndex = cursor;\n this.match = null;\n }\n this.expander.dispatchEvent(new CustomEvent('text-expander-committed', { cancelable: false, detail: { input: this.input } }));\n }\n onBlur() {\n if (this.interactingWithList) {\n this.interactingWithList = false;\n return;\n }\n this.deactivate();\n }\n onPaste() {\n this.justPasted = true;\n }\n async onInput() {\n if (this.justPasted) {\n this.justPasted = false;\n return;\n }\n const match = this.findMatch();\n if (match) {\n this.match = match;\n const menu = await this.notifyProviders(match);\n if (!this.match)\n return;\n if (menu) {\n this.activate(match, menu);\n }\n else {\n this.deactivate();\n }\n }\n else {\n this.match = null;\n this.deactivate();\n }\n }\n findMatch() {\n const cursor = this.input.selectionEnd || 0;\n const text = this.input.value;\n if (cursor <= this.lookBackIndex) {\n this.lookBackIndex = cursor - 1;\n }\n for (const { key, multiWord } of this.expander.keys) {\n const found = query(text, key, cursor, {\n multiWord,\n lookBackIndex: this.lookBackIndex,\n lastMatchPosition: this.match ? this.match.position : null\n });\n if (found) {\n return { text: found.text, key, position: found.position };\n }\n }\n }\n async notifyProviders(match) {\n const providers = [];\n const provide = (result) => providers.push(result);\n const changeEvent = new CustomEvent('text-expander-change', {\n cancelable: true,\n detail: { provide, text: match.text, key: match.key }\n });\n const canceled = !this.expander.dispatchEvent(changeEvent);\n if (canceled)\n return;\n const all = await Promise.all(providers);\n const fragments = all.filter(x => x.matched).map(x => x.fragment);\n return fragments[0];\n }\n onMousedown() {\n this.interactingWithList = true;\n }\n onKeydown(event) {\n if (event.key === 'Escape') {\n this.match = null;\n if (this.deactivate()) {\n this.lookBackIndex = this.input.selectionEnd || this.lookBackIndex;\n event.stopImmediatePropagation();\n event.preventDefault();\n }\n }\n }\n}\nclass TextExpanderElement extends HTMLElement {\n get keys() {\n const keysAttr = this.getAttribute('keys');\n const keys = keysAttr ? keysAttr.split(' ') : [];\n const multiWordAttr = this.getAttribute('multiword');\n const multiWord = multiWordAttr ? multiWordAttr.split(' ') : [];\n const globalMultiWord = multiWord.length === 0 && this.hasAttribute('multiword');\n return keys.map(key => ({ key, multiWord: globalMultiWord || multiWord.includes(key) }));\n }\n set keys(value) {\n this.setAttribute('keys', value);\n }\n connectedCallback() {\n const input = this.querySelector('input[type=\"text\"], textarea');\n if (!(input instanceof HTMLInputElement || input instanceof HTMLTextAreaElement))\n return;\n const state = new TextExpander(this, input);\n states.set(this, state);\n }\n disconnectedCallback() {\n const state = states.get(this);\n if (!state)\n return;\n state.destroy();\n states.delete(this);\n }\n dismiss() {\n const state = states.get(this);\n if (!state)\n return;\n state.dismissMenu();\n }\n}\n\nif (!window.customElements.get('text-expander')) {\n window.TextExpanderElement = TextExpanderElement;\n window.customElements.define('text-expander', TextExpanderElement);\n}\n\nexport { TextExpanderElement as default };\n","import Fuse from 'fuse.js';\n\nimport pmem from './pmem';\n\nasync function _getCustomEmojis(instance, masto) {\n const emojis = await masto.v1.customEmojis.list();\n const visibleEmojis = emojis.filter((e) => e.visibleInPicker);\n const searcher = new Fuse(visibleEmojis, {\n keys: ['shortcode'],\n findAllMatches: true,\n });\n return [visibleEmojis, searcher];\n}\n\nconst getCustomEmojis = pmem(_getCustomEmojis, {\n // Limit by time to reduce memory usage\n // Cached by instance\n matchesArg: (cacheKeyArg, keyArg) => cacheKeyArg.instance === keyArg.instance,\n maxAge: 30 * 60 * 1000, // 30 minutes\n});\n\nexport { getCustomEmojis, _getCustomEmojis };\nexport default getCustomEmojis;\n","import '@github/text-expander-element';\n\nimport { useLingui } from '@lingui/react/macro';\nimport { forwardRef, useImperativeHandle } from 'preact/compat';\nimport { useEffect, useRef } from 'preact/hooks';\n\nimport { api } from '../utils/api';\nimport getCustomEmojis from '../utils/custom-emojis';\nimport emojifyText from '../utils/emojify-text';\nimport getDomain from '../utils/get-domain';\nimport isRTL from '../utils/is-rtl';\nimport shortenNumber from '../utils/shorten-number';\n\nconst menu = document.createElement('ul');\nmenu.role = 'listbox';\nmenu.className = 'text-expander-menu';\n\n// Set IntersectionObserver on menu, reposition it because text-expander doesn't handle it\nconst windowMargin = 16;\nconst observer = new IntersectionObserver((entries) => {\n entries.forEach((entry) => {\n if (entry.isIntersecting) {\n const { left, width } = entry.boundingClientRect;\n const { innerWidth } = window;\n if (left + width > innerWidth) {\n const insetInlineStart = isRTL() ? 'right' : 'left';\n menu.style[insetInlineStart] = innerWidth - width - windowMargin + 'px';\n }\n }\n });\n});\nobserver.observe(menu);\n\nfunction encodeHTML(str) {\n return str.replace(/[&<>\"']/g, function (char) {\n return '&#' + char.charCodeAt(0) + ';';\n });\n}\n\nfunction TextExpander({ onTrigger = null, ...props }, ref) {\n const { t } = useLingui();\n const textExpanderRef = useRef();\n const { masto, instance } = api();\n const searcherRef = useRef();\n const textExpanderTextRef = useRef('');\n const hasTextExpanderRef = useRef(false);\n\n // Expose the activated state to parent components\n useImperativeHandle(ref, () => ({\n setStyle: (style) => {\n if (textExpanderRef.current) {\n Object.assign(textExpanderRef.current.style, style);\n }\n },\n activated: () => hasTextExpanderRef.current,\n }));\n\n // Setup emoji search if not already set up\n useEffect(() => {\n if (searcherRef.current) return; // Already set up\n\n getCustomEmojis(instance, masto)\n .then(([, searcher]) => {\n searcherRef.current = searcher;\n })\n .catch((e) => {\n console.error(e);\n });\n }, [instance, masto]);\n\n useEffect(() => {\n const textExpander = textExpanderRef.current;\n if (!textExpander) return;\n\n const handleChange = (e) => {\n const { key, provide, text } = e.detail;\n textExpanderTextRef.current = text;\n\n if (text === '') {\n provide(\n Promise.resolve({\n matched: false,\n }),\n );\n return;\n }\n\n if (key === ':') {\n const showMore = !!onTrigger;\n const results = searcherRef.current?.search(text, {\n limit: 5,\n });\n\n let html = '';\n results?.forEach(({ item: emoji }) => {\n const { shortcode, url } = emoji;\n html += `\n <li role=\"option\" data-value=\"${encodeHTML(shortcode)}\">\n <img src=\"${encodeHTML(\n url,\n )}\" width=\"16\" height=\"16\" alt=\"\" loading=\"lazy\" />\n ${encodeHTML(shortcode)}\n </li>`;\n });\n if (showMore) {\n html += `<li role=\"option\" data-value=\"\" data-more=\"${text}\">${'More…'}</li>`;\n }\n menu.innerHTML = html;\n\n provide(\n Promise.resolve({\n matched: (results?.length || 0) > 0,\n fragment: menu,\n }),\n );\n return;\n }\n\n // Handle @ mentions and # hashtags\n const type = {\n '@': 'accounts',\n '#': 'hashtags',\n }[key];\n\n if (type) {\n provide(\n new Promise(async (resolve) => {\n try {\n let searchResults;\n if (type === 'accounts') {\n searchResults = await masto.v1.accounts.search.list({\n q: text,\n limit: 5,\n resolve: false,\n });\n } else {\n const response = await masto.v2.search.list({\n type,\n q: text,\n limit: 5,\n });\n searchResults = response[type] || response;\n }\n\n if (text !== textExpanderTextRef.current) {\n return;\n }\n\n const results = searchResults;\n let html = '';\n results.forEach((result) => {\n const {\n name,\n avatarStatic,\n displayName,\n username,\n acct,\n emojis,\n history,\n roles,\n url,\n } = result;\n const displayNameWithEmoji = emojifyText(displayName, emojis);\n const accountInstance = getDomain(url);\n\n if (acct) {\n html += `\n <li role=\"option\" data-value=\"${encodeHTML(acct)}\">\n <span class=\"avatar\">\n <img src=\"${encodeHTML(\n avatarStatic,\n )}\" width=\"16\" height=\"16\" alt=\"\" loading=\"lazy\" />\n </span>\n <span>\n <b>${displayNameWithEmoji || username}</b>\n <br><span class=\"bidi-isolate\">@${encodeHTML(\n acct,\n )}</span>\n ${\n roles?.map(\n (role) => ` <span class=\"tag collapsed\">\n ${role.name}\n ${\n !!accountInstance &&\n `<span class=\"more-insignificant\">\n ${accountInstance}\n </span>`\n }\n </span>`,\n ) || ''\n }\n </span>\n </li>\n `;\n } else {\n const total = history?.reduce?.(\n (acc, cur) => acc + +cur.uses,\n 0,\n );\n html += `\n <li role=\"option\" data-value=\"${encodeHTML(name)}\">\n <span class=\"grow\">#<b>${encodeHTML(name)}</b></span>\n ${\n total\n ? `<span class=\"count\">${shortenNumber(total)}</span>`\n : ''\n }\n </li>\n `;\n }\n });\n if (type === 'accounts') {\n html += `<li role=\"option\" data-value=\"\" data-more=\"${text}\">${t`More…`}</li>`;\n }\n menu.innerHTML = html;\n resolve({\n matched: results.length > 0,\n fragment: menu,\n });\n } catch (error) {\n console.error('Search error:', error);\n resolve({\n matched: false,\n });\n }\n }),\n );\n return;\n }\n\n // No other keys supported\n provide(\n Promise.resolve({\n matched: false,\n }),\n );\n };\n\n const handleValue = (e) => {\n const { key, item } = e.detail;\n const { value, more } = item.dataset;\n\n if (key === ':') {\n e.detail.value = value ? `:${value}:` : ''; // zero-width space\n if (more) {\n // Prevent adding space after the above value\n e.detail.continue = true;\n\n setTimeout(() => {\n // Trigger custom emoji picker modal for more options\n onTrigger?.({\n name: 'custom-emojis',\n defaultSearchTerm: more,\n });\n }, 300);\n }\n } else if (key === '@') {\n e.detail.value = value ? `@${value}` : ''; // zero-width space\n if (more) {\n e.detail.continue = true;\n setTimeout(() => {\n onTrigger?.({\n name: 'mention',\n defaultSearchTerm: more,\n });\n }, 300);\n }\n } else {\n e.detail.value = `${key}${value}`;\n }\n };\n\n const handleCommited = (e) => {\n const { input } = e.detail;\n\n if (input) {\n const event = new Event('input', { bubbles: true });\n input.dispatchEvent(event);\n }\n };\n\n const handleActivate = () => {\n hasTextExpanderRef.current = true;\n };\n\n const handleDeactivate = () => {\n hasTextExpanderRef.current = false;\n };\n\n textExpander.addEventListener('text-expander-change', handleChange);\n textExpander.addEventListener('text-expander-value', handleValue);\n textExpander.addEventListener('text-expander-committed', handleCommited);\n textExpander.addEventListener('text-expander-activate', handleActivate);\n textExpander.addEventListener('text-expander-deactivate', handleDeactivate);\n\n return () => {\n textExpander.removeEventListener('text-expander-change', handleChange);\n textExpander.removeEventListener('text-expander-value', handleValue);\n textExpander.removeEventListener(\n 'text-expander-committed',\n handleCommited,\n );\n textExpander.removeEventListener(\n 'text-expander-activate',\n handleActivate,\n );\n textExpander.removeEventListener(\n 'text-expander-deactivate',\n handleDeactivate,\n );\n };\n }, [searcherRef.current, onTrigger, t, masto]);\n\n return <text-expander ref={textExpanderRef} {...props} />;\n}\n\nexport default forwardRef(TextExpander);\n","import { Trans, useLingui } from '@lingui/react/macro';\n\nimport i18nDuration from '../utils/i18n-duration';\n\nimport Icon from './icon';\nimport TextExpander from './text-expander';\n\nexport const expiryOptions = {\n 300: i18nDuration(5, 'minute'),\n 1_800: i18nDuration(30, 'minute'),\n 3_600: i18nDuration(1, 'hour'),\n 21_600: i18nDuration(6, 'hour'),\n 86_400: i18nDuration(1, 'day'),\n 259_200: i18nDuration(3, 'day'),\n 604_800: i18nDuration(1, 'week'),\n};\n\nfunction ComposePoll({\n lang,\n poll,\n disabled,\n onInput = () => {},\n maxOptions,\n maxExpiration,\n minExpiration,\n maxCharactersPerOption,\n}) {\n const { t } = useLingui();\n const { options, expiresIn, multiple } = poll;\n\n return (\n <div class={`poll ${multiple ? 'multiple' : ''}`}>\n <div class=\"poll-choices\">\n {options.map((option, i) => (\n <div class=\"poll-choice\" key={i}>\n <TextExpander keys=\":\" class=\"poll-field-container\">\n <input\n required\n type=\"text\"\n value={option}\n disabled={disabled}\n maxlength={maxCharactersPerOption}\n placeholder={t`Choice ${i + 1}`}\n lang={lang}\n spellCheck=\"true\"\n autocomplete=\"off\"\n dir=\"auto\"\n data-allow-custom-emoji=\"true\"\n onInput={(e) => {\n const { value } = e.target;\n options[i] = value;\n onInput(poll);\n }}\n />\n </TextExpander>\n <button\n type=\"button\"\n class=\"plain4 poll-button\"\n disabled={disabled || options.length <= 1}\n onClick={() => {\n options.splice(i, 1);\n onInput(poll);\n }}\n title={t`Remove`}\n >\n −\n </button>\n </div>\n ))}\n </div>\n <div class=\"poll-toolbar\">\n <button\n type=\"button\"\n class=\"plain2 poll-button\"\n disabled={disabled || options.length >= maxOptions}\n onClick={() => {\n options.push('');\n onInput(poll);\n }}\n title={t`Add`}\n >\n +\n </button>{' '}\n <div class=\"poll-config\">\n <label class=\"multiple-choices\">\n <input\n type=\"checkbox\"\n checked={multiple}\n disabled={disabled}\n onChange={(e) => {\n const { checked } = e.target;\n poll.multiple = checked;\n onInput(poll);\n }}\n />{' '}\n <Trans>Multiple choice</Trans>\n </label>\n <label class=\"expires-in\">\n <Trans>Duration</Trans>{' '}\n <select\n value={expiresIn}\n disabled={disabled}\n onChange={(e) => {\n const { value } = e.target;\n poll.expiresIn = value;\n onInput(poll);\n }}\n >\n {Object.entries(expiryOptions)\n .filter(([value]) => {\n return value >= minExpiration && value <= maxExpiration;\n })\n .map(([value, label]) => (\n <option value={value} key={value}>\n {label()}\n </option>\n ))}\n </select>\n </label>\n <div class=\"spacer\" />\n <button\n type=\"button\"\n class=\"light danger small\"\n disabled={disabled}\n onClick={() => {\n onInput(null);\n }}\n >\n <Trans>Remove poll</Trans>\n </button>\n </div>\n </div>\n </div>\n );\n}\n\nexport default ComposePoll;\n","import { forwardRef } from 'preact/compat';\nimport { useEffect, useRef, useState } from 'preact/hooks';\nimport { useDebouncedCallback, useThrottledCallback } from 'use-debounce';\n\nimport { langDetector } from '../utils/browser-translator';\nimport escapeHTML from '../utils/escape-html';\nimport states from '../utils/states';\nimport urlRegexObj from '../utils/url-regex';\n\nimport TextExpander from './text-expander';\n\n// https://github.com/mastodon/mastodon/blob/c03bd2a238741a012aa4b98dc4902d6cf948ab63/app/models/account.rb#L69\nconst USERNAME_RE = /[a-z0-9_]+([a-z0-9_.-]+[a-z0-9_]+)?/i;\nconst MENTION_RE = new RegExp(\n `(^|[^=\\\\/\\\\w])(@${USERNAME_RE.source}(?:@[\\\\p{L}\\\\w.-]+[\\\\w]+)?)`,\n 'uig',\n);\n\n// AI-generated, all other regexes are too complicated\nconst HASHTAG_RE = new RegExp(\n `(^|[^=\\\\/\\\\w])(#[\\\\p{L}\\\\p{N}_]+([\\\\p{L}\\\\p{N}_.]+[\\\\p{L}\\\\p{N}_]+)?)(?![\\\\/\\\\w])`,\n 'iug',\n);\n\n// https://github.com/mastodon/mastodon/blob/23e32a4b3031d1da8b911e0145d61b4dd47c4f96/app/models/custom_emoji.rb#L31\nconst SHORTCODE_RE_FRAGMENT = '[a-zA-Z0-9_]{2,}';\nconst SCAN_RE = new RegExp(\n `(^|[^=\\\\/\\\\w])(:${SHORTCODE_RE_FRAGMENT}:)(?=[^A-Za-z0-9_:]|$)`,\n 'g',\n);\n\nconst segmenter = new Intl.Segmenter();\n\nfunction highlightText(text, { maxCharacters = Infinity }) {\n // Exceeded characters limit\n const { composerCharacterCount } = states;\n if (composerCharacterCount > maxCharacters) {\n // Highlight exceeded characters\n let withinLimitHTML = '',\n exceedLimitHTML = '';\n const htmlSegments = segmenter.segment(text);\n for (const { segment, index } of htmlSegments) {\n if (index < maxCharacters) {\n withinLimitHTML += segment;\n } else {\n exceedLimitHTML += segment;\n }\n }\n if (exceedLimitHTML) {\n exceedLimitHTML =\n '<mark class=\"compose-highlight-exceeded\">' +\n escapeHTML(exceedLimitHTML) +\n '</mark>';\n }\n return escapeHTML(withinLimitHTML) + exceedLimitHTML;\n }\n\n return escapeHTML(text)\n .replace(urlRegexObj, '$2<mark class=\"compose-highlight-url\">$3</mark>') // URLs\n .replace(MENTION_RE, '$1<mark class=\"compose-highlight-mention\">$2</mark>') // Mentions\n .replace(HASHTAG_RE, '$1<mark class=\"compose-highlight-hashtag\">$2</mark>') // Hashtags\n .replace(\n SCAN_RE,\n '$1<mark class=\"compose-highlight-emoji-shortcode\">$2</mark>',\n ); // Emoji shortcodes\n}\n\nfunction autoResizeTextarea(textarea) {\n if (!textarea) return;\n const { value, offsetHeight, scrollHeight, clientHeight } = textarea;\n if (offsetHeight < window.innerHeight) {\n // NOTE: This check is needed because the offsetHeight return 50000 (really large number) on first render\n // No idea why it does that, will re-investigate in far future\n const offset = offsetHeight - clientHeight;\n const height = value ? scrollHeight + offset + 'px' : null;\n textarea.style.height = height;\n }\n}\n\nconst detectLangs = async (text) => {\n if (langDetector) {\n const langs = await langDetector.detect(text);\n if (langs?.length) {\n return langs.slice(0, 2).map((lang) => lang.detectedLanguage);\n }\n }\n const { detectAll } = await import('tinyld/light');\n const langs = detectAll(text);\n if (langs?.length) {\n // return max 2\n return langs.slice(0, 2).map((lang) => lang.lang);\n }\n return null;\n};\n\nconst Textarea = forwardRef((props, ref) => {\n const [text, setText] = useState(ref.current?.value || '');\n const { maxCharacters, onTrigger = null, ...textareaProps } = props;\n\n const textExpanderRef = useRef();\n\n useEffect(() => {\n // Resize observer for textarea\n const textarea = ref.current;\n if (!textarea) return;\n const resizeObserver = new ResizeObserver(() => {\n // Get height of textarea, set height to textExpander\n if (textExpanderRef.current) {\n const { height } = textarea.getBoundingClientRect();\n // textExpanderRef.current.style.height = height + 'px';\n if (height) {\n textExpanderRef.current.setStyle({ minHeight: height + 'px' });\n }\n }\n });\n resizeObserver.observe(textarea);\n }, []);\n\n const slowHighlightPerf = useRef(0); // increment if slow\n const composeHighlightRef = useRef();\n const throttleHighlightText = useThrottledCallback((text) => {\n if (!composeHighlightRef.current) return;\n if (slowHighlightPerf.current > 3) {\n // After 3 times of lag, disable highlighting\n composeHighlightRef.current.innerHTML = '';\n composeHighlightRef.current = null; // Destroy the whole thing\n throttleHighlightText?.cancel?.();\n return;\n }\n let start;\n let end;\n if (slowHighlightPerf.current <= 3) start = Date.now();\n composeHighlightRef.current.innerHTML =\n highlightText(text, {\n maxCharacters,\n }) + '\\n';\n if (slowHighlightPerf.current <= 3) end = Date.now();\n console.debug('HIGHLIGHT PERF', { start, end, diff: end - start });\n if (start && end && end - start > 50) {\n // if slow, increment\n slowHighlightPerf.current++;\n }\n // Newline to prevent multiple line breaks at the end from being collapsed, no idea why\n }, 500);\n\n const debouncedAutoDetectLanguage = useDebouncedCallback(() => {\n // Make use of the highlightRef to get the DOM\n // Clone the dom\n const dom = composeHighlightRef.current?.cloneNode(true);\n if (!dom) return;\n // Remove mark\n dom.querySelectorAll('mark').forEach((mark) => {\n mark.remove();\n });\n const text = dom.innerText?.trim();\n if (!text) return;\n (async () => {\n const langs = await detectLangs(text);\n if (langs?.length) {\n onTrigger?.({\n name: 'auto-detect-language',\n languages: langs,\n });\n }\n })();\n }, 2000);\n\n return (\n <TextExpander\n ref={textExpanderRef}\n keys=\"@ # :\"\n class=\"compose-field-container\"\n onTrigger={onTrigger}\n >\n <textarea\n class=\"compose-field\"\n autoCapitalize=\"sentences\"\n autoComplete=\"on\"\n autoCorrect=\"on\"\n spellCheck=\"true\"\n dir=\"auto\"\n rows=\"6\"\n cols=\"50\"\n {...textareaProps}\n ref={ref}\n name=\"status\"\n value={text}\n onKeyDown={(e) => {\n // Get line before cursor position after pressing 'Enter'\n const { key, target } = e;\n const hasTextExpander = textExpanderRef.current?.activated();\n if (\n key === 'Enter' &&\n !(e.ctrlKey || e.metaKey || hasTextExpander) &&\n !e.isComposing\n ) {\n try {\n const { value, selectionStart } = target;\n const textBeforeCursor = value.slice(0, selectionStart);\n const lastLine = textBeforeCursor.split('\\n').slice(-1)[0];\n if (lastLine) {\n // If line starts with \"- \" or \"12. \"\n if (/^\\s*(-|\\d+\\.)\\s/.test(lastLine)) {\n // insert \"- \" at cursor position\n const [_, preSpaces, bullet, postSpaces, anything] =\n lastLine.match(/^(\\s*)(-|\\d+\\.)(\\s+)(.+)?/) || [];\n if (anything) {\n e.preventDefault();\n const [number] = bullet.match(/\\d+/) || [];\n const newBullet = number ? `${+number + 1}.` : '-';\n const text = `\\n${preSpaces}${newBullet}${postSpaces}`;\n target.setRangeText(text, selectionStart, selectionStart);\n const pos = selectionStart + text.length;\n target.setSelectionRange(pos, pos);\n } else {\n // trim the line before the cursor, then insert new line\n const pos = selectionStart - lastLine.length;\n target.setRangeText('', pos, selectionStart);\n }\n autoResizeTextarea(target);\n target.dispatchEvent(new Event('input'));\n }\n }\n } catch (e) {\n // silent fail\n console.error(e);\n }\n }\n if (composeHighlightRef.current) {\n composeHighlightRef.current.scrollTop = target.scrollTop;\n }\n }}\n onInput={(e) => {\n const { target } = e;\n const text = target.value;\n setText(text);\n autoResizeTextarea(target);\n props.onInput?.(e);\n throttleHighlightText(text);\n debouncedAutoDetectLanguage();\n }}\n onScroll={(e) => {\n if (composeHighlightRef.current) {\n const { scrollTop } = e.target;\n composeHighlightRef.current.scrollTop = scrollTop;\n }\n }}\n />\n <div\n ref={composeHighlightRef}\n class=\"compose-highlight\"\n aria-hidden=\"true\"\n />\n </TextExpander>\n );\n});\n\nexport default Textarea;\n","import './custom-emojis-modal.css';\n\nimport { Trans, useLingui } from '@lingui/react/macro';\nimport { memo } from 'preact/compat';\nimport {\n useCallback,\n useEffect,\n useMemo,\n useRef,\n useState,\n} from 'preact/hooks';\n\nimport getCustomEmojis from '../utils/custom-emojis';\nimport store from '../utils/store';\n\nimport Icon from './icon';\nimport Loader from './loader';\n\nconst CUSTOM_EMOJIS_COUNT = 100;\nconst EMOJI_SIZE_MIN = 1;\nconst EMOJI_SIZE_MAX = 2;\nconst EMOJI_SIZE_STEP = 0.5;\n\nconst CustomEmojiButton = memo(({ emoji, onClick, showCode }) => {\n const addEdges = (e) => {\n // Add edge-left or edge-right class based on self position relative to scrollable parent\n // If near left edge, add edge-left, if near right edge, add edge-right\n const buffer = 88;\n const parent = e.currentTarget.closest('main');\n if (parent) {\n const rect = parent.getBoundingClientRect();\n const selfRect = e.currentTarget.getBoundingClientRect();\n const targetClassList = e.currentTarget.classList;\n if (selfRect.left < rect.left + buffer) {\n targetClassList.add('edge-left');\n targetClassList.remove('edge-right');\n } else if (selfRect.right > rect.right - buffer) {\n targetClassList.add('edge-right');\n targetClassList.remove('edge-left');\n } else {\n targetClassList.remove('edge-left', 'edge-right');\n }\n }\n };\n\n return (\n <button\n type=\"button\"\n className=\"plain4\"\n onClick={onClick}\n data-title={showCode ? undefined : emoji.shortcode}\n onPointerEnter={addEdges}\n onFocus={addEdges}\n >\n <picture>\n {!!emoji.staticUrl && (\n <source\n srcSet={emoji.staticUrl}\n media=\"(prefers-reduced-motion: reduce)\"\n />\n )}\n <img\n className=\"shortcode-emoji\"\n src={emoji.url || emoji.staticUrl}\n alt={emoji.shortcode}\n width=\"24\"\n height=\"24\"\n loading=\"lazy\"\n decoding=\"async\"\n />\n </picture>\n {showCode && (\n <>\n {' '}\n <code>{emoji.shortcode}</code>\n </>\n )}\n </button>\n );\n});\n\nconst CustomEmojisList = memo(({ emojis, onSelect }) => {\n const { i18n } = useLingui();\n const [max, setMax] = useState(CUSTOM_EMOJIS_COUNT);\n const showMore = emojis.length > max;\n return (\n <section>\n {emojis.slice(0, max).map((emoji) => (\n <CustomEmojiButton\n key={emoji.shortcode}\n emoji={emoji}\n onClick={() => {\n onSelect(`:${emoji.shortcode}:`);\n }}\n />\n ))}\n {showMore && (\n <button\n type=\"button\"\n class=\"plain small\"\n onClick={() => setMax(max + CUSTOM_EMOJIS_COUNT)}\n >\n <Trans>{i18n.number(emojis.length - max)} more…</Trans>\n </button>\n )}\n </section>\n );\n});\n\nfunction CustomEmojisModal({\n masto,\n instance,\n onClose = () => {},\n onSelect = () => {},\n defaultSearchTerm,\n}) {\n const { t } = useLingui();\n const [uiState, setUIState] = useState('default');\n const customEmojisList = useRef([]);\n const [customEmojis, setCustomEmojis] = useState([]);\n const recentlyUsedCustomEmojis = useMemo(\n () => store.account.get('recentlyUsedCustomEmojis') || [],\n );\n const searcherRef = useRef();\n useEffect(() => {\n setUIState('loading');\n (async () => {\n try {\n const [emojis, searcher] = await getCustomEmojis(instance, masto);\n console.log('emojis', emojis);\n searcherRef.current = searcher;\n setCustomEmojis(emojis);\n setUIState('default');\n } catch (e) {\n setUIState('error');\n console.error(e);\n }\n })();\n }, []);\n\n const customEmojisCatList = useMemo(() => {\n // Group emojis by category\n const emojisCat = {\n '--recent--': recentlyUsedCustomEmojis.filter((emoji) =>\n customEmojis.find((e) => e.shortcode === emoji.shortcode),\n ),\n };\n const othersCat = [];\n customEmojis.forEach((emoji) => {\n customEmojisList.current?.push?.(emoji);\n if (!emoji.category) {\n othersCat.push(emoji);\n return;\n }\n if (!emojisCat[emoji.category]) {\n emojisCat[emoji.category] = [];\n }\n emojisCat[emoji.category].push(emoji);\n });\n if (othersCat.length) {\n emojisCat['--others--'] = othersCat;\n }\n return emojisCat;\n }, [customEmojis]);\n\n const scrollableRef = useRef();\n const [matches, setMatches] = useState(null);\n const [emojiSize, setEmojiSize] = useState(EMOJI_SIZE_MIN);\n const onEmojiSizeDecrease = useCallback(() => {\n const newSize = Math.max(EMOJI_SIZE_MIN, emojiSize - EMOJI_SIZE_STEP);\n setEmojiSize(newSize);\n }, [emojiSize]);\n\n const onEmojiSizeIncrease = useCallback(() => {\n const newSize = Math.min(EMOJI_SIZE_MAX, emojiSize + EMOJI_SIZE_STEP);\n setEmojiSize(newSize);\n }, [emojiSize]);\n\n const onFind = useCallback(\n (e) => {\n const { value } = e.target;\n if (value) {\n const results = searcherRef.current?.search(value, {\n limit: CUSTOM_EMOJIS_COUNT,\n });\n setMatches(results.map((r) => r.item));\n scrollableRef.current?.scrollTo?.(0, 0);\n } else {\n setMatches(null);\n }\n },\n [customEmojis],\n );\n useEffect(() => {\n if (defaultSearchTerm && customEmojis?.length) {\n onFind({ target: { value: defaultSearchTerm } });\n }\n }, [defaultSearchTerm, onFind, customEmojis]);\n\n const onSelectEmoji = useCallback(\n (emoji) => {\n onSelect?.(emoji);\n onClose?.();\n\n queueMicrotask(() => {\n let recentlyUsedCustomEmojis =\n store.account.get('recentlyUsedCustomEmojis') || [];\n const recentlyUsedEmojiIndex = recentlyUsedCustomEmojis.findIndex(\n (e) => e.shortcode === emoji.shortcode,\n );\n if (recentlyUsedEmojiIndex !== -1) {\n // Move emoji to index 0\n recentlyUsedCustomEmojis.splice(recentlyUsedEmojiIndex, 1);\n recentlyUsedCustomEmojis.unshift(emoji);\n } else {\n recentlyUsedCustomEmojis.unshift(emoji);\n // Remove unavailable ones\n recentlyUsedCustomEmojis = recentlyUsedCustomEmojis.filter((e) =>\n customEmojisList.current?.find?.(\n (emoji) => emoji.shortcode === e.shortcode,\n ),\n );\n // Limit to 10\n recentlyUsedCustomEmojis = recentlyUsedCustomEmojis.slice(0, 10);\n }\n\n // Store back\n store.account.set('recentlyUsedCustomEmojis', recentlyUsedCustomEmojis);\n });\n },\n [onSelect],\n );\n\n const inputRef = useRef();\n useEffect(() => {\n if (inputRef.current) {\n inputRef.current.focus();\n // Put cursor at the end\n if (inputRef.current.value) {\n inputRef.current.selectionStart = inputRef.current.value.length;\n inputRef.current.selectionEnd = inputRef.current.value.length;\n }\n }\n }, []);\n\n return (\n <div\n id=\"custom-emojis-sheet\"\n class=\"sheet\"\n style={{\n '--custom-emoji-size': emojiSize,\n }}\n >\n {!!onClose && (\n <button type=\"button\" class=\"sheet-close\" onClick={onClose}>\n <Icon icon=\"x\" alt={t`Close`} />\n </button>\n )}\n <header>\n <div>\n <b>\n <Trans>Custom emojis</Trans>\n </b>{' '}\n {uiState === 'loading' ? (\n <Loader />\n ) : (\n <small class=\"insignificant\"> • {instance}</small>\n )}\n </div>\n <form\n onSubmit={(e) => {\n e.preventDefault();\n const emoji = matches[0];\n if (emoji) {\n onSelectEmoji(`:${emoji.shortcode}:`);\n }\n }}\n >\n <input\n ref={inputRef}\n type=\"search\"\n placeholder={t`Search emoji`}\n onInput={onFind}\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellCheck=\"false\"\n dir=\"auto\"\n defaultValue={defaultSearchTerm || ''}\n />\n </form>\n </header>\n <main ref={scrollableRef}>\n {matches !== null ? (\n <ul class=\"custom-emojis-matches custom-emojis-list\">\n {matches.map((emoji) => (\n <li key={emoji.shortcode} class=\"custom-emojis-match\">\n <CustomEmojiButton\n emoji={emoji}\n onClick={() => {\n onSelectEmoji(`:${emoji.shortcode}:`);\n }}\n showCode\n />\n </li>\n ))}\n </ul>\n ) : (\n <div class=\"custom-emojis-list\">\n {uiState === 'error' && (\n <div class=\"ui-state\">\n <p>\n <Trans>Error loading custom emojis</Trans>\n </p>\n </div>\n )}\n {uiState === 'default' &&\n Object.entries(customEmojisCatList).map(\n ([category, emojis]) =>\n !!emojis?.length && (\n <div class=\"section-container\">\n <div class=\"section-header\">\n {{\n '--recent--': t`Recently used`,\n '--others--': t`Others`,\n }[category] || category}\n </div>\n <CustomEmojisList\n emojis={emojis}\n onSelect={onSelectEmoji}\n />\n </div>\n ),\n )}\n </div>\n )}\n <div class=\"size-range\">\n <button\n type=\"button\"\n class=\"plain4\"\n onClick={onEmojiSizeDecrease}\n disabled={emojiSize <= EMOJI_SIZE_MIN}\n >\n <Icon icon=\"zoom-out\" size=\"l\" alt={t`Zoom out`} />\n </button>\n <button\n type=\"button\"\n class=\"plain4\"\n onClick={onEmojiSizeIncrease}\n disabled={emojiSize >= EMOJI_SIZE_MAX}\n >\n <Icon icon=\"zoom-in\" size=\"l\" alt={t`Zoom in`} />\n </button>\n </div>\n </main>\n </div>\n );\n}\n\nexport default CustomEmojisModal;\n","import { plural } from '@lingui/core/macro';\n\nfunction FilePickerInput({\n hidden,\n supportedMimeTypes,\n maxMediaAttachments,\n mediaAttachments,\n disabled = false,\n setMediaAttachments,\n}) {\n return (\n <input\n type=\"file\"\n hidden={hidden}\n accept={supportedMimeTypes?.join(',')}\n multiple={\n maxMediaAttachments === undefined ||\n maxMediaAttachments - mediaAttachments >= 2\n }\n disabled={disabled}\n onChange={(e) => {\n const files = e.target.files;\n if (!files) return;\n\n const mediaFiles = Array.from(files).map((file) => ({\n file,\n type: file.type,\n size: file.size,\n url: URL.createObjectURL(file),\n id: null, // indicate uploaded state\n description: null,\n }));\n console.log('MEDIA ATTACHMENTS', files, mediaFiles);\n\n // Validate max media attachments\n if (mediaAttachments.length + mediaFiles.length > maxMediaAttachments) {\n alert(\n plural(maxMediaAttachments, {\n one: 'You can only attach up to 1 file.',\n other: 'You can only attach up to # files.',\n }),\n );\n } else {\n setMediaAttachments((attachments) => {\n return attachments.concat(mediaFiles);\n });\n }\n // Reset\n e.target.value = '';\n }}\n />\n );\n}\n\nexport default FilePickerInput;\n","export default \"data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20version='1.0'%20viewBox='0%200%20641%20223'%3e%3cpath%20fill='%23aaa'%20d='M86%20214c-9-1-17-4-24-8l-6-3-5-5-5-4-4-6-4-6-3-8-2-8v-27l2-9%203-9%204-6%204-6%205-5%205-5%207-3%206-4%207-2%207-2%2012-1h12l7%201%208%202%207%204%207%203%205%205%205%204-10%2010-10%209-4-3-10-5-5-1H88l-5%202-6%203-3%204-4%204-2%205-2%206v6l-1%207%201%207%202%207%203%205%202%204%204%203%204%203%205%202%206%202h9l10-1%205-2%206-3v-16H91v-27h59v54l-1%203-2%203-5%204-4%204-5%203-5%202-8%202-8%202-10%201H92l-6-1zm266-62V91h34v46h44V91h34v121h-34v-46h-44v46h-34v-61zm-182-1V90h34v121h-34v-60zm59-1V90h35l36%201%205%202c3%200%208%202%2010%204l5%202%204%205%205%204%203%207%203%207%201%2013v13l-4%206-3%207-4%204-5%205-5%202-5%203-6%202-5%201-18%201h-18v32h-34v-61zm67-2%203-2%202-4%202-5v-5l-2-4-2-4-3-2-3-3h-30v31h30l3-2zm226%2039v-24l-8-12-18-28a1751%201751%200%200%200-20-31v-2h39l7%2012%2012%2021%206%209%2013-21%2013-21h38v2l-41%2061-7%2010v48h-34v-24zM109%2066l-4-1-5-5-5-4-1-5-3-9v-5l1-5c2-7%203-10%208-15l4-4%207-2%207-2h7l6%201%205%202%205%202%203%204%204%203%202%206%202%205v13l-2%205-2%206-4%204-3%203-5%202-4%202-9%201h-9l-5-2zm22-11%204-2%203-4%202-5V34l-2-4-2-4-3-2-4-3-5-1h-6l-4%202-5%202-2%204-3%205-1%203v4l1%205%202%205%202%202%205%203%204%202h10l4-2zM37%2039V11h33l3%201%203%202%204%203%203%203%201%205%201%204v5l-1%204-3%204-3%205-4%201-3%202-11%201H49v16H37V39zm31%200%203-2%201-2%201-2v-4l-1-3-3-2-2-2H49v18h15l4-1zm107%2025a512%20512%200%200%200-19-53h14l4%2014%206%2019%201%204%201-1%207-19%205-17h9l6%2019%207%2018v-1l2-6%205-17%204-13h14v1l-4%2012-16%2041v2h-5l-5-1-6-15-6-15-1%201-3%207-6%2015-2%208h-11l-1-3zm74-25V11h42v11h-29v2l-1%205v4h29v11h-28v11h2l15%201h13v11h-43V39zm55%200V11h33l5%203%205%202%202%204%202%205v10l-2%203-1%204-5%203-5%203%205%205%208%2010%203%204h-14l-7-9-8-10h-9v19h-12V39zm33-3%202-3v-6l-3-3-2-3h-18v16h1v1h17l2-2zm26%203V11h42v11h-29l-1%206v5h29v11h-28v5l-1%205%201%201v1h30v11h-43V39zm54%200V11h17l18%201%204%202%205%203%202%204%203%204%202%206%201%206v5c-1%206-3%2012-6%2015l-3%204-5%203-5%202-17%201h-16V39zm33%2014%205-5%202-3v-6l-1-6-1-3-1-3-4-3-3-2h-5l-6-1-3%201h-3v34h9l8-1%203-2zm50-14V11h34l5%202%204%202%202%203%202%203v9l-2%202-3%204-1%201%203%203%203%204%201%203%201%204-1%204-1%204-3%203-3%203-5%201-5%201h-31V39zm34%2015%202-1v-6l-2-2-2-2h-20v13h20l2-2zm-3-22%204-2v-6l-2-1-2-2h-19v12h16l4-1zm42%2024V45l-6-9-11-17-5-8h15l4%208%207%2011%202%203%207-11%207-11h14l-11%2016-11%2017v23h-12V56z'/%3e%3c/svg%3e\"","import { Trans, useLingui } from '@lingui/react/macro';\nimport { useEffect, useRef, useState } from 'preact/hooks';\nimport { useDebouncedCallback } from 'use-debounce';\n\nimport poweredByGiphyURL from '../assets/powered-by-giphy.svg';\n\nimport Icon from './icon';\nimport Loader from './loader';\n\nconst { PHANPY_GIPHY_API_KEY: GIPHY_API_KEY } = import.meta.env;\n\nconst GIFS_PER_PAGE = 20;\n\nfunction GIFPickerModal({ onClose = () => {}, onSelect = () => {} }) {\n const { i18n, t } = useLingui();\n const [uiState, setUIState] = useState('default');\n const [results, setResults] = useState([]);\n const formRef = useRef(null);\n const qRef = useRef(null);\n const currentOffset = useRef(0);\n const scrollableRef = useRef(null);\n\n function fetchGIFs({ offset }) {\n console.log('fetchGIFs', { offset });\n if (!qRef.current?.value) return;\n setUIState('loading');\n scrollableRef.current?.scrollTo?.({\n top: 0,\n left: 0,\n behavior: 'smooth',\n });\n (async () => {\n try {\n const query = {\n api_key: GIPHY_API_KEY,\n q: qRef.current.value,\n rating: 'g',\n limit: GIFS_PER_PAGE,\n bundle: 'messaging_non_clips',\n offset,\n lang: i18n.locale || 'en',\n };\n const response = await fetch(\n 'https://api.giphy.com/v1/gifs/search?' + new URLSearchParams(query),\n {\n referrerPolicy: 'no-referrer',\n },\n ).then((r) => r.json());\n currentOffset.current = response.pagination?.offset || 0;\n setResults(response);\n setUIState('results');\n } catch (e) {\n setUIState('error');\n console.error(e);\n }\n })();\n }\n\n useEffect(() => {\n qRef.current?.focus();\n }, []);\n\n const debouncedOnInput = useDebouncedCallback(() => {\n fetchGIFs({ offset: 0 });\n }, 1000);\n\n return (\n <div id=\"gif-picker-sheet\" class=\"sheet\">\n {!!onClose && (\n <button type=\"button\" class=\"sheet-close\" onClick={onClose}>\n <Icon icon=\"x\" alt={t`Close`} />\n </button>\n )}\n <header>\n <form\n ref={formRef}\n onSubmit={(e) => {\n e.preventDefault();\n fetchGIFs({ offset: 0 });\n }}\n >\n <input\n ref={qRef}\n type=\"search\"\n name=\"q\"\n placeholder={t`Search GIFs`}\n required\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellCheck=\"false\"\n dir=\"auto\"\n onInput={debouncedOnInput}\n />\n <input\n type=\"image\"\n class=\"powered-button\"\n src={poweredByGiphyURL}\n width=\"86\"\n height=\"30\"\n alt={t`Powered by GIPHY`}\n />\n </form>\n </header>\n <main ref={scrollableRef} class={uiState === 'loading' ? 'loading' : ''}>\n {uiState === 'default' && (\n <div class=\"ui-state\">\n <p class=\"insignificant\">\n <Trans>Type to search GIFs</Trans>\n </p>\n </div>\n )}\n {uiState === 'loading' && !results?.data?.length && (\n <div class=\"ui-state\">\n <Loader abrupt />\n </div>\n )}\n {results?.data?.length > 0 ? (\n <>\n <ul>\n {results.data.map((gif) => {\n const { id, images, title, alt_text } = gif;\n const {\n fixed_height_small,\n fixed_height_downsampled,\n fixed_height,\n original,\n } = images;\n const theImage = fixed_height_small?.url\n ? fixed_height_small\n : fixed_height_downsampled?.url\n ? fixed_height_downsampled\n : fixed_height;\n let { url, webp, width, height } = theImage;\n if (+height > 100) {\n width = (width / height) * 100;\n height = 100;\n }\n const urlObj = URL.parse(url);\n const strippedURL = urlObj.origin + urlObj.pathname;\n let strippedWebP;\n if (webp) {\n const webpObj = URL.parse(webp);\n strippedWebP = webpObj.origin + webpObj.pathname;\n }\n return (\n <li key={id}>\n <button\n type=\"button\"\n onClick={() => {\n const { mp4, url } = original;\n const theURL = mp4 || url;\n const urlObj = URL.parse(theURL);\n const strippedURL = urlObj.origin + urlObj.pathname;\n onClose();\n onSelect({\n url: strippedURL,\n type: mp4 ? 'video/mp4' : 'image/gif',\n alt_text: alt_text || title,\n });\n }}\n >\n <figure\n style={{\n '--figure-width': width + 'px',\n // width: width + 'px'\n }}\n >\n <picture>\n {strippedWebP && (\n <source srcset={strippedWebP} type=\"image/webp\" />\n )}\n <img\n src={strippedURL}\n width={width}\n height={height}\n loading=\"lazy\"\n decoding=\"async\"\n alt={alt_text}\n referrerpolicy=\"no-referrer\"\n onLoad={(e) => {\n e.target.style.backgroundColor = 'transparent';\n }}\n />\n </picture>\n <figcaption>{alt_text || title}</figcaption>\n </figure>\n </button>\n </li>\n );\n })}\n </ul>\n <p class=\"pagination\">\n {results.pagination?.offset > 0 && (\n <button\n type=\"button\"\n class=\"light small\"\n disabled={uiState === 'loading'}\n onClick={() => {\n fetchGIFs({\n offset: results.pagination?.offset - GIFS_PER_PAGE,\n });\n }}\n >\n <Icon icon=\"chevron-left\" />\n <span>\n <Trans>Previous</Trans>\n </span>\n </button>\n )}\n <span />\n {results.pagination?.offset + results.pagination?.count <\n results.pagination?.total_count && (\n <button\n type=\"button\"\n class=\"light small\"\n disabled={uiState === 'loading'}\n onClick={() => {\n fetchGIFs({\n offset: results.pagination?.offset + GIFS_PER_PAGE,\n });\n }}\n >\n <span>\n <Trans>Next</Trans>\n </span>{' '}\n <Icon icon=\"chevron-right\" />\n </button>\n )}\n </p>\n </>\n ) : (\n uiState === 'results' && (\n <div class=\"ui-state\">\n <p>No results</p>\n </div>\n )\n )}\n {uiState === 'error' && (\n <div class=\"ui-state\">\n <p>\n <Trans>Error loading GIFs</Trans>\n </p>\n </div>\n )}\n </main>\n </div>\n );\n}\n\nexport default GIFPickerModal;\n","import { Trans, useLingui } from '@lingui/react/macro';\nimport { MenuItem } from '@szhsin/react-menu';\nimport { useEffect, useMemo, useRef, useState } from 'preact/hooks';\nimport { useDebouncedCallback } from 'use-debounce';\n\nimport localeCode2Text from '../utils/localeCode2Text';\nimport prettyBytes from '../utils/pretty-bytes';\nimport showToast from '../utils/show-toast';\nimport states from '../utils/states';\nimport { getCurrentInstanceConfiguration } from '../utils/store-utils';\nimport supports from '../utils/supports';\n\nimport Icon from './icon';\nimport Menu2 from './menu2';\nimport Modal from './modal';\n\nconst { PHANPY_IMG_ALT_API_URL: IMG_ALT_API_URL } = import.meta.env;\n\nfunction scaleDimension(matrix, matrixLimit, width, height) {\n // matrix = number of pixels\n // matrixLimit = max number of pixels\n // Calculate new width and height, downsize to within the limit, preserve aspect ratio, no decimals\n const scalingFactor = Math.sqrt(matrixLimit / matrix);\n const newWidth = Math.floor(width * scalingFactor);\n const newHeight = Math.floor(height * scalingFactor);\n return { newWidth, newHeight };\n}\n\nfunction MediaAttachment({\n attachment,\n disabled,\n lang,\n descriptionLimit = 1500,\n onDescriptionChange = () => {},\n onRemove = () => {},\n}) {\n const { i18n, t } = useLingui();\n const [uiState, setUIState] = useState('default');\n const supportsEdit = supports('@mastodon/edit-media-attributes');\n const { type, id, file } = attachment;\n const url = useMemo(\n () => (file ? URL.createObjectURL(file) : attachment.url),\n [file, attachment.url],\n );\n console.log({ attachment });\n\n const checkMaxError = !!file?.size;\n const configuration = checkMaxError ? getCurrentInstanceConfiguration() : {};\n const {\n mediaAttachments: {\n imageSizeLimit,\n imageMatrixLimit,\n videoSizeLimit,\n videoMatrixLimit,\n videoFrameRateLimit,\n } = {},\n } = configuration || {};\n\n const [maxError, setMaxError] = useState(() => {\n if (!checkMaxError) return null;\n if (\n type.startsWith('image') &&\n imageSizeLimit &&\n file.size > imageSizeLimit\n ) {\n return {\n type: 'imageSizeLimit',\n details: {\n imageSize: file.size,\n imageSizeLimit,\n },\n };\n } else if (\n type.startsWith('video') &&\n videoSizeLimit &&\n file.size > videoSizeLimit\n ) {\n return {\n type: 'videoSizeLimit',\n details: {\n videoSize: file.size,\n videoSizeLimit,\n },\n };\n }\n return null;\n });\n\n const [imageMatrix, setImageMatrix] = useState({});\n useEffect(() => {\n if (!checkMaxError || !imageMatrixLimit) return;\n if (imageMatrix?.matrix > imageMatrixLimit) {\n setMaxError({\n type: 'imageMatrixLimit',\n details: {\n imageMatrix: imageMatrix?.matrix,\n imageMatrixLimit,\n width: imageMatrix?.width,\n height: imageMatrix?.height,\n },\n });\n }\n }, [imageMatrix, imageMatrixLimit, checkMaxError]);\n\n const [videoMatrix, setVideoMatrix] = useState({});\n useEffect(() => {\n if (!checkMaxError || !videoMatrixLimit) return;\n if (videoMatrix?.matrix > videoMatrixLimit) {\n setMaxError({\n type: 'videoMatrixLimit',\n details: {\n videoMatrix: videoMatrix?.matrix,\n videoMatrixLimit,\n width: videoMatrix?.width,\n height: videoMatrix?.height,\n },\n });\n }\n }, [videoMatrix, videoMatrixLimit, checkMaxError]);\n\n const [description, setDescription] = useState(attachment.description);\n const [suffixType, subtype] = type.split('/');\n const debouncedOnDescriptionChange = useDebouncedCallback(\n onDescriptionChange,\n 250,\n );\n useEffect(() => {\n debouncedOnDescriptionChange(description);\n }, [description, debouncedOnDescriptionChange]);\n\n const [showModal, setShowModal] = useState(false);\n const textareaRef = useRef(null);\n useEffect(() => {\n let timer;\n if (showModal && textareaRef.current) {\n timer = setTimeout(() => {\n textareaRef.current.focus();\n }, 100);\n }\n return () => {\n clearTimeout(timer);\n };\n }, [showModal]);\n\n const descTextarea = (\n <>\n {!!id && !supportsEdit ? (\n <div class=\"media-desc\">\n <span class=\"tag\">\n <Trans>Uploaded</Trans>\n </span>\n <p title={description}>\n {attachment.description || <i>No description</i>}\n </p>\n </div>\n ) : (\n <textarea\n ref={textareaRef}\n value={description || ''}\n lang={lang}\n placeholder={\n {\n image: t`Image description`,\n video: t`Video description`,\n audio: t`Audio description`,\n }[suffixType]\n }\n autoCapitalize=\"sentences\"\n autoComplete=\"on\"\n autoCorrect=\"on\"\n spellCheck=\"true\"\n dir=\"auto\"\n disabled={disabled || uiState === 'loading'}\n class={uiState === 'loading' ? 'loading' : ''}\n maxlength={descriptionLimit} // Not unicode-aware :(\n onInput={(e) => {\n const { value } = e.target;\n setDescription(value);\n // debouncedOnDescriptionChange(value);\n }}\n ></textarea>\n )}\n </>\n );\n\n const toastRef = useRef(null);\n useEffect(() => {\n return () => {\n toastRef.current?.hideToast?.();\n };\n }, []);\n\n const maxErrorToast = useRef(null);\n\n const maxErrorText = (err) => {\n const { type, details } = err;\n switch (type) {\n case 'imageSizeLimit': {\n const { imageSize, imageSizeLimit } = details;\n return t`File size too large. Uploading might encounter issues. Try reduce the file size from ${prettyBytes(\n imageSize,\n )} to ${prettyBytes(imageSizeLimit)} or lower.`;\n }\n case 'imageMatrixLimit': {\n const { imageMatrix, imageMatrixLimit, width, height } = details;\n const { newWidth, newHeight } = scaleDimension(\n imageMatrix,\n imageMatrixLimit,\n width,\n height,\n );\n return t`Dimension too large. Uploading might encounter issues. Try reduce dimension from ${i18n.number(\n width,\n )}×${i18n.number(height)}px to ${i18n.number(newWidth)}×${i18n.number(\n newHeight,\n )}px.`;\n }\n case 'videoSizeLimit': {\n const { videoSize, videoSizeLimit } = details;\n return t`File size too large. Uploading might encounter issues. Try reduce the file size from ${prettyBytes(\n videoSize,\n )} to ${prettyBytes(videoSizeLimit)} or lower.`;\n }\n case 'videoMatrixLimit': {\n const { videoMatrix, videoMatrixLimit, width, height } = details;\n const { newWidth, newHeight } = scaleDimension(\n videoMatrix,\n videoMatrixLimit,\n width,\n height,\n );\n return t`Dimension too large. Uploading might encounter issues. Try reduce dimension from ${i18n.number(\n width,\n )}×${i18n.number(height)}px to ${i18n.number(newWidth)}×${i18n.number(\n newHeight,\n )}px.`;\n }\n case 'videoFrameRateLimit': {\n // Not possible to detect this on client-side for now\n return t`Frame rate too high. Uploading might encounter issues.`;\n }\n }\n };\n\n return (\n <>\n <div class=\"media-attachment\">\n <div\n class=\"media-preview\"\n tabIndex=\"0\"\n onClick={() => {\n setShowModal(true);\n }}\n >\n {suffixType === 'image' ? (\n <img\n src={url}\n alt=\"\"\n onLoad={(e) => {\n if (!checkMaxError) return;\n const { naturalWidth, naturalHeight } = e.target;\n setImageMatrix({\n matrix: naturalWidth * naturalHeight,\n width: naturalWidth,\n height: naturalHeight,\n });\n }}\n />\n ) : suffixType === 'video' || suffixType === 'gifv' ? (\n <video\n src={url + '#t=0.1'} // Make Safari show 1st-frame preview\n playsinline\n muted\n disablePictureInPicture\n preload=\"metadata\"\n onLoadedMetadata={(e) => {\n if (!checkMaxError) return;\n const { videoWidth, videoHeight } = e.target;\n if (videoWidth && videoHeight) {\n setVideoMatrix({\n matrix: videoWidth * videoHeight,\n width: videoWidth,\n height: videoHeight,\n });\n }\n }}\n />\n ) : suffixType === 'audio' ? (\n <audio src={url} controls />\n ) : null}\n </div>\n {descTextarea}\n <div class=\"media-aside\">\n <button\n type=\"button\"\n class=\"plain close-button\"\n disabled={disabled}\n onClick={onRemove}\n >\n <Icon icon=\"x\" alt={t`Remove`} />\n </button>\n {!!maxError && (\n <button\n type=\"button\"\n class=\"media-error\"\n title={maxErrorText(maxError)}\n onClick={() => {\n if (maxErrorToast.current) {\n maxErrorToast.current.hideToast();\n }\n maxErrorToast.current = showToast({\n text: maxErrorText(maxError),\n duration: 10_000,\n });\n }}\n >\n <Icon icon=\"alert\" alt={t`Error`} />\n </button>\n )}\n </div>\n </div>\n {showModal && (\n <Modal\n onClose={() => {\n setShowModal(false);\n }}\n >\n <div id=\"media-sheet\" class=\"sheet sheet-max\">\n <button\n type=\"button\"\n class=\"sheet-close\"\n onClick={() => {\n setShowModal(false);\n }}\n >\n <Icon icon=\"x\" alt={t`Close`} />\n </button>\n <header>\n <h2>\n {\n {\n image: t`Edit image description`,\n video: t`Edit video description`,\n audio: t`Edit audio description`,\n }[suffixType]\n }\n </h2>\n </header>\n <main tabIndex=\"-1\">\n <div class=\"media-preview\">\n {suffixType === 'image' ? (\n <img src={url} alt=\"\" />\n ) : suffixType === 'video' || suffixType === 'gifv' ? (\n <video src={url} playsinline controls />\n ) : suffixType === 'audio' ? (\n <audio src={url} controls />\n ) : null}\n </div>\n <div class=\"media-form\">\n {descTextarea}\n <footer>\n {suffixType === 'image' &&\n /^(png|jpe?g|gif|webp)$/i.test(subtype) &&\n !!states.settings.mediaAltGenerator &&\n !!IMG_ALT_API_URL && (\n <Menu2\n portal={{\n target: document.body,\n }}\n containerProps={{\n style: {\n zIndex: 1001,\n },\n }}\n align=\"center\"\n position=\"anchor\"\n overflow=\"auto\"\n menuButton={\n <button type=\"button\" class=\"plain\">\n <Icon icon=\"more\" size=\"l\" alt={t`More`} />\n </button>\n }\n >\n <MenuItem\n disabled={uiState === 'loading'}\n onClick={() => {\n setUIState('loading');\n toastRef.current = showToast({\n text: t`Generating description. Please wait…`,\n duration: -1,\n });\n // POST with multipart\n (async function () {\n try {\n const body = new FormData();\n body.append('image', file);\n const response = await fetch(IMG_ALT_API_URL, {\n method: 'POST',\n body,\n }).then((r) => r.json());\n if (response.error) {\n throw new Error(response.error);\n }\n setDescription(response.description);\n } catch (e) {\n console.error(e);\n showToast(\n e.message\n ? t`Failed to generate description: ${e.message}`\n : t`Failed to generate description`,\n );\n } finally {\n setUIState('default');\n toastRef.current?.hideToast?.();\n }\n })();\n }}\n >\n <Icon icon=\"sparkles2\" />\n {lang && lang !== 'en' ? (\n <small>\n <Trans>Generate description…</Trans>\n <br />\n (English)\n </small>\n ) : (\n <span>\n <Trans>Generate description…</Trans>\n </span>\n )}\n </MenuItem>\n {!!lang && lang !== 'en' && (\n <MenuItem\n disabled={uiState === 'loading'}\n onClick={() => {\n setUIState('loading');\n toastRef.current = showToast({\n text: t`Generating description. Please wait…`,\n duration: -1,\n });\n // POST with multipart\n (async function () {\n try {\n const body = new FormData();\n body.append('image', file);\n const params = `?lang=${lang}`;\n const response = await fetch(\n IMG_ALT_API_URL + params,\n {\n method: 'POST',\n body,\n },\n ).then((r) => r.json());\n if (response.error) {\n throw new Error(response.error);\n }\n setDescription(response.description);\n } catch (e) {\n console.error(e);\n showToast(\n t`Failed to generate description${\n e?.message ? `: ${e.message}` : ''\n }`,\n );\n } finally {\n setUIState('default');\n toastRef.current?.hideToast?.();\n }\n })();\n }}\n >\n <Icon icon=\"sparkles2\" />\n <small>\n <Trans>Generate description…</Trans>\n <br />\n <Trans>\n ({localeCode2Text(lang)}){' '}\n <span class=\"more-insignificant\">\n — experimental\n </span>\n </Trans>\n </small>\n </MenuItem>\n )}\n </Menu2>\n )}\n <button\n type=\"button\"\n class=\"light block\"\n onClick={() => {\n setShowModal(false);\n }}\n disabled={uiState === 'loading'}\n >\n <Trans>Done</Trans>\n </button>\n </footer>\n </div>\n </main>\n </div>\n </Modal>\n )}\n </>\n );\n}\n\nexport default MediaAttachment;\n","import { Trans, useLingui } from '@lingui/react/macro';\nimport { useEffect, useRef, useState } from 'preact/hooks';\nimport { useHotkeys } from 'react-hotkeys-hook';\nimport { useDebouncedCallback } from 'use-debounce';\n\nimport { api } from '../utils/api';\nimport { fetchRelationships } from '../utils/relationships';\n\nimport AccountBlock from './account-block';\nimport Icon from './icon';\nimport Loader from './loader';\n\nfunction MentionModal({\n onClose = () => {},\n onSelect = () => {},\n defaultSearchTerm,\n}) {\n const { t } = useLingui();\n const { masto } = api();\n const [uiState, setUIState] = useState('default');\n const [accounts, setAccounts] = useState([]);\n const [relationshipsMap, setRelationshipsMap] = useState({});\n\n const [selectedIndex, setSelectedIndex] = useState(0);\n\n const loadRelationships = async (accounts) => {\n if (!accounts?.length) return;\n const relationships = await fetchRelationships(accounts, relationshipsMap);\n if (relationships) {\n setRelationshipsMap({\n ...relationshipsMap,\n ...relationships,\n });\n }\n };\n\n const loadAccounts = (term) => {\n if (!term) return;\n setUIState('loading');\n (async () => {\n try {\n const accounts = await masto.v1.accounts.search.list({\n q: term,\n limit: 40,\n resolve: false,\n });\n setAccounts(accounts);\n loadRelationships(accounts);\n setUIState('default');\n } catch (e) {\n setUIState('error');\n console.error(e);\n }\n })();\n };\n\n const debouncedLoadAccounts = useDebouncedCallback(loadAccounts, 1000);\n\n useEffect(() => {\n loadAccounts();\n }, [loadAccounts]);\n\n const inputRef = useRef();\n useEffect(() => {\n if (inputRef.current) {\n inputRef.current.focus();\n // Put cursor at the end\n if (inputRef.current.value) {\n inputRef.current.selectionStart = inputRef.current.value.length;\n inputRef.current.selectionEnd = inputRef.current.value.length;\n }\n }\n }, []);\n\n useEffect(() => {\n if (defaultSearchTerm) {\n loadAccounts(defaultSearchTerm);\n }\n }, [defaultSearchTerm]);\n\n const selectAccount = (account) => {\n const socialAddress = account.acct;\n onSelect(socialAddress);\n onClose();\n };\n\n useHotkeys(\n 'enter',\n () => {\n const selectedAccount = accounts[selectedIndex];\n if (selectedAccount) {\n selectAccount(selectedAccount);\n }\n },\n {\n preventDefault: true,\n enableOnFormTags: ['input'],\n useKey: true,\n ignoreEventWhen: (e) => e.metaKey || e.ctrlKey || e.altKey || e.shiftKey,\n },\n );\n\n const listRef = useRef();\n useHotkeys(\n 'down',\n () => {\n if (selectedIndex < accounts.length - 1) {\n setSelectedIndex(selectedIndex + 1);\n } else {\n setSelectedIndex(0);\n }\n setTimeout(() => {\n const selectedItem = listRef.current.querySelector('.selected');\n if (selectedItem) {\n selectedItem.scrollIntoView({\n behavior: 'smooth',\n block: 'center',\n inline: 'center',\n });\n }\n }, 1);\n },\n {\n preventDefault: true,\n enableOnFormTags: ['input'],\n useKey: true,\n ignoreEventWhen: (e) => e.metaKey || e.ctrlKey || e.altKey || e.shiftKey,\n },\n );\n\n useHotkeys(\n 'up',\n () => {\n if (selectedIndex > 0) {\n setSelectedIndex(selectedIndex - 1);\n } else {\n setSelectedIndex(accounts.length - 1);\n }\n setTimeout(() => {\n const selectedItem = listRef.current.querySelector('.selected');\n if (selectedItem) {\n selectedItem.scrollIntoView({\n behavior: 'smooth',\n block: 'center',\n inline: 'center',\n });\n }\n }, 1);\n },\n {\n preventDefault: true,\n enableOnFormTags: ['input'],\n useKey: true,\n ignoreEventWhen: (e) => e.metaKey || e.ctrlKey || e.altKey || e.shiftKey,\n },\n );\n\n return (\n <div id=\"mention-sheet\" class=\"sheet\">\n {!!onClose && (\n <button type=\"button\" class=\"sheet-close\" onClick={onClose}>\n <Icon icon=\"x\" alt={t`Close`} />\n </button>\n )}\n <header>\n <form\n onSubmit={(e) => {\n e.preventDefault();\n debouncedLoadAccounts.flush?.();\n // const searchTerm = inputRef.current.value;\n // debouncedLoadAccounts(searchTerm);\n }}\n >\n <input\n ref={inputRef}\n required\n type=\"search\"\n class=\"block\"\n placeholder={t`Search accounts`}\n onInput={(e) => {\n const { value } = e.target;\n debouncedLoadAccounts(value);\n }}\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellCheck=\"false\"\n dir=\"auto\"\n defaultValue={defaultSearchTerm || ''}\n />\n </form>\n </header>\n <main>\n {accounts?.length > 0 ? (\n <ul\n ref={listRef}\n class={`accounts-list ${uiState === 'loading' ? 'loading' : ''}`}\n >\n {accounts.map((account, i) => {\n const relationship = relationshipsMap[account.id];\n return (\n <li\n key={account.id}\n class={i === selectedIndex ? 'selected' : ''}\n >\n <AccountBlock\n avatarSize=\"xxl\"\n account={account}\n relationship={relationship}\n showStats\n showActivity\n />\n <button\n type=\"button\"\n class=\"plain2\"\n onClick={() => {\n selectAccount(account);\n }}\n >\n <Icon icon=\"plus\" size=\"xl\" alt={t`Add`} />\n </button>\n </li>\n );\n })}\n </ul>\n ) : uiState === 'loading' ? (\n <div class=\"ui-state\">\n <Loader abrupt />\n </div>\n ) : uiState === 'error' ? (\n <div class=\"ui-state\">\n <p>\n <Trans>Error loading accounts</Trans>\n </p>\n </div>\n ) : null}\n </main>\n </div>\n );\n}\n\nexport default MentionModal;\n","import './compose.css';\n\nimport { msg, plural } from '@lingui/core/macro';\nimport { Trans, useLingui } from '@lingui/react/macro';\nimport { MenuDivider, MenuItem } from '@szhsin/react-menu';\nimport { deepEqual } from 'fast-equals';\nimport { useEffect, useMemo, useRef, useState } from 'preact/hooks';\nimport { useHotkeys } from 'react-hotkeys-hook';\nimport stringLength from 'string-length';\nimport { uid } from 'uid/single';\nimport useResizeObserver from 'use-resize-observer';\nimport { useSnapshot } from 'valtio';\n\nimport supportedLanguages from '../data/status-supported-languages';\nimport { api, getPreferences } from '../utils/api';\nimport db from '../utils/db';\nimport localeMatch from '../utils/locale-match';\nimport localeCode2Text from '../utils/localeCode2Text';\nimport mem from '../utils/mem';\nimport openCompose from '../utils/open-compose';\nimport RTF from '../utils/relative-time-format';\nimport showToast from '../utils/show-toast';\nimport states, { saveStatus } from '../utils/states';\nimport store from '../utils/store';\nimport {\n getCurrentAccount,\n getCurrentAccountNS,\n getCurrentInstanceConfiguration,\n} from '../utils/store-utils';\nimport supports from '../utils/supports';\nimport urlRegexObj from '../utils/url-regex';\nimport useCloseWatcher from '../utils/useCloseWatcher';\nimport useInterval from '../utils/useInterval';\nimport visibilityIconsMap from '../utils/visibility-icons-map';\nimport visibilityText from '../utils/visibility-text';\n\nimport AccountBlock from './account-block';\n// import Avatar from './avatar';\nimport CameraCaptureInput, {\n supportsCameraCapture,\n} from './camera-capture-input';\nimport CharCountMeter from './char-count-meter';\nimport ComposePoll, { expiryOptions } from './compose-poll';\nimport Textarea from './compose-textarea';\nimport CustomEmojisModal from './custom-emojis-modal';\nimport FilePickerInput from './file-picker-input';\nimport GIFPickerModal from './gif-picker-modal';\nimport Icon from './icon';\nimport Loader from './loader';\nimport MediaAttachment from './media-attachment';\nimport MentionModal from './mention-modal';\nimport Menu2 from './menu2';\nimport Modal from './modal';\nimport ScheduledAtField, {\n getLocalTimezoneName,\n MIN_SCHEDULED_AT,\n} from './ScheduledAtField';\nimport Status from './status';\nimport TextExpander from './text-expander';\n\nconst supportedLanguagesMap = supportedLanguages.reduce((acc, l) => {\n const [code, common, native] = l;\n acc[code] = {\n common,\n native,\n };\n return acc;\n}, {});\n\n/* NOTES:\n - Max character limit includes BOTH status text and Content Warning text\n*/\n\nconst expirySeconds = Object.keys(expiryOptions);\nconst oneDay = 24 * 60 * 60;\n\nconst expiresInFromExpiresAt = (expiresAt) => {\n if (!expiresAt) return oneDay;\n const delta = (Date.parse(expiresAt) - Date.now()) / 1000;\n return expirySeconds.find((s) => s >= delta) || oneDay;\n};\n\nconst DEFAULT_LANG = localeMatch(\n [new Intl.DateTimeFormat().resolvedOptions().locale, ...navigator.languages],\n supportedLanguages.map((l) => l[0]),\n 'en',\n);\n\n// https://github.com/mastodon/mastodon/blob/c4a429ed47e85a6bbf0d470a41cc2f64cf120c19/app/javascript/mastodon/features/compose/util/counter.js\nconst usernameRegex = /(^|[^\\/\\w])@(([a-z0-9_]+)@[a-z0-9\\.\\-]+[a-z0-9]+)/gi;\nconst urlPlaceholder = '$2xxxxxxxxxxxxxxxxxxxxxxx';\nfunction countableText(inputText) {\n return inputText\n .replace(urlRegexObj, urlPlaceholder)\n .replace(usernameRegex, '$1@$3');\n}\n\n// const rtf = new Intl.RelativeTimeFormat();\nconst LF = mem((locale) => new Intl.ListFormat(locale || undefined));\n\nconst ADD_LABELS = {\n camera: msg`Take photo or video`,\n media: msg`Add media`,\n customEmoji: msg`Add custom emoji`,\n gif: msg`Add GIF`,\n poll: msg`Add poll`,\n sensitive: msg`Add content warning`,\n scheduledPost: msg`Schedule post`,\n};\n\nconst DEFAULT_SCHEDULED_AT = Math.max(10 * 60 * 1000, MIN_SCHEDULED_AT); // 10 mins\n\nfunction Compose({\n onClose,\n replyToStatus,\n editStatus,\n draftStatus,\n standalone,\n hasOpener,\n}) {\n const { i18n, _, t } = useLingui();\n const rtf = RTF(i18n.locale);\n const lf = LF(i18n.locale);\n\n console.warn('RENDER COMPOSER');\n const { masto, instance } = api();\n const [uiState, setUIState] = useState('default');\n const UID = useRef(draftStatus?.uid || uid());\n console.log('Compose UID', UID.current);\n\n const currentAccount = getCurrentAccount();\n const currentAccountInfo = currentAccount.info;\n\n const configuration = getCurrentInstanceConfiguration();\n console.log('⚙️ Configuration', configuration);\n\n const {\n statuses: {\n maxCharacters,\n maxMediaAttachments, // Beware: it can be undefined!\n charactersReservedPerUrl,\n } = {},\n mediaAttachments: {\n supportedMimeTypes,\n imageSizeLimit,\n imageMatrixLimit,\n videoSizeLimit,\n videoMatrixLimit,\n videoFrameRateLimit,\n descriptionLimit,\n } = {},\n polls: {\n maxOptions,\n maxCharactersPerOption,\n maxExpiration,\n minExpiration,\n } = {},\n } = configuration || {};\n const supportedImagesVideosTypes = supportedMimeTypes?.filter((mimeType) =>\n /^(image|video)/i.test(mimeType),\n );\n\n const textareaRef = useRef();\n const spoilerTextRef = useRef();\n\n const [visibility, setVisibility] = useState('public');\n const [sensitive, setSensitive] = useState(false);\n const [sensitiveMedia, setSensitiveMedia] = useState(false);\n const [language, setLanguage] = useState(\n store.session.get('currentLanguage') || DEFAULT_LANG,\n );\n const prevLanguage = useRef(language);\n const [mediaAttachments, setMediaAttachments] = useState([]);\n const [poll, setPoll] = useState(null);\n const [scheduledAt, setScheduledAt] = useState(null);\n\n const prefs = getPreferences();\n\n const oninputTextarea = () => {\n if (!textareaRef.current) return;\n textareaRef.current.dispatchEvent(new Event('input'));\n };\n const focusTextarea = () => {\n setTimeout(() => {\n if (!textareaRef.current) return;\n // status starts with newline or space, focus on first position\n if (/^\\n|\\s/.test(draftStatus?.status)) {\n textareaRef.current.selectionStart = 0;\n textareaRef.current.selectionEnd = 0;\n }\n console.debug('FOCUS textarea');\n textareaRef.current?.focus();\n }, 300);\n };\n const insertTextAtCursor = ({ targetElement, text }) => {\n if (!targetElement) return;\n\n const { selectionStart, selectionEnd, value } = targetElement;\n let textBeforeInsert = value.slice(0, selectionStart);\n\n // Remove zero-width space from end of text\n textBeforeInsert = textBeforeInsert.replace(/\\u200B$/, '');\n\n const spaceBeforeInsert = textBeforeInsert\n ? /[\\s\\t\\n\\r]$/.test(textBeforeInsert)\n ? ''\n : ' '\n : '';\n\n const textAfterInsert = value.slice(selectionEnd);\n const spaceAfterInsert = /^[\\s\\t\\n\\r]/.test(textAfterInsert) ? '' : ' ';\n\n const newText =\n textBeforeInsert +\n spaceBeforeInsert +\n text +\n spaceAfterInsert +\n textAfterInsert;\n\n targetElement.value = newText;\n targetElement.selectionStart = targetElement.selectionEnd =\n selectionEnd + text.length + spaceAfterInsert.length;\n targetElement.focus();\n targetElement.dispatchEvent(new Event('input'));\n };\n\n const lastFocusedFieldRef = useRef(null);\n const lastFocusedEmojiFieldRef = useRef(null);\n const focusLastFocusedField = () => {\n setTimeout(() => {\n if (!lastFocusedFieldRef.current) return;\n lastFocusedFieldRef.current.focus();\n }, 0);\n };\n const composeContainerRef = useRef(null);\n useEffect(() => {\n const handleFocus = (e) => {\n // Toggle focused if in or out if any fields are focused\n composeContainerRef.current.classList.toggle(\n 'focused',\n e.type === 'focusin',\n );\n\n const target = e.target;\n if (target.hasAttribute('data-allow-custom-emoji')) {\n lastFocusedEmojiFieldRef.current = target;\n }\n const isFormElement = ['INPUT', 'BUTTON', 'SELECT', 'TEXTAREA'].includes(\n target.tagName,\n );\n if (isFormElement) {\n lastFocusedFieldRef.current = target;\n }\n };\n\n const composeContainer = composeContainerRef.current;\n if (composeContainer) {\n composeContainer.addEventListener('focusin', handleFocus);\n composeContainer.addEventListener('focusout', handleFocus);\n }\n\n return () => {\n if (composeContainer) {\n composeContainer.removeEventListener('focusin', handleFocus);\n composeContainer.removeEventListener('focusout', handleFocus);\n }\n };\n }, []);\n\n useEffect(() => {\n if (replyToStatus) {\n const { spoilerText, visibility, language, sensitive } = replyToStatus;\n if (spoilerText && spoilerTextRef.current) {\n spoilerTextRef.current.value = spoilerText;\n }\n const mentions = new Set([\n replyToStatus.account.acct,\n ...replyToStatus.mentions.map((m) => m.acct),\n ]);\n const allMentions = [...mentions].filter(\n (m) => m !== currentAccountInfo.acct,\n );\n if (allMentions.length > 0) {\n textareaRef.current.value = `${allMentions\n .map((m) => `@${m}`)\n .join(' ')} `;\n oninputTextarea();\n }\n focusTextarea();\n setVisibility(\n visibility === 'public' && prefs['posting:default:visibility']\n ? prefs['posting:default:visibility'].toLowerCase()\n : visibility,\n );\n setLanguage(\n language ||\n prefs['posting:default:language']?.toLowerCase() ||\n DEFAULT_LANG,\n );\n setSensitive(!!spoilerText);\n } else if (editStatus) {\n const { visibility, language, sensitive, poll, mediaAttachments } =\n editStatus;\n const composablePoll = !!poll?.options && {\n ...poll,\n options: poll.options.map((o) => o?.title || o),\n expiresIn: poll?.expiresIn || expiresInFromExpiresAt(poll.expiresAt),\n };\n setUIState('loading');\n (async () => {\n try {\n const statusSource = await masto.v1.statuses\n .$select(editStatus.id)\n .source.fetch();\n console.log({ statusSource });\n const { text, spoilerText } = statusSource;\n textareaRef.current.value = text;\n textareaRef.current.dataset.source = text;\n oninputTextarea();\n focusTextarea();\n spoilerTextRef.current.value = spoilerText;\n setVisibility(visibility);\n setLanguage(\n language ||\n prefs['posting:default:language']?.toLowerCase() ||\n DEFAULT_LANG,\n );\n setSensitive(sensitive);\n if (composablePoll) setPoll(composablePoll);\n setMediaAttachments(mediaAttachments);\n setUIState('default');\n } catch (e) {\n console.error(e);\n alert(e?.reason || e);\n setUIState('error');\n }\n })();\n } else {\n focusTextarea();\n console.log('Apply prefs', prefs);\n if (prefs['posting:default:visibility']) {\n setVisibility(prefs['posting:default:visibility'].toLowerCase());\n }\n if (prefs['posting:default:language']) {\n setLanguage(prefs['posting:default:language'].toLowerCase());\n }\n if (prefs['posting:default:sensitive']) {\n setSensitive(!!prefs['posting:default:sensitive']);\n }\n }\n if (draftStatus) {\n const {\n status,\n spoilerText,\n visibility,\n language,\n sensitive,\n sensitiveMedia,\n poll,\n mediaAttachments,\n scheduledAt,\n } = draftStatus;\n const composablePoll = !!poll?.options && {\n ...poll,\n options: poll.options.map((o) => o?.title || o),\n expiresIn: poll?.expiresIn || expiresInFromExpiresAt(poll.expiresAt),\n };\n textareaRef.current.value = status;\n oninputTextarea();\n focusTextarea();\n if (spoilerText) spoilerTextRef.current.value = spoilerText;\n if (visibility) setVisibility(visibility);\n setLanguage(\n language ||\n prefs['posting:default:language']?.toLowerCase() ||\n DEFAULT_LANG,\n );\n if (sensitiveMedia !== null) setSensitiveMedia(sensitiveMedia);\n if (sensitive !== null) setSensitive(sensitive);\n if (composablePoll) setPoll(composablePoll);\n if (mediaAttachments) setMediaAttachments(mediaAttachments);\n if (scheduledAt) setScheduledAt(scheduledAt);\n }\n }, [draftStatus, editStatus, replyToStatus]);\n\n // focus textarea when state.composerState.minimized turns false\n const snapStates = useSnapshot(states);\n useEffect(() => {\n if (!snapStates.composerState.minimized) {\n focusTextarea();\n }\n }, [snapStates.composerState.minimized]);\n\n const formRef = useRef();\n\n const beforeUnloadCopy = t`You have unsaved changes. Discard this post?`;\n const canClose = () => {\n const { value, dataset } = textareaRef.current;\n\n // check if loading\n if (uiState === 'loading') {\n console.log('canClose', { uiState });\n return false;\n }\n\n // check for status and media attachments\n const hasValue = (value || '')\n .trim()\n .replace(/^\\p{White_Space}+|\\p{White_Space}+$/gu, '');\n const hasMediaAttachments = mediaAttachments.length > 0;\n if (!hasValue && !hasMediaAttachments) {\n console.log('canClose', { value, mediaAttachments });\n return true;\n }\n\n // check if all media attachments have IDs\n const hasIDMediaAttachments =\n mediaAttachments.length > 0 &&\n mediaAttachments.every((media) => media.id);\n if (hasIDMediaAttachments) {\n console.log('canClose', { hasIDMediaAttachments });\n return true;\n }\n\n // check if status contains only \"@acct\", if replying\n const isSelf = replyToStatus?.account.id === currentAccountInfo.id;\n const hasOnlyAcct =\n replyToStatus && value.trim() === `@${replyToStatus.account.acct}`;\n // TODO: check for mentions, or maybe just generic \"@username<space>\", including multiple mentions like \"@username1<space>@username2<space>\"\n if (!isSelf && hasOnlyAcct) {\n console.log('canClose', { isSelf, hasOnlyAcct });\n return true;\n }\n\n // check if status is same with source\n const sameWithSource = value === dataset?.source;\n if (sameWithSource) {\n console.log('canClose', { sameWithSource });\n return true;\n }\n\n console.log('canClose', {\n value,\n hasMediaAttachments,\n hasIDMediaAttachments,\n poll,\n isSelf,\n hasOnlyAcct,\n sameWithSource,\n uiState,\n });\n\n return false;\n };\n\n const confirmClose = () => {\n if (!canClose()) {\n const yes = confirm(beforeUnloadCopy);\n return yes;\n }\n return true;\n };\n\n useEffect(() => {\n // Show warning if user tries to close window with unsaved changes\n const handleBeforeUnload = (e) => {\n if (!canClose()) {\n e.preventDefault();\n e.returnValue = beforeUnloadCopy;\n }\n };\n window.addEventListener('beforeunload', handleBeforeUnload, {\n capture: true,\n });\n return () =>\n window.removeEventListener('beforeunload', handleBeforeUnload, {\n capture: true,\n });\n }, []);\n\n const getCharCount = () => {\n const { value } = textareaRef.current;\n const { value: spoilerText } = spoilerTextRef.current;\n return stringLength(countableText(value)) + stringLength(spoilerText);\n };\n const updateCharCount = () => {\n const count = getCharCount();\n states.composerCharacterCount = count;\n };\n useEffect(updateCharCount, []);\n\n const supportsCloseWatcher = window.CloseWatcher;\n const escDownRef = useRef(false);\n useHotkeys(\n 'esc',\n () => {\n escDownRef.current = true;\n // This won't be true if this event is already handled and not propagated 🤞\n },\n {\n enabled: !supportsCloseWatcher,\n enableOnFormTags: true,\n useKey: true,\n ignoreEventWhen: (e) => e.metaKey || e.ctrlKey || e.altKey || e.shiftKey,\n },\n );\n useHotkeys(\n 'esc',\n () => {\n if (!standalone && escDownRef.current && confirmClose()) {\n onClose();\n }\n escDownRef.current = false;\n },\n {\n enabled: !supportsCloseWatcher,\n enableOnFormTags: true,\n // Use keyup because Esc keydown will close the confirm dialog on Safari\n keyup: true,\n ignoreEventWhen: (e) => {\n const modals = document.querySelectorAll('#modal-container > *');\n const hasModal = !!modals;\n const hasOnlyComposer =\n modals.length === 1 && modals[0].querySelector('#compose-container');\n return (\n (hasModal && !hasOnlyComposer) ||\n e.metaKey ||\n e.ctrlKey ||\n e.altKey ||\n e.shiftKey\n );\n },\n useKey: true,\n },\n );\n useCloseWatcher(() => {\n if (!standalone && confirmClose()) {\n onClose();\n }\n }, []);\n\n const prevBackgroundDraft = useRef({});\n const draftKey = () => {\n const ns = getCurrentAccountNS();\n return `${ns}#${UID.current}`;\n };\n const saveUnsavedDraft = () => {\n // Not enabling this for editing status\n // I don't think this warrant a draft mode for a status that's already posted\n // Maybe it could be a big edit change but it should be rare\n if (editStatus) return;\n if (states.composerState.minimized) return;\n const key = draftKey();\n const backgroundDraft = {\n key,\n replyTo: replyToStatus\n ? {\n /* Smaller payload of replyToStatus. Reasons:\n - No point storing whole thing\n - Could have media attachments\n - Could be deleted/edited later\n */\n id: replyToStatus.id,\n account: {\n id: replyToStatus.account.id,\n username: replyToStatus.account.username,\n acct: replyToStatus.account.acct,\n },\n }\n : null,\n draftStatus: {\n uid: UID.current,\n status: textareaRef.current.value,\n spoilerText: spoilerTextRef.current.value,\n visibility,\n language,\n sensitive,\n sensitiveMedia,\n poll,\n mediaAttachments,\n scheduledAt,\n },\n };\n if (\n !deepEqual(backgroundDraft, prevBackgroundDraft.current) &&\n !canClose()\n ) {\n console.debug('not equal', backgroundDraft, prevBackgroundDraft.current);\n db.drafts\n .set(key, {\n ...backgroundDraft,\n state: 'unsaved',\n updatedAt: Date.now(),\n })\n .then(() => {\n console.debug('DRAFT saved', key, backgroundDraft);\n })\n .catch((e) => {\n console.error('DRAFT failed', key, e);\n });\n prevBackgroundDraft.current = structuredClone(backgroundDraft);\n }\n };\n useInterval(saveUnsavedDraft, 5000); // background save every 5s\n useEffect(() => {\n saveUnsavedDraft();\n // If unmounted, means user discarded the draft\n // Also means pop-out 🙈, but it's okay because the pop-out will persist the ID and re-create the draft\n return () => {\n db.drafts.del(draftKey());\n };\n }, []);\n\n useEffect(() => {\n const handleItems = (e) => {\n const { items } = e.clipboardData || e.dataTransfer;\n const files = [];\n const unsupportedFiles = [];\n for (let i = 0; i < items.length; i++) {\n const item = items[i];\n if (item.kind === 'file') {\n const file = item.getAsFile();\n if (\n supportedMimeTypes !== undefined &&\n !supportedMimeTypes.includes(file.type)\n ) {\n unsupportedFiles.push(file);\n } else {\n files.push(file);\n }\n }\n }\n if (unsupportedFiles.length > 0) {\n alert(\n plural(unsupportedFiles.length, {\n one: `File ${unsupportedFiles[0].name} is not supported.`,\n other: `Files ${lf.format(\n unsupportedFiles.map((f) => f.name),\n )} are not supported.`,\n }),\n );\n }\n if (files.length > 0 && mediaAttachments.length >= maxMediaAttachments) {\n alert(\n plural(maxMediaAttachments, {\n one: 'You can only attach up to 1 file.',\n other: 'You can only attach up to # files.',\n }),\n );\n return;\n }\n console.log({ files });\n if (files.length > 0) {\n e.preventDefault();\n e.stopPropagation();\n // Auto-cut-off files to avoid exceeding maxMediaAttachments\n let allowedFiles = files;\n if (maxMediaAttachments !== undefined) {\n const max = maxMediaAttachments - mediaAttachments.length;\n allowedFiles = allowedFiles.slice(0, max);\n if (allowedFiles.length <= 0) {\n alert(\n plural(maxMediaAttachments, {\n one: 'You can only attach up to 1 file.',\n other: 'You can only attach up to # files.',\n }),\n );\n return;\n }\n }\n const mediaFiles = allowedFiles.map((file) => ({\n file,\n type: file.type,\n size: file.size,\n url: URL.createObjectURL(file),\n id: null,\n description: null,\n }));\n setMediaAttachments([...mediaAttachments, ...mediaFiles]);\n }\n };\n window.addEventListener('paste', handleItems);\n const handleDragover = (e) => {\n // Prevent default if there's files\n if (e.dataTransfer.items.length > 0) {\n e.preventDefault();\n e.stopPropagation();\n }\n };\n window.addEventListener('dragover', handleDragover);\n window.addEventListener('drop', handleItems);\n return () => {\n window.removeEventListener('paste', handleItems);\n window.removeEventListener('dragover', handleDragover);\n window.removeEventListener('drop', handleItems);\n };\n }, [mediaAttachments]);\n\n const [showMentionPicker, setShowMentionPicker] = useState(false);\n const [showEmoji2Picker, setShowEmoji2Picker] = useState(false);\n const [showGIFPicker, setShowGIFPicker] = useState(false);\n\n const [autoDetectedLanguages, setAutoDetectedLanguages] = useState(null);\n const [topSupportedLanguages, restSupportedLanguages] = useMemo(() => {\n const topLanguages = [];\n const restLanguages = [];\n const { contentTranslationHideLanguages = [] } = states.settings;\n supportedLanguages.forEach((l) => {\n const [code] = l;\n if (\n code === language ||\n code === prevLanguage.current ||\n code === DEFAULT_LANG ||\n contentTranslationHideLanguages.includes(code) ||\n (autoDetectedLanguages?.length && autoDetectedLanguages.includes(code))\n ) {\n topLanguages.push(l);\n } else {\n restLanguages.push(l);\n }\n });\n topLanguages.sort(([codeA, commonA], [codeB, commonB]) => {\n if (codeA === language) return -1;\n if (codeB === language) return 1;\n return commonA.localeCompare(commonB);\n });\n restLanguages.sort(([codeA, commonA], [codeB, commonB]) =>\n commonA.localeCompare(commonB),\n );\n return [topLanguages, restLanguages];\n }, [language, autoDetectedLanguages]);\n\n const replyToStatusMonthsAgo = useMemo(\n () =>\n !!replyToStatus?.createdAt &&\n Math.floor(\n (Date.now() - Date.parse(replyToStatus.createdAt)) /\n (1000 * 60 * 60 * 24 * 30),\n ),\n [replyToStatus],\n );\n\n const onMinimize = () => {\n saveUnsavedDraft();\n states.composerState.minimized = true;\n };\n\n const mediaButtonDisabled =\n uiState === 'loading' ||\n (maxMediaAttachments !== undefined &&\n mediaAttachments.length >= maxMediaAttachments) ||\n !!poll;\n\n const cwButtonDisabled = uiState === 'loading' || !!sensitive;\n const onCWButtonClick = () => {\n setSensitive(true);\n setTimeout(() => {\n spoilerTextRef.current?.focus();\n }, 0);\n };\n\n // If maxOptions is not defined or defined and is greater than 1, show poll button\n const showPollButton = maxOptions == null || maxOptions > 1;\n const pollButtonDisabled =\n uiState === 'loading' || !!poll || !!mediaAttachments.length;\n const onPollButtonClick = () => {\n setPoll({\n options: ['', ''],\n expiresIn: 24 * 60 * 60, // 1 day\n multiple: false,\n });\n // Focus first choice field\n setTimeout(() => {\n composeContainerRef.current\n ?.querySelector('.poll-choice input[type=\"text\"]')\n ?.focus();\n }, 0);\n };\n\n const highlightLanguageField =\n language !== prevLanguage.current ||\n (autoDetectedLanguages?.length &&\n !autoDetectedLanguages.includes(language));\n const highlightVisibilityField = visibility !== 'public';\n\n const addSubToolbarRef = useRef();\n const [showAddButton, setShowAddButton] = useState(true);\n const BUTTON_WIDTH = 42; // roughly one button width\n useResizeObserver({\n ref: addSubToolbarRef,\n box: 'border-box',\n onResize: ({ width }) => {\n // If scrollable, it's truncated\n const { scrollWidth } = addSubToolbarRef.current;\n const truncated = scrollWidth > width;\n const overTruncated = width < BUTTON_WIDTH * 4;\n setShowAddButton(overTruncated || truncated);\n addSubToolbarRef.current.hidden = overTruncated;\n },\n });\n\n const showScheduledAt = !editStatus;\n const scheduledAtButtonDisabled = uiState === 'loading' || !!scheduledAt;\n const onScheduledAtClick = () => {\n const date = new Date(Date.now() + DEFAULT_SCHEDULED_AT);\n setScheduledAt(date);\n };\n\n return (\n <div id=\"compose-container-outer\" ref={composeContainerRef}>\n <div\n id=\"compose-container\"\n tabIndex={-1}\n class={standalone ? 'standalone' : ''}\n >\n <div class=\"compose-top\">\n {currentAccountInfo?.avatarStatic && (\n // <Avatar\n // url={currentAccountInfo.avatarStatic}\n // size=\"xl\"\n // alt={currentAccountInfo.username}\n // squircle={currentAccountInfo?.bot}\n // />\n <AccountBlock\n account={currentAccountInfo}\n accountInstance={currentAccount.instanceURL}\n hideDisplayName\n useAvatarStatic\n />\n )}\n {!standalone ? (\n <span class=\"compose-controls\">\n <button\n type=\"button\"\n class=\"plain4 pop-button\"\n disabled={uiState === 'loading'}\n onClick={() => {\n // If there are non-ID media attachments (not yet uploaded), show confirmation dialog because they are not going to be passed to the new window\n // const containNonIDMediaAttachments =\n // mediaAttachments.length > 0 &&\n // mediaAttachments.some((media) => !media.id);\n // if (containNonIDMediaAttachments) {\n // const yes = confirm(\n // 'You have media attachments that are not yet uploaded. Opening a new window will discard them and you will need to re-attach them. Are you sure you want to continue?',\n // );\n // if (!yes) {\n // return;\n // }\n // }\n\n // const mediaAttachmentsWithIDs = mediaAttachments.filter(\n // (media) => media.id,\n // );\n\n const newWin = openCompose({\n editStatus,\n replyToStatus,\n draftStatus: {\n uid: UID.current,\n status: textareaRef.current.value,\n spoilerText: spoilerTextRef.current.value,\n visibility,\n language,\n sensitive,\n poll,\n mediaAttachments,\n scheduledAt,\n },\n });\n\n if (!newWin) {\n return;\n }\n\n onClose();\n }}\n >\n <Icon icon=\"popout\" alt={t`Pop out`} />\n </button>\n <button\n type=\"button\"\n class=\"plain4 min-button\"\n onClick={onMinimize}\n >\n <Icon icon=\"minimize\" alt={t`Minimize`} />\n </button>{' '}\n <button\n type=\"button\"\n class=\"plain4 close-button\"\n disabled={uiState === 'loading'}\n onClick={() => {\n if (confirmClose()) {\n onClose();\n }\n }}\n >\n <Icon icon=\"x\" alt={t`Close`} />\n </button>\n </span>\n ) : (\n hasOpener && (\n <button\n type=\"button\"\n class=\"light pop-button\"\n disabled={uiState === 'loading'}\n onClick={() => {\n // If there are non-ID media attachments (not yet uploaded), show confirmation dialog because they are not going to be passed to the new window\n // const containNonIDMediaAttachments =\n // mediaAttachments.length > 0 &&\n // mediaAttachments.some((media) => !media.id);\n // if (containNonIDMediaAttachments) {\n // const yes = confirm(\n // 'You have media attachments that are not yet uploaded. Opening a new window will discard them and you will need to re-attach them. Are you sure you want to continue?',\n // );\n // if (!yes) {\n // return;\n // }\n // }\n\n if (!window.opener) {\n alert(t`Looks like you closed the parent window.`);\n return;\n }\n\n if (window.opener.__STATES__.showCompose) {\n if (window.opener.__STATES__.composerState?.publishing) {\n alert(\n t`Looks like you already have a compose field open in the parent window and currently publishing. Please wait for it to be done and try again later.`,\n );\n return;\n }\n\n let confirmText = t`Looks like you already have a compose field open in the parent window. Popping in this window will discard the changes you made in the parent window. Continue?`;\n const yes = confirm(confirmText);\n if (!yes) return;\n }\n\n // const mediaAttachmentsWithIDs = mediaAttachments.filter(\n // (media) => media.id,\n // );\n\n onClose({\n fn: () => {\n const passData = {\n editStatus,\n replyToStatus,\n draftStatus: {\n uid: UID.current,\n status: textareaRef.current.value,\n spoilerText: spoilerTextRef.current.value,\n visibility,\n language,\n sensitive,\n sensitiveMedia,\n poll,\n mediaAttachments,\n scheduledAt,\n },\n };\n window.opener.__COMPOSE__ = passData; // Pass it here instead of `showCompose` due to some weird proxy issue again\n if (window.opener.__STATES__.showCompose) {\n window.opener.__STATES__.showCompose = false;\n setTimeout(() => {\n window.opener.__STATES__.showCompose = true;\n }, 10);\n } else {\n window.opener.__STATES__.showCompose = true;\n }\n if (window.opener.__STATES__.composerState.minimized) {\n // Maximize it\n window.opener.__STATES__.composerState.minimized = false;\n }\n },\n });\n }}\n >\n <Icon icon=\"popin\" alt={t`Pop in`} />\n </button>\n )\n )}\n </div>\n {!!replyToStatus && (\n <div class=\"status-preview\">\n <Status status={replyToStatus} size=\"s\" previewMode />\n <div class=\"status-preview-legend reply-to\">\n {replyToStatusMonthsAgo > 0 ? (\n <Trans>\n Replying to @\n {replyToStatus.account.acct || replyToStatus.account.username}\n ’s post (\n <strong>\n {rtf.format(-replyToStatusMonthsAgo, 'month')}\n </strong>\n )\n </Trans>\n ) : (\n <Trans>\n Replying to @\n {replyToStatus.account.acct || replyToStatus.account.username}\n ’s post\n </Trans>\n )}\n </div>\n </div>\n )}\n {!!editStatus && (\n <div class=\"status-preview\">\n <Status status={editStatus} size=\"s\" previewMode />\n <div class=\"status-preview-legend\">\n <Trans>Editing source post</Trans>\n </div>\n </div>\n )}\n <form\n ref={formRef}\n class={`form-visibility-${visibility}`}\n style={{\n pointerEvents: uiState === 'loading' ? 'none' : 'auto',\n opacity: uiState === 'loading' ? 0.5 : 1,\n }}\n onClick={() => {\n setTimeout(() => {\n if (!document.activeElement) {\n lastFocusedFieldRef.current?.focus?.();\n }\n }, 10);\n }}\n onKeyDown={(e) => {\n if (e.key === 'Enter' && (e.ctrlKey || e.metaKey)) {\n formRef.current.dispatchEvent(\n new Event('submit', { cancelable: true }),\n );\n }\n }}\n onSubmit={(e) => {\n e.preventDefault();\n\n const formData = new FormData(e.target);\n const entries = Object.fromEntries(formData.entries());\n console.log('ENTRIES', entries);\n let {\n status,\n visibility,\n sensitive,\n sensitiveMedia,\n spoilerText,\n scheduledAt,\n } = entries;\n\n // Pre-cleanup\n // checkboxes return \"on\" if checked\n sensitive = sensitive === 'on';\n sensitiveMedia = sensitiveMedia === 'on';\n\n // Convert datetime-local input value to RFC3339 Date string value\n scheduledAt = scheduledAt\n ? new Date(scheduledAt).toISOString()\n : undefined;\n\n // Validation\n /* Let the backend validate this\n if (stringLength(status) > maxCharacters) {\n alert(`Status is too long! Max characters: ${maxCharacters}`);\n return;\n }\n if (\n sensitive &&\n stringLength(status) + stringLength(spoilerText) > maxCharacters\n ) {\n alert(\n `Status and content warning is too long! Max characters: ${maxCharacters}`,\n );\n return;\n }\n */\n if (poll) {\n if (poll.options.length < 2) {\n alert(t`Poll must have at least 2 options`);\n return;\n }\n if (poll.options.some((option) => option === '')) {\n alert(t`Some poll choices are empty`);\n return;\n }\n }\n // TODO: check for URLs and use `charactersReservedPerUrl` to calculate max characters\n\n if (mediaAttachments.length > 0) {\n // If there are media attachments, check if they have no descriptions\n const hasNoDescriptions = mediaAttachments.some(\n (media) => !media.description?.trim?.(),\n );\n if (hasNoDescriptions) {\n const yes = confirm(\n t`Some media have no descriptions. Continue?`,\n );\n if (!yes) return;\n }\n }\n\n // Post-cleanup\n spoilerText = (sensitive && spoilerText) || undefined;\n status = status === '' ? undefined : status;\n\n // states.composerState.minimized = true;\n states.composerState.publishing = true;\n setUIState('loading');\n (async () => {\n try {\n console.log('MEDIA ATTACHMENTS', mediaAttachments);\n if (mediaAttachments.length > 0) {\n // Upload media attachments first\n const mediaPromises = mediaAttachments.map((attachment) => {\n const { file, description, id } = attachment;\n console.log('UPLOADING', attachment);\n if (id) {\n // If already uploaded\n return attachment;\n } else {\n const params = removeNullUndefined({\n file,\n description,\n });\n return masto.v2.media.create(params).then((res) => {\n if (res.id) {\n attachment.id = res.id;\n }\n return res;\n });\n }\n });\n const results = await Promise.allSettled(mediaPromises);\n\n // If any failed, return\n if (\n results.some((result) => {\n return result.status === 'rejected' || !result.value?.id;\n })\n ) {\n states.composerState.publishing = false;\n states.composerState.publishingError = true;\n setUIState('error');\n // Alert all the reasons\n results.forEach((result) => {\n if (result.status === 'rejected') {\n console.error(result);\n alert(result.reason || t`Attachment #${i} failed`);\n }\n });\n return;\n }\n\n console.log({ results, mediaAttachments });\n }\n\n /* NOTE:\n Using snakecase here because masto.js's `isObject` returns false for `params`, ONLY happens when opening in pop-out window. This is maybe due to `window.masto` variable being passed from the parent window. The check that failed is `x.constructor === Object`, so maybe the `Object` in new window is different than parent window's?\n Code: https://github.com/neet/masto.js/blob/dd0d649067b6a2b6e60fbb0a96597c373a255b00/src/serializers/is-object.ts#L2\n\n // TODO: Note above is no longer true in Masto.js v6. Revisit this.\n */\n let params = {\n status,\n // spoilerText,\n spoiler_text: spoilerText,\n language,\n sensitive: sensitive || sensitiveMedia,\n poll,\n // mediaIds: mediaAttachments.map((attachment) => attachment.id),\n media_ids: mediaAttachments.map(\n (attachment) => attachment.id,\n ),\n };\n if (editStatus && supports('@mastodon/edit-media-attributes')) {\n params.media_attributes = mediaAttachments.map(\n (attachment) => {\n return {\n id: attachment.id,\n description: attachment.description,\n // focus\n // thumbnail\n };\n },\n );\n } else if (!editStatus) {\n params.visibility = visibility;\n // params.inReplyToId = replyToStatus?.id || undefined;\n params.in_reply_to_id = replyToStatus?.id || undefined;\n params.scheduled_at = scheduledAt;\n }\n params = removeNullUndefined(params);\n console.log('POST', params);\n\n let newStatus;\n if (editStatus) {\n newStatus = await masto.v1.statuses\n .$select(editStatus.id)\n .update(params);\n saveStatus(newStatus, instance, {\n skipThreading: true,\n });\n } else {\n try {\n newStatus = await masto.v1.statuses.create(params, {\n requestInit: {\n headers: {\n 'Idempotency-Key': UID.current,\n },\n },\n });\n } catch (_) {\n // If idempotency key fails, try again without it\n newStatus = await masto.v1.statuses.create(params);\n }\n }\n states.composerState.minimized = false;\n states.composerState.publishing = false;\n setUIState('default');\n\n // Close\n onClose({\n // type: post, reply, edit\n type: editStatus ? 'edit' : replyToStatus ? 'reply' : 'post',\n newStatus,\n instance,\n scheduledAt,\n });\n } catch (e) {\n states.composerState.publishing = false;\n states.composerState.publishingError = true;\n console.error(e);\n alert(e?.reason || e);\n setUIState('error');\n }\n })();\n }}\n >\n <div>\n <div class={`compose-cw-container ${sensitive ? '' : 'collapsed'}`}>\n <input\n type=\"hidden\"\n name=\"sensitive\"\n value={sensitive ? 'on' : 'off'}\n />\n {/* mimic the old checkbox */}\n <TextExpander\n keys=\":\"\n class=\"spoiler-text-field-container\"\n onTrigger={(action) => {\n if (action?.name === 'custom-emojis') {\n setShowEmoji2Picker({\n targetElement: spoilerTextRef,\n defaultSearchTerm: action?.defaultSearchTerm || null,\n });\n }\n }}\n >\n <input\n ref={spoilerTextRef}\n type=\"text\"\n name=\"spoilerText\"\n placeholder={t`Content warning`}\n data-allow-custom-emoji=\"true\"\n disabled={uiState === 'loading'}\n class=\"spoiler-text-field\"\n lang={language}\n spellCheck=\"true\"\n autocomplete=\"off\"\n dir=\"auto\"\n onInput={() => {\n updateCharCount();\n }}\n />\n </TextExpander>\n <button\n type=\"button\"\n class=\"close-button plain4 small\"\n onClick={() => {\n setSensitive(false);\n textareaRef.current.focus();\n }}\n >\n <Icon icon=\"x\" alt={t`Cancel`} />\n </button>\n </div>\n <Textarea\n ref={textareaRef}\n data-allow-custom-emoji=\"true\"\n placeholder={\n replyToStatus\n ? t`Post your reply`\n : editStatus\n ? t`Edit your post`\n : !!poll\n ? t`Ask a question`\n : t`What are you doing?`\n }\n required={mediaAttachments?.length === 0}\n disabled={uiState === 'loading'}\n lang={language}\n onInput={() => {\n updateCharCount();\n }}\n maxCharacters={maxCharacters}\n onTrigger={(action) => {\n if (action?.name === 'custom-emojis') {\n setShowEmoji2Picker({\n targetElement: lastFocusedEmojiFieldRef,\n defaultSearchTerm: action?.defaultSearchTerm || null,\n });\n } else if (action?.name === 'mention') {\n setShowMentionPicker({\n defaultSearchTerm: action?.defaultSearchTerm || null,\n });\n } else if (\n action?.name === 'auto-detect-language' &&\n action?.languages\n ) {\n setAutoDetectedLanguages(action.languages);\n }\n }}\n />\n </div>\n {mediaAttachments?.length > 0 && (\n <div class=\"media-attachments\">\n {mediaAttachments.map((attachment, i) => {\n const { id, file } = attachment;\n const fileID = file?.size + file?.type + file?.name;\n return (\n <MediaAttachment\n key={id || fileID || i}\n attachment={attachment}\n disabled={uiState === 'loading'}\n lang={language}\n descriptionLimit={descriptionLimit}\n onDescriptionChange={(value) => {\n setMediaAttachments((attachments) => {\n const newAttachments = [...attachments];\n newAttachments[i] = {\n ...newAttachments[i],\n description: value,\n };\n return newAttachments;\n });\n }}\n onRemove={() => {\n setMediaAttachments((attachments) => {\n return attachments.filter((_, j) => j !== i);\n });\n }}\n />\n );\n })}\n <label class=\"media-sensitive\">\n <input\n name=\"sensitiveMedia\"\n type=\"checkbox\"\n checked={sensitiveMedia}\n disabled={uiState === 'loading'}\n onChange={(e) => {\n const sensitiveMedia = e.target.checked;\n setSensitiveMedia(sensitiveMedia);\n }}\n />{' '}\n <span>\n <Trans>Mark media as sensitive</Trans>\n </span>{' '}\n <Icon icon={`eye-${sensitiveMedia ? 'close' : 'open'}`} />\n </label>\n </div>\n )}\n {!!poll && (\n <ComposePoll\n lang={language}\n maxOptions={maxOptions}\n maxExpiration={maxExpiration}\n minExpiration={minExpiration}\n maxCharactersPerOption={maxCharactersPerOption}\n poll={poll}\n disabled={uiState === 'loading'}\n onInput={(poll) => {\n if (poll) {\n const newPoll = { ...poll };\n setPoll(newPoll);\n } else {\n setPoll(null);\n focusLastFocusedField();\n }\n }}\n />\n )}\n {scheduledAt && (\n <div class=\"toolbar scheduled-at\">\n <span>\n <label>\n <Trans>\n Posting on{' '}\n <ScheduledAtField\n scheduledAt={scheduledAt}\n setScheduledAt={setScheduledAt}\n />\n </Trans>\n </label>{' '}\n <small class=\"tag insignificant\">\n {getLocalTimezoneName()}\n </small>\n </span>\n <button\n type=\"button\"\n class=\"plain4 close-button small\"\n onClick={() => {\n setScheduledAt(null);\n focusLastFocusedField();\n }}\n >\n <Icon icon=\"x\" alt={t`Cancel`} />\n </button>\n </div>\n )}\n <div class=\"toolbar compose-footer\">\n <span class=\"add-toolbar-button-group spacer\">\n {showAddButton && (\n <Menu2\n portal={{\n target: document.body,\n }}\n containerProps={{\n style: {\n zIndex: 1001,\n },\n }}\n menuButton={({ open }) => (\n <button\n type=\"button\"\n class={`toolbar-button add-button ${\n open ? 'active' : ''\n }`}\n >\n <Icon icon=\"plus\" title={t`Add`} />\n </button>\n )}\n >\n {supportsCameraCapture && (\n <MenuItem\n disabled={mediaButtonDisabled}\n className=\"compose-menu-add-media\"\n >\n <label class=\"compose-menu-add-media-field\">\n <CameraCaptureInput\n hidden\n supportedMimeTypes={supportedImagesVideosTypes}\n disabled={mediaButtonDisabled}\n setMediaAttachments={setMediaAttachments}\n />\n </label>\n <Icon icon=\"camera\" /> <span>{_(ADD_LABELS.camera)}</span>\n </MenuItem>\n )}\n <MenuItem\n disabled={mediaButtonDisabled}\n className=\"compose-menu-add-media\"\n >\n <label class=\"compose-menu-add-media-field\">\n <FilePickerInput\n hidden\n supportedMimeTypes={supportedMimeTypes}\n maxMediaAttachments={maxMediaAttachments}\n mediaAttachments={mediaAttachments}\n disabled={mediaButtonDisabled}\n setMediaAttachments={setMediaAttachments}\n />\n </label>\n <Icon icon=\"media\" /> <span>{_(ADD_LABELS.media)}</span>\n </MenuItem>\n <MenuItem\n disabled={cwButtonDisabled}\n onClick={onCWButtonClick}\n >\n <Icon icon={`eye-${sensitive ? 'close' : 'open'}`} />{' '}\n <span>{_(ADD_LABELS.sensitive)}</span>\n </MenuItem>\n {showPollButton && (\n <MenuItem\n disabled={pollButtonDisabled}\n onClick={onPollButtonClick}\n >\n <Icon icon=\"poll\" /> <span>{_(ADD_LABELS.poll)}</span>\n </MenuItem>\n )}\n <MenuDivider />\n <MenuItem\n onClick={() => {\n setShowEmoji2Picker({\n targetElement: lastFocusedEmojiFieldRef,\n });\n }}\n >\n <Icon icon=\"emoji2\" />{' '}\n <span>{_(ADD_LABELS.customEmoji)}</span>\n </MenuItem>\n {!!states.settings.composerGIFPicker && (\n <MenuItem\n disabled={mediaButtonDisabled}\n onClick={() => {\n setShowGIFPicker(true);\n }}\n >\n <span class=\"icon icon-gif\" role=\"img\" />\n <span>{_(ADD_LABELS.gif)}</span>\n </MenuItem>\n )}\n {showScheduledAt && (\n <>\n <MenuDivider />\n <MenuItem\n disabled={scheduledAtButtonDisabled}\n onClick={onScheduledAtClick}\n >\n <Icon icon=\"schedule\" />{' '}\n <span>{_(ADD_LABELS.scheduledPost)}</span>\n </MenuItem>\n </>\n )}\n </Menu2>\n )}\n <span\n class=\"add-sub-toolbar-button-group\"\n ref={addSubToolbarRef}\n hidden\n >\n {supportsCameraCapture && (\n <label class=\"toolbar-button\">\n <CameraCaptureInput\n supportedMimeTypes={supportedImagesVideosTypes}\n mediaAttachments={mediaAttachments}\n disabled={mediaButtonDisabled}\n setMediaAttachments={setMediaAttachments}\n />\n <Icon icon=\"camera\" alt={_(ADD_LABELS.camera)} />\n </label>\n )}\n <label class=\"toolbar-button\">\n <FilePickerInput\n supportedMimeTypes={supportedMimeTypes}\n maxMediaAttachments={maxMediaAttachments}\n mediaAttachments={mediaAttachments}\n disabled={mediaButtonDisabled}\n setMediaAttachments={setMediaAttachments}\n />\n <Icon icon=\"media\" alt={_(ADD_LABELS.media)} />\n </label>\n <button\n type=\"button\"\n class=\"toolbar-button\"\n disabled={cwButtonDisabled}\n onClick={onCWButtonClick}\n >\n <Icon\n icon={`eye-${sensitive ? 'close' : 'open'}`}\n alt={_(ADD_LABELS.sensitive)}\n />\n </button>\n {showPollButton && (\n <button\n type=\"button\"\n class=\"toolbar-button\"\n disabled={pollButtonDisabled}\n onClick={onPollButtonClick}\n >\n <Icon icon=\"poll\" alt={_(ADD_LABELS.poll)} />\n </button>\n )}\n <div class=\"toolbar-divider\" />\n {/* <button\n type=\"button\"\n class=\"toolbar-button\"\n disabled={uiState === 'loading'}\n onClick={() => {\n setShowMentionPicker(true);\n }}\n >\n <Icon icon=\"at\" />\n </button> */}\n <button\n type=\"button\"\n class=\"toolbar-button\"\n disabled={uiState === 'loading'}\n onClick={() => {\n setShowEmoji2Picker({\n targetElement: lastFocusedEmojiFieldRef,\n });\n }}\n >\n <Icon icon=\"emoji2\" alt={_(ADD_LABELS.customEmoji)} />\n </button>\n {!!states.settings.composerGIFPicker && (\n <button\n type=\"button\"\n class=\"toolbar-button gif-picker-button\"\n disabled={mediaButtonDisabled}\n onClick={() => {\n setShowGIFPicker(true);\n }}\n >\n <span\n class=\"icon icon-gif\"\n aria-label={_(ADD_LABELS.gif)}\n />\n </button>\n )}\n {showScheduledAt && (\n <>\n <div class=\"toolbar-divider\" />\n <button\n type=\"button\"\n class={`toolbar-button ${scheduledAt ? 'highlight' : ''}`}\n disabled={scheduledAtButtonDisabled}\n onClick={onScheduledAtClick}\n >\n <Icon icon=\"schedule\" alt={_(ADD_LABELS.scheduledPost)} />\n </button>\n </>\n )}\n </span>\n </span>\n {uiState === 'loading' ? (\n <Loader abrupt />\n ) : (\n <CharCountMeter\n maxCharacters={maxCharacters}\n hidden={uiState === 'loading'}\n />\n )}\n <label\n class={`toolbar-button ${\n highlightLanguageField ? 'highlight' : ''\n }`}\n >\n <span class=\"icon-text\">\n {supportedLanguagesMap[language]?.native}\n </span>\n <select\n name=\"language\"\n value={language}\n onChange={(e) => {\n const { value } = e.target;\n setLanguage(value || DEFAULT_LANG);\n store.session.set('currentLanguage', value || DEFAULT_LANG);\n }}\n disabled={uiState === 'loading'}\n dir=\"auto\"\n >\n {topSupportedLanguages.map(([code, common, native]) => {\n const commonText = localeCode2Text({\n code,\n fallback: common,\n });\n const showCommon = commonText !== native;\n return (\n <option value={code} key={code}>\n {showCommon ? `${native} - ${commonText}` : commonText}\n </option>\n );\n })}\n <hr />\n {restSupportedLanguages.map(([code, common, native]) => {\n const commonText = localeCode2Text({\n code,\n fallback: common,\n });\n const showCommon = commonText !== native;\n return (\n <option value={code} key={code}>\n {showCommon ? `${native} - ${commonText}` : commonText}\n </option>\n );\n })}\n </select>\n </label>{' '}\n <label\n class={`toolbar-button ${highlightVisibilityField ? 'highlight' : ''}`}\n title={_(visibilityText[visibility])}\n >\n {visibility === 'public' || visibility === 'direct' ? (\n <Icon\n icon={visibilityIconsMap[visibility]}\n alt={_(visibilityText[visibility])}\n />\n ) : (\n <span class=\"icon-text\">{_(visibilityText[visibility])}</span>\n )}\n <select\n name=\"visibility\"\n value={visibility}\n onChange={(e) => {\n setVisibility(e.target.value);\n }}\n disabled={uiState === 'loading' || !!editStatus}\n dir=\"auto\"\n >\n <option value=\"public\">\n <Trans>Public</Trans>\n </option>\n {(supports('@pleroma/local-visibility-post') ||\n supports('@akkoma/local-visibility-post')) && (\n <option value=\"local\">\n <Trans>Local</Trans>\n </option>\n )}\n <option value=\"unlisted\">\n <Trans>Quiet public</Trans>\n </option>\n <option value=\"private\">\n <Trans>Followers</Trans>\n </option>\n <option value=\"direct\">\n <Trans>Private mention</Trans>\n </option>\n </select>\n </label>{' '}\n <button type=\"submit\" disabled={uiState === 'loading'}>\n {scheduledAt\n ? t`Schedule`\n : replyToStatus\n ? t`Reply`\n : editStatus\n ? t`Update`\n : t({\n message: 'Post',\n context: 'Submit button in composer',\n })}\n </button>\n </div>\n </form>\n </div>\n {showMentionPicker && (\n <Modal\n onClose={() => {\n setShowMentionPicker(false);\n focusLastFocusedField();\n }}\n >\n <MentionModal\n masto={masto}\n instance={instance}\n onClose={() => {\n setShowMentionPicker(false);\n }}\n defaultSearchTerm={showMentionPicker?.defaultSearchTerm}\n onSelect={(socialAddress) => {\n const textarea = textareaRef.current;\n if (textarea) {\n insertTextAtCursor({\n targetElement: textarea,\n text: '@' + socialAddress,\n });\n }\n }}\n />\n </Modal>\n )}\n {showEmoji2Picker && (\n <Modal\n onClose={() => {\n setShowEmoji2Picker(false);\n focusLastFocusedField();\n }}\n >\n <CustomEmojisModal\n masto={masto}\n instance={instance}\n onClose={() => {\n setShowEmoji2Picker(false);\n }}\n defaultSearchTerm={showEmoji2Picker?.defaultSearchTerm}\n onSelect={(emojiShortcode) => {\n const targetElement =\n showEmoji2Picker?.targetElement?.current || textareaRef.current;\n if (targetElement) {\n insertTextAtCursor({ targetElement, text: emojiShortcode });\n }\n }}\n />\n </Modal>\n )}\n {showGIFPicker && (\n <Modal\n onClose={() => {\n setShowGIFPicker(false);\n focusLastFocusedField();\n }}\n >\n <GIFPickerModal\n onClose={() => setShowGIFPicker(false)}\n onSelect={({ url, type, alt_text }) => {\n console.log('GIF URL', url);\n if (mediaAttachments.length >= maxMediaAttachments) {\n alert(\n plural(maxMediaAttachments, {\n one: 'You can only attach up to 1 file.',\n other: 'You can only attach up to # files.',\n }),\n );\n return;\n }\n // Download the GIF and insert it as media attachment\n (async () => {\n let theToast;\n try {\n theToast = showToast({\n text: t`Downloading GIF…`,\n duration: -1,\n });\n const blob = await fetch(url, {\n referrerPolicy: 'no-referrer',\n }).then((res) => res.blob());\n const file = new File(\n [blob],\n type === 'video/mp4' ? 'video.mp4' : 'image.gif',\n {\n type,\n },\n );\n const newMediaAttachments = [\n ...mediaAttachments,\n {\n file,\n type,\n size: file.size,\n id: null,\n description: alt_text || '',\n },\n ];\n setMediaAttachments(newMediaAttachments);\n theToast?.hideToast?.();\n } catch (err) {\n console.error(err);\n theToast?.hideToast?.();\n showToast(t`Failed to download GIF`);\n }\n })();\n }}\n />\n </Modal>\n )}\n </div>\n );\n}\n\nfunction removeNullUndefined(obj) {\n for (let key in obj) {\n if (obj[key] === null || obj[key] === undefined) {\n delete obj[key];\n }\n }\n return obj;\n}\n\nexport default Compose;\n"],"file":"assets/compose-BAFFp8_Z.js"} |