@ecodev/natural-editor 1.1.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/README.md +42 -0
- package/bundles/ecodev-natural-editor.umd.js +858 -0
- package/bundles/ecodev-natural-editor.umd.js.map +1 -0
- package/ecodev-natural-editor.d.ts +5 -0
- package/esm2015/ecodev-natural-editor.js +5 -0
- package/esm2015/lib/editor.component.js +161 -0
- package/esm2015/lib/editor.module.js +72 -0
- package/esm2015/lib/link-dialog/link-dialog.component.js +43 -0
- package/esm2015/lib/menu.js +220 -0
- package/esm2015/lib/schema.js +21 -0
- package/esm2015/public-api.js +8 -0
- package/fesm2015/ecodev-natural-editor.js +508 -0
- package/fesm2015/ecodev-natural-editor.js.map +1 -0
- package/lib/editor.component.d.ts +47 -0
- package/lib/editor.module.d.ts +19 -0
- package/lib/link-dialog/link-dialog.component.d.ts +18 -0
- package/lib/menu.d.ts +39 -0
- package/lib/schema.d.ts +2 -0
- package/package.json +37 -0
- package/public-api.d.ts +3 -0
|
@@ -0,0 +1,220 @@
|
|
|
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 && !result.title) {
|
|
133
|
+
delete result.title;
|
|
134
|
+
}
|
|
135
|
+
toggleMark(markType, result)(view.state, view.dispatch);
|
|
136
|
+
view.focus();
|
|
137
|
+
});
|
|
138
|
+
},
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
function wrapListItem(nodeType) {
|
|
142
|
+
return cmdItem(wrapInList(nodeType));
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Given a schema, look for default mark and node types in it and
|
|
146
|
+
* return an object with relevant menu items relating to those marks:
|
|
147
|
+
*/
|
|
148
|
+
export function buildMenuItems(schema, dialog) {
|
|
149
|
+
const r = {
|
|
150
|
+
joinUp: toItem(joinUpItem),
|
|
151
|
+
lift: toItem(liftItem),
|
|
152
|
+
selectParentNode: toItem(selectParentNodeItem),
|
|
153
|
+
undo: toItem(undoItem),
|
|
154
|
+
redo: toItem(redoItem),
|
|
155
|
+
};
|
|
156
|
+
let type;
|
|
157
|
+
type = schema.marks.strong;
|
|
158
|
+
if (type) {
|
|
159
|
+
r.toggleStrong = markItem(type);
|
|
160
|
+
}
|
|
161
|
+
type = schema.marks.em;
|
|
162
|
+
if (type) {
|
|
163
|
+
r.toggleEm = markItem(type);
|
|
164
|
+
}
|
|
165
|
+
type = schema.marks.code;
|
|
166
|
+
if (type) {
|
|
167
|
+
r.toggleCode = markItem(type);
|
|
168
|
+
}
|
|
169
|
+
type = schema.marks.link;
|
|
170
|
+
if (type) {
|
|
171
|
+
r.toggleLink = linkItem(type, dialog);
|
|
172
|
+
}
|
|
173
|
+
type = schema.nodes.image;
|
|
174
|
+
if (type) {
|
|
175
|
+
// r.insertImage = insertImageItem(type);
|
|
176
|
+
}
|
|
177
|
+
type = schema.nodes.bullet_list;
|
|
178
|
+
if (type) {
|
|
179
|
+
r.wrapBulletList = wrapListItem(type);
|
|
180
|
+
}
|
|
181
|
+
type = schema.nodes.ordered_list;
|
|
182
|
+
if (type) {
|
|
183
|
+
r.wrapOrderedList = wrapListItem(type);
|
|
184
|
+
}
|
|
185
|
+
type = schema.nodes.blockquote;
|
|
186
|
+
if (type) {
|
|
187
|
+
r.wrapBlockQuote = toItem(wrapItem(type, {}));
|
|
188
|
+
}
|
|
189
|
+
type = schema.nodes.paragraph;
|
|
190
|
+
if (type) {
|
|
191
|
+
r.makeParagraph = toItem(blockTypeItem(type, {}));
|
|
192
|
+
}
|
|
193
|
+
type = schema.nodes.code_block;
|
|
194
|
+
if (type) {
|
|
195
|
+
r.makeCodeBlock = toItem(blockTypeItem(type, {}));
|
|
196
|
+
}
|
|
197
|
+
type = schema.nodes.heading;
|
|
198
|
+
if (type) {
|
|
199
|
+
r.makeHead1 = toItem(blockTypeItem(type, { attrs: { level: 1 } }));
|
|
200
|
+
r.makeHead2 = toItem(blockTypeItem(type, { attrs: { level: 2 } }));
|
|
201
|
+
r.makeHead3 = toItem(blockTypeItem(type, { attrs: { level: 3 } }));
|
|
202
|
+
r.makeHead4 = toItem(blockTypeItem(type, { attrs: { level: 4 } }));
|
|
203
|
+
r.makeHead5 = toItem(blockTypeItem(type, { attrs: { level: 5 } }));
|
|
204
|
+
r.makeHead6 = toItem(blockTypeItem(type, { attrs: { level: 6 } }));
|
|
205
|
+
}
|
|
206
|
+
type = schema.nodes.horizontal_rule;
|
|
207
|
+
if (type) {
|
|
208
|
+
const hr = type;
|
|
209
|
+
r.insertHorizontalRule = new Item({
|
|
210
|
+
enable(state) {
|
|
211
|
+
return canInsert(state, hr);
|
|
212
|
+
},
|
|
213
|
+
run(state, dispatch) {
|
|
214
|
+
dispatch(state.tr.replaceSelectionWith(hr.create()));
|
|
215
|
+
},
|
|
216
|
+
});
|
|
217
|
+
}
|
|
218
|
+
return r;
|
|
219
|
+
}
|
|
220
|
+
//# 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 && !result.title) {\n                        delete result.title;\n                    }\n\n                    toggleMark(markType, result)(view.state, view.dispatch);\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"]}
|
|
@@ -0,0 +1,21 @@
|
|
|
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==
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
// Load `$localize` onto the global scope - to be able to use that function to translate strings in components/services.
|
|
2
|
+
import '@angular/localize/init';
|
|
3
|
+
/*
|
|
4
|
+
* Public API Surface of natural-editor
|
|
5
|
+
*/
|
|
6
|
+
export { NaturalEditorComponent } from './lib/editor.component';
|
|
7
|
+
export { NaturalEditorModule } from './lib/editor.module';
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwtZWRpdG9yL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLHdIQUF3SDtBQUN4SCxPQUFPLHdCQUF3QixDQUFDO0FBR2hDOztHQUVHO0FBRUgsT0FBTyxFQUFDLHNCQUFzQixFQUFDLE1BQU0sd0JBQXdCLENBQUM7QUFDOUQsT0FBTyxFQUFDLG1CQUFtQixFQUFDLE1BQU0scUJBQXFCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBMb2FkIGAkbG9jYWxpemVgIG9udG8gdGhlIGdsb2JhbCBzY29wZSAtIHRvIGJlIGFibGUgdG8gdXNlIHRoYXQgZnVuY3Rpb24gdG8gdHJhbnNsYXRlIHN0cmluZ3MgaW4gY29tcG9uZW50cy9zZXJ2aWNlcy5cbmltcG9ydCAnQGFuZ3VsYXIvbG9jYWxpemUvaW5pdCc7XG5pbXBvcnQge05hdHVyYWxFZGl0b3JDb21wb25lbnR9IGZyb20gJy4vbGliL2VkaXRvci5jb21wb25lbnQnO1xuXG4vKlxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIG5hdHVyYWwtZWRpdG9yXG4gKi9cblxuZXhwb3J0IHtOYXR1cmFsRWRpdG9yQ29tcG9uZW50fSBmcm9tICcuL2xpYi9lZGl0b3IuY29tcG9uZW50JztcbmV4cG9ydCB7TmF0dXJhbEVkaXRvck1vZHVsZX0gZnJvbSAnLi9saWIvZWRpdG9yLm1vZHVsZSc7XG4iXX0=
|