richtext-editor.vue 31 KB

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