@ecodev/natural-editor 41.0.0 → 41.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/ecodev-natural-editor.umd.js +247 -80
- package/bundles/ecodev-natural-editor.umd.js.map +1 -1
- package/esm2015/lib/editor/editor.component.js +200 -0
- package/esm2015/lib/editor.module.js +15 -5
- package/esm2015/lib/link-dialog/link-dialog.component.js +3 -4
- package/esm2015/lib/utils/image.js +81 -0
- package/esm2015/lib/utils/menu.js +207 -0
- package/esm2015/lib/utils/schema.js +46 -0
- package/esm2015/lib/utils/table.js +37 -0
- package/esm2015/public-api.js +2 -2
- package/fesm2015/ecodev-natural-editor.js +233 -69
- package/fesm2015/ecodev-natural-editor.js.map +1 -1
- package/lib/{editor.component.d.ts → editor/editor.component.d.ts} +19 -4
- package/lib/editor.module.d.ts +4 -2
- package/lib/utils/image.d.ts +15 -0
- package/lib/{menu.d.ts → utils/menu.d.ts} +1 -1
- package/lib/utils/schema.d.ts +3 -0
- package/lib/utils/table.d.ts +8 -0
- package/package.json +3 -1
- package/public-api.d.ts +1 -1
- package/src/lib/editor/_editor.theme.scss +44 -0
- package/theming/_natural-editor.theme.scss +1 -1
- package/esm2015/lib/editor.component.js +0 -160
- package/esm2015/lib/menu.js +0 -222
- package/esm2015/lib/schema.js +0 -21
- package/lib/schema.d.ts +0 -2
- package/src/lib/_editor.theme.scss +0 -23
|
@@ -1,160 +0,0 @@
|
|
|
1
|
-
import { Component, ElementRef, EventEmitter, Inject, Optional, Output, Self, ViewChild, } from '@angular/core';
|
|
2
|
-
import { EditorView } from 'prosemirror-view';
|
|
3
|
-
import { EditorState, Plugin } from 'prosemirror-state';
|
|
4
|
-
// @ts-ignore
|
|
5
|
-
import { exampleSetup } from 'prosemirror-example-setup';
|
|
6
|
-
import { DOMParser, DOMSerializer } from 'prosemirror-model';
|
|
7
|
-
import { schema } from './schema';
|
|
8
|
-
import { DOCUMENT } from '@angular/common';
|
|
9
|
-
import { buildMenuItems } from './menu';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
import * as i1 from "@angular/forms";
|
|
12
|
-
import * as i2 from "@angular/material/dialog";
|
|
13
|
-
import * as i3 from "@angular/material/button-toggle";
|
|
14
|
-
import * as i4 from "@angular/material/icon";
|
|
15
|
-
import * as i5 from "@angular/material/button";
|
|
16
|
-
import * as i6 from "@angular/material/menu";
|
|
17
|
-
import * as i7 from "@angular/common";
|
|
18
|
-
import * as i8 from "@angular/material/tooltip";
|
|
19
|
-
/**
|
|
20
|
-
* Prosemirror component
|
|
21
|
-
*
|
|
22
|
-
* Usage :
|
|
23
|
-
*
|
|
24
|
-
* ```html
|
|
25
|
-
* <natural-editor [(ngModel)]="htmlString"></natural-editor>
|
|
26
|
-
* ```
|
|
27
|
-
*/
|
|
28
|
-
// @dynamic
|
|
29
|
-
export class NaturalEditorComponent {
|
|
30
|
-
constructor(ngControl, document, dialog) {
|
|
31
|
-
this.ngControl = ngControl;
|
|
32
|
-
this.document = document;
|
|
33
|
-
this.dialog = dialog;
|
|
34
|
-
this.view = null;
|
|
35
|
-
this.contentChange = new EventEmitter();
|
|
36
|
-
/**
|
|
37
|
-
* HTML string
|
|
38
|
-
*/
|
|
39
|
-
this.content = '';
|
|
40
|
-
this.menu = null;
|
|
41
|
-
if (this.ngControl !== null) {
|
|
42
|
-
this.ngControl.valueAccessor = this;
|
|
43
|
-
}
|
|
44
|
-
}
|
|
45
|
-
ngOnInit() {
|
|
46
|
-
this.menu = buildMenuItems(schema, this.dialog);
|
|
47
|
-
const serializer = DOMSerializer.fromSchema(schema);
|
|
48
|
-
const state = this.createState();
|
|
49
|
-
this.view = new EditorView(this.editor.nativeElement, {
|
|
50
|
-
state: state,
|
|
51
|
-
dispatchTransaction: (transaction) => {
|
|
52
|
-
if (!this.view) {
|
|
53
|
-
return;
|
|
54
|
-
}
|
|
55
|
-
const newState = this.view.state.apply(transaction);
|
|
56
|
-
this.view.updateState(newState);
|
|
57
|
-
// Transform doc into HTML string
|
|
58
|
-
const dom = serializer.serializeFragment(this.view.state.doc);
|
|
59
|
-
const el = this.document.createElement('_');
|
|
60
|
-
el.appendChild(dom);
|
|
61
|
-
const newContent = el.innerHTML;
|
|
62
|
-
if (this.content === newContent) {
|
|
63
|
-
return;
|
|
64
|
-
}
|
|
65
|
-
this.content = el.innerHTML;
|
|
66
|
-
if (this.onChange) {
|
|
67
|
-
this.onChange(this.content);
|
|
68
|
-
}
|
|
69
|
-
this.contentChange.emit(this.content);
|
|
70
|
-
},
|
|
71
|
-
});
|
|
72
|
-
this.update();
|
|
73
|
-
}
|
|
74
|
-
writeValue(val) {
|
|
75
|
-
if (typeof val === 'string' && val !== this.content) {
|
|
76
|
-
this.content = val;
|
|
77
|
-
}
|
|
78
|
-
if (this.view !== null) {
|
|
79
|
-
const state = this.createState();
|
|
80
|
-
this.view.updateState(state);
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
createState() {
|
|
84
|
-
const template = this.document.createElement('_');
|
|
85
|
-
template.innerHTML = '<div>' + this.content + '</div>';
|
|
86
|
-
if (!template.firstChild) {
|
|
87
|
-
throw new Error('child of template element could not be created');
|
|
88
|
-
}
|
|
89
|
-
const parser = DOMParser.fromSchema(schema);
|
|
90
|
-
const doc = parser.parse(template.firstChild);
|
|
91
|
-
const self = this;
|
|
92
|
-
return EditorState.create({
|
|
93
|
-
doc: doc,
|
|
94
|
-
plugins: [
|
|
95
|
-
...exampleSetup({ schema, menuBar: false }),
|
|
96
|
-
new Plugin({
|
|
97
|
-
view: () => self,
|
|
98
|
-
}),
|
|
99
|
-
],
|
|
100
|
-
});
|
|
101
|
-
}
|
|
102
|
-
/**
|
|
103
|
-
* Called by Prosemirror whenever the editor state changes. So we update our menu states.
|
|
104
|
-
*/
|
|
105
|
-
update() {
|
|
106
|
-
if (!this.view || !this.menu) {
|
|
107
|
-
return;
|
|
108
|
-
}
|
|
109
|
-
for (const item of Object.values(this.menu)) {
|
|
110
|
-
item.update(this.view, this.view.state);
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
registerOnChange(fn) {
|
|
114
|
-
this.onChange = fn;
|
|
115
|
-
}
|
|
116
|
-
registerOnTouched(fn) { }
|
|
117
|
-
setDisabledState(isDisabled) {
|
|
118
|
-
// TODO disable editor ?
|
|
119
|
-
}
|
|
120
|
-
ngOnDestroy() {
|
|
121
|
-
if (this.view) {
|
|
122
|
-
this.view.destroy();
|
|
123
|
-
this.view = null;
|
|
124
|
-
}
|
|
125
|
-
}
|
|
126
|
-
run(event, key) {
|
|
127
|
-
if (!this.view || !this.menu) {
|
|
128
|
-
return;
|
|
129
|
-
}
|
|
130
|
-
const item = this.menu[key];
|
|
131
|
-
if (!item || item.disabled || !item.show) {
|
|
132
|
-
return;
|
|
133
|
-
}
|
|
134
|
-
item.spec.run(this.view.state, this.view.dispatch, this.view, event);
|
|
135
|
-
this.view.focus();
|
|
136
|
-
}
|
|
137
|
-
}
|
|
138
|
-
NaturalEditorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.9", ngImport: i0, type: NaturalEditorComponent, deps: [{ token: i1.NgControl, optional: true, self: true }, { token: DOCUMENT }, { token: i2.MatDialog }], target: i0.ɵɵFactoryTarget.Component });
|
|
139
|
-
NaturalEditorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.9", type: NaturalEditorComponent, selector: "natural-editor", outputs: { contentChange: "contentChange" }, viewQueries: [{ propertyName: "editor", first: true, predicate: ["editor"], descendants: true, read: ElementRef, static: true }], ngImport: i0, template: "<div class=\"menu\" *ngIf=\"menu\">\n <mat-button-toggle-group multiple>\n <mat-button-toggle\n *ngIf=\"menu.toggleStrong\"\n [disabled]=\"menu.toggleStrong.disabled\"\n [checked]=\"menu.toggleStrong.active\"\n (click)=\"run($event, 'toggleStrong')\"\n i18n-matTooltip\n matTooltip=\"Gras\"\n >\n <mat-icon>format_bold</mat-icon>\n </mat-button-toggle>\n\n <mat-button-toggle\n *ngIf=\"menu.toggleEm\"\n [disabled]=\"menu.toggleEm.disabled\"\n [checked]=\"menu.toggleEm.active\"\n (click)=\"run($event, 'toggleEm')\"\n i18n-matTooltip\n matTooltip=\"Italique\"\n >\n <mat-icon>format_italic</mat-icon>\n </mat-button-toggle>\n\n <mat-button-toggle\n *ngIf=\"menu.toggleCode\"\n [disabled]=\"menu.toggleCode.disabled\"\n [checked]=\"menu.toggleCode.active\"\n (click)=\"run($event, 'toggleCode')\"\n i18n-matTooltip\n matTooltip=\"Code\"\n >\n <mat-icon>code</mat-icon>\n </mat-button-toggle>\n\n <mat-button-toggle\n *ngIf=\"menu.toggleLink\"\n [disabled]=\"menu.toggleLink.disabled\"\n [checked]=\"menu.toggleLink.active\"\n (click)=\"run($event, 'toggleLink')\"\n i18n-matTooltip\n matTooltip=\"Ins\u00E9rer un lien...\"\n >\n <mat-icon>insert_link</mat-icon>\n </mat-button-toggle>\n </mat-button-toggle-group>\n\n <button mat-button [matMenuTriggerFor]=\"blockMenu\">\n <span i18n>Type</span>\n <mat-icon>arrow_drop_down</mat-icon>\n </button>\n\n <mat-menu #blockMenu=\"matMenu\">\n <button\n mat-menu-item\n *ngIf=\"menu.makeParagraph\"\n [disabled]=\"menu.makeParagraph.disabled\"\n (click)=\"run($event, 'makeParagraph')\"\n i18n\n >Paragraphe\n </button>\n <button\n mat-menu-item\n *ngIf=\"menu.makeCodeBlock\"\n [disabled]=\"menu.makeCodeBlock.disabled\"\n (click)=\"run($event, 'makeCodeBlock')\"\n i18n\n >Code\n </button>\n <button\n mat-menu-item\n *ngIf=\"menu.makeHead1\"\n [disabled]=\"menu.makeHead1.disabled\"\n (click)=\"run($event, 'makeHead1')\"\n i18n\n >Titre 1\n </button>\n <button\n mat-menu-item\n *ngIf=\"menu.makeHead2\"\n [disabled]=\"menu.makeHead2.disabled\"\n (click)=\"run($event, 'makeHead2')\"\n i18n\n >Titre 2\n </button>\n <button\n mat-menu-item\n *ngIf=\"menu.makeHead3\"\n [disabled]=\"menu.makeHead3.disabled\"\n (click)=\"run($event, 'makeHead3')\"\n i18n\n >Titre 3\n </button>\n <button\n mat-menu-item\n *ngIf=\"menu.makeHead4\"\n [disabled]=\"menu.makeHead4.disabled\"\n (click)=\"run($event, 'makeHead4')\"\n i18n\n >Titre 4\n </button>\n <button\n mat-menu-item\n *ngIf=\"menu.makeHead5\"\n [disabled]=\"menu.makeHead5.disabled\"\n (click)=\"run($event, 'makeHead5')\"\n i18n\n >Titre 5\n </button>\n <button\n mat-menu-item\n *ngIf=\"menu.makeHead6\"\n [disabled]=\"menu.makeHead6.disabled\"\n (click)=\"run($event, 'makeHead6')\"\n i18n\n >Titre 6\n </button>\n </mat-menu>\n\n <button\n mat-button\n *ngIf=\"menu.undo\"\n [disabled]=\"menu.undo.disabled\"\n (click)=\"run($event, 'undo')\"\n i18n-matTooltip\n matTooltip=\"Annuler\"\n >\n <mat-icon>undo</mat-icon>\n </button>\n\n <button\n mat-button\n *ngIf=\"menu.redo\"\n [disabled]=\"menu.redo.disabled\"\n (click)=\"run($event, 'redo')\"\n i18n-matTooltip\n matTooltip=\"Refaire\"\n >\n <mat-icon>redo</mat-icon>\n </button>\n\n <button\n mat-button\n *ngIf=\"menu.wrapBulletList && menu.wrapBulletList.show\"\n [disabled]=\"menu.wrapBulletList.disabled\"\n (click)=\"run($event, 'wrapBulletList')\"\n i18n-matTooltip\n matTooltip=\"Liste \u00E0 puce\"\n >\n <mat-icon>format_list_bulleted</mat-icon>\n </button>\n\n <button\n mat-button\n *ngIf=\"menu.wrapOrderedList && menu.wrapOrderedList.show\"\n [disabled]=\"menu.wrapOrderedList.disabled\"\n (click)=\"run($event, 'wrapOrderedList')\"\n i18n-matTooltip\n matTooltip=\"Liste \u00E0 num\u00E9ro\"\n >\n <mat-icon>format_list_numbered</mat-icon>\n </button>\n\n <button\n mat-button\n *ngIf=\"menu.wrapBlockQuote && menu.wrapBlockQuote.show\"\n [disabled]=\"menu.wrapBlockQuote.disabled\"\n (click)=\"run($event, 'wrapBlockQuote')\"\n i18n-matTooltip\n matTooltip=\"Citation\"\n >\n <mat-icon>format_quote</mat-icon>\n </button>\n\n <button\n mat-button\n *ngIf=\"menu.joinUp && menu.joinUp.show\"\n [disabled]=\"menu.joinUp.disabled\"\n (click)=\"run($event, 'joinUp')\"\n i18n-matTooltip\n matTooltip=\"Fusionner avec l'\u00E9l\u00E9ment du haut\"\n >\n <mat-icon>move_up</mat-icon>\n </button>\n\n <button\n mat-button\n *ngIf=\"menu.lift && menu.lift.show\"\n [disabled]=\"menu.lift.disabled\"\n (click)=\"run($event, 'lift')\"\n i18n-matTooltip\n matTooltip=\"D\u00E9sindenter\"\n >\n <mat-icon>format_indent_decrease</mat-icon>\n </button>\n\n <button\n mat-button\n *ngIf=\"menu.selectParentNode && menu.selectParentNode.show\"\n [disabled]=\"menu.selectParentNode.disabled\"\n (click)=\"run($event, 'selectParentNode')\"\n i18n-matTooltip\n matTooltip=\"S\u00E9lectionner l'\u00E9l\u00E9ment parent\"\n >\n <mat-icon>select_all</mat-icon>\n </button>\n</div>\n<div #editor></div>\n", styles: [".menu{border-bottom:1px solid;display:flex;flex-wrap:wrap;padding:10px 18px}::ng-deep .ProseMirror{position:relative}::ng-deep .ProseMirror{word-wrap:break-word;white-space:pre-wrap;-webkit-font-variant-ligatures:none;font-feature-settings:none;font-variant-ligatures:none}::ng-deep .ProseMirror pre{white-space:pre-wrap}::ng-deep .ProseMirror li{position:relative}::ng-deep .ProseMirror-hideselection *::selection{background:transparent}::ng-deep .ProseMirror-hideselection *::-moz-selection{background:transparent}::ng-deep .ProseMirror-hideselection{caret-color:transparent}::ng-deep .ProseMirror-selectednode{outline:2px solid #8cf}::ng-deep li.ProseMirror-selectednode{outline:none}::ng-deep li.ProseMirror-selectednode:after{content:\"\";position:absolute;left:-32px;right:-2px;top:-2px;bottom:-2px;border:2px solid #8cf;pointer-events:none}::ng-deep .ProseMirror-gapcursor{display:none;pointer-events:none;position:absolute}::ng-deep .ProseMirror-gapcursor:after{content:\"\";display:block;position:absolute;top:-2px;width:20px;border-top:1px solid black;animation:ProseMirror-cursor-blink 1.1s steps(2,start) infinite}@keyframes ProseMirror-cursor-blink{to{visibility:hidden}}::ng-deep .ProseMirror-focused .ProseMirror-gapcursor{display:block}::ng-deep .ProseMirror-example-setup-style hr{padding:2px 10px;border:none;margin:1em 0}::ng-deep .ProseMirror-example-setup-style hr:after{content:\"\";display:block;height:1px;background-color:silver;line-height:2px}::ng-deep .ProseMirror ul,::ng-deep .ProseMirror ol{padding-left:30px}::ng-deep .ProseMirror blockquote{padding-left:1em;border-left:3px solid #eee;margin-left:0;margin-right:0}::ng-deep .ProseMirror-example-setup-style img{cursor:default}::ng-deep .ProseMirror p:first-child,::ng-deep .ProseMirror h1:first-child,::ng-deep .ProseMirror h2:first-child,::ng-deep .ProseMirror h3:first-child,::ng-deep .ProseMirror h4:first-child,::ng-deep .ProseMirror h5:first-child,::ng-deep .ProseMirror h6:first-child{margin-top:10px}::ng-deep .ProseMirror{padding:4px 8px 4px 14px;line-height:1.2;outline:none}::ng-deep .ProseMirror p{margin-bottom:1em}\n"], components: [{ type: i3.MatButtonToggle, selector: "mat-button-toggle", inputs: ["disableRipple", "aria-labelledby", "tabIndex", "appearance", "checked", "disabled", "id", "name", "aria-label", "value"], outputs: ["change"], exportAs: ["matButtonToggle"] }, { type: i4.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i5.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { type: i6.MatMenu, selector: "mat-menu", exportAs: ["matMenu"] }, { type: i6.MatMenuItem, selector: "[mat-menu-item]", inputs: ["disabled", "disableRipple", "role"], exportAs: ["matMenuItem"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.MatButtonToggleGroup, selector: "mat-button-toggle-group", inputs: ["appearance", "name", "vertical", "value", "multiple", "disabled"], outputs: ["valueChange", "change"], exportAs: ["matButtonToggleGroup"] }, { type: i8.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { type: i6.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", exportAs: ["matMenuTrigger"] }] });
|
|
140
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.9", ngImport: i0, type: NaturalEditorComponent, decorators: [{
|
|
141
|
-
type: Component,
|
|
142
|
-
args: [{
|
|
143
|
-
selector: 'natural-editor',
|
|
144
|
-
templateUrl: './editor.component.html',
|
|
145
|
-
styleUrls: ['./editor.component.scss'],
|
|
146
|
-
}]
|
|
147
|
-
}], ctorParameters: function () { return [{ type: i1.NgControl, decorators: [{
|
|
148
|
-
type: Optional
|
|
149
|
-
}, {
|
|
150
|
-
type: Self
|
|
151
|
-
}] }, { type: Document, decorators: [{
|
|
152
|
-
type: Inject,
|
|
153
|
-
args: [DOCUMENT]
|
|
154
|
-
}] }, { type: i2.MatDialog }]; }, propDecorators: { editor: [{
|
|
155
|
-
type: ViewChild,
|
|
156
|
-
args: ['editor', { read: ElementRef, static: true }]
|
|
157
|
-
}], contentChange: [{
|
|
158
|
-
type: Output
|
|
159
|
-
}] } });
|
|
160
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"editor.component.js","sourceRoot":"","sources":["../../../../projects/natural-editor/src/lib/editor.component.ts","../../../../projects/natural-editor/src/lib/editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    Inject,\n    OnDestroy,\n    OnInit,\n    Optional,\n    Output,\n    Self,\n    ViewChild,\n} from '@angular/core';\nimport {ControlValueAccessor, NgControl} from '@angular/forms';\nimport {EditorView} from 'prosemirror-view';\nimport {EditorState, Plugin, Transaction} from 'prosemirror-state';\n// @ts-ignore\nimport {exampleSetup} from 'prosemirror-example-setup';\nimport {DOMParser, DOMSerializer} from 'prosemirror-model';\nimport {schema} from './schema';\nimport {DOCUMENT} from '@angular/common';\nimport {MatDialog} from '@angular/material/dialog';\nimport {buildMenuItems, Key, MenuItems} from './menu';\n\n/**\n * Prosemirror component\n *\n * Usage :\n *\n * ```html\n * <natural-editor [(ngModel)]=\"htmlString\"></natural-editor>\n * ```\n */\n// @dynamic\n@Component({\n    selector: 'natural-editor',\n    templateUrl: './editor.component.html',\n    styleUrls: ['./editor.component.scss'],\n})\nexport class NaturalEditorComponent implements OnInit, OnDestroy, ControlValueAccessor {\n    private view: EditorView | null = null;\n\n    @ViewChild('editor', {read: ElementRef, static: true}) private editor!: ElementRef;\n\n    @Output() public readonly contentChange = new EventEmitter<string>();\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    private onChange?: (value: string | null) => void;\n\n    /**\n     * HTML string\n     */\n    private content = '';\n\n    public menu: MenuItems | null = null;\n\n    constructor(\n        @Optional() @Self() public readonly ngControl: NgControl,\n        @Inject(DOCUMENT) private readonly document: Document,\n        private readonly dialog: MatDialog,\n    ) {\n        if (this.ngControl !== null) {\n            this.ngControl.valueAccessor = this;\n        }\n    }\n\n    public ngOnInit(): void {\n        this.menu = buildMenuItems(schema, this.dialog);\n        const serializer = DOMSerializer.fromSchema(schema);\n        const state = this.createState();\n\n        this.view = new EditorView(this.editor.nativeElement, {\n            state: state,\n            dispatchTransaction: (transaction: Transaction) => {\n                if (!this.view) {\n                    return;\n                }\n\n                const newState = this.view.state.apply(transaction);\n                this.view.updateState(newState);\n\n                // Transform doc into HTML string\n                const dom = serializer.serializeFragment(this.view.state.doc as any);\n                const el = this.document.createElement('_');\n                el.appendChild(dom);\n\n                const newContent = el.innerHTML;\n                if (this.content === newContent) {\n                    return;\n                }\n\n                this.content = el.innerHTML;\n\n                if (this.onChange) {\n                    this.onChange(this.content);\n                }\n                this.contentChange.emit(this.content);\n            },\n        });\n        this.update();\n    }\n\n    public writeValue(val: string | undefined): void {\n        if (typeof val === 'string' && val !== this.content) {\n            this.content = val;\n        }\n\n        if (this.view !== null) {\n            const state = this.createState();\n            this.view.updateState(state);\n        }\n    }\n\n    private createState(): EditorState {\n        const template = this.document.createElement('_');\n        template.innerHTML = '<div>' + this.content + '</div>';\n        if (!template.firstChild) {\n            throw new Error('child of template element could not be created');\n        }\n\n        const parser = DOMParser.fromSchema(schema);\n        const doc = parser.parse(template.firstChild);\n        const self = this;\n\n        return EditorState.create({\n            doc: doc,\n            plugins: [\n                ...exampleSetup({schema, menuBar: false}),\n                new Plugin({\n                    view: () => self,\n                }),\n            ],\n        });\n    }\n\n    /**\n     * Called by Prosemirror whenever the editor state changes. So we update our menu states.\n     */\n    public update(): void {\n        if (!this.view || !this.menu) {\n            return;\n        }\n\n        for (const item of Object.values(this.menu)) {\n            item.update(this.view, this.view.state);\n        }\n    }\n\n    public registerOnChange(fn: any): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: any): void {}\n\n    public setDisabledState(isDisabled: boolean): void {\n        // TODO disable editor ?\n    }\n\n    public ngOnDestroy(): void {\n        if (this.view) {\n            this.view.destroy();\n            this.view = null;\n        }\n    }\n\n    public run(event: Event, key: Key): void {\n        if (!this.view || !this.menu) {\n            return;\n        }\n\n        const item = this.menu[key];\n        if (!item || item.disabled || !item.show) {\n            return;\n        }\n\n        item.spec.run(this.view.state, this.view.dispatch, this.view, event);\n        this.view.focus();\n    }\n}\n","<div class=\"menu\" *ngIf=\"menu\">\n    <mat-button-toggle-group multiple>\n        <mat-button-toggle\n            *ngIf=\"menu.toggleStrong\"\n            [disabled]=\"menu.toggleStrong.disabled\"\n            [checked]=\"menu.toggleStrong.active\"\n            (click)=\"run($event, 'toggleStrong')\"\n            i18n-matTooltip\n            matTooltip=\"Gras\"\n        >\n            <mat-icon>format_bold</mat-icon>\n        </mat-button-toggle>\n\n        <mat-button-toggle\n            *ngIf=\"menu.toggleEm\"\n            [disabled]=\"menu.toggleEm.disabled\"\n            [checked]=\"menu.toggleEm.active\"\n            (click)=\"run($event, 'toggleEm')\"\n            i18n-matTooltip\n            matTooltip=\"Italique\"\n        >\n            <mat-icon>format_italic</mat-icon>\n        </mat-button-toggle>\n\n        <mat-button-toggle\n            *ngIf=\"menu.toggleCode\"\n            [disabled]=\"menu.toggleCode.disabled\"\n            [checked]=\"menu.toggleCode.active\"\n            (click)=\"run($event, 'toggleCode')\"\n            i18n-matTooltip\n            matTooltip=\"Code\"\n        >\n            <mat-icon>code</mat-icon>\n        </mat-button-toggle>\n\n        <mat-button-toggle\n            *ngIf=\"menu.toggleLink\"\n            [disabled]=\"menu.toggleLink.disabled\"\n            [checked]=\"menu.toggleLink.active\"\n            (click)=\"run($event, 'toggleLink')\"\n            i18n-matTooltip\n            matTooltip=\"Insérer un lien...\"\n        >\n            <mat-icon>insert_link</mat-icon>\n        </mat-button-toggle>\n    </mat-button-toggle-group>\n\n    <button mat-button [matMenuTriggerFor]=\"blockMenu\">\n        <span i18n>Type</span>\n        <mat-icon>arrow_drop_down</mat-icon>\n    </button>\n\n    <mat-menu #blockMenu=\"matMenu\">\n        <button\n            mat-menu-item\n            *ngIf=\"menu.makeParagraph\"\n            [disabled]=\"menu.makeParagraph.disabled\"\n            (click)=\"run($event, 'makeParagraph')\"\n            i18n\n            >Paragraphe\n        </button>\n        <button\n            mat-menu-item\n            *ngIf=\"menu.makeCodeBlock\"\n            [disabled]=\"menu.makeCodeBlock.disabled\"\n            (click)=\"run($event, 'makeCodeBlock')\"\n            i18n\n            >Code\n        </button>\n        <button\n            mat-menu-item\n            *ngIf=\"menu.makeHead1\"\n            [disabled]=\"menu.makeHead1.disabled\"\n            (click)=\"run($event, 'makeHead1')\"\n            i18n\n            >Titre 1\n        </button>\n        <button\n            mat-menu-item\n            *ngIf=\"menu.makeHead2\"\n            [disabled]=\"menu.makeHead2.disabled\"\n            (click)=\"run($event, 'makeHead2')\"\n            i18n\n            >Titre 2\n        </button>\n        <button\n            mat-menu-item\n            *ngIf=\"menu.makeHead3\"\n            [disabled]=\"menu.makeHead3.disabled\"\n            (click)=\"run($event, 'makeHead3')\"\n            i18n\n            >Titre 3\n        </button>\n        <button\n            mat-menu-item\n            *ngIf=\"menu.makeHead4\"\n            [disabled]=\"menu.makeHead4.disabled\"\n            (click)=\"run($event, 'makeHead4')\"\n            i18n\n            >Titre 4\n        </button>\n        <button\n            mat-menu-item\n            *ngIf=\"menu.makeHead5\"\n            [disabled]=\"menu.makeHead5.disabled\"\n            (click)=\"run($event, 'makeHead5')\"\n            i18n\n            >Titre 5\n        </button>\n        <button\n            mat-menu-item\n            *ngIf=\"menu.makeHead6\"\n            [disabled]=\"menu.makeHead6.disabled\"\n            (click)=\"run($event, 'makeHead6')\"\n            i18n\n            >Titre 6\n        </button>\n    </mat-menu>\n\n    <button\n        mat-button\n        *ngIf=\"menu.undo\"\n        [disabled]=\"menu.undo.disabled\"\n        (click)=\"run($event, 'undo')\"\n        i18n-matTooltip\n        matTooltip=\"Annuler\"\n    >\n        <mat-icon>undo</mat-icon>\n    </button>\n\n    <button\n        mat-button\n        *ngIf=\"menu.redo\"\n        [disabled]=\"menu.redo.disabled\"\n        (click)=\"run($event, 'redo')\"\n        i18n-matTooltip\n        matTooltip=\"Refaire\"\n    >\n        <mat-icon>redo</mat-icon>\n    </button>\n\n    <button\n        mat-button\n        *ngIf=\"menu.wrapBulletList && menu.wrapBulletList.show\"\n        [disabled]=\"menu.wrapBulletList.disabled\"\n        (click)=\"run($event, 'wrapBulletList')\"\n        i18n-matTooltip\n        matTooltip=\"Liste à puce\"\n    >\n        <mat-icon>format_list_bulleted</mat-icon>\n    </button>\n\n    <button\n        mat-button\n        *ngIf=\"menu.wrapOrderedList && menu.wrapOrderedList.show\"\n        [disabled]=\"menu.wrapOrderedList.disabled\"\n        (click)=\"run($event, 'wrapOrderedList')\"\n        i18n-matTooltip\n        matTooltip=\"Liste à numéro\"\n    >\n        <mat-icon>format_list_numbered</mat-icon>\n    </button>\n\n    <button\n        mat-button\n        *ngIf=\"menu.wrapBlockQuote && menu.wrapBlockQuote.show\"\n        [disabled]=\"menu.wrapBlockQuote.disabled\"\n        (click)=\"run($event, 'wrapBlockQuote')\"\n        i18n-matTooltip\n        matTooltip=\"Citation\"\n    >\n        <mat-icon>format_quote</mat-icon>\n    </button>\n\n    <button\n        mat-button\n        *ngIf=\"menu.joinUp && menu.joinUp.show\"\n        [disabled]=\"menu.joinUp.disabled\"\n        (click)=\"run($event, 'joinUp')\"\n        i18n-matTooltip\n        matTooltip=\"Fusionner avec l'élément du haut\"\n    >\n        <mat-icon>move_up</mat-icon>\n    </button>\n\n    <button\n        mat-button\n        *ngIf=\"menu.lift && menu.lift.show\"\n        [disabled]=\"menu.lift.disabled\"\n        (click)=\"run($event, 'lift')\"\n        i18n-matTooltip\n        matTooltip=\"Désindenter\"\n    >\n        <mat-icon>format_indent_decrease</mat-icon>\n    </button>\n\n    <button\n        mat-button\n        *ngIf=\"menu.selectParentNode && menu.selectParentNode.show\"\n        [disabled]=\"menu.selectParentNode.disabled\"\n        (click)=\"run($event, 'selectParentNode')\"\n        i18n-matTooltip\n        matTooltip=\"Sélectionner l'élément parent\"\n    >\n        <mat-icon>select_all</mat-icon>\n    </button>\n</div>\n<div #editor></div>\n"]}
|
package/esm2015/lib/menu.js
DELETED
|
@@ -1,222 +0,0 @@
|
|
|
1
|
-
import { blockTypeItem, joinUpItem, liftItem, redoItem, selectParentNodeItem, undoItem, wrapItem, } from 'prosemirror-menu';
|
|
2
|
-
import { toggleMark } from 'prosemirror-commands';
|
|
3
|
-
import { wrapInList } from 'prosemirror-schema-list';
|
|
4
|
-
import { LinkDialogComponent } from './link-dialog/link-dialog.component';
|
|
5
|
-
/**
|
|
6
|
-
* One item of the menu.
|
|
7
|
-
*
|
|
8
|
-
* This is the equivalent of `MenuItem` but without all the rendering logic since we use Angular
|
|
9
|
-
* templates for rendering. Also it caches the state of the item everytime the editor state changes,
|
|
10
|
-
* so Angular can query the state as often as needed without performance hit.
|
|
11
|
-
*/
|
|
12
|
-
export class Item {
|
|
13
|
-
constructor(spec) {
|
|
14
|
-
this.spec = spec;
|
|
15
|
-
/**
|
|
16
|
-
* Whether the item is 'active' (for example, the item for toggling the strong mark might be active when the cursor is in strong text).
|
|
17
|
-
*/
|
|
18
|
-
this.active = false;
|
|
19
|
-
/**
|
|
20
|
-
* Button is shown but disabled, because the item cannot be (un-)applied
|
|
21
|
-
*/
|
|
22
|
-
this.disabled = false;
|
|
23
|
-
/**
|
|
24
|
-
* Whether the item is shown at the moment
|
|
25
|
-
*/
|
|
26
|
-
this.show = true;
|
|
27
|
-
}
|
|
28
|
-
/**
|
|
29
|
-
* Update the item state according to the editor state
|
|
30
|
-
*/
|
|
31
|
-
update(view, state) {
|
|
32
|
-
if (this.spec.active) {
|
|
33
|
-
this.active = this.spec.active(state);
|
|
34
|
-
}
|
|
35
|
-
if (this.spec.enable) {
|
|
36
|
-
this.disabled = !this.spec.enable(state);
|
|
37
|
-
}
|
|
38
|
-
if (this.spec.select) {
|
|
39
|
-
this.show = this.spec.select(state);
|
|
40
|
-
}
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
/**
|
|
44
|
-
* Convert built-in `MenuItem` into our Angular specific `Item`
|
|
45
|
-
*/
|
|
46
|
-
function toItem(item) {
|
|
47
|
-
return new Item(item.spec);
|
|
48
|
-
}
|
|
49
|
-
function canInsert(state, nodeType) {
|
|
50
|
-
const $from = state.selection.$from;
|
|
51
|
-
for (let d = $from.depth; d >= 0; d--) {
|
|
52
|
-
const index = $from.index(d);
|
|
53
|
-
if ($from.node(d).canReplaceWith(index, index, nodeType)) {
|
|
54
|
-
return true;
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
return false;
|
|
58
|
-
}
|
|
59
|
-
// function insertImageItem(nodeType: NodeType) {
|
|
60
|
-
// return new MenuItem({
|
|
61
|
-
// enable(state): boolean {
|
|
62
|
-
// return canInsert(state, nodeType);
|
|
63
|
-
// },
|
|
64
|
-
// run(state, _, view) {
|
|
65
|
-
// let {from, to} = state.selection,
|
|
66
|
-
// attrs = null;
|
|
67
|
-
// if (state.selection instanceof NodeSelection && state.selection.node.type == nodeType) {
|
|
68
|
-
// attrs = state.selection.node.attrs;
|
|
69
|
-
// }
|
|
70
|
-
// openPrompt({
|
|
71
|
-
// title: 'Insert image',
|
|
72
|
-
// fields: {
|
|
73
|
-
// src: new TextField({label: 'Location', required: true, value: attrs && attrs.src}),
|
|
74
|
-
// title: new TextField({label: 'Title', value: attrs && attrs.title}),
|
|
75
|
-
// alt: new TextField({
|
|
76
|
-
// label: 'Description',
|
|
77
|
-
// value: attrs ? attrs.alt : state.doc.textBetween(from, to, ' '),
|
|
78
|
-
// }),
|
|
79
|
-
// },
|
|
80
|
-
// callback(attrs) {
|
|
81
|
-
// view.dispatch(view.state.tr.replaceSelectionWith(nodeType.createAndFill(attrs)));
|
|
82
|
-
// view.focus();
|
|
83
|
-
// },
|
|
84
|
-
// });
|
|
85
|
-
// },
|
|
86
|
-
// });
|
|
87
|
-
// }
|
|
88
|
-
function cmdItem(cmd, options = {}, useEnable = false) {
|
|
89
|
-
const passedOptions = Object.assign({ run: cmd }, options);
|
|
90
|
-
if ((!options.enable || useEnable) && !options.select) {
|
|
91
|
-
passedOptions[options.enable ? 'enable' : 'select'] = (state) => cmd(state);
|
|
92
|
-
}
|
|
93
|
-
return new Item(passedOptions);
|
|
94
|
-
}
|
|
95
|
-
function markActive(state, type) {
|
|
96
|
-
const { from, $from, to, empty } = state.selection;
|
|
97
|
-
if (empty) {
|
|
98
|
-
return !!type.isInSet(state.storedMarks || $from.marks());
|
|
99
|
-
}
|
|
100
|
-
else {
|
|
101
|
-
return state.doc.rangeHasMark(from, to, type);
|
|
102
|
-
}
|
|
103
|
-
}
|
|
104
|
-
function markItem(markType, options = {}) {
|
|
105
|
-
const passedOptions = Object.assign({ active(state) {
|
|
106
|
-
return markActive(state, markType);
|
|
107
|
-
} }, options);
|
|
108
|
-
return cmdItem(toggleMark(markType), passedOptions, true);
|
|
109
|
-
}
|
|
110
|
-
function linkItem(markType, dialog) {
|
|
111
|
-
return new Item({
|
|
112
|
-
active(state) {
|
|
113
|
-
return markActive(state, markType);
|
|
114
|
-
},
|
|
115
|
-
enable(state) {
|
|
116
|
-
return !state.selection.empty;
|
|
117
|
-
},
|
|
118
|
-
run(state, dispatch, view) {
|
|
119
|
-
if (markActive(state, markType)) {
|
|
120
|
-
toggleMark(markType)(state, dispatch);
|
|
121
|
-
return true;
|
|
122
|
-
}
|
|
123
|
-
dialog
|
|
124
|
-
.open(LinkDialogComponent, {
|
|
125
|
-
data: {
|
|
126
|
-
href: '',
|
|
127
|
-
title: '',
|
|
128
|
-
},
|
|
129
|
-
})
|
|
130
|
-
.afterClosed()
|
|
131
|
-
.subscribe(result => {
|
|
132
|
-
if (result) {
|
|
133
|
-
if (!result.title) {
|
|
134
|
-
delete result.title;
|
|
135
|
-
}
|
|
136
|
-
toggleMark(markType, result)(view.state, view.dispatch);
|
|
137
|
-
}
|
|
138
|
-
view.focus();
|
|
139
|
-
});
|
|
140
|
-
},
|
|
141
|
-
});
|
|
142
|
-
}
|
|
143
|
-
function wrapListItem(nodeType) {
|
|
144
|
-
return cmdItem(wrapInList(nodeType));
|
|
145
|
-
}
|
|
146
|
-
/**
|
|
147
|
-
* Given a schema, look for default mark and node types in it and
|
|
148
|
-
* return an object with relevant menu items relating to those marks:
|
|
149
|
-
*/
|
|
150
|
-
export function buildMenuItems(schema, dialog) {
|
|
151
|
-
const r = {
|
|
152
|
-
joinUp: toItem(joinUpItem),
|
|
153
|
-
lift: toItem(liftItem),
|
|
154
|
-
selectParentNode: toItem(selectParentNodeItem),
|
|
155
|
-
undo: toItem(undoItem),
|
|
156
|
-
redo: toItem(redoItem),
|
|
157
|
-
};
|
|
158
|
-
let type;
|
|
159
|
-
type = schema.marks.strong;
|
|
160
|
-
if (type) {
|
|
161
|
-
r.toggleStrong = markItem(type);
|
|
162
|
-
}
|
|
163
|
-
type = schema.marks.em;
|
|
164
|
-
if (type) {
|
|
165
|
-
r.toggleEm = markItem(type);
|
|
166
|
-
}
|
|
167
|
-
type = schema.marks.code;
|
|
168
|
-
if (type) {
|
|
169
|
-
r.toggleCode = markItem(type);
|
|
170
|
-
}
|
|
171
|
-
type = schema.marks.link;
|
|
172
|
-
if (type) {
|
|
173
|
-
r.toggleLink = linkItem(type, dialog);
|
|
174
|
-
}
|
|
175
|
-
type = schema.nodes.image;
|
|
176
|
-
if (type) {
|
|
177
|
-
// r.insertImage = insertImageItem(type);
|
|
178
|
-
}
|
|
179
|
-
type = schema.nodes.bullet_list;
|
|
180
|
-
if (type) {
|
|
181
|
-
r.wrapBulletList = wrapListItem(type);
|
|
182
|
-
}
|
|
183
|
-
type = schema.nodes.ordered_list;
|
|
184
|
-
if (type) {
|
|
185
|
-
r.wrapOrderedList = wrapListItem(type);
|
|
186
|
-
}
|
|
187
|
-
type = schema.nodes.blockquote;
|
|
188
|
-
if (type) {
|
|
189
|
-
r.wrapBlockQuote = toItem(wrapItem(type, {}));
|
|
190
|
-
}
|
|
191
|
-
type = schema.nodes.paragraph;
|
|
192
|
-
if (type) {
|
|
193
|
-
r.makeParagraph = toItem(blockTypeItem(type, {}));
|
|
194
|
-
}
|
|
195
|
-
type = schema.nodes.code_block;
|
|
196
|
-
if (type) {
|
|
197
|
-
r.makeCodeBlock = toItem(blockTypeItem(type, {}));
|
|
198
|
-
}
|
|
199
|
-
type = schema.nodes.heading;
|
|
200
|
-
if (type) {
|
|
201
|
-
r.makeHead1 = toItem(blockTypeItem(type, { attrs: { level: 1 } }));
|
|
202
|
-
r.makeHead2 = toItem(blockTypeItem(type, { attrs: { level: 2 } }));
|
|
203
|
-
r.makeHead3 = toItem(blockTypeItem(type, { attrs: { level: 3 } }));
|
|
204
|
-
r.makeHead4 = toItem(blockTypeItem(type, { attrs: { level: 4 } }));
|
|
205
|
-
r.makeHead5 = toItem(blockTypeItem(type, { attrs: { level: 5 } }));
|
|
206
|
-
r.makeHead6 = toItem(blockTypeItem(type, { attrs: { level: 6 } }));
|
|
207
|
-
}
|
|
208
|
-
type = schema.nodes.horizontal_rule;
|
|
209
|
-
if (type) {
|
|
210
|
-
const hr = type;
|
|
211
|
-
r.insertHorizontalRule = new Item({
|
|
212
|
-
enable(state) {
|
|
213
|
-
return canInsert(state, hr);
|
|
214
|
-
},
|
|
215
|
-
run(state, dispatch) {
|
|
216
|
-
dispatch(state.tr.replaceSelectionWith(hr.create()));
|
|
217
|
-
},
|
|
218
|
-
});
|
|
219
|
-
}
|
|
220
|
-
return r;
|
|
221
|
-
}
|
|
222
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu.js","sourceRoot":"","sources":["../../../../projects/natural-editor/src/lib/menu.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    blockTypeItem,\n    joinUpItem,\n    liftItem,\n    MenuItem,\n    MenuItemSpec,\n    redoItem,\n    selectParentNodeItem,\n    undoItem,\n    wrapItem,\n} from 'prosemirror-menu';\nimport {EditorState, Transaction} from 'prosemirror-state';\nimport {Command, toggleMark} from 'prosemirror-commands';\nimport {wrapInList} from 'prosemirror-schema-list';\nimport {MarkType, NodeType, Schema} from 'prosemirror-model';\nimport {MatDialog} from '@angular/material/dialog';\nimport {LinkDialogComponent, LinkDialogData} from './link-dialog/link-dialog.component';\nimport {EditorView} from 'prosemirror-view';\n\n/**\n * One item of the menu.\n *\n * This is the equivalent of `MenuItem` but without all the rendering logic since we use Angular\n * templates for rendering. Also it caches the state of the item everytime the editor state changes,\n * so Angular can query the state as often as needed without performance hit.\n */\nexport class Item {\n    /**\n     * Whether the item is 'active' (for example, the item for toggling the strong mark might be active when the cursor is in strong text).\n     */\n    public active = false;\n\n    /**\n     * Button is shown but disabled, because the item cannot be (un-)applied\n     */\n    public disabled = false;\n\n    /**\n     * Whether the item is shown at the moment\n     */\n    public show = true;\n\n    constructor(public readonly spec: MenuItemSpec) {}\n\n    /**\n     * Update the item state according to the editor state\n     */\n    public update(view: EditorView, state: EditorState): void {\n        if (this.spec.active) {\n            this.active = this.spec.active(state);\n        }\n\n        if (this.spec.enable) {\n            this.disabled = !this.spec.enable(state);\n        }\n\n        if (this.spec.select) {\n            this.show = this.spec.select(state);\n        }\n    }\n}\n\n/**\n * Convert built-in `MenuItem` into our Angular specific `Item`\n */\nfunction toItem(item: MenuItem): Item {\n    return new Item(item.spec);\n}\n\nfunction canInsert(state: EditorState, nodeType: NodeType): boolean {\n    const $from = state.selection.$from;\n    for (let d = $from.depth; d >= 0; d--) {\n        const index = $from.index(d);\n        if ($from.node(d).canReplaceWith(index, index, nodeType)) {\n            return true;\n        }\n    }\n\n    return false;\n}\n\n// function insertImageItem(nodeType: NodeType) {\n//     return new MenuItem({\n//         enable(state): boolean {\n//             return canInsert(state, nodeType);\n//         },\n//         run(state, _, view) {\n//             let {from, to} = state.selection,\n//                 attrs = null;\n//             if (state.selection instanceof NodeSelection && state.selection.node.type == nodeType) {\n//                 attrs = state.selection.node.attrs;\n//             }\n//             openPrompt({\n//                 title: 'Insert image',\n//                 fields: {\n//                     src: new TextField({label: 'Location', required: true, value: attrs && attrs.src}),\n//                     title: new TextField({label: 'Title', value: attrs && attrs.title}),\n//                     alt: new TextField({\n//                         label: 'Description',\n//                         value: attrs ? attrs.alt : state.doc.textBetween(from, to, ' '),\n//                     }),\n//                 },\n//                 callback(attrs) {\n//                     view.dispatch(view.state.tr.replaceSelectionWith(nodeType.createAndFill(attrs)));\n//                     view.focus();\n//                 },\n//             });\n//         },\n//     });\n// }\n\nfunction cmdItem(cmd: Command, options: Partial<MenuItemSpec> = {}, useEnable = false): Item {\n    const passedOptions: MenuItemSpec = {\n        run: cmd,\n        ...options,\n    };\n\n    if ((!options.enable || useEnable) && !options.select) {\n        passedOptions[options.enable ? 'enable' : 'select'] = (state: EditorState) => cmd(state);\n    }\n\n    return new Item(passedOptions);\n}\n\nfunction markActive(state: EditorState, type: MarkType): boolean {\n    const {from, $from, to, empty} = state.selection;\n    if (empty) {\n        return !!type.isInSet(state.storedMarks || $from.marks());\n    } else {\n        return state.doc.rangeHasMark(from, to, type);\n    }\n}\n\nfunction markItem(markType: MarkType, options: Partial<MenuItemSpec> = {}): Item {\n    const passedOptions: Partial<MenuItemSpec> = {\n        active(state: EditorState): boolean {\n            return markActive(state, markType);\n        },\n        ...options,\n    };\n\n    return cmdItem(toggleMark(markType), passedOptions, true);\n}\n\nfunction linkItem(markType: MarkType, dialog: MatDialog): Item {\n    return new Item({\n        active(state: EditorState): boolean {\n            return markActive(state, markType);\n        },\n        enable(state: EditorState): boolean {\n            return !state.selection.empty;\n        },\n        run(state: EditorState, dispatch: (p: Transaction) => void, view: EditorView): boolean | void {\n            if (markActive(state, markType)) {\n                toggleMark(markType)(state, dispatch);\n                return true;\n            }\n\n            dialog\n                .open<LinkDialogComponent, LinkDialogData, LinkDialogData>(LinkDialogComponent, {\n                    data: {\n                        href: '',\n                        title: '',\n                    },\n                })\n                .afterClosed()\n                .subscribe(result => {\n                    if (result) {\n                        if (!result.title) {\n                            delete result.title;\n                        }\n\n                        toggleMark(markType, result)(view.state, view.dispatch);\n                    }\n\n                    view.focus();\n                });\n        },\n    });\n}\n\nfunction wrapListItem(nodeType: NodeType): Item {\n    return cmdItem(wrapInList(nodeType));\n}\n\nexport type Key =\n    | 'toggleStrong'\n    | 'toggleEm'\n    | 'toggleCode'\n    | 'toggleLink'\n    | 'insertImage'\n    | 'wrapBulletList'\n    | 'wrapOrderedList'\n    | 'wrapBlockQuote'\n    | 'makeParagraph'\n    | 'makeCodeBlock'\n    | 'makeHead1'\n    | 'makeHead2'\n    | 'makeHead3'\n    | 'makeHead4'\n    | 'makeHead5'\n    | 'makeHead6'\n    | 'insertHorizontalRule'\n    | 'joinUp'\n    | 'lift'\n    | 'selectParentNode'\n    | 'undo'\n    | 'redo';\n\nexport type MenuItems = Partial<Record<Key, Item>>;\n\n/**\n * Given a schema, look for default mark and node types in it and\n * return an object with relevant menu items relating to those marks:\n */\nexport function buildMenuItems(schema: Schema, dialog: MatDialog): MenuItems {\n    const r: MenuItems = {\n        joinUp: toItem(joinUpItem),\n        lift: toItem(liftItem),\n        selectParentNode: toItem(selectParentNodeItem),\n        undo: toItem(undoItem as unknown as MenuItem), // Typing is incorrect, so we force it\n        redo: toItem(redoItem as unknown as MenuItem),\n    };\n\n    let type: MarkType | NodeType | undefined;\n    type = schema.marks.strong;\n    if (type) {\n        r.toggleStrong = markItem(type);\n    }\n\n    type = schema.marks.em;\n    if (type) {\n        r.toggleEm = markItem(type);\n    }\n\n    type = schema.marks.code;\n    if (type) {\n        r.toggleCode = markItem(type);\n    }\n\n    type = schema.marks.link;\n    if (type) {\n        r.toggleLink = linkItem(type, dialog);\n    }\n\n    type = schema.nodes.image;\n    if (type) {\n        // r.insertImage = insertImageItem(type);\n    }\n\n    type = schema.nodes.bullet_list;\n    if (type) {\n        r.wrapBulletList = wrapListItem(type);\n    }\n\n    type = schema.nodes.ordered_list;\n    if (type) {\n        r.wrapOrderedList = wrapListItem(type);\n    }\n\n    type = schema.nodes.blockquote;\n    if (type) {\n        r.wrapBlockQuote = toItem(wrapItem(type, {}));\n    }\n\n    type = schema.nodes.paragraph;\n    if (type) {\n        r.makeParagraph = toItem(blockTypeItem(type, {}));\n    }\n\n    type = schema.nodes.code_block;\n    if (type) {\n        r.makeCodeBlock = toItem(blockTypeItem(type, {}));\n    }\n\n    type = schema.nodes.heading;\n    if (type) {\n        r.makeHead1 = toItem(blockTypeItem(type, {attrs: {level: 1}}));\n        r.makeHead2 = toItem(blockTypeItem(type, {attrs: {level: 2}}));\n        r.makeHead3 = toItem(blockTypeItem(type, {attrs: {level: 3}}));\n        r.makeHead4 = toItem(blockTypeItem(type, {attrs: {level: 4}}));\n        r.makeHead5 = toItem(blockTypeItem(type, {attrs: {level: 5}}));\n        r.makeHead6 = toItem(blockTypeItem(type, {attrs: {level: 6}}));\n    }\n\n    type = schema.nodes.horizontal_rule;\n    if (type) {\n        const hr = type;\n        r.insertHorizontalRule = new Item({\n            enable(state): boolean {\n                return canInsert(state, hr);\n            },\n            run(state, dispatch): void {\n                dispatch(state.tr.replaceSelectionWith(hr.create()));\n            },\n        });\n    }\n\n    return r;\n}\n"]}
|
package/esm2015/lib/schema.js
DELETED
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
import { marks, nodes } from 'prosemirror-schema-basic';
|
|
2
|
-
import { addListNodes } from 'prosemirror-schema-list';
|
|
3
|
-
import { Schema } from 'prosemirror-model';
|
|
4
|
-
const myNodes = {
|
|
5
|
-
heading: nodes.heading,
|
|
6
|
-
doc: nodes.doc,
|
|
7
|
-
paragraph: nodes.paragraph,
|
|
8
|
-
text: nodes.text,
|
|
9
|
-
hard_break: nodes.hard_break,
|
|
10
|
-
};
|
|
11
|
-
const myMarks = {
|
|
12
|
-
link: marks.link,
|
|
13
|
-
em: marks.em,
|
|
14
|
-
strong: marks.strong,
|
|
15
|
-
};
|
|
16
|
-
const basicSchema = new Schema({ nodes: myNodes, marks: myMarks });
|
|
17
|
-
export const schema = new Schema({
|
|
18
|
-
nodes: addListNodes(basicSchema.spec.nodes, 'paragraph block*', 'block'),
|
|
19
|
-
marks: basicSchema.spec.marks,
|
|
20
|
-
});
|
|
21
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NoZW1hLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC1lZGl0b3Ivc3JjL2xpYi9zY2hlbWEudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLEtBQUssRUFBRSxLQUFLLEVBQUMsTUFBTSwwQkFBMEIsQ0FBQztBQUN0RCxPQUFPLEVBQUMsWUFBWSxFQUFDLE1BQU0seUJBQXlCLENBQUM7QUFDckQsT0FBTyxFQUFDLE1BQU0sRUFBQyxNQUFNLG1CQUFtQixDQUFDO0FBSXpDLE1BQU0sT0FBTyxHQUFlO0lBQ3hCLE9BQU8sRUFBRSxLQUFLLENBQUMsT0FBTztJQUN0QixHQUFHLEVBQUUsS0FBSyxDQUFDLEdBQUc7SUFDZCxTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVM7SUFDMUIsSUFBSSxFQUFFLEtBQUssQ0FBQyxJQUFJO0lBQ2hCLFVBQVUsRUFBRSxLQUFLLENBQUMsVUFBVTtDQUMvQixDQUFDO0FBR0YsTUFBTSxPQUFPLEdBQWU7SUFDeEIsSUFBSSxFQUFFLEtBQUssQ0FBQyxJQUFJO0lBQ2hCLEVBQUUsRUFBRSxLQUFLLENBQUMsRUFBRTtJQUNaLE1BQU0sRUFBRSxLQUFLLENBQUMsTUFBTTtDQUN2QixDQUFDO0FBRUYsTUFBTSxXQUFXLEdBQUcsSUFBSSxNQUFNLENBQUMsRUFBQyxLQUFLLEVBQUUsT0FBTyxFQUFFLEtBQUssRUFBRSxPQUFPLEVBQUMsQ0FBQyxDQUFDO0FBRWpFLE1BQU0sQ0FBQyxNQUFNLE1BQU0sR0FBRyxJQUFJLE1BQU0sQ0FBQztJQUM3QixLQUFLLEVBQUUsWUFBWSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsS0FBWSxFQUFFLGtCQUFrQixFQUFFLE9BQU8sQ0FBQztJQUMvRSxLQUFLLEVBQUUsV0FBVyxDQUFDLElBQUksQ0FBQyxLQUFLO0NBQ2hDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7bWFya3MsIG5vZGVzfSBmcm9tICdwcm9zZW1pcnJvci1zY2hlbWEtYmFzaWMnO1xuaW1wb3J0IHthZGRMaXN0Tm9kZXN9IGZyb20gJ3Byb3NlbWlycm9yLXNjaGVtYS1saXN0JztcbmltcG9ydCB7U2NoZW1hfSBmcm9tICdwcm9zZW1pcnJvci1tb2RlbCc7XG5cbi8vIEtlZXAgb25seSBiYXNpYyBlbGVtZW50c1xudHlwZSBCYXNpY05vZGVzID0gT21pdDx0eXBlb2Ygbm9kZXMsICdpbWFnZScgfCAnY29kZV9ibG9jaycgfCAnYmxvY2txdW90ZScgfCAnaG9yaXpvbnRhbF9ydWxlJz47XG5jb25zdCBteU5vZGVzOiBCYXNpY05vZGVzID0ge1xuICAgIGhlYWRpbmc6IG5vZGVzLmhlYWRpbmcsXG4gICAgZG9jOiBub2Rlcy5kb2MsXG4gICAgcGFyYWdyYXBoOiBub2Rlcy5wYXJhZ3JhcGgsXG4gICAgdGV4dDogbm9kZXMudGV4dCxcbiAgICBoYXJkX2JyZWFrOiBub2Rlcy5oYXJkX2JyZWFrLFxufTtcblxudHlwZSBCYXNpY01hcmtzID0gT21pdDx0eXBlb2YgbWFya3MsICdjb2RlJz47XG5jb25zdCBteU1hcmtzOiBCYXNpY01hcmtzID0ge1xuICAgIGxpbms6IG1hcmtzLmxpbmssXG4gICAgZW06IG1hcmtzLmVtLFxuICAgIHN0cm9uZzogbWFya3Muc3Ryb25nLFxufTtcblxuY29uc3QgYmFzaWNTY2hlbWEgPSBuZXcgU2NoZW1hKHtub2RlczogbXlOb2RlcywgbWFya3M6IG15TWFya3N9KTtcblxuZXhwb3J0IGNvbnN0IHNjaGVtYSA9IG5ldyBTY2hlbWEoe1xuICAgIG5vZGVzOiBhZGRMaXN0Tm9kZXMoYmFzaWNTY2hlbWEuc3BlYy5ub2RlcyBhcyBhbnksICdwYXJhZ3JhcGggYmxvY2sqJywgJ2Jsb2NrJyksXG4gICAgbWFya3M6IGJhc2ljU2NoZW1hLnNwZWMubWFya3MsXG59KTtcbiJdfQ==
|
package/lib/schema.d.ts
DELETED
|
@@ -1,23 +0,0 @@
|
|
|
1
|
-
@use 'sass:map';
|
|
2
|
-
@use '~@angular/material' as mat;
|
|
3
|
-
|
|
4
|
-
@mixin natural-editor($theme) {
|
|
5
|
-
$config: mat.get-color-config($theme);
|
|
6
|
-
$foreground: map.get($config, foreground);
|
|
7
|
-
$is-dark-theme: map.get($config, is-dark);
|
|
8
|
-
|
|
9
|
-
// Copied from Material Form Field
|
|
10
|
-
$fill-background: mat.get-color-from-palette($foreground, base, if($is-dark-theme, 0.1, 0.04));
|
|
11
|
-
$underline-color: mat.get-color-from-palette($foreground, divider, if($is-dark-theme, 0.5, 0.42));
|
|
12
|
-
|
|
13
|
-
natural-editor {
|
|
14
|
-
.menu,
|
|
15
|
-
.ProseMirror {
|
|
16
|
-
background-color: $fill-background;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
.menu {
|
|
20
|
-
border-color: $underline-color;
|
|
21
|
-
}
|
|
22
|
-
}
|
|
23
|
-
}
|