实现功能:基于tinymce TinyMCE中文文档中文手册 拖拽图片排序,去掉全部上传按钮,点击保存上传图片并关闭弹窗,优化了一些交互提示
声明:本人不是做vue前端的,可能有些更好的方法实现拖拽或者其他一些开源库实现。
代码plugin.js:
tinymce.PluginManager.add('axupimgs', function(editor, url) {
var pluginName='Ax多图片上传';
window.axupimgs={}; //扔外部公共变量,也可以扔一个自定义的位置
var baseURL=tinymce.baseURL;
// var iframe1 = baseURL+'/plugins/axupimgs/upfiles.html';
var iframe1 = location.origin+'/tinymce/axupimgs/upfiles_new.html';
axupimgs.images_upload_handler = editor.getParam('images_upload_handler', undefined, 'function');
axupimgs.images_upload_base_path = editor.getParam('images_upload_base_path', '', 'string');
axupimgs.axupimgs_filetype = editor.getParam('axupimgs_filetype', '.png,.gif,.jpg,.jpeg', 'string');
axupimgs.res=[];
var openDialog = function() {
isAddEventListener(true)
return editor.windowManager.openUrl({
title: pluginName,
size: 'large',
url:iframe1,
buttons: [
{
type: 'custom',
text: 'Close',
name: 'close',
},
{
type: 'custom',
text: 'Save',
name: 'save',
primary: true
},
],
onAction: function (api, details) {
switch (details.name) {
case 'save':
//发送给upfiles,上传图片
window.postMessage({ action: 'upLoadPictureFile' }, '*');
break;
case 'close':
window.postMessage({ action: 'closeUpLoadPictureFile' }, '*');
isAddEventListener(false)
break;
default:
break;
}
},
});
};
//保存
function save() {
// 链接插入富文本显示
var html = '';
var imgs = axupimgs.res;
var len = imgs.length;
for (let i = 0; i < len; i++) {
if (imgs[i].url) {
html += '<img src="' + imgs[i].url + '" />';
}
}
editor.insertContent(html);
axupimgs.res = [];
tinymce.activeEditor.windowManager.close()
};
//处理消息
function handleMessage(event){
var data = event.data;
if (data.action === 'savePictureFile') {
save();
isAddEventListener(false)
}
}
//添加监听
function isAddEventListener(isAdd){
if(isAdd){
// 接收来自 iframe1 的消息
window.addEventListener('message', handleMessage, false)
}else{
window.removeEventListener('message',handleMessage, false)
}
}
editor.ui.registry.getAll().icons.axupimgs || editor.ui.registry.addIcon('axupimgs','<svg viewBox="0 0 1280 1024" xmlns="http://www.w3.org/2000/svg" width="24" height="24"><path d="M1126.2,779.8V87.6c0-24-22.6-86.9-83.5-86.9H83.5C14.7,0.7,0,63.7,0,87.7v692c0,36.2,29.2,89.7,83.5,89.7l959.3-1.3c51.7,0,83.5-42.5,83.5-88.3zm-1044,4V86.3h961.6V783.7H82.2v0.1z" fill="#53565A"/><path d="M603,461.6L521.1,366.3,313,629.8,227.2,546.8,102.4,716.8H972.8v-170L768.2,235.2,603.1,461.6zM284.6,358.4a105.4,105.4,0,0,0,73.5-30c19.5-19.1,30.3-45,30.2-71.8,0-56.8-45.9-103-102.4-103-56.6,0-102.4,46.1-102.4,103C183.4,313.5,228,358.4,284.6,358.4z" fill="#9598A0"/><path d="M1197.7,153.6l-0.3,669.3s13.5,113.9-67.4,113.9H153.6c0,24.1,23.9,87.2,83.5,87.2h959.3c58.3,0,83.6-49.5,83.6-89.9V240.8c-0.1-41.8-44.9-87.2-82.3-87.2z" fill="#53565A"/></svg>');
editor.ui.registry.addButton('axupimgs', {
icon: 'axupimgs',
tooltip: pluginName,
onAction: function() {
openDialog();
}
});
editor.ui.registry.addMenuItem('axupimgs', {
icon: 'axupimgs',
text: '图片批量上传...',
onAction: function() {
openDialog();
}
});
return {
getMetadata: function() {
return {
name: pluginName,
url: "http://tinymce.ax-z.cn/more-plugins/axupimgs.php",
};
}
};
});
htmlupfiles_new.html
<!doctype html>
<html>
<head>
<meta charset="utf-8" />
<title>axupimgs</title>
<meta name="viewport" content="user-scalable=no, width=device-width, initial-scale=1.0, maximum-scale=1.0" />
<meta name="apple-mobile-web-app-capable" content="yes" />
<meta name="apple-mobile-web-app-status-bar-style" content="black" />
<meta name="format-detection" content="telephone=no">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1" />
<style>
html,
body {
height: 100%;
margin: 0;
padding: 0;
background: #fff;
}
ul {
margin: 0;
padding: 0;
list-style: none;
}
#wrap {
padding: 10px;
}
#topbar {
padding: 10px 0;
border-bottom: 1px solid #ccc;
/* text-align: right; */
}
#topbar button {
margin: 0;
margin-left: 5px;
outline: none;
padding: 4px 16px;
box-sizing: border-box;
display: inline-block;
border: none;
border-radius: 3px;
text-align: center;
cursor: pointer;
font-size: 14px;
line-height: 1.5;
background-color: #f0f0f0;
color: #223;
}
#topbar button.primary {
background-color: #3d97d4;
color: #fff;
}
#topbar button:hover {
background-color: #207ab7;
color: #fff;
}
#topbar button.addfile {
float: right;
}
#file_list {
display: grid;
grid-gap: 10px;
grid-template-columns: repeat(auto-fill, minmax(160px, 1fr));
padding-top: 10px;
}
#file_list:empty:after {
content: '可以直接拖拽文件到这里';
color: #777;
font-size: 0.8em;
}
#file_list li {
position: relative;
display: block;
vertical-align: top;
padding: 5px 5px;
border-radius: 5px;
}
#file_list li.up-over {}
#file_list li.up-now {}
#file_list li.up-now:after {
content: '';
position: absolute;
top: 0;
left: 0;
display: block;
width: 100%;
height: 100%;
background: rgba(255, 255, 255, 0.8) url(loading.gif) center center no-repeat;
border-radius: 5px;
z-index: 999;
}
#file_list li:hover {
background-color: #ddd;
}
#file_list li .picbox {
display: flex;
flex: 0 0 auto;
justify-content: center;
overflow: hidden;
position: relative;
width: 100%;
padding-top: 100%;
align-items: center;
}
#file_list li .picbox img {
display: block;
max-width: 100%;
max-height: 100%;
position: absolute;
top: 50%;
left: 50%;
transform: translateX(-50%) translateY(-50%);
}
#file_list li.up-over .picbox:after {
content: url('data:image/svg+xml;%20charset=utf8,%3Csvg%20viewBox%3D%220%200%201024%201024%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2224%22%20height%3D%2224%22%3E%3Cpath%20d%3D%22M512%200C229.376%200%200%20229.376%200%20512s229.376%20512%20512%20512%20512-229.376%20512-512S794.624%200%20512%200z%22%20fill%3D%22%234AC711%22%3E%3C%2Fpath%3E%3Cpath%20d%3D%22M855.552%20394.752l-358.4%20358.4a50.9952%2050.9952%200%200%201-72.192%200l-204.8-204.8c-18.944-19.968-18.944-51.2%200-71.168a50.5344%2050.5344%200%200%201%2072.192-1.024L460.8%20644.608l322.048-322.048c19.968-18.944%2051.2-18.944%2071.168%200%2020.48%2019.456%2020.992%2051.712%201.536%2072.192z%22%20fill%3D%22%23FFFFFF%22%3E%3C%2Fpath%3E%3C%2Fsvg%3E');
position: absolute;
bottom: 2px;
right: 2px;
z-index: 9;
}
#file_list li .tools {
display: none;
position: absolute;
bottom: 5px;
right: 5px;
z-index: 99;
}
#file_list li:hover .tools {
display: block;
}
#file_list li .tools .remove {
cursor: pointer;
}
#file_list li .tools .remove:after {
content: url('data:image/svg+xml;%20charset=utf8,%3Csvg%20width=%2224%22%20height=%2224%22%20viewBox=%220%200%2024%2024%22%20xmlns=%22http://www.w3.org/2000/svg%22%3E%3Cpath%20d=%22M17%206h3a1%201%200%200%201%200%202h-1v11a3%203%200%200%201-3%203H8a3%203%200%200%201-3-3V8H4a1%201%200%201%201%200-2h3V5a3%203%200%200%201%203-3h4a3%203%200%200%201%203%203v1zm-2%200V5a1%201%200%200%200-1-1h-4a1%201%200%200%200-1%201v1h6zm2%202H7v11a1%201%200%200%200%201%201h8a1%201%200%200%200%201-1V8zm-8%203a1%201%200%200%201%202%200v6a1%201%200%200%201-2%200v-6zm4%200a1%201%200%200%201%202%200v6a1%201%200%200%201-2%200v-6z%22%3E%3C/path%3E%3C/svg%3E');
}
#file_list li .namebox {
font-size: 14px;
line-height: 20px;
max-height: 40px;
overflow: hidden;
padding: 5px 10px;
text-align: center;
display: flex;
justify-content: center;
align-items: flex-start;
}
#file_list li .namebox span {
word-break: break-all;
vertical-align: top;
}
#loading_overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(255, 255, 255, 0.8);
/* 半透明白色背景 */
z-index: 1000;
/* 确保遮罩层在最顶层 */
display: none;
/* 初始隐藏 */
justify-content: center;
align-items: center;
display: flex;
}
.loading_spinner {
border: 4px solid #f3f3f3;
/* Light grey */
border-top: 4px solid #3498db;
/* Blue */
border-radius: 50%;
width: 40px;
height: 40px;
animation: spin 1s linear infinite;
}
.loading_text {
margin-top: 10px;
font-size: 16px;
}
@keyframes spin {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(360deg);
}
}
</style>
</head>
<body>
<div id="wrap">
<div id="topbar"><button class="removeall">清空列表</button><button class="addfile primary">+ 添加文件</button></div>
<ul id="file_list"></ul>
</div>
<!-- 加载遮罩层和动画 -->
<div id="loading_overlay" style="display: none;">
<div class="loading_spinner"></div>
<!-- <div class="loading_text">加载中...</div> -->
</div>
<script>
var editor = parent.tinymce.activeEditor;
var axupimgs = parent.axupimgs;
axupimgs.res = []; //存放本地文件的数组
var blobInfo = { file: null }
blobInfo.blob = function () { return this.file; }
var upload_handler = axupimgs.images_upload_handler;
var upload_base_path = axupimgs.images_upload_base_path;
//为列表添加排序
function reSort() {
document.querySelectorAll('#file_list li').forEach((el, i) => {
el.setAttribute('data-num', i);
});
}
function addList(files) {
var files_sum = files.length;
var vDom = document.createDocumentFragment();
for (let i = 0; i < files_sum; i++) {
let file = files[i];
let blobUrl = window.URL.createObjectURL(file)
axupimgs.res.push({ file: file, blobUrl: blobUrl, url: '' });
let li = document.createElement('li');
li.setAttribute('class', 'up-no');
li.setAttribute('data-time', file.lastModified);
li.setAttribute('draggable', 'true');
li.innerHTML = '<div class="picbox"><img src="' + blobUrl + '"></div><div class="namebox"><span>' + file.name + '</span></div><div class="tools"><a class="remove"></a></div>';
vDom.appendChild(li);
}
document.querySelector('#file_list').appendChild(vDom);
}
//清空列表
document.querySelector('#topbar .removeall').addEventListener('click', () => {
if (axupimgs.res.length > 0) {
if (confirm("确定清空列表吗")) {
axupimgs.res = []
document.querySelectorAll('#file_list li').forEach((el, i) => {
el.parentNode.removeChild(el)
});
}
}
});
//是否内部拖动
var isInputDrag = false
//拖拽添加
document.addEventListener('dragover', (e) => {
e.stopPropagation();
e.preventDefault();
if (isInputDrag) {
e.dataTransfer.dropEffect = 'move';
} else {
e.dataTransfer.dropEffect = 'copy';
}
});
document.addEventListener('dragend', (e) => {
isInputDrag = false
});
document.addEventListener('dragstart', (e) => {
isInputDrag = true
e.dataTransfer.effectAllowed = 'move';
var li = e.target.parentNode.parentNode;
var n = li.getAttribute('data-num');
e.dataTransfer.setData('text/plain', n);
});
document.addEventListener('drop', (e) => {
e.stopPropagation();
e.preventDefault();
// 获取目标元素
let target = e.target;
// 获取鼠标指针相对于目标元素的位置
let mouseX = e.clientX;
let targetRect = target.getBoundingClientRect();
let targetCenterX = targetRect.left + targetRect.width / 2;
var isLeft = true
// 判断鼠标指针在目标元素的左侧还是右侧
if (mouseX < targetCenterX) {
// 在左侧的处理逻辑
isLeft = true
} else {
isLeft = false
// 在右侧的处理逻辑
}
//内部拖拽
if (isInputDrag) {
//拖拽有问题未完成(有时候切换不了位置)
var li = e.target.parentNode.parentNode;
if (li) {
var oldData = e.dataTransfer.getData('text/plain');
try {
var newData = li.getAttribute('data-num');
if (newData == null) {
newData = e.target.parentNode.getAttribute('data-num')
}
// 可能会引发异常的代码块
} catch (error) {
newData = e.target.parentNode.getAttribute('data-num')
// 异常处理代码块
}
console.log("打印切换位置:", oldData, newData)
if (newData && oldData && (oldData != newData)) {
var oldEl = document.querySelectorAll('#file_list li')[oldData];
var newEl = document.querySelectorAll('#file_list li')[newData];
if (newEl && oldEl) {
if (newData !== oldData) {
if (isLeft) {
document.querySelector('#file_list').insertBefore(oldEl, newEl);
} else {
insertAfter(oldEl, newEl)
}
changePosition(oldData, newData, isLeft)
}
}
}
}
return false
}
if (!e.dataTransfer.files) { return false; }
var dropfiles = e.dataTransfer.files;
if (!(dropfiles.length > 0)) { return false; }
var exts = axupimgs.axupimgs_filetype.replace(/(\s)+/g, '').toLowerCase().split(',');
var files = [];
for (let file of dropfiles) {
ext = file.name.split('.');
ext = '.' + ext[ext.length - 1];
for (let s of exts) {
if (s == ext) {
files.push(file);
break;
}
}
}
if (files.length > 0) { addList(files) }
});
//添加文件
document.querySelector('#topbar .addfile').addEventListener('click', () => {
var input = document.createElement('input');
input.setAttribute('type', 'file');
input.setAttribute('multiple', 'multiple');
input.setAttribute('accept', axupimgs.axupimgs_filetype);
input.click();
input.onchange = function () {
var files = this.files;
addList(files);
}
});
var file_i = 0;
//功能: 在targetElement之后插入 新节点newElement
function insertAfter(newElement, targetElement) {
var parent = targetElement.parentNode;
if (parent.lastChild == targetElement) {
parent.appendChild(newElement);
} else {
parent.insertBefore(newElement, targetElement.nextSibling);
}
}
//上传所有文件
function upAllFiles() {
// isShowAddFile(false)
isShowLoading(true)
axupimgs.res.forEach((data, index) => {
console.log(`Index ${index}: ${data.isUpLoad}`, data.url);
//isUpLoad:undefined未上传 1成功 -1:失败 2:上传中
if (data.isUpLoad == undefined || data.isUpLoad == -1) {
blobInfo.file = data.file;
uploadFile(index, blobInfo)
}
});
}
//单独上传文件
function uploadFile(index, blobInfo) {
axupimgs.res[index].isUpLoad = 2;
upload_handler(blobInfo, function (url) {
if (upload_base_path) {
if (upload_base_path.slice(-1) == '/' && url.substr(0, 1) == '/') {
url = upload_base_path + url.slice(1);
} else if (upload_base_path.slice(-1) != '/' && url.substr(0, 1) != '/') {
url = upload_base_path + '/' + url;
} else {
url = upload_base_path + url;
}
}
axupimgs.res[index].url = url;
axupimgs.res[index].isUpLoad = 1;
filename = url.split('/').pop();
var li = document.querySelectorAll('#file_list li')[index];
li.setAttribute('class', 'up-over');
li.querySelector('.namebox span').innerText = filename;
window.parent.postMessage({ action: 'upLoadPictureFileProgress' }, '*');
}, function (err) {
document.querySelectorAll('#file_list li.up-now').forEach((el, i) => {
el.setAttribute('class', 'up-no');
});
axupimgs.res[index].isUpLoad = -1;
window.parent.postMessage({ action: 'upLoadPictureFileProgress' }, '*');
var name = axupimgs.res[index].file.name
if (alert("图片" + name + "上传失败:" + err)) {
}
});
}
//判断是否全部上传
function checkisAllUpLoad() {
const finedData = axupimgs.res.find(item => item.isUpLoad !== 1);
if (finedData) {
return false;
} else {
return true;
}
}
//判断是否正在上传中
function checkisUpLoadLoading() {
const finedData = axupimgs.res.find(item => item.isUpLoad === 2);
if (finedData) {
return true;
} else {
return false;
}
}
//切换位置
function changePosition(oldIndex, newIndex, isLeft) {
let oldData = axupimgs.res[oldIndex];
let newData = axupimgs.res[newIndex];
//右移
if (newIndex > oldIndex) {
axupimgs.res.splice(oldIndex, 1);
console.log("axupimgs.res-右移", isLeft, oldIndex, newIndex)
axupimgs.res.splice(isLeft ? newIndex - 1 : newIndex, 0, oldData);
}
//左移
if (newIndex < oldIndex) {
axupimgs.res.splice(oldIndex, 1);
console.log("axupimgs.res-左移", isLeft, oldIndex, newIndex)
if (!isLeft) {
if (newIndex >= axupimgs.res.length) {
newIndex = newIndex
} else {
newIndex++
}
}
axupimgs.res.splice(newIndex, 0, oldData);
}
axupimgs.res.forEach((element) => {
console.log("name:", element.file.name);
});
}
//接收js发送的message
window.parent.addEventListener('message', function (event) {
var data = event.data;
//上传图片
if (data.action === 'upLoadPictureFile') {
if (axupimgs.res.length > 0) {
document.querySelectorAll('#file_list li.up-no').forEach((el, i) => {
el.classList ? el.classList.add('up-now') : el.className += ' up-now';
});
if (checkisAllUpLoad()) {
console.log("upLoadPictureFile-图片全部上传成功")
window.parent.postMessage({ action: 'savePictureFile' }, '*');
} else {
console.log("upLoadPictureFile-图片正在上传或者上传失败")
upAllFiles();
}
} else {
if (confirm("暂未上传图片,确定要关闭吗")) {
close();
}
}
}
//进度
if (data.action === 'upLoadPictureFileProgress') {
if (checkisAllUpLoad()) {
window.parent.postMessage({ action: 'savePictureFile' }, '*');
}
// isShowAddFile(!checkisUpLoadLoading())
isShowLoading(checkisUpLoadLoading())
}
//关闭弹窗
if (data.action === 'closeUpLoadPictureFile') {
checkClose();
}
}, false);
// 检测关闭弹窗
function checkClose() {
console.log("checkClose", axupimgs.res)
if (axupimgs.res.length > 0) {
if (confirm("系统可不会保存您的编辑,确定要关闭吗")) {
close();
}
} else {
close();
}
}
//关闭弹窗
function close() {
window.parent.tinymce.activeEditor.windowManager.close();
}
//是否显示添加文件按钮
function isShowAddFile(isShow) {
document.querySelector('.addfile').style.display = isShow ? 'inline' : 'none';
}
// 获取 loading 遮罩层元素
var loadingOverlay = document.getElementById('loading_overlay');
// 当需要显示 loading 时调用此函数
function isShowLoading(isShow) {
loadingOverlay.style.display = isShow ? 'flex' : 'none'; // 显示遮罩层
document.body.style.pointerEvents = isShow ? 'none' : 'auto'; // 禁用页面点击事件
}
var observ_flist = new MutationObserver((muList, observe) => {
if (muList[0].addedNodes.length > 0) {
muList[0].addedNodes.forEach((el) => {
el.querySelector('.remove').addEventListener('click', (e) => {
var li = e.target.parentNode.parentNode;
var n = li.getAttribute('data-num');
var el = document.querySelectorAll('#file_list li')[n];
el.parentNode.removeChild(el);
axupimgs.res.splice(n, 1);
});
});
}
reSort();
});
observ_flist.observe(document.querySelector('#file_list'), { childList: true });
</script>
</body>
</html>