/* globals jQuery, $, JSZip, saveAs, waitForKeyElements */
// ==UserScript==
// @name 幕布导出Xmind文件
// @namespace http://tampermonkey.net/
// @version 0.2
// @description mubu html export to xmind For Lily
// @author xyan9
// @match file:///*
// @license MIT
// @grant none
// @require https://code.jquery.com/jquery-3.6.0.min.js
// @require https://cdn.bootcdn.net/ajax/libs/jszip/3.6.0/jszip.min.js
// @require https://cdn.bootcdn.net/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js
// ==/UserScript==
function Father(name){
let o = new Object()
o = {
"id": genFather(),
"class": "sheet",
"title": "画布 1",
"rootTopic": {
"id": genFather(),
"class": "topic",
"title": name,
"structureClass": "org.xmind.ui.logic.right",
"style": {
"id": genId(),
"properties": {
"svg:fill": "#3949AB",
"border-line-color": "none",
"line-color": "#141414",
"fo:color": "#FFFFFF",
"shape-class": "org.xmind.topicShape.roundedRect",
"line-class": "org.xmind.branchConnection.roundedElbow",
"line-width": "3pt",
"line-pattern": "solid",
"fill-pattern": "solid",
"border-line-width": "0pt",
"border-line-pattern": "solid",
"arrow-end-class": "org.xmind.arrowShape.none",
"alignment-by-level": "inactived",
"fo:font-family": "'NeverMind','Microsoft YaHei','PingFang SC','Microsoft JhengHei',sans-serif",
"fo:font-style": "normal",
"fo:font-weight": 500,
"fo:font-size": "30pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "center"
}
},
"children": {
"attached": [
]
}
},
"extensions": [
{
"provider": "org.xmind.ui.skeleton.structure.style",
"content": {
"centralTopic": "org.xmind.ui.map.clockwise",
"mainTopic": "org.xmind.ui.logic.right"
}
}
],
"theme": {
"id": genId(),
"centralTopic": {
"id": genId(),
"properties": {
"svg:fill": "#000229",
"line-color": "#000229",
"shape-class": "org.xmind.topicShape.roundedRect",
"line-class": "org.xmind.branchConnection.curve",
"line-width": "3pt",
"line-pattern": "solid",
"fill-pattern": "solid",
"border-line-width": "0pt",
"arrow-end-class": "org.xmind.arrowShape.none",
"alignment-by-level": "inactived",
"fo:font-family": "NeverMind",
"fo:font-style": "normal",
"fo:font-weight": 500,
"fo:font-size": "30pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "center"
}
},
"mainTopic": {
"id": genId(),
"properties": {
"shape-class": "org.xmind.topicShape.roundedRect",
"line-class": "org.xmind.branchConnection.roundedElbow",
"line-width": "2pt",
"fill-pattern": "solid",
"border-line-width": "0pt",
"fo:font-family": "NeverMind",
"fo:font-style": "normal",
"fo:font-weight": 500,
"fo:font-size": "18pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "left"
}
},
"subTopic": {
"id":genId(),
"properties": {
"shape-class": "org.xmind.topicShape.roundedRect",
"line-class": "org.xmind.branchConnection.roundedElbow",
"line-width": "2pt",
"fill-pattern": "solid",
"border-line-width": "0pt",
"fo:font-family": "NeverMind",
"fo:font-style": "normal",
"fo:font-weight": 400,
"fo:font-size": "14pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "left"
}
},
"summaryTopic": {
"id": genId(),
"properties": {
"svg:fill": "none",
"border-line-color": "#000229",
"shape-class": "org.xmind.topicShape.roundedRect",
"line-class": "org.xmind.branchConnection.roundedElbow",
"fill-pattern": "solid",
"fo:font-family": "NeverMind",
"fo:font-style": "normal",
"fo:font-weight": "400",
"fo:font-size": "14pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "left"
}
},
"calloutTopic": {
"id": genId(),
"properties": {
"svg:fill": "#000229",
"border-line-color": "#000229",
"callout-shape-class": "org.xmind.calloutTopicShape.balloon.roundedRect",
"fill-pattern": "solid",
"fo:font-family": "NeverMind",
"fo:font-style": "normal",
"fo:font-weight": 400,
"fo:font-size": "14pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "left"
}
},
"floatingTopic": {
"id": genId(),
"properties": {
"svg:fill": "#EEEBEE",
"border-line-color": "#EEEBEE",
"shape-class": "org.xmind.topicShape.roundedRect",
"line-class": "org.xmind.branchConnection.roundedElbow",
"line-width": "2pt",
"line-pattern": "solid",
"fill-pattern": "solid",
"border-line-width": "0pt",
"arrow-end-class": "org.xmind.arrowShape.none",
"fo:font-family": "NeverMind",
"fo:font-style": "normal",
"fo:font-weight": 500,
"fo:font-size": "14pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "left"
}
},
"boundary": {
"id": genId(),
"properties": {
"svg:fill": "#000229",
"line-color": "#000229",
"shape-class": "org.xmind.boundaryShape.roundedRect",
"shape-corner": "20pt",
"line-width": "2",
"line-pattern": "dash",
"fill-pattern": "solid",
"fo:font-family": "'NeverMind','Microsoft YaHei','PingFang SC','Microsoft JhengHei','sans-serif',sans-serif",
"fo:font-style": "normal",
"fo:font-weight": 400,
"fo:font-size": "14pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "center"
}
},
"summary": {
"id":genId(),
"properties": {
"line-color": "#000229",
"shape-class": "org.xmind.summaryShape.round",
"line-width": "2pt",
"line-pattern": "solid",
"line-corner": "8pt"
}
},
"relationship": {
"id": genId(),
"properties": {
"line-color": "#000229",
"shape-class": "org.xmind.relationshipShape.curved",
"line-width": "2",
"line-pattern": "dash",
"arrow-begin-class": "org.xmind.arrowShape.none",
"arrow-end-class": "org.xmind.arrowShape.triangle",
"fo:font-family": "'NeverMind','Microsoft YaHei','PingFang SC','Microsoft JhengHei','sans-serif',sans-serif",
"fo:font-style": "normal",
"fo:font-weight": 400,
"fo:font-size": "13pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "center"
}
},
"map": {
"id": genId(),
"properties": {
"svg:fill": "#FFFFFF",
"multi-line-colors": "#F9423A #F6A04D #F3D321 #00BC7B #486AFF #4D49BE",
"color-list": "#000229 #1F2766 #52CC83 #4D86DB #99142F #245570",
"line-tapered": "none"
}
},
"importantTopic": {
"id": genId(),
"properties": { "svg:fill": "#460400", "fill-pattern": "solid" }
},
"minorTopic": {
"id": genId(),
"properties": { "svg:fill": "#703D00", "fill-pattern": "solid" }
},
"colorThemeId": "Rainbow-#000229-MULTI_LINE_COLORS",
"expiredTopic": {
"id":genId(),
"properties": {
"fo:text-decoration": "line-through",
"svg:fill": "none"
}
},
"global": {
"id": genId(),
"properties": {}
},
"skeletonThemeId": genFather(),
},
"topicPositioning": "fixed",
"coreVersion": "2.74.0",
"relationships": []
}
return o
}
function Manifest(imglist){
let o = new Object()
o = {
"file-entries": {
"content.json": {},
"metadata.json": {},
"Thumbnails/thumbnail.png": {}
}
}
//console.log(o)
for(let i = 0;i < imglist.length;i ++){
o["file-entries"]["resources/" + imglist[i]]= {}
}
//console.log(o)
return o;
}
function MetaData(sheet){
let o = new Object()
o = {
"creator": { "name": "Vana", "version": "22.09.3168.202209272032" },
"activeSheetId": sheet
}
return o
}
let imgList = [];
let zip
let thumbnail = ''
function toZip() {
var file = document.getElementById("fileID");
zip = new JSZip();
let res = getFather()
//return
zip.file('content.json', '[' + JSON.stringify(res) + ']')
zip.file('manifest.json', JSON.stringify(Manifest(imgList)))
zip.file('metadata.json', JSON.stringify(MetaData(res.id)))
zip.file('Thumbnails/thumbnail.png', base64StrToBlob(thumbnail))
zip.generateAsync({
type: "blob",
}).then(function (content) {
saveAs(content, 'mubuexport' + new Date().getTime() + '.xmind');
});
}
(function() {
'use strict';
// Your code here...
var topBox = "<div style='position:fixed;z-index:999999;background-color:#ccc;cursor:pointer;top:0;left:0px;'>" +
"<div id='fun1' style='font-size:12px;padding:12px 30px;color:#FFF;background-color:#25AE84;'>导出</div>" +
"</div>";
$("body").append(topBox);
$("body").on("click", "#fun1", function() {
toZip()
});
// exportRaw('content.json', JSON.stringify(father))
})();
function getFather(){
let title = $(".title").eq(0).html();
let fatherList = $(".node-list:eq(0)").children()
let father = new Father(title)
for(let i = 0;i < fatherList.length;i ++){
let sonList = fatherList.eq(i)
let node = createTree(sonList)
father.rootTopic.children.attached.push(node)
}
return father
}
function genWord(len){
let arr = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z'];
let str = ""
for (let i = 0; i < len; i++) {
let index = Math.round(Math.random() * (arr.length - 1));
str += arr[index];
}
return str;
}
function genId(){
return genWord(8) + '-' + genWord(4) + '-' + genWord(4) + '-' + genWord(4) + '-' + genWord(12)
}
function genFather(){
return genWord(26)
}
function Node(name){
let o = new Object()
o = {
"title": name,
"id": genId(),
"style": {
"id": genId(),
"properties": {
"svg:fill": "none",
"border-line-color": "#232323",
"line-color": "#232323",
"fo:color": "#0A0E16",
"shape-class": "org.xmind.topicShape.underline",
"line-class": "org.xmind.branchConnection.roundedElbow",
"line-width": "2pt",
"line-pattern": "solid",
"fill-pattern": "solid",
"border-line-width": "2pt",
"border-line-pattern": "solid",
"arrow-end-class": "org.xmind.arrowShape.none",
"fo:font-family": "'NeverMind','Microsoft YaHei','PingFang SC','Microsoft JhengHei',sans-serif",
"fo:font-style": "normal",
"fo:font-weight": 400,
"fo:font-size": "14pt",
"fo:text-transform": "manual",
"fo:text-decoration": "none",
"fo:text-align": "left"
},
"children": {
"attached": []
}
},
"children": {
"attached": []
}
}
return o;
}
function createTree(sonNode){
// console.log(sonList)
let title = sonNode.children(".content").text()
let childNum = sonNode.children(".children").length
let noteNum = sonNode.children(".note").length
let imgNum = sonNode.children(".image-list").length
let fatherNode = new Node(title)
if(noteNum != 0){
let nodeTitle = sonNode.children(".note").text()
fatherNode.children.callout= [
{
"id": genId(),
"title": nodeTitle,
"children": {}
}
]
}
if(imgNum != 0){
let imageSrc = sonNode.children(".image-list").children(".image-item").eq(0).children("img").attr("src")
// console.log("imgsrc:" + imageSrc)
let imgUrl = imageSrc
imageSrc = imageSrc.split("/")
imageSrc = imageSrc[imageSrc.length - 1]
imgList.push(imageSrc)
let request = new XMLHttpRequest();
request.overrideMimeType('text/plain; charset=x-user-defined');
request.open('GET', imgUrl, false);
request.send();
let pic = Uint8Array.from(request.response, c => c.charCodeAt(0));
// console.log(pic)
zip.file('resources/' + imageSrc, pic)
fatherNode.image = {
"src": "xap:resources/" + imageSrc
}
}
// console.log(title)
if(childNum != 0){
let sonList = sonNode.children(".children").children(".node-list").children()
for(let i = 0;i < sonList.length;i ++){
let tempNode = sonList.eq(i)
let node = createTree(tempNode)
fatherNode.children.attached.push(node)
}
}
return fatherNode
}
function base64StrToBlob(base64Str) {
const arr = base64Str.split(',');
const mime = arr[0].match(/:(.*?);/)[1];
const bstr = atob(arr[1]);
let n = bstr.length;
let u8arr = new Uint8Array(n);
while (n--) {
u8arr[n] = bstr.charCodeAt(n);
}
return new Blob([u8arr], {
type: mime
});
}