richtext-editor.vue 31 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153
  1. <template>
  2. <view class="container">
  3. <!-- 编辑器内容@focus="modeShow" -->
  4. <view class="main">
  5. <!-- //如果不需要跟随键盘弹出注释:style="{ 'height': editorHeight + 'px' }" mode默认改为true即可 -->
  6. <editor @focus="modeShow" :style="{ 'height': editorHeight + 'px' }" @statuschange="selected" :mode="false"
  7. class="myEditor" placeholder="写点什么儿~" show-img-size show-img-toolbar show-img-resize
  8. @ready="onEditorReady" @blur="onsuccess"></editor>
  9. </view>
  10. <!-- 模块 -->
  11. <view class="modes">
  12. <!-- 功能选项 :style="{position: 'relative',bottom:keyboardHeight+'px'}"-->
  13. <view class="tabName hflex acenter jbetween" @click="ontabtap">
  14. <view class="hflex acenter">
  15. <image @click="changeImg" v-show="!img" src="@/static/richtext/img.png" :data-current="3"
  16. mode="aspectFill"></image>
  17. <image @click="changeImg" v-show="img" src="@/static/richtext/imgselect.png" :data-current="3"
  18. mode="aspectFill">
  19. </image>
  20. <image @click="changeFont" v-show="!font" src="@/static/richtext/font.png" :data-current="1"
  21. mode="aspectFill"></image>
  22. <image @click="changeFont" v-show="font" src="@/static/richtext/fontselect.png" :data-current="1"
  23. mode="aspectFill">
  24. </image>
  25. </view>
  26. <!-- <image @click="changeAlign" v-show="!align" src="@/static/richtext/leftcenter.png" :data-current="2" mode="">
  27. </image>
  28. <image @click="changeAlign" v-show="align" src="@/static/richtext/leftcenterSelect.png" :data-current="2"
  29. mode=""></image> -->
  30. <view class="hflex acenter">
  31. <image @click="repeal" src="@/static/richtext/backone.png" mode=""></image>
  32. <image @click="recover" src="@/static/richtext/afterone.png" mode=""></image>
  33. </view>
  34. <!-- <image @click="modeShow" src="@/static/richtext/keyboard.png" mode=""></image> -->
  35. </view>
  36. <!-- 功能选项内容 -->
  37. <view class="tab_card" v-show="tabIndex==1 && font==true">
  38. <!-- 其他 -->
  39. <view class="one">
  40. <view @click="addBold" class="oneitem" :style="Bold?'background-color: #ffffff;':''">
  41. <image src="@/static/richtext/bold.png" mode=""></image>
  42. </view>
  43. <view @click="bias" class="oneitem" :style="slash?'background-color: #ffffff;':''">
  44. <image src="@/static/richtext/incline.png" mode=""></image>
  45. </view>
  46. <view @click="addDivider" class="oneitem" :style="halvingline?'background-color: #ffffff;':''">
  47. <image src="@/static/richtext/halvingline.png" mode=""></image>
  48. </view>
  49. </view>
  50. <!-- H2 -->
  51. <view class="one">
  52. <view @click="Hfirst" class="oneitem" :style="H1?'background-color: #ffffff;':''">
  53. <image src="@/static/richtext/h-1.png" mode=""></image>
  54. </view>
  55. <view @click="Htwo" class="oneitem" :style="H2?'background-color: #ffffff;':''">
  56. <image src="@/static/richtext/h-2.png" mode=""></image>
  57. </view>
  58. <view @click="Hthree" class="oneitem" :style="H3?'background-color: #ffffff;':''">
  59. <image src="@/static/richtext/h-3.png" mode=""></image>
  60. </view>
  61. <view @click="Hfour" class="oneitem" :style="H4?'background-color: #ffffff;':''">
  62. <image src="@/static/richtext/h-4.png" mode=""></image>
  63. </view>
  64. </view>
  65. <!-- 字号 单位px-->
  66. <view class="three">
  67. <view @click="thirtySix" class="threeitem" :style="thirtysix?'background-color: #ffffff;':''">
  68. <text style="font-weight: 700;font-size: 38upx;">36</text>
  69. </view>
  70. <view @click="thirtyTwo" class="threeitem" :style="thirtytwo?'background-color: #ffffff;':''">
  71. <text style="font-weight: 600;font-size: 36upx;">32</text>
  72. </view>
  73. <view @click="twentyEight" class="threeitem" :style="twentyeight?'background-color: #ffffff;':''">
  74. <text style="font-size: 34upx;">28</text>
  75. </view>
  76. <view @click="twentySix" class="threeitem" :style="twentysix?'background-color: #ffffff;':''">
  77. <text style="font-size: 32upx;">26</text>
  78. </view>
  79. <view @click="twentyFour" class="threeitem" :style="twentyfour?'background-color: #ffffff;':''">
  80. <text style="font-size: 28upx;">24</text>
  81. </view>
  82. <view @click="Twenty" class="threeitem" :style="twenty?'background-color: #ffffff;':''">
  83. <text style="font-size: 26upx;">20</text>
  84. </view>
  85. </view>
  86. <!-- 字体颜色 -->
  87. <!-- <view class="four">
  88. <view @click="Black" class="fouritema" :style="black?'border: 4upx solid black;':''">
  89. <text></text>
  90. </view>
  91. <view @click="Red" class="fouritemb" :style="red?'border: 4upx solid #92a5c7;':''">
  92. <text></text>
  93. </view>
  94. <view @click="Gray" class="fouritemc" :style="gray?'border: 4upx solid #92a5c7;':''">
  95. <text></text>
  96. </view>
  97. <view @click="Blue" class="fouritemd" :style="blue?'border: 4upx solid #55aaff;':''">
  98. <text></text>
  99. </view>
  100. <view @click="Orange" class="fouriteme" :style="orange?'border: 4upx solid orange;':''">
  101. <text></text>
  102. </view>
  103. <view @click="Green" class="fouritemf" :style="green?'border: 4upx solid green;':''">
  104. <text></text>
  105. </view>
  106. </view> -->
  107. <!-- 对齐方式 -->
  108. <view class="one">
  109. <view class="oneitem" @click="alignLeft" :style="leftcenter?'background-color: #ffffff;':''">
  110. <image src="@/static/richtext/leftcenter.png" mode=""></image>
  111. </view>
  112. <view class="oneitem" @click="alignCenter" :style="center?'background-color: #ffffff;':''">
  113. <image src="@/static/richtext/center.png" mode=""></image>
  114. </view>
  115. <view class="oneitem" @click="alignright" :style="rightcenter?'background-color: #ffffff;':''">
  116. <image src="@/static/richtext/rightcenter.png" mode=""></image>
  117. </view>
  118. <view class="oneitem" @click="Sort" :style="underline?'background-color: #ffffff;':''">
  119. <image src="@/static/richtext/underline.png" mode=""></image>
  120. </view>
  121. <view class="oneitem" @click="Refine" :style="refine?'background-color: #ffffff;':''">
  122. <image src="@/static/richtext/dot.png" mode=""></image>
  123. </view>
  124. <view class="oneitem" @click="Check" :style="check?'background-color: #ffffff;':''">
  125. <image src="@/static/richtext/addrefine.png" mode=""></image>
  126. </view>
  127. </view>
  128. </view>
  129. <view class="tab_card" v-show="tabIndex==2 && align==true">
  130. <!-- 对齐方式 -->
  131. <view class="one">
  132. <view class="oneitem" @click="alignLeft" :style="leftcenter?'background-color: #ffffff;':''">
  133. <image src="@/static/richtext/leftcenter.png" mode=""></image>
  134. </view>
  135. <view class="oneitem" @click="alignCenter" :style="center?'background-color: #ffffff;':''">
  136. <image src="@/static/richtext/center.png" mode=""></image>
  137. </view>
  138. <view class="oneitem" @click="alignright" :style="rightcenter?'background-color: #ffffff;':''">
  139. <image src="@/static/richtext/rightcenter.png" mode=""></image>
  140. </view>
  141. <view class="oneitem" @click="Sort" :style="underline?'background-color: #ffffff;':''">
  142. <image src="@/static/richtext/underline.png" mode=""></image>
  143. </view>
  144. <view class="oneitem" @click="Refine" :style="refine?'background-color: #ffffff;':''">
  145. <image src="@/static/richtext/dot.png" mode=""></image>
  146. </view>
  147. <view class="oneitem" @click="Check" :style="check?'background-color: #ffffff;':''">
  148. <image src="@/static/richtext/addrefine.png" mode=""></image>
  149. </view>
  150. </view>
  151. <view class="two">
  152. <view class="twoitem">
  153. <image src="@/static/richtext/lineheight.png" mode=""></image>
  154. </view>
  155. <view @click="lineHeightone" class="twoitem" :style="heightone?'background-color: #ffffff;':''">
  156. <text>1</text>
  157. </view>
  158. <view @click="lineHeightonea" class="twoitem" :style="heightonea?'background-color: #ffffff;':''">
  159. <text>1.5</text>
  160. </view>
  161. <view @click="lineHeighttwo" class="twoitem" :style="heighttwo?'background-color: #ffffff;':''">
  162. <text>2</text>
  163. </view>
  164. <view @click="lineHeighttwoa" class="twoitem" :style="heighttwoa?'background-color: #ffffff;':''">
  165. <text>2.5</text>
  166. </view>
  167. <view @click="lineHeightthree" class="twoitem" :style="heightthree?'background-color: #ffffff;':''">
  168. <text>3</text>
  169. </view>
  170. </view>
  171. <!-- 背景色 -->
  172. <view class="fours">
  173. <view @click="Blackback" class="foursitema" :style="blackback?'border: 4upx solid black;':''">
  174. <text></text>
  175. </view>
  176. <view @click="Redback" class="foursitemb" :style="redback?'border: 4upx solid red;':''">
  177. <text></text>
  178. </view>
  179. <view @click="Grayback" class="foursitemc" :style="grayback?'border: 4upx solid #92a5c7;':''">
  180. <text></text>
  181. </view>
  182. <view @click="Blueback" class="foursitemd" :style="blueback?'border: 4upx solid #55aaff;':''">
  183. <text></text>
  184. </view>
  185. <view @click="Orangeback" class="foursiteme" :style="orangeback?'border: 4upx solid orange;':''">
  186. <text></text>
  187. </view>
  188. <view @click="Green" class="foursitemf" :style="greenback?'border: 4upx solid green;':''">
  189. <text></text>
  190. </view>
  191. </view>
  192. </view>
  193. <!-- 图片 -->
  194. <view class="tab_cardimg" v-show="tabIndex==3 && img==true">
  195. <view class="mo">
  196. <view class="moitem" @click="insertImg">
  197. <image src="@/static/richtext/img.png" mode=""></image>
  198. <text>来自图库</text>
  199. </view>
  200. <!-- <view class="moitem">
  201. <image src="@/static/richtext/photograph.png" mode=""></image>
  202. <text>拍照</text>
  203. </view> -->
  204. <!-- @click="insertVideo" -->
  205. <!-- <view class="moitem" @click="insertVideo">
  206. <image src="@/static/richtext/insertvideo.png" mode=""></image>
  207. <text>插入视频</text>
  208. </view> -->
  209. </view>
  210. </view>
  211. </view>
  212. <!-- 利用rich-text来回显html内容 data是富文本编辑器html的内容-->
  213. <!-- <rich-text class="ql-editor" :nodes="data"></rich-text> -->
  214. <!-- uparse解析 -->
  215. <!-- <u-parse :content="data" @preview="preview" @navigate="navigate"></u-parse> -->
  216. </view>
  217. </template>
  218. <script>
  219. import $api from '@/static/js/api.js'
  220. export default {
  221. props: {
  222. init: {
  223. typeof: String,
  224. default: ''
  225. }
  226. },
  227. data() {
  228. return {
  229. mode: false, //显示模块 //如果不需要跟随键盘弹出 改为true默认显示
  230. tabIndex: 1, //下标
  231. Height: 0, //高度
  232. font: true, //字体选项卡
  233. align: false, //对齐方式选项卡
  234. img: false, //图片选项卡
  235. halvingline: false, //下划线
  236. slash: false, //倾斜
  237. Bold: false, //变量
  238. H1: false, //H1
  239. H2: false, //H2
  240. H3: false, //H3
  241. H4: false, //H4
  242. thirtysix: false, //36字号px
  243. thirtytwo: false, //32字号px
  244. twentyeight: false, //28字号px
  245. twentysix: false, //26字号px
  246. twentyfour: false, //24字号px
  247. twenty: false, //20字号px
  248. black: true, //黑色字体
  249. red: false, //红色字体
  250. gray: false, //灰蓝色字体
  251. blue: false, //浅蓝色字体
  252. orange: false, //橙色字体
  253. green: false, //绿色字体
  254. blackback: true, //黑色背景
  255. redback: false, //红色字体
  256. grayback: false, //灰蓝色字体
  257. blueback: false, //浅蓝色字体
  258. orangeback: false, //橙色字体
  259. greenback: false, //绿色字体
  260. center: false, //居中
  261. leftcenter: false, //居左
  262. rightcenter: false, //居右
  263. underline: false, //排序
  264. refine: false, //加圆点
  265. check: false, //加方格
  266. spacingone: false, //字母间距1
  267. heightone: false, //行高1
  268. heightonea: false, //行高1.5
  269. heighttwo: false, //行高2
  270. heighttwoa: false, //行高2.5
  271. heightthree: false, //行高3
  272. editorCtx: {},
  273. //键盘高度
  274. keyboardHeight: 0,
  275. editorHeight: 0,
  276. data: ""
  277. }
  278. },
  279. mounted() {
  280. var _this = this
  281. uni.getSystemInfo({
  282. success: function(res) {
  283. _this.Height = res.windowHeight;
  284. }
  285. });
  286. },
  287. methods: {
  288. onsuccess() {
  289. var that = this
  290. that.editorCtx.getContents({
  291. success: function(data) {
  292. that.data = data.html
  293. that.$emit('onsuccess', that.data)
  294. },
  295. })
  296. },
  297. modeShow() { //
  298. this.mode = !this.mode
  299. //如果不需要跟随键盘弹出注释即可
  300. var _this = this
  301. // 监听键盘高度变化
  302. uni.onKeyboardHeightChange((obj) => {
  303. _this.keyboardHeight = obj
  304. _this.editorHeight = _this.Height - obj - 30
  305. })
  306. },
  307. ontabtap(e) {
  308. let index = e.target.dataset.current || e.currentTarget.dataset.current;
  309. this.switchTab(index);
  310. this.tabIndex = index;
  311. },
  312. switchTab(index) {
  313. if (this.tabIndex == index) {
  314. return
  315. }
  316. this.tabIndex = index;
  317. },
  318. onEditorReady(e) { //初始化
  319. this.data = this.init
  320. uni.createSelectorQuery().in(this).select('.myEditor').fields({
  321. context: true
  322. }, res => {
  323. this.editorCtx = res.context
  324. res.context.setContents({
  325. html: this.data
  326. })
  327. }).exec()
  328. },
  329. changeFont() { //字体选项卡
  330. this.font = !this.font
  331. this.align = false
  332. this.img = false
  333. },
  334. changeAlign() { //对齐方式选项卡
  335. this.align = !this.align
  336. this.font = false
  337. this.img = false
  338. },
  339. changeImg() { //图片选项卡
  340. this.img = !this.img
  341. this.font = false
  342. this.align = false
  343. },
  344. addDivider() { //插入分割线 halvingline
  345. this.halvingline = !this.halvingline //改变图标颜色
  346. this.editorCtx.insertDivider()
  347. this.editorCtx.getContents()
  348. },
  349. repeal() { //撤销
  350. this.editorCtx.undo()
  351. },
  352. recover() { //恢复
  353. this.editorCtx.redo()
  354. },
  355. addBold() { //字体加粗
  356. this.Bold = !this.Bold //变量
  357. this.editorCtx.format('bold', 'bold')
  358. },
  359. bias() { //字体倾斜
  360. this.slash = !this.slash //改变图标颜色
  361. this.editorCtx.format('italic')
  362. },
  363. alignCenter() { //居中对齐
  364. this.center = !this.center //改变图标颜色
  365. this.leftcenter = false //改变图标颜色
  366. this.rightcenter = false //改变图标颜色
  367. this.editorCtx.format('align', 'center') //属性,值
  368. },
  369. alignLeft() { //居左对齐
  370. this.leftcenter = !this.leftcenter //改变图标颜色
  371. this.center = false //改变图标颜色
  372. this.rightcenter = false //改变图标颜色
  373. this.editorCtx.format('align', 'left') //属性,值
  374. },
  375. alignright() { //居右对齐
  376. this.rightcenter = !this.rightcenter //改变图标颜色
  377. this.center = false //改变图标颜色
  378. this.leftcenter = false //改变图标颜色
  379. this.editorCtx.format('align', 'right') //属性,值
  380. },
  381. Sort() { //排序
  382. this.underline = !this.underline //改变图标颜色
  383. this.editorCtx.format('list', 'ordered') //属性,值 ordered 排序 bullet 点 check 方格
  384. },
  385. Refine() { //加圆点
  386. this.refine = !this.refine //改变图标颜色
  387. this.editorCtx.format('list', 'bullet') //属性,值 ordered 排序 bullet 点 check 方格
  388. },
  389. Check() { //加方格
  390. this.check = !this.check //改变图标颜色
  391. this.editorCtx.format('list', 'check') //属性,值 ordered 排序 bullet 点 check 方格
  392. },
  393. Hfirst() { //标题字体变大
  394. this.H1 = !this.H1
  395. if (this.H1) {
  396. this.editorCtx.format('header', 'H1')
  397. } else if (!this.H1) {
  398. this.editorCtx.format('header', '0')
  399. }
  400. },
  401. Htwo() { //标题字体变大
  402. this.H2 = !this.H2
  403. if (this.H2) {
  404. this.editorCtx.format('header', 'H2')
  405. } else if (!this.H2) {
  406. this.editorCtx.format('header', '0')
  407. }
  408. },
  409. Hthree() { //标题字体变大
  410. this.H3 = !this.H3
  411. if (this.H3) {
  412. this.editorCtx.format('header', 'H3')
  413. } else if (!this.H3) {
  414. this.editorCtx.format('header', '0')
  415. }
  416. },
  417. Hfour() { //标题字体变大
  418. this.H4 = !this.H4
  419. if (this.H4) {
  420. this.editorCtx.format('header', 'H4')
  421. } else if (!this.H4) {
  422. this.editorCtx.format('header', '0')
  423. }
  424. },
  425. thirtySix() { //36
  426. this.thirtysix = !this.thirtysix
  427. if (this.thirtysix) {
  428. this.editorCtx.format('fontSize', '36px')
  429. } else if (!this.thirtysix) {
  430. this.editorCtx.format('fontSize', '')
  431. }
  432. },
  433. thirtyTwo() { //32
  434. this.thirtytwo = !this.thirtytwo
  435. if (this.thirtytwo) {
  436. this.editorCtx.format('fontSize', '32px')
  437. } else if (!this.thirtytwo) {
  438. this.editorCtx.format('fontSize', '')
  439. }
  440. },
  441. twentyEight() { //28
  442. this.twentyeight = !this.twentyeight
  443. if (this.twentyeight) {
  444. this.editorCtx.format('fontSize', '28px')
  445. } else if (!this.twentyeight) {
  446. this.editorCtx.format('fontSize', '')
  447. }
  448. },
  449. twentySix() { //26
  450. this.twentysix = !this.twentysix
  451. if (this.twentysix) {
  452. this.editorCtx.format('fontSize', '26px')
  453. } else if (!this.twentysix) {
  454. this.editorCtx.format('fontSize', '')
  455. }
  456. },
  457. twentyFour() { //24
  458. this.twentyfour = !this.twentyfour
  459. if (this.twentyfour) {
  460. this.editorCtx.format('fontSize', '24px')
  461. } else if (!this.twentyfour) {
  462. this.editorCtx.format('fontSize', '')
  463. }
  464. },
  465. Twenty() { //20
  466. this.twenty = !this.twenty
  467. if (this.twenty) {
  468. this.editorCtx.format('fontSize', '20px')
  469. } else if (!this.twenty) {
  470. this.editorCtx.format('fontSize', '')
  471. }
  472. },
  473. spacingOne() { //字母间距1
  474. this.spacingone = !this.spacingone
  475. if (this.spacingone) {
  476. this.editorCtx.format('letterSpacing', '100px')
  477. } else if (!this.spacingone) {
  478. this.editorCtx.format('letterSpacing', '')
  479. }
  480. },
  481. lineHeightone() { //行高1
  482. this.heightone = !this.heightone
  483. if (this.heightone) {
  484. this.editorCtx.format('lineHeight', '20px') //20upx
  485. } else if (!this.heightone) {
  486. this.editorCtx.format('lineHeight', '')
  487. }
  488. },
  489. lineHeightonea() { //行高1.5
  490. this.heightonea = !this.heightonea
  491. // 当点击1.5时取消其他行高 根据需求自行更改 其他功能同理
  492. this.heightone = false
  493. this.heighttwo = false
  494. this.heighttwoa = false
  495. this.heightthree = false
  496. if (this.heightonea) {
  497. this.editorCtx.format('lineHeight', '30px') //20upx
  498. } else if (!this.heightonea) {
  499. this.editorCtx.format('lineHeight', '')
  500. }
  501. },
  502. lineHeighttwo() { //行高2
  503. this.heighttwo = !this.heighttwo
  504. if (this.heighttwo) {
  505. this.editorCtx.format('lineHeight', '40px') //20upx
  506. } else if (!this.heighttwo) {
  507. this.editorCtx.format('lineHeight', '')
  508. }
  509. },
  510. lineHeighttwoa() { //行高2.5
  511. this.heighttwoa = !this.heighttwoa
  512. if (this.heighttwoa) {
  513. this.editorCtx.format('lineHeight', '50px') //50upx
  514. } else if (!this.heighttwoa) {
  515. this.editorCtx.format('lineHeight', '')
  516. }
  517. },
  518. lineHeightthree() { //行高3
  519. this.heightthree = !this.heightthree
  520. if (this.heightthree) {
  521. this.editorCtx.format('lineHeight', '60px') //50upx
  522. } else if (!this.heightthree) {
  523. this.editorCtx.format('lineHeight', '')
  524. }
  525. },
  526. Black() { //黑色字体
  527. this.black = !this.black
  528. if (this.black) {
  529. this.editorCtx.format('color', 'black')
  530. } else if (!this.black) {
  531. this.editorCtx.format('color', '')
  532. }
  533. },
  534. Red() { //黑色字体
  535. this.red = !this.red
  536. if (this.red) {
  537. this.editorCtx.format('color', 'red')
  538. } else if (!this.red) {
  539. this.editorCtx.format('color', '')
  540. }
  541. },
  542. Gray() { //灰蓝色字体
  543. this.gray = !this.gray
  544. if (this.gray) {
  545. this.editorCtx.format('color', '#92a5c7')
  546. } else if (!this.gray) {
  547. this.editorCtx.format('color', '')
  548. }
  549. },
  550. Blue() { //浅蓝色字体
  551. this.blue = !this.blue
  552. if (this.blue) {
  553. this.editorCtx.format('color', '#00aaff')
  554. } else if (!this.blue) {
  555. this.editorCtx.format('color', '')
  556. }
  557. },
  558. Orange() { //橙色字体
  559. this.orange = !this.orange
  560. if (this.orange) {
  561. this.editorCtx.format('color', 'orange')
  562. } else if (!this.orange) {
  563. this.editorCtx.format('color', '')
  564. }
  565. },
  566. Green() { //绿色字体
  567. this.green = !this.green
  568. if (this.green) {
  569. this.editorCtx.format('color', 'green')
  570. } else if (!this.green) {
  571. this.editorCtx.format('color', '')
  572. }
  573. },
  574. Blackback() { //黑色背景
  575. this.blackback = !this.blackback
  576. if (this.blackback) {
  577. this.editorCtx.format('backgroundColor', 'black')
  578. } else if (!this.blackback) {
  579. this.editorCtx.format('backgroundColor', '')
  580. }
  581. },
  582. Redback() { //黑色背景
  583. this.redback = !this.redback
  584. if (this.redback) {
  585. this.editorCtx.format('backgroundColor', 'red')
  586. } else if (!this.redback) {
  587. this.editorCtx.format('backgroundColor', '')
  588. }
  589. },
  590. Grayback() { //灰蓝色背景
  591. this.grayback = !this.grayback
  592. if (this.grayback) {
  593. this.editorCtx.format('backgroundColor', '#92a5c7')
  594. } else if (!this.grayback) {
  595. this.editorCtx.format('backgroundColor', '')
  596. }
  597. },
  598. Blueback() { //浅蓝色字体
  599. this.blueback = !this.blueback
  600. if (this.blueback) {
  601. this.editorCtx.format('backgroundColor', '#00aaff')
  602. } else if (!this.blueback) {
  603. this.editorCtx.format('backgroundColor', '')
  604. }
  605. },
  606. Orangeback() { //橙色字体
  607. this.orangeback = !this.orangeback
  608. if (this.orangeback) {
  609. this.editorCtx.format('backgroundColor', 'orange')
  610. } else if (!this.orangeback) {
  611. this.editorCtx.format('backgroundColor', '')
  612. }
  613. },
  614. Greenback() { //绿色背景
  615. this.greenback = !this.greenback
  616. if (this.greenback) {
  617. this.editorCtx.format('backgroundColor', 'green')
  618. } else if (!this.greenback) {
  619. this.editorCtx.format('backgroundColor', '')
  620. }
  621. },
  622. selected() { //选中内容
  623. this.editorCtx.getSelectionText()
  624. },
  625. insertImg() { //插入图片
  626. var _this = this
  627. uni.chooseImage({
  628. success: (chooseImageRes) => {
  629. uni.showLoading({
  630. title: '上传中...'
  631. })
  632. const tempFilePaths = chooseImageRes.tempFilePaths;
  633. uni.uploadFile({
  634. url: $api.config.baseUrl + 'upload/image', //仅为示例,非真实的接口地址
  635. filePath: tempFilePaths[0],
  636. name: 'image',
  637. // formData: {
  638. // 'user': 'test'
  639. // },
  640. success: (uploadFileRes) => {
  641. let res = JSON.parse(uploadFileRes.data)
  642. for (let i = 0; i < tempFilePaths.length; i++) {
  643. _this.editorCtx.insertImage({
  644. src: res.data.url //图片路径 换成服务器地址
  645. })
  646. }
  647. uni.hideLoading()
  648. }
  649. });
  650. }
  651. });
  652. /* uni.chooseImage({
  653. success: res => {
  654. uni.showLoading({
  655. title: 'loading...'
  656. })
  657. //在这里先把封面截图保存到服务器拿到服务器存储地址放在这里
  658. for (let i = 0; i < res.tempFilePaths.length; i++) {
  659. this.editorCtx.insertImage({
  660. src: res.tempFilePaths[i] //图片路径 换成服务器地址
  661. })
  662. }
  663. uni.hideLoading()
  664. }
  665. }) */
  666. },
  667. insertVideo() { //视频 //uni.chooseMedia()
  668. let that = this
  669. uni.chooseMedia({
  670. count: 1,
  671. mediaType: ['video'],
  672. sourceType: ['album', 'camera'],
  673. maxDuration: 30,
  674. camera: 'back',
  675. success(res) {
  676. that.src = res.tempFiles[0].tempFilePath
  677. let imgArr = res.tempFiles[0].thumbTempFilePath
  678. //在这里先把封面截图保存到服务器拿到服务器存储地址放在这里
  679. that.editorCtx.insertImage({
  680. //https://mp-2a0f3e44-b47a-4bd5-a925-ba63d3e2a8ab.cdn.bspapp.com/add_img/1693998858778-2023-08-21 133334.png
  681. src: 'https://mp-2a0f3e44-b47a-4bd5-a925-ba63d3e2a8ab.cdn.bspapp.com/add_img/1693998858778-2023-08-21 133334.png', //视频封面图 //服务器地址
  682. alt: '图片出差了',
  683. extClass: 'videothumb',
  684. })
  685. that.getCtx()
  686. }
  687. })
  688. },
  689. // 获取编辑器内容
  690. getCtx() {
  691. var that = this
  692. that.editorCtx.getContents({
  693. success: function(data) {
  694. that.data = data.html
  695. //需要把编辑器内的图片/视频地址更换成储存在服务器后的图片/视频路径
  696. //在要显示的页面,利用rich-text来回显html内容 data是富文本编辑器html的内容
  697. // <rich-text class="ql-editor" :nodes="data"></rich-text>
  698. //这里把编辑器里的视频封面截图替换成了视频
  699. //uniapp富文本编辑器内不支持video无法显示视频,rich-text可以显示但需要uparse插件解析
  700. //https://ext.dcloud.net.cn/plugin?id=183 这里有详细教程
  701. // let reg = /<img[^>]*class=['videothumb"]([^'"]+)[^>]*>/g
  702. // that.data = data.html.replace(reg,
  703. // '<video id="myVideo" class="videosrc" src="https://mp-2a0f3e44-b47a-4bd5-a925-ba63d3e2a8ab.cdn.bspapp.com/add_file/2023-09-17 172002.mp4"></video>'
  704. // );
  705. }
  706. })
  707. },
  708. }
  709. }
  710. </script>
  711. <style lang="scss">
  712. /* 样式二 */
  713. .tabName {
  714. width: 100%;
  715. height: 60upx;
  716. line-height: 60upx;
  717. padding: 15upx 28rpx;
  718. border-top: 1upx solid #d9e4f9;
  719. border-bottom: 1upx solid #d9e4f9;
  720. image {
  721. width: 40rpx;
  722. height: 40rpx;
  723. margin-right: 52rpx;
  724. }
  725. }
  726. /* .tabName>image {
  727. width: 40upx;
  728. height: 40upx;
  729. } */
  730. /* 内容高度根据需求自定义 */
  731. .tab_card {
  732. width: 750upx;
  733. height: 430upx;
  734. /* margin-top: 10upx; */
  735. background-color: #ffffff;
  736. text-align: center;
  737. line-height: 430upx;
  738. padding: 5upx 0;
  739. }
  740. .tab_card>.one {
  741. width: 690upx;
  742. height: 60upx;
  743. line-height: 60upx;
  744. margin: 25upx 30upx;
  745. display: flex;
  746. justify-content: space-around;
  747. align-items: center;
  748. background-color: #edf3fa;
  749. padding: 12upx 0;
  750. }
  751. .tab_card>.one>.oneitem {
  752. width: 90upx;
  753. height: 60upx;
  754. line-height: 60upx;
  755. border-radius: 5upx;
  756. }
  757. .tab_card>.one>.oneitem>image {
  758. width: 50upx;
  759. height: 50upx;
  760. margin-top: 5upx;
  761. }
  762. /* 行高 */
  763. .tab_card>.two {
  764. width: 690upx;
  765. height: 60upx;
  766. line-height: 60upx;
  767. margin: 25upx 30upx;
  768. display: flex;
  769. justify-content: space-around;
  770. align-items: center;
  771. background-color: #edf3fa;
  772. padding: 12upx 0;
  773. }
  774. .tab_card>.two>.twoitem {
  775. width: 90upx;
  776. height: 60upx;
  777. line-height: 60upx;
  778. border-radius: 5upx;
  779. }
  780. .tab_card>.two>.twoitem>image {
  781. width: 50upx;
  782. height: 50upx;
  783. margin-top: 5upx;
  784. }
  785. .tab_card>.two>.twoitem>text {
  786. width: 50upx;
  787. height: 50upx;
  788. color: #899cba;
  789. font-size: 32upx;
  790. }
  791. /* 字号 */
  792. .tab_card>.three {
  793. width: 690upx;
  794. height: 60upx;
  795. line-height: 60upx;
  796. margin: 25upx 30upx;
  797. display: flex;
  798. justify-content: space-around;
  799. align-items: center;
  800. background-color: #edf3fa;
  801. padding: 12upx 0;
  802. }
  803. .tab_card>.three>.threeitem {
  804. width: 90upx;
  805. height: 60upx;
  806. line-height: 60upx;
  807. border-radius: 5upx;
  808. }
  809. /* 字体颜色 */
  810. .tab_card>.four {
  811. width: 690upx;
  812. height: 60upx;
  813. line-height: 60upx;
  814. margin: 25upx 30upx;
  815. display: flex;
  816. justify-content: space-around;
  817. align-items: center;
  818. background-color: #edf3fa;
  819. padding: 12upx 0;
  820. }
  821. .tab_card>.four>.fouritema {
  822. width: 55upx;
  823. height: 55upx;
  824. line-height: 55upx;
  825. display: flex;
  826. align-items: center;
  827. justify-content: center;
  828. border-radius: 50%;
  829. /* border: 4upx solid black; */
  830. }
  831. .tab_card>.four>.fouritema>text {
  832. width: 46upx;
  833. height: 46upx;
  834. display: inline-block;
  835. background-color: black;
  836. border-radius: 50%;
  837. }
  838. .tab_card>.four>.fouritemb {
  839. width: 55upx;
  840. height: 55upx;
  841. line-height: 55upx;
  842. display: flex;
  843. align-items: center;
  844. justify-content: center;
  845. border-radius: 50%;
  846. /* border: 4upx solid red; */
  847. }
  848. .tab_card>.four>.fouritemb>text {
  849. width: 46upx;
  850. height: 46upx;
  851. display: inline-block;
  852. background-color: red;
  853. border-radius: 50%;
  854. }
  855. .tab_card>.four>.fouritemc {
  856. width: 55upx;
  857. height: 55upx;
  858. line-height: 55upx;
  859. display: flex;
  860. align-items: center;
  861. justify-content: center;
  862. border-radius: 50%;
  863. /* border: 4upx solid #92a5c7; */
  864. }
  865. .tab_card>.four>.fouritemc>text {
  866. width: 46upx;
  867. height: 46upx;
  868. display: inline-block;
  869. background-color: #92a5c7;
  870. border-radius: 50%;
  871. }
  872. .tab_card>.four>.fouritemd {
  873. width: 55upx;
  874. height: 55upx;
  875. line-height: 55upx;
  876. display: flex;
  877. align-items: center;
  878. justify-content: center;
  879. border-radius: 50%;
  880. /* border: 4upx solid #00aaff; */
  881. }
  882. .tab_card>.four>.fouritemd>text {
  883. width: 46upx;
  884. height: 46upx;
  885. display: inline-block;
  886. background-color: #00aaff;
  887. border-radius: 50%;
  888. }
  889. .tab_card>.four>.fouriteme {
  890. width: 55upx;
  891. height: 55upx;
  892. line-height: 55upx;
  893. display: flex;
  894. align-items: center;
  895. justify-content: center;
  896. border-radius: 50%;
  897. /* border: 4upx solid orange; */
  898. }
  899. .tab_card>.four>.fouriteme>text {
  900. width: 46upx;
  901. height: 46upx;
  902. display: inline-block;
  903. background-color: orange;
  904. border-radius: 50%;
  905. }
  906. .tab_card>.four>.fouritemf {
  907. width: 55upx;
  908. height: 55upx;
  909. line-height: 55upx;
  910. display: flex;
  911. align-items: center;
  912. justify-content: center;
  913. border-radius: 50%;
  914. /* border: 4upx solid green; */
  915. }
  916. .tab_card>.four>.fouritemf>text {
  917. width: 46upx;
  918. height: 46upx;
  919. display: inline-block;
  920. background-color: green;
  921. border-radius: 50%;
  922. }
  923. /* 对齐方式 背景颜色*/
  924. .tab_card>.fours {
  925. width: 690upx;
  926. height: 60upx;
  927. line-height: 60upx;
  928. margin: 25upx 30upx;
  929. display: flex;
  930. justify-content: space-around;
  931. align-items: center;
  932. background-color: #edf3fa;
  933. padding: 12upx 0;
  934. }
  935. .tab_card>.fours>.foursitema {
  936. width: 50upx;
  937. height: 50upx;
  938. line-height: 50upx;
  939. display: flex;
  940. align-items: center;
  941. justify-content: center;
  942. /* border: 4upx solid black; */
  943. }
  944. .tab_card>.fours>.foursitema>text {
  945. width: 46upx;
  946. height: 46upx;
  947. display: inline-block;
  948. background-color: black;
  949. }
  950. .tab_card>.fours>.foursitemb {
  951. width: 50upx;
  952. height: 50upx;
  953. line-height: 50upx;
  954. display: flex;
  955. align-items: center;
  956. justify-content: center;
  957. /* border: 4upx solid red; */
  958. }
  959. .tab_card>.fours>.foursitemb>text {
  960. width: 46upx;
  961. height: 46upx;
  962. display: inline-block;
  963. background-color: red;
  964. }
  965. .tab_card>.fours>.foursitemc {
  966. width: 50upx;
  967. height: 50upx;
  968. line-height: 50upx;
  969. display: flex;
  970. align-items: center;
  971. justify-content: center;
  972. /* border: 4upx solid #92a5c7; */
  973. }
  974. .tab_card>.fours>.foursitemc>text {
  975. width: 46upx;
  976. height: 46upx;
  977. display: inline-block;
  978. background-color: #92a5c7;
  979. }
  980. .tab_card>.fours>.foursitemd {
  981. width: 50upx;
  982. height: 50upx;
  983. line-height: 50upx;
  984. display: flex;
  985. align-items: center;
  986. justify-content: center;
  987. /* border: 4upx solid #00aaff; */
  988. }
  989. .tab_card>.fours>.foursitemd>text {
  990. width: 46upx;
  991. height: 46upx;
  992. display: inline-block;
  993. background-color: #00aaff;
  994. }
  995. .tab_card>.fours>.foursiteme {
  996. width: 50upx;
  997. height: 50upx;
  998. line-height: 50upx;
  999. display: flex;
  1000. align-items: center;
  1001. justify-content: center;
  1002. /* border-radius: 50%; */
  1003. /* border: 4upx solid orange; */
  1004. }
  1005. .tab_card>.fours>.foursiteme>text {
  1006. width: 46upx;
  1007. height: 46upx;
  1008. display: inline-block;
  1009. background-color: orange;
  1010. }
  1011. .tab_card>.fours>.foursitemf {
  1012. width: 50upx;
  1013. height: 50upx;
  1014. line-height: 50upx;
  1015. display: flex;
  1016. align-items: center;
  1017. justify-content: center;
  1018. /* border: 4upx solid green; */
  1019. }
  1020. .tab_card>.fours>.foursitemf>text {
  1021. width: 46upx;
  1022. height: 46upx;
  1023. display: inline-block;
  1024. background-color: green;
  1025. }
  1026. /* 图片选项卡 */
  1027. .tab_cardimg {
  1028. width: 750upx;
  1029. height: 430upx;
  1030. }
  1031. .tab_cardimg>.mo {
  1032. width: 750upx;
  1033. height: 200upx;
  1034. display: flex;
  1035. }
  1036. .tab_cardimg>.mo>.moitem {
  1037. width: 140upx;
  1038. height: 140upx;
  1039. display: flex;
  1040. justify-content: center;
  1041. align-items: center;
  1042. flex-direction: column;
  1043. margin-left: 40upx;
  1044. margin-top: 30upx;
  1045. }
  1046. .tab_cardimg>.mo>.moitem>image {
  1047. width: 80upx;
  1048. height: 80upx;
  1049. }
  1050. .tab_cardimg>.mo>.moitem>text {
  1051. width: 120upx;
  1052. height: 40upx;
  1053. margin-top: 10upx;
  1054. color: #92a5c7;
  1055. font-size: 24upx;
  1056. text-align: center;
  1057. }
  1058. /* */
  1059. .main {
  1060. width: 100%;
  1061. min-height: 500rpx;
  1062. max-height: 500rpx;
  1063. overflow: auto;
  1064. /* height: max-content; */
  1065. }
  1066. .modes {
  1067. height: 500rpx;
  1068. }
  1069. .myEditor {
  1070. min-height: 500rpx;
  1071. max-height: 500rpx;
  1072. overflow: auto;
  1073. /* height: max-content; */
  1074. padding: 0 20upx;
  1075. font-size: 36upx;
  1076. }
  1077. </style>