diff --git a/demo/index.tsx b/demo/index.tsx index 37af6ea..6dd6a31 100644 --- a/demo/index.tsx +++ b/demo/index.tsx @@ -1,10 +1,10 @@ import * as React from "react"; import ReactDOM from "react-dom"; import { Box, ChakraProvider, HStack, Textarea } from "@chakra-ui/react"; -import { useTextAreaMarkdownEditor } from "../src/hooks/use-markdown-editor"; +import { useTextAreaMarkdownEditor } from "../src"; import { faBold, faItalic, faCode } from "@fortawesome/free-solid-svg-icons"; import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { bold, code, italic } from "../src"; +import { boldCommand, codeCommand, italicCommand } from "../src"; import { ToolbarButton } from "./toolbar-button"; export type DemoProps = {}; @@ -12,9 +12,9 @@ export type DemoProps = {}; export const Demo: React.FunctionComponent = () => { const { ref, commandController } = useTextAreaMarkdownEditor({ commandMap: { - bold: bold, - italic: italic, - code: code + bold: boldCommand, + italic: italicCommand, + code: codeCommand } }); diff --git a/package.json b/package.json index 4fd00ab..f254939 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "react-mde", - "version": "12.0.6", + "version": "12.0.7", "description": "React Markdown Editor", "main": "dist/cjs/index.js", "module": "dist/esm/index.js", diff --git a/src/commands/markdown-commands/bold.tsx b/src/commands/markdown-commands/boldCommand.tsx similarity index 61% rename from src/commands/markdown-commands/bold.tsx rename to src/commands/markdown-commands/boldCommand.tsx index c4dac04..8a51f51 100644 --- a/src/commands/markdown-commands/bold.tsx +++ b/src/commands/markdown-commands/boldCommand.tsx @@ -1,34 +1,36 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; -export const bold: Command = { +import { + getCharactersAfterSelection, + getCharactersBeforeSelection, + getSelectedText, + selectWord +} from "../../helpers/textHelpers"; + +export const boldCommand: Command = { shouldUndo: options => { return ( - textHelpers.getCharactersBeforeSelection(options.initialState, 2) === - "**" && - textHelpers.getCharactersAfterSelection(options.initialState, 2) === "**" + getCharactersBeforeSelection(options.initialState, 2) === "**" && + getCharactersAfterSelection(options.initialState, 2) === "**" ); }, execute: ({ initialState, textApi }) => { // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ + const newSelectionRange = selectWord({ text: initialState.text, selection: initialState.selection }); const state1 = textApi.setSelectionRange(newSelectionRange); // Replaces the current selection with the bold mark up - const state2 = textApi.replaceSelection( - `**${textHelpers.getSelectedText(state1)}**` - ); + const state2 = textApi.replaceSelection(`**${getSelectedText(state1)}**`); // Adjust the selection to not contain the ** textApi.setSelectionRange({ - start: - state2.selection.end - 2 - textHelpers.getSelectedText(state1).length, + start: state2.selection.end - 2 - getSelectedText(state1).length, end: state2.selection.end - 2 }); }, undo: ({ initialState, textApi }) => { - const text = textHelpers.getSelectedText(initialState); + const text = getSelectedText(initialState); textApi.setSelectionRange({ start: initialState.selection.start - 2, end: initialState.selection.end + 2 diff --git a/src/commands/markdown-commands/checkedListCommand.ts b/src/commands/markdown-commands/checkedListCommand.ts new file mode 100644 index 0000000..6e69fbd --- /dev/null +++ b/src/commands/markdown-commands/checkedListCommand.ts @@ -0,0 +1,8 @@ +import { Command } from "../command"; +import { makeList } from "../../helpers/listHelpers"; + +export const checkedListCommand: Command = { + execute: ({ initialState, textApi }) => { + makeList(initialState, textApi, () => `- [ ] `); + } +}; diff --git a/src/commands/markdown-commands/codeBlock.tsx b/src/commands/markdown-commands/codeBlockCommand.tsx similarity index 59% rename from src/commands/markdown-commands/codeBlock.tsx rename to src/commands/markdown-commands/codeBlockCommand.tsx index e79f82d..eda238c 100644 --- a/src/commands/markdown-commands/codeBlock.tsx +++ b/src/commands/markdown-commands/codeBlockCommand.tsx @@ -1,24 +1,28 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; - -export const codeBlock: Command = { +import { + getBreaksNeededForEmptyLineAfter, + getBreaksNeededForEmptyLineBefore, + getSelectedText, + selectWord +} from "../../helpers/textHelpers"; + +export const codeBlockCommand: Command = { execute: async ({ textApi, initialState }) => { // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ + const newSelectionRange = selectWord({ text: initialState.text, selection: initialState.selection }); const state1 = textApi.setSelectionRange(newSelectionRange); // when there's no breaking line - if (textHelpers.getSelectedText(state1).indexOf("\n") === -1) { - textApi.replaceSelection(`\`${textHelpers.getSelectedText(state1)}\``); + if (getSelectedText(state1).indexOf("\n") === -1) { + textApi.replaceSelection(`\`${getSelectedText(state1)}\``); // Adjust the selection to not contain the ** const selectionStart = state1.selection.start + 1; - const selectionEnd = - selectionStart + textHelpers.getSelectedText(state1).length; + const selectionEnd = selectionStart + getSelectedText(state1).length; textApi.setSelectionRange({ start: selectionStart, @@ -27,27 +31,24 @@ export const codeBlock: Command = { return; } - const breaksBeforeCount = textHelpers.getBreaksNeededForEmptyLineBefore( + const breaksBeforeCount = getBreaksNeededForEmptyLineBefore( state1.text, state1.selection.start ); const breaksBefore = Array(breaksBeforeCount + 1).join("\n"); - const breaksAfterCount = textHelpers.getBreaksNeededForEmptyLineAfter( + const breaksAfterCount = getBreaksNeededForEmptyLineAfter( state1.text, state1.selection.end ); const breaksAfter = Array(breaksAfterCount + 1).join("\n"); textApi.replaceSelection( - `${breaksBefore}\`\`\`\n${textHelpers.getSelectedText( - state1 - )}\n\`\`\`${breaksAfter}` + `${breaksBefore}\`\`\`\n${getSelectedText(state1)}\n\`\`\`${breaksAfter}` ); const selectionStart = state1.selection.start + breaksBeforeCount + 4; - const selectionEnd = - selectionStart + textHelpers.getSelectedText(state1).length; + const selectionEnd = selectionStart + getSelectedText(state1).length; textApi.setSelectionRange({ start: selectionStart, diff --git a/src/commands/markdown-commands/code.tsx b/src/commands/markdown-commands/codeCommand.tsx similarity index 64% rename from src/commands/markdown-commands/code.tsx rename to src/commands/markdown-commands/codeCommand.tsx index d8e760c..9511ebc 100644 --- a/src/commands/markdown-commands/code.tsx +++ b/src/commands/markdown-commands/codeCommand.tsx @@ -1,35 +1,36 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; +import { + getCharactersAfterSelection, + getCharactersBeforeSelection, + getSelectedText, + selectWord +} from "../../helpers/textHelpers"; -export const code: Command = { +export const codeCommand: Command = { shouldUndo: options => { return ( - textHelpers.getCharactersBeforeSelection(options.initialState, 1) === - "`" && - textHelpers.getCharactersAfterSelection(options.initialState, 1) === "`" + getCharactersBeforeSelection(options.initialState, 1) === "`" && + getCharactersAfterSelection(options.initialState, 1) === "`" ); }, execute: ({ initialState, textApi }) => { // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ + const newSelectionRange = selectWord({ text: initialState.text, selection: initialState.selection }); const state1 = textApi.setSelectionRange(newSelectionRange); // Replaces the current selection with the italic mark up - const state2 = textApi.replaceSelection( - `\`${textHelpers.getSelectedText(state1)}\`` - ); + const state2 = textApi.replaceSelection(`\`${getSelectedText(state1)}\``); // Adjust the selection to not contain the * textApi.setSelectionRange({ - start: - state2.selection.end - 1 - textHelpers.getSelectedText(state1).length, + start: state2.selection.end - 1 - getSelectedText(state1).length, end: state2.selection.end - 1 }); }, undo: ({ initialState, textApi }) => { - const text = textHelpers.getSelectedText(initialState); + const text = getSelectedText(initialState); textApi.setSelectionRange({ start: initialState.selection.start - 1, end: initialState.selection.end + 1 diff --git a/src/commands/markdown-commands/header.tsx b/src/commands/markdown-commands/headerCommand.tsx similarity index 68% rename from src/commands/markdown-commands/header.tsx rename to src/commands/markdown-commands/headerCommand.tsx index 186dd8f..56fcd8a 100644 --- a/src/commands/markdown-commands/header.tsx +++ b/src/commands/markdown-commands/headerCommand.tsx @@ -1,7 +1,7 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; import { TextController, TextState } from "../../types/CommandOptions"; +import { getSelectedText, selectWord } from "../../helpers/textHelpers"; function setHeader( initialState: TextState, @@ -9,23 +9,21 @@ function setHeader( prefix: string ) { // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ + const newSelectionRange = selectWord({ text: initialState.text, selection: initialState.selection }); const state1 = api.setSelectionRange(newSelectionRange); // Add the prefix to the selection - const state2 = api.replaceSelection( - `${prefix}${textHelpers.getSelectedText(state1)}` - ); + const state2 = api.replaceSelection(`${prefix}${getSelectedText(state1)}`); // Adjust the selection to not contain the prefix api.setSelectionRange({ - start: state2.selection.end - textHelpers.getSelectedText(state1).length, + start: state2.selection.end - getSelectedText(state1).length, end: state2.selection.end }); } -export const header: Command = { +export const headerCommand: Command = { execute: ({ initialState, textApi }) => { setHeader(initialState, textApi, "### "); } diff --git a/src/commands/markdown-commands/image.tsx b/src/commands/markdown-commands/imageCommand.tsx similarity index 75% rename from src/commands/markdown-commands/image.tsx rename to src/commands/markdown-commands/imageCommand.tsx index 8b5c063..9296726 100644 --- a/src/commands/markdown-commands/image.tsx +++ b/src/commands/markdown-commands/imageCommand.tsx @@ -1,20 +1,19 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; +import { getSelectedText, selectWord } from "../../helpers/textHelpers"; -export const image: Command = { +export const imageCommand: Command = { execute: ({ initialState, textApi }) => { // Replaces the current selection with the whole word selected const state1 = textApi.setSelectionRange( - textHelpers.selectWord({ + selectWord({ text: initialState.text, selection: initialState.selection }) ); // Replaces the current selection with the image const imageTemplate = - textHelpers.getSelectedText(state1) || - "https://example.com/your-image.png"; + getSelectedText(state1) || "https://example.com/your-image.png"; textApi.replaceSelection(`![](${imageTemplate})`); // Adjust the selection to not contain the ** textApi.setSelectionRange({ diff --git a/src/commands/markdown-commands/italic.tsx b/src/commands/markdown-commands/italicCommand.tsx similarity index 62% rename from src/commands/markdown-commands/italic.tsx rename to src/commands/markdown-commands/italicCommand.tsx index 65287d0..2d6fc90 100644 --- a/src/commands/markdown-commands/italic.tsx +++ b/src/commands/markdown-commands/italicCommand.tsx @@ -1,35 +1,36 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; +import { + getCharactersAfterSelection, + getCharactersBeforeSelection, + getSelectedText, + selectWord +} from "../../helpers/textHelpers"; -export const italic: Command = { +export const italicCommand: Command = { shouldUndo: options => { return ( - textHelpers.getCharactersBeforeSelection(options.initialState, 1) === - "*" && - textHelpers.getCharactersAfterSelection(options.initialState, 1) === "*" + getCharactersBeforeSelection(options.initialState, 1) === "*" && + getCharactersAfterSelection(options.initialState, 1) === "*" ); }, execute: ({ initialState, textApi }) => { // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ + const newSelectionRange = selectWord({ text: initialState.text, selection: initialState.selection }); const state1 = textApi.setSelectionRange(newSelectionRange); // Replaces the current selection with the italic mark up - const state2 = textApi.replaceSelection( - `*${textHelpers.getSelectedText(state1)}*` - ); + const state2 = textApi.replaceSelection(`*${getSelectedText(state1)}*`); // Adjust the selection to not contain the * textApi.setSelectionRange({ - start: - state2.selection.end - 1 - textHelpers.getSelectedText(state1).length, + start: state2.selection.end - 1 - getSelectedText(state1).length, end: state2.selection.end - 1 }); }, undo: ({ initialState, textApi }) => { - const text = textHelpers.getSelectedText(initialState); + const text = getSelectedText(initialState); textApi.setSelectionRange({ start: initialState.selection.start - 1, end: initialState.selection.end + 1 diff --git a/src/commands/markdown-commands/link.tsx b/src/commands/markdown-commands/linkCommand.tsx similarity index 67% rename from src/commands/markdown-commands/link.tsx rename to src/commands/markdown-commands/linkCommand.tsx index 68459af..24e5b26 100644 --- a/src/commands/markdown-commands/link.tsx +++ b/src/commands/markdown-commands/linkCommand.tsx @@ -1,23 +1,22 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; +import { getSelectedText, selectWord } from "../../helpers/textHelpers"; -export const link: Command = { +export const linkCommand: Command = { execute: ({ initialState, textApi }) => { // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ + const newSelectionRange = selectWord({ text: initialState.text, selection: initialState.selection }); const state1 = textApi.setSelectionRange(newSelectionRange); // Replaces the current selection with the bold mark up const state2 = textApi.replaceSelection( - `[${textHelpers.getSelectedText(state1)}](url)` + `[${getSelectedText(state1)}](url)` ); // Adjust the selection to not contain the ** textApi.setSelectionRange({ - start: - state2.selection.end - 6 - textHelpers.getSelectedText(state1).length, + start: state2.selection.end - 6 - getSelectedText(state1).length, end: state2.selection.end - 6 }); } diff --git a/src/commands/markdown-commands/list.tsx b/src/commands/markdown-commands/list.tsx index aa21299..4b7c4cb 100644 --- a/src/commands/markdown-commands/list.tsx +++ b/src/commands/markdown-commands/list.tsx @@ -1,102 +1 @@ import * as React from "react"; -import { TextController, TextState } from "../../types/CommandOptions"; -import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; - -export type AlterLineFunction = (line: string, index: number) => string; - -/** - * Inserts insertionString before each line - */ -export function insertBeforeEachLine( - selectedText: string, - insertBefore: string | AlterLineFunction -): { modifiedText: string; insertionLength: number } { - const lines = selectedText.split(/\n/); - - let insertionLength = 0; - const modifiedText = lines - .map((item, index) => { - if (typeof insertBefore === "string") { - insertionLength += insertBefore.length; - return insertBefore + item; - } else if (typeof insertBefore === "function") { - const insertionResult = insertBefore(item, index); - insertionLength += insertionResult.length; - return insertBefore(item, index) + item; - } - throw Error("insertion is expected to be either a string or a function"); - }) - .join("\n"); - - return { modifiedText, insertionLength }; -} - -export const makeList = ( - state0: TextState, - textController: TextController, - insertBefore: string | AlterLineFunction -) => { - // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ - text: state0.text, - selection: state0.selection - }); - const state1 = textController.setSelectionRange(newSelectionRange); - - const breaksBeforeCount = textHelpers.getBreaksNeededForEmptyLineBefore( - state1.text, - state1.selection.start - ); - const breaksBefore = Array(breaksBeforeCount + 1).join("\n"); - - const breaksAfterCount = textHelpers.getBreaksNeededForEmptyLineAfter( - state1.text, - state1.selection.end - ); - const breaksAfter = Array(breaksAfterCount + 1).join("\n"); - - const modifiedText = insertBeforeEachLine( - textHelpers.getSelectedText(state1), - insertBefore - ); - - textController.replaceSelection( - `${breaksBefore}${modifiedText.modifiedText}${breaksAfter}` - ); - - // Specifically when the text has only one line, we can exclude the "- ", for example, from the selection - const oneLinerOffset = - textHelpers.getSelectedText(state1).indexOf("\n") === -1 - ? modifiedText.insertionLength - : 0; - - const selectionStart = - state1.selection.start + breaksBeforeCount + oneLinerOffset; - const selectionEnd = - selectionStart + modifiedText.modifiedText.length - oneLinerOffset; - - // Adjust the selection to not contain the ** - textController.setSelectionRange({ - start: selectionStart, - end: selectionEnd - }); -}; - -export const unorderedListCommand: Command = { - execute: ({ initialState, textApi }) => { - makeList(initialState, textApi, "- "); - } -}; - -export const orderedListCommand: Command = { - execute: ({ initialState, textApi }) => { - makeList(initialState, textApi, (item, index) => `${index + 1}. `); - } -}; - -export const checkedListCommand: Command = { - execute: ({ initialState, textApi }) => { - makeList(initialState, textApi, (item, index) => `- [ ] `); - } -}; diff --git a/src/commands/markdown-commands/orderedListCommand.ts b/src/commands/markdown-commands/orderedListCommand.ts new file mode 100644 index 0000000..ea79dfd --- /dev/null +++ b/src/commands/markdown-commands/orderedListCommand.ts @@ -0,0 +1,8 @@ +import { Command } from "../command"; +import { makeList } from "../../helpers/listHelpers"; + +export const orderedListCommand: Command = { + execute: ({ initialState, textApi }) => { + makeList(initialState, textApi, (item, index) => `${index + 1}. `); + } +}; diff --git a/src/commands/markdown-commands/quote.tsx b/src/commands/markdown-commands/quoteCommand.tsx similarity index 63% rename from src/commands/markdown-commands/quote.tsx rename to src/commands/markdown-commands/quoteCommand.tsx index 5eca543..b4c26eb 100644 --- a/src/commands/markdown-commands/quote.tsx +++ b/src/commands/markdown-commands/quoteCommand.tsx @@ -1,23 +1,28 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; +import { + getBreaksNeededForEmptyLineAfter, + getBreaksNeededForEmptyLineBefore, + getSelectedText, + selectWord +} from "../../helpers/textHelpers"; -export const quote: Command = { +export const quoteCommand: Command = { execute: ({ initialState, textApi }) => { // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ + const newSelectionRange = selectWord({ text: initialState.text, selection: initialState.selection }); const state1 = textApi.setSelectionRange(newSelectionRange); - const breaksBeforeCount = textHelpers.getBreaksNeededForEmptyLineBefore( + const breaksBeforeCount = getBreaksNeededForEmptyLineBefore( state1.text, state1.selection.start ); const breaksBefore = Array(breaksBeforeCount + 1).join("\n"); - const breaksAfterCount = textHelpers.getBreaksNeededForEmptyLineAfter( + const breaksAfterCount = getBreaksNeededForEmptyLineAfter( state1.text, state1.selection.end ); @@ -25,12 +30,11 @@ export const quote: Command = { // Replaces the current selection with the quote mark up textApi.replaceSelection( - `${breaksBefore}> ${textHelpers.getSelectedText(state1)}${breaksAfter}` + `${breaksBefore}> ${getSelectedText(state1)}${breaksAfter}` ); const selectionStart = state1.selection.start + breaksBeforeCount + 2; - const selectionEnd = - selectionStart + textHelpers.getSelectedText(state1).length; + const selectionEnd = selectionStart + getSelectedText(state1).length; textApi.setSelectionRange({ start: selectionStart, diff --git a/src/commands/markdown-commands/strikethrough.tsx b/src/commands/markdown-commands/strikethroughCommand.tsx similarity index 61% rename from src/commands/markdown-commands/strikethrough.tsx rename to src/commands/markdown-commands/strikethroughCommand.tsx index 764efed..7a507cd 100644 --- a/src/commands/markdown-commands/strikethrough.tsx +++ b/src/commands/markdown-commands/strikethroughCommand.tsx @@ -1,23 +1,20 @@ import * as React from "react"; import { Command } from "../command"; -import { textHelpers } from "../../helpers/textHelpers"; +import { getSelectedText, selectWord } from "../../helpers/textHelpers"; -export const strikethrough: Command = { +export const strikethroughCommand: Command = { execute: ({ initialState, textApi }) => { // Adjust the selection to encompass the whole word if the caret is inside one - const newSelectionRange = textHelpers.selectWord({ + const newSelectionRange = selectWord({ text: initialState.text, selection: initialState.selection }); const state1 = textApi.setSelectionRange(newSelectionRange); // Replaces the current selection with the strikethrough mark up - const state2 = textApi.replaceSelection( - `~~${textHelpers.getSelectedText(state1)}~~` - ); + const state2 = textApi.replaceSelection(`~~${getSelectedText(state1)}~~`); // Adjust the selection to not contain the ~~ textApi.setSelectionRange({ - start: - state2.selection.end - 2 - textHelpers.getSelectedText(state1).length, + start: state2.selection.end - 2 - getSelectedText(state1).length, end: state2.selection.end - 2 }); } diff --git a/src/commands/markdown-commands/unorderedListCommand.ts b/src/commands/markdown-commands/unorderedListCommand.ts new file mode 100644 index 0000000..69481f5 --- /dev/null +++ b/src/commands/markdown-commands/unorderedListCommand.ts @@ -0,0 +1,8 @@ +import { Command } from "../command"; +import { makeList } from "../../helpers/listHelpers"; + +export const unorderedListCommand: Command = { + execute: ({ initialState, textApi }) => { + makeList(initialState, textApi, "- "); + } +}; diff --git a/src/helpers/listHelpers.ts b/src/helpers/listHelpers.ts new file mode 100644 index 0000000..9ab0d3d --- /dev/null +++ b/src/helpers/listHelpers.ts @@ -0,0 +1,62 @@ +import { TextController, TextState } from "../types/CommandOptions"; + +export type AlterLineFunction = (line: string, index: number) => string; + +import { + getBreaksNeededForEmptyLineAfter, + getBreaksNeededForEmptyLineBefore, + getSelectedText, + insertBeforeEachLine, + selectWord +} from "./textHelpers"; + +export function makeList( + state0: TextState, + textController: TextController, + insertBefore: string | AlterLineFunction +) { + // Adjust the selection to encompass the whole word if the caret is inside one + const newSelectionRange = selectWord({ + text: state0.text, + selection: state0.selection + }); + const state1 = textController.setSelectionRange(newSelectionRange); + + const breaksBeforeCount = getBreaksNeededForEmptyLineBefore( + state1.text, + state1.selection.start + ); + const breaksBefore = Array(breaksBeforeCount + 1).join("\n"); + + const breaksAfterCount = getBreaksNeededForEmptyLineAfter( + state1.text, + state1.selection.end + ); + const breaksAfter = Array(breaksAfterCount + 1).join("\n"); + + const modifiedText = insertBeforeEachLine( + getSelectedText(state1), + insertBefore + ); + + textController.replaceSelection( + `${breaksBefore}${modifiedText.modifiedText}${breaksAfter}` + ); + + // Specifically when the text has only one line, we can exclude the "- ", for example, from the selection + const oneLinerOffset = + getSelectedText(state1).indexOf("\n") === -1 + ? modifiedText.insertionLength + : 0; + + const selectionStart = + state1.selection.start + breaksBeforeCount + oneLinerOffset; + const selectionEnd = + selectionStart + modifiedText.modifiedText.length - oneLinerOffset; + + // Adjust the selection to not contain the ** + textController.setSelectionRange({ + start: selectionStart, + end: selectionEnd + }); +} diff --git a/src/helpers/textHelpers.ts b/src/helpers/textHelpers.ts index aa0c87a..a3b4555 100644 --- a/src/helpers/textHelpers.ts +++ b/src/helpers/textHelpers.ts @@ -1,137 +1,170 @@ import { SelectionRange } from "../types/SelectionRange"; import { TextState } from "../types/CommandOptions"; +import { AlterLineFunction } from "./listHelpers"; /** * A list of helpers for manipulating markdown text. * These helpers do not interface with a textarea. For that, see */ -export const textHelpers = { - getSurroundingWord: function(text: string, position: number): SelectionRange { - if (!text) throw Error("Argument 'text' should be truthy"); +export function getSurroundingWord( + text: string, + position: number +): SelectionRange { + if (!text) throw Error("Argument 'text' should be truthy"); - const isWordDelimiter = (c: string) => c === " " || c.charCodeAt(0) === 10; + const isWordDelimiter = (c: string) => c === " " || c.charCodeAt(0) === 10; - // leftIndex is initialized to 0 because if selection is 0, it won't even enter the iteration - let start = 0; - // rightIndex is initialized to text.length because if selection is equal to text.length it won't even enter the interation - let end = text.length; + // leftIndex is initialized to 0 because if selection is 0, it won't even enter the iteration + let start = 0; + // rightIndex is initialized to text.length because if selection is equal to text.length it won't even enter the interation + let end = text.length; - // iterate to the left - for (let i = position; i - 1 > -1; i--) { - if (isWordDelimiter(text[i - 1])) { - start = i; - break; - } + // iterate to the left + for (let i = position; i - 1 > -1; i--) { + if (isWordDelimiter(text[i - 1])) { + start = i; + break; } + } - // iterate to the right - for (let i = position; i < text.length; i++) { - if (isWordDelimiter(text[i])) { - end = i; - break; - } + // iterate to the right + for (let i = position; i < text.length; i++) { + if (isWordDelimiter(text[i])) { + end = i; + break; } + } - return { start, end }; - }, - - /** - * If the cursor is inside a word and (selection.start === selection.end) - * returns a new Selection where the whole word is selected - * @param text - * @param selection - */ - selectWord: function({ text, selection }: TextState): SelectionRange { - if (text && text.length && selection.start === selection.end) { - // the user is pointing to a word - return this.getSurroundingWord(text, selection.start); - } - return selection; - }, - - /** - * Gets the number of line-breaks that would have to be inserted before the given 'startPosition' - * to make sure there's an empty line between 'startPosition' and the previous text - */ - getBreaksNeededForEmptyLineBefore: function( - text = "", - startPosition: number - ): number { - if (startPosition === 0) return 0; - - // rules: - // - If we're in the first line, no breaks are needed - // - Otherwise there must be 2 breaks before the previous character. Depending on how many breaks exist already, we - // may need to insert 0, 1 or 2 breaks - - let neededBreaks = 2; - let isInFirstLine = true; - for (let i = startPosition - 1; i >= 0 && neededBreaks >= 0; i--) { - switch (text.charCodeAt(i)) { - case 32: // blank space - continue; - case 10: // line break - neededBreaks--; - isInFirstLine = false; - break; - default: - return neededBreaks; - } + return { start, end }; +} + +/** + * If the cursor is inside a word and (selection.start === selection.end) + * returns a new Selection where the whole word is selected + * @param text + * @param selection + */ +export function selectWord({ text, selection }: TextState): SelectionRange { + if (text && text.length && selection.start === selection.end) { + // the user is pointing to a word + return getSurroundingWord(text, selection.start); + } + return selection; +} + +/** + * Gets the number of line-breaks that would have to be inserted before the given 'startPosition' + * to make sure there's an empty line between 'startPosition' and the previous text + */ +export function getBreaksNeededForEmptyLineBefore( + text = "", + startPosition: number +): number { + if (startPosition === 0) return 0; + + // rules: + // - If we're in the first line, no breaks are needed + // - Otherwise there must be 2 breaks before the previous character. Depending on how many breaks exist already, we + // may need to insert 0, 1 or 2 breaks + + let neededBreaks = 2; + let isInFirstLine = true; + for (let i = startPosition - 1; i >= 0 && neededBreaks >= 0; i--) { + switch (text.charCodeAt(i)) { + case 32: // blank space + continue; + case 10: // line break + neededBreaks--; + isInFirstLine = false; + break; + default: + return neededBreaks; } - return isInFirstLine ? 0 : neededBreaks; - }, - - /** - * Gets the number of line-breaks that would have to be inserted after the given 'startPosition' - * to make sure there's an empty line between 'startPosition' and the next text - */ - getBreaksNeededForEmptyLineAfter(text = "", startPosition: number): number { - if (startPosition === text.length - 1) return 0; - - // rules: - // - If we're in the first line, no breaks are needed - // - Otherwise there must be 2 breaks before the previous character. Depending on how many breaks exist already, we - // may need to insert 0, 1 or 2 breaks - - let neededBreaks = 2; - let isInLastLine = true; - for (let i = startPosition; i < text.length && neededBreaks >= 0; i++) { - switch (text.charCodeAt(i)) { - case 32: - continue; - case 10: { - neededBreaks--; - isInLastLine = false; - break; - } - default: - return neededBreaks; + } + return isInFirstLine ? 0 : neededBreaks; +} + +/** + * Gets the number of line-breaks that would have to be inserted after the given 'startPosition' + * to make sure there's an empty line between 'startPosition' and the next text + */ +export function getBreaksNeededForEmptyLineAfter( + text = "", + startPosition: number +) { + if (startPosition === text.length - 1) return 0; + + // rules: + // - If we're in the first line, no breaks are needed + // - Otherwise there must be 2 breaks before the previous character. Depending on how many breaks exist already, we + // may need to insert 0, 1 or 2 breaks + + let neededBreaks = 2; + let isInLastLine = true; + for (let i = startPosition; i < text.length && neededBreaks >= 0; i++) { + switch (text.charCodeAt(i)) { + case 32: + continue; + case 10: { + neededBreaks--; + isInLastLine = false; + break; } + default: + return neededBreaks; } - return isInLastLine ? 0 : neededBreaks; - }, - getSelectedText(textSection: TextState): string { - return textSection.text.slice( - textSection.selection.start, - textSection.selection.end - ); - }, - getCharactersBeforeSelection( - textState: TextState, - characters: number - ): string { - return textState.text.slice( - textState.selection.start - characters, - textState.selection.start - ); - }, - getCharactersAfterSelection( - textState: TextState, - characters: number - ): string { - return textState.text.slice( - textState.selection.end, - textState.selection.end + characters - ); } -}; + return isInLastLine ? 0 : neededBreaks; +} +export function getSelectedText(textSection: TextState): string { + return textSection.text.slice( + textSection.selection.start, + textSection.selection.end + ); +} +export function getCharactersBeforeSelection( + textState: TextState, + characters: number +): string { + return textState.text.slice( + textState.selection.start - characters, + textState.selection.start + ); +} + +export function getCharactersAfterSelection( + textState: TextState, + characters: number +): string { + return textState.text.slice( + textState.selection.end, + textState.selection.end + characters + ); +} + +/** + * Inserts insertionString before each line + */ +export function insertBeforeEachLine( + selectedText: string, + insertBefore: string | AlterLineFunction +): { modifiedText: string; insertionLength: number } { + const lines = selectedText.split(/\n/); + + let insertionLength = 0; + const modifiedText = lines + .map((item, index) => { + if (typeof insertBefore === "string") { + insertionLength += insertBefore.length; + return insertBefore + item; + } else if (typeof insertBefore === "function") { + const insertionResult = insertBefore(item, index); + insertionLength += insertionResult.length; + return insertBefore(item, index) + item; + } + throw Error("insertion is expected to be either a string or a function"); + }) + .join("\n"); + + return { modifiedText, insertionLength }; +} diff --git a/src/index.ts b/src/index.ts index 02c8d61..44b8018 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,44 +1,44 @@ // Individual commands -import { header } from "./commands/markdown-commands/header"; -import { bold } from "./commands/markdown-commands/bold"; -import { italic } from "./commands/markdown-commands/italic"; -import { strikethrough } from "./commands/markdown-commands/strikethrough"; -import { link } from "./commands/markdown-commands/link"; -import { quote } from "./commands/markdown-commands/quote"; -import { codeBlock } from "./commands/markdown-commands/codeBlock"; -import { - checkedListCommand, - orderedListCommand, - unorderedListCommand -} from "./commands/markdown-commands/list"; -import { image } from "./commands/markdown-commands/image"; +import { headerCommand } from "./commands/markdown-commands/headerCommand"; +import { boldCommand } from "./commands/markdown-commands/boldCommand"; +import { italicCommand } from "./commands/markdown-commands/italicCommand"; +import { strikethroughCommand } from "./commands/markdown-commands/strikethroughCommand"; +import { linkCommand } from "./commands/markdown-commands/linkCommand"; +import { quoteCommand } from "./commands/markdown-commands/quoteCommand"; +import { imageCommand } from "./commands/markdown-commands/imageCommand"; import { CommandController } from "./commands/command-controller"; import type { TextController } from "./types/CommandOptions"; import { TextAreaTextController } from "./text/textarea-text-controller"; -import { textHelpers } from "./helpers/textHelpers"; -import { code } from "./commands/markdown-commands/code"; +import * as textHelpers from "./helpers/textHelpers"; +import * as listHelpers from "./helpers/listHelpers"; +import { codeCommand } from "./commands/markdown-commands/codeCommand"; import { useTextAreaMarkdownEditor } from "./hooks/use-markdown-editor"; +import { codeBlockCommand } from "./commands/markdown-commands/codeBlockCommand"; +import { checkedListCommand } from "./commands/markdown-commands/checkedListCommand"; +import { orderedListCommand } from "./commands/markdown-commands/orderedListCommand"; +import { unorderedListCommand } from "./commands/markdown-commands/unorderedListCommand"; export { // helpers textHelpers, + listHelpers, // controllers CommandController, TextController, TextAreaTextController, // commands - header, - bold, - italic, - strikethrough, - link, - quote, - code, - codeBlock, + headerCommand, + boldCommand, + italicCommand, + strikethroughCommand, + linkCommand, + quoteCommand, + codeCommand, + codeBlockCommand, checkedListCommand, orderedListCommand, unorderedListCommand, - image, + imageCommand, // hooks useTextAreaMarkdownEditor };