cherry-muse 1.0.1

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.
Files changed (226) hide show
  1. package/LICENSE +162 -0
  2. package/README.md +139 -0
  3. package/dist/addons/cherry-code-block-card-plugin.js +1 -0
  4. package/dist/addons/cherry-code-block-echarts-plugin.js +1 -0
  5. package/dist/addons/cherry-code-block-mermaid-plugin.js +1 -0
  6. package/dist/cherry-markdown.core.common.js +1 -0
  7. package/dist/cherry-markdown.core.js +1 -0
  8. package/dist/cherry-markdown.css +4089 -0
  9. package/dist/cherry-markdown.engine.core.common.js +1 -0
  10. package/dist/cherry-markdown.engine.core.esm.js +1 -0
  11. package/dist/cherry-markdown.engine.core.js +1 -0
  12. package/dist/cherry-markdown.esm.js +1 -0
  13. package/dist/cherry-markdown.js +70837 -0
  14. package/dist/cherry-markdown.js.map +1 -0
  15. package/dist/cherry-markdown.min.css +1 -0
  16. package/dist/cherry-markdown.min.js +1 -0
  17. package/dist/stats.html +4838 -0
  18. package/examples/scripts/pinyin/README.md +53 -0
  19. package/package.json +167 -0
  20. package/src/Cherry.config.js +411 -0
  21. package/src/Cherry.js +788 -0
  22. package/src/CherryStatic.js +70 -0
  23. package/src/Editor.js +746 -0
  24. package/src/Engine.js +334 -0
  25. package/src/Event.js +74 -0
  26. package/src/Factory.js +180 -0
  27. package/src/Logger.js +31 -0
  28. package/src/Previewer.js +1147 -0
  29. package/src/Sanitizer.js +4 -0
  30. package/src/Sanitizer.node.js +7 -0
  31. package/src/Stats.js +101 -0
  32. package/src/Theme.js +46 -0
  33. package/src/UrlCache.js +98 -0
  34. package/src/addons/cherry-code-block-card-plugin.js +213 -0
  35. package/src/addons/cherry-code-block-echarts-plugin.js +161 -0
  36. package/src/addons/cherry-code-block-mermaid-plugin.js +118 -0
  37. package/src/core/HookCenter.js +303 -0
  38. package/src/core/HooksConfig.js +106 -0
  39. package/src/core/ParagraphBase.js +314 -0
  40. package/src/core/SentenceBase.js +65 -0
  41. package/src/core/SyntaxBase.js +197 -0
  42. package/src/core/hooks/AutoLink.js +251 -0
  43. package/src/core/hooks/BackgroundColor.js +46 -0
  44. package/src/core/hooks/Badge.js +100 -0
  45. package/src/core/hooks/Blockquote.js +113 -0
  46. package/src/core/hooks/Br.js +85 -0
  47. package/src/core/hooks/CodeBlock.js +876 -0
  48. package/src/core/hooks/Color.js +78 -0
  49. package/src/core/hooks/CommentReference.js +96 -0
  50. package/src/core/hooks/Detail.js +138 -0
  51. package/src/core/hooks/Emoji.config.js +9388 -0
  52. package/src/core/hooks/Emoji.js +223 -0
  53. package/src/core/hooks/Emphasis.js +113 -0
  54. package/src/core/hooks/Footnote.js +125 -0
  55. package/src/core/hooks/FrontMatter.js +52 -0
  56. package/src/core/hooks/FrontMatterVars.js +82 -0
  57. package/src/core/hooks/Header.js +229 -0
  58. package/src/core/hooks/HighLight.js +37 -0
  59. package/src/core/hooks/Hr.js +52 -0
  60. package/src/core/hooks/HtmlBlock.js +159 -0
  61. package/src/core/hooks/Iframe.js +80 -0
  62. package/src/core/hooks/Image.js +276 -0
  63. package/src/core/hooks/InlineCode.js +45 -0
  64. package/src/core/hooks/InlineMath.js +142 -0
  65. package/src/core/hooks/Link.js +169 -0
  66. package/src/core/hooks/List.js +260 -0
  67. package/src/core/hooks/Mark.js +55 -0
  68. package/src/core/hooks/MathBlock.js +97 -0
  69. package/src/core/hooks/Panel.js +170 -0
  70. package/src/core/hooks/Paragraph.js +84 -0
  71. package/src/core/hooks/Ruby.js +34 -0
  72. package/src/core/hooks/Size.js +84 -0
  73. package/src/core/hooks/Strikethrough.js +54 -0
  74. package/src/core/hooks/Sub.js +47 -0
  75. package/src/core/hooks/SuggestList.js +317 -0
  76. package/src/core/hooks/Suggester.js +759 -0
  77. package/src/core/hooks/Sup.js +47 -0
  78. package/src/core/hooks/Table.js +315 -0
  79. package/src/core/hooks/Toc.js +290 -0
  80. package/src/core/hooks/Transfer.js +47 -0
  81. package/src/core/hooks/Underline.js +37 -0
  82. package/src/index.core.js +29 -0
  83. package/src/index.engine.core.js +62 -0
  84. package/src/index.engine.js +30 -0
  85. package/src/index.js +28 -0
  86. package/src/locales/index.js +21 -0
  87. package/src/locales/zh_CN.js +170 -0
  88. package/src/sass/cherry.scss +122 -0
  89. package/src/sass/components/bubble.scss +122 -0
  90. package/src/sass/components/codemirror.scss +628 -0
  91. package/src/sass/components/dropdown.scss +37 -0
  92. package/src/sass/components/editor.scss +78 -0
  93. package/src/sass/components/preview.scss +71 -0
  94. package/src/sass/components/prism.scss +142 -0
  95. package/src/sass/components/stats.scss +32 -0
  96. package/src/sass/components/toc.scss +184 -0
  97. package/src/sass/components/toolbar.scss +117 -0
  98. package/src/sass/core/AutoLink.scss +20 -0
  99. package/src/sass/core/BackgroundColor.scss +0 -0
  100. package/src/sass/core/Badge.scss +116 -0
  101. package/src/sass/core/Blockquote.scss +12 -0
  102. package/src/sass/core/Br.scss +0 -0
  103. package/src/sass/core/Card.scss +219 -0
  104. package/src/sass/core/CodeBlock.scss +205 -0
  105. package/src/sass/core/Color.scss +37 -0
  106. package/src/sass/core/CommentReference.scss +0 -0
  107. package/src/sass/core/Detail.scss +107 -0
  108. package/src/sass/core/Emoji.scss +127 -0
  109. package/src/sass/core/Emphasis.scss +9 -0
  110. package/src/sass/core/Footnote.scss +21 -0
  111. package/src/sass/core/FrontMatterVars.scss +19 -0
  112. package/src/sass/core/Header.scss +103 -0
  113. package/src/sass/core/HighLight.scss +0 -0
  114. package/src/sass/core/Hr.scss +10 -0
  115. package/src/sass/core/HtmlBlock.scss +0 -0
  116. package/src/sass/core/Iframe.scss +36 -0
  117. package/src/sass/core/Image.scss +59 -0
  118. package/src/sass/core/InlineCode.scss +10 -0
  119. package/src/sass/core/InlineMath.scss +11 -0
  120. package/src/sass/core/Link.scss +16 -0
  121. package/src/sass/core/List.scss +61 -0
  122. package/src/sass/core/Mark.scss +15 -0
  123. package/src/sass/core/MathBlock.scss +0 -0
  124. package/src/sass/core/Panel.scss +150 -0
  125. package/src/sass/core/Paragraph.scss +6 -0
  126. package/src/sass/core/Ruby.scss +0 -0
  127. package/src/sass/core/Size.scss +8 -0
  128. package/src/sass/core/Strikethrough.scss +0 -0
  129. package/src/sass/core/Sub.scss +5 -0
  130. package/src/sass/core/Suggester.scss +62 -0
  131. package/src/sass/core/Sup.scss +5 -0
  132. package/src/sass/core/Table.scss +127 -0
  133. package/src/sass/core/Toc.scss +28 -0
  134. package/src/sass/core/Transfer.scss +0 -0
  135. package/src/sass/core/Underline.scss +0 -0
  136. package/src/sass/google-fonts.scss +34 -0
  137. package/src/sass/index.scss +3 -0
  138. package/src/sass/prism/dark.scss +131 -0
  139. package/src/sass/prism/light.scss +143 -0
  140. package/src/sass/variables/colors.scss +96 -0
  141. package/src/toolbars/Bubble.js +232 -0
  142. package/src/toolbars/BubbleTable.js +147 -0
  143. package/src/toolbars/HookCenter.js +185 -0
  144. package/src/toolbars/MenuBase.js +357 -0
  145. package/src/toolbars/PreviewerBubble.js +558 -0
  146. package/src/toolbars/Toc.js +246 -0
  147. package/src/toolbars/Toolbar.js +401 -0
  148. package/src/toolbars/hooks/Audio.js +53 -0
  149. package/src/toolbars/hooks/Badge.js +80 -0
  150. package/src/toolbars/hooks/BarTable.js +41 -0
  151. package/src/toolbars/hooks/Bold.js +70 -0
  152. package/src/toolbars/hooks/Br.js +34 -0
  153. package/src/toolbars/hooks/Card.js +64 -0
  154. package/src/toolbars/hooks/CheckList.js +41 -0
  155. package/src/toolbars/hooks/Code.js +46 -0
  156. package/src/toolbars/hooks/Color.js +285 -0
  157. package/src/toolbars/hooks/Copy.js +139 -0
  158. package/src/toolbars/hooks/Detail.js +70 -0
  159. package/src/toolbars/hooks/ECharts.js +303 -0
  160. package/src/toolbars/hooks/Emoji.js +303 -0
  161. package/src/toolbars/hooks/Export.js +47 -0
  162. package/src/toolbars/hooks/File.js +54 -0
  163. package/src/toolbars/hooks/Formula.js +36 -0
  164. package/src/toolbars/hooks/FullScreen.js +55 -0
  165. package/src/toolbars/hooks/Graph.js +281 -0
  166. package/src/toolbars/hooks/H1.js +71 -0
  167. package/src/toolbars/hooks/H2.js +71 -0
  168. package/src/toolbars/hooks/H3.js +71 -0
  169. package/src/toolbars/hooks/Header.js +100 -0
  170. package/src/toolbars/hooks/Hr.js +35 -0
  171. package/src/toolbars/hooks/Iframe.js +35 -0
  172. package/src/toolbars/hooks/Image.js +60 -0
  173. package/src/toolbars/hooks/Insert.js +36 -0
  174. package/src/toolbars/hooks/Italic.js +70 -0
  175. package/src/toolbars/hooks/LineTable.js +41 -0
  176. package/src/toolbars/hooks/Link.js +46 -0
  177. package/src/toolbars/hooks/List.js +55 -0
  178. package/src/toolbars/hooks/Ol.js +41 -0
  179. package/src/toolbars/hooks/Panel.js +155 -0
  180. package/src/toolbars/hooks/Quote.js +45 -0
  181. package/src/toolbars/hooks/Redo.js +33 -0
  182. package/src/toolbars/hooks/Ruby.js +59 -0
  183. package/src/toolbars/hooks/Size.js +100 -0
  184. package/src/toolbars/hooks/Split.js +37 -0
  185. package/src/toolbars/hooks/Strikethrough.js +65 -0
  186. package/src/toolbars/hooks/Sub.js +58 -0
  187. package/src/toolbars/hooks/Sup.js +58 -0
  188. package/src/toolbars/hooks/SwitchModel.js +78 -0
  189. package/src/toolbars/hooks/Table.js +56 -0
  190. package/src/toolbars/hooks/Toc.js +35 -0
  191. package/src/toolbars/hooks/TogglePreview.js +79 -0
  192. package/src/toolbars/hooks/Ul.js +41 -0
  193. package/src/toolbars/hooks/Underline.js +65 -0
  194. package/src/toolbars/hooks/Undo.js +30 -0
  195. package/src/toolbars/hooks/Video.js +53 -0
  196. package/src/utils/LazyLoadImg.js +341 -0
  197. package/src/utils/autoindent.js +58 -0
  198. package/src/utils/codeBlockContentHandler.js +351 -0
  199. package/src/utils/config.js +98 -0
  200. package/src/utils/copy.js +55 -0
  201. package/src/utils/dialog.js +196 -0
  202. package/src/utils/dom.js +162 -0
  203. package/src/utils/downloadUtil.js +23 -0
  204. package/src/utils/env.js +22 -0
  205. package/src/utils/error.js +61 -0
  206. package/src/utils/event.js +38 -0
  207. package/src/utils/export.js +115 -0
  208. package/src/utils/file.js +121 -0
  209. package/src/utils/formulaUtilsHandler.js +230 -0
  210. package/src/utils/htmlparser.js +977 -0
  211. package/src/utils/image.js +99 -0
  212. package/src/utils/imgSizeHandler.js +279 -0
  213. package/src/utils/jsonUtils.js +17 -0
  214. package/src/utils/lineFeed.js +49 -0
  215. package/src/utils/listContentHandler.js +227 -0
  216. package/src/utils/lookbehind-replace.js +81 -0
  217. package/src/utils/mathjax.js +89 -0
  218. package/src/utils/myersDiff.js +211 -0
  219. package/src/utils/pasteHelper.js +253 -0
  220. package/src/utils/recount-pos.js +59 -0
  221. package/src/utils/regexp.js +295 -0
  222. package/src/utils/sanitize.js +477 -0
  223. package/src/utils/selection.js +50 -0
  224. package/src/utils/svgUtils.js +96 -0
  225. package/src/utils/tableContentHandler.js +592 -0
  226. package/tools/README.md +3 -0
@@ -0,0 +1,30 @@
1
+ /**
2
+ * Copyright (C) 2021 THL A29 Limited, a Tencent company.
3
+ *
4
+ * Licensed under the Apache License, Version 2.0 (the "License");
5
+ * you may not use this file except in compliance with the License.
6
+ * You may obtain a copy of the License at
7
+ *
8
+ * http://www.apache.org/licenses/LICENSE-2.0
9
+ *
10
+ * Unless required by applicable law or agreed to in writing, software
11
+ * distributed under the License is distributed on an "AS IS" BASIS,
12
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ * See the License for the specific language governing permissions and
14
+ * limitations under the License.
15
+ */
16
+ import MenuBase from '@/toolbars/MenuBase';
17
+ /**
18
+ * 撤销回退按钮,点击后触发编辑器的undo操作
19
+ * 依赖codemirror的undo接口
20
+ **/
21
+ export default class Undo extends MenuBase {
22
+ constructor($cherry) {
23
+ super($cherry);
24
+ this.setName('undo', 'undo');
25
+ }
26
+
27
+ onClick() {
28
+ this.editor.editor.undo();
29
+ }
30
+ }
@@ -0,0 +1,53 @@
1
+ /**
2
+ * Copyright (C) 2021 THL A29 Limited, a Tencent company.
3
+ *
4
+ * Licensed under the Apache License, Version 2.0 (the "License");
5
+ * you may not use this file except in compliance with the License.
6
+ * You may obtain a copy of the License at
7
+ *
8
+ * http://www.apache.org/licenses/LICENSE-2.0
9
+ *
10
+ * Unless required by applicable law or agreed to in writing, software
11
+ * distributed under the License is distributed on an "AS IS" BASIS,
12
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ * See the License for the specific language governing permissions and
14
+ * limitations under the License.
15
+ */
16
+ import MenuBase from '@/toolbars/MenuBase';
17
+ import { handleUpload, handleParams } from '@/utils/file';
18
+ /**
19
+ * 插入视频
20
+ */
21
+ export default class Video extends MenuBase {
22
+ constructor($cherry) {
23
+ super($cherry);
24
+ this.setName('video', 'videocam');
25
+ }
26
+
27
+ /**
28
+ * 响应点击事件
29
+ * @param {string} selection 被用户选中的文本内容
30
+ * @returns {string} 回填到编辑器光标位置/选中文本区域的内容
31
+ */
32
+ onClick(selection, shortKey = '') {
33
+ if (this.hasCacheOnce()) {
34
+ // @ts-ignore
35
+ const { name, url, params } = this.getAndCleanCacheOnce();
36
+ const begin = '!video[';
37
+ const end = params.poster ? `](${url}){poster=${params.poster}}` : `](${url})`;
38
+ this.registerAfterClickCb(() => {
39
+ this.setLessSelection(begin, end);
40
+ });
41
+ const finalName = params.name ? params.name : name;
42
+ return `${begin}${finalName}${handleParams(params)}${end}`;
43
+ }
44
+ const accept = this.$cherry.options?.fileTypeLimitMap?.video ?? '*';
45
+ // 插入图片,调用上传文件逻辑
46
+ handleUpload(this.editor, 'video', accept, (name, url, params) => {
47
+ this.setCacheOnce({ name, url, params });
48
+ this.fire(null);
49
+ });
50
+ this.updateMarkdown = false;
51
+ return selection;
52
+ }
53
+ }
@@ -0,0 +1,341 @@
1
+ /**
2
+ * Copyright (C) 2021 Tencent.
3
+ *
4
+ * Licensed under the Apache License, Version 2.0 (the "License");
5
+ * you may not use this file except in compliance with the License.
6
+ * You may obtain a copy of the License at
7
+ *
8
+ * http://www.apache.org/licenses/LICENSE-2.0
9
+ *
10
+ * Unless required by applicable law or agreed to in writing, software
11
+ * distributed under the License is distributed on an "AS IS" BASIS,
12
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ * See the License for the specific language governing permissions and
14
+ * limitations under the License.
15
+ */
16
+
17
+ /**
18
+ * 懒加载媒体资源(图片、视频、音频、iframe)
19
+ *
20
+ * - 只缓存资源的src的原因
21
+ * - 1、因为浏览器的缓存机制,相同的src第二次请求时,浏览器会直接返回缓存的资源
22
+ * - 2、编辑状态时预览区域dom结构不稳定,并不能准确的缓存到元素dom对象
23
+ *
24
+ * - 当浏览器**禁用**了缓存时,本机制效果有限
25
+ * - 依然还是可以实现懒加载的效果
26
+ * - 但是会把资源请求次数翻倍
27
+ */
28
+ export default class LazyLoadImg {
29
+ options = {
30
+ // 加载图片时如果需要展示loading图,则配置loading图的地址
31
+ loadingImgPath: '',
32
+ // 同一时间最多有几个资源请求,最大同时加载6个资源
33
+ maxNumPerTime: 2,
34
+ // 不进行懒加载处理的资源数量,如果为0,即所有资源都进行懒加载处理, 如果设置为-1,则所有资源都不进行懒加载处理
35
+ noLoadImgNum: 5,
36
+ // 首次自动加载几个资源(不论资源是否滚动到视野内),autoLoadImgNum = -1 表示会自动加载完所有资源
37
+ autoLoadImgNum: 5,
38
+ // 针对加载失败的资源 或 beforeLoadOneImgCallback 返回false 的资源,最多尝试加载几次,为了防止死循环,最多5次。以资源的src为纬度统计重试次数
39
+ maxTryTimesPerSrc: 2,
40
+ // 加载一个资源之前的回调函数,函数return false 会终止加载操作
41
+ beforeLoadOneImgCallback: (element) => {},
42
+ // 加载一个资源失败之后的回调函数
43
+ failLoadOneImgCallback: (element) => {},
44
+ // 加载一个资源之后的回调函数,如果资源加载失败,则不会回调该函数
45
+ afterLoadOneImgCallback: (element) => {},
46
+ // 加载完所有资源后调用的回调函数,只表示某一个时刻所有资源都加在完时的回调,如果预览区域又有了新资源,当新资源加载完后还会产生这个回调
47
+ afterLoadAllImgCallback: () => {},
48
+ };
49
+
50
+ constructor(options, previewer) {
51
+ Object.assign(this.options, options);
52
+ this.previewer = previewer;
53
+ // 记录已经加载过的资源src
54
+ this.srcLoadedList = [];
55
+ // 记录加载失败的资源src,key是src,value是失败次数
56
+ this.srcFailLoadedList = {};
57
+ // 记录正在加载的资源src
58
+ this.srcLoadingList = [];
59
+ // 记录所有懒加载的资源src
60
+ this.srcList = [];
61
+ // 记录当前时刻有多少资源正在加载
62
+ this.loadingImgNum = 0;
63
+ // 记录上次加载完所有资源的个数
64
+ this.lastLoadAllNum = 0;
65
+ this.previewerDom = this.previewer.getDomContainer();
66
+ }
67
+
68
+ /**
69
+ * 判断图片的src是否加载过
70
+ * @param {String} src
71
+ * @return {Boolean}
72
+ */
73
+ isLoaded(src) {
74
+ return this.srcLoadedList.includes(src);
75
+ }
76
+
77
+ /**
78
+ * 判断图片是否正在加载
79
+ * @param {String} src
80
+ * @return {Boolean}
81
+ */
82
+ isLoading(src) {
83
+ return this.srcLoadingList.includes(src);
84
+ }
85
+
86
+ /**
87
+ * 加载失败时,把src加入到失败队列中,并记录失败次数
88
+ * @param {*} src
89
+ */
90
+ loadFailed(src) {
91
+ this.srcFailLoadedList[src] = this.srcFailLoadedList[src] ? this.srcFailLoadedList[src] + 1 : 1;
92
+ }
93
+
94
+ /**
95
+ * 判断图片失败次数是否超过最大次数
96
+ * @param {*} src
97
+ * @return {Boolean}
98
+ */
99
+ isFailLoadedMax(src) {
100
+ return this.srcFailLoadedList[src] && this.srcFailLoadedList[src] > this.options.maxTryTimesPerSrc;
101
+ }
102
+
103
+ /**
104
+ * 判断当前时刻所有资源是否都完成过加载
105
+ * 当出现新资源后,完成加载后,当前函数还是会再次触发加载完的回调函数(afterLoadAllImgCallback)
106
+ * 该函数并不是实时返回的,最大有1s的延时
107
+ */
108
+ isLoadedAllDone() {
109
+ const elements = this.previewerDom.querySelectorAll('img[data-src], video[data-src], audio[data-src], iframe[data-src]');
110
+ const allLoadedNum = this.srcLoadedList.length;
111
+ // const dataSrcRemain = allLoadNum - this.srcLoadedList.length;
112
+ if (elements.length <= 0 && this.lastLoadAllNum < allLoadedNum) {
113
+ this.lastLoadAllNum = allLoadedNum;
114
+ this.options.afterLoadAllImgCallback();
115
+ return true;
116
+ }
117
+ return false;
118
+ }
119
+
120
+ /**
121
+ * 当向下滚动时,提前100px加载资源
122
+ * 当向上滚动时,不提前加载资源,一定要资源完全进入可视区域(top > 0)再加载资源,否则当锚点定位时,会由于上面的资源加载出现定位不准的情况
123
+ *
124
+ */
125
+ loadOneImg() {
126
+ const elements = this.previewerDom.querySelectorAll('img[data-src], video[data-src], audio[data-src], iframe[data-src]');
127
+ const { height, top } = this.previewerDom.getBoundingClientRect();
128
+ const previewerHeight = height + top + 100; // 冗余一定高度用于提前加载
129
+ const windowsHeight = window?.innerHeight ?? 0 + 100; // 浏览器的视口高度
130
+ const maxHeight = Math.min(previewerHeight, windowsHeight); // 目标视区高度一定是小于浏览器视口高度的,也一定是小于预览区高度的
131
+ const minHeight = top - 30; // 计算顶部高度时,需要预加载一行高
132
+ const { autoLoadImgNum } = this.options;
133
+ for (let i = 0; i < elements.length; i++) {
134
+ const element = elements[i];
135
+ const position = element.getBoundingClientRect();
136
+ // 判断是否在视区内
137
+ const testPosition = position.top >= minHeight && position.top <= maxHeight;
138
+ // 判断是否需要自动加载
139
+ const testAutoLoad = this.srcList.length < autoLoadImgNum;
140
+ if (!testPosition && !testAutoLoad) {
141
+ continue;
142
+ }
143
+ let originSrc = element.getAttribute('data-src');
144
+ if (!originSrc) {
145
+ continue;
146
+ }
147
+ if (this.isLoaded(originSrc) || this.isFailLoadedMax(originSrc)) {
148
+ // 如果已经加载过相同的资源,或者已经超过失败最大重试次数,则直接加载
149
+ element.setAttribute('src', originSrc);
150
+ element.removeAttribute('data-src');
151
+ }
152
+ // 如果当前src正在加载,则忽略这个src,继续找下个符合条件的src
153
+ if (this.isLoading(originSrc)) {
154
+ continue;
155
+ }
156
+ // 超过最大并发量时停止加载
157
+ if (this.loadingImgNum >= this.options.maxNumPerTime) {
158
+ return false;
159
+ }
160
+ const test = this.options.beforeLoadOneImgCallback(element);
161
+ if (typeof test === 'undefined' || test) {
162
+ originSrc = element.getAttribute('data-src') ?? originSrc;
163
+ } else {
164
+ this.loadFailed(originSrc);
165
+ continue;
166
+ }
167
+ this.loadingImgNum += 1;
168
+ this.srcList.push(originSrc);
169
+ this.srcLoadingList.push(originSrc);
170
+ this.tryLoadOneImg(
171
+ element,
172
+ originSrc,
173
+ () => {
174
+ element.setAttribute('src', originSrc);
175
+ element.removeAttribute('data-src');
176
+ this.srcLoadedList.push(originSrc);
177
+ this.loadingImgNum -= 1;
178
+ this.srcLoadingList.splice(this.srcLoadingList.indexOf(originSrc), 1);
179
+ this.options.afterLoadOneImgCallback(element);
180
+ this.loadOneImg();
181
+ },
182
+ () => {
183
+ this.loadFailed(originSrc);
184
+ this.loadingImgNum -= 1;
185
+ this.srcLoadingList.splice(this.srcLoadingList.indexOf(originSrc), 1);
186
+ this.options.failLoadOneImgCallback(element);
187
+ this.loadOneImg();
188
+ },
189
+ );
190
+ }
191
+ return false;
192
+ }
193
+
194
+ /**
195
+ * 尝试加载资源
196
+ * @param {HTMLElement} element 要加载的元素
197
+ * @param {String} src 资源地址
198
+ * @param {Function} successCallback 成功回调
199
+ * @param {Function} failCallback 失败回调
200
+ */
201
+ tryLoadOneImg(element, src, successCallback, failCallback) {
202
+ const tagName = element.tagName.toLowerCase();
203
+
204
+ // 图片使用预加载策略
205
+ if (tagName === 'img') {
206
+ const img = document.createElement('img');
207
+ img.onload = () => {
208
+ successCallback();
209
+ img.remove();
210
+ };
211
+ img.onerror = () => {
212
+ failCallback();
213
+ img.remove();
214
+ };
215
+ img.setAttribute('src', src);
216
+ return;
217
+ }
218
+
219
+ // video/audio/iframe 不需要预加载,直接让浏览器决定何时加载
220
+ // 直接调用成功回调,让外层设置 src 即可
221
+ successCallback();
222
+ }
223
+
224
+ /**
225
+ * 开始进行懒加载
226
+ *
227
+ * **关于实现方式的思考**
228
+ * 实现媒体资源懒加载一般有三种方式:
229
+ * 1、监听滚动事件,滚动到视野内的资源开始加载
230
+ * 2、定时检测当前视窗内是否有资源需要加载
231
+ * 3、当前一个资源加载完成后,自动加载下一个资源
232
+ *
233
+ * 方式1监听滚动事件的弊端:
234
+ * 1、需要限频率
235
+ * 2、不能实现自动加载所有资源的功能(autoLoadImgNum = -1)
236
+ * 3、如果业务方对预览区域做了个性化加工,有可能导致监听不到滚动事件
237
+ * 4、在自动滚动到锚点的场景,会在页面滚动时加载资源,资源的加载会导致锚点上方的元素高度发生变化,最终导致锚点定位失败
238
+ * (所以在这个场景下,需要特殊处理资源加载的时机,但并不好判断是否锚点引发的滚动)
239
+ * 5、浏览器尺寸发生变化或者浏览器缩放比例发生变化的场景(当然还有横屏竖屏切换、系统分辨率改变等)不好监听和响应
240
+ *
241
+ * 方式2轮询的弊端:
242
+ * 1、需要额外的逻辑来控制并发
243
+ * 2、消耗计算资源,所以需要尽量优化单次计算量,并尽量避免在轮询里进行大范围dom操作
244
+ * 3、两次资源加载中间可能有最大轮询间隔的空闲时间浪费
245
+ *
246
+ * 方式3依次加载的弊端:
247
+ * 1、没办法实现滚动到视野内再加载资源
248
+ *
249
+ * 综合考虑决定用方式2(轮询)+方式3(依次加载)的组合方式,并且每次只做一次dom写操作
250
+ * 轮询带来的性能开销就让受摩尔定律加持的硬件和每月都会更新版本的浏览器们愁去吧
251
+ */
252
+ doLazyLoad() {
253
+ // 防止重复调用
254
+ if (this.isRunning) {
255
+ return;
256
+ }
257
+ this.isRunning = true;
258
+ const { maxNumPerTime } = this.options;
259
+ const polling = () => {
260
+ // 保证至少有一次自动加载
261
+ this.loadOneImg();
262
+ for (let i = 1; i < maxNumPerTime; i++) {
263
+ this.loadOneImg();
264
+ }
265
+ setTimeout(polling, 200);
266
+ };
267
+ polling();
268
+ // setTimeout(polling, 200);
269
+ setInterval(() => {
270
+ this.isLoadedAllDone();
271
+ }, 1000);
272
+ }
273
+
274
+ /**
275
+ * 把媒体资源里的data-src替换为src
276
+ * @param {*} content
277
+ * @returns {String}
278
+ */
279
+ changeDataSrc2Src(content) {
280
+ return content.replace(/<(img|video|audio|iframe) ([^>]*?)data-src="([^"]+)"([^>]*?)>/g, (match, tag, m1, src, m3) => {
281
+ return `<${tag} ${this.$removeSrc(m1)} src="${src}" ${this.$removeSrc(m3)}>`.replace(/ {2,}/g, ' ');
282
+ });
283
+ }
284
+
285
+ /**
286
+ * 把已经加载的媒体资源里的data-src替换为src
287
+ * @param {*} content
288
+ * @returns {String}
289
+ */
290
+ changeLoadedDataSrc2Src(content) {
291
+ return content.replace(/<(img|video|audio|iframe) ([^>]*?)data-src="([^"]+)"([^>]*?)>/g, (match, tag, m1, src, m3) => {
292
+ if (!this.isLoaded(src)) {
293
+ return match;
294
+ }
295
+ return `<${tag} ${this.$removeSrc(m1)} src="${src}" ${this.$removeSrc(m3)}>`.replace(/ {2,}/g, ' ');
296
+ });
297
+ }
298
+
299
+ /**
300
+ * 移除元素的src属性
301
+ * @param {String} element
302
+ * @returns {String}
303
+ */
304
+ $removeSrc(element) {
305
+ return ` ${element}`.replace(/^(.*?) src=".*?"(.*?$)/, '$1$2');
306
+ }
307
+
308
+ /**
309
+ * 把媒体资源里的src替换为data-src,如果src已经加载过,则不替换
310
+ * @param {String} content
311
+ * @param {Boolean} focus 强制替换
312
+ * @returns {String}
313
+ */
314
+ changeSrc2DataSrc(content, focus = false) {
315
+ const { loadingImgPath } = this.options;
316
+ const { noLoadImgNum } = this.options;
317
+ let currentNoLoadImgNum = 0;
318
+ return content.replace(/<(img|video|audio|iframe) ([^>]*?)src="([^"]+)"([^>]*?)>/g, (match, tag, m1, src, m3) => {
319
+ // 如果已经替换过data-src了,或者没有src属性,或者关闭了懒加载功能,则不替换
320
+ if (/data-src="/.test(match) || !/ src="/.test(match) || noLoadImgNum < 0) {
321
+ return match;
322
+ }
323
+ if (focus === false) {
324
+ // 前noLoadImgNum个资源不替换
325
+ if (currentNoLoadImgNum < noLoadImgNum) {
326
+ currentNoLoadImgNum += 1;
327
+ return match;
328
+ }
329
+ // 如果src已经加载过,则不替换
330
+ if (this.isLoaded(src)) {
331
+ return match;
332
+ }
333
+ }
334
+ // 如果配置了loadingImgPath且是img标签,则替换src为loadingImgPath
335
+ if (loadingImgPath && tag === 'img') {
336
+ return `<${tag} ${m1}src="${loadingImgPath}" data-src="${src}"${m3}>`;
337
+ }
338
+ return `<${tag} ${m1}data-src="${src}"${m3}>`;
339
+ });
340
+ }
341
+ }
@@ -0,0 +1,58 @@
1
+ /**
2
+ * Copyright (C) 2021 THL A29 Limited, a Tencent company.
3
+ *
4
+ * Licensed under the Apache License, Version 2.0 (the "License");
5
+ * you may not use this file except in compliance with the License.
6
+ * You may obtain a copy of the License at
7
+ *
8
+ * http://www.apache.org/licenses/LICENSE-2.0
9
+ *
10
+ * Unless required by applicable law or agreed to in writing, software
11
+ * distributed under the License is distributed on an "AS IS" BASIS,
12
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ * See the License for the specific language governing permissions and
14
+ * limitations under the License.
15
+ */
16
+
17
+ /**
18
+ * @param {CodeMirror.Editor} cm
19
+ */
20
+ export function handleNewlineIndentList(cm) {
21
+ if (handleCherryList(cm)) return;
22
+ cm.execCommand('newlineAndIndentContinueMarkdownList');
23
+ }
24
+
25
+ function handleCherryList(cm) {
26
+ const cherryListRE = /^(\s*)([I一二三四五六七八九十]+)\.(\s+)/;
27
+ const cherryListEmptyRE = /^(\s*)([I一二三四五六七八九十]+)\.(\s+)$/;
28
+ if (cm.getOption('disableInput')) return false;
29
+ const ranges = cm.listSelections();
30
+ const replacements = [];
31
+ for (let i = 0; i < ranges.length; i++) {
32
+ const pos = ranges[i].head;
33
+ const line = cm.getLine(pos.line);
34
+ const match = cherryListRE.exec(line);
35
+ const cursorBeforeBullet = /^\s*$/.test(line.slice(0, pos.ch));
36
+ if (!ranges[i].empty() || cursorBeforeBullet || !match) return;
37
+ if (cherryListEmptyRE.test(line)) {
38
+ cm.replaceRange(
39
+ '',
40
+ {
41
+ line: pos.line,
42
+ ch: 0,
43
+ },
44
+ {
45
+ line: pos.line,
46
+ ch: pos.ch + 1,
47
+ },
48
+ );
49
+ replacements[i] = '\n';
50
+ } else {
51
+ const indent = match[1];
52
+ const after = match[3];
53
+ replacements[i] = `\n${indent}I.${after}`;
54
+ }
55
+ }
56
+ cm.replaceSelections(replacements);
57
+ return true;
58
+ }