From 607deb31dc728727ed3ae980566488817c07ba98 Mon Sep 17 00:00:00 2001 From: Andrey Antukh Date: Thu, 31 Oct 2024 16:43:20 +0100 Subject: [PATCH 1/3] :recycle: Refactor bundle mechanism Mainly leave shadow-cljs for build cljs stuff and use esbuild for bundle all js dependencies, completly avoiding all possible incompatibility issues between js libraries and google closure compiler. --- .circleci/config.yml | 3 +- .gitignore | 2 + common/src/app/common/types/shape/impl.cljc | 6 +- frontend/package.json | 23 +- frontend/resources/polyfills/dynamicImport.js | 6 +- frontend/resources/templates/index.mustache | 3 +- .../resources/templates/rasterizer.mustache | 3 +- frontend/resources/templates/render.mustache | 1 + frontend/scripts/_helpers.js | 23 +- frontend/scripts/build | 4 +- frontend/shadow-cljs.edn | 92 +- frontend/src/app/main/data/events.cljs | 4 +- frontend/src/app/main/data/shortcuts.cljs | 2 +- frontend/src/app/main/data/shortcuts_impl.js | 48 - .../src/app/main/data/workspace/media.cljs | 2 +- .../src/app/main/data/workspace/texts.cljs | 2 +- frontend/src/app/main/ui/dashboard/grid.cljs | 2 +- .../src/app/main/ui/viewer/thumbnails.cljs | 8 +- .../app/main/ui/workspace/color_palette.cljs | 5 +- .../main/ui/workspace/shapes/text/editor.cljs | 4 +- .../ui/workspace/sidebar/assets/common.cljs | 7 +- .../app/main/ui/workspace/text_palette.cljs | 5 +- .../main/ui/workspace/viewport/actions.cljs | 17 +- .../viewport/grid_layout_editor.cljs | 5 +- .../main/ui/workspace/viewport/widgets.cljs | 2 +- frontend/src/app/util/code_highlight.cljs | 2 +- frontend/src/app/util/dom.cljs | 11 +- frontend/src/app/util/text_editor.cljs | 7 +- frontend/src/app/util/time.cljs | 75 +- frontend/src/app/util/time_impl.js | 71 -- frontend/test/frontend_tests/runner.cljs | 34 + .../draft-js/index.js} | 7 + frontend/vendor/draft-js/package.json | 20 + frontend/vendor/draft-js/yarn.lock | 416 +++++++ frontend/vendor/hljs/index.js | 5 + frontend/vendor/hljs/package.json | 19 + frontend/vendor/hljs/yarn.lock | 273 +++++ frontend/vendor/mousetrap/.gitignore | 3 + frontend/vendor/mousetrap/LICENSE | 193 ++++ frontend/vendor/mousetrap/index.js | 1025 +++++++++++++++++ frontend/vendor/mousetrap/package.json | 15 + frontend/vendor/mousetrap/yarn.lock | 265 +++++ frontend/vendor/text_editor_v2.js | 3 - frontend/vendor/text_editor_v2.js.map | 7 - frontend/yarn.lock | 360 +++++- 45 files changed, 2833 insertions(+), 257 deletions(-) delete mode 100644 frontend/src/app/main/data/shortcuts_impl.js delete mode 100644 frontend/src/app/util/time_impl.js create mode 100644 frontend/test/frontend_tests/runner.cljs rename frontend/{src/app/util/text_editor_impl.js => vendor/draft-js/index.js} (99%) create mode 100644 frontend/vendor/draft-js/package.json create mode 100644 frontend/vendor/draft-js/yarn.lock create mode 100644 frontend/vendor/hljs/index.js create mode 100644 frontend/vendor/hljs/package.json create mode 100644 frontend/vendor/hljs/yarn.lock create mode 100644 frontend/vendor/mousetrap/.gitignore create mode 100644 frontend/vendor/mousetrap/LICENSE create mode 100644 frontend/vendor/mousetrap/index.js create mode 100644 frontend/vendor/mousetrap/package.json create mode 100644 frontend/vendor/mousetrap/yarn.lock delete mode 100644 frontend/vendor/text_editor_v2.js delete mode 100644 frontend/vendor/text_editor_v2.js.map diff --git a/.circleci/config.yml b/.circleci/config.yml index 5747eed39..75955b882 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -86,7 +86,7 @@ jobs: working_directory: "./frontend" command: | yarn install - yarn test + yarn run test - save_cache: paths: @@ -120,6 +120,7 @@ jobs: yarn install yarn run build:app:assets yarn run build:app + yarn run build:app:libs yarn run playwright install --with-deps chromium yarn run test:e2e diff --git a/.gitignore b/.gitignore index b0b2074d8..806990990 100644 --- a/.gitignore +++ b/.gitignore @@ -74,3 +74,5 @@ node_modules /playwright-report/ /blob-report/ /playwright/.cache/ +/frontend/vendor/draft-js/.yarn/ +/frontend/vendor/hljs/.yarn \ No newline at end of file diff --git a/common/src/app/common/types/shape/impl.cljc b/common/src/app/common/types/shape/impl.cljc index 11eade8a8..407ee9b34 100644 --- a/common/src/app/common/types/shape/impl.cljc +++ b/common/src/app/common/types/shape/impl.cljc @@ -158,9 +158,9 @@ (if (= k :selrect) (let [buffer (clone-f32-array (.-buffer coll))] (write-selrect buffer v) - (ShapeWithBuffer. buffer (.-delegate coll))) + (ShapeWithBuffer. buffer (.-delegate ^ShapeWithBuffer coll))) - (let [delegate (.-delegate coll) + (let [delegate (.-delegate ^ShapeWithBuffer coll) delegate' (assoc delegate k v)] (if (identical? delegate' delegate) coll @@ -170,7 +170,7 @@ #?(:cljs (defn- impl-dissoc [coll k] - (let [delegate (.-delegate coll) + (let [delegate (.-delegate ^ShapeWithBuffer coll) delegate' (dissoc delegate k)] (if (identical? delegate delegate') coll diff --git a/frontend/package.json b/frontend/package.json index 6cb2afd11..4d5a2cf3f 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -21,7 +21,7 @@ "build:storybook": "yarn run build:storybook:assets && yarn run build:storybook:cljs && storybook build", "build:storybook:assets": "node ./scripts/build-storybook-assets.js", "build:storybook:cljs": "clojure -M:dev:shadow-cljs compile storybook", - "build:app": "clojure -M:dev:shadow-cljs release main", + "build:app": "clojure -M:dev:shadow-cljs release main worker", "e2e:server": "node ./scripts/e2e-server.js", "fmt:clj": "cljfmt fix --parallel=true src/ test/", "fmt:clj:check": "cljfmt check --parallel=false src/ test/", @@ -30,12 +30,13 @@ "lint:clj": "clj-kondo --parallel --lint src/", "lint:scss": "yarn run prettier -c resources/styles -c src/**/*.scss", "lint:scss:fix": "yarn run prettier -c resources/styles -c src/**/*.scss -w", - "test": "yarn run test:compile && yarn run test:run", - "test:compile": "clojure -M:dev:shadow-cljs compile test --config-merge '{:autorun false}'", - "test:run": "node target/tests.cjs", - "test:watch": "clojure -M:dev:shadow-cljs watch test", + "build:test": "clojure -M:dev:shadow-cljs compile test-esm", + "test": "yarn run build:test && node target/tests/test.js", + "watch:test": "concurrently \"clojure -M:dev:shadow-cljs watch test-esm\" \"nodemon -w target/tests/test.js --exec 'sleep 2 && node target/tests/test.js'\"", "test:e2e": "playwright test --project default", "translations": "node ./scripts/translations.js", + "build:app:libs": "esbuild --bundle --outfile=resources/public/js/libs.js --format=iife target/index.js --minify", + "watch:app:libs": "esbuild --bundle --outfile=resources/public/js/libs.js --format=iife target/index.js --watch", "watch": "yarn run watch:app:assets", "watch:app:assets": "node ./scripts/watch.js", "watch:storybook": "concurrently \"clojure -M:dev:shadow-cljs watch storybook\" \"storybook dev -p 6006 --no-open\" \"yarn run watch:storybook:assets\"", @@ -51,7 +52,7 @@ "@types/node": "^22.7.7", "autoprefixer": "^10.4.20", "concurrently": "^9.0.1", - "draft-js": "git+https://github.com/penpot/draft-js.git#commit=4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0", + "esbuild": "^0.24.0", "express": "^4.21.1", "fancy-log": "^2.0.0", "getopts": "^2.3.0", @@ -91,15 +92,18 @@ "workerpool": "^9.2.0" }, "dependencies": { + "@penpot/draft-js": "file:./vendor/draft-js", + "@penpot/hljs": "file:./vendor/hljs", + "@penpot/mousetrap": "file:./vendor/mousetrap", + "@penpot/svgo": "penpot/svgo#c6fba7a4dcfbc27b643e7fc0c94fc98cf680b77b", + "@penpot/text-editor": "penpot/penpot-text-editor#bae79a69c3d484f4a9271c2ed44022e36fce3cfb", "compression": "^1.7.4", "date-fns": "^4.1.0", "eventsource-parser": "^3.0.0", - "highlight.js": "^11.10.0", "js-beautify": "^1.15.1", "jszip": "^3.10.1", "lodash": "^4.17.21", "luxon": "^3.5.0", - "mousetrap": "^1.6.5", "opentype.js": "^1.3.4", "postcss-modules": "^6.0.0", "randomcolor": "^0.6.2", @@ -110,9 +114,8 @@ "rxjs": "8.0.0-alpha.14", "sax": "^1.4.1", "source-map-support": "^0.5.21", - "svgo": "penpot/svgo#v3", "tdigest": "^0.1.2", - "ua-parser-js": "^1.0.39", + "ua-parser-js": "2.0.0-rc.1", "xregexp": "^5.1.1" } } diff --git a/frontend/resources/polyfills/dynamicImport.js b/frontend/resources/polyfills/dynamicImport.js index 7e354e13c..11bc68e51 100644 --- a/frontend/resources/polyfills/dynamicImport.js +++ b/frontend/resources/polyfills/dynamicImport.js @@ -1,5 +1,7 @@ -if (!('dynamicImport' in window)) { - window.dynamicImport = function(uri) { +if (!('dynamicImport' in globalThis)) { + globalThis.dynamicImport = function(uri) { return import(uri); } }; + +var global = globalThis; diff --git a/frontend/resources/templates/index.mustache b/frontend/resources/templates/index.mustache index e6ab3c4eb..b310d2363 100644 --- a/frontend/resources/templates/index.mustache +++ b/frontend/resources/templates/index.mustache @@ -24,7 +24,7 @@ {{# manifest}} - + {{/manifest}} @@ -46,6 +46,7 @@ {{# manifest}} + {{/manifest}} diff --git a/frontend/resources/templates/rasterizer.mustache b/frontend/resources/templates/rasterizer.mustache index 46372c48d..2bba47a86 100644 --- a/frontend/resources/templates/rasterizer.mustache +++ b/frontend/resources/templates/rasterizer.mustache @@ -11,7 +11,7 @@ {{# manifest}} - + {{/manifest}} @@ -19,6 +19,7 @@ {{# manifest}} + {{/manifest}} diff --git a/frontend/resources/templates/render.mustache b/frontend/resources/templates/render.mustache index cbaad7514..c36e4f9a7 100644 --- a/frontend/resources/templates/render.mustache +++ b/frontend/resources/templates/render.mustache @@ -18,6 +18,7 @@
{{# manifest}} + {{/manifest}} diff --git a/frontend/scripts/_helpers.js b/frontend/scripts/_helpers.js index 390dbe1a1..95dce0ac3 100644 --- a/frontend/scripts/_helpers.js +++ b/frontend/scripts/_helpers.js @@ -180,12 +180,21 @@ export async function watch(baseDir, predicate, callback) { }); } +async function readManifestFile(path) { + const manifestPath = "resources/public/js/manifest.json"; + let content = await fs.readFile(manifestPath, { encoding: "utf8" }); + return JSON.parse(content); +} + async function readShadowManifest() { const ts = Date.now(); try { - const manifestPath = "resources/public/js/manifest.json"; - let content = await fs.readFile(manifestPath, { encoding: "utf8" }); - content = JSON.parse(content); + const content1 = await readManifestFile( + "resources/public/js/manifest.json", + ); + const content2 = await readManifestFile( + "resources/public/js/worker/manifest.json", + ); const index = { ts: ts, @@ -193,10 +202,14 @@ async function readShadowManifest() { polyfills: "js/polyfills.js?ts=" + ts, }; - for (let item of content) { + for (let item of content1) { index[item.name] = "js/" + item["output-name"]; } + for (let item of content2) { + index["worker_" + item.name] = "js/worker/" + item["output-name"]; + } + return index; } catch (cause) { return { @@ -205,7 +218,7 @@ async function readShadowManifest() { polyfills: "js/polyfills.js?ts=" + ts, main: "js/main.js?ts=" + ts, shared: "js/shared.js?ts=" + ts, - worker: "js/worker.js?ts=" + ts, + worker_main: "js/worker/main.js?ts=" + ts, rasterizer: "js/rasterizer.js?ts=" + ts, }; } diff --git a/frontend/scripts/build b/frontend/scripts/build index 97199d20d..c3b3bcf48 100755 --- a/frontend/scripts/build +++ b/frontend/scripts/build @@ -20,8 +20,8 @@ yarn install || exit 1; rm -rf resources/public; rm -rf target/dist; -clojure -M:dev:shadow-cljs release main --config-merge "{:release-version \"${CURRENT_HASH}-${TS}\"}" $EXTRA_PARAMS || exit 1 - +yarn run build:app --config-merge "{:release-version \"${CURRENT_HASH}-${TS}\"}" $EXTRA_PARAMS || exit 1 +yarn run build:app:libs || exit 1; yarn run build:app:assets || exit 1; mkdir -p target/dist; diff --git a/frontend/shadow-cljs.edn b/frontend/shadow-cljs.edn index 959fe99b6..dad9d34f0 100644 --- a/frontend/shadow-cljs.edn +++ b/frontend/shadow-cljs.edn @@ -52,11 +52,6 @@ :depends-on #{:shared} :init-fn app.render/init} - :worker - {:entries [app.worker] - :web-worker true - :depends-on #{:shared}} - :rasterizer {:entries [app.rasterizer] :depends-on #{:shared} @@ -64,9 +59,10 @@ :js-options {:entry-keys ["module" "browser" "main"] - :resolve {"penpot/vendor/text-editor-v2" - {:target :file - :file "vendor/text_editor_v2.js"}}} + :export-conditions ["module" "import", "browser" "require" "default"] + :js-provider :external + :external-index "target/index.js" + :external-index-format :esm} :compiler-options {:output-feature-set :es2020 @@ -86,6 +82,40 @@ :anon-fn-naming-policy :off :source-map-detail-level :all}}} + :worker + {:target :browser + :output-dir "resources/public/js/worker/" + :asset-path "/js/worker" + :devtools {:browser-inject :main + :watch-dir "resources/public" + :reload-strategy :full} + :build-options {:manifest-name "manifest.json"} + :modules + {:main + {:entries [app.worker] + :web-worker true + :depends-on #{}}} + + :js-options + {:entry-keys ["module" "browser" "main"] + :export-conditions ["module" "import", "browser" "require" "default"]} + + :compiler-options + {:output-feature-set :es2020 + :output-wrapper false + :warnings {:fn-deprecated false}} + + :release + {:compiler-options + {:fn-invoke-direct true + :optimizations #shadow/env ["PENPOT_BUILD_OPTIMIZATIONS" :as :keyword :default :advanced] + :output-wrapper true + :rename-prefix-namespace "PENPOT" + :source-map true + :elide-asserts true + :anon-fn-naming-policy :off + :source-map-detail-level :all}}} + ;; FIXME: maybe rename to :components ? (there are nothing storybook ;; related, is just an ESM export of components that will be used ;; initially on storybook but not limited to storybook) @@ -106,26 +136,38 @@ :output-wrapper false :warnings {:fn-deprecated false}}} + :test-esm + {:target :esm + :output-dir "target/tests" + :runtime :custom + + :modules + {:test {:init-fn frontend-tests.runner/init}}} + + ;; :compiler-options + ;; {:output-feature-set :es2020 + ;; :warnings {:fn-deprecated false}}} + :lib-penpot {:target :esm :output-dir "resources/public/libs" - :modules - {:penpot {:exports {:renderPage app.libs.render/render-page-export - :createFile app.libs.file-builder/create-file-export}}} + :modules + {:penpot {:exports {:renderPage app.libs.render/render-page-export + :createFile app.libs.file-builder/create-file-export}}} - :compiler-options - {:output-feature-set :es2020 - :output-wrapper false - :warnings {:fn-deprecated false}} + :compiler-options + {:output-feature-set :es2020 + :output-wrapper false + :warnings {:fn-deprecated false}} - :release - {:compiler-options - {:fn-invoke-direct true - :source-map true - :elide-asserts true - :anon-fn-naming-policy :off - :source-map-detail-level :all}}} + :release + {:compiler-options + {:fn-invoke-direct true + :source-map true + :elide-asserts true + :anon-fn-naming-policy :off + :source-map-detail-level :all}}} :bench {:target :node-script @@ -152,10 +194,7 @@ :autorun true :js-options - {:entry-keys ["module" "browser" "main"] - :resolve {"penpot/vendor/text-editor-v2" - {:target :file - :file "vendor/text_editor_v2.js"}}} + {:entry-keys ["module" "browser" "main"]} :compiler-options {:output-feature-set :es2020 @@ -166,4 +205,3 @@ :warnings {:fn-deprecated false}}} }} - diff --git a/frontend/src/app/main/data/events.cljs b/frontend/src/app/main/data/events.cljs index 06ca2def3..27001909a 100644 --- a/frontend/src/app/main/data/events.cljs +++ b/frontend/src/app/main/data/events.cljs @@ -6,7 +6,7 @@ (ns app.main.data.events (:require - ["ua-parser-js" :as UAParser] + ["ua-parser-js" :as ua] [app.common.data :as d] [app.common.logging :as l] [app.config :as cf] @@ -38,7 +38,7 @@ (defn- collect-context [] - (let [uagent (UAParser.)] + (let [uagent (new ua/UAParser)] (merge {:app-version (:full cf/version) :locale @i18n/locale} diff --git a/frontend/src/app/main/data/shortcuts.cljs b/frontend/src/app/main/data/shortcuts.cljs index 34f6e0496..228ae82c7 100644 --- a/frontend/src/app/main/data/shortcuts.cljs +++ b/frontend/src/app/main/data/shortcuts.cljs @@ -7,7 +7,7 @@ (ns app.main.data.shortcuts (:refer-clojure :exclude [meta reset!]) (:require - ["./shortcuts_impl.js$default" :as mousetrap] + ["@penpot/mousetrap$default" :as mousetrap] [app.common.data.macros :as dm] [app.common.logging :as log] [app.common.schema :as sm] diff --git a/frontend/src/app/main/data/shortcuts_impl.js b/frontend/src/app/main/data/shortcuts_impl.js deleted file mode 100644 index 0dcc27538..000000000 --- a/frontend/src/app/main/data/shortcuts_impl.js +++ /dev/null @@ -1,48 +0,0 @@ -/** - * This Source Code Form is subject to the terms of the Mozilla Public - * License, v. 2.0. If a copy of the MPL was not distributed with this - * file, You can obtain one at http://mozilla.org/MPL/2.0/. - * - * Copyright (c) KALEIDOS INC - */ -"use strict"; - -import Mousetrap from 'mousetrap' - -if (Mousetrap.addKeycodes) { - Mousetrap.addKeycodes({ - 219: '219' - }); -} - -const target = Mousetrap.prototype || Mousetrap; -target.stopCallback = function (e, element, combo) { - // if the element has the data attribute "mousetrap-dont-stop" then no need - // to stop. It should be used like
...
- // or :div {:data-mousetrap-dont-stop true} - if ('mousetrapDontStop' in element.dataset) { - return false - } - - if ((' ' + element.className + ' ').indexOf(' mousetrap ') > -1) { - return false; - } - - if ('composedPath' in e && typeof e.composedPath === 'function') { - // For open shadow trees, update `element` so that the following check works. - const initialEventTarget = e.composedPath()[0]; - if (initialEventTarget !== e.target) { - element = initialEventTarget; - } - } - - // stop for input, select, textarea and button - const shouldStop = element.tagName == "INPUT" || - element.tagName == "SELECT" || - element.tagName == "TEXTAREA" || - (element.tagName == "BUTTON" && combo.includes("tab")) || - (element.contentEditable && element.contentEditable == "true"); - return shouldStop; -} - -export default Mousetrap; diff --git a/frontend/src/app/main/data/workspace/media.cljs b/frontend/src/app/main/data/workspace/media.cljs index 30ad9d735..976dc2cfe 100644 --- a/frontend/src/app/main/data/workspace/media.cljs +++ b/frontend/src/app/main/data/workspace/media.cljs @@ -6,7 +6,7 @@ (ns app.main.data.workspace.media (:require - ["svgo" :as svgo] + ["@penpot/svgo$default" :as svgo] [app.common.data :as d] [app.common.data.macros :as dm] [app.common.exceptions :as ex] diff --git a/frontend/src/app/main/data/workspace/texts.cljs b/frontend/src/app/main/data/workspace/texts.cljs index 898c22e3c..20e24611b 100644 --- a/frontend/src/app/main/data/workspace/texts.cljs +++ b/frontend/src/app/main/data/workspace/texts.cljs @@ -6,7 +6,7 @@ (ns app.main.data.workspace.texts (:require - ["penpot/vendor/text-editor-v2" :as editor.v2] + ["@penpot/text-editor" :as editor.v2] [app.common.attrs :as attrs] [app.common.data :as d] [app.common.data.macros :as dm] diff --git a/frontend/src/app/main/ui/dashboard/grid.cljs b/frontend/src/app/main/ui/dashboard/grid.cljs index 4d45a664a..8883149d0 100644 --- a/frontend/src/app/main/ui/dashboard/grid.cljs +++ b/frontend/src/app/main/ui/dashboard/grid.cljs @@ -280,7 +280,7 @@ (fn [event] (st/emit! (dd/hide-file-menu)) (when can-edit - (let [offset (dom/get-offset-position (.-nativeEvent event)) + (let [offset (dom/get-offset-position (dom/event->native-event event)) select-current? (not (contains? selected-files (:id file))) diff --git a/frontend/src/app/main/ui/viewer/thumbnails.cljs b/frontend/src/app/main/ui/viewer/thumbnails.cljs index 4a20d35fe..3978d8382 100644 --- a/frontend/src/app/main/ui/viewer/thumbnails.cljs +++ b/frontend/src/app/main/ui/viewer/thumbnails.cljs @@ -15,6 +15,7 @@ [app.main.render :as render] [app.main.store :as st] [app.main.ui.icons :as i] + [app.util.dom :as dom] [app.util.i18n :as i18n :refer [tr]] [app.util.object :as obj] [app.util.timers :as ts] @@ -46,9 +47,10 @@ on-scroll (fn [event] - (if (pos? (.. event -nativeEvent -deltaY)) - (on-right-arrow-click event) - (on-left-arrow-click event))) + (let [event (dom/event->native-event event)] + (if (pos? (.-deltaY ^js event)) + (on-right-arrow-click event) + (on-left-arrow-click event)))) on-mount (fn [] diff --git a/frontend/src/app/main/ui/workspace/color_palette.cljs b/frontend/src/app/main/ui/workspace/color_palette.cljs index 5dccc51dc..4ce8947e4 100644 --- a/frontend/src/app/main/ui/workspace/color_palette.cljs +++ b/frontend/src/app/main/ui/workspace/color_palette.cljs @@ -16,6 +16,7 @@ [app.main.ui.hooks :as h] [app.main.ui.icons :as i] [app.util.color :as uc] + [app.util.dom :as dom] [app.util.i18n :refer [tr]] [app.util.keyboard :as kbd] [app.util.object :as obj] @@ -92,7 +93,9 @@ (mf/use-callback (mf/deps max-offset) (fn [event] - (let [delta (+ (.. event -nativeEvent -deltaY) (.. event -nativeEvent -deltaX))] + (let [event (dom/event->native-event event) + delta (+ (.. ^js event -deltaY) + (.. ^js event -deltaX))] (if (pos? delta) (on-right-arrow-click event) (on-left-arrow-click event)))))] diff --git a/frontend/src/app/main/ui/workspace/shapes/text/editor.cljs b/frontend/src/app/main/ui/workspace/shapes/text/editor.cljs index 654e83653..8fe5c8920 100644 --- a/frontend/src/app/main/ui/workspace/shapes/text/editor.cljs +++ b/frontend/src/app/main/ui/workspace/shapes/text/editor.cljs @@ -61,8 +61,8 @@ nil))) (defn- styles-fn [shape styles content] - (let [data (if (= (.getText content) "") - (-> (.getData content) + (let [data (if (= (.getText ^js content) "") + (-> ^js (.getData content) (.toJS) (js->clj :keywordize-keys true)) (txt/styles-to-attrs styles))] diff --git a/frontend/src/app/main/ui/workspace/sidebar/assets/common.cljs b/frontend/src/app/main/ui/workspace/sidebar/assets/common.cljs index 080e64c40..19f1d746f 100644 --- a/frontend/src/app/main/ui/workspace/sidebar/assets/common.cljs +++ b/frontend/src/app/main/ui/workspace/sidebar/assets/common.cljs @@ -134,8 +134,8 @@ (let [children (-> (array/normalize-to-array children) (array/without-nils)) - is-button? #(= :title-button (.. % -props -role)) - is-content? #(= :content (.. % -props -role)) + is-button? #(as-> % $ (= :title-button (.. ^js $ -props -role))) + is-content? #(as-> % $ (= :content (.. ^js $ -props -role))) buttons (array/filter is-button? children) content (array/filter is-content? children) @@ -222,7 +222,8 @@ (defn set-drag-image [event item-ref num-selected] - (let [offset (dom/get-offset-position (.-nativeEvent event)) + (let [offset (dom/get-offset-position + (dom/event->native-event event)) item-el (mf/ref-val item-ref) counter-el (create-counter-element num-selected)] diff --git a/frontend/src/app/main/ui/workspace/text_palette.cljs b/frontend/src/app/main/ui/workspace/text_palette.cljs index 3869a3ec3..d2418c031 100644 --- a/frontend/src/app/main/ui/workspace/text_palette.cljs +++ b/frontend/src/app/main/ui/workspace/text_palette.cljs @@ -14,6 +14,7 @@ [app.main.store :as st] [app.main.ui.context :as ctx] [app.main.ui.icons :as i] + [app.util.dom :as dom] [app.util.i18n :refer [tr]] [app.util.object :as obj] [cuerdas.core :as str] @@ -111,7 +112,9 @@ (mf/use-callback (mf/deps max-offset) (fn [event] - (let [delta (+ (.. event -nativeEvent -deltaY) (.. event -nativeEvent -deltaX))] + (let [event (dom/event->native-event event) + delta (+ (.. ^js event -deltaY) + (.. ^js event -deltaX))] (if (pos? delta) (on-right-arrow-click event) (on-left-arrow-click event)))))] diff --git a/frontend/src/app/main/ui/workspace/viewport/actions.cljs b/frontend/src/app/main/ui/workspace/viewport/actions.cljs index e3d65258f..edc556025 100644 --- a/frontend/src/app/main/ui/workspace/viewport/actions.cljs +++ b/frontend/src/app/main/ui/workspace/viewport/actions.cljs @@ -65,15 +65,15 @@ (dom/stop-propagation bevent) (when-not @z? - (let [event (.-nativeEvent bevent) + (let [event (dom/event->native-event bevent) ctrl? (kbd/ctrl? event) meta? (kbd/meta? event) shift? (kbd/shift? event) alt? (kbd/alt? event) mod? (kbd/mod? event) - left-click? (and (not panning) (= 1 (.-which event))) - middle-click? (and (not panning) (= 2 (.-which event)))] + left-click? (and (not panning) (dom/left-mouse? bevent)) + middle-click? (and (not panning) (dom/middle-mouse? bevent))] (cond (or middle-click? (and left-click? @space?)) @@ -120,12 +120,11 @@ (mf/use-callback (mf/deps @hover @hover-ids selected @space? @z? read-only?) (fn [bevent] - (let [event (.-nativeEvent bevent) + (let [event (dom/event->native-event bevent) shift? (kbd/shift? event) - mod? (kbd/mod? event) - left-click? (= 1 (.-which event))] + mod? (kbd/mod? event)] - (when (and left-click? + (when (and (dom/left-mouse? bevent) (not mod?) (not shift?) (not @space?)) @@ -275,7 +274,7 @@ ;; Release pointer on mouse up (.releasePointerCapture target (.-pointerId event))) - (let [event (.-nativeEvent event) + (let [event (dom/event->native-event event) ctrl? (kbd/ctrl? event) shift? (kbd/shift? event) alt? (kbd/alt? event) @@ -475,7 +474,7 @@ (assoc :y final-y))))) (dnd/has-type? event "penpot/component") - (let [event (.-nativeEvent event) + (let [event (dom/event->native-event event) ctrl? (kbd/ctrl? event) shift? (kbd/shift? event) alt? (kbd/alt? event) diff --git a/frontend/src/app/main/ui/workspace/viewport/grid_layout_editor.cljs b/frontend/src/app/main/ui/workspace/viewport/grid_layout_editor.cljs index 49570f1c9..2f6ae94a6 100644 --- a/frontend/src/app/main/ui/workspace/viewport/grid_layout_editor.cljs +++ b/frontend/src/app/main/ui/workspace/viewport/grid_layout_editor.cljs @@ -959,9 +959,8 @@ handle-pointer-down (mf/use-fn (fn [event] - (let [left-click? (= 1 (.-which (.-nativeEvent event)))] - (when left-click? - (dom/stop-propagation event))))) + (when (dom/left-mouse? event) + (dom/stop-propagation event)))) handle-add-column (mf/use-fn diff --git a/frontend/src/app/main/ui/workspace/viewport/widgets.cljs b/frontend/src/app/main/ui/workspace/viewport/widgets.cljs index 3082435d4..fbf6b37b7 100644 --- a/frontend/src/app/main/ui/workspace/viewport/widgets.cljs +++ b/frontend/src/app/main/ui/workspace/viewport/widgets.cljs @@ -110,7 +110,7 @@ (mf/use-fn (mf/deps frame workspace-read-only?) (fn [bevent] - (let [event (.-nativeEvent bevent) + (let [event (dom/event->native-event bevent) position (dom/get-client-position event)] (dom/prevent-default event) (dom/stop-propagation event) diff --git a/frontend/src/app/util/code_highlight.cljs b/frontend/src/app/util/code_highlight.cljs index 62c761720..c435dc3e7 100644 --- a/frontend/src/app/util/code_highlight.cljs +++ b/frontend/src/app/util/code_highlight.cljs @@ -6,7 +6,7 @@ (ns app.util.code-highlight (:require - ["highlight.js" :as hljs] + ["@penpot/hljs" :as hljs] [app.util.dom :as dom])) (defn highlight! diff --git a/frontend/src/app/util/dom.cljs b/frontend/src/app/util/dom.cljs index 874d9c927..797946bae 100644 --- a/frontend/src/app/util/dom.cljs +++ b/frontend/src/app/util/dom.cljs @@ -756,10 +756,17 @@ (trigger-download-uri filename mtype uri))) -(defn left-mouse? [bevent] - (let [event (.-nativeEvent ^js bevent)] +(defn left-mouse? + [bevent] + (let [event (.-nativeEvent ^js bevent)] (= 1 (.-which event)))) +(defn middle-mouse? + [bevent] + (let [event (.-nativeEvent ^js bevent)] + (= 2 (.-which event)))) + + ;; Warning: need to protect against reverse tabnabbing attack ;; https://www.comparitech.com/blog/information-security/reverse-tabnabbing/ (defn open-new-window diff --git a/frontend/src/app/util/text_editor.cljs b/frontend/src/app/util/text_editor.cljs index a6cedcce8..da4820b67 100644 --- a/frontend/src/app/util/text_editor.cljs +++ b/frontend/src/app/util/text_editor.cljs @@ -7,8 +7,7 @@ (ns app.util.text-editor "Draft related abstraction functions." (:require - ["./text_editor_impl.js" :as impl] - ["draft-js" :as draft] + ["@penpot/draft-js" :as impl] [app.common.text :as txt])) ;; --- CONVERSION @@ -34,12 +33,12 @@ (defn import-content [content] - (-> content txt/convert-to-draft clj->js draft/convertFromRaw)) + (-> content txt/convert-to-draft clj->js impl/convertFromRaw)) (defn export-content [content] (-> content - (draft/convertToRaw) + (impl/convertToRaw) (js->clj :keywordize-keys true) (txt/convert-from-draft))) diff --git a/frontend/src/app/util/time.cljs b/frontend/src/app/util/time.cljs index 85cd03319..760b32704 100644 --- a/frontend/src/app/util/time.cljs +++ b/frontend/src/app/util/time.cljs @@ -6,7 +6,36 @@ (ns app.util.time (:require - ["./time_impl" :as impl] + ["date-fns/format$default" :as dfn-format] + ["date-fns/formatDistanceToNowStrict$default" :as dfn-distance-to-now] + ["date-fns/locale/ar-SA$default" :as dfn-ar] + ["date-fns/locale/ca$default" :as dfn-ca] + ["date-fns/locale/cs$default" :as dfn-cs] + ["date-fns/locale/de$default" :as dfn-de] + ["date-fns/locale/el$default" :as dfn-el] + ["date-fns/locale/en-US$default" :as df-en-us] + ["date-fns/locale/es$default" :as dfn-es] + ["date-fns/locale/eu$default" :as dfn-eu] + ["date-fns/locale/fa-IR$default" :as dfn-fa-ir] + ["date-fns/locale/fr$default" :as dfn-fr] + ["date-fns/locale/gl$default" :as dfn-gl] + ["date-fns/locale/he$default" :as dfn-he] + ["date-fns/locale/hr$default" :as dfn-hr] + ["date-fns/locale/id$default" :as dfn-id] + ["date-fns/locale/it$default" :as dfn-it] + ["date-fns/locale/ja$default" :as dfn-ja] + ["date-fns/locale/ko$default" :as dfn-ko] + ["date-fns/locale/lv$default" :as dfn-lv] + ["date-fns/locale/nb$default" :as dfn-nb] + ["date-fns/locale/nl$default" :as dfn-nl] + ["date-fns/locale/pl$default" :as dfn-pl] + ["date-fns/locale/pt$default" :as dfn-pt] + ["date-fns/locale/pt-BR$default" :as dfn-pt-br] + ["date-fns/locale/ro$default" :as dfn-ro] + ["date-fns/locale/ru$default" :as dfn-ru] + ["date-fns/locale/tr$default" :as dfn-tr] + ["date-fns/locale/uk$default" :as dfn-uk] + ["date-fns/locale/zh-CN$default" :as dfn-zh-cn] [app.common.data.macros :as dm] [app.common.time :as common-time] [app.util.object :as obj] @@ -15,6 +44,42 @@ (dm/export common-time/DateTime) (dm/export common-time/Duration) +(def locales + #js {:ar dfn-ar + :ca dfn-ca + :de dfn-de + :el dfn-el + :en df-en-us + :en_us df-en-us + :es dfn-es + :es_es dfn-es + :fa dfn-fa-ir + :fa_ir dfn-fa-ir + :fr dfn-fr + :he dfn-he + :pt dfn-pt + :pt_pt dfn-pt + :pt_br dfn-pt-br + :ro dfn-ro + :ru dfn-ru + :tr dfn-tr + :zh-cn dfn-zh-cn + :nl dfn-nl + :eu dfn-eu + :gl dfn-gl + :hr dfn-hr + :it dfn-it + :nb dfn-nb + :nb_no dfn-nb + :pl dfn-pl + :id dfn-id + :uk dfn-uk + :cs dfn-cs + :lv dfn-lv + :ko dfn-ko + :ja dfn-ja + :ja_jp dfn-ja}) + (defprotocol ITimeMath (plus [_ o]) (minus [_ o])) @@ -199,18 +264,18 @@ (let [v (if (datetime? v) (format v :date) v)] (->> #js {:includeSeconds true :addSuffix true - :locale (obj/get impl/locales locale)} - (impl/format-distance-to-now v)))))) + :locale (obj/get locales locale)} + (dfn-distance-to-now v)))))) (defn format-date-locale ([v] (format-date-locale v nil)) ([v {:keys [locale] :or {locale "en"}}] (when v (let [v (if (datetime? v) (format v :date) v) - locale (obj/get impl/locales locale) + locale (obj/get locales locale) f (.date (.-formatLong ^js locale) v)] (->> #js {:locale locale} - (impl/format v f)))))) + (dfn-format v f)))))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; Measurement Helpers diff --git a/frontend/src/app/util/time_impl.js b/frontend/src/app/util/time_impl.js deleted file mode 100644 index 6bffde623..000000000 --- a/frontend/src/app/util/time_impl.js +++ /dev/null @@ -1,71 +0,0 @@ -import {format as dfnFormat} from "date-fns/format"; -import {formatDistanceToNowStrict as dfnFormatDistance} from "date-fns/formatDistanceToNowStrict"; - -import {arSA} from "date-fns/locale/ar-SA"; -import {ca} from "date-fns/locale/ca"; -import {de} from "date-fns/locale/de"; -import {el} from "date-fns/locale/el"; -import {enUS} from "date-fns/locale/en-US"; -import {es} from "date-fns/locale/es"; -import {faIR} from "date-fns/locale/fa-IR"; -import {fr} from "date-fns/locale/fr"; -import {he} from "date-fns/locale/he"; -import {pt} from "date-fns/locale/pt"; -import {ptBR} from "date-fns/locale/pt-BR"; -import {ro} from "date-fns/locale/ro"; -import {ru} from "date-fns/locale/ru"; -import {tr} from "date-fns/locale/tr"; -import {zhCN} from "date-fns/locale/zh-CN"; -import {nl} from "date-fns/locale/nl"; -import {eu} from "date-fns/locale/eu"; -import {gl} from "date-fns/locale/gl"; -import {hr} from "date-fns/locale/hr"; -import {it} from "date-fns/locale/it"; -import {nb} from "date-fns/locale/nb"; -import {pl} from "date-fns/locale/pl"; -import {id} from "date-fns/locale/id"; -import {uk} from "date-fns/locale/uk"; -import {cs} from "date-fns/locale/cs"; -import {lv} from "date-fns/locale/lv"; -import {ko} from "date-fns/locale/ko"; -import {ja} from "date-fns/locale/ja"; - -export const locales = { - "ar": arSA, - "ca": ca, - "de": de, - "el": el, - "en": enUS, - "en_us": enUS, - "es": es, - "es_es": es, - "fa": faIR, - "fa_ir": faIR, - "fr": fr, - "he": he, - "pt": pt, - "pt_pt": pt, - "pt_br": ptBR, - "ro": ro, - "ru": ru, - "tr": tr, - "zh_cn": zhCN, - "nl": nl, - "eu": eu, - "gl": gl, - "hr": hr, - "it": it, - "nb": nb, - "nb_no": nb, - "pl": pl, - "id": id, - "uk": uk, - "cs": cs, - "lv": lv, - "ko": ko, - "ja": ja, - "ja_jp": ja, -}; - -export const format = dfnFormat; -export const format_distance_to_now = dfnFormatDistance; diff --git a/frontend/test/frontend_tests/runner.cljs b/frontend/test/frontend_tests/runner.cljs new file mode 100644 index 000000000..59388fe8e --- /dev/null +++ b/frontend/test/frontend_tests/runner.cljs @@ -0,0 +1,34 @@ +(ns frontend-tests.runner + (:require + [cljs.test :as t] + [frontend-tests.basic-shapes-test] + [frontend-tests.helpers-shapes-test] + [frontend-tests.logic.comp-remove-swap-slots-test] + [frontend-tests.logic.copying-and-duplicating-test] + [frontend-tests.logic.frame-guides-test] + [frontend-tests.logic.groups-test] + [frontend-tests.plugins.context-shapes-test] + [frontend-tests.util-range-tree-test] + [frontend-tests.util-simple-math-test] + [frontend-tests.util-snap-data-test])) + +(enable-console-print!) + +(defmethod cljs.test/report [:cljs.test/default :end-run-tests] [m] + (if (cljs.test/successful? m) + (.exit js/process 0) + (.exit js/process 1))) + + +(defn init + [] + (t/run-tests 'frontend-tests.helpers-shapes-test + 'frontend-tests.logic.comp-remove-swap-slots-test + 'frontend-tests.logic.copying-and-duplicating-test + 'frontend-tests.logic.frame-guides-test + 'frontend-tests.logic.groups-test + 'frontend-tests.plugins.context-shapes-test + 'frontend-tests.util-range-tree-test + 'frontend-tests.util-snap-data-test + 'frontend-tests.util-simple-math-test + 'frontend-tests.basic-shapes-test)) diff --git a/frontend/src/app/util/text_editor_impl.js b/frontend/vendor/draft-js/index.js similarity index 99% rename from frontend/src/app/util/text_editor_impl.js rename to frontend/vendor/draft-js/index.js index 16f8b2aab..b47c75077 100644 --- a/frontend/src/app/util/text_editor_impl.js +++ b/frontend/vendor/draft-js/index.js @@ -16,6 +16,8 @@ import { Modifier, RichTextEditorUtil, SelectionState, + convertFromRaw, + convertToRaw } from "draft-js"; import DraftPasteProcessor from 'draft-js/lib/DraftPasteProcessor'; @@ -406,3 +408,8 @@ export function selectionEquals(selection, other) { selection.getFocusOffset() === other.getFocusOffset() && selection.getIsBackward() === other.getIsBackward(); } + +export { + convertToRaw, + convertFromRaw +}; diff --git a/frontend/vendor/draft-js/package.json b/frontend/vendor/draft-js/package.json new file mode 100644 index 000000000..5efe90c80 --- /dev/null +++ b/frontend/vendor/draft-js/package.json @@ -0,0 +1,20 @@ +{ + "name": "@penpot/draft-js-wrapper", + "version": "1.0.0", + "description": "Penpot Draft-JS Wrapper", + "main": "index.js", + "module": " index.js", + "packageManager": "yarn@4.3.1", + "author": "Andrey Antukh", + "license": "MPL-2.0", + "dependencies": { + "draft-js": "penpot/draft-js.git#4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0" + }, + "peerDependencies": { + "react": ">=0.17.0", + "react-dom": ">=0.17.0" + }, + "devDependencies": { + "esbuild": "^0.24.0" + } +} diff --git a/frontend/vendor/draft-js/yarn.lock b/frontend/vendor/draft-js/yarn.lock new file mode 100644 index 000000000..5cf49ca79 --- /dev/null +++ b/frontend/vendor/draft-js/yarn.lock @@ -0,0 +1,416 @@ +# This file is generated by running "yarn install" inside your project. +# Manual changes might be lost - proceed with caution! + +__metadata: + version: 8 + cacheKey: 10c0 + +"@esbuild/aix-ppc64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/aix-ppc64@npm:0.24.0" + conditions: os=aix & cpu=ppc64 + languageName: node + linkType: hard + +"@esbuild/android-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-arm64@npm:0.24.0" + conditions: os=android & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/android-arm@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-arm@npm:0.24.0" + conditions: os=android & cpu=arm + languageName: node + linkType: hard + +"@esbuild/android-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-x64@npm:0.24.0" + conditions: os=android & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/darwin-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/darwin-arm64@npm:0.24.0" + conditions: os=darwin & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/darwin-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/darwin-x64@npm:0.24.0" + conditions: os=darwin & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/freebsd-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/freebsd-arm64@npm:0.24.0" + conditions: os=freebsd & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/freebsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/freebsd-x64@npm:0.24.0" + conditions: os=freebsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/linux-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-arm64@npm:0.24.0" + conditions: os=linux & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/linux-arm@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-arm@npm:0.24.0" + conditions: os=linux & cpu=arm + languageName: node + linkType: hard + +"@esbuild/linux-ia32@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-ia32@npm:0.24.0" + conditions: os=linux & cpu=ia32 + languageName: node + linkType: hard + +"@esbuild/linux-loong64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-loong64@npm:0.24.0" + conditions: os=linux & cpu=loong64 + languageName: node + linkType: hard + +"@esbuild/linux-mips64el@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-mips64el@npm:0.24.0" + conditions: os=linux & cpu=mips64el + languageName: node + linkType: hard + +"@esbuild/linux-ppc64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-ppc64@npm:0.24.0" + conditions: os=linux & cpu=ppc64 + languageName: node + linkType: hard + +"@esbuild/linux-riscv64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-riscv64@npm:0.24.0" + conditions: os=linux & cpu=riscv64 + languageName: node + linkType: hard + +"@esbuild/linux-s390x@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-s390x@npm:0.24.0" + conditions: os=linux & cpu=s390x + languageName: node + linkType: hard + +"@esbuild/linux-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-x64@npm:0.24.0" + conditions: os=linux & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/netbsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/netbsd-x64@npm:0.24.0" + conditions: os=netbsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/openbsd-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/openbsd-arm64@npm:0.24.0" + conditions: os=openbsd & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/openbsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/openbsd-x64@npm:0.24.0" + conditions: os=openbsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/sunos-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/sunos-x64@npm:0.24.0" + conditions: os=sunos & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/win32-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-arm64@npm:0.24.0" + conditions: os=win32 & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/win32-ia32@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-ia32@npm:0.24.0" + conditions: os=win32 & cpu=ia32 + languageName: node + linkType: hard + +"@esbuild/win32-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-x64@npm:0.24.0" + conditions: os=win32 & cpu=x64 + languageName: node + linkType: hard + +"@penpot/draft-js-wrapper@workspace:.": + version: 0.0.0-use.local + resolution: "@penpot/draft-js-wrapper@workspace:." + dependencies: + draft-js: "penpot/draft-js.git#4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0" + esbuild: "npm:^0.24.0" + peerDependencies: + react: ">=0.17.0" + react-dom: ">=0.17.0" + languageName: unknown + linkType: soft + +"asap@npm:~2.0.3": + version: 2.0.6 + resolution: "asap@npm:2.0.6" + checksum: 10c0/c6d5e39fe1f15e4b87677460bd66b66050cd14c772269cee6688824c1410a08ab20254bb6784f9afb75af9144a9f9a7692d49547f4d19d715aeb7c0318f3136d + languageName: node + linkType: hard + +"cross-fetch@npm:^3.1.5": + version: 3.1.8 + resolution: "cross-fetch@npm:3.1.8" + dependencies: + node-fetch: "npm:^2.6.12" + checksum: 10c0/4c5e022ffe6abdf380faa6e2373c0c4ed7ef75e105c95c972b6f627c3f083170b6886f19fb488a7fa93971f4f69dcc890f122b0d97f0bf5f41ca1d9a8f58c8af + languageName: node + linkType: hard + +"draft-js@penpot/draft-js.git#4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0": + version: 0.11.7 + resolution: "draft-js@https://github.com/penpot/draft-js.git#commit=4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0" + dependencies: + fbjs: "npm:^3.0.4" + immutable: "npm:~3.7.4" + object-assign: "npm:^4.1.1" + peerDependencies: + react: ">=0.14.0" + react-dom: ">=0.14.0" + checksum: 10c0/dcd6fd9481b445c0df31a414d5bf0b84ad691d50ac90d805b65c36fb4d26b1ada787f37a63cb437e2a1b6d8dc0f95b4f3c41f6a8082480235ab48b391900a43b + languageName: node + linkType: hard + +"esbuild@npm:^0.24.0": + version: 0.24.0 + resolution: "esbuild@npm:0.24.0" + dependencies: + "@esbuild/aix-ppc64": "npm:0.24.0" + "@esbuild/android-arm": "npm:0.24.0" + "@esbuild/android-arm64": "npm:0.24.0" + "@esbuild/android-x64": "npm:0.24.0" + "@esbuild/darwin-arm64": "npm:0.24.0" + "@esbuild/darwin-x64": "npm:0.24.0" + "@esbuild/freebsd-arm64": "npm:0.24.0" + "@esbuild/freebsd-x64": "npm:0.24.0" + "@esbuild/linux-arm": "npm:0.24.0" + "@esbuild/linux-arm64": "npm:0.24.0" + "@esbuild/linux-ia32": "npm:0.24.0" + "@esbuild/linux-loong64": "npm:0.24.0" + "@esbuild/linux-mips64el": "npm:0.24.0" + "@esbuild/linux-ppc64": "npm:0.24.0" + "@esbuild/linux-riscv64": "npm:0.24.0" + "@esbuild/linux-s390x": "npm:0.24.0" + "@esbuild/linux-x64": "npm:0.24.0" + "@esbuild/netbsd-x64": "npm:0.24.0" + "@esbuild/openbsd-arm64": "npm:0.24.0" + "@esbuild/openbsd-x64": "npm:0.24.0" + "@esbuild/sunos-x64": "npm:0.24.0" + "@esbuild/win32-arm64": "npm:0.24.0" + "@esbuild/win32-ia32": "npm:0.24.0" + "@esbuild/win32-x64": "npm:0.24.0" + dependenciesMeta: + "@esbuild/aix-ppc64": + optional: true + "@esbuild/android-arm": + optional: true + "@esbuild/android-arm64": + optional: true + "@esbuild/android-x64": + optional: true + "@esbuild/darwin-arm64": + optional: true + "@esbuild/darwin-x64": + optional: true + "@esbuild/freebsd-arm64": + optional: true + "@esbuild/freebsd-x64": + optional: true + "@esbuild/linux-arm": + optional: true + "@esbuild/linux-arm64": + optional: true + "@esbuild/linux-ia32": + optional: true + "@esbuild/linux-loong64": + optional: true + "@esbuild/linux-mips64el": + optional: true + "@esbuild/linux-ppc64": + optional: true + "@esbuild/linux-riscv64": + optional: true + "@esbuild/linux-s390x": + optional: true + "@esbuild/linux-x64": + optional: true + "@esbuild/netbsd-x64": + optional: true + "@esbuild/openbsd-arm64": + optional: true + "@esbuild/openbsd-x64": + optional: true + "@esbuild/sunos-x64": + optional: true + "@esbuild/win32-arm64": + optional: true + "@esbuild/win32-ia32": + optional: true + "@esbuild/win32-x64": + optional: true + bin: + esbuild: bin/esbuild + checksum: 10c0/9f1aadd8d64f3bff422ae78387e66e51a5e09de6935a6f987b6e4e189ed00fdc2d1bc03d2e33633b094008529c8b6e06c7ad1a9782fb09fec223bf95998c0683 + languageName: node + linkType: hard + +"fbjs-css-vars@npm:^1.0.0": + version: 1.0.2 + resolution: "fbjs-css-vars@npm:1.0.2" + checksum: 10c0/dfb64116b125a64abecca9e31477b5edb9a2332c5ffe74326fe36e0a72eef7fc8a49b86adf36c2c293078d79f4524f35e80f5e62546395f53fb7c9e69821f54f + languageName: node + linkType: hard + +"fbjs@npm:^3.0.4": + version: 3.0.5 + resolution: "fbjs@npm:3.0.5" + dependencies: + cross-fetch: "npm:^3.1.5" + fbjs-css-vars: "npm:^1.0.0" + loose-envify: "npm:^1.0.0" + object-assign: "npm:^4.1.0" + promise: "npm:^7.1.1" + setimmediate: "npm:^1.0.5" + ua-parser-js: "npm:^1.0.35" + checksum: 10c0/66d0a2fc9a774f9066e35ac2ac4bf1245931d27f3ac287c7d47e6aa1fc152b243c2109743eb8f65341e025621fb51a12038fadb9fd8fda2e3ddae04ebab06f91 + languageName: node + linkType: hard + +"immutable@npm:~3.7.4": + version: 3.7.6 + resolution: "immutable@npm:3.7.6" + checksum: 10c0/efe2bbb2620aa897afbb79545b9eda4dd3dc072e05ae7004895a7efb43187e4265612a88f8723f391eb1c87c46c52fd11e2d1968e42404450c63e49558d7ca4e + languageName: node + linkType: hard + +"js-tokens@npm:^3.0.0 || ^4.0.0": + version: 4.0.0 + resolution: "js-tokens@npm:4.0.0" + checksum: 10c0/e248708d377aa058eacf2037b07ded847790e6de892bbad3dac0abba2e759cb9f121b00099a65195616badcb6eca8d14d975cb3e89eb1cfda644756402c8aeed + languageName: node + linkType: hard + +"loose-envify@npm:^1.0.0": + version: 1.4.0 + resolution: "loose-envify@npm:1.4.0" + dependencies: + js-tokens: "npm:^3.0.0 || ^4.0.0" + bin: + loose-envify: cli.js + checksum: 10c0/655d110220983c1a4b9c0c679a2e8016d4b67f6e9c7b5435ff5979ecdb20d0813f4dec0a08674fcbdd4846a3f07edbb50a36811fd37930b94aaa0d9daceb017e + languageName: node + linkType: hard + +"node-fetch@npm:^2.6.12": + version: 2.7.0 + resolution: "node-fetch@npm:2.7.0" + dependencies: + whatwg-url: "npm:^5.0.0" + peerDependencies: + encoding: ^0.1.0 + peerDependenciesMeta: + encoding: + optional: true + checksum: 10c0/b55786b6028208e6fbe594ccccc213cab67a72899c9234eb59dba51062a299ea853210fcf526998eaa2867b0963ad72338824450905679ff0fa304b8c5093ae8 + languageName: node + linkType: hard + +"object-assign@npm:^4.1.0, object-assign@npm:^4.1.1": + version: 4.1.1 + resolution: "object-assign@npm:4.1.1" + checksum: 10c0/1f4df9945120325d041ccf7b86f31e8bcc14e73d29171e37a7903050e96b81323784ec59f93f102ec635bcf6fa8034ba3ea0a8c7e69fa202b87ae3b6cec5a414 + languageName: node + linkType: hard + +"promise@npm:^7.1.1": + version: 7.3.1 + resolution: "promise@npm:7.3.1" + dependencies: + asap: "npm:~2.0.3" + checksum: 10c0/742e5c0cc646af1f0746963b8776299701ad561ce2c70b49365d62c8db8ea3681b0a1bf0d4e2fe07910bf72f02d39e51e8e73dc8d7503c3501206ac908be107f + languageName: node + linkType: hard + +"setimmediate@npm:^1.0.5": + version: 1.0.5 + resolution: "setimmediate@npm:1.0.5" + checksum: 10c0/5bae81bfdbfbd0ce992893286d49c9693c82b1bcc00dcaaf3a09c8f428fdeacf4190c013598b81875dfac2b08a572422db7df779a99332d0fce186d15a3e4d49 + languageName: node + linkType: hard + +"tr46@npm:~0.0.3": + version: 0.0.3 + resolution: "tr46@npm:0.0.3" + checksum: 10c0/047cb209a6b60c742f05c9d3ace8fa510bff609995c129a37ace03476a9b12db4dbf975e74600830ef0796e18882b2381fb5fb1f6b4f96b832c374de3ab91a11 + languageName: node + linkType: hard + +"ua-parser-js@npm:^1.0.35": + version: 1.0.39 + resolution: "ua-parser-js@npm:1.0.39" + bin: + ua-parser-js: script/cli.js + checksum: 10c0/c6452b0c683000f10975cb0a7e74cb1119ea95d4522ae85f396fa53b0b17884358a24ffdd86a66030c6b2981bdc502109a618c79fdaa217ee9032c9e46fcc78a + languageName: node + linkType: hard + +"webidl-conversions@npm:^3.0.0": + version: 3.0.1 + resolution: "webidl-conversions@npm:3.0.1" + checksum: 10c0/5612d5f3e54760a797052eb4927f0ddc01383550f542ccd33d5238cfd65aeed392a45ad38364970d0a0f4fea32e1f4d231b3d8dac4a3bdd385e5cf802ae097db + languageName: node + linkType: hard + +"whatwg-url@npm:^5.0.0": + version: 5.0.0 + resolution: "whatwg-url@npm:5.0.0" + dependencies: + tr46: "npm:~0.0.3" + webidl-conversions: "npm:^3.0.0" + checksum: 10c0/1588bed84d10b72d5eec1d0faa0722ba1962f1821e7539c535558fb5398d223b0c50d8acab950b8c488b4ba69043fd833cc2697056b167d8ad46fac3995a55d5 + languageName: node + linkType: hard diff --git a/frontend/vendor/hljs/index.js b/frontend/vendor/hljs/index.js new file mode 100644 index 000000000..74c045616 --- /dev/null +++ b/frontend/vendor/hljs/index.js @@ -0,0 +1,5 @@ +import h from "highlight.js"; + +export function highlightElement(node) { + return h.highlightElement(node); +} diff --git a/frontend/vendor/hljs/package.json b/frontend/vendor/hljs/package.json new file mode 100644 index 000000000..9999bbfca --- /dev/null +++ b/frontend/vendor/hljs/package.json @@ -0,0 +1,19 @@ +{ + "name": "@penpot/hljs", + "version": "1.0.0", + "description": "Penpot Hightlight.js ESM wrapper", + "module": "index.js", + "packageManager": "yarn@4.3.1", + "author": "Andrey Antukh", + "license": "MPL-2.0", + "type": "module", + "scripts": { + "build": "esbuild --bundle --platform=browser --format=esm --target=es2021 --outfile=dist/index.mjs index.js --minify" + }, + "dependencies": { + "highlight.js": "^11.10.0" + }, + "devDependencies": { + "esbuild": "^0.24.0" + } +} diff --git a/frontend/vendor/hljs/yarn.lock b/frontend/vendor/hljs/yarn.lock new file mode 100644 index 000000000..354330f59 --- /dev/null +++ b/frontend/vendor/hljs/yarn.lock @@ -0,0 +1,273 @@ +# This file is generated by running "yarn install" inside your project. +# Manual changes might be lost - proceed with caution! + +__metadata: + version: 8 + cacheKey: 10c0 + +"@esbuild/aix-ppc64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/aix-ppc64@npm:0.24.0" + conditions: os=aix & cpu=ppc64 + languageName: node + linkType: hard + +"@esbuild/android-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-arm64@npm:0.24.0" + conditions: os=android & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/android-arm@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-arm@npm:0.24.0" + conditions: os=android & cpu=arm + languageName: node + linkType: hard + +"@esbuild/android-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-x64@npm:0.24.0" + conditions: os=android & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/darwin-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/darwin-arm64@npm:0.24.0" + conditions: os=darwin & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/darwin-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/darwin-x64@npm:0.24.0" + conditions: os=darwin & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/freebsd-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/freebsd-arm64@npm:0.24.0" + conditions: os=freebsd & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/freebsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/freebsd-x64@npm:0.24.0" + conditions: os=freebsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/linux-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-arm64@npm:0.24.0" + conditions: os=linux & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/linux-arm@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-arm@npm:0.24.0" + conditions: os=linux & cpu=arm + languageName: node + linkType: hard + +"@esbuild/linux-ia32@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-ia32@npm:0.24.0" + conditions: os=linux & cpu=ia32 + languageName: node + linkType: hard + +"@esbuild/linux-loong64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-loong64@npm:0.24.0" + conditions: os=linux & cpu=loong64 + languageName: node + linkType: hard + +"@esbuild/linux-mips64el@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-mips64el@npm:0.24.0" + conditions: os=linux & cpu=mips64el + languageName: node + linkType: hard + +"@esbuild/linux-ppc64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-ppc64@npm:0.24.0" + conditions: os=linux & cpu=ppc64 + languageName: node + linkType: hard + +"@esbuild/linux-riscv64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-riscv64@npm:0.24.0" + conditions: os=linux & cpu=riscv64 + languageName: node + linkType: hard + +"@esbuild/linux-s390x@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-s390x@npm:0.24.0" + conditions: os=linux & cpu=s390x + languageName: node + linkType: hard + +"@esbuild/linux-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-x64@npm:0.24.0" + conditions: os=linux & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/netbsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/netbsd-x64@npm:0.24.0" + conditions: os=netbsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/openbsd-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/openbsd-arm64@npm:0.24.0" + conditions: os=openbsd & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/openbsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/openbsd-x64@npm:0.24.0" + conditions: os=openbsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/sunos-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/sunos-x64@npm:0.24.0" + conditions: os=sunos & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/win32-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-arm64@npm:0.24.0" + conditions: os=win32 & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/win32-ia32@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-ia32@npm:0.24.0" + conditions: os=win32 & cpu=ia32 + languageName: node + linkType: hard + +"@esbuild/win32-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-x64@npm:0.24.0" + conditions: os=win32 & cpu=x64 + languageName: node + linkType: hard + +"@penpot/hljs@workspace:.": + version: 0.0.0-use.local + resolution: "@penpot/hljs@workspace:." + dependencies: + esbuild: "npm:^0.24.0" + highlight.js: "npm:^11.10.0" + languageName: unknown + linkType: soft + +"esbuild@npm:^0.24.0": + version: 0.24.0 + resolution: "esbuild@npm:0.24.0" + dependencies: + "@esbuild/aix-ppc64": "npm:0.24.0" + "@esbuild/android-arm": "npm:0.24.0" + "@esbuild/android-arm64": "npm:0.24.0" + "@esbuild/android-x64": "npm:0.24.0" + "@esbuild/darwin-arm64": "npm:0.24.0" + "@esbuild/darwin-x64": "npm:0.24.0" + "@esbuild/freebsd-arm64": "npm:0.24.0" + "@esbuild/freebsd-x64": "npm:0.24.0" + "@esbuild/linux-arm": "npm:0.24.0" + "@esbuild/linux-arm64": "npm:0.24.0" + "@esbuild/linux-ia32": "npm:0.24.0" + "@esbuild/linux-loong64": "npm:0.24.0" + "@esbuild/linux-mips64el": "npm:0.24.0" + "@esbuild/linux-ppc64": "npm:0.24.0" + "@esbuild/linux-riscv64": "npm:0.24.0" + "@esbuild/linux-s390x": "npm:0.24.0" + "@esbuild/linux-x64": "npm:0.24.0" + "@esbuild/netbsd-x64": "npm:0.24.0" + "@esbuild/openbsd-arm64": "npm:0.24.0" + "@esbuild/openbsd-x64": "npm:0.24.0" + "@esbuild/sunos-x64": "npm:0.24.0" + "@esbuild/win32-arm64": "npm:0.24.0" + "@esbuild/win32-ia32": "npm:0.24.0" + "@esbuild/win32-x64": "npm:0.24.0" + dependenciesMeta: + "@esbuild/aix-ppc64": + optional: true + "@esbuild/android-arm": + optional: true + "@esbuild/android-arm64": + optional: true + "@esbuild/android-x64": + optional: true + "@esbuild/darwin-arm64": + optional: true + "@esbuild/darwin-x64": + optional: true + "@esbuild/freebsd-arm64": + optional: true + "@esbuild/freebsd-x64": + optional: true + "@esbuild/linux-arm": + optional: true + "@esbuild/linux-arm64": + optional: true + "@esbuild/linux-ia32": + optional: true + "@esbuild/linux-loong64": + optional: true + "@esbuild/linux-mips64el": + optional: true + "@esbuild/linux-ppc64": + optional: true + "@esbuild/linux-riscv64": + optional: true + "@esbuild/linux-s390x": + optional: true + "@esbuild/linux-x64": + optional: true + "@esbuild/netbsd-x64": + optional: true + "@esbuild/openbsd-arm64": + optional: true + "@esbuild/openbsd-x64": + optional: true + "@esbuild/sunos-x64": + optional: true + "@esbuild/win32-arm64": + optional: true + "@esbuild/win32-ia32": + optional: true + "@esbuild/win32-x64": + optional: true + bin: + esbuild: bin/esbuild + checksum: 10c0/9f1aadd8d64f3bff422ae78387e66e51a5e09de6935a6f987b6e4e189ed00fdc2d1bc03d2e33633b094008529c8b6e06c7ad1a9782fb09fec223bf95998c0683 + languageName: node + linkType: hard + +"highlight.js@npm:^11.10.0": + version: 11.10.0 + resolution: "highlight.js@npm:11.10.0" + checksum: 10c0/cd8bf7ef06cbd72ddb83580ecabe769f08f062be8bb82d2eb492d31c17f7480d1f8d14a66fc81deee0601645435f19f04c470510563f847242a41ccff0ab873e + languageName: node + linkType: hard diff --git a/frontend/vendor/mousetrap/.gitignore b/frontend/vendor/mousetrap/.gitignore new file mode 100644 index 000000000..0db2a5c1d --- /dev/null +++ b/frontend/vendor/mousetrap/.gitignore @@ -0,0 +1,3 @@ +node_modules +*.sublime-workspace +/.yarn \ No newline at end of file diff --git a/frontend/vendor/mousetrap/LICENSE b/frontend/vendor/mousetrap/LICENSE new file mode 100644 index 000000000..33bb7d290 --- /dev/null +++ b/frontend/vendor/mousetrap/LICENSE @@ -0,0 +1,193 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + +TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + +1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + +2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + +3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + +4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + +5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + +6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + +7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + +8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + +9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + +END OF TERMS AND CONDITIONS + +--- Exceptions to the Apache 2.0 License ---- + +As an exception, if, as a result of your compiling your source code, portions +of this Software are embedded into an Object form of such source code, you +may redistribute such embedded portions in such Object form without complying +with the conditions of Sections 4(a), 4(b) and 4(d) of the License. + +In addition, if you combine or link compiled forms of this Software with +software that is licensed under the GPLv2 ("Combined Software") and if a +court of competent jurisdiction determines that the patent provision (Section +3), the indemnity provision (Section 9) or other Section of the License +conflicts with the conditions of the GPLv2, you may retroactively and +prospectively choose to deem waived or otherwise exclude such Section(s) of +the License, but only in their entirety and only with respect to the Combined +Software. diff --git a/frontend/vendor/mousetrap/index.js b/frontend/vendor/mousetrap/index.js new file mode 100644 index 000000000..07a5cb9b0 --- /dev/null +++ b/frontend/vendor/mousetrap/index.js @@ -0,0 +1,1025 @@ +/** + * Copyright 2012-2017 Craig Campbell + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * Mousetrap is a simple keyboard shortcut library for Javascript with + * no external dependencies + * + * @version 1.6.5 + * @url craig.is/killing/mice + */ +const globalDocument = globalThis?.document; +/** + * mapping of special keycodes to their corresponding keys + * + * everything in this dictionary cannot use keypress events + * so it has to be here to map to the correct keycodes for + * keyup/keydown events + * + * @type {Object} + */ +var _MAP = { + 8: 'backspace', + 9: 'tab', + 13: 'enter', + 16: 'shift', + 17: 'ctrl', + 18: 'alt', + 20: 'capslock', + 27: 'esc', + 32: 'space', + 33: 'pageup', + 34: 'pagedown', + 35: 'end', + 36: 'home', + 37: 'left', + 38: 'up', + 39: 'right', + 40: 'down', + 45: 'ins', + 46: 'del', + 91: 'meta', + 93: 'meta', + 224: 'meta', + 219: '219' +}; + +/** + * mapping for special characters so they can support + * + * this dictionary is only used incase you want to bind a + * keyup or keydown event to one of these keys + * + * @type {Object} + */ +var _KEYCODE_MAP = { + 106: '*', + 107: '+', + 109: '-', + 110: '.', + 111 : '/', + 186: ';', + 187: '=', + 188: ',', + 189: '-', + 190: '.', + 191: '/', + 192: '`', + 219: '[', + 220: '\\', + 221: ']', + 222: '\'' +}; + +/** + * this is a mapping of keys that require shift on a US keypad + * back to the non shift equivelents + * + * this is so you can use keyup events with these keys + * + * note that this will only work reliably on US keyboards + * + * @type {Object} + */ +var _SHIFT_MAP = { + '~': '`', + '!': '1', + '@': '2', + '#': '3', + '$': '4', + '%': '5', + '^': '6', + '&': '7', + '*': '8', + '(': '9', + ')': '0', + '_': '-', + '+': '=', + ':': ';', + '\"': '\'', + '<': ',', + '>': '.', + '?': '/', + '|': '\\' +}; + +/** + * this is a list of special strings you can use to map + * to modifier keys when you specify your keyboard shortcuts + * + * @type {Object} + */ + +var globalNavigator = globalThis.navigator; + +var _SPECIAL_ALIASES = { + 'option': 'alt', + 'command': 'meta', + 'return': 'enter', + 'escape': 'esc', + 'plus': '+', + 'mod': /Mac|iPod|iPhone|iPad/.test(globalNavigator?.platform) ? 'meta' : 'ctrl' +}; + +/** + * variable to store the flipped version of _MAP from above + * needed to check if we should use keypress or not when no action + * is specified + * + * @type {Object|undefined} + */ +var _REVERSE_MAP; + +/** + * loop through the f keys, f1 to f19 and add them to the map + * programatically + */ +for (var i = 1; i < 20; ++i) { + _MAP[111 + i] = 'f' + i; +} + +/** + * loop through to map numbers on the numeric keypad + */ +for (i = 0; i <= 9; ++i) { + + // This needs to use a string cause otherwise since 0 is falsey + // mousetrap will never fire for numpad 0 pressed as part of a keydown + // event. + // + // @see https://github.com/ccampbell/mousetrap/pull/258 + _MAP[i + 96] = i.toString(); +} + +/** + * cross browser add event method + * + * @param {Element|HTMLDocument} object + * @param {string} type + * @param {Function} callback + * @returns void + */ +function _addEvent(object, type, callback) { + if (object.addEventListener) { + object.addEventListener(type, callback, false); + return; + } + + object.attachEvent('on' + type, callback); +} + +/** + * takes the event and returns the key character + * + * @param {Event} e + * @return {string} + */ +function _characterFromEvent(e) { + + // for keypress events we should return the character as is + if (e.type == 'keypress') { + var character = String.fromCharCode(e.which); + + // if the shift key is not pressed then it is safe to assume + // that we want the character to be lowercase. this means if + // you accidentally have caps lock on then your key bindings + // will continue to work + // + // the only side effect that might not be desired is if you + // bind something like 'A' cause you want to trigger an + // event when capital A is pressed caps lock will no longer + // trigger the event. shift+a will though. + if (!e.shiftKey) { + character = character.toLowerCase(); + } + + return character; + } + + // for non keypress events the special maps are needed + if (_MAP[e.which]) { + return _MAP[e.which]; + } + + if (_KEYCODE_MAP[e.which]) { + return _KEYCODE_MAP[e.which]; + } + + // if it is not in the special map + + // with keydown and keyup events the character seems to always + // come in as an uppercase character whether you are pressing shift + // or not. we should make sure it is always lowercase for comparisons + return String.fromCharCode(e.which).toLowerCase(); +} + +/** + * checks if two arrays are equal + * + * @param {Array} modifiers1 + * @param {Array} modifiers2 + * @returns {boolean} + */ +function _modifiersMatch(modifiers1, modifiers2) { + return modifiers1.sort().join(',') === modifiers2.sort().join(','); +} + +/** + * takes a key event and figures out what the modifiers are + * + * @param {Event} e + * @returns {Array} + */ +function _eventModifiers(e) { + var modifiers = []; + + if (e.shiftKey) { + modifiers.push('shift'); + } + + if (e.altKey) { + modifiers.push('alt'); + } + + if (e.ctrlKey) { + modifiers.push('ctrl'); + } + + if (e.metaKey) { + modifiers.push('meta'); + } + + return modifiers; +} + +/** + * prevents default for this event + * + * @param {Event} e + * @returns void + */ +function _preventDefault(e) { + if (e.preventDefault) { + e.preventDefault(); + return; + } + + e.returnValue = false; +} + +/** + * stops propogation for this event + * + * @param {Event} e + * @returns void + */ +function _stopPropagation(e) { + if (e.stopPropagation) { + e.stopPropagation(); + return; + } + + e.cancelBubble = true; +} + +/** + * determines if the keycode specified is a modifier key or not + * + * @param {string} key + * @returns {boolean} + */ +function _isModifier(key) { + return key == 'shift' || key == 'ctrl' || key == 'alt' || key == 'meta'; +} + +/** + * reverses the map lookup so that we can look for specific keys + * to see what can and can't use keypress + * + * @return {Object} + */ +function _getReverseMap() { + if (!_REVERSE_MAP) { + _REVERSE_MAP = {}; + for (var key in _MAP) { + + // pull out the numeric keypad from here cause keypress should + // be able to detect the keys from the character + if (key > 95 && key < 112) { + continue; + } + + if (_MAP.hasOwnProperty(key)) { + _REVERSE_MAP[_MAP[key]] = key; + } + } + } + return _REVERSE_MAP; +} + +/** + * picks the best action based on the key combination + * + * @param {string} key - character for key + * @param {Array} modifiers + * @param {string=} action passed in + */ +function _pickBestAction(key, modifiers, action) { + + // if no action was picked in we should try to pick the one + // that we think would work best for this key + if (!action) { + action = _getReverseMap()[key] ? 'keydown' : 'keypress'; + } + + // modifier keys don't work as expected with keypress, + // switch to keydown + if (action == 'keypress' && modifiers.length) { + action = 'keydown'; + } + + return action; +} + +/** + * Converts from a string key combination to an array + * + * @param {string} combination like "command+shift+l" + * @return {Array} + */ +function _keysFromString(combination) { + if (combination === '+') { + return ['+']; + } + + combination = combination.replace(/\+{2}/g, '+plus'); + return combination.split('+'); +} + +/** + * Gets info for a specific key combination + * + * @param {string} combination key combination ("command+s" or "a" or "*") + * @param {string=} action + * @returns {Object} + */ +function _getKeyInfo(combination, action) { + var keys; + var key; + var i; + var modifiers = []; + + // take the keys from this pattern and figure out what the actual + // pattern is all about + keys = _keysFromString(combination); + + for (i = 0; i < keys.length; ++i) { + key = keys[i]; + + // normalize key names + if (_SPECIAL_ALIASES[key]) { + key = _SPECIAL_ALIASES[key]; + } + + // if this is not a keypress event then we should + // be smart about using shift keys + // this will only work for US keyboards however + if (action && action != 'keypress' && _SHIFT_MAP[key]) { + key = _SHIFT_MAP[key]; + modifiers.push('shift'); + } + + // if this key is a modifier then add it to the list of modifiers + if (_isModifier(key)) { + modifiers.push(key); + } + } + + // depending on what the key combination is + // we will try to pick the best event for it + action = _pickBestAction(key, modifiers, action); + + return { + key: key, + modifiers: modifiers, + action: action + }; +} + +function _belongsTo(element, ancestor) { + if (element === null || element === globalDocument) { + return false; + } + + if (element === ancestor) { + return true; + } + + return _belongsTo(element.parentNode, ancestor); +} + +function Mousetrap(targetElement) { + var self = this; + + targetElement = targetElement || globalDocument; + + /** + * element to attach key events to + * + * @type {Element} + */ + self.target = targetElement; + + /** + * a list of all the callbacks setup via Mousetrap.bind() + * + * @type {Object} + */ + self._callbacks = {}; + + /** + * direct map of string combinations to callbacks used for trigger() + * + * @type {Object} + */ + self._directMap = {}; + + /** + * keeps track of what level each sequence is at since multiple + * sequences can start out with the same sequence + * + * @type {Object} + */ + var _sequenceLevels = {}; + + /** + * variable to store the setTimeout call + * + * @type {null|number} + */ + var _resetTimer; + + /** + * temporary state where we will ignore the next keyup + * + * @type {boolean|string} + */ + var _ignoreNextKeyup = false; + + /** + * temporary state where we will ignore the next keypress + * + * @type {boolean} + */ + var _ignoreNextKeypress = false; + + /** + * are we currently inside of a sequence? + * type of action ("keyup" or "keydown" or "keypress") or false + * + * @type {boolean|string} + */ + var _nextExpectedAction = false; + + /** + * resets all sequence counters except for the ones passed in + * + * @param {Object} doNotReset + * @returns void + */ + function _resetSequences(doNotReset) { + doNotReset = doNotReset || {}; + + var activeSequences = false, + key; + + for (key in _sequenceLevels) { + if (doNotReset[key]) { + activeSequences = true; + continue; + } + _sequenceLevels[key] = 0; + } + + if (!activeSequences) { + _nextExpectedAction = false; + } + } + + /** + * finds all callbacks that match based on the keycode, modifiers, + * and action + * + * @param {string} character + * @param {Array} modifiers + * @param {Event|Object} e + * @param {string=} sequenceName - name of the sequence we are looking for + * @param {string=} combination + * @param {number=} level + * @returns {Array} + */ + function _getMatches(character, modifiers, e, sequenceName, combination, level) { + var i; + var callback; + var matches = []; + var action = e.type; + + // if there are no events related to this keycode + if (!self._callbacks[character]) { + return []; + } + + // if a modifier key is coming up on its own we should allow it + if (action == 'keyup' && _isModifier(character)) { + modifiers = [character]; + } + + // loop through all callbacks for the key that was pressed + // and see if any of them match + for (i = 0; i < self._callbacks[character].length; ++i) { + callback = self._callbacks[character][i]; + + // if a sequence name is not specified, but this is a sequence at + // the wrong level then move onto the next match + if (!sequenceName && callback.seq && _sequenceLevels[callback.seq] != callback.level) { + continue; + } + + // if the action we are looking for doesn't match the action we got + // then we should keep going + if (action != callback.action) { + continue; + } + + // if this is a keypress event and the meta key and control key + // are not pressed that means that we need to only look at the + // character, otherwise check the modifiers as well + // + // chrome will not fire a keypress if meta or control is down + // safari will fire a keypress if meta or meta+shift is down + // firefox will fire a keypress if meta or control is down + if ((action == 'keypress' && !e.metaKey && !e.ctrlKey) || _modifiersMatch(modifiers, callback.modifiers)) { + + // when you bind a combination or sequence a second time it + // should overwrite the first one. if a sequenceName or + // combination is specified in this call it does just that + // + // @todo make deleting its own method? + var deleteCombo = !sequenceName && callback.combo == combination; + var deleteSequence = sequenceName && callback.seq == sequenceName && callback.level == level; + if (deleteCombo || deleteSequence) { + self._callbacks[character].splice(i, 1); + } + + matches.push(callback); + } + } + + return matches; + } + + /** + * actually calls the callback function + * + * if your callback function returns false this will use the jquery + * convention - prevent default and stop propogation on the event + * + * @param {Function} callback + * @param {Event} e + * @returns void + */ + function _fireCallback(callback, e, combo, sequence) { + + // if this event should not happen stop here + if (self.stopCallback(e, e.target || e.srcElement, combo, sequence)) { + return; + } + + if (callback(e, combo) === false) { + _preventDefault(e); + _stopPropagation(e); + } + } + + /** + * handles a character key event + * + * @param {string} character + * @param {Array} modifiers + * @param {Event} e + * @returns void + */ + self._handleKey = function(character, modifiers, e) { + var callbacks = _getMatches(character, modifiers, e); + var i; + var doNotReset = {}; + var maxLevel = 0; + var processedSequenceCallback = false; + + // Calculate the maxLevel for sequences so we can only execute the longest callback sequence + for (i = 0; i < callbacks.length; ++i) { + if (callbacks[i].seq) { + maxLevel = Math.max(maxLevel, callbacks[i].level); + } + } + + // loop through matching callbacks for this key event + for (i = 0; i < callbacks.length; ++i) { + + // fire for all sequence callbacks + // this is because if for example you have multiple sequences + // bound such as "g i" and "g t" they both need to fire the + // callback for matching g cause otherwise you can only ever + // match the first one + if (callbacks[i].seq) { + + // only fire callbacks for the maxLevel to prevent + // subsequences from also firing + // + // for example 'a option b' should not cause 'option b' to fire + // even though 'option b' is part of the other sequence + // + // any sequences that do not match here will be discarded + // below by the _resetSequences call + if (callbacks[i].level != maxLevel) { + continue; + } + + processedSequenceCallback = true; + + // keep a list of which sequences were matches for later + doNotReset[callbacks[i].seq] = 1; + _fireCallback(callbacks[i].callback, e, callbacks[i].combo, callbacks[i].seq); + continue; + } + + // if there were no sequence matches but we are still here + // that means this is a regular match so we should fire that + if (!processedSequenceCallback) { + _fireCallback(callbacks[i].callback, e, callbacks[i].combo); + } + } + + // if the key you pressed matches the type of sequence without + // being a modifier (ie "keyup" or "keypress") then we should + // reset all sequences that were not matched by this event + // + // this is so, for example, if you have the sequence "h a t" and you + // type "h e a r t" it does not match. in this case the "e" will + // cause the sequence to reset + // + // modifier keys are ignored because you can have a sequence + // that contains modifiers such as "enter ctrl+space" and in most + // cases the modifier key will be pressed before the next key + // + // also if you have a sequence such as "ctrl+b a" then pressing the + // "b" key will trigger a "keypress" and a "keydown" + // + // the "keydown" is expected when there is a modifier, but the + // "keypress" ends up matching the _nextExpectedAction since it occurs + // after and that causes the sequence to reset + // + // we ignore keypresses in a sequence that directly follow a keydown + // for the same character + var ignoreThisKeypress = e.type == 'keypress' && _ignoreNextKeypress; + if (e.type == _nextExpectedAction && !_isModifier(character) && !ignoreThisKeypress) { + _resetSequences(doNotReset); + } + + _ignoreNextKeypress = processedSequenceCallback && e.type == 'keydown'; + }; + + /** + * handles a keydown event + * + * @param {Event} e + * @returns void + */ + function _handleKeyEvent(e) { + + // normalize e.which for key events + // @see http://stackoverflow.com/questions/4285627/javascript-keycode-vs-charcode-utter-confusion + if (typeof e.which !== 'number') { + e.which = e.keyCode; + } + + var character = _characterFromEvent(e); + + // no character found then stop + if (!character) { + return; + } + + // need to use === for the character check because the character can be 0 + if (e.type == 'keyup' && _ignoreNextKeyup === character) { + _ignoreNextKeyup = false; + return; + } + + self.handleKey(character, _eventModifiers(e), e); + } + + /** + * called to set a 1 second timeout on the specified sequence + * + * this is so after each key press in the sequence you have 1 second + * to press the next key before you have to start over + * + * @returns void + */ + function _resetSequenceTimer() { + clearTimeout(_resetTimer); + _resetTimer = setTimeout(_resetSequences, 1000); + } + + /** + * binds a key sequence to an event + * + * @param {string} combo - combo specified in bind call + * @param {Array} keys + * @param {Function} callback + * @param {string=} action + * @returns void + */ + function _bindSequence(combo, keys, callback, action) { + + // start off by adding a sequence level record for this combination + // and setting the level to 0 + _sequenceLevels[combo] = 0; + + /** + * callback to increase the sequence level for this sequence and reset + * all other sequences that were active + * + * @param {string} nextAction + * @returns {Function} + */ + function _increaseSequence(nextAction) { + return function() { + _nextExpectedAction = nextAction; + ++_sequenceLevels[combo]; + _resetSequenceTimer(); + }; + } + + /** + * wraps the specified callback inside of another function in order + * to reset all sequence counters as soon as this sequence is done + * + * @param {Event} e + * @returns void + */ + function _callbackAndReset(e) { + _fireCallback(callback, e, combo); + + // we should ignore the next key up if the action is key down + // or keypress. this is so if you finish a sequence and + // release the key the final key will not trigger a keyup + if (action !== 'keyup') { + _ignoreNextKeyup = _characterFromEvent(e); + } + + // weird race condition if a sequence ends with the key + // another sequence begins with + setTimeout(_resetSequences, 10); + } + + // loop through keys one at a time and bind the appropriate callback + // function. for any key leading up to the final one it should + // increase the sequence. after the final, it should reset all sequences + // + // if an action is specified in the original bind call then that will + // be used throughout. otherwise we will pass the action that the + // next key in the sequence should match. this allows a sequence + // to mix and match keypress and keydown events depending on which + // ones are better suited to the key provided + for (var i = 0; i < keys.length; ++i) { + var isFinal = i + 1 === keys.length; + var wrappedCallback = isFinal ? _callbackAndReset : _increaseSequence(action || _getKeyInfo(keys[i + 1]).action); + _bindSingle(keys[i], wrappedCallback, action, combo, i); + } + } + + /** + * binds a single keyboard combination + * + * @param {string} combination + * @param {Function} callback + * @param {string=} action + * @param {string=} sequenceName - name of sequence if part of sequence + * @param {number=} level - what part of the sequence the command is + * @returns void + */ + function _bindSingle(combination, callback, action, sequenceName, level) { + + // store a direct mapped reference for use with Mousetrap.trigger + self._directMap[combination + ':' + action] = callback; + + // make sure multiple spaces in a row become a single space + combination = combination.replace(/\s+/g, ' '); + + var sequence = combination.split(' '); + var info; + + // if this pattern is a sequence of keys then run through this method + // to reprocess each pattern one key at a time + if (sequence.length > 1) { + _bindSequence(combination, sequence, callback, action); + return; + } + + info = _getKeyInfo(combination, action); + + // make sure to initialize array if this is the first time + // a callback is added for this key + self._callbacks[info.key] = self._callbacks[info.key] || []; + + // remove an existing match if there is one + _getMatches(info.key, info.modifiers, {type: info.action}, sequenceName, combination, level); + + // add this call back to the array + // if it is a sequence put it at the beginning + // if not put it at the end + // + // this is important because the way these are processed expects + // the sequence ones to come first + self._callbacks[info.key][sequenceName ? 'unshift' : 'push']({ + callback: callback, + modifiers: info.modifiers, + action: info.action, + seq: sequenceName, + level: level, + combo: combination + }); + } + + /** + * binds multiple combinations to the same callback + * + * @param {Array} combinations + * @param {Function} callback + * @param {string|undefined} action + * @returns void + */ + self._bindMultiple = function(combinations, callback, action) { + for (var i = 0; i < combinations.length; ++i) { + _bindSingle(combinations[i], callback, action); + } + }; + + if (targetElement) { + _addEvent(targetElement, 'keypress', _handleKeyEvent); + _addEvent(targetElement, 'keydown', _handleKeyEvent); + _addEvent(targetElement, 'keyup', _handleKeyEvent); + } +} + +/** + * binds an event to mousetrap + * + * can be a single key, a combination of keys separated with +, + * an array of keys, or a sequence of keys separated by spaces + * + * be sure to list the modifier keys first to make sure that the + * correct key ends up getting bound (the last key in the pattern) + * + * @param {string|Array} keys + * @param {Function} callback + * @param {string=} action - 'keypress', 'keydown', or 'keyup' + * @returns void + */ +Mousetrap.prototype.bind = function(keys, callback, action) { + var self = this; + keys = keys instanceof Array ? keys : [keys]; + self._bindMultiple.call(self, keys, callback, action); + return self; +}; + +/** + * unbinds an event to mousetrap + * + * the unbinding sets the callback function of the specified key combo + * to an empty function and deletes the corresponding key in the + * _directMap dict. + * + * TODO: actually remove this from the _callbacks dictionary instead + * of binding an empty function + * + * the keycombo+action has to be exactly the same as + * it was defined in the bind method + * + * @param {string|Array} keys + * @param {string} action + * @returns void + */ +Mousetrap.prototype.unbind = function(keys, action) { + var self = this; + return self.bind.call(self, keys, function() {}, action); +}; + +/** + * triggers an event that has already been bound + * + * @param {string} keys + * @param {string=} action + * @returns void + */ +Mousetrap.prototype.trigger = function(keys, action) { + var self = this; + if (self._directMap[keys + ':' + action]) { + self._directMap[keys + ':' + action]({}, keys); + } + return self; +}; + +/** + * resets the library back to its initial state. this is useful + * if you want to clear out the current keyboard shortcuts and bind + * new ones - for example if you switch to another page + * + * @returns void + */ +Mousetrap.prototype.reset = function() { + var self = this; + self._callbacks = {}; + self._directMap = {}; + return self; +}; + +/** + * should we stop this event before firing off callbacks + * + * @param {Event} e + * @param {Element} element + * @return {boolean} + */ +Mousetrap.prototype.stopCallback = function (e, element, combo) { + // if the element has the data attribute "mousetrap-dont-stop" then no need + // to stop. It should be used like
...
+ // or :div {:data-mousetrap-dont-stop true} + if ('mousetrapDontStop' in element.dataset) { + return false + } + + if ((' ' + element.className + ' ').indexOf(' mousetrap ') > -1) { + return false; + } + + if ('composedPath' in e && typeof e.composedPath === 'function') { + // For open shadow trees, update `element` so that the following check works. + const initialEventTarget = e.composedPath()[0]; + if (initialEventTarget !== e.target) { + element = initialEventTarget; + } + } + + // stop for input, select, textarea and button + const shouldStop = element.tagName == "INPUT" || + element.tagName == "SELECT" || + element.tagName == "TEXTAREA" || + (element.tagName == "BUTTON" && combo.includes("tab")) || + (element.contentEditable && element.contentEditable == "true"); + return shouldStop; +} + + +/** + * exposes _handleKey publicly so it can be overwritten by extensions + */ +Mousetrap.prototype.handleKey = function() { + var self = this; + return self._handleKey.apply(self, arguments); +}; + +/** + * allow custom key mappings + */ +export function addKeycodes(object) { + for (var key in object) { + if (object.hasOwnProperty(key)) { + _MAP[key] = object[key]; + } + } + _REVERSE_MAP = null; +}; + +/** + * Init the global mousetrap functions + * + * This method is needed to allow the global mousetrap functions to work + * now that mousetrap is a constructor function. + */ +var instance = new Mousetrap(globalDocument); + +export default instance; diff --git a/frontend/vendor/mousetrap/package.json b/frontend/vendor/mousetrap/package.json new file mode 100644 index 000000000..f02e97f38 --- /dev/null +++ b/frontend/vendor/mousetrap/package.json @@ -0,0 +1,15 @@ +{ + "name": "@penpot/mousetrap", + "version": "1.6.5", + "description": "Simple library for handling keyboard shortcuts", + "module": "index.js", + "scripts": { + "build": "esbuild --bundle --outfile=dist/index.js --format=esm index.js" + }, + "packageManager": "yarn@4.3.1", + "author": "Craig Campbell", + "license": "Apache-2.0 WITH LLVM-exception", + "devDependencies": { + "esbuild": "^0.24.0" + } +} diff --git a/frontend/vendor/mousetrap/yarn.lock b/frontend/vendor/mousetrap/yarn.lock new file mode 100644 index 000000000..1d8f60533 --- /dev/null +++ b/frontend/vendor/mousetrap/yarn.lock @@ -0,0 +1,265 @@ +# This file is generated by running "yarn install" inside your project. +# Manual changes might be lost - proceed with caution! + +__metadata: + version: 8 + cacheKey: 10c0 + +"@esbuild/aix-ppc64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/aix-ppc64@npm:0.24.0" + conditions: os=aix & cpu=ppc64 + languageName: node + linkType: hard + +"@esbuild/android-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-arm64@npm:0.24.0" + conditions: os=android & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/android-arm@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-arm@npm:0.24.0" + conditions: os=android & cpu=arm + languageName: node + linkType: hard + +"@esbuild/android-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-x64@npm:0.24.0" + conditions: os=android & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/darwin-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/darwin-arm64@npm:0.24.0" + conditions: os=darwin & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/darwin-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/darwin-x64@npm:0.24.0" + conditions: os=darwin & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/freebsd-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/freebsd-arm64@npm:0.24.0" + conditions: os=freebsd & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/freebsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/freebsd-x64@npm:0.24.0" + conditions: os=freebsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/linux-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-arm64@npm:0.24.0" + conditions: os=linux & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/linux-arm@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-arm@npm:0.24.0" + conditions: os=linux & cpu=arm + languageName: node + linkType: hard + +"@esbuild/linux-ia32@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-ia32@npm:0.24.0" + conditions: os=linux & cpu=ia32 + languageName: node + linkType: hard + +"@esbuild/linux-loong64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-loong64@npm:0.24.0" + conditions: os=linux & cpu=loong64 + languageName: node + linkType: hard + +"@esbuild/linux-mips64el@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-mips64el@npm:0.24.0" + conditions: os=linux & cpu=mips64el + languageName: node + linkType: hard + +"@esbuild/linux-ppc64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-ppc64@npm:0.24.0" + conditions: os=linux & cpu=ppc64 + languageName: node + linkType: hard + +"@esbuild/linux-riscv64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-riscv64@npm:0.24.0" + conditions: os=linux & cpu=riscv64 + languageName: node + linkType: hard + +"@esbuild/linux-s390x@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-s390x@npm:0.24.0" + conditions: os=linux & cpu=s390x + languageName: node + linkType: hard + +"@esbuild/linux-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-x64@npm:0.24.0" + conditions: os=linux & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/netbsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/netbsd-x64@npm:0.24.0" + conditions: os=netbsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/openbsd-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/openbsd-arm64@npm:0.24.0" + conditions: os=openbsd & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/openbsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/openbsd-x64@npm:0.24.0" + conditions: os=openbsd & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/sunos-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/sunos-x64@npm:0.24.0" + conditions: os=sunos & cpu=x64 + languageName: node + linkType: hard + +"@esbuild/win32-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-arm64@npm:0.24.0" + conditions: os=win32 & cpu=arm64 + languageName: node + linkType: hard + +"@esbuild/win32-ia32@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-ia32@npm:0.24.0" + conditions: os=win32 & cpu=ia32 + languageName: node + linkType: hard + +"@esbuild/win32-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-x64@npm:0.24.0" + conditions: os=win32 & cpu=x64 + languageName: node + linkType: hard + +"@penpot/mousetrap@workspace:.": + version: 0.0.0-use.local + resolution: "@penpot/mousetrap@workspace:." + dependencies: + esbuild: "npm:^0.24.0" + languageName: unknown + linkType: soft + +"esbuild@npm:^0.24.0": + version: 0.24.0 + resolution: "esbuild@npm:0.24.0" + dependencies: + "@esbuild/aix-ppc64": "npm:0.24.0" + "@esbuild/android-arm": "npm:0.24.0" + "@esbuild/android-arm64": "npm:0.24.0" + "@esbuild/android-x64": "npm:0.24.0" + "@esbuild/darwin-arm64": "npm:0.24.0" + "@esbuild/darwin-x64": "npm:0.24.0" + "@esbuild/freebsd-arm64": "npm:0.24.0" + "@esbuild/freebsd-x64": "npm:0.24.0" + "@esbuild/linux-arm": "npm:0.24.0" + "@esbuild/linux-arm64": "npm:0.24.0" + "@esbuild/linux-ia32": "npm:0.24.0" + "@esbuild/linux-loong64": "npm:0.24.0" + "@esbuild/linux-mips64el": "npm:0.24.0" + "@esbuild/linux-ppc64": "npm:0.24.0" + "@esbuild/linux-riscv64": "npm:0.24.0" + "@esbuild/linux-s390x": "npm:0.24.0" + "@esbuild/linux-x64": "npm:0.24.0" + "@esbuild/netbsd-x64": "npm:0.24.0" + "@esbuild/openbsd-arm64": "npm:0.24.0" + "@esbuild/openbsd-x64": "npm:0.24.0" + "@esbuild/sunos-x64": "npm:0.24.0" + "@esbuild/win32-arm64": "npm:0.24.0" + "@esbuild/win32-ia32": "npm:0.24.0" + "@esbuild/win32-x64": "npm:0.24.0" + dependenciesMeta: + "@esbuild/aix-ppc64": + optional: true + "@esbuild/android-arm": + optional: true + "@esbuild/android-arm64": + optional: true + "@esbuild/android-x64": + optional: true + "@esbuild/darwin-arm64": + optional: true + "@esbuild/darwin-x64": + optional: true + "@esbuild/freebsd-arm64": + optional: true + "@esbuild/freebsd-x64": + optional: true + "@esbuild/linux-arm": + optional: true + "@esbuild/linux-arm64": + optional: true + "@esbuild/linux-ia32": + optional: true + "@esbuild/linux-loong64": + optional: true + "@esbuild/linux-mips64el": + optional: true + "@esbuild/linux-ppc64": + optional: true + "@esbuild/linux-riscv64": + optional: true + "@esbuild/linux-s390x": + optional: true + "@esbuild/linux-x64": + optional: true + "@esbuild/netbsd-x64": + optional: true + "@esbuild/openbsd-arm64": + optional: true + "@esbuild/openbsd-x64": + optional: true + "@esbuild/sunos-x64": + optional: true + "@esbuild/win32-arm64": + optional: true + "@esbuild/win32-ia32": + optional: true + "@esbuild/win32-x64": + optional: true + bin: + esbuild: bin/esbuild + checksum: 10c0/9f1aadd8d64f3bff422ae78387e66e51a5e09de6935a6f987b6e4e189ed00fdc2d1bc03d2e33633b094008529c8b6e06c7ad1a9782fb09fec223bf95998c0683 + languageName: node + linkType: hard diff --git a/frontend/vendor/text_editor_v2.js b/frontend/vendor/text_editor_v2.js deleted file mode 100644 index f11ea57f1..000000000 --- a/frontend/vendor/text_editor_v2.js +++ /dev/null @@ -1,3 +0,0 @@ -var pe=r=>{throw TypeError(r)};var jt=(r,i,t)=>i.has(r)||pe("Cannot "+t);var e=(r,i,t)=>(jt(r,i,"read from private field"),t?t.call(r):i.get(r)),u=(r,i,t)=>i.has(r)?pe("Cannot add the same private member more than once"):i instanceof WeakSet?i.add(r):i.set(r,t),a=(r,i,t,n)=>(jt(r,i,"write to private field"),n?n.call(r,t):i.set(r,t),t),y=(r,i,t)=>(jt(r,i,"access private method"),t);function de(r,i){}function ge(r,i){}var Qt=null,Jt=null;function gr(){return Qt||(Qt=mr(1,1)),Jt||(Jt=Qt.getContext("2d")),Jt}function mr(r,i){return"OffscreenCanvas"in globalThis?new OffscreenCanvas(r,i):document.createElement("canvas")}function qt(r){return r.toString(16).padStart(2,"0")}function yr(r){let i=gr();i.fillStyle=r,i.fillRect(0,0,1,1);let t=i.getImageData(0,0,1,1),[n,s,o,h]=t.data;return[`#${qt(n)}${qt(s)}${qt(o)}`,h/255]}function me(r){let[i,t]=yr(r);return`[["^ ","~:fill-color","${i}","~:fill-opacity",${t}]]`}function ye(r,i){for(let t=0;tt.setAttribute(n,s)),i?.data&&Object.entries(i.data).forEach(([n,s])=>t.dataset[n]=s),i?.styles&&i?.allowedStyles&&j(t,i.allowedStyles,i.styles),i?.children&&(Array.isArray(i.children)?t.append(...i.children):t.appendChild(i.children)),t}function ht(r,i){return r.nodeType===Node.ELEMENT_NODE&&r.nodeName===i.toUpperCase()}function Ft(r,i){return i===0}function Rt(r,i){return r.nodeType===Node.TEXT_NODE?r.nodeValue.length===i:!0}var Ce="BR";function lt(){return document.createElement(Ce)}function N(r){return r.nodeType===Node.ELEMENT_NODE&&r.nodeName===Ce}var Ie="SPAN",te="inline",Nr=`[data-itype="${te}"]`,ee=[["--typography-ref-id"],["--typography-ref-file"],["--font-id"],["--font-variant-id"],["--fills"],["font-variant"],["font-family"],["font-size","px"],["font-weight"],["font-style"],["line-height"],["letter-spacing","px"],["text-decoration"],["text-transform"]];function A(r){return!(!r||!ht(r,Ie)||r.dataset.itype!==te)}function Se(r){return r?["A","ABBR","ACRONYM","B","BDO","BIG","BR","BUTTON","CITE","CODE","DFN","EM","I","IMG","INPUT","KBD","LABEL","MAP","OBJECT","OUTPUT","Q","SAMP","SCRIPT","SELECT","SMALL","SPAN","STRONG","SUB","SUP","TEXTAREA","TIME","TT","VAR"].includes(r.nodeName):!1}function D(r,i,t){if(!(r instanceof HTMLBRElement)&&!(r instanceof Text))throw new TypeError("Invalid inline child");if(r instanceof Text&&r.nodeValue.length===0)throw console.trace("nodeValue",r.nodeValue),new TypeError("Invalid inline child, cannot be an empty text");return at(Ie,{attributes:{id:ot(),...t},data:{itype:te},styles:i,allowedStyles:ee,children:r})}function ve(r,i,t,n){return D(i,Te(ee,r.style,t),n)}function tt(r){return D(lt(),r)}function gt(r,i){return j(r,ee,i)}function P(r){if(!r)return null;if(A(r))return r;if(r.nodeType===Node.TEXT_NODE){let i=r?.parentElement;return!i||!A(i)?null:i}return r.closest(Nr)}function we(r,i){let t=P(r);return t?Ft(t,i):!1}function Lt(r,i){let t=P(r);return t?Rt(t.firstChild,i):!1}function mt(r,i){let t=r.firstChild,n=r.style,s=t.splitText(i);return D(s,n)}function Oe(r){let i=[],t=r,n=0;for(;t;)n>0&&i.push(t),t=t.nextElementSibling,n++;return i}function Ae(r){if(!A(r))throw new Error("Invalid inline");return N(r.firstChild)?0:r.firstChild.nodeValue.length}var De="DIV",re="root",$r=`[data-itype="${re}"]`,Fe=[["--vertical-align"]];function Re(r){return!(!r||!ht(r,De)||r.dataset.itype!==re)}function ie(r,i,t){if(!Array.isArray(r)||!r.every(M))throw new TypeError("Invalid root children");return at(De,{attributes:{id:ot(),...t},data:{itype:re},styles:i,allowedStyles:Fe,children:r})}function Le(r){return ie([et(r)],r)}function Ve(r,i){return j(r,Fe,i)}function ne(r){if(!r)throw new TypeError("Invalid text node");return r.nodeType===Node.TEXT_NODE||N(r)}function Be(r){if(!r)throw new TypeError("Invalid text node");return N(r)?0:r.nodeValue.length}function yt(r){if(ne(r))return r;if(A(r))return r.firstChild;if(M(r))return r.firstChild.firstChild;if(Re(r))return r.firstChild.firstChild.firstChild;throw new Error("Cannot find a text node")}var be="DIV",se="paragraph",Pr=`[data-itype="${se}"]`,Me=[["--typography-ref-id"],["--typography-ref-file"],["--font-id"],["--font-variant-id"],["--fills"],["font-variant"],["font-family"],["font-size","px"],["font-weight"],["font-style"],["line-height"],["letter-spacing","px"],["text-decoration"],["text-transform"],["text-align"],["direction"]];function Ue(r){return!Se(r)}function ke(r){if(!M(r))throw new TypeError("Invalid paragraph");let i=r.firstChild;if(!A(i))throw new TypeError("Invalid inline");return N(i.firstChild)}function M(r){return!(!r||!ht(r,be)||r.dataset.itype!==se)}function X(r,i,t){if(r&&(!Array.isArray(r)||!r.every(A)))throw new TypeError("Invalid paragraph children");return at(be,{attributes:{id:ot(),...t},data:{itype:se},styles:i,allowedStyles:Me,children:r})}function et(r){return X([tt(r)],r)}function xt(r,i){return j(r,Me,i)}function F(r){if(!r)return null;if(M(r))return r;if(r.nodeType===Node.TEXT_NODE||N(r)){let i=r?.parentElement?.parentElement;return!i||!M(i)?null:i}return r.closest(Pr)}function We(r,i){let t=F(r);if(!t)throw new Error("Can't find the paragraph");let n=P(r);if(!n)throw new Error("Can't find the inline");return t.firstElementChild===n&&Ft(n.firstChild,i)}function _e(r,i){let t=F(r);if(!t)throw new Error("Cannot find the paragraph");let n=P(r);if(!n)throw new Error("Cannot find the inline");return t.lastElementChild===n&&Rt(n.firstChild,i)}function Vt(r,i,t){let n=r.style;if(Lt(i,t))return X(Oe(i),n);let s=mt(i,t);return X([s],n)}function Bt(r,i){return r.append(...i.children),i.remove(),r}function Tr(r,i,t){let n=r.createNodeIterator(i,NodeFilter.SHOW_TEXT),s=r.createDocumentFragment(),o=null,h=n.nextNode();for(;h;){let I=Ee(ye(t,xe(h.parentElement)));Zt(h.parentElement.style)||Zt(I)||Ue(h.parentElement)?(o&&s.appendChild(o),o=X(void 0,I)):o===null&&(o=X()),o.appendChild(D(new Text(h.nodeValue),I)),h=n.nextNode()}return s.appendChild(o),s}function Ge(r,i){let n=new DOMParser().parseFromString(r,"text/html");return Tr(n,n.documentElement,i)}function Xe(r,i){let t=r.replace(/\r/g,"").split(` -`),n=document.createDocumentFragment();for(let s of t)s===""?n.appendChild(et(i)):n.appendChild(X([D(new Text(s),i)],i));return n}function $e(r,i,t){r.preventDefault();let n=null;if(r.clipboardData.types.includes("text/html")){let s=r.clipboardData.getData("text/html");n=Ge(s,t.currentStyle)}else if(r.clipboardData.types.includes("text/plain")){let s=r.clipboardData.getData("text/plain");n=Xe(s,t.currentStyle)}n&&(t.isCollapsed?t.insertPaste(n):t.replaceWithPaste(n))}var bt={copy:de,cut:ge,paste:$e};function Ye(r,i,t){if(r.preventDefault(),t.isCollapsed){if(t.isTextFocus)return t.insertText(r.data);if(t.isLineBreakFocus)return t.replaceLineBreak(r.data)}else{if(t.isMultiParagraph)return t.replaceParagraphs(r.data);if(t.isMultiInline)return t.replaceInlines(r.data);if(t.isTextSame)return t.replaceText(r.data)}}function ze(r,i,t){return r.preventDefault(),t.isCollapsed?t.insertParagraph():t.replaceWithParagraph()}function He(r,i,t){if(r.preventDefault(),t.isCollapsed)throw new Error("This should be impossible");return t.removeSelected()}function Ke(r,i,t){if(r.preventDefault(),!i.isEmpty){if(!t.isCollapsed)return t.removeSelected({direction:"backward"});if(t.isTextFocus&&t.focusOffset>0)return t.removeBackwardText();if(t.isTextFocus&&t.focusAtStart)return t.mergeBackwardParagraph();if(t.isInlineFocus||t.isLineBreakFocus)return t.removeBackwardParagraph()}}function je(r,i,t){if(r.preventDefault(),!i.isEmpty){if(!t.isCollapsed)return t.removeSelected({direction:"forward"});if(t.isTextFocus&&t.focusOffset>=0)return t.removeForwardText();if(t.isTextFocus&&t.focusAtEnd)return t.mergeForwardParagraph();if((t.isInlineFocus||t.isLineBreakFocus)&&i.numParagraphs>1)return t.removeForwardParagraph()}}var Mt={insertText:Ye,insertParagraph:ze,deleteByCut:He,deleteContentBackward:Ke,deleteContentForward:je};var rt,Et,Nt,Pt,oe=class extends EventTarget{constructor(t=500){super();u(this,rt,null);u(this,Et,1e3);u(this,Nt,!1);u(this,Pt,()=>{this.dispatchEvent(new Event("change"))});if(typeof t=="number"&&(!Number.isInteger(t)||t<=0))throw new TypeError("Invalid time");a(this,Et,t??500)}get hasPendingChanges(){return e(this,Nt)}notifyDebounced(){a(this,Nt,!0),clearTimeout(e(this,rt)),a(this,rt,setTimeout(e(this,Pt),e(this,Et)))}notifyImmediately(){clearTimeout(e(this,rt)),e(this,Pt).call(this)}dispose(){this.hasPendingChanges&&this.notifyImmediately(),clearTimeout(e(this,rt))}};rt=new WeakMap,Et=new WeakMap,Nt=new WeakMap,Pt=new WeakMap;var Qe=oe;function it(r){if(!Number.isInteger(r)||r<0)throw new TypeError("Invalid offset")}function nt(r){if(typeof r!="string")throw new TypeError("Invalid string")}function Ut(r,i,t){return nt(r),it(i),nt(t),r.slice(0,i)+t+r.slice(i)}function Je(r,i,t,n){return nt(r),it(i),it(t),nt(n),r.slice(0,i)+n+r.slice(t)}function qe(r,i){return nt(r),it(i),i===0?r:r.slice(0,i-1)+r.slice(i)}function Ze(r,i){return nt(r),it(i),r.slice(0,i)+r.slice(i+1)}function tr(r,i,t){return nt(r),it(i),it(t),r.slice(0,i)+r.slice(t)}var Q={FORWARD:1,BACKWARD:0},st,w,T=class T{constructor(i){u(this,st,null);u(this,w,null);if(!(i instanceof HTMLElement))throw new TypeError("Invalid root node");a(this,st,i),a(this,w,T.findDown(i,i))}static isTextNode(i){return i.nodeType===Node.TEXT_NODE||i.nodeType===Node.ELEMENT_NODE&&i.nodeName==="BR"}static isContainerNode(i){return i.nodeType===Node.ELEMENT_NODE&&i.nodeName!=="BR"}static findDown(i,t,n=new Set,s=Q.FORWARD){if(i===t)return T.findDown(s===Q.FORWARD?i.firstChild:i.lastChild,t,n,s);let o=Date.now(),h=i;for(;h;){if(Date.now()-o>=1e3)throw new Error("Iteration timeout");if(n.has(h)){h=s===Q.FORWARD?h.nextSibling:h.previousSibling;continue}if(T.isTextNode(h))return h;if(T.isContainerNode(h))return T.findDown(s===Q.FORWARD?h.firstChild:h.lastChild,t,n,s);h=s===Q.FORWARD?h.nextSibling:h.previousSibling}return null}static findUp(i,t,n=new Set,s=Q.FORWARD){if(n.add(i),T.isTextNode(i))return T.findUp(i.parentNode,t,n,s);if(T.isContainerNode(i)){let o=T.findDown(i,t,n,s);if(o)return o;if(i!==t)return T.findUp(i.parentNode,t,n,s)}return null}get currentNode(){return e(this,w)}set currentNode(i){let t=(i.compareDocumentPosition(e(this,st))&Node.DOCUMENT_POSITION_CONTAINS)===Node.DOCUMENT_POSITION_CONTAINS;if(!(i instanceof Node)||!T.isTextNode(i)||!t)throw new TypeError("Invalid new current node");a(this,w,i)}nextNode(){if(!e(this,w))return null;let i=T.findUp(e(this,w),e(this,st),new Set,Q.FORWARD);return i?(a(this,w,i),e(this,w)):null}previousNode(){if(!e(this,w))return null;let i=T.findUp(e(this,w),e(this,st),new Set,Q.BACKWARD);return i?(a(this,w,i),e(this,w)):null}};st=new WeakMap,w=new WeakMap;var ae=T,er=ae;var $,Y,z,he=class{constructor(i,t,n){u(this,$,new Set);u(this,Y,new Set);u(this,z,new Set);i&&Array.isArray(i)&&a(this,$,new Set(i)),t&&Array.isArray(t)&&a(this,z,new Set(t)),n&&Array.isArray(n)&&a(this,Y,new Set(n))}get added(){return e(this,$)}get removed(){return e(this,Y)}get updated(){return e(this,z)}clear(){e(this,$).clear(),e(this,Y).clear(),e(this,z).clear()}dispose(){e(this,$).clear(),a(this,$,null),e(this,Y).clear(),a(this,Y,null),e(this,z).clear(),a(this,z,null)}add(i){return e(this,$).add(i),this}remove(i){return e(this,Y).add(i),this}update(i){return e(this,z).add(i),this}};$=new WeakMap,Y=new WeakMap,z=new WeakMap;var rr=he;var ut={FORWARD:1,NONE:0,BACKWARD:-1};var ir=Date.now();function Cr(){ir=Date.now()}function Ir(){if(Date.now-ir>=1e3)throw new Error("Safe guard timeout")}var Tt={start:Cr,update:Ir};var U,f,R,g,L,J,H,ft,l,c,V,Ct,ct,d,pt,x,ue,kt,nr,It,Wt,sr,or,ar,le=class extends EventTarget{constructor(t,n,s){super();u(this,x);u(this,U,null);u(this,f,null);u(this,R,new Set);u(this,g,null);u(this,L,null);u(this,J,0);u(this,H,null);u(this,ft,0);u(this,l,null);u(this,c,null);u(this,V,null);u(this,Ct,null);u(this,ct,null);u(this,d,new rr);u(this,pt,null);u(this,It,t=>{if(!this.hasFocus)return;let n=!1,s=!1;if(e(this,L)!==e(this,f).focusNode&&(a(this,L,e(this,f).focusNode),n=!0),a(this,J,e(this,f).focusOffset),e(this,H)!==e(this,f).anchorNode&&(a(this,H,e(this,f).anchorNode),s=!0),a(this,ft,e(this,f).anchorOffset),e(this,f).rangeCount>1)for(let o=0;o0){let o=e(this,f).getRangeAt(0);a(this,g,o),e(this,R).clear(),e(this,R).add(o)}else a(this,g,null),e(this,R).clear();n&&y(this,x,Wt).call(this),e(this,ct)&&e(this,ct).update(this)});a(this,ct,s?.debug),a(this,pt,s?.styleDefaults),a(this,f,n),a(this,U,t),a(this,c,new er(e(this,U).element)),y(this,x,sr).call(this)}get currentStyle(){return e(this,V)}saveSelection(){return a(this,l,{isCollapsed:e(this,f).isCollapsed,focusNode:e(this,f).focusNode,focusOffset:e(this,f).focusOffset,anchorNode:e(this,f).anchorNode,anchorOffset:e(this,f).anchorOffset,range:y(this,x,or).call(this)}),!0}restoreSelection(){return e(this,l)?(e(this,l).anchorNode&&e(this,l).focusNode&&(e(this,l).anchorNode===e(this,l).focusNode?e(this,f).setPosition(e(this,l).focusNode,e(this,l).focusOffset):e(this,f).setBaseAndExtent(e(this,l).anchorNode,e(this,l).anchorOffset,e(this,l).focusNode,e(this,l).focusOffset)),a(this,l,null),!0):!1}startMutation(){return e(this,d).clear(),!!e(this,L)}endMutation(){return e(this,d)}selectAll(){return e(this,f).selectAllChildren(e(this,U).root),this}cursorToEnd(){let t=document.createRange();return t.selectNodeContents(e(this,U).element),t.collapse(!1),e(this,f).removeAllRanges(),e(this,f).addRange(t),this}collapse(t,n){let s=t.nodeType===Node.TEXT_NODE&&n>=t.nodeValue.length?t.nodeValue.length:n;return this.setSelection(t,s,t,s)}setSelection(t,n,s=t,o=n){if(!t.isConnected)throw new Error("Invalid anchorNode");if(!s.isConnected)throw new Error("Invalid focusNode");e(this,l)?(e(this,l).isCollapsed=s===t&&n===o,e(this,l).focusNode=s,e(this,l).focusOffset=o,e(this,l).anchorNode=t,e(this,l).anchorOffset=n,e(this,l).range.collapsed=e(this,l).isCollapsed,s.compareDocumentPosition(t)&Node.DOCUMENT_POSITION_FOLLOWING?(e(this,l).range.startContainer=s,e(this,l).range.startOffset=o,e(this,l).range.endContainer=t,e(this,l).range.endOffset=n):(e(this,l).range.startContainer=t,e(this,l).range.startOffset=n,e(this,l).range.endContainer=s,e(this,l).range.endOffset=o)):(a(this,H,t),a(this,ft,n),t===s?(a(this,L,e(this,H)),a(this,J,e(this,ft)),e(this,f).setPosition(t,n)):(a(this,L,s),a(this,J,o),e(this,f).setBaseAndExtent(t,n,s,o)))}dispose(){document.removeEventListener("selectionchange",e(this,It)),a(this,U,null),e(this,R).clear(),a(this,R,null),a(this,g,null),a(this,f,null),a(this,L,null),a(this,H,null),e(this,d).dispose(),a(this,d,null)}get selection(){return e(this,f)}get range(){return e(this,g)}get direction(){return this.isCollapsed?ut.NONE:this.focusNode!==this.anchorNode?this.startContainer===this.focusNode?ut.BACKWARD:ut.FORWARD:this.focusOffset1&&s.childNodes.length===0)return s.remove(),this.collapse(o,0);return this.collapse(this.focusNode,this.focusOffset)}removeBackwardText(){e(this,c).currentNode=this.focusNode;let t=qe(this.focusNode.nodeValue,this.focusOffset);if(this.focusNode.nodeValue!==t&&(this.focusNode.nodeValue=t),this.focusOffset-1>0)return this.collapse(this.focusNode,this.focusOffset-1);let n=this.focusParagraph;if(!n)throw new Error("Cannot find paragraph");let s=this.focusInline;if(!s)throw new Error("Cannot find inline");let o=e(this,c).previousNode();if(this.focusNode.nodeValue===""&&this.focusNode.remove(),n.children.length===1&&s.childNodes.length===0){let h=lt();return s.appendChild(h),this.collapse(h,0)}else if(n.children.length>1&&s.childNodes.length===0)return s.remove(),this.collapse(o,Be(o));return this.collapse(this.focusNode,this.focusOffset-1)}insertText(t){return this.focusNode.nodeValue=Ut(this.focusNode.nodeValue,this.focusOffset,t),e(this,d).update(this.focusInline),this.collapse(this.focusNode,this.focusOffset+t.length)}replaceText(t){let n=Math.min(this.anchorOffset,this.focusOffset),s=Math.max(this.anchorOffset,this.focusOffset);return this.focusNode.nodeValue=Je(this.focusNode.nodeValue,n,s,t),e(this,d).update(this.focusInline),this.collapse(this.focusNode,n+t.length)}replaceInlines(t){let n=this.focusParagraph;if(this.startInline===n.firstChild&&this.startOffset===0&&this.endInline===n.lastChild&&this.endOffset===n.lastChild.textContent.length){let s=new Text(t);return n.replaceChildren(D(s,this.anchorInline.style)),this.collapse(s,s.nodeValue.length)}return this.removeSelected(),this.focusNode.nodeValue=Ut(this.focusNode.nodeValue,this.focusOffset,t),this.collapse(this.focusNode,this.focusOffset+t.length)}replaceParagraphs(t){let n=this.focusParagraph;this.removeSelected(),this.focusNode.nodeValue=Ut(this.focusNode.nodeValue,this.focusOffset,t);for(let s of n.children)s.textContent===""&&s.remove()}insertParagraphAfter(){let t=this.focusParagraph,n=et(e(this,V));return t.after(n),e(this,d).update(t),e(this,d).add(n),this.collapse(n.firstChild.firstChild,0)}insertParagraphBefore(){let t=this.focusParagraph,n=et(e(this,V));return t.before(n),e(this,d).update(t),e(this,d).add(n),this.collapse(t.firstChild.firstChild,0)}splitParagraph(){let t=this.focusParagraph,n=Vt(this.focusParagraph,this.focusInline,e(this,J));return this.focusParagraph.after(n),e(this,d).update(t),e(this,d).add(n),this.collapse(n.firstChild.firstChild,0)}insertParagraph(){return this.isParagraphEnd?this.insertParagraphAfter():this.isParagraphStart?this.insertParagraphBefore():this.splitParagraph()}replaceWithParagraph(){let t=this.focusParagraph,n=this.focusInline;this.removeSelected();let s=Vt(t,n,this.focusOffset);t.after(s),e(this,d).update(t),e(this,d).add(s)}removeBackwardParagraph(){let t=this.focusParagraph.previousElementSibling;if(!t)return;let n=this.focusParagraph;n.remove();let s=t.children.length>1?t.lastElementChild:t.firstChild,o=N(s.firstChild)?0:s.firstChild.nodeValue.length;return e(this,d).remove(n),this.collapse(s.firstChild,o)}mergeBackwardParagraph(){let t=this.focusParagraph,n=this.focusParagraph.previousElementSibling;if(!n)return;let s=n.lastChild,o=Ae(s);return ke(n)?(n.replaceChildren(...t.children),s=n.firstChild,t.remove()):Bt(n,t),e(this,d).remove(t),e(this,d).update(n),this.collapse(s.firstChild,o)}mergeForwardParagraph(){let t=this.focusParagraph,n=this.focusParagraph.nextElementSibling;n&&(Bt(this.focusParagraph,n),e(this,d).update(t),e(this,d).remove(n))}removeForwardParagraph(){let t=this.focusParagraph.nextSibling;if(!t)return;let n=this.focusParagraph;n.remove();let s=t.firstChild,o=this.focusOffset;return e(this,d).remove(n),this.collapse(s.firstChild,o)}cleanUp(t,n){for(let s of n)s.textContent===""&&(s.remove(),e(this,d).remove(s));for(let s of t)s.children.length===0&&(s.remove(),e(this,d).remove(s))}removeSelected(t){if(this.isCollapsed)return;let n=new Set,s=new Set,o=yt(e(this,g).startContainer),h=yt(e(this,g).endContainer),I=e(this,g).startOffset,B=e(this,g).endOffset,_=null,S=null;if(o===h){e(this,c).currentNode=o,_=e(this,c).previousNode(),e(this,c).currentNode=o,S=e(this,c).nextNode();let E=P(o),b=F(o);n.add(E),s.add(b);let G=tr(o.nodeValue,I,B);if(G===""){let Z=lt();return E.replaceChildren(Z),this.collapse(Z,0)}return o.nodeValue=G,this.collapse(o,I)}e(this,c).currentNode=o;let v=P(o),q=F(o),K=P(h),ce=F(h);Tt.start();do{Tt.update();let E=e(this,c).currentNode,b=P(e(this,c).currentNode),G=F(e(this,c).currentNode),Z=!1;if(e(this,c).currentNode===o?I===0?Z=!0:E.nodeValue=E.nodeValue.slice(0,I):e(this,c).currentNode===h?N(h)||ne(h)&&B===h.nodeValue.length?Z=!0:E.nodeValue=E.nodeValue.slice(B):Z=!0,e(this,c).nextNode(),Z){if(E.remove(),E===o)continue;if(E===h)break;b.childNodes.length===0&&b.remove(),G!==q&&G.children.length===0&&G.remove()}if(E===h)break}while(e(this,c).currentNode);if(q!==ce){let E=Bt(q,ce);if(E.children.length===0){let b=tt(e(this,V));return E.appendChild(b),this.collapse(b.firstChild,0)}}if(v.childNodes.length===0&&K.childNodes.length>0)return v.remove(),this.collapse(h,0);if(v.childNodes.length>0&&K.childNodes.length===0)return K.remove(),this.collapse(o,I);if(v.childNodes.length===0&&K.childNodes.length===0){let E=v.previousElementSibling,b=K.nextElementSibling;if(v.remove(),K.remove(),E)return this.collapse(E.firstChild,E.firstChild.nodeValue.length);if(b)return this.collapse(b.firstChild,0);let G=tt(e(this,V));return q.appendChild(G),this.collapse(G.firstChild,0)}return this.collapse(o,I)}applyStyles(t){return y(this,x,ar).call(this,this.startContainer,this.startOffset,this.endContainer,this.endOffset,t)}};U=new WeakMap,f=new WeakMap,R=new WeakMap,g=new WeakMap,L=new WeakMap,J=new WeakMap,H=new WeakMap,ft=new WeakMap,l=new WeakMap,c=new WeakMap,V=new WeakMap,Ct=new WeakMap,ct=new WeakMap,d=new WeakMap,pt=new WeakMap,x=new WeakSet,ue=function(){if(e(this,pt))for(let[t,n]of Object.entries(e(this,pt)))e(this,V).setProperty(t,n+(t==="font-size"?"px":""))},kt=function(t){for(let n=0;n0){let t=e(this,f).getRangeAt(0);a(this,g,t),e(this,R).add(t)}if(e(this,f).rangeCount>1)for(let t=1;t0){let K=D(v,_.style);q.after(K)}this.setSelection(S,0,S,S.nodeValue.length)}else{let B=this.startParagraph;xt(B,h)}return y(this,x,Wt).call(this)}else if(t!==s){Tt.start();let B=yt(s);e(this,c).currentNode=yt(t);do{Tt.update();let _=F(e(this,c).currentNode);xt(_,h);let S=P(e(this,c).currentNode);if(e(this,c).currentNode===t&&n>0){let v=mt(S,n);gt(v,h),S.after(v)}else if(e(this,c).currentNode===t&&n===0||e(this,c).currentNode!==t&&e(this,c).currentNode!==s||e(this,c).currentNode===s&&o===s.nodeValue.length)gt(S,h);else if(e(this,c).currentNode===s&&or.addEventListener(n,s,t))}function fr(r,i){Object.entries(i).forEach(([t,n])=>r.removeEventListener(t,n))}var Sr={FULL:"full",PARTIAL:"partial"},St=Sr;var p,dt,O,k,m,W,wt,C,cr,pr,Ot,At,dr,fe,Gt,Xt,$t,Yt,zt,Ht,Kt,_t,vt=class extends EventTarget{constructor(t,n){super();u(this,C);u(this,p,null);u(this,dt,null);u(this,O,null);u(this,k,null);u(this,m,null);u(this,W,null);u(this,wt,null);u(this,Ot,t=>this.dispatchEvent(new t.constructor(t.type,t)));u(this,At,t=>{e(this,W).children.length>0&&y(this,C,fe).call(this),this.dispatchEvent(new t.constructor(t.type,t))});u(this,Gt,t=>{e(this,k).notifyImmediately(),e(this,m).saveSelection(),y(this,C,fe).call(this),this.dispatchEvent(new FocusEvent(t.type,t))});u(this,Xt,t=>{e(this,m).restoreSelection(),e(this,W)&&e(this,W).replaceChildren(),this.dispatchEvent(new FocusEvent(t.type,t))});u(this,$t,t=>bt.paste(t,this,e(this,m)));u(this,Yt,t=>bt.cut(t,this,e(this,m)));u(this,zt,t=>bt.copy(t,this,e(this,m)));u(this,Ht,t=>{if(!(t.inputType==="historyUndo"||t.inputType==="historyRedo")){if(!(t.inputType in Mt)){t.inputType!=="insertCompositionText"&&t.preventDefault();return}if(t.inputType in Mt){let n=Mt[t.inputType];if(!e(this,m).startMutation())return;n(t,this,e(this,m));let s=e(this,m).endMutation();y(this,C,_t).call(this,St.FULL,s)}}});u(this,Kt,t=>{t.inputType==="historyUndo"||t.inputType==="historyRedo"||t.inputType==="insertCompositionText"&&y(this,C,_t).call(this,St.FULL,null)});if(!(t instanceof HTMLElement))throw new TypeError("Invalid text editor element");a(this,p,t),a(this,W,n?.selectionImposterElement),a(this,dt,{blur:e(this,Gt),focus:e(this,Xt),paste:e(this,$t),cut:e(this,Yt),copy:e(this,zt),beforeinput:e(this,Ht),input:e(this,Kt)}),a(this,wt,n?.styleDefaults),y(this,C,dr).call(this,n)}get root(){return e(this,O)}set root(t){let n=e(this,O);a(this,O,t),n.replaceWith(t)}get element(){return e(this,p)}get isEmpty(){return e(this,O).children.length===1&&e(this,O).firstElementChild.children.length===1&&N(e(this,O).firstElementChild.firstElementChild.firstChild)}get numParagraphs(){return e(this,O).children.length}get currentStyle(){return e(this,m).currentStyle}focus(){return e(this,p).focus()}blur(){return e(this,p).blur()}createRoot(...t){return ie(...t)}createParagraph(...t){return X(...t)}createInlineFromString(t,n){return t===""?tt(n):D(new Text(t),n)}createInline(...t){return D(...t)}applyStylesToSelection(t){e(this,m).startMutation(),e(this,m).applyStyles(t);let n=e(this,m).endMutation();return y(this,C,_t).call(this,St.FULL,n),e(this,k).notifyImmediately(),this}selectAll(){return e(this,m).selectAll(),this}cursorToEnd(){return e(this,m).cursorToEnd(),this}dispose(){e(this,k).removeEventListener("change",e(this,Ot)),e(this,k).dispose(),a(this,k,null),e(this,m).removeEventListener("stylechange",e(this,At)),e(this,m).dispose(),a(this,m,null),fr(e(this,p),e(this,dt)),a(this,p,null),a(this,O,null)}};p=new WeakMap,dt=new WeakMap,O=new WeakMap,k=new WeakMap,m=new WeakMap,W=new WeakMap,wt=new WeakMap,C=new WeakSet,cr=function(){e(this,p).isContentEditable||(e(this,p).contentEditable="true",e(this,p).isContentEditable||e(this,p).setAttribute("contenteditable","true")),e(this,p).spellcheck&&(e(this,p).spellcheck=!1),e(this,p).autocapitalize&&(e(this,p).autocapitalize=!1),e(this,p).autofocus||(e(this,p).autofocus=!0),(!e(this,p).role||e(this,p).role!=="textbox")&&(e(this,p).role="textbox"),e(this,p).ariaAutoComplete&&(e(this,p).ariaAutoComplete=!1),e(this,p).ariaMultiLine||(e(this,p).ariaMultiLine=!0),e(this,p).dataset.itype="editor"},pr=function(){a(this,O,Le(e(this,wt))),e(this,p).appendChild(e(this,O))},Ot=new WeakMap,At=new WeakMap,dr=function(t){y(this,C,cr).call(this),y(this,C,pr).call(this),a(this,k,new Qe(this)),e(this,k).addEventListener("change",e(this,Ot)),a(this,m,new hr(this,document.getSelection(),t)),e(this,m).addEventListener("stylechange",e(this,At)),ur(e(this,p),e(this,dt),{capture:!0})},fe=function(){if(e(this,W)&&!e(this,m).isCollapsed){let t=e(this,m).range?.getClientRects();if(t){let n=e(this,W).getBoundingClientRect();e(this,W).replaceChildren(lr(n,t))}}},Gt=new WeakMap,Xt=new WeakMap,$t=new WeakMap,Yt=new WeakMap,zt=new WeakMap,Ht=new WeakMap,Kt=new WeakMap,_t=function(t=St.FULL,n){this.dispatchEvent(new CustomEvent("needslayout",{detail:{type:t,mutations:n}}))};function Dt(r){return r instanceof vt}function pn(r){return Dt(r)?r.root:null}function dn(r,i){return Dt(r)&&(r.root=i),r}function gn(r,i){return new vt(r,{...i})}function mn(r){if(Dt(r))return r.currentStyle}function yn(r,i){if(Dt(r))return r.applyStylesToSelection(i)}function xn(r){Dt(r)&&r.dispose()}var En=vt;export{vt as TextEditor,yn as applyStylesToSelection,gn as create,En as default,xn as dispose,mn as getCurrentStyle,pn as getRoot,Dt as isEditor,dn as setRoot}; -//# sourceMappingURL=text_editor_v2.js.map diff --git a/frontend/vendor/text_editor_v2.js.map b/frontend/vendor/text_editor_v2.js.map deleted file mode 100644 index ae55f1c87..000000000 --- a/frontend/vendor/text_editor_v2.js.map +++ /dev/null @@ -1,7 +0,0 @@ -{ - "version": 3, - "sources": ["../editor/clipboard/copy.js", "../editor/clipboard/cut.js", "../editor/content/dom/Color.js", "../editor/content/dom/Style.js", "../editor/content/dom/Element.js", "../editor/content/dom/LineBreak.js", "../editor/content/dom/Inline.js", "../editor/content/dom/Root.js", "../editor/content/dom/TextNode.js", "../editor/content/dom/Paragraph.js", "../editor/content/dom/Content.js", "../editor/clipboard/paste.js", "../editor/clipboard/index.js", "../editor/commands/insertText.js", "../editor/commands/insertParagraph.js", "../editor/commands/deleteByCut.js", "../editor/commands/deleteContentBackward.js", "../editor/commands/deleteContentForward.js", "../editor/commands/index.js", "../editor/controllers/ChangeController.js", "../editor/content/Text.js", "../editor/content/dom/TextNodeIterator.js", "../editor/commands/CommandMutations.js", "../editor/controllers/SelectionDirection.js", "../editor/controllers/SafeGuard.js", "../editor/controllers/SelectionController.js", "../editor/selection/Imposter.js", "../editor/Event.js", "../editor/layout/LayoutType.js", "../editor/TextEditor.js"], - "sourcesContent": ["/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * This event is called when the user copies a text from the\n * editor.\n *\n * TODO: We could transform `--fills` in here to CSS `color`, `background-image`,\n * etc. to be more compatible with other applications.\n *\n * @param {ClipboardEvent} event\n * @param {TextEditor} editor\n */\nexport function copy(event, editor) {}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * This event is called when the user copies a text from the\n * editor.\n *\n * TODO: We could transform `--fills` in here to CSS `color`, `background-image`,\n * etc. to be more compatible with other applications.\n *\n * @param {ClipboardEvent} event\n * @param {TextEditor} editor\n */\nexport function cut(event, editor) {}\n", "/**\n * Canvas used to retrieve colors as CSS hexadecimals.\n *\n * @type {OffscreenCanvas|HTMLCanvasElement}\n */\nlet canvas = null; // createCanvas(1, 1);\n\n/**\n * Context used to retrieve colors as CSS hexadecimals.\n *\n * @type {CanvasRenderingContext2D}\n */\nlet context = null; // canvas.getContext(\"2d\");\n\n/**\n * Returns the canvas context.\n *\n * @returns {CanvasRenderingContext2D}\n */\nfunction getContext() {\n if (!canvas) {\n canvas = createCanvas(1, 1);\n }\n if (!context) {\n context = canvas.getContext(\"2d\");\n }\n return context\n}\n\n/**\n * Creates a new canvas element.\n *\n * @param {number} width\n * @param {number} height\n * @returns {OffscreenCanvas|HTMLCanvasElement}\n */\nfunction createCanvas(width, height) {\n if (\"OffscreenCanvas\" in globalThis) {\n return new OffscreenCanvas(width, height);\n }\n return document.createElement(\"canvas\");\n}\n\n/**\n * Returns a byte representation as an hex.\n *\n * @param {number} byte\n * @returns {string}\n */\nexport function getByteAsHex(byte) {\n return byte.toString(16).padStart(2, \"0\");\n}\n\n/**\n * Returns a color definition from a fillStyle color.\n *\n * @param {string} fillStyle\n * @returns {[string, number]}\n */\nexport function getColor(fillStyle) {\n const context = getContext();\n context.fillStyle = fillStyle;\n context.fillRect(0, 0, 1, 1);\n const imageData = context.getImageData(0, 0, 1, 1);\n const [r, g, b, a] = imageData.data;\n return [`#${getByteAsHex(r)}${getByteAsHex(g)}${getByteAsHex(b)}`, a / 255.0];\n}\n\n/**\n * Returns a fill from a fillStyle color.\n *\n * @param {string} fillStyle\n * @returns {string}\n */\nexport function getFills(fillStyle) {\n const [color, opacity] = getColor(fillStyle);\n return `[[\"^ \",\"~:fill-color\",\"${color}\",\"~:fill-opacity\",${opacity}]]`;\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { getFills } from \"./Color\";\n\n/**\n * Merges two style declarations. `source` -> `target`.\n *\n * @param {CSSStyleDeclaration} target\n * @param {CSSStyleDeclaration} source\n * @returns {CSSStyleDeclaration}\n */\nexport function mergeStyleDeclarations(target, source) {\n // This is better but it doesn't work in JSDOM\n // for (const styleName of source) {\n for (let index = 0; index < source.length; index++) {\n const styleName = source.item(index);\n target.setProperty(styleName, source.getPropertyValue(styleName));\n }\n return target\n}\n\n/**\n * Computes the styles of an element the same way `window.getComputedStyle` does.\n *\n * @param {Element} element\n * @returns {CSSStyleDeclaration}\n */\nexport function getComputedStyle(element) {\n const inertElement = document.createElement(\"div\");\n let currentElement = element;\n while (currentElement) {\n // This is better but it doesn't work in JSDOM.\n // for (const styleName of currentElement.style) {\n for (let index = 0; index < currentElement.style.length; index++) {\n const styleName = currentElement.style.item(index);\n const currentValue = inertElement.style.getPropertyValue(styleName);\n if (currentValue) {\n const priority = currentElement.style.getPropertyPriority(styleName);\n if (priority === \"important\") {\n const newValue = currentElement.style.getPropertyValue(styleName);\n inertElement.style.setProperty(styleName, newValue);\n }\n } else {\n inertElement.style.setProperty(\n styleName,\n currentElement.style.getPropertyValue(styleName)\n );\n }\n }\n currentElement = currentElement.parentElement;\n }\n return inertElement.style;\n}\n\n/**\n * Normalizes style declaration.\n *\n * TODO: I think that this also needs to remove some \"conflicting\"\n * CSS properties like `font-family` or some CSS variables.\n *\n * @param {CSSStyleDeclaration} styleDeclaration\n * @returns {CSSStyleDeclaration}\n */\nexport function normalizeStyles(styleDeclaration) {\n // If there's a color property, we should convert it to\n // a --fills CSS variable property.\n const color = styleDeclaration.getPropertyValue(\"color\");\n if (color) {\n styleDeclaration.removeProperty(\"color\");\n styleDeclaration.setProperty(\"--fills\", getFills(color));\n }\n // If there's a font-family property and not a --font-id, then\n // we remove the font-family because it will not work.\n const fontFamily = styleDeclaration.getPropertyValue(\"font-family\");\n const fontId = styleDeclaration.getPropertyPriority(\"--font-id\");\n if (fontFamily && !fontId) {\n styleDeclaration.removeProperty(\"font-family\");\n }\n return styleDeclaration\n}\n/**\n * Sets a single style property value of an element.\n *\n * @param {HTMLElement} element\n * @param {string} styleName\n * @param {*} styleValue\n * @param {string} [styleUnit]\n * @returns {HTMLElement}\n */\nexport function setStyle(element, styleName, styleValue, styleUnit) {\n if (\n styleName.startsWith(\"--\") &&\n typeof styleValue !== \"string\" &&\n typeof styleValue !== \"number\"\n ) {\n if (styleName === \"--fills\" && styleValue === null) debugger;\n element.style.setProperty(styleName, JSON.stringify(styleValue));\n } else {\n element.style.setProperty(styleName, styleValue + (styleUnit ?? \"\"));\n }\n return element;\n}\n\n/**\n * Returns the value of a style from a declaration.\n *\n * @param {CSSStyleDeclaration} style\n * @param {string} styleName\n * @param {string|undefined} [styleUnit]\n * @returns {*}\n */\nexport function getStyleFromDeclaration(style, styleName, styleUnit) {\n if (styleName.startsWith(\"--\")) {\n return style.getPropertyValue(styleName);\n }\n const styleValue = style.getPropertyValue(styleName);\n if (styleValue.endsWith(styleUnit)) {\n return styleValue.slice(0, -styleUnit.length);\n }\n return styleValue;\n}\n\n/**\n * Returns the value of a style.\n *\n * @param {HTMLElement} element\n * @param {string} styleName\n * @param {string|undefined} [styleUnit]\n * @returns {*}\n */\nexport function getStyle(element, styleName, styleUnit) {\n return getStyleFromDeclaration(element.style, styleName, styleUnit);\n}\n\n/**\n * Sets the styles of an element using an object and a list of\n * allowed styles.\n *\n * @param {HTMLElement} element\n * @param {Array<[string,?string]>} allowedStyles\n * @param {Object.} styleObject\n * @returns {HTMLElement}\n */\nexport function setStylesFromObject(element, allowedStyles, styleObject) {\n for (const [styleName, styleUnit] of allowedStyles) {\n if (!(styleName in styleObject)) {\n continue;\n }\n const styleValue = styleObject[styleName];\n if (styleValue) {\n setStyle(element, styleName, styleValue, styleUnit);\n }\n }\n return element;\n}\n\n/**\n * Sets the styles of an element using a CSS Style Declaration and a list\n * of allowed styles.\n *\n * @param {HTMLElement} element\n * @param {Array<[string,?string]>} allowedStyles\n * @param {CSSStyleDeclaration} styleDeclaration\n * @returns {HTMLElement}\n */\nexport function setStylesFromDeclaration(\n element,\n allowedStyles,\n styleDeclaration\n) {\n for (const [styleName, styleUnit] of allowedStyles) {\n const styleValue = getStyleFromDeclaration(styleDeclaration, styleName, styleUnit);\n if (styleValue) {\n setStyle(element, styleName, styleValue, styleUnit);\n }\n }\n return element;\n}\n\n/**\n * Sets the styles of an element using an Object or a CSS Style Declaration and\n * a list of allowed styles.\n *\n * @param {HTMLElement} element\n * @param {Array<[string,?string]} allowedStyles\n * @param {Object.|CSSStyleDeclaration} styleObjectOrDeclaration\n * @returns {HTMLElement}\n */\nexport function setStyles(element, allowedStyles, styleObjectOrDeclaration) {\n if (styleObjectOrDeclaration instanceof CSSStyleDeclaration) {\n return setStylesFromDeclaration(\n element,\n allowedStyles,\n styleObjectOrDeclaration\n );\n }\n return setStylesFromObject(element, allowedStyles, styleObjectOrDeclaration);\n}\n\n/**\n * Gets the styles of an element using a list of allowed styles.\n *\n * @param {HTMLElement} element\n * @param {Array<[string,?string]} allowedStyles\n * @returns {Object.}\n */\nexport function getStyles(element, allowedStyles) {\n const styleObject = {};\n for (const [styleName, styleUnit] of allowedStyles) {\n const styleValue = getStyle(element, styleName, styleUnit);\n if (styleValue) {\n styleObject[styleName] = styleValue;\n }\n }\n return styleObject;\n}\n\n/**\n * Returns a series of merged styles.\n *\n * @param {Array<[string,?string]} allowedStyles\n * @param {CSSStyleDeclaration} styleDeclaration\n * @param {Object.} newStyles\n * @returns {Object.}\n */\nexport function mergeStyles(allowedStyles, styleDeclaration, newStyles) {\n const mergedStyles = {};\n for (const [styleName, styleUnit] of allowedStyles) {\n if (styleName in newStyles) {\n mergedStyles[styleName] = newStyles[styleName];\n } else {\n mergedStyles[styleName] = getStyleFromDeclaration(styleDeclaration, styleName, styleUnit);\n }\n }\n return mergedStyles;\n}\n\n/**\n * Returns true if the specified style declaration has a display block.\n *\n * @param {CSSStyleDeclaration} style\n * @returns {boolean}\n */\nexport function isDisplayBlock(style) {\n return style.display === \"block\";\n}\n\n/**\n * Returns true if the specified style declaration has a display inline\n * or inline-block.\n *\n * @param {CSSStyleDeclaration} style\n * @returns {boolean}\n */\nexport function isDisplayInline(style) {\n return style.display === \"inline\" || style.display === \"inline-block\";\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { setStyles } from \"./Style\";\n\n/**\n * @typedef {Object} CreateElementOptions\n * @property {Object.} [attributes]\n * @property {Object.} [data]\n * @property {Object.|CSSStyleDeclaration} [styles]\n * @property {Array<[string,?string]>} [allowedStyles]\n * @property {Array|Node} [children]\n */\n\n/**\n * Creates a new random id to identify content nodes.\n *\n * @returns {string}\n */\nexport function createRandomId() {\n return Math.floor(Math.random() * Number.MAX_SAFE_INTEGER).toString(36);\n}\n\n/**\n * Creates a new HTML element.\n *\n * @param {string} tag\n * @param {*} options\n * @returns {HTMLElement}\n */\nexport function createElement(tag, options) {\n const element = document.createElement(tag);\n if (options?.attributes) {\n Object.entries(options.attributes).forEach(([name, value]) =>\n element.setAttribute(name, value)\n );\n }\n if (options?.data) {\n Object.entries(options.data).forEach(\n ([name, value]) => (element.dataset[name] = value)\n );\n }\n if (options?.styles && options?.allowedStyles) {\n setStyles(element, options.allowedStyles, options.styles);\n }\n if (options?.children) {\n if (Array.isArray(options.children)) {\n element.append(...options.children);\n } else {\n element.appendChild(options.children);\n }\n }\n return element;\n}\n\n/**\n * Returns true if passed node is an element.\n *\n * @param {Node} element\n * @param {string} nodeName\n * @returns {boolean}\n */\nexport function isElement(element, nodeName) {\n return (\n element.nodeType === Node.ELEMENT_NODE &&\n element.nodeName === nodeName.toUpperCase()\n );\n}\n\n/**\n * Returns true if the specified offset is at the start of the element.\n *\n * @param {Node} node\n * @param {number} offset\n * @returns {boolean}\n */\nexport function isOffsetAtStart(node, offset) {\n return offset === 0;\n}\n\n/**\n * Returns true if the specified offset is at the end of the element.\n *\n * @param {Node} node\n * @param {number} offset\n * @returns {boolean}\n */\nexport function isOffsetAtEnd(node, offset) {\n if (node.nodeType === Node.TEXT_NODE) {\n return node.nodeValue.length === offset;\n }\n return true;\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nexport const TAG = \"BR\";\n\n/**\n * Creates a new line break.\n *\n * @returns {HTMLBRElement}\n */\nexport function createLineBreak() {\n return document.createElement(TAG);\n}\n\n/**\n * Returns true if the passed node is a line break.\n *\n * @param {Node} node\n * @returns {boolean}\n */\nexport function isLineBreak(node) {\n return node.nodeType === Node.ELEMENT_NODE && node.nodeName === TAG;\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport {\n createElement,\n isElement,\n isOffsetAtStart,\n isOffsetAtEnd,\n} from \"./Element\";\nimport { createLineBreak, isLineBreak } from \"./LineBreak\";\nimport { setStyles, mergeStyles } from \"./Style\";\nimport { createRandomId } from \"./Element\";\n\nexport const TAG = \"SPAN\";\nexport const TYPE = \"inline\";\nexport const QUERY = `[data-itype=\"${TYPE}\"]`;\nexport const STYLES = [\n [\"--typography-ref-id\"],\n [\"--typography-ref-file\"],\n [\"--font-id\"],\n [\"--font-variant-id\"],\n [\"--fills\"],\n [\"font-variant\"],\n [\"font-family\"],\n [\"font-size\", \"px\"],\n [\"font-weight\"],\n [\"font-style\"],\n [\"line-height\"],\n [\"letter-spacing\", \"px\"],\n [\"text-decoration\"],\n [\"text-transform\"],\n];\n\n/**\n * Returns true if passed node is an inline.\n *\n * @param {Node} node\n * @returns {boolean}\n */\nexport function isInline(node) {\n if (!node) return false;\n if (!isElement(node, TAG)) return false;\n if (node.dataset.itype !== TYPE) return false;\n return true;\n}\n\n/**\n * Returns true if the passed node \"behaves\" like an\n * inline.\n *\n * @param {Node} element\n * @returns {boolean}\n */\nexport function isLikeInline(element) {\n return element\n ? [\n \"A\",\n \"ABBR\",\n \"ACRONYM\",\n \"B\",\n \"BDO\",\n \"BIG\",\n \"BR\",\n \"BUTTON\",\n \"CITE\",\n \"CODE\",\n \"DFN\",\n \"EM\",\n \"I\",\n \"IMG\",\n \"INPUT\",\n \"KBD\",\n \"LABEL\",\n \"MAP\",\n \"OBJECT\",\n \"OUTPUT\",\n \"Q\",\n \"SAMP\",\n \"SCRIPT\",\n \"SELECT\",\n \"SMALL\",\n \"SPAN\",\n \"STRONG\",\n \"SUB\",\n \"SUP\",\n \"TEXTAREA\",\n \"TIME\",\n \"TT\",\n \"VAR\",\n ].includes(element.nodeName)\n : false;\n}\n\n/**\n * Creates a new Inline\n *\n * @param {Text|HTMLBRElement} text\n * @param {Object.|CSSStyleDeclaration} styles\n * @param {Object.} [attrs]\n * @returns {HTMLSpanElement}\n */\nexport function createInline(textOrLineBreak, styles, attrs) {\n if (\n !(textOrLineBreak instanceof HTMLBRElement) &&\n !(textOrLineBreak instanceof Text)\n ) {\n throw new TypeError(\"Invalid inline child\");\n }\n if (textOrLineBreak instanceof Text\n && textOrLineBreak.nodeValue.length === 0) {\n console.trace(\"nodeValue\", textOrLineBreak.nodeValue)\n throw new TypeError(\"Invalid inline child, cannot be an empty text\");\n }\n return createElement(TAG, {\n attributes: { id: createRandomId(), ...attrs },\n data: { itype: TYPE },\n styles: styles,\n allowedStyles: STYLES,\n children: textOrLineBreak,\n });\n}\n\n/**\n * Creates a new inline from an older inline. This only\n * merges styles from the older inline to the new inline.\n *\n * @param {HTMLSpanElement} inline\n * @param {Object.} textOrLineBreak\n * @param {Object.|CSSStyleDeclaration} styles\n * @param {Object.} [attrs]\n * @returns {HTMLSpanElement}\n */\nexport function createInlineFrom(inline, textOrLineBreak, styles, attrs) {\n return createInline(\n textOrLineBreak,\n mergeStyles(STYLES, inline.style, styles),\n attrs\n );\n}\n\n/**\n * Creates a new empty inline.\n *\n * @param {Object.|CSSStyleDeclaration} styles\n * @returns {HTMLSpanElement}\n */\nexport function createEmptyInline(styles) {\n return createInline(createLineBreak(), styles);\n}\n\n/**\n * Sets the inline styles.\n *\n * @param {HTMLSpanElement} element\n * @param {Object.|CSSStyleDeclaration} styles\n * @returns {HTMLSpanElement}\n */\nexport function setInlineStyles(element, styles) {\n return setStyles(element, STYLES, styles);\n}\n\n/**\n * Gets the closest inline from a node.\n *\n * @param {Node} node\n * @returns {HTMLElement|null}\n */\nexport function getInline(node) {\n if (!node) return null; // FIXME: Should throw?\n if (isInline(node)) return node;\n if (node.nodeType === Node.TEXT_NODE) {\n const inline = node?.parentElement;\n if (!inline) return null;\n if (!isInline(inline)) return null;\n return inline;\n }\n return node.closest(QUERY);\n}\n\n/**\n * Returns true if we are at the start offset\n * of an inline.\n *\n * NOTE: Only the first inline returns this as true\n *\n * @param {TextNode|HTMLBRElement} node\n * @param {number} offset\n * @returns {boolean}\n */\nexport function isInlineStart(node, offset) {\n const inline = getInline(node);\n if (!inline) return false;\n return isOffsetAtStart(inline, offset);\n}\n\n/**\n * Returns true if we are at the end offset\n * of an inline.\n *\n * @param {TextNode|HTMLBRElement} node\n * @param {number} offset\n * @returns {boolean}\n */\nexport function isInlineEnd(node, offset) {\n const inline = getInline(node);\n if (!inline) return false;\n return isOffsetAtEnd(inline.firstChild, offset);\n}\n\n/**\n * Splits an inline.\n *\n * @param {HTMLSpanElement} inline\n * @param {number} offset\n */\nexport function splitInline(inline, offset) {\n const textNode = inline.firstChild;\n const style = inline.style;\n const newTextNode = textNode.splitText(offset);\n return createInline(newTextNode, style);\n}\n\n/**\n * Returns all the inlines of a paragraph starting at\n * the specified inline.\n *\n * @param {HTMLSpanElement} startInline\n * @returns {Array}\n */\nexport function getInlinesFrom(startInline) {\n const inlines = [];\n let currentInline = startInline;\n let index = 0;\n while (currentInline) {\n if (index > 0) inlines.push(currentInline);\n currentInline = currentInline.nextElementSibling;\n index++;\n }\n return inlines;\n}\n\n/**\n * Returns the length of an inline.\n *\n * @param {HTMLElement} inline\n * @returns {number}\n */\nexport function getInlineLength(inline) {\n if (!isInline(inline)) throw new Error(\"Invalid inline\");\n if (isLineBreak(inline.firstChild)) return 0;\n return inline.firstChild.nodeValue.length;\n}\n\n/**\n * Merges two inlines.\n *\n * @param {HTMLSpanElement} a\n * @param {HTMLSpanElement} b\n * @returns {HTMLSpanElement}\n */\nexport function mergeInlines(a, b) {\n a.append(...b.childNodes);\n b.remove();\n // We need to normalize Text nodes.\n a.normalize();\n return a;\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { createElement, isElement } from \"./Element\";\nimport { createEmptyParagraph, isParagraph } from \"./Paragraph\";\nimport { setStyles } from \"./Style\";\nimport { createRandomId } from \"./Element\";\n\nexport const TAG = \"DIV\";\nexport const TYPE = \"root\";\nexport const QUERY = `[data-itype=\"${TYPE}\"]`;\nexport const STYLES = [[\"--vertical-align\"]];\n\n/**\n * Returns true if passed node is a root.\n *\n * @param {Node} node\n * @returns {boolean}\n */\nexport function isRoot(node) {\n if (!node) return false;\n if (!isElement(node, TAG)) return false;\n if (node.dataset.itype !== TYPE) return false;\n return true;\n}\n\n/**\n * Create a new root element\n *\n * @param {Array} paragraphs\n * @param {Object.|CSSStyleDeclaration} styles,\n * @param {Object.} [attrs]\n * @returns {HTMLDivElement}\n */\nexport function createRoot(paragraphs, styles, attrs) {\n if (!Array.isArray(paragraphs) || !paragraphs.every(isParagraph))\n throw new TypeError(\"Invalid root children\");\n\n return createElement(TAG, {\n attributes: { id: createRandomId(), ...attrs },\n data: { itype: TYPE },\n styles: styles,\n allowedStyles: STYLES,\n children: paragraphs,\n });\n}\n\n/**\n * Creates a new empty root element\n *\n * @param {Object.|CSSStyleDeclaration} styles\n */\nexport function createEmptyRoot(styles) {\n return createRoot([createEmptyParagraph(styles)], styles);\n}\n\n/**\n * Sets the root styles.\n *\n * @param {HTMLDivElement} element\n * @param {Object.|CSSStyleDeclaration} styles\n * @returns {HTMLDivElement}\n */\nexport function setRootStyles(element, styles) {\n return setStyles(element, STYLES, styles);\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { isInline } from \"./Inline\";\nimport { isLineBreak } from \"./LineBreak\";\nimport { isParagraph } from \"./Paragraph\";\nimport { isRoot } from \"./Root\";\n\n/**\n * Returns true if the node is \"like\"\n * text, this means that it is a Text\n * node or a
element.\n *\n * @param {Node} node\n * @returns {boolean}\n */\nexport function isTextNode(node) {\n if (!node) throw new TypeError(\"Invalid text node\");\n return node.nodeType === Node.TEXT_NODE\n || isLineBreak(node);\n}\n\n/**\n * Returns true if the text node is empty.\n *\n * @param {Node} node\n * @returns {boolean}\n */\nexport function isEmptyTextNode(node) {\n return node.nodeType === Node.TEXT_NODE\n && node.nodeValue === \"\";\n}\n\n/**\n * Returns the content length of the\n * node.\n *\n * @param {Node} node\n * @returns {number}\n */\nexport function getTextNodeLength(node) {\n if (!node) throw new TypeError(\"Invalid text node\");\n if (isLineBreak(node)) return 0;\n return node.nodeValue.length;\n}\n\n/**\n * Gets the closest text node.\n *\n * @param {Node} node\n * @returns {Node}\n */\nexport function getClosestTextNode(node) {\n if (isTextNode(node)) return node;\n if (isInline(node)) return node.firstChild;\n if (isParagraph(node)) return node.firstChild.firstChild;\n if (isRoot(node)) return node.firstChild.firstChild.firstChild;\n throw new Error(\"Cannot find a text node\");\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport {\n createElement,\n isElement,\n isOffsetAtStart,\n isOffsetAtEnd,\n} from \"./Element\";\nimport {\n isInline,\n isLikeInline,\n getInline,\n getInlinesFrom,\n createEmptyInline,\n isInlineEnd,\n splitInline,\n} from \"./Inline\";\nimport { isLineBreak } from \"./LineBreak\";\nimport { setStyles } from \"./Style\";\nimport { createRandomId } from \"./Element\";\nimport { isEmptyTextNode, isTextNode } from './TextNode';\n\nexport const TAG = \"DIV\";\nexport const TYPE = \"paragraph\";\nexport const QUERY = `[data-itype=\"${TYPE}\"]`;\nexport const STYLES = [\n [\"--typography-ref-id\"],\n [\"--typography-ref-file\"],\n [\"--font-id\"],\n [\"--font-variant-id\"],\n [\"--fills\"],\n [\"font-variant\"],\n [\"font-family\"],\n [\"font-size\", \"px\"],\n [\"font-weight\"],\n [\"font-style\"],\n [\"line-height\"],\n [\"letter-spacing\", \"px\"],\n [\"text-decoration\"],\n [\"text-transform\"],\n [\"text-align\"],\n [\"direction\"]\n];\n\n/**\n * Returns true if the passed node behaves like a paragraph.\n *\n * NOTE: This is mainly used in paste operations. Every element node\n * it's going to be treated as paragraph it\n *\n * @param {Node} element\n * @returns {boolean}\n */\nexport function isLikeParagraph(element) {\n return !isLikeInline(element);\n}\n\n/**\n * Returns true if we have an empty paragraph.\n *\n * @param {Node} element\n * @returns {boolean}\n */\nexport function isEmptyParagraph(element) {\n if (!isParagraph(element)) throw new TypeError(\"Invalid paragraph\");\n const inline = element.firstChild;\n if (!isInline(inline)) throw new TypeError(\"Invalid inline\");\n return isLineBreak(inline.firstChild);\n}\n\n/**\n * Returns true if passed node is a paragraph.\n *\n * @param {Node} node\n * @returns {boolean}\n */\nexport function isParagraph(node) {\n if (!node) return false;\n if (!isElement(node, TAG)) return false;\n if (node.dataset.itype !== TYPE) return false;\n return true;\n}\n\n/**\n * Creates a new paragraph.\n *\n * @param {Array} inlines\n * @param {Object.|CSSStyleDeclaration} styles\n * @param {Object.} [attrs]\n * @returns {HTMLDivElement}\n */\nexport function createParagraph(inlines, styles, attrs) {\n if (inlines && (!Array.isArray(inlines) || !inlines.every(isInline)))\n throw new TypeError(\"Invalid paragraph children\");\n return createElement(TAG, {\n attributes: { id: createRandomId(), ...attrs },\n data: { itype: TYPE },\n styles: styles,\n allowedStyles: STYLES,\n children: inlines,\n });\n}\n\n/**\n * Returns a new empty paragraph\n *\n * @param {Object.} styles\n * @returns {HTMLDivElement}\n */\nexport function createEmptyParagraph(styles) {\n return createParagraph([\n createEmptyInline(styles)\n ], styles);\n}\n\n/**\n * Sets the paragraph styles.\n *\n * @param {HTMLDivElement} element\n * @param {Object.|CSSStyleDeclaration} styles\n * @returns {HTMLDivElement}\n */\nexport function setParagraphStyles(element, styles) {\n return setStyles(element, STYLES, styles);\n}\n\n/**\n * Gets the closest paragraph from a node.\n *\n * @param {Text|HTMLBRElement} node\n * @returns {HTMLElement|null}\n */\nexport function getParagraph(node) {\n if (!node) return null;\n if (isParagraph(node)) return node;\n if (node.nodeType === Node.TEXT_NODE\n || isLineBreak(node)) {\n const paragraph = node?.parentElement?.parentElement;\n if (!paragraph) {\n return null;\n }\n if (!isParagraph(paragraph)) {\n return null;\n }\n return paragraph;\n }\n return node.closest(QUERY);\n}\n\n/**\n * Returns if the specified node and offset represents\n * the start of the paragraph.\n *\n * @param {Text|HTMLBRElement} node\n * @param {number} offset\n * @returns {boolean}\n */\nexport function isParagraphStart(node, offset) {\n const paragraph = getParagraph(node);\n if (!paragraph) throw new Error(\"Can't find the paragraph\");\n const inline = getInline(node);\n if (!inline) throw new Error(\"Can't find the inline\");\n return (\n paragraph.firstElementChild === inline &&\n isOffsetAtStart(inline.firstChild, offset)\n );\n}\n\n/**\n * Returns if the specified node and offset represents\n * the end of the paragraph.\n *\n * @param {Text|HTMLBRElement} node\n * @param {number} offset\n * @returns {boolean}\n */\nexport function isParagraphEnd(node, offset) {\n const paragraph = getParagraph(node);\n if (!paragraph) throw new Error(\"Cannot find the paragraph\");\n const inline = getInline(node);\n if (!inline) throw new Error(\"Cannot find the inline\");\n return (\n paragraph.lastElementChild === inline &&\n isOffsetAtEnd(inline.firstChild, offset)\n );\n}\n\n/**\n * Splits a paragraph.\n *\n * @param {HTMLDivElement} paragraph\n * @param {HTMLSpanElement} inline\n * @param {number} offset\n */\nexport function splitParagraph(paragraph, inline, offset) {\n const style = paragraph.style;\n if (isInlineEnd(inline, offset)) {\n const newParagraph = createParagraph(getInlinesFrom(inline), style);\n return newParagraph;\n }\n const newInline = splitInline(inline, offset);\n const newParagraph = createParagraph([newInline], style);\n return newParagraph;\n}\n\n/**\n * Splits a paragraph at a specified child node index\n *\n * @param {HTMLDivElement} paragraph\n * @param {number} startIndex\n */\nexport function splitParagraphAtNode(paragraph, startIndex) {\n const style = paragraph.style;\n const newParagraph = createParagraph(null, style);\n const newInlines = [];\n for (let index = startIndex; index < paragraph.children.length; index++) {\n newInlines.push(paragraph.children.item(index));\n }\n newParagraph.append(...newInlines);\n return newParagraph;\n}\n\n/**\n * Merges two paragraphs.\n *\n * @param {HTMLDivElement} a\n * @param {HTMLDivElement} b\n * @returns {HTMLDivElement}\n */\nexport function mergeParagraphs(a, b) {\n a.append(...b.children);\n b.remove();\n return a;\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { createInline } from \"./Inline\";\nimport {\n createEmptyParagraph,\n createParagraph,\n isLikeParagraph,\n} from \"./Paragraph\";\nimport { isDisplayBlock, normalizeStyles, getComputedStyle, mergeStyleDeclarations } from \"./Style\";\n\n/**\n * Maps any HTML into a valid content DOM element.\n *\n * @param {Document} document\n * @param {HTMLElement} root\n * @param {CSSStyleDeclaration} [styleDefaults]\n * @returns {DocumentFragment}\n */\nexport function mapContentFragmentFromDocument(document, root, styleDefaults) {\n const nodeIterator = document.createNodeIterator(root, NodeFilter.SHOW_TEXT);\n const fragment = document.createDocumentFragment();\n\n let currentParagraph = null;\n let currentNode = nodeIterator.nextNode();\n while (currentNode) {\n // We cannot call document.defaultView because it is `null`.\n const parentStyle = normalizeStyles(mergeStyleDeclarations(styleDefaults, getComputedStyle(currentNode.parentElement)));\n if (\n isDisplayBlock(currentNode.parentElement.style) ||\n isDisplayBlock(parentStyle) ||\n isLikeParagraph(currentNode.parentElement)\n ) {\n if (currentParagraph) {\n fragment.appendChild(currentParagraph);\n }\n currentParagraph = createParagraph(undefined, parentStyle);\n } else {\n if (currentParagraph === null) {\n currentParagraph = createParagraph();\n }\n }\n\n currentParagraph.appendChild(\n createInline(new Text(currentNode.nodeValue), parentStyle)\n );\n\n currentNode = nodeIterator.nextNode();\n }\n\n fragment.appendChild(currentParagraph);\n return fragment;\n}\n\n/**\n * Maps any HTML into a valid content DOM element.\n *\n * @param {string} html\n * @param {CSSStyleDeclaration} [styleDefaults]\n * @returns {DocumentFragment}\n */\nexport function mapContentFragmentFromHTML(html, styleDefaults) {\n const parser = new DOMParser();\n const htmlDocument = parser.parseFromString(html, \"text/html\");\n return mapContentFragmentFromDocument(\n htmlDocument,\n htmlDocument.documentElement,\n styleDefaults\n );\n}\n\n/**\n * Maps a plain text into a valid content DOM element.\n *\n * @param {string} string\n * @param {CSSStyleDeclaration} [styleDefaults]\n * @returns {DocumentFragment}\n */\nexport function mapContentFragmentFromString(string, styleDefaults) {\n const lines = string.replace(/\\r/g, \"\").split(\"\\n\");\n const fragment = document.createDocumentFragment();\n for (const line of lines) {\n if (line === \"\") {\n fragment.appendChild(createEmptyParagraph(styleDefaults));\n } else {\n fragment.appendChild(createParagraph([createInline(new Text(line), styleDefaults)], styleDefaults));\n }\n }\n return fragment;\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { mapContentFragmentFromHTML, mapContentFragmentFromString } from '~/editor/content/dom/Content';\n\n/**\n * When the user pastes some HTML, what we do is generate\n * a new DOM based on what the user pasted and then we\n * insert it in the appropiate part (see `insertFromPaste` command).\n *\n * @param {ClipboardEvent} event\n * @param {TextEditor} editor\n * @param {SelectionController} selectionController\n * @returns {void}\n */\nexport function paste(event, editor, selectionController) {\n // We need to prevent default behavior\n // because we don't allow any HTML to\n // be pasted.\n event.preventDefault();\n\n let fragment = null;\n if (event.clipboardData.types.includes(\"text/html\")) {\n const html = event.clipboardData.getData(\"text/html\");\n fragment = mapContentFragmentFromHTML(html, selectionController.currentStyle);\n } else if (event.clipboardData.types.includes(\"text/plain\")) {\n const plain = event.clipboardData.getData(\"text/plain\");\n fragment = mapContentFragmentFromString(plain, selectionController.currentStyle);\n }\n\n if (!fragment) {\n return;\n }\n\n if (selectionController.isCollapsed) {\n selectionController.insertPaste(fragment);\n } else {\n selectionController.replaceWithPaste(fragment);\n }\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { copy } from \"./copy.js\";\nimport { cut } from \"./cut.js\";\nimport { paste } from \"./paste.js\";\n\nexport default {\n copy,\n cut,\n paste,\n};\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Insert typed plain text\n *\n * @see https://w3c.github.io/input-events/#interface-InputEvent\n * @param {InputEvent} event\n * @param {TextEditor} editor\n * @param {SelectionController} selectionController\n */\nexport function insertText(event, editor, selectionController) {\n event.preventDefault();\n if (selectionController.isCollapsed) {\n if (selectionController.isTextFocus) {\n return selectionController.insertText(event.data);\n } else if (selectionController.isLineBreakFocus) {\n return selectionController.replaceLineBreak(event.data);\n }\n } else {\n if (selectionController.isMultiParagraph) {\n return selectionController.replaceParagraphs(event.data);\n } else if (selectionController.isMultiInline) {\n return selectionController.replaceInlines(event.data);\n } else if (selectionController.isTextSame) {\n return selectionController.replaceText(event.data);\n }\n }\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Insert a paragraph\n *\n * @see https://w3c.github.io/input-events/#interface-InputEvent\n * @param {InputEvent} event\n * @param {TextEditor} editor\n * @param {SelectionController} selectionController\n */\nexport function insertParagraph(event, editor, selectionController) {\n event.preventDefault();\n if (selectionController.isCollapsed) {\n return selectionController.insertParagraph();\n }\n return selectionController.replaceWithParagraph();\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Remove the current selection as part of a cut.\n *\n * @param {InputEvent} event\n * @param {TextEditor} editor\n * @param {SelectionController} selectionController\n */\nexport function deleteByCut(event, editor, selectionController) {\n event.preventDefault();\n if (selectionController.isCollapsed) {\n throw new Error(\"This should be impossible\");\n }\n return selectionController.removeSelected();\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * delete the content directly before the caret position and this intention is\n * not covered by another `inputType` or delete the selection with the\n * selection collapsing to its start after the deletion.\n *\n * @param {InputEvent} event\n * @param {TextEditor} editor\n * @param {SelectionController} selectionController\n */\nexport function deleteContentBackward(event, editor, selectionController) {\n event.preventDefault();\n // If the editor is empty this is a no op.\n if (editor.isEmpty) return;\n\n // If not is collapsed AKA is a selection, then\n // we removeSelected.\n if (!selectionController.isCollapsed) {\n return selectionController.removeSelected({ direction: 'backward' });\n }\n\n // If we're in a text node and the offset is\n // greater than 0 (not at the start of the inline)\n // we simple remove a character from the text.\n if (selectionController.isTextFocus && selectionController.focusOffset > 0) {\n return selectionController.removeBackwardText();\n\n // If we're in a text node but we're at the end of the\n // paragraph, we should merge the current paragraph\n // with the following paragraph.\n } else if (\n selectionController.isTextFocus &&\n selectionController.focusAtStart\n ) {\n return selectionController.mergeBackwardParagraph();\n\n // If we're at an inline or a line break paragraph\n // and there's more than one paragraph, then we should\n // remove the next paragraph.\n } else if (\n selectionController.isInlineFocus ||\n selectionController.isLineBreakFocus\n ) {\n return selectionController.removeBackwardParagraph();\n }\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * delete the content directly after the caret position and this intention is not covered by\n * another inputType or delete the selection with the selection collapsing to its end after the deletion\n *\n * @param {InputEvent} event\n * @param {TextEditor} editor\n * @param {SelectionController} selectionController\n */\nexport function deleteContentForward(event, editor, selectionController) {\n event.preventDefault();\n // If the editor is empty this is a no op.\n if (editor.isEmpty) return;\n\n // If not is collapsed AKA is a selection, then\n // we removeSelected.\n if (!selectionController.isCollapsed) {\n return selectionController.removeSelected({ direction: \"forward\" });\n }\n\n // If we're in a text node and the offset is\n // greater than 0 (not at the start of the inline)\n // we simple remove a character from the text.\n if (\n selectionController.isTextFocus &&\n selectionController.focusOffset >= 0\n ) {\n return selectionController.removeForwardText();\n\n // If we're in a text node but we're at the end of the\n // paragraph, we should merge the current paragraph\n // with the following paragraph.\n } else if (\n selectionController.isTextFocus &&\n selectionController.focusAtEnd\n ) {\n return selectionController.mergeForwardParagraph();\n\n // If we're at an inline or a line break paragraph\n // and there's more than one paragraph, then we should\n // remove the next paragraph.\n } else if (\n (selectionController.isInlineFocus ||\n selectionController.isLineBreakFocus) &&\n editor.numParagraphs > 1\n ) {\n return selectionController.removeForwardParagraph();\n }\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { insertText } from \"./insertText.js\";\nimport { insertParagraph } from \"./insertParagraph.js\";\nimport { deleteByCut } from \"./deleteByCut.js\";\nimport { deleteContentBackward } from \"./deleteContentBackward.js\";\nimport { deleteContentForward } from \"./deleteContentForward.js\";\n\nexport default {\n insertText,\n insertParagraph,\n deleteByCut,\n deleteContentBackward,\n deleteContentForward,\n};\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Change controller is responsible of notifying when a change happens.\n */\nexport class ChangeController extends EventTarget {\n /**\n * Keeps the timeout id.\n *\n * @type {number}\n */\n #timeout = null;\n\n /**\n * Keeps the time at which we're going to\n * call the debounced change calls.\n *\n * @type {number}\n */\n #time = 1000;\n\n /**\n * Keeps if we have some pending changes or not.\n *\n * @type {boolean}\n */\n #hasPendingChanges = false;\n\n /**\n * Constructor\n *\n * @param {number} [time=500]\n */\n constructor(time = 500) {\n super()\n if (typeof time === \"number\" && (!Number.isInteger(time) || time <= 0)) {\n throw new TypeError(\"Invalid time\");\n }\n this.#time = time ?? 500;\n }\n\n /**\n * Indicates that there are some pending changes.\n *\n * @type {boolean}\n */\n get hasPendingChanges() {\n return this.#hasPendingChanges;\n }\n\n #onTimeout = () => {\n this.dispatchEvent(new Event(\"change\"));\n };\n\n /**\n * Tells the ChangeController that a change has been made\n * but that you need to delay the notification (and debounce)\n * for sometime.\n */\n notifyDebounced() {\n this.#hasPendingChanges = true;\n clearTimeout(this.#timeout);\n this.#timeout = setTimeout(this.#onTimeout, this.#time);\n }\n\n /**\n * Tells the ChangeController that a change should be notified\n * immediately.\n */\n notifyImmediately() {\n clearTimeout(this.#timeout);\n this.#onTimeout();\n }\n\n /**\n * Disposes the referenced resources.\n */\n dispose() {\n if (this.hasPendingChanges) {\n this.notifyImmediately();\n }\n clearTimeout(this.#timeout);\n }\n}\n\nexport default ChangeController;\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Throws if the passed value is not a valid offset value.\n *\n * @param {*} offset\n * @throws {TypeError}\n */\nfunction tryOffset(offset) {\n if (!Number.isInteger(offset) || offset < 0)\n throw new TypeError(\"Invalid offset\");\n}\n\n/**\n * Throws if the passed value is not a valid string.\n *\n * @param {*} str\n * @throws {TypeError}\n */\nfunction tryString(str) {\n if (typeof str !== \"string\") throw new TypeError(\"Invalid string\");\n}\n\n/**\n * Inserts string into a string.\n *\n * @param {string} str\n * @param {number} offset\n * @param {string} text\n * @returns {string}\n */\nexport function insertInto(str, offset, text) {\n tryString(str);\n tryOffset(offset);\n tryString(text);\n return str.slice(0, offset) + text + str.slice(offset);\n}\n\n/**\n * Replaces a part of a string with a string.\n *\n * @param {string} str\n * @param {number} startOffset\n * @param {number} endOffset\n * @param {string} text\n * @returns {string}\n */\nexport function replaceWith(str, startOffset, endOffset, text) {\n tryString(str);\n tryOffset(startOffset);\n tryOffset(endOffset);\n tryString(text);\n return str.slice(0, startOffset) + text + str.slice(endOffset);\n}\n\n/**\n * Removes text backward from specified offset.\n *\n * @param {string} str\n * @param {number} offset\n * @returns {string}\n */\nexport function removeBackward(str, offset) {\n tryString(str);\n tryOffset(offset);\n if (offset === 0) {\n return str;\n }\n return str.slice(0, offset - 1) + str.slice(offset);\n}\n\n/**\n * Removes text forward from specified offset.\n *\n * @param {string} str\n * @param {number} offset\n * @returns {string}\n */\nexport function removeForward(str, offset) {\n tryString(str);\n tryOffset(offset);\n return str.slice(0, offset) + str.slice(offset + 1);\n}\n\n/**\n * Removes a slice of text.\n *\n * @param {string} str\n * @param {number} start\n * @param {number} end\n * @returns {string}\n */\nexport function removeSlice(str, start, end) {\n tryString(str);\n tryOffset(start);\n tryOffset(end);\n return str.slice(0, start) + str.slice(end);\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Iterator direction.\n *\n * @enum {number}\n */\nexport const TextNodeIteratorDirection = {\n FORWARD: 1,\n BACKWARD: 0,\n};\n\n/**\n * TextNodeIterator\n */\nexport class TextNodeIterator {\n /**\n * Returns if a specific node is a text node.\n *\n * @param {Node} node\n * @returns {boolean}\n */\n static isTextNode(node) {\n return (\n node.nodeType === Node.TEXT_NODE ||\n (node.nodeType === Node.ELEMENT_NODE && node.nodeName === \"BR\")\n );\n }\n\n /**\n * Returns if a specific node is a container node.\n *\n * @param {Node} node\n * @returns {boolean}\n */\n static isContainerNode(node) {\n return node.nodeType === Node.ELEMENT_NODE && node.nodeName !== \"BR\";\n }\n\n /**\n * Finds a node from an initial node and down the tree.\n *\n * @param {Node} startNode\n * @param {Node} rootNode\n * @param {Set} skipNodes\n * @param {number} direction\n * @returns {Node}\n */\n static findDown(\n startNode,\n rootNode,\n skipNodes = new Set(),\n direction = TextNodeIteratorDirection.FORWARD\n ) {\n if (startNode === rootNode) {\n return TextNodeIterator.findDown(\n direction === TextNodeIteratorDirection.FORWARD\n ? startNode.firstChild\n : startNode.lastChild,\n rootNode,\n skipNodes,\n direction\n );\n }\n\n // NOTE: This should not use the SafeGuard\n // module.\n let safeGuard = Date.now();\n let currentNode = startNode;\n while (currentNode) {\n if (Date.now() - safeGuard >= 1000) {\n throw new Error(\"Iteration timeout\");\n }\n if (skipNodes.has(currentNode)) {\n currentNode =\n direction === TextNodeIteratorDirection.FORWARD\n ? currentNode.nextSibling\n : currentNode.previousSibling;\n continue;\n }\n if (TextNodeIterator.isTextNode(currentNode)) {\n return currentNode;\n } else if (TextNodeIterator.isContainerNode(currentNode)) {\n return TextNodeIterator.findDown(\n direction === TextNodeIteratorDirection.FORWARD\n ? currentNode.firstChild\n : currentNode.lastChild,\n rootNode,\n skipNodes,\n direction\n );\n }\n currentNode =\n direction === TextNodeIteratorDirection.FORWARD\n ? currentNode.nextSibling\n : currentNode.previousSibling;\n }\n return null;\n }\n\n /**\n * Finds a node from an initial node and up the tree.\n *\n * @param {Node} startNode\n * @param {Node} rootNode\n * @param {Set} backTrack\n * @param {number} direction\n * @returns {Node}\n */\n static findUp(\n startNode,\n rootNode,\n backTrack = new Set(),\n direction = TextNodeIteratorDirection.FORWARD\n ) {\n backTrack.add(startNode);\n if (TextNodeIterator.isTextNode(startNode)) {\n return TextNodeIterator.findUp(\n startNode.parentNode,\n rootNode,\n backTrack,\n direction\n );\n } else if (TextNodeIterator.isContainerNode(startNode)) {\n const found = TextNodeIterator.findDown(\n startNode,\n rootNode,\n backTrack,\n direction\n );\n if (found) {\n return found;\n }\n if (startNode !== rootNode) {\n return TextNodeIterator.findUp(\n startNode.parentNode,\n rootNode,\n backTrack,\n direction\n );\n }\n }\n return null;\n }\n\n /**\n * This is the root text node.\n *\n * @type {HTMLElement}\n */\n #rootNode = null;\n\n /**\n * This is the current text node.\n *\n * @type {Text|null}\n */\n #currentNode = null;\n\n /**\n * Constructor\n *\n * @param {HTMLElement} rootNode\n */\n constructor(rootNode) {\n if (!(rootNode instanceof HTMLElement)) {\n throw new TypeError(\"Invalid root node\");\n }\n this.#rootNode = rootNode;\n this.#currentNode = TextNodeIterator.findDown(rootNode, rootNode);\n }\n\n /**\n * Current node we're into.\n *\n * @type {TextNode|HTMLBRElement}\n */\n get currentNode() {\n return this.#currentNode;\n }\n\n set currentNode(newCurrentNode) {\n const isContained =\n (newCurrentNode.compareDocumentPosition(this.#rootNode) &\n Node.DOCUMENT_POSITION_CONTAINS) ===\n Node.DOCUMENT_POSITION_CONTAINS;\n if (\n !(newCurrentNode instanceof Node) ||\n !TextNodeIterator.isTextNode(newCurrentNode) ||\n !isContained\n ) {\n throw new TypeError(\"Invalid new current node\");\n }\n this.#currentNode = newCurrentNode;\n }\n\n /**\n * Returns the next Text node or
element or null if there are.\n *\n * @returns {Text|HTMLBRElement}\n */\n nextNode() {\n if (!this.#currentNode) return null;\n\n const nextNode = TextNodeIterator.findUp(\n this.#currentNode,\n this.#rootNode,\n new Set(),\n TextNodeIteratorDirection.FORWARD\n );\n\n if (!nextNode) {\n return null;\n }\n\n this.#currentNode = nextNode;\n return this.#currentNode;\n }\n\n /**\n * Returns the previous Text node or
element or null.\n *\n * @returns {Text|HTMLBRElement}\n */\n previousNode() {\n if (!this.#currentNode) return null;\n\n const previousNode = TextNodeIterator.findUp(\n this.#currentNode,\n this.#rootNode,\n new Set(),\n TextNodeIteratorDirection.BACKWARD\n );\n\n if (!previousNode) {\n return null;\n }\n\n this.#currentNode = previousNode;\n return this.#currentNode;\n }\n}\n\nexport default TextNodeIterator;\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Command mutations\n */\nexport class CommandMutations {\n #added = new Set();\n #removed = new Set();\n #updated = new Set();\n\n constructor(added, updated, removed) {\n if (added && Array.isArray(added)) this.#added = new Set(added);\n if (updated && Array.isArray(updated)) this.#updated = new Set(updated);\n if (removed && Array.isArray(removed)) this.#removed = new Set(removed);\n }\n\n get added() {\n return this.#added;\n }\n\n get removed() {\n return this.#removed;\n }\n\n get updated() {\n return this.#updated;\n }\n\n clear() {\n this.#added.clear();\n this.#removed.clear();\n this.#updated.clear();\n }\n\n dispose() {\n this.#added.clear();\n this.#added = null;\n this.#removed.clear();\n this.#removed = null;\n this.#updated.clear();\n this.#updated = null;\n }\n\n add(node) {\n this.#added.add(node);\n return this;\n }\n\n remove(node) {\n this.#removed.add(node);\n return this;\n }\n\n update(node) {\n this.#updated.add(node);\n return this;\n }\n}\n\nexport default CommandMutations;\n", "/**\n * Indicates the direction of the selection.\n *\n * @readonly\n * @enum {number}\n */\nexport const SelectionDirection = {\n /** The anchorNode is behind the focusNode */\n FORWARD: 1,\n /** The focusNode and the anchorNode are collapsed */\n NONE: 0,\n /** The focusNode is behind the anchorNode */\n BACKWARD: -1,\n};\n\nexport default SelectionDirection;\n", "/**\n * Max. amount of time we should allow.\n *\n * @type {number}\n */\nconst SAFE_GUARD_TIME = 1000;\n\n/**\n * Time at which the safeguard started.\n *\n * @type {number}\n */\nlet startTime = Date.now();\n\n/**\n * Marks the start of the safeguard.\n */\nexport function start() {\n startTime = Date.now();\n}\n\n/**\n * Checks if the safeguard should throw.\n */\nexport function update() {\n if (Date.now - startTime >= SAFE_GUARD_TIME) {\n throw new Error('Safe guard timeout');\n }\n}\n\nexport default {\n start,\n update,\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport { createLineBreak, isLineBreak } from \"~/editor/content/dom/LineBreak\";\nimport {\n createInline,\n createInlineFrom,\n getInline,\n getInlineLength,\n isInline,\n isInlineStart,\n isInlineEnd,\n setInlineStyles,\n mergeInlines,\n splitInline,\n createEmptyInline,\n} from \"~/editor/content/dom/Inline\";\nimport {\n createEmptyParagraph,\n isEmptyParagraph,\n getParagraph,\n isParagraph,\n isParagraphStart,\n isParagraphEnd,\n setParagraphStyles,\n splitParagraph,\n splitParagraphAtNode,\n mergeParagraphs,\n} from \"~/editor/content/dom/Paragraph\";\nimport {\n removeBackward,\n removeForward,\n replaceWith,\n insertInto,\n removeSlice,\n} from \"~/editor/content/Text\";\nimport { getTextNodeLength, getClosestTextNode, isTextNode } from \"~/editor/content/dom/TextNode\";\nimport TextNodeIterator from \"~/editor/content/dom/TextNodeIterator\";\nimport TextEditor from \"~/editor/TextEditor\";\nimport CommandMutations from \"~/editor/commands/CommandMutations\";\nimport { setRootStyles } from \"~/editor/content/dom/Root\";\nimport { SelectionDirection } from \"./SelectionDirection\";\nimport SafeGuard from './SafeGuard';\n\nconst SAFE_GUARD = true;\nconst SAFE_GUARD_TIME = true;\n\n/**\n * Supported options for the SelectionController.\n *\n * @typedef {Object} SelectionControllerOptions\n * @property {Object} [debug] An object with references to DOM elements that will keep all the debugging values.\n */\n\n/**\n * SelectionController uses the same concepts used by the Selection API but extending it to support\n * our own internal model based on paragraphs (in drafconst textEditorMock = TextEditorMock.createTextEditorMockWithParagraphs([\n createParagraph([createInline(new Text(\"Hello, \"))]),\n createEmptyParagraph(),\n createParagraph([createInline(new Text(\"World!\"))]),\n ]);\n const root = textEditorMock.root;\n const selection = document.getSelection();\n const selectionController = new SelectionController(\n textEditorMock,\n selection\n );\n focus(\n selection,\n textEditorMock,\n root.childNodes.item(2).firstChild.firstChild,\n 0\n );\n selectionController.mergeBackwardParagraph();\n expect(textEditorMock.root).toBeInstanceOf(HTMLDivElement);\n expect(textEditorMock.root.children.length).toBe(2);\n expect(textEditorMock.root.dataset.itype).toBe(\"root\");\n expect(textEditorMock.root.firstChild).toBeInstanceOf(HTMLDivElement);\n expect(textEditorMock.root.firstChild.dataset.itype).toBe(\"paragraph\");\n expect(textEditorMock.root.firstChild.firstChild).toBeInstanceOf(\n HTMLSpanElement\n );\n expect(textEditorMock.root.firstChild.firstChild.dataset.itype).toBe(\n \"inline\"\n );\n expect(textEditorMock.root.textContent).toBe(\"Hello, World!\");\n expect(textEditorMock.root.firstChild.textContent).toBe(\"Hello, \");\n expect(textEditorMock.root.lastChild.textContent).toBe(\"World!\");\n t.js they were called blocks) and inlines.\n */\nexport class SelectionController extends EventTarget {\n /**\n * Reference to the text editor.\n *\n * @type {TextEditor}\n */\n #textEditor = null;\n\n /**\n * Selection.\n *\n * @type {Selection}\n */\n #selection = null;\n\n /**\n * Set of ranges (this should always have one)\n *\n * @type {Set}\n */\n #ranges = new Set();\n\n /**\n * Current range (.rangeAt 0)\n *\n * @type {Range}\n */\n #range = null;\n\n /**\n * @type {Node}\n */\n #focusNode = null;\n\n /**\n * @type {number}\n */\n #focusOffset = 0;\n\n /**\n * @type {Node}\n */\n #anchorNode = null;\n\n /**\n * @type {number}\n */\n #anchorOffset = 0;\n\n /**\n * Saved selection.\n *\n * @type {object}\n */\n #savedSelection = null;\n\n /**\n * TextNodeIterator that allows us to move\n * around the root element but only through\n *
and #text nodes.\n *\n * @type {TextNodeIterator}\n */\n #textNodeIterator = null;\n\n /**\n * CSSStyleDeclaration that we can mutate\n * to handle style changes.\n *\n * @type {CSSStyleDeclaration}\n */\n #currentStyle = null;\n\n /**\n * Element used to have a custom CSSStyleDeclaration\n * that we can modify to handle style changes when the\n * selection is changed.\n *\n * @type {HTMLDivElement}\n */\n #inertElement = null;\n\n /**\n * @type {SelectionControllerDebug}\n */\n #debug = null;\n\n /**\n * Command Mutations.\n *\n * @type {CommandMutations}\n */\n #mutations = new CommandMutations();\n\n /**\n * Style defaults.\n *\n * @type {Object.}\n */\n #styleDefaults = null;\n\n /**\n * Constructor\n *\n * @param {TextEditor} textEditor\n * @param {Selection} selection\n * @param {SelectionControllerOptions} [options]\n */\n constructor(textEditor, selection, options) {\n super();\n // FIXME: We can't check if it is an instanceof TextEditor\n // because tests use TextEditorMock.\n /*\n if (!(textEditor instanceof TextEditor)) {\n throw new TypeError(\"Invalid EventTarget\");\n }\n */\n this.#debug = options?.debug;\n this.#styleDefaults = options?.styleDefaults;\n this.#selection = selection;\n this.#textEditor = textEditor;\n this.#textNodeIterator = new TextNodeIterator(this.#textEditor.element);\n\n // Setups everything.\n this.#setup();\n }\n\n /**\n * Styles of the current inline.\n *\n * @type {CSSStyleDeclaration}\n */\n get currentStyle() {\n return this.#currentStyle;\n }\n\n /**\n * Applies the default styles to the currentStyle\n * CSSStyleDeclaration.\n */\n #applyDefaultStylesToCurrentStyle() {\n if (this.#styleDefaults) {\n for (const [name, value] of Object.entries(this.#styleDefaults)) {\n this.#currentStyle.setProperty(\n name,\n value + (name === \"font-size\" ? \"px\" : \"\")\n );\n }\n }\n }\n\n /**\n * Applies some styles to the currentStyle\n * CSSStyleDeclaration\n *\n * @param {HTMLElement} element\n */\n #applyStylesToCurrentStyle(element) {\n for (let index = 0; index < element.style.length; index++) {\n const styleName = element.style.item(index);\n const styleValue = element.style.getPropertyValue(styleName);\n this.#currentStyle.setProperty(styleName, styleValue);\n }\n }\n\n /**\n * Updates current styles based on the currently selected inline.\n *\n * @param {HTMLSpanElement} inline\n * @returns {SelectionController}\n */\n #updateCurrentStyle(inline) {\n this.#applyDefaultStylesToCurrentStyle();\n const root = inline.parentElement.parentElement;\n this.#applyStylesToCurrentStyle(root);\n const paragraph = inline.parentElement;\n this.#applyStylesToCurrentStyle(paragraph);\n this.#applyStylesToCurrentStyle(inline);\n return this;\n }\n\n /**\n * This is called on every `selectionchange` because it is dispatched\n * only by the `document` object.\n *\n * @param {Event} e\n */\n #onSelectionChange = (e) => {\n // If we're outside the contenteditable element, then\n // we return.\n if (!this.hasFocus) return;\n\n let focusNodeChanges = false;\n let anchorNodeChanges = false;\n\n if (this.#focusNode !== this.#selection.focusNode) {\n this.#focusNode = this.#selection.focusNode;\n focusNodeChanges = true;\n }\n this.#focusOffset = this.#selection.focusOffset;\n\n if (this.#anchorNode !== this.#selection.anchorNode) {\n this.#anchorNode = this.#selection.anchorNode;\n anchorNodeChanges = true;\n }\n this.#anchorOffset = this.#selection.anchorOffset;\n\n // We need to handle multi selection from firefox\n // and remove all the old ranges and just keep the\n // last one added.\n if (this.#selection.rangeCount > 1) {\n for (let index = 0; index < this.#selection.rangeCount; index++) {\n const range = this.#selection.getRangeAt(index);\n if (this.#ranges.has(range)) {\n this.#ranges.delete(range);\n this.#selection.removeRange(range);\n } else {\n this.#ranges.add(range);\n this.#range = range;\n }\n }\n } else if (this.#selection.rangeCount > 0) {\n const range = this.#selection.getRangeAt(0);\n this.#range = range;\n this.#ranges.clear();\n this.#ranges.add(range);\n } else {\n this.#range = null;\n this.#ranges.clear();\n }\n\n // If focus node changed, we need to retrieve all the\n // styles of the current inline and dispatch an event\n // to notify that the styles have changed.\n if (focusNodeChanges) {\n this.#notifyStyleChange();\n }\n\n if (this.#debug) {\n this.#debug.update(this);\n }\n };\n\n /**\n * Notifies that the styles have changed.\n */\n #notifyStyleChange() {\n const inline = this.focusInline;\n if (inline) {\n this.#updateCurrentStyle(inline);\n this.dispatchEvent(\n new CustomEvent(\"stylechange\", {\n detail: this.#currentStyle,\n })\n );\n }\n }\n\n /**\n * Setups\n */\n #setup() {\n // This element is not attached to the DOM\n // so it doesn't trigger style or layout calculations.\n // That's why it's called \"inertElement\".\n this.#inertElement = document.createElement(\"div\");\n this.#currentStyle = this.#inertElement.style;\n this.#applyDefaultStylesToCurrentStyle();\n\n if (this.#selection.rangeCount > 0) {\n const range = this.#selection.getRangeAt(0);\n this.#range = range;\n this.#ranges.add(range);\n }\n\n // If there are more than one range, we should remove\n // them because this is a feature not supported by browsers\n // like Safari and Chrome.\n if (this.#selection.rangeCount > 1) {\n for (let index = 1; index < this.#selection.rangeCount; index++) {\n this.#selection.removeRange(index);\n }\n }\n document.addEventListener(\"selectionchange\", this.#onSelectionChange);\n }\n\n /**\n * Returns a Range-like object.\n *\n * @returns {RangeLike}\n */\n #getSavedRange() {\n if (!this.#range) {\n return {\n collapsed: true,\n commonAncestorContainer: null,\n startContainer: null,\n startOffset: 0,\n endContainer: null,\n endOffset: 0,\n };\n }\n return {\n collapsed: this.#range.collapsed,\n commonAncestorContainer: this.#range.commonAncestorContainer,\n startContainer: this.#range.startContainer,\n startOffset: this.#range.startOffset,\n endContainer: this.#range.endContainer,\n endOffset: this.#range.endOffset,\n };\n }\n\n /**\n * Saves the current selection and returns the client rects.\n *\n * @returns {boolean}\n */\n saveSelection() {\n this.#savedSelection = {\n isCollapsed: this.#selection.isCollapsed,\n focusNode: this.#selection.focusNode,\n focusOffset: this.#selection.focusOffset,\n anchorNode: this.#selection.anchorNode,\n anchorOffset: this.#selection.anchorOffset,\n range: this.#getSavedRange(),\n };\n return true;\n }\n\n /**\n * Restores a saved selection if there's any.\n *\n * @returns {boolean}\n */\n restoreSelection() {\n if (!this.#savedSelection) return false;\n\n if (this.#savedSelection.anchorNode && this.#savedSelection.focusNode) {\n if (this.#savedSelection.anchorNode === this.#savedSelection.focusNode) {\n this.#selection.setPosition(this.#savedSelection.focusNode, this.#savedSelection.focusOffset);\n } else {\n this.#selection.setBaseAndExtent(\n this.#savedSelection.anchorNode,\n this.#savedSelection.anchorOffset,\n this.#savedSelection.focusNode,\n this.#savedSelection.focusOffset\n );\n }\n }\n this.#savedSelection = null;\n return true;\n }\n\n /**\n * Marks the start of a mutation.\n *\n * Clears all the mutations kept in CommandMutations.\n */\n startMutation() {\n this.#mutations.clear();\n if (!this.#focusNode) return false;\n return true;\n }\n\n /**\n * Marks the end of a mutation.\n *\n * @returns\n */\n endMutation() {\n return this.#mutations;\n }\n\n /**\n * Selects all content.\n */\n selectAll() {\n this.#selection.selectAllChildren(this.#textEditor.root);\n return this;\n }\n\n /**\n * Moves cursor to end.\n */\n cursorToEnd() {\n const range = document.createRange(); //Create a range (a range is a like the selection but invisible)\n range.selectNodeContents(this.#textEditor.element);\n range.collapse(false);\n this.#selection.removeAllRanges();\n this.#selection.addRange(range);\n return this;\n }\n\n /**\n * Collapses a selection.\n *\n * @param {Node} node\n * @param {number} offset\n */\n collapse(node, offset) {\n const nodeOffset = (node.nodeType === Node.TEXT_NODE && offset >= node.nodeValue.length)\n ? node.nodeValue.length\n : offset\n\n return this.setSelection(\n node,\n nodeOffset,\n node,\n nodeOffset\n );\n }\n\n /**\n * Sets base and extent.\n *\n * @param {Node} anchorNode\n * @param {number} anchorOffset\n * @param {Node} [focusNode=anchorNode]\n * @param {number} [focusOffset=anchorOffset]\n */\n setSelection(anchorNode, anchorOffset, focusNode = anchorNode, focusOffset = anchorOffset) {\n if (!anchorNode.isConnected) {\n throw new Error('Invalid anchorNode')\n }\n if (!focusNode.isConnected) {\n throw new Error('Invalid focusNode')\n }\n if (this.#savedSelection) {\n this.#savedSelection.isCollapsed =\n focusNode === anchorNode && anchorOffset === focusOffset;\n this.#savedSelection.focusNode = focusNode;\n this.#savedSelection.focusOffset = focusOffset;\n this.#savedSelection.anchorNode = anchorNode;\n this.#savedSelection.anchorOffset = anchorOffset;\n\n this.#savedSelection.range.collapsed = this.#savedSelection.isCollapsed;\n const position = focusNode.compareDocumentPosition(anchorNode);\n if (position & Node.DOCUMENT_POSITION_FOLLOWING) {\n this.#savedSelection.range.startContainer = focusNode;\n this.#savedSelection.range.startOffset = focusOffset;\n this.#savedSelection.range.endContainer = anchorNode;\n this.#savedSelection.range.endOffset = anchorOffset;\n } else {\n this.#savedSelection.range.startContainer = anchorNode;\n this.#savedSelection.range.startOffset = anchorOffset;\n this.#savedSelection.range.endContainer = focusNode;\n this.#savedSelection.range.endOffset = focusOffset;\n }\n } else {\n this.#anchorNode = anchorNode;\n this.#anchorOffset = anchorOffset;\n if (anchorNode === focusNode) {\n this.#focusNode = this.#anchorNode;\n this.#focusOffset = this.#anchorOffset;\n this.#selection.setPosition(anchorNode, anchorOffset);\n } else {\n this.#focusNode = focusNode;\n this.#focusOffset = focusOffset;\n this.#selection.setBaseAndExtent(\n anchorNode,\n anchorOffset,\n focusNode,\n focusOffset\n );\n }\n }\n }\n\n /**\n * Disposes the current resources.\n */\n dispose() {\n document.removeEventListener(\"selectionchange\", this.#onSelectionChange);\n this.#textEditor = null;\n this.#ranges.clear();\n this.#ranges = null;\n this.#range = null;\n this.#selection = null;\n this.#focusNode = null;\n this.#anchorNode = null;\n this.#mutations.dispose();\n this.#mutations = null;\n }\n\n /**\n * Returns the current selection.\n *\n * @type {Selection}\n */\n get selection() {\n return this.#selection;\n }\n\n /**\n * Returns the current range.\n *\n * @type {Range}\n */\n get range() {\n return this.#range;\n }\n\n /**\n * Indicates the direction of the selection\n *\n * @type {SelectionDirection}\n */\n get direction() {\n if (this.isCollapsed) {\n return SelectionDirection.NONE;\n }\n if (this.focusNode !== this.anchorNode) {\n return this.startContainer === this.focusNode\n ? SelectionDirection.BACKWARD\n : SelectionDirection.FORWARD;\n }\n return this.focusOffset < this.anchorOffset\n ? SelectionDirection.BACKWARD\n : SelectionDirection.FORWARD;\n }\n\n /**\n * Indicates that the editor element has the\n * focus.\n *\n * @type {boolean}\n */\n get hasFocus() {\n return document.activeElement === this.#textEditor.element;\n }\n\n /**\n * Returns true if the selection is collapsed (caret)\n * or false otherwise.\n *\n * @type {boolean}\n */\n get isCollapsed() {\n if (this.#savedSelection) {\n return this.#savedSelection.isCollapsed;\n }\n return this.#selection.isCollapsed;\n }\n\n /**\n * Current or saved anchor node.\n *\n * @type {Node}\n */\n get anchorNode() {\n if (this.#savedSelection) {\n return this.#savedSelection.anchorNode;\n }\n return this.#anchorNode;\n }\n\n /**\n * Current or saved anchor offset.\n *\n * @type {number}\n */\n get anchorOffset() {\n if (this.#savedSelection) {\n return this.#savedSelection.anchorOffset;\n }\n return this.#selection.anchorOffset;\n }\n\n /**\n * Indicates that the caret is at the start of the node.\n *\n * @type {boolean}\n */\n get anchorAtStart() {\n return this.anchorOffset === 0;\n }\n\n /**\n * Indicates that the caret is at the end of the node.\n *\n * @type {boolean}\n */\n get anchorAtEnd() {\n return this.anchorOffset === this.anchorNode.nodeValue.length;\n }\n\n /**\n * Current or saved focus node.\n *\n * @type {Node}\n */\n get focusNode() {\n if (this.#savedSelection) {\n return this.#savedSelection.focusNode;\n }\n if (!this.#focusNode)\n console.trace(\"focusNode\", this.#focusNode);\n return this.#focusNode;\n }\n\n /**\n * Current or saved focus offset.\n *\n * @type {number}\n */\n get focusOffset() {\n if (this.#savedSelection) {\n return this.#savedSelection.focusOffset;\n }\n return this.#focusOffset;\n }\n\n /**\n * Indicates that the caret is at the start of the node.\n *\n * @type {boolean}\n */\n get focusAtStart() {\n return this.focusOffset === 0;\n }\n\n /**\n * Indicates that the caret is at the end of the node.\n *\n * @type {boolean}\n */\n get focusAtEnd() {\n return this.focusOffset === this.focusNode.nodeValue.length;\n }\n\n /**\n * Returns the paragraph in the focus node\n * of the current selection.\n *\n * @type {HTMLElement|null}\n */\n get focusParagraph() {\n return getParagraph(this.focusNode);\n }\n\n /**\n * Returns the inline in the focus node\n * of the current selection.\n *\n * @type {HTMLElement|null}\n */\n get focusInline() {\n return getInline(this.focusNode);\n }\n\n /**\n * Returns the current paragraph in the anchor\n * node of the current selection.\n *\n * @type {HTMLElement|null}\n */\n get anchorParagraph() {\n return getParagraph(this.anchorNode);\n }\n\n /**\n * Returns the current inline in the anchor\n * node of the current selection.\n *\n * @type {HTMLElement|null}\n */\n get anchorInline() {\n return getInline(this.anchorNode);\n }\n\n /**\n * Start container of the current range.\n */\n get startContainer() {\n if (this.#savedSelection) {\n return this.#savedSelection?.range?.startContainer;\n }\n return this.#range?.startContainer;\n }\n\n /**\n * `startOffset` of the current range.\n *\n * @type {number|null}\n */\n get startOffset() {\n if (this.#savedSelection) {\n return this.#savedSelection?.range?.startOffset;\n }\n return this.#range?.startOffset;\n }\n\n /**\n * Start paragraph of the current range.\n *\n * @type {HTMLElement|null}\n */\n get startParagraph() {\n const startContainer = this.startContainer;\n if (!startContainer) return null;\n return getParagraph(startContainer);\n }\n\n /**\n * Start inline of the current page.\n *\n * @type {HTMLElement|null}\n */\n get startInline() {\n const startContainer = this.startContainer;\n if (!startContainer) return null;\n return getInline(startContainer);\n }\n\n /**\n * End container of the current range.\n *\n * @type {Node}\n */\n get endContainer() {\n if (this.#savedSelection) {\n return this.#savedSelection?.range?.endContainer;\n }\n return this.#range?.endContainer;\n }\n\n /**\n * `endOffset` of the current range\n *\n * @type {HTMLElement|null}\n */\n get endOffset() {\n if (this.#savedSelection) {\n return this.#savedSelection?.range?.endOffset;\n }\n return this.#range?.endOffset;\n }\n\n /**\n * Paragraph element of the `endContainer` of\n * the current range.\n *\n * @type {HTMLElement|null}\n */\n get endParagraph() {\n const endContainer = this.endContainer;\n if (!endContainer) return null;\n return getParagraph(endContainer);\n }\n\n /**\n * Inline element of the `endContainer` of\n * the current range.\n *\n * @type {HTMLElement|null}\n */\n get endInline() {\n const endContainer = this.endContainer;\n if (!endContainer) return null;\n return getInline(endContainer);\n }\n\n /**\n * Returns true if the anchor node and the focus\n * node are the same text nodes.\n *\n * @type {boolean}\n */\n get isTextSame() {\n return (\n this.isTextFocus === this.isTextAnchor &&\n this.focusNode === this.anchorNode\n );\n }\n\n /**\n * Indicates that focus node is a text node.\n *\n * @type {boolean}\n */\n get isTextFocus() {\n return this.focusNode.nodeType === Node.TEXT_NODE;\n }\n\n /**\n * Indicates that anchor node is a text node.\n *\n * @type {boolean}\n */\n get isTextAnchor() {\n return this.anchorNode.nodeType === Node.TEXT_NODE;\n }\n\n /**\n * Is true if the current focus node is a inline.\n *\n * @type {boolean}\n */\n get isInlineFocus() {\n return isInline(this.focusNode);\n }\n\n /**\n * Is true if the current anchor node is a inline.\n *\n * @type {boolean}\n */\n get isInlineAnchor() {\n return isInline(this.anchorNode);\n }\n\n /**\n * Is true if the current focus node is a paragraph.\n *\n * @type {boolean}\n */\n get isParagraphFocus() {\n return isParagraph(this.focusNode);\n }\n\n /**\n * Is true if the current anchor node is a paragraph.\n *\n * @type {boolean}\n */\n get isParagraphAnchor() {\n return isParagraph(this.anchorNode);\n }\n\n /**\n * Is true if the current focus node is a line break.\n *\n * @type {boolean}\n */\n get isLineBreakFocus() {\n return (\n isLineBreak(this.focusNode) ||\n (isInline(this.focusNode) && isLineBreak(this.focusNode.firstChild))\n );\n }\n\n /**\n * Indicates that we have multiple nodes selected.\n *\n * @type {boolean}\n */\n get isMulti() {\n return this.focusNode !== this.anchorNode;\n }\n\n /**\n * Indicates that we have selected multiple\n * paragraph elements.\n *\n * @type {boolean}\n */\n get isMultiParagraph() {\n return this.isMulti && this.focusParagraph !== this.anchorParagraph;\n }\n\n /**\n * Indicates that we have selected multiple\n * inline elements.\n *\n * @type {boolean}\n */\n get isMultiInline() {\n return this.isMulti && this.focusInline !== this.anchorInline;\n }\n\n /**\n * Indicates that the caret (only the caret)\n * is at the start of an inline.\n *\n * @type {boolean}\n */\n get isInlineStart() {\n if (!this.isCollapsed) return false;\n return isInlineStart(this.focusNode, this.focusOffset);\n }\n\n /**\n * Indicates that the caret (only the caret)\n * is at the end of an inline. This value doesn't\n * matter when dealing with selections.\n *\n * @type {boolean}\n */\n get isInlineEnd() {\n if (!this.isCollapsed) return false;\n return isInlineEnd(this.focusNode, this.focusOffset);\n }\n\n /**\n * Indicates that we're in the starting position of a paragraph.\n *\n * @type {boolean}\n */\n get isParagraphStart() {\n if (!this.isCollapsed) return false;\n return isParagraphStart(this.focusNode, this.focusOffset);\n }\n\n /**\n * Indicates that we're in the ending position of a paragraph.\n *\n * @type {boolean}\n */\n get isParagraphEnd() {\n if (!this.isCollapsed) return false;\n return isParagraphEnd(this.focusNode, this.focusOffset);\n }\n\n /**\n * Insert pasted fragment.\n *\n * @param {DocumentFragment} fragment\n */\n insertPaste(fragment) {\n const numParagraphs = fragment.children.length;\n if (this.isParagraphStart) {\n this.focusParagraph.before(fragment);\n } else if (this.isParagraphEnd) {\n this.focusParagraph.after(fragment);\n } else {\n const newParagraph = splitParagraph(\n this.focusParagraph,\n this.focusInline,\n this.focusOffset\n );\n this.focusParagraph.after(fragment, newParagraph);\n }\n }\n\n /**\n * Replaces data with pasted fragment\n *\n * @param {DocumentFragment} fragment\n */\n replaceWithPaste(fragment) {\n const numParagraphs = fragment.children.length;\n this.removeSelected();\n this.insertPaste(fragment);\n }\n\n /**\n * Replaces the current line break with text\n *\n * @param {string} text\n */\n replaceLineBreak(text) {\n const newText = new Text(text);\n this.focusInline.replaceChildren(newText);\n this.collapse(newText, text.length);\n }\n\n /**\n * Removes text forward from the current position.\n */\n removeForwardText() {\n this.#textNodeIterator.currentNode = this.focusNode;\n\n const removedData = removeForward(\n this.focusNode.nodeValue,\n this.focusOffset\n );\n\n if (this.focusNode.nodeValue !== removedData) {\n this.focusNode.nodeValue = removedData;\n }\n\n const paragraph = this.focusParagraph;\n if (!paragraph) throw new Error(\"Cannot find paragraph\");\n const inline = this.focusInline;\n if (!inline) throw new Error(\"Cannot find inline\");\n\n const nextTextNode = this.#textNodeIterator.nextNode();\n if (this.focusNode.nodeValue === \"\") {\n this.focusNode.remove();\n }\n\n if (paragraph.childNodes.length === 1 && inline.childNodes.length === 0) {\n const lineBreak = createLineBreak();\n inline.appendChild(lineBreak);\n return this.collapse(lineBreak, 0);\n } else if (\n paragraph.childNodes.length > 1 &&\n inline.childNodes.length === 0\n ) {\n inline.remove();\n return this.collapse(nextTextNode, 0);\n }\n return this.collapse(this.focusNode, this.focusOffset);\n }\n\n /**\n * Removes text backward from the current caret position.\n */\n removeBackwardText() {\n this.#textNodeIterator.currentNode = this.focusNode;\n\n // Remove the character from the string.\n const removedData = removeBackward(\n this.focusNode.nodeValue,\n this.focusOffset\n );\n\n if (this.focusNode.nodeValue !== removedData) {\n this.focusNode.nodeValue = removedData;\n }\n\n // If the focusNode has content we don't need to do\n // anything else.\n if (this.focusOffset - 1 > 0) {\n return this.collapse(this.focusNode, this.focusOffset - 1);\n }\n\n const paragraph = this.focusParagraph;\n if (!paragraph) throw new Error(\"Cannot find paragraph\");\n const inline = this.focusInline;\n if (!inline) throw new Error(\"Cannot find inline\");\n\n const previousTextNode = this.#textNodeIterator.previousNode();\n if (this.focusNode.nodeValue === \"\") {\n this.focusNode.remove();\n }\n\n if (paragraph.children.length === 1 && inline.childNodes.length === 0) {\n const lineBreak = createLineBreak();\n inline.appendChild(lineBreak);\n return this.collapse(lineBreak, 0);\n } else if (\n paragraph.children.length > 1 &&\n inline.childNodes.length === 0\n ) {\n inline.remove();\n return this.collapse(previousTextNode, getTextNodeLength(previousTextNode));\n }\n\n return this.collapse(this.focusNode, this.focusOffset - 1);\n }\n\n /**\n * Inserts some text in the caret position.\n *\n * @param {string} newText\n */\n insertText(newText) {\n this.focusNode.nodeValue = insertInto(\n this.focusNode.nodeValue,\n this.focusOffset,\n newText\n );\n this.#mutations.update(this.focusInline);\n return this.collapse(this.focusNode, this.focusOffset + newText.length);\n }\n\n /**\n * Replaces currently selected text.\n *\n * @param {string} newText\n */\n replaceText(newText) {\n const startOffset = Math.min(this.anchorOffset, this.focusOffset);\n const endOffset = Math.max(this.anchorOffset, this.focusOffset);\n this.focusNode.nodeValue = replaceWith(\n this.focusNode.nodeValue,\n startOffset,\n endOffset,\n newText\n );\n this.#mutations.update(this.focusInline);\n return this.collapse(this.focusNode, startOffset + newText.length);\n }\n\n /**\n * Replaces the selected inlines with new text.\n *\n * @param {string} newText\n */\n replaceInlines(newText) {\n const currentParagraph = this.focusParagraph;\n\n // This is the special (and fast) case where we're\n // removing everything inside a paragraph.\n if (\n this.startInline === currentParagraph.firstChild &&\n this.startOffset === 0 &&\n this.endInline === currentParagraph.lastChild &&\n this.endOffset === currentParagraph.lastChild.textContent.length\n ) {\n const newTextNode = new Text(newText);\n currentParagraph.replaceChildren(\n createInline(newTextNode, this.anchorInline.style)\n );\n return this.collapse(newTextNode, newTextNode.nodeValue.length);\n }\n\n this.removeSelected();\n\n this.focusNode.nodeValue = insertInto(\n this.focusNode.nodeValue,\n this.focusOffset,\n newText\n );\n\n // FIXME: I'm not sure if we should merge inlines when they share the same styles.\n // For example: if we have > 2 inlines and the start inline and the end inline\n // share the same styles, maybe we should merge them?\n // mergeInlines(startInline, endInline);\n return this.collapse(this.focusNode, this.focusOffset + newText.length);\n }\n\n /**\n * Replaces paragraphs with text.\n *\n * @param {string} newText\n */\n replaceParagraphs(newText) {\n const currentParagraph = this.focusParagraph;\n\n this.removeSelected();\n\n this.focusNode.nodeValue = insertInto(\n this.focusNode.nodeValue,\n this.focusOffset,\n newText\n );\n\n for (const child of currentParagraph.children) {\n if (child.textContent === \"\") {\n child.remove();\n }\n }\n }\n\n /**\n * Inserts a new paragraph after the current paragraph.\n */\n insertParagraphAfter() {\n const currentParagraph = this.focusParagraph;\n const newParagraph = createEmptyParagraph(this.#currentStyle);\n currentParagraph.after(newParagraph);\n this.#mutations.update(currentParagraph);\n this.#mutations.add(newParagraph);\n return this.collapse(newParagraph.firstChild.firstChild, 0);\n }\n\n /**\n * Inserts a new paragraph before the current paragraph.\n */\n insertParagraphBefore() {\n const currentParagraph = this.focusParagraph;\n const newParagraph = createEmptyParagraph(this.#currentStyle);\n currentParagraph.before(newParagraph);\n this.#mutations.update(currentParagraph);\n this.#mutations.add(newParagraph);\n return this.collapse(currentParagraph.firstChild.firstChild, 0);\n }\n\n /**\n * Splits the current paragraph.\n */\n splitParagraph() {\n const currentParagraph = this.focusParagraph;\n const newParagraph = splitParagraph(\n this.focusParagraph,\n this.focusInline,\n this.#focusOffset\n );\n this.focusParagraph.after(newParagraph);\n this.#mutations.update(currentParagraph);\n this.#mutations.add(newParagraph);\n return this.collapse(newParagraph.firstChild.firstChild, 0);\n }\n\n /**\n * Inserts a new paragraph.\n */\n insertParagraph() {\n if (this.isParagraphEnd) {\n return this.insertParagraphAfter();\n } else if (this.isParagraphStart) {\n return this.insertParagraphBefore();\n }\n return this.splitParagraph();\n }\n\n /**\n * Replaces the currently selected content with\n * a paragraph.\n */\n replaceWithParagraph() {\n const currentParagraph = this.focusParagraph;\n const currentInline = this.focusInline;\n\n this.removeSelected();\n\n const newParagraph = splitParagraph(\n currentParagraph,\n currentInline,\n this.focusOffset\n );\n currentParagraph.after(newParagraph);\n\n this.#mutations.update(currentParagraph);\n this.#mutations.add(newParagraph);\n\n // FIXME: Missing collapse?\n }\n\n /**\n * Removes a paragraph in backward direction.\n */\n removeBackwardParagraph() {\n const previousParagraph = this.focusParagraph.previousElementSibling;\n if (!previousParagraph) {\n return;\n }\n const paragraphToBeRemoved = this.focusParagraph;\n paragraphToBeRemoved.remove();\n const previousInline =\n previousParagraph.children.length > 1\n ? previousParagraph.lastElementChild\n : previousParagraph.firstChild;\n const previousOffset = isLineBreak(previousInline.firstChild)\n ? 0\n : previousInline.firstChild.nodeValue.length;\n this.#mutations.remove(paragraphToBeRemoved);\n return this.collapse(previousInline.firstChild, previousOffset);\n }\n\n /**\n * Merges the previous paragraph with the current paragraph.\n */\n mergeBackwardParagraph() {\n const currentParagraph = this.focusParagraph;\n const previousParagraph = this.focusParagraph.previousElementSibling;\n if (!previousParagraph) {\n return;\n }\n let previousInline = previousParagraph.lastChild;\n const previousOffset = getInlineLength(previousInline);\n if (isEmptyParagraph(previousParagraph)) {\n previousParagraph.replaceChildren(...currentParagraph.children);\n previousInline = previousParagraph.firstChild;\n currentParagraph.remove();\n } else {\n mergeParagraphs(previousParagraph, currentParagraph);\n }\n this.#mutations.remove(currentParagraph);\n this.#mutations.update(previousParagraph);\n return this.collapse(previousInline.firstChild, previousOffset);\n }\n\n /**\n * Merges the next paragraph with the current paragraph.\n */\n mergeForwardParagraph() {\n const currentParagraph = this.focusParagraph;\n const nextParagraph = this.focusParagraph.nextElementSibling;\n if (!nextParagraph) {\n return;\n }\n mergeParagraphs(this.focusParagraph, nextParagraph);\n this.#mutations.update(currentParagraph);\n this.#mutations.remove(nextParagraph);\n\n // FIXME: Missing collapse?\n }\n\n /**\n * Removes the forward paragraph.\n */\n removeForwardParagraph() {\n const nextParagraph = this.focusParagraph.nextSibling;\n if (!nextParagraph) {\n return;\n }\n const paragraphToBeRemoved = this.focusParagraph;\n paragraphToBeRemoved.remove();\n const nextInline = nextParagraph.firstChild;\n const nextOffset = this.focusOffset;\n this.#mutations.remove(paragraphToBeRemoved);\n return this.collapse(nextInline.firstChild, nextOffset);\n }\n\n /**\n * Cleans up all the affected paragraphs.\n *\n * @param {Set} affectedParagraphs\n * @param {Set} affectedInlines\n */\n cleanUp(affectedParagraphs, affectedInlines) {\n // Remove empty inlines\n for (const inline of affectedInlines) {\n if (inline.textContent === \"\") {\n inline.remove();\n this.#mutations.remove(inline);\n }\n }\n\n // Remove empty paragraphs.\n for (const paragraph of affectedParagraphs) {\n if (paragraph.children.length === 0) {\n paragraph.remove();\n this.#mutations.remove(paragraph);\n }\n }\n }\n\n /**\n * Removes the selected content.\n *\n * @param {RemoveSelectedOptions} [options]\n */\n removeSelected(options) {\n if (this.isCollapsed) return;\n\n const affectedInlines = new Set();\n const affectedParagraphs = new Set();\n\n const startNode = getClosestTextNode(this.#range.startContainer);\n const endNode = getClosestTextNode(this.#range.endContainer);\n const startOffset = this.#range.startOffset;\n const endOffset = this.#range.endOffset;\n\n let previousNode = null;\n let nextNode = null;\n\n // This is the simplest case, when the startNode and the endNode\n // are the same and they're a textNode.\n if (startNode === endNode) {\n this.#textNodeIterator.currentNode = startNode;\n previousNode = this.#textNodeIterator.previousNode();\n\n this.#textNodeIterator.currentNode = startNode;\n nextNode = this.#textNodeIterator.nextNode();\n\n const inline = getInline(startNode);\n const paragraph = getParagraph(startNode);\n affectedInlines.add(inline);\n affectedParagraphs.add(paragraph);\n\n const newNodeValue = removeSlice(\n startNode.nodeValue,\n startOffset,\n endOffset\n );\n if (newNodeValue === \"\") {\n const lineBreak = createLineBreak();\n inline.replaceChildren(lineBreak);\n return this.collapse(lineBreak, 0);\n }\n startNode.nodeValue = newNodeValue;\n return this.collapse(startNode, startOffset);\n }\n\n // If startNode and endNode are different,\n // then we should process every text node from\n // start to end.\n\n // Select initial node.\n this.#textNodeIterator.currentNode = startNode;\n\n const startInline = getInline(startNode);\n const startParagraph = getParagraph(startNode);\n const endInline = getInline(endNode);\n const endParagraph = getParagraph(endNode);\n\n SafeGuard.start();\n do {\n SafeGuard.update();\n\n const currentNode = this.#textNodeIterator.currentNode;\n\n // We retrieve the inline and paragraph of the\n // current node.\n const inline = getInline(this.#textNodeIterator.currentNode);\n const paragraph = getParagraph(this.#textNodeIterator.currentNode);\n\n let shouldRemoveNodeCompletely = false;\n if (this.#textNodeIterator.currentNode === startNode) {\n if (startOffset === 0) {\n // We should remove this node completely.\n shouldRemoveNodeCompletely = true;\n } else {\n // We should remove this node partially.\n currentNode.nodeValue = currentNode.nodeValue.slice(0, startOffset);\n }\n } else if (this.#textNodeIterator.currentNode === endNode) {\n if (isLineBreak(endNode)\n || (isTextNode(endNode)\n && endOffset === endNode.nodeValue.length)) {\n // We should remove this node completely.\n shouldRemoveNodeCompletely = true;\n } else {\n // We should remove this node partially.\n currentNode.nodeValue = currentNode.nodeValue.slice(endOffset);\n }\n } else {\n // We should remove this node completely.\n shouldRemoveNodeCompletely = true;\n }\n\n this.#textNodeIterator.nextNode();\n\n // Realizamos el borrado del nodo actual.\n if (shouldRemoveNodeCompletely) {\n currentNode.remove();\n if (currentNode === startNode) {\n continue;\n }\n if (currentNode === endNode) {\n break;\n }\n\n if (inline.childNodes.length === 0) {\n inline.remove();\n }\n if (paragraph !== startParagraph && paragraph.children.length === 0) {\n paragraph.remove();\n }\n }\n\n if (currentNode === endNode) {\n break;\n }\n\n } while (this.#textNodeIterator.currentNode);\n\n if (startParagraph !== endParagraph) {\n const mergedParagraph = mergeParagraphs(startParagraph, endParagraph);\n if (mergedParagraph.children.length === 0) {\n const newEmptyInline = createEmptyInline(this.#currentStyle);\n mergedParagraph.appendChild(newEmptyInline);\n return this.collapse(newEmptyInline.firstChild, 0);\n }\n }\n\n if (startInline.childNodes.length === 0 && endInline.childNodes.length > 0) {\n startInline.remove();\n return this.collapse(endNode, 0);\n } else if (startInline.childNodes.length > 0 && endInline.childNodes.length === 0) {\n endInline.remove();\n return this.collapse(startNode, startOffset);\n } else if (startInline.childNodes.length === 0 && endInline.childNodes.length === 0) {\n const previousInline = startInline.previousElementSibling;\n const nextInline = endInline.nextElementSibling;\n startInline.remove();\n endInline.remove();\n if (previousInline) {\n return this.collapse(previousInline.firstChild, previousInline.firstChild.nodeValue.length);\n }\n if (nextInline) {\n return this.collapse(nextInline.firstChild, 0);\n }\n const newEmptyInline = createEmptyInline(this.#currentStyle);\n startParagraph.appendChild(newEmptyInline);\n return this.collapse(newEmptyInline.firstChild, 0);\n }\n\n return this.collapse(startNode, startOffset);\n }\n\n /**\n * Applies styles from the startNode to the endNode.\n *\n * @param {Node} startNode\n * @param {number} startOffset\n * @param {Node} endNode\n * @param {number} endOffset\n * @param {Object.|CSSStyleDeclaration} newStyles\n * @returns {void}\n */\n #applyStylesTo(startNode, startOffset, endNode, endOffset, newStyles) {\n // Applies the necessary styles to the root element.\n const root = this.#textEditor.root;\n setRootStyles(root, newStyles);\n\n // If the startContainer and endContainer are the same\n // node, then we can apply styles directly to that\n // node.\n if (startNode === endNode && startNode.nodeType === Node.TEXT_NODE) {\n // The styles are applied to the node completelly.\n if (startOffset === 0 && endOffset === endNode.nodeValue.length) {\n const paragraph = this.startParagraph;\n const inline = this.startInline;\n setParagraphStyles(paragraph, newStyles);\n setInlineStyles(inline, newStyles);\n\n // The styles are applied to a part of the node.\n } else if (startOffset !== endOffset) {\n const paragraph = this.startParagraph;\n setParagraphStyles(paragraph, newStyles);\n const inline = this.startInline;\n const midText = startNode.splitText(startOffset);\n const endText = midText.splitText(endOffset - startOffset);\n const midInline = createInlineFrom(inline, midText, newStyles);\n inline.after(midInline);\n if (endText.length > 0) {\n const endInline = createInline(endText, inline.style);\n midInline.after(endInline);\n }\n\n // FIXME: This can change focus <-> anchor order.\n this.setSelection(midText, 0, midText, midText.nodeValue.length);\n\n // The styles are applied to the paragraph.\n } else {\n const paragraph = this.startParagraph;\n setParagraphStyles(paragraph, newStyles);\n }\n return this.#notifyStyleChange();\n\n // If the startContainer and endContainer are different\n // then we need to iterate through those nodes to apply\n // the styles.\n } else if (startNode !== endNode) {\n SafeGuard.start();\n const expectedEndNode = getClosestTextNode(endNode);\n this.#textNodeIterator.currentNode = getClosestTextNode(startNode);\n do {\n SafeGuard.update();\n\n const paragraph = getParagraph(this.#textNodeIterator.currentNode);\n setParagraphStyles(paragraph, newStyles);\n const inline = getInline(this.#textNodeIterator.currentNode);\n // If we're at the start node and offset is greater than 0\n // then we should split the inline and apply styles to that\n // new inline.\n if (\n this.#textNodeIterator.currentNode === startNode &&\n startOffset > 0\n ) {\n const newInline = splitInline(inline, startOffset);\n setInlineStyles(newInline, newStyles);\n inline.after(newInline);\n // If we're at the start node and offset is equal to 0\n // or current node is different to start node and\n // different to end node or we're at the end node\n // and the offset is equalto the node length\n } else if (\n (this.#textNodeIterator.currentNode === startNode &&\n startOffset === 0) ||\n (this.#textNodeIterator.currentNode !== startNode &&\n this.#textNodeIterator.currentNode !== endNode) ||\n (this.#textNodeIterator.currentNode === endNode &&\n endOffset === endNode.nodeValue.length)\n ) {\n setInlineStyles(inline, newStyles);\n\n // If we're at end node\n } else if (\n this.#textNodeIterator.currentNode === endNode &&\n endOffset < endNode.nodeValue.length\n ) {\n const newInline = splitInline(inline, endOffset);\n setInlineStyles(inline, newStyles);\n inline.after(newInline);\n }\n\n // We've reached the final node so we can return safely.\n if (this.#textNodeIterator.currentNode === expectedEndNode) return;\n\n this.#textNodeIterator.nextNode();\n } while (this.#textNodeIterator.currentNode);\n }\n\n return this.#notifyStyleChange();\n }\n\n /**\n * Applies styles to selection\n *\n * @param {Object.} newStyles\n * @returns {void}\n */\n applyStyles(newStyles) {\n return this.#applyStylesTo(\n this.startContainer,\n this.startOffset,\n this.endContainer,\n this.endOffset,\n newStyles\n );\n }\n}\n\nexport default SelectionController;\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Creates a new selection imposter from a list of client rects.\n *\n * @param {DOMRect} referenceRect\n * @param {DOMRectList} clientRects\n * @returns {DocumentFragment}\n */\nexport function createSelectionImposterFromClientRects(\n referenceRect,\n clientRects\n) {\n const fragment = document.createDocumentFragment();\n for (const rect of clientRects) {\n const rectElement = document.createElement(\"div\");\n rectElement.className = \"selection-imposter-rect\";\n rectElement.style.left = `${rect.x - referenceRect.x}px`;\n rectElement.style.top = `${rect.y - referenceRect.y}px`;\n rectElement.style.width = `${rect.width}px`;\n rectElement.style.height = `${rect.height}px`;\n fragment.appendChild(rectElement);\n }\n return fragment;\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Adds a series of listeners.\n *\n * @param {EventTarget} target\n * @param {Object.} object\n * @param {EventListenerOptions} [options]\n */\nexport function addEventListeners(target, object, options) {\n Object.entries(object).forEach(([type, listener]) =>\n target.addEventListener(type, listener, options)\n );\n}\n\n/**\n * Removes a series of listeners.\n *\n * @param {EventTarget} target\n * @param {Object.} object\n */\nexport function removeEventListeners(target, object) {\n Object.entries(object).forEach(([type, listener]) =>\n target.removeEventListener(type, listener)\n );\n}\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\n/**\n * Enumeration of types of layout.\n *\n * @enum {string}\n */\nexport const LayoutType = {\n FULL: \"full\",\n PARTIAL: \"partial\",\n};\n\nexport default LayoutType;\n", "/**\n * This Source Code Form is subject to the terms of the Mozilla Public\n * License, v. 2.0. If a copy of the MPL was not distributed with this\n * file, You can obtain one at http://mozilla.org/MPL/2.0/.\n *\n * Copyright (c) KALEIDOS INC\n */\n\nimport clipboard from \"./clipboard/index.js\";\nimport commands from \"./commands/index.js\";\nimport ChangeController from './controllers/ChangeController.js';\nimport SelectionController from './controllers/SelectionController.js';\nimport { createSelectionImposterFromClientRects } from './selection/Imposter.js';\nimport { addEventListeners, removeEventListeners } from \"./Event\";\nimport { createRoot, createEmptyRoot } from './content/dom/Root.js';\nimport { createParagraph } from './content/dom/Paragraph.js';\nimport { createEmptyInline, createInline } from './content/dom/Inline.js';\nimport { isLineBreak } from './content/dom/LineBreak.js';\nimport LayoutType from './layout/LayoutType.js';\n\n/**\n * Text Editor.\n */\nexport class TextEditor extends EventTarget {\n /**\n * Element content editable to be used by the TextEditor\n *\n * @type {HTMLElement}\n */\n #element = null;\n\n /**\n * Map/Dictionary of events.\n *\n * @type {Object.}\n */\n #events = null;\n\n /**\n * Root element that will contain the content.\n *\n * @type {HTMLElement}\n */\n #root = null;\n\n /**\n * Change controller controls when we should notify changes.\n *\n * @type {ChangeController}\n */\n #changeController = null;\n\n /**\n * Selection controller controls the current/saved selection.\n *\n * @type {SelectionController}\n */\n #selectionController = null;\n\n /**\n * Selection imposter keeps selection elements.\n *\n * @type {HTMLElement}\n */\n #selectionImposterElement = null;\n\n /**\n * Style defaults.\n *\n * @type {Object.}\n */\n #styleDefaults = null;\n\n /**\n * Constructor.\n *\n * @param {HTMLElement} element\n */\n constructor(element, options) {\n super();\n if (!(element instanceof HTMLElement))\n throw new TypeError(\"Invalid text editor element\");\n\n this.#element = element;\n this.#selectionImposterElement = options?.selectionImposterElement;\n this.#events = {\n blur: this.#onBlur,\n focus: this.#onFocus,\n\n paste: this.#onPaste,\n cut: this.#onCut,\n copy: this.#onCopy,\n\n beforeinput: this.#onBeforeInput,\n input: this.#onInput,\n };\n this.#styleDefaults = options?.styleDefaults;\n this.#setup(options);\n }\n\n /**\n * Setups editor properties.\n */\n #setupElementProperties() {\n if (!this.#element.isContentEditable) {\n this.#element.contentEditable = \"true\";\n // In `jsdom` it isn't enough to set the attribute 'contentEditable'\n // to `true` to work.\n // FIXME: Remove this when `jsdom` implements this interface.\n if (!this.#element.isContentEditable) {\n this.#element.setAttribute(\"contenteditable\", \"true\");\n }\n }\n if (this.#element.spellcheck) this.#element.spellcheck = false;\n if (this.#element.autocapitalize) this.#element.autocapitalize = false;\n if (!this.#element.autofocus) this.#element.autofocus = true;\n if (!this.#element.role || this.#element.role !== \"textbox\")\n this.#element.role = \"textbox\";\n if (this.#element.ariaAutoComplete) this.#element.ariaAutoComplete = false;\n if (!this.#element.ariaMultiLine) this.#element.ariaMultiLine = true;\n this.#element.dataset.itype = \"editor\";\n }\n\n /**\n * Setups the root element.\n */\n #setupRoot() {\n this.#root = createEmptyRoot(this.#styleDefaults);\n this.#element.appendChild(this.#root);\n }\n\n /**\n * Dispatchs a `change` event.\n *\n * @param {CustomEvent} e\n * @returns {void}\n */\n #onChange = (e) => this.dispatchEvent(new e.constructor(e.type, e));\n\n /**\n * Dispatchs a `stylechange` event.\n *\n * @param {CustomEvent} e\n * @returns {void}\n */\n #onStyleChange = (e) => {\n if (this.#selectionImposterElement.children.length > 0) {\n // We need to recreate the selection imposter when we've\n // already have one.\n this.#createSelectionImposter();\n }\n this.dispatchEvent(new e.constructor(e.type, e));\n };\n\n /**\n * Setups the elements, the properties and the\n * initial content.\n */\n #setup(options) {\n this.#setupElementProperties();\n this.#setupRoot();\n this.#changeController = new ChangeController(this);\n this.#changeController.addEventListener(\"change\", this.#onChange);\n this.#selectionController = new SelectionController(\n this,\n document.getSelection(),\n options\n );\n this.#selectionController.addEventListener(\n \"stylechange\",\n this.#onStyleChange\n );\n addEventListeners(this.#element, this.#events, {\n capture: true\n });\n }\n\n /**\n * Creates the selection imposter.\n */\n #createSelectionImposter() {\n // We only create a selection imposter if there's any selection\n // and if there is a selection imposter element to attach the\n // rects.\n if (\n this.#selectionImposterElement &&\n !this.#selectionController.isCollapsed\n ) {\n const rects = this.#selectionController.range?.getClientRects();\n if (rects) {\n const rect = this.#selectionImposterElement.getBoundingClientRect();\n this.#selectionImposterElement.replaceChildren(\n createSelectionImposterFromClientRects(rect, rects)\n );\n }\n }\n }\n\n /**\n * On blur we create a new FakeSelection if there's any.\n *\n * @param {FocusEvent} e\n */\n #onBlur = (e) => {\n this.#changeController.notifyImmediately();\n this.#selectionController.saveSelection();\n this.#createSelectionImposter();\n this.dispatchEvent(new FocusEvent(e.type, e));\n };\n\n /**\n * On focus we should restore the FakeSelection from the current\n * selection.\n *\n * @param {FocusEvent} e\n */\n #onFocus = (e) => {\n this.#selectionController.restoreSelection();\n if (this.#selectionImposterElement) {\n this.#selectionImposterElement.replaceChildren();\n }\n this.dispatchEvent(new FocusEvent(e.type, e));\n };\n\n /**\n * Event called when the user pastes some text into the\n * editor.\n *\n * @param {ClipboardEvent} e\n */\n #onPaste = (e) => clipboard.paste(e, this, this.#selectionController);\n\n /**\n * Event called when the user cuts some text from the\n * editor.\n *\n * @param {ClipboardEvent} e\n */\n #onCut = (e) => clipboard.cut(e, this, this.#selectionController);\n\n /**\n * Event called when the user copies some text from the\n * editor.\n *\n * @param {ClipboardEvent} e\n */\n #onCopy = (e) => clipboard.copy(e, this, this.#selectionController);\n\n /**\n * Event called before the DOM is modified.\n *\n * @param {InputEvent} e\n */\n #onBeforeInput = (e) => {\n if (e.inputType === \"historyUndo\"\n || e.inputType === \"historyRedo\") {\n return;\n }\n\n if (!(e.inputType in commands)) {\n if (e.inputType !== \"insertCompositionText\") {\n e.preventDefault();\n }\n return;\n }\n\n if (e.inputType in commands) {\n const command = commands[e.inputType];\n if (!this.#selectionController.startMutation())\n return;\n\n command(e, this, this.#selectionController);\n const mutations = this.#selectionController.endMutation();\n this.#notifyLayout(LayoutType.FULL, mutations);\n }\n };\n\n /**\n * Event called after the DOM is modified.\n *\n * @param {InputEvent} e\n */\n #onInput = (e) => {\n if (e.inputType === \"historyUndo\" || e.inputType === \"historyRedo\") {\n return;\n }\n\n if (e.inputType === \"insertCompositionText\") {\n this.#notifyLayout(LayoutType.FULL, null);\n }\n };\n\n /**\n * Notifies that the edited texts needs layout.\n *\n * @param {'full'|'partial'} type\n * @param {CommandMutations} mutations\n */\n #notifyLayout(type = LayoutType.FULL, mutations) {\n this.dispatchEvent(\n new CustomEvent(\"needslayout\", {\n detail: {\n type: type,\n mutations: mutations,\n },\n })\n );\n }\n\n /**\n * Root element that contains all the paragraphs.\n *\n * @type {HTMLDivElement}\n */\n get root() {\n return this.#root;\n }\n\n set root(newRoot) {\n const previousRoot = this.#root;\n this.#root = newRoot;\n previousRoot.replaceWith(newRoot);\n }\n\n /**\n * Element that contains the root and that has the\n * contenteditable attribute.\n *\n * @type {HTMLElement}\n */\n get element() {\n return this.#element;\n }\n\n /**\n * Returns true if the content is in an empty state.\n *\n * @type {boolean}\n */\n get isEmpty() {\n return (\n this.#root.children.length === 1 &&\n this.#root.firstElementChild.children.length === 1 &&\n isLineBreak(this.#root.firstElementChild.firstElementChild.firstChild)\n );\n }\n\n /**\n * Indicates the amount of paragraphs in the current content.\n *\n * @type {number}\n */\n get numParagraphs() {\n return this.#root.children.length;\n }\n\n /**\n * CSS Style declaration for the current inline. From here we\n * can infer root, paragraph and inline declarations.\n *\n * @type {CSSStyleDeclaration}\n */\n get currentStyle() {\n return this.#selectionController.currentStyle;\n }\n\n /**\n * Focus the element\n */\n focus() {\n return this.#element.focus();\n }\n\n /**\n * Blurs the element\n */\n blur() {\n return this.#element.blur();\n }\n\n /**\n * Creates a new root.\n *\n * @param {...any} args\n * @returns {HTMLDivElement}\n */\n createRoot(...args) {\n return createRoot(...args);\n }\n\n /**\n * Creates a new paragraph.\n *\n * @param {...any} args\n * @returns {HTMLDivElement}\n */\n createParagraph(...args) {\n return createParagraph(...args);\n }\n\n /**\n * Creates a new inline from a string.\n *\n * @param {string} text\n * @param {Object.|CSSStyleDeclaration} styles\n * @returns {HTMLSpanElement}\n */\n createInlineFromString(text, styles) {\n if (text === \"\") {\n return createEmptyInline(styles);\n }\n return createInline(new Text(text), styles);\n }\n\n /**\n * Creates a new inline.\n *\n * @param {...any} args\n * @returns {HTMLSpanElement}\n */\n createInline(...args) {\n return createInline(...args);\n }\n\n /**\n * Applies the current styles to the selection or\n * the current DOM node at the caret.\n *\n * @param {*} styles\n */\n applyStylesToSelection(styles) {\n this.#selectionController.startMutation();\n this.#selectionController.applyStyles(styles);\n const mutations = this.#selectionController.endMutation();\n this.#notifyLayout(LayoutType.FULL, mutations);\n this.#changeController.notifyImmediately();\n return this;\n }\n\n /**\n * Selects all content.\n */\n selectAll() {\n this.#selectionController.selectAll();\n return this;\n }\n\n /**\n * Moves cursor to end.\n *\n * @returns\n */\n cursorToEnd() {\n this.#selectionController.cursorToEnd();\n return this;\n }\n\n /**\n * Disposes everything.\n */\n dispose() {\n this.#changeController.removeEventListener(\"change\", this.#onChange);\n this.#changeController.dispose();\n this.#changeController = null;\n this.#selectionController.removeEventListener(\n \"stylechange\",\n this.#onStyleChange\n );\n this.#selectionController.dispose();\n this.#selectionController = null;\n removeEventListeners(this.#element, this.#events);\n this.#element = null;\n this.#root = null;\n }\n}\n\nexport function isEditor(instance) {\n return (instance instanceof TextEditor);\n}\n\n/* Convenience function based API for Text Editor */\nexport function getRoot(instance) {\n if (isEditor(instance)) {\n return instance.root;\n } else {\n return null;\n }\n}\n\nexport function setRoot(instance, root) {\n if (isEditor(instance)) {\n instance.root = root;\n }\n\n return instance;\n}\n\nexport function create(element, options) {\n return new TextEditor(element, {...options});\n}\n\nexport function getCurrentStyle(instance) {\n if (isEditor(instance)) {\n return instance.currentStyle;\n }\n}\n\nexport function applyStylesToSelection(instance, styles) {\n if (isEditor(instance)) {\n return instance.applyStylesToSelection(styles);\n }\n}\n\nexport function dispose(instance) {\n if (isEditor(instance)) {\n instance.dispose();\n }\n}\n\nexport default TextEditor;\n"], - "mappings": "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", - "names": ["copy", "event", "editor", "cut", "event", "editor", "canvas", "context", "getContext", "createCanvas", "width", "height", "getByteAsHex", "byte", "getColor", "fillStyle", "imageData", "r", "g", "b", "a", "getFills", "color", "opacity", "mergeStyleDeclarations", "target", "source", "index", "styleName", "getComputedStyle", "element", "inertElement", "currentElement", "newValue", "normalizeStyles", "styleDeclaration", "color", "getFills", "fontFamily", "fontId", "setStyle", "styleValue", "styleUnit", "getStyleFromDeclaration", "style", "setStylesFromObject", "element", "allowedStyles", "styleObject", "styleName", "styleUnit", "styleValue", "setStyle", "setStylesFromDeclaration", "styleDeclaration", "getStyleFromDeclaration", "setStyles", "styleObjectOrDeclaration", "mergeStyles", "allowedStyles", "styleDeclaration", "newStyles", "mergedStyles", "styleName", "styleUnit", "getStyleFromDeclaration", "isDisplayBlock", "style", "createRandomId", "createElement", "tag", "options", "element", "name", "value", "setStyles", "isElement", "nodeName", "isOffsetAtStart", "node", "offset", "isOffsetAtEnd", "TAG", "createLineBreak", "isLineBreak", "node", "TAG", "TYPE", "QUERY", "STYLES", "isInline", "node", "isElement", "isLikeInline", "element", "createInline", "textOrLineBreak", "styles", "attrs", "createElement", "createRandomId", "createInlineFrom", "inline", "mergeStyles", "createEmptyInline", "createLineBreak", "setInlineStyles", "setStyles", "getInline", "isInlineStart", "offset", "isOffsetAtStart", "isInlineEnd", "isOffsetAtEnd", "splitInline", "textNode", "style", "newTextNode", "getInlinesFrom", "startInline", "inlines", "currentInline", "index", "getInlineLength", "isLineBreak", "TAG", "TYPE", "QUERY", "STYLES", "isRoot", "node", "isElement", "createRoot", "paragraphs", "styles", "attrs", "isParagraph", "createElement", "createRandomId", "createEmptyRoot", "createEmptyParagraph", "setRootStyles", "element", "setStyles", "isTextNode", "node", "isLineBreak", "getTextNodeLength", "node", "isLineBreak", "getClosestTextNode", "isTextNode", "isInline", "isParagraph", "isRoot", "TAG", "TYPE", "QUERY", "STYLES", "isLikeParagraph", "element", "isLikeInline", "isEmptyParagraph", "isParagraph", "inline", "isInline", "isLineBreak", "node", "isElement", "createParagraph", "inlines", "styles", "attrs", "createElement", "createRandomId", "createEmptyParagraph", "createEmptyInline", "setParagraphStyles", "setStyles", "getParagraph", "paragraph", "isParagraphStart", "offset", "getInline", "isOffsetAtStart", "isParagraphEnd", "isOffsetAtEnd", "splitParagraph", "style", "isInlineEnd", "getInlinesFrom", "newInline", "splitInline", "mergeParagraphs", "a", "b", "mapContentFragmentFromDocument", "document", "root", "styleDefaults", "nodeIterator", "fragment", "currentParagraph", "currentNode", "parentStyle", "normalizeStyles", "mergeStyleDeclarations", "getComputedStyle", "isDisplayBlock", "isLikeParagraph", "createParagraph", "createInline", "mapContentFragmentFromHTML", "html", "htmlDocument", "mapContentFragmentFromString", "string", "lines", "line", "createEmptyParagraph", "paste", "event", "editor", "selectionController", "fragment", "html", "mapContentFragmentFromHTML", "plain", "mapContentFragmentFromString", "clipboard_default", "copy", "cut", "paste", "insertText", "event", "editor", "selectionController", "insertParagraph", "event", "editor", "selectionController", "deleteByCut", "event", "editor", "selectionController", "deleteContentBackward", "event", "editor", "selectionController", "deleteContentForward", "event", "editor", "selectionController", "commands_default", "insertText", "insertParagraph", "deleteByCut", "deleteContentBackward", "deleteContentForward", "_timeout", "_time", "_hasPendingChanges", "_onTimeout", "ChangeController", "time", "__privateAdd", "__privateSet", "__privateGet", "ChangeController_default", "tryOffset", "offset", "tryString", "str", "insertInto", "text", "replaceWith", "startOffset", "endOffset", "removeBackward", "removeForward", "removeSlice", "start", "end", "TextNodeIteratorDirection", "_rootNode", "_currentNode", "_TextNodeIterator", "rootNode", "__privateAdd", "__privateSet", "node", "startNode", "skipNodes", "direction", "safeGuard", "currentNode", "backTrack", "found", "__privateGet", "newCurrentNode", "isContained", "nextNode", "previousNode", "TextNodeIterator", "TextNodeIterator_default", "_added", "_removed", "_updated", "CommandMutations", "added", "updated", "removed", "__privateAdd", "__privateSet", "__privateGet", "node", "CommandMutations_default", "SelectionDirection", "startTime", "start", "update", "SafeGuard_default", "_textEditor", "_selection", "_ranges", "_range", "_focusNode", "_focusOffset", "_anchorNode", "_anchorOffset", "_savedSelection", "_textNodeIterator", "_currentStyle", "_inertElement", "_debug", "_mutations", "_styleDefaults", "_SelectionController_instances", "applyDefaultStylesToCurrentStyle_fn", "applyStylesToCurrentStyle_fn", "updateCurrentStyle_fn", "_onSelectionChange", "notifyStyleChange_fn", "setup_fn", "getSavedRange_fn", "applyStylesTo_fn", "SelectionController", "textEditor", "selection", "options", "__privateAdd", "CommandMutations_default", "e", "focusNodeChanges", "anchorNodeChanges", "__privateGet", "__privateSet", "index", "range", "__privateMethod", "TextNodeIterator_default", "node", "offset", "nodeOffset", "anchorNode", "anchorOffset", "focusNode", "focusOffset", "SelectionDirection", "getParagraph", "getInline", "startContainer", "endContainer", "isInline", "isParagraph", "isLineBreak", "isInlineStart", "isInlineEnd", "isParagraphStart", "isParagraphEnd", "fragment", "numParagraphs", "newParagraph", "splitParagraph", "text", "newText", "removedData", "removeForward", "paragraph", "inline", "nextTextNode", "lineBreak", "createLineBreak", "removeBackward", "previousTextNode", "getTextNodeLength", "insertInto", "startOffset", "endOffset", "replaceWith", "currentParagraph", "newTextNode", "createInline", "child", "createEmptyParagraph", "currentInline", "previousParagraph", "paragraphToBeRemoved", "previousInline", "previousOffset", "getInlineLength", "isEmptyParagraph", "mergeParagraphs", "nextParagraph", "nextInline", "nextOffset", "affectedParagraphs", "affectedInlines", "startNode", "getClosestTextNode", "endNode", "previousNode", "nextNode", "newNodeValue", "removeSlice", "startInline", "startParagraph", "endInline", "endParagraph", "SafeGuard_default", "currentNode", "shouldRemoveNodeCompletely", "isTextNode", "mergedParagraph", "newEmptyInline", "createEmptyInline", "newStyles", "name", "value", "element", "styleName", "styleValue", "root", "setRootStyles", "setParagraphStyles", "setInlineStyles", "midText", "endText", "midInline", "createInlineFrom", "expectedEndNode", "newInline", "splitInline", "SelectionController_default", "createSelectionImposterFromClientRects", "referenceRect", "clientRects", "fragment", "rect", "rectElement", "addEventListeners", "target", "object", "options", "type", "listener", "removeEventListeners", "LayoutType", "LayoutType_default", "_element", "_events", "_root", "_changeController", "_selectionController", "_selectionImposterElement", "_styleDefaults", "_TextEditor_instances", "setupElementProperties_fn", "setupRoot_fn", "_onChange", "_onStyleChange", "setup_fn", "createSelectionImposter_fn", "_onBlur", "_onFocus", "_onPaste", "_onCut", "_onCopy", "_onBeforeInput", "_onInput", "notifyLayout_fn", "TextEditor", "element", "options", "__privateAdd", "e", "__privateGet", "__privateMethod", "clipboard_default", "commands_default", "command", "mutations", "LayoutType_default", "__privateSet", "newRoot", "previousRoot", "isLineBreak", "args", "createRoot", "createParagraph", "text", "styles", "createEmptyInline", "createInline", "removeEventListeners", "createEmptyRoot", "ChangeController_default", "SelectionController_default", "addEventListeners", "rects", "rect", "createSelectionImposterFromClientRects", "type", "isEditor", "instance", "getRoot", "setRoot", "root", "create", "getCurrentStyle", "applyStylesToSelection", "dispose", "TextEditor_default"] -} diff --git a/frontend/yarn.lock b/frontend/yarn.lock index 17b8acb7b..b17d25357 100644 --- a/frontend/yarn.lock +++ b/frontend/yarn.lock @@ -322,6 +322,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/aix-ppc64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/aix-ppc64@npm:0.24.0" + conditions: os=aix & cpu=ppc64 + languageName: node + linkType: hard + "@esbuild/android-arm64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/android-arm64@npm:0.20.2" @@ -343,6 +350,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/android-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-arm64@npm:0.24.0" + conditions: os=android & cpu=arm64 + languageName: node + linkType: hard + "@esbuild/android-arm@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/android-arm@npm:0.20.2" @@ -364,6 +378,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/android-arm@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-arm@npm:0.24.0" + conditions: os=android & cpu=arm + languageName: node + linkType: hard + "@esbuild/android-x64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/android-x64@npm:0.20.2" @@ -385,6 +406,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/android-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/android-x64@npm:0.24.0" + conditions: os=android & cpu=x64 + languageName: node + linkType: hard + "@esbuild/darwin-arm64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/darwin-arm64@npm:0.20.2" @@ -406,6 +434,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/darwin-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/darwin-arm64@npm:0.24.0" + conditions: os=darwin & cpu=arm64 + languageName: node + linkType: hard + "@esbuild/darwin-x64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/darwin-x64@npm:0.20.2" @@ -427,6 +462,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/darwin-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/darwin-x64@npm:0.24.0" + conditions: os=darwin & cpu=x64 + languageName: node + linkType: hard + "@esbuild/freebsd-arm64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/freebsd-arm64@npm:0.20.2" @@ -448,6 +490,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/freebsd-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/freebsd-arm64@npm:0.24.0" + conditions: os=freebsd & cpu=arm64 + languageName: node + linkType: hard + "@esbuild/freebsd-x64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/freebsd-x64@npm:0.20.2" @@ -469,6 +518,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/freebsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/freebsd-x64@npm:0.24.0" + conditions: os=freebsd & cpu=x64 + languageName: node + linkType: hard + "@esbuild/linux-arm64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-arm64@npm:0.20.2" @@ -490,6 +546,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-arm64@npm:0.24.0" + conditions: os=linux & cpu=arm64 + languageName: node + linkType: hard + "@esbuild/linux-arm@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-arm@npm:0.20.2" @@ -511,6 +574,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-arm@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-arm@npm:0.24.0" + conditions: os=linux & cpu=arm + languageName: node + linkType: hard + "@esbuild/linux-ia32@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-ia32@npm:0.20.2" @@ -532,6 +602,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-ia32@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-ia32@npm:0.24.0" + conditions: os=linux & cpu=ia32 + languageName: node + linkType: hard + "@esbuild/linux-loong64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-loong64@npm:0.20.2" @@ -553,6 +630,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-loong64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-loong64@npm:0.24.0" + conditions: os=linux & cpu=loong64 + languageName: node + linkType: hard + "@esbuild/linux-mips64el@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-mips64el@npm:0.20.2" @@ -574,6 +658,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-mips64el@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-mips64el@npm:0.24.0" + conditions: os=linux & cpu=mips64el + languageName: node + linkType: hard + "@esbuild/linux-ppc64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-ppc64@npm:0.20.2" @@ -595,6 +686,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-ppc64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-ppc64@npm:0.24.0" + conditions: os=linux & cpu=ppc64 + languageName: node + linkType: hard + "@esbuild/linux-riscv64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-riscv64@npm:0.20.2" @@ -616,6 +714,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-riscv64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-riscv64@npm:0.24.0" + conditions: os=linux & cpu=riscv64 + languageName: node + linkType: hard + "@esbuild/linux-s390x@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-s390x@npm:0.20.2" @@ -637,6 +742,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-s390x@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-s390x@npm:0.24.0" + conditions: os=linux & cpu=s390x + languageName: node + linkType: hard + "@esbuild/linux-x64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/linux-x64@npm:0.20.2" @@ -658,6 +770,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/linux-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/linux-x64@npm:0.24.0" + conditions: os=linux & cpu=x64 + languageName: node + linkType: hard + "@esbuild/netbsd-x64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/netbsd-x64@npm:0.20.2" @@ -679,6 +798,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/netbsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/netbsd-x64@npm:0.24.0" + conditions: os=netbsd & cpu=x64 + languageName: node + linkType: hard + "@esbuild/openbsd-arm64@npm:0.23.1": version: 0.23.1 resolution: "@esbuild/openbsd-arm64@npm:0.23.1" @@ -686,6 +812,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/openbsd-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/openbsd-arm64@npm:0.24.0" + conditions: os=openbsd & cpu=arm64 + languageName: node + linkType: hard + "@esbuild/openbsd-x64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/openbsd-x64@npm:0.20.2" @@ -707,6 +840,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/openbsd-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/openbsd-x64@npm:0.24.0" + conditions: os=openbsd & cpu=x64 + languageName: node + linkType: hard + "@esbuild/sunos-x64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/sunos-x64@npm:0.20.2" @@ -728,6 +868,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/sunos-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/sunos-x64@npm:0.24.0" + conditions: os=sunos & cpu=x64 + languageName: node + linkType: hard + "@esbuild/win32-arm64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/win32-arm64@npm:0.20.2" @@ -749,6 +896,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/win32-arm64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-arm64@npm:0.24.0" + conditions: os=win32 & cpu=arm64 + languageName: node + linkType: hard + "@esbuild/win32-ia32@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/win32-ia32@npm:0.20.2" @@ -770,6 +924,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/win32-ia32@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-ia32@npm:0.24.0" + conditions: os=win32 & cpu=ia32 + languageName: node + linkType: hard + "@esbuild/win32-x64@npm:0.20.2": version: 0.20.2 resolution: "@esbuild/win32-x64@npm:0.20.2" @@ -791,6 +952,13 @@ __metadata: languageName: node linkType: hard +"@esbuild/win32-x64@npm:0.24.0": + version: 0.24.0 + resolution: "@esbuild/win32-x64@npm:0.24.0" + conditions: os=win32 & cpu=x64 + languageName: node + linkType: hard + "@gulp-sourcemaps/identity-map@npm:^2.0.1": version: 2.0.1 resolution: "@gulp-sourcemaps/identity-map@npm:2.0.1" @@ -1070,6 +1238,54 @@ __metadata: languageName: node linkType: hard +"@penpot/draft-js@file:./vendor/draft-js::locator=frontend%40workspace%3A.": + version: 1.0.0 + resolution: "@penpot/draft-js@file:./vendor/draft-js#./vendor/draft-js::hash=841941&locator=frontend%40workspace%3A." + dependencies: + draft-js: "penpot/draft-js.git#4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0" + peerDependencies: + react: ">=0.17.0" + react-dom: ">=0.17.0" + checksum: 10c0/49daa7a5ca10f585d6f3fb21a45a8a4809f4e9933b49758747cdcacfe7660ec0d184ae0cabecefcc62895382c43c548127408c1a6c5884c772a8cddbd74236be + languageName: node + linkType: hard + +"@penpot/hljs@file:./vendor/hljs::locator=frontend%40workspace%3A.": + version: 1.0.0 + resolution: "@penpot/hljs@file:./vendor/hljs#./vendor/hljs::hash=06811a&locator=frontend%40workspace%3A." + dependencies: + highlight.js: "npm:^11.10.0" + checksum: 10c0/6a6d213fbb9608fc70706ff04609f10744ebee614f25b6df664e411e1998603f0f0714b52aaabe1015ec5d48807f4a3ce4e80d73348bb0f11ddc588bf3a10c43 + languageName: node + linkType: hard + +"@penpot/mousetrap@file:./vendor/mousetrap::locator=frontend%40workspace%3A.": + version: 1.6.5 + resolution: "@penpot/mousetrap@file:./vendor/mousetrap#./vendor/mousetrap::hash=0dc831&locator=frontend%40workspace%3A." + checksum: 10c0/d23776bc184b42cc692e1ff2127e7ec0b94b4db33d87c21f1746f9f626e20edb1877083c1886cd71d4e3bcd4845943d0d9198016c0d3a53acecd237698d4d8d4 + languageName: node + linkType: hard + +"@penpot/svgo@penpot/svgo#c6fba7a4dcfbc27b643e7fc0c94fc98cf680b77b": + version: 4.0.0 + resolution: "@penpot/svgo@https://github.com/penpot/svgo.git#commit=c6fba7a4dcfbc27b643e7fc0c94fc98cf680b77b" + dependencies: + "@trysound/sax": "npm:0.2.0" + css-select: "npm:^5.1.0" + css-tree: "npm:^3.0.0" + csso: "npm:^5.0.5" + lodash: "npm:^4.17.21" + checksum: 10c0/af452f31196bcd237c390a12fea2da4c1d4005ae6d759c38f2169c9975c2178f85ec72077e96a8a40ded38748c2c1449dbdaf0d15f37ca3237622d766ac49ec8 + languageName: node + linkType: hard + +"@penpot/text-editor@penpot/penpot-text-editor#bae79a69c3d484f4a9271c2ed44022e36fce3cfb": + version: 0.0.0 + resolution: "@penpot/text-editor@https://github.com/penpot/penpot-text-editor.git#commit=bae79a69c3d484f4a9271c2ed44022e36fce3cfb" + checksum: 10c0/3559ddafb87df9197da00dc91fbe196ddf720ea00fa37e9fba2d9521ccd229f28c42ee0c942bd0abf1674ae43496a16f2c80d82e0ca14168cadfea2d3444d746 + languageName: node + linkType: hard + "@pkgjs/parseargs@npm:^0.11.0": version: 0.11.0 resolution: "@pkgjs/parseargs@npm:0.11.0" @@ -3746,6 +3962,13 @@ __metadata: languageName: node linkType: hard +"detect-europe-js@npm:^0.1.1": + version: 0.1.1 + resolution: "detect-europe-js@npm:0.1.1" + checksum: 10c0/4d283ca6ee4c297c6c199dab8536666f2a17f73efd846cbfd226a525c4a1de1de6afdb10dcac00eb36e5f3a095539d8d4c235ab24baa5230d9f562178a9eb2d4 + languageName: node + linkType: hard + "detect-libc@npm:^1.0.3": version: 1.0.3 resolution: "detect-libc@npm:1.0.3" @@ -3875,7 +4098,7 @@ __metadata: languageName: node linkType: hard -"draft-js@git+https://github.com/penpot/draft-js.git#commit=4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0": +"draft-js@penpot/draft-js.git#4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0": version: 0.11.7 resolution: "draft-js@https://github.com/penpot/draft-js.git#commit=4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0" dependencies: @@ -4434,6 +4657,89 @@ __metadata: languageName: node linkType: hard +"esbuild@npm:^0.24.0": + version: 0.24.0 + resolution: "esbuild@npm:0.24.0" + dependencies: + "@esbuild/aix-ppc64": "npm:0.24.0" + "@esbuild/android-arm": "npm:0.24.0" + "@esbuild/android-arm64": "npm:0.24.0" + "@esbuild/android-x64": "npm:0.24.0" + "@esbuild/darwin-arm64": "npm:0.24.0" + "@esbuild/darwin-x64": "npm:0.24.0" + "@esbuild/freebsd-arm64": "npm:0.24.0" + "@esbuild/freebsd-x64": "npm:0.24.0" + "@esbuild/linux-arm": "npm:0.24.0" + "@esbuild/linux-arm64": "npm:0.24.0" + "@esbuild/linux-ia32": "npm:0.24.0" + "@esbuild/linux-loong64": "npm:0.24.0" + "@esbuild/linux-mips64el": "npm:0.24.0" + "@esbuild/linux-ppc64": "npm:0.24.0" + "@esbuild/linux-riscv64": "npm:0.24.0" + "@esbuild/linux-s390x": "npm:0.24.0" + "@esbuild/linux-x64": "npm:0.24.0" + "@esbuild/netbsd-x64": "npm:0.24.0" + "@esbuild/openbsd-arm64": "npm:0.24.0" + "@esbuild/openbsd-x64": "npm:0.24.0" + "@esbuild/sunos-x64": "npm:0.24.0" + "@esbuild/win32-arm64": "npm:0.24.0" + "@esbuild/win32-ia32": "npm:0.24.0" + "@esbuild/win32-x64": "npm:0.24.0" + dependenciesMeta: + "@esbuild/aix-ppc64": + optional: true + "@esbuild/android-arm": + optional: true + "@esbuild/android-arm64": + optional: true + "@esbuild/android-x64": + optional: true + "@esbuild/darwin-arm64": + optional: true + "@esbuild/darwin-x64": + optional: true + "@esbuild/freebsd-arm64": + optional: true + "@esbuild/freebsd-x64": + optional: true + "@esbuild/linux-arm": + optional: true + "@esbuild/linux-arm64": + optional: true + "@esbuild/linux-ia32": + optional: true + "@esbuild/linux-loong64": + optional: true + "@esbuild/linux-mips64el": + optional: true + "@esbuild/linux-ppc64": + optional: true + "@esbuild/linux-riscv64": + optional: true + "@esbuild/linux-s390x": + optional: true + "@esbuild/linux-x64": + optional: true + "@esbuild/netbsd-x64": + optional: true + "@esbuild/openbsd-arm64": + optional: true + "@esbuild/openbsd-x64": + optional: true + "@esbuild/sunos-x64": + optional: true + "@esbuild/win32-arm64": + optional: true + "@esbuild/win32-ia32": + optional: true + "@esbuild/win32-x64": + optional: true + bin: + esbuild: bin/esbuild + checksum: 10c0/9f1aadd8d64f3bff422ae78387e66e51a5e09de6935a6f987b6e4e189ed00fdc2d1bc03d2e33633b094008529c8b6e06c7ad1a9782fb09fec223bf95998c0683 + languageName: node + linkType: hard + "escalade@npm:^3.1.1, escalade@npm:^3.1.2": version: 3.1.2 resolution: "escalade@npm:3.1.2" @@ -4884,6 +5190,11 @@ __metadata: version: 0.0.0-use.local resolution: "frontend@workspace:." dependencies: + "@penpot/draft-js": "file:./vendor/draft-js" + "@penpot/hljs": "file:./vendor/hljs" + "@penpot/mousetrap": "file:./vendor/mousetrap" + "@penpot/svgo": "penpot/svgo#c6fba7a4dcfbc27b643e7fc0c94fc98cf680b77b" + "@penpot/text-editor": "penpot/penpot-text-editor#bae79a69c3d484f4a9271c2ed44022e36fce3cfb" "@playwright/test": "npm:1.48.1" "@storybook/addon-essentials": "npm:^8.3.6" "@storybook/addon-themes": "npm:^8.3.6" @@ -4895,7 +5206,7 @@ __metadata: compression: "npm:^1.7.4" concurrently: "npm:^9.0.1" date-fns: "npm:^4.1.0" - draft-js: "git+https://github.com/penpot/draft-js.git#commit=4a99b2a6020b2af97f6dc5fa1b4275ec16b559a0" + esbuild: "npm:^0.24.0" eventsource-parser: "npm:^3.0.0" express: "npm:^4.21.1" fancy-log: "npm:^2.0.0" @@ -4909,7 +5220,6 @@ __metadata: gulp-sass: "npm:^5.1.0" gulp-sourcemaps: "npm:^3.0.0" gulp-svg-sprite: "npm:^2.0.3" - highlight.js: "npm:^11.10.0" js-beautify: "npm:^1.15.1" jsdom: "npm:^25.0.1" jszip: "npm:^3.10.1" @@ -4918,7 +5228,6 @@ __metadata: map-stream: "npm:0.0.7" marked: "npm:^14.1.3" mkdirp: "npm:^3.0.1" - mousetrap: "npm:^1.6.5" mustache: "npm:^4.2.0" nodemon: "npm:^3.1.7" npm-run-all: "npm:^4.1.5" @@ -4944,10 +5253,9 @@ __metadata: source-map-support: "npm:^0.5.21" storybook: "npm:^8.3.6" svg-sprite: "npm:^2.0.4" - svgo: "penpot/svgo#v3" tdigest: "npm:^0.1.2" typescript: "npm:^5.6.3" - ua-parser-js: "npm:^1.0.39" + ua-parser-js: "npm:2.0.0-rc.1" vite: "npm:^5.4.9" vitest: "npm:^2.1.3" wasm-pack: "npm:^0.13.0" @@ -6730,13 +7038,6 @@ __metadata: languageName: node linkType: hard -"mousetrap@npm:^1.6.5": - version: 1.6.5 - resolution: "mousetrap@npm:1.6.5" - checksum: 10c0/5c361bdbbff3966fd58d70f39b9fe1f8e32c78f3ce65989d83af7aad32a3a95313ce835a8dd8a55cb5de9eeb7c1f0c2b9048631a3073b5606241589e8fc0ba53 - languageName: node - linkType: hard - "ms@npm:2.0.0": version: 2.0.0 resolution: "ms@npm:2.0.0" @@ -9317,19 +9618,6 @@ __metadata: languageName: node linkType: hard -"svgo@penpot/svgo#v3": - version: 4.0.0 - resolution: "svgo@https://github.com/penpot/svgo.git#commit=71c0db44c3c2665f2ffc0c4c5383acaebd5c524f" - dependencies: - "@trysound/sax": "npm:0.2.0" - css-select: "npm:^5.1.0" - css-tree: "npm:^3.0.0" - csso: "npm:^5.0.5" - lodash: "npm:^4.17.21" - checksum: 10c0/642c583372a610e484382cbf8a8fe28256dd354598d2e65ade2a3a63bf841b4d3dab4106f929f183ae3610007db2fc1413e82acc23793fe1a2e882bc923acc72 - languageName: node - linkType: hard - "symbol-tree@npm:^3.2.4": version: 3.2.4 resolution: "symbol-tree@npm:3.2.4" @@ -9706,6 +9994,17 @@ __metadata: languageName: node linkType: hard +"ua-parser-js@npm:2.0.0-rc.1": + version: 2.0.0-rc.1 + resolution: "ua-parser-js@npm:2.0.0-rc.1" + dependencies: + detect-europe-js: "npm:^0.1.1" + bin: + ua-parser-js: script/cli.js + checksum: 10c0/7b57e397ec617eefb391ee67e5a09f6f94ab486e4c261d86eb35bea35a647a439c614dc6f9f348ff46c0293f18239701ebe1bcba7b60102fb07b9400266425ac + languageName: node + linkType: hard + "ua-parser-js@npm:^1.0.35": version: 1.0.37 resolution: "ua-parser-js@npm:1.0.37" @@ -9713,15 +10012,6 @@ __metadata: languageName: node linkType: hard -"ua-parser-js@npm:^1.0.39": - version: 1.0.39 - resolution: "ua-parser-js@npm:1.0.39" - bin: - ua-parser-js: script/cli.js - checksum: 10c0/c6452b0c683000f10975cb0a7e74cb1119ea95d4522ae85f396fa53b0b17884358a24ffdd86a66030c6b2981bdc502109a618c79fdaa217ee9032c9e46fcc78a - languageName: node - linkType: hard - "unbox-primitive@npm:^1.0.2": version: 1.0.2 resolution: "unbox-primitive@npm:1.0.2" From 39106c1e140a41a1f7fba647f92f7acdec3f0c52 Mon Sep 17 00:00:00 2001 From: Andrey Antukh Date: Fri, 1 Nov 2024 09:52:39 +0100 Subject: [PATCH 2/3] :sparkles: Improve e2e performance on CI --- .circleci/config.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 75955b882..3fb56cbef 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -99,10 +99,10 @@ jobs: - image: penpotapp/devenv:latest working_directory: ~/repo - resource_class: medium+ + resource_class: large environment: - JAVA_OPTS: -Xmx4g -Xms100m -XX:+UseSerialGC + JAVA_OPTS: -Xmx6g -Xms2g NODE_OPTIONS: --max-old-space-size=4096 steps: @@ -122,7 +122,7 @@ jobs: yarn run build:app yarn run build:app:libs yarn run playwright install --with-deps chromium - yarn run test:e2e + yarn run test:e2e -x --workers=4 test-backend: docker: From 96102fc87867c1d78ab4395176849e3174bb53ea Mon Sep 17 00:00:00 2001 From: Andrey Antukh Date: Mon, 4 Nov 2024 14:31:59 +0100 Subject: [PATCH 3/3] :sparkles: Update devenv default watch command for handle libs --- docker/devenv/files/start-tmux.sh | 2 +- frontend/package.json | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/docker/devenv/files/start-tmux.sh b/docker/devenv/files/start-tmux.sh index ec6470ea2..b5198358a 100755 --- a/docker/devenv/files/start-tmux.sh +++ b/docker/devenv/files/start-tmux.sh @@ -26,7 +26,7 @@ tmux send-keys -t penpot 'yarn run watch' enter tmux new-window -t penpot:1 -n 'frontend shadow' tmux select-window -t penpot:1 tmux send-keys -t penpot 'cd penpot/frontend' enter C-l -tmux send-keys -t penpot 'clojure -M:dev:shadow-cljs watch main' enter +tmux send-keys -t penpot 'yarn run watch:app' enter tmux new-window -t penpot:2 -n 'exporter' tmux select-window -t penpot:2 diff --git a/frontend/package.json b/frontend/package.json index 4d5a2cf3f..1c2651729 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -37,6 +37,7 @@ "translations": "node ./scripts/translations.js", "build:app:libs": "esbuild --bundle --outfile=resources/public/js/libs.js --format=iife target/index.js --minify", "watch:app:libs": "esbuild --bundle --outfile=resources/public/js/libs.js --format=iife target/index.js --watch", + "watch:app": "concurrently \"clojure -M:dev:shadow-cljs watch main worker\" \"yarn run watch:app:libs\"", "watch": "yarn run watch:app:assets", "watch:app:assets": "node ./scripts/watch.js", "watch:storybook": "concurrently \"clojure -M:dev:shadow-cljs watch storybook\" \"storybook dev -p 6006 --no-open\" \"yarn run watch:storybook:assets\"",