diff --git a/.eslintrc.js b/.eslintrc.js index b9508e9a..a489703f 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -31,6 +31,7 @@ module.exports = { root: true, rules: { 'rulesdir/prefer-underscore-method': 'off', + 'rulesdir/prefer-import-module-contents': 'off', 'react/jsx-props-no-spreading': 'off', 'react/require-default-props': 'off', 'react/jsx-filename-extension': ['error', { extensions: ['.tsx', '.jsx'] }], diff --git a/.gitignore b/.gitignore index 6c4e8770..d6cadc5d 100644 --- a/.gitignore +++ b/.gitignore @@ -76,3 +76,6 @@ android/keystores/debug.keystore # generated by bob lib/ + +# react-native-live-markdown +.build_complete diff --git a/README.md b/README.md index 61e01a1d..04157863 100644 --- a/README.md +++ b/README.md @@ -124,10 +124,10 @@ The style object can be passed to multiple `MarkdownTextInput` components using ## Parsing logic -`MarkdownTextInput` behavior can be customized via `parser` property. Parser is a function that accepts a plaintext string and returns an array of `Range` objects: +`MarkdownTextInput` behavior can be customized via `parser` property. Parser is a function that accepts a plaintext string and returns an array of `MarkdownRange` objects: ```ts -interface Range { +interface MarkdownRange { type: MarkdownType; start: number; length: number; @@ -172,10 +172,10 @@ Currently, `react-native-live-markdown` supports only [ExpensiMark](https://gith `MarkdownTextInput` inherits all props of React Native's `TextInput` component as well as introduces the following properties: -| Prop | Type | Default | Note | -| --------------- | ---------------------------- | ----------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -| `parser` | `(value: string) => Range[]` | `undefined` | A function that parses the current value and returns an array of ranges. | -| `markdownStyle` | `MarkdownStyle` | `undefined` | Adds custom styling to Markdown text. The provided value is merged with default style object. See [Styling](https://github.com/expensify/react-native-live-markdown/blob/main/README.md#styling) for more information. | +| Prop | Type | Default | Note | +| --------------- | ------------------------------------ | ----------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| `parser` | `(value: string) => MarkdownRange[]` | `undefined` | A function that parses the current value and returns an array of ranges. | +| `markdownStyle` | `MarkdownStyle` | `undefined` | Adds custom styling to Markdown text. The provided value is merged with default style object. See [Styling](https://github.com/expensify/react-native-live-markdown/blob/main/README.md#styling) for more information. | ## Compatibility diff --git a/WebExample/__tests__/input.spec.ts b/WebExample/__tests__/input.spec.ts index f3accc26..3ec26376 100644 --- a/WebExample/__tests__/input.spec.ts +++ b/WebExample/__tests__/input.spec.ts @@ -1,6 +1,6 @@ import {test, expect} from '@playwright/test'; import * as TEST_CONST from '../../example/src/testConstants'; -import {checkCursorPosition, setupInput} from './utils'; +import {getCursorPosition, getElementValue, setupInput} from './utils'; test.beforeEach(async ({page}) => { await page.goto(TEST_CONST.LOCAL_URL, {waitUntil: 'load'}); @@ -12,8 +12,8 @@ test.describe('typing', () => { await inputLocator.focus(); await inputLocator.pressSequentially(TEST_CONST.EXAMPLE_CONTENT); - const value = await inputLocator.innerText(); - expect(value).toEqual(TEST_CONST.EXAMPLE_CONTENT); + + expect(await getElementValue(inputLocator)).toEqual(TEST_CONST.EXAMPLE_CONTENT); }); test('fast type cursor position', async ({page}) => { @@ -23,10 +23,10 @@ test.describe('typing', () => { await inputLocator.pressSequentially(EXAMPLE_LONG_CONTENT); - expect(await inputLocator.innerText()).toBe(EXAMPLE_LONG_CONTENT); + expect(await getElementValue(inputLocator)).toBe(EXAMPLE_LONG_CONTENT); - const cursorPosition = await page.evaluate(checkCursorPosition); + const cursorPosition = await getCursorPosition(inputLocator); - expect(cursorPosition).toBe(EXAMPLE_LONG_CONTENT.length); + expect(cursorPosition.end).toBe(EXAMPLE_LONG_CONTENT.length); }); }); diff --git a/WebExample/__tests__/textManipulation.spec.ts b/WebExample/__tests__/textManipulation.spec.ts index 5c83e3b4..b789bd47 100644 --- a/WebExample/__tests__/textManipulation.spec.ts +++ b/WebExample/__tests__/textManipulation.spec.ts @@ -1,7 +1,7 @@ import {test, expect} from '@playwright/test'; import type {Locator, Page} from '@playwright/test'; import * as TEST_CONST from '../../example/src/testConstants'; -import {checkCursorPosition, setupInput, getElementStyle, pressCmd} from './utils'; +import {getCursorPosition, setupInput, getElementStyle, pressCmd, getElementValue} from './utils'; const pasteContent = async ({text, page, inputLocator}: {text: string; page: Page; inputLocator: Locator}) => { await page.evaluate(async (pasteText) => navigator.clipboard.writeText(pasteText), text); @@ -43,7 +43,7 @@ test.describe('paste content', () => { const newText = '*bold*'; await pasteContent({text: newText, page, inputLocator}); - expect(await inputLocator.innerText()).toBe(newText); + expect(await getElementValue(inputLocator)).toBe(newText); }); test('paste undo', async ({page, browserName}) => { @@ -61,10 +61,9 @@ test.describe('paste content', () => { await page.evaluate(async (pasteText) => navigator.clipboard.writeText(pasteText), PASTE_TEXT_SECOND); await pressCmd({inputLocator, command: 'v'}); await page.waitForTimeout(TEST_CONST.INPUT_HISTORY_DEBOUNCE_TIME_MS); - await pressCmd({inputLocator, command: 'z'}); - - expect(await inputLocator.innerText()).toBe(PASTE_TEXT_FIRST); + await page.waitForTimeout(TEST_CONST.INPUT_HISTORY_DEBOUNCE_TIME_MS); + expect(await getElementValue(inputLocator)).toBe(PASTE_TEXT_FIRST); }); test('paste redo', async ({page}) => { @@ -84,7 +83,7 @@ test.describe('paste content', () => { await pressCmd({inputLocator, command: 'z'}); await pressCmd({inputLocator, command: 'Shift+z'}); - expect(await inputLocator.innerText()).toBe(`${PASTE_TEXT_FIRST}${PASTE_TEXT_SECOND}`); + expect(await getElementValue(inputLocator)).toBe(`${PASTE_TEXT_FIRST}${PASTE_TEXT_SECOND}`); }); }); @@ -93,9 +92,9 @@ test('select all', async ({page}) => { await inputLocator.focus(); await pressCmd({inputLocator, command: 'a'}); - const cursorPosition = await page.evaluate(checkCursorPosition); + const cursorPosition = await getCursorPosition(inputLocator); - expect(cursorPosition).toBe(TEST_CONST.EXAMPLE_CONTENT.length); + expect(cursorPosition.end).toBe(TEST_CONST.EXAMPLE_CONTENT.length); }); test('cut content changes', async ({page, browserName}) => { @@ -107,15 +106,12 @@ test('cut content changes', async ({page, browserName}) => { const inputLocator = await setupInput(page, 'clear'); await pasteContent({text: WRAPPED_CONTENT, page, inputLocator}); - const rootHandle = await inputLocator.locator('span.root').first(); - await page.evaluate(async (initialContent) => { - const filteredNode = Array.from(document.querySelectorAll('div[contenteditable="true"] > span.root span')).find((node) => { - return node.textContent?.includes(initialContent) && node.nextElementSibling && node.nextElementSibling.textContent?.includes('*'); - }); + await page.evaluate(async () => { + const filteredNode = Array.from(document.querySelectorAll('span[data-type="text"]')); - const startNode = filteredNode; - const endNode = filteredNode?.nextElementSibling; + const startNode = filteredNode[1]; + const endNode = filteredNode[2]; if (startNode?.firstChild && endNode?.lastChild) { const range = new Range(); @@ -126,10 +122,16 @@ test('cut content changes', async ({page, browserName}) => { selection?.removeAllRanges(); selection?.addRange(range); } - }, INITIAL_CONTENT); + + return filteredNode; + }); await inputLocator.focus(); await pressCmd({inputLocator, command: 'x'}); - expect(await rootHandle.innerHTML()).toBe(EXPECTED_CONTENT); + expect(await getElementValue(inputLocator)).toBe(EXPECTED_CONTENT); + + // Ckeck if there is no markdown elements after the cut operation + const spans = await inputLocator.locator('span[data-type="text"]'); + expect(await spans.count()).toBe(1); }); diff --git a/WebExample/__tests__/utils.ts b/WebExample/__tests__/utils.ts index 8b588348..dfc97bd8 100644 --- a/WebExample/__tests__/utils.ts +++ b/WebExample/__tests__/utils.ts @@ -10,16 +10,16 @@ const setupInput = async (page: Page, action?: 'clear' | 'reset') => { return inputLocator; }; -const checkCursorPosition = () => { - const editableDiv = document.querySelector('div[contenteditable="true"]') as HTMLElement; - const range = window.getSelection()?.getRangeAt(0); - if (!range || !editableDiv) { - return null; - } - const preCaretRange = range.cloneRange(); - preCaretRange.selectNodeContents(editableDiv); - preCaretRange.setEnd(range.endContainer, range.endOffset); - return preCaretRange.toString().length; +const getCursorPosition = async (elementHandle: Locator) => { + const inputSelectionHandle = await elementHandle.evaluateHandle( + ( + div: HTMLInputElement & { + selection: {start: number; end: number}; + }, + ) => div.selection, + ); + const selection = await inputSelectionHandle.jsonValue(); + return selection; }; const setCursorPosition = ({startNode, endNode}: {startNode?: Element; endNode?: Element | null}) => { @@ -43,8 +43,11 @@ const getElementStyle = async (elementHandle: Locator) => { if (elementHandle) { await elementHandle.waitFor({state: 'attached'}); - - elementStyle = await elementHandle.getAttribute('style'); + // We need to get styles from the parent element because every text node is wrapped additionally with a span element + const parentElementHandle = await elementHandle.evaluateHandle((element) => { + return element.parentElement; + }); + elementStyle = await parentElementHandle.asElement()?.getAttribute('style'); } return elementStyle; }; @@ -55,4 +58,10 @@ const pressCmd = async ({inputLocator, command}: {inputLocator: Locator; command await inputLocator.press(`${OPERATION_MODIFIER}+${command}`); }; -export {setupInput, checkCursorPosition, setCursorPosition, getElementStyle, pressCmd}; +const getElementValue = async (elementHandle: Locator) => { + const inputValueHandle = await elementHandle.evaluateHandle((div: HTMLInputElement) => div.value); + const value = await inputValueHandle.jsonValue(); + return value; +}; + +export {setupInput, getCursorPosition, setCursorPosition, getElementStyle, pressCmd, getElementValue}; diff --git a/example/ios/Podfile.lock b/example/ios/Podfile.lock index 991ac005..7e8a69f6 100644 --- a/example/ios/Podfile.lock +++ b/example/ios/Podfile.lock @@ -1053,13 +1053,13 @@ PODS: - React-jsi (= 0.73.4) - React-logger (= 0.73.4) - React-perflogger (= 0.73.4) - - RNLiveMarkdown (0.1.111): + - RNLiveMarkdown (0.1.120): - glog - RCT-Folly (= 2022.05.16.00) - React-Core - - RNLiveMarkdown/common (= 0.1.111) + - RNLiveMarkdown/common (= 0.1.120) - RNReanimated/worklets - - RNLiveMarkdown/common (0.1.111): + - RNLiveMarkdown/common (0.1.120): - glog - RCT-Folly (= 2022.05.16.00) - React-Core @@ -1298,7 +1298,7 @@ SPEC CHECKSUMS: React-runtimescheduler: ed48e5faac6751e66ee1261c4bd01643b436f112 React-utils: 6e5ad394416482ae21831050928ae27348f83487 ReactCommon: 840a955d37b7f3358554d819446bffcf624b2522 - RNLiveMarkdown: 0ab41fe829862ca9e9ef3b6604201d1de9c9e416 + RNLiveMarkdown: c69f957652deeda303840cd1bbe612f203a20a9e RNReanimated: 1353558d1fba65da0e0d0dde4146d5690f78276f SocketRocket: f32cd54efbe0f095c4d7594881e52619cfe80b17 Yoga: 64cd2a583ead952b0315d5135bf39e053ae9be70 diff --git a/example/package.json b/example/package.json index 58414ccc..d5f6127f 100644 --- a/example/package.json +++ b/example/package.json @@ -8,10 +8,10 @@ "start": "react-native start" }, "dependencies": { - "expensify-common": "2.0.72", + "expensify-common": "2.0.76", "react": "18.2.0", "react-native": "0.73.4", - "react-native-reanimated": "3.15.0-nightly-20240804-7df5fd57d" + "react-native-reanimated": "3.15.0" }, "devDependencies": { "@babel/core": "^7.20.0", diff --git a/example/src/App.tsx b/example/src/App.tsx index 2aee3426..85dbe092 100644 --- a/example/src/App.tsx +++ b/example/src/App.tsx @@ -1,7 +1,5 @@ import * as React from 'react'; - import {Button, Platform, StyleSheet, Text, View} from 'react-native'; - import {MarkdownTextInput, parseExpensiMark} from '@expensify/react-native-live-markdown'; import type {TextInput} from 'react-native'; import * as TEST_CONST from './testConstants'; diff --git a/package.json b/package.json index 1a32430b..afe2133f 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@expensify/react-native-live-markdown", - "version": "0.1.112", + "version": "0.1.120", "description": "Drop-in replacement for React Native's TextInput component with Markdown formatting.", "main": "lib/commonjs/index", "module": "lib/module/index", @@ -36,6 +36,7 @@ "lint:WebExample": "eslint WebExample --ext .js,.ts,.tsx", "clean": "del-cli android/build example/android/build example/android/app/build example/ios/build lib", "prepare": "bob build", + "build:watch": "nodemon --watch src --ext .ts,.tsx,.css --exec \"rm -f .build_complete && yarn prepare && yarn pack && touch .build_complete\"", "release": "release-it" }, "keywords": [ @@ -83,14 +84,15 @@ "eslint-plugin-prettier": "^4.0.0", "eslint-plugin-promise": "^6.1.1", "eslint-plugin-tsdoc": "^0.2.17", - "expensify-common": "2.0.72", + "expensify-common": "2.0.76", "jest": "^29.6.3", "jest-environment-jsdom": "^29.7.0", + "nodemon": "^3.1.3", "prettier": "2.8.8", "react": "18.2.0", "react-native": "0.73.4", "react-native-builder-bob": "^0.20.0", - "react-native-reanimated": "3.15.0-nightly-20240804-7df5fd57d", + "react-native-reanimated": "3.15.0", "react-native-web": "^0.19.10", "release-it": "^15.0.0", "turbo": "^1.10.7", @@ -99,10 +101,11 @@ "resolutions": { "@types/react": "17.0.21", "react-native@0.73.4": "patch:react-native@npm%3A0.73.4#./.yarn/patches/react-native-npm-0.73.4-297ae78b8f.patch", - "@react-native/gradle-plugin@0.73.4": "patch:@react-native/gradle-plugin@npm%3A0.73.4#./.yarn/patches/@react-native-gradle-plugin-npm-0.73.4-6535082666.patch" + "@react-native/gradle-plugin@0.73.4": "patch:@react-native/gradle-plugin@npm%3A0.73.4#./.yarn/patches/@react-native-gradle-plugin-npm-0.73.4-6535082666.patch", + "link@^2.1.1": "patch:link@npm%3A2.1.1#./.yarn/patches/link-npm-2.1.1-1c9fea076e.patch" }, "peerDependencies": { - "expensify-common": ">=2.0.72", + "expensify-common": ">=2.0.76", "react": "*", "react-native": "*", "react-native-reanimated": ">=3.11.0" diff --git a/src/MarkdownTextInput.tsx b/src/MarkdownTextInput.tsx index 61619727..83ae40db 100644 --- a/src/MarkdownTextInput.tsx +++ b/src/MarkdownTextInput.tsx @@ -6,10 +6,11 @@ import type {WorkletRuntime} from 'react-native-reanimated'; import type {ShareableRef, WorkletFunction} from 'react-native-reanimated/lib/typescript/commonTypes'; import MarkdownTextInputDecoratorViewNativeComponent from './MarkdownTextInputDecoratorViewNativeComponent'; +import type {MarkdownStyle} from './MarkdownTextInputDecoratorViewNativeComponent'; import NativeLiveMarkdownModule from './NativeLiveMarkdownModule'; -import type * as MarkdownTextInputDecoratorViewNativeComponentTypes from './MarkdownTextInputDecoratorViewNativeComponent'; -import * as StyleUtils from './styleUtils'; -import type * as StyleUtilsTypes from './styleUtils'; +import {mergeMarkdownStyleWithDefault} from './styleUtils'; +import type {PartialMarkdownStyle} from './styleUtils'; +import type {MarkdownRange} from './commonTypes'; declare global { // eslint-disable-next-line no-var @@ -39,7 +40,7 @@ function initializeLiveMarkdownIfNeeded() { initialized = true; } -function registerParser(parser: (input: string) => Range[]): number { +function registerParser(parser: (input: string) => MarkdownRange[]): number { initializeLiveMarkdownIfNeeded(); const shareableWorklet = makeShareableCloneRecursive(parser) as ShareableRef>; const parserId = global.jsi_registerMarkdownWorklet(shareableWorklet); @@ -50,21 +51,9 @@ function unregisterParser(parserId: number) { global.jsi_unregisterMarkdownWorklet(parserId); } -type PartialMarkdownStyle = StyleUtilsTypes.PartialMarkdownStyle; -type MarkdownStyle = MarkdownTextInputDecoratorViewNativeComponentTypes.MarkdownStyle; - -type MarkdownType = 'bold' | 'italic' | 'strikethrough' | 'emoji' | 'mention-here' | 'mention-user' | 'mention-report' | 'link' | 'code' | 'pre' | 'blockquote' | 'h1' | 'syntax'; - -interface Range { - type: MarkdownType; - start: number; - length: number; - depth?: number; -} - interface MarkdownTextInputProps extends TextInputProps { markdownStyle?: PartialMarkdownStyle; - parser: (value: string) => Range[]; + parser: (value: string) => MarkdownRange[]; } function processColorsInMarkdownStyle(input: MarkdownStyle): MarkdownStyle { @@ -85,7 +74,7 @@ function processColorsInMarkdownStyle(input: MarkdownStyle): MarkdownStyle { } function processMarkdownStyle(input: PartialMarkdownStyle | undefined): MarkdownStyle { - return processColorsInMarkdownStyle(StyleUtils.mergeMarkdownStyleWithDefault(input)); + return processColorsInMarkdownStyle(mergeMarkdownStyleWithDefault(input)); } const MarkdownTextInput = React.forwardRef((props, ref) => { @@ -138,6 +127,6 @@ const styles = StyleSheet.create({ }, }); -export type {PartialMarkdownStyle as MarkdownStyle, MarkdownTextInputProps, Range, MarkdownType}; +export type {PartialMarkdownStyle as MarkdownStyle, MarkdownTextInputProps}; export default MarkdownTextInput; diff --git a/src/MarkdownTextInput.web.tsx b/src/MarkdownTextInput.web.tsx index 2961e0c4..fb66c7dd 100644 --- a/src/MarkdownTextInput.web.tsx +++ b/src/MarkdownTextInput.web.tsx @@ -11,51 +11,23 @@ import type { TextInputContentSizeChangeEventData, } from 'react-native'; import React, {useEffect, useRef, useCallback, useMemo, useLayoutEffect} from 'react'; -import type {CSSProperties, MutableRefObject, ReactEventHandler, FocusEventHandler, MouseEvent, KeyboardEvent, SyntheticEvent} from 'react'; +import type {CSSProperties, MutableRefObject, ReactEventHandler, FocusEventHandler, MouseEvent, KeyboardEvent, SyntheticEvent, ClipboardEventHandler} from 'react'; import {StyleSheet} from 'react-native'; -import * as ParseUtils from './web/parserUtils'; -import * as CursorUtils from './web/cursorUtils'; -import * as StyleUtils from './styleUtils'; -import * as BrowserUtils from './web/browserUtils'; -import type * as MarkdownTextInputDecoratorViewNativeComponent from './MarkdownTextInputDecoratorViewNativeComponent'; -import './web/MarkdownTextInput.css'; +import {updateInputStructure} from './web/utils/parserUtils'; import InputHistory from './web/InputHistory'; -import type * as MarkdownTextInputRange from './MarkdownTextInput'; +import type {MarkdownRange} from './commonTypes'; +import type {TreeNode} from './web/utils/treeUtils'; +import {getCurrentCursorPosition, removeSelection, setCursorPosition} from './web/utils/cursorUtils'; +import './web/MarkdownTextInput.css'; +import type {MarkdownStyle} from './MarkdownTextInputDecoratorViewNativeComponent'; +import {getElementHeight, getPlaceholderValue, isEventComposing, normalizeValue, parseInnerHTMLToText} from './web/utils/inputUtils'; +import {parseToReactDOMStyle, processMarkdownStyle} from './web/utils/webStyleUtils'; const useClientEffect = typeof window === 'undefined' ? useEffect : useLayoutEffect; -let createReactDOMStyle: (style: any) => any; -try { - createReactDOMStyle = - // eslint-disable-next-line @typescript-eslint/no-var-requires - require('react-native-web/dist/exports/StyleSheet/compiler/createReactDOMStyle').default; -} catch (e) { - throw new Error('[react-native-live-markdown] Function `createReactDOMStyle` from react-native-web not found. Please make sure that you are using React Native Web 0.18 or newer.'); -} - -let preprocessStyle: (style: any) => any; -try { - preprocessStyle = - // eslint-disable-next-line @typescript-eslint/no-var-requires - require('react-native-web/dist/exports/StyleSheet/preprocess').default; -} catch (e) { - throw new Error('[react-native-live-markdown] Function `preprocessStyle` from react-native-web not found.'); -} - -let dangerousStyleValue: (name: string, value: any, isCustomProperty: boolean) => any; -try { - dangerousStyleValue = - // eslint-disable-next-line @typescript-eslint/no-var-requires - require('react-native-web/dist/modules/setValueForStyles/dangerousStyleValue').default; -} catch (e) { - throw new Error('[react-native-live-markdown] Function `dangerousStyleValue` from react-native-web not found.'); -} - -type MarkdownStyle = MarkdownTextInputDecoratorViewNativeComponent.MarkdownStyle; - interface MarkdownTextInputProps extends TextInputProps { markdownStyle?: MarkdownStyle; - parser: (text: string) => MarkdownTextInputRange.Range[]; + parser: (text: string) => MarkdownRange[]; onClick?: (e: MouseEvent) => void; dir?: string; disabled?: boolean; @@ -82,62 +54,15 @@ type ParseTextResult = { let focusTimeout: NodeJS.Timeout | null = null; -// Removes one '\n' from the end of the string that were added by contentEditable div -function normalizeValue(value: string) { - return value.replace(/\n$/, ''); -} -// Adds one '\n' at the end of the string if it's missing -function denormalizeValue(value: string) { - return value.endsWith('\n') ? `${value}\n` : value; -} - -// If an Input Method Editor is processing key input, the 'keyCode' is 229. -// https://www.w3.org/TR/uievents/#determine-keydown-keyup-keyCode -function isEventComposing(nativeEvent: globalThis.KeyboardEvent) { - return nativeEvent.isComposing || nativeEvent.keyCode === 229; -} - -const ZERO_WIDTH_SPACE = '\u200B'; - -function getPlaceholderValue(placeholder: string | undefined) { - if (!placeholder) { - return ZERO_WIDTH_SPACE; - } - return placeholder.length ? placeholder : ZERO_WIDTH_SPACE; -} - -function processUnitsInMarkdownStyle(input: MarkdownStyle): MarkdownStyle { - const output = JSON.parse(JSON.stringify(input)); - - Object.keys(output).forEach((key) => { - const obj = output[key]; - Object.keys(obj).forEach((prop) => { - obj[prop] = dangerousStyleValue(prop, obj[prop], false); - }); - }); - - return output as MarkdownStyle; -} +type MarkdownTextInputElement = HTMLDivElement & + HTMLInputElement & { + tree: TreeNode; + selection: Selection; + }; -function processMarkdownStyle(input: MarkdownStyle | undefined): MarkdownStyle { - return processUnitsInMarkdownStyle(StyleUtils.mergeMarkdownStyleWithDefault(input)); -} - -function getElementHeight(node: HTMLDivElement, styles: CSSProperties, numberOfLines: number | undefined) { - if (numberOfLines) { - const tempElement = document.createElement('div'); - tempElement.setAttribute('contenteditable', 'true'); - Object.assign(tempElement.style, styles); - tempElement.innerText = Array(numberOfLines).fill('A').join('\n'); - if (node.parentElement) { - node.parentElement.appendChild(tempElement); - const height = tempElement.clientHeight; - node.parentElement.removeChild(tempElement); - return height ? `${height}px` : 'auto'; - } - } - return styles.height ? `${styles.height}px` : 'auto'; -} +type HTMLMarkdownElement = HTMLElement & { + value: string; +}; const MarkdownTextInput = React.forwardRef( ( @@ -173,6 +98,7 @@ const MarkdownTextInput = React.forwardRef( autoFocus = false, onContentSizeChange, id, + inputMode, }, ref, ) => { @@ -184,13 +110,13 @@ const MarkdownTextInput = React.forwardRef( } const compositionRef = useRef(false); - const pasteRef = useRef(false); - const divRef = useRef(null); + const divRef = useRef(null); const currentlyFocusedField = useRef(null); const contentSelection = useRef(null); const className = `react-native-live-markdown-input-${multiline ? 'multiline' : 'singleline'}`; const history = useRef(); - const dimensions = React.useRef(null); + const dimensions = useRef(null); + const pasteContent = useRef(null); if (!history.current) { history.current = new InputHistory(100, 150, value || ''); @@ -203,7 +129,7 @@ const MarkdownTextInput = React.forwardRef( const setEventProps = useCallback((e: NativeSyntheticEvent) => { if (divRef.current) { - const text = normalizeValue(divRef.current.innerText || ''); + const text = divRef.current.value; if (e.target) { // TODO: change the logic here so every event have value property (e.target as unknown as HTMLInputElement).value = text; @@ -217,20 +143,26 @@ const MarkdownTextInput = React.forwardRef( const parseText = useCallback( ( - parserFunction: (input: string) => MarkdownTextInputRange.Range[], - target: HTMLDivElement, + parserFunction: (input: string) => MarkdownRange[], + target: MarkdownTextInputElement, text: string | null, customMarkdownStyles: MarkdownStyle, cursorPosition: number | null = null, shouldAddToHistory = true, + shouldForceDOMUpdate = false, ): ParseTextResult => { + if (!divRef.current) { + return {text: text || '', cursorPosition: null}; + } + if (text === null) { - return {text: target.innerText, cursorPosition: null}; + return {text: divRef.current.value, cursorPosition: null}; } - const parsedText = ParseUtils.parseText(parserFunction, target, text, cursorPosition, customMarkdownStyles, !multiline); + const parsedText = updateInputStructure(parserFunction, target, text, cursorPosition, customMarkdownStyles, !multiline, shouldForceDOMUpdate); + divRef.current.value = parsedText.text; + if (history.current && shouldAddToHistory) { - // We need to normalize the value before saving it to the history to prevent situations when additional new lines break the cursor position calculation logic - history.current.throttledAdd(normalizeValue(parsedText.text), parsedText.cursorPosition); + history.current.throttledAdd(parsedText.text, parsedText.cursorPosition); } return parsedText; @@ -241,7 +173,7 @@ const MarkdownTextInput = React.forwardRef( const processedMarkdownStyle = useMemo(() => { const newMarkdownStyle = processMarkdownStyle(markdownStyle); if (divRef.current) { - parseText(parser, divRef.current, divRef.current.innerText, newMarkdownStyle, null, false); + parseText(parser, divRef.current, divRef.current.value, newMarkdownStyle, null, false); } return newMarkdownStyle; }, [parser, markdownStyle, parseText]); @@ -254,13 +186,13 @@ const MarkdownTextInput = React.forwardRef( caretColor: (flattenedStyle as TextStyle).color || 'black', }, disabled && styles.disabledInputStyles, - createReactDOMStyle(preprocessStyle(flattenedStyle)), + parseToReactDOMStyle(flattenedStyle), ]) as CSSProperties, [flattenedStyle, disabled], ); const undo = useCallback( - (target: HTMLDivElement): ParseTextResult => { + (target: MarkdownTextInputElement): ParseTextResult => { if (!history.current) { return { text: '', @@ -268,14 +200,14 @@ const MarkdownTextInput = React.forwardRef( }; } const item = history.current.undo(); - const undoValue = item ? denormalizeValue(item.text) : null; + const undoValue = item ? item.text : null; return parseText(parser, target, undoValue, processedMarkdownStyle, item ? item.cursorPosition : null, false); }, [parser, parseText, processedMarkdownStyle], ); const redo = useCallback( - (target: HTMLDivElement): ParseTextResult => { + (target: MarkdownTextInputElement): ParseTextResult => { if (!history.current) { return { text: '', @@ -283,20 +215,12 @@ const MarkdownTextInput = React.forwardRef( }; } const item = history.current.redo(); - const redoValue = item ? denormalizeValue(item.text) : null; + const redoValue = item ? item.text : null; return parseText(parser, target, redoValue, processedMarkdownStyle, item ? item.cursorPosition : null, false); }, [parser, parseText, processedMarkdownStyle], ); - // We have to process value property since contentEditable div adds one additional '\n' at the end of the text if we are entering new line - const processedValue = useMemo(() => { - if (value && value[value.length - 1] === '\n') { - return `${value}\n`; - } - return value; - }, [value]); - // Placeholder text color logic const updateTextColor = useCallback( (node: HTMLDivElement, text: string) => { @@ -319,26 +243,25 @@ const MarkdownTextInput = React.forwardRef( ); const updateRefSelectionVariables = useCallback((newSelection: Selection) => { + if (!divRef.current) { + return; + } const {start, end} = newSelection; - const markdownHTMLInput = divRef.current as HTMLInputElement; - markdownHTMLInput.selectionStart = start; - markdownHTMLInput.selectionEnd = end; + divRef.current.selection = {start, end}; }, []); const updateSelection = useCallback( - (e: SyntheticEvent | null = null, predefinedSelection: Selection | null = null) => { + (e: SyntheticEvent, predefinedSelection: Selection | null = null) => { if (!divRef.current) { return; } - const newSelection = predefinedSelection || CursorUtils.getCurrentCursorPosition(divRef.current); + const newSelection = predefinedSelection || getCurrentCursorPosition(divRef.current); if (newSelection && (!contentSelection.current || contentSelection.current.start !== newSelection.start || contentSelection.current.end !== newSelection.end)) { updateRefSelectionVariables(newSelection); contentSelection.current = newSelection; - if (e) { - handleSelectionChange(e); - } + handleSelectionChange(e); } }, [handleSelectionChange, updateRefSelectionVariables], @@ -364,21 +287,31 @@ const MarkdownTextInput = React.forwardRef( const handleOnChangeText = useCallback( (e: SyntheticEvent) => { - if (!divRef.current || !(e.target instanceof HTMLElement)) { + if (!divRef.current || !(e.target instanceof HTMLElement) || !contentSelection.current) { return; } + const nativeEvent = e.nativeEvent as MarkdownNativeEvent; + const inputType = nativeEvent.inputType; + + updateTextColor(divRef.current, e.target.textContent ?? ''); + const previousText = divRef.current.value; + const parsedText = normalizeValue(inputType === 'pasteText' ? pasteContent.current || '' : parseInnerHTMLToText(e.target as MarkdownTextInputElement)); + + if (pasteContent.current) { + pasteContent.current = null; + } + const prevSelection = contentSelection.current ?? {start: 0, end: 0}; - const prevTextLength = CursorUtils.getPrevTextLength() ?? 0; - const changedText = e.target.innerText; - if (compositionRef.current && !BrowserUtils.isMobile) { - updateTextColor(divRef.current, changedText); + const newCursorPosition = Math.max(Math.max(contentSelection.current.end, 0) + (parsedText.length - previousText.length), 0); + + if (compositionRef.current) { + divRef.current.value = parsedText; compositionRef.current = false; + contentSelection.current.end = newCursorPosition; return; } let newInputUpdate: ParseTextResult; - const nativeEvent = e.nativeEvent as MarkdownNativeEvent; - const inputType = nativeEvent.inputType; switch (inputType) { case 'historyUndo': newInputUpdate = undo(divRef.current); @@ -386,26 +319,15 @@ const MarkdownTextInput = React.forwardRef( case 'historyRedo': newInputUpdate = redo(divRef.current); break; - case 'insertFromPaste': - // if there is no newline at the end of the copied text, contentEditable adds invisible
tag at the end of the text, so we need to normalize it - if (changedText.length > 2 && changedText[changedText.length - 2] !== '\n' && changedText[changedText.length - 1] === '\n') { - newInputUpdate = parseText(parser, divRef.current, normalizeValue(changedText), processedMarkdownStyle); - break; - } - newInputUpdate = parseText(parser, divRef.current, changedText, processedMarkdownStyle); - break; default: - newInputUpdate = parseText(parser, divRef.current, changedText, processedMarkdownStyle); + newInputUpdate = parseText(parser, divRef.current, parsedText, processedMarkdownStyle, newCursorPosition, true, !inputType); } - const {text, cursorPosition} = newInputUpdate; - const normalizedText = normalizeValue(text); - - if (pasteRef?.current) { - pasteRef.current = false; - updateSelection(e); - } updateTextColor(divRef.current, text); + updateSelection(e, { + start: cursorPosition ?? 0, + end: cursorPosition ?? 0, + }); if (onChange) { const event = e as unknown as NativeSyntheticEvent<{ @@ -416,7 +338,7 @@ const MarkdownTextInput = React.forwardRef( setEventProps(event); // The new text is between the prev start selection and the new end selection, can be empty - const addedText = normalizedText.slice(prevSelection.start, cursorPosition ?? 0); + const addedText = text.slice(prevSelection.start, cursorPosition ?? 0); // The length of the text that replaced the before text const count = addedText.length; // The start index of the replacement operation @@ -427,7 +349,7 @@ const MarkdownTextInput = React.forwardRef( let before = prevSelectionRange; if (prevSelectionRange === 0 && (inputType === 'deleteContentBackward' || inputType === 'deleteContentForward')) { // its possible the user pressed a delete key without a selection range, so we need to adjust the before value to have the length of the deleted text - before = prevTextLength - normalizedText.length; + before = previousText.length - text.length; } if (inputType === 'deleteContentBackward') { @@ -445,12 +367,32 @@ const MarkdownTextInput = React.forwardRef( } if (onChangeText) { - onChangeText(normalizedText); + onChangeText(text); } handleContentSizeChange(); }, - [updateTextColor, handleContentSizeChange, onChange, onChangeText, undo, redo, parseText, processedMarkdownStyle, updateSelection, setEventProps], + [updateTextColor, updateSelection, onChange, onChangeText, handleContentSizeChange, undo, redo, parseText, processedMarkdownStyle, setEventProps], + ); + + const insertText = useCallback( + (e: SyntheticEvent, text: string) => { + if (!contentSelection.current || !divRef.current) { + return; + } + + const previousText = divRef.current.value; + const newText = `${divRef.current.value.substring(0, contentSelection.current.start)}${text}${divRef.current.value.substring(contentSelection.current.end)}`; + if (previousText === newText) { + document.execCommand('delete'); + } + + pasteContent.current = newText; + (e.nativeEvent as MarkdownNativeEvent).inputType = 'pasteText'; + + handleOnChangeText(e); + }, + [handleOnChangeText], ); const handleKeyPress = useCallback( @@ -462,7 +404,7 @@ const MarkdownTextInput = React.forwardRef( const hostNode = e.target; e.stopPropagation(); - if (e.key === 'z' && e.metaKey) { + if (e.key === 'z' && (e.ctrlKey || e.metaKey)) { e.preventDefault(); const nativeEvent = e.nativeEvent as unknown as MarkdownNativeEvent; if (e.shiftKey) { @@ -487,8 +429,6 @@ const MarkdownTextInput = React.forwardRef( onKeyPress(event); } - updateSelection(event as unknown as SyntheticEvent); - if ( e.key === 'Enter' && // Do not call submit if composition is occuring. @@ -502,16 +442,14 @@ const MarkdownTextInput = React.forwardRef( } else if (multiline) { // We need to change normal behavior of "Enter" key to insert a line breaks, to prevent wrapping contentEditable text in
tags. // Thanks to that in every situation we have proper amount of new lines in our parsed text. Without it pressing enter in empty lines will add 2 more new lines. - document.execCommand('insertLineBreak'); - CursorUtils.scrollCursorIntoView(divRef.current as HTMLInputElement); + insertText(e, '\n'); } - if (!e.shiftKey && ((shouldBlurOnSubmit && hostNode !== null) || !multiline)) { setTimeout(() => divRef.current && divRef.current.blur(), 0); } } }, - [multiline, blurOnSubmit, setEventProps, onKeyPress, updateSelection, handleOnChangeText, onSubmitEditing], + [multiline, blurOnSubmit, setEventProps, onKeyPress, handleOnChangeText, onSubmitEditing, insertText], ); const handleFocus: FocusEventHandler = useCallback( @@ -522,12 +460,11 @@ const MarkdownTextInput = React.forwardRef( setEventProps(e); if (divRef.current) { if (contentSelection.current) { - CursorUtils.setCursorPosition(divRef.current, contentSelection.current.start, contentSelection.current.end); + setCursorPosition(divRef.current, contentSelection.current.start, contentSelection.current.end); } else { - const valueLength = value ? value.length : divRef.current.innerText.length; - CursorUtils.setCursorPosition(divRef.current, valueLength, null); + const valueLength = value ? value.length : divRef.current.value.length; + setCursorPosition(divRef.current, valueLength, null); } - updateSelection(event, contentSelection.current); } if (onFocus) { @@ -537,7 +474,7 @@ const MarkdownTextInput = React.forwardRef( if (hostNode !== null) { if (clearTextOnFocus && divRef.current) { - divRef.current.innerText = ''; + divRef.current.textContent = ''; } if (selectTextOnFocus) { // Safari requires selection to occur in a setTimeout @@ -553,13 +490,13 @@ const MarkdownTextInput = React.forwardRef( } } }, - [clearTextOnFocus, onFocus, selectTextOnFocus, setEventProps, updateSelection, value], + [clearTextOnFocus, onFocus, selectTextOnFocus, setEventProps, value], ); const handleBlur: FocusEventHandler = useCallback( (event) => { const e = event as unknown as NativeSyntheticEvent; - CursorUtils.removeSelection(); + removeSelection(); currentlyFocusedField.current = null; if (onBlur) { setEventProps(e); @@ -571,44 +508,74 @@ const MarkdownTextInput = React.forwardRef( const handleClick = useCallback( (e: MouseEvent) => { - updateSelection(e); if (!onClick || !divRef.current) { return; } - (e.target as HTMLInputElement).value = normalizeValue(divRef.current.innerText || ''); + (e.target as HTMLInputElement).value = divRef.current.value; onClick(e); }, - [onClick, updateSelection], + [onClick], ); - const handlePaste = useCallback((e) => { - pasteRef.current = true; - if (e.isDefaultPrevented()) { + const handleCopy: ClipboardEventHandler = useCallback((e) => { + if (!divRef.current || !contentSelection.current) { return; } - e.preventDefault(); - const clipboardData = e.clipboardData; - const text = clipboardData.getData('text/plain'); - document.execCommand('insertText', false, text); + const text = divRef.current?.value.substring(contentSelection.current.start, contentSelection.current.end); + e.clipboardData.setData('text/plain', text ?? ''); }, []); + const handleCut = useCallback( + (e) => { + if (!divRef.current || !contentSelection.current) { + return; + } + handleCopy(e); + if (contentSelection.current.start !== contentSelection.current.end) { + document.execCommand('delete'); + } + }, + [handleCopy], + ); + + const handlePaste = useCallback( + (e) => { + if (e.isDefaultPrevented() || !divRef.current || !contentSelection.current) { + return; + } + e.preventDefault(); + const clipboardData = e.clipboardData; + const text = clipboardData.getData('text/plain'); + insertText(e, text); + }, + [insertText], + ); + const startComposition = useCallback(() => { compositionRef.current = true; }, []); + const endComposition = useCallback( + (e) => { + compositionRef.current = false; + handleOnChangeText(e); + }, + [handleOnChangeText], + ); + const setRef = (currentRef: HTMLDivElement | null) => { const r = currentRef; if (r) { (r as unknown as TextInput).isFocused = () => document.activeElement === r; (r as unknown as TextInput).clear = () => { - r.innerText = ''; + r.textContent = ''; updateTextColor(r, ''); }; if (value === '' || value === undefined) { // update to placeholder color when value is empty - updateTextColor(r, r.innerText); + updateTextColor(r, r.textContent ?? ''); } } @@ -620,26 +587,25 @@ const MarkdownTextInput = React.forwardRef( (ref as (elementRef: HTMLDivElement | null) => void)(r); } } - divRef.current = r; + divRef.current = r as MarkdownTextInputElement; }; useClientEffect( function parseAndStyleValue() { - if (!divRef.current || processedValue === divRef.current.innerText) { + if (!divRef.current || value === divRef.current.value) { return; } if (value === undefined) { - parseText(parser, divRef.current, divRef.current.innerText, processedMarkdownStyle); + parseText(parser, divRef.current, divRef.current.value, processedMarkdownStyle); return; } - - const text = processedValue !== undefined ? processedValue : ''; - - parseText(parser, divRef.current, text, processedMarkdownStyle, text.length); + const normalizedValue = normalizeValue(value); + divRef.current.value = normalizedValue; + parseText(parser, divRef.current, normalizedValue, processedMarkdownStyle); updateTextColor(divRef.current, value); }, - [multiline, processedMarkdownStyle, processedValue], + [multiline, processedMarkdownStyle, value], ); useClientEffect( @@ -674,11 +640,10 @@ const MarkdownTextInput = React.forwardRef( if (!divRef.current || !selection || (contentSelection.current && selection.start === contentSelection.current.start && selection.end === contentSelection.current.end)) { return; } - const newSelection: Selection = {start: selection.start, end: selection.end ?? selection.start}; contentSelection.current = newSelection; updateRefSelectionVariables(newSelection); - CursorUtils.setCursorPosition(divRef.current, newSelection.start, newSelection.end); + setCursorPosition(divRef.current, newSelection.start, newSelection.end); }, [selection, updateRefSelectionVariables]); return ( @@ -698,15 +663,19 @@ const MarkdownTextInput = React.forwardRef( className={className} onKeyDown={handleKeyPress} onCompositionStart={startComposition} - onKeyUp={updateSelection} + onCompositionEnd={endComposition} onInput={handleOnChangeText} onClick={handleClick} onFocus={handleFocus} onBlur={handleBlur} + onCopy={handleCopy} + onCut={handleCut} onPaste={handlePaste} placeholder={heightSafePlaceholder} spellCheck={spellCheck} dir={dir} + inputMode={inputMode} + onSelect={updateSelection} /> ); }, @@ -733,4 +702,4 @@ const styles = StyleSheet.create({ export default MarkdownTextInput; -export type {MarkdownTextInputProps}; +export type {MarkdownTextInputProps, MarkdownTextInputElement, HTMLMarkdownElement}; diff --git a/src/__tests__/parseExpensiMark.test.ts b/src/__tests__/parseExpensiMark.test.ts index c17334c7..4ef9b18f 100644 --- a/src/__tests__/parseExpensiMark.test.ts +++ b/src/__tests__/parseExpensiMark.test.ts @@ -1,14 +1,14 @@ import {expect} from '@jest/globals'; -import type * as ParserTypes from '../index'; import parseExpensiMark from '../parseExpensiMark'; +import type {MarkdownRange} from '../commonTypes'; declare module 'expect' { interface Matchers { - toBeParsedAs(expectedRanges: ParserTypes.Range[]): R; + toBeParsedAs(expectedRanges: MarkdownRange[]): R; } } -const toBeParsedAs = function (actual: string, expectedRanges: ParserTypes.Range[]) { +const toBeParsedAs = function (actual: string, expectedRanges: MarkdownRange[]) { const actualRanges = parseExpensiMark(actual); if (JSON.stringify(actualRanges) !== JSON.stringify(expectedRanges)) { return { diff --git a/src/__tests__/webParser.test.tsx b/src/__tests__/webParser.test.tsx index 388e3c8e..6f5c662c 100644 --- a/src/__tests__/webParser.test.tsx +++ b/src/__tests__/webParser.test.tsx @@ -1,7 +1,6 @@ /* eslint-disable @typescript-eslint/no-explicit-any */ import {expect} from '@jest/globals'; -import * as ParserUtils from '../web/parserUtils'; -import type * as MarkdownTypes from '../web/parserUtils'; +import {parseRangesToHTMLNodes} from '../web/utils/parserUtils'; import parseExpensiMark from '../parseExpensiMark'; declare module 'expect' { @@ -15,10 +14,9 @@ const toBeParsedAsHTML = function (actual: string, expectedHTML: string) { throw new Error('Actual value must be a string'); } let expected = expectedHTML; - const ranges = parseExpensiMark(actual); - const markdownRanges = ranges as MarkdownTypes.MarkdownRange[]; + const markdownRanges = parseExpensiMark(actual); - const actualDOM = ParserUtils.parseRangesToHTMLNodes(actual, markdownRanges, {}, true); + const actualDOM = parseRangesToHTMLNodes(actual, markdownRanges, {}, true).dom; const actualHTML = actualDOM.innerHTML; if (actualHTML === expected) { @@ -38,83 +36,107 @@ expect.extend({ }); test('empty string', () => { - expect('').toBeParsedAsHTML(''); + expect('').toBeParsedAsHTML('


'); }); test('no formatting', () => { - expect('Hello, world!').toBeParsedAsHTML('Hello, world!'); + expect('Hello, world!').toBeParsedAsHTML('

Hello, world!

'); }); test('bold', () => { - expect('Hello, *world*').toBeParsedAsHTML('Hello, *world*'); + expect('Hello, *world*!').toBeParsedAsHTML( + '

Hello, *world*!

', + ); }); test('italic', () => { - expect('Hello, _world_!').toBeParsedAsHTML('Hello, _world_!'); + expect('Hello, _world_!').toBeParsedAsHTML( + '

Hello, _world_!

', + ); }); test('strikethrough', () => { - expect('Hello, ~world~!').toBeParsedAsHTML('Hello, ~world~!'); + expect('Hello, ~world~!').toBeParsedAsHTML( + '

Hello, ~world~!

', + ); }); describe('mention-here', () => { test('normal', () => { - expect('@here Hello!').toBeParsedAsHTML('@here Hello!'); + expect('@here Hello!').toBeParsedAsHTML( + '

@here Hello!

', + ); }); test('with punctation marks', () => { - expect('@here!').toBeParsedAsHTML('@here!'); + expect('@here!').toBeParsedAsHTML( + '

@here!

', + ); }); test('at the beginning of a heading', () => { - expect('# @here').toBeParsedAsHTML('# @here'); + expect('# @here').toBeParsedAsHTML( + '

# @here

', + ); }); }); describe('mention-user', () => { test('normal', () => { - expect('@mail@mail.com Hello!').toBeParsedAsHTML('@mail@mail.com Hello!'); + expect('@mail@mail.com Hello!').toBeParsedAsHTML( + '

@mail@mail.com Hello!

', + ); }); test('with punctation marks', () => { - expect('@mail@mail.com!').toBeParsedAsHTML('@mail@mail.com!'); + expect('@mail@mail.com!').toBeParsedAsHTML( + '

@mail@mail.com!

', + ); }); test('at the beginning of a heading', () => { - expect('# @mail@mail.com').toBeParsedAsHTML('# @mail@mail.com'); + expect('# @mail@mail.com').toBeParsedAsHTML( + '

# @mail@mail.com

', + ); }); }); describe('link', () => { test('plain link', () => { - expect('https://example.com').toBeParsedAsHTML('https://example.com'); + expect('https://example.com').toBeParsedAsHTML( + '

https://example.com

', + ); }); test('labeled link', () => { expect('[Link](https://example.com)').toBeParsedAsHTML( - '[Link](https://example.com)', + '

[Link](https://example.com)

', ); }); test('link with same label as href', () => { expect('[https://example.com](https://example.com)').toBeParsedAsHTML( - '[https://example.com](https://example.com)', + '

[https://example.com](https://example.com)

', ); }); test('link with query string', () => { - expect('https://example.com?name=John&age=25&city=NewYork').toBeParsedAsHTML('https://example.com?name=John&age=25&city=NewYork'); + expect('https://example.com?name=John&age=25&city=NewYork').toBeParsedAsHTML( + '

https://example.com?name=John&age=25&city=NewYork

', + ); }); }); describe('email', () => { test('plain email', () => { - expect('someone@example.com').toBeParsedAsHTML('someone@example.com'); + expect('someone@example.com').toBeParsedAsHTML( + '

someone@example.com

', + ); }); test('labeled email', () => { expect('[Email](mailto:someone@example.com)').toBeParsedAsHTML( - '[Email](mailto:someone@example.com)', + '

[Email](mailto:someone@example.com)

', ); }); }); @@ -122,120 +144,148 @@ describe('email', () => { describe('email with same label as address', () => { test('label and address without "mailto:"', () => { expect('[someone@example.com](someone@example.com)').toBeParsedAsHTML( - '[someone@example.com](someone@example.com)', + '

[someone@example.com](someone@example.com)

', ); }); test('label with "mailto:"', () => { expect('[mailto:someone@example.com](someone@example.com)').toBeParsedAsHTML( - '[mailto:someone@example.com](someone@example.com)', + '

[mailto:someone@example.com](someone@example.com)

', ); }); test('address with "mailto:"', () => { expect('[someone@example.com](mailto:someone@example.com)').toBeParsedAsHTML( - '[someone@example.com](mailto:someone@example.com)', + '

[someone@example.com](mailto:someone@example.com)

', ); }); test('label and address with "mailto:"', () => { expect('[mailto:someone@example.com](mailto:someone@example.com)').toBeParsedAsHTML( - '[mailto:someone@example.com](mailto:someone@example.com)', + '

[mailto:someone@example.com](mailto:someone@example.com)

', ); }); }); test('inline code', () => { - expect('Hello `world`!').toBeParsedAsHTML('Hello `world`!'); + expect('Hello `world`!').toBeParsedAsHTML( + '

Hello `world`!

', + ); }); test('codeblock', () => { - expect('```\nHello world!\n```').toBeParsedAsHTML('```\nHello world!\n```'); + expect('```\nHello world!\n```').toBeParsedAsHTML( + '

```
Hello world!
```

', + ); }); describe('quote', () => { test('with single space', () => { - expect('> Hello world!').toBeParsedAsHTML('> Hello world!'); + expect('> Hello world!').toBeParsedAsHTML( + '

> Hello world!

', + ); }); test('with multiple spaces', () => { - expect('> Hello world!').toBeParsedAsHTML('> Hello world!'); + expect('> Hello world!').toBeParsedAsHTML( + '

> Hello world!

', + ); }); }); test('multiple blockquotes', () => { expect('> Hello\n> beautiful\n> world').toBeParsedAsHTML( - '> Hello\n> beautiful\n> world', + '

> Hello

> beautiful

> world

', ); }); test('separate blockquotes', () => { expect('> Lorem ipsum\ndolor\n> sit amet').toBeParsedAsHTML( - '> Lorem ipsum\ndolor\n> sit amet', + '

> Lorem ipsum

dolor

> sit amet

', ); }); test('nested blockquotes', () => { expect('> > > > Lorem ipsum dolor sit amet').toBeParsedAsHTML( - '> > > > Lorem ipsum dolor sit amet', + '

> > > > Lorem ipsum dolor sit amet

', ); }); test('heading', () => { - expect('# Hello world').toBeParsedAsHTML('# Hello world'); + expect('# Hello world').toBeParsedAsHTML( + '

# Hello world

', + ); }); test('nested bold and italic', () => { expect('*_Hello_*, _*world*_!').toBeParsedAsHTML( - '*_Hello_*, _*world*_!', + '

*_Hello_*, _*world*_!

', ); }); describe('nested heading in blockquote', () => { test('with single space', () => { - expect('> # Hello world').toBeParsedAsHTML('> # Hello world'); + expect('> # Hello world').toBeParsedAsHTML( + '

> # Hello world

', + ); }); test('with multiple spaces after #', () => { - expect('> # Hello world').toBeParsedAsHTML('> # Hello world'); + expect('> # Hello world').toBeParsedAsHTML( + '

> # Hello world

', + ); }); }); describe('trailing whitespace', () => { describe('after blockquote', () => { test('nothing', () => { - expect('> Hello world').toBeParsedAsHTML('> Hello world'); + expect('> Hello world').toBeParsedAsHTML( + '

> Hello world

', + ); }); test('single space', () => { - expect('> Hello world ').toBeParsedAsHTML('> Hello world '); + expect('> Hello world ').toBeParsedAsHTML( + '

> Hello world

', + ); }); test('newline', () => { - expect('> Hello world\n').toBeParsedAsHTML('> Hello world\n'); + expect('> Hello world\n').toBeParsedAsHTML( + '

> Hello world


', + ); }); }); describe('after heading', () => { test('nothing', () => { - expect('# Hello world').toBeParsedAsHTML('# Hello world'); + expect('# Hello world').toBeParsedAsHTML( + '

# Hello world

', + ); }); test('single space', () => { - expect('# Hello world ').toBeParsedAsHTML('# Hello world '); + expect('# Hello world ').toBeParsedAsHTML( + '

# Hello world

', + ); }); test('multiple spaces', () => { - expect('# Hello world ').toBeParsedAsHTML('# Hello world '); + expect('# Hello world ').toBeParsedAsHTML( + '

# Hello world

', + ); }); test('newline', () => { - expect('# Hello world\n').toBeParsedAsHTML('# Hello world\n'); + expect('# Hello world\n').toBeParsedAsHTML( + '

# Hello world


', + ); }); test('multiple quotes', () => { expect('> # Hello\n> # world').toBeParsedAsHTML( - '> # Hello\n> # world', + '

> # Hello

> # world

', ); }); }); diff --git a/src/commonTypes.ts b/src/commonTypes.ts new file mode 100644 index 00000000..fa2f0de5 --- /dev/null +++ b/src/commonTypes.ts @@ -0,0 +1,10 @@ +type MarkdownType = 'bold' | 'italic' | 'strikethrough' | 'emoji' | 'mention-here' | 'mention-user' | 'mention-report' | 'link' | 'code' | 'pre' | 'blockquote' | 'h1' | 'syntax'; + +interface MarkdownRange { + type: MarkdownType; + start: number; + length: number; + depth?: number; +} + +export type {MarkdownType, MarkdownRange}; diff --git a/src/index.tsx b/src/index.tsx index e8185a30..4f3b1d60 100644 --- a/src/index.tsx +++ b/src/index.tsx @@ -1,3 +1,4 @@ export {default as MarkdownTextInput} from './MarkdownTextInput'; -export type {MarkdownTextInputProps, MarkdownStyle, Range} from './MarkdownTextInput'; +export type {MarkdownTextInputProps, MarkdownStyle} from './MarkdownTextInput'; +export type {MarkdownType, MarkdownRange} from './commonTypes'; export {default as parseExpensiMark} from './parseExpensiMark'; diff --git a/src/parseExpensiMark.ts b/src/parseExpensiMark.ts index 6cf0801c..2965085e 100644 --- a/src/parseExpensiMark.ts +++ b/src/parseExpensiMark.ts @@ -1,9 +1,6 @@ import ExpensiMark from 'expensify-common/dist/ExpensiMark'; -import * as Utils from 'expensify-common/dist/utils'; -import type * as MarkdownTextInputTypes from './MarkdownTextInput'; - -type Range = MarkdownTextInputTypes.Range; -type MarkdownType = MarkdownTextInputTypes.MarkdownType; +import {unescapeText} from 'expensify-common/dist/utils'; +import type {MarkdownType, MarkdownRange} from './commonTypes'; type Token = ['TEXT' | 'HTML', string]; type StackItem = {tag: string; children: Array}; @@ -51,7 +48,7 @@ function parseTokensToTree(tokens: Token[]): StackItem { const stack: StackItem[] = [{tag: '<>', children: []}]; tokens.forEach(([type, payload]) => { if (type === 'TEXT') { - const text = Utils.unescapeText(payload); + const text = unescapeText(payload); const top = stack[stack.length - 1]; top!.children.push(text); } else if (type === 'HTML') { @@ -89,7 +86,7 @@ function parseTokensToTree(tokens: Token[]): StackItem { return stack[0]!; } -function parseTreeToTextAndRanges(tree: StackItem): [string, Range[]] { +function parseTreeToTextAndRanges(tree: StackItem): [string, MarkdownRange[]] { 'worklet'; let text = ''; @@ -113,7 +110,7 @@ function parseTreeToTextAndRanges(tree: StackItem): [string, Range[]] { ranges.push({type, start, length: end - start}); } - const ranges: Range[] = []; + const ranges: MarkdownRange[] = []; function dfs(node: StackItem | string) { if (typeof node === 'string') { text += node; @@ -166,10 +163,10 @@ function parseTreeToTextAndRanges(tree: StackItem): [string, Range[]] { appendSyntax('```'); } else if (node.tag.startsWith(' a.start - b.start || b.length - a.length || getTagPriority(b.type) - getTagPriority(a.type) || 0); } -function groupRanges(ranges: Range[]) { +function groupRanges(ranges: MarkdownRange[]) { 'worklet'; const lastVisibleRangeIndex: {[key in MarkdownType]?: number} = {}; @@ -260,10 +257,10 @@ function groupRanges(ranges: Range[]) { } return acc; - }, [] as Range[]); + }, [] as MarkdownRange[]); } -function parseExpensiMark(markdown: string): Range[] { +function parseExpensiMark(markdown: string): MarkdownRange[] { 'worklet'; try { @@ -288,4 +285,3 @@ function parseExpensiMark(markdown: string): Range[] { } export default parseExpensiMark; -export type {MarkdownType, Range}; diff --git a/src/styleUtils.ts b/src/styleUtils.ts index a6c161ac..4e66dcb8 100644 --- a/src/styleUtils.ts +++ b/src/styleUtils.ts @@ -1,7 +1,5 @@ import {Platform} from 'react-native'; -import type * as MarkdownTextInputDecoractorView from './MarkdownTextInputDecoratorViewNativeComponent'; - -type MarkdownStyle = MarkdownTextInputDecoractorView.MarkdownStyle; +import type {MarkdownStyle} from './MarkdownTextInputDecoratorViewNativeComponent'; type PartialMarkdownStyle = Partial<{ [K in keyof MarkdownStyle]: Partial; diff --git a/src/web/browserUtils.ts b/src/web/browserUtils.ts deleted file mode 100644 index 2bb782f6..00000000 --- a/src/web/browserUtils.ts +++ /dev/null @@ -1,11 +0,0 @@ -const isFirefox = navigator.userAgent?.toLowerCase().includes('firefox'); -const isChromium = 'chrome' in window; - -/** - * Whether the platform is a mobile browser. - * Copied from Expensify App https://github.com/Expensify/App/blob/90dee7accae79c49debf30354c160cab6c52c423/src/libs/Browser/index.website.ts#L41 - * - */ -const isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|BB|PlayBook|IEMobile|Windows Phone|Silk|Opera Mini/i.test(navigator?.userAgent ?? ''); - -export {isFirefox, isChromium, isMobile}; diff --git a/src/web/cursorUtils.ts b/src/web/cursorUtils.ts deleted file mode 100644 index 9d2ca9ae..00000000 --- a/src/web/cursorUtils.ts +++ /dev/null @@ -1,165 +0,0 @@ -import * as BrowserUtils from './browserUtils'; - -let prevTextLength: number | undefined; - -function getPrevTextLength() { - return prevTextLength; -} - -function findTextNodes(textNodes: Text[], node: ChildNode) { - if (node.nodeType === Node.TEXT_NODE) { - textNodes.push(node as Text); - } else { - for (let i = 0, length = node.childNodes.length; i < length; ++i) { - const childNode = node.childNodes[i]; - if (childNode) { - findTextNodes(textNodes, childNode); - } - } - } -} - -function setPrevText(target: HTMLElement) { - let text = []; - const textNodes: Text[] = []; - findTextNodes(textNodes, target); - text = textNodes - .map((e) => e.nodeValue ?? '') - ?.join('') - ?.split(''); - - prevTextLength = text.length; -} - -function setCursorPosition(target: HTMLElement, start: number, end: number | null = null) { - // We don't want to move the cursor if the target is not focused - if (target !== document.activeElement) { - return; - } - - const range = document.createRange(); - range.selectNodeContents(target); - - const textNodes: Text[] = []; - findTextNodes(textNodes, target); - - // These are utilities for handling the boundary cases (especially onEnter) - // prevChar & nextChar are characters before & after the target cursor position - const textCharacters = textNodes - .map((e) => e.nodeValue ?? '') - ?.join('') - ?.split(''); - const prevChar = textCharacters?.[start - 1] ?? ''; - const nextChar = textCharacters?.[start] ?? ''; - - let charCount = 0; - let startNode: Text | null = null; - let endNode: Text | null = null; - const n = textNodes.length; - for (let i = 0; i < n; ++i) { - const textNode = textNodes[i]; - if (textNode) { - const nextCharCount = charCount + textNode.length; - - if (!startNode && start >= charCount && (start <= nextCharCount || (start === nextCharCount && i < n - 1))) { - startNode = textNode; - - // There are 4 cases to consider here: - // 1. Caret in front of a character, when pressing enter - // 2. Caret at the end of a line (not last one) - // 3. Caret at the end of whole input, when pressing enter - // 4. All other placements - if (prevChar === '\n' && prevTextLength !== undefined && prevTextLength < textCharacters.length) { - if (nextChar && nextChar !== '\n' && i !== n - 1) { - range.setStart(textNodes[i + 1] as Node, 0); - } else if (i !== textNodes.length - 1) { - range.setStart(textNodes[i] as Node, 1); - } else { - range.setStart(textNode, start - charCount); - } - } else { - range.setStart(textNode, start - charCount); - } - if (!end) { - break; - } - } - if (end && !endNode && end >= charCount && (end <= nextCharCount || (end === nextCharCount && i < n - 1))) { - endNode = textNode; - range.setEnd(textNode, end - charCount); - } - charCount = nextCharCount; - } - } - - if (!end) { - range.collapse(true); - } - - const selection = window.getSelection(); - if (selection) { - selection.setBaseAndExtent(range.startContainer, range.startOffset, range.endContainer, range.endOffset); - } - - scrollCursorIntoView(target as HTMLInputElement); -} - -function moveCursorToEnd(target: HTMLElement) { - const range = document.createRange(); - const selection = window.getSelection(); - if (selection) { - range.setStart(target, target.childNodes.length); - range.collapse(true); - selection.setBaseAndExtent(range.startContainer, range.startOffset, range.endContainer, range.endOffset); - } -} - -function getCurrentCursorPosition(target: HTMLElement) { - const selection = window.getSelection(); - if (!selection || (selection && selection.rangeCount === 0)) { - return null; - } - const range = selection.getRangeAt(0); - const preSelectionRange = range.cloneRange(); - preSelectionRange.selectNodeContents(target); - preSelectionRange.setEnd(range.startContainer, range.startOffset); - const start = preSelectionRange.toString().length; - const end = start + range.toString().length; - return {start, end}; -} - -function removeSelection() { - const selection = window.getSelection(); - if (selection) { - selection.removeAllRanges(); - } -} - -function scrollCursorIntoView(target: HTMLInputElement) { - if (target.selectionStart === null || !target.value || BrowserUtils.isFirefox) { - return; - } - - const selection = window.getSelection(); - if (!selection || (selection && selection.rangeCount === 0)) { - return; - } - - const caretRect = selection.getRangeAt(0).getClientRects()[0]; - const editableRect = target.getBoundingClientRect(); - - // Adjust for padding and border - const paddingTop = parseFloat(window.getComputedStyle(target).paddingTop); - const borderTop = parseFloat(window.getComputedStyle(target).borderTopWidth); - - if (caretRect && !(caretRect.top >= editableRect.top + paddingTop + borderTop && caretRect.bottom <= editableRect.bottom - 2 * (paddingTop - borderTop))) { - const topToCaret = caretRect.top - editableRect.top; - const inputHeight = editableRect.height; - // Chrome Rects don't include padding & border, so we're adding them manually - const inputOffset = caretRect.height - inputHeight + paddingTop + borderTop + (BrowserUtils.isChromium ? 0 : 4 * (paddingTop + borderTop)); - - target.scrollTo(0, topToCaret + target.scrollTop + inputOffset); - } -} - -export {getCurrentCursorPosition, moveCursorToEnd, setCursorPosition, setPrevText, removeSelection, scrollCursorIntoView, getPrevTextLength}; diff --git a/src/web/parserUtils.ts b/src/web/parserUtils.ts deleted file mode 100644 index c3175eb6..00000000 --- a/src/web/parserUtils.ts +++ /dev/null @@ -1,236 +0,0 @@ -import * as CursorUtils from './cursorUtils'; -import type * as StyleUtilsTypes from '../styleUtils'; -import * as BrowserUtils from './browserUtils'; -import type * as MarkdownTextInputTypes from '../MarkdownTextInput'; - -type PartialMarkdownStyle = StyleUtilsTypes.PartialMarkdownStyle; -type Range = MarkdownTextInputTypes.Range; -type MarkdownType = MarkdownTextInputTypes.MarkdownType; - -type MarkdownRange = { - type: MarkdownType; - start: number; - length: number; - depth?: number; -}; - -type NestedNode = { - node: HTMLElement; - endIndex: number; -}; - -function addStyling(targetElement: HTMLElement, type: MarkdownType, markdownStyle: PartialMarkdownStyle) { - const node = targetElement; - switch (type) { - case 'syntax': - Object.assign(node.style, markdownStyle.syntax); - break; - case 'bold': - node.style.fontWeight = 'bold'; - break; - case 'italic': - node.style.fontStyle = 'italic'; - break; - case 'strikethrough': - node.style.textDecoration = 'line-through'; - break; - case 'emoji': - Object.assign(node.style, {...markdownStyle.emoji, verticalAlign: 'middle'}); - break; - case 'mention-here': - Object.assign(node.style, markdownStyle.mentionHere); - break; - case 'mention-user': - Object.assign(node.style, markdownStyle.mentionUser); - break; - case 'mention-report': - Object.assign(node.style, markdownStyle.mentionReport); - break; - case 'link': - Object.assign(node.style, { - ...markdownStyle.link, - textDecoration: 'underline', - }); - break; - case 'code': - Object.assign(node.style, markdownStyle.code); - break; - case 'pre': - Object.assign(node.style, markdownStyle.pre); - break; - - case 'blockquote': - Object.assign(node.style, { - ...markdownStyle.blockquote, - borderLeftStyle: 'solid', - display: 'inline-block', - maxWidth: '100%', - boxSizing: 'border-box', - }); - break; - case 'h1': - Object.assign(node.style, { - ...markdownStyle.h1, - fontWeight: 'bold', - }); - break; - default: - break; - } -} - -function addSubstringAsTextNode(root: HTMLElement, text: string, startIndex: number, endIndex: number) { - const substring = text.substring(startIndex, endIndex); - if (substring.length > 0) { - root.appendChild(document.createTextNode(substring)); - } -} - -function ungroupRanges(ranges: MarkdownRange[]): MarkdownRange[] { - const ungroupedRanges: MarkdownRange[] = []; - ranges.forEach((range) => { - if (!range.depth) { - ungroupedRanges.push(range); - } - const {depth, ...rangeWithoutDepth} = range; - Array.from({length: depth!}).forEach(() => { - ungroupedRanges.push(rangeWithoutDepth); - }); - }); - return ungroupedRanges; -} - -function parseRangesToHTMLNodes(text: string, ranges: MarkdownRange[], markdownStyle: PartialMarkdownStyle = {}, disableInlineStyles = false): HTMLElement { - const root: HTMLElement = document.createElement('span'); - root.className = 'root'; - const textLength = text.length; - if (ranges.length === 0) { - addSubstringAsTextNode(root, text, 0, textLength); - return root; - } - - const stack = ungroupRanges(ranges); - const nestedStack: NestedNode[] = [{node: root, endIndex: textLength}]; - let lastRangeEndIndex = 0; - while (stack.length > 0) { - const range = stack.shift(); - if (!range) { - break; - } - let currentRoot = nestedStack[nestedStack.length - 1]; - if (!currentRoot) { - break; - } - - const endOfCurrentRange = range.start + range.length; - const nextRangeStartIndex = stack.length > 0 && !!stack[0] ? stack[0].start || 0 : textLength; - - addSubstringAsTextNode(currentRoot.node, text, lastRangeEndIndex, range.start); // add text with newlines before current range - - const span = document.createElement('span'); - if (disableInlineStyles) { - span.className = range.type; - } else { - addStyling(span, range.type, markdownStyle); - } - - if (stack.length > 0 && nextRangeStartIndex < endOfCurrentRange && range.type !== 'syntax') { - // tag nesting - currentRoot.node.appendChild(span); - nestedStack.push({node: span, endIndex: endOfCurrentRange}); - lastRangeEndIndex = range.start; - } else { - addSubstringAsTextNode(span, text, range.start, endOfCurrentRange); - currentRoot.node.appendChild(span); - lastRangeEndIndex = endOfCurrentRange; - - // end of tag nesting - while (nestedStack.length - 1 > 0 && nextRangeStartIndex >= currentRoot.endIndex) { - addSubstringAsTextNode(currentRoot.node, text, lastRangeEndIndex, currentRoot.endIndex); - const prevRoot = nestedStack.pop(); - if (!prevRoot) { - break; - } - lastRangeEndIndex = prevRoot.endIndex; - currentRoot = nestedStack[nestedStack.length - 1] || currentRoot; - } - } - } - - if (nestedStack.length > 1) { - const lastNestedNode = nestedStack[nestedStack.length - 1]; - if (lastNestedNode) { - root.appendChild(lastNestedNode.node); - } - } - - addSubstringAsTextNode(root, text, lastRangeEndIndex, textLength); - return root; -} - -function moveCursor(isFocused: boolean, alwaysMoveCursorToTheEnd: boolean, cursorPosition: number | null, target: HTMLElement) { - if (!isFocused) { - return; - } - - if (alwaysMoveCursorToTheEnd || cursorPosition === null) { - CursorUtils.moveCursorToEnd(target); - } else if (cursorPosition !== null) { - CursorUtils.setCursorPosition(target, cursorPosition); - } -} - -function parseText( - parser: (input: string) => Range[], - target: HTMLElement, - text: string, - cursorPositionIndex: number | null, - markdownStyle: PartialMarkdownStyle = {}, - alwaysMoveCursorToTheEnd = false, -) { - const targetElement = target; - - // in case the cursorPositionIndex is larger than text length, cursorPosition will be null, i.e: move the caret to the end - let cursorPosition: number | null = cursorPositionIndex && cursorPositionIndex <= text.length ? cursorPositionIndex : null; - const isFocused = document.activeElement === target; - if (isFocused && cursorPositionIndex === null) { - const selection = CursorUtils.getCurrentCursorPosition(target); - cursorPosition = selection ? selection.end : null; - } - const ranges = parser(text); - - const markdownRanges: MarkdownRange[] = ranges as MarkdownRange[]; - const rootSpan = targetElement.firstChild as HTMLElement | null; - - if (!text || targetElement.innerHTML === '
' || (rootSpan && rootSpan.innerHTML === '\n')) { - targetElement.innerHTML = ''; - targetElement.innerText = ''; - } - - // We don't want to parse text with single '\n', because contentEditable represents it as invisible
- if (text) { - const dom = parseRangesToHTMLNodes(text, markdownRanges, markdownStyle); - - if (!rootSpan || !rootSpan?.classList?.contains('root') || rootSpan.innerHTML !== dom.innerHTML) { - targetElement.innerHTML = ''; - targetElement.innerText = ''; - target.appendChild(dom); - - if (BrowserUtils.isChromium) { - moveCursor(isFocused, alwaysMoveCursorToTheEnd, cursorPosition, target); - } - } - - if (!BrowserUtils.isChromium) { - moveCursor(isFocused, alwaysMoveCursorToTheEnd, cursorPosition, target); - } - } - - CursorUtils.setPrevText(target); - - return {text: target.innerText, cursorPosition: cursorPosition || 0}; -} - -export {parseText, parseRangesToHTMLNodes}; - -export type {MarkdownRange, MarkdownType}; diff --git a/src/web/utils/blockUtils.ts b/src/web/utils/blockUtils.ts new file mode 100644 index 00000000..887ef4e1 --- /dev/null +++ b/src/web/utils/blockUtils.ts @@ -0,0 +1,72 @@ +import type {PartialMarkdownStyle} from '../../styleUtils'; +import type {NodeType} from './treeUtils'; + +function addStyleToBlock(targetElement: HTMLElement, type: NodeType, markdownStyle: PartialMarkdownStyle) { + const node = targetElement; + switch (type) { + case 'line': + Object.assign(node.style, { + display: 'block', + margin: '0', + padding: '0', + }); + break; + case 'syntax': + Object.assign(node.style, markdownStyle.syntax); + break; + case 'bold': + node.style.fontWeight = 'bold'; + break; + case 'italic': + node.style.fontStyle = 'italic'; + break; + case 'strikethrough': + node.style.textDecoration = 'line-through'; + break; + case 'emoji': + Object.assign(node.style, {...markdownStyle.emoji, verticalAlign: 'middle'}); + break; + case 'mention-here': + Object.assign(node.style, markdownStyle.mentionHere); + break; + case 'mention-user': + Object.assign(node.style, markdownStyle.mentionUser); + break; + case 'mention-report': + Object.assign(node.style, markdownStyle.mentionReport); + break; + case 'link': + Object.assign(node.style, { + ...markdownStyle.link, + textDecoration: 'underline', + }); + break; + case 'code': + Object.assign(node.style, markdownStyle.code); + break; + case 'pre': + Object.assign(node.style, markdownStyle.pre); + break; + + case 'blockquote': + Object.assign(node.style, { + ...markdownStyle.blockquote, + borderLeftStyle: 'solid', + display: 'inline-block', + maxWidth: '100%', + boxSizing: 'border-box', + }); + break; + case 'h1': + Object.assign(node.style, { + ...markdownStyle.h1, + fontWeight: 'bold', + }); + break; + default: + break; + } +} + +// eslint-disable-next-line import/prefer-default-export +export {addStyleToBlock}; diff --git a/src/web/utils/browserUtils.ts b/src/web/utils/browserUtils.ts new file mode 100644 index 00000000..cfe157c5 --- /dev/null +++ b/src/web/utils/browserUtils.ts @@ -0,0 +1,13 @@ +const BrowserUtils = { + isFirefox: navigator.userAgent.toLowerCase().includes('firefox'), + isChromium: 'chrome' in window, + + /** + * Whether the platform is a mobile browser. + * Copied from Expensify App https://github.com/Expensify/App/blob/90dee7accae79c49debf30354c160cab6c52c423/src/libs/Browser/index.website.ts#L41 + * + */ + isMobile: /Android|webOS|iPhone|iPad|iPod|BlackBerry|BB|PlayBook|IEMobile|Windows Phone|Silk|Opera Mini/i.test(navigator.userAgent), +}; + +export default BrowserUtils; diff --git a/src/web/utils/cursorUtils.ts b/src/web/utils/cursorUtils.ts new file mode 100644 index 00000000..9ad986c9 --- /dev/null +++ b/src/web/utils/cursorUtils.ts @@ -0,0 +1,120 @@ +import type {MarkdownTextInputElement} from '../../MarkdownTextInput.web'; +import {findHTMLElementInTree, getTreeNodeByIndex} from './treeUtils'; +import type {TreeNode} from './treeUtils'; + +function setCursorPosition(target: MarkdownTextInputElement, startIndex: number, endIndex: number | null = null) { + // We don't want to move the cursor if the target is not focused + if (!target.tree || target !== document.activeElement) { + return; + } + + const start = Math.max(0, Math.min(startIndex, target.tree.length)); + const end = endIndex ? Math.max(0, Math.min(endIndex, target.tree.length)) : null; + if (end && end < start) { + return; + } + + const range = document.createRange(); + range.selectNodeContents(target); + + const startTreeNode = getTreeNodeByIndex(target.tree, start); + const endTreeNode = end && startTreeNode && (end < startTreeNode.start || end >= startTreeNode.start + startTreeNode.length) ? getTreeNodeByIndex(target.tree, end) : startTreeNode; + if (!startTreeNode || !endTreeNode) { + console.error('Invalid start or end tree node'); + return; + } + + if (startTreeNode.type === 'br') { + range.setStartBefore(startTreeNode.element); + } else { + range.setStart(startTreeNode.element.childNodes[0] as ChildNode, start - startTreeNode.start); + } + + if (endTreeNode.type === 'br') { + range.setEndBefore(endTreeNode.element); + } else { + range.setEnd(endTreeNode.element.childNodes[0] as ChildNode, (end || start) - endTreeNode.start); + } + + if (!end) { + range.collapse(true); + } + + const selection = window.getSelection(); + if (selection) { + selection.setBaseAndExtent(range.startContainer, range.startOffset, range.endContainer, range.endOffset); + } + + // Update the focused elements zIndex to make sure they are on top and the cursor is beeing set correctly + startTreeNode.element.style.zIndex = '1'; + endTreeNode.element.style.zIndex = '1'; + + scrollIntoView(startTreeNode); +} + +function scrollIntoView(node: TreeNode) { + if (node.type === 'br' && node.parentNode?.parentNode?.type === 'line') { + // If the node is a line break, scroll to the parent paragraph, because Safari doesn't support scrollIntoView on br elements + node.parentNode.parentNode.element.scrollIntoView({ + block: 'nearest', + }); + } else { + node.element.scrollIntoView({ + block: 'nearest', + }); + } +} + +function moveCursorToEnd(target: HTMLElement) { + const range = document.createRange(); + const selection = window.getSelection(); + if (selection) { + range.setStart(target, target.childNodes.length); + range.collapse(true); + selection.setBaseAndExtent(range.startContainer, range.startOffset, range.endContainer, range.endOffset); + } +} + +function getCurrentCursorPosition(target: MarkdownTextInputElement) { + function getHTMLElement(node: Node) { + let element = node as HTMLElement | Text; + if (element instanceof Text) { + element = node.parentElement as HTMLElement; + } + return element; + } + + const selection = window.getSelection(); + if (!selection || (selection && selection.rangeCount === 0)) { + return null; + } + const range = selection.getRangeAt(0); + const startElement = getHTMLElement(range.startContainer); + const endElement = range.startContainer === range.endContainer ? startElement : getHTMLElement(range.endContainer); + + const startTreeNode = findHTMLElementInTree(target.tree, startElement); + const endTreeNode = findHTMLElementInTree(target.tree, endElement); + + let start = -1; + let end = -1; + if (startTreeNode && endTreeNode) { + start = startTreeNode.start + range.startOffset; + + // If the end node is a root node, we need to set the end to the end of the text (FireFox fix) + if (endTreeNode?.parentNode === null) { + end = target.value.length; + } else { + end = endTreeNode.start + range.endOffset; + } + } + return {start, end}; +} + +function removeSelection() { + const selection = window.getSelection(); + if (selection) { + selection.removeAllRanges(); + } +} + +export {getCurrentCursorPosition, moveCursorToEnd, setCursorPosition, removeSelection, scrollIntoView}; diff --git a/src/web/utils/inputUtils.ts b/src/web/utils/inputUtils.ts new file mode 100644 index 00000000..0e006990 --- /dev/null +++ b/src/web/utils/inputUtils.ts @@ -0,0 +1,106 @@ +import type {CSSProperties} from 'react'; +import type {MarkdownTextInputElement} from '../../MarkdownTextInput.web'; + +const ZERO_WIDTH_SPACE = '\u200B'; + +// If an Input Method Editor is processing key input, the 'keyCode' is 229. +// https://www.w3.org/TR/uievents/#determine-keydown-keyup-keyCode +function isEventComposing(nativeEvent: globalThis.KeyboardEvent) { + return nativeEvent.isComposing || nativeEvent.keyCode === 229; +} + +function getPlaceholderValue(placeholder: string | undefined) { + if (!placeholder) { + return ZERO_WIDTH_SPACE; + } + return placeholder.length ? placeholder : ZERO_WIDTH_SPACE; +} + +function getElementHeight(node: HTMLDivElement, styles: CSSProperties, numberOfLines: number | undefined) { + if (numberOfLines) { + const tempElement = document.createElement('div'); + tempElement.setAttribute('contenteditable', 'true'); + Object.assign(tempElement.style, styles); + tempElement.textContent = Array(numberOfLines).fill('A').join('\n'); + if (node.parentElement) { + node.parentElement.appendChild(tempElement); + const height = tempElement.clientHeight; + node.parentElement.removeChild(tempElement); + return height ? `${height}px` : 'auto'; + } + } + return styles.height ? `${styles.height}px` : 'auto'; +} + +function normalizeValue(value: string) { + return value.replaceAll('\r\n', '\n'); +} + +// Parses the HTML structure of a MarkdownTextInputElement to a plain text string. Used for getting the correct value of the input element. +function parseInnerHTMLToText(target: MarkdownTextInputElement): string { + // Returns the parent of a given node that is higher in the hierarchy and is of a different type than 'text', 'br' or 'line' + function getTopParentNode(node: ChildNode) { + let currentParentNode = node.parentNode; + while (currentParentNode && ['text', 'br', 'line'].includes(currentParentNode.parentElement?.getAttribute('data-type') || '')) { + currentParentNode = currentParentNode?.parentNode || null; + } + return currentParentNode; + } + + const stack: ChildNode[] = [target]; + let text = ''; + let shouldAddNewline = false; + const lastNode = target.childNodes[target.childNodes.length - 1]; + // Remove the last
element if it's the last child of the target element. Fixes the issue with adding extra newline when pasting into the empty input. + if (lastNode?.nodeName === 'DIV' && (lastNode as HTMLElement)?.innerHTML === '
') { + target.removeChild(lastNode); + } + + while (stack.length > 0) { + const node = stack.pop(); + if (!node) { + break; + } + + // If we are operating on the nodes that are children of the MarkdownTextInputElement, we need to add a newline after each + const isTopComponent = node.parentElement?.contentEditable === 'true'; + if (isTopComponent) { + // Replaced text is beeing added as text node, so we need to not add the newline before and after it + if (node.nodeType === Node.TEXT_NODE) { + shouldAddNewline = false; + } else { + if (shouldAddNewline) { + text += '\n'; + shouldAddNewline = false; + } + shouldAddNewline = true; + } + } + + if (node.nodeType === Node.TEXT_NODE) { + // Parse text nodes into text + text += node.textContent; + } else if (node.nodeName === 'BR') { + const parentNode = getTopParentNode(node); + if (parentNode && parentNode.parentElement?.contentEditable !== 'true') { + // Parse br elements into newlines only if their parent is not a child of the MarkdownTextInputElement (a paragraph when writing or a div when pasting). + // It prevents adding extra newlines when entering text + text += '\n'; + } + } else { + let i = node.childNodes.length - 1; + while (i > -1) { + const child = node.childNodes[i]; + if (!child) { + break; + } + stack.push(child); + i--; + } + } + } + + return text.replaceAll('\r\n', '\n'); +} + +export {isEventComposing, getPlaceholderValue, getElementHeight, parseInnerHTMLToText, normalizeValue}; diff --git a/src/web/utils/parserUtils.ts b/src/web/utils/parserUtils.ts new file mode 100644 index 00000000..22e514a8 --- /dev/null +++ b/src/web/utils/parserUtils.ts @@ -0,0 +1,292 @@ +import type {HTMLMarkdownElement, MarkdownTextInputElement} from '../../MarkdownTextInput.web'; +import {addNodeToTree} from './treeUtils'; +import type {NodeType, TreeNode} from './treeUtils'; +import type {PartialMarkdownStyle} from '../../styleUtils'; +import {getCurrentCursorPosition, moveCursorToEnd, setCursorPosition} from './cursorUtils'; +import {addStyleToBlock} from './blockUtils'; +import type {MarkdownRange} from '../../commonTypes'; + +type Paragraph = { + text: string; + start: number; + length: number; + Ranges: MarkdownRange[]; +}; + +function ungroupRanges(ranges: MarkdownRange[]): MarkdownRange[] { + const ungroupedRanges: MarkdownRange[] = []; + ranges.forEach((range) => { + if (!range.depth) { + ungroupedRanges.push(range); + } + const {depth, ...rangeWithoutDepth} = range; + Array.from({length: depth!}).forEach(() => { + ungroupedRanges.push(rangeWithoutDepth); + }); + }); + return ungroupedRanges; +} + +function splitTextIntoLines(text: string): Paragraph[] { + let lineStartIndex = 0; + const lines: Paragraph[] = text.split('\n').map((line) => { + const lineObject: Paragraph = { + text: line, + start: lineStartIndex, + length: line.length, + Ranges: [], + }; + lineStartIndex += line.length + 1; // Adding 1 for the newline character + return lineObject; + }); + + return lines; +} + +/** Merges lines that contain multiline markdown tags into one line */ +function mergeLinesWithMultilineTags(lines: Paragraph[], ranges: MarkdownRange[]) { + let mergedLines = [...lines]; + const lineIndexes = mergedLines.map((_line, index) => index); + + ranges.forEach((range) => { + const beginLineIndex = mergedLines.findLastIndex((line) => line.start <= range.start); + const endLineIndex = mergedLines.findIndex((line) => line.start + line.length >= range.start + range.length); + const correspondingLineIndexes = lineIndexes.slice(beginLineIndex, endLineIndex + 1); + + if (correspondingLineIndexes.length > 0) { + const mainLineIndex = correspondingLineIndexes[0] as number; + const mainLine = mergedLines[mainLineIndex] as Paragraph; + + mainLine.Ranges.push(range); + + const otherLineIndexes = correspondingLineIndexes.slice(1); + otherLineIndexes.forEach((lineIndex) => { + const otherLine = mergedLines[lineIndex] as Paragraph; + + mainLine.text += `\n${otherLine.text}`; + mainLine.length += otherLine.length + 1; + mainLine.Ranges.push(...otherLine.Ranges); + }); + if (otherLineIndexes.length > 0) { + mergedLines = mergedLines.filter((_line, index) => !otherLineIndexes.includes(index)); + } + } + }); + + return mergedLines; +} + +/** Adds a value prop to the element and appends the value to the parent node element */ +function appendValueToElement(element: HTMLMarkdownElement, parentTreeNode: TreeNode, value: string) { + const targetElement = element; + const parentNode = parentTreeNode; + targetElement.value = value; + parentNode.element.value = (parentNode.element.value || '') + value; +} + +function appendNode(element: HTMLMarkdownElement, parentTreeNode: TreeNode, type: NodeType, length: number) { + const node = addNodeToTree(element, parentTreeNode, type, length); + parentTreeNode.element.appendChild(element); + return node; +} + +function addBrElement(node: TreeNode) { + const span = document.createElement('span') as HTMLMarkdownElement; + span.setAttribute('data-type', 'br'); + appendValueToElement(span, node, '\n'); + const spanNode = appendNode(span, node, 'br', 1); + appendNode(document.createElement('br') as unknown as HTMLMarkdownElement, spanNode, 'br', 1); + return spanNode; +} + +function addTextToElement(node: TreeNode, text: string) { + const lines = text.split('\n'); + lines.forEach((line, index) => { + if (line !== '') { + const span = document.createElement('span') as HTMLMarkdownElement; + appendValueToElement(span, node, line); + span.setAttribute('data-type', 'text'); + span.appendChild(document.createTextNode(line)); + appendNode(span, node, 'text', line.length); + } + + if (index < lines.length - 1 || (index === 0 && line === '')) { + addBrElement(node); + } + }); +} + +function addParagraph(node: TreeNode, text: string | null = null, length: number, disableInlineStyles = false) { + const p = document.createElement('p'); + p.setAttribute('data-type', 'line'); + if (!disableInlineStyles) { + addStyleToBlock(p, 'line', {}); + } + + const pNode = appendNode(p as unknown as HTMLMarkdownElement, node, 'line', length); + + if (text === '') { + // If the line is empty, we still need to add a br element to keep the line height + addBrElement(pNode); + } else if (text) { + addTextToElement(pNode, text); + } + + return pNode; +} + +/** Builds HTML DOM structure based on passed text and markdown ranges */ +function parseRangesToHTMLNodes(text: string, ranges: MarkdownRange[], markdownStyle: PartialMarkdownStyle = {}, disableInlineStyles = false) { + const rootElement: HTMLMarkdownElement = document.createElement('span') as HTMLMarkdownElement; + const textLength = text.length; + const rootNode: TreeNode = { + element: rootElement, + start: 0, + length: textLength, + parentNode: null, + childNodes: [], + type: 'root', + orderIndex: '', + isGeneratingNewline: false, + }; + let currentParentNode: TreeNode = rootNode; + let lines = splitTextIntoLines(text); + + if (ranges.length === 0) { + lines.forEach((line) => { + addParagraph(rootNode, line.text, line.length, disableInlineStyles); + }); + return {dom: rootElement, tree: rootNode}; + } + + const Ranges = ungroupRanges(ranges); + lines = mergeLinesWithMultilineTags(lines, Ranges); + + let lastRangeEndIndex = 0; + while (lines.length > 0) { + const line = lines.shift(); + if (!line) { + break; + } + + // preparing line paragraph element for markdown text + currentParentNode = addParagraph(rootNode, null, line.length, disableInlineStyles); + rootElement.value = (rootElement.value || '') + line.text; + if (lines.length > 0) { + rootElement.value = `${rootElement.value || ''}\n`; + } + + if (line.Ranges.length === 0) { + addTextToElement(currentParentNode, line.text); + } + + lastRangeEndIndex = line.start; + const lineRanges = line.Ranges; + // go through all markdown ranges in the line + while (lineRanges.length > 0) { + const range = lineRanges.shift(); + if (!range) { + break; + } + + const endOfCurrentRange = range.start + range.length; + const nextRangeStartIndex = lineRanges.length > 0 && !!lineRanges[0] ? lineRanges[0].start || 0 : textLength; + + // add text before the markdown range + const textBeforeRange = line.text.substring(lastRangeEndIndex - line.start, range.start - line.start); + if (textBeforeRange) { + addTextToElement(currentParentNode, textBeforeRange); + } + + // create markdown span element + const span = document.createElement('span') as HTMLMarkdownElement; + span.setAttribute('data-type', range.type); + if (!disableInlineStyles) { + addStyleToBlock(span, range.type, markdownStyle); + } + + const spanNode = appendNode(span, currentParentNode, range.type, range.length); + + if (lineRanges.length > 0 && nextRangeStartIndex < endOfCurrentRange && range.type !== 'syntax') { + // tag nesting + currentParentNode = spanNode; + lastRangeEndIndex = range.start; + } else { + // adding markdown tag + addTextToElement(spanNode, text.substring(range.start, endOfCurrentRange)); + currentParentNode.element.value = (currentParentNode.element.value || '') + (spanNode.element.value || ''); + lastRangeEndIndex = endOfCurrentRange; + // tag unnesting and adding text after the tag + while (currentParentNode.parentNode !== null && nextRangeStartIndex >= currentParentNode.start + currentParentNode.length) { + const textAfterRange = line.text.substring(lastRangeEndIndex - line.start, currentParentNode.start - line.start + currentParentNode.length); + if (textAfterRange) { + addTextToElement(currentParentNode, textAfterRange); + } + lastRangeEndIndex = currentParentNode.start + currentParentNode.length; + if (currentParentNode.parentNode.type !== 'root') { + currentParentNode.parentNode.element.value = currentParentNode.element.value || ''; + } + currentParentNode = currentParentNode.parentNode || rootNode; + } + } + } + } + + return {dom: rootElement, tree: rootNode}; +} + +function moveCursor(isFocused: boolean, alwaysMoveCursorToTheEnd: boolean, cursorPosition: number | null, target: MarkdownTextInputElement) { + if (!isFocused) { + return; + } + + if (alwaysMoveCursorToTheEnd || cursorPosition === null) { + moveCursorToEnd(target); + } else if (cursorPosition !== null) { + setCursorPosition(target, cursorPosition); + } +} +function updateInputStructure( + parserFunction: (input: string) => MarkdownRange[], + target: MarkdownTextInputElement, + text: string, + cursorPositionIndex: number | null, + markdownStyle: PartialMarkdownStyle = {}, + alwaysMoveCursorToTheEnd = false, + shouldForceDOMUpdate = false, +) { + const targetElement = target; + + // in case the cursorPositionIndex is larger than text length, cursorPosition will be null, i.e: move the caret to the end + let cursorPosition: number | null = cursorPositionIndex !== null && cursorPositionIndex <= text.length ? cursorPositionIndex : null; + const isFocused = document.activeElement === target; + if (isFocused && cursorPositionIndex === null) { + const selection = getCurrentCursorPosition(target); + cursorPosition = selection ? selection.start : null; + } + const ranges = parserFunction(text); + if (!text || targetElement.innerHTML === '
' || (targetElement && targetElement.innerHTML === '\n')) { + targetElement.innerHTML = ''; + targetElement.innerText = ''; + } + + // We don't want to parse text with single '\n', because contentEditable represents it as invisible
+ if (text) { + const {dom, tree} = parseRangesToHTMLNodes(text, ranges, markdownStyle); + + if (shouldForceDOMUpdate || targetElement.innerHTML !== dom.innerHTML) { + targetElement.innerHTML = ''; + targetElement.innerText = ''; + Array.from(dom.children).forEach((child) => { + targetElement.appendChild(child); + }); + } + + targetElement.tree = tree; + moveCursor(isFocused, alwaysMoveCursorToTheEnd, cursorPosition, targetElement); + } + + return {text, cursorPosition: cursorPosition || 0}; +} + +export {updateInputStructure, parseRangesToHTMLNodes}; diff --git a/src/web/utils/treeUtils.ts b/src/web/utils/treeUtils.ts new file mode 100644 index 00000000..f9354487 --- /dev/null +++ b/src/web/utils/treeUtils.ts @@ -0,0 +1,103 @@ +import type {HTMLMarkdownElement} from '../../MarkdownTextInput.web'; +import type {MarkdownRange, MarkdownType} from '../../commonTypes'; + +type NodeType = MarkdownType | 'line' | 'text' | 'br' | 'root'; + +type TreeNode = Omit & { + element: HTMLMarkdownElement; + parentNode: TreeNode | null; + childNodes: TreeNode[]; + type: NodeType; + orderIndex: string; + isGeneratingNewline: boolean; +}; + +function addNodeToTree(element: HTMLMarkdownElement, parentTreeNode: TreeNode, type: NodeType, length: number | null = null) { + const contentLength = length || (element.nodeName === 'BR' || type === 'br' ? 1 : element.value?.length) || 0; + const isGeneratingNewline = type === 'line' && !(element.childNodes.length === 1 && (element.childNodes[0] as HTMLElement)?.getAttribute?.('data-type') === 'br'); + const parentChildrenCount = parentTreeNode?.childNodes.length || 0; + let startIndex = parentTreeNode.start; + if (parentChildrenCount > 0) { + const lastParentChild = parentTreeNode.childNodes[parentChildrenCount - 1]; + if (lastParentChild) { + startIndex = lastParentChild.start + lastParentChild.length; + startIndex += lastParentChild.isGeneratingNewline || element.style.display === 'block' ? 1 : 0; + } + } + + const item: TreeNode = { + element, + parentNode: parentTreeNode, + childNodes: [], + start: startIndex, + length: contentLength, + type, + orderIndex: parentTreeNode.parentNode === null ? `${parentChildrenCount}` : `${parentTreeNode.orderIndex},${parentChildrenCount}`, + isGeneratingNewline, + }; + + element.setAttribute('data-id', item.orderIndex); + parentTreeNode.childNodes.push(item); + return item; +} + +function findHTMLElementInTree(treeRoot: TreeNode, element: HTMLElement): TreeNode | null { + if (element.hasAttribute?.('contenteditable')) { + return treeRoot; + } + + if (!element || !element.hasAttribute?.('data-id')) { + return null; + } + const indexes = element.getAttribute('data-id')?.split(','); + let el: TreeNode | null = treeRoot; + + while (el && indexes && indexes.length > 0) { + const index = Number(indexes.shift() || -1); + if (index < 0) { + break; + } + + if (el) { + el = el.childNodes[index] || null; + } + } + + return el; +} + +function getTreeNodeByIndex(treeRoot: TreeNode, index: number): TreeNode | null { + let el: TreeNode | null = treeRoot; + + let i = 0; + let newLineGenerated = false; + while (el && el.childNodes.length > 0 && i < el.childNodes.length) { + const child = el.childNodes[i] as TreeNode; + + if (!child) { + break; + } + + if (index >= child.start && index < child.start + child.length) { + if (child.childNodes.length === 0) { + return child; + } + el = child; + i = 0; + } else if ((child.isGeneratingNewline || newLineGenerated || i === el.childNodes.length - 1) && index === child.start + child.length) { + newLineGenerated = true; + if (child.childNodes.length === 0) { + return child; + } + el = child; + i = 0; + } else { + i++; + } + } + return null; +} + +export {addNodeToTree, findHTMLElementInTree, getTreeNodeByIndex}; + +export type {TreeNode, NodeType}; diff --git a/src/web/utils/webStyleUtils.ts b/src/web/utils/webStyleUtils.ts new file mode 100644 index 00000000..e0f84510 --- /dev/null +++ b/src/web/utils/webStyleUtils.ts @@ -0,0 +1,54 @@ +/* eslint-disable @typescript-eslint/no-explicit-any */ +import type {TextStyle} from 'react-native'; +import type {MarkdownStyle} from '../../MarkdownTextInputDecoratorViewNativeComponent'; +import {mergeMarkdownStyleWithDefault} from '../../styleUtils'; + +let createReactDOMStyle: (style: any) => any; +try { + createReactDOMStyle = + // eslint-disable-next-line @typescript-eslint/no-var-requires + require('react-native-web/dist/exports/StyleSheet/compiler/createReactDOMStyle').default; +} catch (e) { + throw new Error('[react-native-live-markdown] Function `createReactDOMStyle` from react-native-web not found. Please make sure that you are using React Native Web 0.18 or newer.'); +} + +let preprocessStyle: (style: any) => any; +try { + preprocessStyle = + // eslint-disable-next-line @typescript-eslint/no-var-requires + require('react-native-web/dist/exports/StyleSheet/preprocess').default; +} catch (e) { + throw new Error('[react-native-live-markdown] Function `preprocessStyle` from react-native-web not found.'); +} + +let dangerousStyleValue: (name: string, value: any, isCustomProperty: boolean) => any; +try { + dangerousStyleValue = + // eslint-disable-next-line @typescript-eslint/no-var-requires + require('react-native-web/dist/modules/setValueForStyles/dangerousStyleValue').default; +} catch (e) { + throw new Error('[react-native-live-markdown] Function `dangerousStyleValue` from react-native-web not found.'); +} + +function processUnitsInMarkdownStyle(input: MarkdownStyle): MarkdownStyle { + const output = JSON.parse(JSON.stringify(input)); + + Object.keys(output).forEach((key) => { + const obj = output[key]; + Object.keys(obj).forEach((prop) => { + obj[prop] = dangerousStyleValue(prop, obj[prop], false); + }); + }); + + return output as MarkdownStyle; +} + +function processMarkdownStyle(input: MarkdownStyle | undefined): MarkdownStyle { + return processUnitsInMarkdownStyle(mergeMarkdownStyleWithDefault(input)); +} + +function parseToReactDOMStyle(style: TextStyle): any { + return createReactDOMStyle(preprocessStyle(style)); +} + +export {parseToReactDOMStyle, processMarkdownStyle}; diff --git a/yarn.lock b/yarn.lock index 23dbf1e5..8e381236 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2613,10 +2613,10 @@ __metadata: "@react-native/babel-preset": 0.73.21 "@react-native/metro-config": 0.73.5 babel-plugin-module-resolver: ^5.0.0 - expensify-common: 2.0.72 + expensify-common: 2.0.76 react: 18.2.0 react-native: 0.73.4 - react-native-reanimated: 3.15.0-nightly-20240804-7df5fd57d + react-native-reanimated: 3.15.0 languageName: unknown linkType: soft @@ -2650,20 +2650,21 @@ __metadata: eslint-plugin-prettier: ^4.0.0 eslint-plugin-promise: ^6.1.1 eslint-plugin-tsdoc: ^0.2.17 - expensify-common: 2.0.72 + expensify-common: 2.0.76 jest: ^29.6.3 jest-environment-jsdom: ^29.7.0 + nodemon: ^3.1.3 prettier: 2.8.8 react: 18.2.0 react-native: 0.73.4 react-native-builder-bob: ^0.20.0 - react-native-reanimated: 3.15.0-nightly-20240804-7df5fd57d + react-native-reanimated: 3.15.0 react-native-web: ^0.19.10 release-it: ^15.0.0 turbo: ^1.10.7 typescript: ^5.3.3 peerDependencies: - expensify-common: ">=2.0.72" + expensify-common: ">=2.0.76" react: "*" react-native: "*" react-native-reanimated: ">=3.11.0" @@ -6900,7 +6901,7 @@ __metadata: languageName: node linkType: hard -"chokidar@npm:^3.5.3": +"chokidar@npm:^3.5.2, chokidar@npm:^3.5.3": version: 3.6.0 resolution: "chokidar@npm:3.6.0" dependencies: @@ -8110,6 +8111,18 @@ __metadata: languageName: node linkType: hard +"debug@npm:^4": + version: 4.3.5 + resolution: "debug@npm:4.3.5" + dependencies: + ms: 2.1.2 + peerDependenciesMeta: + supports-color: + optional: true + checksum: 7c002b51e256257f936dda09eb37167df952758c57badf6bf44bdc40b89a4bcb8e5a0a2e4c7b53f97c69e2970dd5272d33a757378a12c8f8e64ea7bf99e8e86e + languageName: node + linkType: hard + "decamelize-keys@npm:^1.1.0": version: 1.1.1 resolution: "decamelize-keys@npm:1.1.1" @@ -9864,9 +9877,9 @@ __metadata: languageName: node linkType: hard -"expensify-common@npm:2.0.72": - version: 2.0.72 - resolution: "expensify-common@npm:2.0.72" +"expensify-common@npm:2.0.76": + version: 2.0.76 + resolution: "expensify-common@npm:2.0.76" dependencies: awesome-phonenumber: ^5.4.0 classnames: 2.5.0 @@ -9881,7 +9894,7 @@ __metadata: semver: ^7.6.3 simply-deferred: "git+https://github.com/Expensify/simply-deferred.git#77a08a95754660c7bd6e0b6979fdf84e8e831bf5" ua-parser-js: ^1.0.38 - checksum: b1331aad5a512ffeb67fcb657025f2a2bfe86d3d905cbc0d09b8dff856562ecb2be5c0bbd37bd218f6bc8309f0417e52b918ea4132dfddd555dd3bfa1c89cb89 + checksum: bbf5ebf994abd164816ccd3b8f33432cb8b95545460868b1d5b52f50c502ca83ef71ed913f9ff1da9dd74aa18aa2d5ef954e2045f9213fda7b1008673e0ce384 languageName: node linkType: hard @@ -11648,6 +11661,13 @@ __metadata: languageName: node linkType: hard +"ignore-by-default@npm:^1.0.1": + version: 1.0.1 + resolution: "ignore-by-default@npm:1.0.1" + checksum: 441509147b3615e0365e407a3c18e189f78c07af08564176c680be1fabc94b6c789cad1342ad887175d4ecd5225de86f73d376cec8e06b42fd9b429505ffcf8a + languageName: node + linkType: hard + "ignore@npm:^4.0.6": version: 4.0.6 resolution: "ignore@npm:4.0.6" @@ -15089,6 +15109,26 @@ __metadata: languageName: node linkType: hard +"nodemon@npm:^3.1.3": + version: 3.1.3 + resolution: "nodemon@npm:3.1.3" + dependencies: + chokidar: ^3.5.2 + debug: ^4 + ignore-by-default: ^1.0.1 + minimatch: ^3.1.2 + pstree.remy: ^1.1.8 + semver: ^7.5.3 + simple-update-notifier: ^2.0.0 + supports-color: ^5.5.0 + touch: ^3.1.0 + undefsafe: ^2.0.5 + bin: + nodemon: bin/nodemon.js + checksum: ac2fa8865ab292b7ddf66731487acca4b4282b2728361e0de633c0c74cd705d6a0852b52f785c09469d959241d038ba824f50375622e687a2a318be747d9cd9d + languageName: node + linkType: hard + "nopt@npm:^7.0.0": version: 7.2.0 resolution: "nopt@npm:7.2.0" @@ -16633,6 +16673,13 @@ __metadata: languageName: node linkType: hard +"pstree.remy@npm:^1.1.8": + version: 1.1.8 + resolution: "pstree.remy@npm:1.1.8" + checksum: 5cb53698d6bb34dfb278c8a26957964aecfff3e161af5fbf7cee00bbe9d8547c7aced4bd9cb193bce15fb56e9e4220fc02a5bf9c14345ffb13a36b858701ec2d + languageName: node + linkType: hard + "pump@npm:^3.0.0": version: 3.0.0 resolution: "pump@npm:3.0.0" @@ -16860,9 +16907,9 @@ __metadata: languageName: node linkType: hard -"react-native-reanimated@npm:3.15.0-nightly-20240804-7df5fd57d": - version: 3.15.0-nightly-20240804-7df5fd57d - resolution: "react-native-reanimated@npm:3.15.0-nightly-20240804-7df5fd57d" +"react-native-reanimated@npm:3.15.0": + version: 3.15.0 + resolution: "react-native-reanimated@npm:3.15.0" dependencies: "@babel/plugin-transform-arrow-functions": ^7.0.0-0 "@babel/plugin-transform-class-properties": ^7.0.0-0 @@ -16879,7 +16926,7 @@ __metadata: "@babel/core": ^7.0.0-0 react: "*" react-native: "*" - checksum: 3b42a49644374ac0a2561378d29d19f903c962fed8ebd9fb0c03b6d76956110ac4cffda9efb0c0cbc91d1fad9e10e1096977a115655442de4793a5b0c56f04f8 + checksum: fad4f54d7f005f4fb3d90d983bedcf1ec7de936259b8e286feb35edaa9d1ac16d8b927fcd59eeea7f91446310432b342fd9785301a304955844785fef52a3e31 languageName: node linkType: hard @@ -18233,6 +18280,15 @@ __metadata: languageName: node linkType: hard +"simple-update-notifier@npm:^2.0.0": + version: 2.0.0 + resolution: "simple-update-notifier@npm:2.0.0" + dependencies: + semver: ^7.5.3 + checksum: 9ba00d38ce6a29682f64a46213834e4eb01634c2f52c813a9a7b8873ca49cdbb703696f3290f3b27dc067de6d9418b0b84bef22c3eb074acf352529b2d6c27fd + languageName: node + linkType: hard + "simply-deferred@git+https://github.com/Expensify/simply-deferred.git#77a08a95754660c7bd6e0b6979fdf84e8e831bf5": version: 3.0.0 resolution: "simply-deferred@https://github.com/Expensify/simply-deferred.git#commit=77a08a95754660c7bd6e0b6979fdf84e8e831bf5" @@ -18888,7 +18944,7 @@ __metadata: languageName: node linkType: hard -"supports-color@npm:^5.3.0": +"supports-color@npm:^5.3.0, supports-color@npm:^5.5.0": version: 5.5.0 resolution: "supports-color@npm:5.5.0" dependencies: @@ -19246,6 +19302,15 @@ __metadata: languageName: node linkType: hard +"touch@npm:^3.1.0": + version: 3.1.1 + resolution: "touch@npm:3.1.1" + bin: + nodetouch: bin/nodetouch.js + checksum: fb8c54207500eb760b6b9d77b9c5626cc027c9ad44431eed4268845f00f8c6bbfc95ce7e9da8e487f020aa921982a8bc5d8e909d0606e82686bd0a08a8e0539b + languageName: node + linkType: hard + "tough-cookie@npm:^4.1.2": version: 4.1.3 resolution: "tough-cookie@npm:4.1.3" @@ -19739,6 +19804,13 @@ __metadata: languageName: node linkType: hard +"undefsafe@npm:^2.0.5": + version: 2.0.5 + resolution: "undefsafe@npm:2.0.5" + checksum: f42ab3b5770fedd4ada175fc1b2eb775b78f609156f7c389106aafd231bfc210813ee49f54483d7191d7b76e483bc7f537b5d92d19ded27156baf57592eb02cc + languageName: node + linkType: hard + "underscore@npm:^1.13.1": version: 1.13.6 resolution: "underscore@npm:1.13.6"