基於VUE接入TinyMCE富文本編輯器 漂亮簡潔 封裝成組件隨用隨調

来源:https://www.cnblogs.com/iwillrich/archive/2022/06/23/16404236.html
-Advertisement-
Play Games

TinyMCE是一款易用、且功能強大的所見即所得的富文本編輯器。同類程式有:UEditor、Kindeditor、Simditor、CKEditor、wangEditor、Suneditor、froala等等。 配置靈活,界面簡潔,支持自定義插件。 TinyMCE中文手冊:http://tinymc ...


TinyMCE是一款易用、且功能強大的所見即所得的富文本編輯器。同類程式有:UEditor、Kindeditor、Simditor、CKEditor、wangEditor、Suneditor、froala等等。
配置靈活,界面簡潔,支持自定義插件。
TinyMCE中文手冊:http://tinymce.ax-z.cn

一、安裝環境

1、安裝需要的包

我使用的是v5版本的,需要搭配tinymce-vue包來使用

npm install tinymce/[email protected]

npm install [email protected]

2、將安裝的tinymce包copy放在public下

cp ./node_modules/tinymce ./public/tinymce 

3、下載語言包

打開鏈接 http://tinymce.ax-z.cn/static/tiny/langs/zh_CN.js ,將zh_CN.js 另存到 public/tinymce/langs下,沒有langs文件夾的可以手動創建一下

二、代碼引用

附上我封裝的組件,可以作為參考,上傳文件的邏輯需要結合業務邏輯自行實現

<template>
	<Editor :id="editorId" v-if="reFresh" v-model="newData" :init="init" />
</template>
<script>
// 引入組件
import tinymce from "tinymce/tinymce";
import Editor from "@tinymce/tinymce-vue";
import 'tinymce/themes/silver/theme'
import "tinymce/skins/ui/oxide/skin.min.css";

// 引入富文本編輯器主題的js和css

export default {
	name: "TinymceEditor",
	components: { Editor },
	data() {
		return {
			init: {    // 配置文件
				base_url: '/tinymce',
				menubar: false,
				external_plugins: {  //引入需要的插件
					anchor: "/tinymce/plugins/anchor/plugin.min.js",
					code: "/tinymce/plugins/code/plugin.min.js",
					print: "/tinymce/plugins/print/plugin.min.js",
					preview: "/tinymce/plugins/preview/plugin.min.js",
					searchreplace: "/tinymce/plugins/searchreplace/plugin.min.js",
					autolink: "/tinymce/plugins/autolink/plugin.min.js",
					directionality: "/tinymce/plugins/directionality/plugin.min.js",
					visualblocks: "/tinymce/plugins/visualblocks/plugin.min.js",
					visualchars: "/tinymce/plugins/visualchars/plugin.min.js",
					fullscreen: "/tinymce/plugins/fullscreen/plugin.min.js",
					image: "/tinymce/plugins/image/plugin.min.js",
					link: "/tinymce/plugins/link/plugin.min.js",
					media: "/tinymce/plugins/media/plugin.min.js",
					template: "/tinymce/plugins/template/plugin.min.js",
					codesample: "/tinymce/plugins/codesample/plugin.min.js",
					table: "/tinymce/plugins/table/plugin.min.js",
					charmap: "/tinymce/plugins/charmap/plugin.min.js",
					pagebreak: "/tinymce/plugins/pagebreak/plugin.min.js",
					nonbreaking: "/tinymce/plugins/nonbreaking/plugin.min.js",
					insertdatetime: "/tinymce/plugins/insertdatetime/plugin.min.js",
					advlist: "/tinymce/plugins/advlist/plugin.min.js",
					lists: "/tinymce/plugins/lists/plugin.min.js",
					wordcount: "/tinymce/plugins/wordcount/plugin.min.js",
					imagetools: "/tinymce/plugins/imagetools/plugin.min.js",
					textpattern: "/tinymce/plugins/textpattern/plugin.min.js",
					help: "/tinymce/plugins/help/plugin.min.js",
					emoticons: "/tinymce/plugins/emoticons/plugin.min.js",
					autosave: "/tinymce/plugins/autosave/plugin.min.js",
					iframe: "/tinymce/plugins/iframe/plugin.min.js",
					hr: "/tinymce/plugins/hr/plugin.min.js",
					// formatpainter: "/tinymce/plugins/formatpainter/plugin.min.js",
				},
				language_url:
					"/tinymce/langs/zh_CN.js",   //語言文件
				language: "zh_CN",
				font_formats:
					"微軟雅黑=Microsoft YaHei,Helvetica Neue,PingFang SC,sans-serif;蘋果蘋方=PingFang SC,Microsoft YaHei,sans-serif;宋體=simsun,serif;仿宋體=FangSong,serif;黑體=SimHei,sans-serif;Arial=arial,helvetica,sans-serif;Arial Black=arial black,avant garde;Book Antiqua=book antiqua,palatino;",

				toolbar: [
					"template code undo redo restoredraft cut copy paste pastetext forecolor backcolor bold italic underline strikethrough link unlink anchor alignleft aligncenter alignright alignjustify outdent indent formatselect fontselect fontsizeselect bullist numlist blockquote subscript superscript removeformat table image media charmap emoticons pagebreak insertdatetime print preview fullscreen formatpainter iframe hr",
				],
				templates: [],
				// content_css : ['/layui/css/layui.css','/css/public.css?v=1'],
				content_css: [],
				height: 800, //編輯器高度
				min_height: 200,
				max_height: 600,
				branding: false,
				paste_data_images: true, // 允許粘貼圖像
				file_picker_types: "file image media",
				images_upload_handler: (blobInfo, success) => {
					var xhr, formData;
					var file = blobInfo.blob();//轉化為易於理解的file對象
					xhr = new XMLHttpRequest();
					xhr.withCredentials = false;
					xhr.open("POST",`${process.env.VUE_APP_API_URL}/file/upload`);  //上傳文件的地址,需要替換成自己的
					xhr.setRequestHeader("Authorization",window.localStorage.escourse_token);
					xhr.onload = function () {
						var json;
						json = JSON.parse(xhr.responseText);
					};
					formData = new FormData();
					formData.append("file", file, file.name);
					formData.append("module", "post");
					formData.append("file_source", "1");
					// formData.append("token",window.localStorage.escourse_token)
					xhr.send(formData);
					xhr.onload = function () {
							let json = JSON.parse(xhr.responseText);
							if (json.code == 200) {
								success(process.env.VUE_APP_API_URL + json.data.file.path)
								return
							}
						};
				},
				media_url_resolver: function (data, resolve) {
					try {
						let videoUri = encodeURI(data.url);
						let embedHtml = `<p>
                            <span
                                class="mce-object mce-object-video"
                                data-mce-selected="1"
                                data-mce-object="video"
                                data-mce-p-width="100%"
                                data-mce-p-height="auto"
                                data-mce-p-controls="controls"
                                data-mce-p-controlslist="nodownload"
                                data-mce-p-allowfullscreen="true"
                                data-mce-p-src=${videoUri} >
                                <video src=${data.url} width="100%" height="auto" controls="controls" controlslist="nodownload">
                                </video>
                            </span>
                            </p>
                            <p style="text-align: left;"></p>`;
						resolve({ html: embedHtml });
					} catch (e) {
						resolve({ html: "" });
					}
				},
				file_picker_callback(cb, value, meta) {
					//當點擊meidia圖標上傳時,判斷meta.filetype == 'media'有必要,因為file_picker_callback是media(媒體)、image(圖片)、file(文件)的共同入口
					console.log('meta', meta);

					//創建一個隱藏的type=file的文件選擇input
					let input = document.createElement("input");
					input.setAttribute("type", "file");
					input.onchange = function () {
						var xhr, formData;
						let file = this.files[0]; //只選取第一個文件。如果要選取全部,後面註意做修改
						xhr = new XMLHttpRequest();
						xhr.withCredentials = false;
						xhr.open("POST",`${process.env.VUE_APP_API_URL}/file/upload`);   //上傳文件的地址,需要替換成自己的
						xhr.setRequestHeader("Authorization",window.localStorage.escourse_token);
						xhr.onload = function () {
							var json;
							json = JSON.parse(xhr.responseText);
						};
						formData = new FormData();
						formData.append("file", file, file.name);
						formData.append("module", "post");
						formData.append("file_source", "1");
						// formData.append("token",window.localStorage.escourse_token)
						xhr.send(formData);

						xhr.upload.onprogress = function () {
							// 進度(e.loaded / e.total * 100)
							// progress(e.loaded / e.total * 100);
						};
						// xhr.onerror = function () {
						//   //根據自己的需要添加代碼
						//   console.log(xhr.status);
						//   return;
						// };
						xhr.onload = function () {
							let json = JSON.parse(xhr.responseText);
							if (json.code == 200) {
								// if (meta.filetype == "media") {
									cb(process.env.VUE_APP_API_URL + json.data.file.path, { text: json.data.file.name })
								// }
								// if (meta.filetype == "file") {
								// 	cb(json.data.url, { text: json.data.name })
								// }
								return
							}
						};
					};
					//觸發點擊
					input.click();

				},
			},
			editorId: this.id,
			newData: this.data,
			reFresh:false,
			laws:[]
		};
	},
    created(){
        this.getBandData();
    },
	mounted() {
		tinymce.init({});
	},
	props: {
		data: String,
		onChange: Function   // 回調函數,將新修改的內容做為第一個參數返回
	},
	watch: {
		newData: 'updateData',
		data: 'uptData',
		"init.templates"(newValue) {   // 如果配置發生改變,則刷新編輯器
			this.reFresh = false;
			this.$nextTick(() => {
				this.reFresh = true;
			});
		},
	},
	methods: {
		async getBandData() { 
			let urlResult = await getTemplates();  //這邊我是自己封裝了個插件,通過getTemplates介面,從後端讀取模板,再顯示到編輯器插件上,如果不需要可以刪掉
			if (urlResult.code == 200) {
				let array = urlResult.data;
				let arr = [];
				for (let index = 0; index < array.length; index++) {
					const element = array[index];
					let data = {
						title: element.name,
						content: element.html,
						description: element.html,
						// description: "<img src='"+this.urlConfig.adminUrl + element.thumbnail+"'/>",
					};
					arr.push(data);
				}
				this.laws = arr; //得到的數據賦值給laws
				this.init.templates = this.laws;
			}
		},
		addContent(html) {
			console.log('html', html)
			this.newData = this.newData + html
		},
		uptData() {
			this.newData = JSON.parse(JSON.stringify(this.data))
		},
		updateData() {
			this.onChange(this.newData)
		},
		clear() {
			this.editorValue = "";
		},
	},
};
</script>

在頁面中引用

<template>
	<Editor v-bind:data="data" :onChange="onchange"></Editor>
</template>
import Editor from "../../../components/tinymce/editor";
export default {
	data() {
		return {
			data:''
		};
	},
	components: {
		Editor
	},
	methods: {
		onchange(e){
			this.data = e
		}
	}
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 問題導入 在之前項目的基礎功能實現中,後臺管理和移動端在進行數據訪問的時候,都是直接操作資料庫MySQL。此時的系統有且僅有一臺MySQL伺服器,則可能會出現如下問題 ①、讀和寫所有壓力都由一臺資料庫承擔,壓力大 ②、資料庫伺服器磁碟損壞導致數據丟失,單點故障 解決方案 很簡單,一臺伺服器撐不住,那 ...
  • 資料庫設計的設計內容包括:需求分析、概念結構設計、邏輯結構設計、物理結構設計、資料庫的實施和資料庫的運行和維護。 ...
  • 原文鏈接:走好數據中台最後一公裡,數據服務API是數據中台的標配 視頻回顧:點擊這裡 課件獲取:點擊這裡 一、數據服務API建設背景 在數字化轉型的時代背景下,新需求的大量增長、新技術的不斷迭代,“互聯網化、數字化”進程的不斷深入,越來越多的業務被遷移到互聯網上,產生大量的業務交互和對外服務需求,對 ...
  • E-R圖也稱實體-聯繫圖(Entity Relationship Diagram),它提供了表示實體類型、屬性和聯繫的方法,用來描述現實世界的概念模型。 ...
  • Flutter IOS 鍵盤焦點 關閉打開鍵盤 Android 的TextField 獲取焦點打開的鍵盤中有關閉鍵盤的箭頭 可以進行關閉鍵盤 IOS 則就不行,它的鍵盤沒有關閉鍵盤的按鈕 就很噁心!! IOS 的輸入框在你輸入完數據之後, 你就沒辦法關閉鍵盤。 雖然你可以滑動界面顯示被隱藏的按鈕,也 ...
  • 隨著信息化的發展,很多具有重要價值的知識隱藏分佈在海量數據中,影響了人們獲取知識的效率,如何處理繁雜的非結構化文本數據成為難題。 近日,HMS Core機器學習服務6.5.0版本新增線上文本實體抽取能力,該能力可以檢測出文本中是否存在比如日期、姓名、專有名詞等實體信息,並將此類實體抽取出來,即自動處 ...
  • vue2中只能有一個根標簽,但是在vue3中根組件已經可以有多個根節點了 在vue2中只所以這麼做是因為vdom是一顆單根樹形結構,patch方法在遍歷的時候從根節點開始遍歷,它要求只有一個根節點,組件也會轉換為一個vdom,自然滿足這個要求 vue3中值所以可以有多個節點,是因為引入了Fragme ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 1. 防抖節流 這也是一個經典題目了,首先要知道什麼是防抖,什麼是節流。 防抖: 在一段時間內,事件只會最後觸發一次。 節流: 事件,按照一段時間的間隔來進行觸發。 實在不懂的話,可以去這個大佬的Demo地址玩玩防抖節流DEMO // 防 ...
一周排行
    -Advertisement-
    Play Games
  • 一:背景 準備開個系列來聊一下 PerfView 這款工具,熟悉我的朋友都知道我喜歡用 WinDbg,這東西雖然很牛,但也不是萬能的,也有一些場景他解決不了或者很難解決,這時候藉助一些其他的工具來輔助,是一個很不錯的主意。 很多朋友喜歡在項目中以記錄日誌的方式來監控項目的流轉情況,其實 CoreCL ...
  • 本來閑來無事,準備看看Dapper擴展的源碼學習學習其中的編程思想,同時整理一下自己代碼的單元測試,為以後的進一步改進打下基礎。 突然就發現問題了,源碼也不看了,開始改代碼,改了好久。 測試Dapper.LiteSql數據批量插入的時候,耗時20秒,感覺不正常,於是我測試了非Dapper版的Lite ...
  • 需求如下,在DEV框架項目中,需要在表格中增加一列顯示圖片,並且能編輯該列圖片,然後進行保存等操作,最終效果如下 這裡使用的是PictureEdit控制項來實現,打開DEV GridControl設計器,在ColumnEdit選擇PictureEdit: 綁定圖片代碼如下: DataTable dtO ...
  • 前兩天微軟偷偷更新了Visual Studio 2022 正式版版本 17.3 發佈,發佈摘要: MAUI 工作負荷 GA 生成 MAUI/Blazor CSS 熱重載支持 現在,你將能夠使用我們的新增功能在 Visual Studio 中使用每個更新試用一系列新功能。 選擇每個功能以瞭解有關特定功 ...
  • 航天和軍工領域的數字化轉型和建設正在積極推進,在與航天二院、航天三院、航天六院、航天九院、無線電廠、兵工廠等單位交流的過程中,用戶更聚焦試驗和生產過程中的痛點,迫切需要解決軟體平臺統一監測和控制設備及軟體與設備協同的問題。 ...
  • .NET 項目預設情況下 日誌是使用的 ILogger 介面,預設提供一下四種日誌記錄程式: 控制台 調試 EventSource EventLog 這四種記錄程式都是預設包含在 .NET 運行時庫中。關於這四種記錄程式的詳細介紹可以直接查看微軟的官方文檔 https://docs.microsof ...
  • 一:背景 上一篇我們聊到瞭如何去找 熱點函數,這一篇我們來看下當你的程式出現了 非托管記憶體泄漏 時如何去尋找可疑的代碼源頭,其實思路很簡單,就是在 HeapAlloc 或者 VirtualAlloc 時做 Hook 攔截,記錄它的調用棧以及分配的記憶體量, PerfView 會將這個 分配量 做成一個 ...
  • 背景 在 CI/CD 流程當中,測試是 CI 中很重要的部分。跟開發人員關係最大的就是單元測試,單元測試編寫完成之後,我們可以使用 IDE 或者 dot cover 等工具獲得單元測試對於業務代碼的覆蓋率。不過我們需要一個獨立的 CLI 工具,這樣我們才能夠在 Jenkins 的 CI 流程集成。 ...
  • 一、應用場景 大家在使用Mybatis進行開發的時候,經常會遇到一種情況:按照月份month將數據放在不同的表裡面,查詢數據的時候需要跟不同的月份month去查詢不同的表。 但是我們都知道,Mybatis是ORM持久層框架,即:實體關係映射,實體Object與資料庫表之間是存在一一對應的映射關係。比 ...
  • 我國目前並未出台專門針對網路爬蟲技術的法律規範,但在司法實踐中,相關判決已屢見不鮮,K 哥特設了“K哥爬蟲普法”專欄,本欄目通過對真實案例的分析,旨在提高廣大爬蟲工程師的法律意識,知曉如何合法合規利用爬蟲技術,警鐘長鳴,做一個守法、護法、有原則的技術人員。 案情介紹 深圳市快鴿互聯網科技有限公司 2 ...