From 26bd694c6a4a78376621927562c87767fe5843f8 Mon Sep 17 00:00:00 2001 From: Bruno Windels <brunow@matrix.org> Date: Fri, 6 Sep 2019 16:25:06 +0200 Subject: [PATCH] support toggling inline formatting --- .../views/rooms/BasicMessageComposer.js | 8 +++---- src/editor/operations.js | 23 ++++++++++++++++--- 2 files changed, 24 insertions(+), 7 deletions(-) diff --git a/src/components/views/rooms/BasicMessageComposer.js b/src/components/views/rooms/BasicMessageComposer.js index 2b4693646a..99e7b8ea07 100644 --- a/src/components/views/rooms/BasicMessageComposer.js +++ b/src/components/views/rooms/BasicMessageComposer.js @@ -24,7 +24,7 @@ import {setSelection} from '../../../editor/caret'; import { formatRangeAsQuote, formatRangeAsCode, - formatInline, + toggleInlineFormat, replaceRangeAndMoveCaret, } from '../../../editor/operations'; import {getCaretOffsetAndText, getRangeForSelection} from '../../../editor/dom'; @@ -457,13 +457,13 @@ export default class BasicMessageEditor extends React.Component { this.historyManager.ensureLastChangesPushed(this.props.model); switch (action) { case "bold": - formatInline(range, "**"); + toggleInlineFormat(range, "**"); break; case "italics": - formatInline(range, "*"); + toggleInlineFormat(range, "*"); break; case "strikethrough": - formatInline(range, "<del>", "</del>"); + toggleInlineFormat(range, "<del>", "</del>"); break; case "code": formatRangeAsCode(range); diff --git a/src/editor/operations.js b/src/editor/operations.js index 4645e7d805..e2661faf59 100644 --- a/src/editor/operations.js +++ b/src/editor/operations.js @@ -100,10 +100,27 @@ export function formatRangeAsCode(range) { replaceRangeAndExpandSelection(range, parts); } -export function formatInline(range, prefix, suffix = prefix) { +export function toggleInlineFormat(range, prefix, suffix = prefix) { const {model, parts} = range; const {partCreator} = model; - parts.unshift(partCreator.plain(prefix)); - parts.push(partCreator.plain(suffix)); + + const isFormatted = parts.length && + parts[0].text.startsWith(prefix) && + parts[parts.length - 1].text.endsWith(suffix); + + if (isFormatted) { + // remove prefix and suffix + const partWithoutPrefix = parts[0].serialize(); + partWithoutPrefix.text = partWithoutPrefix.text.substr(prefix.length); + parts[0] = partCreator.deserializePart(partWithoutPrefix); + + const partWithoutSuffix = parts[parts.length - 1].serialize(); + const suffixPartText = partWithoutSuffix.text; + partWithoutSuffix.text = suffixPartText.substring(0, suffixPartText.length - suffix.length); + parts[parts.length - 1] = partCreator.deserializePart(partWithoutSuffix); + } else { + parts.unshift(partCreator.plain(prefix)); + parts.push(partCreator.plain(suffix)); + } replaceRangeAndExpandSelection(range, parts); }