2021-10-18 14:01:15 +02:00
|
|
|
<!--
|
|
|
|
Copyright: Ankitects Pty Ltd and contributors
|
|
|
|
License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
-->
|
2022-02-03 05:52:11 +01:00
|
|
|
<script context="module" lang="ts">
|
|
|
|
import type { Writable } from "svelte/store";
|
2022-02-04 09:36:34 +01:00
|
|
|
|
Field redesign (#2002)
* Adjust size of legacy buttons
* Revert "Adjust size of legacy buttons"
This reverts commit fb888fe1db9050c34b1a7b0820e6da5ac91ccee6.
* Remove unused function from #1476
* Use outline version for tag icon
* Add chevron icons
* Remove code icons, keep one pin icon version
* Add code-bg color
* Redesign fields
* Remove unused import
* Fix imports
* Move PlainTextBadge between editing inputs
where it belongs :)
* Make whole separator line clickable
* Fix transition
and format
* Don't show toggle when field is collapsed
* Show toggle only on hover
for mobile I'd like to implement a swipe mechanism.
* Use tweened SVG for triangle instead of CSS hack
* Implement more obvious HTML toggle on bottom right
* Reduce field height by a few pixels
* Reduce field height by two pixels
* Show HTML toggle when PlainTextInput is active, regardless of hover/focus
* Remove RichTextBadge.svelte
* Create separate collapsed field state
this means users can collapse fields with the HTML editor open and it will stay open when the field is expanded again.
* Add slide out animation to EditingArea, RichTextInput and PlainTextInput
only for collapsing, because it is choppy on expansion (common issue with Svelte transitions).
* Fix aliasing issue on focused field corners
* Make StickyBadge feel more responsive
* Move StickyBadge closer to field border
* Adjust field gutter/margins
* Make LabelContainer sticky
to make field operations accessible on fields with a lot of content.
* Add back html icons, remove visual editor icons
* Revert "Add code-bg color"
This reverts commit 4200f354193710b3acd9bcf84b67958e200ddcdb.
* Add rich text icon, remove strikethrough code icon
* Revert PlainTextBadge to original position
* Adjust margins in FieldState
* Rename PlainTextBadge to SecondaryInputBadge
in preparation for #1987
* Run eslint and prettier
* Make whole LabelContainer clickable area for collapse/expand
* Revert "Add slide out animation to EditingArea, RichTextInput and PlainTextInput"
This reverts commit 9a2b3410d0ead37ae1da408d68e14507a058a613.
* Fix error on collapse/expansion
this was caused by the {#if} blocks, which resulted in the deletion of original EditingAreas.
* Refocus when toggling chevron and secondary input badge
* Revert "Revert "Add code-bg color""
This reverts commit 1cfd3bda65354ab90c1ab4cbbef47596a1be8754.
* Use single rotating chevron icon and make it RTL-compatible
* Remove redundant CSS transition rule
* Introduce animated Collapsible component and fix refocus on toggle
* Do not try to force repaint, as it is not required
* Remove RTL store from LabelContainer
the direction is already applied globally.
* Collapse secondary input with field
* Add focusedField to NoteEditorAPI
* Replace :global CSS selector with class .visible
thus removing the assumption that the component is used inside an EditorField.
https://github.com/ankitects/anki/pull/2002#discussion_r944876448
* Use named function syntax instead of function expressions
* Add explanation comment
* Remove unnecessary :bind directive
* Create CollapseBadge component
* Move :global selector into .plain-text-input
* Add comment explaining box-shadow pseudo-element
* Move Collapsible from EditingArea, PlainTextInput and RichTextInput into user components
* Rename SecondaryInputBadge to PlainTextBadge and remove generalization logic
I kept the rich text icon inside icons.ts for future use.
* Sort imports
* Fix background-color for duplicates not showing
with yet another pseudo-element :)
The pseudo-element that covers up field borders on scroll caused this issue. Fighting fire with fire here.
* Increase size of plain text toggle to original value again
This makes the clickable area a bit bigger and looks slightly more consistent with StickyBadge.
* Scrap pseudo-element mess in LabelContainer and tackle the actual issue
* Add class .visible to StickyBadge too
This introduces a peculiar bug: The active prop of StickyBadge resets to false when the mouse leaves the field - regardless of the actual back-end value.
* Fix sticky badge resetting on mouseleave/blur
* Apply overflow: hidden only during transition
fixes MathJax handle getting cut off by fields
* Remove unused variable
* Fix visual bug caused by overflow:hidden not applying in time
I tried several asynchronous approaches, but they all caused issues: either they prevented the CSS transition or they made field inputs lose focus.
In the end I resorted to direct, synchronous DOM-manipulation and added an explanatory comment.
* Decrease Collapsible load time by blocking first transition
I noticed the sliding animation has a hefty performance impact when a large number of fields is loaded simultaneously.
Blocking the first transition (which isn't even visible) results in a big boost in load time.
* Replace usages of gap with margins for children
* Revert unnecessary removal of grid-gap definition
* Correct comments about flex-gap property
mistook that for grid-gap.
* Resolve style issues
* Add minimum targets to gap comment
Co-authored-by: Henrik Giesel <hengiesel@gmail.com>
2022-08-19 02:02:28 +02:00
|
|
|
import Collapsible from "../components/Collapsible.svelte";
|
2022-02-03 05:52:11 +01:00
|
|
|
import type { EditingInputAPI } from "./EditingArea.svelte";
|
|
|
|
import type { EditorToolbarAPI } from "./editor-toolbar";
|
2022-02-04 09:36:34 +01:00
|
|
|
import type { EditorFieldAPI } from "./EditorField.svelte";
|
Field redesign (#2002)
* Adjust size of legacy buttons
* Revert "Adjust size of legacy buttons"
This reverts commit fb888fe1db9050c34b1a7b0820e6da5ac91ccee6.
* Remove unused function from #1476
* Use outline version for tag icon
* Add chevron icons
* Remove code icons, keep one pin icon version
* Add code-bg color
* Redesign fields
* Remove unused import
* Fix imports
* Move PlainTextBadge between editing inputs
where it belongs :)
* Make whole separator line clickable
* Fix transition
and format
* Don't show toggle when field is collapsed
* Show toggle only on hover
for mobile I'd like to implement a swipe mechanism.
* Use tweened SVG for triangle instead of CSS hack
* Implement more obvious HTML toggle on bottom right
* Reduce field height by a few pixels
* Reduce field height by two pixels
* Show HTML toggle when PlainTextInput is active, regardless of hover/focus
* Remove RichTextBadge.svelte
* Create separate collapsed field state
this means users can collapse fields with the HTML editor open and it will stay open when the field is expanded again.
* Add slide out animation to EditingArea, RichTextInput and PlainTextInput
only for collapsing, because it is choppy on expansion (common issue with Svelte transitions).
* Fix aliasing issue on focused field corners
* Make StickyBadge feel more responsive
* Move StickyBadge closer to field border
* Adjust field gutter/margins
* Make LabelContainer sticky
to make field operations accessible on fields with a lot of content.
* Add back html icons, remove visual editor icons
* Revert "Add code-bg color"
This reverts commit 4200f354193710b3acd9bcf84b67958e200ddcdb.
* Add rich text icon, remove strikethrough code icon
* Revert PlainTextBadge to original position
* Adjust margins in FieldState
* Rename PlainTextBadge to SecondaryInputBadge
in preparation for #1987
* Run eslint and prettier
* Make whole LabelContainer clickable area for collapse/expand
* Revert "Add slide out animation to EditingArea, RichTextInput and PlainTextInput"
This reverts commit 9a2b3410d0ead37ae1da408d68e14507a058a613.
* Fix error on collapse/expansion
this was caused by the {#if} blocks, which resulted in the deletion of original EditingAreas.
* Refocus when toggling chevron and secondary input badge
* Revert "Revert "Add code-bg color""
This reverts commit 1cfd3bda65354ab90c1ab4cbbef47596a1be8754.
* Use single rotating chevron icon and make it RTL-compatible
* Remove redundant CSS transition rule
* Introduce animated Collapsible component and fix refocus on toggle
* Do not try to force repaint, as it is not required
* Remove RTL store from LabelContainer
the direction is already applied globally.
* Collapse secondary input with field
* Add focusedField to NoteEditorAPI
* Replace :global CSS selector with class .visible
thus removing the assumption that the component is used inside an EditorField.
https://github.com/ankitects/anki/pull/2002#discussion_r944876448
* Use named function syntax instead of function expressions
* Add explanation comment
* Remove unnecessary :bind directive
* Create CollapseBadge component
* Move :global selector into .plain-text-input
* Add comment explaining box-shadow pseudo-element
* Move Collapsible from EditingArea, PlainTextInput and RichTextInput into user components
* Rename SecondaryInputBadge to PlainTextBadge and remove generalization logic
I kept the rich text icon inside icons.ts for future use.
* Sort imports
* Fix background-color for duplicates not showing
with yet another pseudo-element :)
The pseudo-element that covers up field borders on scroll caused this issue. Fighting fire with fire here.
* Increase size of plain text toggle to original value again
This makes the clickable area a bit bigger and looks slightly more consistent with StickyBadge.
* Scrap pseudo-element mess in LabelContainer and tackle the actual issue
* Add class .visible to StickyBadge too
This introduces a peculiar bug: The active prop of StickyBadge resets to false when the mouse leaves the field - regardless of the actual back-end value.
* Fix sticky badge resetting on mouseleave/blur
* Apply overflow: hidden only during transition
fixes MathJax handle getting cut off by fields
* Remove unused variable
* Fix visual bug caused by overflow:hidden not applying in time
I tried several asynchronous approaches, but they all caused issues: either they prevented the CSS transition or they made field inputs lose focus.
In the end I resorted to direct, synchronous DOM-manipulation and added an explanatory comment.
* Decrease Collapsible load time by blocking first transition
I noticed the sliding animation has a hefty performance impact when a large number of fields is loaded simultaneously.
Blocking the first transition (which isn't even visible) results in a big boost in load time.
* Replace usages of gap with margins for children
* Revert unnecessary removal of grid-gap definition
* Correct comments about flex-gap property
mistook that for grid-gap.
* Resolve style issues
* Add minimum targets to gap comment
Co-authored-by: Henrik Giesel <hengiesel@gmail.com>
2022-08-19 02:02:28 +02:00
|
|
|
import FieldState from "./FieldState.svelte";
|
|
|
|
import LabelContainer from "./LabelContainer.svelte";
|
|
|
|
import LabelName from "./LabelName.svelte";
|
2022-02-03 05:52:11 +01:00
|
|
|
|
|
|
|
export interface NoteEditorAPI {
|
|
|
|
fields: EditorFieldAPI[];
|
Field redesign (#2002)
* Adjust size of legacy buttons
* Revert "Adjust size of legacy buttons"
This reverts commit fb888fe1db9050c34b1a7b0820e6da5ac91ccee6.
* Remove unused function from #1476
* Use outline version for tag icon
* Add chevron icons
* Remove code icons, keep one pin icon version
* Add code-bg color
* Redesign fields
* Remove unused import
* Fix imports
* Move PlainTextBadge between editing inputs
where it belongs :)
* Make whole separator line clickable
* Fix transition
and format
* Don't show toggle when field is collapsed
* Show toggle only on hover
for mobile I'd like to implement a swipe mechanism.
* Use tweened SVG for triangle instead of CSS hack
* Implement more obvious HTML toggle on bottom right
* Reduce field height by a few pixels
* Reduce field height by two pixels
* Show HTML toggle when PlainTextInput is active, regardless of hover/focus
* Remove RichTextBadge.svelte
* Create separate collapsed field state
this means users can collapse fields with the HTML editor open and it will stay open when the field is expanded again.
* Add slide out animation to EditingArea, RichTextInput and PlainTextInput
only for collapsing, because it is choppy on expansion (common issue with Svelte transitions).
* Fix aliasing issue on focused field corners
* Make StickyBadge feel more responsive
* Move StickyBadge closer to field border
* Adjust field gutter/margins
* Make LabelContainer sticky
to make field operations accessible on fields with a lot of content.
* Add back html icons, remove visual editor icons
* Revert "Add code-bg color"
This reverts commit 4200f354193710b3acd9bcf84b67958e200ddcdb.
* Add rich text icon, remove strikethrough code icon
* Revert PlainTextBadge to original position
* Adjust margins in FieldState
* Rename PlainTextBadge to SecondaryInputBadge
in preparation for #1987
* Run eslint and prettier
* Make whole LabelContainer clickable area for collapse/expand
* Revert "Add slide out animation to EditingArea, RichTextInput and PlainTextInput"
This reverts commit 9a2b3410d0ead37ae1da408d68e14507a058a613.
* Fix error on collapse/expansion
this was caused by the {#if} blocks, which resulted in the deletion of original EditingAreas.
* Refocus when toggling chevron and secondary input badge
* Revert "Revert "Add code-bg color""
This reverts commit 1cfd3bda65354ab90c1ab4cbbef47596a1be8754.
* Use single rotating chevron icon and make it RTL-compatible
* Remove redundant CSS transition rule
* Introduce animated Collapsible component and fix refocus on toggle
* Do not try to force repaint, as it is not required
* Remove RTL store from LabelContainer
the direction is already applied globally.
* Collapse secondary input with field
* Add focusedField to NoteEditorAPI
* Replace :global CSS selector with class .visible
thus removing the assumption that the component is used inside an EditorField.
https://github.com/ankitects/anki/pull/2002#discussion_r944876448
* Use named function syntax instead of function expressions
* Add explanation comment
* Remove unnecessary :bind directive
* Create CollapseBadge component
* Move :global selector into .plain-text-input
* Add comment explaining box-shadow pseudo-element
* Move Collapsible from EditingArea, PlainTextInput and RichTextInput into user components
* Rename SecondaryInputBadge to PlainTextBadge and remove generalization logic
I kept the rich text icon inside icons.ts for future use.
* Sort imports
* Fix background-color for duplicates not showing
with yet another pseudo-element :)
The pseudo-element that covers up field borders on scroll caused this issue. Fighting fire with fire here.
* Increase size of plain text toggle to original value again
This makes the clickable area a bit bigger and looks slightly more consistent with StickyBadge.
* Scrap pseudo-element mess in LabelContainer and tackle the actual issue
* Add class .visible to StickyBadge too
This introduces a peculiar bug: The active prop of StickyBadge resets to false when the mouse leaves the field - regardless of the actual back-end value.
* Fix sticky badge resetting on mouseleave/blur
* Apply overflow: hidden only during transition
fixes MathJax handle getting cut off by fields
* Remove unused variable
* Fix visual bug caused by overflow:hidden not applying in time
I tried several asynchronous approaches, but they all caused issues: either they prevented the CSS transition or they made field inputs lose focus.
In the end I resorted to direct, synchronous DOM-manipulation and added an explanatory comment.
* Decrease Collapsible load time by blocking first transition
I noticed the sliding animation has a hefty performance impact when a large number of fields is loaded simultaneously.
Blocking the first transition (which isn't even visible) results in a big boost in load time.
* Replace usages of gap with margins for children
* Revert unnecessary removal of grid-gap definition
* Correct comments about flex-gap property
mistook that for grid-gap.
* Resolve style issues
* Add minimum targets to gap comment
Co-authored-by: Henrik Giesel <hengiesel@gmail.com>
2022-08-19 02:02:28 +02:00
|
|
|
hoveredField: Writable<EditorFieldAPI | null>;
|
2022-02-03 05:52:11 +01:00
|
|
|
focusedField: Writable<EditorFieldAPI | null>;
|
|
|
|
focusedInput: Writable<EditingInputAPI | null>;
|
|
|
|
toolbar: EditorToolbarAPI;
|
|
|
|
}
|
|
|
|
|
2022-02-04 09:36:34 +01:00
|
|
|
import { registerPackage } from "../lib/runtime-require";
|
2022-02-03 05:52:11 +01:00
|
|
|
import contextProperty from "../sveltelib/context-property";
|
|
|
|
import lifecycleHooks from "../sveltelib/lifecycle-hooks";
|
|
|
|
|
|
|
|
const key = Symbol("noteEditor");
|
|
|
|
const [context, setContextProperty] = contextProperty<NoteEditorAPI>(key);
|
|
|
|
const [lifecycle, instances, setupLifecycleHooks] = lifecycleHooks<NoteEditorAPI>();
|
|
|
|
|
|
|
|
export { context };
|
|
|
|
|
|
|
|
registerPackage("anki/NoteEditor", {
|
|
|
|
context,
|
|
|
|
lifecycle,
|
|
|
|
instances,
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<script lang="ts">
|
2022-02-05 06:58:31 +01:00
|
|
|
import { onMount, tick } from "svelte";
|
2022-02-04 09:36:34 +01:00
|
|
|
import { get, writable } from "svelte/store";
|
|
|
|
|
2022-02-03 05:52:11 +01:00
|
|
|
import Absolute from "../components/Absolute.svelte";
|
|
|
|
import Badge from "../components/Badge.svelte";
|
2022-09-28 06:02:32 +02:00
|
|
|
import HorizontalResizer from "../components/HorizontalResizer.svelte";
|
|
|
|
import Pane from "../components/Pane.svelte";
|
|
|
|
import PaneContent from "../components/PaneContent.svelte";
|
|
|
|
import { ResizablePane } from "../components/types";
|
2022-02-03 05:52:11 +01:00
|
|
|
import { bridgeCommand } from "../lib/bridgecommand";
|
2022-06-01 12:26:16 +02:00
|
|
|
import { TagEditor } from "../tag-editor";
|
2022-09-28 06:02:32 +02:00
|
|
|
import TagAddButton from "../tag-editor/tag-options-button/TagAddButton.svelte";
|
2022-02-04 09:36:34 +01:00
|
|
|
import { ChangeTimer } from "./change-timer";
|
2022-02-03 05:52:11 +01:00
|
|
|
import DecoratedElements from "./DecoratedElements.svelte";
|
2022-02-04 09:36:34 +01:00
|
|
|
import { clearableArray } from "./destroyable";
|
|
|
|
import DuplicateLink from "./DuplicateLink.svelte";
|
2022-08-15 05:34:16 +02:00
|
|
|
import EditorToolbar from "./editor-toolbar";
|
2022-02-04 09:36:34 +01:00
|
|
|
import type { FieldData } from "./EditorField.svelte";
|
|
|
|
import EditorField from "./EditorField.svelte";
|
2022-08-15 05:34:16 +02:00
|
|
|
import FieldDescription from "./FieldDescription.svelte";
|
2022-02-04 09:36:34 +01:00
|
|
|
import Fields from "./Fields.svelte";
|
|
|
|
import FrameElement from "./FrameElement.svelte";
|
|
|
|
import { alertIcon } from "./icons";
|
2022-08-15 05:34:16 +02:00
|
|
|
import ImageHandle from "./image-overlay";
|
2022-09-26 01:47:50 +02:00
|
|
|
import { shrinkImagesByDefault } from "./image-overlay/ImageOverlay.svelte";
|
2022-08-15 05:34:16 +02:00
|
|
|
import MathjaxHandle from "./mathjax-overlay";
|
2022-02-03 05:52:11 +01:00
|
|
|
import MathjaxElement from "./MathjaxElement.svelte";
|
2022-02-04 09:36:34 +01:00
|
|
|
import Notification from "./Notification.svelte";
|
2022-08-15 05:34:16 +02:00
|
|
|
import PlainTextInput from "./plain-text-input";
|
2022-10-03 05:14:57 +02:00
|
|
|
import { closeHTMLTags } from "./plain-text-input/PlainTextInput.svelte";
|
2022-02-03 05:52:11 +01:00
|
|
|
import PlainTextBadge from "./PlainTextBadge.svelte";
|
2022-08-15 05:34:16 +02:00
|
|
|
import RichTextInput, { editingInputIsRichText } from "./rich-text-input";
|
2022-08-31 15:34:39 +02:00
|
|
|
import RichTextBadge from "./RichTextBadge.svelte";
|
2022-09-10 10:46:59 +02:00
|
|
|
import SymbolsOverlay from "./symbols-overlay";
|
2022-02-03 05:52:11 +01:00
|
|
|
|
|
|
|
function quoteFontFamily(fontFamily: string): string {
|
|
|
|
// generic families (e.g. sans-serif) must not be quoted
|
|
|
|
if (!/^[-a-z]+$/.test(fontFamily)) {
|
|
|
|
fontFamily = `"${fontFamily}"`;
|
|
|
|
}
|
|
|
|
return fontFamily;
|
|
|
|
}
|
|
|
|
|
2022-05-10 03:48:11 +02:00
|
|
|
const size = 1.6;
|
2022-02-04 09:36:34 +01:00
|
|
|
const wrap = true;
|
2022-02-03 05:52:11 +01:00
|
|
|
|
2022-02-04 09:36:34 +01:00
|
|
|
const fieldStores: Writable<string>[] = [];
|
2022-02-03 05:52:11 +01:00
|
|
|
let fieldNames: string[] = [];
|
|
|
|
export function setFields(fs: [string, string][]): void {
|
|
|
|
// this is a bit of a mess -- when moving to Rust calls, we should make
|
|
|
|
// sure to have two backend endpoints for:
|
|
|
|
// * the note, which can be set through this view
|
|
|
|
// * the fieldname, font, etc., which cannot be set
|
|
|
|
|
|
|
|
const newFieldNames: string[] = [];
|
|
|
|
|
|
|
|
for (const [index, [fieldName]] of fs.entries()) {
|
|
|
|
newFieldNames[index] = fieldName;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = fieldStores.length; i < newFieldNames.length; i++) {
|
|
|
|
const newStore = writable("");
|
|
|
|
fieldStores[i] = newStore;
|
|
|
|
newStore.subscribe((value) => updateField(i, value));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (
|
|
|
|
let i = fieldStores.length;
|
|
|
|
i > newFieldNames.length;
|
|
|
|
i = fieldStores.length
|
|
|
|
) {
|
|
|
|
fieldStores.pop();
|
|
|
|
}
|
|
|
|
|
2022-02-04 09:36:34 +01:00
|
|
|
for (const [index, [, fieldContent]] of fs.entries()) {
|
2022-02-03 05:52:11 +01:00
|
|
|
fieldStores[index].set(fieldContent);
|
|
|
|
}
|
|
|
|
|
|
|
|
fieldNames = newFieldNames;
|
|
|
|
}
|
|
|
|
|
2022-08-31 15:34:39 +02:00
|
|
|
let fieldsCollapsed: boolean[] = [];
|
|
|
|
export function setCollapsed(fs: boolean[]): void {
|
|
|
|
fieldsCollapsed = fs;
|
|
|
|
}
|
|
|
|
|
2022-08-18 04:30:18 +02:00
|
|
|
let richTextsHidden: boolean[] = [];
|
|
|
|
let plainTextsHidden: boolean[] = [];
|
2022-08-31 15:34:39 +02:00
|
|
|
let plainTextDefaults: boolean[] = [];
|
2022-08-18 04:30:18 +02:00
|
|
|
|
|
|
|
export function setPlainTexts(fs: boolean[]): void {
|
2022-08-31 15:34:39 +02:00
|
|
|
richTextsHidden = fs;
|
2022-08-18 04:30:18 +02:00
|
|
|
plainTextsHidden = Array.from(fs, (v) => !v);
|
2022-08-31 15:34:39 +02:00
|
|
|
plainTextDefaults = [...richTextsHidden];
|
2022-08-18 04:30:18 +02:00
|
|
|
}
|
|
|
|
|
2022-08-18 04:06:06 +02:00
|
|
|
function setMathjaxEnabled(enabled: boolean): void {
|
|
|
|
mathjaxConfig.enabled = enabled;
|
|
|
|
}
|
|
|
|
|
2022-02-03 05:52:11 +01:00
|
|
|
let fieldDescriptions: string[] = [];
|
|
|
|
export function setDescriptions(fs: string[]): void {
|
|
|
|
fieldDescriptions = fs;
|
|
|
|
}
|
|
|
|
|
|
|
|
let fonts: [string, number, boolean][] = [];
|
Field redesign (#2002)
* Adjust size of legacy buttons
* Revert "Adjust size of legacy buttons"
This reverts commit fb888fe1db9050c34b1a7b0820e6da5ac91ccee6.
* Remove unused function from #1476
* Use outline version for tag icon
* Add chevron icons
* Remove code icons, keep one pin icon version
* Add code-bg color
* Redesign fields
* Remove unused import
* Fix imports
* Move PlainTextBadge between editing inputs
where it belongs :)
* Make whole separator line clickable
* Fix transition
and format
* Don't show toggle when field is collapsed
* Show toggle only on hover
for mobile I'd like to implement a swipe mechanism.
* Use tweened SVG for triangle instead of CSS hack
* Implement more obvious HTML toggle on bottom right
* Reduce field height by a few pixels
* Reduce field height by two pixels
* Show HTML toggle when PlainTextInput is active, regardless of hover/focus
* Remove RichTextBadge.svelte
* Create separate collapsed field state
this means users can collapse fields with the HTML editor open and it will stay open when the field is expanded again.
* Add slide out animation to EditingArea, RichTextInput and PlainTextInput
only for collapsing, because it is choppy on expansion (common issue with Svelte transitions).
* Fix aliasing issue on focused field corners
* Make StickyBadge feel more responsive
* Move StickyBadge closer to field border
* Adjust field gutter/margins
* Make LabelContainer sticky
to make field operations accessible on fields with a lot of content.
* Add back html icons, remove visual editor icons
* Revert "Add code-bg color"
This reverts commit 4200f354193710b3acd9bcf84b67958e200ddcdb.
* Add rich text icon, remove strikethrough code icon
* Revert PlainTextBadge to original position
* Adjust margins in FieldState
* Rename PlainTextBadge to SecondaryInputBadge
in preparation for #1987
* Run eslint and prettier
* Make whole LabelContainer clickable area for collapse/expand
* Revert "Add slide out animation to EditingArea, RichTextInput and PlainTextInput"
This reverts commit 9a2b3410d0ead37ae1da408d68e14507a058a613.
* Fix error on collapse/expansion
this was caused by the {#if} blocks, which resulted in the deletion of original EditingAreas.
* Refocus when toggling chevron and secondary input badge
* Revert "Revert "Add code-bg color""
This reverts commit 1cfd3bda65354ab90c1ab4cbbef47596a1be8754.
* Use single rotating chevron icon and make it RTL-compatible
* Remove redundant CSS transition rule
* Introduce animated Collapsible component and fix refocus on toggle
* Do not try to force repaint, as it is not required
* Remove RTL store from LabelContainer
the direction is already applied globally.
* Collapse secondary input with field
* Add focusedField to NoteEditorAPI
* Replace :global CSS selector with class .visible
thus removing the assumption that the component is used inside an EditorField.
https://github.com/ankitects/anki/pull/2002#discussion_r944876448
* Use named function syntax instead of function expressions
* Add explanation comment
* Remove unnecessary :bind directive
* Create CollapseBadge component
* Move :global selector into .plain-text-input
* Add comment explaining box-shadow pseudo-element
* Move Collapsible from EditingArea, PlainTextInput and RichTextInput into user components
* Rename SecondaryInputBadge to PlainTextBadge and remove generalization logic
I kept the rich text icon inside icons.ts for future use.
* Sort imports
* Fix background-color for duplicates not showing
with yet another pseudo-element :)
The pseudo-element that covers up field borders on scroll caused this issue. Fighting fire with fire here.
* Increase size of plain text toggle to original value again
This makes the clickable area a bit bigger and looks slightly more consistent with StickyBadge.
* Scrap pseudo-element mess in LabelContainer and tackle the actual issue
* Add class .visible to StickyBadge too
This introduces a peculiar bug: The active prop of StickyBadge resets to false when the mouse leaves the field - regardless of the actual back-end value.
* Fix sticky badge resetting on mouseleave/blur
* Apply overflow: hidden only during transition
fixes MathJax handle getting cut off by fields
* Remove unused variable
* Fix visual bug caused by overflow:hidden not applying in time
I tried several asynchronous approaches, but they all caused issues: either they prevented the CSS transition or they made field inputs lose focus.
In the end I resorted to direct, synchronous DOM-manipulation and added an explanatory comment.
* Decrease Collapsible load time by blocking first transition
I noticed the sliding animation has a hefty performance impact when a large number of fields is loaded simultaneously.
Blocking the first transition (which isn't even visible) results in a big boost in load time.
* Replace usages of gap with margins for children
* Revert unnecessary removal of grid-gap definition
* Correct comments about flex-gap property
mistook that for grid-gap.
* Resolve style issues
* Add minimum targets to gap comment
Co-authored-by: Henrik Giesel <hengiesel@gmail.com>
2022-08-19 02:02:28 +02:00
|
|
|
|
2022-02-04 09:36:34 +01:00
|
|
|
const fields = clearableArray<EditorFieldAPI>();
|
2022-02-03 05:52:11 +01:00
|
|
|
|
|
|
|
export function setFonts(fs: [string, number, boolean][]): void {
|
|
|
|
fonts = fs;
|
|
|
|
}
|
|
|
|
|
2022-02-05 06:58:31 +01:00
|
|
|
export function focusField(index: number | null): void {
|
|
|
|
tick().then(() => {
|
|
|
|
if (typeof index === "number") {
|
|
|
|
if (!(index in fields)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fields[index].editingArea?.refocus();
|
|
|
|
} else {
|
|
|
|
$focusedInput?.refocus();
|
|
|
|
}
|
|
|
|
});
|
2022-02-03 05:52:11 +01:00
|
|
|
}
|
|
|
|
|
2022-02-04 09:36:34 +01:00
|
|
|
const tags = writable<string[]>([]);
|
2022-02-03 05:52:11 +01:00
|
|
|
export function setTags(ts: string[]): void {
|
|
|
|
$tags = ts;
|
|
|
|
}
|
|
|
|
|
2022-09-28 06:02:32 +02:00
|
|
|
const tagsCollapsed = writable<boolean>();
|
|
|
|
export function setTagsCollapsed(collapsed: boolean): void {
|
|
|
|
$tagsCollapsed = collapsed;
|
|
|
|
if (collapsed) {
|
|
|
|
lowerResizer.move([tagsPane, fieldsPane], tagsPane.minHeight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 05:52:11 +01:00
|
|
|
let noteId: number | null = null;
|
|
|
|
export function setNoteId(ntid: number): void {
|
2022-02-04 09:40:50 +01:00
|
|
|
// TODO this is a hack, because it requires the NoteEditor to know implementation details of the PlainTextInput.
|
|
|
|
// It should be refactored once we work on our own Undo stack
|
|
|
|
for (const pi of plainTextInputs) {
|
2022-02-25 02:14:26 +01:00
|
|
|
pi.api.codeMirror.editor.then((editor) => editor.clearHistory());
|
2022-02-04 09:40:50 +01:00
|
|
|
}
|
2022-02-03 05:52:11 +01:00
|
|
|
noteId = ntid;
|
|
|
|
}
|
|
|
|
|
2022-09-10 10:46:59 +02:00
|
|
|
let insertSymbols = false;
|
|
|
|
|
|
|
|
function setInsertSymbolsEnabled() {
|
|
|
|
insertSymbols = true;
|
|
|
|
}
|
|
|
|
|
2022-02-03 05:52:11 +01:00
|
|
|
function getNoteId(): number | null {
|
|
|
|
return noteId;
|
|
|
|
}
|
|
|
|
|
|
|
|
let cols: ("dupe" | "")[] = [];
|
|
|
|
export function setBackgrounds(cls: ("dupe" | "")[]): void {
|
|
|
|
cols = cls;
|
|
|
|
}
|
|
|
|
|
|
|
|
let hint: string = "";
|
|
|
|
export function setClozeHint(hnt: string): void {
|
|
|
|
hint = hnt;
|
|
|
|
}
|
|
|
|
|
|
|
|
$: fieldsData = fieldNames.map((name, index) => ({
|
|
|
|
name,
|
2022-08-31 15:34:39 +02:00
|
|
|
plainText: plainTextDefaults[index],
|
2022-02-03 05:52:11 +01:00
|
|
|
description: fieldDescriptions[index],
|
|
|
|
fontFamily: quoteFontFamily(fonts[index][0]),
|
|
|
|
fontSize: fonts[index][1],
|
|
|
|
direction: fonts[index][2] ? "rtl" : "ltr",
|
2022-08-31 15:34:39 +02:00
|
|
|
collapsed: fieldsCollapsed[index],
|
2022-02-03 05:52:11 +01:00
|
|
|
})) as FieldData[];
|
|
|
|
|
|
|
|
function saveTags({ detail }: CustomEvent): void {
|
2022-09-28 06:02:32 +02:00
|
|
|
tagAmount = detail.tags.filter((tag: string) => tag != "").length;
|
2022-02-03 05:52:11 +01:00
|
|
|
bridgeCommand(`saveTags:${JSON.stringify(detail.tags)}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
const fieldSave = new ChangeTimer();
|
|
|
|
|
2022-09-26 01:47:50 +02:00
|
|
|
function transformContentBeforeSave(content: string): string {
|
|
|
|
return content.replace(/ data-editor-shrink="(true|false)"/g, "");
|
|
|
|
}
|
|
|
|
|
2022-02-03 05:52:11 +01:00
|
|
|
function updateField(index: number, content: string): void {
|
|
|
|
fieldSave.schedule(
|
2022-09-26 01:47:50 +02:00
|
|
|
() =>
|
|
|
|
bridgeCommand(
|
|
|
|
`key:${index}:${getNoteId()}:${transformContentBeforeSave(
|
|
|
|
content,
|
|
|
|
)}`,
|
|
|
|
),
|
2022-02-03 05:52:11 +01:00
|
|
|
600,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function saveFieldNow(): void {
|
|
|
|
/* this will always be a key save */
|
|
|
|
fieldSave.fireImmediately();
|
|
|
|
}
|
|
|
|
|
|
|
|
export function saveOnPageHide() {
|
|
|
|
if (document.visibilityState === "hidden") {
|
|
|
|
// will fire on session close and minimize
|
|
|
|
saveFieldNow();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function focusIfField(x: number, y: number): boolean {
|
|
|
|
const elements = document.elementsFromPoint(x, y);
|
|
|
|
const first = elements[0];
|
|
|
|
|
|
|
|
if (first.shadowRoot) {
|
|
|
|
const richTextInput = first.shadowRoot.lastElementChild! as HTMLElement;
|
|
|
|
richTextInput.focus();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
let richTextInputs: RichTextInput[] = [];
|
|
|
|
$: richTextInputs = richTextInputs.filter(Boolean);
|
|
|
|
|
|
|
|
let plainTextInputs: PlainTextInput[] = [];
|
|
|
|
$: plainTextInputs = plainTextInputs.filter(Boolean);
|
|
|
|
|
2022-02-04 09:36:34 +01:00
|
|
|
const toolbar: Partial<EditorToolbarAPI> = {};
|
2022-02-03 05:52:11 +01:00
|
|
|
|
2022-09-26 01:47:50 +02:00
|
|
|
function setShrinkImages(shrinkByDefault: boolean) {
|
|
|
|
$shrinkImagesByDefault = shrinkByDefault;
|
|
|
|
}
|
|
|
|
|
2022-10-03 05:14:57 +02:00
|
|
|
function setCloseHTMLTags(closeTags: boolean) {
|
|
|
|
$closeHTMLTags = closeTags;
|
|
|
|
}
|
|
|
|
|
2022-08-18 04:06:06 +02:00
|
|
|
import { mathjaxConfig } from "../editable/mathjax-element";
|
2022-02-03 05:52:11 +01:00
|
|
|
import { wrapInternal } from "../lib/wrap";
|
2022-08-31 15:34:39 +02:00
|
|
|
import { refocusInput } from "./helpers";
|
2022-02-03 05:52:11 +01:00
|
|
|
import * as oldEditorAdapter from "./old-editor-adapter";
|
|
|
|
|
|
|
|
onMount(() => {
|
|
|
|
function wrap(before: string, after: string): void {
|
|
|
|
if (!$focusedInput || !editingInputIsRichText($focusedInput)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$focusedInput.element.then((element) => {
|
|
|
|
wrapInternal(element, before, after, false);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Object.assign(globalThis, {
|
|
|
|
setFields,
|
2022-08-31 15:34:39 +02:00
|
|
|
setCollapsed,
|
2022-08-18 04:30:18 +02:00
|
|
|
setPlainTexts,
|
2022-02-03 05:52:11 +01:00
|
|
|
setDescriptions,
|
|
|
|
setFonts,
|
|
|
|
focusField,
|
|
|
|
setTags,
|
2022-09-28 06:02:32 +02:00
|
|
|
setTagsCollapsed,
|
2022-02-03 05:52:11 +01:00
|
|
|
setBackgrounds,
|
|
|
|
setClozeHint,
|
|
|
|
saveNow: saveFieldNow,
|
|
|
|
focusIfField,
|
2022-05-23 01:14:36 +02:00
|
|
|
getNoteId,
|
2022-02-03 05:52:11 +01:00
|
|
|
setNoteId,
|
|
|
|
wrap,
|
2022-08-18 04:06:06 +02:00
|
|
|
setMathjaxEnabled,
|
2022-09-10 10:46:59 +02:00
|
|
|
setInsertSymbolsEnabled,
|
2022-09-26 01:47:50 +02:00
|
|
|
setShrinkImages,
|
2022-10-03 05:14:57 +02:00
|
|
|
setCloseHTMLTags,
|
2022-02-03 05:52:11 +01:00
|
|
|
...oldEditorAdapter,
|
|
|
|
});
|
|
|
|
|
|
|
|
document.addEventListener("visibilitychange", saveOnPageHide);
|
|
|
|
return () => document.removeEventListener("visibilitychange", saveOnPageHide);
|
|
|
|
});
|
|
|
|
|
|
|
|
let apiPartial: Partial<NoteEditorAPI> = {};
|
|
|
|
export { apiPartial as api };
|
|
|
|
|
Field redesign (#2002)
* Adjust size of legacy buttons
* Revert "Adjust size of legacy buttons"
This reverts commit fb888fe1db9050c34b1a7b0820e6da5ac91ccee6.
* Remove unused function from #1476
* Use outline version for tag icon
* Add chevron icons
* Remove code icons, keep one pin icon version
* Add code-bg color
* Redesign fields
* Remove unused import
* Fix imports
* Move PlainTextBadge between editing inputs
where it belongs :)
* Make whole separator line clickable
* Fix transition
and format
* Don't show toggle when field is collapsed
* Show toggle only on hover
for mobile I'd like to implement a swipe mechanism.
* Use tweened SVG for triangle instead of CSS hack
* Implement more obvious HTML toggle on bottom right
* Reduce field height by a few pixels
* Reduce field height by two pixels
* Show HTML toggle when PlainTextInput is active, regardless of hover/focus
* Remove RichTextBadge.svelte
* Create separate collapsed field state
this means users can collapse fields with the HTML editor open and it will stay open when the field is expanded again.
* Add slide out animation to EditingArea, RichTextInput and PlainTextInput
only for collapsing, because it is choppy on expansion (common issue with Svelte transitions).
* Fix aliasing issue on focused field corners
* Make StickyBadge feel more responsive
* Move StickyBadge closer to field border
* Adjust field gutter/margins
* Make LabelContainer sticky
to make field operations accessible on fields with a lot of content.
* Add back html icons, remove visual editor icons
* Revert "Add code-bg color"
This reverts commit 4200f354193710b3acd9bcf84b67958e200ddcdb.
* Add rich text icon, remove strikethrough code icon
* Revert PlainTextBadge to original position
* Adjust margins in FieldState
* Rename PlainTextBadge to SecondaryInputBadge
in preparation for #1987
* Run eslint and prettier
* Make whole LabelContainer clickable area for collapse/expand
* Revert "Add slide out animation to EditingArea, RichTextInput and PlainTextInput"
This reverts commit 9a2b3410d0ead37ae1da408d68e14507a058a613.
* Fix error on collapse/expansion
this was caused by the {#if} blocks, which resulted in the deletion of original EditingAreas.
* Refocus when toggling chevron and secondary input badge
* Revert "Revert "Add code-bg color""
This reverts commit 1cfd3bda65354ab90c1ab4cbbef47596a1be8754.
* Use single rotating chevron icon and make it RTL-compatible
* Remove redundant CSS transition rule
* Introduce animated Collapsible component and fix refocus on toggle
* Do not try to force repaint, as it is not required
* Remove RTL store from LabelContainer
the direction is already applied globally.
* Collapse secondary input with field
* Add focusedField to NoteEditorAPI
* Replace :global CSS selector with class .visible
thus removing the assumption that the component is used inside an EditorField.
https://github.com/ankitects/anki/pull/2002#discussion_r944876448
* Use named function syntax instead of function expressions
* Add explanation comment
* Remove unnecessary :bind directive
* Create CollapseBadge component
* Move :global selector into .plain-text-input
* Add comment explaining box-shadow pseudo-element
* Move Collapsible from EditingArea, PlainTextInput and RichTextInput into user components
* Rename SecondaryInputBadge to PlainTextBadge and remove generalization logic
I kept the rich text icon inside icons.ts for future use.
* Sort imports
* Fix background-color for duplicates not showing
with yet another pseudo-element :)
The pseudo-element that covers up field borders on scroll caused this issue. Fighting fire with fire here.
* Increase size of plain text toggle to original value again
This makes the clickable area a bit bigger and looks slightly more consistent with StickyBadge.
* Scrap pseudo-element mess in LabelContainer and tackle the actual issue
* Add class .visible to StickyBadge too
This introduces a peculiar bug: The active prop of StickyBadge resets to false when the mouse leaves the field - regardless of the actual back-end value.
* Fix sticky badge resetting on mouseleave/blur
* Apply overflow: hidden only during transition
fixes MathJax handle getting cut off by fields
* Remove unused variable
* Fix visual bug caused by overflow:hidden not applying in time
I tried several asynchronous approaches, but they all caused issues: either they prevented the CSS transition or they made field inputs lose focus.
In the end I resorted to direct, synchronous DOM-manipulation and added an explanatory comment.
* Decrease Collapsible load time by blocking first transition
I noticed the sliding animation has a hefty performance impact when a large number of fields is loaded simultaneously.
Blocking the first transition (which isn't even visible) results in a big boost in load time.
* Replace usages of gap with margins for children
* Revert unnecessary removal of grid-gap definition
* Correct comments about flex-gap property
mistook that for grid-gap.
* Resolve style issues
* Add minimum targets to gap comment
Co-authored-by: Henrik Giesel <hengiesel@gmail.com>
2022-08-19 02:02:28 +02:00
|
|
|
const hoveredField: NoteEditorAPI["hoveredField"] = writable(null);
|
2022-02-03 05:52:11 +01:00
|
|
|
const focusedField: NoteEditorAPI["focusedField"] = writable(null);
|
|
|
|
const focusedInput: NoteEditorAPI["focusedInput"] = writable(null);
|
|
|
|
|
|
|
|
const api: NoteEditorAPI = {
|
|
|
|
...apiPartial,
|
Field redesign (#2002)
* Adjust size of legacy buttons
* Revert "Adjust size of legacy buttons"
This reverts commit fb888fe1db9050c34b1a7b0820e6da5ac91ccee6.
* Remove unused function from #1476
* Use outline version for tag icon
* Add chevron icons
* Remove code icons, keep one pin icon version
* Add code-bg color
* Redesign fields
* Remove unused import
* Fix imports
* Move PlainTextBadge between editing inputs
where it belongs :)
* Make whole separator line clickable
* Fix transition
and format
* Don't show toggle when field is collapsed
* Show toggle only on hover
for mobile I'd like to implement a swipe mechanism.
* Use tweened SVG for triangle instead of CSS hack
* Implement more obvious HTML toggle on bottom right
* Reduce field height by a few pixels
* Reduce field height by two pixels
* Show HTML toggle when PlainTextInput is active, regardless of hover/focus
* Remove RichTextBadge.svelte
* Create separate collapsed field state
this means users can collapse fields with the HTML editor open and it will stay open when the field is expanded again.
* Add slide out animation to EditingArea, RichTextInput and PlainTextInput
only for collapsing, because it is choppy on expansion (common issue with Svelte transitions).
* Fix aliasing issue on focused field corners
* Make StickyBadge feel more responsive
* Move StickyBadge closer to field border
* Adjust field gutter/margins
* Make LabelContainer sticky
to make field operations accessible on fields with a lot of content.
* Add back html icons, remove visual editor icons
* Revert "Add code-bg color"
This reverts commit 4200f354193710b3acd9bcf84b67958e200ddcdb.
* Add rich text icon, remove strikethrough code icon
* Revert PlainTextBadge to original position
* Adjust margins in FieldState
* Rename PlainTextBadge to SecondaryInputBadge
in preparation for #1987
* Run eslint and prettier
* Make whole LabelContainer clickable area for collapse/expand
* Revert "Add slide out animation to EditingArea, RichTextInput and PlainTextInput"
This reverts commit 9a2b3410d0ead37ae1da408d68e14507a058a613.
* Fix error on collapse/expansion
this was caused by the {#if} blocks, which resulted in the deletion of original EditingAreas.
* Refocus when toggling chevron and secondary input badge
* Revert "Revert "Add code-bg color""
This reverts commit 1cfd3bda65354ab90c1ab4cbbef47596a1be8754.
* Use single rotating chevron icon and make it RTL-compatible
* Remove redundant CSS transition rule
* Introduce animated Collapsible component and fix refocus on toggle
* Do not try to force repaint, as it is not required
* Remove RTL store from LabelContainer
the direction is already applied globally.
* Collapse secondary input with field
* Add focusedField to NoteEditorAPI
* Replace :global CSS selector with class .visible
thus removing the assumption that the component is used inside an EditorField.
https://github.com/ankitects/anki/pull/2002#discussion_r944876448
* Use named function syntax instead of function expressions
* Add explanation comment
* Remove unnecessary :bind directive
* Create CollapseBadge component
* Move :global selector into .plain-text-input
* Add comment explaining box-shadow pseudo-element
* Move Collapsible from EditingArea, PlainTextInput and RichTextInput into user components
* Rename SecondaryInputBadge to PlainTextBadge and remove generalization logic
I kept the rich text icon inside icons.ts for future use.
* Sort imports
* Fix background-color for duplicates not showing
with yet another pseudo-element :)
The pseudo-element that covers up field borders on scroll caused this issue. Fighting fire with fire here.
* Increase size of plain text toggle to original value again
This makes the clickable area a bit bigger and looks slightly more consistent with StickyBadge.
* Scrap pseudo-element mess in LabelContainer and tackle the actual issue
* Add class .visible to StickyBadge too
This introduces a peculiar bug: The active prop of StickyBadge resets to false when the mouse leaves the field - regardless of the actual back-end value.
* Fix sticky badge resetting on mouseleave/blur
* Apply overflow: hidden only during transition
fixes MathJax handle getting cut off by fields
* Remove unused variable
* Fix visual bug caused by overflow:hidden not applying in time
I tried several asynchronous approaches, but they all caused issues: either they prevented the CSS transition or they made field inputs lose focus.
In the end I resorted to direct, synchronous DOM-manipulation and added an explanatory comment.
* Decrease Collapsible load time by blocking first transition
I noticed the sliding animation has a hefty performance impact when a large number of fields is loaded simultaneously.
Blocking the first transition (which isn't even visible) results in a big boost in load time.
* Replace usages of gap with margins for children
* Revert unnecessary removal of grid-gap definition
* Correct comments about flex-gap property
mistook that for grid-gap.
* Resolve style issues
* Add minimum targets to gap comment
Co-authored-by: Henrik Giesel <hengiesel@gmail.com>
2022-08-19 02:02:28 +02:00
|
|
|
hoveredField,
|
2022-02-03 05:52:11 +01:00
|
|
|
focusedField,
|
|
|
|
focusedInput,
|
|
|
|
toolbar: toolbar as EditorToolbarAPI,
|
|
|
|
fields,
|
|
|
|
};
|
|
|
|
|
|
|
|
setContextProperty(api);
|
|
|
|
setupLifecycleHooks(api);
|
2022-09-28 06:02:32 +02:00
|
|
|
|
|
|
|
let clientHeight: number;
|
|
|
|
|
|
|
|
const fieldsPane = new ResizablePane();
|
|
|
|
const tagsPane = new ResizablePane();
|
|
|
|
|
|
|
|
let lowerResizer: HorizontalResizer;
|
|
|
|
let tagEditor: TagEditor;
|
|
|
|
|
|
|
|
$: tagAmount = $tags.length;
|
|
|
|
|
2022-10-13 02:13:19 +02:00
|
|
|
let snapTags = $tagsCollapsed;
|
|
|
|
|
2022-09-28 06:02:32 +02:00
|
|
|
function collapseTags(): void {
|
|
|
|
lowerResizer.move([tagsPane, fieldsPane], tagsPane.minHeight);
|
2022-10-13 02:13:19 +02:00
|
|
|
$tagsCollapsed = snapTags = true;
|
2022-09-28 06:02:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function expandTags(): void {
|
|
|
|
lowerResizer.move([tagsPane, fieldsPane], tagsPane.maxHeight);
|
2022-10-13 02:13:19 +02:00
|
|
|
$tagsCollapsed = snapTags = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
window.addEventListener("resize", () => snapResizer(snapTags));
|
|
|
|
|
|
|
|
function snapResizer(collapse: boolean): void {
|
|
|
|
if (collapse) {
|
|
|
|
collapseTags();
|
|
|
|
bridgeCommand("collapseTags");
|
|
|
|
} else {
|
|
|
|
expandTags();
|
|
|
|
bridgeCommand("expandTags");
|
|
|
|
}
|
2022-09-28 06:02:32 +02:00
|
|
|
}
|
2022-02-03 05:52:11 +01:00
|
|
|
</script>
|
|
|
|
|
2022-04-19 09:20:26 +02:00
|
|
|
<!--
|
|
|
|
@component
|
|
|
|
Serves as a pre-slotted convenience component which combines all the common
|
|
|
|
components and functionality for general note editing.
|
|
|
|
|
|
|
|
Functionality exclusive to specifc note-editing views (e.g. in the browser or
|
|
|
|
the AddCards dialog) should be implemented in the user of this component.
|
|
|
|
-->
|
2022-09-28 06:02:32 +02:00
|
|
|
<div class="note-editor" bind:clientHeight>
|
2022-09-12 11:22:22 +02:00
|
|
|
<EditorToolbar {size} {wrap} api={toolbar}>
|
|
|
|
<slot slot="notetypeButtons" name="notetypeButtons" />
|
|
|
|
</EditorToolbar>
|
|
|
|
|
|
|
|
{#if hint}
|
|
|
|
<Absolute bottom right --margin="10px">
|
|
|
|
<Notification>
|
2022-09-28 06:02:32 +02:00
|
|
|
<Badge --badge-color="tomato" --icon-align="top"
|
2022-09-12 11:22:22 +02:00
|
|
|
>{@html alertIcon}</Badge
|
|
|
|
>
|
|
|
|
<span>{@html hint}</span>
|
|
|
|
</Notification>
|
|
|
|
</Absolute>
|
|
|
|
{/if}
|
|
|
|
|
2022-09-28 06:02:32 +02:00
|
|
|
<Pane
|
|
|
|
bind:this={fieldsPane.resizable}
|
2022-10-13 02:13:19 +02:00
|
|
|
on:resize={(e) => {
|
|
|
|
fieldsPane.height = e.detail.height;
|
|
|
|
}}
|
2022-09-28 06:02:32 +02:00
|
|
|
>
|
|
|
|
<PaneContent>
|
|
|
|
<Fields>
|
|
|
|
<DecoratedElements>
|
|
|
|
{#each fieldsData as field, index}
|
|
|
|
{@const content = fieldStores[index]}
|
|
|
|
|
|
|
|
<EditorField
|
|
|
|
{field}
|
|
|
|
{content}
|
|
|
|
flipInputs={plainTextDefaults[index]}
|
|
|
|
api={fields[index]}
|
|
|
|
on:focusin={() => {
|
|
|
|
$focusedField = fields[index];
|
|
|
|
bridgeCommand(`focus:${index}`);
|
|
|
|
}}
|
|
|
|
on:focusout={() => {
|
|
|
|
$focusedField = null;
|
|
|
|
bridgeCommand(
|
|
|
|
`blur:${index}:${getNoteId()}:${transformContentBeforeSave(
|
|
|
|
get(content),
|
|
|
|
)}`,
|
|
|
|
);
|
2022-09-12 11:22:22 +02:00
|
|
|
}}
|
2022-09-28 06:02:32 +02:00
|
|
|
on:mouseenter={() => {
|
|
|
|
$hoveredField = fields[index];
|
|
|
|
}}
|
|
|
|
on:mouseleave={() => {
|
|
|
|
$hoveredField = null;
|
|
|
|
}}
|
|
|
|
collapsed={fieldsCollapsed[index]}
|
2022-10-03 05:16:54 +02:00
|
|
|
--dupes-color={cols[index] === "dupe"
|
|
|
|
? "var(--accent-danger)"
|
|
|
|
: "transparent"}
|
2022-09-12 11:22:22 +02:00
|
|
|
>
|
2022-09-28 06:02:32 +02:00
|
|
|
<svelte:fragment slot="field-label">
|
|
|
|
<LabelContainer
|
|
|
|
collapsed={fieldsCollapsed[index]}
|
|
|
|
on:toggle={async () => {
|
|
|
|
fieldsCollapsed[index] =
|
|
|
|
!fieldsCollapsed[index];
|
|
|
|
|
|
|
|
const defaultInput = !plainTextDefaults[index]
|
|
|
|
? richTextInputs[index]
|
|
|
|
: plainTextInputs[index];
|
|
|
|
|
|
|
|
if (!fieldsCollapsed[index]) {
|
|
|
|
refocusInput(defaultInput.api);
|
|
|
|
} else if (!plainTextDefaults[index]) {
|
|
|
|
plainTextsHidden[index] = true;
|
|
|
|
} else {
|
|
|
|
richTextsHidden[index] = true;
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<svelte:fragment slot="field-name">
|
|
|
|
<LabelName>
|
|
|
|
{field.name}
|
|
|
|
</LabelName>
|
|
|
|
</svelte:fragment>
|
|
|
|
<FieldState>
|
|
|
|
{#if cols[index] === "dupe"}
|
|
|
|
<DuplicateLink />
|
|
|
|
{/if}
|
|
|
|
{#if plainTextDefaults[index]}
|
|
|
|
<RichTextBadge
|
|
|
|
visible={!fieldsCollapsed[index] &&
|
|
|
|
(fields[index] === $hoveredField ||
|
|
|
|
fields[index] ===
|
|
|
|
$focusedField)}
|
|
|
|
bind:off={richTextsHidden[index]}
|
|
|
|
on:toggle={async () => {
|
|
|
|
richTextsHidden[index] =
|
|
|
|
!richTextsHidden[index];
|
|
|
|
|
|
|
|
if (!richTextsHidden[index]) {
|
|
|
|
refocusInput(
|
|
|
|
richTextInputs[index].api,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
{:else}
|
|
|
|
<PlainTextBadge
|
|
|
|
visible={!fieldsCollapsed[index] &&
|
|
|
|
(fields[index] === $hoveredField ||
|
|
|
|
fields[index] ===
|
|
|
|
$focusedField)}
|
|
|
|
bind:off={plainTextsHidden[index]}
|
|
|
|
on:toggle={async () => {
|
|
|
|
plainTextsHidden[index] =
|
|
|
|
!plainTextsHidden[index];
|
|
|
|
|
|
|
|
if (!plainTextsHidden[index]) {
|
|
|
|
refocusInput(
|
|
|
|
plainTextInputs[index].api,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
{/if}
|
|
|
|
<slot
|
|
|
|
name="field-state"
|
|
|
|
{field}
|
|
|
|
{index}
|
|
|
|
visible={fields[index] === $hoveredField ||
|
|
|
|
fields[index] === $focusedField}
|
|
|
|
/>
|
|
|
|
</FieldState>
|
|
|
|
</LabelContainer>
|
2022-09-12 11:22:22 +02:00
|
|
|
</svelte:fragment>
|
2022-09-28 06:02:32 +02:00
|
|
|
<svelte:fragment slot="rich-text-input">
|
|
|
|
<Collapsible
|
|
|
|
collapse={richTextsHidden[index]}
|
|
|
|
let:collapsed={hidden}
|
|
|
|
>
|
|
|
|
<RichTextInput
|
|
|
|
{hidden}
|
|
|
|
on:focusout={() => {
|
|
|
|
saveFieldNow();
|
|
|
|
$focusedInput = null;
|
2022-09-12 11:22:22 +02:00
|
|
|
}}
|
2022-09-28 06:02:32 +02:00
|
|
|
bind:this={richTextInputs[index]}
|
|
|
|
>
|
|
|
|
<ImageHandle maxWidth={250} maxHeight={125} />
|
|
|
|
<MathjaxHandle />
|
|
|
|
{#if insertSymbols}
|
|
|
|
<SymbolsOverlay />
|
|
|
|
{/if}
|
|
|
|
<FieldDescription>
|
|
|
|
{field.description}
|
|
|
|
</FieldDescription>
|
|
|
|
</RichTextInput>
|
|
|
|
</Collapsible>
|
|
|
|
</svelte:fragment>
|
|
|
|
<svelte:fragment slot="plain-text-input">
|
|
|
|
<Collapsible
|
|
|
|
collapse={plainTextsHidden[index]}
|
|
|
|
let:collapsed={hidden}
|
|
|
|
>
|
|
|
|
<PlainTextInput
|
|
|
|
{hidden}
|
|
|
|
isDefault={plainTextDefaults[index]}
|
|
|
|
richTextHidden={richTextsHidden[index]}
|
|
|
|
on:focusout={() => {
|
|
|
|
saveFieldNow();
|
|
|
|
$focusedInput = null;
|
2022-09-12 11:22:22 +02:00
|
|
|
}}
|
2022-09-28 06:02:32 +02:00
|
|
|
bind:this={plainTextInputs[index]}
|
2022-09-12 11:22:22 +02:00
|
|
|
/>
|
2022-09-28 06:02:32 +02:00
|
|
|
</Collapsible>
|
|
|
|
</svelte:fragment>
|
|
|
|
</EditorField>
|
|
|
|
{/each}
|
|
|
|
|
|
|
|
<MathjaxElement />
|
|
|
|
<FrameElement />
|
|
|
|
</DecoratedElements>
|
|
|
|
</Fields>
|
|
|
|
</PaneContent>
|
|
|
|
</Pane>
|
|
|
|
|
2022-10-13 02:13:19 +02:00
|
|
|
<HorizontalResizer
|
|
|
|
panes={[fieldsPane, tagsPane]}
|
|
|
|
showIndicator={$tagsCollapsed || snapTags}
|
|
|
|
tip={`Double click to ${$tagsCollapsed ? "expand" : "collapse"} tag editor`}
|
|
|
|
{clientHeight}
|
|
|
|
bind:this={lowerResizer}
|
|
|
|
on:dblclick={() => snapResizer(!$tagsCollapsed)}
|
|
|
|
on:release={() => {
|
|
|
|
snapResizer(snapTags);
|
|
|
|
}}
|
|
|
|
>
|
2022-09-28 06:02:32 +02:00
|
|
|
<div class="tags-expander">
|
|
|
|
<TagAddButton
|
|
|
|
on:tagappend={() => {
|
|
|
|
tagEditor.appendEmptyTag();
|
|
|
|
}}
|
|
|
|
keyCombination="Control+Shift+T"
|
|
|
|
>
|
|
|
|
{@html tagAmount > 0 ? `${tagAmount} Tags` : ""}
|
|
|
|
</TagAddButton>
|
|
|
|
</div>
|
2022-10-13 02:13:19 +02:00
|
|
|
</HorizontalResizer>
|
2022-09-28 06:02:32 +02:00
|
|
|
|
|
|
|
<Pane
|
|
|
|
bind:this={tagsPane.resizable}
|
|
|
|
on:resize={(e) => {
|
|
|
|
tagsPane.height = e.detail.height;
|
2022-10-13 02:13:19 +02:00
|
|
|
if (tagsPane.maxHeight > 0) {
|
|
|
|
snapTags = tagsPane.height < tagsPane.maxHeight / 2;
|
|
|
|
}
|
2022-09-28 06:02:32 +02:00
|
|
|
}}
|
2022-10-13 02:13:19 +02:00
|
|
|
--opacity={(() => {
|
|
|
|
if (!$tagsCollapsed) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return snapTags ? tagsPane.height / tagsPane.maxHeight : 1;
|
|
|
|
}
|
|
|
|
})()}
|
2022-09-28 06:02:32 +02:00
|
|
|
>
|
|
|
|
<PaneContent scroll={false}>
|
|
|
|
<TagEditor
|
|
|
|
{tags}
|
2022-10-13 02:13:19 +02:00
|
|
|
--button-opacity={snapTags ? 0 : 1}
|
2022-09-28 06:02:32 +02:00
|
|
|
bind:this={tagEditor}
|
|
|
|
on:tagsupdate={saveTags}
|
|
|
|
on:tagsFocused={() => {
|
|
|
|
expandTags();
|
|
|
|
$tagsCollapsed = false;
|
|
|
|
}}
|
|
|
|
on:heightChange={(e) => {
|
|
|
|
tagsPane.maxHeight = e.detail.height;
|
|
|
|
if (!$tagsCollapsed) {
|
|
|
|
expandTags();
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</PaneContent>
|
|
|
|
</Pane>
|
2021-10-18 14:01:15 +02:00
|
|
|
</div>
|
|
|
|
|
|
|
|
<style lang="scss">
|
|
|
|
.note-editor {
|
|
|
|
display: flex;
|
|
|
|
flex-direction: column;
|
2022-09-12 11:22:22 +02:00
|
|
|
height: 100%;
|
|
|
|
}
|
2021-10-18 14:01:15 +02:00
|
|
|
</style>
|