@pie-lib/mask-markup 2.0.0-beta.1 → 2.0.0-next.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/CHANGELOG.json +1 -871
- package/CHANGELOG.md +434 -32
- package/LICENSE.md +5 -0
- package/NEXT.CHANGELOG.json +1 -0
- package/lib/choices/choice.js +101 -129
- package/lib/choices/choice.js.map +1 -1
- package/lib/choices/index.js +28 -48
- package/lib/choices/index.js.map +1 -1
- package/lib/componentize.js +2 -6
- package/lib/componentize.js.map +1 -1
- package/lib/components/blank.js +315 -246
- package/lib/components/blank.js.map +1 -1
- package/lib/components/correct-input.js +47 -66
- package/lib/components/correct-input.js.map +1 -1
- package/lib/components/dropdown.js +399 -156
- package/lib/components/dropdown.js.map +1 -1
- package/lib/components/input.js +15 -19
- package/lib/components/input.js.map +1 -1
- package/lib/constructed-response.js +81 -28
- package/lib/constructed-response.js.map +1 -1
- package/lib/customizable.js +44 -0
- package/lib/customizable.js.map +1 -0
- package/lib/drag-in-the-blank.js +160 -96
- package/lib/drag-in-the-blank.js.map +1 -1
- package/lib/index.js +8 -7
- package/lib/index.js.map +1 -1
- package/lib/inline-dropdown.js +10 -14
- package/lib/inline-dropdown.js.map +1 -1
- package/lib/mask.js +93 -101
- package/lib/mask.js.map +1 -1
- package/lib/serialization.js +36 -81
- package/lib/serialization.js.map +1 -1
- package/lib/with-mask.js +53 -49
- package/lib/with-mask.js.map +1 -1
- package/package.json +26 -15
- package/src/__tests__/drag-in-the-blank.test.js +111 -0
- package/src/__tests__/index.test.js +39 -0
- package/src/__tests__/mask.test.js +187 -0
- package/src/__tests__/serialization.test.js +54 -0
- package/src/__tests__/utils.js +1 -0
- package/src/__tests__/with-mask.test.js +76 -0
- package/src/choices/__tests__/index.test.js +75 -0
- package/src/choices/choice.jsx +84 -83
- package/src/choices/index.jsx +25 -15
- package/src/components/__tests__/blank.test.js +138 -0
- package/src/components/__tests__/correct-input.test.js +90 -0
- package/src/components/__tests__/dropdown.test.js +93 -0
- package/src/components/__tests__/input.test.js +102 -0
- package/src/components/blank.jsx +319 -195
- package/src/components/correct-input.jsx +45 -46
- package/src/components/dropdown.jsx +374 -139
- package/src/components/input.jsx +6 -3
- package/src/constructed-response.jsx +81 -18
- package/src/customizable.jsx +35 -0
- package/src/drag-in-the-blank.jsx +159 -47
- package/src/index.js +3 -1
- package/src/inline-dropdown.jsx +6 -3
- package/src/mask.jsx +75 -30
- package/src/serialization.js +37 -44
- package/src/with-mask.jsx +36 -3
- package/README.md +0 -14
- package/lib/new-serialization.js +0 -320
- package/lib/parse-html.js +0 -16
- package/lib/test-serializer.js +0 -215
- package/src/new-serialization.jsx +0 -291
- package/src/parse-html.js +0 -8
- package/src/test-serializer.js +0 -163
package/lib/test-serializer.js
DELETED
|
@@ -1,215 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
|
|
3
|
-
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
-
|
|
5
|
-
Object.defineProperty(exports, "__esModule", {
|
|
6
|
-
value: true
|
|
7
|
-
});
|
|
8
|
-
exports["default"] = void 0;
|
|
9
|
-
|
|
10
|
-
var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
|
|
11
|
-
|
|
12
|
-
var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
|
|
13
|
-
|
|
14
|
-
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
|
|
15
|
-
|
|
16
|
-
var _react = _interopRequireDefault(require("react"));
|
|
17
|
-
|
|
18
|
-
var _server = _interopRequireDefault(require("react-dom/server"));
|
|
19
|
-
|
|
20
|
-
var _escapeHtml = _interopRequireDefault(require("escape-html"));
|
|
21
|
-
|
|
22
|
-
var _slate = require("slate");
|
|
23
|
-
|
|
24
|
-
var _slateHyperscript = require("slate-hyperscript");
|
|
25
|
-
|
|
26
|
-
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
27
|
-
|
|
28
|
-
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { (0, _defineProperty2["default"])(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
29
|
-
|
|
30
|
-
function allWhitespace(node) {
|
|
31
|
-
// Use ECMA-262 Edition 3 String and RegExp features
|
|
32
|
-
return !/[^\t\n\r ]/.test(node.textContent);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
function defaultParseHtml(html) {
|
|
36
|
-
if (typeof DOMParser === 'undefined') {
|
|
37
|
-
throw new Error('The native `DOMParser` global which the `Html` serializer uses by default is not present in this environment. You must supply the `options.parseHtml` function instead.');
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
var parsed = new DOMParser().parseFromString(html, 'text/html');
|
|
41
|
-
var body = parsed.body;
|
|
42
|
-
var textNodes = document.createTreeWalker(body, NodeFilter.SHOW_TEXT, null, null);
|
|
43
|
-
var n = textNodes.nextNode();
|
|
44
|
-
|
|
45
|
-
while (n) {
|
|
46
|
-
if (allWhitespace(n) || n.nodeValue === "\u200B") {
|
|
47
|
-
n.parentNode.removeChild(n);
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
n = textNodes.nextNode();
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
return body;
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
var Html = /*#__PURE__*/(0, _createClass2["default"])(function Html(props) {
|
|
57
|
-
var _this = this;
|
|
58
|
-
|
|
59
|
-
(0, _classCallCheck2["default"])(this, Html);
|
|
60
|
-
(0, _defineProperty2["default"])(this, "serializeEls", function (node) {
|
|
61
|
-
if (_slate.Text.isText(node)) {
|
|
62
|
-
var string = (0, _escapeHtml["default"])(node.text);
|
|
63
|
-
|
|
64
|
-
if (node.bold) {
|
|
65
|
-
string = /*#__PURE__*/_react["default"].createElement("strong", null, string);
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
return string;
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
var children = (node.children || []).map(function (n) {
|
|
72
|
-
return _this.serializeEls(n);
|
|
73
|
-
});
|
|
74
|
-
|
|
75
|
-
var correctRule = _this.rules.reduce(function (res, rule) {
|
|
76
|
-
return res || rule.serialize(node, children);
|
|
77
|
-
}, null);
|
|
78
|
-
|
|
79
|
-
if (correctRule) {
|
|
80
|
-
return correctRule;
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
switch (node.type) {
|
|
84
|
-
case 'quote':
|
|
85
|
-
return /*#__PURE__*/_react["default"].createElement("blockquote", null, /*#__PURE__*/_react["default"].createElement("p", null, children));
|
|
86
|
-
|
|
87
|
-
case 'paragraph':
|
|
88
|
-
return /*#__PURE__*/_react["default"].createElement("p", null, children);
|
|
89
|
-
|
|
90
|
-
case 'link':
|
|
91
|
-
return /*#__PURE__*/_react["default"].createElement("a", {
|
|
92
|
-
href: (0, _escapeHtml["default"])(node.url)
|
|
93
|
-
}, children);
|
|
94
|
-
|
|
95
|
-
default:
|
|
96
|
-
return children;
|
|
97
|
-
}
|
|
98
|
-
});
|
|
99
|
-
(0, _defineProperty2["default"])(this, "serialize", function (node) {
|
|
100
|
-
var deserialized = _this.serializeEls(node);
|
|
101
|
-
|
|
102
|
-
var html = _server["default"].renderToStaticMarkup( /*#__PURE__*/_react["default"].createElement('body', null, deserialized));
|
|
103
|
-
|
|
104
|
-
var inner = html.slice(6, -7);
|
|
105
|
-
return inner;
|
|
106
|
-
});
|
|
107
|
-
(0, _defineProperty2["default"])(this, "deserialize", function (html) {
|
|
108
|
-
var body = _this.parseHtml(html);
|
|
109
|
-
|
|
110
|
-
if (body.firstChild && body.firstChild.nodeType === Node.TEXT_NODE) {
|
|
111
|
-
body = _this.parseHtml("<p>".concat(html, "</p>"));
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
return _this.deserializeEls(body);
|
|
115
|
-
});
|
|
116
|
-
(0, _defineProperty2["default"])(this, "deserializeEls", function (element) {
|
|
117
|
-
var markAttributes = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
|
|
118
|
-
|
|
119
|
-
if (element.nodeType === Node.TEXT_NODE) {
|
|
120
|
-
return (0, _slateHyperscript.jsx)('text', markAttributes, element.textContent);
|
|
121
|
-
} else if (element.nodeType !== Node.ELEMENT_NODE) {
|
|
122
|
-
return null;
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
var nodeAttributes = _objectSpread({}, markAttributes); // define attributes for text nodes
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
if (element.nodeName === 'STRONG') {
|
|
129
|
-
nodeAttributes.bold = true;
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
var nextFn = function nextFn(nodes) {
|
|
133
|
-
var childNodes = Array.from(nodes);
|
|
134
|
-
var children = Array.from(childNodes).map(function (node) {
|
|
135
|
-
return _this.deserializeEls(node, nodeAttributes);
|
|
136
|
-
}).flat();
|
|
137
|
-
|
|
138
|
-
if (children.length === 0) {
|
|
139
|
-
children.push((0, _slateHyperscript.jsx)('text', nodeAttributes, ''));
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
return children;
|
|
143
|
-
};
|
|
144
|
-
|
|
145
|
-
var correctRule = _this.rules.reduce(function (res, rule) {
|
|
146
|
-
return res || rule.deserialize(element, nextFn);
|
|
147
|
-
}, null);
|
|
148
|
-
|
|
149
|
-
if (correctRule) {
|
|
150
|
-
return correctRule;
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
var childNodes = Array.from(element.childNodes);
|
|
154
|
-
var children = Array.from(childNodes).map(function (node) {
|
|
155
|
-
return _this.deserializeEls(node, nodeAttributes);
|
|
156
|
-
}).flat();
|
|
157
|
-
|
|
158
|
-
if (children.length === 0) {
|
|
159
|
-
children.push((0, _slateHyperscript.jsx)('text', nodeAttributes, ''));
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
switch (element.nodeName) {
|
|
163
|
-
case 'TABLE':
|
|
164
|
-
return (0, _slateHyperscript.jsx)('element', {
|
|
165
|
-
type: 'table'
|
|
166
|
-
}, children);
|
|
167
|
-
|
|
168
|
-
case 'TBODY':
|
|
169
|
-
return (0, _slateHyperscript.jsx)('element', {
|
|
170
|
-
type: 'tbody'
|
|
171
|
-
}, children);
|
|
172
|
-
|
|
173
|
-
case 'TR':
|
|
174
|
-
return (0, _slateHyperscript.jsx)('element', {
|
|
175
|
-
type: 'tr'
|
|
176
|
-
}, children);
|
|
177
|
-
|
|
178
|
-
case 'TD':
|
|
179
|
-
return (0, _slateHyperscript.jsx)('element', {
|
|
180
|
-
type: 'td'
|
|
181
|
-
}, children);
|
|
182
|
-
|
|
183
|
-
case 'BODY':
|
|
184
|
-
return (0, _slateHyperscript.jsx)('fragment', {}, children);
|
|
185
|
-
|
|
186
|
-
case 'BR':
|
|
187
|
-
return '\n';
|
|
188
|
-
|
|
189
|
-
case 'BLOCKQUOTE':
|
|
190
|
-
return (0, _slateHyperscript.jsx)('element', {
|
|
191
|
-
type: 'quote'
|
|
192
|
-
}, children);
|
|
193
|
-
|
|
194
|
-
case 'P':
|
|
195
|
-
return (0, _slateHyperscript.jsx)('element', {
|
|
196
|
-
type: 'paragraph'
|
|
197
|
-
}, children);
|
|
198
|
-
|
|
199
|
-
case 'A':
|
|
200
|
-
return (0, _slateHyperscript.jsx)('element', {
|
|
201
|
-
type: 'link',
|
|
202
|
-
url: element.getAttribute('href')
|
|
203
|
-
}, children);
|
|
204
|
-
|
|
205
|
-
default:
|
|
206
|
-
return children;
|
|
207
|
-
}
|
|
208
|
-
});
|
|
209
|
-
this.defaultBlock = props.defaultBlock;
|
|
210
|
-
this.parseHtml = defaultParseHtml;
|
|
211
|
-
this.rules = props.rules;
|
|
212
|
-
});
|
|
213
|
-
var _default = Html;
|
|
214
|
-
exports["default"] = _default;
|
|
215
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["allWhitespace","node","test","textContent","defaultParseHtml","html","DOMParser","Error","parsed","parseFromString","body","textNodes","document","createTreeWalker","NodeFilter","SHOW_TEXT","n","nextNode","nodeValue","parentNode","removeChild","Html","props","Text","isText","string","escapeHtml","text","bold","children","map","serializeEls","correctRule","rules","reduce","res","rule","serialize","type","url","deserialized","ReactServer","renderToStaticMarkup","React","createElement","inner","slice","parseHtml","firstChild","nodeType","Node","TEXT_NODE","deserializeEls","element","markAttributes","jsx","ELEMENT_NODE","nodeAttributes","nodeName","nextFn","nodes","childNodes","Array","from","flat","length","push","deserialize","getAttribute","defaultBlock"],"sources":["../src/test-serializer.js"],"sourcesContent":["import React from 'react';\nimport ReactServer from 'react-dom/server';\nimport escapeHtml from 'escape-html';\nimport { Text } from 'slate';\nimport { jsx } from 'slate-hyperscript';\n\nfunction allWhitespace(node) {\n  // Use ECMA-262 Edition 3 String and RegExp features\n  return !/[^\\t\\n\\r ]/.test(node.textContent);\n}\n\nfunction defaultParseHtml(html) {\n  if (typeof DOMParser === 'undefined') {\n    throw new Error(\n      'The native `DOMParser` global which the `Html` serializer uses by default is not present in this environment. You must supply the `options.parseHtml` function instead.'\n    );\n  }\n\n  const parsed = new DOMParser().parseFromString(html, 'text/html');\n\n  const { body } = parsed;\n  const textNodes = document.createTreeWalker(body, NodeFilter.SHOW_TEXT, null, null);\n  let n = textNodes.nextNode();\n\n  while (n) {\n    if (allWhitespace(n) || n.nodeValue === '\\u200B') {\n      n.parentNode.removeChild(n);\n    }\n    n = textNodes.nextNode();\n  }\n\n  return body;\n}\n\nclass Html {\n  constructor(props) {\n    this.defaultBlock = props.defaultBlock;\n    this.parseHtml = defaultParseHtml;\n    this.rules = props.rules;\n  }\n\n  serializeEls = node => {\n    if (Text.isText(node)) {\n      let string = escapeHtml(node.text);\n      if (node.bold) {\n        string = <strong>{string}</strong>;\n      }\n      return string;\n    }\n\n    let children = (node.children || []).map(n => this.serializeEls(n));\n\n    const correctRule = this.rules.reduce((res, rule) => {\n      return res || rule.serialize(node, children);\n    }, null);\n\n    if (correctRule) {\n      return correctRule;\n    }\n\n    switch (node.type) {\n      case 'quote':\n        return (\n          <blockquote>\n            <p>{children}</p>\n          </blockquote>\n        );\n      case 'paragraph':\n        return <p>{children}</p>;\n      case 'link':\n        return <a href={escapeHtml(node.url)}>{children}</a>;\n      default:\n        return children;\n    }\n  };\n\n  serialize = node => {\n    const deserialized = this.serializeEls(node);\n    const html = ReactServer.renderToStaticMarkup(React.createElement(\n      'body',\n      null,\n      deserialized\n    ));\n    const inner = html.slice(6, -7);\n    return inner;\n  };\n\n  deserialize = html => {\n    let body = this.parseHtml(html);\n\n    if (body.firstChild && body.firstChild.nodeType === Node.TEXT_NODE) {\n      body = this.parseHtml(`<p>${html}</p>`);\n    }\n\n    return this.deserializeEls(body);\n  };\n\n  deserializeEls = (element, markAttributes = {}) => {\n    if (element.nodeType === Node.TEXT_NODE) {\n      return jsx('text', markAttributes, element.textContent);\n    } else if (element.nodeType !== Node.ELEMENT_NODE) {\n      return null;\n    }\n\n    const nodeAttributes = { ...markAttributes };\n\n    // define attributes for text nodes\n    if (element.nodeName === 'STRONG') {\n      nodeAttributes.bold = true;\n    }\n\n    const nextFn = nodes => {\n      const childNodes = Array.from(nodes);\n      const children = Array.from(childNodes)\n        .map(node => this.deserializeEls(node, nodeAttributes))\n        .flat();\n\n      if (children.length === 0) {\n        children.push(jsx('text', nodeAttributes, ''));\n      }\n\n      return children;\n    };\n\n    const correctRule = this.rules.reduce((res, rule) => {\n      return res || rule.deserialize(element, nextFn);\n    }, null);\n\n    if (correctRule) {\n      return correctRule;\n    }\n\n    const childNodes = Array.from(element.childNodes);\n    const children = Array.from(childNodes)\n      .map(node => this.deserializeEls(node, nodeAttributes))\n      .flat();\n\n    if (children.length === 0) {\n      children.push(jsx('text', nodeAttributes, ''));\n    }\n\n    switch (element.nodeName) {\n      case 'TABLE':\n        return jsx('element', { type: 'table' }, children);\n      case 'TBODY':\n        return jsx('element', { type: 'tbody' }, children);\n      case 'TR':\n        return jsx('element', { type: 'tr' }, children);\n      case 'TD':\n        return jsx('element', { type: 'td' }, children);\n      case 'BODY':\n        return jsx('fragment', {}, children);\n      case 'BR':\n        return '\\n';\n      case 'BLOCKQUOTE':\n        return jsx('element', { type: 'quote' }, children);\n      case 'P':\n        return jsx('element', { type: 'paragraph' }, children);\n      case 'A':\n        return jsx('element', { type: 'link', url: element.getAttribute('href') }, children);\n      default:\n        return children;\n    }\n  };\n}\n\nexport default Html;\n"],"mappings":"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"}
|
|
@@ -1,291 +0,0 @@
|
|
|
1
|
-
import TestSerializer from './test-serializer';
|
|
2
|
-
import React from 'react';
|
|
3
|
-
import debug from 'debug';
|
|
4
|
-
import { object as toStyleObject } from 'to-style';
|
|
5
|
-
|
|
6
|
-
import { serialization as imgSerialization } from './plugins/image';
|
|
7
|
-
import { serialization as mathSerialization } from './plugins/math';
|
|
8
|
-
import { serialization as mediaSerialization } from './plugins/media';
|
|
9
|
-
import { serialization as listSerialization } from './plugins/list';
|
|
10
|
-
import { serialization as tableSerialization } from './plugins/table';
|
|
11
|
-
import { serialization as responseAreaSerialization } from './plugins/respArea';
|
|
12
|
-
import { Mark, Value } from 'slate';
|
|
13
|
-
import { jsx } from 'slate-hyperscript';
|
|
14
|
-
|
|
15
|
-
const log = debug('@pie-lib:editable-html:serialization');
|
|
16
|
-
|
|
17
|
-
/**
|
|
18
|
-
* Tags to blocks.
|
|
19
|
-
*
|
|
20
|
-
* @type {Object}
|
|
21
|
-
*/
|
|
22
|
-
|
|
23
|
-
export const BLOCK_TAGS = {
|
|
24
|
-
div: 'div',
|
|
25
|
-
span: 'span',
|
|
26
|
-
p: 'paragraph',
|
|
27
|
-
blockquote: 'quote',
|
|
28
|
-
pre: 'code',
|
|
29
|
-
h1: 'heading-one',
|
|
30
|
-
h2: 'heading-two',
|
|
31
|
-
h3: 'heading-three',
|
|
32
|
-
h4: 'heading-four',
|
|
33
|
-
h5: 'heading-five',
|
|
34
|
-
h6: 'heading-six'
|
|
35
|
-
};
|
|
36
|
-
|
|
37
|
-
/**
|
|
38
|
-
* Tags to marks.
|
|
39
|
-
*
|
|
40
|
-
* @type {Object}
|
|
41
|
-
*/
|
|
42
|
-
|
|
43
|
-
const MARK_TAGS = {
|
|
44
|
-
b: 'bold',
|
|
45
|
-
em: 'italic',
|
|
46
|
-
u: 'underline',
|
|
47
|
-
s: 'strikethrough',
|
|
48
|
-
code: 'code',
|
|
49
|
-
strong: 'bold'
|
|
50
|
-
};
|
|
51
|
-
|
|
52
|
-
export const parseStyleString = s => {
|
|
53
|
-
const regex = /([\w-]*)\s*:\s*([^;]*)/g;
|
|
54
|
-
let match;
|
|
55
|
-
const result = {};
|
|
56
|
-
while ((match = regex.exec(s))) {
|
|
57
|
-
result[match[1]] = match[2].trim();
|
|
58
|
-
}
|
|
59
|
-
return result;
|
|
60
|
-
};
|
|
61
|
-
|
|
62
|
-
export const getBase64 = file => {
|
|
63
|
-
return new Promise((resolve, reject) => {
|
|
64
|
-
const reader = new FileReader();
|
|
65
|
-
reader.readAsDataURL(file);
|
|
66
|
-
reader.onload = () => resolve(reader.result);
|
|
67
|
-
reader.onerror = error => reject(error);
|
|
68
|
-
});
|
|
69
|
-
};
|
|
70
|
-
|
|
71
|
-
export const reactAttributes = o => toStyleObject(o, { camelize: true, addUnits: false });
|
|
72
|
-
|
|
73
|
-
const attributesToMap = el => (acc, attribute) => {
|
|
74
|
-
const value = el.getAttribute(attribute);
|
|
75
|
-
if (value) {
|
|
76
|
-
if (attribute === 'style') {
|
|
77
|
-
const styleString = el.getAttribute(attribute);
|
|
78
|
-
const reactStyleObject = reactAttributes(parseStyleString(styleString));
|
|
79
|
-
acc['style'] = reactStyleObject;
|
|
80
|
-
} else {
|
|
81
|
-
acc[attribute] = el.getAttribute(attribute);
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
return acc;
|
|
85
|
-
};
|
|
86
|
-
|
|
87
|
-
const attributes = ['border', 'cellpadding', 'cellspacing', 'class', 'style'];
|
|
88
|
-
|
|
89
|
-
/**
|
|
90
|
-
* Serializer rules.
|
|
91
|
-
*
|
|
92
|
-
* @type {Array}
|
|
93
|
-
*/
|
|
94
|
-
|
|
95
|
-
const blocks = {
|
|
96
|
-
deserialize(el, next) {
|
|
97
|
-
log('[blocks:deserialize] block: ', el);
|
|
98
|
-
const block = BLOCK_TAGS[el.tagName.toLowerCase()];
|
|
99
|
-
if (!block) return;
|
|
100
|
-
log('[blocks:deserialize] block: ', block);
|
|
101
|
-
|
|
102
|
-
if (el.childNodes.length === 1) {
|
|
103
|
-
const cn = el.childNodes[0];
|
|
104
|
-
if (cn && cn.tagName && cn.tagName.toLowerCase() === block) {
|
|
105
|
-
log('[we have a child node of the same]...');
|
|
106
|
-
return;
|
|
107
|
-
}
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
return jsx(
|
|
111
|
-
'element',
|
|
112
|
-
{
|
|
113
|
-
type: block,
|
|
114
|
-
/**
|
|
115
|
-
* Here for rendering styles for all block elements
|
|
116
|
-
*/
|
|
117
|
-
data: { attributes: attributes.reduce(attributesToMap(el), {}) }
|
|
118
|
-
},
|
|
119
|
-
next(el.childNodes)
|
|
120
|
-
);
|
|
121
|
-
},
|
|
122
|
-
serialize: (object, children) => {
|
|
123
|
-
if (object.object !== 'block') return;
|
|
124
|
-
|
|
125
|
-
const jsonData = object.data.toJSON();
|
|
126
|
-
|
|
127
|
-
log('[blocks:serialize] object: ', object, children);
|
|
128
|
-
let key;
|
|
129
|
-
|
|
130
|
-
for (key in BLOCK_TAGS) {
|
|
131
|
-
if (BLOCK_TAGS[key] === object.type) {
|
|
132
|
-
const Tag = key;
|
|
133
|
-
|
|
134
|
-
return <Tag {...jsonData.attributes}>{children}</Tag>;
|
|
135
|
-
}
|
|
136
|
-
}
|
|
137
|
-
}
|
|
138
|
-
};
|
|
139
|
-
|
|
140
|
-
const marks = {
|
|
141
|
-
deserialize(el, next) {
|
|
142
|
-
const mark = MARK_TAGS[el.tagName.toLowerCase()];
|
|
143
|
-
if (!mark) {
|
|
144
|
-
return;
|
|
145
|
-
}
|
|
146
|
-
log('[deserialize] mark: ', mark);
|
|
147
|
-
|
|
148
|
-
return jsx('element', { type: mark }, next(el.childNodes));
|
|
149
|
-
},
|
|
150
|
-
serialize(object, children) {
|
|
151
|
-
/*if (Mark.isMark(object)) {
|
|
152
|
-
for (var key in MARK_TAGS) {
|
|
153
|
-
if (MARK_TAGS[key] === object.type) {
|
|
154
|
-
const Tag = key;
|
|
155
|
-
return <Tag>{children}</Tag>;
|
|
156
|
-
}
|
|
157
|
-
}
|
|
158
|
-
}*/
|
|
159
|
-
}
|
|
160
|
-
};
|
|
161
|
-
|
|
162
|
-
const findPreviousText = el => {
|
|
163
|
-
if (el.nodeName === '#text') {
|
|
164
|
-
return el;
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
if (el.previousSibling) {
|
|
168
|
-
return findPreviousText(el.previousSibling);
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
return null;
|
|
172
|
-
};
|
|
173
|
-
|
|
174
|
-
export const TEXT_RULE = {
|
|
175
|
-
deserialize(el) {
|
|
176
|
-
/**
|
|
177
|
-
* This needs to be called on the dom element in order to merge the adjacent text nodes together
|
|
178
|
-
* */
|
|
179
|
-
el.normalize();
|
|
180
|
-
|
|
181
|
-
if (el.tagName && el.tagName.toLowerCase() === 'br') {
|
|
182
|
-
return jsx('text', {});
|
|
183
|
-
}
|
|
184
|
-
|
|
185
|
-
if (el.nodeName === '#text') {
|
|
186
|
-
if (el.nodeValue && el.nodeValue.match(/<!--.*?-->/)) return;
|
|
187
|
-
|
|
188
|
-
log('[text:deserialize] return text object..');
|
|
189
|
-
return jsx('text', {}, el.nodeValue);
|
|
190
|
-
}
|
|
191
|
-
},
|
|
192
|
-
|
|
193
|
-
serialize(obj, children) {
|
|
194
|
-
if (obj.object === 'string') {
|
|
195
|
-
return children.split('\n').reduce((array, text, i) => {
|
|
196
|
-
if (i !== 0) array.push(<br />);
|
|
197
|
-
array.push(text);
|
|
198
|
-
return array;
|
|
199
|
-
}, []);
|
|
200
|
-
}
|
|
201
|
-
}
|
|
202
|
-
};
|
|
203
|
-
|
|
204
|
-
const RULES = [
|
|
205
|
-
listSerialization,
|
|
206
|
-
mathSerialization,
|
|
207
|
-
mediaSerialization,
|
|
208
|
-
imgSerialization,
|
|
209
|
-
tableSerialization,
|
|
210
|
-
responseAreaSerialization,
|
|
211
|
-
TEXT_RULE,
|
|
212
|
-
blocks,
|
|
213
|
-
marks
|
|
214
|
-
];
|
|
215
|
-
|
|
216
|
-
function allWhitespace(node) {
|
|
217
|
-
// Use ECMA-262 Edition 3 String and RegExp features
|
|
218
|
-
return !/[^\t\n\r ]/.test(node.textContent);
|
|
219
|
-
}
|
|
220
|
-
|
|
221
|
-
function defaultParseHtml(html) {
|
|
222
|
-
if (typeof DOMParser === 'undefined') {
|
|
223
|
-
throw new Error(
|
|
224
|
-
'The native `DOMParser` global which the `Html` serializer uses by default is not present in this environment. You must supply the `options.parseHtml` function instead.'
|
|
225
|
-
);
|
|
226
|
-
}
|
|
227
|
-
|
|
228
|
-
const parsed = new DOMParser().parseFromString(html, 'text/html');
|
|
229
|
-
|
|
230
|
-
const { body } = parsed;
|
|
231
|
-
const textNodes = document.createTreeWalker(body, NodeFilter.SHOW_TEXT, null, null);
|
|
232
|
-
let n = textNodes.nextNode();
|
|
233
|
-
|
|
234
|
-
while (n) {
|
|
235
|
-
if (allWhitespace(n) || n.nodeValue === '\u200B') {
|
|
236
|
-
n.parentNode.removeChild(n);
|
|
237
|
-
}
|
|
238
|
-
n = textNodes.nextNode();
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
return body;
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
/** If this lib is used on the server side, we need to bypass using the DOMParser - just put in a stub. */
|
|
245
|
-
const parseHtml =
|
|
246
|
-
typeof window === 'undefined'
|
|
247
|
-
? () => ({
|
|
248
|
-
childNodes: []
|
|
249
|
-
})
|
|
250
|
-
: defaultParseHtml;
|
|
251
|
-
|
|
252
|
-
const serializer = new TestSerializer({
|
|
253
|
-
defaultBlock: 'div',
|
|
254
|
-
rules: RULES,
|
|
255
|
-
parseHtml
|
|
256
|
-
});
|
|
257
|
-
|
|
258
|
-
const _extends =
|
|
259
|
-
Object.assign ||
|
|
260
|
-
function(target) {
|
|
261
|
-
for (var i = 1; i < arguments.length; i++) {
|
|
262
|
-
var source = arguments[i];
|
|
263
|
-
|
|
264
|
-
for (var key in source) {
|
|
265
|
-
if (Object.prototype.hasOwnProperty.call(source, key)) {
|
|
266
|
-
target[key] = source[key];
|
|
267
|
-
}
|
|
268
|
-
}
|
|
269
|
-
}
|
|
270
|
-
|
|
271
|
-
return target;
|
|
272
|
-
};
|
|
273
|
-
|
|
274
|
-
export const htmlToValue = html => {
|
|
275
|
-
try {
|
|
276
|
-
return serializer.deserialize(html);
|
|
277
|
-
} catch (e) {
|
|
278
|
-
console.log("Couldn't parse html: ", e);
|
|
279
|
-
return {};
|
|
280
|
-
}
|
|
281
|
-
};
|
|
282
|
-
|
|
283
|
-
export const valueToHtml = value => serializer.serialize(value);
|
|
284
|
-
|
|
285
|
-
/**
|
|
286
|
-
*
|
|
287
|
-
* <div><div>a</div></div> -> <div>a</div>
|
|
288
|
-
*
|
|
289
|
-
* <div><div>a</div><div>b</div></div> -> <div>a</div><div>b</div>
|
|
290
|
-
* <div><div>a</div>4444<div>b</div></div> -> <div>a</div>4444<div>b</div>
|
|
291
|
-
*/
|
package/src/parse-html.js
DELETED
package/src/test-serializer.js
DELETED
|
@@ -1,163 +0,0 @@
|
|
|
1
|
-
import React from 'react';
|
|
2
|
-
import ReactServer from 'react-dom/server';
|
|
3
|
-
import escapeHtml from 'escape-html';
|
|
4
|
-
import { Text } from 'slate';
|
|
5
|
-
import { jsx } from 'slate-hyperscript';
|
|
6
|
-
|
|
7
|
-
function allWhitespace(node) {
|
|
8
|
-
// Use ECMA-262 Edition 3 String and RegExp features
|
|
9
|
-
return !/[^\t\n\r ]/.test(node.textContent);
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
function defaultParseHtml(html) {
|
|
13
|
-
if (typeof DOMParser === 'undefined') {
|
|
14
|
-
throw new Error(
|
|
15
|
-
'The native `DOMParser` global which the `Html` serializer uses by default is not present in this environment. You must supply the `options.parseHtml` function instead.'
|
|
16
|
-
);
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
const parsed = new DOMParser().parseFromString(html, 'text/html');
|
|
20
|
-
|
|
21
|
-
const { body } = parsed;
|
|
22
|
-
const textNodes = document.createTreeWalker(body, NodeFilter.SHOW_TEXT, null, null);
|
|
23
|
-
let n = textNodes.nextNode();
|
|
24
|
-
|
|
25
|
-
while (n) {
|
|
26
|
-
if (allWhitespace(n) || n.nodeValue === '\u200B') {
|
|
27
|
-
n.parentNode.removeChild(n);
|
|
28
|
-
}
|
|
29
|
-
n = textNodes.nextNode();
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
return body;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
class Html {
|
|
36
|
-
constructor(props) {
|
|
37
|
-
this.defaultBlock = props.defaultBlock;
|
|
38
|
-
this.parseHtml = defaultParseHtml;
|
|
39
|
-
this.rules = props.rules;
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
serializeEls = node => {
|
|
43
|
-
if (Text.isText(node)) {
|
|
44
|
-
let string = escapeHtml(node.text);
|
|
45
|
-
if (node.bold) {
|
|
46
|
-
string = <strong>{string}</strong>;
|
|
47
|
-
}
|
|
48
|
-
return string;
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
let children = (node.children || []).map(n => this.serializeEls(n));
|
|
52
|
-
|
|
53
|
-
const correctRule = this.rules.reduce((res, rule) => {
|
|
54
|
-
return res || rule.serialize(node, children);
|
|
55
|
-
}, null);
|
|
56
|
-
|
|
57
|
-
if (correctRule) {
|
|
58
|
-
return correctRule;
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
switch (node.type) {
|
|
62
|
-
case 'quote':
|
|
63
|
-
return (
|
|
64
|
-
<blockquote>
|
|
65
|
-
<p>{children}</p>
|
|
66
|
-
</blockquote>
|
|
67
|
-
);
|
|
68
|
-
case 'paragraph':
|
|
69
|
-
return <p>{children}</p>;
|
|
70
|
-
case 'link':
|
|
71
|
-
return <a href={escapeHtml(node.url)}>{children}</a>;
|
|
72
|
-
default:
|
|
73
|
-
return children;
|
|
74
|
-
}
|
|
75
|
-
};
|
|
76
|
-
|
|
77
|
-
serialize = node => {
|
|
78
|
-
const deserialized = this.serializeEls(node);
|
|
79
|
-
const html = ReactServer.renderToStaticMarkup(React.createElement('body', null, deserialized));
|
|
80
|
-
const inner = html.slice(6, -7);
|
|
81
|
-
return inner;
|
|
82
|
-
};
|
|
83
|
-
|
|
84
|
-
deserialize = html => {
|
|
85
|
-
let body = this.parseHtml(html);
|
|
86
|
-
|
|
87
|
-
if (body.firstChild && body.firstChild.nodeType === Node.TEXT_NODE) {
|
|
88
|
-
body = this.parseHtml(`<p>${html}</p>`);
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
return this.deserializeEls(body);
|
|
92
|
-
};
|
|
93
|
-
|
|
94
|
-
deserializeEls = (element, markAttributes = {}) => {
|
|
95
|
-
if (element.nodeType === Node.TEXT_NODE) {
|
|
96
|
-
return jsx('text', markAttributes, element.textContent);
|
|
97
|
-
} else if (element.nodeType !== Node.ELEMENT_NODE) {
|
|
98
|
-
return null;
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
const nodeAttributes = { ...markAttributes };
|
|
102
|
-
|
|
103
|
-
// define attributes for text nodes
|
|
104
|
-
if (element.nodeName === 'STRONG') {
|
|
105
|
-
nodeAttributes.bold = true;
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
const nextFn = nodes => {
|
|
109
|
-
const childNodes = Array.from(nodes);
|
|
110
|
-
const children = Array.from(childNodes)
|
|
111
|
-
.map(node => this.deserializeEls(node, nodeAttributes))
|
|
112
|
-
.flat();
|
|
113
|
-
|
|
114
|
-
if (children.length === 0) {
|
|
115
|
-
children.push(jsx('text', nodeAttributes, ''));
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
return children;
|
|
119
|
-
};
|
|
120
|
-
|
|
121
|
-
const correctRule = this.rules.reduce((res, rule) => {
|
|
122
|
-
return res || rule.deserialize(element, nextFn);
|
|
123
|
-
}, null);
|
|
124
|
-
|
|
125
|
-
if (correctRule) {
|
|
126
|
-
return correctRule;
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
const childNodes = Array.from(element.childNodes);
|
|
130
|
-
const children = Array.from(childNodes)
|
|
131
|
-
.map(node => this.deserializeEls(node, nodeAttributes))
|
|
132
|
-
.flat();
|
|
133
|
-
|
|
134
|
-
if (children.length === 0) {
|
|
135
|
-
children.push(jsx('text', nodeAttributes, ''));
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
switch (element.nodeName) {
|
|
139
|
-
case 'TABLE':
|
|
140
|
-
return jsx('element', { type: 'table' }, children);
|
|
141
|
-
case 'TBODY':
|
|
142
|
-
return jsx('element', { type: 'tbody' }, children);
|
|
143
|
-
case 'TR':
|
|
144
|
-
return jsx('element', { type: 'tr' }, children);
|
|
145
|
-
case 'TD':
|
|
146
|
-
return jsx('element', { type: 'td' }, children);
|
|
147
|
-
case 'BODY':
|
|
148
|
-
return jsx('fragment', {}, children);
|
|
149
|
-
case 'BR':
|
|
150
|
-
return '\n';
|
|
151
|
-
case 'BLOCKQUOTE':
|
|
152
|
-
return jsx('element', { type: 'quote' }, children);
|
|
153
|
-
case 'P':
|
|
154
|
-
return jsx('element', { type: 'paragraph' }, children);
|
|
155
|
-
case 'A':
|
|
156
|
-
return jsx('element', { type: 'link', url: element.getAttribute('href') }, children);
|
|
157
|
-
default:
|
|
158
|
-
return children;
|
|
159
|
-
}
|
|
160
|
-
};
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
export default Html;
|