// ==UserScript== // @name U2实时预览BBCODE // @namespace https://u2.dmhy.org/ // @version 1.1.8 // @description 实时预览BBCODE // @author kysdm // @grant GM_xmlhttpRequest // @connect p.sda1.dev // @connect sm.ms // @connect smms.app // @match *://u2.dmhy.org/* // @exclude *://u2.dmhy.org/shoutbox.php* // @icon https://u2.dmhy.org/favicon.ico // @require https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.0/jquery.min.js // @license Apache-2.0 // ==/UserScript== /* 本脚本基于 Bamboo Green 界面风格进行修改 为什么会有近似功能的函数呢,问就是历史原因 等不能跑的时候再动祖传代码 /* /* GreasyFork 地址 https://greasyfork.org/zh-CN/scripts/426268 */ /* 更新日志 https://github.com/kysdm/u2_share/commits/main/u2share_bbcode.user.js */ 'use strict'; (async () => { // 声明全局变量 // https://api.jquery.com/jQuery.noConflict/ const jq = jQuery.noConflict(); // 网站语言 const lang = new lang_init(jq('#locale_selection').val());; // CSS jq('body').append(``); jq('body').append(``); // JS jq('body').append(``); await loadScript('https://cdnjs.cloudflare.com/ajax/libs/localforage/1.10.0/localforage.min.js') await loadScript('https://userscript.kysdm.com/js/mediainfo.js?v=1.0') await loadScript('https://userscript.kysdm.com/js/conversion.js?v=1.0') // DB const db = localforage.createInstance({ name: "bbcodejs" }); const attachmap_db = localforage.createInstance({ name: "attachmap" }); const history_db = localforage.createInstance({ name: "history" }); // 现存BBCODE元素 (async () => { if (jq('.bbcode').length === 0) return; // 判断页面是否存在 bbcode 输入框 new init(); const url = location.href.match(/u2\.dmhy\.org\/(upload|forums|comment|contactstaff|sendmessage|edit)\.php/i); if (url === null) return; await syncScroll('#bbcodejs_tbody', url[1], '.bbcode', '#bbcode2'); if (url[1] === 'upload') { await autoSaveUpload(); } else { await autoSaveMessage('#bbcodejs_tbody', '.bbcode', '#qr', url[1], '#compose'); } jq('.bbcode').parents("tr:eq(1)").after('' + lang['preview'] + '
' + '
' + '
' + await bbcode2html(jq('.bbcode').val()) + '
'); syncWindowChange('.bbcode', '#bbcode2'); jq('.bbcode').bind('input propertychange', async function updateValue() { let html = await bbcode2html(jq(this).val()); jq('#bbcode2').children('.child').html(html); }); jq('.codebuttons').click(async function updateValue() { let html = await bbcode2html(jq('.bbcode').val()); jq('#bbcode2').children('.child').html(html); }); jq('#compose').find("td.embedded.smile-icon a").each(function () { jq(this).attr('href', jq(this).attr('href').replace(/javascript: SmileIT\('\[(.+?)\]','[^']+?','[^']+?'\)/gis, function (s, x) { return `javascript:void('${x}');` })); }) .click(async function () { onEditorActionBox(jq(this).attr('href'), '.bbcode'); jq('#bbcode2').children('.child').html(await bbcode2html(jq('.bbcode').val())); }); if (/u2\.dmhy\.org\/upload\.php/i.test(location.href)) { // 添加中括号 function add_brackets(txt) { if (txt === '') { return ''; } else { return '[' + txt + ']'; } }; // 检查重叠的中括号 function check_title(txt) { if (/\[{2,}|\]{2,}/g.test(txt)) { return '' + txt + ''; } else { return txt; } }; var main_title = '' + lang['select_type'] + ''; function addMainTitle() { const custom_title = jq('#custom_title').val(); let type_id = jq('#browsecat').val(); if (custom_title !== '') { main_title = `${custom_title}` } else if (type_id === '0') { main_title = '' + lang['select_type'] + ''; } else if (['9', '411', '413', '12', '13', '14', '15', '16', '17', '410', '412'].indexOf(type_id) !== -1) { main_title = '' + add_brackets(jq('#anime_chinese-input').val()) + add_brackets(jq('#anime_english-input').val()) + add_brackets(jq('#anime_original-input').val()) + add_brackets(jq('#anime_source-input').val()) + add_brackets(jq('#anime_resolution-input').val()) + add_brackets(jq('#anime_episode-input').val()) + add_brackets(jq('#anime_container-input').val()) + add_brackets(jq('#anime_extra-input').val()) + ''; } else if (['21', '22', '23'].indexOf(type_id) !== -1) { main_title = '' + add_brackets(jq('#manga_title-input').val()) + add_brackets(jq('#manga_author-input').val()) + add_brackets(jq('#manga_volume-input').val()) + add_brackets(jq('#manga_ended').find("select").val()) + add_brackets(jq('#manga_publisher-input').val()) + add_brackets(jq('#manga_remark-input').val()) + ''; } else if (type_id === '30') { var prefix_1 = jq('#music_prefix').find("select").val(); var prefix_2 = jq('#music_collection').find("select").val(); if (['EAC', 'XLD'].indexOf(prefix_1) !== -1) { var music_quality = false; } else if (['Hi-Res', 'Web'].indexOf(prefix_1) !== -1) { var music_quality = true; }; switch (prefix_2) { case "0": // 单张 main_title = '' + add_brackets(prefix_1) + add_brackets(jq('#music_date-input').val()) + add_brackets(jq('#music_category-input').val()) + add_brackets(jq('#music_artist-input').val()) + add_brackets(jq('#music_title-input').val()) + add_brackets(jq('#music_serial_number-input').val()) + add_brackets((() => { if (music_quality) { return jq('#music_quality-input').val(); } else { return ''; } })()) + add_brackets(jq('#music_format-input').val()) + ''; break; case "1": // 合集 main_title = '' + add_brackets(prefix_1) + add_brackets('合集') + add_brackets(jq('#music_category-input').val()) + add_brackets(jq('#music_title-input').val()) + add_brackets(jq('#music_quantity-input').val()) + add_brackets((() => { if (music_quality) { return jq('#music_quality-input').val(); } else { return ''; } })()) + ''; break; } } else if (type_id === '40') { main_title = '' + jq('#other_title-input').val() + ''; } jq('#checktitle').html(check_title(main_title)); } jq("#browsecat").change(() => { new addMainTitle; }); jq(".torrent-info-input").bind('input propertychange', () => { new addMainTitle; }); jq('#other_title').after('' + lang['main_title'] + '' + '' + main_title + '' ); const token = await history_db.getItem('token'); let token_waring = '' if (token === null || token.length !== 96) { token_waring = `API Token 不存在或无效  鉴权脚本 (安装后打开任意种子页面触发鉴权)`}; jq('#anime_chinese').before(` 引用种子 ${token_waring}
输入种子ID,复制该种子的描述信息。 自定义标题
除非你确切知道你在做什么,否则请不要在此处输入任何内容。 ` ) const browsecat_options = {}; const uid = jq("#info_block a:first").attr("href").match(/id=(\d+)/)[1]; jq("#browsecat option").each(function () { const text = jq(this).text(); const value = jq(this).val(); browsecat_options[text] = value; }); jq("#custom_title").bind('input propertychange', () => { new addMainTitle; }); // console.log(browsecat_options); jq('#copyButton').click(async function (ev) { ev.preventDefault(); // 阻止表单的提交行为 let tid = jq('#copytorrentinfo').val().trim();; if (isNaN(tid)) { window.alert('无效种子ID'); return; } let api = await getApi(token, uid, tid); if (api.msg !== 'success') { window.alert(`API获取发生错误\n\n${api.msg}`); console.log(api); return; } let torrents = api.data.history; if (Object.keys(torrents).length === 0) { window.alert('API没有此种子数据'); return; } jq("#compose input[id]").map(function () { // 预先清空所有字段 if (this.id.endsWith("-input") || this.id === 'poster') jq(`#${this.id}`).val(''); jq('#custom_title').val(torrents[0].title) jq('[name="small_descr"]').val(torrents[0].subtitle); jq('[name="anidburl"]').val(torrents[0].anidb === null ? '' : `https://anidb.net/anime/${torrents[0].anidb}`); jq('#browsecat').val(browsecat_options[torrents[0]['category']]); document.getElementById('browsecat').dispatchEvent(new Event('change')); // 手动触发列表更改事件 jq('.bbcode').val(torrents[0].description_info); jq('[class^="torrent-info-input"]').trigger("input"); // 手动触发标题更改 jq('.bbcode').trigger("input"); // 手动触发bbcode更改 }); }) // 种子文件 jq('#torrent').parent().html(`
`); jq('#compose').find("tr:eq(1)").after(` 种子信息
-
种子列表
[全部展开] - `); await loadScript('https://userscript.kysdm.com/js/torrent-creator.js?v=1.1') jq('.progress').css({ 'width': '99%', 'height': '8px', 'border': '1px solid #ccc', 'border-radius': '5px', // 圆角 'margin': '8px 2px', 'overflow': 'hidden', }); jq('.progress > div').css({ 'width': '0px', 'height': '100%', 'background-color': '#8db8ff', 'transition': 'all 300ms ease' }); // 设置进度条颜色 jq('[name="progress"]').hide(); // 隐藏进度条 // 显示上传的文件名 & 去除其余上传框内的值 jq('#torrent').change(async function () { const response = await fetch(URL.createObjectURL(this.files[0])); const torrent_blob = await response.blob(); console.log(torrent_blob); await db.setItem(`upload_autoSaveMessageTorrentBlob`, torrent_blob); await db.setItem(`upload_autoSaveMessageTorrentName`, this.files[0].name); jq('#upload_chooser').text(this.files[0].name); jq('#upload_chooser').prop('title', this.files[0].name); jq('#filechooser').val(''); jq('#folderchooser').val(''); jq('#qr').attr('disabled', false); // 解除上传按钮限制 jq('#torrent_download').attr('disabled', false); // 解除按钮禁用 jq('#upload_torrent,#upload_file,#upload_folder,#torrent_create').attr('disabled', true); // 禁止其余种子处理按钮 await pageTorrentInfo(); }); jq('#filechooser').change(function () { const encoder = new TextEncoder(); const maxPathName = this.files[0].name; const maxPathUtf8Bytes = encoder.encode(maxPathName).length; if (maxPathUtf8Bytes > 230 && !confirm(`文件路径超长,是否继续?\n\n${maxPathUtf8Bytes}\n\n${maxPathName}`)) return; jq('#upload_chooser').text(this.files[0].name); jq('#upload_chooser').prop('title', this.files[0].name); jq('#torrent').val(''); jq('#folderchooser').val(''); }); jq('#folderchooser').change(function () { const encoder = new TextEncoder(); let maxPathUtf8Bytes = 0; let maxPathName = new Array(); for (const file of this.files) { const currentPath = file.webkitRelativePath; const currentPathUtf8Bytes = encoder.encode(currentPath).length; if (maxPathUtf8Bytes < currentPathUtf8Bytes) { maxPathUtf8Bytes = currentPathUtf8Bytes; maxPathName = [currentPath]; } else if (maxPathUtf8Bytes === currentPathUtf8Bytes) { maxPathName.push(currentPath); }; }; if (maxPathUtf8Bytes > 230 && !confirm(`文件路径超长,是否继续?\n\n${maxPathUtf8Bytes}\n\n${maxPathName.join('\n')}`)) return; jq('#upload_chooser').text((this.files[0].webkitRelativePath).split("/")[0]); jq('#upload_chooser').prop('title', (this.files[0].webkitRelativePath).split("/")[0]); jq('#torrent').val(''); jq('#filechooser').val(''); }); jq('#qr').attr('disabled', true); // 未上传种子前,禁止上传按钮 // 种子创建按钮 jq('#torrent_create').click(async function () { let file = jq('#filechooser')[0].files; let folder = jq('#folderchooser')[0].files; if (file.length !== 0) { torrent_start(); await CreateTorrentFile(file); await pageTorrentInfo(); } else if (folder.length !== 0) { torrent_start(); await CreateTorrentFolder(folder); await pageTorrentInfo(); } else { window.alert('没有选择任何文件'); return; }; }); // 清空 jq('#torrent_clean').click(async function () { jq('#torrent').val(''); jq('#filechooser').val(''); jq('#folderchooser').val(''); jq('#upload_chooser').text(''); jq('#upload_chooser').prop('title', ''); jq('[name="progress"]').hide(); // 隐藏进度条 jq('#upload_torrent,#upload_file,#upload_folder,#torrent_create').attr('disabled', false); jq('#torrent_download,#qr').attr('disabled', true); // 禁用按钮 jq('.progress > div').css('width', "0%"); jq('[name="progress-total"],[name="progress-name"],[name="progress-percent"]').text(''); jq('#download_link').attr('href', 'javascript:void(0)').attr('download', ''); // 删除下载按钮的URL await db.removeItem(`upload_autoSaveMessageTorrentBlob`); await db.removeItem(`upload_autoSaveMessageTorrentName`); jq('#torrentinfo1').html('-'); jq('#torrentinfo2').html(''); jq('#torrentinfo3').html(''); jq('#file_tree').html('-'); }); var downloadUrl; jq('#torrent_download').click(async function () { let a_1 = document.getElementById("download_link"); const blob = await db.getItem(`upload_autoSaveMessageTorrentBlob`); const filename = await db.getItem(`upload_autoSaveMessageTorrentName`); window.URL.revokeObjectURL(downloadUrl); downloadUrl = window.URL.createObjectURL(blob); a_1.href = downloadUrl; a_1.download = filename; a_1.click(); }); const torrent_start = () => { jq('#upload_torrent,#upload_file,#upload_folder,#torrent_create,#torrent_download,#torrent_clean').attr('disabled', true); jq('[name="progress"]').show(); }; // 拖拽 jq('#compose > table > tbody > tr:lt(5)').on({ dragenter: function (e) { e.preventDefault(); e.stopPropagation(); }, dragover: function (e) { e.preventDefault(); e.stopPropagation(); }, drop: async function (e) { e.preventDefault(); e.stopPropagation(); // 递归扫描文件夹 const filesList = new Array(); const encoder = new TextEncoder(); const scanFiles = (entry) => { return new Promise(async (resolve, reject) => { if (entry.isDirectory) { const directoryReader = entry.createReader() const read = () => { return new Promise((resolve, reject) => { directoryReader.readEntries( async (entries) => { for (let i = 0; i <= entries.length - 1; i++) await scanFiles(entries[i]); resolve(entries); }, (e) => { reject(e) } ); }); }; const entries = await read(); if (entries.length > 0) await read(); // console.log('完成读取当前文件夹'); resolve(); } else { entry.file( async (file) => { const path = entry.fullPath.substring(1); const newFile = Object.defineProperty(file, 'webkitRelativePath', { value: path, }); filesList.push(newFile) resolve(); }, (e) => { reject(e) } ); }; }); }; let f = e.originalEvent.dataTransfer.files; // 获取文件对象 if (f.length === 0) return false; if (f.length !== 1) { window.alert('只允许单个文件/文件夹'); return false; }; let items = e.originalEvent.dataTransfer.items; for (let i = 0; i <= items.length - 1; i++) { // 实际这个循环只会运行一次 <不允许多文件夹上传> let item = items[i]; if (item.kind === "file") { let entry = item.webkitGetAsEntry(); await scanFiles(entry); }; }; if (filesList.length === 1) { // 可能是单文件也可能是文件夹内只有一个文件 if (filesList[0].webkitRelativePath === filesList[0].name) { if (filesList[0].name.toLowerCase().match(/.+\.torrent$/)) { // console.log('是种子文件'); const response = await fetch(URL.createObjectURL(filesList[0])); const torrent_blob = await response.blob(); await db.setItem(`upload_autoSaveMessageTorrentBlob`, torrent_blob); await db.setItem(`upload_autoSaveMessageTorrentName`, filesList[0].name); jq('#upload_chooser').text(filesList[0].name); jq('#upload_chooser').prop('title', filesList[0].name); jq('#qr').attr('disabled', false); // 解除上传按钮锁定 jq('#torrent_download').attr('disabled', false); // 解除按钮禁用 jq('#upload_torrent,#upload_file,#upload_folder,#torrent_create').attr('disabled', true); // 禁止其余种子处理按钮 await pageTorrentInfo(); } else { // console.log('是普通单文件'); const maxPathName = filesList[0].name; const maxPathUtf8Bytes = encoder.encode(maxPathName).length; if (maxPathUtf8Bytes > 230 && !confirm(`文件路径超长,是否继续?\n\n${maxPathUtf8Bytes}\n\n${maxPathName}`)) return; jq('#upload_chooser').text(filesList[0].name); jq('#upload_chooser').prop('title', filesList[0].name); torrent_start(); await CreateTorrentFile(filesList); jq('#qr').attr('disabled', false); jq('#torrent_download').attr('disabled', false); await pageTorrentInfo(); }; } else { // console.log('文件夹内有一个文件'); const maxPathName = filesList[0].webkitRelativePath const maxPathUtf8Bytes = encoder.encode(maxPathName).length; if (maxPathUtf8Bytes > 230 && !confirm(`文件路径超长,是否继续?\n\n${maxPathUtf8Bytes}\n\n${maxPathName}`)) return; jq('#upload_chooser').text((filesList[0].webkitRelativePath).split("/")[0]); jq('#upload_chooser').prop('title', (filesList[0].webkitRelativePath).split("/")[0]); torrent_start(); await CreateTorrentFolder(filesList); jq('#qr').attr('disabled', false); jq('#torrent_download').attr('disabled', false); await pageTorrentInfo(); }; } else { // console.log('文件夹内有多个文件'); let maxPathUtf8Bytes = 0; let maxPathName = new Array(); for (const file of filesList) { const currentPath = file.webkitRelativePath; const currentPathUtf8Bytes = encoder.encode(currentPath).length; if (maxPathUtf8Bytes < currentPathUtf8Bytes) { maxPathUtf8Bytes = currentPathUtf8Bytes; maxPathName = [currentPath]; } else if (maxPathUtf8Bytes === currentPathUtf8Bytes) { maxPathName.push(currentPath); }; }; if (maxPathUtf8Bytes > 230 && !confirm(`文件路径超长,是否继续?\n\n${maxPathUtf8Bytes}\n\n${maxPathName.join('\n')}`)) return; jq('#upload_chooser').text((filesList[0].webkitRelativePath).split("/")[0]); jq('#upload_chooser').prop('title', (filesList[0].webkitRelativePath).split("/")[0]); torrent_start(); await CreateTorrentFolder(filesList); jq('#qr').attr('disabled', false); jq('#torrent_download').attr('disabled', false); await pageTorrentInfo(); }; } }); jq('#qr').click(async function (e) { e.preventDefault(); this.disabled = true; // 禁止按钮重复点击 const torrentBlob = await db.getItem(`upload_autoSaveMessageTorrentBlob`); if (torrentBlob === null || typeof torrentBlob === 'undefined') { window.alert('种子文件不存在,在其他页面点击清除按钮了?'); return; } console.log(new File([torrentBlob], "a.torrent", { type: "application/octet-stream" })); const p = () => { return new Promise(function (resolve, reject) { // https://developer.mozilla.org/zh-CN/docs/Web/API/FormData let formdata = new FormData(document.getElementById('compose')); // 处理自定义标题的结构 let customTitle = formdata.get("custom_title"); if (!isWhitespace(customTitle)) { // 自定义标题中有数据时 const category = formdata.get("type"); customTitle = customTitle.replace(/^\[|\]$/g, ''); // 去除自定义标题两边的括号,提交后系统会自动补全 if (['9', '411', '413', '12', '13', '14', '15', '16', '17', '410', '412'].includes(category)) { for (let pair of formdata.entries()) if (pair[0].startsWith('anime_')) formdata.set(pair[0], ''); formdata.set("anime_chinese", customTitle); } else if (['21', '22', '23'].includes(category)) { for (let pair of formdata.entries()) if (pair[0].startsWith('manga_')) formdata.set(pair[0], ''); formdata.set("manga_title", customTitle); } else if (category === '30') { for (let pair of formdata.entries()) if (pair[0].startsWith('music_')) formdata.set(pair[0], ''); formdata.set("music_title", customTitle); } else if (category === '40') { formdata.set("other_title", customTitle); } } if (torrentBlob) formdata.set("file", new File([torrentBlob], "a.torrent", { type: "application/octet-stream" })); const request = new XMLHttpRequest(); request.open("POST", "takeupload.php"); request.timeout = 5000; // 超时时间 单位毫秒 request.onload = function () { if (request.status >= 200 && request.status < 300) { resolve({ status: request.status, response: request.response, responseURL: request.responseURL }); } else { reject({ status: request.status, statusText: request.statusText }); }; }; request.onerror = function () { reject({ status: request.status, statusText: request.statusText }); }; request.ontimeout = function () { reject({ status: 408, statusText: "Request timed out" }); }; request.send(formdata); }); }; p().then(async r => { if (!r.responseURL.includes("takeupload.php")) { // 成功上传 // console.log('成功上传'); clearInterval(jq(`#upload_auto_save_text`).attr('title')); // 停止自动保存 await db.removeItem(`upload_autoSaveMessageTime`); await db.removeItem(`upload_autoSaveMessageBbcode`); await db.removeItem(`upload_autoSaveMessageSmallDescr`); await db.removeItem(`upload_autoSaveMessagePoster`); await db.removeItem(`upload_autoSaveMessageAnidbUrl`); await db.removeItem(`upload_autoSaveMessageInfo`); await db.removeItem(`upload_autoSaveMessageTorrentBlob`); await db.removeItem(`upload_autoSaveMessageTorrentName`); // console.log(`upload-已清空保存的记录`); window.open(r.responseURL, '_self'); return; }; const h = document.createElement('div'); h.innerHTML = r.response; let warn = jq(h).find('#outer').text(); warn = warn ? warn.trim() : warn; window.alert(warn) console.log(warn); this.disabled = false; // 解除按钮禁止点击 }).catch(e => { console.error(e); window.alert('上传发生错误\n' + e) this.disabled = false; // 解除按钮禁止点击 }); }); }; function init() { const type = 'original'; let h1 = jq('.codebuttons').eq(6).parent().html(); let h2 = jq('.codebuttons').eq(7).parent().html(); let h3 = jq('.codebuttons').eq(8).parent().html(); jq('.codebuttons').eq(8).parent().remove(); jq('.codebuttons').eq(7).parent().remove(); jq('.codebuttons').eq(6).parent().remove(); jq('input[value="URL"]').parent().remove(); jq('input[value="IMG"]').parent() .before(``) .before(``) .after(``) .after(``) .after(``) .after(``) .after(``); jq('input[value="QUOTE"]').parent() .after(``) .after(``) .after(``) .after(``) .after(``); jq('.codebuttons').parents('table').eq(0).after('
'); jq('#bbcodejs_tbody').append('
' + h1 + h2 + h3 + '
'); const margin = jq('.codebuttons').parents('tbody').eq(0).width() - jq("#bbcodejs_select").width() - 2.6; jq("#bbcodejs_select").css("margin-left", margin + "px"); jq(`[name="${type}_bbcode_button"]`).click(function () { onEditorActionBox(this.value, `.bbcode`); }); } })(); async function bbcode2html(bbcodestr) { var tempCode = new Array(); var tempCodeCount = 0; let lost_tags = new Array(); function addTempCode(value) { tempCode[tempCodeCount] = value; let returnstr = ""; tempCodeCount++; return returnstr; }; const escape_reg = new RegExp("[&\"\'<>]", "g"); bbcodestr = bbcodestr.replace(escape_reg, function (s, x) { switch (s) { case '&': return '&'; case '"': return '"'; case "'": return '''; case '<': return '<'; case '>': return '>'; default: return s; }; }); bbcodestr = bbcodestr.replace(/\r\n/g, () => { return '
' }); bbcodestr = bbcodestr.replace(/\n/g, () => { return '
' }); bbcodestr = bbcodestr.replace(/\r/g, () => { return '
' }); bbcodestr = bbcodestr.replace(/ /g, '  '); let br_end = ''; // 对结尾的换行符进行计数 let br; if (br = bbcodestr.match(/(?:
)+$/)) { br_end = br[0]; const regex = new RegExp(`${br_end}$`, ""); bbcodestr = bbcodestr.replace(regex, ''); }; const checkLostTags = (value, r_tag_start, r_tag_end) => { let state = false; let r_tag_start_exec = r_tag_start.exec(value); let index_start = r_tag_start_exec ? (r_tag_start_exec.index + r_tag_start_exec[0].length) : 0; let r_tag_end_exec = r_tag_end.exec(value.slice(index_start)); if (r_tag_start_exec && !r_tag_end_exec) { let tag_start_val = r_tag_start_exec.groups.tag;; console.log('检测到丢失的标签 => ' + `[/${tag_start_val}]`); lost_tags.push(`[/${tag_start_val}]`); state = true; }; return { "state": state }; }; const url = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { textarea = textarea.replace(/\[url=.(?:"){0,2}\]/i, function (s) { return '[url]'; }); } if (val) { const lost = checkLostTags(textarea, /\[(?url)=[^\[]*?/i, /\[\/(?url)\]/i); if (lost.state) { return textarea.replace(/\[url=[^\[]*?/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[url=(.+?)\](.*?)\[\/url\]/i, function (all, url, text) { if (url.match(/\s|\[/)) return addTempCode(all); let tmp = url.replace(/^(?:")?(.*?)(?:")?$/, "$1"); if (!tmp.match(/"/)) url = tmp; else { if (url.match(/"/g).length === 1) url = url.replace('"', ''); } return addTempCode('' + text + ''); }); } else { const lost = checkLostTags(textarea, /\[(?url)\]/i, /\[\/(?url)\]/i); if (lost.state) { return textarea.replace(/\[url\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[url\](.+?)\[\/url\]/i, function (s, x) { if (x.match(/\s|\[/i)) return addTempCode(s); return addTempCode('' + x + ''); }); }; }; // 注释 const rt = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { return textarea.replace(/\[rt=.*?\]/i, function (s) { return addTempCode(s); }); } else if (!val) { return textarea.replace('[rt]', function (s) { return addTempCode(s); }) } else { const lost = checkLostTags(textarea, /\[(?rt)=[^\[]*?/i, /\[\/(?rt)\]/i); if (lost.state) { return textarea.replace(/\[rt=[^\[]*?/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[rt=(.+?)\](.*?)\[\/rt\]/i, function (all, tval, text) { if (tval.match(/\[/i)) return addTempCode(all); let tmp = tval.replace(/^(?:")?(.*?)(?:")?$/, "$1"); if (!tmp.match(/"/)) tval = tmp; return addTempCode('' + text + '(' + tval + ')'); }); }; }; // 字体 const font = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { return textarea.replace(/\[font=.*?]/i, function (s) { return addTempCode(s); }); } else if (!val) { return textarea.replace('[font]', function (s) { return addTempCode(s); }) } else { const lost = checkLostTags(textarea, /\[(?font)=[^\[]*?\]/i, /\[\/(?font)\]/i); if (lost.state) { return textarea.replace(/\[font=[^\[]*?/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[font=(.+?)\](.*?)\[\/font\]/i, function (all, tval, text) { if (tval.match(/\[/i)) return '[' + addTempCode(`font=`) + `${tval}]${text}`; let tmp = tval.replace(/^(?:")?(.*?)(?:")?$/, "$1"); if (!/"/.test(tmp)) { tval = tmp; } else { if (tval.match(/"/g).length === 1) tval = tval.replace('"', ''); }; return '' + text + ''; }); }; }; // 颜色 const color = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { return textarea.replace(/\[color=.*?\]/i, function (s) { return addTempCode(s); }); } else if (!val) { return textarea.replace('[color]', function (s) { return addTempCode(s); }) } else { const lost = checkLostTags(textarea, /\[(?color)=[^\[]*?\]/i, /\[\/(?color)\]/i); if (lost.state) { return textarea.replace(/\[color=[^\[]*?\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[color=(.+?)\](.*?)\[\/color\]/i, function (all, tval, text) { if (tval.match(/\[/i)) return addTempCode(all);; let tmp = tval.replace(/^(?:")?(.*?)(?:")?$/, "$1"); if (!/"/.test(tmp)) { tval = tmp; } else { if (tval.match(/"/g).length === 1) tval = tval.replace('"', ''); }; return '' + text + ''; }); }; }; // 文字大小 const size = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { return textarea.replace(/\[size=.*?\]/i, function (s) { return addTempCode(s); }); } else if (!val) { return textarea.replace('[size]', function (s) { return addTempCode(s); }) } else { const lost = checkLostTags(textarea, /\[(?size)=[^\[]*?\]/i, /\[\/(?size)\]/i); if (lost.state) { return textarea.replace(/\[size=[^\[]*?\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[size=(.+?)\](.*?)\[\/size\]/i, function (all, tval, text) { // size只允许1-9的数字 if (!tval.match(/^(?:")?[0-9](?:")?$/)) return addTempCode(all); let tmp = tval.replace(/^(?:")?(.*?)(?:")?$/, "$1"); if (!/"/.test(tmp)) { tval = tmp; } else { if (tval.match(/"/g).length === 1) tval = tval.replace('"', ''); }; return '' + text + ''; }); }; }; const pre = (val, textarea) => { if (val) { return textarea.replace(/\[pre=(.*?)\]/i, function (s, v) { return addTempCode('[pre=') + v + ']'; }); }; const lost = checkLostTags(textarea, /\[(?pre)\]/i, /\[\/(?pre)\]/i); if (lost.state) { return textarea.replace(/\[pre\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[pre\](.*?)\[\/pre\]/i, function (all, text) { return '
' + text + '
'; }); }; const b = (val, textarea) => { if (val) { return textarea.replace(/\[b=(.*?)\]/i, function (s, v) { return addTempCode('[b=') + v + ']'; }); }; const lost = checkLostTags(textarea, /\[(?b)\]/i, /\[\/(?b)\]/i); if (lost.state) { return textarea.replace(/\[b\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[b\](.*?)\[\/b\]/i, function (all, text) { return '' + text + ''; }); }; const i = (val, textarea) => { if (val) { return textarea.replace(/\[i=(.*?)\]/i, function (s, v) { return addTempCode('[i=') + v + ']'; }); }; const lost = checkLostTags(textarea, /\[(?i)\]/i, /\[\/(?i)\]/i); if (lost.state) { return textarea.replace(/\[i\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[i\](.*?)\[\/i\]/i, function (all, text) { return '' + text + ''; }); }; const u = (val, textarea) => { if (val) { return textarea.replace(/\[u=(.*?)\]/i, function (s, v) { return addTempCode('[u=') + v + ']'; }); }; const lost = checkLostTags(textarea, /\[(?u)\]/i, /\[\/(?u)\]/i); if (lost.state) { return textarea.replace(/\[u\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[u\](.*?)\[\/u\]/i, function (all, text) { return '' + text + ''; }); }; const s = (val, textarea) => { if (val) { return textarea.replace(/\[s=(.*?)\]/i, function (s, v) { return addTempCode('[s=') + v + ']'; }); }; const lost = checkLostTags(textarea, /\[(?s)\]/i, /\[\/(?s)\]/i); if (lost.state) { return textarea.replace(/\[s\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[s\](.*?)\[\/s\]/i, function (all, text) { return '' + text + ''; }); }; const img = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { return textarea.replace(/\[img=.*?\]/i, function (s) { return addTempCode(s); }); } else if (val) { return textarea.replace(/\[img=(.*?)\]/i, function (all, url) { // [img=http://u2.dmhy.org/pic/logo.png] url = url.replace('&', '&'); if (/^((?!"|'|>|<|;|#).)+\.(?:png|jpg|jpeg|gif|svg|bmp|webp)$/i.test(url)) { // url 以 .png 之类结尾 return addTempCode('image'); } else { return addTempCode(all); }; }); } else { // [img]http://u2.dmhy.org/pic/logo.png[/img] const lost = checkLostTags(textarea, /\[(?img)\]/i, /\[\/(?img)\]/i); if (lost.state) { return textarea.replace(/\[img\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[img\](.*?)\[\/img\]/i, function (all, url) { url = url.replace('&', '&'); if (/^((?!"|'|>|<|;|#).)+\.(?:png|jpg|jpeg|gif|svg|bmp|webp)$/i.test(url)) { // url 以 .png 之类结尾 return addTempCode('image'); } else { return addTempCode(all); }; }); }; }; const imglnk = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { return textarea.replace(/\[imglnk=.*?\]/i, function (s) { return addTempCode(s); }); } else if (val) { return textarea.replace(/\[imglnk=(.*?)\]/i, function (all, url) { return addTempCode('[imglnk=') + url + ']'; }); } else { // [img]http://u2.dmhy.org/pic/logo.png[/img] const lost = checkLostTags(textarea, /\[(?imglnk)\]/i, /\[\/(?imglnk)\]/i); if (lost.state) { return textarea.replace(/\[imglnk\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[imglnk\](.*?)\[\/imglnk\]/i, function (all, url) { url = url.replace('&', '&'); if (/^((?!"|'|>|<|;|\[|\]|#).)+\.(?:png|jpg|jpeg|gif|svg|bmp|webp)$/i.test(url)) { // url 以 .png 之类结尾 return addTempCode(`image`); } else { return addTempCode(all); }; }); }; }; const code = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { textarea = textarea.replace(/\[code=(?:"){0,2}/, '[code]'); }; if (val) { textarea = textarea.replace(/\[code=(.*?)\]/i, function (s, v) { return addTempCode('[code=') + v + ']'; }); }; const lost = checkLostTags(textarea, /\[(?code)\]/i, /\[\/(?code)\]/i); if (lost.state) { return textarea.replace(/\[code\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[code\](.*?)\[\/code\]/i, function (all, text) { return addTempCode(`
${lang['code']}
${text.replace(/  /g, ' ')}

`); }); }; const info = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { textarea = textarea.replace(/\[info=(?:"){0,2}/, '[info]'); }; if (val) { textarea = textarea.replace(/\[info=(.*?)\]/i, function (s, v) { return addTempCode('[info=') + v + ']'; }); }; const lost = checkLostTags(textarea, /\[(?info)\]/i, /\[\/(?info)\]/i); if (lost.state) { return textarea.replace(/\[info\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[info\](.*?)\[\/info\]/i, function (all, text) { return addTempCode(`
${lang['info']}${text.replace(/  /g, ' ')}
`); }); }; const mediainfo = (val, textarea) => { if (val === '=' || val === '="' || val === '=""') { textarea = textarea.replace(/\[mediainfo=(?:"){0,2}/, '[mediainfo]'); }; if (val) { textarea = textarea.replace(/\[mediainfo=(.*?)\]/i, function (s, v) { return addTempCode('[mediainfo=') + v + ']'; }); }; const lost = checkLostTags(textarea, /\[(?mediainfo)\]/i, /\[\/(?mediainfo)\]/i); if (lost.state) { return textarea.replace(/\[mediainfo\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[mediainfo\](.*?)\[\/mediainfo\]/i, function (all, text) { return addTempCode(`
${lang['mediainfo']}${text.replace(/  /g, ' ')}
`); }); }; const quote = (val, textarea) => { if (!val) { // [quote]我爱U2分享園@動漫花園。[/quote] const lost = checkLostTags(textarea, /\[(?quote)]/i, /\[\/(?quote)\]/i); if (lost.state) { return textarea.replace(/\[quote\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[quote\](.*?)\[\/quote\]/i, function (s, x) { return '
' + lang['quote'] + '' + x.replace(/(
)*$/, '') + '
'; }); } else if (val === '=' || val === '="' || val === '=""') { // [quote=""]我爱U2分享園@動漫花園。[/quote] const lost = checkLostTags(textarea, /\[(?quote)=[^\[]*?\]/i, /\[\/(?quote)\]/i); if (lost.state) { return textarea.replace(/\[quote=[^\[]*?\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[quote=[^\[]*?\](.*?)\[\/quote\]/i, function (s, x) { return '
' + lang['quote'] + '' + x.replace(/(
)*$/, '') + '
'; }); } else { // [quote="ABC"]我爱U2分享園@動漫花園。[/quote] const lost = checkLostTags(textarea, /\[(?quote)=[^\[]*?\]/i, /\[\/(?quote)\]/i); if (lost.state) { return textarea.replace(/\[quote=[^\[]*?\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[quote=([^\[]*?)\](.*?)\[\/quote\]/i, function (all, tval, text) { if (tval.match(/\[/i)) return addTempCode(all);; let tmp = tval.replace(/^(?:")?(.*?)(?:")?$/, "$1"); if (!/"/.test(tmp)) { tval = tmp; }; return '
' + lang['quote'] + ': ' + tval + '' + text.replace(/(
)*$/, '') + '
'; }); }; }; const spoiler = (val, textarea) => { if (!val) { // [spoiler]我要剧透了![/spoiler] const lost = checkLostTags(textarea, /\[(?spoiler)]/i, /\[\/(?spoiler)\]/i); if (lost.state) { return textarea.replace(/\[spoiler\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[spoiler\](.*?)\[\/spoiler\]/i, function (s, x) { return `` + `` + `` + `
${lang['spoiler']}  ` + `
${x.replace(/(
)*$/, '')}
`; }); } else if (val === '=' || val === '="' || val === '=""') { // [spoiler=""]真的![/spoiler] const lost = checkLostTags(textarea, /\[(?spoiler)=.+?\]/i, /\[\/(?spoiler)\]/i); if (lost.state) { return textarea.replace(/\[spoiler=[^\[]*?\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[spoiler=.*?\](.*?)\[\/spoiler\]/i, function (s, x) { return `` + `` + `` + `
${lang['spoiler']}  ` + `
${x.replace(/(
)*$/, '')}
`; }); } else { // [spoiler="剧透是不可能的!"]真的![/spoiler] const lost = checkLostTags(textarea, /\[(?spoiler)=.+?\]/i, /\[\/(?spoiler)\]/i); if (lost.state) { return textarea.replace(/\[spoiler=[^\[]*?\]/i, function (s) { return addTempCode(s); }); }; return textarea.replace(/\[spoiler=(.*?)\](.*?)\[\/spoiler\]/i, function (all, tval, text) { if (tval.match(/\[/i)) return addTempCode(all);; let tmp = tval.replace(/^(?:")?(.*?)(?:")?$/, "$1"); if (!/"/.test(tmp)) tval = tmp; return `` + `` + `` + `
${tval}  ` + `
${text.replace(/(
)*$/, '')}
`; }); }; }; // 附件 const attach = async (val, textarea) => { const lost = checkLostTags(textarea, /\[(?attach)\]/i, /\[\/(?attach)\]/i); if (lost.state) { return textarea.replace(/\[attach\]/i, function (s) { return addTempCode(s); }); }; return await replaceAsync(textarea, /\[attach(?=[^\]]*?)?\](?.*?)\[\/attach\]/i, async (...args) => { const { tag, hash } = args.slice(-1)[0]; if (tag) { return '[' + addTempCode(`attach`) + tag + `]${hash}[/attach]`; }; if (/
/.test(hash)) { return addTempCode(`[attach]`) + hash + addTempCode('[/attach]'); }; if (!hash) { console.log('内部为空'); return addTempCode(args[0]); }; // attach 标签内为空时 if (!/^\w{32}$/.test(hash)) { return `
附件 ${hash} 无效。
`; }; // attach 标签内hash不符合要求 return await attachmap_db.getItem(hash).then(async (value) => { if (value !== null && value.attach_id) { // console.log('数据已存在'); if (value.attach_type === 'img') { if (Number.isFinite(value.attach_thumb)) { if (value.attach_thumb === 0) { // console.log('没有触发缩图'); return `${value.attach_name}` } else if (value.attach_thumb === 1) { // console.log('触发缩图'); return `${value.attach_name}` }; }; // 正常情况是不会到这一步的,就不判断缩图状态了 return `${value.attach_name}` } else if (value.attach_type === 'other') { return '
' + `other  ` + `${value.attach_name}` + '  ' + `(${value.attach_size})` + '
' } else if (value.attach_type === 'invalid') { // 会不会发生碰撞呢 xd return `
附件 ${args[1]} 无效。
`; }; } else { // console.log('数据不存在'); return await new Promise((resolve, reject) => { jq.ajax({ type: 'post', url: 'https://u2.dmhy.org/preview.php', contentType: "application/x-www-form-urlencoded", data: ({ "body": `[attach]${hash}[/attach]` }), success: async function (d) { // console.log('成功'); let htmlobj = document.createElement('div'); htmlobj.innerHTML = d; let span = jq(htmlobj).find('span'); let attach_normal = jq(span).children('bdo').children('div.attach'); // 普通附件 let attach_image = jq(span).children('bdo').children('img'); // 图片附件 if (attach_normal.length !== 0 && attach_image.length === 0) { // console.log('普通附件'); let attach_info_obj = /(?