-
Notifications
You must be signed in to change notification settings - Fork 3k
/
index.js
146 lines (125 loc) · 5.53 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
import render from 'dom-serializer';
import {parseDocument} from 'htmlparser2';
import _ from 'underscore';
import Str from 'expensify-common/lib/str';
const elementsWillBeSkipped = ['html', 'body'];
const tagAttribute = 'data-testid';
/**
* Reads html of selection. If browser doesn't support Selection API, returns empty string.
* @returns {String} HTML of selection as String
*/
const getHTMLOfSelection = () => {
// If browser doesn't support Selection API, return an empty string.
if (!window.getSelection) {
return '';
}
const selection = window.getSelection();
if (selection.rangeCount <= 0) {
return window.getSelection().toString();
}
const div = document.createElement('div');
// HTML tag of markdown comments is in data-testid attribute (em, strong, blockquote..). Our goal here is to
// find that nodes and replace that tag with the one inside data-testid, so ExpensiMark can parse it.
// Simply, we want to replace this:
// <span class="..." style="..." data-testid="strong">bold</span>
// to this:
// <strong>bold</strong>
//
// We traverse all ranges, and get closest node with data-testid and replace its contents with contents of
// range.
for (let i = 0; i < selection.rangeCount; i++) {
const range = selection.getRangeAt(i).cloneRange();
while (range.endOffset === 0) {
range.setEndBefore(range.endContainer);
}
const clonedSelection = range.cloneContents();
// If clonedSelection has no text content this data has no meaning to us.
if (clonedSelection.textContent) {
let node = null;
// If selection starts and ends within same text node we use its parentNode. This is because we can't
// use closest function on a [Text](https://developer.mozilla.org/en-US/docs/Web/API/Text) node.
// We are selecting closest node because nodes with data-testid can be one of the parents of the actual node.
// Assuming we selected only "block" part of following html:
// <div className="..." style="..." data-testid="pre">
// <div dir="auto" class="..." style="...">
// this is block code
// </div>
// </div>
// commonAncestorContainer: #text "this is block code"
// commonAncestorContainer.parentNode:
// <div dir="auto" class="..." style="...">
// this is block code
// </div>
// and finally commonAncestorContainer.parentNode.closest('data-testid') is targeted dom.
if (range.commonAncestorContainer instanceof HTMLElement) {
node = range.commonAncestorContainer.closest(`[${tagAttribute}]`);
} else {
node = range.commonAncestorContainer.parentNode.closest(`[${tagAttribute}]`);
}
// This means "range.commonAncestorContainer" is a text node. We simply get its parent node.
if (!node) {
node = range.commonAncestorContainer.parentNode;
}
node = node.cloneNode();
node.appendChild(clonedSelection);
div.appendChild(node);
}
}
return div.innerHTML;
};
/**
* Clears all attributes from dom elements
* @param {Object} dom htmlparser2 dom representation
* @returns {Object} htmlparser2 dom representation
*/
const replaceNodes = (dom) => {
let domName = dom.name;
let domChildren;
const domAttribs = {};
let data;
// Encoding HTML chars '< >' in the text, because any HTML will be removed in stripHTML method.
if (dom.type === 'text') {
data = Str.htmlEncode(dom.data);
}
// We are skipping elements which has html and body in data-testid, since ExpensiMark can't parse it. Also this data
// has no meaning for us.
if (dom.attribs && dom.attribs[tagAttribute]) {
if (!elementsWillBeSkipped.includes(dom.attribs[tagAttribute])) {
domName = dom.attribs[tagAttribute];
}
} else if (dom.name === 'div' && dom.children.length === 1 && dom.children[0].type !== 'text') {
// We are excluding divs that have only one child and no text nodes and don't have a tagAttribute to prevent
// additional newlines from being added in the HTML to Markdown conversion process.
return replaceNodes(dom.children[0]);
}
// We need to preserve href attribute in order to copy links.
if (dom.attribs && dom.attribs.href) {
domAttribs.href = dom.attribs.href;
}
if (dom.children) {
domChildren = _.map(dom.children, c => replaceNodes(c));
}
return {
...dom,
data,
name: domName,
attribs: domAttribs,
children: domChildren,
};
};
/**
* Resolves the current selection to values and produces clean HTML.
* @returns {String} resolved selection in the HTML format
*/
const getCurrentSelection = () => {
const domRepresentation = parseDocument(getHTMLOfSelection());
domRepresentation.children = _.map(domRepresentation.children, replaceNodes);
// Newline characters need to be removed here because the HTML could contain both newlines and <br> tags, and when
// <br> tags are converted later to markdown, it creates duplicate newline characters. This means that when the content
// is pasted, there are extra newlines in the content that we want to avoid.
const newHtml = render(domRepresentation).replace(/<br>\n/g, '<br>');
return newHtml || '';
};
export default {
getCurrentSelection,
};