fusions-ui 1.2.6 → 1.2.8

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 (547) hide show
  1. package/LICENSE +1 -1
  2. package/changelog.md +2 -0
  3. package/components/avatar/index.ts +1 -0
  4. package/components/avatar/src/avatar.ts +131 -0
  5. package/components/avatar/src/avatar.vue +104 -0
  6. package/components/avatar/src/composables/index.ts +1 -0
  7. package/components/avatar/src/composables/use-avatar.ts +61 -0
  8. package/components/avatar-group/index.ts +1 -0
  9. package/components/avatar-group/src/avatar-group.ts +86 -0
  10. package/components/{fu-avatar-group/fu-avatar-group.vue → avatar-group/src/avatar-group.vue} +27 -34
  11. package/components/avatar-group/src/composables/index.ts +1 -0
  12. package/components/avatar-group/src/composables/use-avatar-group.ts +17 -0
  13. package/components/badge/index.ts +1 -0
  14. package/components/badge/src/badge.ts +126 -0
  15. package/components/{fu-badge/fu-badge.vue → badge/src/badge.vue} +26 -62
  16. package/components/badge/src/composables/badge-custom.ts +51 -0
  17. package/components/badge/src/composables/index.ts +1 -0
  18. package/components/banner-arc/index.ts +1 -0
  19. package/components/banner-arc/src/banner-arc.ts +41 -0
  20. package/components/banner-arc/src/banner-arc.vue +49 -0
  21. package/components/banner-arc/src/composables/banner-arc-custom.ts +19 -0
  22. package/components/banner-arc/src/composables/index.ts +1 -0
  23. package/components/button/index.ts +1 -0
  24. package/components/button/src/button.ts +332 -0
  25. package/components/{fu-button/fu-button.vue → button/src/button.vue} +57 -163
  26. package/components/button/src/composables/button-custom.ts +90 -0
  27. package/components/button/src/composables/index.ts +2 -0
  28. package/components/button/src/composables/use-button.ts +137 -0
  29. package/components/cell/index.ts +1 -0
  30. package/components/cell/src/cell.ts +167 -0
  31. package/components/{fu-cell/fu-cell.vue → cell/src/cell.vue} +29 -38
  32. package/components/cell/src/composables/index.ts +1 -0
  33. package/components/cell/src/composables/use-cell.ts +20 -0
  34. package/components/cell-group/index.ts +1 -0
  35. package/components/cell-group/src/cell-group.ts +23 -0
  36. package/components/{fu-cell-group/fu-cell-group.vue → cell-group/src/cell-group.vue} +8 -13
  37. package/components/checkbox/index.ts +1 -0
  38. package/components/checkbox/src/checkbox.ts +144 -0
  39. package/components/checkbox/src/checkbox.vue +270 -0
  40. package/components/checkbox/src/composables/checkbox-custom.ts +15 -0
  41. package/components/checkbox/src/composables/index.ts +2 -0
  42. package/components/checkbox/src/composables/use-checkbox.ts +200 -0
  43. package/components/code-input/index.ts +1 -0
  44. package/components/code-input/src/code-input.ts +157 -0
  45. package/components/code-input/src/code-input.vue +158 -0
  46. package/components/code-input/src/composables/code-input-custom.ts +74 -0
  47. package/components/code-input/src/composables/index.ts +2 -0
  48. package/components/code-input/src/composables/use-code-input.ts +67 -0
  49. package/components/code-verify/index.ts +1 -0
  50. package/components/code-verify/src/code-verify.ts +160 -0
  51. package/components/code-verify/src/code-verify.vue +87 -0
  52. package/components/code-verify/src/composables/code-verify-custom.ts +23 -0
  53. package/components/code-verify/src/composables/index.ts +2 -0
  54. package/components/code-verify/src/composables/use-code-verify.ts +132 -0
  55. package/components/collapse/index.ts +1 -0
  56. package/components/collapse/src/collapse.ts +41 -0
  57. package/components/collapse/src/collapse.vue +48 -0
  58. package/components/collapse/src/composables/index.ts +1 -0
  59. package/components/collapse/src/composables/use-collapse.ts +150 -0
  60. package/components/collapse-item/index.ts +1 -0
  61. package/components/collapse-item/src/collapse-item.ts +102 -0
  62. package/components/{fu-collapse-item/fu-collapse-item.vue → collapse-item/src/collapse-item.vue} +20 -159
  63. package/components/collapse-item/src/composables/index.ts +1 -0
  64. package/components/collapse-item/src/composables/use-collapse-item.ts +139 -0
  65. package/components/countdown/index.ts +1 -0
  66. package/components/countdown/src/composables/index.ts +1 -0
  67. package/components/countdown/src/composables/use-countdown.ts +133 -0
  68. package/components/countdown/src/countdown.ts +63 -0
  69. package/components/countdown/src/countdown.vue +60 -0
  70. package/components/{fu-countdown/util.js → countdown/src/util.ts} +5 -5
  71. package/components/datetime-picker/index.ts +1 -0
  72. package/components/datetime-picker/src/composables/index.ts +1 -0
  73. package/components/datetime-picker/src/composables/use-datetime-picker.ts +325 -0
  74. package/components/datetime-picker/src/datetime-picker.ts +223 -0
  75. package/components/datetime-picker/src/datetime-picker.vue +96 -0
  76. package/components/form/index.ts +1 -0
  77. package/components/form/src/composables/index.ts +1 -0
  78. package/components/form/src/composables/use-form.ts +256 -0
  79. package/components/form/src/form.ts +108 -0
  80. package/components/form/src/form.vue +64 -0
  81. package/components/{fu-form/utils.js → form/src/utils.ts} +49 -42
  82. package/components/{fu-form/validate.js → form/src/validate.ts} +58 -49
  83. package/components/form-item/index.ts +1 -0
  84. package/components/form-item/src/composables/index.ts +1 -0
  85. package/components/form-item/src/composables/use-form-item.ts +292 -0
  86. package/components/form-item/src/form-item.ts +81 -0
  87. package/components/form-item/src/form-item.vue +229 -0
  88. package/components/fusions-ui/src/fusions-ui.vue +8 -0
  89. package/components/grid/index.ts +1 -0
  90. package/components/grid/src/composables/index.ts +1 -0
  91. package/components/grid/src/composables/use-grid.ts +67 -0
  92. package/components/grid/src/grid.ts +54 -0
  93. package/components/grid/src/grid.vue +69 -0
  94. package/components/grid-item/index.ts +1 -0
  95. package/components/grid-item/src/composables/grid-item-custom.ts +16 -0
  96. package/components/grid-item/src/composables/index.ts +2 -0
  97. package/components/grid-item/src/composables/use-grid-item.ts +55 -0
  98. package/components/grid-item/src/grid-item.ts +15 -0
  99. package/components/grid-item/src/grid-item.vue +102 -0
  100. package/components/icon/index.ts +1 -0
  101. package/components/icon/src/composables/icon-custom.ts +41 -0
  102. package/components/icon/src/composables/index.ts +2 -0
  103. package/components/icon/src/composables/use-icon.ts +14 -0
  104. package/components/{fu-icon/icon.css → icon/src/fuicon.css} +584 -584
  105. package/components/icon/src/fuicon.ts +1023 -0
  106. package/components/icon/src/icon.ts +84 -0
  107. package/components/icon/src/icon.vue +92 -0
  108. package/components/image/index.ts +1 -0
  109. package/components/image/src/composables/image-custom.ts +36 -0
  110. package/components/image/src/composables/index.ts +2 -0
  111. package/components/image/src/composables/use-image.ts +135 -0
  112. package/components/image/src/image.ts +185 -0
  113. package/components/image/src/image.vue +152 -0
  114. package/components/index-anchor/index.ts +1 -0
  115. package/components/index-anchor/src/composables/index.ts +1 -0
  116. package/components/index-anchor/src/composables/use-index-anchor.ts +44 -0
  117. package/components/index-anchor/src/index-anchor.ts +45 -0
  118. package/components/index-anchor/src/index-anchor.vue +60 -0
  119. package/components/index-item/index.ts +1 -0
  120. package/components/index-item/src/composables/index.ts +1 -0
  121. package/components/index-item/src/composables/use-index-item.ts +75 -0
  122. package/components/index-item/src/index-item.ts +13 -0
  123. package/components/index-item/src/index-item.vue +32 -0
  124. package/components/index-list/index.ts +1 -0
  125. package/components/index-list/src/composables/index.ts +1 -0
  126. package/components/index-list/src/composables/use-index-list.ts +297 -0
  127. package/components/index-list/src/index-list.ts +52 -0
  128. package/components/index-list/src/index-list.vue +178 -0
  129. package/components/index.ts +62 -0
  130. package/components/input/index.ts +1 -0
  131. package/components/input/src/composables/index.ts +2 -0
  132. package/components/input/src/composables/input-custom.ts +69 -0
  133. package/components/input/src/composables/use-input.ts +117 -0
  134. package/components/input/src/input.ts +360 -0
  135. package/components/input/src/input.vue +241 -0
  136. package/components/keyboard/index.ts +1 -0
  137. package/components/keyboard/src/composables/index.ts +3 -0
  138. package/components/keyboard/src/composables/use-keyboard-car.ts +99 -0
  139. package/components/keyboard/src/composables/use-keyboard-number.ts +81 -0
  140. package/components/keyboard/src/composables/use-keyboard.ts +61 -0
  141. package/components/keyboard/src/keyboard-car.ts +53 -0
  142. package/components/{fu-keyboard → keyboard/src}/keyboard-car.vue +37 -199
  143. package/components/keyboard/src/keyboard-number.ts +44 -0
  144. package/components/keyboard/src/keyboard-number.vue +108 -0
  145. package/components/keyboard/src/keyboard.ts +174 -0
  146. package/components/{fu-keyboard/fu-keyboard.vue → keyboard/src/keyboard.vue} +38 -63
  147. package/components/line/index.ts +1 -0
  148. package/components/line/src/composables/index.ts +1 -0
  149. package/components/line/src/composables/line-custom.ts +30 -0
  150. package/components/line/src/line.ts +82 -0
  151. package/components/line/src/line.vue +44 -0
  152. package/components/link/index.ts +1 -0
  153. package/components/link/src/composables/index.ts +2 -0
  154. package/components/link/src/composables/link-custom.ts +19 -0
  155. package/components/link/src/composables/use-link.ts +33 -0
  156. package/components/link/src/link.ts +61 -0
  157. package/components/link/src/link.vue +62 -0
  158. package/components/loading/index.ts +1 -0
  159. package/{libs/function/colorGradient.js → components/loading/src/composables/colorGradient.ts} +4 -4
  160. package/components/loading/src/composables/index.ts +2 -0
  161. package/components/loading/src/composables/loading-custom.ts +21 -0
  162. package/components/loading/src/composables/use-loading.ts +100 -0
  163. package/components/loading/src/loading.ts +77 -0
  164. package/components/{fu-loading/fu-loading.vue → loading/src/loading.vue} +23 -119
  165. package/components/loading-more/index.ts +1 -0
  166. package/components/loading-more/src/loading-more.ts +75 -0
  167. package/components/{fu-loading-more/fu-loading-more.vue → loading-more/src/loading-more.vue} +18 -19
  168. package/components/modal/index.ts +1 -0
  169. package/components/modal/src/composables/index.ts +2 -0
  170. package/components/modal/src/composables/modal-custom.ts +17 -0
  171. package/components/modal/src/composables/use-modal.ts +56 -0
  172. package/components/modal/src/modal.ts +152 -0
  173. package/components/{fu-modal/fu-modal.vue → modal/src/modal.vue} +41 -69
  174. package/components/navbar/index.ts +1 -0
  175. package/components/navbar/src/composables/index.ts +2 -0
  176. package/components/navbar/src/composables/navbar-custom.ts +62 -0
  177. package/components/navbar/src/composables/use-navbar.ts +42 -0
  178. package/components/navbar/src/navbar.ts +151 -0
  179. package/components/navbar/src/navbar.vue +249 -0
  180. package/components/notice-bar/index.ts +3 -0
  181. package/components/notice-bar/src/composables/index.ts +5 -0
  182. package/components/notice-bar/src/composables/notice-column-custom.ts +43 -0
  183. package/components/notice-bar/src/composables/notice-row-custom.ts +26 -0
  184. package/components/notice-bar/src/composables/use-notice-bar.ts +34 -0
  185. package/components/notice-bar/src/composables/use-notice-column.ts +31 -0
  186. package/components/notice-bar/src/composables/use-notice-row.ts +88 -0
  187. package/components/notice-bar/src/notice-bar.ts +197 -0
  188. package/components/{fu-notice-bar/fu-notice-bar.vue → notice-bar/src/notice-bar.vue} +28 -41
  189. package/components/notice-bar/src/notice-column.ts +172 -0
  190. package/components/notice-bar/src/notice-column.vue +129 -0
  191. package/components/notice-bar/src/notice-row.ts +155 -0
  192. package/components/notice-bar/src/notice-row.vue +135 -0
  193. package/components/number-box/index.ts +1 -0
  194. package/components/number-box/src/composables/index.ts +2 -0
  195. package/components/number-box/src/composables/number-box-custom.ts +16 -0
  196. package/components/number-box/src/composables/use-number-box.ts +84 -0
  197. package/components/number-box/src/number-box.ts +98 -0
  198. package/components/number-box/src/number-box.vue +123 -0
  199. package/components/parse/index.ts +1 -0
  200. package/components/parse/src/app-plus/html/js/handler.js +254 -0
  201. package/components/parse/src/app-plus/html/js/uni.webview.min.js +188 -0
  202. package/components/parse/src/app-plus/html/local.html +32 -0
  203. package/components/parse/src/composables/index.ts +1 -0
  204. package/components/parse/src/composables/use-parse.ts +428 -0
  205. package/components/parse/src/node/node.vue +635 -0
  206. package/components/parse/src/parse.ts +156 -0
  207. package/components/parse/src/parse.vue +131 -0
  208. package/components/parse/src/parser.ts +1402 -0
  209. package/components/picker/index.ts +1 -0
  210. package/components/picker/src/composables/index.ts +2 -0
  211. package/components/picker/src/composables/picker-custom.ts +40 -0
  212. package/components/picker/src/composables/use-picker.ts +179 -0
  213. package/components/picker/src/picker.ts +172 -0
  214. package/components/picker/src/picker.vue +197 -0
  215. package/components/popup/index.ts +1 -0
  216. package/components/popup/src/composables/index.ts +1 -0
  217. package/components/popup/src/composables/use-popup.ts +346 -0
  218. package/components/popup/src/keypress.ts +55 -0
  219. package/components/popup/src/popup.ts +101 -0
  220. package/components/popup/src/popup.vue +204 -0
  221. package/components/progress/index.ts +1 -0
  222. package/components/progress/src/composables/index.ts +2 -0
  223. package/components/progress/src/composables/progress-custom.ts +24 -0
  224. package/components/progress/src/composables/use-progress.ts +159 -0
  225. package/components/progress/src/progress.ts +84 -0
  226. package/components/progress/src/progress.vue +150 -0
  227. package/components/rate/index.ts +1 -0
  228. package/components/rate/src/composables/index.ts +1 -0
  229. package/components/rate/src/composables/use-rate.ts +196 -0
  230. package/components/rate/src/rate.ts +114 -0
  231. package/components/rate/src/rate.vue +113 -0
  232. package/components/read-more/index.ts +1 -0
  233. package/components/read-more/src/composables/index.ts +2 -0
  234. package/components/read-more/src/composables/read-more-custom.ts +46 -0
  235. package/components/read-more/src/composables/use-read-more.ts +72 -0
  236. package/components/read-more/src/read-more.ts +102 -0
  237. package/components/read-more/src/read-more.vue +104 -0
  238. package/components/safe-bottom/index.ts +1 -0
  239. package/components/safe-bottom/src/composables/index.ts +2 -0
  240. package/components/safe-bottom/src/composables/safe-bottom-custom.ts +18 -0
  241. package/components/safe-bottom/src/composables/use-safe-bottom.ts +17 -0
  242. package/components/safe-bottom/src/safe-bottom.ts +8 -0
  243. package/components/safe-bottom/src/safe-bottom.vue +43 -0
  244. package/components/scroll-list/index.ts +1 -0
  245. package/components/scroll-list/src/composables/index.ts +2 -0
  246. package/components/scroll-list/src/composables/scroll-list-custom.ts +41 -0
  247. package/components/scroll-list/src/composables/use-scroll-list.ts +126 -0
  248. package/components/scroll-list/src/scroll-list.ts +57 -0
  249. package/components/scroll-list/src/scroll-list.vue +143 -0
  250. package/components/{fu-scroll-list/scrollWxs.wxs → scroll-list/src/scroll-wxs.wxs} +2 -2
  251. package/components/search/index.ts +1 -0
  252. package/components/search/src/composables/index.ts +2 -0
  253. package/components/search/src/composables/search-custom.ts +25 -0
  254. package/components/search/src/composables/use-search.ts +106 -0
  255. package/components/search/src/search.ts +186 -0
  256. package/components/{fu-search/fu-search.vue → search/src/search.vue} +42 -154
  257. package/components/section/index.ts +1 -0
  258. package/components/section/src/composables/index.ts +1 -0
  259. package/components/section/src/composables/use-section.ts +19 -0
  260. package/components/section/src/section.ts +83 -0
  261. package/components/{fu-section/fu-section.vue → section/src/section.vue} +36 -35
  262. package/components/sign-board/index.ts +1 -0
  263. package/components/sign-board/src/composables/index.ts +1 -0
  264. package/components/sign-board/src/composables/use-sign-board.ts +552 -0
  265. package/components/sign-board/src/sign-board.ts +48 -0
  266. package/components/sign-board/src/sign-board.vue +191 -0
  267. package/components/skeleton/index.ts +1 -0
  268. package/components/skeleton/src/composables/index.ts +2 -0
  269. package/components/skeleton/src/composables/skeleton-custom.ts +23 -0
  270. package/components/skeleton/src/composables/use-skeleton.ts +102 -0
  271. package/components/skeleton/src/skeleton.ts +37 -0
  272. package/components/skeleton/src/skeleton.vue +147 -0
  273. package/components/status-bar/src/composables/index.ts +1 -0
  274. package/components/status-bar/src/composables/use-status-bar.ts +29 -0
  275. package/components/status-bar/src/status-bar.vue +21 -0
  276. package/components/steps/index.ts +1 -0
  277. package/components/steps/src/composables/index.ts +1 -0
  278. package/components/steps/src/composables/use-steps.ts +27 -0
  279. package/components/steps/src/steps.ts +65 -0
  280. package/components/steps/src/steps.vue +46 -0
  281. package/components/steps-item/index.ts +1 -0
  282. package/components/steps-item/src/composables/index.ts +2 -0
  283. package/components/steps-item/src/composables/steps-item-custom.ts +81 -0
  284. package/components/steps-item/src/composables/use-steps-item.ts +87 -0
  285. package/components/steps-item/src/steps-item.ts +38 -0
  286. package/components/steps-item/src/steps-item.vue +224 -0
  287. package/components/sticky/index.ts +1 -0
  288. package/components/sticky/src/composables/index.ts +2 -0
  289. package/components/sticky/src/composables/sticky-custom.ts +57 -0
  290. package/components/sticky/src/composables/use-sticky.ts +152 -0
  291. package/components/sticky/src/sticky.ts +62 -0
  292. package/components/sticky/src/sticky.vue +65 -0
  293. package/components/subsection/index.ts +1 -0
  294. package/components/subsection/src/composables/index.ts +2 -0
  295. package/components/subsection/src/composables/subsection-custom.ts +71 -0
  296. package/components/subsection/src/composables/use-subsection.ts +85 -0
  297. package/components/subsection/src/subsection.ts +107 -0
  298. package/components/subsection/src/subsection.vue +165 -0
  299. package/components/swipe-action/src/composables/index.ts +1 -0
  300. package/components/swipe-action/src/composables/use-swipe-action.ts +58 -0
  301. package/components/swipe-action/src/swipe-action.vue +81 -0
  302. package/components/swipe-action-item/index.ts +1 -0
  303. package/components/{fu-swipe-action-item/mpwxs.js → swipe-action-item/src/mpwxs.ts} +22 -27
  304. package/components/swipe-action-item/src/swipe-action-item.ts +68 -0
  305. package/components/{fu-swipe-action-item/fu-swipe-action-item.vue → swipe-action-item/src/swipe-action-item.vue} +24 -17
  306. package/components/{fu-swipe-action-item → swipe-action-item/src}/wx.wxs +2 -2
  307. package/components/swiper/index.ts +1 -0
  308. package/components/swiper/src/composables/index.ts +2 -0
  309. package/components/swiper/src/composables/swiper-custom.ts +40 -0
  310. package/components/swiper/src/composables/use-swiper.ts +75 -0
  311. package/components/swiper/src/swiper.ts +189 -0
  312. package/components/{fu-swiper/fu-swiper.vue → swiper/src/swiper.vue} +49 -118
  313. package/components/switch/index.ts +1 -0
  314. package/components/switch/src/composables/index.ts +2 -0
  315. package/components/switch/src/composables/switch-custom.ts +32 -0
  316. package/components/switch/src/composables/use-switch.ts +38 -0
  317. package/components/switch/src/switch.ts +115 -0
  318. package/components/{fu-switch/fu-switch.vue → switch/src/switch.vue} +25 -73
  319. package/components/tabs/index.ts +1 -0
  320. package/components/tabs/src/composables/index.ts +2 -0
  321. package/components/tabs/src/composables/tabs-custom.ts +43 -0
  322. package/components/tabs/src/composables/use-tabs.ts +210 -0
  323. package/components/tabs/src/tabs.ts +123 -0
  324. package/components/tabs/src/tabs.vue +153 -0
  325. package/components/tag/index.ts +1 -0
  326. package/components/tag/src/composables/index.ts +2 -0
  327. package/components/tag/src/composables/tag-custom.ts +61 -0
  328. package/components/tag/src/composables/use-tag.ts +23 -0
  329. package/components/tag/src/tag.ts +159 -0
  330. package/components/{fu-tag/fu-tag.vue → tag/src/tag.vue} +53 -93
  331. package/components/text/index.ts +1 -0
  332. package/components/text/src/composables/index.ts +2 -0
  333. package/components/text/src/composables/text-custom.ts +120 -0
  334. package/components/text/src/composables/use-text.ts +83 -0
  335. package/components/text/src/text.ts +261 -0
  336. package/components/text/src/text.vue +186 -0
  337. package/components/textarea/index.ts +1 -0
  338. package/components/textarea/src/composables/index.ts +2 -0
  339. package/components/textarea/src/composables/textarea-custom.ts +50 -0
  340. package/components/textarea/src/composables/use-textarea.ts +88 -0
  341. package/components/textarea/src/textarea.ts +250 -0
  342. package/components/textarea/src/textarea.vue +162 -0
  343. package/components/{fu-timeaxis/fu-timeaxis.vue → timeaxis/src/timeaxis.vue} +3 -9
  344. package/components/timeaxis-item/index.ts +1 -0
  345. package/components/timeaxis-item/src/timeaxis-item.ts +20 -0
  346. package/components/timeaxis-item/src/timeaxis-item.vue +50 -0
  347. package/components/transition/index.ts +1 -0
  348. package/components/transition/src/composables/createAnimation.ts +171 -0
  349. package/components/transition/src/composables/index.ts +2 -0
  350. package/components/transition/src/composables/transform-custom.ts +18 -0
  351. package/components/transition/src/composables/use-transition.ts +251 -0
  352. package/components/transition/src/transition.ts +64 -0
  353. package/components/transition/src/transition.vue +70 -0
  354. package/components/upload/index.ts +1 -0
  355. package/components/upload/src/composables/index.ts +5 -0
  356. package/components/upload/src/composables/upload-file-custom.ts +68 -0
  357. package/components/upload/src/composables/upload-image-custom.ts +76 -0
  358. package/components/upload/src/composables/use-upload-file.ts +27 -0
  359. package/components/upload/src/composables/use-upload-image.ts +38 -0
  360. package/components/upload/src/composables/use-upload.ts +388 -0
  361. package/components/upload/src/upload-file.ts +84 -0
  362. package/components/upload/src/upload-file.vue +193 -0
  363. package/components/upload/src/upload-image.ts +90 -0
  364. package/components/upload/src/upload-image.vue +158 -0
  365. package/components/upload/src/upload.ts +196 -0
  366. package/components/upload/src/upload.vue +151 -0
  367. package/components/{fu-upload/utils.js → upload/src/utils.ts} +12 -12
  368. package/components/vtabs/index.ts +1 -0
  369. package/components/vtabs/src/composables/index.ts +2 -0
  370. package/components/vtabs/src/composables/use-vtabs.ts +276 -0
  371. package/components/vtabs/src/composables/vtabs-custom.ts +54 -0
  372. package/components/vtabs/src/vtabs.ts +130 -0
  373. package/components/vtabs/src/vtabs.vue +189 -0
  374. package/components/vtabs-item/index.ts +1 -0
  375. package/components/vtabs-item/src/composables/index.ts +1 -0
  376. package/components/vtabs-item/src/composables/use-vtabs-item.ts +60 -0
  377. package/components/vtabs-item/src/vtabs-item.ts +10 -0
  378. package/components/vtabs-item/src/vtabs-item.vue +20 -0
  379. package/components/waterfall/index.ts +1 -0
  380. package/components/waterfall/src/composables/index.ts +2 -0
  381. package/components/waterfall/src/composables/use-waterfall.ts +139 -0
  382. package/components/waterfall/src/composables/waterfall-custom.ts +18 -0
  383. package/components/waterfall/src/waterfall.ts +114 -0
  384. package/components/waterfall/src/waterfall.vue +104 -0
  385. package/{libs/config/config.js → config/config.ts} +13 -7
  386. package/constants/images.ts +18 -0
  387. package/constants/index.ts +6 -0
  388. package/constants/open-types.ts +33 -0
  389. package/constants/props.ts +18 -0
  390. package/constants/shapes.ts +6 -0
  391. package/constants/types.ts +10 -0
  392. package/constants/z-index.ts +27 -0
  393. package/global.d.ts +103 -0
  394. package/hooks/index.ts +25 -0
  395. package/index.scss +3 -5
  396. package/index.ts +56 -0
  397. package/package.json +1 -1
  398. package/{libs/route/min.route.config.js → route/min.route.config.ts} +62 -23
  399. package/route/route.config.ts +93 -0
  400. package/style/color.scss +26 -0
  401. package/{libs/style → style}/common.scss +1 -14
  402. package/{libs/function/applyEven.js → utils/applyEven.ts} +21 -12
  403. package/utils/check.ts +276 -0
  404. package/{libs/function/chooseUploadFile.js → utils/chooseUploadFile.ts} +22 -26
  405. package/utils/common.ts +609 -0
  406. package/{libs/function/digit.js → utils/digit.ts} +63 -33
  407. package/{components/fu-swipe-action-item/isPC.js → utils/isPC.ts} +1 -1
  408. package/utils/toast.ts +42 -0
  409. package/components/fu-avatar/fu-avatar.vue +0 -157
  410. package/components/fu-avatar/props.js +0 -117
  411. package/components/fu-avatar-group/props.js +0 -74
  412. package/components/fu-badge/props.js +0 -112
  413. package/components/fu-banner-arc/fu-banner-arc.vue +0 -58
  414. package/components/fu-banner-arc/props.js +0 -35
  415. package/components/fu-button/props.js +0 -228
  416. package/components/fu-cell/props.js +0 -108
  417. package/components/fu-cell-group/props.js +0 -19
  418. package/components/fu-checkbox/fu-checkbox.vue +0 -469
  419. package/components/fu-checkbox/props.js +0 -126
  420. package/components/fu-code-input/fu-code-input.vue +0 -269
  421. package/components/fu-code-input/props.js +0 -136
  422. package/components/fu-code-verify/fu-code-verify.vue +0 -221
  423. package/components/fu-code-verify/props.js +0 -136
  424. package/components/fu-collapse/fu-collapse.vue +0 -143
  425. package/components/fu-collapse/props.js +0 -29
  426. package/components/fu-collapse-item/props.js +0 -101
  427. package/components/fu-countdown/fu-countdown.vue +0 -164
  428. package/components/fu-countdown/props.js +0 -35
  429. package/components/fu-datetime-picker/fu-datetime-picker.vue +0 -375
  430. package/components/fu-datetime-picker/props.js +0 -194
  431. package/components/fu-form/fu-form.vue +0 -284
  432. package/components/fu-form/props.js +0 -74
  433. package/components/fu-form-item/fu-form-item.vue +0 -535
  434. package/components/fu-form-item/props.js +0 -66
  435. package/components/fu-grid/fu-grid.vue +0 -113
  436. package/components/fu-grid/props.js +0 -44
  437. package/components/fu-grid-item/fu-grid-item.vue +0 -136
  438. package/components/fu-grid-item/props.js +0 -27
  439. package/components/fu-icon/fu-icon.vue +0 -107
  440. package/components/fu-icon/icon.js +0 -1031
  441. package/components/fu-icon/props.js +0 -55
  442. package/components/fu-image/fu-image.vue +0 -268
  443. package/components/fu-image/props.js +0 -145
  444. package/components/fu-index-anchor/fu-index-anchor.vue +0 -91
  445. package/components/fu-index-anchor/props.js +0 -43
  446. package/components/fu-index-item/fu-index-item.vue +0 -79
  447. package/components/fu-index-list/fu-index-list.vue +0 -426
  448. package/components/fu-index-list/props.js +0 -42
  449. package/components/fu-input/fu-input.vue +0 -393
  450. package/components/fu-input/props.js +0 -291
  451. package/components/fu-keyboard/keyboard-number.vue +0 -195
  452. package/components/fu-keyboard/props.js +0 -138
  453. package/components/fu-line/fu-line.vue +0 -64
  454. package/components/fu-line/props.js +0 -58
  455. package/components/fu-link/fu-link.vue +0 -91
  456. package/components/fu-link/props.js +0 -49
  457. package/components/fu-loading/props.js +0 -60
  458. package/components/fu-loading-more/props.js +0 -59
  459. package/components/fu-modal/props.js +0 -128
  460. package/components/fu-navbar/fu-navbar.vue +0 -271
  461. package/components/fu-navbar/props.js +0 -72
  462. package/components/fu-navbar/status-bar.vue +0 -27
  463. package/components/fu-notice-bar/notice-column.vue +0 -313
  464. package/components/fu-notice-bar/notice-row.vue +0 -363
  465. package/components/fu-notice-bar/props.js +0 -173
  466. package/components/fu-number-box/fu-number-box.vue +0 -210
  467. package/components/fu-number-box/props.js +0 -79
  468. package/components/fu-parse/fu-parse.vue +0 -508
  469. package/components/fu-parse/node/node.vue +0 -576
  470. package/components/fu-parse/parser.js +0 -1335
  471. package/components/fu-picker/fu-picker.vue +0 -347
  472. package/components/fu-picker/props.js +0 -150
  473. package/components/fu-popup/fu-popup.vue +0 -475
  474. package/components/fu-popup/keypress.js +0 -45
  475. package/components/fu-popup/props.js +0 -83
  476. package/components/fu-progress/fu-progress.vue +0 -292
  477. package/components/fu-progress/props.js +0 -76
  478. package/components/fu-rate/fu-rate.vue +0 -301
  479. package/components/fu-rate/props.js +0 -103
  480. package/components/fu-read-more/fu-read-more.vue +0 -191
  481. package/components/fu-read-more/props.js +0 -72
  482. package/components/fu-ribbon/fu-ribbon.vue +0 -106
  483. package/components/fu-ribbon/props.js +0 -35
  484. package/components/fu-safe-area/fu-safe-area.vue +0 -61
  485. package/components/fu-scroll-list/fu-scroll-list.vue +0 -196
  486. package/components/fu-scroll-list/nvue.js +0 -28
  487. package/components/fu-scroll-list/props.js +0 -43
  488. package/components/fu-search/props.js +0 -149
  489. package/components/fu-section/props.js +0 -59
  490. package/components/fu-sign-board/fu-sign-board.vue +0 -728
  491. package/components/fu-sign-board/props.js +0 -34
  492. package/components/fu-skeleton/fu-skeleton.vue +0 -238
  493. package/components/fu-skeleton/props.js +0 -35
  494. package/components/fu-steps/fu-steps.vue +0 -81
  495. package/components/fu-steps/props.js +0 -55
  496. package/components/fu-steps-item/fu-steps-item.vue +0 -274
  497. package/components/fu-steps-item/props.js +0 -34
  498. package/components/fu-sticky/fu-sticky.vue +0 -232
  499. package/components/fu-sticky/props.js +0 -54
  500. package/components/fu-subsection/fu-subsection.vue +0 -288
  501. package/components/fu-subsection/props.js +0 -88
  502. package/components/fu-swipe-action/fu-swipe-action.vue +0 -64
  503. package/components/fu-swipe-action-item/props.js +0 -54
  504. package/components/fu-swiper/props.js +0 -163
  505. package/components/fu-switch/props.js +0 -102
  506. package/components/fu-tabs/fu-tabs.vue +0 -361
  507. package/components/fu-tabs/props.js +0 -109
  508. package/components/fu-tag/props.js +0 -135
  509. package/components/fu-text/button.js +0 -13
  510. package/components/fu-text/fu-text.vue +0 -254
  511. package/components/fu-text/openType.js +0 -47
  512. package/components/fu-text/props.js +0 -173
  513. package/components/fu-text/value.js +0 -88
  514. package/components/fu-textarea/fu-textarea.vue +0 -288
  515. package/components/fu-textarea/props.js +0 -206
  516. package/components/fu-timeaxis-item/fu-timeaxis-item.vue +0 -68
  517. package/components/fu-transition/createAnimation.js +0 -131
  518. package/components/fu-transition/fu-transition.vue +0 -292
  519. package/components/fu-transition/props.js +0 -21
  520. package/components/fu-upload/fu-upload.vue +0 -563
  521. package/components/fu-upload/props.js +0 -171
  522. package/components/fu-upload/upload-file.vue +0 -332
  523. package/components/fu-upload/upload-image.vue +0 -304
  524. package/components/fu-vtabs/fu-vtabs.vue +0 -443
  525. package/components/fu-vtabs/props.js +0 -114
  526. package/components/fu-vtabs-item/fu-vtabs-item.vue +0 -71
  527. package/components/fu-waterfall/fu-waterfall.vue +0 -238
  528. package/components/fu-waterfall/props.js +0 -85
  529. package/index.js +0 -71
  530. package/libs/function/check.js +0 -215
  531. package/libs/function/common.js +0 -527
  532. package/libs/function/message.js +0 -47
  533. package/libs/mixin/button.js +0 -85
  534. package/libs/mixin/mixin.js +0 -94
  535. package/libs/mixin/mpMixin.js +0 -8
  536. package/libs/mixin/mpShare.js +0 -15
  537. package/libs/route/route.config.js +0 -66
  538. package/libs/style/color.scss +0 -24
  539. /package/components/{fu-icon/icon.ttf → icon/src/fuicon.ttf} +0 -0
  540. /package/components/{fu-swipe-action-item → swipe-action-item/src}/bindingx.js +0 -0
  541. /package/components/{fu-swipe-action-item → swipe-action-item/src}/mpalipay.js +0 -0
  542. /package/components/{fu-swipe-action-item → swipe-action-item/src}/mpother.js +0 -0
  543. /package/components/{fu-swipe-action-item → swipe-action-item/src}/render.js +0 -0
  544. /package/{libs/style → style}/components.scss +0 -0
  545. /package/{libs/style → style}/style.h5.scss +0 -0
  546. /package/{libs/style → style}/style.mp.scss +0 -0
  547. /package/{libs/function → utils}/dayjs.js +0 -0
@@ -1,1335 +0,0 @@
1
- /**
2
- * @fileoverview html 解析器
3
- */
4
-
5
- // 配置
6
- const config = {
7
- // 信任的标签(保持标签名不变)
8
- trustTags: makeMap('a,abbr,ad,audio,b,blockquote,br,code,col,colgroup,dd,del,dl,dt,div,em,fieldset,h1,h2,h3,h4,h5,h6,hr,i,img,ins,label,legend,li,ol,p,q,ruby,rt,source,span,strong,sub,sup,table,tbody,td,tfoot,th,thead,tr,title,ul,video'),
9
-
10
- // 块级标签(转为 div,其他的非信任标签转为 span)
11
- blockTags: makeMap('address,article,aside,body,caption,center,cite,footer,header,html,nav,pre,section'),
12
-
13
- // #ifdef (MP-WEIXIN || MP-QQ || APP-PLUS || MP-360) && VUE3
14
- // 行内标签
15
- inlineTags: makeMap('abbr,b,big,code,del,em,i,ins,label,q,small,span,strong,sub,sup'),
16
- // #endif
17
-
18
- // 要移除的标签
19
- ignoreTags: makeMap('area,base,canvas,embed,frame,head,iframe,input,link,map,meta,param,rp,script,source,style,textarea,title,track,wbr'),
20
-
21
- // 自闭合的标签
22
- voidTags: makeMap('area,base,br,col,circle,ellipse,embed,frame,hr,img,input,line,link,meta,param,path,polygon,rect,source,track,use,wbr'),
23
-
24
- // html 实体
25
- entities: {
26
- lt: '<',
27
- gt: '>',
28
- quot: '"',
29
- apos: "'",
30
- ensp: '\u2002',
31
- emsp: '\u2003',
32
- nbsp: '\xA0',
33
- semi: ';',
34
- ndash: '–',
35
- mdash: '—',
36
- middot: '·',
37
- lsquo: '‘',
38
- rsquo: '’',
39
- ldquo: '“',
40
- rdquo: '”',
41
- bull: '•',
42
- hellip: '…',
43
- larr: '←',
44
- uarr: '↑',
45
- rarr: '→',
46
- darr: '↓'
47
- },
48
-
49
- // 默认的标签样式
50
- tagStyle: {
51
- // #ifndef APP-PLUS-NVUE
52
- address: 'font-style:italic',
53
- big: 'display:inline;font-size:1.2em',
54
- caption: 'display:table-caption;text-align:center',
55
- center: 'text-align:center',
56
- cite: 'font-style:italic',
57
- dd: 'margin-left:40px',
58
- mark: 'background-color:yellow',
59
- pre: 'font-family:monospace;white-space:pre',
60
- s: 'text-decoration:line-through',
61
- small: 'display:inline;font-size:0.8em',
62
- strike: 'text-decoration:line-through',
63
- u: 'text-decoration:underline'
64
- // #endif
65
- },
66
-
67
- // svg 大小写对照表
68
- svgDict: {
69
- animatetransform: 'animateTransform',
70
- lineargradient: 'linearGradient',
71
- viewbox: 'viewBox',
72
- attributename: 'attributeName',
73
- repeatcount: 'repeatCount',
74
- repeatdur: 'repeatDur'
75
- }
76
- }
77
- const tagSelector={}
78
- const {
79
- windowWidth,
80
- // #ifdef MP-WEIXIN
81
- system
82
- // #endif
83
- } = uni.getSystemInfoSync()
84
- const blankChar = makeMap(' ,\r,\n,\t,\f')
85
- let idIndex = 0
86
-
87
- // #ifdef H5 || APP-PLUS
88
- config.ignoreTags.iframe = undefined
89
- config.trustTags.iframe = true
90
- config.ignoreTags.embed = undefined
91
- config.trustTags.embed = true
92
- // #endif
93
- // #ifdef APP-PLUS-NVUE
94
- config.ignoreTags.source = undefined
95
- config.ignoreTags.style = undefined
96
- // #endif
97
-
98
- /**
99
- * @description 创建 map
100
- * @param {String} str 逗号分隔
101
- */
102
- function makeMap (str) {
103
- const map = Object.create(null)
104
- const list = str.split(',')
105
- for (let i = list.length; i--;) {
106
- map[list[i]] = true
107
- }
108
- return map
109
- }
110
-
111
- /**
112
- * @description 解码 html 实体
113
- * @param {String} str 要解码的字符串
114
- * @param {Boolean} amp 要不要解码 &amp;
115
- * @returns {String} 解码后的字符串
116
- */
117
- function decodeEntity (str, amp) {
118
- let i = str.indexOf('&')
119
- while (i !== -1) {
120
- const j = str.indexOf(';', i + 3)
121
- let code
122
- if (j === -1) break
123
- if (str[i + 1] === '#') {
124
- // &#123; 形式的实体
125
- code = parseInt((str[i + 2] === 'x' ? '0' : '') + str.substring(i + 2, j))
126
- if (!isNaN(code)) {
127
- str = str.substr(0, i) + String.fromCharCode(code) + str.substr(j + 1)
128
- }
129
- } else {
130
- // &nbsp; 形式的实体
131
- code = str.substring(i + 1, j)
132
- if (config.entities[code] || (code === 'amp' && amp)) {
133
- str = str.substr(0, i) + (config.entities[code] || '&') + str.substr(j + 1)
134
- }
135
- }
136
- i = str.indexOf('&', i + 1)
137
- }
138
- return str
139
- }
140
-
141
- /**
142
- * @description 合并多个块级标签,加快长内容渲染
143
- * @param {Array} nodes 要合并的标签数组
144
- */
145
- function mergeNodes (nodes) {
146
- let i = nodes.length - 1
147
- for (let j = i; j >= -1; j--) {
148
- if (j === -1 || nodes[j].c || !nodes[j].name || (nodes[j].name !== 'div' && nodes[j].name !== 'p' && nodes[j].name[0] !== 'h') || (nodes[j].attrs.style || '').includes('inline')) {
149
- if (i - j >= 5) {
150
- nodes.splice(j + 1, i - j, {
151
- name: 'div',
152
- attrs: {},
153
- children: nodes.slice(j + 1, i + 1)
154
- })
155
- }
156
- i = j - 1
157
- }
158
- }
159
- }
160
-
161
- /**
162
- * @description html 解析器
163
- * @param {Object} vm 组件实例
164
- */
165
- function Parser (vm) {
166
- this.options = vm || {}
167
- this.tagStyle = Object.assign({}, config.tagStyle, this.options.tagStyle)
168
- this.imgList = vm.imgList || []
169
- this.imgList._unloadimgs = 0
170
- this.plugins = vm.plugins || []
171
- this.attrs = Object.create(null)
172
- this.stack = []
173
- this.nodes = []
174
- this.pre = (this.options.containerStyle || '').includes('white-space') && this.options.containerStyle.includes('pre') ? 2 : 0
175
- }
176
-
177
- /**
178
- * @description 执行解析
179
- * @param {String} content 要解析的文本
180
- */
181
- Parser.prototype.parse = function (content) {
182
- // 插件处理
183
- for (let i = this.plugins.length; i--;) {
184
- if (this.plugins[i].onUpdate) {
185
- content = this.plugins[i].onUpdate(content, config) || content
186
- }
187
- }
188
-
189
- new Lexer(this).parse(content)
190
- // 出栈未闭合的标签
191
- while (this.stack.length) {
192
- this.popNode()
193
- }
194
- if (this.nodes.length > 50) {
195
- mergeNodes(this.nodes)
196
- }
197
- return this.nodes
198
- }
199
-
200
- /**
201
- * @description 将标签暴露出来(不被 rich-text 包含)
202
- */
203
- Parser.prototype.expose = function () {
204
- // #ifndef APP-PLUS-NVUE
205
- for (let i = this.stack.length; i--;) {
206
- const item = this.stack[i]
207
- if (item.c || item.name === 'a' || item.name === 'video' || item.name === 'audio') return
208
- item.c = 1
209
- }
210
- // #endif
211
- }
212
-
213
- /**
214
- * @description 处理插件
215
- * @param {Object} node 要处理的标签
216
- * @returns {Boolean} 是否要移除此标签
217
- */
218
- Parser.prototype.hook = function (node) {
219
- for (let i = this.plugins.length; i--;) {
220
- if (this.plugins[i].onParse && this.plugins[i].onParse(node, this) === false) {
221
- return false
222
- }
223
- }
224
- return true
225
- }
226
-
227
- /**
228
- * @description 将链接拼接上主域名
229
- * @param {String} url 需要拼接的链接
230
- * @returns {String} 拼接后的链接
231
- */
232
- Parser.prototype.getUrl = function (url) {
233
- const domain = this.options.domain
234
- if (url[0] === '/') {
235
- if (url[1] === '/') {
236
- // // 开头的补充协议名
237
- url = (domain ? domain.split('://')[0] : 'http') + ':' + url
238
- } else if (domain) {
239
- // 否则补充整个域名
240
- url = domain + url
241
- } /* #ifdef APP-PLUS */ else {
242
- url = plus.io.convertLocalFileSystemURL(url)
243
- } /* #endif */
244
- } else if (!url.includes('data:') && !url.includes('://')) {
245
- if (domain) {
246
- url = domain + '/' + url
247
- } /* #ifdef APP-PLUS */ else {
248
- url = plus.io.convertLocalFileSystemURL(url)
249
- } /* #endif */
250
- }
251
- return url
252
- }
253
-
254
- /**
255
- * @description 解析样式表
256
- * @param {Object} node 标签
257
- * @returns {Object}
258
- */
259
- Parser.prototype.parseStyle = function (node) {
260
- const attrs = node.attrs
261
- const list = (this.tagStyle[node.name] || '').split(';').concat((attrs.style || '').split(';'))
262
- const styleObj = {}
263
- let tmp = ''
264
-
265
- if (attrs.id && !this.xml) {
266
- // 暴露锚点
267
- if (this.options.useAnchor) {
268
- this.expose()
269
- } else if (node.name !== 'img' && node.name !== 'a' && node.name !== 'video' && node.name !== 'audio') {
270
- attrs.id = undefined
271
- }
272
- }
273
-
274
- // 转换 width 和 height 属性
275
- if (attrs.width) {
276
- styleObj.width = parseFloat(attrs.width) + (attrs.width.includes('%') ? '%' : 'px')
277
- attrs.width = undefined
278
- }
279
- if (attrs.height) {
280
- styleObj.height = parseFloat(attrs.height) + (attrs.height.includes('%') ? '%' : 'px')
281
- attrs.height = undefined
282
- }
283
-
284
- for (let i = 0, len = list.length; i < len; i++) {
285
- const info = list[i].split(':')
286
- if (info.length < 2) continue
287
- const key = info.shift().trim().toLowerCase()
288
- let value = info.join(':').trim()
289
- if ((value[0] === '-' && value.lastIndexOf('-') > 0) || value.includes('safe')) {
290
- // 兼容性的 css 不压缩
291
- tmp += `;${key}:${value}`
292
- } else if (!styleObj[key] || value.includes('import') || !styleObj[key].includes('import')) {
293
- // 重复的样式进行覆盖
294
- if (value.includes('url')) {
295
- // 填充链接
296
- let j = value.indexOf('(') + 1
297
- if (j) {
298
- while (value[j] === '"' || value[j] === "'" || blankChar[value[j]]) {
299
- j++
300
- }
301
- value = value.substr(0, j) + this.getUrl(value.substr(j))
302
- }
303
- } else if (value.includes('rpx')) {
304
- // 转换 rpx(rich-text 内部不支持 rpx)
305
- value = value.replace(/[0-9.]+\s*rpx/g, $ => parseFloat($) * windowWidth / 750 + 'px')
306
- }
307
- styleObj[key] = value
308
- }
309
- }
310
-
311
- node.attrs.style = tmp
312
- return styleObj
313
- }
314
-
315
- /**
316
- * @description 解析到标签名
317
- * @param {String} name 标签名
318
- * @private
319
- */
320
- Parser.prototype.onTagName = function (name) {
321
- this.tagName = this.xml ? name : name.toLowerCase()
322
- if (this.tagName === 'svg') {
323
- this.xml = (this.xml || 0) + 1 // svg 标签内大小写敏感
324
- config.ignoreTags.style = undefined // svg 标签内 style 可用
325
- }
326
- }
327
-
328
- /**
329
- * @description 解析到属性名
330
- * @param {String} name 属性名
331
- * @private
332
- */
333
- Parser.prototype.onAttrName = function (name) {
334
- name = this.xml ? name : name.toLowerCase()
335
- if (name.substr(0, 5) === 'data-') {
336
- if (name === 'data-src' && !this.attrs.src) {
337
- // data-src 自动转为 src
338
- this.attrName = 'src'
339
- } else if (this.tagName === 'img' || this.tagName === 'a') {
340
- // a 和 img 标签保留 data- 的属性,可以在 imgtap 和 linktap 事件中使用
341
- this.attrName = name
342
- } else {
343
- // 剩余的移除以减小大小
344
- this.attrName = undefined
345
- }
346
- } else {
347
- this.attrName = name
348
- this.attrs[name] = 'T' // boolean 型属性缺省设置
349
- }
350
- }
351
-
352
- /**
353
- * @description 解析到属性值
354
- * @param {String} val 属性值
355
- * @private
356
- */
357
- Parser.prototype.onAttrVal = function (val) {
358
- const name = this.attrName || ''
359
- if (name === 'style' || name === 'href') {
360
- // 部分属性进行实体解码
361
- this.attrs[name] = decodeEntity(val, true)
362
- } else if (name.includes('src')) {
363
- // 拼接主域名
364
- this.attrs[name] = this.getUrl(decodeEntity(val, true))
365
- } else if (name) {
366
- this.attrs[name] = val
367
- }
368
- }
369
-
370
- /**
371
- * @description 解析到标签开始
372
- * @param {Boolean} selfClose 是否有自闭合标识 />
373
- * @private
374
- */
375
- Parser.prototype.onOpenTag = function (selfClose) {
376
- // 拼装 node
377
- const node = Object.create(null)
378
- node.name = this.tagName
379
- node.attrs = this.attrs
380
- // 避免因为自动 diff 使得 type 被设置为 null 导致部分内容不显示
381
- if (this.options.nodes.length) {
382
- node.type = 'node'
383
- }
384
- this.attrs = Object.create(null)
385
-
386
- const attrs = node.attrs
387
- const parent = this.stack[this.stack.length - 1]
388
- const siblings = parent ? parent.children : this.nodes
389
- const close = this.xml ? selfClose : config.voidTags[node.name]
390
-
391
- // 替换标签名选择器
392
- if (tagSelector[node.name]) {
393
- attrs.class = tagSelector[node.name] + (attrs.class ? ' ' + attrs.class : '')
394
- }
395
-
396
- // 转换 embed 标签
397
- if (node.name === 'embed') {
398
- // #ifndef H5 || APP-PLUS
399
- const src = attrs.src || ''
400
- // 按照后缀名和 type 将 embed 转为 video 或 audio
401
- if (src.includes('.mp4') || src.includes('.3gp') || src.includes('.m3u8') || (attrs.type || '').includes('video')) {
402
- node.name = 'video'
403
- } else if (src.includes('.mp3') || src.includes('.wav') || src.includes('.aac') || src.includes('.m4a') || (attrs.type || '').includes('audio')) {
404
- node.name = 'audio'
405
- }
406
- if (attrs.autostart) {
407
- attrs.autoplay = 'T'
408
- }
409
- attrs.controls = 'T'
410
- // #endif
411
- // #ifdef H5 || APP-PLUS
412
- this.expose()
413
- // #endif
414
- }
415
-
416
- // #ifndef APP-PLUS-NVUE
417
- // 处理音视频
418
- if (node.name === 'video' || node.name === 'audio') {
419
- // 设置 id 以便获取 context
420
- if (node.name === 'video' && !attrs.id) {
421
- attrs.id = 'v' + idIndex++
422
- }
423
- // 没有设置 controls 也没有设置 autoplay 的自动设置 controls
424
- if (!attrs.controls && !attrs.autoplay) {
425
- attrs.controls = 'T'
426
- }
427
- // 用数组存储所有可用的 source
428
- node.src = []
429
- if (attrs.src) {
430
- node.src.push(attrs.src)
431
- attrs.src = undefined
432
- }
433
- this.expose()
434
- }
435
- // #endif
436
-
437
- // 处理自闭合标签
438
- if (close) {
439
- if (!this.hook(node) || config.ignoreTags[node.name]) {
440
- // 通过 base 标签设置主域名
441
- if (node.name === 'base' && !this.options.domain) {
442
- this.options.domain = attrs.href
443
- } /* #ifndef APP-PLUS-NVUE */ else if (node.name === 'source' && parent && (parent.name === 'video' || parent.name === 'audio') && attrs.src) {
444
- // 设置 source 标签(仅父节点为 video 或 audio 时有效)
445
- parent.src.push(attrs.src)
446
- } /* #endif */
447
- return
448
- }
449
-
450
- // 解析 style
451
- const styleObj = this.parseStyle(node)
452
-
453
- // 处理图片
454
- if (node.name === 'img') {
455
- if (attrs.src) {
456
- // 标记 webp
457
- if (attrs.src.includes('webp')) {
458
- node.webp = 'T'
459
- }
460
- // data url 图片如果没有设置 original-src 默认为不可预览的小图片
461
- if (attrs.src.includes('data:') && !attrs['original-src']) {
462
- attrs.ignore = 'T'
463
- }
464
- if (!attrs.ignore || node.webp || attrs.src.includes('cloud://')) {
465
- for (let i = this.stack.length; i--;) {
466
- const item = this.stack[i]
467
- if (item.name === 'a') {
468
- node.a = item.attrs
469
- }
470
- if (item.name === 'table' && !node.webp && !attrs.src.includes('cloud://')) {
471
- if (!styleObj.display || styleObj.display.includes('inline')) {
472
- node.t = 'inline-block'
473
- } else {
474
- node.t = styleObj.display
475
- }
476
- styleObj.display = undefined
477
- }
478
- // #ifndef H5 || APP-PLUS
479
- const style = item.attrs.style || ''
480
- if (style.includes('flex:') && !style.includes('flex:0') && !style.includes('flex: 0') && (!styleObj.width || parseInt(styleObj.width) > 100)) {
481
- styleObj.width = '100% !important'
482
- styleObj.height = ''
483
- for (let j = i + 1; j < this.stack.length; j++) {
484
- this.stack[j].attrs.style = (this.stack[j].attrs.style || '').replace('inline-', '')
485
- }
486
- } else if (style.includes('flex') && styleObj.width === '100%') {
487
- for (let j = i + 1; j < this.stack.length; j++) {
488
- const style = this.stack[j].attrs.style || ''
489
- if (!style.includes(';width') && !style.includes(' width') && style.indexOf('width') !== 0) {
490
- styleObj.width = ''
491
- break
492
- }
493
- }
494
- } else if (style.includes('inline-block')) {
495
- if (styleObj.width && styleObj.width[styleObj.width.length - 1] === '%') {
496
- item.attrs.style += ';max-width:' + styleObj.width
497
- styleObj.width = ''
498
- } else {
499
- item.attrs.style += ';max-width:100%'
500
- }
501
- }
502
- // #endif
503
- item.c = 1
504
- }
505
- attrs.i = this.imgList.length.toString()
506
- let src = attrs['original-src'] || attrs.src
507
- // #ifndef H5 || MP-ALIPAY || APP-PLUS || MP-360
508
- if (this.imgList.includes(src)) {
509
- // 如果有重复的链接则对域名进行随机大小写变换避免预览时错位
510
- let i = src.indexOf('://')
511
- if (i !== -1) {
512
- i += 3
513
- let newSrc = src.substr(0, i)
514
- for (; i < src.length; i++) {
515
- if (src[i] === '/') break
516
- newSrc += Math.random() > 0.5 ? src[i].toUpperCase() : src[i]
517
- }
518
- newSrc += src.substr(i)
519
- src = newSrc
520
- }
521
- }
522
- // #endif
523
- this.imgList.push(src)
524
- if (!node.t) {
525
- this.imgList._unloadimgs += 1
526
- }
527
- // #ifdef H5 || APP-PLUS
528
- if (this.options.lazyLoad) {
529
- attrs['data-src'] = attrs.src
530
- attrs.src = undefined
531
- }
532
- // #endif
533
- }
534
- }
535
- if (styleObj.display === 'inline') {
536
- styleObj.display = ''
537
- }
538
- // #ifndef APP-PLUS-NVUE
539
- if (attrs.ignore) {
540
- styleObj['max-width'] = styleObj['max-width'] || '100%'
541
- attrs.style += ';-webkit-touch-callout:none'
542
- }
543
- // #endif
544
- // 设置的宽度超出屏幕,为避免变形,高度转为自动
545
- if (parseInt(styleObj.width) > windowWidth) {
546
- styleObj.height = undefined
547
- }
548
- // 记录是否设置了宽高
549
- if (!isNaN(parseInt(styleObj.width))) {
550
- node.w = 'T'
551
- }
552
- if (!isNaN(parseInt(styleObj.height)) && (!styleObj.height.includes('%') || (parent && (parent.attrs.style || '').includes('height')))) {
553
- node.h = 'T'
554
- }
555
- } else if (node.name === 'svg') {
556
- siblings.push(node)
557
- this.stack.push(node)
558
- this.popNode()
559
- return
560
- }
561
- for (const key in styleObj) {
562
- if (styleObj[key]) {
563
- attrs.style += `;${key}:${styleObj[key].replace(' !important', '')}`
564
- }
565
- }
566
- attrs.style = attrs.style.substr(1) || undefined
567
- // #ifdef (MP-WEIXIN || MP-QQ) && VUE3
568
- if (!attrs.style) {
569
- delete attrs.style
570
- }
571
- // #endif
572
- } else {
573
- if ((node.name === 'pre' || ((attrs.style || '').includes('white-space') && attrs.style.includes('pre'))) && this.pre !== 2) {
574
- this.pre = node.pre = 1
575
- }
576
- node.children = []
577
- this.stack.push(node)
578
- }
579
-
580
- // 加入节点树
581
- siblings.push(node)
582
- }
583
-
584
- /**
585
- * @description 解析到标签结束
586
- * @param {String} name 标签名
587
- * @private
588
- */
589
- Parser.prototype.onCloseTag = function (name) {
590
- // 依次出栈到匹配为止
591
- name = this.xml ? name : name.toLowerCase()
592
- let i
593
- for (i = this.stack.length; i--;) {
594
- if (this.stack[i].name === name) break
595
- }
596
- if (i !== -1) {
597
- while (this.stack.length > i) {
598
- this.popNode()
599
- }
600
- } else if (name === 'p' || name === 'br') {
601
- const siblings = this.stack.length ? this.stack[this.stack.length - 1].children : this.nodes
602
- siblings.push({
603
- name,
604
- attrs: {
605
- class: tagSelector[name] || '',
606
- style: this.tagStyle[name] || ''
607
- }
608
- })
609
- }
610
- }
611
-
612
- /**
613
- * @description 处理标签出栈
614
- * @private
615
- */
616
- Parser.prototype.popNode = function () {
617
- const node = this.stack.pop()
618
- let attrs = node.attrs
619
- const children = node.children
620
- const parent = this.stack[this.stack.length - 1]
621
- const siblings = parent ? parent.children : this.nodes
622
-
623
- if (!this.hook(node) || config.ignoreTags[node.name]) {
624
- // 获取标题
625
- if (node.name === 'title' && children.length && children[0].type === 'text' && this.options.setTitle) {
626
- uni.setNavigationBarTitle({
627
- title: children[0].text
628
- })
629
- }
630
- siblings.pop()
631
- return
632
- }
633
-
634
- if (node.pre && this.pre !== 2) {
635
- // 是否合并空白符标识
636
- this.pre = node.pre = undefined
637
- for (let i = this.stack.length; i--;) {
638
- if (this.stack[i].pre) {
639
- this.pre = 1
640
- }
641
- }
642
- }
643
-
644
- const styleObj = {}
645
-
646
- // 转换 svg
647
- if (node.name === 'svg') {
648
- if (this.xml > 1) {
649
- // 多层 svg 嵌套
650
- this.xml--
651
- return
652
- }
653
- // #ifdef APP-PLUS-NVUE
654
- (function traversal (node) {
655
- if (node.name) {
656
- // 调整 svg 的大小写
657
- node.name = config.svgDict[node.name] || node.name
658
- for (const item in node.attrs) {
659
- if (config.svgDict[item]) {
660
- node.attrs[config.svgDict[item]] = node.attrs[item]
661
- node.attrs[item] = undefined
662
- }
663
- }
664
- for (let i = 0; i < (node.children || []).length; i++) {
665
- traversal(node.children[i])
666
- }
667
- }
668
- })(node)
669
- // #endif
670
- // #ifndef APP-PLUS-NVUE
671
- let src = ''
672
- const style = attrs.style
673
- attrs.style = ''
674
- attrs.xmlns = 'http://www.w3.org/2000/svg';
675
- (function traversal (node) {
676
- if (node.type === 'text') {
677
- src += node.text
678
- return
679
- }
680
- const name = config.svgDict[node.name] || node.name
681
- src += '<' + name
682
- for (const item in node.attrs) {
683
- const val = node.attrs[item]
684
- if (val) {
685
- src += ` ${config.svgDict[item] || item}="${val}"`
686
- }
687
- }
688
- if (!node.children) {
689
- src += '/>'
690
- } else {
691
- src += '>'
692
- for (let i = 0; i < node.children.length; i++) {
693
- traversal(node.children[i])
694
- }
695
- src += '</' + name + '>'
696
- }
697
- })(node)
698
- node.name = 'img'
699
- node.attrs = {
700
- src: 'data:image/svg+xml;utf8,' + src.replace(/#/g, '%23'),
701
- style,
702
- ignore: 'T'
703
- }
704
- node.children = undefined
705
- // #endif
706
- this.xml = false
707
- config.ignoreTags.style = true
708
- return
709
- }
710
-
711
- // #ifndef APP-PLUS-NVUE
712
- // 转换 align 属性
713
- if (attrs.align) {
714
- if (node.name === 'table') {
715
- if (attrs.align === 'center') {
716
- styleObj['margin-inline-start'] = styleObj['margin-inline-end'] = 'auto'
717
- } else {
718
- styleObj.float = attrs.align
719
- }
720
- } else {
721
- styleObj['text-align'] = attrs.align
722
- }
723
- attrs.align = undefined
724
- }
725
-
726
- // 转换 dir 属性
727
- if (attrs.dir) {
728
- styleObj.direction = attrs.dir
729
- attrs.dir = undefined
730
- }
731
-
732
- // 转换 font 标签的属性
733
- if (node.name === 'font') {
734
- if (attrs.color) {
735
- styleObj.color = attrs.color
736
- attrs.color = undefined
737
- }
738
- if (attrs.face) {
739
- styleObj['font-family'] = attrs.face
740
- attrs.face = undefined
741
- }
742
- if (attrs.size) {
743
- let size = parseInt(attrs.size)
744
- if (!isNaN(size)) {
745
- if (size < 1) {
746
- size = 1
747
- } else if (size > 7) {
748
- size = 7
749
- }
750
- styleObj['font-size'] = ['x-small', 'small', 'medium', 'large', 'x-large', 'xx-large', 'xxx-large'][size - 1]
751
- }
752
- attrs.size = undefined
753
- }
754
- }
755
- // #endif
756
-
757
- // 一些编辑器的自带 class
758
- if ((attrs.class || '').includes('align-center')) {
759
- styleObj['text-align'] = 'center'
760
- }
761
-
762
- Object.assign(styleObj, this.parseStyle(node))
763
-
764
- if (node.name !== 'table' && parseInt(styleObj.width) > windowWidth) {
765
- styleObj['max-width'] = '100%'
766
- styleObj['box-sizing'] = 'border-box'
767
- }
768
-
769
- // #ifndef APP-PLUS-NVUE
770
- if (config.blockTags[node.name]) {
771
- node.name = 'div'
772
- } else if (!config.trustTags[node.name] && !this.xml) {
773
- // 未知标签转为 span,避免无法显示
774
- node.name = 'span'
775
- }
776
-
777
- if (node.name === 'a' || node.name === 'ad'
778
- // #ifdef H5 || APP-PLUS
779
- || node.name === 'iframe' // eslint-disable-line
780
- // #endif
781
- ) {
782
- this.expose()
783
- } else if (node.name === 'video') {
784
- if ((styleObj.height || '').includes('auto')) {
785
- styleObj.height = undefined
786
- }
787
- /* #ifdef APP-PLUS */
788
- let str = '<video style="width:100%;height:100%"'
789
- for (const item in attrs) {
790
- if (attrs[item]) {
791
- str += ' ' + item + '="' + attrs[item] + '"'
792
- }
793
- }
794
- if (this.options.pauseVideo) {
795
- str += ' onplay="this.dispatchEvent(new CustomEvent(\'vplay\',{bubbles:!0}));for(var e=document.getElementsByTagName(\'video\'),t=0;t<e.length;t++)e[t]!=this&&e[t].pause()"'
796
- }
797
- str += '>'
798
- for (let i = 0; i < node.src.length; i++) {
799
- str += '<source src="' + node.src[i] + '">'
800
- }
801
- str += '</video>'
802
- node.html = str
803
- /* #endif */
804
- } else if ((node.name === 'ul' || node.name === 'ol') && node.c) {
805
- // 列表处理
806
- const types = {
807
- a: 'lower-alpha',
808
- A: 'upper-alpha',
809
- i: 'lower-roman',
810
- I: 'upper-roman'
811
- }
812
- if (types[attrs.type]) {
813
- attrs.style += ';list-style-type:' + types[attrs.type]
814
- attrs.type = undefined
815
- }
816
- for (let i = children.length; i--;) {
817
- if (children[i].name === 'li') {
818
- children[i].c = 1
819
- }
820
- }
821
- } else if (node.name === 'table') {
822
- // 表格处理
823
- // cellpadding、cellspacing、border 这几个常用表格属性需要通过转换实现
824
- let padding = parseFloat(attrs.cellpadding)
825
- let spacing = parseFloat(attrs.cellspacing)
826
- const border = parseFloat(attrs.border)
827
- const bordercolor = styleObj['border-color']
828
- const borderstyle = styleObj['border-style']
829
- if (node.c) {
830
- // padding 和 spacing 默认 2
831
- if (isNaN(padding)) {
832
- padding = 2
833
- }
834
- if (isNaN(spacing)) {
835
- spacing = 2
836
- }
837
- }
838
- if (border) {
839
- attrs.style += `;border:${border}px ${borderstyle || 'solid'} ${bordercolor || 'gray'}`
840
- }
841
- if (node.flag && node.c) {
842
- // 有 colspan 或 rowspan 且含有链接的表格通过 grid 布局实现
843
- styleObj.display = 'grid'
844
- if (spacing) {
845
- styleObj['grid-gap'] = spacing + 'px'
846
- styleObj.padding = spacing + 'px'
847
- } else if (border) {
848
- // 无间隔的情况下避免边框重叠
849
- attrs.style += ';border-left:0;border-top:0'
850
- }
851
-
852
- const width = [] // 表格的列宽
853
- const trList = [] // tr 列表
854
- const cells = [] // 保存新的单元格
855
- const map = {}; // 被合并单元格占用的格子
856
-
857
- (function traversal (nodes) {
858
- for (let i = 0; i < nodes.length; i++) {
859
- if (nodes[i].name === 'tr') {
860
- trList.push(nodes[i])
861
- } else {
862
- traversal(nodes[i].children || [])
863
- }
864
- }
865
- })(children)
866
-
867
- for (let row = 1; row <= trList.length; row++) {
868
- let col = 1
869
- for (let j = 0; j < trList[row - 1].children.length; j++) {
870
- const td = trList[row - 1].children[j]
871
- if (td.name === 'td' || td.name === 'th') {
872
- // 这个格子被上面的单元格占用,则列号++
873
- while (map[row + '.' + col]) {
874
- col++
875
- }
876
- let style = td.attrs.style || ''
877
- let start = style.indexOf('width') ? style.indexOf(';width') : 0
878
- // 提取出 td 的宽度
879
- if (start !== -1) {
880
- let end = style.indexOf(';', start + 6)
881
- if (end === -1) {
882
- end = style.length
883
- }
884
- if (!td.attrs.colspan) {
885
- width[col] = style.substring(start ? start + 7 : 6, end)
886
- }
887
- style = style.substr(0, start) + style.substr(end)
888
- }
889
- // 设置竖直对齐
890
- style += ';display:flex'
891
- start = style.indexOf('vertical-align')
892
- if (start !== -1) {
893
- const val = style.substr(start + 15, 10)
894
- if (val.includes('middle')) {
895
- style += ';align-items:center'
896
- } else if (val.includes('bottom')) {
897
- style += ';align-items:flex-end'
898
- }
899
- } else {
900
- style += ';align-items:center'
901
- }
902
- // 设置水平对齐
903
- start = style.indexOf('text-align')
904
- if (start !== -1) {
905
- const val = style.substr(start + 11, 10)
906
- if (val.includes('center')) {
907
- style += ';justify-content: center'
908
- } else if (val.includes('right')) {
909
- style += ';justify-content: right'
910
- }
911
- }
912
- style = (border ? `;border:${border}px ${borderstyle || 'solid'} ${bordercolor || 'gray'}` + (spacing ? '' : ';border-right:0;border-bottom:0') : '') + (padding ? `;padding:${padding}px` : '') + ';' + style
913
- // 处理列合并
914
- if (td.attrs.colspan) {
915
- style += `;grid-column-start:${col};grid-column-end:${col + parseInt(td.attrs.colspan)}`
916
- if (!td.attrs.rowspan) {
917
- style += `;grid-row-start:${row};grid-row-end:${row + 1}`
918
- }
919
- col += parseInt(td.attrs.colspan) - 1
920
- }
921
- // 处理行合并
922
- if (td.attrs.rowspan) {
923
- style += `;grid-row-start:${row};grid-row-end:${row + parseInt(td.attrs.rowspan)}`
924
- if (!td.attrs.colspan) {
925
- style += `;grid-column-start:${col};grid-column-end:${col + 1}`
926
- }
927
- // 记录下方单元格被占用
928
- for (let rowspan = 1; rowspan < td.attrs.rowspan; rowspan++) {
929
- for (let colspan = 0; colspan < (td.attrs.colspan || 1); colspan++) {
930
- map[(row + rowspan) + '.' + (col - colspan)] = 1
931
- }
932
- }
933
- }
934
- if (style) {
935
- td.attrs.style = style
936
- }
937
- cells.push(td)
938
- col++
939
- }
940
- }
941
- if (row === 1) {
942
- let temp = ''
943
- for (let i = 1; i < col; i++) {
944
- temp += (width[i] ? width[i] : 'auto') + ' '
945
- }
946
- styleObj['grid-template-columns'] = temp
947
- }
948
- }
949
- node.children = cells
950
- } else {
951
- // 没有使用合并单元格的表格通过 table 布局实现
952
- if (node.c) {
953
- styleObj.display = 'table'
954
- }
955
- if (!isNaN(spacing)) {
956
- styleObj['border-spacing'] = spacing + 'px'
957
- }
958
- if (border || padding) {
959
- // 遍历
960
- (function traversal (nodes) {
961
- for (let i = 0; i < nodes.length; i++) {
962
- const td = nodes[i]
963
- if (td.name === 'th' || td.name === 'td') {
964
- if (border) {
965
- td.attrs.style = `border:${border}px ${borderstyle || 'solid'} ${bordercolor || 'gray'};${td.attrs.style || ''}`
966
- }
967
- if (padding) {
968
- td.attrs.style = `padding:${padding}px;${td.attrs.style || ''}`
969
- }
970
- } else if (td.children) {
971
- traversal(td.children)
972
- }
973
- }
974
- })(children)
975
- }
976
- }
977
- // 给表格添加一个单独的横向滚动层
978
- if (this.options.scrollTable && !(attrs.style || '').includes('inline')) {
979
- const table = Object.assign({}, node)
980
- node.name = 'div'
981
- node.attrs = {
982
- style: 'overflow:auto'
983
- }
984
- node.children = [table]
985
- attrs = table.attrs
986
- }
987
- } else if ((node.name === 'td' || node.name === 'th') && (attrs.colspan || attrs.rowspan)) {
988
- for (let i = this.stack.length; i--;) {
989
- if (this.stack[i].name === 'table') {
990
- this.stack[i].flag = 1 // 指示含有合并单元格
991
- break
992
- }
993
- }
994
- } else if (node.name === 'ruby') {
995
- // 转换 ruby
996
- node.name = 'span'
997
- for (let i = 0; i < children.length - 1; i++) {
998
- if (children[i].type === 'text' && children[i + 1].name === 'rt') {
999
- children[i] = {
1000
- name: 'div',
1001
- attrs: {
1002
- style: 'display:inline-block;text-align:center'
1003
- },
1004
- children: [{
1005
- name: 'div',
1006
- attrs: {
1007
- style: 'font-size:50%;' + (children[i + 1].attrs.style || '')
1008
- },
1009
- children: children[i + 1].children
1010
- }, children[i]]
1011
- }
1012
- children.splice(i + 1, 1)
1013
- }
1014
- }
1015
- } else if (node.c) {
1016
- (function traversal (node) {
1017
- node.c = 2
1018
- for (let i = node.children.length; i--;) {
1019
- const child = node.children[i]
1020
- // #ifdef (MP-WEIXIN || MP-QQ || APP-PLUS || MP-360) && VUE3
1021
- if (child.name && (config.inlineTags[child.name] || ((child.attrs.style || '').includes('inline') && child.children)) && !child.c) {
1022
- traversal(child)
1023
- }
1024
- // #endif
1025
- if (!child.c || child.name === 'table') {
1026
- node.c = 1
1027
- }
1028
- }
1029
- })(node)
1030
- }
1031
-
1032
- if ((styleObj.display || '').includes('flex') && !node.c) {
1033
- for (let i = children.length; i--;) {
1034
- const item = children[i]
1035
- if (item.f) {
1036
- item.attrs.style = (item.attrs.style || '') + item.f
1037
- item.f = undefined
1038
- }
1039
- }
1040
- }
1041
- // flex 布局时部分样式需要提取到 rich-text 外层
1042
- const flex = parent && ((parent.attrs.style || '').includes('flex') || (parent.attrs.style || '').includes('grid'))
1043
- // #ifdef MP-WEIXIN
1044
- // 检查基础库版本 virtualHost 是否可用
1045
- && !(node.c && wx.getNFCAdapter) // eslint-disable-line
1046
- // #endif
1047
- // #ifndef MP-WEIXIN || MP-QQ || MP-BAIDU || MP-TOUTIAO
1048
- && !node.c // eslint-disable-line
1049
- // #endif
1050
- if (flex) {
1051
- node.f = ';max-width:100%'
1052
- }
1053
-
1054
- if (children.length >= 50 && node.c && !(styleObj.display || '').includes('flex')) {
1055
- mergeNodes(children)
1056
- }
1057
- // #endif
1058
-
1059
- for (const key in styleObj) {
1060
- if (styleObj[key]) {
1061
- const val = `;${key}:${styleObj[key].replace(' !important', '')}`
1062
- /* #ifndef APP-PLUS-NVUE */
1063
- if (flex && ((key.includes('flex') && key !== 'flex-direction') || key === 'align-self' || key.includes('grid') || styleObj[key][0] === '-' || (key.includes('width') && val.includes('%')))) {
1064
- node.f += val
1065
- if (key === 'width') {
1066
- attrs.style += ';width:100%'
1067
- }
1068
- } else /* #endif */ {
1069
- attrs.style += val
1070
- }
1071
- }
1072
- }
1073
- attrs.style = attrs.style.substr(1) || undefined
1074
- // #ifdef (MP-WEIXIN || MP-QQ) && VUE3
1075
- for (const key in attrs) {
1076
- if (!attrs[key]) {
1077
- delete attrs[key]
1078
- }
1079
- }
1080
- // #endif
1081
- }
1082
-
1083
- /**
1084
- * @description 解析到文本
1085
- * @param {String} text 文本内容
1086
- */
1087
- Parser.prototype.onText = function (text) {
1088
- if (!this.pre) {
1089
- // 合并空白符
1090
- let trim = ''
1091
- let flag
1092
- for (let i = 0, len = text.length; i < len; i++) {
1093
- if (!blankChar[text[i]]) {
1094
- trim += text[i]
1095
- } else {
1096
- if (trim[trim.length - 1] !== ' ') {
1097
- trim += ' '
1098
- }
1099
- if (text[i] === '\n' && !flag) {
1100
- flag = true
1101
- }
1102
- }
1103
- }
1104
- // 去除含有换行符的空串
1105
- if (trim === ' ') {
1106
- if (flag) return
1107
- // #ifdef VUE3
1108
- else {
1109
- const parent = this.stack[this.stack.length - 1]
1110
- if (parent && parent.name[0] === 't') return
1111
- }
1112
- // #endif
1113
- }
1114
- text = trim
1115
- }
1116
- const node = Object.create(null)
1117
- node.type = 'text'
1118
- // #ifdef (MP-BAIDU || MP-ALIPAY || MP-TOUTIAO) && VUE3
1119
- node.attrs = {}
1120
- // #endif
1121
- node.text = decodeEntity(text)
1122
- if (this.hook(node)) {
1123
- // #ifdef MP-WEIXIN
1124
- if (this.options.selectable === 'force' && system.includes('iOS') && !uni.canIUse('rich-text.user-select')) {
1125
- this.expose()
1126
- }
1127
- // #endif
1128
- const siblings = this.stack.length ? this.stack[this.stack.length - 1].children : this.nodes
1129
- siblings.push(node)
1130
- }
1131
- }
1132
-
1133
- /**
1134
- * @description html 词法分析器
1135
- * @param {Object} handler 高层处理器
1136
- */
1137
- function Lexer (handler) {
1138
- this.handler = handler
1139
- }
1140
-
1141
- /**
1142
- * @description 执行解析
1143
- * @param {String} content 要解析的文本
1144
- */
1145
- Lexer.prototype.parse = function (content) {
1146
- this.content = content || ''
1147
- this.i = 0 // 标记解析位置
1148
- this.start = 0 // 标记一个单词的开始位置
1149
- this.state = this.text // 当前状态
1150
- for (let len = this.content.length; this.i !== -1 && this.i < len;) {
1151
- this.state()
1152
- }
1153
- }
1154
-
1155
- /**
1156
- * @description 检查标签是否闭合
1157
- * @param {String} method 如果闭合要进行的操作
1158
- * @returns {Boolean} 是否闭合
1159
- * @private
1160
- */
1161
- Lexer.prototype.checkClose = function (method) {
1162
- const selfClose = this.content[this.i] === '/'
1163
- if (this.content[this.i] === '>' || (selfClose && this.content[this.i + 1] === '>')) {
1164
- if (method) {
1165
- this.handler[method](this.content.substring(this.start, this.i))
1166
- }
1167
- this.i += selfClose ? 2 : 1
1168
- this.start = this.i
1169
- this.handler.onOpenTag(selfClose)
1170
- if (this.handler.tagName === 'script') {
1171
- this.i = this.content.indexOf('</', this.i)
1172
- if (this.i !== -1) {
1173
- this.i += 2
1174
- this.start = this.i
1175
- }
1176
- this.state = this.endTag
1177
- } else {
1178
- this.state = this.text
1179
- }
1180
- return true
1181
- }
1182
- return false
1183
- }
1184
-
1185
- /**
1186
- * @description 文本状态
1187
- * @private
1188
- */
1189
- Lexer.prototype.text = function () {
1190
- this.i = this.content.indexOf('<', this.i) // 查找最近的标签
1191
- if (this.i === -1) {
1192
- // 没有标签了
1193
- if (this.start < this.content.length) {
1194
- this.handler.onText(this.content.substring(this.start, this.content.length))
1195
- }
1196
- return
1197
- }
1198
- const c = this.content[this.i + 1]
1199
- if ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z')) {
1200
- // 标签开头
1201
- if (this.start !== this.i) {
1202
- this.handler.onText(this.content.substring(this.start, this.i))
1203
- }
1204
- this.start = ++this.i
1205
- this.state = this.tagName
1206
- } else if (c === '/' || c === '!' || c === '?') {
1207
- if (this.start !== this.i) {
1208
- this.handler.onText(this.content.substring(this.start, this.i))
1209
- }
1210
- const next = this.content[this.i + 2]
1211
- if (c === '/' && ((next >= 'a' && next <= 'z') || (next >= 'A' && next <= 'Z'))) {
1212
- // 标签结尾
1213
- this.i += 2
1214
- this.start = this.i
1215
- this.state = this.endTag
1216
- return
1217
- }
1218
- // 处理注释
1219
- let end = '-->'
1220
- if (c !== '!' || this.content[this.i + 2] !== '-' || this.content[this.i + 3] !== '-') {
1221
- end = '>'
1222
- }
1223
- this.i = this.content.indexOf(end, this.i)
1224
- if (this.i !== -1) {
1225
- this.i += end.length
1226
- this.start = this.i
1227
- }
1228
- } else {
1229
- this.i++
1230
- }
1231
- }
1232
-
1233
- /**
1234
- * @description 标签名状态
1235
- * @private
1236
- */
1237
- Lexer.prototype.tagName = function () {
1238
- if (blankChar[this.content[this.i]]) {
1239
- // 解析到标签名
1240
- this.handler.onTagName(this.content.substring(this.start, this.i))
1241
- while (blankChar[this.content[++this.i]]);
1242
- if (this.i < this.content.length && !this.checkClose()) {
1243
- this.start = this.i
1244
- this.state = this.attrName
1245
- }
1246
- } else if (!this.checkClose('onTagName')) {
1247
- this.i++
1248
- }
1249
- }
1250
-
1251
- /**
1252
- * @description 属性名状态
1253
- * @private
1254
- */
1255
- Lexer.prototype.attrName = function () {
1256
- let c = this.content[this.i]
1257
- if (blankChar[c] || c === '=') {
1258
- // 解析到属性名
1259
- this.handler.onAttrName(this.content.substring(this.start, this.i))
1260
- let needVal = c === '='
1261
- const len = this.content.length
1262
- while (++this.i < len) {
1263
- c = this.content[this.i]
1264
- if (!blankChar[c]) {
1265
- if (this.checkClose()) return
1266
- if (needVal) {
1267
- // 等号后遇到第一个非空字符
1268
- this.start = this.i
1269
- this.state = this.attrVal
1270
- return
1271
- }
1272
- if (this.content[this.i] === '=') {
1273
- needVal = true
1274
- } else {
1275
- this.start = this.i
1276
- this.state = this.attrName
1277
- return
1278
- }
1279
- }
1280
- }
1281
- } else if (!this.checkClose('onAttrName')) {
1282
- this.i++
1283
- }
1284
- }
1285
-
1286
- /**
1287
- * @description 属性值状态
1288
- * @private
1289
- */
1290
- Lexer.prototype.attrVal = function () {
1291
- const c = this.content[this.i]
1292
- const len = this.content.length
1293
- if (c === '"' || c === "'") {
1294
- // 有冒号的属性
1295
- this.start = ++this.i
1296
- this.i = this.content.indexOf(c, this.i)
1297
- if (this.i === -1) return
1298
- this.handler.onAttrVal(this.content.substring(this.start, this.i))
1299
- } else {
1300
- // 没有冒号的属性
1301
- for (; this.i < len; this.i++) {
1302
- if (blankChar[this.content[this.i]]) {
1303
- this.handler.onAttrVal(this.content.substring(this.start, this.i))
1304
- break
1305
- } else if (this.checkClose('onAttrVal')) return
1306
- }
1307
- }
1308
- while (blankChar[this.content[++this.i]]);
1309
- if (this.i < len && !this.checkClose()) {
1310
- this.start = this.i
1311
- this.state = this.attrName
1312
- }
1313
- }
1314
-
1315
- /**
1316
- * @description 结束标签状态
1317
- * @returns {String} 结束的标签名
1318
- * @private
1319
- */
1320
- Lexer.prototype.endTag = function () {
1321
- const c = this.content[this.i]
1322
- if (blankChar[c] || c === '>' || c === '/') {
1323
- this.handler.onCloseTag(this.content.substring(this.start, this.i))
1324
- if (c !== '>') {
1325
- this.i = this.content.indexOf('>', this.i)
1326
- if (this.i === -1) return
1327
- }
1328
- this.start = ++this.i
1329
- this.state = this.text
1330
- } else {
1331
- this.i++
1332
- }
1333
- }
1334
-
1335
- export default Parser