From 9c6aec78e0d01804873b1f53586b2ac14b94aa28 Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Thu, 30 Jun 2022 20:38:30 +0800 Subject: [PATCH 001/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E8=87=AA=E5=B7=B1?= =?UTF-8?q?=E7=9A=84bug:=E8=A7=92=E5=BA=A6=E5=BA=94=E8=AF=A5=E6=98=AF180?= =?UTF-8?q?=E8=80=8C=E4=B8=8D=E6=98=AF-180?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index 36b6891e..dadec488 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -238,7 +238,9 @@ class ArkosExtensions { const a = Cast.toNumber(args.a) const b = Cast.toNumber(args.b) let dif = b - a - return dif - Math.round(dif / 360) * 360 + dif -= Math.round(dif / 360) * 360 + if (dif === -180) dif = 180 + return dif } disFromAToB(args) { From 9ed430960c3572337396209485b154b1d3fc2198 Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Thu, 30 Jun 2022 19:34:09 +0800 Subject: [PATCH 002/357] =?UTF-8?q?LZString=E5=8E=8B=E7=BC=A9=E7=AE=97?= =?UTF-8?q?=E6=B3=95(=E4=BF=AE=E6=94=B9=E8=BF=87=EF=BC=8C=E6=94=AF?= =?UTF-8?q?=E6=8C=81=E5=8A=A0=E5=AF=86)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/Archivecode.js | 439 ++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 437 insertions(+), 2 deletions(-) diff --git a/Arkos/Archivecode.js b/Arkos/Archivecode.js index adf74a86..4fc8bb5d 100644 --- a/Arkos/Archivecode.js +++ b/Arkos/Archivecode.js @@ -6,6 +6,8 @@ import Cast from '../utils/cast.js' // console.log(Cast.toNumber('123')) //console.log(Cast.toNumber('aab')) +let LZString = _LZString(); + class Archive_code { constructor(runtime) { this.runtime = runtime @@ -49,6 +51,7 @@ class Archive_code { 'ArchiveCodeExt.getCharByUnicode': 'Unicode[code]对应字符', 'ArchiveCodeExt.methodMenu.1': 'Arkos加密法', 'ArchiveCodeExt.methodMenu.2': '分裂加密法', + 'ArchiveCodeExt.methodMenu.3': 'LZString压缩', 'ArchiveCodeExt.infoMenu.1': '名称', 'ArchiveCodeExt.infoMenu.2': '内容', 'ArchiveCodeExt.infoMenu.3': '类型', @@ -83,6 +86,7 @@ class Archive_code { 'ArchiveCodeExt.getCharByUnicode': ' character of Unicode[code]', 'ArchiveCodeExt.methodMenu.1': 'Arkos cipher', 'ArchiveCodeExt.methodMenu.2': 'mitotic encryption', + 'ArchiveCodeExt.methodMenu.3': 'LZString', 'ArchiveCodeExt.infoMenu.1': 'name', 'ArchiveCodeExt.infoMenu.2': 'value', 'ArchiveCodeExt.infoMenu.3': 'type', @@ -442,9 +446,12 @@ class Archive_code { // text: this.formatMessage('ArchiveCodeExt.methodMenu.2'), // value: '2' // }, - ], - infoMenu: [ { + text: this.formatMessage('ArchiveCodeExt.methodMenu.3'), + value: '3' + }, + ], + infoMenu: [{ text: this.formatMessage('ArchiveCodeExt.infoMenu.1'), value: '1' }, @@ -697,6 +704,8 @@ class Archive_code { return this.ArkosEncrypt(args); case '2': return this.encrypt2(args); + case '3': + return LZString.scompress(args.str, args.key); default: return ''; } @@ -709,6 +718,8 @@ class Archive_code { return this.ArkosDecrypt(args); case '2': return this.decrypt2(args); + case '3': + return LZString.sdecompress(args.str, args.key); default: return ''; } @@ -943,6 +954,430 @@ class Archive_code { } +// - lz-strings.js (modified) - + +// Copyright (c) 2013 Pieroxy +// This work is free. You can redistribute it and/or modify it +// under the terms of the WTFPL, Version 2 +// For more information see LICENSE.txt or http://www.wtfpl.net/ +// +// For more information, the home page: +// http://pieroxy.net/blog/pages/lz-string/testing.html +// +// LZ-based compression algorithm, version 1.4.4 +function _LZString() { + + // private property + var f = String.fromCharCode; + + var LZString = { + scompress: function(uncompressed, key) { + if (key.length !== 0) + key = this.scompress(key, ""); + return this._compress(uncompressed, 14, function(x, i) { + if (key.length !== 0) + x ^= (key.charCodeAt(i % key.length) * (i + 1)) & ((1 << 14) - 1); + return f(x + 0x4E00); + }); + }, + + _compress: function(uncompressed, bitsPerChar, getCharFromInt) { + if (uncompressed == null) return ""; + var i, value, + context_dictionary = {}, + context_dictionaryToCreate = {}, + context_c = "", + context_wc = "", + context_w = "", + context_enlargeIn = 2, // Compensate for the first entry which should not count + context_dictSize = 3, + context_numBits = 2, + context_data = [], + context_data_val = 0, + context_data_position = 0, + ii; + + for (ii = 0; ii < uncompressed.length; ii += 1) { + context_c = uncompressed.charAt(ii); + if (!Object.prototype.hasOwnProperty.call(context_dictionary, context_c)) { + context_dictionary[context_c] = context_dictSize++; + context_dictionaryToCreate[context_c] = true; + } + + context_wc = context_w + context_c; + if (Object.prototype.hasOwnProperty.call(context_dictionary, context_wc)) { + context_w = context_wc; + } else { + if (Object.prototype.hasOwnProperty.call(context_dictionaryToCreate, context_w)) { + if (context_w.charCodeAt(0) < 256) { + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + } + value = context_w.charCodeAt(0); + for (i = 0; i < 8; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } else { + value = 1; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | value; + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = 0; + } + value = context_w.charCodeAt(0); + for (i = 0; i < 16; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + delete context_dictionaryToCreate[context_w]; + } else { + value = context_dictionary[context_w]; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + // Add wc to the dictionary. + context_dictionary[context_wc] = context_dictSize++; + context_w = String(context_c); + } + } + + // Output the code for w. + if (context_w !== "") { + if (Object.prototype.hasOwnProperty.call(context_dictionaryToCreate, context_w)) { + if (context_w.charCodeAt(0) < 256) { + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + } + value = context_w.charCodeAt(0); + for (i = 0; i < 8; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } else { + value = 1; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | value; + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = 0; + } + value = context_w.charCodeAt(0); + for (i = 0; i < 16; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + delete context_dictionaryToCreate[context_w]; + } else { + value = context_dictionary[context_w]; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + } + + // Mark the end of the stream + value = 2; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + // Flush the last char + while (true) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data.push(getCharFromInt(context_data_val, context_data.length)); + break; + } else context_data_position++; + } + return context_data.join(''); + }, + + sdecompress: function(compressed, key) { + if (key.length !== 0) + key = this.scompress(key, ""); + return this._decompress(compressed.length, 1 << (14 - 1), function(i) { + let x = compressed.charCodeAt(i) - 0x4E00; + if (key.length !== 0) + x ^= (key.charCodeAt(i % key.length) * (i + 1)) & ((1 << 14) - 1); + return x; + }); + }, + + _decompress: function(length, resetValue, getNextValue) { + var dictionary = [], + next, + enlargeIn = 4, + dictSize = 4, + numBits = 3, + entry = "", + result = [], + i, + w, + bits, resb, maxpower, power, + c, + data = { + val: getNextValue(0), + position: resetValue, + index: 1 + }; + + for (i = 0; i < 3; i += 1) { + dictionary[i] = i; + } + + bits = 0; + maxpower = Math.pow(2, 2); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + switch (next = bits) { + case 0: + bits = 0; + maxpower = Math.pow(2, 8); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + c = f(bits); + break; + case 1: + bits = 0; + maxpower = Math.pow(2, 16); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + c = f(bits); + break; + case 2: + return ""; + } + dictionary[3] = c; + w = c; + result.push(c); + while (true) { + if (data.index > length) { + return ""; + } + + bits = 0; + maxpower = Math.pow(2, numBits); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + switch (c = bits) { + case 0: + bits = 0; + maxpower = Math.pow(2, 8); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + dictionary[dictSize++] = f(bits); + c = dictSize - 1; + enlargeIn--; + break; + case 1: + bits = 0; + maxpower = Math.pow(2, 16); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + dictionary[dictSize++] = f(bits); + c = dictSize - 1; + enlargeIn--; + break; + case 2: + return result.join(''); + } + + if (enlargeIn == 0) { + enlargeIn = Math.pow(2, numBits); + numBits++; + } + + if (dictionary[c]) { + entry = dictionary[c]; + } else { + if (c === dictSize) { + entry = w + w.charAt(0); + } else { + // throw new Error("LZString: decompress error."); + // return null; + return ""; + } + } + result.push(entry); + + // Add w+entry[0] to the dictionary. + dictionary[dictSize++] = w + entry.charAt(0); + enlargeIn--; + + w = entry; + + if (enlargeIn == 0) { + enlargeIn = Math.pow(2, numBits); + numBits++; + } + + } + } + }; + return LZString; +} window.tempExt = { From b3bd325ca750cc6e3a9f35a3d328182f087e3ab5 Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Fri, 24 Jun 2022 00:11:07 +0800 Subject: [PATCH 003/357] =?UTF-8?q?=E6=AD=A3=E5=88=99=E8=A1=A8=E8=BE=BE?= =?UTF-8?q?=E5=BC=8F=E6=8B=93=E5=B1=95=E5=88=9D=E7=A8=BF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- six-6/regexp.js | 577 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 577 insertions(+) create mode 100644 six-6/regexp.js diff --git a/six-6/regexp.js b/six-6/regexp.js new file mode 100644 index 00000000..b1c8bc63 --- /dev/null +++ b/six-6/regexp.js @@ -0,0 +1,577 @@ +// import Cast from "cast.js"; + +class RegExpVI { + constructor(runtime) { + this.regexp = new RegExp("", "g"); + this.text = ""; + this.result = null; + this.runtime = runtime; + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "RegExpVI.name": "正则表达式vi", + "RegExpVI.info.1": "创建匹配模式", + "RegExpVI.info.2": "查找文本", + "RegExpVI.regexp.dight": "数字", + "RegExpVI.regexp.chars": "数字字母下划线", + "RegExpVI.regexp.white": "空白字符", + "RegExpVI.regexp.notdight": "非数字", + "RegExpVI.regexp.notchars": "非数字字母下划线", + "RegExpVI.regexp.notwhite": "非空白字符", + "RegExpVI.regexp.number": "数", + "RegExpVI.regexp.integer": "整数", + "RegExpVI.regexp.positivenumber": "非负数", + "RegExpVI.regexp.positiveinteger": "非负整数", + "RegExpVI.regexp.scientific": "科学记数法", + "RegExpVI.regexp.any": "任意字符", + "RegExpVI.regexp.word": "英文单词", + "RegExpVI.regexp.upper": "大写字母", + "RegExpVI.regexp.lower": "小写字母", + "RegExpVI.regexp.chinese": "汉字", + "RegExpVI.range": "[INCLUDE] [TEXT] 中的字符", + "RegExpVI.range.include": "在", + "RegExpVI.range.exclude": "不在", + "RegExpVI.text": "文字 [TEXT]", + "RegExpVI.quantifier": "[COUNT] [REGEXP]", + "RegExpVI.quantifier.anyornone": "可选的任意个", + "RegExpVI.quantifier.anyornonelazy": "可选的任意个 (尽量少)", + "RegExpVI.quantifier.optional": "可选的", + "RegExpVI.quantifier.optionallazy": "可选的 (尽量无)", + "RegExpVI.quantifier.any": "任意个", + "RegExpVI.quantifier.anylazy": "任意个 (尽量少)", + "RegExpVI.countn": "[MODE] [N] 个 [REGEXP]", + "RegExpVI.countn.equal": "恰好", + "RegExpVI.countn.greater": "至少", + "RegExpVI.countn.greaterlazy": "至少 (尽量少)", + "RegExpVI.countrange": "[MODE] [X] 到 [Y] 个 [REGEXP]", + "RegExpVI.countrange.greedy": "(尽量多)", + "RegExpVI.countrange.lazy": "(尽量少)", + "RegExpVI.group": "捕获组: ([REGEXP])", + "RegExpVI.groupreference": "第 [N] 组的内容", + "RegExpVI.concat": "[REGEXP1] 和 [REGEXP2]", + "RegExpVI.concatwithseparator": "[REGEXP1] 和 [TEXT] 和 [REGEXP2]", + "RegExpVI.or": "[REGEXP1] 或者 [REGEXP2]", + "RegExpVI.findtext": "在 [TEXT] 里查找 [REGEXP]", + "RegExpVI.findnext": "继续查找", + "RegExpVI.isfound": "找到了?", + "RegExpVI.foundtext": "找到的文本", + "RegExpVI.foundtextbegin": "找到的文本开始位置", + "RegExpVI.foundtextend": "找到的文本结束位置", + "RegExpVI.foundtextgroups": "找到的第 [N] 个捕获组的内容" + }, + en: { + "RegExpVI.name": "Regular expression vi", + "RegExpVI.info.1": "Creating pattern", + "RegExpVI.info.2": "Finding texts", + "RegExpVI.regexp.dight": "dight", + "RegExpVI.regexp.chars": "[a-zA-Z_]", + "RegExpVI.regexp.white": "white character", + "RegExpVI.regexp.notdight": "not dight", + "RegExpVI.regexp.notchars": "not [a-zA-Z_]", + "RegExpVI.regexp.notwhite": "not white character", + "RegExpVI.regexp.number": "number", + "RegExpVI.regexp.integer": "integer", + "RegExpVI.regexp.positivenumber": "not-negative number", + "RegExpVI.regexp.positiveinteger": "not-negative integer", + "RegExpVI.regexp.scientific": "scientific notation", + "RegExpVI.regexp.any": "any character", + "RegExpVI.regexp.word": "word", + "RegExpVI.regexp.upper": "captical letter", + "RegExpVI.regexp.lower": "non-capital letter", + "RegExpVI.regexp.chinese": "chinese character", + "RegExpVI.range": "chars [INCLUDE] [TEXT]", + "RegExpVI.range.include": "in", + "RegExpVI.range.exclude": "not in", + "RegExpVI.text": "text [TEXT]", + "RegExpVI.quantifier": "[COUNT] [REGEXP]", + "RegExpVI.quantifier.anyornone": "any numbers or none of (greedy)", + "RegExpVI.quantifier.anyornonelazy": "any numbers or none of (lazy)", + "RegExpVI.quantifier.optional": "optional (greedy)", + "RegExpVI.quantifier.optionallazy": "optional (lazy)", + "RegExpVI.quantifier.any": "any numbers of (greedy)", + "RegExpVI.quantifier.anylazy": "any numbers of (lazy)", + "RegExpVI.countn": "[MODE] [N] [REGEXP]", + "RegExpVI.countn.equal": "exactly", + "RegExpVI.countn.greater": "at least (greedy)", + "RegExpVI.countn.greaterlazy": "at least (lazy)", + "RegExpVI.countrange": "[MODE] [X] to [Y] [REGEXP]", + "RegExpVI.countrange.greedy": "(greedy)", + "RegExpVI.countrange.lazy": "(lazy)", + "RegExpVI.group": "group [REGEXP]", + "RegExpVI.groupreference": "group [N]", + "RegExpVI.concat": "[REGEXP1] and [REGEXP2]", + "RegExpVI.concatwithseparator": "[REGEXP1] and [TEXT] and [REGEXP2]", + "RegExpVI.or": "[REGEXP1] or [REGEXP2]", + "RegExpVI.findtext": "find [REGEXP] in [TEXT]", + "RegExpVI.findnext": "find next", + "RegExpVI.isfound": "found?", + "RegExpVI.foundtext": "found text", + "RegExpVI.foundtextbegin": "begin index", + "RegExpVI.foundtextend": "end index", + "RegExpVI.foundtextgroups": "founded content of group [N]" + } + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: "RegExpVI", // 拓展id + name: this.formatMessage("RegExpVI.name"), // 拓展名 + // docsURI: "https://www.baidu.com", + color1: "#3333FF", + color2: "#0000CC", + blocks: [{ + opcode: "info1", + blockType: "button", + text: this.formatMessage("RegExpVI.info.1"), + onClick: function(e) {} + }, { + opcode: "str", + blockType: "reporter", + text: this.formatMessage("RegExpVI.text"), + arguments: { + TEXT: { + type: "string", + defaultValue: "," + } + } + }, { + opcode: "range", + blockType: "reporter", + text: this.formatMessage("RegExpVI.range"), + arguments: { + INCLUDE: { + type: "string", + menu: "CharsetInclude", + defaultValue: "-" + }, + TEXT: { + type: "string", + defaultValue: "a-zA-Z" + } + } + }, { + opcode: "quantifier", + blockType: "reporter", + text: this.formatMessage("RegExpVI.quantifier"), + arguments: { + COUNT: { + type: "string", + menu: "CountPicker", + defaultValue: "+" + }, + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + } + } + }, { + opcode: "countn", + blockType: "reporter", + text: this.formatMessage("RegExpVI.countn"), + arguments: { + N: { + type: "number", + defaultValue: 2 + }, + MODE: { + type: "string", + menu: "CountnumPicker", + defaultValue: "}" + }, + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + } + } + }, { + opcode: "countrange", + blockType: "reporter", + text: this.formatMessage("RegExpVI.countrange"), + arguments: { + X: { + type: "number", + defaultValue: 2 + }, + Y: { + type: "number", + defaultValue: 3 + }, + MODE: { + type: "string", + menu: "CountrangePicker", + defaultValue: "}" + }, + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + } + } + }, { + opcode: "or", + blockType: "reporter", + text: this.formatMessage("RegExpVI.or"), + arguments: { + REGEXP1: { + type: "string", + menu: "RegExpPicker", + defaultValue: "-?[0-9]+" + }, + REGEXP2: { + type: "string", + menu: "RegExpPicker", + defaultValue: "[a-zA-Z']+" + } + } + }, { + opcode: "concat", + blockType: "reporter", + text: this.formatMessage("RegExpVI.concat"), + arguments: { + REGEXP1: { + type: "string", + menu: "RegExpPicker", + defaultValue: "[A-Z]" + }, + REGEXP2: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + } + } + }, { + opcode: "concatwithseparator", + blockType: "reporter", + text: this.formatMessage("RegExpVI.concatwithseparator"), + arguments: { + REGEXP1: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + }, + TEXT: { + type: "string", + defaultValue: "," + }, + REGEXP2: { + type: "string", + menu: "RegExpPicker", + defaultValue: "[A-Z]" + } + } + }, { + opcode: "group", + blockType: "reporter", + text: this.formatMessage("RegExpVI.group"), + arguments: { + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "[a-zA-Z']+" + } + } + }, { + opcode: "groupreference", + blockType: "reporter", + text: this.formatMessage("RegExpVI.groupreference"), + arguments: { + N: { + type: "number", + defaultValue: 1 + } + } + }, { + opcode: "info2", + blockType: "button", + text: this.formatMessage("RegExpVI.info.2"), + onClick: function(e) {} + }, { + opcode: "findtext", + blockType: "command", + text: this.formatMessage("RegExpVI.findtext"), + arguments: { + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "-?[0-9]+(?:\\.[0-9]*)?" + }, + TEXT: { + type: "string", + defaultValue: "I have 23 apples." + } + } + }, { + opcode: "findnext", + blockType: "command", + text: this.formatMessage("RegExpVI.findnext"), + arguments: {} + }, { + opcode: "isfound", + blockType: "Boolean", + text: this.formatMessage("RegExpVI.isfound"), + arguments: {} + }, { + opcode: "foundtext", + blockType: "reporter", + text: this.formatMessage("RegExpVI.foundtext"), + arguments: {} + }, { + opcode: "foundtextbegin", + blockType: "reporter", + text: this.formatMessage("RegExpVI.foundtextbegin"), + arguments: {} + }, { + opcode: "foundtextend", + blockType: "reporter", + text: this.formatMessage("RegExpVI.foundtextend"), + arguments: {} + }, { + opcode: "foundtextgroups", + blockType: "reporter", + text: this.formatMessage("RegExpVI.foundtextgroups"), + arguments: { + N: { + type: "number", + defaultValue: 1 + } + } + }], + menus: { + RegExpPicker: { + items: [{ + text: this.formatMessage("RegExpVI.regexp.dight"), + value: "\\d" + }, { + text: this.formatMessage("RegExpVI.regexp.chars"), + value: "\\w" + }, { + text: this.formatMessage("RegExpVI.regexp.white"), + value: "\\s" + }, { + text: this.formatMessage("RegExpVI.regexp.notdight"), + value: "\\D" + }, { + text: this.formatMessage("RegExpVI.regexp.notchars"), + value: "\\W" + }, { + text: this.formatMessage("RegExpVI.regexp.notwhite"), + value: "\\S" + }, { + text: this.formatMessage("RegExpVI.regexp.number"), + value: "-?[0-9]+(?:\\.[0-9]*)?" + }, { + text: this.formatMessage("RegExpVI.regexp.integer"), + value: "-?[0-9]+" + }, { + text: this.formatMessage("RegExpVI.regexp.positivenumber"), + value: "[0-9]+(?:\\.[0-9]*)?" + }, { + text: this.formatMessage("RegExpVI.regexp.positiveinteger"), + value: "[0-9]+" + }, { + text: this.formatMessage("RegExpVI.regexp.scientific"), + value: "-?[0-9]+(?:\\.[0-9]*)?(?:[eE]-?[0-9]+)" + }, { + text: this.formatMessage("RegExpVI.regexp.any"), + value: "." + }, { + text: this.formatMessage("RegExpVI.regexp.word"), + value: "[a-zA-Z']+" + }, { + text: this.formatMessage("RegExpVI.regexp.upper"), + value: "[A-Z]" + }, { + text: this.formatMessage("RegExpVI.regexp.lower"), + value: "[a-z]" + }, { + text: this.formatMessage("RegExpVI.regexp.chinese"), + value: "[\u4e00-\u9fa5]" + }], + acceptReporters: true + }, + CountPicker: { + items: [{ + text: this.formatMessage("RegExpVI.quantifier.anyornone"), + value: "*" + }, { + text: this.formatMessage("RegExpVI.quantifier.any"), + value: "+" + }, { + text: this.formatMessage("RegExpVI.quantifier.optional"), + value: "?" + }, { + text: this.formatMessage("RegExpVI.quantifier.anyornonelazy"), + value: "*?" + }, { + text: this.formatMessage("RegExpVI.quantifier.anylazy"), + value: "+?" + }, { + text: this.formatMessage("RegExpVI.quantifier.optionallazy"), + value: "??" + }] + }, + CountnumPicker: { + items: [{ + text: this.formatMessage("RegExpVI.countn.equal"), + value: "}" + }, { + text: this.formatMessage("RegExpVI.countn.greater"), + value: ",}" + }, { + text: this.formatMessage("RegExpVI.countn.greaterlazy"), + value: ",}?" + }] + }, + CountrangePicker: { + items: [{ + text: this.formatMessage("RegExpVI.countrange.lazy"), + value: "}?" + }, { + text: this.formatMessage("RegExpVI.countrange.greedy"), + value: "}" + }] + }, + CharsetInclude: { + items: [{ + text: this.formatMessage("RegExpVI.range.include"), + value: "-" + }, { + text: this.formatMessage("RegExpVI.range.exclude"), + value: "-^" + }] + } + } + }; + } + + range(args) { + const re = /[\\\]\^]/g; + return "[" + String(args.INCLUDE).slice(1) + String(args.TEXT).replace(re, "\\$&") + "]"; + } + + str(args) { + const re = /[\\\[\].*?+(){}^$|]/g; + return String(args.TEXT).replace(re, "\\$&"); + } + + concatwithseparator(args) { + return "(?:" + String(args.REGEXP1) + ")(?:" + this.str(args) + ")(?:" + String(args.REGEXP2) + ")"; + } + + concat(args) { + return "(?:" + String(args.REGEXP1) + ")(?:" + String(args.REGEXP2) + ")"; + } + + or(args) { + return "(?:" + String(args.REGEXP1) + ")|(?:" + String(args.REGEXP2) + ")"; + } + + quantifier(args) { + return "(?:" + String(args.REGEXP) + ")" + String(args.COUNT); + } + + countn(args) { + let n = Math.floor(Number(args.N)); + if (n < 0) n = 0; + return "(?:" + String(args.REGEXP) + "){" + n + String(args.MODE); + } + + countrange(args) { + let x = Math.floor(Number(args.X)); + let y = Math.floor(Number(args.Y)); + if (x < 0) x = 0; + if (y < 0) y = 0; + if (x > y) { + let t = x; + x = y; + y = t + }; + return "(?:" + String(args.REGEXP) + "){" + + (x === y ? x : x + "," + y) + + String(args.MODE); + } + + group(args) { + return "(" + String(args.REGEXP) + ")"; + } + + groupreference(args) { + let N = Math.floor(Number(args.N)); + return N > 0 ? "\\" + N : ""; + } + + findtext(args) { + this.regexp = new RegExp(String(args.REGEXP), "g"); + this.text = args.TEXT; + this.result = this.regexp.exec(this.text); + } + + findnext(args) { + if (this.result !== null) { + if (this.result[0].length === 0) + this.regexp.lastIndex++; + this.result = this.regexp.exec(this.text); + } + } + + isfound(args) { + return this.result !== null; + } + + foundtext(args) { + return this.result === null ? "" : this.result[0]; + } + + foundtextbegin(args) { + return this.regexp.lastIndex - this.foundtext().length + 1; + } + + foundtextend(args) { + return this.regexp.lastIndex; + } + + foundtextgroups(args) { + if (this.result === null) return ""; + let N = Math.floor(Number(args.N)); + if (N >= this.result.length) { + return ""; + } else { + return this.result[N] === null ? "" : this.result[N]; + }; + } + +} + +window.tempExt = { + Extension: RegExpVI, + info: { + name: "RegExpVI.name", + description: "RegExpVI.descp", + extensionId: "RegExpVI", + featured: true, + disabled: false, + collaborator: "vi" + }, + l10n: { + "zh-cn": { + "RegExpVI.name": "正则表达式vi", + "RegExpVI.descp": "用积木创建正则表达式并匹配文本。" + }, + en: { + "RegExpVI.name": "Regular expression vi", + "RegExpVI.descp": "Create RegExp and find texts." + } + } +}; + +/* vim: set expandtab tabstop=2 shiftwidth=2: */ From 54d02a9b826eaa6734e8d91b97bbc77c796e188f Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Sun, 3 Jul 2022 10:22:13 +0800 Subject: [PATCH 004/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E8=8C=83=E5=9B=B4?= =?UTF-8?q?=E9=97=AE=E9=A2=98=EF=BC=9B=E5=90=88=E9=80=82=E7=9A=84=E5=88=86?= =?UTF-8?q?=E9=9A=94=E7=AC=A6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- six-6/regexp.js | 445 +++++++++++++++++++++++++----------------------- 1 file changed, 236 insertions(+), 209 deletions(-) diff --git a/six-6/regexp.js b/six-6/regexp.js index b1c8bc63..909bcc15 100644 --- a/six-6/regexp.js +++ b/six-6/regexp.js @@ -27,9 +27,11 @@ class RegExpVI { "RegExpVI.regexp.upper": "大写字母", "RegExpVI.regexp.lower": "小写字母", "RegExpVI.regexp.chinese": "汉字", - "RegExpVI.range": "[INCLUDE] [TEXT] 中的字符", + "RegExpVI.range": "[INCLUDE] [TEXT] 中的字符 [RANGE]", "RegExpVI.range.include": "在", "RegExpVI.range.exclude": "不在", + "RegExpVI.range.range": "(启用范围)", + "RegExpVI.range.norange": "(禁用范围)", "RegExpVI.text": "文字 [TEXT]", "RegExpVI.quantifier": "[COUNT] [REGEXP]", "RegExpVI.quantifier.anyornone": "可选的任意个", @@ -78,9 +80,11 @@ class RegExpVI { "RegExpVI.regexp.upper": "captical letter", "RegExpVI.regexp.lower": "non-capital letter", "RegExpVI.regexp.chinese": "chinese character", - "RegExpVI.range": "chars [INCLUDE] [TEXT]", + "RegExpVI.range": "chars [INCLUDE] [RANGE] [TEXT]", "RegExpVI.range.include": "in", "RegExpVI.range.exclude": "not in", + "RegExpVI.range.range": "(range enabled)", + "RegExpVI.range.norange": "(range disabled)", "RegExpVI.text": "text [TEXT]", "RegExpVI.quantifier": "[COUNT] [REGEXP]", "RegExpVI.quantifier.anyornone": "any numbers or none of (greedy)", @@ -127,225 +131,225 @@ class RegExpVI { // docsURI: "https://www.baidu.com", color1: "#3333FF", color2: "#0000CC", - blocks: [{ - opcode: "info1", - blockType: "button", - text: this.formatMessage("RegExpVI.info.1"), - onClick: function(e) {} - }, { - opcode: "str", - blockType: "reporter", - text: this.formatMessage("RegExpVI.text"), - arguments: { - TEXT: { - type: "string", - defaultValue: "," + blocks: [ + "---" + this.formatMessage("RegExpVI.info.1"), + { + opcode: "str", + blockType: "reporter", + text: this.formatMessage("RegExpVI.text"), + arguments: { + TEXT: { + type: "string", + defaultValue: "," + } } - } - }, { - opcode: "range", - blockType: "reporter", - text: this.formatMessage("RegExpVI.range"), - arguments: { - INCLUDE: { - type: "string", - menu: "CharsetInclude", - defaultValue: "-" - }, - TEXT: { - type: "string", - defaultValue: "a-zA-Z" + }, { + opcode: "range", + blockType: "reporter", + text: this.formatMessage("RegExpVI.range"), + arguments: { + INCLUDE: { + type: "string", + menu: "CharsetInclude", + defaultValue: "-" + }, + RANGE: { + type: "string", + menu: "CharsetRange", + defaultValue: "true" + }, + TEXT: { + type: "string", + defaultValue: "a-zA-Z" + } } - } - }, { - opcode: "quantifier", - blockType: "reporter", - text: this.formatMessage("RegExpVI.quantifier"), - arguments: { - COUNT: { - type: "string", - menu: "CountPicker", - defaultValue: "+" - }, - REGEXP: { - type: "string", - menu: "RegExpPicker", - defaultValue: "\\d" + }, { + opcode: "quantifier", + blockType: "reporter", + text: this.formatMessage("RegExpVI.quantifier"), + arguments: { + COUNT: { + type: "string", + menu: "CountPicker", + defaultValue: "+" + }, + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + } } - } - }, { - opcode: "countn", - blockType: "reporter", - text: this.formatMessage("RegExpVI.countn"), - arguments: { - N: { - type: "number", - defaultValue: 2 - }, - MODE: { - type: "string", - menu: "CountnumPicker", - defaultValue: "}" - }, - REGEXP: { - type: "string", - menu: "RegExpPicker", - defaultValue: "\\d" + }, { + opcode: "countn", + blockType: "reporter", + text: this.formatMessage("RegExpVI.countn"), + arguments: { + N: { + type: "number", + defaultValue: 2 + }, + MODE: { + type: "string", + menu: "CountnumPicker", + defaultValue: "}" + }, + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + } } - } - }, { - opcode: "countrange", - blockType: "reporter", - text: this.formatMessage("RegExpVI.countrange"), - arguments: { - X: { - type: "number", - defaultValue: 2 - }, - Y: { - type: "number", - defaultValue: 3 - }, - MODE: { - type: "string", - menu: "CountrangePicker", - defaultValue: "}" - }, - REGEXP: { - type: "string", - menu: "RegExpPicker", - defaultValue: "\\d" + }, { + opcode: "countrange", + blockType: "reporter", + text: this.formatMessage("RegExpVI.countrange"), + arguments: { + X: { + type: "number", + defaultValue: 2 + }, + Y: { + type: "number", + defaultValue: 3 + }, + MODE: { + type: "string", + menu: "CountrangePicker", + defaultValue: "}" + }, + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + } } - } - }, { - opcode: "or", - blockType: "reporter", - text: this.formatMessage("RegExpVI.or"), - arguments: { - REGEXP1: { - type: "string", - menu: "RegExpPicker", - defaultValue: "-?[0-9]+" - }, - REGEXP2: { - type: "string", - menu: "RegExpPicker", - defaultValue: "[a-zA-Z']+" + }, { + opcode: "or", + blockType: "reporter", + text: this.formatMessage("RegExpVI.or"), + arguments: { + REGEXP1: { + type: "string", + menu: "RegExpPicker", + defaultValue: "-?[0-9]+" + }, + REGEXP2: { + type: "string", + menu: "RegExpPicker", + defaultValue: "[a-zA-Z']+" + } } - } - }, { - opcode: "concat", - blockType: "reporter", - text: this.formatMessage("RegExpVI.concat"), - arguments: { - REGEXP1: { - type: "string", - menu: "RegExpPicker", - defaultValue: "[A-Z]" - }, - REGEXP2: { - type: "string", - menu: "RegExpPicker", - defaultValue: "\\d" + }, { + opcode: "concat", + blockType: "reporter", + text: this.formatMessage("RegExpVI.concat"), + arguments: { + REGEXP1: { + type: "string", + menu: "RegExpPicker", + defaultValue: "[A-Z]" + }, + REGEXP2: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + } } - } - }, { - opcode: "concatwithseparator", - blockType: "reporter", - text: this.formatMessage("RegExpVI.concatwithseparator"), - arguments: { - REGEXP1: { - type: "string", - menu: "RegExpPicker", - defaultValue: "\\d" - }, - TEXT: { - type: "string", - defaultValue: "," - }, - REGEXP2: { - type: "string", - menu: "RegExpPicker", - defaultValue: "[A-Z]" + }, { + opcode: "concatwithseparator", + blockType: "reporter", + text: this.formatMessage("RegExpVI.concatwithseparator"), + arguments: { + REGEXP1: { + type: "string", + menu: "RegExpPicker", + defaultValue: "\\d" + }, + TEXT: { + type: "string", + defaultValue: "," + }, + REGEXP2: { + type: "string", + menu: "RegExpPicker", + defaultValue: "[A-Z]" + } } - } - }, { - opcode: "group", - blockType: "reporter", - text: this.formatMessage("RegExpVI.group"), - arguments: { - REGEXP: { - type: "string", - menu: "RegExpPicker", - defaultValue: "[a-zA-Z']+" + }, { + opcode: "group", + blockType: "reporter", + text: this.formatMessage("RegExpVI.group"), + arguments: { + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "[a-zA-Z']+" + } } - } - }, { - opcode: "groupreference", - blockType: "reporter", - text: this.formatMessage("RegExpVI.groupreference"), - arguments: { - N: { - type: "number", - defaultValue: 1 + }, { + opcode: "groupreference", + blockType: "reporter", + text: this.formatMessage("RegExpVI.groupreference"), + arguments: { + N: { + type: "number", + defaultValue: 1 + } } - } - }, { - opcode: "info2", - blockType: "button", - text: this.formatMessage("RegExpVI.info.2"), - onClick: function(e) {} - }, { - opcode: "findtext", - blockType: "command", - text: this.formatMessage("RegExpVI.findtext"), - arguments: { - REGEXP: { - type: "string", - menu: "RegExpPicker", - defaultValue: "-?[0-9]+(?:\\.[0-9]*)?" - }, - TEXT: { - type: "string", - defaultValue: "I have 23 apples." + }, + "---" + this.formatMessage("RegExpVI.info.2"), + { + opcode: "findtext", + blockType: "command", + text: this.formatMessage("RegExpVI.findtext"), + arguments: { + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "-?[0-9]+(?:\\.[0-9]*)?" + }, + TEXT: { + type: "string", + defaultValue: "I have 23 apples." + } } - } - }, { - opcode: "findnext", - blockType: "command", - text: this.formatMessage("RegExpVI.findnext"), - arguments: {} - }, { - opcode: "isfound", - blockType: "Boolean", - text: this.formatMessage("RegExpVI.isfound"), - arguments: {} - }, { - opcode: "foundtext", - blockType: "reporter", - text: this.formatMessage("RegExpVI.foundtext"), - arguments: {} - }, { - opcode: "foundtextbegin", - blockType: "reporter", - text: this.formatMessage("RegExpVI.foundtextbegin"), - arguments: {} - }, { - opcode: "foundtextend", - blockType: "reporter", - text: this.formatMessage("RegExpVI.foundtextend"), - arguments: {} - }, { - opcode: "foundtextgroups", - blockType: "reporter", - text: this.formatMessage("RegExpVI.foundtextgroups"), - arguments: { - N: { - type: "number", - defaultValue: 1 + }, { + opcode: "findnext", + blockType: "command", + text: this.formatMessage("RegExpVI.findnext"), + arguments: {} + }, { + opcode: "isfound", + blockType: "Boolean", + text: this.formatMessage("RegExpVI.isfound"), + arguments: {} + }, { + opcode: "foundtext", + blockType: "reporter", + text: this.formatMessage("RegExpVI.foundtext"), + arguments: {} + }, { + opcode: "foundtextbegin", + blockType: "reporter", + text: this.formatMessage("RegExpVI.foundtextbegin"), + arguments: {} + }, { + opcode: "foundtextend", + blockType: "reporter", + text: this.formatMessage("RegExpVI.foundtextend"), + arguments: {} + }, { + opcode: "foundtextgroups", + blockType: "reporter", + text: this.formatMessage("RegExpVI.foundtextgroups"), + arguments: { + N: { + type: "number", + defaultValue: 1 + } } } - }], + ], menus: { RegExpPicker: { items: [{ @@ -449,6 +453,15 @@ class RegExpVI { text: this.formatMessage("RegExpVI.range.exclude"), value: "-^" }] + }, + CharsetRange: { + items: [{ + text: this.formatMessage("RegExpVI.range.range"), + value: "true" + }, { + text: this.formatMessage("RegExpVI.range.norange"), + value: "false" + }] } } }; @@ -456,7 +469,21 @@ class RegExpVI { range(args) { const re = /[\\\]\^]/g; - return "[" + String(args.INCLUDE).slice(1) + String(args.TEXT).replace(re, "\\$&") + "]"; + let text = args.TEXT; + if (args.RANGE === "true") { + text = text.replace(/.-./g, function(x) { + let a = x.codePointAt(0); + let b = x.codePointAt(2); + if (a < b) return x; + else if (a === b) return x[0]; + else return x[2] + "-" + x[0]; + }); + } else { + if (text.includes("-")) { + text = "-" + text.replace(/-/g, ""); + } + } + return "[" + String(args.INCLUDE).slice(1) + String(text).replace(re, "\\$&") + "]"; } str(args) { From 22c28d990eb270aac19aad0207bb36bffc352810 Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Sun, 3 Jul 2022 10:26:23 +0800 Subject: [PATCH 005/357] =?UTF-8?q?=E8=A1=A8=E8=BE=BE=E5=BC=8F=E5=87=BA?= =?UTF-8?q?=E9=94=99=E5=A4=84=E7=90=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- six-6/regexp.js | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/six-6/regexp.js b/six-6/regexp.js index 909bcc15..ce4b1391 100644 --- a/six-6/regexp.js +++ b/six-6/regexp.js @@ -538,7 +538,14 @@ class RegExpVI { } findtext(args) { - this.regexp = new RegExp(String(args.REGEXP), "g"); + try { + this.regexp = new RegExp(String(args.REGEXP), "g"); + } catch (e) { + // 这里应该提示用户表达式出错了 + this.regexp = new RegExp("", "g"); + this.text = ""; + this.result = null; + } this.text = args.TEXT; this.result = this.regexp.exec(this.text); } From c41dda1ccd1cc1505a46f5dcbf9564f95a6ad4ba Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Sun, 3 Jul 2022 12:18:01 +0800 Subject: [PATCH 006/357] =?UTF-8?q?=E5=88=86=E5=89=B2=EF=BC=8C=E6=9B=BF?= =?UTF-8?q?=E6=8D=A2=EF=BC=8C=E5=85=A8=E9=83=A8=E6=9F=A5=E6=89=BE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- six-6/regexp.js | 184 +++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 181 insertions(+), 3 deletions(-) diff --git a/six-6/regexp.js b/six-6/regexp.js index ce4b1391..a28ca9f8 100644 --- a/six-6/regexp.js +++ b/six-6/regexp.js @@ -58,7 +58,12 @@ class RegExpVI { "RegExpVI.foundtext": "找到的文本", "RegExpVI.foundtextbegin": "找到的文本开始位置", "RegExpVI.foundtextend": "找到的文本结束位置", - "RegExpVI.foundtextgroups": "找到的第 [N] 个捕获组的内容" + "RegExpVI.foundtextgroups": "找到的第 [N] 个捕获组的内容", + "RegExpVI.findall": "在 [TEXT] 里查找所有 [REGEXP] 并保存到 [LIST]", + "RegExpVI.split": "在 [TEXT] 里以 [REGEXP] 分割并保存到 [LIST]", + "RegExpVI.replace": "将 [TEXT] 里的所有 [REGEXP] 替换为 [DEST] [GROUP]", + "RegExpVI.replace.group": "(使用引用记号)", + "RegExpVI.replace.nogroup": "(不使用引用记号)", }, en: { "RegExpVI.name": "Regular expression vi", @@ -111,7 +116,12 @@ class RegExpVI { "RegExpVI.foundtext": "found text", "RegExpVI.foundtextbegin": "begin index", "RegExpVI.foundtextend": "end index", - "RegExpVI.foundtextgroups": "founded content of group [N]" + "RegExpVI.foundtextgroups": "founded content of group [N]", + "RegExpVI.findall": "find all [REGEXP] in [TEXT] and save to [LIST]", + "RegExpVI.split": "split [TEXT] by [REGEXP] and save to [LIST]", + "RegExpVI.replace": "replace all [REGEXP] in [TEXT] with [DEST] [GROUP]", + "RegExpVI.replace.group": "(refs enabled)", + "RegExpVI.replace.nogroup": "(refs disabled)", } }) } @@ -348,6 +358,69 @@ class RegExpVI { defaultValue: 1 } } + }, + { + opcode: "findall", + blockType: "command", + text: this.formatMessage("RegExpVI.findall"), + arguments: { + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "-?[0-9]+(?:\\.[0-9]*)?" + }, + TEXT: { + type: "string", + defaultValue: "I have 23 45." + }, + LIST: { + type: "string", + menu: "List" + } + } + }, { + opcode: "split", + blockType: "command", + text: this.formatMessage("RegExpVI.split"), + arguments: { + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "-?[0-9]+(?:\\.[0-9]*)?" + }, + TEXT: { + type: "string", + defaultValue: "I have 23 apples." + }, + LIST: { + type: "string", + menu: "List" + } + } + }, { + opcode: "replace", + blockType: "reporter", + text: this.formatMessage("RegExpVI.replace"), + arguments: { + REGEXP: { + type: "string", + menu: "RegExpPicker", + defaultValue: "-?[0-9]+(?:\\.[0-9]*)?" + }, + TEXT: { + type: "string", + defaultValue: "I have 23 apples." + }, + DEST: { + type: "string", + defaultValue: "45" + }, + GROUP: { + type: "string", + menu: "ReplaceGroup", + defaultValue: "false" + } + } } ], menus: { @@ -462,6 +535,18 @@ class RegExpVI { text: this.formatMessage("RegExpVI.range.norange"), value: "false" }] + }, + ReplaceGroup: { + items: [{ + text: this.formatMessage("RegExpVI.replace.group"), + value: "true" + }, { + text: this.formatMessage("RegExpVI.replace.nogroup"), + value: "false" + }] + }, + List: { + items: '_getListOfList' } } }; @@ -545,8 +630,9 @@ class RegExpVI { this.regexp = new RegExp("", "g"); this.text = ""; this.result = null; + return; } - this.text = args.TEXT; + this.text = String(args.TEXT); this.result = this.regexp.exec(this.text); } @@ -584,6 +670,98 @@ class RegExpVI { }; } + findall(args, util) { + let regexp; + let result = []; + try { + regexp = new RegExp(String(args.REGEXP), "g"); + } catch (e) { + // 这里应该提示用户表达式出错了 + return; + } + let text = String(args.TEXT); + let res; + while (res = regexp.exec(text), res !== null) { + result.push(res[0]); + } + let vari = util.target.lookupVariableById(args.LIST); + if (vari !== null) + vari.value = result; + } + + split(args, util) { + let regexp; + let result = []; + try { + regexp = new RegExp(String(args.REGEXP), "g"); + } catch (e) { + // 这里应该提示用户表达式出错了 + return; + } + let text = String(args.TEXT); + result = text.split(regexp); + let vari = util.target.lookupVariableById(args.LIST); + if (vari !== null) + vari.value = result; + } + + replace(args) { + let regexp; + let result = []; + try { + regexp = new RegExp(String(args.REGEXP), "g"); + } catch (e) { + // 这里应该提示用户表达式出错了 + return ""; + } + let text = String(args.TEXT); + let dest = String(args.DEST); + if (String(args.GROUP) === "false") { + dest = dest.replace(/\$/g, "$$$$"); + } + return text.replace(regexp, dest); + } + + _getListOfList() { + if (this.runtime === undefined) { + return []; + } + let list = []; + let varis = this.runtime._stageTarget.variables; + Object.keys(varis).forEach(i => { + let v = varis[i]; + if (v.type === "list") { + list.push({ + text: v.name, + value: v.id + }); + } + }); + if (this.runtime._editingTarget !== null && + !this.runtime._editingTarget.isStage) { + varis = this.runtime._editingTarget.variables; + Object.keys(varis).forEach(i => { + let v = varis[i]; + if (v.type === "list") { + list.push({ + text: v.name, + value: v.id + }); + } + }); + } + if (list.length === 0) { + list.push({ + text: "?", + value: "?" + }); + } + list.sort(function(a, b) { + return a.text < b.text ? -1 : a.text > b.text ? 1 : 0; + }); + return list; + } + } window.tempExt = { From 244525c0b7491908c44f70de059df6a27b89c1d3 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sun, 17 Jul 2022 00:09:55 +0800 Subject: [PATCH 007/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E6=97=A0=E6=B3=95?= =?UTF-8?q?=E5=8A=A0=E5=AF=86=E8=A7=A3=E5=AF=86=E6=95=B0=E5=AD=97=E7=9A=84?= =?UTF-8?q?bug(=E5=BF=98=E8=AE=B0=E8=BD=AC=E5=AD=97=E7=AC=A6=E4=B8=B2?= =?UTF-8?q?=E4=BA=86)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/Archivecode.js | 28 +++++++++++++++++++++++----- 1 file changed, 23 insertions(+), 5 deletions(-) diff --git a/Arkos/Archivecode.js b/Arkos/Archivecode.js index 4fc8bb5d..a0c1c9f9 100644 --- a/Arkos/Archivecode.js +++ b/Arkos/Archivecode.js @@ -2,7 +2,10 @@ import Cast from '../utils/cast.js' //import cover from './assets/cover2.png' //import icon from './assets/icon2.svg' -//合作者:Nights:帮忙写了代码框架,并且给予了一些技术上的指导 +//合作者: +// Nights: 搭了框架,以及一些技术帮助 +// -6: 修复了许多纰漏和 bug ,加入了新的加密算法 +// Arkos: 什么都不会的屑蒟蒻 // console.log(Cast.toNumber('123')) //console.log(Cast.toNumber('aab')) @@ -699,6 +702,8 @@ class Archive_code { //加密 encrypt(args) { + args.key = Cast.toString(args.key) + args.str = Cast.toString(args.str) switch (args.method) { case '1': return this.ArkosEncrypt(args); @@ -713,6 +718,8 @@ class Archive_code { //解密 decrypt(args) { + args.key = Cast.toString(args.key) + args.str = Cast.toString(args.str) switch (args.method) { case '1': return this.ArkosDecrypt(args); @@ -725,6 +732,17 @@ class Archive_code { } } + //发现 Unicode 为 0 10 13 55296~57343(2047个字符) 的字符无法被正常复制,故排除掉这些字符。 + getCode(c) { + c = Cast.toString(c).charCodeAt(0) + if (c === 0) return NaN + else if (c < 10) return c-1 //排除0 + else if (c < 13) return c-2 //排除0 10 + else if (c < 55296) return c-3 //排除0 10 13 + else if (c > 55296) return c-2050 //排除0 10 13 55296~57343(2047个字符) + else return NaN + } + //Arkos加密法 ArkosEncrypt(args) { args.key = this.keyVar(args.key) @@ -857,7 +875,7 @@ class Archive_code { } if (list.length === 0) { list.push({ - text: `*没有变量*`, + text: `-`, value: 'empty', }); } @@ -900,7 +918,7 @@ class Archive_code { } if (list.length === 0) { list.push({ - text: `*没有列表*`, + text: `-`, value: 'empty', }); } @@ -921,7 +939,7 @@ class Archive_code { }); if (list.length === 0) { list.push({ - text: '*没有变量*', + text: '-', value: 'empty', }); } @@ -942,7 +960,7 @@ class Archive_code { }); if (list.length === 0) { list.push({ - text: '*没有列表*', + text: '-', value: 'empty', }); } From 3ac4ac659ee94a4c30fccbd532c4c4650a68fa7c Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sun, 17 Jul 2022 09:59:01 +0800 Subject: [PATCH 008/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95Cast.toString?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/test.js | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/Arkos/test.js b/Arkos/test.js index 8098ca69..18641d80 100644 --- a/Arkos/test.js +++ b/Arkos/test.js @@ -1,3 +1,5 @@ +import Cast from '../utils/cast.js' + class StrictEqualityExtension { constructor(runtime) { this.runtime = runtime @@ -22,6 +24,17 @@ class StrictEqualityExtension { defaultValue: 'Second value' } } + }, + { + opcode: 'getType', + blockType: 'reporter', + text: '获得[n]的类型', + arguments: { + n: { + type: 'string', + defaultValue: '1' + } + } } ], menus: { @@ -42,6 +55,13 @@ class StrictEqualityExtension { getDynamicMenuItems() { return ['ddd 1', '大V 2', 'dd了 3']; } + + getType(args) { + console.log(Cast.toString(args.n)); + console.log(typeof Cast.toString(args.n)); + return typeof(args.n); + } + } From 4dc7da6b7fcf33ef4bb425cf385dde6d237d3951 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sun, 17 Jul 2022 10:22:13 +0800 Subject: [PATCH 009/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E7=9B=B4=E6=8E=A5?= =?UTF-8?q?=E8=AF=BB=E5=8F=96XX=E5=AE=B9=E5=99=A8=E4=B8=ADXX=E5=86=85?= =?UTF-8?q?=E5=AE=B9=E7=9A=84=E7=A7=AF=E6=9C=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/Archivecode.js | 36 ++++++++++++++++++++++++++++++++++-- 1 file changed, 34 insertions(+), 2 deletions(-) diff --git a/Arkos/Archivecode.js b/Arkos/Archivecode.js index a0c1c9f9..33d25928 100644 --- a/Arkos/Archivecode.js +++ b/Arkos/Archivecode.js @@ -60,7 +60,7 @@ class Archive_code { 'ArchiveCodeExt.infoMenu.3': '类型', 'ArchiveCodeExt.infoMenu.4': '列表长度', 'ArchiveCodeExt.delete': '删除容器中名为[key]的内容', - + 'ArchiveCodeExt.getContentInContainer': '获得容器[container]中名为[key]的内容', }, en: { @@ -94,7 +94,8 @@ class Archive_code { 'ArchiveCodeExt.infoMenu.2': 'value', 'ArchiveCodeExt.infoMenu.3': 'type', 'ArchiveCodeExt.infoMenu.4': 'lenth of list', - 'ArchiveCodeExt.delete': 'Delete content[key] in Container' + 'ArchiveCodeExt.delete': 'delete content[key] in Container', + 'ArchiveCodeExt.getContentInContainer': 'get [key] in [container]', }, }) @@ -349,6 +350,22 @@ class Archive_code { }, } }, + { + //直接获得container容器中的key内容 + opcode: 'getContentInContainer', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContentInContainer'), + arguments: { + container: { + type: 'string', + defaultValue: '{"金币":100,"经验值":50}' + }, + key: { + type: 'string', + defaultValue: '金币' + }, + } + }, { //加密 opcode: 'encrypt', @@ -526,6 +543,21 @@ class Archive_code { this.content[args.name] = list.value; } } + + //直接获得container容器中的key内容 + getContentInContainer(args) { + let content; + try { + content = JSON.parse(Cast.toString(args.container)) + if(typeof(content) === 'object' && !Array.isArray(content) && content !== null) { + return this._anythingToNumberString(content[args.key]); + }else{ + return '' + } + } catch (e) { + return '' + } + } deserialization(args) { let content; From 53828a496f12f429d3e77fd3ced2b0fb9ff718b8 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sun, 17 Jul 2022 10:56:09 +0800 Subject: [PATCH 010/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E8=8E=B7=E5=8F=96?= =?UTF-8?q?=E7=89=B9=E6=95=88=E7=A7=AF=E6=9C=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 76 +++++++++++++++++++++++++++++++++++++++++++++--- 1 file changed, 72 insertions(+), 4 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index dadec488..15b515ca 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -18,6 +18,14 @@ class ArkosExtensions { 'ArkosExt.insertString': '在[str]的第[pos]个字符前插入[substr]', 'ArkosExt.replaceString': '将[str]中的第[start]个到第[end]个字符,替换为[substr]', 'ArkosExt.turnDegreesToDir': '朝方向[dir]旋转[degree]度', + 'ArkosExt.getEffect': '获取特效[EFFECT]的值', + 'ArchiveCodeExt.color': '颜色', + 'ArchiveCodeExt.fisheye': '语言', + 'ArchiveCodeExt.whirl': '旋涡', + 'ArchiveCodeExt.pixelate': '像素化', + 'ArchiveCodeExt.mosaic': '马赛克', + 'ArchiveCodeExt.brightness': '亮度', + 'ArchiveCodeExt.ghost': '虚像', }, en: { @@ -30,6 +38,14 @@ class ArkosExtensions { 'ArkosExt.insertString': 'insert[substr]at[pos]of[str]', 'ArkosExt.replaceString': 'replace from[start]to[end]of[str],with[substr]', 'ArkosExt.turnDegreesToDir': 'turn[degree] degrees toward direction[dir]', + 'ArkosExt.getEffect': 'effect[EFFECT]', + 'ArchiveCodeExt.color': 'color', + 'ArchiveCodeExt.fisheye': 'fisheye', + 'ArchiveCodeExt.whirl': 'whirl', + 'ArchiveCodeExt.pixelate': 'pixelate', + 'ArchiveCodeExt.mosaic': 'mosaic', + 'ArchiveCodeExt.brightness': 'brightness', + 'ArchiveCodeExt.ghost': 'ghost', }, }) } @@ -202,15 +218,58 @@ class ArkosExtensions { arguments: { degree: { type: 'number', - defaultValue: 0, + defaultValue: 45, }, dir: { type: 'angle', - defaultValue: 0, + defaultValue: 10, + }, + }, + }, + { + //获取特效值 + opcode: 'getEffect', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getEffect'), + arguments: { + EFFECT: { + type: 'string', + menu: 'effectMenu', }, }, }, ], + menus: { + effectMenu: [{ + text: this.formatMessage('ArchiveCodeExt.color'), + value: 'color' + }, + { + text: this.formatMessage('ArchiveCodeExt.fisheye'), + value: 'fisheye' + }, + { + text: this.formatMessage('ArchiveCodeExt.whirl'), + value: 'whirl' + }, + { + text: this.formatMessage('ArchiveCodeExt.pixelate'), + value: 'pixelate' + }, + { + text: this.formatMessage('ArchiveCodeExt.mosaic'), + value: 'mosaic' + }, + { + text: this.formatMessage('ArchiveCodeExt.brightness'), + value: 'brightness' + }, + { + text: this.formatMessage('ArchiveCodeExt.ghost'), + value: 'ghost' + } + ] + }, } } @@ -286,17 +345,26 @@ class ArkosExtensions { return str.slice(0, start - 1) + substr + str.slice(end) } + turnDegreesToDir(args, util) { const degree = Cast.toNumber(args.degree); const dir = Cast.toNumber(args.dir); - const dif = this.differenceBetweenDirections(degree, dir); - if(Math.abs(dif) < degrees) + const dif = this.differenceBetweenDirections(util.target.direction, dir); + if(Math.abs(dif) < degree) util.target.setDirection(dir); else if(dif < 0) util.target.setDirection(util.target.direction - degree); else util.target.setDirection(util.target.direction + degree); } + + //获取特效的数值(照抄官方做法) + getEffect (args, util) { + const effect = Cast.toString(args.EFFECT).toLowerCase(); + if (!util.target.effects.hasOwnProperty(effect)) return 0; + return util.target.effects[effect]; + } + } From 46d4837e8f5d78f450ee1e29912cadca8100ee04 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sun, 17 Jul 2022 15:49:03 +0800 Subject: [PATCH 011/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E5=88=A4=E6=96=AD?= =?UTF-8?q?=E9=9A=90=E8=97=8F=E3=80=81=E6=97=8B=E8=BD=AC=E6=96=B9=E5=BC=8F?= =?UTF-8?q?=E3=80=81=E9=80=A0=E5=9E=8B=E9=95=BF=E5=AE=BD=E3=80=81=E5=BC=BA?= =?UTF-8?q?=E8=A1=8C=E8=AE=BE=E7=BD=AE=E5=A4=A7=E5=B0=8F=EF=BC=9B=20?= =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E6=9C=9D..=E6=96=B9=E5=90=91=E6=97=8B?= =?UTF-8?q?=E8=BD=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 155 +++++++++++++++++++++++++++++++++++++++-------- 1 file changed, 129 insertions(+), 26 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 15b515ca..722e70fb 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -19,13 +19,19 @@ class ArkosExtensions { 'ArkosExt.replaceString': '将[str]中的第[start]个到第[end]个字符,替换为[substr]', 'ArkosExt.turnDegreesToDir': '朝方向[dir]旋转[degree]度', 'ArkosExt.getEffect': '获取特效[EFFECT]的值', - 'ArchiveCodeExt.color': '颜色', - 'ArchiveCodeExt.fisheye': '语言', - 'ArchiveCodeExt.whirl': '旋涡', - 'ArchiveCodeExt.pixelate': '像素化', - 'ArchiveCodeExt.mosaic': '马赛克', - 'ArchiveCodeExt.brightness': '亮度', - 'ArchiveCodeExt.ghost': '虚像', + 'ArkosExt.color': '颜色', + 'ArkosExt.fisheye': '鱼眼', + 'ArkosExt.whirl': '旋涡', + 'ArkosExt.pixelate': '像素化', + 'ArkosExt.mosaic': '马赛克', + 'ArkosExt.brightness': '亮度', + 'ArkosExt.ghost': '虚像', + 'ArkosExt.ifVisible': '角色可见?', + 'ArkosExt.getRotationStyle': '当前旋转方式', + 'ArkosExt.getWidthOrHeight': '获取当前造型的[t]', + 'ArkosExt.setSize': '强行将大小设为[size]', + 'ArkosExt.width': '长', + 'ArkosExt.height': '宽', }, en: { @@ -39,13 +45,19 @@ class ArkosExtensions { 'ArkosExt.replaceString': 'replace from[start]to[end]of[str],with[substr]', 'ArkosExt.turnDegreesToDir': 'turn[degree] degrees toward direction[dir]', 'ArkosExt.getEffect': 'effect[EFFECT]', - 'ArchiveCodeExt.color': 'color', - 'ArchiveCodeExt.fisheye': 'fisheye', - 'ArchiveCodeExt.whirl': 'whirl', - 'ArchiveCodeExt.pixelate': 'pixelate', - 'ArchiveCodeExt.mosaic': 'mosaic', - 'ArchiveCodeExt.brightness': 'brightness', - 'ArchiveCodeExt.ghost': 'ghost', + 'ArkosExt.color': 'color', + 'ArkosExt.fisheye': 'fisheye', + 'ArkosExt.whirl': 'whirl', + 'ArkosExt.pixelate': 'pixelate', + 'ArkosExt.mosaic': 'mosaic', + 'ArkosExt.brightness': 'brightness', + 'ArkosExt.ghost': 'ghost', + 'ArkosExt.ifVisible': 'visible?', + 'ArkosExt.getRotationStyle': 'rotation style', + 'ArkosExt.getWidthOrHeight': 'get [t] of the current costume', + 'ArkosExt.setSize': 'Force the size to [size] %', + 'ArkosExt.width': 'width', + 'ArkosExt.height': 'height', }, }) } @@ -238,34 +250,83 @@ class ArkosExtensions { }, }, }, + { + //是否隐藏 + opcode: 'ifVisible', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.ifVisible'), + }, + { + //获取旋转方式 + opcode: 'getRotationStyle', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getRotationStyle'), + }, + { + //获取造型长宽 + opcode: 'getWidthOrHeight', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getWidthOrHeight'), + arguments: { + t: { + type: 'string', + menu: 'WOrH', + }, + }, + }, + { + //强行设置大小 + opcode: 'setSize', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.setSize'), + arguments: { + size: { + type: 'number', + defaultValue: 500, + }, + }, + }, ], menus: { - effectMenu: [{ - text: this.formatMessage('ArchiveCodeExt.color'), + //长0宽1 菜单 + WOrH: [ + { + text: this.formatMessage('ArkosExt.width'), + value: '0' + }, + { + text: this.formatMessage('ArkosExt.height'), + value: '1' + }, + ], + //特效菜单 + effectMenu: [ + { + text: this.formatMessage('ArkosExt.color'), value: 'color' }, { - text: this.formatMessage('ArchiveCodeExt.fisheye'), + text: this.formatMessage('ArkosExt.fisheye'), value: 'fisheye' }, { - text: this.formatMessage('ArchiveCodeExt.whirl'), + text: this.formatMessage('ArkosExt.whirl'), value: 'whirl' }, { - text: this.formatMessage('ArchiveCodeExt.pixelate'), + text: this.formatMessage('ArkosExt.pixelate'), value: 'pixelate' }, { - text: this.formatMessage('ArchiveCodeExt.mosaic'), + text: this.formatMessage('ArkosExt.mosaic'), value: 'mosaic' }, { - text: this.formatMessage('ArchiveCodeExt.brightness'), + text: this.formatMessage('ArkosExt.brightness'), value: 'brightness' }, { - text: this.formatMessage('ArchiveCodeExt.ghost'), + text: this.formatMessage('ArkosExt.ghost'), value: 'ghost' } ] @@ -349,7 +410,7 @@ class ArkosExtensions { turnDegreesToDir(args, util) { const degree = Cast.toNumber(args.degree); const dir = Cast.toNumber(args.dir); - const dif = this.differenceBetweenDirections(util.target.direction, dir); + const dif = this.differenceBetweenDirections({a: util.target.direction, b: dir}); if(Math.abs(dif) < degree) util.target.setDirection(dir); else if(dif < 0) @@ -358,11 +419,53 @@ class ArkosExtensions { util.target.setDirection(util.target.direction + degree); } - //获取特效的数值(照抄官方做法) + //获取特效的数值 getEffect (args, util) { - const effect = Cast.toString(args.EFFECT).toLowerCase(); + let effect = Cast.toString(args.EFFECT).toLowerCase(); if (!util.target.effects.hasOwnProperty(effect)) return 0; - return util.target.effects[effect]; + effect = util.target.effects[effect] + return effect; + } + + //角色是否可见 + ifVisible (args, util) { + return util.target.visible; + } + + //获取图层(逝一逝) + // getLayer (args, util) { + // return util.target.layer; + // } + + //获取当前角色的旋转方式 + getRotationStyle (args, util) { + let t = util.target.rotationStyle + return t; + } + + //获取当前造型的长/宽 + getWidthOrHeight (args, util) { + const costumeSize = util.target.renderer.getCurrentSkinSize(this.drawableID); + return costumeSize[args.t]; + } + + //强行设置大小(逝一逝) + setSize (args, util) { + if (util.target.isStage) { + return; + } + if (util.target.renderer) { + args.size = Cast.toNumber(args.size); + if(args.size < 0.1) args.size = 0.1; + util.target.size = args.size; + const {direction, scale} = util.target._getRenderedDirectionAndScale(); + util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); + if (util.target.visible) { + util.target.emit(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, util.target); + util.target.runtime.requestRedraw(); + } + } + util.target.runtime.requestTargetsUpdate(util.target); } } From e2ee2e795307e35d9c96871fc02cf15ed99ecf0b Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sun, 17 Jul 2022 15:58:48 +0800 Subject: [PATCH 012/357] =?UTF-8?q?=E5=B0=86=E8=AE=BE=E7=BD=AE=E5=A4=A7?= =?UTF-8?q?=E5=B0=8F=E7=9A=84=E7=A7=AF=E6=9C=A8=E6=94=B9=E6=88=90command?= =?UTF-8?q?=E7=B1=BB=E5=9E=8B;=20=E4=BF=AE=E5=A4=8D=E6=97=A0=E6=B3=95?= =?UTF-8?q?=E8=8E=B7=E5=8F=96=E5=9B=BE=E7=89=87=E9=95=BF=E5=AE=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 16 +++++++--------- 1 file changed, 7 insertions(+), 9 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 722e70fb..c3cd5387 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -29,7 +29,7 @@ class ArkosExtensions { 'ArkosExt.ifVisible': '角色可见?', 'ArkosExt.getRotationStyle': '当前旋转方式', 'ArkosExt.getWidthOrHeight': '获取当前造型的[t]', - 'ArkosExt.setSize': '强行将大小设为[size]', + 'ArkosExt.setSize': '⚠️强行将大小设为[size](无视限制)', 'ArkosExt.width': '长', 'ArkosExt.height': '宽', }, @@ -55,7 +55,7 @@ class ArkosExtensions { 'ArkosExt.ifVisible': 'visible?', 'ArkosExt.getRotationStyle': 'rotation style', 'ArkosExt.getWidthOrHeight': 'get [t] of the current costume', - 'ArkosExt.setSize': 'Force the size to [size] %', + 'ArkosExt.setSize': '⚠️Force the size to [size] % (regardless of limitation) ', 'ArkosExt.width': 'width', 'ArkosExt.height': 'height', }, @@ -277,7 +277,7 @@ class ArkosExtensions { { //强行设置大小 opcode: 'setSize', - blockType: 'reporter', + blockType: 'command', text: this.formatMessage('ArkosExt.setSize'), arguments: { size: { @@ -423,8 +423,7 @@ class ArkosExtensions { getEffect (args, util) { let effect = Cast.toString(args.EFFECT).toLowerCase(); if (!util.target.effects.hasOwnProperty(effect)) return 0; - effect = util.target.effects[effect] - return effect; + return util.target.effects[effect]; } //角色是否可见 @@ -439,13 +438,12 @@ class ArkosExtensions { //获取当前角色的旋转方式 getRotationStyle (args, util) { - let t = util.target.rotationStyle - return t; + return util.target.rotationStyle; } //获取当前造型的长/宽 getWidthOrHeight (args, util) { - const costumeSize = util.target.renderer.getCurrentSkinSize(this.drawableID); + const costumeSize = util.target.renderer.getCurrentSkinSize(util.target.drawableID); return costumeSize[args.t]; } @@ -461,7 +459,7 @@ class ArkosExtensions { const {direction, scale} = util.target._getRenderedDirectionAndScale(); util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); if (util.target.visible) { - util.target.emit(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, util.target); + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); util.target.runtime.requestRedraw(); } } From 979e547e69b1a4e497f65d719b5a72ab357b33c9 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 00:22:05 +0800 Subject: [PATCH 013/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E5=BC=BA=E8=A1=8C?= =?UTF-8?q?=E7=A7=BB=E5=88=B0xy=E3=80=81=E8=8E=B7=E5=8F=96=E8=A7=92?= =?UTF-8?q?=E8=89=B2=E8=BE=B9=E7=BC=98=E5=9D=90=E6=A0=87=E3=80=81=E5=88=A4?= =?UTF-8?q?=E6=96=AD=E8=A7=92=E8=89=B2=E6=98=AF=E5=90=A6=E5=9C=A8=E8=88=9E?= =?UTF-8?q?=E5=8F=B0=E5=A4=96?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 209 +++++++++++++++++++++++++++++++++++++++-------- 1 file changed, 176 insertions(+), 33 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index c3cd5387..37b58906 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -6,8 +6,8 @@ console.log(Cast.toNumber('123')) console.log(Cast.toNumber('aab')) class ArkosExtensions { constructor(runtime) { - this.runtime = runtime - this._formatMessage = runtime.getFormatMessage({ + util.target.runtime = runtime + util.target._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', @@ -18,6 +18,7 @@ class ArkosExtensions { 'ArkosExt.insertString': '在[str]的第[pos]个字符前插入[substr]', 'ArkosExt.replaceString': '将[str]中的第[start]个到第[end]个字符,替换为[substr]', 'ArkosExt.turnDegreesToDir': '朝方向[dir]旋转[degree]度', + 'ArkosExt.getEffect': '获取特效[EFFECT]的值', 'ArkosExt.color': '颜色', 'ArkosExt.fisheye': '鱼眼', @@ -26,12 +27,21 @@ class ArkosExtensions { 'ArkosExt.mosaic': '马赛克', 'ArkosExt.brightness': '亮度', 'ArkosExt.ghost': '虚像', + 'ArkosExt.ifVisible': '角色可见?', 'ArkosExt.getRotationStyle': '当前旋转方式', 'ArkosExt.getWidthOrHeight': '获取当前造型的[t]', 'ArkosExt.setSize': '⚠️强行将大小设为[size](无视限制)', - 'ArkosExt.width': '长', - 'ArkosExt.height': '宽', + 'ArkosExt.width': '宽', + 'ArkosExt.height': '高', + + 'ArkosExt.setXY': '⚠️强行移到x:[x]y:[y](无视边界)', + 'ArkosExt.getBoundaryCoord': '获取角色的[t]', + 'ArkosExt.top': '上边缘y', + 'ArkosExt.bottom': '下边缘y', + 'ArkosExt.left': '左边缘x', + 'ArkosExt.right': '右边缘x', + 'ArkosExt.isOutOfSight': '角色移到舞台区外?', }, en: { @@ -55,15 +65,23 @@ class ArkosExtensions { 'ArkosExt.ifVisible': 'visible?', 'ArkosExt.getRotationStyle': 'rotation style', 'ArkosExt.getWidthOrHeight': 'get [t] of the current costume', - 'ArkosExt.setSize': '⚠️Force the size to [size] % (regardless of limitation) ', + 'ArkosExt.setSize': '⚠️force the size to [size] % (regardless of limitation) ', 'ArkosExt.width': 'width', 'ArkosExt.height': 'height', + + 'ArkosExt.setXY': '⚠️force to x:[x]y:[y] (regardless of the boundary)', + 'ArkosExt.getBoundaryCoord': 'get [t] of the sprite', + 'ArkosExt.top': 'top y', + 'ArkosExt.bottom': 'bottom y', + 'ArkosExt.left': 'left x', + 'ArkosExt.right': 'right x', + 'ArkosExt.isOutOfSight': 'is out of stage?', }, }) } formatMessage(id) { - return this._formatMessage({ + return util.target._formatMessage({ id, default: id, description: id, @@ -73,7 +91,7 @@ class ArkosExtensions { getInfo() { return { id: 'hcnTest', // 拓展id - name: this.formatMessage('ArkosExt.extensionName'), // 拓展名 + name: util.target.formatMessage('ArkosExt.extensionName'), // 拓展名 color1: '#FF8383', // menuIconURI: icon, // blockIconURI: icon, @@ -82,7 +100,7 @@ class ArkosExtensions { // 判断相等(区分大小写) opcode: 'strictlyEquals', blockType: 'Boolean', - text: this.formatMessage('ArkosExt.stringEquality'), + text: util.target.formatMessage('ArkosExt.stringEquality'), arguments: { ONE: { type: 'string', @@ -98,7 +116,7 @@ class ArkosExtensions { // 计算点A到点B的方向 opcode: 'getDirFromAToB', blockType: 'reporter', - text: this.formatMessage('ArkosExt.directionFromAtoB'), + text: util.target.formatMessage('ArkosExt.directionFromAtoB'), arguments: { X1: { type: 'number', @@ -122,7 +140,7 @@ class ArkosExtensions { // 计算角b-角a的角度差 opcode: 'differenceBetweenDirections', blockType: 'reporter', - text: this.formatMessage('ArkosExt.differenceBetweenDirections'), + text: util.target.formatMessage('ArkosExt.differenceBetweenDirections'), arguments: { a: { type: 'number', @@ -138,7 +156,7 @@ class ArkosExtensions { // 两点距离 opcode: 'disFromAToB', blockType: 'reporter', - text: this.formatMessage('ArkosExt.distance'), + text: util.target.formatMessage('ArkosExt.distance'), arguments: { X1: { type: 'number', @@ -162,7 +180,7 @@ class ArkosExtensions { // 查找子字符串,从pos开始 opcode: 'indexof', blockType: 'reporter', - text: this.formatMessage('ArkosExt.searchString'), + text: util.target.formatMessage('ArkosExt.searchString'), arguments: { str: { type: 'string', @@ -182,7 +200,7 @@ class ArkosExtensions { // 在字符串中插入子字符串 opcode: 'insertStr', blockType: 'reporter', - text: this.formatMessage('ArkosExt.insertString'), + text: util.target.formatMessage('ArkosExt.insertString'), arguments: { str: { type: 'string', @@ -202,7 +220,7 @@ class ArkosExtensions { // 替换字符串中的从..到..的字符串 opcode: 'replaceStr', blockType: 'reporter', - text: this.formatMessage('ArkosExt.replaceString'), + text: util.target.formatMessage('ArkosExt.replaceString'), arguments: { str: { type: 'string', @@ -226,7 +244,7 @@ class ArkosExtensions { //朝..方向旋转..角度 opcode: 'turnDegreesToDir', blockType: 'command', - text: this.formatMessage('ArkosExt.turnDegreesToDir'), + text: util.target.formatMessage('ArkosExt.turnDegreesToDir'), arguments: { degree: { type: 'number', @@ -242,7 +260,7 @@ class ArkosExtensions { //获取特效值 opcode: 'getEffect', blockType: 'reporter', - text: this.formatMessage('ArkosExt.getEffect'), + text: util.target.formatMessage('ArkosExt.getEffect'), arguments: { EFFECT: { type: 'string', @@ -254,19 +272,19 @@ class ArkosExtensions { //是否隐藏 opcode: 'ifVisible', blockType: 'Boolean', - text: this.formatMessage('ArkosExt.ifVisible'), + text: util.target.formatMessage('ArkosExt.ifVisible'), }, { //获取旋转方式 opcode: 'getRotationStyle', blockType: 'reporter', - text: this.formatMessage('ArkosExt.getRotationStyle'), + text: util.target.formatMessage('ArkosExt.getRotationStyle'), }, { - //获取造型长宽 + //获取造型0宽1高 opcode: 'getWidthOrHeight', blockType: 'reporter', - text: this.formatMessage('ArkosExt.getWidthOrHeight'), + text: util.target.formatMessage('ArkosExt.getWidthOrHeight'), arguments: { t: { type: 'string', @@ -278,55 +296,120 @@ class ArkosExtensions { //强行设置大小 opcode: 'setSize', blockType: 'command', - text: this.formatMessage('ArkosExt.setSize'), + text: util.target.formatMessage('ArkosExt.setSize'), arguments: { size: { type: 'number', - defaultValue: 500, + defaultValue: 9999, }, }, }, + { + //强行移到xy + opcode: 'setXY', + blockType: 'command', + text: util.target.formatMessage('ArkosExt.setXY'), + arguments: { + x: { + type: 'number', + defaultValue: 100000, + }, + y: { + type: 'number', + defaultValue: 100000, + }, + }, + }, + { + //获取角色边缘xy + opcode: 'getBoundaryCoord', + blockType: 'command', + text: util.target.formatMessage('ArkosExt.getBoundaryCoord'), + arguments: { + t: { + type: 'string', + menu: 'boundaryMenu', + } + }, + }, + { + //是否跑到舞台外 + opcode: 'isOutOfSight', + blockType: 'Boolean', + text: util.target.formatMessage('ArkosExt.isOutOfSight'), + }, + { + //返回值转bool积木 + opcode: 'reporterToBoolean', + blockType: 'Boolean', + text: '[t]', + arguments: { + t: { + type: 'string', + menu: '1', + } + }, + }, ], menus: { - //长0宽1 菜单 + //角色上下左右边缘 + boundaryMenu: [ + { + text: util.target.formatMessage('ArkosExt.top'), + value: '1' + }, + { + text: util.target.formatMessage('ArkosExt.bottom'), + value: '2' + }, + { + text: util.target.formatMessage('ArkosExt.left'), + value: '3' + }, + { + text: util.target.formatMessage('ArkosExt.right'), + value: '4' + }, + ], + //0宽1高 菜单 WOrH: [ { - text: this.formatMessage('ArkosExt.width'), + text: util.target.formatMessage('ArkosExt.width'), value: '0' }, { - text: this.formatMessage('ArkosExt.height'), + text: util.target.formatMessage('ArkosExt.height'), value: '1' }, ], //特效菜单 effectMenu: [ { - text: this.formatMessage('ArkosExt.color'), + text: util.target.formatMessage('ArkosExt.color'), value: 'color' }, { - text: this.formatMessage('ArkosExt.fisheye'), + text: util.target.formatMessage('ArkosExt.fisheye'), value: 'fisheye' }, { - text: this.formatMessage('ArkosExt.whirl'), + text: util.target.formatMessage('ArkosExt.whirl'), value: 'whirl' }, { - text: this.formatMessage('ArkosExt.pixelate'), + text: util.target.formatMessage('ArkosExt.pixelate'), value: 'pixelate' }, { - text: this.formatMessage('ArkosExt.mosaic'), + text: util.target.formatMessage('ArkosExt.mosaic'), value: 'mosaic' }, { - text: this.formatMessage('ArkosExt.brightness'), + text: util.target.formatMessage('ArkosExt.brightness'), value: 'brightness' }, { - text: this.formatMessage('ArkosExt.ghost'), + text: util.target.formatMessage('ArkosExt.ghost'), value: 'ghost' } ] @@ -410,7 +493,7 @@ class ArkosExtensions { turnDegreesToDir(args, util) { const degree = Cast.toNumber(args.degree); const dir = Cast.toNumber(args.dir); - const dif = this.differenceBetweenDirections({a: util.target.direction, b: dir}); + const dif = util.target.differenceBetweenDirections({a: util.target.direction, b: dir}); if(Math.abs(dif) < degree) util.target.setDirection(dir); else if(dif < 0) @@ -466,6 +549,66 @@ class ArkosExtensions { util.target.runtime.requestTargetsUpdate(util.target); } + //强行设置XY(逝一逝) + setXY (args, util) { + if (util.target.isStage) return; + const oldX = util.target.x; + const oldY = util.target.y; + util.target.x = args.x; + util.target.y = args.y; + if (util.target.renderer) { + util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); + if (util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } + } else { + util.target.x = x; + util.target.y = y; + } + util.target.emit('TARGET_MOVED', util.target, oldX, oldY, false); + util.target.runtime.requestTargetsUpdate(util.target); + } + + //获取角色边缘的坐标 + getBoundaryCoord (args) { + switch(args.t){ + case '1': + return bounds.top; + case '2': + return bounds.bottom; + case '3': + return bounds.left; + case '4': + return bounds.right; + default: + return ''; + } + } + + //是否在舞台外 + isOutOfSight (util) { + if (util.target.renderer) { + const stageWidth = util.target.runtime.constructor.STAGE_WIDTH; + const stageHeight = util.target.runtime.constructor.STAGE_HEIGHT; + const bounds = util.target.getBounds(); + if (bounds.right < -stageWidth / 2 || + bounds.left > stageWidth / 2 || + bounds.bottom > stageHeight / 2 || + bounds.top < -stageHeight / 2) { + return true; + } + } + return false; + } + + //形如:<() > + reporterToBoolean (args){ + return args.t ? true : false; + } + + + } From 59c0c6092ee0200deb154c3dcd29aa58e79e1a2b Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 00:30:34 +0800 Subject: [PATCH 014/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E4=B8=80=E4=B8=AA?= =?UTF-8?q?=E4=B8=A5=E9=87=8D=E7=9A=84=E9=94=99=E8=AF=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 68 ++++++++++++++++++++++++------------------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 37b58906..96f9b3cb 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -6,8 +6,8 @@ console.log(Cast.toNumber('123')) console.log(Cast.toNumber('aab')) class ArkosExtensions { constructor(runtime) { - util.target.runtime = runtime - util.target._formatMessage = runtime.getFormatMessage({ + this.runtime = runtime + this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', @@ -81,7 +81,7 @@ class ArkosExtensions { } formatMessage(id) { - return util.target._formatMessage({ + return this._formatMessage({ id, default: id, description: id, @@ -91,7 +91,7 @@ class ArkosExtensions { getInfo() { return { id: 'hcnTest', // 拓展id - name: util.target.formatMessage('ArkosExt.extensionName'), // 拓展名 + name: this.formatMessage('ArkosExt.extensionName'), // 拓展名 color1: '#FF8383', // menuIconURI: icon, // blockIconURI: icon, @@ -100,7 +100,7 @@ class ArkosExtensions { // 判断相等(区分大小写) opcode: 'strictlyEquals', blockType: 'Boolean', - text: util.target.formatMessage('ArkosExt.stringEquality'), + text: this.formatMessage('ArkosExt.stringEquality'), arguments: { ONE: { type: 'string', @@ -116,7 +116,7 @@ class ArkosExtensions { // 计算点A到点B的方向 opcode: 'getDirFromAToB', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.directionFromAtoB'), + text: this.formatMessage('ArkosExt.directionFromAtoB'), arguments: { X1: { type: 'number', @@ -140,7 +140,7 @@ class ArkosExtensions { // 计算角b-角a的角度差 opcode: 'differenceBetweenDirections', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.differenceBetweenDirections'), + text: this.formatMessage('ArkosExt.differenceBetweenDirections'), arguments: { a: { type: 'number', @@ -156,7 +156,7 @@ class ArkosExtensions { // 两点距离 opcode: 'disFromAToB', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.distance'), + text: this.formatMessage('ArkosExt.distance'), arguments: { X1: { type: 'number', @@ -180,7 +180,7 @@ class ArkosExtensions { // 查找子字符串,从pos开始 opcode: 'indexof', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.searchString'), + text: this.formatMessage('ArkosExt.searchString'), arguments: { str: { type: 'string', @@ -200,7 +200,7 @@ class ArkosExtensions { // 在字符串中插入子字符串 opcode: 'insertStr', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.insertString'), + text: this.formatMessage('ArkosExt.insertString'), arguments: { str: { type: 'string', @@ -220,7 +220,7 @@ class ArkosExtensions { // 替换字符串中的从..到..的字符串 opcode: 'replaceStr', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.replaceString'), + text: this.formatMessage('ArkosExt.replaceString'), arguments: { str: { type: 'string', @@ -244,7 +244,7 @@ class ArkosExtensions { //朝..方向旋转..角度 opcode: 'turnDegreesToDir', blockType: 'command', - text: util.target.formatMessage('ArkosExt.turnDegreesToDir'), + text: this.formatMessage('ArkosExt.turnDegreesToDir'), arguments: { degree: { type: 'number', @@ -260,7 +260,7 @@ class ArkosExtensions { //获取特效值 opcode: 'getEffect', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.getEffect'), + text: this.formatMessage('ArkosExt.getEffect'), arguments: { EFFECT: { type: 'string', @@ -272,19 +272,19 @@ class ArkosExtensions { //是否隐藏 opcode: 'ifVisible', blockType: 'Boolean', - text: util.target.formatMessage('ArkosExt.ifVisible'), + text: this.formatMessage('ArkosExt.ifVisible'), }, { //获取旋转方式 opcode: 'getRotationStyle', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.getRotationStyle'), + text: this.formatMessage('ArkosExt.getRotationStyle'), }, { //获取造型0宽1高 opcode: 'getWidthOrHeight', blockType: 'reporter', - text: util.target.formatMessage('ArkosExt.getWidthOrHeight'), + text: this.formatMessage('ArkosExt.getWidthOrHeight'), arguments: { t: { type: 'string', @@ -296,7 +296,7 @@ class ArkosExtensions { //强行设置大小 opcode: 'setSize', blockType: 'command', - text: util.target.formatMessage('ArkosExt.setSize'), + text: this.formatMessage('ArkosExt.setSize'), arguments: { size: { type: 'number', @@ -308,7 +308,7 @@ class ArkosExtensions { //强行移到xy opcode: 'setXY', blockType: 'command', - text: util.target.formatMessage('ArkosExt.setXY'), + text: this.formatMessage('ArkosExt.setXY'), arguments: { x: { type: 'number', @@ -324,7 +324,7 @@ class ArkosExtensions { //获取角色边缘xy opcode: 'getBoundaryCoord', blockType: 'command', - text: util.target.formatMessage('ArkosExt.getBoundaryCoord'), + text: this.formatMessage('ArkosExt.getBoundaryCoord'), arguments: { t: { type: 'string', @@ -336,7 +336,7 @@ class ArkosExtensions { //是否跑到舞台外 opcode: 'isOutOfSight', blockType: 'Boolean', - text: util.target.formatMessage('ArkosExt.isOutOfSight'), + text: this.formatMessage('ArkosExt.isOutOfSight'), }, { //返回值转bool积木 @@ -355,61 +355,61 @@ class ArkosExtensions { //角色上下左右边缘 boundaryMenu: [ { - text: util.target.formatMessage('ArkosExt.top'), + text: this.formatMessage('ArkosExt.top'), value: '1' }, { - text: util.target.formatMessage('ArkosExt.bottom'), + text: this.formatMessage('ArkosExt.bottom'), value: '2' }, { - text: util.target.formatMessage('ArkosExt.left'), + text: this.formatMessage('ArkosExt.left'), value: '3' }, { - text: util.target.formatMessage('ArkosExt.right'), + text: this.formatMessage('ArkosExt.right'), value: '4' }, ], //0宽1高 菜单 WOrH: [ { - text: util.target.formatMessage('ArkosExt.width'), + text: this.formatMessage('ArkosExt.width'), value: '0' }, { - text: util.target.formatMessage('ArkosExt.height'), + text: this.formatMessage('ArkosExt.height'), value: '1' }, ], //特效菜单 effectMenu: [ { - text: util.target.formatMessage('ArkosExt.color'), + text: this.formatMessage('ArkosExt.color'), value: 'color' }, { - text: util.target.formatMessage('ArkosExt.fisheye'), + text: this.formatMessage('ArkosExt.fisheye'), value: 'fisheye' }, { - text: util.target.formatMessage('ArkosExt.whirl'), + text: this.formatMessage('ArkosExt.whirl'), value: 'whirl' }, { - text: util.target.formatMessage('ArkosExt.pixelate'), + text: this.formatMessage('ArkosExt.pixelate'), value: 'pixelate' }, { - text: util.target.formatMessage('ArkosExt.mosaic'), + text: this.formatMessage('ArkosExt.mosaic'), value: 'mosaic' }, { - text: util.target.formatMessage('ArkosExt.brightness'), + text: this.formatMessage('ArkosExt.brightness'), value: 'brightness' }, { - text: util.target.formatMessage('ArkosExt.ghost'), + text: this.formatMessage('ArkosExt.ghost'), value: 'ghost' } ] @@ -493,7 +493,7 @@ class ArkosExtensions { turnDegreesToDir(args, util) { const degree = Cast.toNumber(args.degree); const dir = Cast.toNumber(args.dir); - const dif = util.target.differenceBetweenDirections({a: util.target.direction, b: dir}); + const dif = this.differenceBetweenDirections({a: util.target.direction, b: dir}); if(Math.abs(dif) < degree) util.target.setDirection(dir); else if(dif < 0) From 1c493b09f9857535a458778e62f96a986c5f8feb Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 00:39:56 +0800 Subject: [PATCH 015/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E7=A7=AF=E6=9C=A8?= =?UTF-8?q?=E7=B1=BB=E5=9E=8B=E9=94=99=E8=AF=AFbug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 96f9b3cb..3e37e85a 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -323,7 +323,7 @@ class ArkosExtensions { { //获取角色边缘xy opcode: 'getBoundaryCoord', - blockType: 'command', + blockType: 'reporter', text: this.formatMessage('ArkosExt.getBoundaryCoord'), arguments: { t: { @@ -572,6 +572,7 @@ class ArkosExtensions { //获取角色边缘的坐标 getBoundaryCoord (args) { + const bounds = util.target.getBounds(); switch(args.t){ case '1': return bounds.top; @@ -604,7 +605,7 @@ class ArkosExtensions { //形如:<() > reporterToBoolean (args){ - return args.t ? true : false; + return (args.t) ? true : false; } From bff241a4748110273f2e1d0fec7c5e6caa6a9856 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 00:53:17 +0800 Subject: [PATCH 016/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E7=A7=AF=E6=9C=A8?= =?UTF-8?q?=E4=B8=8D=E6=98=BE=E7=A4=BAbug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 3e37e85a..91754094 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -346,7 +346,7 @@ class ArkosExtensions { arguments: { t: { type: 'string', - menu: '1', + defaultValue: '1', } }, }, @@ -571,8 +571,9 @@ class ArkosExtensions { } //获取角色边缘的坐标 - getBoundaryCoord (args) { - const bounds = util.target.getBounds(); + getBoundaryCoord (args, util) { + console.log(util.target.runtime.renderer) + const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); switch(args.t){ case '1': return bounds.top; From a3641f9b3ad1c07e477d331e16f6416cca5d608f Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 01:02:01 +0800 Subject: [PATCH 017/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E6=A3=80=E6=B5=8B?= =?UTF-8?q?=E6=98=AF=E5=90=A6=E5=86=8D=E8=88=9E=E5=8F=B0=E5=86=85=E7=A7=AF?= =?UTF-8?q?=E6=9C=A8=E6=97=A0=E6=B3=95=E8=BF=90=E8=A1=8C=E7=9A=84bug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index 91754094..e16c98b4 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -593,7 +593,7 @@ class ArkosExtensions { if (util.target.renderer) { const stageWidth = util.target.runtime.constructor.STAGE_WIDTH; const stageHeight = util.target.runtime.constructor.STAGE_HEIGHT; - const bounds = util.target.getBounds(); + const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); if (bounds.right < -stageWidth / 2 || bounds.left > stageWidth / 2 || bounds.bottom > stageHeight / 2 || @@ -606,6 +606,7 @@ class ArkosExtensions { //形如:<() > reporterToBoolean (args){ + if(Cast.toString(args.t).toLowerCase() === 'false') return false; return (args.t) ? true : false; } From b1dbdd9c3b5b9535e3e48283fbc5a46f23f0f2bf Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 01:05:57 +0800 Subject: [PATCH 018/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E5=88=A4=E6=96=AD0?= =?UTF-8?q?=E7=9A=84=E6=83=85=E5=86=B5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 1 + 1 file changed, 1 insertion(+) diff --git a/Arkos/project.js b/Arkos/project.js index e16c98b4..9bbc919d 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -607,6 +607,7 @@ class ArkosExtensions { //形如:<() > reporterToBoolean (args){ if(Cast.toString(args.t).toLowerCase() === 'false') return false; + if(args.t === '0') return false; return (args.t) ? true : false; } From 259c1f3738481913767ef5947c8fb9ed47396e4c Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 01:12:31 +0800 Subject: [PATCH 019/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95=EF=BC=8Clog=20render?= =?UTF-8?q?er?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 9bbc919d..edf83be0 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -572,7 +572,6 @@ class ArkosExtensions { //获取角色边缘的坐标 getBoundaryCoord (args, util) { - console.log(util.target.runtime.renderer) const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); switch(args.t){ case '1': @@ -590,7 +589,9 @@ class ArkosExtensions { //是否在舞台外 isOutOfSight (util) { - if (util.target.renderer) { + console.log(util.target.runtime.renderer) + console.log(util.target.renderer) + //if (util.target.renderer) { const stageWidth = util.target.runtime.constructor.STAGE_WIDTH; const stageHeight = util.target.runtime.constructor.STAGE_HEIGHT; const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); @@ -600,7 +601,7 @@ class ArkosExtensions { bounds.top < -stageHeight / 2) { return true; } - } + //} return false; } From e31e31b4798c848eaec6e2760d7ea00c3c9f4821 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 01:20:51 +0800 Subject: [PATCH 020/357] =?UTF-8?q?=E8=AF=95=E5=9B=BE=E4=BF=AE=E5=A4=8D?= =?UTF-8?q?=E7=A7=AF=E6=9C=A8=E8=BF=90=E8=A1=8C=E4=B8=8D=E4=BA=86=E7=9A=84?= =?UTF-8?q?bug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index edf83be0..e64e6799 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -589,8 +589,6 @@ class ArkosExtensions { //是否在舞台外 isOutOfSight (util) { - console.log(util.target.runtime.renderer) - console.log(util.target.renderer) //if (util.target.renderer) { const stageWidth = util.target.runtime.constructor.STAGE_WIDTH; const stageHeight = util.target.runtime.constructor.STAGE_HEIGHT; From 7dd13615089cddcba64e7473402f2f68fe2a9bac Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Mon, 18 Jul 2022 10:31:35 +0800 Subject: [PATCH 021/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E4=BD=BF=E7=A7=AF?= =?UTF-8?q?=E6=9C=A8=E8=83=BD=E8=BF=90=E8=A1=8C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index edf83be0..a4909a4d 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -589,8 +589,8 @@ class ArkosExtensions { //是否在舞台外 isOutOfSight (util) { - console.log(util.target.runtime.renderer) - console.log(util.target.renderer) + // console.log(util.target.runtime.renderer) + // console.log(util.target.renderer) //if (util.target.renderer) { const stageWidth = util.target.runtime.constructor.STAGE_WIDTH; const stageHeight = util.target.runtime.constructor.STAGE_HEIGHT; From 49552f33a176103f47d9f283e3717d7f0730fe1a Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Mon, 18 Jul 2022 11:08:17 +0800 Subject: [PATCH 022/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E4=BD=BF=E7=A7=AF?= =?UTF-8?q?=E6=9C=A8=E8=83=BD=E8=BF=90=E8=A1=8C*2?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index a4909a4d..d926e32f 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -588,7 +588,7 @@ class ArkosExtensions { } //是否在舞台外 - isOutOfSight (util) { + isOutOfSight (args, util) { // console.log(util.target.runtime.renderer) // console.log(util.target.renderer) //if (util.target.renderer) { From 496e9e909179491b9529dbd75a918775a5369a82 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Mon, 18 Jul 2022 11:34:06 +0800 Subject: [PATCH 023/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E4=BD=BF=E7=A7=AF?= =?UTF-8?q?=E6=9C=A8=E8=83=BD=E8=BF=90=E8=A1=8C*3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index d926e32f..d7312c30 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -591,9 +591,11 @@ class ArkosExtensions { isOutOfSight (args, util) { // console.log(util.target.runtime.renderer) // console.log(util.target.renderer) - //if (util.target.renderer) { + if (util.target.renderer) { const stageWidth = util.target.runtime.constructor.STAGE_WIDTH; const stageHeight = util.target.runtime.constructor.STAGE_HEIGHT; + console.log('stageWidth',stageWidth); + console.log('stageHeight',stageHeight); const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); if (bounds.right < -stageWidth / 2 || bounds.left > stageWidth / 2 || @@ -601,7 +603,7 @@ class ArkosExtensions { bounds.top < -stageHeight / 2) { return true; } - //} + } return false; } From e79c99cd5fd92ffe9a1ce0a6d07ef9a41383dc04 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Mon, 18 Jul 2022 11:46:43 +0800 Subject: [PATCH 024/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E6=94=B9=E8=88=9E?= =?UTF-8?q?=E5=8F=B0=E5=AE=BD=E5=BA=A6=E4=B8=8D=E6=AD=A3=E7=A1=AE=E7=9A=84?= =?UTF-8?q?bug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index d7312c30..85f49207 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -592,8 +592,8 @@ class ArkosExtensions { // console.log(util.target.runtime.renderer) // console.log(util.target.renderer) if (util.target.renderer) { - const stageWidth = util.target.runtime.constructor.STAGE_WIDTH; - const stageHeight = util.target.runtime.constructor.STAGE_HEIGHT; + const stageWidth = util.target.runtime.STAGE_WIDTH; + const stageHeight = util.target.runtime.STAGE_HEIGHT; console.log('stageWidth',stageWidth); console.log('stageHeight',stageHeight); const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); From 25bc97611776dc37d556e5b32a2b7e8a288e50c5 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Mon, 18 Jul 2022 12:32:46 +0800 Subject: [PATCH 025/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E4=BD=BF=E7=A7=AF?= =?UTF-8?q?=E6=9C=A8=E8=83=BD=E8=BF=90=E8=A1=8C*3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 85f49207..fe59e3bc 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -592,8 +592,8 @@ class ArkosExtensions { // console.log(util.target.runtime.renderer) // console.log(util.target.renderer) if (util.target.renderer) { - const stageWidth = util.target.runtime.STAGE_WIDTH; - const stageHeight = util.target.runtime.STAGE_HEIGHT; + const stageWidth = this.runtime.STAGE_WIDTH; + const stageHeight = this.runtime.STAGE_HEIGHT; console.log('stageWidth',stageWidth); console.log('stageHeight',stageHeight); const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); From 300b2c0842f20a6a4ecea8ae62828dee662eb724 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Mon, 18 Jul 2022 12:47:19 +0800 Subject: [PATCH 026/357] =?UTF-8?q?=E4=BD=BF=E7=A7=AF=E6=9C=A8=E8=83=BD?= =?UTF-8?q?=E8=BF=90=E8=A1=8C*4?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index fe59e3bc..f329c2f4 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -592,8 +592,8 @@ class ArkosExtensions { // console.log(util.target.runtime.renderer) // console.log(util.target.renderer) if (util.target.renderer) { - const stageWidth = this.runtime.STAGE_WIDTH; - const stageHeight = this.runtime.STAGE_HEIGHT; + const stageWidth = util.target.runtime.stageWidth; + const stageHeight = util.target.runtime.stageHeight; console.log('stageWidth',stageWidth); console.log('stageHeight',stageHeight); const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); From aeddc154641ee3a6befba5bff676d89eec0177c2 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Mon, 18 Jul 2022 19:08:31 +0800 Subject: [PATCH 027/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E5=BD=A2=E5=A6=821<2?= =?UTF-8?q?<3=E7=9A=84=E5=88=A4=E6=96=AD=E7=A7=AF=E6=9C=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 55 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 55 insertions(+) diff --git a/Arkos/project.js b/Arkos/project.js index f329c2f4..4ea62887 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -350,8 +350,40 @@ class ArkosExtensions { } }, }, + { + //形如 a≤b≤c + opcode: 'compareTwoSides', + blockType: 'Boolean', + text: '[a][op1][b][op2][c]', + arguments: { + a: { + type: 'string', + defaultValue: '1', + }, + b: { + type: 'string', + defaultValue: 'x', + }, + c: { + type: 'string', + defaultValue: '3', + }, + op1: { + type: 'string', + menu: 'opMenu1', + }, + op2: { + type: 'string', + menu: 'opMenu1', + }, + }, + }, ], menus: { + //判断符菜单 + opMenu1: ['<','≤','=','≠',], + opMenu2: ['<','>','≤','≥','=','≠',], + //角色上下左右边缘 boundaryMenu: [ { @@ -614,6 +646,29 @@ class ArkosExtensions { return (args.t) ? true : false; } + compare(a,b,op){ + switch(op){ + case '<': + return Cast.compare(a, b) < 0; + case '>': + return Cast.compare(a, b) > 0; + case '=': + return Cast.compare(a, b) === 0; + case '≤': + return Cast.compare(a, b) <= 0; + case '≥': + return Cast.compare(a, b) >= 0; + case '≠': + return Cast.compare(a, b) !== 0; + } + } + + //形如:a≤b≤c op1,op2 + compareTwoSides (args){ + if(this.compare(args.a, args.b, args.op1)&&this.compare(args.b, args.c, args.op2)) return true; + else return false; + } + } From 73b54910b12ce42ffae9155208c94c2bea18d34b Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Mon, 18 Jul 2022 21:12:03 +0800 Subject: [PATCH 028/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E5=BD=A2=E5=A6=82a?= =?UTF-8?q?=E2=89=A4b=E4=B8=94>c=E7=A7=AF=E6=9C=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 71 +++++++++++++++++++++++++++++++++++++++++++++--- 1 file changed, 67 insertions(+), 4 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 4ea62887..0b3c84c9 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -42,6 +42,9 @@ class ArkosExtensions { 'ArkosExt.left': '左边缘x', 'ArkosExt.right': '右边缘x', 'ArkosExt.isOutOfSight': '角色移到舞台区外?', + + 'ArkosExt.and': '且', + 'ArkosExt.or': '或', }, en: { @@ -76,6 +79,9 @@ class ArkosExtensions { 'ArkosExt.left': 'left x', 'ArkosExt.right': 'right x', 'ArkosExt.isOutOfSight': 'is out of stage?', + + 'ArkosExt.and': 'and', + 'ArkosExt.or': 'or', }, }) } @@ -378,12 +384,57 @@ class ArkosExtensions { }, }, }, + { + //形如:a≤b且/或>c op1,op2 logic compareTwoSidesPlus + opcode: 'compareTwoSidesPlus', + blockType: 'Boolean', + text: '[a][op1][b][logic][op2][c]', + arguments: { + a: { + type: 'string', + defaultValue: '1', + }, + b: { + type: 'string', + defaultValue: 'x', + }, + c: { + type: 'string', + defaultValue: '3', + }, + op1: { + type: 'string', + menu: 'opMenu2', + defaultValue: '<', + }, + op2: { + type: 'string', + menu: 'opMenu2', + defaultValue: '>', + }, + logic: { + type: 'string', + menu: 'logicMenu', + defaultValue: 'or', + }, + }, + }, ], menus: { //判断符菜单 opMenu1: ['<','≤','=','≠',], opMenu2: ['<','>','≤','≥','=','≠',], - + //logicMenu + logicMenu: [ + { + text: this.formatMessage('ArkosExt.or'), + value: 'or' + }, + { + text: this.formatMessage('ArkosExt.and'), + value: 'and' + }, + ], //角色上下左右边缘 boundaryMenu: [ { @@ -660,15 +711,27 @@ class ArkosExtensions { return Cast.compare(a, b) >= 0; case '≠': return Cast.compare(a, b) !== 0; + default: + return false; } } //形如:a≤b≤c op1,op2 compareTwoSides (args){ - if(this.compare(args.a, args.b, args.op1)&&this.compare(args.b, args.c, args.op2)) return true; - else return false; + return this.compare(args.a, args.b, args.op1)&&this.compare(args.b, args.c, args.op2) + } + + //形如:a≤b且/或>c op1,op2 logic + compareTwoSidesPlus(args){ + switch(args.logic){ + case 'or': + return this.compare(args.a, args.b, args.op1)||this.compare(args.b, args.c, args.op2) + case 'and': + return this.compare(args.a, args.b, args.op1)&&this.compare(args.b, args.c, args.op2) + default: + return false; + } } - } From 3d9b9248f138df47f4eb85d7e028e697dfd45dfd Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 00:49:00 +0800 Subject: [PATCH 029/357] =?UTF-8?q?=E5=A4=A7=E6=9B=B4=E6=96=B0=EF=BC=9A?= =?UTF-8?q?=E5=8A=A0=E5=85=A5=E6=8E=92=E5=BA=8F=E8=A1=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 348 +++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 338 insertions(+), 10 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 0b3c84c9..3fbdaeed 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -7,6 +7,10 @@ console.log(Cast.toNumber('aab')) class ArkosExtensions { constructor(runtime) { this.runtime = runtime + this.sortedTable={ + list1:{order:'desc',list:[]}, + list2:{order:'desc',list:[]}, + } this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', @@ -28,7 +32,7 @@ class ArkosExtensions { 'ArkosExt.brightness': '亮度', 'ArkosExt.ghost': '虚像', - 'ArkosExt.ifVisible': '角色可见?', + 'ArkosExt.isHiding': '角色隐藏?', 'ArkosExt.getRotationStyle': '当前旋转方式', 'ArkosExt.getWidthOrHeight': '获取当前造型的[t]', 'ArkosExt.setSize': '⚠️强行将大小设为[size](无视限制)', @@ -45,6 +49,21 @@ class ArkosExtensions { 'ArkosExt.and': '且', 'ArkosExt.or': '或', + + 'ArkosExt.clearSortedTable': '📊清空排序表[list]', + 'ArkosExt.setTypeOfSortedTable': '📊将排序表[list]的排序方式设为[type]', + 'ArkosExt.addToSortedTable': '📊将内容(重名的则覆盖)[name],排序值[value]加入排序表[list],附加信息[extra]', + 'ArkosExt.getFromSortedTableByNo': '📊获取排序表[list]中第[n]项的[t]', + 'ArkosExt.getFromSortedTableByName': '📊获取[name]在排序表[list]中的[t]', + 'ArkosExt.lengthOfSortedTable': '📊排序表[list]中内容数', + 'ArkosExt.deleteNameOfSortedTable': '📊删除排序表[list]中名为[name]的项', + 'ArkosExt.asc': '升序', + 'ArkosExt.desc': '降序', + + 'ArkosExt.name': '名称', + 'ArkosExt.rank': '表中位置', + 'ArkosExt.rankValue': '排序值', + 'ArkosExt.extra': '附加信息', }, en: { @@ -65,7 +84,7 @@ class ArkosExtensions { 'ArkosExt.mosaic': 'mosaic', 'ArkosExt.brightness': 'brightness', 'ArkosExt.ghost': 'ghost', - 'ArkosExt.ifVisible': 'visible?', + 'ArkosExt.isHiding': 'is hiding?', 'ArkosExt.getRotationStyle': 'rotation style', 'ArkosExt.getWidthOrHeight': 'get [t] of the current costume', 'ArkosExt.setSize': '⚠️force the size to [size] % (regardless of limitation) ', @@ -82,6 +101,21 @@ class ArkosExtensions { 'ArkosExt.and': 'and', 'ArkosExt.or': 'or', + + 'ArkosExt.clearSortedTable': '📊empty sorted table[list]', + 'ArkosExt.setTypeOfSortedTable': '📊set sort order of[list]to[type]', + 'ArkosExt.addToSortedTable': '📊add (overwrite if existed)[name]to table[list] with sort index value[value],extra data[extra] and sort', + 'ArkosExt.getFromSortedTableByNo': '📊get[t]of #[n] from [list]', + 'ArkosExt.getFromSortedTableByName': '📊get[t]of [name] from [list]', + 'ArkosExt.lengthOfSortedTable': '📊length of sorted table[list]', + 'ArkosExt.deleteNameOfSortedTable': '📊delete [name] in[list]', + 'ArkosExt.asc': 'ascending order', + 'ArkosExt.desc': 'descending order', + + 'ArkosExt.name': 'name', + 'ArkosExt.rank': 'rank', + 'ArkosExt.rankValue': 'rankValue', + 'ArkosExt.extra': 'extra', }, }) } @@ -276,9 +310,9 @@ class ArkosExtensions { }, { //是否隐藏 - opcode: 'ifVisible', + opcode: 'isHiding', blockType: 'Boolean', - text: this.formatMessage('ArkosExt.ifVisible'), + text: this.formatMessage('ArkosExt.isHiding'), }, { //获取旋转方式 @@ -392,11 +426,11 @@ class ArkosExtensions { arguments: { a: { type: 'string', - defaultValue: '1', + defaultValue: 'x', }, b: { type: 'string', - defaultValue: 'x', + defaultValue: '1', }, c: { type: 'string', @@ -419,8 +453,161 @@ class ArkosExtensions { }, }, }, + { + //📊清空排序表 + opcode: 'clearSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + }, + }, + { + //📊排序表排序方式 + opcode: 'setTypeOfSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.setTypeOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + type: { + type: 'string', + menu: 'sortOrder', + }, + }, + }, + { + //📊将XX加入排序表 + opcode: 'addToSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.addToSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + value: { + type: 'number', + defaultValue: '95', + }, + extra: { + type: 'string', + defaultValue: '20212490', + }, + }, + }, + { + //📊获取排序表第n项 + opcode: 'getFromSortedTableByNo', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getFromSortedTableByNo'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + defaultValue: '1', + menu: 'tableItemPropertyMenu', + }, + }, + }, + { + //📊获取排序表特定名字内容 + opcode: 'getFromSortedTableByName', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getFromSortedTableByName'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + t: { + type: 'string', + defaultValue: '2', + menu: 'tableItemPropertyMenu', + }, + }, + }, + { + //📊获取排序表长度 + opcode: 'lengthOfSortedTable', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + }, + }, + { + //📊删除表中内容 + opcode: 'deleteNameOfSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.deleteNameOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + }, + }, ], menus: { + tableItemPropertyMenu: [ + { + text: this.formatMessage('ArkosExt.name'), + value: '1' + }, + { + text: this.formatMessage('ArkosExt.rank'), + value: '2' + }, + { + text: this.formatMessage('ArkosExt.rankValue'), + value: '3' + }, + { + text: this.formatMessage('ArkosExt.extra'), + value: '4' + }, + ], + sortOrder: [ + { + text: this.formatMessage('ArkosExt.asc'), + value: 'asc'//升序 + }, + { + text: this.formatMessage('ArkosExt.desc'), + value: 'desc'//降序 + }, + ], + sortedTableMenu: { + items: 'findAllSortedTable' + }, //判断符菜单 opMenu1: ['<','≤','=','≠',], opMenu2: ['<','>','≤','≥','=','≠',], @@ -593,8 +780,8 @@ class ArkosExtensions { } //角色是否可见 - ifVisible (args, util) { - return util.target.visible; + isHiding (args, util) { + return !util.target.visible; } //获取图层(逝一逝) @@ -725,14 +912,155 @@ class ArkosExtensions { compareTwoSidesPlus(args){ switch(args.logic){ case 'or': - return this.compare(args.a, args.b, args.op1)||this.compare(args.b, args.c, args.op2) + return this.compare(args.a, args.b, args.op1)||this.compare(args.a, args.c, args.op2) case 'and': - return this.compare(args.a, args.b, args.op1)&&this.compare(args.b, args.c, args.op2) + return this.compare(args.a, args.b, args.op1)&&this.compare(args.a, args.c, args.op2) default: return false; } } + + //数组排序规则 + sortRule(propName,order) { + return (a, b) => { + a = a[propName] + b = b[propName] + if (a > b) return order === 'asc' ? 1 : -1; + else if (a < b) return order === 'asc' ? -1 : 1; + else return 0; + } + } + + //查找所有排序表 + findAllSortedTable() { + const list = []; + let temp = this.sortedTable; + Object.keys(temp).forEach(obj => { + //if ( Array.isArray (temp[obj]) ) { + list.push({ + text: obj, + value: obj, + }); + //} + }); + if (list.length === 0) { + list.push({ + text: '-', + value: 'empty', + }); + } + //list.sort(this.sortRule("text")); + return list; + } + createTableIfNotExist(list){ + if(!(list in this.sortedTable)) + this.sortedTable[list]=[]; + } + + sortTable(list){ + this.sortedTable[list].list.sort(this.sortRule("rankValue",this.sortedTable[list].order)); + } + + //📊清空排序表 + clearSortedTable (args){ + this.sortedTable[args.list]=[]; + } + + //📊设置排序方式 + setTypeOfSortedTable (args){ + this.createTableIfNotExist(args.list) + this.sortedTable[args.list].order=args.type; + this.sortTable(args.list) + } + + //查找在列表中的插入位置(已有则覆盖) + _findPlaceAndInsert(list, order, item){ + //删除已存在的内容 + for (let i = 0; i < list.length; i++) { + if (list[i].name === item.name){ + //删除同名项 + list.splice(i,1); + break; + } + } + //查找插入位置并插入 + for (let i = 0; i < list.length; i++) { + if ((list[i].rankValue > item.rankValue && order ==='asc')|| + (list[i].rankValue < item.rankValue && order ==='desc')){ + //插入在该项前 + list.splice(i,0,item); + return; + } + } + //没找到插入位置,加在末尾 + list.push(item); + } + + //📊将内容加入表 + addToSortedTable (args){ + this.createTableIfNotExist(args.list) + this._findPlaceAndInsert( + this.sortedTable[args.list].list, + this.sortedTable[args.list].order, + {name:args.name, rankValue:args.value, extra:args.extra}); + } + + _getTInItem (item, t, rank){ + if(item === undefined) return ''; + switch(t){ + case '1': + return item.name; + case '2': + return rank; + case '3': + return item.rankValue; + case '4': + return item.extra; + default: + return ''; + } + } + + //📊获取第n项 + getFromSortedTableByNo (args){ + if(!(args.list in this.sortedTable)) return ''; + let list = this.sortedTable[args.list].list; + return this._getTInItem (list[args.n-1], args.t, args.n); + } + + _getItemIdxByName(list, name){ + for (let i = 0; i < list.length; i++) { + if (list[i].name === name){ + return i; + } + } + return -1; + } + + //📊获取名为XX的项 + getFromSortedTableByName (args){ + if(!(args.list in this.sortedTable)) return ''; + let list = this.sortedTable[args.list].list; + let n = this._getItemIdxByName(list ,args.name) ; + if(n === -1) return ''; + return this._getTInItem (list[n], args.t, n+1); + } + + //📊获取排序表长度 + lengthOfSortedTable (args){ + if(!(args.list in this.sortedTable)) return 0; + return this.sortedTable[args.list].list.length; + } + + //📊删除排序表名为XX的内容 + deleteNameOfSortedTable (args){ + if(!(args.list in this.sortedTable)) return; + let list = this.sortedTable[args.list].list; + let n = this._getItemIdxByName(list ,args.name) ; + if(n === -1) return; + list.splice(n, 1); + } } From f0cf58d7cc10f6673f5174b57cfd4f6542b71816 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 00:51:07 +0800 Subject: [PATCH 030/357] =?UTF-8?q?=E7=AE=80=E5=86=99=E4=BB=A3=E7=A0=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 3fbdaeed..51dddd38 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -937,10 +937,7 @@ class ArkosExtensions { let temp = this.sortedTable; Object.keys(temp).forEach(obj => { //if ( Array.isArray (temp[obj]) ) { - list.push({ - text: obj, - value: obj, - }); + list.push(obj); //} }); if (list.length === 0) { From 556a50be492e7793acfb7d290130b91e8d2b4b3b Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 01:04:14 +0800 Subject: [PATCH 031/357] =?UTF-8?q?=E8=AE=BE=E7=BD=AE=E6=8E=92=E5=BA=8F?= =?UTF-8?q?=E8=A1=A8=E8=8F=9C=E5=8D=95=E5=8F=AF=E5=A1=9E=E5=85=A5=E7=A7=AF?= =?UTF-8?q?=E6=9C=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 51dddd38..49109480 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -462,6 +462,7 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', + acceptReporters: true, }, }, }, @@ -474,6 +475,7 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', + acceptReporters: true, }, type: { type: 'string', @@ -490,6 +492,7 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', + acceptReporters: true, }, name: { type: 'string', @@ -514,6 +517,7 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', + acceptReporters: true, }, n: { type: 'number', @@ -535,6 +539,7 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', + acceptReporters: true, }, name: { type: 'string', @@ -556,6 +561,7 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', + acceptReporters: true, }, }, }, @@ -568,6 +574,7 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', + acceptReporters: true, }, name: { type: 'string', @@ -952,7 +959,7 @@ class ArkosExtensions { createTableIfNotExist(list){ if(!(list in this.sortedTable)) - this.sortedTable[list]=[]; + this.sortedTable[list]={order:'desc',list:[]}; } sortTable(list){ @@ -961,7 +968,8 @@ class ArkosExtensions { //📊清空排序表 clearSortedTable (args){ - this.sortedTable[args.list]=[]; + this.createTableIfNotExist(args.list) + this.sortedTable[args.list].list=[]; } //📊设置排序方式 From 35067c0f35969af42af7dcdfd1ae4ec8be406b4e Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 01:11:30 +0800 Subject: [PATCH 032/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E6=97=A0=E6=B3=95?= =?UTF-8?q?=E5=A1=9E=E5=8F=82=E6=95=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 49109480..d64c41c8 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -462,7 +462,6 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', - acceptReporters: true, }, }, }, @@ -475,7 +474,6 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', - acceptReporters: true, }, type: { type: 'string', @@ -492,7 +490,6 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', - acceptReporters: true, }, name: { type: 'string', @@ -517,7 +514,6 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', - acceptReporters: true, }, n: { type: 'number', @@ -539,7 +535,6 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', - acceptReporters: true, }, name: { type: 'string', @@ -561,7 +556,6 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', - acceptReporters: true, }, }, }, @@ -574,7 +568,6 @@ class ArkosExtensions { list: { type: 'string', menu: 'sortedTableMenu', - acceptReporters: true, }, name: { type: 'string', @@ -613,7 +606,8 @@ class ArkosExtensions { }, ], sortedTableMenu: { - items: 'findAllSortedTable' + items: 'findAllSortedTable', + acceptReporters: true, }, //判断符菜单 opMenu1: ['<','≤','=','≠',], From abd9f4e9bfc5307460935e85457374dc026bf05d Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Mon, 18 Jul 2022 17:48:19 +0800 Subject: [PATCH 033/357] =?UTF-8?q?=E5=A2=9E=E5=8A=A0=E6=97=A5=E6=9C=9F?= =?UTF-8?q?=E6=97=B6=E9=97=B4=E8=AE=A1=E7=AE=97=E6=8B=93=E5=B1=95=E4=B8=80?= =?UTF-8?q?=E6=AD=A5=E5=88=B0=E4=BD=8D?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- six-6/DateTime.svg | 1 + six-6/DateTimePost.svg | 1 + six-6/datetime.js | 447 +++++++++++++++++++++++++++++++++++++++++ 3 files changed, 449 insertions(+) create mode 100644 six-6/DateTime.svg create mode 100644 six-6/DateTimePost.svg create mode 100644 six-6/datetime.js diff --git a/six-6/DateTime.svg b/six-6/DateTime.svg new file mode 100644 index 00000000..2e842290 --- /dev/null +++ b/six-6/DateTime.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/six-6/DateTimePost.svg b/six-6/DateTimePost.svg new file mode 100644 index 00000000..07b86243 --- /dev/null +++ b/six-6/DateTimePost.svg @@ -0,0 +1 @@ +2h18星期一182022七月 \ No newline at end of file diff --git a/six-6/datetime.js b/six-6/datetime.js new file mode 100644 index 00000000..f043f9de --- /dev/null +++ b/six-6/datetime.js @@ -0,0 +1,447 @@ +// import Cast from "cast.js"; + +let _picture = "" + + "byIgc3Ryb2tlPSJub25lIiBzdHJva2Utd2lkdGg9IjEiIHN0cm9rZS1saW5lY2FwPSJidXR0IiBzdHJva2UtbGluZWpvaW49Im1pdGVyIiBzdHJva2UtbWl0ZXJsaW1pdD0iMTAiIHN0cm9rZS1kYXNoYXJyYXk9IiIgc3Ryb2tlLWRhc2hvZmZzZXQ9IjAiIGZvbnQtZmFtaWx5PSJTYW5zIFNlcmlmIiBmb250LXdlaWdodD0ibm9ybWFsIiB0ZXh0LWFuY2hvcj0ic3RhcnQiIHN0eWxlPSJtaXgtYmxlbmQtbW9kZTogbm9ybWFsIj48dHNwYW4geD0iMCIgZHk9IjAiPjJoPC90c3Bhbj48L3RleHQ+PHBhdGggZD0iTTE4MS4wMDAwMSwxODYuOTM5NnYtMThoMjR2MTh6IiBmaWxsPSIjZmZmZmZmIiBzdHJva2U9Im5vbmUiIHN0cm9rZS13aWR0aD0iMCIgc3Ryb2tlLWxpbmVjYXA9ImJ1dHQiLz48dGV4dCB0cmFuc2Zvcm09InRyYW5zbGF0ZSgxODIuMDAwMDEsMTg0LjkzOTYpIHNjYWxlKDAuNSwwLjUpIiBmb250LXNpemU9IjQwIiB4bWw6c3BhY2U9InByZXNlcnZlIiBmaWxsPSIjNDhiMzY4IiBmaWxsLXJ1bGU9Im5vbnplcm8iIHN0cm9rZT0ibm9uZSIgc3Ryb2tlLXdpZHRoPSIxIiBzdHJva2UtbGluZWNhcD0iYnV0dCIgc3Ryb2tlLWxpbmVqb2luPSJtaXRlciIgc3Ryb2tlLW1pdGVybGltaXQ9IjEwIiBzdHJva2UtZGFzaGFycmF5PSIiIHN0cm9rZS1kYXNob2Zmc2V0PSIwIiBmb250LWZhbWlseT0iU2FucyBTZXJpZiIgZm9udC13ZWlnaHQ9Im5vcm1hbCIgdGV4dC1hbmNob3I9InN0YXJ0IiBzdHlsZT0ibWl4LWJsZW5kLW1vZGU6IG5vcm1hbCI+PHRzcGFuIHg9IjAiIGR5PSIwIj4xODwvdHNwYW4+PC90ZXh0PjwvZz48cGF0aCBkPSJNMjc0LjAwMDAxLDI3OC42MDYyN3YtMTg0aDE4NXYxODR6IiBmaWxsPSIjNDhiMzY4IiBzdHJva2U9IiM0OGIzNjgiIHN0cm9rZS13aWR0aD0iMTUiIHN0cm9rZS1saW5lY2FwPSJidXR0Ii8+PHBhdGggZD0iTTI2NS4wMDAwMSwyNjkuNjA2Mjd2LTE4NGgxODV2MTg0eiIgZmlsbD0iI2JlZmZkMiIgc3Ryb2tlPSIjNWJlMjg0IiBzdHJva2Utd2lkdGg9IjE1IiBzdHJva2UtbGluZWNhcD0iYnV0dCIvPjx0ZXh0IHRyYW5zZm9ybT0idHJhbnNsYXRlKDMxOC4wMDAwMSwxMjEuNzc5MTcpIHNjYWxlKDAuNjU4MzYsMC42NTgzNikiIGZvbnQtc2l6ZT0iNDAiIHhtbDpzcGFjZT0icHJlc2VydmUiIGZpbGw9IiM0OGIzNjgiIGZpbGwtcnVsZT0ibm9uemVybyIgc3Ryb2tlPSJub25lIiBzdHJva2Utd2lkdGg9IjEiIHN0cm9rZS1saW5lY2FwPSJidXR0IiBzdHJva2UtbGluZWpvaW49Im1pdGVyIiBzdHJva2UtbWl0ZXJsaW1pdD0iMTAiIHN0cm9rZS1kYXNoYXJyYXk9IiIgc3Ryb2tlLWRhc2hvZmZzZXQ9IjAiIGZvbnQtZmFtaWx5PSJTYW5zIFNlcmlmIiBmb250LXdlaWdodD0ibm9ybWFsIiB0ZXh0LWFuY2hvcj0ic3RhcnQiIHN0eWxlPSJtaXgtYmxlbmQtbW9kZTogbm9ybWFsIj48dHNwYW4geD0iMCIgZHk9IjAiPuaYn+acn+S4gDwvdHNwYW4+PC90ZXh0Pjx0ZXh0IHRyYW5zZm9ybT0idHJhbnNsYXRlKDI5Ny4wMDAwMSwyMTEuOTUwNzYpIHNjYWxlKDIuNTMyMzQsMi41MzIzNCkiIGZvbnQtc2l6ZT0iNDAiIHhtbDpzcGFjZT0icHJlc2VydmUiIGZpbGw9IiM0OGIzNjgiIGZpbGwtcnVsZT0ibm9uemVybyIgc3Ryb2tlPSJub25lIiBzdHJva2Utd2lkdGg9IjEiIHN0cm9rZS1saW5lY2FwPSJidXR0IiBzdHJva2UtbGluZWpvaW49Im1pdGVyIiBzdHJva2UtbWl0ZXJsaW1pdD0iMTAiIHN0cm9rZS1kYXNoYXJyYXk9IiIgc3Ryb2tlLWRhc2hvZmZzZXQ9IjAiIGZvbnQtZmFtaWx5PSJTYW5zIFNlcmlmIiBmb250LXdlaWdodD0ibm9ybWFsIiB0ZXh0LWFuY2hvcj0ic3RhcnQiIHN0eWxlPSJtaXgtYmxlbmQtbW9kZTogbm9ybWFsIj48dHNwYW4geD0iMCIgZHk9IjAiPjE4PC90c3Bhbj48L3RleHQ+PHRleHQgdHJhbnNmb3JtPSJ0cmFuc2xhdGUoMjc2LjAwMDAxLDI1NC4xMDYyNykgc2NhbGUoMC42MzAwMSwwLjYzMDAxKSIgZm9udC1zaXplPSI0MCIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSIgZmlsbD0iIzQ4YjM2OCIgZmlsbC1ydWxlPSJub256ZXJvIiBzdHJva2U9Im5vbmUiIHN0cm9rZS13aWR0aD0iMSIgc3Ryb2tlLWxpbmVjYXA9ImJ1dHQiIHN0cm9rZS1saW5lam9pbj0ibWl0ZXIiIHN0cm9rZS1taXRlcmxpbWl0PSIxMCIgc3Ryb2tlLWRhc2hhcnJheT0iIiBzdHJva2UtZGFzaG9mZnNldD0iMCIgZm9udC1mYW1pbHk9IlNhbnMgU2VyaWYiIGZvbnQtd2VpZ2h0PSJub3JtYWwiIHRleHQtYW5jaG9yPSJzdGFydCIgc3R5bGU9Im1p" + + "eC1ibGVuZC1tb2RlOiBub3JtYWwiPjx0c3BhbiB4PSIwIiBkeT0iMCI+MjAyMjwvdHNwYW4+PC90ZXh0Pjx0ZXh0IHRyYW5zZm9ybT0idHJhbnNsYXRlKDM5Mi4zMzI2MiwyNTQuMDU4Mykgc2NhbGUoMC41NzA4NCwwLjU3MDg0KSIgZm9udC1zaXplPSI0MCIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSIgZmlsbD0iIzQ4YjM2OCIgZmlsbC1ydWxlPSJub256ZXJvIiBzdHJva2U9Im5vbmUiIHN0cm9rZS13aWR0aD0iMSIgc3Ryb2tlLWxpbmVjYXA9ImJ1dHQiIHN0cm9rZS1saW5lam9pbj0ibWl0ZXIiIHN0cm9rZS1taXRlcmxpbWl0PSIxMCIgc3Ryb2tlLWRhc2hhcnJheT0iIiBzdHJva2UtZGFzaG9mZnNldD0iMCIgZm9udC1mYW1pbHk9IlNhbnMgU2VyaWYiIGZvbnQtd2VpZ2h0PSJub3JtYWwiIHRleHQtYW5jaG9yPSJzdGFydCIgc3R5bGU9Im1peC1ibGVuZC1tb2RlOiBub3JtYWwiPjx0c3BhbiB4PSIwIiBkeT0iMCI+5LiD5pyIPC90c3Bhbj48L3RleHQ+PC9nPjwvZz48L3N2Zz4="; + +let _icon = ""; + +class DateTime { + constructor(runtime) { + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "DateTime.name": "时间计算", + "DateTime.date": "[YEAR] 年 [MONTH] 月 [DATE] 日", + "DateTime.time": "[YEAR] 年 [MONTH] 月 [DATE] 日 [HOUR] 时 [MINUTE] 分 [SECOND] 秒", + "DateTime.now": "现在时刻", + "DateTime.today": "今天", + "DateTime.after2000": " 2000年过 [DAYS] 天的时刻", + "DateTime.diff2000": "[TIME] 距2000年的天数", + "DateTime.diff": "[TIME1] 距 [TIME2] 的 [UNIT] 数", + "DateTime.add": "[TIME1] 过 [TIME2] [UNIT] 的时刻", + "DateTime.get": "[TIME] 的 [UNIT]", + "DateTime.get.year": "年", + "DateTime.get.month": "月", + "DateTime.get.date": "日", + "DateTime.get.day": "星期", + "DateTime.get.days": "天", + "DateTime.get.hour": "小时", + "DateTime.get.minute": "分钟", + "DateTime.get.second": "秒", + "DateTime.get.millisecond": "毫秒", + "DateTime.timezone": "当前时区", + "DateTime.timezonemin": "当前时区分钟数", + }, + en: { + "DateTime.name": "Date/Time", + "DateTime.date": "[MONTH]/[DATE], [YEAR]", + "DateTime.time": "[MONTH]/[DATE],[YEAR] [HOUR]:[MINUTE]:[SECOND]", + "DateTime.now": "now", + "DateTime.today": "today", + "DateTime.after2000": "[DAYS] days after 2000", + "DateTime.diff2000": "days since 2000 of [TIME]", + "DateTime.diff": "difference between [TIME1] and [TIME2] in [UNIT]", + "DateTime.add": "[TIME2] [UNIT] after [TIME1]", + "DateTime.get": "[UNIT] of [TIME]", + "DateTime.get.year": "year", + "DateTime.get.month": "month", + "DateTime.get.date": "date", + "DateTime.get.day": "day", + "DateTime.get.days": "days", + "DateTime.get.hour": "hours", + "DateTime.get.minute": "minutes", + "DateTime.get.second": "seconds", + "DateTime.get.millisecond": "milliseconds", + "DateTime.timezone": "timezone", + "DateTime.timezonemin": "timezone offset in minutes", + } + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: "DateTime", // 拓展id + name: this.formatMessage("DateTime.name"), // 拓展名 + // docsURI: "https://www.baidu.com", + blockIconURI: _icon, + menuIconURI: _icon, + color1: "#40ff80", + color2: "#00CC80", + blocks: [{ + opcode: "date", + blockType: "reporter", + text: this.formatMessage("DateTime.date"), + arguments: { + YEAR: { + type: "number", + defaultValue: new Date().getFullYear() + }, + MONTH: { + type: "number", + defaultValue: new Date().getMonth() + 1 + }, + DATE: { + type: "number", + defaultValue: new Date().getDate() + } + } + }, { + opcode: "time", + blockType: "reporter", + text: this.formatMessage("DateTime.time"), + arguments: { + YEAR: { + type: "number", + defaultValue: new Date().getFullYear() + }, + MONTH: { + type: "number", + defaultValue: new Date().getMonth() + 1 + }, + DATE: { + type: "number", + defaultValue: new Date().getDate() + }, + HOUR: { + type: "number", + defaultValue: new Date().getHours() + }, + MINUTE: { + type: "number", + defaultValue: new Date().getMinutes() + }, + SECOND: { + type: "number", + defaultValue: new Date().getSeconds() + } + } + }, { + opcode: "now", + blockType: "reporter", + text: this.formatMessage("DateTime.now"), + arguments: {} + }, { + opcode: "today", + blockType: "reporter", + text: this.formatMessage("DateTime.today"), + arguments: {} + }, { + opcode: "after2000", + blockType: "reporter", + text: this.formatMessage("DateTime.after2000"), + arguments: { + DAYS: { + type: "number", + defaultValue: this.diff({ + TIME1: "2000-01-01T00:00:00.000Z", + TIME2: this.now(), + UNIT: "days" + }) + } + } + }, { + opcode: "diff2000", + blockType: "reporter", + text: this.formatMessage("DateTime.diff2000"), + arguments: { + TIME: { + type: "string", + defaultValue: this.now() + } + } + }, { + opcode: "diff", + blockType: "reporter", + text: this.formatMessage("DateTime.diff"), + arguments: { + TIME1: { + type: "string", + defaultValue: "2000-01-23T12:34:56" + }, + TIME2: { + type: "string", + defaultValue: this.now() + }, + UNIT: { + type: "string", + menu: "DiffUnits", + defaultValue: "days" + } + } + }, { + opcode: "add", + blockType: "reporter", + text: this.formatMessage("DateTime.add"), + arguments: { + TIME1: { + type: "string", + defaultValue: "2000-01-23T12:34:56" + }, + TIME2: { + type: "number", + defaultValue: 1234 + }, + UNIT: { + type: "string", + menu: "DiffUnits", + defaultValue: "days" + } + } + }, { + opcode: "get", + blockType: "reporter", + text: this.formatMessage("DateTime.get"), + arguments: { + TIME: { + type: "string", + defaultValue: this.now() + }, + UNIT: { + type: "string", + menu: "Units", + defaultValue: "date" + } + } + }, { + opcode: "timezone", + blockType: "reporter", + text: this.formatMessage("DateTime.timezone"), + arguments: {} + }, { + opcode: "timezonemin", + blockType: "reporter", + text: this.formatMessage("DateTime.timezonemin"), + arguments: {} + }], + menus: { + DiffUnits: { + items: [{ + text: this.formatMessage("DateTime.get.days"), + value: "days" + }, + { + text: this.formatMessage("DateTime.get.hour"), + value: "hour" + }, + { + text: this.formatMessage("DateTime.get.minute"), + value: "minute" + }, + { + text: this.formatMessage("DateTime.get.second"), + value: "second" + }, + { + text: this.formatMessage("DateTime.get.millisecond"), + value: "millisecond" + } + ] + }, + Units: { + items: [{ + text: this.formatMessage("DateTime.get.year"), + value: "year" + }, + { + text: this.formatMessage("DateTime.get.month"), + value: "month" + }, + { + text: this.formatMessage("DateTime.get.date"), + value: "date" + }, + { + text: this.formatMessage("DateTime.get.day"), + value: "day" + }, + { + text: this.formatMessage("DateTime.get.hour"), + value: "hour" + }, + { + text: this.formatMessage("DateTime.get.minute"), + value: "minute" + }, + { + text: this.formatMessage("DateTime.get.second"), + value: "second" + }, + { + text: this.formatMessage("DateTime.get.millisecond"), + value: "millisecond" + } + ] + } + } + }; + } + + _padZero(str, n) { + let num = Number(str); + if (num === NaN || num < 0) throw new Error("`" + str + "' is not a vaild number"); + return String(Math.floor(num)).padStart(n, 0); + } + + _catchret(func) { + try { + return func(); + } catch (e) { + return e; + } + } + + date(args) { + return this._catchret(() => { + return this._localDate(new Date(`${this._padZero(args.YEAR, 4)}-${this._padZero(args.MONTH, 2)}-${this._padZero(args.DATE, 2)}T00:00:00.000`)).split("T")[0]; + }); + } + + time(args) { + return this._catchret(() => { + return this._localDate(new Date(`${this._padZero(args.YEAR, 4)}-${this._padZero(args.MONTH, 2)}-${this._padZero(args.DATE, 2)}T${this._padZero(args.HOUR, 2)}:${this._padZero(args.MINUTE, 2)}:${this._padZero(args.SECOND, 2)}.000`)); + }); + } + + _localDate(date) { + let timeoffs = date.getTimezoneOffset(); + return new Date(date.getTime() - timeoffs * 1000 * 60) + .toISOString().slice(0, -1); + } + + now(args) { + return this._localDate(new Date()); + } + + today(args) { + return this.now({}).split("T")[0]; + } + + after2000(args) { + return this.add({ + TIME1: "2000-01-01T00:00:00Z", + TIME2: args.DAYS, + UNIT: "days" + }); + } + + diff2000(args) { + return this.diff({ + TIME1: "2000-01-01T00:00:00Z", + TIME2: args.TIME, + UNIT: "days" + }); + } + + diff(args) { + return this._catchret(() => { + let diff = new Date(args.TIME2).getTime() - new Date(args.TIME1).getTime(); + switch (args.UNIT) { + case "days": + return diff / (24 * 60 * 60 * 1000); + case "hour": + return diff / (60 * 60 * 1000); + case "minute": + return diff / (60 * 1000); + case "second": + return diff / (1000); + case "millisecond": + return diff; + } + }); + } + + add(args) { + return this._catchret(() => { + let orig = new Date(args.TIME1).getTime(); + let diff = Number(args.TIME2); + switch (args.UNIT) { + case "days": + diff *= 24 * 60 * 60 * 1000; + break; + case "hour": + diff *= 60 * 60 * 1000; + break; + case "minute": + diff *= 60 * 1000; + break; + case "second": + diff *= 1000; + break; + case "millisecond": + // diff *= 1; + break; + } + return this._localDate(new Date(orig + diff)); + }); + } + + get(args) { + return this._catchret(() => { + let orig = new Date(args.TIME); + switch (args.UNIT) { + case "year": + return orig.getFullYear(); + case "month": + return orig.getMonth() + 1; + case "day": + return orig.getDay() + 1; + case "date": + return orig.getDate(); + case "hour": + return orig.getHours(); + case "minute": + return orig.getMinutes(); + case "second": + return orig.getSeconds(); + case "millisecond": + return orig.getMilliseconds(); + } + }); + } + + timezone(args) { + let toff = new Date().getTimezoneOffset(); + return (toff > 0 ? "-" : "+") + + this._padZero(Math.abs(toff) / 60, 2) + + this._padZero(Math.abs(toff) % 60, 2); + } + + timezonemin(args) { + let toff = new Date().getTimezoneOffset(); + return -toff; + } +} + +window.tempExt = { + Extension: DateTime, + info: { + name: "DateTime.name", + description: "DateTime.descp", + extensionId: "DateTime", + iconURL: _picture, + insetIconURL: _icon, + featured: true, + disabled: false, + collaborator: "six-6 x CCW" + }, + l10n: { + "zh-cn": { + "DateTime.name": "时间计算", + "DateTime.descp": "计算日期与时间" + }, + en: { + "DateTime.name": "Date and time calculator", + "DateTime.descp": "Calculate date and time" + } + } +}; + +/* vim: set expandtab tabstop=2 shiftwidth=2: */ From 7710d607167e84c0af2e950ad751a6a4e500ad79 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Tue, 19 Jul 2022 14:26:25 +0800 Subject: [PATCH 034/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E8=8E=B7=E5=8F=96=20?= =?UTF-8?q?HEX=20=E5=80=BC=E7=A7=AF=E6=9C=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/Arkos/project.js b/Arkos/project.js index d64c41c8..f3573a87 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -1,4 +1,5 @@ import Cast from '../utils/cast.js' +import Cast from '../utils/color.js' // import cover from './assets/icon.svg' // import icon from './assets/icon.svg' @@ -64,6 +65,8 @@ class ArkosExtensions { 'ArkosExt.rank': '表中位置', 'ArkosExt.rankValue': '排序值', 'ArkosExt.extra': '附加信息', + + 'ArkosExt.colorToHex': '颜色[COLOR]的代码', }, en: { @@ -116,6 +119,8 @@ class ArkosExtensions { 'ArkosExt.rank': 'rank', 'ArkosExt.rankValue': 'rankValue', 'ArkosExt.extra': 'extra', + + 'ArkosExt.colorToHex': 'get code of color[COLOR]', }, }) } @@ -575,6 +580,17 @@ class ArkosExtensions { }, }, }, + { + //获取颜色HEX + opcode: 'colorToHex', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.colorToHex'), + arguments: { + COLOR: { + type: 'color', + }, + }, + }, ], menus: { tableItemPropertyMenu: [ @@ -1061,6 +1077,11 @@ class ArkosExtensions { list.splice(n, 1); } + //获取颜色HEX码 + colorToHex (args, util) { + return Color.rgbToHex(Cast.toRgbColorList(args.COLOR)); +} + } From 94c2d29c5b9d897d5371eaf7b98289637ebe4316 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Tue, 19 Jul 2022 15:11:21 +0800 Subject: [PATCH 035/357] =?UTF-8?q?=E8=B0=83=E6=95=B4=E7=A7=AF=E6=9C=A8?= =?UTF-8?q?=E9=A1=BA=E5=BA=8F=EF=BC=8C=E5=8A=A0=E6=A0=8F=E7=9B=AE=E6=A0=87?= =?UTF-8?q?=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 277 +++++++++++++++++++++++++---------------------- 1 file changed, 148 insertions(+), 129 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index f3573a87..b2e5fb24 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -1,5 +1,5 @@ import Cast from '../utils/cast.js' -import Cast from '../utils/color.js' +import Color from '../utils/color.js' // import cover from './assets/icon.svg' // import icon from './assets/icon.svg' @@ -15,6 +15,11 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', + 'ArkosExt.info1': '🏃‍♂️ 坐标&角度', + 'ArkosExt.info2': '🔠 字符串处理', + 'ArkosExt.info3': '🛠 实用积木', + 'ArkosExt.info4': '📄 数据获取', + 'ArkosExt.info5': '📊 排序表', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', @@ -67,6 +72,8 @@ class ArkosExtensions { 'ArkosExt.extra': '附加信息', 'ArkosExt.colorToHex': '颜色[COLOR]的代码', + + }, en: { @@ -121,6 +128,12 @@ class ArkosExtensions { 'ArkosExt.extra': 'extra', 'ArkosExt.colorToHex': 'get code of color[COLOR]', + + 'ArkosExt.info1': '🏃‍♂️ Coodinate & Direction', + 'ArkosExt.info2': '🔠 String Processing', + 'ArkosExt.info3': '🛠 Utilities', + 'ArkosExt.info4': '📄 Information', + 'ArkosExt.info5': '📊 Sorted Table', }, }) } @@ -136,29 +149,15 @@ class ArkosExtensions { getInfo() { return { id: 'hcnTest', // 拓展id - name: this.formatMessage('ArkosExt.extensionName'), // 拓展名 + name: this.formatMessage('ArkosExt.extensionName'), + color1: '#FF8383', // menuIconURI: icon, // blockIconURI: icon, blocks: [ + "---" + this.formatMessage("ArkosExt.info1"), // 🏃‍♂️坐标&角度 + // 计算点A到点B的方向 { - // 判断相等(区分大小写) - opcode: 'strictlyEquals', - blockType: 'Boolean', - text: this.formatMessage('ArkosExt.stringEquality'), - arguments: { - ONE: { - type: 'string', - defaultValue: 'A', - }, - TWO: { - type: 'string', - defaultValue: 'a', - }, - }, - }, - { - // 计算点A到点B的方向 opcode: 'getDirFromAToB', blockType: 'reporter', text: this.formatMessage('ArkosExt.directionFromAtoB'), @@ -181,8 +180,8 @@ class ArkosExtensions { }, }, }, + // 计算角b-角a的角度差 { - // 计算角b-角a的角度差 opcode: 'differenceBetweenDirections', blockType: 'reporter', text: this.formatMessage('ArkosExt.differenceBetweenDirections'), @@ -197,8 +196,8 @@ class ArkosExtensions { }, }, }, + // 两点距离 { - // 两点距离 opcode: 'disFromAToB', blockType: 'reporter', text: this.formatMessage('ArkosExt.distance'), @@ -221,8 +220,25 @@ class ArkosExtensions { }, }, }, + //朝..方向旋转..角度 + { + opcode: 'turnDegreesToDir', + blockType: 'command', + text: this.formatMessage('ArkosExt.turnDegreesToDir'), + arguments: { + degree: { + type: 'number', + defaultValue: 45, + }, + dir: { + type: 'angle', + defaultValue: 10, + }, + }, + }, + "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 + // 查找子字符串,从pos开始 { - // 查找子字符串,从pos开始 opcode: 'indexof', blockType: 'reporter', text: this.formatMessage('ArkosExt.searchString'), @@ -241,8 +257,8 @@ class ArkosExtensions { }, }, }, + // 在字符串中插入子字符串 { - // 在字符串中插入子字符串 opcode: 'insertStr', blockType: 'reporter', text: this.formatMessage('ArkosExt.insertString'), @@ -261,8 +277,8 @@ class ArkosExtensions { }, }, }, + // 替换字符串中的从..到..的字符串 { - // 替换字符串中的从..到..的字符串 opcode: 'replaceStr', blockType: 'reporter', text: this.formatMessage('ArkosExt.replaceString'), @@ -285,106 +301,25 @@ class ArkosExtensions { }, }, }, + "---" + this.formatMessage("ArkosExt.info3"), //🔧实用积木 + // 判断相等(区分大小写) { - //朝..方向旋转..角度 - opcode: 'turnDegreesToDir', - blockType: 'command', - text: this.formatMessage('ArkosExt.turnDegreesToDir'), - arguments: { - degree: { - type: 'number', - defaultValue: 45, - }, - dir: { - type: 'angle', - defaultValue: 10, - }, - }, - }, - { - //获取特效值 - opcode: 'getEffect', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getEffect'), - arguments: { - EFFECT: { - type: 'string', - menu: 'effectMenu', - }, - }, - }, - { - //是否隐藏 - opcode: 'isHiding', + opcode: 'strictlyEquals', blockType: 'Boolean', - text: this.formatMessage('ArkosExt.isHiding'), - }, - { - //获取旋转方式 - opcode: 'getRotationStyle', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getRotationStyle'), - }, - { - //获取造型0宽1高 - opcode: 'getWidthOrHeight', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getWidthOrHeight'), + text: this.formatMessage('ArkosExt.stringEquality'), arguments: { - t: { + ONE: { type: 'string', - menu: 'WOrH', - }, - }, - }, - { - //强行设置大小 - opcode: 'setSize', - blockType: 'command', - text: this.formatMessage('ArkosExt.setSize'), - arguments: { - size: { - type: 'number', - defaultValue: 9999, - }, - }, - }, - { - //强行移到xy - opcode: 'setXY', - blockType: 'command', - text: this.formatMessage('ArkosExt.setXY'), - arguments: { - x: { - type: 'number', - defaultValue: 100000, - }, - y: { - type: 'number', - defaultValue: 100000, + defaultValue: 'A', }, - }, - }, - { - //获取角色边缘xy - opcode: 'getBoundaryCoord', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getBoundaryCoord'), - arguments: { - t: { + TWO: { type: 'string', - menu: 'boundaryMenu', - } + defaultValue: 'a', + }, }, }, + //返回值转bool积木 { - //是否跑到舞台外 - opcode: 'isOutOfSight', - blockType: 'Boolean', - text: this.formatMessage('ArkosExt.isOutOfSight'), - }, - { - //返回值转bool积木 opcode: 'reporterToBoolean', blockType: 'Boolean', text: '[t]', @@ -395,8 +330,8 @@ class ArkosExtensions { } }, }, + //形如 a≤b≤c { - //形如 a≤b≤c opcode: 'compareTwoSides', blockType: 'Boolean', text: '[a][op1][b][op2][c]', @@ -423,8 +358,8 @@ class ArkosExtensions { }, }, }, + //形如:a≤b且/或>c op1,op2 logic compareTwoSidesPlus { - //形如:a≤b且/或>c op1,op2 logic compareTwoSidesPlus opcode: 'compareTwoSidesPlus', blockType: 'Boolean', text: '[a][op1][b][logic][op2][c]', @@ -458,8 +393,103 @@ class ArkosExtensions { }, }, }, + //获取颜色HEX + { + opcode: 'colorToHex', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.colorToHex'), + arguments: { + COLOR: { + type: 'color', + }, + }, + }, + //强行设置大小 + { + opcode: 'setSize', + blockType: 'command', + text: this.formatMessage('ArkosExt.setSize'), + arguments: { + size: { + type: 'number', + defaultValue: 9999, + }, + }, + }, + //强行移到xy + { + opcode: 'setXY', + blockType: 'command', + text: this.formatMessage('ArkosExt.setXY'), + arguments: { + x: { + type: 'number', + defaultValue: 100000, + }, + y: { + type: 'number', + defaultValue: 100000, + }, + }, + }, + "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 + //获取特效值 + { + opcode: 'getEffect', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getEffect'), + arguments: { + EFFECT: { + type: 'string', + menu: 'effectMenu', + }, + }, + }, + //是否隐藏 + { + opcode: 'isHiding', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.isHiding'), + }, + //获取旋转方式 + { + opcode: 'getRotationStyle', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getRotationStyle'), + }, + //获取造型0宽1高 + { + opcode: 'getWidthOrHeight', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getWidthOrHeight'), + arguments: { + t: { + type: 'string', + menu: 'WOrH', + }, + }, + }, + //获取角色边缘xy + { + opcode: 'getBoundaryCoord', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getBoundaryCoord'), + arguments: { + t: { + type: 'string', + menu: 'boundaryMenu', + } + }, + }, + //是否跑到舞台外 + { + opcode: 'isOutOfSight', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.isOutOfSight'), + }, + "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 + //📊清空排序表 { - //📊清空排序表 opcode: 'clearSortedTable', blockType: 'command', text: this.formatMessage('ArkosExt.clearSortedTable'), @@ -580,17 +610,6 @@ class ArkosExtensions { }, }, }, - { - //获取颜色HEX - opcode: 'colorToHex', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.colorToHex'), - arguments: { - COLOR: { - type: 'color', - }, - }, - }, ], menus: { tableItemPropertyMenu: [ From 3cb841e91e0c924c4d7f4e0560edd93bd508b936 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Tue, 19 Jul 2022 15:38:08 +0800 Subject: [PATCH 036/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E4=BF=AE=E6=98=BE?= =?UTF-8?q?=E7=A4=BA=E9=94=99=E8=AF=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 21 +++++++++++---------- 1 file changed, 11 insertions(+), 10 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index b2e5fb24..696bd44f 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -394,16 +394,17 @@ class ArkosExtensions { }, }, //获取颜色HEX - { - opcode: 'colorToHex', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.colorToHex'), - arguments: { - COLOR: { - type: 'color', - }, - }, - }, + // { + // opcode: 'colorToHex', + // blockType: 'reporter', + // text: this.formatMessage('ArkosExt.colorToHex'), + // arguments: { + // COLOR: { + // type: 'color', + // defaultValue: '1', + // }, + // }, + // }, //强行设置大小 { opcode: 'setSize', From 1448149ac99b5f7676c657b4404dc189e0c761d6 Mon Sep 17 00:00:00 2001 From: Arkos123 <1726039638@qq.com> Date: Tue, 19 Jul 2022 15:49:28 +0800 Subject: [PATCH 037/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E4=BF=AE=E6=94=B9?= =?UTF-8?q?=E6=98=BE=E7=A4=BAbug=20*=202?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 696bd44f..b04cb11a 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -155,7 +155,7 @@ class ArkosExtensions { // menuIconURI: icon, // blockIconURI: icon, blocks: [ - "---" + this.formatMessage("ArkosExt.info1"), // 🏃‍♂️坐标&角度 + // "---" + this.formatMessage("ArkosExt.info1"), // 🏃‍♂️坐标&角度 // 计算点A到点B的方向 { opcode: 'getDirFromAToB', @@ -236,7 +236,7 @@ class ArkosExtensions { }, }, }, - "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 + // "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 // 查找子字符串,从pos开始 { opcode: 'indexof', @@ -301,7 +301,7 @@ class ArkosExtensions { }, }, }, - "---" + this.formatMessage("ArkosExt.info3"), //🔧实用积木 + // "---" + this.formatMessage("ArkosExt.info3"), //🔧实用积木 // 判断相等(区分大小写) { opcode: 'strictlyEquals', @@ -433,7 +433,7 @@ class ArkosExtensions { }, }, }, - "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 + // "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 //获取特效值 { opcode: 'getEffect', @@ -488,7 +488,7 @@ class ArkosExtensions { blockType: 'Boolean', text: this.formatMessage('ArkosExt.isOutOfSight'), }, - "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 + // "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 //📊清空排序表 { opcode: 'clearSortedTable', From 74f0a41407212e919c1072984ac4c09c9a33d782 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 21:20:05 +0800 Subject: [PATCH 038/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index b04cb11a..462c7859 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -155,7 +155,7 @@ class ArkosExtensions { // menuIconURI: icon, // blockIconURI: icon, blocks: [ - // "---" + this.formatMessage("ArkosExt.info1"), // 🏃‍♂️坐标&角度 + "---" + this.formatMessage("ArkosExt.info1"), // 🏃‍♂️坐标&角度 // 计算点A到点B的方向 { opcode: 'getDirFromAToB', From 7082533bccd87843ef477d57e63ff13fad36311a Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 21:36:31 +0800 Subject: [PATCH 039/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95=20*=202?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index 462c7859..7e9a8e79 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -155,7 +155,7 @@ class ArkosExtensions { // menuIconURI: icon, // blockIconURI: icon, blocks: [ - "---" + this.formatMessage("ArkosExt.info1"), // 🏃‍♂️坐标&角度 + "---" + '反反复复',//this.formatMessage("ArkosExt.info1"), // 🏃‍♂️坐标&角度 // 计算点A到点B的方向 { opcode: 'getDirFromAToB', @@ -235,6 +235,7 @@ class ArkosExtensions { defaultValue: 10, }, }, + filter: [TargetType.SPRITE] }, // "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 // 查找子字符串,从pos开始 @@ -416,6 +417,7 @@ class ArkosExtensions { defaultValue: 9999, }, }, + filter: [TargetType.SPRITE] }, //强行移到xy { @@ -432,6 +434,7 @@ class ArkosExtensions { defaultValue: 100000, }, }, + filter: [TargetType.SPRITE] }, // "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 //获取特效值 @@ -451,12 +454,14 @@ class ArkosExtensions { opcode: 'isHiding', blockType: 'Boolean', text: this.formatMessage('ArkosExt.isHiding'), + filter: [TargetType.SPRITE] }, //获取旋转方式 { opcode: 'getRotationStyle', blockType: 'reporter', text: this.formatMessage('ArkosExt.getRotationStyle'), + filter: [TargetType.SPRITE] }, //获取造型0宽1高 { @@ -481,12 +486,14 @@ class ArkosExtensions { menu: 'boundaryMenu', } }, + filter: [TargetType.SPRITE] }, //是否跑到舞台外 { opcode: 'isOutOfSight', blockType: 'Boolean', text: this.formatMessage('ArkosExt.isOutOfSight'), + filter: [TargetType.SPRITE] }, // "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 //📊清空排序表 From 31c67800b84d9374d7dd0c1ed2c072449e217841 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 21:43:52 +0800 Subject: [PATCH 040/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95*3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 7e9a8e79..b9882d8b 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -235,7 +235,7 @@ class ArkosExtensions { defaultValue: 10, }, }, - filter: [TargetType.SPRITE] + filter: ['sprite'] }, // "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 // 查找子字符串,从pos开始 @@ -417,7 +417,7 @@ class ArkosExtensions { defaultValue: 9999, }, }, - filter: [TargetType.SPRITE] + filter: ['sprite'] }, //强行移到xy { @@ -434,7 +434,7 @@ class ArkosExtensions { defaultValue: 100000, }, }, - filter: [TargetType.SPRITE] + filter: ['sprite'] }, // "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 //获取特效值 @@ -454,14 +454,14 @@ class ArkosExtensions { opcode: 'isHiding', blockType: 'Boolean', text: this.formatMessage('ArkosExt.isHiding'), - filter: [TargetType.SPRITE] + filter: ['sprite'] }, //获取旋转方式 { opcode: 'getRotationStyle', blockType: 'reporter', text: this.formatMessage('ArkosExt.getRotationStyle'), - filter: [TargetType.SPRITE] + filter: ['sprite'] }, //获取造型0宽1高 { @@ -486,14 +486,14 @@ class ArkosExtensions { menu: 'boundaryMenu', } }, - filter: [TargetType.SPRITE] + filter: ['sprite'] }, //是否跑到舞台外 { opcode: 'isOutOfSight', blockType: 'Boolean', text: this.formatMessage('ArkosExt.isOutOfSight'), - filter: [TargetType.SPRITE] + filter: ['sprite'] }, // "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 //📊清空排序表 From a2f268d5de7636713dc243ecd19e1e4d201e2463 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 21:57:53 +0800 Subject: [PATCH 041/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95*5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 36 ++++++++++++++++++------------------ 1 file changed, 18 insertions(+), 18 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index b9882d8b..a3e0c798 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🏃‍♂️ 坐标&角度', + 'ArkosExt.info1': '🏃 坐标&角度', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 数据获取', @@ -129,7 +129,7 @@ class ArkosExtensions { 'ArkosExt.colorToHex': 'get code of color[COLOR]', - 'ArkosExt.info1': '🏃‍♂️ Coodinate & Direction', + 'ArkosExt.info1': '🏃 Coodinate & Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', 'ArkosExt.info4': '📄 Information', @@ -155,7 +155,7 @@ class ArkosExtensions { // menuIconURI: icon, // blockIconURI: icon, blocks: [ - "---" + '反反复复',//this.formatMessage("ArkosExt.info1"), // 🏃‍♂️坐标&角度 + "---" + this.formatMessage("ArkosExt.info1"), // 🏃坐标&角度 // 计算点A到点B的方向 { opcode: 'getDirFromAToB', @@ -237,7 +237,7 @@ class ArkosExtensions { }, filter: ['sprite'] }, - // "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 + "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 // 查找子字符串,从pos开始 { opcode: 'indexof', @@ -302,7 +302,7 @@ class ArkosExtensions { }, }, }, - // "---" + this.formatMessage("ArkosExt.info3"), //🔧实用积木 + "---" + this.formatMessage("ArkosExt.info3"), //🔧实用积木 // 判断相等(区分大小写) { opcode: 'strictlyEquals', @@ -395,17 +395,17 @@ class ArkosExtensions { }, }, //获取颜色HEX - // { - // opcode: 'colorToHex', - // blockType: 'reporter', - // text: this.formatMessage('ArkosExt.colorToHex'), - // arguments: { - // COLOR: { - // type: 'color', - // defaultValue: '1', - // }, - // }, - // }, + { + opcode: 'colorToHex', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.colorToHex'), + arguments: { + COLOR: { + type: 'color', + //defaultValue: '1', + }, + }, + }, //强行设置大小 { opcode: 'setSize', @@ -436,7 +436,7 @@ class ArkosExtensions { }, filter: ['sprite'] }, - // "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 + "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 //获取特效值 { opcode: 'getEffect', @@ -495,7 +495,7 @@ class ArkosExtensions { text: this.formatMessage('ArkosExt.isOutOfSight'), filter: ['sprite'] }, - // "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 + "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 //📊清空排序表 { opcode: 'clearSortedTable', From 342fbc5d92cf6e624ebfde3547ef2b84cdfc35ac Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:01:11 +0800 Subject: [PATCH 042/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95*6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index a3e0c798..318bcb48 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🏃 坐标&角度', + 'ArkosExt.info1': ' 坐标&角度', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 数据获取', From 875502be2de80b517de772d3ffa32badafa48d27 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:08:43 +0800 Subject: [PATCH 043/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95*7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 318bcb48..e57240f1 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': ' 坐标&角度', + 'ArkosExt.info1': '🏃 坐标&角度', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 数据获取', @@ -155,7 +155,7 @@ class ArkosExtensions { // menuIconURI: icon, // blockIconURI: icon, blocks: [ - "---" + this.formatMessage("ArkosExt.info1"), // 🏃坐标&角度 + //"---" + this.formatMessage("ArkosExt.info1"), //🏃 坐标&角度 // 计算点A到点B的方向 { opcode: 'getDirFromAToB', From 7ef6d7aa134d7ee8db8e44182b6297a501f40c0b Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:23:18 +0800 Subject: [PATCH 044/357] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E6=97=A0=E6=B3=95?= =?UTF-8?q?=E8=8E=B7=E5=8F=96=E9=A2=9C=E8=89=B2=E7=9A=84=E9=94=99=E8=AF=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index e57240f1..5944ece7 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🏃 坐标&角度', + 'ArkosExt.info1': '🚶 坐标&角度', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 数据获取', @@ -129,7 +129,7 @@ class ArkosExtensions { 'ArkosExt.colorToHex': 'get code of color[COLOR]', - 'ArkosExt.info1': '🏃 Coodinate & Direction', + 'ArkosExt.info1': '🚶 Coodinate & Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', 'ArkosExt.info4': '📄 Information', @@ -390,7 +390,7 @@ class ArkosExtensions { logic: { type: 'string', menu: 'logicMenu', - defaultValue: 'or', + defaultValue: 'or',// }, }, }, @@ -1106,7 +1106,8 @@ class ArkosExtensions { //获取颜色HEX码 colorToHex (args, util) { - return Color.rgbToHex(Cast.toRgbColorList(args.COLOR)); + let c = Cast.toRgbColorList(args.COLOR) + return Color.rgbToHex({r:c[0],g:c[1],b:c[2]}); } } From 39e536671b81064c0604b952408bf968edc74c9c Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:26:12 +0800 Subject: [PATCH 045/357] =?UTF-8?q?=E6=98=BE=E7=A4=BA=E6=A0=8F=E7=9B=AE?= =?UTF-8?q?=E6=A0=87=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 5944ece7..8ac96164 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -18,7 +18,7 @@ class ArkosExtensions { 'ArkosExt.info1': '🚶 坐标&角度', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', - 'ArkosExt.info4': '📄 数据获取', + 'ArkosExt.info4': '📄 信息获取', 'ArkosExt.info5': '📊 排序表', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', @@ -155,7 +155,7 @@ class ArkosExtensions { // menuIconURI: icon, // blockIconURI: icon, blocks: [ - //"---" + this.formatMessage("ArkosExt.info1"), //🏃 坐标&角度 + "---" + this.formatMessage("ArkosExt.info1"), //🏃 坐标&角度 // 计算点A到点B的方向 { opcode: 'getDirFromAToB', From 443b19afa02bbd7d69e5d4036417ee6ea7891874 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:30:21 +0800 Subject: [PATCH 046/357] =?UTF-8?q?=E6=B5=8B=E8=AF=9510?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 8ac96164..dde8fad4 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🚶 坐标&角度', + 'ArkosExt.info1': '🏄 坐标&方向', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', @@ -129,7 +129,7 @@ class ArkosExtensions { 'ArkosExt.colorToHex': 'get code of color[COLOR]', - 'ArkosExt.info1': '🚶 Coodinate & Direction', + 'ArkosExt.info1': '🏄 Coordinate & Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', 'ArkosExt.info4': '📄 Information', From 39a168cbdca70dbdebb84af34fccb82fd1af5d09 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:31:13 +0800 Subject: [PATCH 047/357] =?UTF-8?q?=E6=B5=8B=E8=AF=959?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index dde8fad4..b0ac65e5 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🏄 坐标&方向', + 'ArkosExt.info1': '🔠 坐标&方向', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', @@ -129,7 +129,7 @@ class ArkosExtensions { 'ArkosExt.colorToHex': 'get code of color[COLOR]', - 'ArkosExt.info1': '🏄 Coordinate & Direction', + 'ArkosExt.info1': '🔠 Coordinate & Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', 'ArkosExt.info4': '📄 Information', From 48eecb48f7a04f74417dfc816fa9e0af20b64adb Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:35:21 +0800 Subject: [PATCH 048/357] =?UTF-8?q?=E6=B5=8B=E8=AF=952520?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index b0ac65e5..a4470d67 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -155,7 +155,7 @@ class ArkosExtensions { // menuIconURI: icon, // blockIconURI: icon, blocks: [ - "---" + this.formatMessage("ArkosExt.info1"), //🏃 坐标&角度 + "---" + this.formatMessage("ArkosExt.info2"), //🏃 坐标&角度 // 计算点A到点B的方向 { opcode: 'getDirFromAToB', From 752b5caac2b0bc488d207bc1b05e260019e464c9 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:43:28 +0800 Subject: [PATCH 049/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index a4470d67..df36fdb9 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🔠 坐标&方向', + 'ArkosExt.info2': '🔠 坐标&方向', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', @@ -129,7 +129,7 @@ class ArkosExtensions { 'ArkosExt.colorToHex': 'get code of color[COLOR]', - 'ArkosExt.info1': '🔠 Coordinate & Direction', + 'ArkosExt.info2': '🔠 Coordinate & Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', 'ArkosExt.info4': '📄 Information', From 125629c2d92cf01e31838e1db3210e5c670bad18 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:46:49 +0800 Subject: [PATCH 050/357] =?UTF-8?q?=E6=B5=8B=E8=AF=952?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index df36fdb9..887a9e16 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -155,7 +155,7 @@ class ArkosExtensions { // menuIconURI: icon, // blockIconURI: icon, blocks: [ - "---" + this.formatMessage("ArkosExt.info2"), //🏃 坐标&角度 + "---" + this.formatMessage("ArkosExt.info1"), //🏃 坐标&角度 // 计算点A到点B的方向 { opcode: 'getDirFromAToB', From 81ff406fe16248da58e25b27d9ec5492a06160f3 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Tue, 19 Jul 2022 22:52:15 +0800 Subject: [PATCH 051/357] =?UTF-8?q?=E6=B5=8B=E8=AF=953?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 887a9e16..a43dcec1 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info2': '🔠 坐标&方向', + 'ArkosExt.info1': '🚶 坐标&方向', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', @@ -129,7 +129,7 @@ class ArkosExtensions { 'ArkosExt.colorToHex': 'get code of color[COLOR]', - 'ArkosExt.info2': '🔠 Coordinate & Direction', + 'ArkosExt.info1': '🚶 Coordinate & Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', 'ArkosExt.info4': '📄 Information', From 05eadb7d83fa8de00890d4cab82e4519a1604ffc Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 19 Jul 2022 22:55:07 +0800 Subject: [PATCH 052/357] =?UTF-8?q?=E6=B5=8B=E8=AF=954?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index a43dcec1..da67f10d 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,11 +15,11 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🚶 坐标&方向', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', 'ArkosExt.info5': '📊 排序表', + 'ArkosExt.info1': '🚶 坐标&方向', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', From cca957f42d63e3efbb308954d4e6053c4013ea8d Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 19 Jul 2022 22:56:28 +0800 Subject: [PATCH 053/357] Update project.js --- Arkos/project.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index da67f10d..65faa523 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,11 +15,11 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', + 'ArkosExt.info1': '🚶 字符串处理', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', 'ArkosExt.info5': '📊 排序表', - 'ArkosExt.info1': '🚶 坐标&方向', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', From 0b378c4e668b24c3bc138b8bfd61df33665db2e0 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 19 Jul 2022 22:58:18 +0800 Subject: [PATCH 054/357] Update project.js --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 65faa523..1f3f089a 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🚶 字符串处理', + 'ArkosExt.info1': '🚶 坐标&方向', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', @@ -128,7 +128,7 @@ class ArkosExtensions { 'ArkosExt.extra': 'extra', 'ArkosExt.colorToHex': 'get code of color[COLOR]', - + 'ArkosExt.info1': '🚶 Coordinate & Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', From 288f8ee9c3bd0e36f093a4814bae84e0294d38f6 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 19 Jul 2022 22:58:57 +0800 Subject: [PATCH 055/357] Update project.js --- Arkos/project.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/project.js b/Arkos/project.js index 1f3f089a..b0d46f93 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🚶 坐标&方向', + 'ArkosExt.info1': '🚶 ', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', From 3d1ba11f7f467d39cf044677c26fade182720975 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 19 Jul 2022 23:00:06 +0800 Subject: [PATCH 056/357] Update project.js --- Arkos/project.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index b0d46f93..30154761 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -15,7 +15,7 @@ class ArkosExtensions { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🚶 ', + 'ArkosExt.info1': '🚶 坐标和方向', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', @@ -129,7 +129,7 @@ class ArkosExtensions { 'ArkosExt.colorToHex': 'get code of color[COLOR]', - 'ArkosExt.info1': '🚶 Coordinate & Direction', + 'ArkosExt.info1': '🚶 Coordinate and Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', 'ArkosExt.info4': '📄 Information', From b397f4929044e892113799b65cc440b79d33d795 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Wed, 20 Jul 2022 00:22:07 +0800 Subject: [PATCH 057/357] =?UTF-8?q?=E9=87=8D=E5=A4=A7=E6=9B=B4=E6=96=B0?= =?UTF-8?q?=EF=BC=9A=E5=AE=B9=E5=99=A8=E5=8F=AF=E9=80=89=E6=8B=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/Archivecode.js | 366 ++++++++++++++++++++++++++----------------- 1 file changed, 222 insertions(+), 144 deletions(-) diff --git a/Arkos/Archivecode.js b/Arkos/Archivecode.js index 33d25928..938a93f9 100644 --- a/Arkos/Archivecode.js +++ b/Arkos/Archivecode.js @@ -14,42 +14,45 @@ let LZString = _LZString(); class Archive_code { constructor(runtime) { this.runtime = runtime - this._archive_code = '' //生成的序列结果 - this.deserializeSuccessfully = false //存反序列化是否成功 - //存需要序列化的内容 + this.convertedSuccessfully = false //JSON转容器是否成功 + //content为一个大容器,内部存很多小容器 this.content = { - 金币: 200, - 背包: ["木头", 233] + "1":{ + 金币: 200, + 背包: ["木头", 233] + }, + "2":{ + Arkos:{score:95}, + yk1boy:{score:100} + } } - //存反序列化结果 - //this.content2 = {}; - //Object.assign(this.content2,this.content) - this.setArchive_code(); - //this.setContent2json(); this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArchiveCodeExt.extensionName': '存档码', - 'ArchiveCodeExt.init': '清空容器', - 'ArchiveCodeExt.serialization': '将内容[value]命名为[name]加入容器(已有则覆盖)', - 'ArchiveCodeExt.serializationForVariable': '将变量[var]内容命名为[name]加入容器(已有则覆盖)', - 'ArchiveCodeExt.serializationForList': '将列表[list]内容命名为[name]加入容器(已有则覆盖)', + 'ArchiveCodeExt.info1': '🏺 容器操作', + 'ArchiveCodeExt.info2': '🔧 JSON处理', + 'ArchiveCodeExt.info3': '💡 附加积木', + 'ArchiveCodeExt.clearContainer': '清空容器[con]', + 'ArchiveCodeExt.addContentToContainer': '将内容[value]命名为[name]加入容器[con](已有则覆盖)', + 'ArchiveCodeExt.addVariableToContainer': '将变量[var]内容命名为[name]加入容器[con](已有则覆盖)', + 'ArchiveCodeExt.addListToContainer': '将列表[list]内容命名为[name]加入容器[con](已有则覆盖)', //'ArchiveCodeExt.stop': '序列化结束', - 'ArchiveCodeExt.result': '容器内容对应字符串', - 'ArchiveCodeExt.deserialization': '读取字符串到容器:[code]', - 'ArchiveCodeExt.getContent': '容器中名称为[key]的内容', + 'ArchiveCodeExt.containerToJSON': '容器[con]内容对应字符串', + 'ArchiveCodeExt.parseJSONToContainer': '读取字符串到容器[con]:[code]', + 'ArchiveCodeExt.getContent': '容器[con]中名称为[key]的内容', //'ArchiveCodeExt.showContent2json':'读取结果', - 'ArchiveCodeExt.saveContentToVar': '将容器中名称为[key]的内容保存到变量[var]', - 'ArchiveCodeExt.saveContentToList': '将容器中名称为[key]的内容保存到列表[list]', - 'ArchiveCodeExt.deserializable': '读取成功?', - 'ArchiveCodeExt.getAmount': '容器中内容的总数', - 'ArchiveCodeExt.ifExist': '容器是否存在名为[key]的内容', - 'ArchiveCodeExt.getContentByNumber': '获取容器中第[index]个内容的[type]', + 'ArchiveCodeExt.saveContentToVar': '将容器[con]中名称为[key]的内容保存到变量[var]', + 'ArchiveCodeExt.saveContentToList': '将容器[con]中名称为[key]的内容保存到列表[list]', + 'ArchiveCodeExt.ifConvertedSuccessfully': '读取成功?', + 'ArchiveCodeExt.getAmount': '容器[con]中内容的总数', + 'ArchiveCodeExt.ifExist': '容器[con]是否存在名为[key]的内容', + 'ArchiveCodeExt.getContentByNumber': '获取容器[con]中第[index]个内容的[type]', 'ArchiveCodeExt.encrypt': '以[method]加密[str],密匙[key]', 'ArchiveCodeExt.decrypt': '以[method]解密[str],密匙[key]', 'ArchiveCodeExt.writeClipboard': '复制[str]到剪贴板', - 'ArchiveCodeExt.getLengthOfList': '容器中名称为[key]的列表的长度', - 'ArchiveCodeExt.getContentOfList': '容器中名称为[key]的列表的第[n]项', + 'ArchiveCodeExt.getLengthOfList': '容器[con]中名称为[key]的列表的长度', + 'ArchiveCodeExt.getContentOfList': '容器[con]中名称为[key]的列表的第[n]项', 'ArchiveCodeExt.getUnicode': '字符[c]的Unicode', 'ArchiveCodeExt.getCharByUnicode': 'Unicode[code]对应字符', 'ArchiveCodeExt.methodMenu.1': 'Arkos加密法', @@ -59,32 +62,31 @@ class Archive_code { 'ArchiveCodeExt.infoMenu.2': '内容', 'ArchiveCodeExt.infoMenu.3': '类型', 'ArchiveCodeExt.infoMenu.4': '列表长度', - 'ArchiveCodeExt.delete': '删除容器中名为[key]的内容', + 'ArchiveCodeExt.delete': '删除容器[con]中名为[key]的内容', 'ArchiveCodeExt.getContentInContainer': '获得容器[container]中名为[key]的内容', }, en: { 'ArchiveCodeExt.extensionName': 'Archive Code', - 'ArchiveCodeExt.init': 'empty Container', - 'ArchiveCodeExt.serialization': 'add content[value] to Container, name as[name]', - 'ArchiveCodeExt.serializationForVariable': 'add variable[var] to Container, name as[name]', - 'ArchiveCodeExt.serializationForList': 'add list[list] to Container, name as[name]', + 'ArchiveCodeExt.clearContainer': 'empty container[con]', + 'ArchiveCodeExt.addContentToContainer': 'add content[value] to container[con], name as[name]', + 'ArchiveCodeExt.addVariableToContainer': 'add variable[var] to container[con], name as[name]', + 'ArchiveCodeExt.addListToContainer': 'add list[list] to container[con], name as[name]', //'ArchiveCodeExt.stop': 'end serialization', - 'ArchiveCodeExt.result': 'Container in string form', - 'ArchiveCodeExt.deserialization': 'parse string[code] to Container', + 'ArchiveCodeExt.containerToJSON': 'container[con] in string form', + 'ArchiveCodeExt.parseJSONToContainer': 'parse string[code] to container[con]', 'ArchiveCodeExt.getContent': 'content of[key]', - 'ArchiveCodeExt.saveContentToVar': 'save [key]to variable[var]', - 'ArchiveCodeExt.saveContentToList': 'save[key]to list[list]', - 'ArchiveCodeExt.deserializable': 'parse successfullly?', - 'ArchiveCodeExt.getAmount': 'the amount of contents in Container', - 'ArchiveCodeExt.ifExist': 'Container contains[key]?', - 'ArchiveCodeExt.getContentByNumber': 'get [type]of #[index]content', + 'ArchiveCodeExt.saveContentToVar': 'save[key]in container[con]to variable[var]', + 'ArchiveCodeExt.saveContentToList': 'save[key]in container[con]to list[list]', + 'ArchiveCodeExt.ifConvertedSuccessfully': 'parse successfullly?', + 'ArchiveCodeExt.getAmount': 'the amount of contents in container[con]', + 'ArchiveCodeExt.ifExist': 'container[con] contains[key]?', + 'ArchiveCodeExt.getContentByNumber': 'get [type]of #[index]content in container[con]', 'ArchiveCodeExt.encrypt': 'use[method]to encrypt[str]with key[key]', 'ArchiveCodeExt.decrypt': 'use[method]to decrypt[str]with key[key]', 'ArchiveCodeExt.writeClipboard': 'copy[str]to clipboard', - //'ArchiveCodeExt.showContent2json':'deserialization result', - 'ArchiveCodeExt.getContentOfList': '#[n] of list[key]', - 'ArchiveCodeExt.getLengthOfList': 'length of list[key]', + 'ArchiveCodeExt.getContentOfList': '#[n] of list[key]in container[con]', + 'ArchiveCodeExt.getLengthOfList': 'length of list[key]in container[con]', 'ArchiveCodeExt.getUnicode': 'get Unicode of[c]', 'ArchiveCodeExt.getCharByUnicode': ' character of Unicode[code]', 'ArchiveCodeExt.methodMenu.1': 'Arkos cipher', @@ -94,28 +96,13 @@ class Archive_code { 'ArchiveCodeExt.infoMenu.2': 'value', 'ArchiveCodeExt.infoMenu.3': 'type', 'ArchiveCodeExt.infoMenu.4': 'lenth of list', - 'ArchiveCodeExt.delete': 'delete content[key] in Container', - 'ArchiveCodeExt.getContentInContainer': 'get [key] in [container]', + 'ArchiveCodeExt.delete': 'delete content[key] in container[con]', + 'ArchiveCodeExt.getContentIncontainer': 'get [key] in [container]', }, }) } - //每次读取archive_code会生成一次 - get archive_code() { - this.setArchive_code(); - return this._archive_code; - } - - //根据content的内容,将其JSON化,存到archive_code - setArchive_code() { - this._archive_code = JSON.stringify(this.content); - } - - // setContent2json(){ - // this.content2json = JSON.stringify(this.content2); - // } - formatMessage(id) { return this._formatMessage({ id, @@ -132,18 +119,31 @@ class Archive_code { // menuIconURI: cover, // blockIconURI: icon, blocks: [ + "---" + this.formatMessage("ArchiveCodeExt.info1"), //🏺容器操作 { - //开始序列化 - opcode: 'init', + //清空容器 + opcode: 'clearContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.init') + text: this.formatMessage('ArchiveCodeExt.clearContainer'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } }, { - //将内容加入序列 名称xx 值xx - opcode: 'serialization', + //将内容加入容器 名称xx 值xx + opcode: 'addContentToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.serialization'), + text: this.formatMessage('ArchiveCodeExt.addContentToContainer'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, name: { type: 'string', defaultValue: '用户id' @@ -156,10 +156,15 @@ class Archive_code { }, { //将变量加入序列 - opcode: 'serializationForVariable', + opcode: 'addVariableToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.serializationForVariable'), + text: this.formatMessage('ArchiveCodeExt.addVariableToContainer'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, name: { type: 'string', defaultValue: '金币' @@ -172,10 +177,15 @@ class Archive_code { }, { //将列表加入序列 - opcode: 'serializationForList', + opcode: 'addListToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.serializationForList'), + text: this.formatMessage('ArchiveCodeExt.addListToContainer'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, name: { type: 'string', defaultValue: '背包' @@ -194,18 +204,30 @@ class Archive_code { // }, { //返回序列化结果 - opcode: 'result', + opcode: 'containerToJSON', blockType: 'reporter', - text: this.formatMessage('ArchiveCodeExt.result'), + text: this.formatMessage('ArchiveCodeExt.containerToJSON'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } }, { //反序列化 - opcode: 'deserialization', + opcode: 'parseJSONToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.deserialization'), + text: this.formatMessage('ArchiveCodeExt.parseJSONToContainer'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, code: { type: 'string', defaultValue: `{"金币":200,"背包":["木头","面包"]}` @@ -214,15 +236,22 @@ class Archive_code { }, { //反序列化是否成功 - opcode: 'deserializable', + opcode: 'ifConvertedSuccessfully', blockType: 'Boolean', - text: this.formatMessage('ArchiveCodeExt.deserializable'), + text: this.formatMessage('ArchiveCodeExt.ifConvertedSuccessfully'), }, { //返回容器中数据数量 opcode: 'getAmount', blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getAmount'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } }, { //获取第n(从1开始)个内容,的(1名称2内容3类型4列表长度) @@ -230,6 +259,11 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContentByNumber'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, index: { type: 'number', //menu: 'varMenu2', @@ -253,6 +287,11 @@ class Archive_code { blockType: 'Boolean', text: this.formatMessage('ArchiveCodeExt.ifExist'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -266,6 +305,11 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContent'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -279,6 +323,11 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getLengthOfList'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -292,6 +341,11 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContentOfList'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -310,6 +364,11 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.saveContentToVar'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -327,6 +386,11 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.saveContentToList'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'listMenu2', @@ -344,12 +408,18 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.delete'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', defaultValue: '金币' }, } }, + "---" + this.formatMessage("ArchiveCodeExt.info2"), //🔧 JSON处理 { //直接获得container容器中的key内容 opcode: 'getContentInContainer', @@ -366,6 +436,7 @@ class Archive_code { }, } }, + "---" + this.formatMessage("ArchiveCodeExt.info3"), //💡 附加积木 { //加密 opcode: 'encrypt', @@ -444,6 +515,10 @@ class Archive_code { }, ], menus: { + containerMenu: { + items: 'findAllContainer', + acceptReporters: true, + }, varMenu: { items: 'findAllVar' }, @@ -492,59 +567,60 @@ class Archive_code { }; } - - - init() { - // console.log('editingTarget :', this.runtime._editingTarget) - // console.log('stageTarget :', this.runtime._stageTarget) - // console.log('_stageTarget.variables',JSON.stringify(this.runtime._stageTarget.variables)) - this.content = {}; - this.setArchive_code(); + findAllContainer() { + const list = []; + let temp = this.content; + Object.keys(temp).forEach(obj => { + //if ( Array.isArray (temp[obj]) ) { + list.push(obj); + //} + }); + if (list.length === 0) { + list.push({ + text: '-', + value: 'empty', + }); + } + return list; } - result() { - return this.archive_code; + _createContainerIfNotExist(con){ + if(!(con in this.content)) + this.content[con]={}; } - // showContent2json() { - // this.setContent2json(); - // return this.content2json; - // } + clearContainer(args) { + this._createContainerIfNotExist(args.con) + this.content[args.con] = {}; + } - stop() { - this.setArchive_code(); + containerToJSON(args) { + if(!(args.con in this.content)) return ''; + return JSON.stringify(this.content[args.con]); } - serialization(args) { - // if (this.archive_code !== '') - // this.archive_code += ',' - // this.archive_code += `"${args.name}":${JSON.stringify(args.value)}` - this.content[args.name] = args.value; + addContentToContainer(args) { + if(!(args.con in this.content)) return; + this.content[args.con][args.name] = args.value; } - serializationForVariable(args, util) { - // if (this.archive_code !== '') - // this.archive_code += ',' - // const variable = util.target.lookupVariableById(args.var); - // this.archive_code += `"${args.name}":${JSON.stringify(variable.value)}` + addVariableToContainer(args, util) { + if(!(args.con in this.content)) return; if (args.var !== 'empty') { const variable = util.target.lookupVariableById(args.var); - this.content[args.name] = variable.value; + this.content[args.con][args.name] = variable.value; } } - serializationForList(args, util) { - // if (this.archive_code !== '') - // this.archive_code += ',' - // const list = util.target.lookupVariableById(args.list); - // this.archive_code += `"${args.name}":${JSON.stringify(list)}` + addListToContainer(args, util) { + if(!(args.con in this.content)) return; if (args.list !== 'empty') { const list = util.target.lookupVariableById(args.list); - this.content[args.name] = list.value; + this.content[args.con][args.name] = list.value; } } - //直接获得container容器中的key内容 + //直接获得{container}中的key内容 getContentInContainer(args) { let content; try { @@ -559,16 +635,17 @@ class Archive_code { } } - deserialization(args) { + parseJSONToContainer(args) { + this._createContainerIfNotExist(args.con) let content; - this.deserializeSuccessfully = false; + this.convertedSuccessfully = false; try { // 如果解析失败,不要修改content。 content = JSON.parse(Cast.toString(args.code)) // 考虑数组[]情况。 if(typeof(content) === 'object' && !Array.isArray(content) && content !== null) { - this.content = content; - this.deserializeSuccessfully = true; + this.content[args.con] = content; + this.convertedSuccessfully = true; } } catch (e) { //this.content2 = {} @@ -576,22 +653,25 @@ class Archive_code { //console.log(typeof this.content) } - deserializable() { - return this.deserializeSuccessfully + ifConvertedSuccessfully() { + return this.convertedSuccessfully } ifExist(args) { - return Cast.toString(args.key) in this.content; + if(!(args.con in this.content)) return false; + return Cast.toString(args.key) in this.content[args.con]; } - getAmount() { - return Object.keys(this.content).length; + getAmount(args) { + if(!(args.con in this.content)) return ''; + return Object.keys(this.content[args.con]).length; } getContentByNumber(args) { - let key = Object.keys(this.content)[args.index - 1] + if(!(args.con in this.content)) return ''; + let key = Object.keys(this.content[args.con])[args.index - 1] if (key === undefined) return ''; - let value = this.content[key] + let value = this.content[args.con][key] switch (args.type) { case '1'://名称 return key; @@ -622,11 +702,9 @@ class Archive_code { } - getContent(args, util) { - // const variable = util.target.lookupVariableById(args.var); - // variable.value = args.key; - - return this.content[args.key] === undefined ? '' : this._anythingToNumberString(this.content[args.key]); + getContent(args) { + if(!(args.con in this.content)) return ''; + return this._anythingToNumberString(this.content[args.con][args.key]); } getUnicode(args) { @@ -637,11 +715,9 @@ class Archive_code { return String.fromCharCode(Cast.toNumber(args.code)) } - getContentOfList(args, util) { - // const variable = util.target.lookupVariableById(args.var); - // variable.value = args.key; - //如果没有这项,或者不是列表 - let t = this.content[args.key] + getContentOfList(args) { + if(!(args.con in this.content)) return ''; + let t = this.content[args.con][args.key] if (Array.isArray(t)) { let i = Cast.toNumber(args.n) - 1; if (i < 0 || i >= t.length) { @@ -654,10 +730,8 @@ class Archive_code { } getLengthOfList(args, util) { - // const variable = util.target.lookupVariableById(args.var); - // variable.value = args.key; - //如果没有这项,或者不是列表 - let t = this.content[args.key] + if(!(args.con in this.content)) return ''; + let t = this.content[args.con][args.key] return Array.isArray(t) ? t.length : ''; } @@ -668,15 +742,16 @@ class Archive_code { break; case "object": if(Array.isArray(value)) { + value = JSON.stringify(value); //列表直接用 JSON 格式显示 // 在原版scratch中如果直接使用列表作为变量,得到的结果是由空格分隔的。如果列表中每一项都是单个字符(数字不算),则结果不用空格分割。这里还原原版行为。 // 如果直接String()的话,项目会默认用逗号分割。 - let areChars = true; - value.forEach((v, i) => { - if (!(typeof v === "string" && v.length === 1)) { - areChars = false; - } - }); - value = value.join(areChars ? '' : ' '); + // let areChars = true; + // value.forEach((v, i) => { + // if (!(typeof v === "string" && v.length === 1)) { + // areChars = false; + // } + // }); + // value = value.join(areChars ? '' : ' '); } else { // 否则,就直接stringify value = JSON.stringify(value); @@ -689,22 +764,24 @@ class Archive_code { } saveContentToVar(args, util) { + if(!(args.con in this.content)) return; if (args.var !== 'empty') { const variable = util.target.lookupVariableById(args.var); - let value = this._anythingToNumberString(this.content[args.key]); + let value = this._anythingToNumberString(this.content[args.con][args.key]); variable.value = value; } } saveContentToList(args, util) { + if(!(args.con in this.content)) return; if (args.list !== 'empty') { const list = util.target.lookupVariableById(args.list); - if (!(args.key in this.content)) { + if (!(args.key in this.content[args.con])) { // 如果啥都没有就清空 list.value = []; return; } - let value = this.content[args.key]; + let value = this.content[args.con][args.key]; if (!Array.isArray(value)) { //如果要读取的内容不是列表而是什么奇奇怪怪的东西,就把它包装成列表 value = [value]; @@ -718,7 +795,8 @@ class Archive_code { } delete(args) { - delete this.content[args.key]; + if(!(args.con in this.content)) return; + delete this.content[args.con][args.key]; } //将密匙转换为一个值 From 68c375fc83730b18680e16b5239b33db367ce666 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Wed, 20 Jul 2022 00:41:20 +0800 Subject: [PATCH 058/357] =?UTF-8?q?=E4=B8=BA=E4=BA=86=E5=85=BC=E5=AE=B9?= =?UTF-8?q?=E6=97=A7=E7=89=88=EF=BC=8Copcode=E6=94=B9=E4=B8=BA=E5=8E=9Fopc?= =?UTF-8?q?ode?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/Archivecode.js | 23 ++++++++++++++--------- 1 file changed, 14 insertions(+), 9 deletions(-) diff --git a/Arkos/Archivecode.js b/Arkos/Archivecode.js index 938a93f9..24ca92e6 100644 --- a/Arkos/Archivecode.js +++ b/Arkos/Archivecode.js @@ -122,7 +122,8 @@ class Archive_code { "---" + this.formatMessage("ArchiveCodeExt.info1"), //🏺容器操作 { //清空容器 - opcode: 'clearContainer', + opcode: 'init',//为了兼容旧版,opcode用原来的 + func: 'clearContainer', blockType: 'command', text: this.formatMessage('ArchiveCodeExt.clearContainer'), arguments: { @@ -135,7 +136,8 @@ class Archive_code { }, { //将内容加入容器 名称xx 值xx - opcode: 'addContentToContainer', + opcode: 'serialization',//'addContentToContainer', + func:'addContentToContainer', blockType: 'command', text: this.formatMessage('ArchiveCodeExt.addContentToContainer'), arguments: { @@ -156,7 +158,8 @@ class Archive_code { }, { //将变量加入序列 - opcode: 'addVariableToContainer', + opcode: 'serializationForVariable',//'addVariableToContainer', + func:'addVariableToContainer', blockType: 'command', text: this.formatMessage('ArchiveCodeExt.addVariableToContainer'), arguments: { @@ -177,7 +180,8 @@ class Archive_code { }, { //将列表加入序列 - opcode: 'addListToContainer', + opcode: 'serializationForList', + func:'addListToContainer', blockType: 'command', text: this.formatMessage('ArchiveCodeExt.addListToContainer'), arguments: { @@ -204,7 +208,8 @@ class Archive_code { // }, { //返回序列化结果 - opcode: 'containerToJSON', + opcode: 'result', + func:'containerToJSON', blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.containerToJSON'), arguments: { @@ -215,11 +220,10 @@ class Archive_code { }, } }, - - { //反序列化 - opcode: 'parseJSONToContainer', + opcode: 'deserialization', + func:'parseJSONToContainer', blockType: 'command', text: this.formatMessage('ArchiveCodeExt.parseJSONToContainer'), arguments: { @@ -236,7 +240,8 @@ class Archive_code { }, { //反序列化是否成功 - opcode: 'ifConvertedSuccessfully', + opcode: 'deserializable', + func:'ifConvertedSuccessfully', blockType: 'Boolean', text: this.formatMessage('ArchiveCodeExt.ifConvertedSuccessfully'), }, From 8805f3d8d2f64c7bf06bd15e54221eb2dca39333 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Wed, 20 Jul 2022 00:48:50 +0800 Subject: [PATCH 059/357] =?UTF-8?q?=E6=8D=A2=E5=9B=9E=E6=97=A7=E7=89=88?= =?UTF-8?q?=EF=BC=8C=E6=B5=8B=E8=AF=95=E5=85=BC=E5=AE=B9=E6=80=A7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/Archivecode.js | 1090 ++++++++---------------------------------- 1 file changed, 202 insertions(+), 888 deletions(-) diff --git a/Arkos/Archivecode.js b/Arkos/Archivecode.js index 24ca92e6..5ef9dfbc 100644 --- a/Arkos/Archivecode.js +++ b/Arkos/Archivecode.js @@ -1,108 +1,103 @@ import Cast from '../utils/cast.js' //import cover from './assets/cover2.png' //import icon from './assets/icon2.svg' - -//合作者: -// Nights: 搭了框架,以及一些技术帮助 -// -6: 修复了许多纰漏和 bug ,加入了新的加密算法 -// Arkos: 什么都不会的屑蒟蒻 +//合作者:Nights:帮忙写了代码框架,并且给予了一些技术上的指导 // console.log(Cast.toNumber('123')) //console.log(Cast.toNumber('aab')) - -let LZString = _LZString(); - class Archive_code { constructor(runtime) { this.runtime = runtime - this.convertedSuccessfully = false //JSON转容器是否成功 - //content为一个大容器,内部存很多小容器 + this._archive_code = '' //生成的序列结果 + this.deserializeSuccessfully = false //存反序列化是否成功 + //存需要序列化的内容 this.content = { - "1":{ - 金币: 200, - 背包: ["木头", 233] - }, - "2":{ - Arkos:{score:95}, - yk1boy:{score:100} - } + 金币: 200, + 背包: ["木头", 233] } - + //存反序列化结果 + //this.content2 = {}; + //Object.assign(this.content2,this.content) + this.setArchive_code(); + //this.setContent2json(); this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArchiveCodeExt.extensionName': '存档码', - 'ArchiveCodeExt.info1': '🏺 容器操作', - 'ArchiveCodeExt.info2': '🔧 JSON处理', - 'ArchiveCodeExt.info3': '💡 附加积木', - 'ArchiveCodeExt.clearContainer': '清空容器[con]', - 'ArchiveCodeExt.addContentToContainer': '将内容[value]命名为[name]加入容器[con](已有则覆盖)', - 'ArchiveCodeExt.addVariableToContainer': '将变量[var]内容命名为[name]加入容器[con](已有则覆盖)', - 'ArchiveCodeExt.addListToContainer': '将列表[list]内容命名为[name]加入容器[con](已有则覆盖)', + 'ArchiveCodeExt.init': '清空容器', + 'ArchiveCodeExt.serialization': '将内容[value]命名为[name]加入容器(已有则覆盖)', + 'ArchiveCodeExt.serializationForVariable': '将变量[var]内容命名为[name]加入容器(已有则覆盖)', + 'ArchiveCodeExt.serializationForList': '将列表[list]内容命名为[name]加入容器(已有则覆盖)', //'ArchiveCodeExt.stop': '序列化结束', - 'ArchiveCodeExt.containerToJSON': '容器[con]内容对应字符串', - 'ArchiveCodeExt.parseJSONToContainer': '读取字符串到容器[con]:[code]', - 'ArchiveCodeExt.getContent': '容器[con]中名称为[key]的内容', + 'ArchiveCodeExt.result': '容器内容对应字符串', + 'ArchiveCodeExt.deserialization': '读取字符串到容器:[code]', + 'ArchiveCodeExt.getContent': '容器中名称为[key]的内容', //'ArchiveCodeExt.showContent2json':'读取结果', - 'ArchiveCodeExt.saveContentToVar': '将容器[con]中名称为[key]的内容保存到变量[var]', - 'ArchiveCodeExt.saveContentToList': '将容器[con]中名称为[key]的内容保存到列表[list]', - 'ArchiveCodeExt.ifConvertedSuccessfully': '读取成功?', - 'ArchiveCodeExt.getAmount': '容器[con]中内容的总数', - 'ArchiveCodeExt.ifExist': '容器[con]是否存在名为[key]的内容', - 'ArchiveCodeExt.getContentByNumber': '获取容器[con]中第[index]个内容的[type]', + 'ArchiveCodeExt.saveContentToVar': '将容器中名称为[key]的内容保存到变量[var]', + 'ArchiveCodeExt.saveContentToList': '将容器中名称为[key]的内容保存到列表[list]', + 'ArchiveCodeExt.deserializable': '读取成功?', + 'ArchiveCodeExt.getAmount': '容器中内容的总数', + 'ArchiveCodeExt.ifExist': '容器是否存在名为[key]的内容', + 'ArchiveCodeExt.getContentByNumber': '获取容器中第[index]个内容的[type]', 'ArchiveCodeExt.encrypt': '以[method]加密[str],密匙[key]', 'ArchiveCodeExt.decrypt': '以[method]解密[str],密匙[key]', 'ArchiveCodeExt.writeClipboard': '复制[str]到剪贴板', - 'ArchiveCodeExt.getLengthOfList': '容器[con]中名称为[key]的列表的长度', - 'ArchiveCodeExt.getContentOfList': '容器[con]中名称为[key]的列表的第[n]项', + 'ArchiveCodeExt.getLengthOfList': '容器中名称为[key]的列表的长度', + 'ArchiveCodeExt.getContentOfList': '容器中名称为[key]的列表的第[n]项', 'ArchiveCodeExt.getUnicode': '字符[c]的Unicode', 'ArchiveCodeExt.getCharByUnicode': 'Unicode[code]对应字符', 'ArchiveCodeExt.methodMenu.1': 'Arkos加密法', 'ArchiveCodeExt.methodMenu.2': '分裂加密法', - 'ArchiveCodeExt.methodMenu.3': 'LZString压缩', 'ArchiveCodeExt.infoMenu.1': '名称', 'ArchiveCodeExt.infoMenu.2': '内容', 'ArchiveCodeExt.infoMenu.3': '类型', 'ArchiveCodeExt.infoMenu.4': '列表长度', - 'ArchiveCodeExt.delete': '删除容器[con]中名为[key]的内容', - 'ArchiveCodeExt.getContentInContainer': '获得容器[container]中名为[key]的内容', + 'ArchiveCodeExt.delete': '删除容器中名为[key]的内容', }, - en: { 'ArchiveCodeExt.extensionName': 'Archive Code', - 'ArchiveCodeExt.clearContainer': 'empty container[con]', - 'ArchiveCodeExt.addContentToContainer': 'add content[value] to container[con], name as[name]', - 'ArchiveCodeExt.addVariableToContainer': 'add variable[var] to container[con], name as[name]', - 'ArchiveCodeExt.addListToContainer': 'add list[list] to container[con], name as[name]', + 'ArchiveCodeExt.init': 'empty Container', + 'ArchiveCodeExt.serialization': 'add content[value] to Container, name as[name]', + 'ArchiveCodeExt.serializationForVariable': 'add variable[var] to Container, name as[name]', + 'ArchiveCodeExt.serializationForList': 'add list[list] to Container, name as[name]', //'ArchiveCodeExt.stop': 'end serialization', - 'ArchiveCodeExt.containerToJSON': 'container[con] in string form', - 'ArchiveCodeExt.parseJSONToContainer': 'parse string[code] to container[con]', + 'ArchiveCodeExt.result': 'Container in string form', + 'ArchiveCodeExt.deserialization': 'parse string[code] to Container', 'ArchiveCodeExt.getContent': 'content of[key]', - 'ArchiveCodeExt.saveContentToVar': 'save[key]in container[con]to variable[var]', - 'ArchiveCodeExt.saveContentToList': 'save[key]in container[con]to list[list]', - 'ArchiveCodeExt.ifConvertedSuccessfully': 'parse successfullly?', - 'ArchiveCodeExt.getAmount': 'the amount of contents in container[con]', - 'ArchiveCodeExt.ifExist': 'container[con] contains[key]?', - 'ArchiveCodeExt.getContentByNumber': 'get [type]of #[index]content in container[con]', + 'ArchiveCodeExt.saveContentToVar': 'save [key]to variable[var]', + 'ArchiveCodeExt.saveContentToList': 'save[key]to list[list]', + 'ArchiveCodeExt.deserializable': 'parse successfullly?', + 'ArchiveCodeExt.getAmount': 'the amount of contents in Container', + 'ArchiveCodeExt.ifExist': 'Container contains[key]?', + 'ArchiveCodeExt.getContentByNumber': 'get [type]of #[index]content', 'ArchiveCodeExt.encrypt': 'use[method]to encrypt[str]with key[key]', 'ArchiveCodeExt.decrypt': 'use[method]to decrypt[str]with key[key]', 'ArchiveCodeExt.writeClipboard': 'copy[str]to clipboard', - 'ArchiveCodeExt.getContentOfList': '#[n] of list[key]in container[con]', - 'ArchiveCodeExt.getLengthOfList': 'length of list[key]in container[con]', + //'ArchiveCodeExt.showContent2json':'deserialization result', + 'ArchiveCodeExt.getContentOfList': '#[n] of list[key]', + 'ArchiveCodeExt.getLengthOfList': 'length of list[key]', 'ArchiveCodeExt.getUnicode': 'get Unicode of[c]', 'ArchiveCodeExt.getCharByUnicode': ' character of Unicode[code]', 'ArchiveCodeExt.methodMenu.1': 'Arkos cipher', 'ArchiveCodeExt.methodMenu.2': 'mitotic encryption', - 'ArchiveCodeExt.methodMenu.3': 'LZString', 'ArchiveCodeExt.infoMenu.1': 'name', 'ArchiveCodeExt.infoMenu.2': 'value', 'ArchiveCodeExt.infoMenu.3': 'type', 'ArchiveCodeExt.infoMenu.4': 'lenth of list', - 'ArchiveCodeExt.delete': 'delete content[key] in container[con]', - 'ArchiveCodeExt.getContentIncontainer': 'get [key] in [container]', + 'ArchiveCodeExt.delete': 'Delete content[key] in Container' }, }) - } - + //每次读取archive_code会生成一次 + get archive_code() { + this.setArchive_code(); + return this._archive_code; + } + //根据content的内容,将其JSON化,存到archive_code + setArchive_code() { + this._archive_code = JSON.stringify(this.content); + } + // setContent2json(){ + // this.content2json = JSON.stringify(this.content2); + // } formatMessage(id) { return this._formatMessage({ id, @@ -110,7 +105,6 @@ class Archive_code { description: id, }) } - getInfo() { return { id: 'Archivecode', // Archivecode @@ -119,33 +113,18 @@ class Archive_code { // menuIconURI: cover, // blockIconURI: icon, blocks: [ - "---" + this.formatMessage("ArchiveCodeExt.info1"), //🏺容器操作 { - //清空容器 - opcode: 'init',//为了兼容旧版,opcode用原来的 - func: 'clearContainer', + //开始序列化 + opcode: 'init', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.clearContainer'), - arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, - } + text: this.formatMessage('ArchiveCodeExt.init') }, { - //将内容加入容器 名称xx 值xx - opcode: 'serialization',//'addContentToContainer', - func:'addContentToContainer', + //将内容加入序列 名称xx 值xx + opcode: 'serialization', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.addContentToContainer'), + text: this.formatMessage('ArchiveCodeExt.serialization'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, name: { type: 'string', defaultValue: '用户id' @@ -158,16 +137,10 @@ class Archive_code { }, { //将变量加入序列 - opcode: 'serializationForVariable',//'addVariableToContainer', - func:'addVariableToContainer', + opcode: 'serializationForVariable', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.addVariableToContainer'), + text: this.formatMessage('ArchiveCodeExt.serializationForVariable'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, name: { type: 'string', defaultValue: '金币' @@ -181,15 +154,9 @@ class Archive_code { { //将列表加入序列 opcode: 'serializationForList', - func:'addListToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.addListToContainer'), + text: this.formatMessage('ArchiveCodeExt.serializationForList'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, name: { type: 'string', defaultValue: '背包' @@ -209,29 +176,15 @@ class Archive_code { { //返回序列化结果 opcode: 'result', - func:'containerToJSON', blockType: 'reporter', - text: this.formatMessage('ArchiveCodeExt.containerToJSON'), - arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, - } + text: this.formatMessage('ArchiveCodeExt.result'), }, { //反序列化 opcode: 'deserialization', - func:'parseJSONToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.parseJSONToContainer'), + text: this.formatMessage('ArchiveCodeExt.deserialization'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, code: { type: 'string', defaultValue: `{"金币":200,"背包":["木头","面包"]}` @@ -241,22 +194,14 @@ class Archive_code { { //反序列化是否成功 opcode: 'deserializable', - func:'ifConvertedSuccessfully', blockType: 'Boolean', - text: this.formatMessage('ArchiveCodeExt.ifConvertedSuccessfully'), + text: this.formatMessage('ArchiveCodeExt.deserializable'), }, { //返回容器中数据数量 opcode: 'getAmount', blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getAmount'), - arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, - } }, { //获取第n(从1开始)个内容,的(1名称2内容3类型4列表长度) @@ -264,11 +209,6 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContentByNumber'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, index: { type: 'number', //menu: 'varMenu2', @@ -292,11 +232,6 @@ class Archive_code { blockType: 'Boolean', text: this.formatMessage('ArchiveCodeExt.ifExist'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, key: { type: 'string', //menu: 'varMenu2', @@ -310,11 +245,6 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContent'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, key: { type: 'string', //menu: 'varMenu2', @@ -328,11 +258,6 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getLengthOfList'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, key: { type: 'string', //menu: 'varMenu2', @@ -346,11 +271,6 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContentOfList'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, key: { type: 'string', //menu: 'varMenu2', @@ -369,11 +289,6 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.saveContentToVar'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, key: { type: 'string', //menu: 'varMenu2', @@ -391,11 +306,6 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.saveContentToList'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, key: { type: 'string', //menu: 'listMenu2', @@ -413,35 +323,12 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.delete'), arguments: { - con: { - type: 'string', - defaultValue: '1', - menu: 'containerMenu' - }, - key: { - type: 'string', - defaultValue: '金币' - }, - } - }, - "---" + this.formatMessage("ArchiveCodeExt.info2"), //🔧 JSON处理 - { - //直接获得container容器中的key内容 - opcode: 'getContentInContainer', - blockType: 'reporter', - text: this.formatMessage('ArchiveCodeExt.getContentInContainer'), - arguments: { - container: { - type: 'string', - defaultValue: '{"金币":100,"经验值":50}' - }, key: { type: 'string', defaultValue: '金币' }, } }, - "---" + this.formatMessage("ArchiveCodeExt.info3"), //💡 附加积木 { //加密 opcode: 'encrypt', @@ -520,10 +407,6 @@ class Archive_code { }, ], menus: { - containerMenu: { - items: 'findAllContainer', - acceptReporters: true, - }, varMenu: { items: 'findAllVar' }, @@ -546,12 +429,9 @@ class Archive_code { // text: this.formatMessage('ArchiveCodeExt.methodMenu.2'), // value: '2' // }, - { - text: this.formatMessage('ArchiveCodeExt.methodMenu.3'), - value: '3' - }, ], - infoMenu: [{ + infoMenu: [ + { text: this.formatMessage('ArchiveCodeExt.infoMenu.1'), value: '1' }, @@ -571,242 +451,153 @@ class Archive_code { }, }; } - - findAllContainer() { - const list = []; - let temp = this.content; - Object.keys(temp).forEach(obj => { - //if ( Array.isArray (temp[obj]) ) { - list.push(obj); - //} - }); - if (list.length === 0) { - list.push({ - text: '-', - value: 'empty', - }); - } - return list; - } - - _createContainerIfNotExist(con){ - if(!(con in this.content)) - this.content[con]={}; - } - - clearContainer(args) { - this._createContainerIfNotExist(args.con) - this.content[args.con] = {}; - } - - containerToJSON(args) { - if(!(args.con in this.content)) return ''; - return JSON.stringify(this.content[args.con]); - } - - addContentToContainer(args) { - if(!(args.con in this.content)) return; - this.content[args.con][args.name] = args.value; - } - - addVariableToContainer(args, util) { - if(!(args.con in this.content)) return; + init() { + // console.log('editingTarget :', this.runtime._editingTarget) + // console.log('stageTarget :', this.runtime._stageTarget) + // console.log('_stageTarget.variables',JSON.stringify(this.runtime._stageTarget.variables)) + this.content = {}; + this.setArchive_code(); + } + result() { + return this.archive_code; + } + // showContent2json() { + // this.setContent2json(); + // return this.content2json; + // } + stop() { + this.setArchive_code(); + } + serialization(args) { + // if (this.archive_code !== '') + // this.archive_code += ',' + // this.archive_code += `"${args.name}":${JSON.stringify(args.value)}` + this.content[args.name] = args.value; + } + serializationForVariable(args, util) { + // if (this.archive_code !== '') + // this.archive_code += ',' + // const variable = util.target.lookupVariableById(args.var); + // this.archive_code += `"${args.name}":${JSON.stringify(variable.value)}` if (args.var !== 'empty') { const variable = util.target.lookupVariableById(args.var); - this.content[args.con][args.name] = variable.value; + this.content[args.name] = variable.value; } } - - addListToContainer(args, util) { - if(!(args.con in this.content)) return; + serializationForList(args, util) { + // if (this.archive_code !== '') + // this.archive_code += ',' + // const list = util.target.lookupVariableById(args.list); + // this.archive_code += `"${args.name}":${JSON.stringify(list)}` if (args.list !== 'empty') { const list = util.target.lookupVariableById(args.list); - this.content[args.con][args.name] = list.value; - } - } - - //直接获得{container}中的key内容 - getContentInContainer(args) { - let content; - try { - content = JSON.parse(Cast.toString(args.container)) - if(typeof(content) === 'object' && !Array.isArray(content) && content !== null) { - return this._anythingToNumberString(content[args.key]); - }else{ - return '' - } - } catch (e) { - return '' + this.content[args.name] = list.value; } } - parseJSONToContainer(args) { - this._createContainerIfNotExist(args.con) - let content; - this.convertedSuccessfully = false; + deserialization(args) { try { - // 如果解析失败,不要修改content。 - content = JSON.parse(Cast.toString(args.code)) - // 考虑数组[]情况。 - if(typeof(content) === 'object' && !Array.isArray(content) && content !== null) { - this.content[args.con] = content; - this.convertedSuccessfully = true; + this.content = JSON.parse(String(args.code)) + this.deserializeSuccessfully = true; + this.content = JSON.parse(args.code) + if(typeof(this.content)==='object'&& this.content !== null ){ + this.deserializeSuccessfully = true; + }else{ + this.content = {}; + this.deserializeSuccessfully = false; } } catch (e) { + this.deserializeSuccessfully = false; //this.content2 = {} } //console.log(typeof this.content) } - - ifConvertedSuccessfully() { - return this.convertedSuccessfully + deserializable() { + return this.deserializeSuccessfully } - ifExist(args) { - if(!(args.con in this.content)) return false; - return Cast.toString(args.key) in this.content[args.con]; + return (this.content.hasOwnProperty(args.key)); } - - getAmount(args) { - if(!(args.con in this.content)) return ''; - return Object.keys(this.content[args.con]).length; + getAmount() { + let count = 0; + for (let key in this.content) { + count++; + } + return count; } - getContentByNumber(args) { - if(!(args.con in this.content)) return ''; - let key = Object.keys(this.content[args.con])[args.index - 1] + let key = Object.keys(this.content)[args.index - 1] if (key === undefined) return ''; - let value = this.content[args.con][key] switch (args.type) { case '1'://名称 return key; case '2'://内容 - return this._anythingToNumberString(value); + return this.content[key]; case '3'://类型 - switch(typeof value){ - case "object": - // 本地化问题:返回的值是中文 - // 这一点可以统一用英文或者符号或者做成判断<名字为(abc)的数值是列表?> - // 或者符号化 - // 列表 容器 变量 没有 - // List Container Variable Unset - // [] {} "" undefined - return Array.isArray(value) ? '列表' : '容器'; - case "string": - case "number": - case "boolean": - return '变量'; - default: - return ''; - } + return (typeof this.content[key] === 'object') ? '列表' : '变量'; case '4'://列表长度 - return Array.isArray(value) ? value.length : ''; + return (typeof this.content[key] === 'object') ? this.content[key].length : ''; default: return ''; } - } - - getContent(args) { - if(!(args.con in this.content)) return ''; - return this._anythingToNumberString(this.content[args.con][args.key]); + getContent(args, util) { + // const variable = util.target.lookupVariableById(args.var); + // variable.value = args.key; + return (this.content[args.key] === undefined) ? '' : String(this.content[args.key]) } - getUnicode(args) { - return Cast.toString(args.c).charCodeAt(0) + return args.c.charCodeAt(0) } - getCharByUnicode(args) { - return String.fromCharCode(Cast.toNumber(args.code)) - } - - getContentOfList(args) { - if(!(args.con in this.content)) return ''; - let t = this.content[args.con][args.key] - if (Array.isArray(t)) { - let i = Cast.toNumber(args.n) - 1; - if (i < 0 || i >= t.length) { - return ''; - } - return t[i]; - } else { + return String.fromCharCode(args.code) + } + getContentOfList(args, util) { + // const variable = util.target.lookupVariableById(args.var); + // variable.value = args.key; + //如果没有这项,或者不是列表 + let t = this.content[args.key] + if (t === undefined || typeof t !== 'object') { return ''; + } else { + if (t[args.n - 1] === undefined) return ''; + else return t[args.n - 1] } } - getLengthOfList(args, util) { - if(!(args.con in this.content)) return ''; - let t = this.content[args.con][args.key] - return Array.isArray(t) ? t.length : ''; - } - - _anythingToNumberString(value) { - switch(typeof(value)){ - case "string": - case "number": - break; - case "object": - if(Array.isArray(value)) { - value = JSON.stringify(value); //列表直接用 JSON 格式显示 - // 在原版scratch中如果直接使用列表作为变量,得到的结果是由空格分隔的。如果列表中每一项都是单个字符(数字不算),则结果不用空格分割。这里还原原版行为。 - // 如果直接String()的话,项目会默认用逗号分割。 - // let areChars = true; - // value.forEach((v, i) => { - // if (!(typeof v === "string" && v.length === 1)) { - // areChars = false; - // } - // }); - // value = value.join(areChars ? '' : ' '); - } else { - // 否则,就直接stringify - value = JSON.stringify(value); - } - break; - default: - value = ''; //包含了undefined + // const variable = util.target.lookupVariableById(args.var); + // variable.value = args.key; + //如果没有这项,或者不是列表 + let t = this.content[args.key] + if (t === undefined || typeof t !== 'object') { + return ''; + } else { + return t.length; } - return value; } - saveContentToVar(args, util) { - if(!(args.con in this.content)) return; if (args.var !== 'empty') { const variable = util.target.lookupVariableById(args.var); - let value = this._anythingToNumberString(this.content[args.con][args.key]); - variable.value = value; + variable.value = String(this.content[args.key]); + if (variable.value === "undefined") variable.value = ''; } } - saveContentToList(args, util) { - if(!(args.con in this.content)) return; if (args.list !== 'empty') { const list = util.target.lookupVariableById(args.list); - if (!(args.key in this.content[args.con])) { - // 如果啥都没有就清空 - list.value = []; - return; - } - let value = this.content[args.con][args.key]; - if (!Array.isArray(value)) { - //如果要读取的内容不是列表而是什么奇奇怪怪的东西,就把它包装成列表 - value = [value]; + list.value = this.content[args.key]; + if (list.value === undefined) list.value = []; + if (typeof list.value !== 'object') { + list.value = [list.value]; } - value.forEach((v, i) => { - // 防止数组内容混入奇奇怪怪的东西 - value[i] = this._anythingToNumberString(v); - }); - list.value = value; } } - delete(args) { - if(!(args.con in this.content)) return; - delete this.content[args.con][args.key]; + Reflect.deleteProperty(this.content, args.key); } - //将密匙转换为一个值 keyVar(k) { - k = Cast.toString(k) + k = String(k) let t = 13; for (let i = 0; i < k.length; i++) { t += k.charCodeAt(i) @@ -814,66 +605,42 @@ class Archive_code { } return t } - //加密 encrypt(args) { - args.key = Cast.toString(args.key) - args.str = Cast.toString(args.str) switch (args.method) { case '1': return this.ArkosEncrypt(args); case '2': return this.encrypt2(args); - case '3': - return LZString.scompress(args.str, args.key); default: return ''; } } - //解密 decrypt(args) { - args.key = Cast.toString(args.key) - args.str = Cast.toString(args.str) switch (args.method) { case '1': return this.ArkosDecrypt(args); case '2': return this.decrypt2(args); - case '3': - return LZString.sdecompress(args.str, args.key); default: return ''; } } - - //发现 Unicode 为 0 10 13 55296~57343(2047个字符) 的字符无法被正常复制,故排除掉这些字符。 - getCode(c) { - c = Cast.toString(c).charCodeAt(0) - if (c === 0) return NaN - else if (c < 10) return c-1 //排除0 - else if (c < 13) return c-2 //排除0 10 - else if (c < 55296) return c-3 //排除0 10 13 - else if (c > 55296) return c-2050 //排除0 10 13 55296~57343(2047个字符) - else return NaN - } - //Arkos加密法 ArkosEncrypt(args) { args.key = this.keyVar(args.key) - args.str = Cast.toString(args.str) + args.str = String(args.str) let b = '' for (let i = 0; i < args.str.length; i++) { b += this.enChar1(args.str[i], args.key + i) } return b } - - //Arkos解密 ArkosDecrypt(args) { args.key = this.keyVar(args.key) - args.str = Cast.toString(args.str) + args.str = String(args.str) let b = '' for (let i = 0; i < args.str.length; i++) { b += this.deChar1(args.str[i], args.key + i) @@ -881,24 +648,19 @@ class Archive_code { //console.log('123') return b } - enChar1(c, p) { - // 目前我知道的unicode字符最大编码是131071 let t = (c.charCodeAt(0) + p) % 54533 // - t += 9 - 2 * (t % 10) - + t = t - t % 10 + (9 - t % 10) return String.fromCharCode(t) } - - deChar1(c, p) { let t = c.charCodeAt(0) - t += 9 - 2 * (t % 10) - t = (t - p + 54533) % 54533 + //t%=65536 + t = t - t % 10 + (9 - t % 10) + t = (t - p) % 54533 + if (t < 0) t += 54533 return String.fromCharCode(t) } - - //分裂加密法 encrypt2(args) { args.key = this.keyVar(args.key) @@ -909,7 +671,6 @@ class Archive_code { } return b } - //分裂解密 decrypt2(args) { args.key = this.keyVar(args.key) @@ -921,16 +682,13 @@ class Archive_code { //console.log('123') return b } - enChar2(c, p) { let t = (c.charCodeAt(0) + p) % 65536 t = t - t % 10 + (9 - t % 10) - let c1 = String.fromCharCode(t >> 8) let c2 = String.fromCharCode(t % 256) return c1 + c2 } - deChar2(c1, c2, p) { let t = c1.charCodeAt(0) * 256 + c2.charCodeAt(0) t %= 65536 @@ -938,20 +696,9 @@ class Archive_code { t = (t - p + 65536) % 65536 return String.fromCharCode(t) } - writeClipboard(args) { - // 错误处理... - if("navigator" in window && "clipboard" in navigator && "writeText" in navigator.clipboard) { - navigator.clipboard.writeText(Cast.toString(args.str)).catch(x => writeClipboard2(args)); - } else { - writeClipboard2(args); - } - } - - writeClipboard2(args) { - prompt("无法访问剪贴板,请选择在下方文字点击右键或按 Ctrl+C 复制。", Cast.toString(args.str)); + navigator.clipboard.writeText(args.str); } - compare(propName) { return (a, b) => { a = a[propName] @@ -961,36 +708,35 @@ class Archive_code { else return 0; } } - findAllVar() { const list = []; - let temp; + let temp = this.runtime._stageTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === '') { + list.push({ + text: `${temp[obj].name}`, + value: temp[obj].id, + }); + } + }); try { - temp = this.runtime._stageTarget.variables + temp = this.runtime._editingTarget.variables + } catch (e) { + temp = 'e' + } + if (temp !=='e' && this.runtime._editingTarget !== this.runtime._stageTarget) { Object.keys(temp).forEach(obj => { if (temp[obj].type === '') { list.push({ - text: temp[obj].name, + text: `[私有变量]${temp[obj].name}`, value: temp[obj].id, }); } }); - if (!this.runtime._editingTarget.isStage) { - temp = this.runtime._editingTarget.variables - Object.keys(temp).forEach(obj => { - if (temp[obj].type === '') { - list.push({ - text: '[私有变量]' + temp[obj].name, - value: temp[obj].id, - }); - } - }); - } - } catch (e) { } if (list.length === 0) { list.push({ - text: `-`, + text: `*没有变量*`, value: 'empty', }); } @@ -1004,43 +750,42 @@ class Archive_code { list.sort(this.compare("text")); return list; } - findAllList() { const list = []; - let temp; + let temp = this.runtime._stageTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === 'list') { + //console.log(temp[obj].type) + list.push({ + text: `${temp[obj].name}`, + value: temp[obj].id, + }); + } + }); try { - temp = this.runtime._stageTarget.variables + temp = this.runtime._editingTarget.variables + } catch (e) { + temp = 'e' + } + if (temp !=='e' && this.runtime._editingTarget !== this.runtime._stageTarget) { Object.keys(temp).forEach(obj => { if (temp[obj].type === 'list') { list.push({ - text: temp[obj].name, + text: `[私有列表]${temp[obj].name}`, value: temp[obj].id, }); } }); - if (!this.runtime._editingTarget.isStage) { - temp = this.runtime._editingTarget.variables - Object.keys(temp).forEach(obj => { - if (temp[obj].type === 'list') { - list.push({ - text: '[私有列表]' + temp[obj].name, - value: temp[obj].id, - }); - } - }); - } - } catch (e) { } if (list.length === 0) { list.push({ - text: `-`, + text: `*没有列表*`, value: 'empty', }); } list.sort(this.compare("text")); return list; } - findAllVarContents() { const list = []; let temp = this.content @@ -1054,14 +799,13 @@ class Archive_code { }); if (list.length === 0) { list.push({ - text: '-', + text: '*没有变量*', value: 'empty', }); } list.sort(this.compare("text")); return list; } - findAllListsContents() { const list = []; let temp = this.content @@ -1075,444 +819,14 @@ class Archive_code { }); if (list.length === 0) { list.push({ - text: '-', + text: '*没有列表*', value: 'empty', }); } list.sort(this.compare("text")); return list; } - - } - - -// - lz-strings.js (modified) - - -// Copyright (c) 2013 Pieroxy -// This work is free. You can redistribute it and/or modify it -// under the terms of the WTFPL, Version 2 -// For more information see LICENSE.txt or http://www.wtfpl.net/ -// -// For more information, the home page: -// http://pieroxy.net/blog/pages/lz-string/testing.html -// -// LZ-based compression algorithm, version 1.4.4 -function _LZString() { - - // private property - var f = String.fromCharCode; - - var LZString = { - scompress: function(uncompressed, key) { - if (key.length !== 0) - key = this.scompress(key, ""); - return this._compress(uncompressed, 14, function(x, i) { - if (key.length !== 0) - x ^= (key.charCodeAt(i % key.length) * (i + 1)) & ((1 << 14) - 1); - return f(x + 0x4E00); - }); - }, - - _compress: function(uncompressed, bitsPerChar, getCharFromInt) { - if (uncompressed == null) return ""; - var i, value, - context_dictionary = {}, - context_dictionaryToCreate = {}, - context_c = "", - context_wc = "", - context_w = "", - context_enlargeIn = 2, // Compensate for the first entry which should not count - context_dictSize = 3, - context_numBits = 2, - context_data = [], - context_data_val = 0, - context_data_position = 0, - ii; - - for (ii = 0; ii < uncompressed.length; ii += 1) { - context_c = uncompressed.charAt(ii); - if (!Object.prototype.hasOwnProperty.call(context_dictionary, context_c)) { - context_dictionary[context_c] = context_dictSize++; - context_dictionaryToCreate[context_c] = true; - } - - context_wc = context_w + context_c; - if (Object.prototype.hasOwnProperty.call(context_dictionary, context_wc)) { - context_w = context_wc; - } else { - if (Object.prototype.hasOwnProperty.call(context_dictionaryToCreate, context_w)) { - if (context_w.charCodeAt(0) < 256) { - for (i = 0; i < context_numBits; i++) { - context_data_val = (context_data_val << 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - } - value = context_w.charCodeAt(0); - for (i = 0; i < 8; i++) { - context_data_val = (context_data_val << 1) | (value & 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = value >> 1; - } - } else { - value = 1; - for (i = 0; i < context_numBits; i++) { - context_data_val = (context_data_val << 1) | value; - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = 0; - } - value = context_w.charCodeAt(0); - for (i = 0; i < 16; i++) { - context_data_val = (context_data_val << 1) | (value & 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = value >> 1; - } - } - context_enlargeIn--; - if (context_enlargeIn == 0) { - context_enlargeIn = Math.pow(2, context_numBits); - context_numBits++; - } - delete context_dictionaryToCreate[context_w]; - } else { - value = context_dictionary[context_w]; - for (i = 0; i < context_numBits; i++) { - context_data_val = (context_data_val << 1) | (value & 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = value >> 1; - } - - - } - context_enlargeIn--; - if (context_enlargeIn == 0) { - context_enlargeIn = Math.pow(2, context_numBits); - context_numBits++; - } - // Add wc to the dictionary. - context_dictionary[context_wc] = context_dictSize++; - context_w = String(context_c); - } - } - - // Output the code for w. - if (context_w !== "") { - if (Object.prototype.hasOwnProperty.call(context_dictionaryToCreate, context_w)) { - if (context_w.charCodeAt(0) < 256) { - for (i = 0; i < context_numBits; i++) { - context_data_val = (context_data_val << 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - } - value = context_w.charCodeAt(0); - for (i = 0; i < 8; i++) { - context_data_val = (context_data_val << 1) | (value & 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = value >> 1; - } - } else { - value = 1; - for (i = 0; i < context_numBits; i++) { - context_data_val = (context_data_val << 1) | value; - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = 0; - } - value = context_w.charCodeAt(0); - for (i = 0; i < 16; i++) { - context_data_val = (context_data_val << 1) | (value & 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = value >> 1; - } - } - context_enlargeIn--; - if (context_enlargeIn == 0) { - context_enlargeIn = Math.pow(2, context_numBits); - context_numBits++; - } - delete context_dictionaryToCreate[context_w]; - } else { - value = context_dictionary[context_w]; - for (i = 0; i < context_numBits; i++) { - context_data_val = (context_data_val << 1) | (value & 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = value >> 1; - } - - - } - context_enlargeIn--; - if (context_enlargeIn == 0) { - context_enlargeIn = Math.pow(2, context_numBits); - context_numBits++; - } - } - - // Mark the end of the stream - value = 2; - for (i = 0; i < context_numBits; i++) { - context_data_val = (context_data_val << 1) | (value & 1); - if (context_data_position == bitsPerChar - 1) { - context_data_position = 0; - context_data.push(getCharFromInt(context_data_val, context_data.length)); - context_data_val = 0; - } else { - context_data_position++; - } - value = value >> 1; - } - - // Flush the last char - while (true) { - context_data_val = (context_data_val << 1); - if (context_data_position == bitsPerChar - 1) { - context_data.push(getCharFromInt(context_data_val, context_data.length)); - break; - } else context_data_position++; - } - return context_data.join(''); - }, - - sdecompress: function(compressed, key) { - if (key.length !== 0) - key = this.scompress(key, ""); - return this._decompress(compressed.length, 1 << (14 - 1), function(i) { - let x = compressed.charCodeAt(i) - 0x4E00; - if (key.length !== 0) - x ^= (key.charCodeAt(i % key.length) * (i + 1)) & ((1 << 14) - 1); - return x; - }); - }, - - _decompress: function(length, resetValue, getNextValue) { - var dictionary = [], - next, - enlargeIn = 4, - dictSize = 4, - numBits = 3, - entry = "", - result = [], - i, - w, - bits, resb, maxpower, power, - c, - data = { - val: getNextValue(0), - position: resetValue, - index: 1 - }; - - for (i = 0; i < 3; i += 1) { - dictionary[i] = i; - } - - bits = 0; - maxpower = Math.pow(2, 2); - power = 1; - while (power != maxpower) { - resb = data.val & data.position; - data.position >>= 1; - if (data.position == 0) { - data.position = resetValue; - data.val = getNextValue(data.index++); - } - bits |= (resb > 0 ? 1 : 0) * power; - power <<= 1; - } - - switch (next = bits) { - case 0: - bits = 0; - maxpower = Math.pow(2, 8); - power = 1; - while (power != maxpower) { - resb = data.val & data.position; - data.position >>= 1; - if (data.position == 0) { - data.position = resetValue; - data.val = getNextValue(data.index++); - } - bits |= (resb > 0 ? 1 : 0) * power; - power <<= 1; - } - c = f(bits); - break; - case 1: - bits = 0; - maxpower = Math.pow(2, 16); - power = 1; - while (power != maxpower) { - resb = data.val & data.position; - data.position >>= 1; - if (data.position == 0) { - data.position = resetValue; - data.val = getNextValue(data.index++); - } - bits |= (resb > 0 ? 1 : 0) * power; - power <<= 1; - } - c = f(bits); - break; - case 2: - return ""; - } - dictionary[3] = c; - w = c; - result.push(c); - while (true) { - if (data.index > length) { - return ""; - } - - bits = 0; - maxpower = Math.pow(2, numBits); - power = 1; - while (power != maxpower) { - resb = data.val & data.position; - data.position >>= 1; - if (data.position == 0) { - data.position = resetValue; - data.val = getNextValue(data.index++); - } - bits |= (resb > 0 ? 1 : 0) * power; - power <<= 1; - } - - switch (c = bits) { - case 0: - bits = 0; - maxpower = Math.pow(2, 8); - power = 1; - while (power != maxpower) { - resb = data.val & data.position; - data.position >>= 1; - if (data.position == 0) { - data.position = resetValue; - data.val = getNextValue(data.index++); - } - bits |= (resb > 0 ? 1 : 0) * power; - power <<= 1; - } - - dictionary[dictSize++] = f(bits); - c = dictSize - 1; - enlargeIn--; - break; - case 1: - bits = 0; - maxpower = Math.pow(2, 16); - power = 1; - while (power != maxpower) { - resb = data.val & data.position; - data.position >>= 1; - if (data.position == 0) { - data.position = resetValue; - data.val = getNextValue(data.index++); - } - bits |= (resb > 0 ? 1 : 0) * power; - power <<= 1; - } - dictionary[dictSize++] = f(bits); - c = dictSize - 1; - enlargeIn--; - break; - case 2: - return result.join(''); - } - - if (enlargeIn == 0) { - enlargeIn = Math.pow(2, numBits); - numBits++; - } - - if (dictionary[c]) { - entry = dictionary[c]; - } else { - if (c === dictSize) { - entry = w + w.charAt(0); - } else { - // throw new Error("LZString: decompress error."); - // return null; - return ""; - } - } - result.push(entry); - - // Add w+entry[0] to the dictionary. - dictionary[dictSize++] = w + entry.charAt(0); - enlargeIn--; - - w = entry; - - if (enlargeIn == 0) { - enlargeIn = Math.pow(2, numBits); - numBits++; - } - - } - } - }; - return LZString; -} - - window.tempExt = { Extension: Archive_code, info: { From ca423c593a90b8d7bfb09fa98f0e998ad8d9f0c9 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Wed, 20 Jul 2022 00:51:29 +0800 Subject: [PATCH 060/357] =?UTF-8?q?=E6=8D=A2=E5=9B=9E=E6=96=B0=E7=89=88?= =?UTF-8?q?=EF=BC=8C=E6=B5=8B=E8=AF=95=E5=85=BC=E5=AE=B9=E6=80=A7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/Archivecode.js | 1090 ++++++++++++++++++++++++++++++++++-------- 1 file changed, 888 insertions(+), 202 deletions(-) diff --git a/Arkos/Archivecode.js b/Arkos/Archivecode.js index 5ef9dfbc..24ca92e6 100644 --- a/Arkos/Archivecode.js +++ b/Arkos/Archivecode.js @@ -1,103 +1,108 @@ import Cast from '../utils/cast.js' //import cover from './assets/cover2.png' //import icon from './assets/icon2.svg' -//合作者:Nights:帮忙写了代码框架,并且给予了一些技术上的指导 + +//合作者: +// Nights: 搭了框架,以及一些技术帮助 +// -6: 修复了许多纰漏和 bug ,加入了新的加密算法 +// Arkos: 什么都不会的屑蒟蒻 // console.log(Cast.toNumber('123')) //console.log(Cast.toNumber('aab')) + +let LZString = _LZString(); + class Archive_code { constructor(runtime) { this.runtime = runtime - this._archive_code = '' //生成的序列结果 - this.deserializeSuccessfully = false //存反序列化是否成功 - //存需要序列化的内容 + this.convertedSuccessfully = false //JSON转容器是否成功 + //content为一个大容器,内部存很多小容器 this.content = { - 金币: 200, - 背包: ["木头", 233] + "1":{ + 金币: 200, + 背包: ["木头", 233] + }, + "2":{ + Arkos:{score:95}, + yk1boy:{score:100} + } } - //存反序列化结果 - //this.content2 = {}; - //Object.assign(this.content2,this.content) - this.setArchive_code(); - //this.setContent2json(); + this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'ArchiveCodeExt.extensionName': '存档码', - 'ArchiveCodeExt.init': '清空容器', - 'ArchiveCodeExt.serialization': '将内容[value]命名为[name]加入容器(已有则覆盖)', - 'ArchiveCodeExt.serializationForVariable': '将变量[var]内容命名为[name]加入容器(已有则覆盖)', - 'ArchiveCodeExt.serializationForList': '将列表[list]内容命名为[name]加入容器(已有则覆盖)', + 'ArchiveCodeExt.info1': '🏺 容器操作', + 'ArchiveCodeExt.info2': '🔧 JSON处理', + 'ArchiveCodeExt.info3': '💡 附加积木', + 'ArchiveCodeExt.clearContainer': '清空容器[con]', + 'ArchiveCodeExt.addContentToContainer': '将内容[value]命名为[name]加入容器[con](已有则覆盖)', + 'ArchiveCodeExt.addVariableToContainer': '将变量[var]内容命名为[name]加入容器[con](已有则覆盖)', + 'ArchiveCodeExt.addListToContainer': '将列表[list]内容命名为[name]加入容器[con](已有则覆盖)', //'ArchiveCodeExt.stop': '序列化结束', - 'ArchiveCodeExt.result': '容器内容对应字符串', - 'ArchiveCodeExt.deserialization': '读取字符串到容器:[code]', - 'ArchiveCodeExt.getContent': '容器中名称为[key]的内容', + 'ArchiveCodeExt.containerToJSON': '容器[con]内容对应字符串', + 'ArchiveCodeExt.parseJSONToContainer': '读取字符串到容器[con]:[code]', + 'ArchiveCodeExt.getContent': '容器[con]中名称为[key]的内容', //'ArchiveCodeExt.showContent2json':'读取结果', - 'ArchiveCodeExt.saveContentToVar': '将容器中名称为[key]的内容保存到变量[var]', - 'ArchiveCodeExt.saveContentToList': '将容器中名称为[key]的内容保存到列表[list]', - 'ArchiveCodeExt.deserializable': '读取成功?', - 'ArchiveCodeExt.getAmount': '容器中内容的总数', - 'ArchiveCodeExt.ifExist': '容器是否存在名为[key]的内容', - 'ArchiveCodeExt.getContentByNumber': '获取容器中第[index]个内容的[type]', + 'ArchiveCodeExt.saveContentToVar': '将容器[con]中名称为[key]的内容保存到变量[var]', + 'ArchiveCodeExt.saveContentToList': '将容器[con]中名称为[key]的内容保存到列表[list]', + 'ArchiveCodeExt.ifConvertedSuccessfully': '读取成功?', + 'ArchiveCodeExt.getAmount': '容器[con]中内容的总数', + 'ArchiveCodeExt.ifExist': '容器[con]是否存在名为[key]的内容', + 'ArchiveCodeExt.getContentByNumber': '获取容器[con]中第[index]个内容的[type]', 'ArchiveCodeExt.encrypt': '以[method]加密[str],密匙[key]', 'ArchiveCodeExt.decrypt': '以[method]解密[str],密匙[key]', 'ArchiveCodeExt.writeClipboard': '复制[str]到剪贴板', - 'ArchiveCodeExt.getLengthOfList': '容器中名称为[key]的列表的长度', - 'ArchiveCodeExt.getContentOfList': '容器中名称为[key]的列表的第[n]项', + 'ArchiveCodeExt.getLengthOfList': '容器[con]中名称为[key]的列表的长度', + 'ArchiveCodeExt.getContentOfList': '容器[con]中名称为[key]的列表的第[n]项', 'ArchiveCodeExt.getUnicode': '字符[c]的Unicode', 'ArchiveCodeExt.getCharByUnicode': 'Unicode[code]对应字符', 'ArchiveCodeExt.methodMenu.1': 'Arkos加密法', 'ArchiveCodeExt.methodMenu.2': '分裂加密法', + 'ArchiveCodeExt.methodMenu.3': 'LZString压缩', 'ArchiveCodeExt.infoMenu.1': '名称', 'ArchiveCodeExt.infoMenu.2': '内容', 'ArchiveCodeExt.infoMenu.3': '类型', 'ArchiveCodeExt.infoMenu.4': '列表长度', - 'ArchiveCodeExt.delete': '删除容器中名为[key]的内容', + 'ArchiveCodeExt.delete': '删除容器[con]中名为[key]的内容', + 'ArchiveCodeExt.getContentInContainer': '获得容器[container]中名为[key]的内容', }, + en: { 'ArchiveCodeExt.extensionName': 'Archive Code', - 'ArchiveCodeExt.init': 'empty Container', - 'ArchiveCodeExt.serialization': 'add content[value] to Container, name as[name]', - 'ArchiveCodeExt.serializationForVariable': 'add variable[var] to Container, name as[name]', - 'ArchiveCodeExt.serializationForList': 'add list[list] to Container, name as[name]', + 'ArchiveCodeExt.clearContainer': 'empty container[con]', + 'ArchiveCodeExt.addContentToContainer': 'add content[value] to container[con], name as[name]', + 'ArchiveCodeExt.addVariableToContainer': 'add variable[var] to container[con], name as[name]', + 'ArchiveCodeExt.addListToContainer': 'add list[list] to container[con], name as[name]', //'ArchiveCodeExt.stop': 'end serialization', - 'ArchiveCodeExt.result': 'Container in string form', - 'ArchiveCodeExt.deserialization': 'parse string[code] to Container', + 'ArchiveCodeExt.containerToJSON': 'container[con] in string form', + 'ArchiveCodeExt.parseJSONToContainer': 'parse string[code] to container[con]', 'ArchiveCodeExt.getContent': 'content of[key]', - 'ArchiveCodeExt.saveContentToVar': 'save [key]to variable[var]', - 'ArchiveCodeExt.saveContentToList': 'save[key]to list[list]', - 'ArchiveCodeExt.deserializable': 'parse successfullly?', - 'ArchiveCodeExt.getAmount': 'the amount of contents in Container', - 'ArchiveCodeExt.ifExist': 'Container contains[key]?', - 'ArchiveCodeExt.getContentByNumber': 'get [type]of #[index]content', + 'ArchiveCodeExt.saveContentToVar': 'save[key]in container[con]to variable[var]', + 'ArchiveCodeExt.saveContentToList': 'save[key]in container[con]to list[list]', + 'ArchiveCodeExt.ifConvertedSuccessfully': 'parse successfullly?', + 'ArchiveCodeExt.getAmount': 'the amount of contents in container[con]', + 'ArchiveCodeExt.ifExist': 'container[con] contains[key]?', + 'ArchiveCodeExt.getContentByNumber': 'get [type]of #[index]content in container[con]', 'ArchiveCodeExt.encrypt': 'use[method]to encrypt[str]with key[key]', 'ArchiveCodeExt.decrypt': 'use[method]to decrypt[str]with key[key]', 'ArchiveCodeExt.writeClipboard': 'copy[str]to clipboard', - //'ArchiveCodeExt.showContent2json':'deserialization result', - 'ArchiveCodeExt.getContentOfList': '#[n] of list[key]', - 'ArchiveCodeExt.getLengthOfList': 'length of list[key]', + 'ArchiveCodeExt.getContentOfList': '#[n] of list[key]in container[con]', + 'ArchiveCodeExt.getLengthOfList': 'length of list[key]in container[con]', 'ArchiveCodeExt.getUnicode': 'get Unicode of[c]', 'ArchiveCodeExt.getCharByUnicode': ' character of Unicode[code]', 'ArchiveCodeExt.methodMenu.1': 'Arkos cipher', 'ArchiveCodeExt.methodMenu.2': 'mitotic encryption', + 'ArchiveCodeExt.methodMenu.3': 'LZString', 'ArchiveCodeExt.infoMenu.1': 'name', 'ArchiveCodeExt.infoMenu.2': 'value', 'ArchiveCodeExt.infoMenu.3': 'type', 'ArchiveCodeExt.infoMenu.4': 'lenth of list', - 'ArchiveCodeExt.delete': 'Delete content[key] in Container' + 'ArchiveCodeExt.delete': 'delete content[key] in container[con]', + 'ArchiveCodeExt.getContentIncontainer': 'get [key] in [container]', }, }) + } - //每次读取archive_code会生成一次 - get archive_code() { - this.setArchive_code(); - return this._archive_code; - } - //根据content的内容,将其JSON化,存到archive_code - setArchive_code() { - this._archive_code = JSON.stringify(this.content); - } - // setContent2json(){ - // this.content2json = JSON.stringify(this.content2); - // } + formatMessage(id) { return this._formatMessage({ id, @@ -105,6 +110,7 @@ class Archive_code { description: id, }) } + getInfo() { return { id: 'Archivecode', // Archivecode @@ -113,18 +119,33 @@ class Archive_code { // menuIconURI: cover, // blockIconURI: icon, blocks: [ + "---" + this.formatMessage("ArchiveCodeExt.info1"), //🏺容器操作 { - //开始序列化 - opcode: 'init', + //清空容器 + opcode: 'init',//为了兼容旧版,opcode用原来的 + func: 'clearContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.init') + text: this.formatMessage('ArchiveCodeExt.clearContainer'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } }, { - //将内容加入序列 名称xx 值xx - opcode: 'serialization', + //将内容加入容器 名称xx 值xx + opcode: 'serialization',//'addContentToContainer', + func:'addContentToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.serialization'), + text: this.formatMessage('ArchiveCodeExt.addContentToContainer'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, name: { type: 'string', defaultValue: '用户id' @@ -137,10 +158,16 @@ class Archive_code { }, { //将变量加入序列 - opcode: 'serializationForVariable', + opcode: 'serializationForVariable',//'addVariableToContainer', + func:'addVariableToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.serializationForVariable'), + text: this.formatMessage('ArchiveCodeExt.addVariableToContainer'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, name: { type: 'string', defaultValue: '金币' @@ -154,9 +181,15 @@ class Archive_code { { //将列表加入序列 opcode: 'serializationForList', + func:'addListToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.serializationForList'), + text: this.formatMessage('ArchiveCodeExt.addListToContainer'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, name: { type: 'string', defaultValue: '背包' @@ -176,15 +209,29 @@ class Archive_code { { //返回序列化结果 opcode: 'result', + func:'containerToJSON', blockType: 'reporter', - text: this.formatMessage('ArchiveCodeExt.result'), + text: this.formatMessage('ArchiveCodeExt.containerToJSON'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } }, { //反序列化 opcode: 'deserialization', + func:'parseJSONToContainer', blockType: 'command', - text: this.formatMessage('ArchiveCodeExt.deserialization'), + text: this.formatMessage('ArchiveCodeExt.parseJSONToContainer'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, code: { type: 'string', defaultValue: `{"金币":200,"背包":["木头","面包"]}` @@ -194,14 +241,22 @@ class Archive_code { { //反序列化是否成功 opcode: 'deserializable', + func:'ifConvertedSuccessfully', blockType: 'Boolean', - text: this.formatMessage('ArchiveCodeExt.deserializable'), + text: this.formatMessage('ArchiveCodeExt.ifConvertedSuccessfully'), }, { //返回容器中数据数量 opcode: 'getAmount', blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getAmount'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } }, { //获取第n(从1开始)个内容,的(1名称2内容3类型4列表长度) @@ -209,6 +264,11 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContentByNumber'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, index: { type: 'number', //menu: 'varMenu2', @@ -232,6 +292,11 @@ class Archive_code { blockType: 'Boolean', text: this.formatMessage('ArchiveCodeExt.ifExist'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -245,6 +310,11 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContent'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -258,6 +328,11 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getLengthOfList'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -271,6 +346,11 @@ class Archive_code { blockType: 'reporter', text: this.formatMessage('ArchiveCodeExt.getContentOfList'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -289,6 +369,11 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.saveContentToVar'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'varMenu2', @@ -306,6 +391,11 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.saveContentToList'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, key: { type: 'string', //menu: 'listMenu2', @@ -323,12 +413,35 @@ class Archive_code { blockType: 'command', text: this.formatMessage('ArchiveCodeExt.delete'), arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + key: { + type: 'string', + defaultValue: '金币' + }, + } + }, + "---" + this.formatMessage("ArchiveCodeExt.info2"), //🔧 JSON处理 + { + //直接获得container容器中的key内容 + opcode: 'getContentInContainer', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContentInContainer'), + arguments: { + container: { + type: 'string', + defaultValue: '{"金币":100,"经验值":50}' + }, key: { type: 'string', defaultValue: '金币' }, } }, + "---" + this.formatMessage("ArchiveCodeExt.info3"), //💡 附加积木 { //加密 opcode: 'encrypt', @@ -407,6 +520,10 @@ class Archive_code { }, ], menus: { + containerMenu: { + items: 'findAllContainer', + acceptReporters: true, + }, varMenu: { items: 'findAllVar' }, @@ -429,9 +546,12 @@ class Archive_code { // text: this.formatMessage('ArchiveCodeExt.methodMenu.2'), // value: '2' // }, - ], - infoMenu: [ { + text: this.formatMessage('ArchiveCodeExt.methodMenu.3'), + value: '3' + }, + ], + infoMenu: [{ text: this.formatMessage('ArchiveCodeExt.infoMenu.1'), value: '1' }, @@ -451,153 +571,242 @@ class Archive_code { }, }; } - init() { - // console.log('editingTarget :', this.runtime._editingTarget) - // console.log('stageTarget :', this.runtime._stageTarget) - // console.log('_stageTarget.variables',JSON.stringify(this.runtime._stageTarget.variables)) - this.content = {}; - this.setArchive_code(); - } - result() { - return this.archive_code; - } - // showContent2json() { - // this.setContent2json(); - // return this.content2json; - // } - stop() { - this.setArchive_code(); - } - serialization(args) { - // if (this.archive_code !== '') - // this.archive_code += ',' - // this.archive_code += `"${args.name}":${JSON.stringify(args.value)}` - this.content[args.name] = args.value; - } - serializationForVariable(args, util) { - // if (this.archive_code !== '') - // this.archive_code += ',' - // const variable = util.target.lookupVariableById(args.var); - // this.archive_code += `"${args.name}":${JSON.stringify(variable.value)}` + + findAllContainer() { + const list = []; + let temp = this.content; + Object.keys(temp).forEach(obj => { + //if ( Array.isArray (temp[obj]) ) { + list.push(obj); + //} + }); + if (list.length === 0) { + list.push({ + text: '-', + value: 'empty', + }); + } + return list; + } + + _createContainerIfNotExist(con){ + if(!(con in this.content)) + this.content[con]={}; + } + + clearContainer(args) { + this._createContainerIfNotExist(args.con) + this.content[args.con] = {}; + } + + containerToJSON(args) { + if(!(args.con in this.content)) return ''; + return JSON.stringify(this.content[args.con]); + } + + addContentToContainer(args) { + if(!(args.con in this.content)) return; + this.content[args.con][args.name] = args.value; + } + + addVariableToContainer(args, util) { + if(!(args.con in this.content)) return; if (args.var !== 'empty') { const variable = util.target.lookupVariableById(args.var); - this.content[args.name] = variable.value; + this.content[args.con][args.name] = variable.value; } } - serializationForList(args, util) { - // if (this.archive_code !== '') - // this.archive_code += ',' - // const list = util.target.lookupVariableById(args.list); - // this.archive_code += `"${args.name}":${JSON.stringify(list)}` + + addListToContainer(args, util) { + if(!(args.con in this.content)) return; if (args.list !== 'empty') { const list = util.target.lookupVariableById(args.list); - this.content[args.name] = list.value; + this.content[args.con][args.name] = list.value; } } - - deserialization(args) { + + //直接获得{container}中的key内容 + getContentInContainer(args) { + let content; try { - this.content = JSON.parse(String(args.code)) - this.deserializeSuccessfully = true; - this.content = JSON.parse(args.code) - if(typeof(this.content)==='object'&& this.content !== null ){ - this.deserializeSuccessfully = true; + content = JSON.parse(Cast.toString(args.container)) + if(typeof(content) === 'object' && !Array.isArray(content) && content !== null) { + return this._anythingToNumberString(content[args.key]); }else{ - this.content = {}; - this.deserializeSuccessfully = false; + return '' + } + } catch (e) { + return '' + } + } + + parseJSONToContainer(args) { + this._createContainerIfNotExist(args.con) + let content; + this.convertedSuccessfully = false; + try { + // 如果解析失败,不要修改content。 + content = JSON.parse(Cast.toString(args.code)) + // 考虑数组[]情况。 + if(typeof(content) === 'object' && !Array.isArray(content) && content !== null) { + this.content[args.con] = content; + this.convertedSuccessfully = true; } } catch (e) { - this.deserializeSuccessfully = false; //this.content2 = {} } //console.log(typeof this.content) } - deserializable() { - return this.deserializeSuccessfully + + ifConvertedSuccessfully() { + return this.convertedSuccessfully } + ifExist(args) { - return (this.content.hasOwnProperty(args.key)); + if(!(args.con in this.content)) return false; + return Cast.toString(args.key) in this.content[args.con]; } - getAmount() { - let count = 0; - for (let key in this.content) { - count++; - } - return count; + + getAmount(args) { + if(!(args.con in this.content)) return ''; + return Object.keys(this.content[args.con]).length; } + getContentByNumber(args) { - let key = Object.keys(this.content)[args.index - 1] + if(!(args.con in this.content)) return ''; + let key = Object.keys(this.content[args.con])[args.index - 1] if (key === undefined) return ''; + let value = this.content[args.con][key] switch (args.type) { case '1'://名称 return key; case '2'://内容 - return this.content[key]; + return this._anythingToNumberString(value); case '3'://类型 - return (typeof this.content[key] === 'object') ? '列表' : '变量'; + switch(typeof value){ + case "object": + // 本地化问题:返回的值是中文 + // 这一点可以统一用英文或者符号或者做成判断<名字为(abc)的数值是列表?> + // 或者符号化 + // 列表 容器 变量 没有 + // List Container Variable Unset + // [] {} "" undefined + return Array.isArray(value) ? '列表' : '容器'; + case "string": + case "number": + case "boolean": + return '变量'; + default: + return ''; + } case '4'://列表长度 - return (typeof this.content[key] === 'object') ? this.content[key].length : ''; + return Array.isArray(value) ? value.length : ''; default: return ''; } + } - getContent(args, util) { - // const variable = util.target.lookupVariableById(args.var); - // variable.value = args.key; - return (this.content[args.key] === undefined) ? '' : String(this.content[args.key]) + + getContent(args) { + if(!(args.con in this.content)) return ''; + return this._anythingToNumberString(this.content[args.con][args.key]); } + getUnicode(args) { - return args.c.charCodeAt(0) + return Cast.toString(args.c).charCodeAt(0) } + getCharByUnicode(args) { - return String.fromCharCode(args.code) - } - getContentOfList(args, util) { - // const variable = util.target.lookupVariableById(args.var); - // variable.value = args.key; - //如果没有这项,或者不是列表 - let t = this.content[args.key] - if (t === undefined || typeof t !== 'object') { - return ''; + return String.fromCharCode(Cast.toNumber(args.code)) + } + + getContentOfList(args) { + if(!(args.con in this.content)) return ''; + let t = this.content[args.con][args.key] + if (Array.isArray(t)) { + let i = Cast.toNumber(args.n) - 1; + if (i < 0 || i >= t.length) { + return ''; + } + return t[i]; } else { - if (t[args.n - 1] === undefined) return ''; - else return t[args.n - 1] + return ''; } } + getLengthOfList(args, util) { - // const variable = util.target.lookupVariableById(args.var); - // variable.value = args.key; - //如果没有这项,或者不是列表 - let t = this.content[args.key] - if (t === undefined || typeof t !== 'object') { - return ''; - } else { - return t.length; + if(!(args.con in this.content)) return ''; + let t = this.content[args.con][args.key] + return Array.isArray(t) ? t.length : ''; + } + + _anythingToNumberString(value) { + switch(typeof(value)){ + case "string": + case "number": + break; + case "object": + if(Array.isArray(value)) { + value = JSON.stringify(value); //列表直接用 JSON 格式显示 + // 在原版scratch中如果直接使用列表作为变量,得到的结果是由空格分隔的。如果列表中每一项都是单个字符(数字不算),则结果不用空格分割。这里还原原版行为。 + // 如果直接String()的话,项目会默认用逗号分割。 + // let areChars = true; + // value.forEach((v, i) => { + // if (!(typeof v === "string" && v.length === 1)) { + // areChars = false; + // } + // }); + // value = value.join(areChars ? '' : ' '); + } else { + // 否则,就直接stringify + value = JSON.stringify(value); + } + break; + default: + value = ''; //包含了undefined } + return value; } + saveContentToVar(args, util) { + if(!(args.con in this.content)) return; if (args.var !== 'empty') { const variable = util.target.lookupVariableById(args.var); - variable.value = String(this.content[args.key]); - if (variable.value === "undefined") variable.value = ''; + let value = this._anythingToNumberString(this.content[args.con][args.key]); + variable.value = value; } } + saveContentToList(args, util) { + if(!(args.con in this.content)) return; if (args.list !== 'empty') { const list = util.target.lookupVariableById(args.list); - list.value = this.content[args.key]; - if (list.value === undefined) list.value = []; - if (typeof list.value !== 'object') { - list.value = [list.value]; + if (!(args.key in this.content[args.con])) { + // 如果啥都没有就清空 + list.value = []; + return; + } + let value = this.content[args.con][args.key]; + if (!Array.isArray(value)) { + //如果要读取的内容不是列表而是什么奇奇怪怪的东西,就把它包装成列表 + value = [value]; } + value.forEach((v, i) => { + // 防止数组内容混入奇奇怪怪的东西 + value[i] = this._anythingToNumberString(v); + }); + list.value = value; } } + delete(args) { - Reflect.deleteProperty(this.content, args.key); + if(!(args.con in this.content)) return; + delete this.content[args.con][args.key]; } + //将密匙转换为一个值 keyVar(k) { - k = String(k) + k = Cast.toString(k) let t = 13; for (let i = 0; i < k.length; i++) { t += k.charCodeAt(i) @@ -605,42 +814,66 @@ class Archive_code { } return t } + //加密 encrypt(args) { + args.key = Cast.toString(args.key) + args.str = Cast.toString(args.str) switch (args.method) { case '1': return this.ArkosEncrypt(args); case '2': return this.encrypt2(args); + case '3': + return LZString.scompress(args.str, args.key); default: return ''; } } + //解密 decrypt(args) { + args.key = Cast.toString(args.key) + args.str = Cast.toString(args.str) switch (args.method) { case '1': return this.ArkosDecrypt(args); case '2': return this.decrypt2(args); + case '3': + return LZString.sdecompress(args.str, args.key); default: return ''; } } + + //发现 Unicode 为 0 10 13 55296~57343(2047个字符) 的字符无法被正常复制,故排除掉这些字符。 + getCode(c) { + c = Cast.toString(c).charCodeAt(0) + if (c === 0) return NaN + else if (c < 10) return c-1 //排除0 + else if (c < 13) return c-2 //排除0 10 + else if (c < 55296) return c-3 //排除0 10 13 + else if (c > 55296) return c-2050 //排除0 10 13 55296~57343(2047个字符) + else return NaN + } + //Arkos加密法 ArkosEncrypt(args) { args.key = this.keyVar(args.key) - args.str = String(args.str) + args.str = Cast.toString(args.str) let b = '' for (let i = 0; i < args.str.length; i++) { b += this.enChar1(args.str[i], args.key + i) } return b } + + //Arkos解密 ArkosDecrypt(args) { args.key = this.keyVar(args.key) - args.str = String(args.str) + args.str = Cast.toString(args.str) let b = '' for (let i = 0; i < args.str.length; i++) { b += this.deChar1(args.str[i], args.key + i) @@ -648,19 +881,24 @@ class Archive_code { //console.log('123') return b } + enChar1(c, p) { + // 目前我知道的unicode字符最大编码是131071 let t = (c.charCodeAt(0) + p) % 54533 // - t = t - t % 10 + (9 - t % 10) + t += 9 - 2 * (t % 10) + return String.fromCharCode(t) } + + deChar1(c, p) { let t = c.charCodeAt(0) - //t%=65536 - t = t - t % 10 + (9 - t % 10) - t = (t - p) % 54533 - if (t < 0) t += 54533 + t += 9 - 2 * (t % 10) + t = (t - p + 54533) % 54533 return String.fromCharCode(t) } + + //分裂加密法 encrypt2(args) { args.key = this.keyVar(args.key) @@ -671,6 +909,7 @@ class Archive_code { } return b } + //分裂解密 decrypt2(args) { args.key = this.keyVar(args.key) @@ -682,13 +921,16 @@ class Archive_code { //console.log('123') return b } + enChar2(c, p) { let t = (c.charCodeAt(0) + p) % 65536 t = t - t % 10 + (9 - t % 10) + let c1 = String.fromCharCode(t >> 8) let c2 = String.fromCharCode(t % 256) return c1 + c2 } + deChar2(c1, c2, p) { let t = c1.charCodeAt(0) * 256 + c2.charCodeAt(0) t %= 65536 @@ -696,9 +938,20 @@ class Archive_code { t = (t - p + 65536) % 65536 return String.fromCharCode(t) } + writeClipboard(args) { - navigator.clipboard.writeText(args.str); + // 错误处理... + if("navigator" in window && "clipboard" in navigator && "writeText" in navigator.clipboard) { + navigator.clipboard.writeText(Cast.toString(args.str)).catch(x => writeClipboard2(args)); + } else { + writeClipboard2(args); + } + } + + writeClipboard2(args) { + prompt("无法访问剪贴板,请选择在下方文字点击右键或按 Ctrl+C 复制。", Cast.toString(args.str)); } + compare(propName) { return (a, b) => { a = a[propName] @@ -708,35 +961,36 @@ class Archive_code { else return 0; } } + findAllVar() { const list = []; - let temp = this.runtime._stageTarget.variables - Object.keys(temp).forEach(obj => { - if (temp[obj].type === '') { - list.push({ - text: `${temp[obj].name}`, - value: temp[obj].id, - }); - } - }); + let temp; try { - temp = this.runtime._editingTarget.variables - } catch (e) { - temp = 'e' - } - if (temp !=='e' && this.runtime._editingTarget !== this.runtime._stageTarget) { + temp = this.runtime._stageTarget.variables Object.keys(temp).forEach(obj => { if (temp[obj].type === '') { list.push({ - text: `[私有变量]${temp[obj].name}`, + text: temp[obj].name, value: temp[obj].id, }); } }); + if (!this.runtime._editingTarget.isStage) { + temp = this.runtime._editingTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === '') { + list.push({ + text: '[私有变量]' + temp[obj].name, + value: temp[obj].id, + }); + } + }); + } + } catch (e) { } if (list.length === 0) { list.push({ - text: `*没有变量*`, + text: `-`, value: 'empty', }); } @@ -750,42 +1004,43 @@ class Archive_code { list.sort(this.compare("text")); return list; } + findAllList() { const list = []; - let temp = this.runtime._stageTarget.variables - Object.keys(temp).forEach(obj => { - if (temp[obj].type === 'list') { - //console.log(temp[obj].type) - list.push({ - text: `${temp[obj].name}`, - value: temp[obj].id, - }); - } - }); + let temp; try { - temp = this.runtime._editingTarget.variables - } catch (e) { - temp = 'e' - } - if (temp !=='e' && this.runtime._editingTarget !== this.runtime._stageTarget) { + temp = this.runtime._stageTarget.variables Object.keys(temp).forEach(obj => { if (temp[obj].type === 'list') { list.push({ - text: `[私有列表]${temp[obj].name}`, + text: temp[obj].name, value: temp[obj].id, }); } }); + if (!this.runtime._editingTarget.isStage) { + temp = this.runtime._editingTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === 'list') { + list.push({ + text: '[私有列表]' + temp[obj].name, + value: temp[obj].id, + }); + } + }); + } + } catch (e) { } if (list.length === 0) { list.push({ - text: `*没有列表*`, + text: `-`, value: 'empty', }); } list.sort(this.compare("text")); return list; } + findAllVarContents() { const list = []; let temp = this.content @@ -799,13 +1054,14 @@ class Archive_code { }); if (list.length === 0) { list.push({ - text: '*没有变量*', + text: '-', value: 'empty', }); } list.sort(this.compare("text")); return list; } + findAllListsContents() { const list = []; let temp = this.content @@ -819,14 +1075,444 @@ class Archive_code { }); if (list.length === 0) { list.push({ - text: '*没有列表*', + text: '-', value: 'empty', }); } list.sort(this.compare("text")); return list; } + + } + + +// - lz-strings.js (modified) - + +// Copyright (c) 2013 Pieroxy +// This work is free. You can redistribute it and/or modify it +// under the terms of the WTFPL, Version 2 +// For more information see LICENSE.txt or http://www.wtfpl.net/ +// +// For more information, the home page: +// http://pieroxy.net/blog/pages/lz-string/testing.html +// +// LZ-based compression algorithm, version 1.4.4 +function _LZString() { + + // private property + var f = String.fromCharCode; + + var LZString = { + scompress: function(uncompressed, key) { + if (key.length !== 0) + key = this.scompress(key, ""); + return this._compress(uncompressed, 14, function(x, i) { + if (key.length !== 0) + x ^= (key.charCodeAt(i % key.length) * (i + 1)) & ((1 << 14) - 1); + return f(x + 0x4E00); + }); + }, + + _compress: function(uncompressed, bitsPerChar, getCharFromInt) { + if (uncompressed == null) return ""; + var i, value, + context_dictionary = {}, + context_dictionaryToCreate = {}, + context_c = "", + context_wc = "", + context_w = "", + context_enlargeIn = 2, // Compensate for the first entry which should not count + context_dictSize = 3, + context_numBits = 2, + context_data = [], + context_data_val = 0, + context_data_position = 0, + ii; + + for (ii = 0; ii < uncompressed.length; ii += 1) { + context_c = uncompressed.charAt(ii); + if (!Object.prototype.hasOwnProperty.call(context_dictionary, context_c)) { + context_dictionary[context_c] = context_dictSize++; + context_dictionaryToCreate[context_c] = true; + } + + context_wc = context_w + context_c; + if (Object.prototype.hasOwnProperty.call(context_dictionary, context_wc)) { + context_w = context_wc; + } else { + if (Object.prototype.hasOwnProperty.call(context_dictionaryToCreate, context_w)) { + if (context_w.charCodeAt(0) < 256) { + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + } + value = context_w.charCodeAt(0); + for (i = 0; i < 8; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } else { + value = 1; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | value; + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = 0; + } + value = context_w.charCodeAt(0); + for (i = 0; i < 16; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + delete context_dictionaryToCreate[context_w]; + } else { + value = context_dictionary[context_w]; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + // Add wc to the dictionary. + context_dictionary[context_wc] = context_dictSize++; + context_w = String(context_c); + } + } + + // Output the code for w. + if (context_w !== "") { + if (Object.prototype.hasOwnProperty.call(context_dictionaryToCreate, context_w)) { + if (context_w.charCodeAt(0) < 256) { + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + } + value = context_w.charCodeAt(0); + for (i = 0; i < 8; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } else { + value = 1; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | value; + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = 0; + } + value = context_w.charCodeAt(0); + for (i = 0; i < 16; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + delete context_dictionaryToCreate[context_w]; + } else { + value = context_dictionary[context_w]; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + } + + // Mark the end of the stream + value = 2; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + // Flush the last char + while (true) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data.push(getCharFromInt(context_data_val, context_data.length)); + break; + } else context_data_position++; + } + return context_data.join(''); + }, + + sdecompress: function(compressed, key) { + if (key.length !== 0) + key = this.scompress(key, ""); + return this._decompress(compressed.length, 1 << (14 - 1), function(i) { + let x = compressed.charCodeAt(i) - 0x4E00; + if (key.length !== 0) + x ^= (key.charCodeAt(i % key.length) * (i + 1)) & ((1 << 14) - 1); + return x; + }); + }, + + _decompress: function(length, resetValue, getNextValue) { + var dictionary = [], + next, + enlargeIn = 4, + dictSize = 4, + numBits = 3, + entry = "", + result = [], + i, + w, + bits, resb, maxpower, power, + c, + data = { + val: getNextValue(0), + position: resetValue, + index: 1 + }; + + for (i = 0; i < 3; i += 1) { + dictionary[i] = i; + } + + bits = 0; + maxpower = Math.pow(2, 2); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + switch (next = bits) { + case 0: + bits = 0; + maxpower = Math.pow(2, 8); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + c = f(bits); + break; + case 1: + bits = 0; + maxpower = Math.pow(2, 16); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + c = f(bits); + break; + case 2: + return ""; + } + dictionary[3] = c; + w = c; + result.push(c); + while (true) { + if (data.index > length) { + return ""; + } + + bits = 0; + maxpower = Math.pow(2, numBits); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + switch (c = bits) { + case 0: + bits = 0; + maxpower = Math.pow(2, 8); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + dictionary[dictSize++] = f(bits); + c = dictSize - 1; + enlargeIn--; + break; + case 1: + bits = 0; + maxpower = Math.pow(2, 16); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + dictionary[dictSize++] = f(bits); + c = dictSize - 1; + enlargeIn--; + break; + case 2: + return result.join(''); + } + + if (enlargeIn == 0) { + enlargeIn = Math.pow(2, numBits); + numBits++; + } + + if (dictionary[c]) { + entry = dictionary[c]; + } else { + if (c === dictSize) { + entry = w + w.charAt(0); + } else { + // throw new Error("LZString: decompress error."); + // return null; + return ""; + } + } + result.push(entry); + + // Add w+entry[0] to the dictionary. + dictionary[dictSize++] = w + entry.charAt(0); + enlargeIn--; + + w = entry; + + if (enlargeIn == 0) { + enlargeIn = Math.pow(2, numBits); + numBits++; + } + + } + } + }; + return LZString; +} + + window.tempExt = { Extension: Archive_code, info: { From 5941ed9bbc2d5d8c618a088b0d33bef2ff727ba0 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Wed, 20 Jul 2022 01:43:10 +0800 Subject: [PATCH 061/357] =?UTF-8?q?=E4=B8=8A=E4=BC=A0=E6=96=B0=E7=89=88?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Archivecode\346\226\260\347\211\210.js" | 1533 +++++++++++++++++ 1 file changed, 1533 insertions(+) create mode 100644 "Arkos/Archivecode\346\226\260\347\211\210.js" diff --git "a/Arkos/Archivecode\346\226\260\347\211\210.js" "b/Arkos/Archivecode\346\226\260\347\211\210.js" new file mode 100644 index 00000000..6255b066 --- /dev/null +++ "b/Arkos/Archivecode\346\226\260\347\211\210.js" @@ -0,0 +1,1533 @@ +import Cast from '../utils/cast.js' +//import cover from './assets/cover2.png' +//import icon from './assets/icon2.svg' + +//合作者: +// Nights: 搭了框架,以及一些技术帮助 +// -6: 修复了许多纰漏和 bug ,加入了新的加密算法 +// Arkos: 什么都不会的屑蒟蒻 +// console.log(Cast.toNumber('123')) +//console.log(Cast.toNumber('aab')) + +let LZString = _LZString(); + +class Archive_code { + constructor(runtime) { + this.runtime = runtime + this.convertedSuccessfully = false //JSON转容器是否成功 + //content为一个大容器,内部存很多小容器 + this.content = { + "1":{ + 金币: 200, + 背包: ["木头", 233] + }, + "2":{ + Arkos:{score:95}, + yk1boy:{score:100} + } + } + + this._formatMessage = runtime.getFormatMessage({ + 'zh-cn': { + 'ArchiveCodeExt.extensionName': '存档码', + 'ArchiveCodeExt.info1': '🏺 容器操作', + 'ArchiveCodeExt.info2': '🔧 JSON处理', + 'ArchiveCodeExt.info3': '💡 附加积木', + 'ArchiveCodeExt.clearContainer': '清空容器[con]', + 'ArchiveCodeExt.addContentToContainer': '将内容[value]命名为[name]加入容器[con](已有则覆盖)', + 'ArchiveCodeExt.addVariableToContainer': '将变量[var]内容命名为[name]加入容器[con](已有则覆盖)', + 'ArchiveCodeExt.addListToContainer': '将列表[list]内容命名为[name]加入容器[con](已有则覆盖)', + //'ArchiveCodeExt.stop': '序列化结束', + 'ArchiveCodeExt.containerToJSON': '容器[con]内容对应字符串', + 'ArchiveCodeExt.parseJSONToContainer': '读取字符串到容器[con]:[code]', + 'ArchiveCodeExt.getContent': '容器[con]中名称为[key]的内容', + //'ArchiveCodeExt.showContent2json':'读取结果', + 'ArchiveCodeExt.saveContentToVar': '将容器[con]中名称为[key]的内容保存到变量[var]', + 'ArchiveCodeExt.saveContentToList': '将容器[con]中名称为[key]的内容保存到列表[list]', + 'ArchiveCodeExt.ifConvertedSuccessfully': '读取成功?', + 'ArchiveCodeExt.getAmount': '容器[con]中内容的总数', + 'ArchiveCodeExt.ifExist': '容器[con]是否存在名为[key]的内容', + 'ArchiveCodeExt.getContentByNumber': '获取容器[con]中第[index]个内容的[type]', + 'ArchiveCodeExt.encrypt': '以[method]加密[str],密匙[key]', + 'ArchiveCodeExt.decrypt': '以[method]解密[str],密匙[key]', + 'ArchiveCodeExt.writeClipboard': '复制[str]到剪贴板', + 'ArchiveCodeExt.getLengthOfList': '容器[con]中名称为[key]的列表的长度', + 'ArchiveCodeExt.getContentOfList': '容器[con]中名称为[key]的列表的第[n]项', + 'ArchiveCodeExt.getUnicode': '字符[c]的Unicode', + 'ArchiveCodeExt.getCharByUnicode': 'Unicode[code]对应字符', + 'ArchiveCodeExt.methodMenu.1': 'Arkos加密法', + 'ArchiveCodeExt.methodMenu.2': '分裂加密法', + 'ArchiveCodeExt.methodMenu.3': 'LZString压缩', + 'ArchiveCodeExt.infoMenu.1': '名称', + 'ArchiveCodeExt.infoMenu.2': '内容', + 'ArchiveCodeExt.infoMenu.3': '类型', + 'ArchiveCodeExt.infoMenu.4': '列表长度', + 'ArchiveCodeExt.delete': '删除容器[con]中名为[key]的内容', + 'ArchiveCodeExt.getContentInContainer': '获得容器[container]中名为[key]的内容', + }, + + en: { + 'ArchiveCodeExt.extensionName': 'Archive Code', + 'ArchiveCodeExt.clearContainer': 'empty container[con]', + 'ArchiveCodeExt.addContentToContainer': 'add content[value] to container[con], name as[name]', + 'ArchiveCodeExt.addVariableToContainer': 'add variable[var] to container[con], name as[name]', + 'ArchiveCodeExt.addListToContainer': 'add list[list] to container[con], name as[name]', + //'ArchiveCodeExt.stop': 'end serialization', + 'ArchiveCodeExt.containerToJSON': 'container[con] in string form', + 'ArchiveCodeExt.parseJSONToContainer': 'parse string[code] to container[con]', + 'ArchiveCodeExt.getContent': 'content of[key]', + 'ArchiveCodeExt.saveContentToVar': 'save[key]in container[con]to variable[var]', + 'ArchiveCodeExt.saveContentToList': 'save[key]in container[con]to list[list]', + 'ArchiveCodeExt.ifConvertedSuccessfully': 'parse successfullly?', + 'ArchiveCodeExt.getAmount': 'the amount of contents in container[con]', + 'ArchiveCodeExt.ifExist': 'container[con] contains[key]?', + 'ArchiveCodeExt.getContentByNumber': 'get [type]of #[index]content in container[con]', + 'ArchiveCodeExt.encrypt': 'use[method]to encrypt[str]with key[key]', + 'ArchiveCodeExt.decrypt': 'use[method]to decrypt[str]with key[key]', + 'ArchiveCodeExt.writeClipboard': 'copy[str]to clipboard', + 'ArchiveCodeExt.getContentOfList': '#[n] of list[key]in container[con]', + 'ArchiveCodeExt.getLengthOfList': 'length of list[key]in container[con]', + 'ArchiveCodeExt.getUnicode': 'get Unicode of[c]', + 'ArchiveCodeExt.getCharByUnicode': ' character of Unicode[code]', + 'ArchiveCodeExt.methodMenu.1': 'Arkos cipher', + 'ArchiveCodeExt.methodMenu.2': 'mitotic encryption', + 'ArchiveCodeExt.methodMenu.3': 'LZString', + 'ArchiveCodeExt.infoMenu.1': 'name', + 'ArchiveCodeExt.infoMenu.2': 'value', + 'ArchiveCodeExt.infoMenu.3': 'type', + 'ArchiveCodeExt.infoMenu.4': 'lenth of list', + 'ArchiveCodeExt.delete': 'delete content[key] in container[con]', + 'ArchiveCodeExt.getContentIncontainer': 'get [key] in [container]', + }, + }) + + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id, + }) + } + + getInfo() { + return { + id: 'Archivecode', // Archivecode + name: this.formatMessage('ArchiveCodeExt.extensionName'), + color1: '#2FBC95', + // menuIconURI: cover, + // blockIconURI: icon, + blocks: [ + "---" + this.formatMessage("ArchiveCodeExt.info1"), //🏺容器操作 + { + //清空容器 + opcode: 'clearContainer', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.clearContainer'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } + }, + { + //将内容加入容器 名称xx 值xx + opcode: 'addContentToContainer', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.addContentToContainer'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + name: { + type: 'string', + defaultValue: '用户id' + }, + value: { + type: 'string', + defaultValue: '114514', + } + } + }, + { + //将变量加入序列 + opcode: 'addVariableToContainer', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.addVariableToContainer'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + name: { + type: 'string', + defaultValue: '金币' + }, + var: { + type: 'string', + menu: 'varMenu' + } + } + }, + { + //将列表加入序列 + opcode: 'addListToContainer', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.addListToContainer'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + name: { + type: 'string', + defaultValue: '背包' + }, + list: { + type: 'string', + menu: 'listMenu' + } + } + }, + // { + // //序列化结束 + // opcode: 'stop', + // blockType: 'command', + // text: this.formatMessage('ArchiveCodeExt.stop'), + // }, + { + //返回序列化结果 + opcode: 'containerToJSON', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.containerToJSON'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } + }, + + + { + //反序列化 + opcode: 'parseJSONToContainer', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.parseJSONToContainer'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + code: { + type: 'string', + defaultValue: `{"金币":200,"背包":["木头","面包"]}` + } + } + }, + { + //反序列化是否成功 + opcode: 'ifConvertedSuccessfully', + blockType: 'Boolean', + text: this.formatMessage('ArchiveCodeExt.ifConvertedSuccessfully'), + }, + { + //返回容器中数据数量 + opcode: 'getAmount', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getAmount'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + } + }, + { + //获取第n(从1开始)个内容,的(1名称2内容3类型4列表长度) + opcode: 'getContentByNumber', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContentByNumber'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + index: { + type: 'number', + //menu: 'varMenu2', + defaultValue: '1' + }, + type: { + type: 'number', + menu: 'infoMenu', + } + } + }, + // { + // //返回反序列化结果 + // opcode: 'showContent2json', + // blockType: 'reporter', + // text: this.formatMessage('ArchiveCodeExt.showContent2json'), + // }, + { + //返回名称为..的内容 + opcode: 'ifExist', + blockType: 'Boolean', + text: this.formatMessage('ArchiveCodeExt.ifExist'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '金币' + } + } + }, + { + //返回名称为..的内容 + opcode: 'getContent', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContent'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '金币' + } + } + }, + { + //返回名称为..的列表的第n项 + opcode: 'getLengthOfList', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getLengthOfList'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '背包' + }, + } + }, + { + //返回名称为..的列表的第n项 + opcode: 'getContentOfList', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContentOfList'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '背包' + }, + n: { + type: 'number', + //menu: 'varMenu2', + defaultValue: '1' + } + } + }, + { + //将内容保存到变量 + opcode: 'saveContentToVar', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.saveContentToVar'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '金币' + }, + var: { + type: 'string', + menu: 'varMenu' + } + } + }, + { + //将内容保存到列表 + opcode: 'saveContentToList', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.saveContentToList'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + key: { + type: 'string', + //menu: 'listMenu2', + defaultValue: '背包' + }, + list: { + type: 'string', + menu: 'listMenu' + } + } + }, + { + //删除内容 + opcode: 'delete', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.delete'), + arguments: { + con: { + type: 'string', + defaultValue: '1', + menu: 'containerMenu' + }, + key: { + type: 'string', + defaultValue: '金币' + }, + } + }, + "---" + this.formatMessage("ArchiveCodeExt.info2"), //🔧 JSON处理 + { + //直接获得container容器中的key内容 + opcode: 'getContentInContainer', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContentInContainer'), + arguments: { + container: { + type: 'string', + defaultValue: '{"金币":100,"经验值":50}' + }, + key: { + type: 'string', + defaultValue: '金币' + }, + } + }, + "---" + this.formatMessage("ArchiveCodeExt.info3"), //💡 附加积木 + { + //加密 + opcode: 'encrypt', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.encrypt'), + arguments: { + str: { + type: 'string', + defaultValue: '我好帅114514' + }, + key: { + type: 'string', + defaultValue: 'Arkos' + }, + method: { + type: 'string', + menu: 'methodMenu' + }, + } + }, + { + //解密 + opcode: 'decrypt', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.decrypt'), + arguments: { + str: { + type: 'string', + defaultValue: '搧宊怑ȼȻɋɉɌɈ' + }, + key: { + type: 'string', + defaultValue: 'Arkos' + }, + method: { + type: 'string', + menu: 'methodMenu' + }, + } + }, + { + //复制到剪切板 + opcode: 'writeClipboard', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.writeClipboard'), + arguments: { + str: { + type: 'string', + defaultValue: '要复制的东西' + } + } + }, + { + //获取字符unicode + opcode: 'getUnicode', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getUnicode'), + arguments: { + c: { + type: 'string', + defaultValue: 'A' + } + } + }, + { + //由unicode得到字符 + opcode: 'getCharByUnicode', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getCharByUnicode'), + arguments: { + code: { + type: 'string', + defaultValue: '65' + } + } + }, + ], + menus: { + containerMenu: { + items: 'findAllContainer', + acceptReporters: true, + }, + varMenu: { + items: 'findAllVar' + }, + listMenu: { + items: 'findAllList' + }, + varMenu2: { + //解析后得到的列表 + items: 'findAllVarContents' + }, + listMenu2: { + items: 'findAllListsContents' + }, + methodMenu: [ + { + text: this.formatMessage('ArchiveCodeExt.methodMenu.1'), + value: '1' + }, + // { + // text: this.formatMessage('ArchiveCodeExt.methodMenu.2'), + // value: '2' + // }, + { + text: this.formatMessage('ArchiveCodeExt.methodMenu.3'), + value: '3' + }, + ], + infoMenu: [{ + text: this.formatMessage('ArchiveCodeExt.infoMenu.1'), + value: '1' + }, + { + text: this.formatMessage('ArchiveCodeExt.infoMenu.2'), + value: '2' + }, + { + text: this.formatMessage('ArchiveCodeExt.infoMenu.3'), + value: '3' + }, + { + text: this.formatMessage('ArchiveCodeExt.infoMenu.4'), + value: '4' + }, + ] + }, + }; + } + + findAllContainer() { + const list = []; + let temp = this.content; + Object.keys(temp).forEach(obj => { + //if ( Array.isArray (temp[obj]) ) { + list.push(obj); + //} + }); + if (list.length === 0) { + list.push({ + text: '-', + value: 'empty', + }); + } + return list; + } + + _createContainerIfNotExist(con){ + if(!(con in this.content)) + this.content[con]={}; + } + + clearContainer(args) { + this._createContainerIfNotExist(args.con) + this.content[args.con] = {}; + } + + containerToJSON(args) { + if(!(args.con in this.content)) return ''; + return JSON.stringify(this.content[args.con]); + } + + addContentToContainer(args) { + if(!(args.con in this.content)) return; + this.content[args.con][args.name] = args.value; + } + + addVariableToContainer(args, util) { + if(!(args.con in this.content)) return; + if (args.var !== 'empty') { + const variable = util.target.lookupVariableById(args.var); + this.content[args.con][args.name] = variable.value; + } + } + + addListToContainer(args, util) { + if(!(args.con in this.content)) return; + if (args.list !== 'empty') { + const list = util.target.lookupVariableById(args.list); + this.content[args.con][args.name] = list.value; + } + } + + //直接获得{container}中的key内容 + getContentInContainer(args) { + let content; + try { + content = JSON.parse(Cast.toString(args.container)) + if(typeof(content) === 'object' && !Array.isArray(content) && content !== null) { + return this._anythingToNumberString(content[args.key]); + }else{ + return '' + } + } catch (e) { + return '' + } + } + + parseJSONToContainer(args) { + this._createContainerIfNotExist(args.con) + let content; + this.convertedSuccessfully = false; + try { + // 如果解析失败,不要修改content。 + content = JSON.parse(Cast.toString(args.code)) + // 考虑数组[]情况。 + if(typeof(content) === 'object' && !Array.isArray(content) && content !== null) { + this.content[args.con] = content; + this.convertedSuccessfully = true; + } + } catch (e) { + //this.content2 = {} + } + //console.log(typeof this.content) + } + + ifConvertedSuccessfully() { + return this.convertedSuccessfully + } + + ifExist(args) { + if(!(args.con in this.content)) return false; + return Cast.toString(args.key) in this.content[args.con]; + } + + getAmount(args) { + if(!(args.con in this.content)) return ''; + return Object.keys(this.content[args.con]).length; + } + + getContentByNumber(args) { + if(!(args.con in this.content)) return ''; + let key = Object.keys(this.content[args.con])[args.index - 1] + if (key === undefined) return ''; + let value = this.content[args.con][key] + switch (args.type) { + case '1'://名称 + return key; + case '2'://内容 + return this._anythingToNumberString(value); + case '3'://类型 + switch(typeof value){ + case "object": + // 本地化问题:返回的值是中文 + // 这一点可以统一用英文或者符号或者做成判断<名字为(abc)的数值是列表?> + // 或者符号化 + // 列表 容器 变量 没有 + // List Container Variable Unset + // [] {} "" undefined + return Array.isArray(value) ? '列表' : '容器'; + case "string": + case "number": + case "boolean": + return '变量'; + default: + return ''; + } + case '4'://列表长度 + return Array.isArray(value) ? value.length : ''; + default: + return ''; + } + + } + + getContent(args) { + if(!(args.con in this.content)) return ''; + return this._anythingToNumberString(this.content[args.con][args.key]); + } + + getUnicode(args) { + return Cast.toString(args.c).charCodeAt(0) + } + + getCharByUnicode(args) { + return String.fromCharCode(Cast.toNumber(args.code)) + } + + getContentOfList(args) { + if(!(args.con in this.content)) return ''; + let t = this.content[args.con][args.key] + if (Array.isArray(t)) { + let i = Cast.toNumber(args.n) - 1; + if (i < 0 || i >= t.length) { + return ''; + } + return t[i]; + } else { + return ''; + } + } + + getLengthOfList(args, util) { + if(!(args.con in this.content)) return ''; + let t = this.content[args.con][args.key] + return Array.isArray(t) ? t.length : ''; + } + + _anythingToNumberString(value) { + switch(typeof(value)){ + case "string": + case "number": + break; + case "object": + if(Array.isArray(value)) { + value = JSON.stringify(value); //列表直接用 JSON 格式显示 + // 在原版scratch中如果直接使用列表作为变量,得到的结果是由空格分隔的。如果列表中每一项都是单个字符(数字不算),则结果不用空格分割。这里还原原版行为。 + // 如果直接String()的话,项目会默认用逗号分割。 + // let areChars = true; + // value.forEach((v, i) => { + // if (!(typeof v === "string" && v.length === 1)) { + // areChars = false; + // } + // }); + // value = value.join(areChars ? '' : ' '); + } else { + // 否则,就直接stringify + value = JSON.stringify(value); + } + break; + default: + value = ''; //包含了undefined + } + return value; + } + + saveContentToVar(args, util) { + if(!(args.con in this.content)) return; + if (args.var !== 'empty') { + const variable = util.target.lookupVariableById(args.var); + let value = this._anythingToNumberString(this.content[args.con][args.key]); + variable.value = value; + } + } + + saveContentToList(args, util) { + if(!(args.con in this.content)) return; + if (args.list !== 'empty') { + const list = util.target.lookupVariableById(args.list); + if (!(args.key in this.content[args.con])) { + // 如果啥都没有就清空 + list.value = []; + return; + } + let value = this.content[args.con][args.key]; + if (!Array.isArray(value)) { + //如果要读取的内容不是列表而是什么奇奇怪怪的东西,就把它包装成列表 + value = [value]; + } + value.forEach((v, i) => { + // 防止数组内容混入奇奇怪怪的东西 + value[i] = this._anythingToNumberString(v); + }); + list.value = value; + } + } + + delete(args) { + if(!(args.con in this.content)) return; + delete this.content[args.con][args.key]; + } + + //将密匙转换为一个值 + keyVar(k) { + k = Cast.toString(k) + let t = 13; + for (let i = 0; i < k.length; i++) { + t += k.charCodeAt(i) + t %= 65536 + } + return t + } + + //加密 + encrypt(args) { + args.key = Cast.toString(args.key) + args.str = Cast.toString(args.str) + switch (args.method) { + case '1': + return this.ArkosEncrypt(args); + case '2': + return this.encrypt2(args); + case '3': + return LZString.scompress(args.str, args.key); + default: + return ''; + } + } + + //解密 + decrypt(args) { + args.key = Cast.toString(args.key) + args.str = Cast.toString(args.str) + switch (args.method) { + case '1': + return this.ArkosDecrypt(args); + case '2': + return this.decrypt2(args); + case '3': + return LZString.sdecompress(args.str, args.key); + default: + return ''; + } + } + + //发现 Unicode 为 0 10 13 55296~57343(2047个字符) 的字符无法被正常复制,故排除掉这些字符。 + getCode(c) { + c = Cast.toString(c).charCodeAt(0) + if (c === 0) return NaN + else if (c < 10) return c-1 //排除0 + else if (c < 13) return c-2 //排除0 10 + else if (c < 55296) return c-3 //排除0 10 13 + else if (c > 55296) return c-2050 //排除0 10 13 55296~57343(2047个字符) + else return NaN + } + + //Arkos加密法 + ArkosEncrypt(args) { + args.key = this.keyVar(args.key) + args.str = Cast.toString(args.str) + let b = '' + for (let i = 0; i < args.str.length; i++) { + b += this.enChar1(args.str[i], args.key + i) + } + return b + } + + + //Arkos解密 + ArkosDecrypt(args) { + args.key = this.keyVar(args.key) + args.str = Cast.toString(args.str) + let b = '' + for (let i = 0; i < args.str.length; i++) { + b += this.deChar1(args.str[i], args.key + i) + } + //console.log('123') + return b + } + + enChar1(c, p) { + // 目前我知道的unicode字符最大编码是131071 + let t = (c.charCodeAt(0) + p) % 54533 // + t += 9 - 2 * (t % 10) + + return String.fromCharCode(t) + } + + + deChar1(c, p) { + let t = c.charCodeAt(0) + t += 9 - 2 * (t % 10) + t = (t - p + 54533) % 54533 + return String.fromCharCode(t) + } + + + //分裂加密法 + encrypt2(args) { + args.key = this.keyVar(args.key) + args.str = String(args.str) + let b = '' + for (let i = 0; i < args.str.length; i++) { + b += this.enChar2(args.str[i], args.key + i) + } + return b + } + + //分裂解密 + decrypt2(args) { + args.key = this.keyVar(args.key) + args.str = String(args.str) + let b = '' + for (let i = 0; i < args.str.length; i += 2) { + b += this.deChar2(args.str[i], (i + 2 > args.str.length) ? '\0' : args.str[i + 1], args.key + i / 2) + } + //console.log('123') + return b + } + + enChar2(c, p) { + let t = (c.charCodeAt(0) + p) % 65536 + t = t - t % 10 + (9 - t % 10) + + let c1 = String.fromCharCode(t >> 8) + let c2 = String.fromCharCode(t % 256) + return c1 + c2 + } + + deChar2(c1, c2, p) { + let t = c1.charCodeAt(0) * 256 + c2.charCodeAt(0) + t %= 65536 + t = t - t % 10 + (9 - t % 10) + t = (t - p + 65536) % 65536 + return String.fromCharCode(t) + } + + writeClipboard(args) { + // 错误处理... + if("navigator" in window && "clipboard" in navigator && "writeText" in navigator.clipboard) { + navigator.clipboard.writeText(Cast.toString(args.str)).catch(x => writeClipboard2(args)); + } else { + writeClipboard2(args); + } + } + + writeClipboard2(args) { + prompt("无法访问剪贴板,请选择在下方文字点击右键或按 Ctrl+C 复制。", Cast.toString(args.str)); + } + + compare(propName) { + return (a, b) => { + a = a[propName] + b = b[propName] + if (a > b) return 1; + else if (a < b) return -1; + else return 0; + } + } + + findAllVar() { + const list = []; + let temp; + try { + temp = this.runtime._stageTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === '') { + list.push({ + text: temp[obj].name, + value: temp[obj].id, + }); + } + }); + if (!this.runtime._editingTarget.isStage) { + temp = this.runtime._editingTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === '') { + list.push({ + text: '[私有变量]' + temp[obj].name, + value: temp[obj].id, + }); + } + }); + } + } catch (e) { + } + if (list.length === 0) { + list.push({ + text: `-`, + value: 'empty', + }); + } + // Object.keys(this.runtime._editingTarget.variables).forEach(key => { + // list.forEach((obj) => { + // if (obj.value === key) { + // obj.text = `*${obj.text}`; + // } + // }); + // }); + list.sort(this.compare("text")); + return list; + } + + findAllList() { + const list = []; + let temp; + try { + temp = this.runtime._stageTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === 'list') { + list.push({ + text: temp[obj].name, + value: temp[obj].id, + }); + } + }); + if (!this.runtime._editingTarget.isStage) { + temp = this.runtime._editingTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === 'list') { + list.push({ + text: '[私有列表]' + temp[obj].name, + value: temp[obj].id, + }); + } + }); + } + } catch (e) { + } + if (list.length === 0) { + list.push({ + text: `-`, + value: 'empty', + }); + } + list.sort(this.compare("text")); + return list; + } + + findAllVarContents() { + const list = []; + let temp = this.content + Object.keys(temp).forEach(obj => { + if (typeof temp[obj] !== 'object') { + list.push({ + text: obj, + value: obj, + }); + } + }); + if (list.length === 0) { + list.push({ + text: '-', + value: 'empty', + }); + } + list.sort(this.compare("text")); + return list; + } + + findAllListsContents() { + const list = []; + let temp = this.content + Object.keys(temp).forEach(obj => { + if (typeof temp[obj] === 'object') { + list.push({ + text: obj, + value: obj, + }); + } + }); + if (list.length === 0) { + list.push({ + text: '-', + value: 'empty', + }); + } + list.sort(this.compare("text")); + return list; + } + + +} + + +// - lz-strings.js (modified) - + +// Copyright (c) 2013 Pieroxy +// This work is free. You can redistribute it and/or modify it +// under the terms of the WTFPL, Version 2 +// For more information see LICENSE.txt or http://www.wtfpl.net/ +// +// For more information, the home page: +// http://pieroxy.net/blog/pages/lz-string/testing.html +// +// LZ-based compression algorithm, version 1.4.4 +function _LZString() { + + // private property + var f = String.fromCharCode; + + var LZString = { + scompress: function(uncompressed, key) { + if (key.length !== 0) + key = this.scompress(key, ""); + return this._compress(uncompressed, 14, function(x, i) { + if (key.length !== 0) + x ^= (key.charCodeAt(i % key.length) * (i + 1)) & ((1 << 14) - 1); + return f(x + 0x4E00); + }); + }, + + _compress: function(uncompressed, bitsPerChar, getCharFromInt) { + if (uncompressed == null) return ""; + var i, value, + context_dictionary = {}, + context_dictionaryToCreate = {}, + context_c = "", + context_wc = "", + context_w = "", + context_enlargeIn = 2, // Compensate for the first entry which should not count + context_dictSize = 3, + context_numBits = 2, + context_data = [], + context_data_val = 0, + context_data_position = 0, + ii; + + for (ii = 0; ii < uncompressed.length; ii += 1) { + context_c = uncompressed.charAt(ii); + if (!Object.prototype.hasOwnProperty.call(context_dictionary, context_c)) { + context_dictionary[context_c] = context_dictSize++; + context_dictionaryToCreate[context_c] = true; + } + + context_wc = context_w + context_c; + if (Object.prototype.hasOwnProperty.call(context_dictionary, context_wc)) { + context_w = context_wc; + } else { + if (Object.prototype.hasOwnProperty.call(context_dictionaryToCreate, context_w)) { + if (context_w.charCodeAt(0) < 256) { + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + } + value = context_w.charCodeAt(0); + for (i = 0; i < 8; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } else { + value = 1; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | value; + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = 0; + } + value = context_w.charCodeAt(0); + for (i = 0; i < 16; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + delete context_dictionaryToCreate[context_w]; + } else { + value = context_dictionary[context_w]; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + // Add wc to the dictionary. + context_dictionary[context_wc] = context_dictSize++; + context_w = String(context_c); + } + } + + // Output the code for w. + if (context_w !== "") { + if (Object.prototype.hasOwnProperty.call(context_dictionaryToCreate, context_w)) { + if (context_w.charCodeAt(0) < 256) { + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + } + value = context_w.charCodeAt(0); + for (i = 0; i < 8; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } else { + value = 1; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | value; + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = 0; + } + value = context_w.charCodeAt(0); + for (i = 0; i < 16; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + delete context_dictionaryToCreate[context_w]; + } else { + value = context_dictionary[context_w]; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + + } + context_enlargeIn--; + if (context_enlargeIn == 0) { + context_enlargeIn = Math.pow(2, context_numBits); + context_numBits++; + } + } + + // Mark the end of the stream + value = 2; + for (i = 0; i < context_numBits; i++) { + context_data_val = (context_data_val << 1) | (value & 1); + if (context_data_position == bitsPerChar - 1) { + context_data_position = 0; + context_data.push(getCharFromInt(context_data_val, context_data.length)); + context_data_val = 0; + } else { + context_data_position++; + } + value = value >> 1; + } + + // Flush the last char + while (true) { + context_data_val = (context_data_val << 1); + if (context_data_position == bitsPerChar - 1) { + context_data.push(getCharFromInt(context_data_val, context_data.length)); + break; + } else context_data_position++; + } + return context_data.join(''); + }, + + sdecompress: function(compressed, key) { + if (key.length !== 0) + key = this.scompress(key, ""); + return this._decompress(compressed.length, 1 << (14 - 1), function(i) { + let x = compressed.charCodeAt(i) - 0x4E00; + if (key.length !== 0) + x ^= (key.charCodeAt(i % key.length) * (i + 1)) & ((1 << 14) - 1); + return x; + }); + }, + + _decompress: function(length, resetValue, getNextValue) { + var dictionary = [], + next, + enlargeIn = 4, + dictSize = 4, + numBits = 3, + entry = "", + result = [], + i, + w, + bits, resb, maxpower, power, + c, + data = { + val: getNextValue(0), + position: resetValue, + index: 1 + }; + + for (i = 0; i < 3; i += 1) { + dictionary[i] = i; + } + + bits = 0; + maxpower = Math.pow(2, 2); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + switch (next = bits) { + case 0: + bits = 0; + maxpower = Math.pow(2, 8); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + c = f(bits); + break; + case 1: + bits = 0; + maxpower = Math.pow(2, 16); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + c = f(bits); + break; + case 2: + return ""; + } + dictionary[3] = c; + w = c; + result.push(c); + while (true) { + if (data.index > length) { + return ""; + } + + bits = 0; + maxpower = Math.pow(2, numBits); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + switch (c = bits) { + case 0: + bits = 0; + maxpower = Math.pow(2, 8); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + + dictionary[dictSize++] = f(bits); + c = dictSize - 1; + enlargeIn--; + break; + case 1: + bits = 0; + maxpower = Math.pow(2, 16); + power = 1; + while (power != maxpower) { + resb = data.val & data.position; + data.position >>= 1; + if (data.position == 0) { + data.position = resetValue; + data.val = getNextValue(data.index++); + } + bits |= (resb > 0 ? 1 : 0) * power; + power <<= 1; + } + dictionary[dictSize++] = f(bits); + c = dictSize - 1; + enlargeIn--; + break; + case 2: + return result.join(''); + } + + if (enlargeIn == 0) { + enlargeIn = Math.pow(2, numBits); + numBits++; + } + + if (dictionary[c]) { + entry = dictionary[c]; + } else { + if (c === dictSize) { + entry = w + w.charAt(0); + } else { + // throw new Error("LZString: decompress error."); + // return null; + return ""; + } + } + result.push(entry); + + // Add w+entry[0] to the dictionary. + dictionary[dictSize++] = w + entry.charAt(0); + enlargeIn--; + + w = entry; + + if (enlargeIn == 0) { + enlargeIn = Math.pow(2, numBits); + numBits++; + } + + } + } + }; + return LZString; +} + + +window.tempExt = { + Extension: Archive_code, + info: { + name: 'hcn.extensionName', + description: 'hcn.description', + extensionId: 'Archivecode', + // iconURL: icon, + // insetIconURL: cover, + featured: true, + disabled: false, + collaborator: 'only for hcn test', + }, + l10n: { + 'zh-cn': { + 'hcn.extensionName': 'hcn 的测试', + 'hcn.description': 'hcn 的测试', + }, + en: { + 'hcn.extensionName': 'hcn test', + 'hcn.description': 'hcn test', + }, + }, +} \ No newline at end of file From c5f878aebaa8d14a803e105e578c71c554856e7d Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Wed, 20 Jul 2022 01:44:46 +0800 Subject: [PATCH 062/357] =?UTF-8?q?=E4=B8=8A=E4=BC=A0=E6=97=A7=E7=89=88?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Archivecode\346\227\247\347\211\210.js" | 917 ++++++++++++++++++ 1 file changed, 917 insertions(+) create mode 100644 "Arkos/Archivecode\346\227\247\347\211\210.js" diff --git "a/Arkos/Archivecode\346\227\247\347\211\210.js" "b/Arkos/Archivecode\346\227\247\347\211\210.js" new file mode 100644 index 00000000..65675ced --- /dev/null +++ "b/Arkos/Archivecode\346\227\247\347\211\210.js" @@ -0,0 +1,917 @@ +import Cast from '../utils/cast.js' +//import cover from './assets/cover2.png' +//import icon from './assets/icon2.svg' + +//合作者:Nights:帮忙写了代码框架,并且给予了一些技术上的指导 +// console.log(Cast.toNumber('123')) +//console.log(Cast.toNumber('aab')) + +class Archive_code { + constructor(runtime) { + this.runtime = runtime + this._archive_code = '' //生成的序列结果 + this.deserializeSuccessfully = false //存反序列化是否成功 + //存需要序列化的内容 + this.content = { + 金币: 200, + 背包: ["木头", 233] + } + //存反序列化结果 + //this.content2 = {}; + //Object.assign(this.content2,this.content) + this.setArchive_code(); + //this.setContent2json(); + + this._formatMessage = runtime.getFormatMessage({ + 'zh-cn': { + 'ArchiveCodeExt.extensionName': '存档码', + 'ArchiveCodeExt.init': '清空容器', + 'ArchiveCodeExt.serialization': '将内容[value]命名为[name]加入容器(已有则覆盖)', + 'ArchiveCodeExt.serializationForVariable': '将变量[var]内容命名为[name]加入容器(已有则覆盖)', + 'ArchiveCodeExt.serializationForList': '将列表[list]内容命名为[name]加入容器(已有则覆盖)', + //'ArchiveCodeExt.stop': '序列化结束', + 'ArchiveCodeExt.result': '容器内容对应字符串', + 'ArchiveCodeExt.deserialization': '读取字符串到容器:[code]', + 'ArchiveCodeExt.getContent': '容器中名称为[key]的内容', + //'ArchiveCodeExt.showContent2json':'读取结果', + 'ArchiveCodeExt.saveContentToVar': '将容器中名称为[key]的内容保存到变量[var]', + 'ArchiveCodeExt.saveContentToList': '将容器中名称为[key]的内容保存到列表[list]', + 'ArchiveCodeExt.deserializable': '读取成功?', + 'ArchiveCodeExt.getAmount': '容器中内容的总数', + 'ArchiveCodeExt.ifExist': '容器是否存在名为[key]的内容', + 'ArchiveCodeExt.getContentByNumber': '获取容器中第[index]个内容的[type]', + 'ArchiveCodeExt.encrypt': '以[method]加密[str],密匙[key]', + 'ArchiveCodeExt.decrypt': '以[method]解密[str],密匙[key]', + 'ArchiveCodeExt.writeClipboard': '复制[str]到剪贴板', + 'ArchiveCodeExt.getLengthOfList': '容器中名称为[key]的列表的长度', + 'ArchiveCodeExt.getContentOfList': '容器中名称为[key]的列表的第[n]项', + 'ArchiveCodeExt.getUnicode': '字符[c]的Unicode', + 'ArchiveCodeExt.getCharByUnicode': 'Unicode[code]对应字符', + 'ArchiveCodeExt.methodMenu.1': 'Arkos加密法', + 'ArchiveCodeExt.methodMenu.2': '分裂加密法', + 'ArchiveCodeExt.infoMenu.1': '名称', + 'ArchiveCodeExt.infoMenu.2': '内容', + 'ArchiveCodeExt.infoMenu.3': '类型', + 'ArchiveCodeExt.infoMenu.4': '列表长度', + 'ArchiveCodeExt.delete': '删除容器中名为[key]的内容', + + }, + + en: { + 'ArchiveCodeExt.extensionName': 'Archive Code', + 'ArchiveCodeExt.init': 'empty Container', + 'ArchiveCodeExt.serialization': 'add content[value] to Container, name as[name]', + 'ArchiveCodeExt.serializationForVariable': 'add variable[var] to Container, name as[name]', + 'ArchiveCodeExt.serializationForList': 'add list[list] to Container, name as[name]', + //'ArchiveCodeExt.stop': 'end serialization', + 'ArchiveCodeExt.result': 'Container in string form', + 'ArchiveCodeExt.deserialization': 'parse string[code] to Container', + 'ArchiveCodeExt.getContent': 'content of[key]', + 'ArchiveCodeExt.saveContentToVar': 'save [key]to variable[var]', + 'ArchiveCodeExt.saveContentToList': 'save[key]to list[list]', + 'ArchiveCodeExt.deserializable': 'parse successfullly?', + 'ArchiveCodeExt.getAmount': 'the amount of contents in Container', + 'ArchiveCodeExt.ifExist': 'Container contains[key]?', + 'ArchiveCodeExt.getContentByNumber': 'get [type]of #[index]content', + 'ArchiveCodeExt.encrypt': 'use[method]to encrypt[str]with key[key]', + 'ArchiveCodeExt.decrypt': 'use[method]to decrypt[str]with key[key]', + 'ArchiveCodeExt.writeClipboard': 'copy[str]to clipboard', + //'ArchiveCodeExt.showContent2json':'deserialization result', + 'ArchiveCodeExt.getContentOfList': '#[n] of list[key]', + 'ArchiveCodeExt.getLengthOfList': 'length of list[key]', + 'ArchiveCodeExt.getUnicode': 'get Unicode of[c]', + 'ArchiveCodeExt.getCharByUnicode': ' character of Unicode[code]', + 'ArchiveCodeExt.methodMenu.1': 'Arkos cipher', + 'ArchiveCodeExt.methodMenu.2': 'mitotic encryption', + 'ArchiveCodeExt.infoMenu.1': 'name', + 'ArchiveCodeExt.infoMenu.2': 'value', + 'ArchiveCodeExt.infoMenu.3': 'type', + 'ArchiveCodeExt.infoMenu.4': 'lenth of list', + 'ArchiveCodeExt.delete': 'Delete content[key] in Container' + }, + }) + + } + + //每次读取archive_code会生成一次 + get archive_code() { + this.setArchive_code(); + return this._archive_code; + } + + //根据content的内容,将其JSON化,存到archive_code + setArchive_code() { + this._archive_code = JSON.stringify(this.content); + } + + // setContent2json(){ + // this.content2json = JSON.stringify(this.content2); + // } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id, + }) + } + + getInfo() { + return { + id: 'Archivecode', // Archivecode + name: this.formatMessage('ArchiveCodeExt.extensionName'), + color1: '#2FBC95', + // menuIconURI: cover, + // blockIconURI: icon, + blocks: [ + { + //开始序列化 + opcode: 'init', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.init') + }, + { + //将内容加入序列 名称xx 值xx + opcode: 'serialization', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.serialization'), + arguments: { + name: { + type: 'string', + defaultValue: '用户id' + }, + value: { + type: 'string', + defaultValue: '114514', + } + } + }, + { + //将变量加入序列 + opcode: 'serializationForVariable', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.serializationForVariable'), + arguments: { + name: { + type: 'string', + defaultValue: '金币' + }, + var: { + type: 'string', + menu: 'varMenu' + } + } + }, + { + //将列表加入序列 + opcode: 'serializationForList', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.serializationForList'), + arguments: { + name: { + type: 'string', + defaultValue: '背包' + }, + list: { + type: 'string', + menu: 'listMenu' + } + } + }, + // { + // //序列化结束 + // opcode: 'stop', + // blockType: 'command', + // text: this.formatMessage('ArchiveCodeExt.stop'), + // }, + { + //返回序列化结果 + opcode: 'result', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.result'), + }, + + + { + //反序列化 + opcode: 'deserialization', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.deserialization'), + arguments: { + code: { + type: 'string', + defaultValue: `{"金币":200,"背包":["木头","面包"]}` + } + } + }, + { + //反序列化是否成功 + opcode: 'deserializable', + blockType: 'Boolean', + text: this.formatMessage('ArchiveCodeExt.deserializable'), + }, + { + //返回容器中数据数量 + opcode: 'getAmount', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getAmount'), + }, + { + //获取第n(从1开始)个内容,的(1名称2内容3类型4列表长度) + opcode: 'getContentByNumber', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContentByNumber'), + arguments: { + index: { + type: 'number', + //menu: 'varMenu2', + defaultValue: '1' + }, + type: { + type: 'number', + menu: 'infoMenu', + } + } + }, + // { + // //返回反序列化结果 + // opcode: 'showContent2json', + // blockType: 'reporter', + // text: this.formatMessage('ArchiveCodeExt.showContent2json'), + // }, + { + //返回名称为..的内容 + opcode: 'ifExist', + blockType: 'Boolean', + text: this.formatMessage('ArchiveCodeExt.ifExist'), + arguments: { + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '金币' + } + } + }, + { + //返回名称为..的内容 + opcode: 'getContent', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContent'), + arguments: { + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '金币' + } + } + }, + { + //返回名称为..的列表的第n项 + opcode: 'getLengthOfList', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getLengthOfList'), + arguments: { + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '背包' + }, + } + }, + { + //返回名称为..的列表的第n项 + opcode: 'getContentOfList', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getContentOfList'), + arguments: { + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '背包' + }, + n: { + type: 'number', + //menu: 'varMenu2', + defaultValue: '1' + } + } + }, + { + //将内容保存到变量 + opcode: 'saveContentToVar', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.saveContentToVar'), + arguments: { + key: { + type: 'string', + //menu: 'varMenu2', + defaultValue: '金币' + }, + var: { + type: 'string', + menu: 'varMenu' + } + } + }, + { + //将内容保存到列表 + opcode: 'saveContentToList', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.saveContentToList'), + arguments: { + key: { + type: 'string', + //menu: 'listMenu2', + defaultValue: '背包' + }, + list: { + type: 'string', + menu: 'listMenu' + } + } + }, + { + //删除内容 + opcode: 'delete', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.delete'), + arguments: { + key: { + type: 'string', + defaultValue: '金币' + }, + } + }, + { + //加密 + opcode: 'encrypt', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.encrypt'), + arguments: { + str: { + type: 'string', + defaultValue: '我好帅114514' + }, + key: { + type: 'string', + defaultValue: 'Arkos' + }, + method: { + type: 'string', + menu: 'methodMenu' + }, + } + }, + { + //解密 + opcode: 'decrypt', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.decrypt'), + arguments: { + str: { + type: 'string', + defaultValue: '搧宊怑ȼȻɋɉɌɈ' + }, + key: { + type: 'string', + defaultValue: 'Arkos' + }, + method: { + type: 'string', + menu: 'methodMenu' + }, + } + }, + { + //复制到剪切板 + opcode: 'writeClipboard', + blockType: 'command', + text: this.formatMessage('ArchiveCodeExt.writeClipboard'), + arguments: { + str: { + type: 'string', + defaultValue: '要复制的东西' + } + } + }, + { + //获取字符unicode + opcode: 'getUnicode', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getUnicode'), + arguments: { + c: { + type: 'string', + defaultValue: 'A' + } + } + }, + { + //由unicode得到字符 + opcode: 'getCharByUnicode', + blockType: 'reporter', + text: this.formatMessage('ArchiveCodeExt.getCharByUnicode'), + arguments: { + code: { + type: 'string', + defaultValue: '65' + } + } + }, + ], + menus: { + varMenu: { + items: 'findAllVar' + }, + listMenu: { + items: 'findAllList' + }, + varMenu2: { + //解析后得到的列表 + items: 'findAllVarContents' + }, + listMenu2: { + items: 'findAllListsContents' + }, + methodMenu: [ + { + text: this.formatMessage('ArchiveCodeExt.methodMenu.1'), + value: '1' + }, + // { + // text: this.formatMessage('ArchiveCodeExt.methodMenu.2'), + // value: '2' + // }, + ], + infoMenu: [ + { + text: this.formatMessage('ArchiveCodeExt.infoMenu.1'), + value: '1' + }, + { + text: this.formatMessage('ArchiveCodeExt.infoMenu.2'), + value: '2' + }, + { + text: this.formatMessage('ArchiveCodeExt.infoMenu.3'), + value: '3' + }, + { + text: this.formatMessage('ArchiveCodeExt.infoMenu.4'), + value: '4' + }, + ] + }, + }; + } + + + + init() { + // console.log('editingTarget :', this.runtime._editingTarget) + // console.log('stageTarget :', this.runtime._stageTarget) + // console.log('_stageTarget.variables',JSON.stringify(this.runtime._stageTarget.variables)) + this.content = {}; + this.setArchive_code(); + } + + result() { + return this.archive_code; + } + + // showContent2json() { + // this.setContent2json(); + // return this.content2json; + // } + + stop() { + this.setArchive_code(); + } + + serialization(args) { + // if (this.archive_code !== '') + // this.archive_code += ',' + // this.archive_code += `"${args.name}":${JSON.stringify(args.value)}` + this.content[args.name] = args.value; + } + + serializationForVariable(args, util) { + // if (this.archive_code !== '') + // this.archive_code += ',' + // const variable = util.target.lookupVariableById(args.var); + // this.archive_code += `"${args.name}":${JSON.stringify(variable.value)}` + if (args.var !== 'empty') { + const variable = util.target.lookupVariableById(args.var); + this.content[args.name] = variable.value; + } + } + + serializationForList(args, util) { + // if (this.archive_code !== '') + // this.archive_code += ',' + // const list = util.target.lookupVariableById(args.list); + // this.archive_code += `"${args.name}":${JSON.stringify(list)}` + if (args.list !== 'empty') { + const list = util.target.lookupVariableById(args.list); + this.content[args.name] = list.value; + } + } + + deserialization(args) { + try { + this.content = JSON.parse(args.code) + if(typeof(this.content)==='object'&& this.content !== null ){ + this.deserializeSuccessfully = true; + }else{ + this.content = {}; + this.deserializeSuccessfully = false; + } + } catch (e) { + this.deserializeSuccessfully = false; + //this.content2 = {} + } + //console.log(typeof this.content) + } + + deserializable() { + return this.deserializeSuccessfully + } + + ifExist(args) { + return (this.content.hasOwnProperty(args.key)); + } + + getAmount() { + let count = 0; + for (let key in this.content) { + count++; + } + return count; + } + + getContentByNumber(args) { + let key = Object.keys(this.content)[args.index - 1] + if (key === undefined) return ''; + switch (args.type) { + case '1'://名称 + return key; + case '2'://内容 + return this.content[key]; + case '3'://类型 + return (typeof this.content[key] === 'object') ? '列表' : '变量'; + case '4'://列表长度 + return (typeof this.content[key] === 'object') ? this.content[key].length : ''; + default: + return ''; + } + + } + + getContent(args, util) { + // const variable = util.target.lookupVariableById(args.var); + // variable.value = args.key; + + return (this.content[args.key] === undefined) ? '' : String(this.content[args.key]) + } + + getUnicode(args) { + return args.c.charCodeAt(0) + } + + getCharByUnicode(args) { + return String.fromCharCode(args.code) + } + + getContentOfList(args, util) { + // const variable = util.target.lookupVariableById(args.var); + // variable.value = args.key; + //如果没有这项,或者不是列表 + let t = this.content[args.key] + if (t === undefined || typeof t !== 'object') { + return ''; + } else { + if (t[args.n - 1] === undefined) return ''; + else return t[args.n - 1] + } + } + + getLengthOfList(args, util) { + // const variable = util.target.lookupVariableById(args.var); + // variable.value = args.key; + //如果没有这项,或者不是列表 + let t = this.content[args.key] + if (t === undefined || typeof t !== 'object') { + return ''; + } else { + return t.length; + } + } + + + + saveContentToVar(args, util) { + if (args.var !== 'empty') { + const variable = util.target.lookupVariableById(args.var); + variable.value = String(this.content[args.key]); + if (variable.value === "undefined") variable.value = ''; + } + } + + saveContentToList(args, util) { + if (args.list !== 'empty') { + const list = util.target.lookupVariableById(args.list); + list.value = this.content[args.key]; + if (list.value === undefined) list.value = []; + if (typeof list.value !== 'object') { + list.value = [list.value]; + } + } + } + + delete(args) { + Reflect.deleteProperty(this.content, args.key); + } + + //将密匙转换为一个值 + keyVar(k) { + k = String(k) + let t = 13; + for (let i = 0; i < k.length; i++) { + t += k.charCodeAt(i) + t %= 65536 + } + return t + } + + //加密 + encrypt(args) { + switch (args.method) { + case '1': + return this.ArkosEncrypt(args); + case '2': + return this.encrypt2(args); + default: + return ''; + } + } + + //解密 + decrypt(args) { + switch (args.method) { + case '1': + return this.ArkosDecrypt(args); + case '2': + return this.decrypt2(args); + default: + return ''; + } + } + + //Arkos加密法 + ArkosEncrypt(args) { + args.key = this.keyVar(args.key) + args.str = String(args.str) + let b = '' + for (let i = 0; i < args.str.length; i++) { + b += this.enChar1(args.str[i], args.key + i) + } + return b + } + + + //Arkos解密 + ArkosDecrypt(args) { + args.key = this.keyVar(args.key) + args.str = String(args.str) + let b = '' + for (let i = 0; i < args.str.length; i++) { + b += this.deChar1(args.str[i], args.key + i) + } + //console.log('123') + return b + } + + enChar1(c, p) { + let t = (c.charCodeAt(0) + p) % 54533 // + t = t - t % 10 + (9 - t % 10) + + return String.fromCharCode(t) + } + + + deChar1(c, p) { + let t = c.charCodeAt(0) + //t%=65536 + t = t - t % 10 + (9 - t % 10) + t = (t - p) % 54533 + if (t < 0) t += 54533 + return String.fromCharCode(t) + } + + + //分裂加密法 + encrypt2(args) { + args.key = this.keyVar(args.key) + args.str = String(args.str) + let b = '' + for (let i = 0; i < args.str.length; i++) { + b += this.enChar2(args.str[i], args.key + i) + } + return b + } + + //分裂解密 + decrypt2(args) { + args.key = this.keyVar(args.key) + args.str = String(args.str) + let b = '' + for (let i = 0; i < args.str.length; i += 2) { + b += this.deChar2(args.str[i], (i + 2 > args.str.length) ? '\0' : args.str[i + 1], args.key + i / 2) + } + //console.log('123') + return b + } + + enChar2(c, p) { + let t = (c.charCodeAt(0) + p) % 65536 + t = t - t % 10 + (9 - t % 10) + + let c1 = String.fromCharCode(t >> 8) + let c2 = String.fromCharCode(t % 256) + return c1 + c2 + } + + deChar2(c1, c2, p) { + let t = c1.charCodeAt(0) * 256 + c2.charCodeAt(0) + t %= 65536 + t = t - t % 10 + (9 - t % 10) + t = (t - p + 65536) % 65536 + return String.fromCharCode(t) + } + + writeClipboard(args) { + navigator.clipboard.writeText(args.str); + } + + compare(propName) { + return (a, b) => { + a = a[propName] + b = b[propName] + if (a > b) return 1; + else if (a < b) return -1; + else return 0; + } + } + + findAllVar() { + const list = []; + let temp = this.runtime._stageTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === '') { + list.push({ + text: `${temp[obj].name}`, + value: temp[obj].id, + }); + } + }); + try { + temp = this.runtime._editingTarget.variables + } catch (e) { + temp = 'e' + } + if (temp !=='e' && this.runtime._editingTarget !== this.runtime._stageTarget) { + Object.keys(temp).forEach(obj => { + if (temp[obj].type === '') { + list.push({ + text: `[私有变量]${temp[obj].name}`, + value: temp[obj].id, + }); + } + }); + } + if (list.length === 0) { + list.push({ + text: `*没有变量*`, + value: 'empty', + }); + } + // Object.keys(this.runtime._editingTarget.variables).forEach(key => { + // list.forEach((obj) => { + // if (obj.value === key) { + // obj.text = `*${obj.text}`; + // } + // }); + // }); + list.sort(this.compare("text")); + return list; + } + + findAllList() { + const list = []; + let temp = this.runtime._stageTarget.variables + Object.keys(temp).forEach(obj => { + if (temp[obj].type === 'list') { + //console.log(temp[obj].type) + list.push({ + text: `${temp[obj].name}`, + value: temp[obj].id, + }); + } + }); + try { + temp = this.runtime._editingTarget.variables + } catch (e) { + temp = 'e' + } + if (temp !=='e' && this.runtime._editingTarget !== this.runtime._stageTarget) { + Object.keys(temp).forEach(obj => { + if (temp[obj].type === 'list') { + list.push({ + text: `[私有列表]${temp[obj].name}`, + value: temp[obj].id, + + }); + } + }); + } + if (list.length === 0) { + list.push({ + text: `*没有列表*`, + value: 'empty', + }); + } + list.sort(this.compare("text")); + return list; + } + + findAllVarContents() { + const list = []; + let temp = this.content + Object.keys(temp).forEach(obj => { + if (typeof temp[obj] !== 'object') { + list.push({ + text: obj, + value: obj, + }); + } + }); + if (list.length === 0) { + list.push({ + text: '*没有变量*', + value: 'empty', + }); + } + list.sort(this.compare("text")); + return list; + } + + findAllListsContents() { + const list = []; + let temp = this.content + Object.keys(temp).forEach(obj => { + if (typeof temp[obj] === 'object') { + list.push({ + text: obj, + value: obj, + }); + } + }); + if (list.length === 0) { + list.push({ + text: '*没有列表*', + value: 'empty', + }); + } + list.sort(this.compare("text")); + return list; + } + + +} + + + + +window.tempExt = { + Extension: Archive_code, + info: { + name: 'hcn.extensionName', + description: 'hcn.description', + extensionId: 'Archivecode', + // iconURL: icon, + // insetIconURL: cover, + featured: true, + disabled: false, + collaborator: 'only for hcn test', + }, + l10n: { + 'zh-cn': { + 'hcn.extensionName': 'hcn 的测试', + 'hcn.description': 'hcn 的测试', + }, + en: { + 'hcn.extensionName': 'hcn test', + 'hcn.description': 'hcn test', + }, + }, +} \ No newline at end of file From f8e975c6e25d63edbbaf1f04901229238691715c Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Sat, 23 Jul 2022 22:00:28 +0800 Subject: [PATCH 063/357] =?UTF-8?q?cast=E5=8F=82=E6=95=B0=EF=BC=8C?= =?UTF-8?q?=E4=BB=A5=E5=8F=8A=E9=99=90=E5=88=B6=E7=A7=BB=E5=88=B0=EF=BC=8C?= =?UTF-8?q?=E8=AE=BE=E7=BD=AE=E5=A4=A7=E5=B0=8F=E7=9A=84=E8=8C=83=E5=9B=B4?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 30154761..bccdb57e 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -850,8 +850,7 @@ class ArkosExtensions { return; } if (util.target.renderer) { - args.size = Cast.toNumber(args.size); - if(args.size < 0.1) args.size = 0.1; + args.size = _clamp(Cast.toNumber(args.size),0.1,100000000) util.target.size = args.size; const {direction, scale} = util.target._getRenderedDirectionAndScale(); util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); @@ -863,9 +862,15 @@ class ArkosExtensions { util.target.runtime.requestTargetsUpdate(util.target); } + _clamp (n, min, max) { + return Math.min(Math.max(n, min), max); + } + //强行设置XY(逝一逝) setXY (args, util) { if (util.target.isStage) return; + args.x = _clamp(Cast.toNumber(args.x),-100000000,100000000) + args.y = _clamp(Cast.toNumber(args.y),-100000000,100000000) const oldX = util.target.x; const oldY = util.target.y; util.target.x = args.x; @@ -1068,7 +1073,7 @@ class ArkosExtensions { getFromSortedTableByNo (args){ if(!(args.list in this.sortedTable)) return ''; let list = this.sortedTable[args.list].list; - return this._getTInItem (list[args.n-1], args.t, args.n); + return this._getTInItem (list[args.n-1], args.t, Cast.toNumber(args.n)); } _getItemIdxByName(list, name){ From 719650eff35b6139fd22773216775cf157da34b0 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Sun, 24 Jul 2022 09:50:23 +0800 Subject: [PATCH 064/357] =?UTF-8?q?=E5=BF=98=E5=8A=A0this.=20=E4=BA=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index bccdb57e..7199d763 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -850,7 +850,7 @@ class ArkosExtensions { return; } if (util.target.renderer) { - args.size = _clamp(Cast.toNumber(args.size),0.1,100000000) + args.size = this._clamp(Cast.toNumber(args.size),0.1,100000000) util.target.size = args.size; const {direction, scale} = util.target._getRenderedDirectionAndScale(); util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); @@ -869,8 +869,8 @@ class ArkosExtensions { //强行设置XY(逝一逝) setXY (args, util) { if (util.target.isStage) return; - args.x = _clamp(Cast.toNumber(args.x),-100000000,100000000) - args.y = _clamp(Cast.toNumber(args.y),-100000000,100000000) + args.x = this._clamp(Cast.toNumber(args.x),-100000000,100000000) + args.y = this._clamp(Cast.toNumber(args.y),-100000000,100000000) const oldX = util.target.x; const oldY = util.target.y; util.target.x = args.x; From 1258431e910a280d49881abb5177e20043fd7db0 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 26 Jul 2022 23:34:55 +0800 Subject: [PATCH 065/357] =?UTF-8?q?=E5=8A=A0=E5=85=A5=E4=B8=B4=E6=97=B6?= =?UTF-8?q?=E5=8F=98=E9=87=8F?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 336 ++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 334 insertions(+), 2 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 7199d763..713dae3c 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -8,6 +8,7 @@ console.log(Cast.toNumber('aab')) class ArkosExtensions { constructor(runtime) { this.runtime = runtime + this.tempData={} this.sortedTable={ list1:{order:'desc',list:[]}, list2:{order:'desc',list:[]}, @@ -20,6 +21,7 @@ class ArkosExtensions { 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', 'ArkosExt.info5': '📊 排序表', + 'ArkosExt.info6': '🗂️ 临时变量', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', @@ -73,7 +75,24 @@ class ArkosExtensions { 'ArkosExt.colorToHex': '颜色[COLOR]的代码', - + 'ArkosExt.deleteAllTempData': '清空所有临时数据', + 'ArkosExt.getCountOfTempData': '临时数据量', + 'ArkosExt.delTempData': '删除名为[data]的临时数据', + + 'ArkosExt.setTempVar': '将临时变量[var]设为[t]', + 'ArkosExt.addTempVar': '将临时变量[var]增加[t]', + 'ArkosExt.getTempVar': '临时变量[var]', + + 'ArkosExt.clearTempList': '创建或清空临时列表[list]', + 'ArkosExt.initTempList': '临时列表[list]内容设为[t]', + 'ArkosExt.addTempList': '向临时列表[list]加入[t]', + 'ArkosExt.opTempList': '将临时列表[list]第[n]项[op][t]', + 'ArkosExt.ListOp1': '前插入', + 'ArkosExt.ListOp2': '替换为', + 'ArkosExt.ListOp3': '增加', + 'ArkosExt.delItemOfTempList': '删除临时列表[list]第[n]项', + 'ArkosExt.getItemOfTempList': '临时列表[list]第[n]项', + 'ArkosExt.lengthOfTempList': '临时列表[list]长度', }, en: { @@ -134,6 +153,27 @@ class ArkosExtensions { 'ArkosExt.info3': '🛠 Utilities', 'ArkosExt.info4': '📄 Information', 'ArkosExt.info5': '📊 Sorted Table', + 'ArkosExt.info6': '🗂️ Temporary Data', + + 'ArkosExt.deleteAllTempData': 'clear all temporary data', + 'ArkosExt.getCountOfTempData': 'count of temporary data', + 'ArkosExt.delTempData': 'delete temporary data[data]', + + 'ArkosExt.setTempVar': 'set temp var[var] to [t]', + 'ArkosExt.addTempVar': 'change temp var[var] by [t]', + 'ArkosExt.getTempVar': 'temp var[var]', + + + 'ArkosExt.clearTempList': 'create or clear temp list[list]', + 'ArkosExt.initTempList': 'set temp list[list]to[t]', + 'ArkosExt.addTempList': 'add[t] to temp list[list]', + 'ArkosExt.opTempList': '[op][t]at [n] of temp list[list]', + 'ArkosExt.ListOp1': 'insert', + 'ArkosExt.ListOp2': 'replace with', + 'ArkosExt.ListOp3': 'change by', + 'ArkosExt.delItemOfTempList': 'delete [n]of temp list[list]', + 'ArkosExt.getItemOfTempList': 'item[n]of temp list[list]', + 'ArkosExt.lengthOfTempList': 'length of temp list[list]', }, }) } @@ -618,8 +658,205 @@ class ArkosExtensions { }, }, }, + + "---" + this.formatMessage("ArkosExt.info6"), //🗂️ 临时变量 + + //清空所有临时数据 + { + opcode: 'deleteAllTempData', + blockType: 'command', + text: this.formatMessage('ArkosExt.deleteAllTempData'), + }, + //临时数据量 + { + opcode: 'getCountOfTempData', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getCountOfTempData'), + }, + //删除临时数据 + { + opcode: 'delTempData', + blockType: 'command', + text: this.formatMessage('ArkosExt.delTempData'), + arguments: { + data: { + type: 'string', + defaultValue: 'i', + }, + }, + }, + //设置临时数据 + { + opcode: 'setTempVar', + blockType: 'command', + text: this.formatMessage('ArkosExt.setTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + t: { + type: 'string', + defaultValue: '0', + }, + }, + }, + //增加临时数据 + { + opcode: 'addTempVar', + blockType: 'command', + text: this.formatMessage('ArkosExt.addTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + t: { + type: 'number', + defaultValue: 1, + }, + }, + }, + //增加临时数据 + { + opcode: 'getTempVar', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + }, + }, + //创建或清空临时列表 + { + opcode: 'clearTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + }, + }, + //设置临时列表 + { + opcode: 'initTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.initTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + t: { + type: 'string', + defaultValue: '[1,2,"apple"]', + }, + }, + }, + //向临时列表加入 + { + opcode: 'addTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.addTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + t: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, + //操作临时列表 + { + opcode: 'opTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.opTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + op: { + type: 'string', + menu: 'ListOpMenu', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, + //删除临时列表XX项 + { + opcode: 'delItemOfTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.delItemOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + t: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, + //获取临时列表XX项 + { + opcode: 'getItemOfTempList', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + n: { + type: 'number', + defaultValue: 1, + }, + }, + }, + //临时列表长度 + { + opcode: 'lengthOfTempList', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + }, + }, ], menus: { + ListOpMenu: [ + { + text: this.formatMessage('ArkosExt.ListOp1'), //插入 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.ListOp2'), //替换 + value: '2' + }, + { + text: this.formatMessage('ArkosExt.ListOp3'), //增加 + value: '3' + }, + ], tableItemPropertyMenu: [ { text: this.formatMessage('ArkosExt.name'), @@ -1113,8 +1350,103 @@ class ArkosExtensions { colorToHex (args, util) { let c = Cast.toRgbColorList(args.COLOR) return Color.rgbToHex({r:c[0],g:c[1],b:c[2]}); -} + } + + + //🗂️ 临时变量积木 + deleteAllTempData (args) { + this.tempData = {}; + } + + getCountOfTempData (args) { + return Object.keys (this.tempData).length; + } + + delTempData (args) { + delete this.tempData[Cast.toString(args.data)]; + } + + setTempVar (args) { + this.tempData[Cast.toString(args.var)] = args.t; + } + + addTempVar (args) { + this.tempData[Cast.toString(args.var)] = Cast.toNumber(this.tempData[Cast.toString(args.var)]) + Cast.toNumber(args.t); + } + + getTempVar (args) { + let temp = this.tempData[Cast.toString(args.var)] + if(Array.isArray(temp)) return JSON.stringify(temp); + return Cast.toString(temp); + } + + //创建或清空 + clearTempList (args) { + this.tempData[Cast.toString(args.list)] = []; + } + + initTempList (args) { + try { + let content = JSON.parse(Cast.toString(args.t)) + if(Array.isArray(content)) { + this.tempData[Cast.toString(args.list)] = content; + } + } catch (e) { + + } + } + + addTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + list.push(Cast.toString(args.t)); + } + + opTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + let n = Cast.toNumber(args.n) + if(n < 1 || n >list.length+1) return; + n -= 1; + switch(args.op){ + case '1'://插入 + list.splice(n,0,Cast.toString(args.t)); + return; + case '2'://替换 + list[n] = Cast.toString(args.t); + return; + case '3'://增加 + list[n] = Cast.toNumber(list[n]) + Cast.toNumber(args.t); + return; + default: + return; + } + } + delItemOfTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + let n = Cast.toNumber(args.n) + if(n < 1 || n >list.length) return; + n -= 1; + list.splice(n,1); + } + + getItemOfTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return ''; + let n = Cast.toNumber(args.n) + if(n < 1 || n >list.length) return ''; + n -= 1; + return Cast.toString(list[n]); + } + + lengthOfTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return 0; + return list.length; + } + } From 55d93b9c2a872d5e4e140e24dbdf300173221c8c Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 26 Jul 2022 23:36:55 +0800 Subject: [PATCH 066/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E7=A7=AF=E6=9C=A8?= =?UTF-8?q?=E5=8F=82=E6=95=B0=E9=94=99=E8=AF=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 713dae3c..8a085bb7 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -807,9 +807,9 @@ class ArkosExtensions { type: 'string', defaultValue: 'list', }, - t: { - type: 'string', - defaultValue: 'thing', + n: { + type: 'number', + defaultValue: 1, }, }, }, From cdfe1c6ab4ee6a789aaa9d4b2c643dd9e5d6b6ee Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Wed, 27 Jul 2022 16:56:51 +0800 Subject: [PATCH 067/357] =?UTF-8?q?=E7=BB=99=E4=B8=B4=E6=97=B6=E6=95=B0?= =?UTF-8?q?=E6=8D=AE=E7=A7=AF=E6=9C=A8=E5=8A=A0=E5=85=A5=F0=9F=97=82?= =?UTF-8?q?=EF=B8=8F=E5=89=8D=E7=BC=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/project.js | 56 ++++++++++++++++++++++++------------------------ 1 file changed, 28 insertions(+), 28 deletions(-) diff --git a/Arkos/project.js b/Arkos/project.js index 8a085bb7..a2925998 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -75,24 +75,24 @@ class ArkosExtensions { 'ArkosExt.colorToHex': '颜色[COLOR]的代码', - 'ArkosExt.deleteAllTempData': '清空所有临时数据', - 'ArkosExt.getCountOfTempData': '临时数据量', - 'ArkosExt.delTempData': '删除名为[data]的临时数据', - - 'ArkosExt.setTempVar': '将临时变量[var]设为[t]', - 'ArkosExt.addTempVar': '将临时变量[var]增加[t]', - 'ArkosExt.getTempVar': '临时变量[var]', - - 'ArkosExt.clearTempList': '创建或清空临时列表[list]', - 'ArkosExt.initTempList': '临时列表[list]内容设为[t]', - 'ArkosExt.addTempList': '向临时列表[list]加入[t]', - 'ArkosExt.opTempList': '将临时列表[list]第[n]项[op][t]', + 'ArkosExt.deleteAllTempData': '🗂️清空所有临时数据', + 'ArkosExt.getCountOfTempData': '🗂️临时数据量', + 'ArkosExt.delTempData': '🗂️删除名为[data]的临时数据', + + 'ArkosExt.setTempVar': '🗂️将临时变量[var]设为[t]', + 'ArkosExt.addTempVar': '🗂️将临时变量[var]增加[t]', + 'ArkosExt.getTempVar': '🗂️临时变量[var]', + + 'ArkosExt.clearTempList': '🗂️创建或清空临时列表[list]', + 'ArkosExt.initTempList': '🗂️临时列表[list]内容设为[t]', + 'ArkosExt.addTempList': '🗂️向临时列表[list]加入[t]', + 'ArkosExt.opTempList': '🗂️将临时列表[list]第[n]项[op][t]', 'ArkosExt.ListOp1': '前插入', 'ArkosExt.ListOp2': '替换为', 'ArkosExt.ListOp3': '增加', - 'ArkosExt.delItemOfTempList': '删除临时列表[list]第[n]项', - 'ArkosExt.getItemOfTempList': '临时列表[list]第[n]项', - 'ArkosExt.lengthOfTempList': '临时列表[list]长度', + 'ArkosExt.delItemOfTempList': '🗂️删除临时列表[list]第[n]项', + 'ArkosExt.getItemOfTempList': '🗂️临时列表[list]第[n]项', + 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', }, en: { @@ -155,25 +155,25 @@ class ArkosExtensions { 'ArkosExt.info5': '📊 Sorted Table', 'ArkosExt.info6': '🗂️ Temporary Data', - 'ArkosExt.deleteAllTempData': 'clear all temporary data', - 'ArkosExt.getCountOfTempData': 'count of temporary data', - 'ArkosExt.delTempData': 'delete temporary data[data]', + 'ArkosExt.deleteAllTempData': '🗂️clear all temporary data', + 'ArkosExt.getCountOfTempData': '🗂️count of temporary data', + 'ArkosExt.delTempData': '🗂️delete temporary data[data]', - 'ArkosExt.setTempVar': 'set temp var[var] to [t]', - 'ArkosExt.addTempVar': 'change temp var[var] by [t]', - 'ArkosExt.getTempVar': 'temp var[var]', + 'ArkosExt.setTempVar': '🗂️set temp var[var] to [t]', + 'ArkosExt.addTempVar': '🗂️change temp var[var] by [t]', + 'ArkosExt.getTempVar': '🗂️temp var[var]', - 'ArkosExt.clearTempList': 'create or clear temp list[list]', - 'ArkosExt.initTempList': 'set temp list[list]to[t]', - 'ArkosExt.addTempList': 'add[t] to temp list[list]', - 'ArkosExt.opTempList': '[op][t]at [n] of temp list[list]', + 'ArkosExt.clearTempList': '🗂️create or clear temp list[list]', + 'ArkosExt.initTempList': '🗂️set temp list[list]to[t]', + 'ArkosExt.addTempList': '🗂️add[t] to temp list[list]', + 'ArkosExt.opTempList': '🗂️[op][t]at [n] of temp list[list]', 'ArkosExt.ListOp1': 'insert', 'ArkosExt.ListOp2': 'replace with', 'ArkosExt.ListOp3': 'change by', - 'ArkosExt.delItemOfTempList': 'delete [n]of temp list[list]', - 'ArkosExt.getItemOfTempList': 'item[n]of temp list[list]', - 'ArkosExt.lengthOfTempList': 'length of temp list[list]', + 'ArkosExt.delItemOfTempList': '🗂️delete [n]of temp list[list]', + 'ArkosExt.getItemOfTempList': '🗂️item[n]of temp list[list]', + 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', }, }) } From cac0937fec1361f999efb7c8ea69f8ed0d9fcffb Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Thu, 28 Jul 2022 23:54:59 +0800 Subject: [PATCH 068/357] add --- java30433/30Ext.js | 191 ++++++++++++++++++ .../1273b53562550d34c80dc66a2737dfa4.png | Bin 0 -> 100847 bytes java30433/assets/cover.icpr | Bin 0 -> 3494732 bytes java30433/assets/cover.png | Bin 0 -> 80445 bytes 4 files changed, 191 insertions(+) create mode 100644 java30433/30Ext.js create mode 100644 java30433/assets/1273b53562550d34c80dc66a2737dfa4.png create mode 100644 java30433/assets/cover.icpr create mode 100644 java30433/assets/cover.png diff --git a/java30433/30Ext.js b/java30433/30Ext.js new file mode 100644 index 00000000..4077219a --- /dev/null +++ b/java30433/30Ext.js @@ -0,0 +1,191 @@ +class _30ExtExtensions { + constructor(runtime) { + this.runtime = runtime; + this._formatMessage = runtime.getFormatMessage({ + 'zh-cn': { + '30Ext.extensionName': '工具箱', + '30Ext.info.1': '造型镜像操作', + '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', + '30Ext.menu.mirrorMethod.1': '左右镜像', + '30Ext.menu.mirrorMethod.1': '上下镜像', + '30Ext.block.clearMirror': '清除角色镜像变换', + '30Ext.info.2': '角色跨域操作', + '30Ext.block.anotherRun': '让[sprite]运行[SUBSTACK]', + '30Ext.block.anotherRunWithClone': '让[sprite]的第[cloneId]个克隆体运行[SUBSTACK]' + }, + + en: { + '30Ext.extensionName': 'Toolbox', + '30Ext.info.1': 'Mirror transform', + '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', + '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', + '30Ext.menu.mirrorMethod.1': 'Vertical mirror transform', + '30Ext.block.clearMirror': 'Clear the mirror transform', + '30Ext.info.2': 'Cross sprite operation', + '30Ext.block.anotherRun': 'Let [sprite] run[SUBSTACK]', + '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [sprite] run[SUBSTACK]' + } + }) + } + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id, + }) + } + getInfo() { + return { + id: '30Ext', // 拓展id + name: this.formatMessage('30Ext.extensionName'), + + color1: '#FF8196', + // menuIconURI: icon, + // blockIconURI: icon, + blocks: [ + "---" + this.formatMessage("30Ext.info.1"), //角色造型操作 + // 镜像造型 + { + opcode: 'mirrorSprite', + blockType: 'command', + text: this.formatMessage('30Ext.block.mirrorSprite'), + arguments: { + mirrorMethod: { + type: 'number', + menu: [{ + text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 + value: 0 + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 + value: 1 + } + ] + } + } + }, + // 清除镜像 + { + opcode: 'clearMirror', + blockType: 'command', + text: this.formatMessage('30Ext.block.clearMirror') + }, + "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 + { + opcode: 'anotherRun', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRun'), + arguments: { + spriteName: { + type: 'string', + menu: spritesMenu + }, + SUBSTACK: { + type: "input_statement" + } + } + }, + { + opcode: 'anotherRunWithClone', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRunWithClone'), + arguments: { + spriteName: { + type: 'string', + menu: spritesMenu + }, + cloneId: { + type: 'number', + defaultValue: 1 + } + SUBSTACK: { + type: "input_statement" + } + } + } + ], + menus: { + spritesMenu: { + items: 'getSpritesMenu' + } + } + } + } + //菜单 + //动态菜单: 角色菜单 + getSpritesMenu(){ + var sprites = []; + for(const targetId in vm.runtime.targets) { + if(!vm.runtime.targets.hasOwnProperty(targetId)) continue; + if(!vm.runtime.targets[targetId].isOriginal) continue; + if(vm.runtime.targets[targetId] === vm.editingTarget) continue; //排除自己 + let name = vm.runtime.targets[targetId].sprite.name; + sprites.push(name); //疑问? 菜单不应该返回诸如[["显示", "内容"],["显示2", "内容值"]]的东西吗 我看扩展库test.js是这样写 + } + return sprites; + } + // + //角色造型操作 + // + //镜像造型 + mirrorSprite(args, util){ + util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; + } + //清除镜像 + clearMirror(args, util){ + for (let i=0; i<2; i++){ + let foo = util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]; + foo = Math.abs(foo); + } + } + // + //角色跨域操作 + // + //跨域执行 + anotherRun(args, util){ + if(!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = util.target.sprite.clones[0]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } + //跨域克隆体执行 + anotherRunWithClone(args, util){ + if(!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = util.target.sprite.clones[args.cloneId]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } +} +window.tempExt = { + Extension: _30Extensions, + info: { + name: '30Ext.extensionName', + description: '30Ext.description', + extensionId: '30Ext', + // iconURL: icon, + // insetIconURL: cover, + featured: true, + disabled: false, + collaborator: '_30 x CCW' + }, + l10n: { + 'zh-cn': { + '30Ext.extensionName': 'java30433的工具箱', + '30Ext.description': '造型镜像&角色跨域操作 持续更新中...' + }, + en: { + '30Ext.extensionName': "java30433's Toolbox", + '30Ext.description': 'Mirror transform&Cross sprite operation Continuous updating.' + } + } +}; \ No newline at end of file diff --git a/java30433/assets/1273b53562550d34c80dc66a2737dfa4.png b/java30433/assets/1273b53562550d34c80dc66a2737dfa4.png new file mode 100644 index 0000000000000000000000000000000000000000..60b19381cfd5b89c49ec5492945da810499ca14e GIT binary patch literal 100847 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

>yE!}~IZsG_n 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 |XT*H?)%R4H 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 literal 0 HcmV?d00001 diff --git a/java30433/assets/cover.icpr b/java30433/assets/cover.icpr new file mode 100644 index 0000000000000000000000000000000000000000..f1161fe69420ba64ad8485a472691f56e85b3165 GIT binary patch literal 3494732 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 literal 0 HcmV?d00001 From 07a81adb79ba7e030fa78976f43cb4d9f010a869 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Thu, 28 Jul 2022 23:55:37 +0800 Subject: [PATCH 069/357] add --- ccw-user-extension | 1 - 1 file changed, 1 deletion(-) delete mode 160000 ccw-user-extension diff --git a/ccw-user-extension b/ccw-user-extension deleted file mode 160000 index 149b828a..00000000 --- a/ccw-user-extension +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 149b828a413cb1a01a2081c7d4dd0e2d5b51a78c From 595521d6b3616d9b8d3224ee788a856488321632 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Fri, 29 Jul 2022 21:45:07 +0800 Subject: [PATCH 070/357] add 30Ext --- Arkos/project.js | 2 +- Arkos/projectWith30.js | 1620 +++++++++++++++++ java30433/30Ext.js | 191 -- .../1273b53562550d34c80dc66a2737dfa4.png | Bin 100847 -> 0 bytes java30433/assets/cover.icpr | Bin 3494732 -> 0 bytes java30433/assets/cover.png | Bin 80445 -> 0 bytes 6 files changed, 1621 insertions(+), 192 deletions(-) create mode 100644 Arkos/projectWith30.js delete mode 100644 java30433/30Ext.js delete mode 100644 java30433/assets/1273b53562550d34c80dc66a2737dfa4.png delete mode 100644 java30433/assets/cover.icpr delete mode 100644 java30433/assets/cover.png diff --git a/Arkos/project.js b/Arkos/project.js index a2925998..dee320da 100644 --- a/Arkos/project.js +++ b/Arkos/project.js @@ -1473,4 +1473,4 @@ window.tempExt = { 'hcn.description': 'hcn test', }, }, - } + } \ No newline at end of file diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js new file mode 100644 index 00000000..1f008242 --- /dev/null +++ b/Arkos/projectWith30.js @@ -0,0 +1,1620 @@ +import Cast from '../utils/cast.js' +import Color from '../utils/color.js' +// import cover from './assets/icon.svg' +// import icon from './assets/icon.svg' + +console.log(Cast.toNumber('123')) +console.log(Cast.toNumber('aab')) +class ArkosExtensions { + constructor(runtime) { + this.runtime = runtime + this.tempData={} + this.sortedTable={ + list1:{order:'desc',list:[]}, + list2:{order:'desc',list:[]}, + } + this._formatMessage = runtime.getFormatMessage({ + 'zh-cn': { + 'ArkosExt.extensionName': 'Arkosの拓展', + 'ArkosExt.info1': '🚶 坐标和方向', + 'ArkosExt.info2': '🔠 字符串处理', + 'ArkosExt.info3': '🛠 实用积木', + 'ArkosExt.info4': '📄 信息获取', + 'ArkosExt.info5': '📊 排序表', + 'ArkosExt.info6': '🗂️ 临时变量', + 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', + 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', + 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', + 'ArkosExt.distance': '点x1:[X1]y1:[Y1]到点x2:[X2]y2:[Y2]的距离', + 'ArkosExt.searchString': '在[str]中查找[substr]的位置(从位置[pos]开始找)', + 'ArkosExt.insertString': '在[str]的第[pos]个字符前插入[substr]', + 'ArkosExt.replaceString': '将[str]中的第[start]个到第[end]个字符,替换为[substr]', + 'ArkosExt.turnDegreesToDir': '朝方向[dir]旋转[degree]度', + + 'ArkosExt.getEffect': '获取特效[EFFECT]的值', + 'ArkosExt.color': '颜色', + 'ArkosExt.fisheye': '鱼眼', + 'ArkosExt.whirl': '旋涡', + 'ArkosExt.pixelate': '像素化', + 'ArkosExt.mosaic': '马赛克', + 'ArkosExt.brightness': '亮度', + 'ArkosExt.ghost': '虚像', + + 'ArkosExt.isHiding': '角色隐藏?', + 'ArkosExt.getRotationStyle': '当前旋转方式', + 'ArkosExt.getWidthOrHeight': '获取当前造型的[t]', + 'ArkosExt.setSize': '⚠️强行将大小设为[size](无视限制)', + 'ArkosExt.width': '宽', + 'ArkosExt.height': '高', + + 'ArkosExt.setXY': '⚠️强行移到x:[x]y:[y](无视边界)', + 'ArkosExt.getBoundaryCoord': '获取角色的[t]', + 'ArkosExt.top': '上边缘y', + 'ArkosExt.bottom': '下边缘y', + 'ArkosExt.left': '左边缘x', + 'ArkosExt.right': '右边缘x', + 'ArkosExt.isOutOfSight': '角色移到舞台区外?', + + 'ArkosExt.and': '且', + 'ArkosExt.or': '或', + + 'ArkosExt.clearSortedTable': '📊清空排序表[list]', + 'ArkosExt.setTypeOfSortedTable': '📊将排序表[list]的排序方式设为[type]', + 'ArkosExt.addToSortedTable': '📊将内容(重名的则覆盖)[name],排序值[value]加入排序表[list],附加信息[extra]', + 'ArkosExt.getFromSortedTableByNo': '📊获取排序表[list]中第[n]项的[t]', + 'ArkosExt.getFromSortedTableByName': '📊获取[name]在排序表[list]中的[t]', + 'ArkosExt.lengthOfSortedTable': '📊排序表[list]中内容数', + 'ArkosExt.deleteNameOfSortedTable': '📊删除排序表[list]中名为[name]的项', + 'ArkosExt.asc': '升序', + 'ArkosExt.desc': '降序', + + 'ArkosExt.name': '名称', + 'ArkosExt.rank': '表中位置', + 'ArkosExt.rankValue': '排序值', + 'ArkosExt.extra': '附加信息', + + 'ArkosExt.colorToHex': '颜色[COLOR]的代码', + + 'ArkosExt.deleteAllTempData': '🗂️清空所有临时数据', + 'ArkosExt.getCountOfTempData': '🗂️临时数据量', + 'ArkosExt.delTempData': '🗂️删除名为[data]的临时数据', + + 'ArkosExt.setTempVar': '🗂️将临时变量[var]设为[t]', + 'ArkosExt.addTempVar': '🗂️将临时变量[var]增加[t]', + 'ArkosExt.getTempVar': '🗂️临时变量[var]', + + 'ArkosExt.clearTempList': '🗂️创建或清空临时列表[list]', + 'ArkosExt.initTempList': '🗂️临时列表[list]内容设为[t]', + 'ArkosExt.addTempList': '🗂️向临时列表[list]加入[t]', + 'ArkosExt.opTempList': '🗂️将临时列表[list]第[n]项[op][t]', + 'ArkosExt.ListOp1': '前插入', + 'ArkosExt.ListOp2': '替换为', + 'ArkosExt.ListOp3': '增加', + 'ArkosExt.delItemOfTempList': '🗂️删除临时列表[list]第[n]项', + 'ArkosExt.getItemOfTempList': '🗂️临时列表[list]第[n]项', + 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', + + '30Ext.info': '以下扩展由_30提供', + '30Ext.info.1': '造型镜像操作', + '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', + '30Ext.menu.mirrorMethod.1': '左右镜像', + '30Ext.menu.mirrorMethod.1': '上下镜像', + '30Ext.block.clearMirror': '清除角色镜像变换', + '30Ext.info.2': '角色跨域操作', + '30Ext.block.anotherRun': '让[sprite]运行[SUBSTACK]', + '30Ext.block.anotherRunWithClone': '让[sprite]的第[cloneId]个克隆体运行[SUBSTACK]' + }, + + en: { + 'ArkosExt.extensionName': "Arkos' Extensions", + 'ArkosExt.stringEquality': '(case sensitive)[ONE]=[TWO]', + 'ArkosExt.directionFromAtoB': 'direction from x1:[X1]y1:[Y1]to x2:[X2]y2:[Y2]', + 'ArkosExt.differenceBetweenDirections': 'direction[b] minus direction[a]', + 'ArkosExt.distance': 'distance between x1:[X1]y1:[Y1]and x2:[X2]y2:[Y2]', + 'ArkosExt.searchString': 'position of[substr]in[str],start from[pos]', + 'ArkosExt.insertString': 'insert[substr]at[pos]of[str]', + 'ArkosExt.replaceString': 'replace from[start]to[end]of[str],with[substr]', + 'ArkosExt.turnDegreesToDir': 'turn[degree] degrees toward direction[dir]', + 'ArkosExt.getEffect': 'effect[EFFECT]', + 'ArkosExt.color': 'color', + 'ArkosExt.fisheye': 'fisheye', + 'ArkosExt.whirl': 'whirl', + 'ArkosExt.pixelate': 'pixelate', + 'ArkosExt.mosaic': 'mosaic', + 'ArkosExt.brightness': 'brightness', + 'ArkosExt.ghost': 'ghost', + 'ArkosExt.isHiding': 'is hiding?', + 'ArkosExt.getRotationStyle': 'rotation style', + 'ArkosExt.getWidthOrHeight': 'get [t] of the current costume', + 'ArkosExt.setSize': '⚠️force the size to [size] % (regardless of limitation) ', + 'ArkosExt.width': 'width', + 'ArkosExt.height': 'height', + + 'ArkosExt.setXY': '⚠️force to x:[x]y:[y] (regardless of the boundary)', + 'ArkosExt.getBoundaryCoord': 'get [t] of the sprite', + 'ArkosExt.top': 'top y', + 'ArkosExt.bottom': 'bottom y', + 'ArkosExt.left': 'left x', + 'ArkosExt.right': 'right x', + 'ArkosExt.isOutOfSight': 'is out of stage?', + + 'ArkosExt.and': 'and', + 'ArkosExt.or': 'or', + + 'ArkosExt.clearSortedTable': '📊empty sorted table[list]', + 'ArkosExt.setTypeOfSortedTable': '📊set sort order of[list]to[type]', + 'ArkosExt.addToSortedTable': '📊add (overwrite if existed)[name]to table[list] with sort index value[value],extra data[extra] and sort', + 'ArkosExt.getFromSortedTableByNo': '📊get[t]of #[n] from [list]', + 'ArkosExt.getFromSortedTableByName': '📊get[t]of [name] from [list]', + 'ArkosExt.lengthOfSortedTable': '📊length of sorted table[list]', + 'ArkosExt.deleteNameOfSortedTable': '📊delete [name] in[list]', + 'ArkosExt.asc': 'ascending order', + 'ArkosExt.desc': 'descending order', + + 'ArkosExt.name': 'name', + 'ArkosExt.rank': 'rank', + 'ArkosExt.rankValue': 'rankValue', + 'ArkosExt.extra': 'extra', + + 'ArkosExt.colorToHex': 'get code of color[COLOR]', + + 'ArkosExt.info1': '🚶 Coordinate and Direction', + 'ArkosExt.info2': '🔠 String Processing', + 'ArkosExt.info3': '🛠 Utilities', + 'ArkosExt.info4': '📄 Information', + 'ArkosExt.info5': '📊 Sorted Table', + 'ArkosExt.info6': '🗂️ Temporary Data', + + 'ArkosExt.deleteAllTempData': '🗂️clear all temporary data', + 'ArkosExt.getCountOfTempData': '🗂️count of temporary data', + 'ArkosExt.delTempData': '🗂️delete temporary data[data]', + + 'ArkosExt.setTempVar': '🗂️set temp var[var] to [t]', + 'ArkosExt.addTempVar': '🗂️change temp var[var] by [t]', + 'ArkosExt.getTempVar': '🗂️temp var[var]', + + + 'ArkosExt.clearTempList': '🗂️create or clear temp list[list]', + 'ArkosExt.initTempList': '🗂️set temp list[list]to[t]', + 'ArkosExt.addTempList': '🗂️add[t] to temp list[list]', + 'ArkosExt.opTempList': '🗂️[op][t]at [n] of temp list[list]', + 'ArkosExt.ListOp1': 'insert', + 'ArkosExt.ListOp2': 'replace with', + 'ArkosExt.ListOp3': 'change by', + 'ArkosExt.delItemOfTempList': '🗂️delete [n]of temp list[list]', + 'ArkosExt.getItemOfTempList': '🗂️item[n]of temp list[list]', + 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', + + '30Ext.info': 'The following extensions are provided by _30', + '30Ext.info.1': 'Mirror transform', + '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', + '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', + '30Ext.menu.mirrorMethod.1': 'Vertical mirror transform', + '30Ext.block.clearMirror': 'Clear the mirror transform', + '30Ext.info.2': 'Cross sprite operation', + '30Ext.block.anotherRun': 'Let [sprite] run[SUBSTACK]', + '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [sprite] run[SUBSTACK]' + }, + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id, + }) + } + + getInfo() { + return { + id: 'hcnTest', // 拓展id + name: this.formatMessage('ArkosExt.extensionName'), + + color1: '#FF8383', + // menuIconURI: icon, + // blockIconURI: icon, + blocks: [ + "---" + this.formatMessage("ArkosExt.info1"), //🏃 坐标&角度 + // 计算点A到点B的方向 + { + opcode: 'getDirFromAToB', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.directionFromAtoB'), + arguments: { + X1: { + type: 'number', + defaultValue: 0, + }, + Y1: { + type: 'number', + defaultValue: 0, + }, + X2: { + type: 'number', + defaultValue: 0, + }, + Y2: { + type: 'number', + defaultValue: 0, + }, + }, + }, + // 计算角b-角a的角度差 + { + opcode: 'differenceBetweenDirections', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.differenceBetweenDirections'), + arguments: { + a: { + type: 'number', + defaultValue: 0, + }, + b: { + type: 'number', + defaultValue: 0, + }, + }, + }, + // 两点距离 + { + opcode: 'disFromAToB', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.distance'), + arguments: { + X1: { + type: 'number', + defaultValue: 0, + }, + Y1: { + type: 'number', + defaultValue: 0, + }, + X2: { + type: 'number', + defaultValue: 0, + }, + Y2: { + type: 'number', + defaultValue: 0, + }, + }, + }, + //朝..方向旋转..角度 + { + opcode: 'turnDegreesToDir', + blockType: 'command', + text: this.formatMessage('ArkosExt.turnDegreesToDir'), + arguments: { + degree: { + type: 'number', + defaultValue: 45, + }, + dir: { + type: 'angle', + defaultValue: 10, + }, + }, + filter: ['sprite'] + }, + "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 + // 查找子字符串,从pos开始 + { + opcode: 'indexof', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.searchString'), + arguments: { + str: { + type: 'string', + defaultValue: 'banana', + }, + substr: { + type: 'string', + defaultValue: 'na', + }, + pos: { + type: 'number', + defaultValue: 1, + }, + }, + }, + // 在字符串中插入子字符串 + { + opcode: 'insertStr', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.insertString'), + arguments: { + str: { + type: 'string', + defaultValue: 'ac', + }, + substr: { + type: 'string', + defaultValue: 'b', + }, + pos: { + type: 'number', + defaultValue: 2, + }, + }, + }, + // 替换字符串中的从..到..的字符串 + { + opcode: 'replaceStr', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.replaceString'), + arguments: { + str: { + type: 'string', + defaultValue: 'ABCDEF', + }, + substr: { + type: 'string', + defaultValue: 'XX', + }, + start: { + type: 'number', + defaultValue: 3, + }, + end: { + type: 'number', + defaultValue: 4, + }, + }, + }, + "---" + this.formatMessage("ArkosExt.info3"), //🔧实用积木 + // 判断相等(区分大小写) + { + opcode: 'strictlyEquals', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.stringEquality'), + arguments: { + ONE: { + type: 'string', + defaultValue: 'A', + }, + TWO: { + type: 'string', + defaultValue: 'a', + }, + }, + }, + //返回值转bool积木 + { + opcode: 'reporterToBoolean', + blockType: 'Boolean', + text: '[t]', + arguments: { + t: { + type: 'string', + defaultValue: '1', + } + }, + }, + //形如 a≤b≤c + { + opcode: 'compareTwoSides', + blockType: 'Boolean', + text: '[a][op1][b][op2][c]', + arguments: { + a: { + type: 'string', + defaultValue: '1', + }, + b: { + type: 'string', + defaultValue: 'x', + }, + c: { + type: 'string', + defaultValue: '3', + }, + op1: { + type: 'string', + menu: 'opMenu1', + }, + op2: { + type: 'string', + menu: 'opMenu1', + }, + }, + }, + //形如:a≤b且/或>c op1,op2 logic compareTwoSidesPlus + { + opcode: 'compareTwoSidesPlus', + blockType: 'Boolean', + text: '[a][op1][b][logic][op2][c]', + arguments: { + a: { + type: 'string', + defaultValue: 'x', + }, + b: { + type: 'string', + defaultValue: '1', + }, + c: { + type: 'string', + defaultValue: '3', + }, + op1: { + type: 'string', + menu: 'opMenu2', + defaultValue: '<', + }, + op2: { + type: 'string', + menu: 'opMenu2', + defaultValue: '>', + }, + logic: { + type: 'string', + menu: 'logicMenu', + defaultValue: 'or',// + }, + }, + }, + //获取颜色HEX + { + opcode: 'colorToHex', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.colorToHex'), + arguments: { + COLOR: { + type: 'color', + //defaultValue: '1', + }, + }, + }, + //强行设置大小 + { + opcode: 'setSize', + blockType: 'command', + text: this.formatMessage('ArkosExt.setSize'), + arguments: { + size: { + type: 'number', + defaultValue: 9999, + }, + }, + filter: ['sprite'] + }, + //强行移到xy + { + opcode: 'setXY', + blockType: 'command', + text: this.formatMessage('ArkosExt.setXY'), + arguments: { + x: { + type: 'number', + defaultValue: 100000, + }, + y: { + type: 'number', + defaultValue: 100000, + }, + }, + filter: ['sprite'] + }, + "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 + //获取特效值 + { + opcode: 'getEffect', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getEffect'), + arguments: { + EFFECT: { + type: 'string', + menu: 'effectMenu', + }, + }, + }, + //是否隐藏 + { + opcode: 'isHiding', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.isHiding'), + filter: ['sprite'] + }, + //获取旋转方式 + { + opcode: 'getRotationStyle', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getRotationStyle'), + filter: ['sprite'] + }, + //获取造型0宽1高 + { + opcode: 'getWidthOrHeight', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getWidthOrHeight'), + arguments: { + t: { + type: 'string', + menu: 'WOrH', + }, + }, + }, + //获取角色边缘xy + { + opcode: 'getBoundaryCoord', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getBoundaryCoord'), + arguments: { + t: { + type: 'string', + menu: 'boundaryMenu', + } + }, + filter: ['sprite'] + }, + //是否跑到舞台外 + { + opcode: 'isOutOfSight', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.isOutOfSight'), + filter: ['sprite'] + }, + "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 + //📊清空排序表 + { + opcode: 'clearSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + }, + }, + { + //📊排序表排序方式 + opcode: 'setTypeOfSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.setTypeOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + type: { + type: 'string', + menu: 'sortOrder', + }, + }, + }, + { + //📊将XX加入排序表 + opcode: 'addToSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.addToSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + value: { + type: 'number', + defaultValue: '95', + }, + extra: { + type: 'string', + defaultValue: '20212490', + }, + }, + }, + { + //📊获取排序表第n项 + opcode: 'getFromSortedTableByNo', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getFromSortedTableByNo'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + defaultValue: '1', + menu: 'tableItemPropertyMenu', + }, + }, + }, + { + //📊获取排序表特定名字内容 + opcode: 'getFromSortedTableByName', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getFromSortedTableByName'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + t: { + type: 'string', + defaultValue: '2', + menu: 'tableItemPropertyMenu', + }, + }, + }, + { + //📊获取排序表长度 + opcode: 'lengthOfSortedTable', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + }, + }, + { + //📊删除表中内容 + opcode: 'deleteNameOfSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.deleteNameOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + }, + }, + + "---" + this.formatMessage("ArkosExt.info6"), //🗂️ 临时变量 + + //清空所有临时数据 + { + opcode: 'deleteAllTempData', + blockType: 'command', + text: this.formatMessage('ArkosExt.deleteAllTempData'), + }, + //临时数据量 + { + opcode: 'getCountOfTempData', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getCountOfTempData'), + }, + //删除临时数据 + { + opcode: 'delTempData', + blockType: 'command', + text: this.formatMessage('ArkosExt.delTempData'), + arguments: { + data: { + type: 'string', + defaultValue: 'i', + }, + }, + }, + //设置临时数据 + { + opcode: 'setTempVar', + blockType: 'command', + text: this.formatMessage('ArkosExt.setTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + t: { + type: 'string', + defaultValue: '0', + }, + }, + }, + //增加临时数据 + { + opcode: 'addTempVar', + blockType: 'command', + text: this.formatMessage('ArkosExt.addTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + t: { + type: 'number', + defaultValue: 1, + }, + }, + }, + //增加临时数据 + { + opcode: 'getTempVar', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + }, + }, + //创建或清空临时列表 + { + opcode: 'clearTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + }, + }, + //设置临时列表 + { + opcode: 'initTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.initTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + t: { + type: 'string', + defaultValue: '[1,2,"apple"]', + }, + }, + }, + //向临时列表加入 + { + opcode: 'addTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.addTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + t: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, + //操作临时列表 + { + opcode: 'opTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.opTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + op: { + type: 'string', + menu: 'ListOpMenu', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, + //删除临时列表XX项 + { + opcode: 'delItemOfTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.delItemOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + n: { + type: 'number', + defaultValue: 1, + }, + }, + }, + //获取临时列表XX项 + { + opcode: 'getItemOfTempList', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + n: { + type: 'number', + defaultValue: 1, + }, + }, + }, + //临时列表长度 + { + opcode: 'lengthOfTempList', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + }, + }, + // + "---" + this.formatMessage("30Ext.info"), //30的扩展 + "---" + this.formatMessage("30Ext.info.1"), //造型镜像 + // 镜像造型 + { + opcode: 'mirrorSprite', + blockType: 'command', + text: this.formatMessage('30Ext.block.mirrorSprite'), + arguments: { + mirrorMethod: { + type: 'number', + menu: [{ + text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 + value: 0 + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 + value: 1 + } + ] + } + } + }, + // 清除镜像 + { + opcode: 'clearMirror', + blockType: 'command', + text: this.formatMessage('30Ext.block.clearMirror') + }, + "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 + { + opcode: 'anotherRun', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRun'), + arguments: { + spriteName: { + type: 'string', + menu: spritesMenu + }, + SUBSTACK: { + type: "input_statement" + } + } + }, + { + opcode: 'anotherRunWithClone', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRunWithClone'), + arguments: { + spriteName: { + type: 'string', + menu: spritesMenu + }, + cloneId: { + type: 'number', + defaultValue: 1 + } + SUBSTACK: { + type: "input_statement" + } + } + }, + + ], + menus: { + ListOpMenu: [ + { + text: this.formatMessage('ArkosExt.ListOp1'), //插入 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.ListOp2'), //替换 + value: '2' + }, + { + text: this.formatMessage('ArkosExt.ListOp3'), //增加 + value: '3' + }, + ], + tableItemPropertyMenu: [ + { + text: this.formatMessage('ArkosExt.name'), + value: '1' + }, + { + text: this.formatMessage('ArkosExt.rank'), + value: '2' + }, + { + text: this.formatMessage('ArkosExt.rankValue'), + value: '3' + }, + { + text: this.formatMessage('ArkosExt.extra'), + value: '4' + }, + ], + sortOrder: [ + { + text: this.formatMessage('ArkosExt.asc'), + value: 'asc'//升序 + }, + { + text: this.formatMessage('ArkosExt.desc'), + value: 'desc'//降序 + }, + ], + sortedTableMenu: { + items: 'findAllSortedTable', + acceptReporters: true, + }, + //判断符菜单 + opMenu1: ['<','≤','=','≠',], + opMenu2: ['<','>','≤','≥','=','≠',], + //logicMenu + logicMenu: [ + { + text: this.formatMessage('ArkosExt.or'), + value: 'or' + }, + { + text: this.formatMessage('ArkosExt.and'), + value: 'and' + }, + ], + //角色上下左右边缘 + boundaryMenu: [ + { + text: this.formatMessage('ArkosExt.top'), + value: '1' + }, + { + text: this.formatMessage('ArkosExt.bottom'), + value: '2' + }, + { + text: this.formatMessage('ArkosExt.left'), + value: '3' + }, + { + text: this.formatMessage('ArkosExt.right'), + value: '4' + }, + ], + //0宽1高 菜单 + WOrH: [ + { + text: this.formatMessage('ArkosExt.width'), + value: '0' + }, + { + text: this.formatMessage('ArkosExt.height'), + value: '1' + }, + ], + //特效菜单 + effectMenu: [ + { + text: this.formatMessage('ArkosExt.color'), + value: 'color' + }, + { + text: this.formatMessage('ArkosExt.fisheye'), + value: 'fisheye' + }, + { + text: this.formatMessage('ArkosExt.whirl'), + value: 'whirl' + }, + { + text: this.formatMessage('ArkosExt.pixelate'), + value: 'pixelate' + }, + { + text: this.formatMessage('ArkosExt.mosaic'), + value: 'mosaic' + }, + { + text: this.formatMessage('ArkosExt.brightness'), + value: 'brightness' + }, + { + text: this.formatMessage('ArkosExt.ghost'), + value: 'ghost' + } + ] + //30Ext + spritesMenu: { + items: 'getSpritesMenu' + } + }, + } + } + + strictlyEquals(args) { + // 实际上在这里直接使用严格相等是不太明智的,因为有一定的可能会遇到数字和字符比较, + // 而在Scratch中数字和字符在表现完全一样的时候几乎没有区别。 + // 因此包上Cast.toString()以使得数字和字符能够正常比较(类似 9 = "9" ) + return Cast.toString(args.ONE) === Cast.toString(args.TWO) + } + + getDirFromAToB(args) { + // 一定要先转化为数字; + const X1 = Cast.toNumber(args.X1) + const X2 = Cast.toNumber(args.X2) + const Y1 = Cast.toNumber(args.Y1) + const Y2 = Cast.toNumber(args.Y2) + + // 这里利用atan函数的性质atan(+inf)=90,atan(-inf)=-90,atan(NaN)=NaN可以省很多代码 + let a = Math.atan((X2 - X1) / (Y2 - Y1)) / Math.PI * 180 + (Y1 > Y2 ? 180 : 0) + if (a > 180) a -= 360 + return a; + } + + differenceBetweenDirections(args) { + const a = Cast.toNumber(args.a) + const b = Cast.toNumber(args.b) + let dif = b - a + dif -= Math.round(dif / 360) * 360 + if (dif === -180) dif = 180 + return dif + } + + disFromAToB(args) { + const X1 = Cast.toNumber(args.X1) + const X2 = Cast.toNumber(args.X2) + const Y1 = Cast.toNumber(args.Y1) + const Y2 = Cast.toNumber(args.Y2) + return Math.sqrt((X1 - X2) * (X1 - X2) + (Y1 - Y2) * (Y1 - Y2)) + } + + indexof(args) { + const str = Cast.toString(args.str) + const substr = Cast.toString(args.substr) + const a = str.indexOf(substr, Cast.toNumber(args.pos) - 1) + if (a === -1) { + // Scratch列表中也有查询积木,其中找不到返回的是0。建议维持原有的风格。 + return 0 + } + return a + 1 + } + + insertStr(args) { + const str = Cast.toString(args.str) + const substr = Cast.toString(args.substr) + let pos = Cast.toNumber(args.pos) - 1 + if (pos < 0) { + pos = 0 + } + return str.slice(0, pos) + substr + str.slice(pos) + } + + replaceStr(args) { + const str = Cast.toString(args.str) + const substr = Cast.toString(args.substr) + let start = Cast.toNumber(args.start) + let end = Cast.toNumber(args.end) + if (start > end) { + const t = end + end = start + start = t + } + if (start < 1) start = 1 + return str.slice(0, start - 1) + substr + str.slice(end) + } + + + turnDegreesToDir(args, util) { + const degree = Cast.toNumber(args.degree); + const dir = Cast.toNumber(args.dir); + const dif = this.differenceBetweenDirections({a: util.target.direction, b: dir}); + if(Math.abs(dif) < degree) + util.target.setDirection(dir); + else if(dif < 0) + util.target.setDirection(util.target.direction - degree); + else + util.target.setDirection(util.target.direction + degree); + } + + //获取特效的数值 + getEffect (args, util) { + let effect = Cast.toString(args.EFFECT).toLowerCase(); + if (!util.target.effects.hasOwnProperty(effect)) return 0; + return util.target.effects[effect]; + } + + //角色是否可见 + isHiding (args, util) { + return !util.target.visible; + } + + //获取图层(逝一逝) + // getLayer (args, util) { + // return util.target.layer; + // } + + //获取当前角色的旋转方式 + getRotationStyle (args, util) { + return util.target.rotationStyle; + } + + //获取当前造型的长/宽 + getWidthOrHeight (args, util) { + const costumeSize = util.target.renderer.getCurrentSkinSize(util.target.drawableID); + return costumeSize[args.t]; + } + + //强行设置大小(逝一逝) + setSize (args, util) { + if (util.target.isStage) { + return; + } + if (util.target.renderer) { + args.size = this._clamp(Cast.toNumber(args.size),0.1,100000000) + util.target.size = args.size; + const {direction, scale} = util.target._getRenderedDirectionAndScale(); + util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); + if (util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } + } + util.target.runtime.requestTargetsUpdate(util.target); + } + + _clamp (n, min, max) { + return Math.min(Math.max(n, min), max); + } + + //强行设置XY(逝一逝) + setXY (args, util) { + if (util.target.isStage) return; + args.x = this._clamp(Cast.toNumber(args.x),-100000000,100000000) + args.y = this._clamp(Cast.toNumber(args.y),-100000000,100000000) + const oldX = util.target.x; + const oldY = util.target.y; + util.target.x = args.x; + util.target.y = args.y; + if (util.target.renderer) { + util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); + if (util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } + } else { + util.target.x = x; + util.target.y = y; + } + util.target.emit('TARGET_MOVED', util.target, oldX, oldY, false); + util.target.runtime.requestTargetsUpdate(util.target); + } + + //获取角色边缘的坐标 + getBoundaryCoord (args, util) { + const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); + switch(args.t){ + case '1': + return bounds.top; + case '2': + return bounds.bottom; + case '3': + return bounds.left; + case '4': + return bounds.right; + default: + return ''; + } + } + + //是否在舞台外 + isOutOfSight (args, util) { + // console.log(util.target.runtime.renderer) + // console.log(util.target.renderer) + if (util.target.renderer) { + const stageWidth = util.target.runtime.stageWidth; + const stageHeight = util.target.runtime.stageHeight; + console.log('stageWidth',stageWidth); + console.log('stageHeight',stageHeight); + const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); + if (bounds.right < -stageWidth / 2 || + bounds.left > stageWidth / 2 || + bounds.bottom > stageHeight / 2 || + bounds.top < -stageHeight / 2) { + return true; + } + } + return false; + } + + //形如:<() > + reporterToBoolean (args){ + if(Cast.toString(args.t).toLowerCase() === 'false') return false; + if(args.t === '0') return false; + return (args.t) ? true : false; + } + + compare(a,b,op){ + switch(op){ + case '<': + return Cast.compare(a, b) < 0; + case '>': + return Cast.compare(a, b) > 0; + case '=': + return Cast.compare(a, b) === 0; + case '≤': + return Cast.compare(a, b) <= 0; + case '≥': + return Cast.compare(a, b) >= 0; + case '≠': + return Cast.compare(a, b) !== 0; + default: + return false; + } + } + + //形如:a≤b≤c op1,op2 + compareTwoSides (args){ + return this.compare(args.a, args.b, args.op1)&&this.compare(args.b, args.c, args.op2) + } + + //形如:a≤b且/或>c op1,op2 logic + compareTwoSidesPlus(args){ + switch(args.logic){ + case 'or': + return this.compare(args.a, args.b, args.op1)||this.compare(args.a, args.c, args.op2) + case 'and': + return this.compare(args.a, args.b, args.op1)&&this.compare(args.a, args.c, args.op2) + default: + return false; + } + } + + //数组排序规则 + sortRule(propName,order) { + return (a, b) => { + a = a[propName] + b = b[propName] + if (a > b) return order === 'asc' ? 1 : -1; + else if (a < b) return order === 'asc' ? -1 : 1; + else return 0; + } + } + + //查找所有排序表 + findAllSortedTable() { + const list = []; + let temp = this.sortedTable; + Object.keys(temp).forEach(obj => { + //if ( Array.isArray (temp[obj]) ) { + list.push(obj); + //} + }); + if (list.length === 0) { + list.push({ + text: '-', + value: 'empty', + }); + } + //list.sort(this.sortRule("text")); + return list; + } + + createTableIfNotExist(list){ + if(!(list in this.sortedTable)) + this.sortedTable[list]={order:'desc',list:[]}; + } + + sortTable(list){ + this.sortedTable[list].list.sort(this.sortRule("rankValue",this.sortedTable[list].order)); + } + + //📊清空排序表 + clearSortedTable (args){ + this.createTableIfNotExist(args.list) + this.sortedTable[args.list].list=[]; + } + + //📊设置排序方式 + setTypeOfSortedTable (args){ + this.createTableIfNotExist(args.list) + this.sortedTable[args.list].order=args.type; + this.sortTable(args.list) + } + + //查找在列表中的插入位置(已有则覆盖) + _findPlaceAndInsert(list, order, item){ + //删除已存在的内容 + for (let i = 0; i < list.length; i++) { + if (list[i].name === item.name){ + //删除同名项 + list.splice(i,1); + break; + } + } + //查找插入位置并插入 + for (let i = 0; i < list.length; i++) { + if ((list[i].rankValue > item.rankValue && order ==='asc')|| + (list[i].rankValue < item.rankValue && order ==='desc')){ + //插入在该项前 + list.splice(i,0,item); + return; + } + } + //没找到插入位置,加在末尾 + list.push(item); + } + + //📊将内容加入表 + addToSortedTable (args){ + this.createTableIfNotExist(args.list) + this._findPlaceAndInsert( + this.sortedTable[args.list].list, + this.sortedTable[args.list].order, + {name:args.name, rankValue:args.value, extra:args.extra}); + } + + _getTInItem (item, t, rank){ + if(item === undefined) return ''; + switch(t){ + case '1': + return item.name; + case '2': + return rank; + case '3': + return item.rankValue; + case '4': + return item.extra; + default: + return ''; + } + } + + //📊获取第n项 + getFromSortedTableByNo (args){ + if(!(args.list in this.sortedTable)) return ''; + let list = this.sortedTable[args.list].list; + return this._getTInItem (list[args.n-1], args.t, Cast.toNumber(args.n)); + } + + _getItemIdxByName(list, name){ + for (let i = 0; i < list.length; i++) { + if (list[i].name === name){ + return i; + } + } + return -1; + } + + //📊获取名为XX的项 + getFromSortedTableByName (args){ + if(!(args.list in this.sortedTable)) return ''; + let list = this.sortedTable[args.list].list; + let n = this._getItemIdxByName(list ,args.name) ; + if(n === -1) return ''; + return this._getTInItem (list[n], args.t, n+1); + } + + //📊获取排序表长度 + lengthOfSortedTable (args){ + if(!(args.list in this.sortedTable)) return 0; + return this.sortedTable[args.list].list.length; + } + + //📊删除排序表名为XX的内容 + deleteNameOfSortedTable (args){ + if(!(args.list in this.sortedTable)) return; + let list = this.sortedTable[args.list].list; + let n = this._getItemIdxByName(list ,args.name) ; + if(n === -1) return; + list.splice(n, 1); + } + + //获取颜色HEX码 + colorToHex (args, util) { + let c = Cast.toRgbColorList(args.COLOR) + return Color.rgbToHex({r:c[0],g:c[1],b:c[2]}); + } + + + //🗂️ 临时变量积木 + deleteAllTempData (args) { + this.tempData = {}; + } + + getCountOfTempData (args) { + return Object.keys (this.tempData).length; + } + + delTempData (args) { + delete this.tempData[Cast.toString(args.data)]; + } + + setTempVar (args) { + this.tempData[Cast.toString(args.var)] = args.t; + } + + addTempVar (args) { + this.tempData[Cast.toString(args.var)] = Cast.toNumber(this.tempData[Cast.toString(args.var)]) + Cast.toNumber(args.t); + } + + getTempVar (args) { + let temp = this.tempData[Cast.toString(args.var)] + if(Array.isArray(temp)) return JSON.stringify(temp); + return Cast.toString(temp); + } + + //创建或清空 + clearTempList (args) { + this.tempData[Cast.toString(args.list)] = []; + } + + initTempList (args) { + try { + let content = JSON.parse(Cast.toString(args.t)) + if(Array.isArray(content)) { + this.tempData[Cast.toString(args.list)] = content; + } + } catch (e) { + + } + } + + addTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + list.push(Cast.toString(args.t)); + } + + opTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + let n = Cast.toNumber(args.n) + if(n < 1 || n >list.length+1) return; + n -= 1; + switch(args.op){ + case '1'://插入 + list.splice(n,0,Cast.toString(args.t)); + return; + case '2'://替换 + list[n] = Cast.toString(args.t); + return; + case '3'://增加 + list[n] = Cast.toNumber(list[n]) + Cast.toNumber(args.t); + return; + default: + return; + } + } + + delItemOfTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + let n = Cast.toNumber(args.n) + if(n < 1 || n >list.length) return; + n -= 1; + list.splice(n,1); + } + + getItemOfTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return ''; + let n = Cast.toNumber(args.n) + if(n < 1 || n >list.length) return ''; + n -= 1; + return Cast.toString(list[n]); + } + + lengthOfTempList (args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return 0; + return list.length; + } + // + //30Ext + //诶诶 HCN居然不给另外开扩展 只好插队了 + // + //菜单 + //动态菜单: 角色菜单 + getSpritesMenu(){ + var sprites = []; + for(const targetId in vm.runtime.targets) { + if(!vm.runtime.targets.hasOwnProperty(targetId)) continue; + if(!vm.runtime.targets[targetId].isOriginal) continue; + if(vm.runtime.targets[targetId] === vm.editingTarget) continue; //排除自己 + let name = vm.runtime.targets[targetId].sprite.name; + sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 + } + return sprites; + } + // + //角色造型操作 + // + //镜像造型 + mirrorSprite(args, util){ + util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; + } + //清除镜像 + clearMirror(args, util){ + for (let i=0; i<2; i++){ + let foo = util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]; + foo = Math.abs(foo); + } + } + // + //角色跨域操作 + // + //跨域执行 + anotherRun(args, util){ + if(!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = util.target.sprite.clones[0]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } + //跨域克隆体执行 + anotherRunWithClone(args, util){ + if(!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = util.target.sprite.clones[args.cloneId]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } +} + + + +window.tempExt = { + Extension: ArkosExtensions, + info: { + name: 'hcn.extensionName', + description: 'hcn.description', + extensionId: 'hcnTest', + // iconURL: icon, + // insetIconURL: cover, + featured: true, + disabled: false, + collaborator: 'only for hcn test', + }, + l10n: { + 'zh-cn': { + 'hcn.extensionName': 'hcn 的测试', + 'hcn.description': 'hcn 的测试', + }, + en: { + 'hcn.extensionName': 'hcn test', + 'hcn.description': 'hcn test', + }, + }, + } diff --git a/java30433/30Ext.js b/java30433/30Ext.js deleted file mode 100644 index 4077219a..00000000 --- a/java30433/30Ext.js +++ /dev/null @@ -1,191 +0,0 @@ -class _30ExtExtensions { - constructor(runtime) { - this.runtime = runtime; - this._formatMessage = runtime.getFormatMessage({ - 'zh-cn': { - '30Ext.extensionName': '工具箱', - '30Ext.info.1': '造型镜像操作', - '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', - '30Ext.menu.mirrorMethod.1': '左右镜像', - '30Ext.menu.mirrorMethod.1': '上下镜像', - '30Ext.block.clearMirror': '清除角色镜像变换', - '30Ext.info.2': '角色跨域操作', - '30Ext.block.anotherRun': '让[sprite]运行[SUBSTACK]', - '30Ext.block.anotherRunWithClone': '让[sprite]的第[cloneId]个克隆体运行[SUBSTACK]' - }, - - en: { - '30Ext.extensionName': 'Toolbox', - '30Ext.info.1': 'Mirror transform', - '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', - '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', - '30Ext.menu.mirrorMethod.1': 'Vertical mirror transform', - '30Ext.block.clearMirror': 'Clear the mirror transform', - '30Ext.info.2': 'Cross sprite operation', - '30Ext.block.anotherRun': 'Let [sprite] run[SUBSTACK]', - '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [sprite] run[SUBSTACK]' - } - }) - } - formatMessage(id) { - return this._formatMessage({ - id, - default: id, - description: id, - }) - } - getInfo() { - return { - id: '30Ext', // 拓展id - name: this.formatMessage('30Ext.extensionName'), - - color1: '#FF8196', - // menuIconURI: icon, - // blockIconURI: icon, - blocks: [ - "---" + this.formatMessage("30Ext.info.1"), //角色造型操作 - // 镜像造型 - { - opcode: 'mirrorSprite', - blockType: 'command', - text: this.formatMessage('30Ext.block.mirrorSprite'), - arguments: { - mirrorMethod: { - type: 'number', - menu: [{ - text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: 0 - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: 1 - } - ] - } - } - }, - // 清除镜像 - { - opcode: 'clearMirror', - blockType: 'command', - text: this.formatMessage('30Ext.block.clearMirror') - }, - "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 - { - opcode: 'anotherRun', - blockType: 'conditional', - text: this.formatMessage('30Ext.block.anotherRun'), - arguments: { - spriteName: { - type: 'string', - menu: spritesMenu - }, - SUBSTACK: { - type: "input_statement" - } - } - }, - { - opcode: 'anotherRunWithClone', - blockType: 'conditional', - text: this.formatMessage('30Ext.block.anotherRunWithClone'), - arguments: { - spriteName: { - type: 'string', - menu: spritesMenu - }, - cloneId: { - type: 'number', - defaultValue: 1 - } - SUBSTACK: { - type: "input_statement" - } - } - } - ], - menus: { - spritesMenu: { - items: 'getSpritesMenu' - } - } - } - } - //菜单 - //动态菜单: 角色菜单 - getSpritesMenu(){ - var sprites = []; - for(const targetId in vm.runtime.targets) { - if(!vm.runtime.targets.hasOwnProperty(targetId)) continue; - if(!vm.runtime.targets[targetId].isOriginal) continue; - if(vm.runtime.targets[targetId] === vm.editingTarget) continue; //排除自己 - let name = vm.runtime.targets[targetId].sprite.name; - sprites.push(name); //疑问? 菜单不应该返回诸如[["显示", "内容"],["显示2", "内容值"]]的东西吗 我看扩展库test.js是这样写 - } - return sprites; - } - // - //角色造型操作 - // - //镜像造型 - mirrorSprite(args, util){ - util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; - } - //清除镜像 - clearMirror(args, util){ - for (let i=0; i<2; i++){ - let foo = util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]; - foo = Math.abs(foo); - } - } - // - //角色跨域操作 - // - //跨域执行 - anotherRun(args, util){ - if(!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[0]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; - } - } - //跨域克隆体执行 - anotherRunWithClone(args, util){ - if(!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[args.cloneId]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; - } - } -} -window.tempExt = { - Extension: _30Extensions, - info: { - name: '30Ext.extensionName', - description: '30Ext.description', - extensionId: '30Ext', - // iconURL: icon, - // insetIconURL: cover, - featured: true, - disabled: false, - collaborator: '_30 x CCW' - }, - l10n: { - 'zh-cn': { - '30Ext.extensionName': 'java30433的工具箱', - '30Ext.description': '造型镜像&角色跨域操作 持续更新中...' - }, - en: { - '30Ext.extensionName': "java30433's Toolbox", - '30Ext.description': 'Mirror transform&Cross sprite operation Continuous updating.' - } - } -}; \ No newline at end of file diff --git a/java30433/assets/1273b53562550d34c80dc66a2737dfa4.png b/java30433/assets/1273b53562550d34c80dc66a2737dfa4.png deleted file mode 100644 index 60b19381cfd5b89c49ec5492945da810499ca14e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 100847 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

>yE!}~IZsG_n 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 |XT*H?)%R4H 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 diff --git a/java30433/assets/cover.icpr b/java30433/assets/cover.icpr deleted file mode 100644 index f1161fe69420ba64ad8485a472691f56e85b3165..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3494732 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 From 1c1cbdce3708505f4eab3c746b57ecb15aff41fe Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Fri, 29 Jul 2022 22:25:54 +0800 Subject: [PATCH 071/357] =?UTF-8?q?=E8=A1=A5=E4=B8=8A=E9=80=97=E5=8F=B7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 1f008242..73c3ed2b 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -94,7 +94,7 @@ class ArkosExtensions { 'ArkosExt.getItemOfTempList': '🗂️临时列表[list]第[n]项', 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', - '30Ext.info': '以下扩展由_30提供', + '30Ext.info': '✨ 以下扩展由_30提供', '30Ext.info.1': '造型镜像操作', '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', '30Ext.menu.mirrorMethod.1': '左右镜像', @@ -185,7 +185,7 @@ class ArkosExtensions { 'ArkosExt.getItemOfTempList': '🗂️item[n]of temp list[list]', 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', - '30Ext.info': 'The following extensions are provided by _30', + '30Ext.info': '✨ The following extensions are provided by _30', '30Ext.info.1': 'Mirror transform', '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', @@ -917,7 +917,7 @@ class ArkosExtensions { cloneId: { type: 'number', defaultValue: 1 - } + }, SUBSTACK: { type: "input_statement" } @@ -1046,7 +1046,7 @@ class ArkosExtensions { text: this.formatMessage('ArkosExt.ghost'), value: 'ghost' } - ] + ], //30Ext spritesMenu: { items: 'getSpritesMenu' From 748f30bfab1727f4f6ac1abd715ff5925e739889 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Fri, 29 Jul 2022 22:41:44 +0800 Subject: [PATCH 072/357] =?UTF-8?q?=E8=8F=9C=E5=8D=95=E9=A1=B9=E5=8A=A0?= =?UTF-8?q?=E4=B8=8A=E5=BC=95=E5=8F=B7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 1f008242..3551e6d3 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -898,7 +898,7 @@ class ArkosExtensions { arguments: { spriteName: { type: 'string', - menu: spritesMenu + menu: 'spritesMenu' }, SUBSTACK: { type: "input_statement" @@ -912,7 +912,7 @@ class ArkosExtensions { arguments: { spriteName: { type: 'string', - menu: spritesMenu + menu: 'spritesMenu' }, cloneId: { type: 'number', From b48fc8bfb3fba85d54c0aee0c5f3b266c8890950 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Fri, 29 Jul 2022 22:44:31 +0800 Subject: [PATCH 073/357] =?UTF-8?q?=E8=8F=9C=E5=8D=95=E5=8A=A0=E5=BC=95?= =?UTF-8?q?=E5=8F=B7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 73c3ed2b..d2f5bbb0 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -95,12 +95,12 @@ class ArkosExtensions { 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', '30Ext.info': '✨ 以下扩展由_30提供', - '30Ext.info.1': '造型镜像操作', + '30Ext.info.1': '🪞 造型镜像操作', '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', '30Ext.menu.mirrorMethod.1': '左右镜像', '30Ext.menu.mirrorMethod.1': '上下镜像', '30Ext.block.clearMirror': '清除角色镜像变换', - '30Ext.info.2': '角色跨域操作', + '30Ext.info.2': '🛸 角色跨域操作', '30Ext.block.anotherRun': '让[sprite]运行[SUBSTACK]', '30Ext.block.anotherRunWithClone': '让[sprite]的第[cloneId]个克隆体运行[SUBSTACK]' }, @@ -186,12 +186,12 @@ class ArkosExtensions { 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', '30Ext.info': '✨ The following extensions are provided by _30', - '30Ext.info.1': 'Mirror transform', + '30Ext.info.1': '🪞 Mirror transform', '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', '30Ext.menu.mirrorMethod.1': 'Vertical mirror transform', '30Ext.block.clearMirror': 'Clear the mirror transform', - '30Ext.info.2': 'Cross sprite operation', + '30Ext.info.2': '🛸 Cross sprite operation', '30Ext.block.anotherRun': 'Let [sprite] run[SUBSTACK]', '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [sprite] run[SUBSTACK]' }, @@ -898,7 +898,7 @@ class ArkosExtensions { arguments: { spriteName: { type: 'string', - menu: spritesMenu + menu: 'spritesMenu' }, SUBSTACK: { type: "input_statement" @@ -912,7 +912,7 @@ class ArkosExtensions { arguments: { spriteName: { type: 'string', - menu: spritesMenu + menu: 'spritesMenu' }, cloneId: { type: 'number', From 44ad86387028c47216544cde95a95e0238e63e9a Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Fri, 29 Jul 2022 22:59:08 +0800 Subject: [PATCH 074/357] =?UTF-8?q?=E5=88=A0=E6=8E=89SUBSTACK=E5=8F=82?= =?UTF-8?q?=E6=95=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 33 +++++++++++++++++---------------- 1 file changed, 17 insertions(+), 16 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index d2f5bbb0..6c6625c0 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -872,15 +872,7 @@ class ArkosExtensions { arguments: { mirrorMethod: { type: 'number', - menu: [{ - text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: 0 - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: 1 - } - ] + menu: 'mirrorMenu' } } }, @@ -900,9 +892,9 @@ class ArkosExtensions { type: 'string', menu: 'spritesMenu' }, - SUBSTACK: { - type: "input_statement" - } + //SUBSTACK: { + // type: "input_statement" + //} } }, { @@ -918,9 +910,9 @@ class ArkosExtensions { type: 'number', defaultValue: 1 }, - SUBSTACK: { - type: "input_statement" - } + //SUBSTACK: { + // type: "input_statement" + //} } }, @@ -1050,7 +1042,16 @@ class ArkosExtensions { //30Ext spritesMenu: { items: 'getSpritesMenu' - } + }, + mirrorMenu: [{ + text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 + value: 0 + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 + value: 1 + } + ] }, } } From 619d9d85aa2115de798b290595ef99e868c0c174 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Fri, 29 Jul 2022 23:04:23 +0800 Subject: [PATCH 075/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E4=BF=AE=E5=A4=8D?= =?UTF-8?q?=E8=B7=A8=E5=9F=9F=E6=89=A7=E8=A1=8C=E5=9D=97?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 44 ++++++++++++++++++++++-------------------- 1 file changed, 23 insertions(+), 21 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index d2f5bbb0..951637ee 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -101,8 +101,8 @@ class ArkosExtensions { '30Ext.menu.mirrorMethod.1': '上下镜像', '30Ext.block.clearMirror': '清除角色镜像变换', '30Ext.info.2': '🛸 角色跨域操作', - '30Ext.block.anotherRun': '让[sprite]运行[SUBSTACK]', - '30Ext.block.anotherRunWithClone': '让[sprite]的第[cloneId]个克隆体运行[SUBSTACK]' + '30Ext.block.anotherRun': '让[sprite]运行', + '30Ext.block.anotherRunWithClone': '让[sprite]的第[cloneId]个克隆体运行' }, en: { @@ -192,8 +192,8 @@ class ArkosExtensions { '30Ext.menu.mirrorMethod.1': 'Vertical mirror transform', '30Ext.block.clearMirror': 'Clear the mirror transform', '30Ext.info.2': '🛸 Cross sprite operation', - '30Ext.block.anotherRun': 'Let [sprite] run[SUBSTACK]', - '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [sprite] run[SUBSTACK]' + '30Ext.block.anotherRun': 'Let [sprite] run', + '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [sprite] run' }, }) } @@ -872,15 +872,7 @@ class ArkosExtensions { arguments: { mirrorMethod: { type: 'number', - menu: [{ - text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: 0 - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: 1 - } - ] + menu: 'mirrorMenu' } } }, @@ -900,9 +892,9 @@ class ArkosExtensions { type: 'string', menu: 'spritesMenu' }, - SUBSTACK: { - type: "input_statement" - } + //SUBSTACK: { + // type: "input_statement" + //} } }, { @@ -918,9 +910,9 @@ class ArkosExtensions { type: 'number', defaultValue: 1 }, - SUBSTACK: { - type: "input_statement" - } + //SUBSTACK: { + // type: "input_statement" + //} } }, @@ -1050,7 +1042,16 @@ class ArkosExtensions { //30Ext spritesMenu: { items: 'getSpritesMenu' - } + }, + mirrorMenu: [{ + text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 + value: 0 + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 + value: 1 + } + ] }, } } @@ -1569,6 +1570,7 @@ class ArkosExtensions { // //跨域执行 anotherRun(args, util){ + console.info(util); // if(!util.thread.ex_30Ext_count) { util.thread.ex_30Ext_count = true; util.thread.ex_30Ext_oldTarget = util.thread.target; @@ -1617,4 +1619,4 @@ window.tempExt = { 'hcn.description': 'hcn test', }, }, - } + } \ No newline at end of file From 7df01cbc2a2dd9dbba6cc0069f4c8ccb22c684f0 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Fri, 29 Jul 2022 23:19:21 +0800 Subject: [PATCH 076/357] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E5=8F=82=E6=95=B0?= =?UTF-8?q?=EF=BC=8C=E4=BF=AE=E6=94=B9=E4=B8=8B=E6=8B=89=E5=88=97=E8=A1=A8?= =?UTF-8?q?=E6=98=BE=E7=A4=BA=E9=94=99=E8=AF=AF=EF=BC=8C=E9=95=9C=E5=83=8F?= =?UTF-8?q?=E5=90=8Eredraw?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 20 ++++++++++++-------- 1 file changed, 12 insertions(+), 8 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 6c6625c0..070f0737 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -98,11 +98,11 @@ class ArkosExtensions { '30Ext.info.1': '🪞 造型镜像操作', '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', '30Ext.menu.mirrorMethod.1': '左右镜像', - '30Ext.menu.mirrorMethod.1': '上下镜像', + '30Ext.menu.mirrorMethod.2': '上下镜像', '30Ext.block.clearMirror': '清除角色镜像变换', '30Ext.info.2': '🛸 角色跨域操作', - '30Ext.block.anotherRun': '让[sprite]运行[SUBSTACK]', - '30Ext.block.anotherRunWithClone': '让[sprite]的第[cloneId]个克隆体运行[SUBSTACK]' + '30Ext.block.anotherRun': '让[spriteName]运行', + '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' }, en: { @@ -189,11 +189,11 @@ class ArkosExtensions { '30Ext.info.1': '🪞 Mirror transform', '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', - '30Ext.menu.mirrorMethod.1': 'Vertical mirror transform', + '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', '30Ext.block.clearMirror': 'Clear the mirror transform', '30Ext.info.2': '🛸 Cross sprite operation', - '30Ext.block.anotherRun': 'Let [sprite] run[SUBSTACK]', - '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [sprite] run[SUBSTACK]' + '30Ext.block.anotherRun': 'Let [spriteName] run', + '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' }, }) } @@ -1557,13 +1557,17 @@ class ArkosExtensions { //镜像造型 mirrorSprite(args, util){ util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; + util.target.emit(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, this); + util.target.runtime.requestRedraw(); } //清除镜像 clearMirror(args, util){ for (let i=0; i<2; i++){ - let foo = util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]; - foo = Math.abs(foo); + util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]= + Math.abs(util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]); } + util.target.emit(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, this); + util.target.runtime.requestRedraw(); } // //角色跨域操作 From 610203f271d92bb209c50dd42970614a2e92f660 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Fri, 29 Jul 2022 23:27:08 +0800 Subject: [PATCH 077/357] =?UTF-8?q?=E6=8A=8A=E5=87=BA=E7=8E=B0vm=E7=9A=84?= =?UTF-8?q?=E5=9C=B0=E6=96=B9=E9=83=BD=E6=8D=A2=E6=8E=89?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 070f0737..042cbb49 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1542,11 +1542,11 @@ class ArkosExtensions { //动态菜单: 角色菜单 getSpritesMenu(){ var sprites = []; - for(const targetId in vm.runtime.targets) { - if(!vm.runtime.targets.hasOwnProperty(targetId)) continue; - if(!vm.runtime.targets[targetId].isOriginal) continue; - if(vm.runtime.targets[targetId] === vm.editingTarget) continue; //排除自己 - let name = vm.runtime.targets[targetId].sprite.name; + for(const targetId in this.runtime.targets) { + if(!this.runtime.targets.hasOwnProperty(targetId)) continue; + if(!this.runtime.targets[targetId].isOriginal) continue; + if(this.runtime.targets[targetId] === this.runtime._editingTarget) continue; //排除自己 + let name = this.runtime.targets[targetId].sprite.name; sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 } return sprites; From 43903a517962a5a116b8da50a5b796ae5ff6ba9c Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Fri, 29 Jul 2022 23:32:41 +0800 Subject: [PATCH 078/357] =?UTF-8?q?=E4=BF=AEbug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 042cbb49..24ce057d 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1557,7 +1557,7 @@ class ArkosExtensions { //镜像造型 mirrorSprite(args, util){ util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; - util.target.emit(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, this); + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); util.target.runtime.requestRedraw(); } //清除镜像 @@ -1566,7 +1566,7 @@ class ArkosExtensions { util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]= Math.abs(util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]); } - util.target.emit(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, this); + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); util.target.runtime.requestRedraw(); } // From 6cc1c56df8a5c27c0e5fa99386ed721fea8445b6 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Fri, 29 Jul 2022 23:36:44 +0800 Subject: [PATCH 079/357] =?UTF-8?q?=E6=B5=8B=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 24ce057d..ecf54f14 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1557,8 +1557,8 @@ class ArkosExtensions { //镜像造型 mirrorSprite(args, util){ util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); + //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + //util.target.runtime.requestRedraw(); } //清除镜像 clearMirror(args, util){ @@ -1566,8 +1566,8 @@ class ArkosExtensions { util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]= Math.abs(util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]); } - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); + //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + //util.target.runtime.requestRedraw(); } // //角色跨域操作 From 2257f7df930647d6830963bf18da8679bdab22e7 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Sat, 30 Jul 2022 00:24:02 +0800 Subject: [PATCH 080/357] test2 --- Arkos/projectWith30.js | 86 ++++++++++++++++++++++++------------------ 1 file changed, 50 insertions(+), 36 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 951637ee..cc9e5f44 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -98,11 +98,11 @@ class ArkosExtensions { '30Ext.info.1': '🪞 造型镜像操作', '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', '30Ext.menu.mirrorMethod.1': '左右镜像', - '30Ext.menu.mirrorMethod.1': '上下镜像', + '30Ext.menu.mirrorMethod.2': '上下镜像', '30Ext.block.clearMirror': '清除角色镜像变换', '30Ext.info.2': '🛸 角色跨域操作', - '30Ext.block.anotherRun': '让[sprite]运行', - '30Ext.block.anotherRunWithClone': '让[sprite]的第[cloneId]个克隆体运行' + '30Ext.block.anotherRun': '让[spriteName]运行', + '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' }, en: { @@ -189,11 +189,11 @@ class ArkosExtensions { '30Ext.info.1': '🪞 Mirror transform', '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', - '30Ext.menu.mirrorMethod.1': 'Vertical mirror transform', + '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', '30Ext.block.clearMirror': 'Clear the mirror transform', '30Ext.info.2': '🛸 Cross sprite operation', - '30Ext.block.anotherRun': 'Let [sprite] run', - '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [sprite] run' + '30Ext.block.anotherRun': 'Let [spriteName] run', + '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' }, }) } @@ -871,7 +871,7 @@ class ArkosExtensions { text: this.formatMessage('30Ext.block.mirrorSprite'), arguments: { mirrorMethod: { - type: 'number', + type: 'string', menu: 'mirrorMenu' } } @@ -892,9 +892,9 @@ class ArkosExtensions { type: 'string', menu: 'spritesMenu' }, - //SUBSTACK: { - // type: "input_statement" - //} + SUBSTACK: { + type: "input_statement" + } } }, { @@ -910,9 +910,9 @@ class ArkosExtensions { type: 'number', defaultValue: 1 }, - //SUBSTACK: { - // type: "input_statement" - //} + SUBSTACK: { + type: "input_statement" + } } }, @@ -1045,11 +1045,11 @@ class ArkosExtensions { }, mirrorMenu: [{ text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: 0 + value: '0' }, { text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: 1 + value: '1' } ] }, @@ -1542,11 +1542,11 @@ class ArkosExtensions { //动态菜单: 角色菜单 getSpritesMenu(){ var sprites = []; - for(const targetId in vm.runtime.targets) { - if(!vm.runtime.targets.hasOwnProperty(targetId)) continue; - if(!vm.runtime.targets[targetId].isOriginal) continue; - if(vm.runtime.targets[targetId] === vm.editingTarget) continue; //排除自己 - let name = vm.runtime.targets[targetId].sprite.name; + for(const targetId in this.runtime.targets) { + if(!this.runtime.targets.hasOwnProperty(targetId)) continue; + if(!this.runtime.targets[targetId].isOriginal) continue; + if(this.runtime.targets[targetId] === this.runtime._editingTarget) continue; //排除自己 + let name = this.runtime.targets[targetId].sprite.name; sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 } return sprites; @@ -1556,41 +1556,55 @@ class ArkosExtensions { // //镜像造型 mirrorSprite(args, util){ - util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; + if (!util.target.ext30_isHook) { + util.target.ext30_mirror0 = 1; + util.target.ext30_mirror1 = 1; + const oldSet = util.target.prototype.setSize; + util.target.prototype.setSize = function (size) { + oldSet.call(this, size); + if (util.target.ext30_mirror0) util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[0] *= util.target.ext30_mirror0; + if (util.target.ext30_mirror1) util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[1] *= util.target.ext30_mirror1; + } + util.target.ext30_isHook = true; + } + util.target['ext30_mirror' + args.mirrorMethod] *= -1; + util.target.setSize(util.target._size); + //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + //util.target.runtime.requestRedraw(); } //清除镜像 clearMirror(args, util){ - for (let i=0; i<2; i++){ - let foo = util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]; - foo = Math.abs(foo); - } + util.target.ext30_mirror0 = 1; + util.target.ext30_mirror1 = 1; + util.target.setSize(util.target._size); + //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + //util.target.runtime.requestRedraw(); } // //角色跨域操作 // //跨域执行 anotherRun(args, util){ - console.info(util); // - if(!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; + if(!util.thread.ext30_count) { + util.thread.ext30_count = true; + util.thread.ext30_oldTarget = util.thread.target; util.thread.target = util.target.sprite.clones[0]; util.startBranch(1, true); } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; + util.thread.target = util.thread.ext30_oldTarget; + util.thread.ext30_count = false; } } //跨域克隆体执行 anotherRunWithClone(args, util){ - if(!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; + if(!util.thread.ext30_count) { + util.thread.ext30_count = true; + util.thread.ext30_oldTarget = util.thread.target; util.thread.target = util.target.sprite.clones[args.cloneId]; util.startBranch(1, true); } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; + util.thread.target = util.thread.ext30_oldTarget; + util.thread.ext30_count = false; } } } From bb654060c0d4807bab4fd87f6b98284cb92c814e Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sat, 30 Jul 2022 19:58:56 +0800 Subject: [PATCH 081/357] =?UTF-8?q?=E5=A2=9E=E5=8A=A0=E4=B8=B4=E6=97=B6?= =?UTF-8?q?=E5=AE=B9=E5=99=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 862 ++++++++++++++++++++++++++--------------- 1 file changed, 556 insertions(+), 306 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index ecf54f14..7f9bbd2a 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -2,16 +2,17 @@ import Cast from '../utils/cast.js' import Color from '../utils/color.js' // import cover from './assets/icon.svg' // import icon from './assets/icon.svg' +//鸣谢:-6 优化代码和修复了一些 bug;_30 提供了部分拓展积木 console.log(Cast.toNumber('123')) console.log(Cast.toNumber('aab')) class ArkosExtensions { constructor(runtime) { this.runtime = runtime - this.tempData={} - this.sortedTable={ - list1:{order:'desc',list:[]}, - list2:{order:'desc',list:[]}, + this.tempData = {} + this.sortedTable = { + list1: { order: 'desc', list: [] }, + list2: { order: 'desc', list: [] }, } this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { @@ -57,7 +58,7 @@ class ArkosExtensions { 'ArkosExt.and': '且', 'ArkosExt.or': '或', - + 'ArkosExt.clearSortedTable': '📊清空排序表[list]', 'ArkosExt.setTypeOfSortedTable': '📊将排序表[list]的排序方式设为[type]', 'ArkosExt.addToSortedTable': '📊将内容(重名的则覆盖)[name],排序值[value]加入排序表[list],附加信息[extra]', @@ -81,7 +82,7 @@ class ArkosExtensions { 'ArkosExt.setTempVar': '🗂️将临时变量[var]设为[t]', 'ArkosExt.addTempVar': '🗂️将临时变量[var]增加[t]', - 'ArkosExt.getTempVar': '🗂️临时变量[var]', + 'ArkosExt.getTempVar': '🗂️临时数据[var]', 'ArkosExt.clearTempList': '🗂️创建或清空临时列表[list]', 'ArkosExt.initTempList': '🗂️临时列表[list]内容设为[t]', @@ -93,16 +94,28 @@ class ArkosExtensions { 'ArkosExt.delItemOfTempList': '🗂️删除临时列表[list]第[n]项', 'ArkosExt.getItemOfTempList': '🗂️临时列表[list]第[n]项', 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', - - '30Ext.info': '✨ 以下扩展由_30提供', - '30Ext.info.1': '🪞 造型镜像操作', - '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', - '30Ext.menu.mirrorMethod.1': '左右镜像', - '30Ext.menu.mirrorMethod.2': '上下镜像', - '30Ext.block.clearMirror': '清除角色镜像变换', - '30Ext.info.2': '🛸 角色跨域操作', - '30Ext.block.anotherRun': '让[spriteName]运行', - '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' + + 'ArkosExt.clearTempCon': '🗂️创建或清空临时容器[con]', + 'ArkosExt.initTempCon': '🗂️临时容器[con]内容设为[t]', + 'ArkosExt.opTempCon': '🗂️临时容器[con]中的[c][op][t]', + 'ArkosExt.conOp1': '设为', + 'ArkosExt.conOp2': '增加', + 'ArkosExt.delItemOfTempCon': '🗂️删除临时容器[con]中名为[c]的内容', + 'ArkosExt.getItemOfTempConByName': '🗂️临时容器[con]中的[c]', + 'ArkosExt.getItemOfTempConByNo': '🗂️临时容器[con]第[n]项的[t]', + 'ArkosExt.conInfo1': '名称', + 'ArkosExt.conInfo2': '内容', + 'ArkosExt.lengthOfTempCon': '🗂️临时容器[con]中内容数', + + '30Ext.info': '✨ 以下扩展由_30提供', + '30Ext.info.1': '🪞 造型镜像操作', + '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', + '30Ext.menu.mirrorMethod.1': '左右镜像', + '30Ext.menu.mirrorMethod.2': '上下镜像', + '30Ext.block.clearMirror': '清除角色镜像变换', + '30Ext.info.2': '🛸 角色跨域操作', + '30Ext.block.anotherRun': '让[spriteName]运行', + '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' }, en: { @@ -157,7 +170,7 @@ class ArkosExtensions { 'ArkosExt.extra': 'extra', 'ArkosExt.colorToHex': 'get code of color[COLOR]', - + 'ArkosExt.info1': '🚶 Coordinate and Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', @@ -171,9 +184,9 @@ class ArkosExtensions { 'ArkosExt.setTempVar': '🗂️set temp var[var] to [t]', 'ArkosExt.addTempVar': '🗂️change temp var[var] by [t]', - 'ArkosExt.getTempVar': '🗂️temp var[var]', + 'ArkosExt.getTempVar': '🗂️temp data[var]', + - 'ArkosExt.clearTempList': '🗂️create or clear temp list[list]', 'ArkosExt.initTempList': '🗂️set temp list[list]to[t]', 'ArkosExt.addTempList': '🗂️add[t] to temp list[list]', @@ -184,16 +197,28 @@ class ArkosExtensions { 'ArkosExt.delItemOfTempList': '🗂️delete [n]of temp list[list]', 'ArkosExt.getItemOfTempList': '🗂️item[n]of temp list[list]', 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', - - '30Ext.info': '✨ The following extensions are provided by _30', - '30Ext.info.1': '🪞 Mirror transform', - '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', - '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', - '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', - '30Ext.block.clearMirror': 'Clear the mirror transform', - '30Ext.info.2': '🛸 Cross sprite operation', - '30Ext.block.anotherRun': 'Let [spriteName] run', - '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' + + 'ArkosExt.clearTempCon': '🗂️create or clear temp container[con]', + 'ArkosExt.initTempCon': '🗂️set temp container[con]to[t]', + 'ArkosExt.opTempCon': '🗂️[c]in temp container[con][op][t]', + 'ArkosExt.conOp1': 'set to', + 'ArkosExt.conOp2': 'change by', + 'ArkosExt.delItemOfTempCon': '🗂️delete[c]in temp container[con]', + 'ArkosExt.getItemOfTempConByName': '🗂️[c]in temp container[con]', + 'ArkosExt.getItemOfTempConByNo': '🗂️[t] of #[n] in temp container[con]', + 'ArkosExt.conInfo1': 'name', + 'ArkosExt.conInfo2': 'content', + 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', + + '30Ext.info': '✨ The following extensions are provided by _30', + '30Ext.info.1': '🪞 Mirror transform', + '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', + '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', + '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', + '30Ext.block.clearMirror': 'Clear the mirror transform', + '30Ext.info.2': '🛸 Cross sprite operation', + '30Ext.block.anotherRun': 'Let [spriteName] run', + '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' }, }) } @@ -209,7 +234,7 @@ class ArkosExtensions { getInfo() { return { id: 'hcnTest', // 拓展id - name: this.formatMessage('ArkosExt.extensionName'), + name: this.formatMessage('ArkosExt.extensionName'), color1: '#FF8383', // menuIconURI: icon, @@ -861,63 +886,204 @@ class ArkosExtensions { }, }, }, - // - "---" + this.formatMessage("30Ext.info"), //30的扩展 - "---" + this.formatMessage("30Ext.info.1"), //造型镜像 - // 镜像造型 - { - opcode: 'mirrorSprite', - blockType: 'command', - text: this.formatMessage('30Ext.block.mirrorSprite'), - arguments: { - mirrorMethod: { - type: 'number', - menu: 'mirrorMenu' - } - } - }, - // 清除镜像 - { - opcode: 'clearMirror', - blockType: 'command', - text: this.formatMessage('30Ext.block.clearMirror') - }, - "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 - { - opcode: 'anotherRun', - blockType: 'conditional', - text: this.formatMessage('30Ext.block.anotherRun'), - arguments: { - spriteName: { - type: 'string', - menu: 'spritesMenu' - }, - //SUBSTACK: { - // type: "input_statement" - //} - } - }, - { - opcode: 'anotherRunWithClone', - blockType: 'conditional', - text: this.formatMessage('30Ext.block.anotherRunWithClone'), - arguments: { - spriteName: { - type: 'string', - menu: 'spritesMenu' - }, - cloneId: { - type: 'number', - defaultValue: 1 - }, - //SUBSTACK: { - // type: "input_statement" - //} - } - }, - + + + //创建或清空临时容器 + { + opcode: 'clearTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + }, + }, + //设置临时容器 + { + opcode: 'initTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.initTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + t: { + type: 'string', + defaultValue: '{"coins":200,"backpack":["wood","bread"]}', + }, + }, + }, + //操作临时容器 + { + opcode: 'opTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.opTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + op: { + type: 'string', + menu: 'conOpMenu', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + t: { + type: 'string', + defaultValue: '520', + }, + }, + }, + //删除临时容器名为xx的内容 + { + opcode: 'delItemOfTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.delItemOfTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + }, + }, + //获取临时容器名为XX的内容 + { + opcode: 'getItemOfTempConByName', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempConByName'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + }, + }, + //获取临时容器第n项的xx + { + opcode: 'getItemOfTempConByNo', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempConByNo'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + menu: 'conInfoMenu', + defaultValue: '2', + }, + }, + }, + //临时容器长度 + { + opcode: 'lengthOfTempCon', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + }, + }, + + + // + "---" + this.formatMessage("30Ext.info"), //感谢30提供的扩展 + "---" + this.formatMessage("30Ext.info.1"), //造型镜像 + // 镜像造型 + { + opcode: 'mirrorSprite', + blockType: 'command', + text: this.formatMessage('30Ext.block.mirrorSprite'), + arguments: { + mirrorMethod: { + type: 'number', + menu: 'mirrorMenu' + } + } + }, + // 清除镜像 + { + opcode: 'clearMirror', + blockType: 'command', + text: this.formatMessage('30Ext.block.clearMirror') + }, + "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 + { + opcode: 'anotherRun', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRun'), + arguments: { + spriteName: { + type: 'string', + menu: 'spritesMenu' + }, + //SUBSTACK: { + // type: "input_statement" + //} + } + }, + { + opcode: 'anotherRunWithClone', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRunWithClone'), + arguments: { + spriteName: { + type: 'string', + menu: 'spritesMenu' + }, + cloneId: { + type: 'number', + defaultValue: 1 + }, + //SUBSTACK: { + // type: "input_statement" + //} + } + }, + ], menus: { + conInfoMenu:[ + { + text: this.formatMessage('ArkosExt.conInfo1'), //名称 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.conInfo2'), //内容 + value: '2' + }, + ], + conOpMenu:[ + { + text: this.formatMessage('ArkosExt.conOp1'), //设为 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.conOp2'), //增加 + value: '2' + }, + ], ListOpMenu: [ { text: this.formatMessage('ArkosExt.ListOp1'), //插入 @@ -965,8 +1131,8 @@ class ArkosExtensions { acceptReporters: true, }, //判断符菜单 - opMenu1: ['<','≤','=','≠',], - opMenu2: ['<','>','≤','≥','=','≠',], + opMenu1: ['<', '≤', '=', '≠',], + opMenu2: ['<', '>', '≤', '≥', '=', '≠',], //logicMenu logicMenu: [ { @@ -1039,19 +1205,19 @@ class ArkosExtensions { value: 'ghost' } ], - //30Ext - spritesMenu: { - items: 'getSpritesMenu' - }, - mirrorMenu: [{ - text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: 0 - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: 1 - } - ] + //30Ext + spritesMenu: { + items: 'getSpritesMenu' + }, + mirrorMenu: [{ + text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 + value: 0 + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 + value: 1 + } + ] }, } } @@ -1132,25 +1298,25 @@ class ArkosExtensions { turnDegreesToDir(args, util) { const degree = Cast.toNumber(args.degree); const dir = Cast.toNumber(args.dir); - const dif = this.differenceBetweenDirections({a: util.target.direction, b: dir}); - if(Math.abs(dif) < degree) + const dif = this.differenceBetweenDirections({ a: util.target.direction, b: dir }); + if (Math.abs(dif) < degree) util.target.setDirection(dir); - else if(dif < 0) + else if (dif < 0) util.target.setDirection(util.target.direction - degree); else util.target.setDirection(util.target.direction + degree); } //获取特效的数值 - getEffect (args, util) { + getEffect(args, util) { let effect = Cast.toString(args.EFFECT).toLowerCase(); if (!util.target.effects.hasOwnProperty(effect)) return 0; return util.target.effects[effect]; } //角色是否可见 - isHiding (args, util) { - return !util.target.visible; + isHiding(args, util) { + return !util.target.visible; } //获取图层(逝一逝) @@ -1159,65 +1325,65 @@ class ArkosExtensions { // } //获取当前角色的旋转方式 - getRotationStyle (args, util) { + getRotationStyle(args, util) { return util.target.rotationStyle; } //获取当前造型的长/宽 - getWidthOrHeight (args, util) { + getWidthOrHeight(args, util) { const costumeSize = util.target.renderer.getCurrentSkinSize(util.target.drawableID); return costumeSize[args.t]; } //强行设置大小(逝一逝) - setSize (args, util) { + setSize(args, util) { if (util.target.isStage) { - return; + return; } if (util.target.renderer) { - args.size = this._clamp(Cast.toNumber(args.size),0.1,100000000) - util.target.size = args.size; - const {direction, scale} = util.target._getRenderedDirectionAndScale(); - util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); - if (util.target.visible) { - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - } + args.size = this._clamp(Cast.toNumber(args.size), 0.1, 100000000) + util.target.size = args.size; + const { direction, scale } = util.target._getRenderedDirectionAndScale(); + util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); + if (util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } } util.target.runtime.requestTargetsUpdate(util.target); } - _clamp (n, min, max) { - return Math.min(Math.max(n, min), max); + _clamp(n, min, max) { + return Math.min(Math.max(n, min), max); } - + //强行设置XY(逝一逝) - setXY (args, util) { + setXY(args, util) { if (util.target.isStage) return; - args.x = this._clamp(Cast.toNumber(args.x),-100000000,100000000) - args.y = this._clamp(Cast.toNumber(args.y),-100000000,100000000) + args.x = this._clamp(Cast.toNumber(args.x), -100000000, 100000000) + args.y = this._clamp(Cast.toNumber(args.y), -100000000, 100000000) const oldX = util.target.x; const oldY = util.target.y; util.target.x = args.x; util.target.y = args.y; if (util.target.renderer) { - util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); - if (util.target.visible) { - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - } + util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); + if (util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } } else { - util.target.x = x; - util.target.y = y; + util.target.x = x; + util.target.y = y; } util.target.emit('TARGET_MOVED', util.target, oldX, oldY, false); util.target.runtime.requestTargetsUpdate(util.target); } //获取角色边缘的坐标 - getBoundaryCoord (args, util) { + getBoundaryCoord(args, util) { const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); - switch(args.t){ + switch (args.t) { case '1': return bounds.top; case '2': @@ -1232,34 +1398,34 @@ class ArkosExtensions { } //是否在舞台外 - isOutOfSight (args, util) { + isOutOfSight(args, util) { // console.log(util.target.runtime.renderer) // console.log(util.target.renderer) if (util.target.renderer) { - const stageWidth = util.target.runtime.stageWidth; - const stageHeight = util.target.runtime.stageHeight; - console.log('stageWidth',stageWidth); - console.log('stageHeight',stageHeight); - const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); - if (bounds.right < -stageWidth / 2 || - bounds.left > stageWidth / 2 || - bounds.bottom > stageHeight / 2 || - bounds.top < -stageHeight / 2) { - return true; - } + const stageWidth = util.target.runtime.stageWidth; + const stageHeight = util.target.runtime.stageHeight; + console.log('stageWidth', stageWidth); + console.log('stageHeight', stageHeight); + const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); + if (bounds.right < -stageWidth / 2 || + bounds.left > stageWidth / 2 || + bounds.bottom > stageHeight / 2 || + bounds.top < -stageHeight / 2) { + return true; + } } return false; } //形如:<() > - reporterToBoolean (args){ - if(Cast.toString(args.t).toLowerCase() === 'false') return false; - if(args.t === '0') return false; + reporterToBoolean(args) { + if (Cast.toString(args.t).toLowerCase() === 'false') return false; + if (args.t === '0') return false; return (args.t) ? true : false; } - compare(a,b,op){ - switch(op){ + compare(a, b, op) { + switch (op) { case '<': return Cast.compare(a, b) < 0; case '>': @@ -1278,29 +1444,29 @@ class ArkosExtensions { } //形如:a≤b≤c op1,op2 - compareTwoSides (args){ - return this.compare(args.a, args.b, args.op1)&&this.compare(args.b, args.c, args.op2) + compareTwoSides(args) { + return this.compare(args.a, args.b, args.op1) && this.compare(args.b, args.c, args.op2) } - + //形如:a≤b且/或>c op1,op2 logic - compareTwoSidesPlus(args){ - switch(args.logic){ + compareTwoSidesPlus(args) { + switch (args.logic) { case 'or': - return this.compare(args.a, args.b, args.op1)||this.compare(args.a, args.c, args.op2) + return this.compare(args.a, args.b, args.op1) || this.compare(args.a, args.c, args.op2) case 'and': - return this.compare(args.a, args.b, args.op1)&&this.compare(args.a, args.c, args.op2) + return this.compare(args.a, args.b, args.op1) && this.compare(args.a, args.c, args.op2) default: return false; } } //数组排序规则 - sortRule(propName,order) { + sortRule(propName, order) { return (a, b) => { a = a[propName] b = b[propName] if (a > b) return order === 'asc' ? 1 : -1; - else if (a < b) return order === 'asc' ? -1 : 1; + else if (a < b) return order === 'asc' ? -1 : 1; else return 0; } } @@ -1311,7 +1477,7 @@ class ArkosExtensions { let temp = this.sortedTable; Object.keys(temp).forEach(obj => { //if ( Array.isArray (temp[obj]) ) { - list.push(obj); + list.push(obj); //} }); if (list.length === 0) { @@ -1323,45 +1489,45 @@ class ArkosExtensions { //list.sort(this.sortRule("text")); return list; } - - createTableIfNotExist(list){ - if(!(list in this.sortedTable)) - this.sortedTable[list]={order:'desc',list:[]}; + + createTableIfNotExist(list) { + if (!(list in this.sortedTable)) + this.sortedTable[list] = { order: 'desc', list: [] }; } - sortTable(list){ - this.sortedTable[list].list.sort(this.sortRule("rankValue",this.sortedTable[list].order)); + sortTable(list) { + this.sortedTable[list].list.sort(this.sortRule("rankValue", this.sortedTable[list].order)); } - + //📊清空排序表 - clearSortedTable (args){ + clearSortedTable(args) { this.createTableIfNotExist(args.list) - this.sortedTable[args.list].list=[]; + this.sortedTable[args.list].list = []; } //📊设置排序方式 - setTypeOfSortedTable (args){ + setTypeOfSortedTable(args) { this.createTableIfNotExist(args.list) - this.sortedTable[args.list].order=args.type; + this.sortedTable[args.list].order = args.type; this.sortTable(args.list) } //查找在列表中的插入位置(已有则覆盖) - _findPlaceAndInsert(list, order, item){ + _findPlaceAndInsert(list, order, item) { //删除已存在的内容 for (let i = 0; i < list.length; i++) { - if (list[i].name === item.name){ + if (list[i].name === item.name) { //删除同名项 - list.splice(i,1); + list.splice(i, 1); break; } } //查找插入位置并插入 for (let i = 0; i < list.length; i++) { - if ((list[i].rankValue > item.rankValue && order ==='asc')|| - (list[i].rankValue < item.rankValue && order ==='desc')){ + if ((list[i].rankValue > item.rankValue && order === 'asc') || + (list[i].rankValue < item.rankValue && order === 'desc')) { //插入在该项前 - list.splice(i,0,item); + list.splice(i, 0, item); return; } } @@ -1370,17 +1536,17 @@ class ArkosExtensions { } //📊将内容加入表 - addToSortedTable (args){ + addToSortedTable(args) { this.createTableIfNotExist(args.list) this._findPlaceAndInsert( this.sortedTable[args.list].list, this.sortedTable[args.list].order, - {name:args.name, rankValue:args.value, extra:args.extra}); + { name: args.name, rankValue: args.value, extra: args.extra }); } - - _getTInItem (item, t, rank){ - if(item === undefined) return ''; - switch(t){ + + _getTInItem(item, t, rank) { + if (item === undefined) return ''; + switch (t) { case '1': return item.name; case '2': @@ -1395,15 +1561,15 @@ class ArkosExtensions { } //📊获取第n项 - getFromSortedTableByNo (args){ - if(!(args.list in this.sortedTable)) return ''; + getFromSortedTableByNo(args) { + if (!(args.list in this.sortedTable)) return ''; let list = this.sortedTable[args.list].list; - return this._getTInItem (list[args.n-1], args.t, Cast.toNumber(args.n)); + return this._getTInItem(list[args.n - 1], args.t, Cast.toNumber(args.n)); } - _getItemIdxByName(list, name){ + _getItemIdxByName(list, name) { for (let i = 0; i < list.length; i++) { - if (list[i].name === name){ + if (list[i].name === name) { return i; } } @@ -1411,72 +1577,88 @@ class ArkosExtensions { } //📊获取名为XX的项 - getFromSortedTableByName (args){ - if(!(args.list in this.sortedTable)) return ''; + getFromSortedTableByName(args) { + if (!(args.list in this.sortedTable)) return ''; let list = this.sortedTable[args.list].list; - let n = this._getItemIdxByName(list ,args.name) ; - if(n === -1) return ''; - return this._getTInItem (list[n], args.t, n+1); + let n = this._getItemIdxByName(list, args.name); + if (n === -1) return ''; + return this._getTInItem(list[n], args.t, n + 1); } //📊获取排序表长度 - lengthOfSortedTable (args){ - if(!(args.list in this.sortedTable)) return 0; + lengthOfSortedTable(args) { + if (!(args.list in this.sortedTable)) return 0; return this.sortedTable[args.list].list.length; } //📊删除排序表名为XX的内容 - deleteNameOfSortedTable (args){ - if(!(args.list in this.sortedTable)) return; + deleteNameOfSortedTable(args) { + if (!(args.list in this.sortedTable)) return; let list = this.sortedTable[args.list].list; - let n = this._getItemIdxByName(list ,args.name) ; - if(n === -1) return; + let n = this._getItemIdxByName(list, args.name); + if (n === -1) return; list.splice(n, 1); } //获取颜色HEX码 - colorToHex (args, util) { + colorToHex(args, util) { let c = Cast.toRgbColorList(args.COLOR) - return Color.rgbToHex({r:c[0],g:c[1],b:c[2]}); + return Color.rgbToHex({ r: c[0], g: c[1], b: c[2] }); } //🗂️ 临时变量积木 - deleteAllTempData (args) { + + //来自 -6 :任意内容转字符或数字 + _anythingToNumberString(value) { + switch(typeof(value)){ + case "string": + case "number": + break; + case "object": + value = JSON.stringify(value); + break; + default: + value = ''; //包含了undefined + } + return value; + } + + deleteAllTempData(args) { this.tempData = {}; } - getCountOfTempData (args) { - return Object.keys (this.tempData).length; + getCountOfTempData(args) { + return Object.keys(this.tempData).length; } - delTempData (args) { + delTempData(args) { delete this.tempData[Cast.toString(args.data)]; } - setTempVar (args) { + setTempVar(args) { this.tempData[Cast.toString(args.var)] = args.t; } - addTempVar (args) { + addTempVar(args) { this.tempData[Cast.toString(args.var)] = Cast.toNumber(this.tempData[Cast.toString(args.var)]) + Cast.toNumber(args.t); } - getTempVar (args) { + getTempVar(args) { let temp = this.tempData[Cast.toString(args.var)] - if(Array.isArray(temp)) return JSON.stringify(temp); + if (Array.isArray(temp)) return JSON.stringify(temp); return Cast.toString(temp); } //创建或清空 - clearTempList (args) { + clearTempList(args) { this.tempData[Cast.toString(args.list)] = []; } - initTempList (args) { + initTempList(args) { try { let content = JSON.parse(Cast.toString(args.t)) - if(Array.isArray(content)) { + if (Array.isArray(content)) { this.tempData[Cast.toString(args.list)] = content; } } catch (e) { @@ -1484,24 +1666,24 @@ class ArkosExtensions { } } - addTempList (args) { + addTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return; + if (!Array.isArray(list)) return; list.push(Cast.toString(args.t)); } - opTempList (args) { + opTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return; + if (!Array.isArray(list)) return; let n = Cast.toNumber(args.n) - if(n < 1 || n >list.length+1) return; + if (n < 1 || n > list.length + 1) return; n -= 1; - switch(args.op){ + switch (args.op) { case '1'://插入 - list.splice(n,0,Cast.toString(args.t)); + list.splice(n, 0, args.t); return; case '2'://替换 - list[n] = Cast.toString(args.t); + list[n] = args.t; return; case '3'://增加 list[n] = Cast.toNumber(list[n]) + Cast.toNumber(args.t); @@ -1511,115 +1693,183 @@ class ArkosExtensions { } } - delItemOfTempList (args) { + delItemOfTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return; + if (!Array.isArray(list)) return; let n = Cast.toNumber(args.n) - if(n < 1 || n >list.length) return; + if (n < 1 || n > list.length) return; n -= 1; - list.splice(n,1); + list.splice(n, 1); } - getItemOfTempList (args) { + getItemOfTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return ''; + if (!Array.isArray(list)) return ''; let n = Cast.toNumber(args.n) - if(n < 1 || n >list.length) return ''; + if (n < 1 || n > list.length) return ''; n -= 1; return Cast.toString(list[n]); } - lengthOfTempList (args) { + lengthOfTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return 0; + if (!Array.isArray(list)) return 0; return list.length; } + + //容器 + clearTempCon(args) { + this.tempData[Cast.toString(args.con)] = {}; + } + + initTempCon(args) { + try { + let content = JSON.parse(Cast.toString(args.t)) + if (typeof(content) === 'object' && content !== null) { + this.tempData[Cast.toString(args.con)] = content; + } + } catch (e) { + + } + } + + opTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return; + let c = Cast.toString(args.c) + switch (args.op) { + case '1'://设为 + con[c] = args.t; + return; + case '2'://增加 + if(!(c in con)) return; + con[c] = Cast.toNumber(con[c]) + Cast.toNumber(args.t); + return; + default: + return; + } + } + + delItemOfTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return; + delete con[Cast.toString(args.c)]; + } + + getItemOfTempConByName(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return ''; + return Cast.toString(con[Cast.toString(args.c)]); + } + + getItemOfTempConByNo(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return ''; + let key = Object.keys(con)[Cast.toNumber(args.n) - 1] + if (key === undefined) return ''; + switch (args.t) { + case '1'://名称 + return key; + case '2'://内容 + return this._anythingToNumberString(con[key]); + default: + return; + } + } + + lengthOfTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && con !== null)) return 0; + return Object.keys(con).length; + } + + // //30Ext //诶诶 HCN居然不给另外开扩展 只好插队了 // //菜单 - //动态菜单: 角色菜单 - getSpritesMenu(){ - var sprites = []; - for(const targetId in this.runtime.targets) { - if(!this.runtime.targets.hasOwnProperty(targetId)) continue; - if(!this.runtime.targets[targetId].isOriginal) continue; - if(this.runtime.targets[targetId] === this.runtime._editingTarget) continue; //排除自己 - let name = this.runtime.targets[targetId].sprite.name; - sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 - } - return sprites; - } - // - //角色造型操作 - // - //镜像造型 - mirrorSprite(args, util){ - util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; - //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - //util.target.runtime.requestRedraw(); - } - //清除镜像 - clearMirror(args, util){ - for (let i=0; i<2; i++){ - util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]= - Math.abs(util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]); - } - //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - //util.target.runtime.requestRedraw(); - } - // - //角色跨域操作 - // - //跨域执行 - anotherRun(args, util){ - if(!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[0]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; - } - } - //跨域克隆体执行 - anotherRunWithClone(args, util){ - if(!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[args.cloneId]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; - } - } + //动态菜单: 角色菜单 + getSpritesMenu() { + var sprites = []; + for (const targetId in this.runtime.targets) { + if (!this.runtime.targets.hasOwnProperty(targetId)) continue; + if (!this.runtime.targets[targetId].isOriginal) continue; + if (this.runtime.targets[targetId] === this.runtime._editingTarget) continue; //排除自己 + let name = this.runtime.targets[targetId].sprite.name; + sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 + } + return sprites; + } + // + //角色造型操作 + // + //镜像造型 + mirrorSprite(args, util) { + util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[args.mirrorMethod] *= -1; + //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + //util.target.runtime.requestRedraw(); + } + //清除镜像 + clearMirror(args, util) { + for (let i = 0; i < 2; i++) { + util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i] = + Math.abs(util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[i]); + } + //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + //util.target.runtime.requestRedraw(); + } + // + //角色跨域操作 + // + //跨域执行 + anotherRun(args, util) { + if (!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = util.target.sprite.clones[0]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } + //跨域克隆体执行 + anotherRunWithClone(args, util) { + if (!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = util.target.sprite.clones[args.cloneId]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } } window.tempExt = { - Extension: ArkosExtensions, - info: { - name: 'hcn.extensionName', - description: 'hcn.description', - extensionId: 'hcnTest', - // iconURL: icon, - // insetIconURL: cover, - featured: true, - disabled: false, - collaborator: 'only for hcn test', + Extension: ArkosExtensions, + info: { + name: 'hcn.extensionName', + description: 'hcn.description', + extensionId: 'hcnTest', + // iconURL: icon, + // insetIconURL: cover, + featured: true, + disabled: false, + collaborator: 'only for hcn test', + }, + l10n: { + 'zh-cn': { + 'hcn.extensionName': 'hcn 的测试', + 'hcn.description': 'hcn 的测试', }, - l10n: { - 'zh-cn': { - 'hcn.extensionName': 'hcn 的测试', - 'hcn.description': 'hcn 的测试', - }, - en: { - 'hcn.extensionName': 'hcn test', - 'hcn.description': 'hcn test', - }, + en: { + 'hcn.extensionName': 'hcn test', + 'hcn.description': 'hcn test', }, - } + }, +} From 29918f53c6ede209952e2f527d1cca9f51146470 Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sat, 30 Jul 2022 20:07:30 +0800 Subject: [PATCH 082/357] =?UTF-8?q?=E6=9F=A5=E7=9C=8B=E5=AE=B9=E5=99=A8?= =?UTF-8?q?=E6=97=B6=E8=BD=AC=E4=B8=BA=20JSON?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 7f9bbd2a..b7d4b738 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1646,7 +1646,7 @@ class ArkosExtensions { getTempVar(args) { let temp = this.tempData[Cast.toString(args.var)] - if (Array.isArray(temp)) return JSON.stringify(temp); + if (typeof(temp) === 'object') return JSON.stringify(temp); return Cast.toString(temp); } From 7db147b2aff55eac1fc36492cabcb58ae630c12c Mon Sep 17 00:00:00 2001 From: Arkos <1726039638@qq.com> Date: Sat, 30 Jul 2022 20:14:59 +0800 Subject: [PATCH 083/357] =?UTF-8?q?=E5=88=86=E5=BC=80=E4=B8=B4=E6=97=B6?= =?UTF-8?q?=E5=8F=98=E9=87=8F=E3=80=81=E5=88=97=E8=A1=A8=E3=80=81=E5=AE=B9?= =?UTF-8?q?=E5=99=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index b7d4b738..58d8cf01 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -22,7 +22,10 @@ class ArkosExtensions { 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', 'ArkosExt.info5': '📊 排序表', - 'ArkosExt.info6': '🗂️ 临时变量', + 'ArkosExt.info6': '🗂️ 临时数据', + 'ArkosExt.info7': '临时变量', + 'ArkosExt.info8': '临时列表', + 'ArkosExt.info9': '临时容器', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', @@ -704,8 +707,7 @@ class ArkosExtensions { }, }, - "---" + this.formatMessage("ArkosExt.info6"), //🗂️ 临时变量 - + "---" + this.formatMessage("ArkosExt.info6"), //🗂️ 临时数据 //清空所有临时数据 { opcode: 'deleteAllTempData', @@ -730,6 +732,7 @@ class ArkosExtensions { }, }, }, + "---" + this.formatMessage("ArkosExt.info7"), //临时变量 //设置临时数据 { opcode: 'setTempVar', @@ -762,7 +765,7 @@ class ArkosExtensions { }, }, }, - //增加临时数据 + //获取临时数据 { opcode: 'getTempVar', blockType: 'reporter', @@ -774,6 +777,7 @@ class ArkosExtensions { }, }, }, + "---" + this.formatMessage("ArkosExt.info8"), //临时列表 //创建或清空临时列表 { opcode: 'clearTempList', @@ -886,8 +890,7 @@ class ArkosExtensions { }, }, }, - - + "---" + this.formatMessage("ArkosExt.info9"), //临时容器 //创建或清空临时容器 { opcode: 'clearTempCon', From a3e5245ad1db8e28657c955c90723fd9c77c4a87 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Sat, 30 Jul 2022 21:45:07 +0800 Subject: [PATCH 084/357] =?UTF-8?q?=E6=9B=B4=E6=8D=A2=E9=80=A0=E5=9E=8B?= =?UTF-8?q?=E9=95=9C=E5=83=8F=E6=96=B9=E6=B3=95=E9=80=BB=E8=BE=91?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 872 ++++++++++++++++++++++++++--------------- 1 file changed, 562 insertions(+), 310 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index cc9e5f44..f0cbd392 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -2,16 +2,17 @@ import Cast from '../utils/cast.js' import Color from '../utils/color.js' // import cover from './assets/icon.svg' // import icon from './assets/icon.svg' +//鸣谢:-6 优化代码和修复了一些 bug;_30 提供了部分拓展积木 console.log(Cast.toNumber('123')) console.log(Cast.toNumber('aab')) class ArkosExtensions { constructor(runtime) { this.runtime = runtime - this.tempData={} - this.sortedTable={ - list1:{order:'desc',list:[]}, - list2:{order:'desc',list:[]}, + this.tempData = {} + this.sortedTable = { + list1: { order: 'desc', list: [] }, + list2: { order: 'desc', list: [] }, } this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { @@ -21,7 +22,10 @@ class ArkosExtensions { 'ArkosExt.info3': '🛠 实用积木', 'ArkosExt.info4': '📄 信息获取', 'ArkosExt.info5': '📊 排序表', - 'ArkosExt.info6': '🗂️ 临时变量', + 'ArkosExt.info6': '🗂️ 临时数据', + 'ArkosExt.info7': '临时变量', + 'ArkosExt.info8': '临时列表', + 'ArkosExt.info9': '临时容器', 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', @@ -57,7 +61,7 @@ class ArkosExtensions { 'ArkosExt.and': '且', 'ArkosExt.or': '或', - + 'ArkosExt.clearSortedTable': '📊清空排序表[list]', 'ArkosExt.setTypeOfSortedTable': '📊将排序表[list]的排序方式设为[type]', 'ArkosExt.addToSortedTable': '📊将内容(重名的则覆盖)[name],排序值[value]加入排序表[list],附加信息[extra]', @@ -81,7 +85,7 @@ class ArkosExtensions { 'ArkosExt.setTempVar': '🗂️将临时变量[var]设为[t]', 'ArkosExt.addTempVar': '🗂️将临时变量[var]增加[t]', - 'ArkosExt.getTempVar': '🗂️临时变量[var]', + 'ArkosExt.getTempVar': '🗂️临时数据[var]', 'ArkosExt.clearTempList': '🗂️创建或清空临时列表[list]', 'ArkosExt.initTempList': '🗂️临时列表[list]内容设为[t]', @@ -93,16 +97,28 @@ class ArkosExtensions { 'ArkosExt.delItemOfTempList': '🗂️删除临时列表[list]第[n]项', 'ArkosExt.getItemOfTempList': '🗂️临时列表[list]第[n]项', 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', - - '30Ext.info': '✨ 以下扩展由_30提供', - '30Ext.info.1': '🪞 造型镜像操作', - '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', - '30Ext.menu.mirrorMethod.1': '左右镜像', - '30Ext.menu.mirrorMethod.2': '上下镜像', - '30Ext.block.clearMirror': '清除角色镜像变换', - '30Ext.info.2': '🛸 角色跨域操作', - '30Ext.block.anotherRun': '让[spriteName]运行', - '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' + + 'ArkosExt.clearTempCon': '🗂️创建或清空临时容器[con]', + 'ArkosExt.initTempCon': '🗂️临时容器[con]内容设为[t]', + 'ArkosExt.opTempCon': '🗂️临时容器[con]中的[c][op][t]', + 'ArkosExt.conOp1': '设为', + 'ArkosExt.conOp2': '增加', + 'ArkosExt.delItemOfTempCon': '🗂️删除临时容器[con]中名为[c]的内容', + 'ArkosExt.getItemOfTempConByName': '🗂️临时容器[con]中的[c]', + 'ArkosExt.getItemOfTempConByNo': '🗂️临时容器[con]第[n]项的[t]', + 'ArkosExt.conInfo1': '名称', + 'ArkosExt.conInfo2': '内容', + 'ArkosExt.lengthOfTempCon': '🗂️临时容器[con]中内容数', + + '30Ext.info': '✨ 以下扩展由_30提供', + '30Ext.info.1': '🪞 造型镜像操作', + '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', + '30Ext.menu.mirrorMethod.1': '左右镜像', + '30Ext.menu.mirrorMethod.2': '上下镜像', + '30Ext.block.clearMirror': '清除角色镜像变换', + '30Ext.info.2': '🛸 角色跨域操作', + '30Ext.block.anotherRun': '让[spriteName]运行', + '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' }, en: { @@ -157,7 +173,7 @@ class ArkosExtensions { 'ArkosExt.extra': 'extra', 'ArkosExt.colorToHex': 'get code of color[COLOR]', - + 'ArkosExt.info1': '🚶 Coordinate and Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', @@ -171,9 +187,9 @@ class ArkosExtensions { 'ArkosExt.setTempVar': '🗂️set temp var[var] to [t]', 'ArkosExt.addTempVar': '🗂️change temp var[var] by [t]', - 'ArkosExt.getTempVar': '🗂️temp var[var]', + 'ArkosExt.getTempVar': '🗂️temp data[var]', + - 'ArkosExt.clearTempList': '🗂️create or clear temp list[list]', 'ArkosExt.initTempList': '🗂️set temp list[list]to[t]', 'ArkosExt.addTempList': '🗂️add[t] to temp list[list]', @@ -184,16 +200,28 @@ class ArkosExtensions { 'ArkosExt.delItemOfTempList': '🗂️delete [n]of temp list[list]', 'ArkosExt.getItemOfTempList': '🗂️item[n]of temp list[list]', 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', - - '30Ext.info': '✨ The following extensions are provided by _30', - '30Ext.info.1': '🪞 Mirror transform', - '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', - '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', - '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', - '30Ext.block.clearMirror': 'Clear the mirror transform', - '30Ext.info.2': '🛸 Cross sprite operation', - '30Ext.block.anotherRun': 'Let [spriteName] run', - '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' + + 'ArkosExt.clearTempCon': '🗂️create or clear temp container[con]', + 'ArkosExt.initTempCon': '🗂️set temp container[con]to[t]', + 'ArkosExt.opTempCon': '🗂️[c]in temp container[con][op][t]', + 'ArkosExt.conOp1': 'set to', + 'ArkosExt.conOp2': 'change by', + 'ArkosExt.delItemOfTempCon': '🗂️delete[c]in temp container[con]', + 'ArkosExt.getItemOfTempConByName': '🗂️[c]in temp container[con]', + 'ArkosExt.getItemOfTempConByNo': '🗂️[t] of #[n] in temp container[con]', + 'ArkosExt.conInfo1': 'name', + 'ArkosExt.conInfo2': 'content', + 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', + + '30Ext.info': '✨ The following extensions are provided by _30', + '30Ext.info.1': '🪞 Mirror transform', + '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', + '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', + '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', + '30Ext.block.clearMirror': 'Clear the mirror transform', + '30Ext.info.2': '🛸 Cross sprite operation', + '30Ext.block.anotherRun': 'Let [spriteName] run', + '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' }, }) } @@ -209,7 +237,7 @@ class ArkosExtensions { getInfo() { return { id: 'hcnTest', // 拓展id - name: this.formatMessage('ArkosExt.extensionName'), + name: this.formatMessage('ArkosExt.extensionName'), color1: '#FF8383', // menuIconURI: icon, @@ -679,8 +707,7 @@ class ArkosExtensions { }, }, - "---" + this.formatMessage("ArkosExt.info6"), //🗂️ 临时变量 - + "---" + this.formatMessage("ArkosExt.info6"), //🗂️ 临时数据 //清空所有临时数据 { opcode: 'deleteAllTempData', @@ -705,6 +732,7 @@ class ArkosExtensions { }, }, }, + "---" + this.formatMessage("ArkosExt.info7"), //临时变量 //设置临时数据 { opcode: 'setTempVar', @@ -737,7 +765,7 @@ class ArkosExtensions { }, }, }, - //增加临时数据 + //获取临时数据 { opcode: 'getTempVar', blockType: 'reporter', @@ -749,6 +777,7 @@ class ArkosExtensions { }, }, }, + "---" + this.formatMessage("ArkosExt.info8"), //临时列表 //创建或清空临时列表 { opcode: 'clearTempList', @@ -861,63 +890,203 @@ class ArkosExtensions { }, }, }, - // - "---" + this.formatMessage("30Ext.info"), //30的扩展 - "---" + this.formatMessage("30Ext.info.1"), //造型镜像 - // 镜像造型 - { - opcode: 'mirrorSprite', - blockType: 'command', - text: this.formatMessage('30Ext.block.mirrorSprite'), - arguments: { - mirrorMethod: { - type: 'string', - menu: 'mirrorMenu' - } - } - }, - // 清除镜像 - { - opcode: 'clearMirror', - blockType: 'command', - text: this.formatMessage('30Ext.block.clearMirror') - }, - "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 - { - opcode: 'anotherRun', - blockType: 'conditional', - text: this.formatMessage('30Ext.block.anotherRun'), - arguments: { - spriteName: { - type: 'string', - menu: 'spritesMenu' - }, - SUBSTACK: { - type: "input_statement" - } - } - }, - { - opcode: 'anotherRunWithClone', - blockType: 'conditional', - text: this.formatMessage('30Ext.block.anotherRunWithClone'), - arguments: { - spriteName: { - type: 'string', - menu: 'spritesMenu' - }, - cloneId: { - type: 'number', - defaultValue: 1 - }, - SUBSTACK: { - type: "input_statement" - } - } - }, - + "---" + this.formatMessage("ArkosExt.info9"), //临时容器 + //创建或清空临时容器 + { + opcode: 'clearTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + }, + }, + //设置临时容器 + { + opcode: 'initTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.initTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + t: { + type: 'string', + defaultValue: '{"coins":200,"backpack":["wood","bread"]}', + }, + }, + }, + //操作临时容器 + { + opcode: 'opTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.opTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + op: { + type: 'string', + menu: 'conOpMenu', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + t: { + type: 'string', + defaultValue: '520', + }, + }, + }, + //删除临时容器名为xx的内容 + { + opcode: 'delItemOfTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.delItemOfTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + }, + }, + //获取临时容器名为XX的内容 + { + opcode: 'getItemOfTempConByName', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempConByName'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + }, + }, + //获取临时容器第n项的xx + { + opcode: 'getItemOfTempConByNo', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempConByNo'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + menu: 'conInfoMenu', + defaultValue: '2', + }, + }, + }, + //临时容器长度 + { + opcode: 'lengthOfTempCon', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + }, + }, + + + // + "---" + this.formatMessage("30Ext.info"), //感谢30提供的扩展 + "---" + this.formatMessage("30Ext.info.1"), //造型镜像 + // 镜像造型 + { + opcode: 'mirrorSprite', + blockType: 'command', + text: this.formatMessage('30Ext.block.mirrorSprite'), + arguments: { + mirrorMethod: { + type: 'number', + menu: 'mirrorMenu' + } + } + }, + // 清除镜像 + { + opcode: 'clearMirror', + blockType: 'command', + text: this.formatMessage('30Ext.block.clearMirror') + }, + "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 + { + opcode: 'anotherRun', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRun'), + arguments: { + spriteName: { + type: 'string', + menu: 'spritesMenu' + }, + SUBSTACK: { //TODO + type: "input_statement" + } + } + }, + { + opcode: 'anotherRunWithClone', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRunWithClone'), + arguments: { + spriteName: { + type: 'string', + menu: 'spritesMenu' + }, + cloneId: { + type: 'number', + defaultValue: 1 + }, + SUBSTACK: { + type: "input_statement" + } + } + }, + ], menus: { + conInfoMenu:[ + { + text: this.formatMessage('ArkosExt.conInfo1'), //名称 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.conInfo2'), //内容 + value: '2' + }, + ], + conOpMenu:[ + { + text: this.formatMessage('ArkosExt.conOp1'), //设为 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.conOp2'), //增加 + value: '2' + }, + ], ListOpMenu: [ { text: this.formatMessage('ArkosExt.ListOp1'), //插入 @@ -965,8 +1134,8 @@ class ArkosExtensions { acceptReporters: true, }, //判断符菜单 - opMenu1: ['<','≤','=','≠',], - opMenu2: ['<','>','≤','≥','=','≠',], + opMenu1: ['<', '≤', '=', '≠',], + opMenu2: ['<', '>', '≤', '≥', '=', '≠',], //logicMenu logicMenu: [ { @@ -1039,19 +1208,19 @@ class ArkosExtensions { value: 'ghost' } ], - //30Ext - spritesMenu: { - items: 'getSpritesMenu' - }, - mirrorMenu: [{ - text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: '0' - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: '1' - } - ] + //30Ext + spritesMenu: { + items: 'getSpritesMenu' + }, + mirrorMenu: [{ + text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 + value: 0 + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 + value: 1 + } + ] }, } } @@ -1132,25 +1301,25 @@ class ArkosExtensions { turnDegreesToDir(args, util) { const degree = Cast.toNumber(args.degree); const dir = Cast.toNumber(args.dir); - const dif = this.differenceBetweenDirections({a: util.target.direction, b: dir}); - if(Math.abs(dif) < degree) + const dif = this.differenceBetweenDirections({ a: util.target.direction, b: dir }); + if (Math.abs(dif) < degree) util.target.setDirection(dir); - else if(dif < 0) + else if (dif < 0) util.target.setDirection(util.target.direction - degree); else util.target.setDirection(util.target.direction + degree); } //获取特效的数值 - getEffect (args, util) { + getEffect(args, util) { let effect = Cast.toString(args.EFFECT).toLowerCase(); if (!util.target.effects.hasOwnProperty(effect)) return 0; return util.target.effects[effect]; } //角色是否可见 - isHiding (args, util) { - return !util.target.visible; + isHiding(args, util) { + return !util.target.visible; } //获取图层(逝一逝) @@ -1159,65 +1328,65 @@ class ArkosExtensions { // } //获取当前角色的旋转方式 - getRotationStyle (args, util) { + getRotationStyle(args, util) { return util.target.rotationStyle; } //获取当前造型的长/宽 - getWidthOrHeight (args, util) { + getWidthOrHeight(args, util) { const costumeSize = util.target.renderer.getCurrentSkinSize(util.target.drawableID); return costumeSize[args.t]; } //强行设置大小(逝一逝) - setSize (args, util) { + setSize(args, util) { if (util.target.isStage) { - return; + return; } if (util.target.renderer) { - args.size = this._clamp(Cast.toNumber(args.size),0.1,100000000) - util.target.size = args.size; - const {direction, scale} = util.target._getRenderedDirectionAndScale(); - util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); - if (util.target.visible) { - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - } + args.size = this._clamp(Cast.toNumber(args.size), 0.1, 100000000) + util.target.size = args.size; + const { direction, scale } = util.target._getRenderedDirectionAndScale(); + util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); + if (util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } } util.target.runtime.requestTargetsUpdate(util.target); } - _clamp (n, min, max) { - return Math.min(Math.max(n, min), max); + _clamp(n, min, max) { + return Math.min(Math.max(n, min), max); } - + //强行设置XY(逝一逝) - setXY (args, util) { + setXY(args, util) { if (util.target.isStage) return; - args.x = this._clamp(Cast.toNumber(args.x),-100000000,100000000) - args.y = this._clamp(Cast.toNumber(args.y),-100000000,100000000) + args.x = this._clamp(Cast.toNumber(args.x), -100000000, 100000000) + args.y = this._clamp(Cast.toNumber(args.y), -100000000, 100000000) const oldX = util.target.x; const oldY = util.target.y; util.target.x = args.x; util.target.y = args.y; if (util.target.renderer) { - util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); - if (util.target.visible) { - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - } + util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); + if (util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } } else { - util.target.x = x; - util.target.y = y; + util.target.x = x; + util.target.y = y; } util.target.emit('TARGET_MOVED', util.target, oldX, oldY, false); util.target.runtime.requestTargetsUpdate(util.target); } //获取角色边缘的坐标 - getBoundaryCoord (args, util) { + getBoundaryCoord(args, util) { const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); - switch(args.t){ + switch (args.t) { case '1': return bounds.top; case '2': @@ -1232,34 +1401,34 @@ class ArkosExtensions { } //是否在舞台外 - isOutOfSight (args, util) { + isOutOfSight(args, util) { // console.log(util.target.runtime.renderer) // console.log(util.target.renderer) if (util.target.renderer) { - const stageWidth = util.target.runtime.stageWidth; - const stageHeight = util.target.runtime.stageHeight; - console.log('stageWidth',stageWidth); - console.log('stageHeight',stageHeight); - const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); - if (bounds.right < -stageWidth / 2 || - bounds.left > stageWidth / 2 || - bounds.bottom > stageHeight / 2 || - bounds.top < -stageHeight / 2) { - return true; - } + const stageWidth = util.target.runtime.stageWidth; + const stageHeight = util.target.runtime.stageHeight; + console.log('stageWidth', stageWidth); + console.log('stageHeight', stageHeight); + const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); + if (bounds.right < -stageWidth / 2 || + bounds.left > stageWidth / 2 || + bounds.bottom > stageHeight / 2 || + bounds.top < -stageHeight / 2) { + return true; + } } return false; } //形如:<() > - reporterToBoolean (args){ - if(Cast.toString(args.t).toLowerCase() === 'false') return false; - if(args.t === '0') return false; + reporterToBoolean(args) { + if (Cast.toString(args.t).toLowerCase() === 'false') return false; + if (args.t === '0') return false; return (args.t) ? true : false; } - compare(a,b,op){ - switch(op){ + compare(a, b, op) { + switch (op) { case '<': return Cast.compare(a, b) < 0; case '>': @@ -1278,29 +1447,29 @@ class ArkosExtensions { } //形如:a≤b≤c op1,op2 - compareTwoSides (args){ - return this.compare(args.a, args.b, args.op1)&&this.compare(args.b, args.c, args.op2) + compareTwoSides(args) { + return this.compare(args.a, args.b, args.op1) && this.compare(args.b, args.c, args.op2) } - + //形如:a≤b且/或>c op1,op2 logic - compareTwoSidesPlus(args){ - switch(args.logic){ + compareTwoSidesPlus(args) { + switch (args.logic) { case 'or': - return this.compare(args.a, args.b, args.op1)||this.compare(args.a, args.c, args.op2) + return this.compare(args.a, args.b, args.op1) || this.compare(args.a, args.c, args.op2) case 'and': - return this.compare(args.a, args.b, args.op1)&&this.compare(args.a, args.c, args.op2) + return this.compare(args.a, args.b, args.op1) && this.compare(args.a, args.c, args.op2) default: return false; } } //数组排序规则 - sortRule(propName,order) { + sortRule(propName, order) { return (a, b) => { a = a[propName] b = b[propName] if (a > b) return order === 'asc' ? 1 : -1; - else if (a < b) return order === 'asc' ? -1 : 1; + else if (a < b) return order === 'asc' ? -1 : 1; else return 0; } } @@ -1311,7 +1480,7 @@ class ArkosExtensions { let temp = this.sortedTable; Object.keys(temp).forEach(obj => { //if ( Array.isArray (temp[obj]) ) { - list.push(obj); + list.push(obj); //} }); if (list.length === 0) { @@ -1323,45 +1492,45 @@ class ArkosExtensions { //list.sort(this.sortRule("text")); return list; } - - createTableIfNotExist(list){ - if(!(list in this.sortedTable)) - this.sortedTable[list]={order:'desc',list:[]}; + + createTableIfNotExist(list) { + if (!(list in this.sortedTable)) + this.sortedTable[list] = { order: 'desc', list: [] }; } - sortTable(list){ - this.sortedTable[list].list.sort(this.sortRule("rankValue",this.sortedTable[list].order)); + sortTable(list) { + this.sortedTable[list].list.sort(this.sortRule("rankValue", this.sortedTable[list].order)); } - + //📊清空排序表 - clearSortedTable (args){ + clearSortedTable(args) { this.createTableIfNotExist(args.list) - this.sortedTable[args.list].list=[]; + this.sortedTable[args.list].list = []; } //📊设置排序方式 - setTypeOfSortedTable (args){ + setTypeOfSortedTable(args) { this.createTableIfNotExist(args.list) - this.sortedTable[args.list].order=args.type; + this.sortedTable[args.list].order = args.type; this.sortTable(args.list) } //查找在列表中的插入位置(已有则覆盖) - _findPlaceAndInsert(list, order, item){ + _findPlaceAndInsert(list, order, item) { //删除已存在的内容 for (let i = 0; i < list.length; i++) { - if (list[i].name === item.name){ + if (list[i].name === item.name) { //删除同名项 - list.splice(i,1); + list.splice(i, 1); break; } } //查找插入位置并插入 for (let i = 0; i < list.length; i++) { - if ((list[i].rankValue > item.rankValue && order ==='asc')|| - (list[i].rankValue < item.rankValue && order ==='desc')){ + if ((list[i].rankValue > item.rankValue && order === 'asc') || + (list[i].rankValue < item.rankValue && order === 'desc')) { //插入在该项前 - list.splice(i,0,item); + list.splice(i, 0, item); return; } } @@ -1370,17 +1539,17 @@ class ArkosExtensions { } //📊将内容加入表 - addToSortedTable (args){ + addToSortedTable(args) { this.createTableIfNotExist(args.list) this._findPlaceAndInsert( this.sortedTable[args.list].list, this.sortedTable[args.list].order, - {name:args.name, rankValue:args.value, extra:args.extra}); + { name: args.name, rankValue: args.value, extra: args.extra }); } - - _getTInItem (item, t, rank){ - if(item === undefined) return ''; - switch(t){ + + _getTInItem(item, t, rank) { + if (item === undefined) return ''; + switch (t) { case '1': return item.name; case '2': @@ -1395,15 +1564,15 @@ class ArkosExtensions { } //📊获取第n项 - getFromSortedTableByNo (args){ - if(!(args.list in this.sortedTable)) return ''; + getFromSortedTableByNo(args) { + if (!(args.list in this.sortedTable)) return ''; let list = this.sortedTable[args.list].list; - return this._getTInItem (list[args.n-1], args.t, Cast.toNumber(args.n)); + return this._getTInItem(list[args.n - 1], args.t, Cast.toNumber(args.n)); } - _getItemIdxByName(list, name){ + _getItemIdxByName(list, name) { for (let i = 0; i < list.length; i++) { - if (list[i].name === name){ + if (list[i].name === name) { return i; } } @@ -1411,72 +1580,88 @@ class ArkosExtensions { } //📊获取名为XX的项 - getFromSortedTableByName (args){ - if(!(args.list in this.sortedTable)) return ''; + getFromSortedTableByName(args) { + if (!(args.list in this.sortedTable)) return ''; let list = this.sortedTable[args.list].list; - let n = this._getItemIdxByName(list ,args.name) ; - if(n === -1) return ''; - return this._getTInItem (list[n], args.t, n+1); + let n = this._getItemIdxByName(list, args.name); + if (n === -1) return ''; + return this._getTInItem(list[n], args.t, n + 1); } //📊获取排序表长度 - lengthOfSortedTable (args){ - if(!(args.list in this.sortedTable)) return 0; + lengthOfSortedTable(args) { + if (!(args.list in this.sortedTable)) return 0; return this.sortedTable[args.list].list.length; } //📊删除排序表名为XX的内容 - deleteNameOfSortedTable (args){ - if(!(args.list in this.sortedTable)) return; + deleteNameOfSortedTable(args) { + if (!(args.list in this.sortedTable)) return; let list = this.sortedTable[args.list].list; - let n = this._getItemIdxByName(list ,args.name) ; - if(n === -1) return; + let n = this._getItemIdxByName(list, args.name); + if (n === -1) return; list.splice(n, 1); } //获取颜色HEX码 - colorToHex (args, util) { + colorToHex(args, util) { let c = Cast.toRgbColorList(args.COLOR) - return Color.rgbToHex({r:c[0],g:c[1],b:c[2]}); + return Color.rgbToHex({ r: c[0], g: c[1], b: c[2] }); } //🗂️ 临时变量积木 - deleteAllTempData (args) { + + //来自 -6 :任意内容转字符或数字 + _anythingToNumberString(value) { + switch(typeof(value)){ + case "string": + case "number": + break; + case "object": + value = JSON.stringify(value); + break; + default: + value = ''; //包含了undefined + } + return value; + } + + deleteAllTempData(args) { this.tempData = {}; } - getCountOfTempData (args) { - return Object.keys (this.tempData).length; + getCountOfTempData(args) { + return Object.keys(this.tempData).length; } - delTempData (args) { + delTempData(args) { delete this.tempData[Cast.toString(args.data)]; } - setTempVar (args) { + setTempVar(args) { this.tempData[Cast.toString(args.var)] = args.t; } - addTempVar (args) { + addTempVar(args) { this.tempData[Cast.toString(args.var)] = Cast.toNumber(this.tempData[Cast.toString(args.var)]) + Cast.toNumber(args.t); } - getTempVar (args) { + getTempVar(args) { let temp = this.tempData[Cast.toString(args.var)] - if(Array.isArray(temp)) return JSON.stringify(temp); + if (typeof(temp) === 'object') return JSON.stringify(temp); return Cast.toString(temp); } //创建或清空 - clearTempList (args) { + clearTempList(args) { this.tempData[Cast.toString(args.list)] = []; } - initTempList (args) { + initTempList(args) { try { let content = JSON.parse(Cast.toString(args.t)) - if(Array.isArray(content)) { + if (Array.isArray(content)) { this.tempData[Cast.toString(args.list)] = content; } } catch (e) { @@ -1484,24 +1669,24 @@ class ArkosExtensions { } } - addTempList (args) { + addTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return; + if (!Array.isArray(list)) return; list.push(Cast.toString(args.t)); } - opTempList (args) { + opTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return; + if (!Array.isArray(list)) return; let n = Cast.toNumber(args.n) - if(n < 1 || n >list.length+1) return; + if (n < 1 || n > list.length + 1) return; n -= 1; - switch(args.op){ + switch (args.op) { case '1'://插入 - list.splice(n,0,Cast.toString(args.t)); + list.splice(n, 0, args.t); return; case '2'://替换 - list[n] = Cast.toString(args.t); + list[n] = args.t; return; case '3'://增加 list[n] = Cast.toNumber(list[n]) + Cast.toNumber(args.t); @@ -1511,52 +1696,120 @@ class ArkosExtensions { } } - delItemOfTempList (args) { + delItemOfTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return; + if (!Array.isArray(list)) return; let n = Cast.toNumber(args.n) - if(n < 1 || n >list.length) return; + if (n < 1 || n > list.length) return; n -= 1; - list.splice(n,1); + list.splice(n, 1); } - getItemOfTempList (args) { + getItemOfTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return ''; + if (!Array.isArray(list)) return ''; let n = Cast.toNumber(args.n) - if(n < 1 || n >list.length) return ''; + if (n < 1 || n > list.length) return ''; n -= 1; return Cast.toString(list[n]); } - lengthOfTempList (args) { + lengthOfTempList(args) { let list = this.tempData[Cast.toString(args.list)] - if(!Array.isArray(list)) return 0; + if (!Array.isArray(list)) return 0; return list.length; } + + //容器 + clearTempCon(args) { + this.tempData[Cast.toString(args.con)] = {}; + } + + initTempCon(args) { + try { + let content = JSON.parse(Cast.toString(args.t)) + if (typeof(content) === 'object' && content !== null) { + this.tempData[Cast.toString(args.con)] = content; + } + } catch (e) { + + } + } + + opTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return; + let c = Cast.toString(args.c) + switch (args.op) { + case '1'://设为 + con[c] = args.t; + return; + case '2'://增加 + if(!(c in con)) return; + con[c] = Cast.toNumber(con[c]) + Cast.toNumber(args.t); + return; + default: + return; + } + } + + delItemOfTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return; + delete con[Cast.toString(args.c)]; + } + + getItemOfTempConByName(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return ''; + return Cast.toString(con[Cast.toString(args.c)]); + } + + getItemOfTempConByNo(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return ''; + let key = Object.keys(con)[Cast.toNumber(args.n) - 1] + if (key === undefined) return ''; + switch (args.t) { + case '1'://名称 + return key; + case '2'://内容 + return this._anythingToNumberString(con[key]); + default: + return; + } + } + + lengthOfTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if (!(typeof(con) === 'object' && con !== null)) return 0; + return Object.keys(con).length; + } + + // //30Ext - //诶诶 HCN居然不给另外开扩展 只好插队了 // //菜单 - //动态菜单: 角色菜单 - getSpritesMenu(){ - var sprites = []; - for(const targetId in this.runtime.targets) { - if(!this.runtime.targets.hasOwnProperty(targetId)) continue; - if(!this.runtime.targets[targetId].isOriginal) continue; - if(this.runtime.targets[targetId] === this.runtime._editingTarget) continue; //排除自己 - let name = this.runtime.targets[targetId].sprite.name; - sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 - } - return sprites; - } - // - //角色造型操作 - // - //镜像造型 - mirrorSprite(args, util){ - if (!util.target.ext30_isHook) { + //动态菜单: 角色菜单 + getSpritesMenu() { + var sprites = []; + for (const targetId in this.runtime.targets) { + if (!this.runtime.targets.hasOwnProperty(targetId)) continue; + if (!this.runtime.targets[targetId].isOriginal) continue; + if (this.runtime.targets[targetId] === this.runtime._editingTarget) continue; //排除自己 + let name = this.runtime.targets[targetId].sprite.name; + sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 + } + return sprites; + } + // + //角色造型操作 + // + //镜像造型 + mirrorSprite(args, util) { + //测试: onSize监听 更换代码执行逻辑 + if (!util.target.ext30_isHook) { util.target.ext30_mirror0 = 1; util.target.ext30_mirror1 = 1; const oldSet = util.target.prototype.setSize; @@ -1569,68 +1822,67 @@ class ArkosExtensions { } util.target['ext30_mirror' + args.mirrorMethod] *= -1; util.target.setSize(util.target._size); - //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - //util.target.runtime.requestRedraw(); - } - //清除镜像 - clearMirror(args, util){ - util.target.ext30_mirror0 = 1; - util.target.ext30_mirror1 = 1; - util.target.setSize(util.target._size); - //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - //util.target.runtime.requestRedraw(); - } - // - //角色跨域操作 - // - //跨域执行 - anotherRun(args, util){ - if(!util.thread.ext30_count) { - util.thread.ext30_count = true; - util.thread.ext30_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[0]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ext30_oldTarget; - util.thread.ext30_count = false; - } - } - //跨域克隆体执行 - anotherRunWithClone(args, util){ - if(!util.thread.ext30_count) { - util.thread.ext30_count = true; - util.thread.ext30_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[args.cloneId]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ext30_oldTarget; - util.thread.ext30_count = false; - } - } + } + //清除镜像 + clearMirror(args, util) { + util.target.ext30_mirror0 = 1; + util.target.ext30_mirror1 = 1; + util.target.setSize(util.target._size); + //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + //util.target.runtime.requestRedraw(); + } + // + //角色跨域操作 + // + //跨域执行 + anotherRun(args, util) { + console.info(util);//TODO + if (!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = util.target.sprite.clones[0]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } + //跨域克隆体执行 + anotherRunWithClone(args, util) { + if (!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = util.target.sprite.clones[args.cloneId]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } } window.tempExt = { - Extension: ArkosExtensions, - info: { - name: 'hcn.extensionName', - description: 'hcn.description', - extensionId: 'hcnTest', - // iconURL: icon, - // insetIconURL: cover, - featured: true, - disabled: false, - collaborator: 'only for hcn test', + Extension: ArkosExtensions, + info: { + name: 'hcn.extensionName', + description: 'hcn.description', + extensionId: 'hcnTest', + // iconURL: icon, + // insetIconURL: cover, + featured: true, + disabled: false, + collaborator: 'only for hcn test', + }, + l10n: { + 'zh-cn': { + 'hcn.extensionName': 'hcn 的测试', + 'hcn.description': 'hcn 的测试', }, - l10n: { - 'zh-cn': { - 'hcn.extensionName': 'hcn 的测试', - 'hcn.description': 'hcn 的测试', - }, - en: { - 'hcn.extensionName': 'hcn test', - 'hcn.description': 'hcn test', - }, + en: { + 'hcn.extensionName': 'hcn test', + 'hcn.description': 'hcn test', }, - } \ No newline at end of file + }, +} \ No newline at end of file From a708b8df42c30b749d3a729ffd7ef3cb10b5f7d9 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Sat, 30 Jul 2022 22:23:32 +0800 Subject: [PATCH 085/357] =?UTF-8?q?prototype=E6=94=B9=E4=B8=BA=5F=5Fproto?= =?UTF-8?q?=5F=5F?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 51ecc7d0..807b2c04 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1813,8 +1813,8 @@ class ArkosExtensions { if (!util.target.ext30_isHook) { util.target.ext30_mirror0 = 1; util.target.ext30_mirror1 = 1; - const oldSet = util.target.prototype.setSize; - util.target.prototype.setSize = function (size) { + const oldSet = util.target.__proto__.setSize; + util.target.__proto__.setSize = function (size) { oldSet.call(this, size); if (util.target.ext30_mirror0) util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[0] *= util.target.ext30_mirror0; if (util.target.ext30_mirror1) util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[1] *= util.target.ext30_mirror1; From 8d18044a560ebbd2edc375bda90e6603301cabec Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Sat, 30 Jul 2022 22:30:25 +0800 Subject: [PATCH 086/357] =?UTF-8?q?=E6=BC=8F=E4=BA=86=E5=87=A0=E4=B8=AA?= =?UTF-8?q?=E8=8B=B1=E6=96=87=E7=BF=BB=E8=AF=91?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 807b2c04..ecdd548b 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -180,6 +180,9 @@ class ArkosExtensions { 'ArkosExt.info4': '📄 Information', 'ArkosExt.info5': '📊 Sorted Table', 'ArkosExt.info6': '🗂️ Temporary Data', + 'ArkosExt.info7': 'temp var', + 'ArkosExt.info8': 'temp list', + 'ArkosExt.info9': 'temp container', 'ArkosExt.deleteAllTempData': '🗂️clear all temporary data', 'ArkosExt.getCountOfTempData': '🗂️count of temporary data', @@ -213,7 +216,7 @@ class ArkosExtensions { 'ArkosExt.conInfo2': 'content', 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', - '30Ext.info': '✨ The following extensions are provided by _30', + '30Ext.info': '✨ Contributed by _30 ↓', '30Ext.info.1': '🪞 Mirror transform', '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', From 4a357f236f70aea3ab2cc3cbd09af13ff51c2726 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Sat, 30 Jul 2022 22:36:20 +0800 Subject: [PATCH 087/357] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E8=B7=A8=E5=9F=9F?= =?UTF-8?q?=E6=89=A7=E8=A1=8C=E7=9A=84=E4=BB=A3=E7=A0=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index ecdd548b..bec38198 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -216,7 +216,7 @@ class ArkosExtensions { 'ArkosExt.conInfo2': 'content', 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', - '30Ext.info': '✨ Contributed by _30 ↓', + '30Ext.info': '✨ Contributed by _30', '30Ext.info.1': '🪞 Mirror transform', '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', @@ -1844,7 +1844,8 @@ class ArkosExtensions { if (!util.thread.ex_30Ext_count) { util.thread.ex_30Ext_count = true; util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[0]; + //util.thread.target = util.target.sprite.clones[0]; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.NAME).sprite.clones[0]; util.startBranch(1, true); } else { util.thread.target = util.thread.ex_30Ext_oldTarget; @@ -1856,7 +1857,8 @@ class ArkosExtensions { if (!util.thread.ex_30Ext_count) { util.thread.ex_30Ext_count = true; util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[args.cloneId]; + //util.thread.target = util.target.sprite.clones[args.cloneId]; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.NAME).sprite.clones[args.cloneId]; util.startBranch(1, true); } else { util.thread.target = util.thread.ex_30Ext_oldTarget; From ffefa64c1ef29112ba4e5e6d884508741b777d4f Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Sat, 30 Jul 2022 23:57:52 +0800 Subject: [PATCH 088/357] =?UTF-8?q?=E5=86=8D=E6=8D=A2=E4=B8=80=E7=A7=8D?= =?UTF-8?q?=E7=9B=91=E5=90=AC=E6=96=B9=E5=BC=8F=20=E5=BA=94=E8=AF=A5?= =?UTF-8?q?=E8=83=BD=E8=B7=91=E4=BA=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 53 ++++++++++++++++++++++-------------------- 1 file changed, 28 insertions(+), 25 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index f0cbd392..464dd08f 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -180,6 +180,9 @@ class ArkosExtensions { 'ArkosExt.info4': '📄 Information', 'ArkosExt.info5': '📊 Sorted Table', 'ArkosExt.info6': '🗂️ Temporary Data', + 'ArkosExt.info7': 'temp var', + 'ArkosExt.info8': 'temp list', + 'ArkosExt.info9': 'temp container', 'ArkosExt.deleteAllTempData': '🗂️clear all temporary data', 'ArkosExt.getCountOfTempData': '🗂️count of temporary data', @@ -213,7 +216,7 @@ class ArkosExtensions { 'ArkosExt.conInfo2': 'content', 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', - '30Ext.info': '✨ The following extensions are provided by _30', + '30Ext.info': '✨ Contributed by _30', '30Ext.info.1': '🪞 Mirror transform', '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', @@ -1020,7 +1023,7 @@ class ArkosExtensions { text: this.formatMessage('30Ext.block.mirrorSprite'), arguments: { mirrorMethod: { - type: 'number', + type: 'string', menu: 'mirrorMenu' } } @@ -1042,7 +1045,7 @@ class ArkosExtensions { menu: 'spritesMenu' }, SUBSTACK: { //TODO - type: "input_statement" + type: "input_statement" } } }, @@ -1060,8 +1063,9 @@ class ArkosExtensions { defaultValue: 1 }, SUBSTACK: { - type: "input_statement" + type: "input_statement" } + } }, @@ -1214,11 +1218,11 @@ class ArkosExtensions { }, mirrorMenu: [{ text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: 0 + value: '0' }, { text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: 1 + value: '1' } ] }, @@ -1808,26 +1812,23 @@ class ArkosExtensions { // //镜像造型 mirrorSprite(args, util) { - //测试: onSize监听 更换代码执行逻辑 + //测试: 换一个监听方式 + if (!util.target.ext30_isHook) { - util.target.ext30_mirror0 = 1; - util.target.ext30_mirror1 = 1; - const oldSet = util.target.prototype.setSize; - util.target.prototype.setSize = function (size) { - oldSet.call(this, size); - if (util.target.ext30_mirror0) util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[0] *= util.target.ext30_mirror0; - if (util.target.ext30_mirror1) util.target.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[1] *= util.target.ext30_mirror1; - } - util.target.ext30_isHook = true; - } - util.target['ext30_mirror' + args.mirrorMethod] *= -1; - util.target.setSize(util.target._size); + util.target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e,t) => { + if (util.target.ext30_mirror0) this.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[0] *= util.target.ext30_mirror0; + if (util.target.ext30_mirror1) this.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[1] *= util.target.ext30_mirror1; + }); + util.target.ext30_isHook = true; + } + util.target['ext30_mirror' + args.mirrorMethod] *= -1; + util.target.setSize(util.target._size); } //清除镜像 clearMirror(args, util) { util.target.ext30_mirror0 = 1; - util.target.ext30_mirror1 = 1; - util.target.setSize(util.target._size); + util.target.ext30_mirror1 = 1; + util.target.setSize(util.target._size); //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); //util.target.runtime.requestRedraw(); } @@ -1836,11 +1837,12 @@ class ArkosExtensions { // //跨域执行 anotherRun(args, util) { - console.info(util);//TODO + console.info(util);//TODO if (!util.thread.ex_30Ext_count) { util.thread.ex_30Ext_count = true; util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[0]; + //util.thread.target = util.target.sprite.clones[0]; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.NAME).sprite.clones[0]; util.startBranch(1, true); } else { util.thread.target = util.thread.ex_30Ext_oldTarget; @@ -1852,7 +1854,8 @@ class ArkosExtensions { if (!util.thread.ex_30Ext_count) { util.thread.ex_30Ext_count = true; util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = util.target.sprite.clones[args.cloneId]; + //util.thread.target = util.target.sprite.clones[args.cloneId]; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.NAME).sprite.clones[args.cloneId]; util.startBranch(1, true); } else { util.thread.target = util.thread.ex_30Ext_oldTarget; @@ -1885,4 +1888,4 @@ window.tempExt = { 'hcn.description': 'hcn test', }, }, -} \ No newline at end of file +} From f788d75a12cad0aef480f346ca1b49e06eabad70 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Sun, 31 Jul 2022 00:07:15 +0800 Subject: [PATCH 089/357] =?UTF-8?q?args=E5=A5=BD=E5=83=8F=E6=9C=89?= =?UTF-8?q?=E9=97=AE=E9=A2=98=20=E8=BE=93=E5=87=BA=E6=B5=8B=E8=AF=95?= =?UTF-8?q?=E4=B8=80=E4=B8=8B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 464dd08f..b686b955 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1837,7 +1837,7 @@ class ArkosExtensions { // //跨域执行 anotherRun(args, util) { - console.info(util);//TODO + console.info(args); if (!util.thread.ex_30Ext_count) { util.thread.ex_30Ext_count = true; util.thread.ex_30Ext_oldTarget = util.thread.target; From 7884e36eaff550d48d2a1db24fa10fbdc680c236 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Sun, 31 Jul 2022 00:15:54 +0800 Subject: [PATCH 090/357] =?UTF-8?q?=E8=B7=A8=E5=9F=9F=E5=A4=A7=E6=8A=B5?= =?UTF-8?q?=E6=98=AF=E6=B2=A1=E9=97=AE=E9=A2=98=E7=9A=84=E4=BA=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index b686b955..8d3f129a 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1837,12 +1837,10 @@ class ArkosExtensions { // //跨域执行 anotherRun(args, util) { - console.info(args); if (!util.thread.ex_30Ext_count) { util.thread.ex_30Ext_count = true; util.thread.ex_30Ext_oldTarget = util.thread.target; - //util.thread.target = util.target.sprite.clones[0]; - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.NAME).sprite.clones[0]; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName).sprite.clones[0]; util.startBranch(1, true); } else { util.thread.target = util.thread.ex_30Ext_oldTarget; @@ -1854,8 +1852,7 @@ class ArkosExtensions { if (!util.thread.ex_30Ext_count) { util.thread.ex_30Ext_count = true; util.thread.ex_30Ext_oldTarget = util.thread.target; - //util.thread.target = util.target.sprite.clones[args.cloneId]; - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.NAME).sprite.clones[args.cloneId]; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName).sprite.clones[args.cloneId]; util.startBranch(1, true); } else { util.thread.target = util.thread.ex_30Ext_oldTarget; From d12584886950ff982fa901b97c05125f373e3f64 Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Sun, 31 Jul 2022 20:39:05 +0800 Subject: [PATCH 091/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E8=A7=A3=E5=86=B3IR?= =?UTF-8?q?=E9=97=AE=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 8d3f129a..21d79d3a 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1037,7 +1037,7 @@ class ArkosExtensions { "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 { opcode: 'anotherRun', - blockType: 'conditional', + blockType: 'loop', text: this.formatMessage('30Ext.block.anotherRun'), arguments: { spriteName: { @@ -1051,7 +1051,7 @@ class ArkosExtensions { }, { opcode: 'anotherRunWithClone', - blockType: 'conditional', + blockType: 'command', text: this.formatMessage('30Ext.block.anotherRunWithClone'), arguments: { spriteName: { @@ -1816,19 +1816,20 @@ class ArkosExtensions { if (!util.target.ext30_isHook) { util.target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e,t) => { - if (util.target.ext30_mirror0) this.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[0] *= util.target.ext30_mirror0; - if (util.target.ext30_mirror1) this.runtime.renderer._allDrawables[util.target.drawableID]._skinScale[1] *= util.target.ext30_mirror1; + let drawable = this.runtime.renderer._allDrawables[util.target.drawableID]; + if (util.target.ext30_mirror0) drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * util.target.ext30_mirror0; + if (util.target.ext30_mirror1) drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * util.target.ext30_mirror1; }); + util.target.ext30_mirror0 = 1; + util.target.ext30_mirror1 = 1; util.target.ext30_isHook = true; } util.target['ext30_mirror' + args.mirrorMethod] *= -1; - util.target.setSize(util.target._size); } //清除镜像 clearMirror(args, util) { util.target.ext30_mirror0 = 1; util.target.ext30_mirror1 = 1; - util.target.setSize(util.target._size); //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); //util.target.runtime.requestRedraw(); } From a913b6def9981e3bd656bf7de479421e74c2540b Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Sun, 31 Jul 2022 21:34:13 +0800 Subject: [PATCH 092/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E8=A7=A3=E5=86=B3IR?= =?UTF-8?q?=E9=97=AE=E9=A2=98=20*3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 2 ++ 1 file changed, 2 insertions(+) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 21d79d3a..6890d68d 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1,3 +1,5 @@ +//Version: 1.1.0(?) r1 +//用来标记一下文件版本 import Cast from '../utils/cast.js' import Color from '../utils/color.js' // import cover from './assets/icon.svg' From de064c011480fd2021f859a97accd37ebdc7c43d Mon Sep 17 00:00:00 2001 From: java30433 <2159116373@qq.com> Date: Sun, 31 Jul 2022 22:39:39 +0800 Subject: [PATCH 093/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E9=95=9C=E5=83=8F?= =?UTF-8?q?=E5=8A=9F=E8=83=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 6890d68d..b8740002 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1039,7 +1039,7 @@ class ArkosExtensions { "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 { opcode: 'anotherRun', - blockType: 'loop', + blockType: 'conditional', text: this.formatMessage('30Ext.block.anotherRun'), arguments: { spriteName: { @@ -1053,7 +1053,7 @@ class ArkosExtensions { }, { opcode: 'anotherRunWithClone', - blockType: 'command', + blockType: 'conditional', text: this.formatMessage('30Ext.block.anotherRunWithClone'), arguments: { spriteName: { @@ -1814,8 +1814,7 @@ class ArkosExtensions { // //镜像造型 mirrorSprite(args, util) { - //测试: 换一个监听方式 - + //OK if (!util.target.ext30_isHook) { util.target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e,t) => { let drawable = this.runtime.renderer._allDrawables[util.target.drawableID]; @@ -1827,13 +1826,16 @@ class ArkosExtensions { util.target.ext30_isHook = true; } util.target['ext30_mirror' + args.mirrorMethod] *= -1; + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } //清除镜像 clearMirror(args, util) { util.target.ext30_mirror0 = 1; util.target.ext30_mirror1 = 1; - //util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - //util.target.runtime.requestRedraw(); + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); } // //角色跨域操作 From c6d60283e35c7e5e688a41d62667b594b4f23915 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Sun, 31 Jul 2022 22:45:40 +0800 Subject: [PATCH 094/357] Update projectWith30.js --- Arkos/projectWith30.js | 9 ++------- 1 file changed, 2 insertions(+), 7 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index b8740002..056885a0 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1,5 +1,3 @@ -//Version: 1.1.0(?) r1 -//用来标记一下文件版本 import Cast from '../utils/cast.js' import Color from '../utils/color.js' // import cover from './assets/icon.svg' @@ -1826,16 +1824,13 @@ class ArkosExtensions { util.target.ext30_isHook = true; } util.target['ext30_mirror' + args.mirrorMethod] *= -1; - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - + util.target.setSize(util.target._size); } //清除镜像 clearMirror(args, util) { util.target.ext30_mirror0 = 1; util.target.ext30_mirror1 = 1; - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); + util.target.setSize(util.target._size); } // //角色跨域操作 From 907922517dfdaea2d0cab09965479a8fcfef56c8 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Sun, 31 Jul 2022 22:48:01 +0800 Subject: [PATCH 095/357] Update projectWith30.js --- Arkos/projectWith30.js | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 056885a0..3f3cc86d 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1824,13 +1824,15 @@ class ArkosExtensions { util.target.ext30_isHook = true; } util.target['ext30_mirror' + args.mirrorMethod] *= -1; - util.target.setSize(util.target._size); + util.target.emitFast(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, util.target); + util.target.runtime.requestRedraw(); } //清除镜像 clearMirror(args, util) { util.target.ext30_mirror0 = 1; util.target.ext30_mirror1 = 1; - util.target.setSize(util.target._size); + util.target.emitFast(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, util.target); + util.target.runtime.requestRedraw(); } // //角色跨域操作 From 2ebea8f8488e7d7cc7159289b8d72ca7ae2801f8 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Sun, 31 Jul 2022 22:48:57 +0800 Subject: [PATCH 096/357] Update projectWith30.js --- Arkos/projectWith30.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 3f3cc86d..adfea7f4 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1824,14 +1824,14 @@ class ArkosExtensions { util.target.ext30_isHook = true; } util.target['ext30_mirror' + args.mirrorMethod] *= -1; - util.target.emitFast(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, util.target); + util.target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); util.target.runtime.requestRedraw(); } //清除镜像 clearMirror(args, util) { util.target.ext30_mirror0 = 1; util.target.ext30_mirror1 = 1; - util.target.emitFast(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, util.target); + util.target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); util.target.runtime.requestRedraw(); } // From 5ae3f1d1e4a2f10bbbbcd5eed38a819e3d0c5341 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Sun, 31 Jul 2022 22:55:40 +0800 Subject: [PATCH 097/357] =?UTF-8?q?=E7=BE=8E=E5=8C=96=E4=BB=A3=E7=A0=81?= =?UTF-8?q?=EF=BC=8C=E7=BB=9F=E4=B8=80tab=E7=BC=A9=E8=BF=9B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 3780 ++++++++++++++++++++-------------------- 1 file changed, 1905 insertions(+), 1875 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index adfea7f4..c318c98d 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -7,1884 +7,1914 @@ import Color from '../utils/color.js' console.log(Cast.toNumber('123')) console.log(Cast.toNumber('aab')) class ArkosExtensions { - constructor(runtime) { - this.runtime = runtime - this.tempData = {} - this.sortedTable = { - list1: { order: 'desc', list: [] }, - list2: { order: 'desc', list: [] }, - } - this._formatMessage = runtime.getFormatMessage({ - 'zh-cn': { - 'ArkosExt.extensionName': 'Arkosの拓展', - 'ArkosExt.info1': '🚶 坐标和方向', - 'ArkosExt.info2': '🔠 字符串处理', - 'ArkosExt.info3': '🛠 实用积木', - 'ArkosExt.info4': '📄 信息获取', - 'ArkosExt.info5': '📊 排序表', - 'ArkosExt.info6': '🗂️ 临时数据', - 'ArkosExt.info7': '临时变量', - 'ArkosExt.info8': '临时列表', - 'ArkosExt.info9': '临时容器', - 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', - 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', - 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', - 'ArkosExt.distance': '点x1:[X1]y1:[Y1]到点x2:[X2]y2:[Y2]的距离', - 'ArkosExt.searchString': '在[str]中查找[substr]的位置(从位置[pos]开始找)', - 'ArkosExt.insertString': '在[str]的第[pos]个字符前插入[substr]', - 'ArkosExt.replaceString': '将[str]中的第[start]个到第[end]个字符,替换为[substr]', - 'ArkosExt.turnDegreesToDir': '朝方向[dir]旋转[degree]度', - - 'ArkosExt.getEffect': '获取特效[EFFECT]的值', - 'ArkosExt.color': '颜色', - 'ArkosExt.fisheye': '鱼眼', - 'ArkosExt.whirl': '旋涡', - 'ArkosExt.pixelate': '像素化', - 'ArkosExt.mosaic': '马赛克', - 'ArkosExt.brightness': '亮度', - 'ArkosExt.ghost': '虚像', - - 'ArkosExt.isHiding': '角色隐藏?', - 'ArkosExt.getRotationStyle': '当前旋转方式', - 'ArkosExt.getWidthOrHeight': '获取当前造型的[t]', - 'ArkosExt.setSize': '⚠️强行将大小设为[size](无视限制)', - 'ArkosExt.width': '宽', - 'ArkosExt.height': '高', - - 'ArkosExt.setXY': '⚠️强行移到x:[x]y:[y](无视边界)', - 'ArkosExt.getBoundaryCoord': '获取角色的[t]', - 'ArkosExt.top': '上边缘y', - 'ArkosExt.bottom': '下边缘y', - 'ArkosExt.left': '左边缘x', - 'ArkosExt.right': '右边缘x', - 'ArkosExt.isOutOfSight': '角色移到舞台区外?', - - 'ArkosExt.and': '且', - 'ArkosExt.or': '或', - - 'ArkosExt.clearSortedTable': '📊清空排序表[list]', - 'ArkosExt.setTypeOfSortedTable': '📊将排序表[list]的排序方式设为[type]', - 'ArkosExt.addToSortedTable': '📊将内容(重名的则覆盖)[name],排序值[value]加入排序表[list],附加信息[extra]', - 'ArkosExt.getFromSortedTableByNo': '📊获取排序表[list]中第[n]项的[t]', - 'ArkosExt.getFromSortedTableByName': '📊获取[name]在排序表[list]中的[t]', - 'ArkosExt.lengthOfSortedTable': '📊排序表[list]中内容数', - 'ArkosExt.deleteNameOfSortedTable': '📊删除排序表[list]中名为[name]的项', - 'ArkosExt.asc': '升序', - 'ArkosExt.desc': '降序', - - 'ArkosExt.name': '名称', - 'ArkosExt.rank': '表中位置', - 'ArkosExt.rankValue': '排序值', - 'ArkosExt.extra': '附加信息', - - 'ArkosExt.colorToHex': '颜色[COLOR]的代码', - - 'ArkosExt.deleteAllTempData': '🗂️清空所有临时数据', - 'ArkosExt.getCountOfTempData': '🗂️临时数据量', - 'ArkosExt.delTempData': '🗂️删除名为[data]的临时数据', - - 'ArkosExt.setTempVar': '🗂️将临时变量[var]设为[t]', - 'ArkosExt.addTempVar': '🗂️将临时变量[var]增加[t]', - 'ArkosExt.getTempVar': '🗂️临时数据[var]', - - 'ArkosExt.clearTempList': '🗂️创建或清空临时列表[list]', - 'ArkosExt.initTempList': '🗂️临时列表[list]内容设为[t]', - 'ArkosExt.addTempList': '🗂️向临时列表[list]加入[t]', - 'ArkosExt.opTempList': '🗂️将临时列表[list]第[n]项[op][t]', - 'ArkosExt.ListOp1': '前插入', - 'ArkosExt.ListOp2': '替换为', - 'ArkosExt.ListOp3': '增加', - 'ArkosExt.delItemOfTempList': '🗂️删除临时列表[list]第[n]项', - 'ArkosExt.getItemOfTempList': '🗂️临时列表[list]第[n]项', - 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', - - 'ArkosExt.clearTempCon': '🗂️创建或清空临时容器[con]', - 'ArkosExt.initTempCon': '🗂️临时容器[con]内容设为[t]', - 'ArkosExt.opTempCon': '🗂️临时容器[con]中的[c][op][t]', - 'ArkosExt.conOp1': '设为', - 'ArkosExt.conOp2': '增加', - 'ArkosExt.delItemOfTempCon': '🗂️删除临时容器[con]中名为[c]的内容', - 'ArkosExt.getItemOfTempConByName': '🗂️临时容器[con]中的[c]', - 'ArkosExt.getItemOfTempConByNo': '🗂️临时容器[con]第[n]项的[t]', - 'ArkosExt.conInfo1': '名称', - 'ArkosExt.conInfo2': '内容', - 'ArkosExt.lengthOfTempCon': '🗂️临时容器[con]中内容数', - - '30Ext.info': '✨ 以下扩展由_30提供', - '30Ext.info.1': '🪞 造型镜像操作', - '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', - '30Ext.menu.mirrorMethod.1': '左右镜像', - '30Ext.menu.mirrorMethod.2': '上下镜像', - '30Ext.block.clearMirror': '清除角色镜像变换', - '30Ext.info.2': '🛸 角色跨域操作', - '30Ext.block.anotherRun': '让[spriteName]运行', - '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' - }, - - en: { - 'ArkosExt.extensionName': "Arkos' Extensions", - 'ArkosExt.stringEquality': '(case sensitive)[ONE]=[TWO]', - 'ArkosExt.directionFromAtoB': 'direction from x1:[X1]y1:[Y1]to x2:[X2]y2:[Y2]', - 'ArkosExt.differenceBetweenDirections': 'direction[b] minus direction[a]', - 'ArkosExt.distance': 'distance between x1:[X1]y1:[Y1]and x2:[X2]y2:[Y2]', - 'ArkosExt.searchString': 'position of[substr]in[str],start from[pos]', - 'ArkosExt.insertString': 'insert[substr]at[pos]of[str]', - 'ArkosExt.replaceString': 'replace from[start]to[end]of[str],with[substr]', - 'ArkosExt.turnDegreesToDir': 'turn[degree] degrees toward direction[dir]', - 'ArkosExt.getEffect': 'effect[EFFECT]', - 'ArkosExt.color': 'color', - 'ArkosExt.fisheye': 'fisheye', - 'ArkosExt.whirl': 'whirl', - 'ArkosExt.pixelate': 'pixelate', - 'ArkosExt.mosaic': 'mosaic', - 'ArkosExt.brightness': 'brightness', - 'ArkosExt.ghost': 'ghost', - 'ArkosExt.isHiding': 'is hiding?', - 'ArkosExt.getRotationStyle': 'rotation style', - 'ArkosExt.getWidthOrHeight': 'get [t] of the current costume', - 'ArkosExt.setSize': '⚠️force the size to [size] % (regardless of limitation) ', - 'ArkosExt.width': 'width', - 'ArkosExt.height': 'height', - - 'ArkosExt.setXY': '⚠️force to x:[x]y:[y] (regardless of the boundary)', - 'ArkosExt.getBoundaryCoord': 'get [t] of the sprite', - 'ArkosExt.top': 'top y', - 'ArkosExt.bottom': 'bottom y', - 'ArkosExt.left': 'left x', - 'ArkosExt.right': 'right x', - 'ArkosExt.isOutOfSight': 'is out of stage?', - - 'ArkosExt.and': 'and', - 'ArkosExt.or': 'or', - - 'ArkosExt.clearSortedTable': '📊empty sorted table[list]', - 'ArkosExt.setTypeOfSortedTable': '📊set sort order of[list]to[type]', - 'ArkosExt.addToSortedTable': '📊add (overwrite if existed)[name]to table[list] with sort index value[value],extra data[extra] and sort', - 'ArkosExt.getFromSortedTableByNo': '📊get[t]of #[n] from [list]', - 'ArkosExt.getFromSortedTableByName': '📊get[t]of [name] from [list]', - 'ArkosExt.lengthOfSortedTable': '📊length of sorted table[list]', - 'ArkosExt.deleteNameOfSortedTable': '📊delete [name] in[list]', - 'ArkosExt.asc': 'ascending order', - 'ArkosExt.desc': 'descending order', - - 'ArkosExt.name': 'name', - 'ArkosExt.rank': 'rank', - 'ArkosExt.rankValue': 'rankValue', - 'ArkosExt.extra': 'extra', - - 'ArkosExt.colorToHex': 'get code of color[COLOR]', - - 'ArkosExt.info1': '🚶 Coordinate and Direction', - 'ArkosExt.info2': '🔠 String Processing', - 'ArkosExt.info3': '🛠 Utilities', - 'ArkosExt.info4': '📄 Information', - 'ArkosExt.info5': '📊 Sorted Table', - 'ArkosExt.info6': '🗂️ Temporary Data', - 'ArkosExt.info7': 'temp var', - 'ArkosExt.info8': 'temp list', - 'ArkosExt.info9': 'temp container', - - 'ArkosExt.deleteAllTempData': '🗂️clear all temporary data', - 'ArkosExt.getCountOfTempData': '🗂️count of temporary data', - 'ArkosExt.delTempData': '🗂️delete temporary data[data]', - - 'ArkosExt.setTempVar': '🗂️set temp var[var] to [t]', - 'ArkosExt.addTempVar': '🗂️change temp var[var] by [t]', - 'ArkosExt.getTempVar': '🗂️temp data[var]', - - - 'ArkosExt.clearTempList': '🗂️create or clear temp list[list]', - 'ArkosExt.initTempList': '🗂️set temp list[list]to[t]', - 'ArkosExt.addTempList': '🗂️add[t] to temp list[list]', - 'ArkosExt.opTempList': '🗂️[op][t]at [n] of temp list[list]', - 'ArkosExt.ListOp1': 'insert', - 'ArkosExt.ListOp2': 'replace with', - 'ArkosExt.ListOp3': 'change by', - 'ArkosExt.delItemOfTempList': '🗂️delete [n]of temp list[list]', - 'ArkosExt.getItemOfTempList': '🗂️item[n]of temp list[list]', - 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', - - 'ArkosExt.clearTempCon': '🗂️create or clear temp container[con]', - 'ArkosExt.initTempCon': '🗂️set temp container[con]to[t]', - 'ArkosExt.opTempCon': '🗂️[c]in temp container[con][op][t]', - 'ArkosExt.conOp1': 'set to', - 'ArkosExt.conOp2': 'change by', - 'ArkosExt.delItemOfTempCon': '🗂️delete[c]in temp container[con]', - 'ArkosExt.getItemOfTempConByName': '🗂️[c]in temp container[con]', - 'ArkosExt.getItemOfTempConByNo': '🗂️[t] of #[n] in temp container[con]', - 'ArkosExt.conInfo1': 'name', - 'ArkosExt.conInfo2': 'content', - 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', - - '30Ext.info': '✨ Contributed by _30', - '30Ext.info.1': '🪞 Mirror transform', - '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', - '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', - '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', - '30Ext.block.clearMirror': 'Clear the mirror transform', - '30Ext.info.2': '🛸 Cross sprite operation', - '30Ext.block.anotherRun': 'Let [spriteName] run', - '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' - }, - }) - } - - formatMessage(id) { - return this._formatMessage({ - id, - default: id, - description: id, - }) - } - - getInfo() { - return { - id: 'hcnTest', // 拓展id - name: this.formatMessage('ArkosExt.extensionName'), - - color1: '#FF8383', - // menuIconURI: icon, - // blockIconURI: icon, - blocks: [ - "---" + this.formatMessage("ArkosExt.info1"), //🏃 坐标&角度 - // 计算点A到点B的方向 - { - opcode: 'getDirFromAToB', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.directionFromAtoB'), - arguments: { - X1: { - type: 'number', - defaultValue: 0, - }, - Y1: { - type: 'number', - defaultValue: 0, - }, - X2: { - type: 'number', - defaultValue: 0, - }, - Y2: { - type: 'number', - defaultValue: 0, - }, - }, - }, - // 计算角b-角a的角度差 - { - opcode: 'differenceBetweenDirections', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.differenceBetweenDirections'), - arguments: { - a: { - type: 'number', - defaultValue: 0, - }, - b: { - type: 'number', - defaultValue: 0, - }, - }, - }, - // 两点距离 - { - opcode: 'disFromAToB', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.distance'), - arguments: { - X1: { - type: 'number', - defaultValue: 0, - }, - Y1: { - type: 'number', - defaultValue: 0, - }, - X2: { - type: 'number', - defaultValue: 0, - }, - Y2: { - type: 'number', - defaultValue: 0, - }, - }, - }, - //朝..方向旋转..角度 - { - opcode: 'turnDegreesToDir', - blockType: 'command', - text: this.formatMessage('ArkosExt.turnDegreesToDir'), - arguments: { - degree: { - type: 'number', - defaultValue: 45, - }, - dir: { - type: 'angle', - defaultValue: 10, - }, - }, - filter: ['sprite'] - }, - "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 - // 查找子字符串,从pos开始 - { - opcode: 'indexof', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.searchString'), - arguments: { - str: { - type: 'string', - defaultValue: 'banana', - }, - substr: { - type: 'string', - defaultValue: 'na', - }, - pos: { - type: 'number', - defaultValue: 1, - }, - }, - }, - // 在字符串中插入子字符串 - { - opcode: 'insertStr', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.insertString'), - arguments: { - str: { - type: 'string', - defaultValue: 'ac', - }, - substr: { - type: 'string', - defaultValue: 'b', - }, - pos: { - type: 'number', - defaultValue: 2, - }, - }, - }, - // 替换字符串中的从..到..的字符串 - { - opcode: 'replaceStr', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.replaceString'), - arguments: { - str: { - type: 'string', - defaultValue: 'ABCDEF', - }, - substr: { - type: 'string', - defaultValue: 'XX', - }, - start: { - type: 'number', - defaultValue: 3, - }, - end: { - type: 'number', - defaultValue: 4, - }, - }, - }, - "---" + this.formatMessage("ArkosExt.info3"), //🔧实用积木 - // 判断相等(区分大小写) - { - opcode: 'strictlyEquals', - blockType: 'Boolean', - text: this.formatMessage('ArkosExt.stringEquality'), - arguments: { - ONE: { - type: 'string', - defaultValue: 'A', - }, - TWO: { - type: 'string', - defaultValue: 'a', - }, - }, - }, - //返回值转bool积木 - { - opcode: 'reporterToBoolean', - blockType: 'Boolean', - text: '[t]', - arguments: { - t: { - type: 'string', - defaultValue: '1', - } - }, - }, - //形如 a≤b≤c - { - opcode: 'compareTwoSides', - blockType: 'Boolean', - text: '[a][op1][b][op2][c]', - arguments: { - a: { - type: 'string', - defaultValue: '1', - }, - b: { - type: 'string', - defaultValue: 'x', - }, - c: { - type: 'string', - defaultValue: '3', - }, - op1: { - type: 'string', - menu: 'opMenu1', - }, - op2: { - type: 'string', - menu: 'opMenu1', - }, - }, - }, - //形如:a≤b且/或>c op1,op2 logic compareTwoSidesPlus - { - opcode: 'compareTwoSidesPlus', - blockType: 'Boolean', - text: '[a][op1][b][logic][op2][c]', - arguments: { - a: { - type: 'string', - defaultValue: 'x', - }, - b: { - type: 'string', - defaultValue: '1', - }, - c: { - type: 'string', - defaultValue: '3', - }, - op1: { - type: 'string', - menu: 'opMenu2', - defaultValue: '<', - }, - op2: { - type: 'string', - menu: 'opMenu2', - defaultValue: '>', - }, - logic: { - type: 'string', - menu: 'logicMenu', - defaultValue: 'or',// - }, - }, - }, - //获取颜色HEX - { - opcode: 'colorToHex', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.colorToHex'), - arguments: { - COLOR: { - type: 'color', - //defaultValue: '1', - }, - }, - }, - //强行设置大小 - { - opcode: 'setSize', - blockType: 'command', - text: this.formatMessage('ArkosExt.setSize'), - arguments: { - size: { - type: 'number', - defaultValue: 9999, - }, - }, - filter: ['sprite'] - }, - //强行移到xy - { - opcode: 'setXY', - blockType: 'command', - text: this.formatMessage('ArkosExt.setXY'), - arguments: { - x: { - type: 'number', - defaultValue: 100000, - }, - y: { - type: 'number', - defaultValue: 100000, - }, - }, - filter: ['sprite'] - }, - "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 - //获取特效值 - { - opcode: 'getEffect', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getEffect'), - arguments: { - EFFECT: { - type: 'string', - menu: 'effectMenu', - }, - }, - }, - //是否隐藏 - { - opcode: 'isHiding', - blockType: 'Boolean', - text: this.formatMessage('ArkosExt.isHiding'), - filter: ['sprite'] - }, - //获取旋转方式 - { - opcode: 'getRotationStyle', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getRotationStyle'), - filter: ['sprite'] - }, - //获取造型0宽1高 - { - opcode: 'getWidthOrHeight', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getWidthOrHeight'), - arguments: { - t: { - type: 'string', - menu: 'WOrH', - }, - }, - }, - //获取角色边缘xy - { - opcode: 'getBoundaryCoord', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getBoundaryCoord'), - arguments: { - t: { - type: 'string', - menu: 'boundaryMenu', - } - }, - filter: ['sprite'] - }, - //是否跑到舞台外 - { - opcode: 'isOutOfSight', - blockType: 'Boolean', - text: this.formatMessage('ArkosExt.isOutOfSight'), - filter: ['sprite'] - }, - "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 - //📊清空排序表 - { - opcode: 'clearSortedTable', - blockType: 'command', - text: this.formatMessage('ArkosExt.clearSortedTable'), - arguments: { - list: { - type: 'string', - menu: 'sortedTableMenu', - }, - }, - }, - { - //📊排序表排序方式 - opcode: 'setTypeOfSortedTable', - blockType: 'command', - text: this.formatMessage('ArkosExt.setTypeOfSortedTable'), - arguments: { - list: { - type: 'string', - menu: 'sortedTableMenu', - }, - type: { - type: 'string', - menu: 'sortOrder', - }, - }, - }, - { - //📊将XX加入排序表 - opcode: 'addToSortedTable', - blockType: 'command', - text: this.formatMessage('ArkosExt.addToSortedTable'), - arguments: { - list: { - type: 'string', - menu: 'sortedTableMenu', - }, - name: { - type: 'string', - defaultValue: '小明', - }, - value: { - type: 'number', - defaultValue: '95', - }, - extra: { - type: 'string', - defaultValue: '20212490', - }, - }, - }, - { - //📊获取排序表第n项 - opcode: 'getFromSortedTableByNo', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getFromSortedTableByNo'), - arguments: { - list: { - type: 'string', - menu: 'sortedTableMenu', - }, - n: { - type: 'number', - defaultValue: 1, - }, - t: { - type: 'string', - defaultValue: '1', - menu: 'tableItemPropertyMenu', - }, - }, - }, - { - //📊获取排序表特定名字内容 - opcode: 'getFromSortedTableByName', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getFromSortedTableByName'), - arguments: { - list: { - type: 'string', - menu: 'sortedTableMenu', - }, - name: { - type: 'string', - defaultValue: '小明', - }, - t: { - type: 'string', - defaultValue: '2', - menu: 'tableItemPropertyMenu', - }, - }, - }, - { - //📊获取排序表长度 - opcode: 'lengthOfSortedTable', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.lengthOfSortedTable'), - arguments: { - list: { - type: 'string', - menu: 'sortedTableMenu', - }, - }, - }, - { - //📊删除表中内容 - opcode: 'deleteNameOfSortedTable', - blockType: 'command', - text: this.formatMessage('ArkosExt.deleteNameOfSortedTable'), - arguments: { - list: { - type: 'string', - menu: 'sortedTableMenu', - }, - name: { - type: 'string', - defaultValue: '小明', - }, - }, - }, - - "---" + this.formatMessage("ArkosExt.info6"), //🗂️ 临时数据 - //清空所有临时数据 - { - opcode: 'deleteAllTempData', - blockType: 'command', - text: this.formatMessage('ArkosExt.deleteAllTempData'), - }, - //临时数据量 - { - opcode: 'getCountOfTempData', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getCountOfTempData'), - }, - //删除临时数据 - { - opcode: 'delTempData', - blockType: 'command', - text: this.formatMessage('ArkosExt.delTempData'), - arguments: { - data: { - type: 'string', - defaultValue: 'i', - }, - }, - }, - "---" + this.formatMessage("ArkosExt.info7"), //临时变量 - //设置临时数据 - { - opcode: 'setTempVar', - blockType: 'command', - text: this.formatMessage('ArkosExt.setTempVar'), - arguments: { - var: { - type: 'string', - defaultValue: 'i', - }, - t: { - type: 'string', - defaultValue: '0', - }, - }, - }, - //增加临时数据 - { - opcode: 'addTempVar', - blockType: 'command', - text: this.formatMessage('ArkosExt.addTempVar'), - arguments: { - var: { - type: 'string', - defaultValue: 'i', - }, - t: { - type: 'number', - defaultValue: 1, - }, - }, - }, - //获取临时数据 - { - opcode: 'getTempVar', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getTempVar'), - arguments: { - var: { - type: 'string', - defaultValue: 'i', - }, - }, - }, - "---" + this.formatMessage("ArkosExt.info8"), //临时列表 - //创建或清空临时列表 - { - opcode: 'clearTempList', - blockType: 'command', - text: this.formatMessage('ArkosExt.clearTempList'), - arguments: { - list: { - type: 'string', - defaultValue: 'list', - }, - }, - }, - //设置临时列表 - { - opcode: 'initTempList', - blockType: 'command', - text: this.formatMessage('ArkosExt.initTempList'), - arguments: { - list: { - type: 'string', - defaultValue: 'list', - }, - t: { - type: 'string', - defaultValue: '[1,2,"apple"]', - }, - }, - }, - //向临时列表加入 - { - opcode: 'addTempList', - blockType: 'command', - text: this.formatMessage('ArkosExt.addTempList'), - arguments: { - list: { - type: 'string', - defaultValue: 'list', - }, - t: { - type: 'string', - defaultValue: 'thing', - }, - }, - }, - //操作临时列表 - { - opcode: 'opTempList', - blockType: 'command', - text: this.formatMessage('ArkosExt.opTempList'), - arguments: { - list: { - type: 'string', - defaultValue: 'list', - }, - op: { - type: 'string', - menu: 'ListOpMenu', - }, - n: { - type: 'number', - defaultValue: 1, - }, - t: { - type: 'string', - defaultValue: 'thing', - }, - }, - }, - //删除临时列表XX项 - { - opcode: 'delItemOfTempList', - blockType: 'command', - text: this.formatMessage('ArkosExt.delItemOfTempList'), - arguments: { - list: { - type: 'string', - defaultValue: 'list', - }, - n: { - type: 'number', - defaultValue: 1, - }, - }, - }, - //获取临时列表XX项 - { - opcode: 'getItemOfTempList', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getItemOfTempList'), - arguments: { - list: { - type: 'string', - defaultValue: 'list', - }, - n: { - type: 'number', - defaultValue: 1, - }, - }, - }, - //临时列表长度 - { - opcode: 'lengthOfTempList', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.lengthOfTempList'), - arguments: { - list: { - type: 'string', - defaultValue: 'list', - }, - }, - }, - "---" + this.formatMessage("ArkosExt.info9"), //临时容器 - //创建或清空临时容器 - { - opcode: 'clearTempCon', - blockType: 'command', - text: this.formatMessage('ArkosExt.clearTempCon'), - arguments: { - con: { - type: 'string', - defaultValue: 'con1', - }, - }, - }, - //设置临时容器 - { - opcode: 'initTempCon', - blockType: 'command', - text: this.formatMessage('ArkosExt.initTempCon'), - arguments: { - con: { - type: 'string', - defaultValue: 'con1', - }, - t: { - type: 'string', - defaultValue: '{"coins":200,"backpack":["wood","bread"]}', - }, - }, - }, - //操作临时容器 - { - opcode: 'opTempCon', - blockType: 'command', - text: this.formatMessage('ArkosExt.opTempCon'), - arguments: { - con: { - type: 'string', - defaultValue: 'con1', - }, - op: { - type: 'string', - menu: 'conOpMenu', - }, - c: { - type: 'string', - defaultValue: 'coins', - }, - t: { - type: 'string', - defaultValue: '520', - }, - }, - }, - //删除临时容器名为xx的内容 - { - opcode: 'delItemOfTempCon', - blockType: 'command', - text: this.formatMessage('ArkosExt.delItemOfTempCon'), - arguments: { - con: { - type: 'string', - defaultValue: 'con1', - }, - c: { - type: 'string', - defaultValue: 'coins', - }, - }, - }, - //获取临时容器名为XX的内容 - { - opcode: 'getItemOfTempConByName', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getItemOfTempConByName'), - arguments: { - con: { - type: 'string', - defaultValue: 'con1', - }, - c: { - type: 'string', - defaultValue: 'coins', - }, - }, - }, - //获取临时容器第n项的xx - { - opcode: 'getItemOfTempConByNo', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.getItemOfTempConByNo'), - arguments: { - con: { - type: 'string', - defaultValue: 'con1', - }, - n: { - type: 'number', - defaultValue: 1, - }, - t: { - type: 'string', - menu: 'conInfoMenu', - defaultValue: '2', - }, - }, - }, - //临时容器长度 - { - opcode: 'lengthOfTempCon', - blockType: 'reporter', - text: this.formatMessage('ArkosExt.lengthOfTempCon'), - arguments: { - con: { - type: 'string', - defaultValue: 'con1', - }, - }, - }, - - - // - "---" + this.formatMessage("30Ext.info"), //感谢30提供的扩展 - "---" + this.formatMessage("30Ext.info.1"), //造型镜像 - // 镜像造型 - { - opcode: 'mirrorSprite', - blockType: 'command', - text: this.formatMessage('30Ext.block.mirrorSprite'), - arguments: { - mirrorMethod: { - type: 'string', - menu: 'mirrorMenu' - } - } - }, - // 清除镜像 - { - opcode: 'clearMirror', - blockType: 'command', - text: this.formatMessage('30Ext.block.clearMirror') - }, - "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 - { - opcode: 'anotherRun', - blockType: 'conditional', - text: this.formatMessage('30Ext.block.anotherRun'), - arguments: { - spriteName: { - type: 'string', - menu: 'spritesMenu' - }, - SUBSTACK: { //TODO - type: "input_statement" - } - } - }, - { - opcode: 'anotherRunWithClone', - blockType: 'conditional', - text: this.formatMessage('30Ext.block.anotherRunWithClone'), - arguments: { - spriteName: { - type: 'string', - menu: 'spritesMenu' - }, - cloneId: { - type: 'number', - defaultValue: 1 - }, - SUBSTACK: { - type: "input_statement" - } - - } - }, - - ], - menus: { - conInfoMenu:[ - { - text: this.formatMessage('ArkosExt.conInfo1'), //名称 - value: '1' - }, - { - text: this.formatMessage('ArkosExt.conInfo2'), //内容 - value: '2' - }, - ], - conOpMenu:[ - { - text: this.formatMessage('ArkosExt.conOp1'), //设为 - value: '1' - }, - { - text: this.formatMessage('ArkosExt.conOp2'), //增加 - value: '2' - }, - ], - ListOpMenu: [ - { - text: this.formatMessage('ArkosExt.ListOp1'), //插入 - value: '1' - }, - { - text: this.formatMessage('ArkosExt.ListOp2'), //替换 - value: '2' - }, - { - text: this.formatMessage('ArkosExt.ListOp3'), //增加 - value: '3' - }, - ], - tableItemPropertyMenu: [ - { - text: this.formatMessage('ArkosExt.name'), - value: '1' - }, - { - text: this.formatMessage('ArkosExt.rank'), - value: '2' - }, - { - text: this.formatMessage('ArkosExt.rankValue'), - value: '3' - }, - { - text: this.formatMessage('ArkosExt.extra'), - value: '4' - }, - ], - sortOrder: [ - { - text: this.formatMessage('ArkosExt.asc'), - value: 'asc'//升序 - }, - { - text: this.formatMessage('ArkosExt.desc'), - value: 'desc'//降序 - }, - ], - sortedTableMenu: { - items: 'findAllSortedTable', - acceptReporters: true, - }, - //判断符菜单 - opMenu1: ['<', '≤', '=', '≠',], - opMenu2: ['<', '>', '≤', '≥', '=', '≠',], - //logicMenu - logicMenu: [ - { - text: this.formatMessage('ArkosExt.or'), - value: 'or' - }, - { - text: this.formatMessage('ArkosExt.and'), - value: 'and' - }, - ], - //角色上下左右边缘 - boundaryMenu: [ - { - text: this.formatMessage('ArkosExt.top'), - value: '1' - }, - { - text: this.formatMessage('ArkosExt.bottom'), - value: '2' - }, - { - text: this.formatMessage('ArkosExt.left'), - value: '3' - }, - { - text: this.formatMessage('ArkosExt.right'), - value: '4' - }, - ], - //0宽1高 菜单 - WOrH: [ - { - text: this.formatMessage('ArkosExt.width'), - value: '0' - }, - { - text: this.formatMessage('ArkosExt.height'), - value: '1' - }, - ], - //特效菜单 - effectMenu: [ - { - text: this.formatMessage('ArkosExt.color'), - value: 'color' - }, - { - text: this.formatMessage('ArkosExt.fisheye'), - value: 'fisheye' - }, - { - text: this.formatMessage('ArkosExt.whirl'), - value: 'whirl' - }, - { - text: this.formatMessage('ArkosExt.pixelate'), - value: 'pixelate' - }, - { - text: this.formatMessage('ArkosExt.mosaic'), - value: 'mosaic' - }, - { - text: this.formatMessage('ArkosExt.brightness'), - value: 'brightness' - }, - { - text: this.formatMessage('ArkosExt.ghost'), - value: 'ghost' - } - ], - //30Ext - spritesMenu: { - items: 'getSpritesMenu' - }, - mirrorMenu: [{ - text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: '0' - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: '1' - } - ] - }, - } - } - - strictlyEquals(args) { - // 实际上在这里直接使用严格相等是不太明智的,因为有一定的可能会遇到数字和字符比较, - // 而在Scratch中数字和字符在表现完全一样的时候几乎没有区别。 - // 因此包上Cast.toString()以使得数字和字符能够正常比较(类似 9 = "9" ) - return Cast.toString(args.ONE) === Cast.toString(args.TWO) - } - - getDirFromAToB(args) { - // 一定要先转化为数字; - const X1 = Cast.toNumber(args.X1) - const X2 = Cast.toNumber(args.X2) - const Y1 = Cast.toNumber(args.Y1) - const Y2 = Cast.toNumber(args.Y2) - - // 这里利用atan函数的性质atan(+inf)=90,atan(-inf)=-90,atan(NaN)=NaN可以省很多代码 - let a = Math.atan((X2 - X1) / (Y2 - Y1)) / Math.PI * 180 + (Y1 > Y2 ? 180 : 0) - if (a > 180) a -= 360 - return a; - } - - differenceBetweenDirections(args) { - const a = Cast.toNumber(args.a) - const b = Cast.toNumber(args.b) - let dif = b - a - dif -= Math.round(dif / 360) * 360 - if (dif === -180) dif = 180 - return dif - } - - disFromAToB(args) { - const X1 = Cast.toNumber(args.X1) - const X2 = Cast.toNumber(args.X2) - const Y1 = Cast.toNumber(args.Y1) - const Y2 = Cast.toNumber(args.Y2) - return Math.sqrt((X1 - X2) * (X1 - X2) + (Y1 - Y2) * (Y1 - Y2)) - } - - indexof(args) { - const str = Cast.toString(args.str) - const substr = Cast.toString(args.substr) - const a = str.indexOf(substr, Cast.toNumber(args.pos) - 1) - if (a === -1) { - // Scratch列表中也有查询积木,其中找不到返回的是0。建议维持原有的风格。 - return 0 - } - return a + 1 - } - - insertStr(args) { - const str = Cast.toString(args.str) - const substr = Cast.toString(args.substr) - let pos = Cast.toNumber(args.pos) - 1 - if (pos < 0) { - pos = 0 - } - return str.slice(0, pos) + substr + str.slice(pos) - } - - replaceStr(args) { - const str = Cast.toString(args.str) - const substr = Cast.toString(args.substr) - let start = Cast.toNumber(args.start) - let end = Cast.toNumber(args.end) - if (start > end) { - const t = end - end = start - start = t - } - if (start < 1) start = 1 - return str.slice(0, start - 1) + substr + str.slice(end) - } - - - turnDegreesToDir(args, util) { - const degree = Cast.toNumber(args.degree); - const dir = Cast.toNumber(args.dir); - const dif = this.differenceBetweenDirections({ a: util.target.direction, b: dir }); - if (Math.abs(dif) < degree) - util.target.setDirection(dir); - else if (dif < 0) - util.target.setDirection(util.target.direction - degree); - else - util.target.setDirection(util.target.direction + degree); - } - - //获取特效的数值 - getEffect(args, util) { - let effect = Cast.toString(args.EFFECT).toLowerCase(); - if (!util.target.effects.hasOwnProperty(effect)) return 0; - return util.target.effects[effect]; - } - - //角色是否可见 - isHiding(args, util) { - return !util.target.visible; - } - - //获取图层(逝一逝) - // getLayer (args, util) { - // return util.target.layer; - // } - - //获取当前角色的旋转方式 - getRotationStyle(args, util) { - return util.target.rotationStyle; - } - - //获取当前造型的长/宽 - getWidthOrHeight(args, util) { - const costumeSize = util.target.renderer.getCurrentSkinSize(util.target.drawableID); - return costumeSize[args.t]; - } - - //强行设置大小(逝一逝) - setSize(args, util) { - if (util.target.isStage) { - return; - } - if (util.target.renderer) { - args.size = this._clamp(Cast.toNumber(args.size), 0.1, 100000000) - util.target.size = args.size; - const { direction, scale } = util.target._getRenderedDirectionAndScale(); - util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); - if (util.target.visible) { - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - } - } - util.target.runtime.requestTargetsUpdate(util.target); - } - - _clamp(n, min, max) { - return Math.min(Math.max(n, min), max); - } - - //强行设置XY(逝一逝) - setXY(args, util) { - if (util.target.isStage) return; - args.x = this._clamp(Cast.toNumber(args.x), -100000000, 100000000) - args.y = this._clamp(Cast.toNumber(args.y), -100000000, 100000000) - const oldX = util.target.x; - const oldY = util.target.y; - util.target.x = args.x; - util.target.y = args.y; - if (util.target.renderer) { - util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); - if (util.target.visible) { - util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - } - } else { - util.target.x = x; - util.target.y = y; - } - util.target.emit('TARGET_MOVED', util.target, oldX, oldY, false); - util.target.runtime.requestTargetsUpdate(util.target); - } - - //获取角色边缘的坐标 - getBoundaryCoord(args, util) { - const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); - switch (args.t) { - case '1': - return bounds.top; - case '2': - return bounds.bottom; - case '3': - return bounds.left; - case '4': - return bounds.right; - default: - return ''; - } - } - - //是否在舞台外 - isOutOfSight(args, util) { - // console.log(util.target.runtime.renderer) - // console.log(util.target.renderer) - if (util.target.renderer) { - const stageWidth = util.target.runtime.stageWidth; - const stageHeight = util.target.runtime.stageHeight; - console.log('stageWidth', stageWidth); - console.log('stageHeight', stageHeight); - const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); - if (bounds.right < -stageWidth / 2 || - bounds.left > stageWidth / 2 || - bounds.bottom > stageHeight / 2 || - bounds.top < -stageHeight / 2) { - return true; - } - } - return false; - } - - //形如:<() > - reporterToBoolean(args) { - if (Cast.toString(args.t).toLowerCase() === 'false') return false; - if (args.t === '0') return false; - return (args.t) ? true : false; - } - - compare(a, b, op) { - switch (op) { - case '<': - return Cast.compare(a, b) < 0; - case '>': - return Cast.compare(a, b) > 0; - case '=': - return Cast.compare(a, b) === 0; - case '≤': - return Cast.compare(a, b) <= 0; - case '≥': - return Cast.compare(a, b) >= 0; - case '≠': - return Cast.compare(a, b) !== 0; - default: - return false; - } - } - - //形如:a≤b≤c op1,op2 - compareTwoSides(args) { - return this.compare(args.a, args.b, args.op1) && this.compare(args.b, args.c, args.op2) - } - - //形如:a≤b且/或>c op1,op2 logic - compareTwoSidesPlus(args) { - switch (args.logic) { - case 'or': - return this.compare(args.a, args.b, args.op1) || this.compare(args.a, args.c, args.op2) - case 'and': - return this.compare(args.a, args.b, args.op1) && this.compare(args.a, args.c, args.op2) - default: - return false; - } - } - - //数组排序规则 - sortRule(propName, order) { - return (a, b) => { - a = a[propName] - b = b[propName] - if (a > b) return order === 'asc' ? 1 : -1; - else if (a < b) return order === 'asc' ? -1 : 1; - else return 0; - } - } - - //查找所有排序表 - findAllSortedTable() { - const list = []; - let temp = this.sortedTable; - Object.keys(temp).forEach(obj => { - //if ( Array.isArray (temp[obj]) ) { - list.push(obj); - //} - }); - if (list.length === 0) { - list.push({ - text: '-', - value: 'empty', - }); - } - //list.sort(this.sortRule("text")); - return list; - } - - createTableIfNotExist(list) { - if (!(list in this.sortedTable)) - this.sortedTable[list] = { order: 'desc', list: [] }; - } - - sortTable(list) { - this.sortedTable[list].list.sort(this.sortRule("rankValue", this.sortedTable[list].order)); - } - - //📊清空排序表 - clearSortedTable(args) { - this.createTableIfNotExist(args.list) - this.sortedTable[args.list].list = []; - } - - //📊设置排序方式 - setTypeOfSortedTable(args) { - this.createTableIfNotExist(args.list) - this.sortedTable[args.list].order = args.type; - this.sortTable(args.list) - } - - //查找在列表中的插入位置(已有则覆盖) - _findPlaceAndInsert(list, order, item) { - //删除已存在的内容 - for (let i = 0; i < list.length; i++) { - if (list[i].name === item.name) { - //删除同名项 - list.splice(i, 1); - break; - } - } - //查找插入位置并插入 - for (let i = 0; i < list.length; i++) { - if ((list[i].rankValue > item.rankValue && order === 'asc') || - (list[i].rankValue < item.rankValue && order === 'desc')) { - //插入在该项前 - list.splice(i, 0, item); - return; - } - } - //没找到插入位置,加在末尾 - list.push(item); - } - - //📊将内容加入表 - addToSortedTable(args) { - this.createTableIfNotExist(args.list) - this._findPlaceAndInsert( - this.sortedTable[args.list].list, - this.sortedTable[args.list].order, - { name: args.name, rankValue: args.value, extra: args.extra }); - } - - _getTInItem(item, t, rank) { - if (item === undefined) return ''; - switch (t) { - case '1': - return item.name; - case '2': - return rank; - case '3': - return item.rankValue; - case '4': - return item.extra; - default: - return ''; - } - } - - //📊获取第n项 - getFromSortedTableByNo(args) { - if (!(args.list in this.sortedTable)) return ''; - let list = this.sortedTable[args.list].list; - return this._getTInItem(list[args.n - 1], args.t, Cast.toNumber(args.n)); - } - - _getItemIdxByName(list, name) { - for (let i = 0; i < list.length; i++) { - if (list[i].name === name) { - return i; - } - } - return -1; - } - - //📊获取名为XX的项 - getFromSortedTableByName(args) { - if (!(args.list in this.sortedTable)) return ''; - let list = this.sortedTable[args.list].list; - let n = this._getItemIdxByName(list, args.name); - if (n === -1) return ''; - return this._getTInItem(list[n], args.t, n + 1); - } - - //📊获取排序表长度 - lengthOfSortedTable(args) { - if (!(args.list in this.sortedTable)) return 0; - return this.sortedTable[args.list].list.length; - } - - //📊删除排序表名为XX的内容 - deleteNameOfSortedTable(args) { - if (!(args.list in this.sortedTable)) return; - let list = this.sortedTable[args.list].list; - let n = this._getItemIdxByName(list, args.name); - if (n === -1) return; - list.splice(n, 1); - } - - //获取颜色HEX码 - colorToHex(args, util) { - let c = Cast.toRgbColorList(args.COLOR) - return Color.rgbToHex({ r: c[0], g: c[1], b: c[2] }); - } - - - //🗂️ 临时变量积木 - - //来自 -6 :任意内容转字符或数字 - _anythingToNumberString(value) { - switch(typeof(value)){ - case "string": - case "number": - break; - case "object": - value = JSON.stringify(value); - break; - default: - value = ''; //包含了undefined - } - return value; - } - - deleteAllTempData(args) { - this.tempData = {}; - } - - getCountOfTempData(args) { - return Object.keys(this.tempData).length; - } - - delTempData(args) { - delete this.tempData[Cast.toString(args.data)]; - } - - setTempVar(args) { - this.tempData[Cast.toString(args.var)] = args.t; - } - - addTempVar(args) { - this.tempData[Cast.toString(args.var)] = Cast.toNumber(this.tempData[Cast.toString(args.var)]) + Cast.toNumber(args.t); - } - - getTempVar(args) { - let temp = this.tempData[Cast.toString(args.var)] - if (typeof(temp) === 'object') return JSON.stringify(temp); - return Cast.toString(temp); - } - - //创建或清空 - clearTempList(args) { - this.tempData[Cast.toString(args.list)] = []; - } - - initTempList(args) { - try { - let content = JSON.parse(Cast.toString(args.t)) - if (Array.isArray(content)) { - this.tempData[Cast.toString(args.list)] = content; - } - } catch (e) { - - } - } - - addTempList(args) { - let list = this.tempData[Cast.toString(args.list)] - if (!Array.isArray(list)) return; - list.push(Cast.toString(args.t)); - } - - opTempList(args) { - let list = this.tempData[Cast.toString(args.list)] - if (!Array.isArray(list)) return; - let n = Cast.toNumber(args.n) - if (n < 1 || n > list.length + 1) return; - n -= 1; - switch (args.op) { - case '1'://插入 - list.splice(n, 0, args.t); - return; - case '2'://替换 - list[n] = args.t; - return; - case '3'://增加 - list[n] = Cast.toNumber(list[n]) + Cast.toNumber(args.t); - return; - default: - return; - } - } - - delItemOfTempList(args) { - let list = this.tempData[Cast.toString(args.list)] - if (!Array.isArray(list)) return; - let n = Cast.toNumber(args.n) - if (n < 1 || n > list.length) return; - n -= 1; - list.splice(n, 1); - } - - getItemOfTempList(args) { - let list = this.tempData[Cast.toString(args.list)] - if (!Array.isArray(list)) return ''; - let n = Cast.toNumber(args.n) - if (n < 1 || n > list.length) return ''; - n -= 1; - return Cast.toString(list[n]); - } - - lengthOfTempList(args) { - let list = this.tempData[Cast.toString(args.list)] - if (!Array.isArray(list)) return 0; - return list.length; - } - - //容器 - clearTempCon(args) { - this.tempData[Cast.toString(args.con)] = {}; - } - - initTempCon(args) { - try { - let content = JSON.parse(Cast.toString(args.t)) - if (typeof(content) === 'object' && content !== null) { - this.tempData[Cast.toString(args.con)] = content; - } - } catch (e) { - - } - } - - opTempCon(args) { - let con = this.tempData[Cast.toString(args.con)] - if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return; - let c = Cast.toString(args.c) - switch (args.op) { - case '1'://设为 - con[c] = args.t; - return; - case '2'://增加 - if(!(c in con)) return; - con[c] = Cast.toNumber(con[c]) + Cast.toNumber(args.t); - return; - default: - return; - } - } - - delItemOfTempCon(args) { - let con = this.tempData[Cast.toString(args.con)] - if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return; - delete con[Cast.toString(args.c)]; - } - - getItemOfTempConByName(args) { - let con = this.tempData[Cast.toString(args.con)] - if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return ''; - return Cast.toString(con[Cast.toString(args.c)]); - } - - getItemOfTempConByNo(args) { - let con = this.tempData[Cast.toString(args.con)] - if (!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return ''; - let key = Object.keys(con)[Cast.toNumber(args.n) - 1] - if (key === undefined) return ''; - switch (args.t) { - case '1'://名称 - return key; - case '2'://内容 - return this._anythingToNumberString(con[key]); - default: - return; - } - } - - lengthOfTempCon(args) { - let con = this.tempData[Cast.toString(args.con)] - if (!(typeof(con) === 'object' && con !== null)) return 0; - return Object.keys(con).length; - } - - - // - //30Ext - // - //菜单 - //动态菜单: 角色菜单 - getSpritesMenu() { - var sprites = []; - for (const targetId in this.runtime.targets) { - if (!this.runtime.targets.hasOwnProperty(targetId)) continue; - if (!this.runtime.targets[targetId].isOriginal) continue; - if (this.runtime.targets[targetId] === this.runtime._editingTarget) continue; //排除自己 - let name = this.runtime.targets[targetId].sprite.name; - sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 - } - return sprites; - } - // - //角色造型操作 - // - //镜像造型 - mirrorSprite(args, util) { - //OK - if (!util.target.ext30_isHook) { - util.target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e,t) => { - let drawable = this.runtime.renderer._allDrawables[util.target.drawableID]; - if (util.target.ext30_mirror0) drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * util.target.ext30_mirror0; - if (util.target.ext30_mirror1) drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * util.target.ext30_mirror1; - }); - util.target.ext30_mirror0 = 1; - util.target.ext30_mirror1 = 1; - util.target.ext30_isHook = true; - } - util.target['ext30_mirror' + args.mirrorMethod] *= -1; - util.target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - } - //清除镜像 - clearMirror(args, util) { - util.target.ext30_mirror0 = 1; - util.target.ext30_mirror1 = 1; - util.target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); - util.target.runtime.requestRedraw(); - } - // - //角色跨域操作 - // - //跨域执行 - anotherRun(args, util) { - if (!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName).sprite.clones[0]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; - } - } - //跨域克隆体执行 - anotherRunWithClone(args, util) { - if (!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName).sprite.clones[args.cloneId]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; - } - } + constructor(runtime) { + this.runtime = runtime + this.tempData = {} + this.sortedTable = { + list1: { + order: 'desc', + list: [] + }, + list2: { + order: 'desc', + list: [] + }, + } + this._formatMessage = runtime.getFormatMessage({ + 'zh-cn': { + 'ArkosExt.extensionName': 'Arkosの拓展', + 'ArkosExt.info1': '🚶 坐标和方向', + 'ArkosExt.info2': '🔠 字符串处理', + 'ArkosExt.info3': '🛠 实用积木', + 'ArkosExt.info4': '📄 信息获取', + 'ArkosExt.info5': '📊 排序表', + 'ArkosExt.info6': '🗂️ 临时数据', + 'ArkosExt.info7': '临时变量', + 'ArkosExt.info8': '临时列表', + 'ArkosExt.info9': '临时容器', + 'ArkosExt.stringEquality': '(区分大小写)[ONE]=[TWO]', + 'ArkosExt.directionFromAtoB': '点x1:[X1]y1:[Y1]朝向点x2:[X2]y2:[Y2]的方向', + 'ArkosExt.differenceBetweenDirections': '由方向1[a]到方向2[b]的角度差', + 'ArkosExt.distance': '点x1:[X1]y1:[Y1]到点x2:[X2]y2:[Y2]的距离', + 'ArkosExt.searchString': '在[str]中查找[substr]的位置(从位置[pos]开始找)', + 'ArkosExt.insertString': '在[str]的第[pos]个字符前插入[substr]', + 'ArkosExt.replaceString': '将[str]中的第[start]个到第[end]个字符,替换为[substr]', + 'ArkosExt.turnDegreesToDir': '朝方向[dir]旋转[degree]度', + + 'ArkosExt.getEffect': '获取特效[EFFECT]的值', + 'ArkosExt.color': '颜色', + 'ArkosExt.fisheye': '鱼眼', + 'ArkosExt.whirl': '旋涡', + 'ArkosExt.pixelate': '像素化', + 'ArkosExt.mosaic': '马赛克', + 'ArkosExt.brightness': '亮度', + 'ArkosExt.ghost': '虚像', + + 'ArkosExt.isHiding': '角色隐藏?', + 'ArkosExt.getRotationStyle': '当前旋转方式', + 'ArkosExt.getWidthOrHeight': '获取当前造型的[t]', + 'ArkosExt.setSize': '⚠️强行将大小设为[size](无视限制)', + 'ArkosExt.width': '宽', + 'ArkosExt.height': '高', + + 'ArkosExt.setXY': '⚠️强行移到x:[x]y:[y](无视边界)', + 'ArkosExt.getBoundaryCoord': '获取角色的[t]', + 'ArkosExt.top': '上边缘y', + 'ArkosExt.bottom': '下边缘y', + 'ArkosExt.left': '左边缘x', + 'ArkosExt.right': '右边缘x', + 'ArkosExt.isOutOfSight': '角色移到舞台区外?', + + 'ArkosExt.and': '且', + 'ArkosExt.or': '或', + + 'ArkosExt.clearSortedTable': '📊清空排序表[list]', + 'ArkosExt.setTypeOfSortedTable': '📊将排序表[list]的排序方式设为[type]', + 'ArkosExt.addToSortedTable': '📊将内容(重名的则覆盖)[name],排序值[value]加入排序表[list],附加信息[extra]', + 'ArkosExt.getFromSortedTableByNo': '📊获取排序表[list]中第[n]项的[t]', + 'ArkosExt.getFromSortedTableByName': '📊获取[name]在排序表[list]中的[t]', + 'ArkosExt.lengthOfSortedTable': '📊排序表[list]中内容数', + 'ArkosExt.deleteNameOfSortedTable': '📊删除排序表[list]中名为[name]的项', + 'ArkosExt.asc': '升序', + 'ArkosExt.desc': '降序', + + 'ArkosExt.name': '名称', + 'ArkosExt.rank': '表中位置', + 'ArkosExt.rankValue': '排序值', + 'ArkosExt.extra': '附加信息', + + 'ArkosExt.colorToHex': '颜色[COLOR]的代码', + + 'ArkosExt.deleteAllTempData': '🗂️清空所有临时数据', + 'ArkosExt.getCountOfTempData': '🗂️临时数据量', + 'ArkosExt.delTempData': '🗂️删除名为[data]的临时数据', + + 'ArkosExt.setTempVar': '🗂️将临时变量[var]设为[t]', + 'ArkosExt.addTempVar': '🗂️将临时变量[var]增加[t]', + 'ArkosExt.getTempVar': '🗂️临时数据[var]', + + 'ArkosExt.clearTempList': '🗂️创建或清空临时列表[list]', + 'ArkosExt.initTempList': '🗂️临时列表[list]内容设为[t]', + 'ArkosExt.addTempList': '🗂️向临时列表[list]加入[t]', + 'ArkosExt.opTempList': '🗂️将临时列表[list]第[n]项[op][t]', + 'ArkosExt.ListOp1': '前插入', + 'ArkosExt.ListOp2': '替换为', + 'ArkosExt.ListOp3': '增加', + 'ArkosExt.delItemOfTempList': '🗂️删除临时列表[list]第[n]项', + 'ArkosExt.getItemOfTempList': '🗂️临时列表[list]第[n]项', + 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', + + 'ArkosExt.clearTempCon': '🗂️创建或清空临时容器[con]', + 'ArkosExt.initTempCon': '🗂️临时容器[con]内容设为[t]', + 'ArkosExt.opTempCon': '🗂️临时容器[con]中的[c][op][t]', + 'ArkosExt.conOp1': '设为', + 'ArkosExt.conOp2': '增加', + 'ArkosExt.delItemOfTempCon': '🗂️删除临时容器[con]中名为[c]的内容', + 'ArkosExt.getItemOfTempConByName': '🗂️临时容器[con]中的[c]', + 'ArkosExt.getItemOfTempConByNo': '🗂️临时容器[con]第[n]项的[t]', + 'ArkosExt.conInfo1': '名称', + 'ArkosExt.conInfo2': '内容', + 'ArkosExt.lengthOfTempCon': '🗂️临时容器[con]中内容数', + + '30Ext.info': '✨ 以下扩展由_30提供', + '30Ext.info.1': '🪞 造型镜像操作', + '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', + '30Ext.menu.mirrorMethod.1': '左右镜像', + '30Ext.menu.mirrorMethod.2': '上下镜像', + '30Ext.block.clearMirror': '清除角色镜像变换', + '30Ext.info.2': '🛸 角色跨域操作', + '30Ext.block.anotherRun': '让[spriteName]运行', + '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' + }, + + en: { + 'ArkosExt.extensionName': "Arkos' Extensions", + 'ArkosExt.stringEquality': '(case sensitive)[ONE]=[TWO]', + 'ArkosExt.directionFromAtoB': 'direction from x1:[X1]y1:[Y1]to x2:[X2]y2:[Y2]', + 'ArkosExt.differenceBetweenDirections': 'direction[b] minus direction[a]', + 'ArkosExt.distance': 'distance between x1:[X1]y1:[Y1]and x2:[X2]y2:[Y2]', + 'ArkosExt.searchString': 'position of[substr]in[str],start from[pos]', + 'ArkosExt.insertString': 'insert[substr]at[pos]of[str]', + 'ArkosExt.replaceString': 'replace from[start]to[end]of[str],with[substr]', + 'ArkosExt.turnDegreesToDir': 'turn[degree] degrees toward direction[dir]', + 'ArkosExt.getEffect': 'effect[EFFECT]', + 'ArkosExt.color': 'color', + 'ArkosExt.fisheye': 'fisheye', + 'ArkosExt.whirl': 'whirl', + 'ArkosExt.pixelate': 'pixelate', + 'ArkosExt.mosaic': 'mosaic', + 'ArkosExt.brightness': 'brightness', + 'ArkosExt.ghost': 'ghost', + 'ArkosExt.isHiding': 'is hiding?', + 'ArkosExt.getRotationStyle': 'rotation style', + 'ArkosExt.getWidthOrHeight': 'get [t] of the current costume', + 'ArkosExt.setSize': '⚠️force the size to [size] % (regardless of limitation) ', + 'ArkosExt.width': 'width', + 'ArkosExt.height': 'height', + + 'ArkosExt.setXY': '⚠️force to x:[x]y:[y] (regardless of the boundary)', + 'ArkosExt.getBoundaryCoord': 'get [t] of the sprite', + 'ArkosExt.top': 'top y', + 'ArkosExt.bottom': 'bottom y', + 'ArkosExt.left': 'left x', + 'ArkosExt.right': 'right x', + 'ArkosExt.isOutOfSight': 'is out of stage?', + + 'ArkosExt.and': 'and', + 'ArkosExt.or': 'or', + + 'ArkosExt.clearSortedTable': '📊empty sorted table[list]', + 'ArkosExt.setTypeOfSortedTable': '📊set sort order of[list]to[type]', + 'ArkosExt.addToSortedTable': '📊add (overwrite if existed)[name]to table[list] with sort index value[value],extra data[extra] and sort', + 'ArkosExt.getFromSortedTableByNo': '📊get[t]of #[n] from [list]', + 'ArkosExt.getFromSortedTableByName': '📊get[t]of [name] from [list]', + 'ArkosExt.lengthOfSortedTable': '📊length of sorted table[list]', + 'ArkosExt.deleteNameOfSortedTable': '📊delete [name] in[list]', + 'ArkosExt.asc': 'ascending order', + 'ArkosExt.desc': 'descending order', + + 'ArkosExt.name': 'name', + 'ArkosExt.rank': 'rank', + 'ArkosExt.rankValue': 'rankValue', + 'ArkosExt.extra': 'extra', + + 'ArkosExt.colorToHex': 'get code of color[COLOR]', + + 'ArkosExt.info1': '🚶 Coordinate and Direction', + 'ArkosExt.info2': '🔠 String Processing', + 'ArkosExt.info3': '🛠 Utilities', + 'ArkosExt.info4': '📄 Information', + 'ArkosExt.info5': '📊 Sorted Table', + 'ArkosExt.info6': '🗂️ Temporary Data', + 'ArkosExt.info7': 'temp var', + 'ArkosExt.info8': 'temp list', + 'ArkosExt.info9': 'temp container', + + 'ArkosExt.deleteAllTempData': '🗂️clear all temporary data', + 'ArkosExt.getCountOfTempData': '🗂️count of temporary data', + 'ArkosExt.delTempData': '🗂️delete temporary data[data]', + + 'ArkosExt.setTempVar': '🗂️set temp var[var] to [t]', + 'ArkosExt.addTempVar': '🗂️change temp var[var] by [t]', + 'ArkosExt.getTempVar': '🗂️temp data[var]', + + + 'ArkosExt.clearTempList': '🗂️create or clear temp list[list]', + 'ArkosExt.initTempList': '🗂️set temp list[list]to[t]', + 'ArkosExt.addTempList': '🗂️add[t] to temp list[list]', + 'ArkosExt.opTempList': '🗂️[op][t]at [n] of temp list[list]', + 'ArkosExt.ListOp1': 'insert', + 'ArkosExt.ListOp2': 'replace with', + 'ArkosExt.ListOp3': 'change by', + 'ArkosExt.delItemOfTempList': '🗂️delete [n]of temp list[list]', + 'ArkosExt.getItemOfTempList': '🗂️item[n]of temp list[list]', + 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', + + 'ArkosExt.clearTempCon': '🗂️create or clear temp container[con]', + 'ArkosExt.initTempCon': '🗂️set temp container[con]to[t]', + 'ArkosExt.opTempCon': '🗂️[c]in temp container[con][op][t]', + 'ArkosExt.conOp1': 'set to', + 'ArkosExt.conOp2': 'change by', + 'ArkosExt.delItemOfTempCon': '🗂️delete[c]in temp container[con]', + 'ArkosExt.getItemOfTempConByName': '🗂️[c]in temp container[con]', + 'ArkosExt.getItemOfTempConByNo': '🗂️[t] of #[n] in temp container[con]', + 'ArkosExt.conInfo1': 'name', + 'ArkosExt.conInfo2': 'content', + 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', + + '30Ext.info': '✨ Contributed by _30', + '30Ext.info.1': '🪞 Mirror transform', + '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', + '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', + '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', + '30Ext.block.clearMirror': 'Clear the mirror transform', + '30Ext.info.2': '🛸 Cross sprite operation', + '30Ext.block.anotherRun': 'Let [spriteName] run', + '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' + }, + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id, + }) + } + + getInfo() { + return { + id: 'hcnTest', // 拓展id + name: this.formatMessage('ArkosExt.extensionName'), + + color1: '#FF8383', + // menuIconURI: icon, + // blockIconURI: icon, + blocks: [ + "---" + this.formatMessage("ArkosExt.info1"), //🏃 坐标&角度 + // 计算点A到点B的方向 + { + opcode: 'getDirFromAToB', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.directionFromAtoB'), + arguments: { + X1: { + type: 'number', + defaultValue: 0, + }, + Y1: { + type: 'number', + defaultValue: 0, + }, + X2: { + type: 'number', + defaultValue: 0, + }, + Y2: { + type: 'number', + defaultValue: 0, + }, + }, + }, + // 计算角b-角a的角度差 + { + opcode: 'differenceBetweenDirections', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.differenceBetweenDirections'), + arguments: { + a: { + type: 'number', + defaultValue: 0, + }, + b: { + type: 'number', + defaultValue: 0, + }, + }, + }, + // 两点距离 + { + opcode: 'disFromAToB', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.distance'), + arguments: { + X1: { + type: 'number', + defaultValue: 0, + }, + Y1: { + type: 'number', + defaultValue: 0, + }, + X2: { + type: 'number', + defaultValue: 0, + }, + Y2: { + type: 'number', + defaultValue: 0, + }, + }, + }, + //朝..方向旋转..角度 + { + opcode: 'turnDegreesToDir', + blockType: 'command', + text: this.formatMessage('ArkosExt.turnDegreesToDir'), + arguments: { + degree: { + type: 'number', + defaultValue: 45, + }, + dir: { + type: 'angle', + defaultValue: 10, + }, + }, + filter: ['sprite'] + }, + "---" + this.formatMessage("ArkosExt.info2"), //🔠字符串处理 + // 查找子字符串,从pos开始 + { + opcode: 'indexof', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.searchString'), + arguments: { + str: { + type: 'string', + defaultValue: 'banana', + }, + substr: { + type: 'string', + defaultValue: 'na', + }, + pos: { + type: 'number', + defaultValue: 1, + }, + }, + }, + // 在字符串中插入子字符串 + { + opcode: 'insertStr', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.insertString'), + arguments: { + str: { + type: 'string', + defaultValue: 'ac', + }, + substr: { + type: 'string', + defaultValue: 'b', + }, + pos: { + type: 'number', + defaultValue: 2, + }, + }, + }, + // 替换字符串中的从..到..的字符串 + { + opcode: 'replaceStr', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.replaceString'), + arguments: { + str: { + type: 'string', + defaultValue: 'ABCDEF', + }, + substr: { + type: 'string', + defaultValue: 'XX', + }, + start: { + type: 'number', + defaultValue: 3, + }, + end: { + type: 'number', + defaultValue: 4, + }, + }, + }, + "---" + this.formatMessage("ArkosExt.info3"), //🔧实用积木 + // 判断相等(区分大小写) + { + opcode: 'strictlyEquals', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.stringEquality'), + arguments: { + ONE: { + type: 'string', + defaultValue: 'A', + }, + TWO: { + type: 'string', + defaultValue: 'a', + }, + }, + }, + //返回值转bool积木 + { + opcode: 'reporterToBoolean', + blockType: 'Boolean', + text: '[t]', + arguments: { + t: { + type: 'string', + defaultValue: '1', + } + }, + }, + //形如 a≤b≤c + { + opcode: 'compareTwoSides', + blockType: 'Boolean', + text: '[a][op1][b][op2][c]', + arguments: { + a: { + type: 'string', + defaultValue: '1', + }, + b: { + type: 'string', + defaultValue: 'x', + }, + c: { + type: 'string', + defaultValue: '3', + }, + op1: { + type: 'string', + menu: 'opMenu1', + }, + op2: { + type: 'string', + menu: 'opMenu1', + }, + }, + }, + //形如:a≤b且/或>c op1,op2 logic compareTwoSidesPlus + { + opcode: 'compareTwoSidesPlus', + blockType: 'Boolean', + text: '[a][op1][b][logic][op2][c]', + arguments: { + a: { + type: 'string', + defaultValue: 'x', + }, + b: { + type: 'string', + defaultValue: '1', + }, + c: { + type: 'string', + defaultValue: '3', + }, + op1: { + type: 'string', + menu: 'opMenu2', + defaultValue: '<', + }, + op2: { + type: 'string', + menu: 'opMenu2', + defaultValue: '>', + }, + logic: { + type: 'string', + menu: 'logicMenu', + defaultValue: 'or', // + }, + }, + }, + //获取颜色HEX + { + opcode: 'colorToHex', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.colorToHex'), + arguments: { + COLOR: { + type: 'color', + //defaultValue: '1', + }, + }, + }, + //强行设置大小 + { + opcode: 'setSize', + blockType: 'command', + text: this.formatMessage('ArkosExt.setSize'), + arguments: { + size: { + type: 'number', + defaultValue: 9999, + }, + }, + filter: ['sprite'] + }, + //强行移到xy + { + opcode: 'setXY', + blockType: 'command', + text: this.formatMessage('ArkosExt.setXY'), + arguments: { + x: { + type: 'number', + defaultValue: 100000, + }, + y: { + type: 'number', + defaultValue: 100000, + }, + }, + filter: ['sprite'] + }, + "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 + //获取特效值 + { + opcode: 'getEffect', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getEffect'), + arguments: { + EFFECT: { + type: 'string', + menu: 'effectMenu', + }, + }, + }, + //是否隐藏 + { + opcode: 'isHiding', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.isHiding'), + filter: ['sprite'] + }, + //获取旋转方式 + { + opcode: 'getRotationStyle', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getRotationStyle'), + filter: ['sprite'] + }, + //获取造型0宽1高 + { + opcode: 'getWidthOrHeight', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getWidthOrHeight'), + arguments: { + t: { + type: 'string', + menu: 'WOrH', + }, + }, + }, + //获取角色边缘xy + { + opcode: 'getBoundaryCoord', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getBoundaryCoord'), + arguments: { + t: { + type: 'string', + menu: 'boundaryMenu', + } + }, + filter: ['sprite'] + }, + //是否跑到舞台外 + { + opcode: 'isOutOfSight', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.isOutOfSight'), + filter: ['sprite'] + }, + "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 + //📊清空排序表 + { + opcode: 'clearSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + }, + }, + { + //📊排序表排序方式 + opcode: 'setTypeOfSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.setTypeOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + type: { + type: 'string', + menu: 'sortOrder', + }, + }, + }, + { + //📊将XX加入排序表 + opcode: 'addToSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.addToSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + value: { + type: 'number', + defaultValue: '95', + }, + extra: { + type: 'string', + defaultValue: '20212490', + }, + }, + }, + { + //📊获取排序表第n项 + opcode: 'getFromSortedTableByNo', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getFromSortedTableByNo'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + defaultValue: '1', + menu: 'tableItemPropertyMenu', + }, + }, + }, + { + //📊获取排序表特定名字内容 + opcode: 'getFromSortedTableByName', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getFromSortedTableByName'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + t: { + type: 'string', + defaultValue: '2', + menu: 'tableItemPropertyMenu', + }, + }, + }, + { + //📊获取排序表长度 + opcode: 'lengthOfSortedTable', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + }, + }, + { + //📊删除表中内容 + opcode: 'deleteNameOfSortedTable', + blockType: 'command', + text: this.formatMessage('ArkosExt.deleteNameOfSortedTable'), + arguments: { + list: { + type: 'string', + menu: 'sortedTableMenu', + }, + name: { + type: 'string', + defaultValue: '小明', + }, + }, + }, + + "---" + this.formatMessage("ArkosExt.info6"), //🗂️ 临时数据 + //清空所有临时数据 + { + opcode: 'deleteAllTempData', + blockType: 'command', + text: this.formatMessage('ArkosExt.deleteAllTempData'), + }, + //临时数据量 + { + opcode: 'getCountOfTempData', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getCountOfTempData'), + }, + //删除临时数据 + { + opcode: 'delTempData', + blockType: 'command', + text: this.formatMessage('ArkosExt.delTempData'), + arguments: { + data: { + type: 'string', + defaultValue: 'i', + }, + }, + }, + "---" + this.formatMessage("ArkosExt.info7"), //临时变量 + //设置临时数据 + { + opcode: 'setTempVar', + blockType: 'command', + text: this.formatMessage('ArkosExt.setTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + t: { + type: 'string', + defaultValue: '0', + }, + }, + }, + //增加临时数据 + { + opcode: 'addTempVar', + blockType: 'command', + text: this.formatMessage('ArkosExt.addTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + t: { + type: 'number', + defaultValue: 1, + }, + }, + }, + //获取临时数据 + { + opcode: 'getTempVar', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getTempVar'), + arguments: { + var: { + type: 'string', + defaultValue: 'i', + }, + }, + }, + "---" + this.formatMessage("ArkosExt.info8"), //临时列表 + //创建或清空临时列表 + { + opcode: 'clearTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + }, + }, + //设置临时列表 + { + opcode: 'initTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.initTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + t: { + type: 'string', + defaultValue: '[1,2,"apple"]', + }, + }, + }, + //向临时列表加入 + { + opcode: 'addTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.addTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + t: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, + //操作临时列表 + { + opcode: 'opTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.opTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + op: { + type: 'string', + menu: 'ListOpMenu', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, + //删除临时列表XX项 + { + opcode: 'delItemOfTempList', + blockType: 'command', + text: this.formatMessage('ArkosExt.delItemOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + n: { + type: 'number', + defaultValue: 1, + }, + }, + }, + //获取临时列表XX项 + { + opcode: 'getItemOfTempList', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + n: { + type: 'number', + defaultValue: 1, + }, + }, + }, + //临时列表长度 + { + opcode: 'lengthOfTempList', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfTempList'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + }, + }, + "---" + this.formatMessage("ArkosExt.info9"), //临时容器 + //创建或清空临时容器 + { + opcode: 'clearTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.clearTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + }, + }, + //设置临时容器 + { + opcode: 'initTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.initTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + t: { + type: 'string', + defaultValue: '{"coins":200,"backpack":["wood","bread"]}', + }, + }, + }, + //操作临时容器 + { + opcode: 'opTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.opTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + op: { + type: 'string', + menu: 'conOpMenu', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + t: { + type: 'string', + defaultValue: '520', + }, + }, + }, + //删除临时容器名为xx的内容 + { + opcode: 'delItemOfTempCon', + blockType: 'command', + text: this.formatMessage('ArkosExt.delItemOfTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + }, + }, + //获取临时容器名为XX的内容 + { + opcode: 'getItemOfTempConByName', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempConByName'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + c: { + type: 'string', + defaultValue: 'coins', + }, + }, + }, + //获取临时容器第n项的xx + { + opcode: 'getItemOfTempConByNo', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getItemOfTempConByNo'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + n: { + type: 'number', + defaultValue: 1, + }, + t: { + type: 'string', + menu: 'conInfoMenu', + defaultValue: '2', + }, + }, + }, + //临时容器长度 + { + opcode: 'lengthOfTempCon', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lengthOfTempCon'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + }, + }, + + + // + "---" + this.formatMessage("30Ext.info"), //感谢30提供的扩展 + "---" + this.formatMessage("30Ext.info.1"), //造型镜像 + // 镜像造型 + { + opcode: 'mirrorSprite', + blockType: 'command', + text: this.formatMessage('30Ext.block.mirrorSprite'), + arguments: { + mirrorMethod: { + type: 'string', + menu: 'mirrorMenu' + } + } + }, + // 清除镜像 + { + opcode: 'clearMirror', + blockType: 'command', + text: this.formatMessage('30Ext.block.clearMirror') + }, + "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 + { + opcode: 'anotherRun', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRun'), + arguments: { + spriteName: { + type: 'string', + menu: 'spritesMenu' + }, + SUBSTACK: { //TODO + type: "input_statement" + } + } + }, + { + opcode: 'anotherRunWithClone', + blockType: 'conditional', + text: this.formatMessage('30Ext.block.anotherRunWithClone'), + arguments: { + spriteName: { + type: 'string', + menu: 'spritesMenu' + }, + cloneId: { + type: 'number', + defaultValue: 1 + }, + SUBSTACK: { + type: "input_statement" + } + + } + }, + + ], + menus: { + conInfoMenu: [{ + text: this.formatMessage('ArkosExt.conInfo1'), //名称 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.conInfo2'), //内容 + value: '2' + }, + ], + conOpMenu: [{ + text: this.formatMessage('ArkosExt.conOp1'), //设为 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.conOp2'), //增加 + value: '2' + }, + ], + ListOpMenu: [{ + text: this.formatMessage('ArkosExt.ListOp1'), //插入 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.ListOp2'), //替换 + value: '2' + }, + { + text: this.formatMessage('ArkosExt.ListOp3'), //增加 + value: '3' + }, + ], + tableItemPropertyMenu: [{ + text: this.formatMessage('ArkosExt.name'), + value: '1' + }, + { + text: this.formatMessage('ArkosExt.rank'), + value: '2' + }, + { + text: this.formatMessage('ArkosExt.rankValue'), + value: '3' + }, + { + text: this.formatMessage('ArkosExt.extra'), + value: '4' + }, + ], + sortOrder: [{ + text: this.formatMessage('ArkosExt.asc'), + value: 'asc' //升序 + }, + { + text: this.formatMessage('ArkosExt.desc'), + value: 'desc' //降序 + }, + ], + sortedTableMenu: { + items: 'findAllSortedTable', + acceptReporters: true, + }, + //判断符菜单 + opMenu1: ['<', '≤', '=', '≠', ], + opMenu2: ['<', '>', '≤', '≥', '=', '≠', ], + //logicMenu + logicMenu: [{ + text: this.formatMessage('ArkosExt.or'), + value: 'or' + }, + { + text: this.formatMessage('ArkosExt.and'), + value: 'and' + }, + ], + //角色上下左右边缘 + boundaryMenu: [{ + text: this.formatMessage('ArkosExt.top'), + value: '1' + }, + { + text: this.formatMessage('ArkosExt.bottom'), + value: '2' + }, + { + text: this.formatMessage('ArkosExt.left'), + value: '3' + }, + { + text: this.formatMessage('ArkosExt.right'), + value: '4' + }, + ], + //0宽1高 菜单 + WOrH: [{ + text: this.formatMessage('ArkosExt.width'), + value: '0' + }, + { + text: this.formatMessage('ArkosExt.height'), + value: '1' + }, + ], + //特效菜单 + effectMenu: [{ + text: this.formatMessage('ArkosExt.color'), + value: 'color' + }, + { + text: this.formatMessage('ArkosExt.fisheye'), + value: 'fisheye' + }, + { + text: this.formatMessage('ArkosExt.whirl'), + value: 'whirl' + }, + { + text: this.formatMessage('ArkosExt.pixelate'), + value: 'pixelate' + }, + { + text: this.formatMessage('ArkosExt.mosaic'), + value: 'mosaic' + }, + { + text: this.formatMessage('ArkosExt.brightness'), + value: 'brightness' + }, + { + text: this.formatMessage('ArkosExt.ghost'), + value: 'ghost' + } + ], + //30Ext + spritesMenu: { + items: 'getSpritesMenu' + }, + mirrorMenu: [{ + text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 + value: '0' + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 + value: '1' + } + ] + }, + } + } + + strictlyEquals(args) { + // 实际上在这里直接使用严格相等是不太明智的,因为有一定的可能会遇到数字和字符比较, + // 而在Scratch中数字和字符在表现完全一样的时候几乎没有区别。 + // 因此包上Cast.toString()以使得数字和字符能够正常比较(类似 9 = "9" ) + return Cast.toString(args.ONE) === Cast.toString(args.TWO) + } + + getDirFromAToB(args) { + // 一定要先转化为数字; + const X1 = Cast.toNumber(args.X1) + const X2 = Cast.toNumber(args.X2) + const Y1 = Cast.toNumber(args.Y1) + const Y2 = Cast.toNumber(args.Y2) + + // 这里利用atan函数的性质atan(+inf)=90,atan(-inf)=-90,atan(NaN)=NaN可以省很多代码 + let a = Math.atan((X2 - X1) / (Y2 - Y1)) / Math.PI * 180 + (Y1 > Y2 ? 180 : 0) + if(a > 180) a -= 360 + return a; + } + + differenceBetweenDirections(args) { + const a = Cast.toNumber(args.a) + const b = Cast.toNumber(args.b) + let dif = b - a + dif -= Math.round(dif / 360) * 360 + if(dif === -180) dif = 180 + return dif + } + + disFromAToB(args) { + const X1 = Cast.toNumber(args.X1) + const X2 = Cast.toNumber(args.X2) + const Y1 = Cast.toNumber(args.Y1) + const Y2 = Cast.toNumber(args.Y2) + return Math.sqrt((X1 - X2) * (X1 - X2) + (Y1 - Y2) * (Y1 - Y2)) + } + + indexof(args) { + const str = Cast.toString(args.str) + const substr = Cast.toString(args.substr) + const a = str.indexOf(substr, Cast.toNumber(args.pos) - 1) + if(a === -1) { + // Scratch列表中也有查询积木,其中找不到返回的是0。建议维持原有的风格。 + return 0 + } + return a + 1 + } + + insertStr(args) { + const str = Cast.toString(args.str) + const substr = Cast.toString(args.substr) + let pos = Cast.toNumber(args.pos) - 1 + if(pos < 0) { + pos = 0 + } + return str.slice(0, pos) + substr + str.slice(pos) + } + + replaceStr(args) { + const str = Cast.toString(args.str) + const substr = Cast.toString(args.substr) + let start = Cast.toNumber(args.start) + let end = Cast.toNumber(args.end) + if(start > end) { + const t = end + end = start + start = t + } + if(start < 1) start = 1 + return str.slice(0, start - 1) + substr + str.slice(end) + } + + + turnDegreesToDir(args, util) { + const degree = Cast.toNumber(args.degree); + const dir = Cast.toNumber(args.dir); + const dif = this.differenceBetweenDirections({ + a: util.target.direction, + b: dir + }); + if(Math.abs(dif) < degree) + util.target.setDirection(dir); + else if(dif < 0) + util.target.setDirection(util.target.direction - degree); + else + util.target.setDirection(util.target.direction + degree); + } + + //获取特效的数值 + getEffect(args, util) { + let effect = Cast.toString(args.EFFECT) + .toLowerCase(); + if(!util.target.effects.hasOwnProperty(effect)) return 0; + return util.target.effects[effect]; + } + + //角色是否可见 + isHiding(args, util) { + return !util.target.visible; + } + + //获取图层(逝一逝) + // getLayer (args, util) { + // return util.target.layer; + // } + + //获取当前角色的旋转方式 + getRotationStyle(args, util) { + return util.target.rotationStyle; + } + + //获取当前造型的长/宽 + getWidthOrHeight(args, util) { + const costumeSize = util.target.renderer.getCurrentSkinSize(util.target.drawableID); + return costumeSize[args.t]; + } + + //强行设置大小(逝一逝) + setSize(args, util) { + if(util.target.isStage) { + return; + } + if(util.target.renderer) { + args.size = this._clamp(Cast.toNumber(args.size), 0.1, 100000000) + util.target.size = args.size; + const { + direction, + scale + } = util.target._getRenderedDirectionAndScale(); + util.target.renderer.updateDrawableDirectionScale(util.target.drawableID, direction, scale); + if(util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } + } + util.target.runtime.requestTargetsUpdate(util.target); + } + + _clamp(n, min, max) { + return Math.min(Math.max(n, min), max); + } + + //强行设置XY(逝一逝) + setXY(args, util) { + if(util.target.isStage) return; + args.x = this._clamp(Cast.toNumber(args.x), -100000000, 100000000) + args.y = this._clamp(Cast.toNumber(args.y), -100000000, 100000000) + const oldX = util.target.x; + const oldY = util.target.y; + util.target.x = args.x; + util.target.y = args.y; + if(util.target.renderer) { + util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); + if(util.target.visible) { + util.target.emit('EVENT_TARGET_VISUAL_CHANGE', util.target); + util.target.runtime.requestRedraw(); + } + } else { + util.target.x = x; + util.target.y = y; + } + util.target.emit('TARGET_MOVED', util.target, oldX, oldY, false); + util.target.runtime.requestTargetsUpdate(util.target); + } + + //获取角色边缘的坐标 + getBoundaryCoord(args, util) { + const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); + switch (args.t) { + case '1': + return bounds.top; + case '2': + return bounds.bottom; + case '3': + return bounds.left; + case '4': + return bounds.right; + default: + return ''; + } + } + + //是否在舞台外 + isOutOfSight(args, util) { + // console.log(util.target.runtime.renderer) + // console.log(util.target.renderer) + if(util.target.renderer) { + const stageWidth = util.target.runtime.stageWidth; + const stageHeight = util.target.runtime.stageHeight; + console.log('stageWidth', stageWidth); + console.log('stageHeight', stageHeight); + const bounds = util.target.runtime.renderer.getBounds(util.target.drawableID); + if(bounds.right < -stageWidth / 2 || + bounds.left > stageWidth / 2 || + bounds.bottom > stageHeight / 2 || + bounds.top < -stageHeight / 2) { + return true; + } + } + return false; + } + + //形如:<() > + reporterToBoolean(args) { + if(Cast.toString(args.t) + .toLowerCase() === 'false') return false; + if(args.t === '0') return false; + return (args.t) ? true : false; + } + + compare(a, b, op) { + switch (op) { + case '<': + return Cast.compare(a, b) < 0; + case '>': + return Cast.compare(a, b) > 0; + case '=': + return Cast.compare(a, b) === 0; + case '≤': + return Cast.compare(a, b) <= 0; + case '≥': + return Cast.compare(a, b) >= 0; + case '≠': + return Cast.compare(a, b) !== 0; + default: + return false; + } + } + + //形如:a≤b≤c op1,op2 + compareTwoSides(args) { + return this.compare(args.a, args.b, args.op1) && this.compare(args.b, args.c, args.op2) + } + + //形如:a≤b且/或>c op1,op2 logic + compareTwoSidesPlus(args) { + switch (args.logic) { + case 'or': + return this.compare(args.a, args.b, args.op1) || this.compare(args.a, args.c, args.op2) + case 'and': + return this.compare(args.a, args.b, args.op1) && this.compare(args.a, args.c, args.op2) + default: + return false; + } + } + + //数组排序规则 + sortRule(propName, order) { + return (a, b) => { + a = a[propName] + b = b[propName] + if(a > b) return order === 'asc' ? 1 : -1; + else if(a < b) return order === 'asc' ? -1 : 1; + else return 0; + } + } + + //查找所有排序表 + findAllSortedTable() { + const list = []; + let temp = this.sortedTable; + Object.keys(temp) + .forEach(obj => { + //if ( Array.isArray (temp[obj]) ) { + list.push(obj); + //} + }); + if(list.length === 0) { + list.push({ + text: '-', + value: 'empty', + }); + } + //list.sort(this.sortRule("text")); + return list; + } + + createTableIfNotExist(list) { + if(!(list in this.sortedTable)) + this.sortedTable[list] = { + order: 'desc', + list: [] + }; + } + + sortTable(list) { + this.sortedTable[list].list.sort(this.sortRule("rankValue", this.sortedTable[list].order)); + } + + //📊清空排序表 + clearSortedTable(args) { + this.createTableIfNotExist(args.list) + this.sortedTable[args.list].list = []; + } + + //📊设置排序方式 + setTypeOfSortedTable(args) { + this.createTableIfNotExist(args.list) + this.sortedTable[args.list].order = args.type; + this.sortTable(args.list) + } + + //查找在列表中的插入位置(已有则覆盖) + _findPlaceAndInsert(list, order, item) { + //删除已存在的内容 + for(let i = 0; i < list.length; i++) { + if(list[i].name === item.name) { + //删除同名项 + list.splice(i, 1); + break; + } + } + //查找插入位置并插入 + for(let i = 0; i < list.length; i++) { + if((list[i].rankValue > item.rankValue && order === 'asc') || + (list[i].rankValue < item.rankValue && order === 'desc')) { + //插入在该项前 + list.splice(i, 0, item); + return; + } + } + //没找到插入位置,加在末尾 + list.push(item); + } + + //📊将内容加入表 + addToSortedTable(args) { + this.createTableIfNotExist(args.list) + this._findPlaceAndInsert( + this.sortedTable[args.list].list, + this.sortedTable[args.list].order, { + name: args.name, + rankValue: args.value, + extra: args.extra + }); + } + + _getTInItem(item, t, rank) { + if(item === undefined) return ''; + switch (t) { + case '1': + return item.name; + case '2': + return rank; + case '3': + return item.rankValue; + case '4': + return item.extra; + default: + return ''; + } + } + + //📊获取第n项 + getFromSortedTableByNo(args) { + if(!(args.list in this.sortedTable)) return ''; + let list = this.sortedTable[args.list].list; + return this._getTInItem(list[args.n - 1], args.t, Cast.toNumber(args.n)); + } + + _getItemIdxByName(list, name) { + for(let i = 0; i < list.length; i++) { + if(list[i].name === name) { + return i; + } + } + return -1; + } + + //📊获取名为XX的项 + getFromSortedTableByName(args) { + if(!(args.list in this.sortedTable)) return ''; + let list = this.sortedTable[args.list].list; + let n = this._getItemIdxByName(list, args.name); + if(n === -1) return ''; + return this._getTInItem(list[n], args.t, n + 1); + } + + //📊获取排序表长度 + lengthOfSortedTable(args) { + if(!(args.list in this.sortedTable)) return 0; + return this.sortedTable[args.list].list.length; + } + + //📊删除排序表名为XX的内容 + deleteNameOfSortedTable(args) { + if(!(args.list in this.sortedTable)) return; + let list = this.sortedTable[args.list].list; + let n = this._getItemIdxByName(list, args.name); + if(n === -1) return; + list.splice(n, 1); + } + + //获取颜色HEX码 + colorToHex(args, util) { + let c = Cast.toRgbColorList(args.COLOR) + return Color.rgbToHex({ + r: c[0], + g: c[1], + b: c[2] + }); + } + + + //🗂️ 临时变量积木 + + //来自 -6 :任意内容转字符或数字 + _anythingToNumberString(value) { + switch (typeof(value)) { + case "string": + case "number": + break; + case "object": + value = JSON.stringify(value); + break; + default: + value = ''; //包含了undefined + } + return value; + } + + deleteAllTempData(args) { + this.tempData = {}; + } + + getCountOfTempData(args) { + return Object.keys(this.tempData) + .length; + } + + delTempData(args) { + delete this.tempData[Cast.toString(args.data)]; + } + + setTempVar(args) { + this.tempData[Cast.toString(args.var)] = args.t; + } + + addTempVar(args) { + this.tempData[Cast.toString(args.var)] = Cast.toNumber(this.tempData[Cast.toString(args.var)]) + Cast.toNumber(args.t); + } + + getTempVar(args) { + let temp = this.tempData[Cast.toString(args.var)] + if(typeof(temp) === 'object') return JSON.stringify(temp); + return Cast.toString(temp); + } + + //创建或清空 + clearTempList(args) { + this.tempData[Cast.toString(args.list)] = []; + } + + initTempList(args) { + try { + let content = JSON.parse(Cast.toString(args.t)) + if(Array.isArray(content)) { + this.tempData[Cast.toString(args.list)] = content; + } + } catch (e) { + + } + } + + addTempList(args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + list.push(Cast.toString(args.t)); + } + + opTempList(args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + let n = Cast.toNumber(args.n) + if(n < 1 || n > list.length + 1) return; + n -= 1; + switch (args.op) { + case '1': //插入 + list.splice(n, 0, args.t); + return; + case '2': //替换 + list[n] = args.t; + return; + case '3': //增加 + list[n] = Cast.toNumber(list[n]) + Cast.toNumber(args.t); + return; + default: + return; + } + } + + delItemOfTempList(args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return; + let n = Cast.toNumber(args.n) + if(n < 1 || n > list.length) return; + n -= 1; + list.splice(n, 1); + } + + getItemOfTempList(args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return ''; + let n = Cast.toNumber(args.n) + if(n < 1 || n > list.length) return ''; + n -= 1; + return Cast.toString(list[n]); + } + + lengthOfTempList(args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return 0; + return list.length; + } + + //容器 + clearTempCon(args) { + this.tempData[Cast.toString(args.con)] = {}; + } + + initTempCon(args) { + try { + let content = JSON.parse(Cast.toString(args.t)) + if(typeof(content) === 'object' && content !== null) { + this.tempData[Cast.toString(args.con)] = content; + } + } catch (e) { + + } + } + + opTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if(!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return; + let c = Cast.toString(args.c) + switch (args.op) { + case '1': //设为 + con[c] = args.t; + return; + case '2': //增加 + if(!(c in con)) return; + con[c] = Cast.toNumber(con[c]) + Cast.toNumber(args.t); + return; + default: + return; + } + } + + delItemOfTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if(!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return; + delete con[Cast.toString(args.c)]; + } + + getItemOfTempConByName(args) { + let con = this.tempData[Cast.toString(args.con)] + if(!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return ''; + return Cast.toString(con[Cast.toString(args.c)]); + } + + getItemOfTempConByNo(args) { + let con = this.tempData[Cast.toString(args.con)] + if(!(typeof(con) === 'object' && !Array.isArray(con) && con !== null)) return ''; + let key = Object.keys(con)[Cast.toNumber(args.n) - 1] + if(key === undefined) return ''; + switch (args.t) { + case '1': //名称 + return key; + case '2': //内容 + return this._anythingToNumberString(con[key]); + default: + return; + } + } + + lengthOfTempCon(args) { + let con = this.tempData[Cast.toString(args.con)] + if(!(typeof(con) === 'object' && con !== null)) return 0; + return Object.keys(con) + .length; + } + + + // + //30Ext + // + //菜单 + //动态菜单: 角色菜单 + getSpritesMenu() { + var sprites = []; + for(const targetId in this.runtime.targets) { + if(!this.runtime.targets.hasOwnProperty(targetId)) continue; + if(!this.runtime.targets[targetId].isOriginal) continue; + if(this.runtime.targets[targetId] === this.runtime._editingTarget) continue; //排除自己 + let name = this.runtime.targets[targetId].sprite.name; + sprites.push(name); //['Stage','角色1','角色2'] Stage暂时懒得换成中文 + } + return sprites; + } + // + //角色造型操作 + // + //镜像造型 + mirrorSprite(args, util) { + //OK + let target = util.target; + if(!.ext30_isHook) { + target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e, t) => { + let drawable = this.runtime.renderer._allDrawables[target.drawableID]; + if(target.ext30_mirror0) drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * target.ext30_mirror0; + if(target.ext30_mirror1) drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * target.ext30_mirror1; + }); + target.ext30_mirror0 = 1; + target.ext30_mirror1 = 1; + target.ext30_isHook = true; + } + target['ext30_mirror' + args.mirrorMethod] *= -1; + if (target.visible) { + target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); + target.runtime.requestRedraw(); + } + target.runtime.requestTargetsUpdate(util.target); + } + //清除镜像 + clearMirror(args, util) { + let target = util.target; + target.ext30_mirror0 = 1; + target.ext30_mirror1 = 1; + target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); + target.runtime.requestRedraw(); + if (target.visible) { + target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); + target.runtime.requestRedraw(); + } + target.runtime.requestTargetsUpdate(util.target); + } + // + //角色跨域操作 + // + //跨域执行 + anotherRun(args, util) { + if(!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName) + .sprite.clones[0]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } + //跨域克隆体执行 + anotherRunWithClone(args, util) { + if(!util.thread.ex_30Ext_count) { + util.thread.ex_30Ext_count = true; + util.thread.ex_30Ext_oldTarget = util.thread.target; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName) + .sprite.clones[args.cloneId]; + util.startBranch(1, true); + } else { + util.thread.target = util.thread.ex_30Ext_oldTarget; + util.thread.ex_30Ext_count = false; + } + } } window.tempExt = { - Extension: ArkosExtensions, - info: { - name: 'hcn.extensionName', - description: 'hcn.description', - extensionId: 'hcnTest', - // iconURL: icon, - // insetIconURL: cover, - featured: true, - disabled: false, - collaborator: 'only for hcn test', - }, - l10n: { - 'zh-cn': { - 'hcn.extensionName': 'hcn 的测试', - 'hcn.description': 'hcn 的测试', - }, - en: { - 'hcn.extensionName': 'hcn test', - 'hcn.description': 'hcn test', - }, - }, + Extension: ArkosExtensions, + info: { + name: 'hcn.extensionName', + description: 'hcn.description', + extensionId: 'hcnTest', + // iconURL: icon, + // insetIconURL: cover, + featured: true, + disabled: false, + collaborator: 'only for hcn test', + }, + l10n: { + 'zh-cn': { + 'hcn.extensionName': 'hcn 的测试', + 'hcn.description': 'hcn 的测试', + }, + en: { + 'hcn.extensionName': 'hcn test', + 'hcn.description': 'hcn test', + }, + }, } From 94b1ec455fbc9e2a926f449c44ae984f32513af8 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Sun, 31 Jul 2022 22:57:04 +0800 Subject: [PATCH 098/357] Update projectWith30.js --- Arkos/projectWith30.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index c318c98d..8bd0cb91 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1832,7 +1832,7 @@ class ArkosExtensions { mirrorSprite(args, util) { //OK let target = util.target; - if(!.ext30_isHook) { + if(!target.ext30_isHook) { target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e, t) => { let drawable = this.runtime.renderer._allDrawables[target.drawableID]; if(target.ext30_mirror0) drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * target.ext30_mirror0; From 85e71920d68e8f45868746e8cc7b3e30ed187706 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Sun, 31 Jul 2022 23:18:51 +0800 Subject: [PATCH 099/357] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E6=9B=B4=E6=96=B0?= =?UTF-8?q?=E6=B8=B2=E6=9F=93=E5=99=A8=E7=9A=84=E6=96=B9=E6=B3=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 21 ++++++++++----------- 1 file changed, 10 insertions(+), 11 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 8bd0cb91..61438eb8 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1832,9 +1832,9 @@ class ArkosExtensions { mirrorSprite(args, util) { //OK let target = util.target; + let drawable = this.runtime.renderer._allDrawables[target.drawableID]; if(!target.ext30_isHook) { target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e, t) => { - let drawable = this.runtime.renderer._allDrawables[target.drawableID]; if(target.ext30_mirror0) drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * target.ext30_mirror0; if(target.ext30_mirror1) drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * target.ext30_mirror1; }); @@ -1843,11 +1843,10 @@ class ArkosExtensions { target.ext30_isHook = true; } target['ext30_mirror' + args.mirrorMethod] *= -1; - if (target.visible) { - target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); - target.runtime.requestRedraw(); - } - target.runtime.requestTargetsUpdate(util.target); + //更新渲染器 + drawable._renderer.dirty = true; + drawable._rotationTransformDirty = true; + drawable.setTransformDirty(); } //清除镜像 clearMirror(args, util) { @@ -1856,11 +1855,11 @@ class ArkosExtensions { target.ext30_mirror1 = 1; target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); target.runtime.requestRedraw(); - if (target.visible) { - target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); - target.runtime.requestRedraw(); - } - target.runtime.requestTargetsUpdate(util.target); + //更新渲染器 + let drawable = this.runtime.renderer._allDrawables[target.drawableID]; + drawable._renderer.dirty = true; + drawable._rotationTransformDirty = true; + drawable.setTransformDirty(); } // //角色跨域操作 From 58e4dd8a3161d543f859c86089e188f76f413ee0 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Sun, 31 Jul 2022 23:43:14 +0800 Subject: [PATCH 100/357] =?UTF-8?q?=E9=87=8D=E5=86=99=E8=B7=A8=E5=9F=9F?= =?UTF-8?q?=E6=89=A7=E8=A1=8C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 74 ++++++++++++------------------------------ 1 file changed, 21 insertions(+), 53 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 61438eb8..c5909032 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1041,23 +1041,22 @@ class ArkosExtensions { text: this.formatMessage('30Ext.block.clearMirror') }, "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 + //跨域执行 { opcode: 'anotherRun', - blockType: 'conditional', + blockType: 'command', text: this.formatMessage('30Ext.block.anotherRun'), arguments: { spriteName: { type: 'string', menu: 'spritesMenu' - }, - SUBSTACK: { //TODO - type: "input_statement" } } }, + //跨域执行(克隆体) { opcode: 'anotherRunWithClone', - blockType: 'conditional', + blockType: 'command', text: this.formatMessage('30Ext.block.anotherRunWithClone'), arguments: { spriteName: { @@ -1067,9 +1066,6 @@ class ArkosExtensions { cloneId: { type: 'number', defaultValue: 1 - }, - SUBSTACK: { - type: "input_statement" } } @@ -1831,64 +1827,36 @@ class ArkosExtensions { //镜像造型 mirrorSprite(args, util) { //OK - let target = util.target; + let targetSize = util.target._size; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; - if(!target.ext30_isHook) { - target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e, t) => { - if(target.ext30_mirror0) drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * target.ext30_mirror0; - if(target.ext30_mirror1) drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * target.ext30_mirror1; - }); - target.ext30_mirror0 = 1; - target.ext30_mirror1 = 1; - target.ext30_isHook = true; - } - target['ext30_mirror' + args.mirrorMethod] *= -1; - //更新渲染器 - drawable._renderer.dirty = true; - drawable._rotationTransformDirty = true; - drawable.setTransformDirty(); + drawable['ext30_mirror' + args.mirrorMethod] *= -1; + drawable.updateScale([ + (drawable.ext30_mirror0?drawable.ext30_mirror0:1) * targetSize, + (drawable.ext30_mirror1?drawable.ext30_mirror1:1) * targetSize + ]); } //清除镜像 clearMirror(args, util) { - let target = util.target; - target.ext30_mirror0 = 1; - target.ext30_mirror1 = 1; - target.emitFast('EVENT_TARGET_VISUAL_CHANGE', util.target); - target.runtime.requestRedraw(); - //更新渲染器 + let targetSize = util.target._size; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; - drawable._renderer.dirty = true; - drawable._rotationTransformDirty = true; - drawable.setTransformDirty(); - } + drawable.ext30_mirror0 = 1; + drawable.ext30_mirror1 = 1; + drawable.updateScale([ + (drawable.ext30_mirror0?drawable.ext30_mirror0:1) * targetSize, + (drawable.ext30_mirror1?drawable.ext30_mirror1:1) * targetSize + ]); + } + //TODO: 拉伸 // //角色跨域操作 // //跨域执行 anotherRun(args, util) { - if(!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName) - .sprite.clones[0]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; - } + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName).sprite.clones[0]; } //跨域克隆体执行 anotherRunWithClone(args, util) { - if(!util.thread.ex_30Ext_count) { - util.thread.ex_30Ext_count = true; - util.thread.ex_30Ext_oldTarget = util.thread.target; - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName) - .sprite.clones[args.cloneId]; - util.startBranch(1, true); - } else { - util.thread.target = util.thread.ex_30Ext_oldTarget; - util.thread.ex_30Ext_count = false; - } + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName).sprite.clones[args.cloneId]; } } From ae553696be952b55c480e89ab55b3af8437565ad Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Sun, 31 Jul 2022 23:51:41 +0800 Subject: [PATCH 101/357] Update projectWith30.js --- Arkos/projectWith30.js | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index c5909032..9067e266 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1827,23 +1827,23 @@ class ArkosExtensions { //镜像造型 mirrorSprite(args, util) { //OK - let targetSize = util.target._size; + let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; drawable['ext30_mirror' + args.mirrorMethod] *= -1; drawable.updateScale([ - (drawable.ext30_mirror0?drawable.ext30_mirror0:1) * targetSize, - (drawable.ext30_mirror1?drawable.ext30_mirror1:1) * targetSize + (drawable.ext30_mirror0?drawable.ext30_mirror0:1) * target._size, + (drawable.ext30_mirror1?drawable.ext30_mirror1:1) * target._size ]); } //清除镜像 clearMirror(args, util) { - let targetSize = util.target._size; + let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; drawable.ext30_mirror0 = 1; drawable.ext30_mirror1 = 1; drawable.updateScale([ - (drawable.ext30_mirror0?drawable.ext30_mirror0:1) * targetSize, - (drawable.ext30_mirror1?drawable.ext30_mirror1:1) * targetSize + (drawable.ext30_mirror0?drawable.ext30_mirror0:1) * target._size, + (drawable.ext30_mirror1?drawable.ext30_mirror1:1) * target._size ]); } //TODO: 拉伸 From 67752bdb0d65c0cf58aa1fb7ee6acc624106b010 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Mon, 1 Aug 2022 00:01:47 +0800 Subject: [PATCH 102/357] Update projectWith30.js --- Arkos/projectWith30.js | 42 +++++++++++++++++++++++++++++++----------- 1 file changed, 31 insertions(+), 11 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 9067e266..674ae6d0 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1826,14 +1826,29 @@ class ArkosExtensions { // //镜像造型 mirrorSprite(args, util) { - //OK let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; drawable['ext30_mirror' + args.mirrorMethod] *= -1; - drawable.updateScale([ - (drawable.ext30_mirror0?drawable.ext30_mirror0:1) * target._size, - (drawable.ext30_mirror1?drawable.ext30_mirror1:1) * target._size - ]); + if(!drawable.ext30_hook) { + drawable.__proto__.updateScale = function(scale) { + if(drawable._scale[0] !== scale[0] || drawable._scale[1] !== scale[1]) { + drawable._scale[0] = scale[0] * (drawable.ext30_mirror0 ? drawable.ext30_mirror0 : 1); + drawable._scale[1] = scale[1] * (drawable.ext30_mirror1 ? drawable.ext30_mirror1 : 1); + drawable._renderer.dirty = true; + drawable._rotationCenterDirty = true; + drawable._skinScaleDirty = true; + drawable.setTransformDirty(); + } + } + drawable.ext30_hook = true; + }; + //更新 + drawable._scale[0] = target._size * (drawable.ext30_mirror0 ? drawable.ext30_mirror0 : 1); + drawable._scale[1] = target._size * (drawable.ext30_mirror1 ? drawable.ext30_mirror1 : 1); + drawable._renderer.dirty = true; + drawable._rotationCenterDirty = true; + drawable._skinScaleDirty = true; + drawable.setTransformDirty(); } //清除镜像 clearMirror(args, util) { @@ -1841,10 +1856,13 @@ class ArkosExtensions { let drawable = this.runtime.renderer._allDrawables[target.drawableID]; drawable.ext30_mirror0 = 1; drawable.ext30_mirror1 = 1; - drawable.updateScale([ - (drawable.ext30_mirror0?drawable.ext30_mirror0:1) * target._size, - (drawable.ext30_mirror1?drawable.ext30_mirror1:1) * target._size - ]); + //更新 + drawable._scale[0] = target._size * (drawable.ext30_mirror0 ? drawable.ext30_mirror0 : 1); + drawable._scale[1] = target._size * (drawable.ext30_mirror1 ? drawable.ext30_mirror1 : 1); + drawable._renderer.dirty = true; + drawable._rotationCenterDirty = true; + drawable._skinScaleDirty = true; + drawable.setTransformDirty(); } //TODO: 拉伸 // @@ -1852,11 +1870,13 @@ class ArkosExtensions { // //跨域执行 anotherRun(args, util) { - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName).sprite.clones[0]; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName) + .sprite.clones[0]; } //跨域克隆体执行 anotherRunWithClone(args, util) { - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName).sprite.clones[args.cloneId]; + util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName) + .sprite.clones[args.cloneId]; } } From 4c69301fb0eced8994c462b646c2565328826f12 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Mon, 1 Aug 2022 00:16:36 +0800 Subject: [PATCH 103/357] Update projectWith30.js --- Arkos/projectWith30.js | 44 +++++++++++++++++++----------------------- 1 file changed, 20 insertions(+), 24 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 674ae6d0..aa99cbc2 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1828,27 +1828,23 @@ class ArkosExtensions { mirrorSprite(args, util) { let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; - drawable['ext30_mirror' + args.mirrorMethod] *= -1; - if(!drawable.ext30_hook) { - drawable.__proto__.updateScale = function(scale) { - if(drawable._scale[0] !== scale[0] || drawable._scale[1] !== scale[1]) { - drawable._scale[0] = scale[0] * (drawable.ext30_mirror0 ? drawable.ext30_mirror0 : 1); - drawable._scale[1] = scale[1] * (drawable.ext30_mirror1 ? drawable.ext30_mirror1 : 1); - drawable._renderer.dirty = true; - drawable._rotationCenterDirty = true; - drawable._skinScaleDirty = true; - drawable.setTransformDirty(); - } - } - drawable.ext30_hook = true; - }; + if(!util.target.ext30_isHook) { + target.ext30_mirror0 = 1; + target.ext30_mirror1 = 1; + target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e, t) => { + drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * target.ext30_mirror0; + drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * target.ext30_mirror1; + }); + target.ext30_isHook = true; + } + util.target['ext30_mirror' + args.mirrorMethod] *= -1; //更新 - drawable._scale[0] = target._size * (drawable.ext30_mirror0 ? drawable.ext30_mirror0 : 1); - drawable._scale[1] = target._size * (drawable.ext30_mirror1 ? drawable.ext30_mirror1 : 1); + target.emitFast(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, this); drawable._renderer.dirty = true; - drawable._rotationCenterDirty = true; - drawable._skinScaleDirty = true; - drawable.setTransformDirty(); + drawable._rotationCenterDirty = true; + drawable._skinScaleDirty = true; + drawable.setTransformDirty(); + this.runtime.requestRedraw(); } //清除镜像 clearMirror(args, util) { @@ -1857,12 +1853,12 @@ class ArkosExtensions { drawable.ext30_mirror0 = 1; drawable.ext30_mirror1 = 1; //更新 - drawable._scale[0] = target._size * (drawable.ext30_mirror0 ? drawable.ext30_mirror0 : 1); - drawable._scale[1] = target._size * (drawable.ext30_mirror1 ? drawable.ext30_mirror1 : 1); + target.emitFast(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, this); drawable._renderer.dirty = true; - drawable._rotationCenterDirty = true; - drawable._skinScaleDirty = true; - drawable.setTransformDirty(); + drawable._rotationCenterDirty = true; + drawable._skinScaleDirty = true; + drawable.setTransformDirty(); + this.runtime.requestRedraw(); } //TODO: 拉伸 // From 4f48f1b4ad83e69977abf9be6936b48564edd219 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Mon, 1 Aug 2022 00:17:32 +0800 Subject: [PATCH 104/357] Update projectWith30.js --- Arkos/projectWith30.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index aa99cbc2..1905ffc8 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1839,7 +1839,7 @@ class ArkosExtensions { } util.target['ext30_mirror' + args.mirrorMethod] *= -1; //更新 - target.emitFast(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, this); + target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); drawable._renderer.dirty = true; drawable._rotationCenterDirty = true; drawable._skinScaleDirty = true; @@ -1853,7 +1853,7 @@ class ArkosExtensions { drawable.ext30_mirror0 = 1; drawable.ext30_mirror1 = 1; //更新 - target.emitFast(RenderedTarget.EVENT_TARGET_VISUAL_CHANGE, this); + target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); drawable._renderer.dirty = true; drawable._rotationCenterDirty = true; drawable._skinScaleDirty = true; From 7c58b570e86d12f38d23b98b5cfe47a62d151b65 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Mon, 1 Aug 2022 13:53:08 +0800 Subject: [PATCH 105/357] Update projectWith30.js --- Arkos/projectWith30.js | 63 ++---------------------------------------- 1 file changed, 3 insertions(+), 60 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 1905ffc8..9267585a 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -122,9 +122,6 @@ class ArkosExtensions { '30Ext.menu.mirrorMethod.1': '左右镜像', '30Ext.menu.mirrorMethod.2': '上下镜像', '30Ext.block.clearMirror': '清除角色镜像变换', - '30Ext.info.2': '🛸 角色跨域操作', - '30Ext.block.anotherRun': '让[spriteName]运行', - '30Ext.block.anotherRunWithClone': '让[spriteName]的第[cloneId]个克隆体运行' }, en: { @@ -228,9 +225,6 @@ class ArkosExtensions { '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', '30Ext.block.clearMirror': 'Clear the mirror transform', - '30Ext.info.2': '🛸 Cross sprite operation', - '30Ext.block.anotherRun': 'Let [spriteName] run', - '30Ext.block.anotherRunWithClone': 'Let the [cloneId] clone of [spriteName] run' }, }) } @@ -1039,37 +1033,7 @@ class ArkosExtensions { opcode: 'clearMirror', blockType: 'command', text: this.formatMessage('30Ext.block.clearMirror') - }, - "---" + this.formatMessage("30Ext.info.2"), //角色跨域操作 - //跨域执行 - { - opcode: 'anotherRun', - blockType: 'command', - text: this.formatMessage('30Ext.block.anotherRun'), - arguments: { - spriteName: { - type: 'string', - menu: 'spritesMenu' - } - } - }, - //跨域执行(克隆体) - { - opcode: 'anotherRunWithClone', - blockType: 'command', - text: this.formatMessage('30Ext.block.anotherRunWithClone'), - arguments: { - spriteName: { - type: 'string', - menu: 'spritesMenu' - }, - cloneId: { - type: 'number', - defaultValue: 1 - } - - } - }, + } ], menus: { @@ -1827,6 +1791,7 @@ class ArkosExtensions { //镜像造型 mirrorSprite(args, util) { let target = util.target; + this.setXY({x: target._x+1, y: target._y+1},util); let drawable = this.runtime.renderer._allDrawables[target.drawableID]; if(!util.target.ext30_isHook) { target.ext30_mirror0 = 1; @@ -1840,11 +1805,7 @@ class ArkosExtensions { util.target['ext30_mirror' + args.mirrorMethod] *= -1; //更新 target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); - drawable._renderer.dirty = true; - drawable._rotationCenterDirty = true; - drawable._skinScaleDirty = true; - drawable.setTransformDirty(); - this.runtime.requestRedraw(); + this.setXY({x: target._x-1, y: target._y-1},util); } //清除镜像 clearMirror(args, util) { @@ -1854,26 +1815,8 @@ class ArkosExtensions { drawable.ext30_mirror1 = 1; //更新 target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); - drawable._renderer.dirty = true; - drawable._rotationCenterDirty = true; - drawable._skinScaleDirty = true; - drawable.setTransformDirty(); - this.runtime.requestRedraw(); } //TODO: 拉伸 - // - //角色跨域操作 - // - //跨域执行 - anotherRun(args, util) { - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName) - .sprite.clones[0]; - } - //跨域克隆体执行 - anotherRunWithClone(args, util) { - util.thread.target = this.runtime.targets.find(target => target.sprite.name === args.spriteName) - .sprite.clones[args.cloneId]; - } } From abb2af9522c4246d09c90b168f34a08c7cacfe74 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Mon, 1 Aug 2022 13:55:36 +0800 Subject: [PATCH 106/357] =?UTF-8?q?=E7=A7=BB=E9=99=A4=E4=BA=86=E8=B7=A8?= =?UTF-8?q?=E5=9F=9F=E6=93=8D=E4=BD=9C=EF=BC=88=E6=97=A0=E6=B3=95=E4=BD=BF?= =?UTF-8?q?=E7=94=A8=EF=BC=89?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 2 ++ 1 file changed, 2 insertions(+) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 9267585a..8d6eeb60 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1810,11 +1810,13 @@ class ArkosExtensions { //清除镜像 clearMirror(args, util) { let target = util.target; + this.setXY({x: target._x+1, y: target._y+1},util); let drawable = this.runtime.renderer._allDrawables[target.drawableID]; drawable.ext30_mirror0 = 1; drawable.ext30_mirror1 = 1; //更新 target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); + this.setXY({x: target._x-1, y: target._y-1},util); } //TODO: 拉伸 } From 60a926d3ab23512264e2ec4183022747e2bb0c0e Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 12:13:58 +0800 Subject: [PATCH 107/357] Update projectWith30.js --- Arkos/projectWith30.js | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 8d6eeb60..6f31fb22 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1788,11 +1788,10 @@ class ArkosExtensions { // //角色造型操作 // - //镜像造型 - mirrorSprite(args, util) { +mirrorSprite(args, util) { let target = util.target; - this.setXY({x: target._x+1, y: target._y+1},util); let drawable = this.runtime.renderer._allDrawables[target.drawableID]; +this.setXY({x: target._x+1, y: target._y+1},util); if(!util.target.ext30_isHook) { target.ext30_mirror0 = 1; target.ext30_mirror1 = 1; @@ -1800,6 +1799,12 @@ class ArkosExtensions { drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * target.ext30_mirror0; drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * target.ext30_mirror1; }); +let oldf = target.__proto__.setSize; +target.__proto__.setSize = function(size) { +this.setXY({x: target._x+1, y: target._y+1},util); +oldf.call(target, size); +this.setXY({x: target._x-1, y: target._y-1},util); +} target.ext30_isHook = true; } util.target['ext30_mirror' + args.mirrorMethod] *= -1; From efc1ebca1b51cc02f4f67dc8e776ab25cbd8ffcf Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 12:42:38 +0800 Subject: [PATCH 108/357] Update projectWith30.js --- Arkos/projectWith30.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 6f31fb22..2619a37f 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1791,7 +1791,7 @@ class ArkosExtensions { mirrorSprite(args, util) { let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; -this.setXY({x: target._x+1, y: target._y+1},util); +this.setXY({x: target.x+1, y: target.y+1},util); if(!util.target.ext30_isHook) { target.ext30_mirror0 = 1; target.ext30_mirror1 = 1; @@ -1801,16 +1801,16 @@ this.setXY({x: target._x+1, y: target._y+1},util); }); let oldf = target.__proto__.setSize; target.__proto__.setSize = function(size) { -this.setXY({x: target._x+1, y: target._y+1},util); +this.setXY({x: target.x+1, y: target.y+1},util); oldf.call(target, size); -this.setXY({x: target._x-1, y: target._y-1},util); +this.setXY({x: target.x-1, y: target.y-1},util); } target.ext30_isHook = true; } util.target['ext30_mirror' + args.mirrorMethod] *= -1; //更新 target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); - this.setXY({x: target._x-1, y: target._y-1},util); + this.setXY({x: target.x-1, y: target.y-1},util); } //清除镜像 clearMirror(args, util) { From c88c9fa22d2e461bac1912d0c030ca5e7d3795e4 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 2 Aug 2022 12:50:01 +0800 Subject: [PATCH 109/357] =?UTF-8?q?=E6=9B=B4=E6=8D=A2=E6=96=87=E5=AD=97?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 2619a37f..cf9e2742 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -117,11 +117,11 @@ class ArkosExtensions { 'ArkosExt.lengthOfTempCon': '🗂️临时容器[con]中内容数', '30Ext.info': '✨ 以下扩展由_30提供', - '30Ext.info.1': '🪞 造型镜像操作', - '30Ext.block.mirrorSprite': '[mirrorMethod]当前角色', + '30Ext.info.1': '🔮 造型镜像操作', + '30Ext.block.mirrorSprite': '(⚠还在测试)[mirrorMethod]当前角色', '30Ext.menu.mirrorMethod.1': '左右镜像', '30Ext.menu.mirrorMethod.2': '上下镜像', - '30Ext.block.clearMirror': '清除角色镜像变换', + '30Ext.block.clearMirror': '(⚠还在测试)清除角色镜像变换', }, en: { @@ -220,11 +220,11 @@ class ArkosExtensions { 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', '30Ext.info': '✨ Contributed by _30', - '30Ext.info.1': '🪞 Mirror transform', - '30Ext.block.mirrorSprite': '[mirrorMethod] current sprite', + '30Ext.info.1': '🔮 Mirror transform', + '30Ext.block.mirrorSprite': '(⚠testing)[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', - '30Ext.block.clearMirror': 'Clear the mirror transform', + '30Ext.block.clearMirror': '(⚠testing)Clear the mirror transform', }, }) } From 0a14a791e23ef82491504aa1b7ecda16b54f1577 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 12:57:23 +0800 Subject: [PATCH 110/357] Update projectWith30.js --- Arkos/projectWith30.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index cf9e2742..ff49c501 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1801,9 +1801,9 @@ this.setXY({x: target.x+1, y: target.y+1},util); }); let oldf = target.__proto__.setSize; target.__proto__.setSize = function(size) { -this.setXY({x: target.x+1, y: target.y+1},util); + util.target.renderer.updateDrawablePosition(util.target.drawableID, [target.x+1,target.y]); oldf.call(target, size); -this.setXY({x: target.x-1, y: target.y-1},util); + util.target.renderer.updateDrawablePosition(util.target.drawableID, [target.x-1,target.y]); } target.ext30_isHook = true; } From ace3c6e0ca992455c8547bf83dfea825da0d6643 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 12:58:17 +0800 Subject: [PATCH 111/357] Update projectWith30.js --- Arkos/projectWith30.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index ff49c501..8f84f445 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1815,13 +1815,13 @@ oldf.call(target, size); //清除镜像 clearMirror(args, util) { let target = util.target; - this.setXY({x: target._x+1, y: target._y+1},util); + this.setXY({x: target.x+1, y: target.y+1},util); let drawable = this.runtime.renderer._allDrawables[target.drawableID]; drawable.ext30_mirror0 = 1; drawable.ext30_mirror1 = 1; //更新 target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); - this.setXY({x: target._x-1, y: target._y-1},util); + this.setXY({x: target.x-1, y: target.y-1},util); } //TODO: 拉伸 } From d60a270ab6a5ce7e8735954664521dc46648bb79 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Tue, 2 Aug 2022 13:05:01 +0800 Subject: [PATCH 112/357] =?UTF-8?q?=E6=9B=B4=E6=94=B9emoji?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 8f84f445..704c11c6 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -118,10 +118,10 @@ class ArkosExtensions { '30Ext.info': '✨ 以下扩展由_30提供', '30Ext.info.1': '🔮 造型镜像操作', - '30Ext.block.mirrorSprite': '(⚠还在测试)[mirrorMethod]当前角色', + '30Ext.block.mirrorSprite': '(⚠️还在测试)[mirrorMethod]当前角色', '30Ext.menu.mirrorMethod.1': '左右镜像', '30Ext.menu.mirrorMethod.2': '上下镜像', - '30Ext.block.clearMirror': '(⚠还在测试)清除角色镜像变换', + '30Ext.block.clearMirror': '(⚠️还在测试)清除角色镜像变换', }, en: { @@ -221,10 +221,10 @@ class ArkosExtensions { '30Ext.info': '✨ Contributed by _30', '30Ext.info.1': '🔮 Mirror transform', - '30Ext.block.mirrorSprite': '(⚠testing)[mirrorMethod] current sprite', + '30Ext.block.mirrorSprite': '(⚠️testing)[mirrorMethod] current sprite', '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', - '30Ext.block.clearMirror': '(⚠testing)Clear the mirror transform', + '30Ext.block.clearMirror': '(⚠️testing)Clear the mirror transform', }, }) } From ba6b25de9fa4bf7e97d6caa931ae08ecfca5869a Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 13:26:13 +0800 Subject: [PATCH 113/357] Update projectWith30.js --- Arkos/projectWith30.js | 28 ++++++++++++++++++++++------ 1 file changed, 22 insertions(+), 6 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 704c11c6..e4f2be87 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1793,18 +1793,34 @@ mirrorSprite(args, util) { let drawable = this.runtime.renderer._allDrawables[target.drawableID]; this.setXY({x: target.x+1, y: target.y+1},util); if(!util.target.ext30_isHook) { + let setXYhook = function(args, util) { + if(util.target.isStage) return; + args.x = this._clamp(Cast.toNumber(args.x), -100000000, 100000000) + args.y = this._clamp(Cast.toNumber(args.y), -100000000, 100000000) + const oldX = util.target.x; + const oldY = util.target.y; + util.target.x = args.x; + util.target.y = args.y; + if(util.target.renderer) { + util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); + if(util.target.visible) { + util.target.runtime.requestRedraw(); + } + } else { + util.target.x = x; + util.target.y = y; + } + util.target.emit('TARGET_MOVED', util.target, oldX, oldY, false); + util.target.runtime.requestTargetsUpdate(util.target); + } target.ext30_mirror0 = 1; target.ext30_mirror1 = 1; target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e, t) => { + setXYhook({x: target.x-1, y: target.y-1},util); drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * target.ext30_mirror0; drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * target.ext30_mirror1; + setXYhook({x: target.x+1, y: target.y+1},util); }); -let oldf = target.__proto__.setSize; -target.__proto__.setSize = function(size) { - util.target.renderer.updateDrawablePosition(util.target.drawableID, [target.x+1,target.y]); -oldf.call(target, size); - util.target.renderer.updateDrawablePosition(util.target.drawableID, [target.x-1,target.y]); -} target.ext30_isHook = true; } util.target['ext30_mirror' + args.mirrorMethod] *= -1; From ee2fc12bc79b199eb7fa3bfa8735af0a8ea39cfb Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 13:34:48 +0800 Subject: [PATCH 114/357] Update projectWith30.js --- Arkos/projectWith30.js | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index e4f2be87..3a5c5f29 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1795,8 +1795,6 @@ this.setXY({x: target.x+1, y: target.y+1},util); if(!util.target.ext30_isHook) { let setXYhook = function(args, util) { if(util.target.isStage) return; - args.x = this._clamp(Cast.toNumber(args.x), -100000000, 100000000) - args.y = this._clamp(Cast.toNumber(args.y), -100000000, 100000000) const oldX = util.target.x; const oldY = util.target.y; util.target.x = args.x; @@ -1832,9 +1830,8 @@ this.setXY({x: target.x+1, y: target.y+1},util); clearMirror(args, util) { let target = util.target; this.setXY({x: target.x+1, y: target.y+1},util); - let drawable = this.runtime.renderer._allDrawables[target.drawableID]; - drawable.ext30_mirror0 = 1; - drawable.ext30_mirror1 = 1; + +target.ext30_mirror0 = 1;target.ext30_mirror1 = 1; //更新 target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); this.setXY({x: target.x-1, y: target.y-1},util); From 4ac9e536939d4a04db0358a3796fd791ea446430 Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Thu, 21 Jul 2022 13:19:52 +0800 Subject: [PATCH 115/357] =?UTF-8?q?=E5=B0=9D=E8=AF=95=E4=BF=AE=E5=A4=8Darg?= =?UTF-8?q?s.LIST=E7=9A=84=E9=97=AE=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- six-6/regexp.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/six-6/regexp.js b/six-6/regexp.js index a28ca9f8..16fca510 100644 --- a/six-6/regexp.js +++ b/six-6/regexp.js @@ -700,7 +700,8 @@ class RegExpVI { } let text = String(args.TEXT); result = text.split(regexp); - let vari = util.target.lookupVariableById(args.LIST); + // 这里估计是vm对名为LIST或者VARIABLE的参数有特殊处理 + let vari = util.target.lookupVariableById(args.LIST.id); if (vari !== null) vari.value = result; } From ff2f7fbc5e95cd8d5a8feb88129ecedb5d9d1fef Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Thu, 21 Jul 2022 13:20:17 +0800 Subject: [PATCH 116/357] =?UTF-8?q?Revert=20"=E5=B0=9D=E8=AF=95=E4=BF=AE?= =?UTF-8?q?=E5=A4=8Dargs.LIST=E7=9A=84=E9=97=AE=E9=A2=98"?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This reverts commit 314a61057464f4543f80c06643b1a65d3c3e8366. --- six-6/regexp.js | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/six-6/regexp.js b/six-6/regexp.js index 16fca510..a28ca9f8 100644 --- a/six-6/regexp.js +++ b/six-6/regexp.js @@ -700,8 +700,7 @@ class RegExpVI { } let text = String(args.TEXT); result = text.split(regexp); - // 这里估计是vm对名为LIST或者VARIABLE的参数有特殊处理 - let vari = util.target.lookupVariableById(args.LIST.id); + let vari = util.target.lookupVariableById(args.LIST); if (vari !== null) vari.value = result; } From 9b3a5409049b17ed7c17e628068b0eba891cc2a3 Mon Sep 17 00:00:00 2001 From: six-6 <1195132772@qq.com> Date: Thu, 21 Jul 2022 13:22:02 +0800 Subject: [PATCH 117/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E4=BA=86RegExpVI=20s?= =?UTF-8?q?plit/matchall=20LIST=20->=20LISTID=20=E7=9A=84=E9=97=AE?= =?UTF-8?q?=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- six-6/regexp.js | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/six-6/regexp.js b/six-6/regexp.js index a28ca9f8..facd2643 100644 --- a/six-6/regexp.js +++ b/six-6/regexp.js @@ -59,8 +59,8 @@ class RegExpVI { "RegExpVI.foundtextbegin": "找到的文本开始位置", "RegExpVI.foundtextend": "找到的文本结束位置", "RegExpVI.foundtextgroups": "找到的第 [N] 个捕获组的内容", - "RegExpVI.findall": "在 [TEXT] 里查找所有 [REGEXP] 并保存到 [LIST]", - "RegExpVI.split": "在 [TEXT] 里以 [REGEXP] 分割并保存到 [LIST]", + "RegExpVI.findall": "在 [TEXT] 里查找所有 [REGEXP] 并保存到列表 [LISTID]", + "RegExpVI.split": "在 [TEXT] 里以 [REGEXP] 分割并保存到列表 [LISTID]", "RegExpVI.replace": "将 [TEXT] 里的所有 [REGEXP] 替换为 [DEST] [GROUP]", "RegExpVI.replace.group": "(使用引用记号)", "RegExpVI.replace.nogroup": "(不使用引用记号)", @@ -117,8 +117,8 @@ class RegExpVI { "RegExpVI.foundtextbegin": "begin index", "RegExpVI.foundtextend": "end index", "RegExpVI.foundtextgroups": "founded content of group [N]", - "RegExpVI.findall": "find all [REGEXP] in [TEXT] and save to [LIST]", - "RegExpVI.split": "split [TEXT] by [REGEXP] and save to [LIST]", + "RegExpVI.findall": "find all [REGEXP] in [TEXT] and save to list [LISTID]", + "RegExpVI.split": "split [TEXT] by [REGEXP] and save to list [LISTID]", "RegExpVI.replace": "replace all [REGEXP] in [TEXT] with [DEST] [GROUP]", "RegExpVI.replace.group": "(refs enabled)", "RegExpVI.replace.nogroup": "(refs disabled)", @@ -373,7 +373,7 @@ class RegExpVI { type: "string", defaultValue: "I have 23 45." }, - LIST: { + LISTID: { type: "string", menu: "List" } @@ -392,7 +392,7 @@ class RegExpVI { type: "string", defaultValue: "I have 23 apples." }, - LIST: { + LISTID: { type: "string", menu: "List" } @@ -684,7 +684,7 @@ class RegExpVI { while (res = regexp.exec(text), res !== null) { result.push(res[0]); } - let vari = util.target.lookupVariableById(args.LIST); + let vari = util.target.lookupVariableById(args.LISTID); if (vari !== null) vari.value = result; } @@ -700,7 +700,7 @@ class RegExpVI { } let text = String(args.TEXT); result = text.split(regexp); - let vari = util.target.lookupVariableById(args.LIST); + let vari = util.target.lookupVariableById(args.LISTID); if (vari !== null) vari.value = result; } From d2dbdf8d36fa416bb920f554ae0a9c89ed995ca3 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 21:35:10 +0800 Subject: [PATCH 118/357] Update projectWith30.js --- Arkos/projectWith30.js | 107 ++++++++++++++++++++--------------------- 1 file changed, 53 insertions(+), 54 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 3a5c5f29..33c1bfc5 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -118,10 +118,11 @@ class ArkosExtensions { '30Ext.info': '✨ 以下扩展由_30提供', '30Ext.info.1': '🔮 造型镜像操作', - '30Ext.block.mirrorSprite': '(⚠️还在测试)[mirrorMethod]当前角色', - '30Ext.menu.mirrorMethod.1': '左右镜像', - '30Ext.menu.mirrorMethod.2': '上下镜像', - '30Ext.block.clearMirror': '(⚠️还在测试)清除角色镜像变换', + '30Ext.block.mirrorSprite': '(⚠️还在测试)将角色的镜像模式设为[mirrorMethod]', + '30Ext.menu.mirrorMethod.1': '无镜像', + '30Ext.menu.mirrorMethod.2': '左右镜像', + '30Ext.menu.mirrorMethod.3': '上下镜像', + '30Ext.menu.mirrorMethod.4': '上下左右镜像', }, en: { @@ -221,10 +222,11 @@ class ArkosExtensions { '30Ext.info': '✨ Contributed by _30', '30Ext.info.1': '🔮 Mirror transform', - '30Ext.block.mirrorSprite': '(⚠️testing)[mirrorMethod] current sprite', - '30Ext.menu.mirrorMethod.1': 'Horizontal mirror transform', - '30Ext.menu.mirrorMethod.2': 'Vertical mirror transform', - '30Ext.block.clearMirror': '(⚠️testing)Clear the mirror transform', + '30Ext.block.mirrorSprite': '(⚠️Testing)Set the mirroring mode of the sprite to [mirrorMethod]', + '30Ext.menu.mirrorMethod.1': 'No mirror', + '30Ext.menu.mirrorMethod.2': 'Horizontal mirror', + '30Ext.menu.mirrorMethod.3': 'Vertical Mirror', + '30Ext.menu.mirrorMethod.4': 'Horizontal & Vertical mirror', }, }) } @@ -1174,12 +1176,20 @@ class ArkosExtensions { items: 'getSpritesMenu' }, mirrorMenu: [{ - text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //左右镜像 - value: '0' + text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //无镜像 + value: '1' }, { - text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //上下镜像 - value: '1' + text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //左右镜像 + value: '2' + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.3'), //上下镜像 + value: '3' + }, + { + text: this.formatMessage('30Ext.menu.mirrorMethod.4'), //上下左右镜像 + value: '4' } ] }, @@ -1788,53 +1798,42 @@ class ArkosExtensions { // //角色造型操作 // -mirrorSprite(args, util) { + mirrorSprite(args, util) { let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; -this.setXY({x: target.x+1, y: target.y+1},util); - if(!util.target.ext30_isHook) { - let setXYhook = function(args, util) { - if(util.target.isStage) return; - const oldX = util.target.x; - const oldY = util.target.y; - util.target.x = args.x; - util.target.y = args.y; - if(util.target.renderer) { - util.target.renderer.updateDrawablePosition(util.target.drawableID, [args.x, args.y]); - if(util.target.visible) { - util.target.runtime.requestRedraw(); - } - } else { - util.target.x = x; - util.target.y = y; + switch (args.mirrorMethod) { + case 1: + target.ext30.mirror.x = 1; + target.ext30.mirror.y = 1; + break; + case 2: + target.ext30.mirror.x = -1; + target.ext30.mirror.y = 1; + break; + case 3: + target.ext30.mirror.x = 1; + target.ext30.mirror.y = -1; + break; + case 4: + target.ext30.mirror.x = -1; + target.ext30.mirror.y = -1; + break; } - util.target.emit('TARGET_MOVED', util.target, oldX, oldY, false); - util.target.runtime.requestTargetsUpdate(util.target); - } - target.ext30_mirror0 = 1; - target.ext30_mirror1 = 1; - target.addListener('EVENT_TARGET_VISUAL_CHANGE', (e, t) => { - setXYhook({x: target.x-1, y: target.y-1},util); - drawable._skinScale[0] = Math.abs(drawable._skinScale[0]) * target.ext30_mirror0; - drawable._skinScale[1] = Math.abs(drawable._skinScale[1]) * target.ext30_mirror1; - setXYhook({x: target.x+1, y: target.y+1},util); - }); - target.ext30_isHook = true; + if(!target.ext30.mirror.hook) { + //初始化 + target.ext30.mirror.x = 1; + target.ext30.mirror.y = 1; + //注入修改函数 + let old_fun = drawable.__proto__.updateScale; + drawable.__proto__.updateScale = function(scale) { + scale[0] *= target.ext30.mirror.x; + scale[1] *= target.ext30.mirror.y; + return old_fun.call(drawable, scale); + } + target.ext30.mirror.hook = true; } - util.target['ext30_mirror' + args.mirrorMethod] *= -1; - //更新 - target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); - this.setXY({x: target.x-1, y: target.y-1},util); - } - //清除镜像 - clearMirror(args, util) { - let target = util.target; - this.setXY({x: target.x+1, y: target.y+1},util); - -target.ext30_mirror0 = 1;target.ext30_mirror1 = 1; //更新 - target.emitFast('EVENT_TARGET_VISUAL_CHANGE', this); - this.setXY({x: target.x-1, y: target.y-1},util); + drawable.updateScale(drawable.scale); } //TODO: 拉伸 } From 6a465c3ddf0025d662bde23f0ed8a7b9a1fa4a06 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 21:46:53 +0800 Subject: [PATCH 119/357] Update projectWith30.js --- Arkos/projectWith30.js | 45 ++++++++++++++++++++++-------------------- 1 file changed, 24 insertions(+), 21 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 33c1bfc5..ef476c76 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1029,14 +1029,7 @@ class ArkosExtensions { menu: 'mirrorMenu' } } - }, - // 清除镜像 - { - opcode: 'clearMirror', - blockType: 'command', - text: this.formatMessage('30Ext.block.clearMirror') } - ], menus: { conInfoMenu: [{ @@ -1782,6 +1775,19 @@ class ArkosExtensions { // //30Ext // + //初始化 + ext30_tryInit(target) { + if (!target.ext30) { + target.ext30 = { + mirror: { + x: 1, + y: 1, + hook: false + } + }; + } + return target; + } //菜单 //动态菜单: 角色菜单 getSpritesMenu() { @@ -1799,8 +1805,18 @@ class ArkosExtensions { //角色造型操作 // mirrorSprite(args, util) { - let target = util.target; + let target = this.ext30_tryInit(util.target); let drawable = this.runtime.renderer._allDrawables[target.drawableID]; + if(!target.ext30.mirror.hook) { + //注入修改函数 + let old_fun = drawable.__proto__.updateScale; + drawable.__proto__.updateScale = function(scale) { + scale[0] *= target.ext30.mirror.x; + scale[1] *= target.ext30.mirror.y; + return old_fun.call(drawable, scale); + } + target.ext30.mirror.hook = true; + } switch (args.mirrorMethod) { case 1: target.ext30.mirror.x = 1; @@ -1819,19 +1835,6 @@ class ArkosExtensions { target.ext30.mirror.y = -1; break; } - if(!target.ext30.mirror.hook) { - //初始化 - target.ext30.mirror.x = 1; - target.ext30.mirror.y = 1; - //注入修改函数 - let old_fun = drawable.__proto__.updateScale; - drawable.__proto__.updateScale = function(scale) { - scale[0] *= target.ext30.mirror.x; - scale[1] *= target.ext30.mirror.y; - return old_fun.call(drawable, scale); - } - target.ext30.mirror.hook = true; - } //更新 drawable.updateScale(drawable.scale); } From 4c4020469ac063703db89436d2e40d2da4ac9045 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 21:49:49 +0800 Subject: [PATCH 120/357] Update projectWith30.js --- Arkos/projectWith30.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index ef476c76..61571a81 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1818,19 +1818,19 @@ class ArkosExtensions { target.ext30.mirror.hook = true; } switch (args.mirrorMethod) { - case 1: + case '1': target.ext30.mirror.x = 1; target.ext30.mirror.y = 1; break; - case 2: + case '2': target.ext30.mirror.x = -1; target.ext30.mirror.y = 1; break; - case 3: + case '3': target.ext30.mirror.x = 1; target.ext30.mirror.y = -1; break; - case 4: + case '4': target.ext30.mirror.x = -1; target.ext30.mirror.y = -1; break; From 314fa1d451f2d78ef30a9e596e13dd3f79c25417 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 22:03:39 +0800 Subject: [PATCH 121/357] Update projectWith30.js --- Arkos/projectWith30.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 61571a81..cf8304cc 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1811,8 +1811,8 @@ class ArkosExtensions { //注入修改函数 let old_fun = drawable.__proto__.updateScale; drawable.__proto__.updateScale = function(scale) { - scale[0] *= target.ext30.mirror.x; - scale[1] *= target.ext30.mirror.y; + scale[0] = Math.abs(scale[0]) * target.ext30.mirror.x; + scale[1] = Math.abs(scale[1]) * target.ext30.mirror.y; return old_fun.call(drawable, scale); } target.ext30.mirror.hook = true; From f59166bc005746e703908bb4039554b09e8f81d5 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 22:18:00 +0800 Subject: [PATCH 122/357] Update projectWith30.js --- Arkos/projectWith30.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index cf8304cc..9ce53ca8 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1028,7 +1028,8 @@ class ArkosExtensions { type: 'string', menu: 'mirrorMenu' } - } + }, + filter: ['sprite'] } ], menus: { From 6ecc99af5c4b66a16e720d7802f4f3fa33aa8390 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 23:52:43 +0800 Subject: [PATCH 123/357] Update projectWith30.js --- Arkos/projectWith30.js | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 9ce53ca8..25101456 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1778,7 +1778,7 @@ class ArkosExtensions { // //初始化 ext30_tryInit(target) { - if (!target.ext30) { + if(!target.ext30) { target.ext30 = { mirror: { x: 1, @@ -1810,11 +1810,16 @@ class ArkosExtensions { let drawable = this.runtime.renderer._allDrawables[target.drawableID]; if(!target.ext30.mirror.hook) { //注入修改函数 - let old_fun = drawable.__proto__.updateScale; - drawable.__proto__.updateScale = function(scale) { + drawable.updateScale.prototype.constructor = function(scale) { scale[0] = Math.abs(scale[0]) * target.ext30.mirror.x; scale[1] = Math.abs(scale[1]) * target.ext30.mirror.y; - return old_fun.call(drawable, scale); + if(drawable._scale[0] !== scale[0] ||drawable._scale[1] !== scale[1]) { + drawable._scale[0] = scale[0]; + drawable._scale[1] = scale[1]; + drawable._rotationCenterDirty = true; + drawable._skinScaleDirty = true; + drawable.setTransformDirty(); + } } target.ext30.mirror.hook = true; } From 8ac7e78daf8dd5491d972f06867165c9ab47f8e4 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 23:54:52 +0800 Subject: [PATCH 124/357] Update projectWith30.js --- Arkos/projectWith30.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 25101456..1291e0e5 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1810,7 +1810,7 @@ class ArkosExtensions { let drawable = this.runtime.renderer._allDrawables[target.drawableID]; if(!target.ext30.mirror.hook) { //注入修改函数 - drawable.updateScale.prototype.constructor = function(scale) { + drawable.updateScale = function(scale) { scale[0] = Math.abs(scale[0]) * target.ext30.mirror.x; scale[1] = Math.abs(scale[1]) * target.ext30.mirror.y; if(drawable._scale[0] !== scale[0] ||drawable._scale[1] !== scale[1]) { From c9bd3cf852480bb7ea9ed3201aacf32d6177676c Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Tue, 2 Aug 2022 23:56:58 +0800 Subject: [PATCH 125/357] Update projectWith30.js --- Arkos/projectWith30.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 1291e0e5..405e2672 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1810,7 +1810,7 @@ class ArkosExtensions { let drawable = this.runtime.renderer._allDrawables[target.drawableID]; if(!target.ext30.mirror.hook) { //注入修改函数 - drawable.updateScale = function(scale) { + drawable.__proto__.updateScale = function(scale) { scale[0] = Math.abs(scale[0]) * target.ext30.mirror.x; scale[1] = Math.abs(scale[1]) * target.ext30.mirror.y; if(drawable._scale[0] !== scale[0] ||drawable._scale[1] !== scale[1]) { From c1504d114a791ba6222e2e50c587aff414230093 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 00:00:59 +0800 Subject: [PATCH 126/357] Update projectWith30.js --- Arkos/projectWith30.js | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 405e2672..8bd58afa 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1813,12 +1813,12 @@ class ArkosExtensions { drawable.__proto__.updateScale = function(scale) { scale[0] = Math.abs(scale[0]) * target.ext30.mirror.x; scale[1] = Math.abs(scale[1]) * target.ext30.mirror.y; - if(drawable._scale[0] !== scale[0] ||drawable._scale[1] !== scale[1]) { - drawable._scale[0] = scale[0]; - drawable._scale[1] = scale[1]; - drawable._rotationCenterDirty = true; - drawable._skinScaleDirty = true; - drawable.setTransformDirty(); + if(this._scale[0] !== scale[0] ||this._scale[1] !== scale[1]) { + this._scale[0] = scale[0]; + this._scale[1] = scale[1]; + this._rotationCenterDirty = true; + this._skinScaleDirty = true; + this.setTransformDirty(); } } target.ext30.mirror.hook = true; From 3e667bb1fac2d2ace2000b125cf416b06dadcc8e Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 00:09:39 +0800 Subject: [PATCH 127/357] Update projectWith30.js --- Arkos/projectWith30.js | 50 ++++++++++++++---------------------------- 1 file changed, 17 insertions(+), 33 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 8bd58afa..44ab43d6 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1776,19 +1776,6 @@ class ArkosExtensions { // //30Ext // - //初始化 - ext30_tryInit(target) { - if(!target.ext30) { - target.ext30 = { - mirror: { - x: 1, - y: 1, - hook: false - } - }; - } - return target; - } //菜单 //动态菜单: 角色菜单 getSpritesMenu() { @@ -1806,39 +1793,36 @@ class ArkosExtensions { //角色造型操作 // mirrorSprite(args, util) { - let target = this.ext30_tryInit(util.target); + let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; - if(!target.ext30.mirror.hook) { + if(!drawable.ext30_mirror_hook) { + drawable.ext30_mirror_x = 1; + drawable.ext30_mirror_y = 1; //注入修改函数 + let old_fun = drawable.__proto__.updateScale; drawable.__proto__.updateScale = function(scale) { - scale[0] = Math.abs(scale[0]) * target.ext30.mirror.x; - scale[1] = Math.abs(scale[1]) * target.ext30.mirror.y; - if(this._scale[0] !== scale[0] ||this._scale[1] !== scale[1]) { - this._scale[0] = scale[0]; - this._scale[1] = scale[1]; - this._rotationCenterDirty = true; - this._skinScaleDirty = true; - this.setTransformDirty(); - } + scale[0] = Math.abs(scale[0]) * this.ext30_mirror_x; + scale[1] = Math.abs(scale[1]) * this.ext30_mirror_y; + return old_fun.call(this, scale); } - target.ext30.mirror.hook = true; + drawable.ext30_mirror_hook = true; } switch (args.mirrorMethod) { case '1': - target.ext30.mirror.x = 1; - target.ext30.mirror.y = 1; + drawable.ext30_mirror_x = 1; + drawable.ext30_mirror_y = 1; break; case '2': - target.ext30.mirror.x = -1; - target.ext30.mirror.y = 1; + drawable.ext30_mirror_x = -1; + drawable.ext30_mirror_y = 1; break; case '3': - target.ext30.mirror.x = 1; - target.ext30.mirror.y = -1; + drawable.ext30_mirror_x = 1; + drawable.ext30_mirror_y = -1; break; case '4': - target.ext30.mirror.x = -1; - target.ext30.mirror.y = -1; + drawable.ext30_mirror_x = -1; + drawable.ext30_mirror_y = -1; break; } //更新 From b0b3713849526fbda8e3145fc55076e6a2428297 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 20:08:17 +0800 Subject: [PATCH 128/357] Update projectWith30.js --- Arkos/projectWith30.js | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 44ab43d6..941042ec 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1800,11 +1800,13 @@ class ArkosExtensions { drawable.ext30_mirror_y = 1; //注入修改函数 let old_fun = drawable.__proto__.updateScale; - drawable.__proto__.updateScale = function(scale) { - scale[0] = Math.abs(scale[0]) * this.ext30_mirror_x; - scale[1] = Math.abs(scale[1]) * this.ext30_mirror_y; - return old_fun.call(this, scale); - } + Object.defineProperty(drawable, "updateScale" , + {value: function(scale) { + scale[0] = Math.abs(scale[0]) * this.ext30_mirror_x; + scale[1] = Math.abs(scale[1]) * this.ext30_mirror_y; + return old_fun.call(this, scale); + }} + ); drawable.ext30_mirror_hook = true; } switch (args.mirrorMethod) { From b926c866abdf63514ca6e93384764e430897e209 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 20:52:52 +0800 Subject: [PATCH 129/357] =?UTF-8?q?=E7=A7=BB=E9=99=A4=E9=95=9C=E5=83=8F=20?= =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E5=AE=9A=E5=90=91=E7=BC=A9=E6=94=BE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 103 ++++++++++++++++------------------------- 1 file changed, 40 insertions(+), 63 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 941042ec..0b0a1dca 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -117,12 +117,9 @@ class ArkosExtensions { 'ArkosExt.lengthOfTempCon': '🗂️临时容器[con]中内容数', '30Ext.info': '✨ 以下扩展由_30提供', - '30Ext.info.1': '🔮 造型镜像操作', - '30Ext.block.mirrorSprite': '(⚠️还在测试)将角色的镜像模式设为[mirrorMethod]', - '30Ext.menu.mirrorMethod.1': '无镜像', - '30Ext.menu.mirrorMethod.2': '左右镜像', - '30Ext.menu.mirrorMethod.3': '上下镜像', - '30Ext.menu.mirrorMethod.4': '上下左右镜像', + '30Ext.info.1': '🔮 定向缩放操作', + '30Ext.block.sclaeSpriteX': '水平缩放角色[input]倍', + '30Ext.block.sclaeSpriteY': '垂直缩放角色[input]倍' }, en: { @@ -221,12 +218,9 @@ class ArkosExtensions { 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', '30Ext.info': '✨ Contributed by _30', - '30Ext.info.1': '🔮 Mirror transform', - '30Ext.block.mirrorSprite': '(⚠️Testing)Set the mirroring mode of the sprite to [mirrorMethod]', - '30Ext.menu.mirrorMethod.1': 'No mirror', - '30Ext.menu.mirrorMethod.2': 'Horizontal mirror', - '30Ext.menu.mirrorMethod.3': 'Vertical Mirror', - '30Ext.menu.mirrorMethod.4': 'Horizontal & Vertical mirror', + '30Ext.info.1': '🔮 Directional scale', + '30Ext.block.sclaeSpriteX': 'Scale the sprite [input] times horizontally', + '30Ext.block.sclaeSpriteY': 'Scale the sprite [input] times vertically' }, }) } @@ -1017,16 +1011,29 @@ class ArkosExtensions { // "---" + this.formatMessage("30Ext.info"), //感谢30提供的扩展 - "---" + this.formatMessage("30Ext.info.1"), //造型镜像 - // 镜像造型 + "---" + this.formatMessage("30Ext.info.1"), //定向缩放 + // x向缩放 { - opcode: 'mirrorSprite', + opcode: 'sclaeSpriteX', blockType: 'command', - text: this.formatMessage('30Ext.block.mirrorSprite'), + text: this.formatMessage('30Ext.block.sclaeSpriteX'), arguments: { - mirrorMethod: { - type: 'string', - menu: 'mirrorMenu' + input: { + type: 'number', + defaultValue: '1' + } + }, + filter: ['sprite'] + } + // y向缩放 + { + opcode: 'sclaeSpriteY', + blockType: 'command', + text: this.formatMessage('30Ext.block.sclaeSpriteY'), + arguments: { + input: { + type: 'number', + defaultValue: '1' } }, filter: ['sprite'] @@ -1168,25 +1175,8 @@ class ArkosExtensions { //30Ext spritesMenu: { items: 'getSpritesMenu' - }, - mirrorMenu: [{ - text: this.formatMessage('30Ext.menu.mirrorMethod.1'), //无镜像 - value: '1' - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.2'), //左右镜像 - value: '2' - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.3'), //上下镜像 - value: '3' - }, - { - text: this.formatMessage('30Ext.menu.mirrorMethod.4'), //上下左右镜像 - value: '4' - } - ] - }, + } + } } } @@ -1792,44 +1782,31 @@ class ArkosExtensions { // //角色造型操作 // - mirrorSprite(args, util) { + scaleSprite(index, value, util) { let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; - if(!drawable.ext30_mirror_hook) { - drawable.ext30_mirror_x = 1; - drawable.ext30_mirror_y = 1; + if(!drawable.ext30_scale) { + drawable.ext30_scale = [1,1]; //注入修改函数 let old_fun = drawable.__proto__.updateScale; Object.defineProperty(drawable, "updateScale" , {value: function(scale) { - scale[0] = Math.abs(scale[0]) * this.ext30_mirror_x; - scale[1] = Math.abs(scale[1]) * this.ext30_mirror_y; + scale[0] = Math.abs(scale[0]) * this.ext30_scale[0]; + scale[1] = Math.abs(scale[1]) * this.ext30_scale[1]; return old_fun.call(this, scale); }} ); - drawable.ext30_mirror_hook = true; - } - switch (args.mirrorMethod) { - case '1': - drawable.ext30_mirror_x = 1; - drawable.ext30_mirror_y = 1; - break; - case '2': - drawable.ext30_mirror_x = -1; - drawable.ext30_mirror_y = 1; - break; - case '3': - drawable.ext30_mirror_x = 1; - drawable.ext30_mirror_y = -1; - break; - case '4': - drawable.ext30_mirror_x = -1; - drawable.ext30_mirror_y = -1; - break; } + drawable.ext30_scale[index] = value; //更新 drawable.updateScale(drawable.scale); } + scaleSpriteX(args, util) { + this.scaleSprite(0, args.input, util); + } + scaleSpriteY(args, util) { + this.scaleSprite(1, args.input, util); + } //TODO: 拉伸 } From 49557578280367150c3ebbbaa99a0104b309490a Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 20:53:57 +0800 Subject: [PATCH 130/357] Update projectWith30.js --- Arkos/projectWith30.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 0b0a1dca..e5b05024 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1024,7 +1024,7 @@ class ArkosExtensions { } }, filter: ['sprite'] - } + }, // y向缩放 { opcode: 'sclaeSpriteY', From 9f6ee850607d30cfab8d4df4a391d6268aab8fab Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 20:57:15 +0800 Subject: [PATCH 131/357] Update projectWith30.js --- Arkos/projectWith30.js | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index e5b05024..69154949 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1014,7 +1014,7 @@ class ArkosExtensions { "---" + this.formatMessage("30Ext.info.1"), //定向缩放 // x向缩放 { - opcode: 'sclaeSpriteX', + opcode: 'scaleSpriteX', blockType: 'command', text: this.formatMessage('30Ext.block.sclaeSpriteX'), arguments: { @@ -1027,7 +1027,7 @@ class ArkosExtensions { }, // y向缩放 { - opcode: 'sclaeSpriteY', + opcode: 'scaleSpriteY', blockType: 'command', text: this.formatMessage('30Ext.block.sclaeSpriteY'), arguments: { @@ -1807,7 +1807,6 @@ class ArkosExtensions { scaleSpriteY(args, util) { this.scaleSprite(1, args.input, util); } - //TODO: 拉伸 } From 90d63d7a12e5c37cb164417e6e82c22545bb25d1 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 20:59:38 +0800 Subject: [PATCH 132/357] Update projectWith30.js --- Arkos/projectWith30.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 69154949..6ec3766b 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -118,8 +118,8 @@ class ArkosExtensions { '30Ext.info': '✨ 以下扩展由_30提供', '30Ext.info.1': '🔮 定向缩放操作', - '30Ext.block.sclaeSpriteX': '水平缩放角色[input]倍', - '30Ext.block.sclaeSpriteY': '垂直缩放角色[input]倍' + '30Ext.block.scaleSpriteX': '水平缩放角色[input]倍', + '30Ext.block.scaleSpriteY': '垂直缩放角色[input]倍' }, en: { @@ -219,8 +219,8 @@ class ArkosExtensions { '30Ext.info': '✨ Contributed by _30', '30Ext.info.1': '🔮 Directional scale', - '30Ext.block.sclaeSpriteX': 'Scale the sprite [input] times horizontally', - '30Ext.block.sclaeSpriteY': 'Scale the sprite [input] times vertically' + '30Ext.block.scaleSpriteX': 'Scale the sprite [input] times horizontally', + '30Ext.block.scaleSpriteY': 'Scale the sprite [input] times vertically' }, }) } @@ -1029,7 +1029,7 @@ class ArkosExtensions { { opcode: 'scaleSpriteY', blockType: 'command', - text: this.formatMessage('30Ext.block.sclaeSpriteY'), + text: this.formatMessage('30Ext.block.scaleSpriteY'), arguments: { input: { type: 'number', From 2bf07ccc9947f271668fd403eb7a470811cc8ef2 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 21:02:06 +0800 Subject: [PATCH 133/357] Update projectWith30.js --- Arkos/projectWith30.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 6ec3766b..d3ec7520 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1016,7 +1016,7 @@ class ArkosExtensions { { opcode: 'scaleSpriteX', blockType: 'command', - text: this.formatMessage('30Ext.block.sclaeSpriteX'), + text: this.formatMessage('30Ext.block.scaleSpriteX'), arguments: { input: { type: 'number', From a10bdbfd58521181ba22daf69894c73926f9f71a Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 21:13:47 +0800 Subject: [PATCH 134/357] Update projectWith30.js --- Arkos/projectWith30.js | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index d3ec7520..d68ce7f1 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -118,8 +118,8 @@ class ArkosExtensions { '30Ext.info': '✨ 以下扩展由_30提供', '30Ext.info.1': '🔮 定向缩放操作', - '30Ext.block.scaleSpriteX': '水平缩放角色[input]倍', - '30Ext.block.scaleSpriteY': '垂直缩放角色[input]倍' + '30Ext.block.scaleSpriteX': '将角色水平缩放比例设为[input](倍)', + '30Ext.block.scaleSpriteY': '将角色垂直缩放比例设为[input](倍)' }, en: { @@ -219,8 +219,8 @@ class ArkosExtensions { '30Ext.info': '✨ Contributed by _30', '30Ext.info.1': '🔮 Directional scale', - '30Ext.block.scaleSpriteX': 'Scale the sprite [input] times horizontally', - '30Ext.block.scaleSpriteY': 'Scale the sprite [input] times vertically' + '30Ext.block.scaleSpriteX': 'Set the horizontal scaling of the sprite to [input] (Times)', + '30Ext.block.scaleSpriteY': 'Set the vertical scaling of the sprite to [input] (Times)' }, }) } @@ -1787,19 +1787,21 @@ class ArkosExtensions { let drawable = this.runtime.renderer._allDrawables[target.drawableID]; if(!drawable.ext30_scale) { drawable.ext30_scale = [1,1]; + drawable.ext30_rawScale = drawable.scale; //注入修改函数 let old_fun = drawable.__proto__.updateScale; Object.defineProperty(drawable, "updateScale" , {value: function(scale) { - scale[0] = Math.abs(scale[0]) * this.ext30_scale[0]; - scale[1] = Math.abs(scale[1]) * this.ext30_scale[1]; + this.ext30_rawScale = scale; + scale[0] = Math.abs(this.ext30_rawScale[0]) * this.ext30_scale[0]; + scale[1] = Math.abs(this.ext30_rawScale[1]) * this.ext30_scale[1]; return old_fun.call(this, scale); }} ); } drawable.ext30_scale[index] = value; //更新 - drawable.updateScale(drawable.scale); + drawable.updateScale(drawable.ext30_rawScale); } scaleSpriteX(args, util) { this.scaleSprite(0, args.input, util); From c5322a42a6d27c833fac631d667e4414581c2616 Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Wed, 3 Aug 2022 21:25:11 +0800 Subject: [PATCH 135/357] Update projectWith30.js --- Arkos/projectWith30.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index d68ce7f1..3e0b75f6 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -1792,16 +1792,16 @@ class ArkosExtensions { let old_fun = drawable.__proto__.updateScale; Object.defineProperty(drawable, "updateScale" , {value: function(scale) { - this.ext30_rawScale = scale; - scale[0] = Math.abs(this.ext30_rawScale[0]) * this.ext30_scale[0]; - scale[1] = Math.abs(this.ext30_rawScale[1]) * this.ext30_scale[1]; + this.ext30_rawSize = scale[0]; + scale[0] = this.ext30_rawSize * this.ext30_scale[0]; + scale[1] = this.ext30_rawSize * this.ext30_scale[1]; return old_fun.call(this, scale); }} ); } drawable.ext30_scale[index] = value; //更新 - drawable.updateScale(drawable.ext30_rawScale); + drawable.updateScale([target.size, target.size]); } scaleSpriteX(args, util) { this.scaleSprite(0, args.input, util); From 37c73268dbad8b922c32ddfc3081d2552ca3950d Mon Sep 17 00:00:00 2001 From: java30433 <110109574+java30433@users.noreply.github.com> Date: Mon, 8 Aug 2022 21:28:46 +0800 Subject: [PATCH 136/357] =?UTF-8?q?=E6=96=B0=E5=A2=9E=E5=9B=BE=E5=B1=82?= =?UTF-8?q?=E7=AE=A1=E7=90=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Arkos/projectWith30.js | 41 ++++++++++++++++++++++++++++++++++++++--- 1 file changed, 38 insertions(+), 3 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 3e0b75f6..8aa6c5ed 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -119,7 +119,10 @@ class ArkosExtensions { '30Ext.info': '✨ 以下扩展由_30提供', '30Ext.info.1': '🔮 定向缩放操作', '30Ext.block.scaleSpriteX': '将角色水平缩放比例设为[input](倍)', - '30Ext.block.scaleSpriteY': '将角色垂直缩放比例设为[input](倍)' + '30Ext.block.scaleSpriteY': '将角色垂直缩放比例设为[input](倍)', + '30Ext.info.2': '图层操作', + '30Ext.block.getLayer': '角色当前图层序数', + '30Ext.block.setLayer': '将角色移到第[input]图层', }, en: { @@ -220,7 +223,10 @@ class ArkosExtensions { '30Ext.info': '✨ Contributed by _30', '30Ext.info.1': '🔮 Directional scale', '30Ext.block.scaleSpriteX': 'Set the horizontal scaling of the sprite to [input] (Times)', - '30Ext.block.scaleSpriteY': 'Set the vertical scaling of the sprite to [input] (Times)' + '30Ext.block.scaleSpriteY': 'Set the vertical scaling of the sprite to [input] (Times)', + '30Ext.info.2': 'Layer Manage', + '30Ext.block.getLayer': 'Current layer of the sprite', + '30Ext.block.setLayer': 'Move the sprite to layer [input]', }, }) } @@ -1037,7 +1043,27 @@ class ArkosExtensions { } }, filter: ['sprite'] - } + }, + "---" + this.formatMessage("30Ext.info.2"), //图层管理 + //获取图层 + { + opcode: 'getLayer', + blockType: 'reporter', + text: this.formatMessage('30Ext.block.getLayer'), + }, + //设置图层 + { + opcode: 'setLayer', + blockType: 'command', + text: this.formatMessage('30Ext.block.setLayer'), + arguments: { + input: { + type: 'number', + defaultValue: '1' + } + }, + filter: ['sprite'] + }, ], menus: { conInfoMenu: [{ @@ -1809,6 +1835,15 @@ class ArkosExtensions { scaleSpriteY(args, util) { this.scaleSprite(1, args.input, util); } + // + //图层操作 + // + getLayer(args, util) { + return util.target.getLayerOrder(); + } + setLayer(args, util) { + util.target.renderer.setDrawableOrder(util.target.drawableID, args.input, 'sprite'); + } } From 2c6d0d8fd5fd1da08a1e6d1ca7501aa1491dd555 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 25 Sep 2022 13:07:14 +0800 Subject: [PATCH 137/357] Add files via upload --- ...07\344\273\266\345\212\251\346\211\213.js" | 249 ++++++++++++++++++ 1 file changed, 249 insertions(+) create mode 100644 "\346\226\207\344\273\266\345\212\251\346\211\213.js" diff --git "a/\346\226\207\344\273\266\345\212\251\346\211\213.js" "b/\346\226\207\344\273\266\345\212\251\346\211\213.js" new file mode 100644 index 00000000..1d9b6614 --- /dev/null +++ "b/\346\226\207\344\273\266\345\212\251\346\211\213.js" @@ -0,0 +1,249 @@ +// import Cast from "cast.js"; + +let _picture = ""; + +let _icon = ""; + +class File_Helper { + constructor(runtime) { + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "File_Helper.name": "文件助手", + "File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", + "File_Helper.upload": "获取键[name]的值", + "File_Helper.save": "将内容[text]命名为[name]并保存", + "File_Helper.delete": "删除键[name]", + "File_Helper.segmentation": "将[text]按[s]分割", + }, + en: { + "File_Helper.name": "File Helper", + "File_Helper.download": "Download split content [text] by [S] named [name]", + "File_Helper.upload": "Get value [name]", + "File_Helper.save": "Save content [text] with [name]", + "File_Helper.delete": "delete value [name]", + "File_Helper.segmentation": "Split [text] by [s]", + } + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: "File_Helper", // 拓展id + name: this.formatMessage("File_Helper.name"), // 拓展名 + // docsURI: "https://www.baidu.com", + blockIconURI: _icon, + menuIconURI: _icon, + color1: "#60D6F4", + color2: "#55a7f7", + blocks: [ + { + opcode: "download", + blockType: "command", + text: this.formatMessage("File_Helper.download"), + arguments: { + text: { + type: "string", + defaultValue: 'awa!!!', + }, + name: { + type: "string", + defaultValue: 'wit_cat.txt', + }, + }, + }, + { + opcode: "upload", + blockType: "reporter", + text: this.formatMessage("File_Helper.upload"), + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "save", + blockType: "command", + text: this.formatMessage("File_Helper.save"), + arguments: { + text: { + type: "string", + defaultValue: '0', + }, + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "segmentation", + blockType: "reporter", + text: this.formatMessage("File_Helper.sasegmentation"), + arguments: { + text: { + type: "string", + defaultValue: 'wow/!!!', + }, + s: { + type: "string", + defaultValue: '/', + }, + }, + }, + ] + }; + } + //下载文件 + download(args){ + const filename = args.name; + var s = args.s; + var a = args.text.split(s); + var h = a[0]; + for(j = 1; j < a.length; j++) { + var h = h + ` +` + a[j]; + } + const content = h; + // 创建隐藏的可下载链接 + var eleLink = document.createElement('a'); + eleLink.download = filename; + eleLink.style.display = 'none'; + // 字符内容转变成blob地址 + var blob = new Blob([content]); + eleLink.href = URL.createObjectURL(blob); + // 触发点击 + document.body.appendChild(eleLink); + eleLink.click(); + // 然后移除 + document.body.removeChild(eleLink); + } + //读取本地变量 + upload(args){ + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + return localStorage.getItem(h+name); + } + //保存本地变量 + save(args){ + const text = args.text; + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + localStorage.setItem(h+name, text); + } + //删除本地变量 + delete(args){ + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + localStorage.setItem(h+name); + } + + segmentation(args){ + var text = args.text; + var s = args.s; + var array = text.split(s); + return array; + } + +} + +window.tempExt = { + Extension: File_Helper, + info: { + name: "File_Helper.name", + description: "File_Helper.descp", + extensionId: "File_Helper", + iconURL: _picture, + insetIconURL: _icon, + featured: true, + disabled: false, + collaborator: "白猫 x CCW" + }, + l10n: { + "zh-cn": { + "File_Helper.name": "文件助手", + "File_Helper.descp": "处理本地数据" + }, + en: { + "File_Helper.name": "File Helper", + "File_Helper.descp": "Handling local data" + } + } +}; + +/* vim: set expandtab tabstop=2 shiftwidth=2: */ \ No newline at end of file From acdc0aa1e627d42043c181c60531d645e3464c0a Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 25 Sep 2022 13:22:07 +0800 Subject: [PATCH 138/357] =?UTF-8?q?Delete=20=E6=96=87=E4=BB=B6=E5=8A=A9?= =?UTF-8?q?=E6=89=8B.js?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...07\344\273\266\345\212\251\346\211\213.js" | 249 ------------------ 1 file changed, 249 deletions(-) delete mode 100644 "\346\226\207\344\273\266\345\212\251\346\211\213.js" diff --git "a/\346\226\207\344\273\266\345\212\251\346\211\213.js" "b/\346\226\207\344\273\266\345\212\251\346\211\213.js" deleted file mode 100644 index 1d9b6614..00000000 --- "a/\346\226\207\344\273\266\345\212\251\346\211\213.js" +++ /dev/null @@ -1,249 +0,0 @@ -// import Cast from "cast.js"; - -let _picture = ""; - -let _icon = ""; - -class File_Helper { - constructor(runtime) { - this._formatMessage = runtime.getFormatMessage({ - "zh-cn": { - "File_Helper.name": "文件助手", - "File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", - "File_Helper.upload": "获取键[name]的值", - "File_Helper.save": "将内容[text]命名为[name]并保存", - "File_Helper.delete": "删除键[name]", - "File_Helper.segmentation": "将[text]按[s]分割", - }, - en: { - "File_Helper.name": "File Helper", - "File_Helper.download": "Download split content [text] by [S] named [name]", - "File_Helper.upload": "Get value [name]", - "File_Helper.save": "Save content [text] with [name]", - "File_Helper.delete": "delete value [name]", - "File_Helper.segmentation": "Split [text] by [s]", - } - }) - } - - formatMessage(id) { - return this._formatMessage({ - id, - default: id, - description: id - }); - } - - getInfo() { - return { - id: "File_Helper", // 拓展id - name: this.formatMessage("File_Helper.name"), // 拓展名 - // docsURI: "https://www.baidu.com", - blockIconURI: _icon, - menuIconURI: _icon, - color1: "#60D6F4", - color2: "#55a7f7", - blocks: [ - { - opcode: "download", - blockType: "command", - text: this.formatMessage("File_Helper.download"), - arguments: { - text: { - type: "string", - defaultValue: 'awa!!!', - }, - name: { - type: "string", - defaultValue: 'wit_cat.txt', - }, - }, - }, - { - opcode: "upload", - blockType: "reporter", - text: this.formatMessage("File_Helper.upload"), - arguments: { - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "save", - blockType: "command", - text: this.formatMessage("File_Helper.save"), - arguments: { - text: { - type: "string", - defaultValue: '0', - }, - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "segmentation", - blockType: "reporter", - text: this.formatMessage("File_Helper.sasegmentation"), - arguments: { - text: { - type: "string", - defaultValue: 'wow/!!!', - }, - s: { - type: "string", - defaultValue: '/', - }, - }, - }, - ] - }; - } - //下载文件 - download(args){ - const filename = args.name; - var s = args.s; - var a = args.text.split(s); - var h = a[0]; - for(j = 1; j < a.length; j++) { - var h = h + ` -` + a[j]; - } - const content = h; - // 创建隐藏的可下载链接 - var eleLink = document.createElement('a'); - eleLink.download = filename; - eleLink.style.display = 'none'; - // 字符内容转变成blob地址 - var blob = new Blob([content]); - eleLink.href = URL.createObjectURL(blob); - // 触发点击 - document.body.appendChild(eleLink); - eleLink.click(); - // 然后移除 - document.body.removeChild(eleLink); - } - //读取本地变量 - upload(args){ - const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ - var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - return localStorage.getItem(h+name); - } - //保存本地变量 - save(args){ - const text = args.text; - const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ - var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - localStorage.setItem(h+name, text); - } - //删除本地变量 - delete(args){ - const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ - var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - localStorage.setItem(h+name); - } - - segmentation(args){ - var text = args.text; - var s = args.s; - var array = text.split(s); - return array; - } - -} - -window.tempExt = { - Extension: File_Helper, - info: { - name: "File_Helper.name", - description: "File_Helper.descp", - extensionId: "File_Helper", - iconURL: _picture, - insetIconURL: _icon, - featured: true, - disabled: false, - collaborator: "白猫 x CCW" - }, - l10n: { - "zh-cn": { - "File_Helper.name": "文件助手", - "File_Helper.descp": "处理本地数据" - }, - en: { - "File_Helper.name": "File Helper", - "File_Helper.descp": "Handling local data" - } - } -}; - -/* vim: set expandtab tabstop=2 shiftwidth=2: */ \ No newline at end of file From 87a3c9dae3070c6824192a83460a62a567911534 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 25 Sep 2022 13:31:20 +0800 Subject: [PATCH 139/357] wit_cat File_Helper ,you can use this to handling local data. --- ...07\344\273\266\345\212\251\346\211\213.js" | 260 ++++++++++++++++++ 1 file changed, 260 insertions(+) create mode 100644 "wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" diff --git "a/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" "b/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" new file mode 100644 index 00000000..eda46902 --- /dev/null +++ "b/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" @@ -0,0 +1,260 @@ +// import Cast from "cast.js"; + +let _picture = ""; + +let _icon = ""; + +class File_Helper { + constructor(runtime) { + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "File_Helper.name": "文件助手", + "File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", + "File_Helper.upload": "获取键[name]的值", + "File_Helper.save": "将内容[text]命名为[name]并保存", + "File_Helper.delete": "删除键[name]", + "File_Helper.segmentation": "将[text]按[s]分割", + }, + en: { + "File_Helper.name": "File Helper", + "File_Helper.download": "Download split content [text] by [S] named [name]", + "File_Helper.upload": "Get value [name]", + "File_Helper.save": "Save content [text] with [name]", + "File_Helper.delete": "delete value [name]", + "File_Helper.segmentation": "Split [text] by [s]", + } + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: "File_Helper", // 拓展id + name: this.formatMessage("File_Helper.name"), // 拓展名 + // docsURI: "https://www.baidu.com", + blockIconURI: _icon, + menuIconURI: _icon, + color1: "#60D6F4", + color2: "#55a7f7", + blocks: [ + { + opcode: "download", + blockType: "command", + text: this.formatMessage("File_Helper.download"), + arguments: { + text: { + type: "string", + defaultValue: 'awa!!!', + }, + name: { + type: "string", + defaultValue: 'wit_cat.txt', + }, + }, + }, + { + opcode: "upload", + blockType: "reporter", + text: this.formatMessage("File_Helper.upload"), + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "save", + blockType: "command", + text: this.formatMessage("File_Helper.save"), + arguments: { + text: { + type: "string", + defaultValue: '0', + }, + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "segmentation", + blockType: "reporter", + text: this.formatMessage("File_Helper.segmentation"), + arguments: { + text: { + type: "string", + defaultValue: 'wow/!!!', + }, + s: { + type: "string", + defaultValue: '/', + }, + }, + }, + { + opcode: "delete", + blockType: "command", + text: this.formatMessage("File_Helper.delete"), + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + ] + }; + } + //下载文件 + download(args){ + const filename = args.name; + var s = args.s; + var a = args.text.split(s); + var h = a[0]; + for(j = 1; j < a.length; j++) { + var h = h + ` +` + a[j]; + } + const content = h; + // 创建隐藏的可下载链接 + var eleLink = document.createElement('a'); + eleLink.download = filename; + eleLink.style.display = 'none'; + // 字符内容转变成blob地址 + var blob = new Blob([content]); + eleLink.href = URL.createObjectURL(blob); + // 触发点击 + document.body.appendChild(eleLink); + eleLink.click(); + // 然后移除 + document.body.removeChild(eleLink); + } + //读取本地变量 + upload(args){ + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + return localStorage.getItem(h+name); + } + //保存本地变量 + save(args){ + const text = args.text; + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + localStorage.setItem(h+name, text); + } + //删除本地变量 + delete(args){ + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + localStorage.removeItem(h+name); + } + + segmentation(args){ + var text = args.text; + var s = args.s; + var array = text.split(s); + return array; + } + +} + +window.tempExt = { + Extension: File_Helper, + info: { + name: "File_Helper.name", + description: "File_Helper.descp", + extensionId: "File_Helper", + iconURL: _picture, + insetIconURL: _icon, + featured: true, + disabled: false, + collaborator: "白猫 x CCW" + }, + l10n: { + "zh-cn": { + "File_Helper.name": "文件助手", + "File_Helper.descp": "处理本地数据" + }, + en: { + "File_Helper.name": "File Helper", + "File_Helper.descp": "Handling local data" + } + } +}; + +/* vim: set expandtab tabstop=2 shiftwidth=2: */ \ No newline at end of file From a4a6faa9332562a99833e115c01f59c8154d11e9 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 25 Sep 2022 15:43:13 +0800 Subject: [PATCH 140/357] =?UTF-8?q?Update=20=E6=96=87=E4=BB=B6=E5=8A=A9?= =?UTF-8?q?=E6=89=8B.js?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...07\344\273\266\345\212\251\346\211\213.js" | 65 ++++++++++++++++--- 1 file changed, 57 insertions(+), 8 deletions(-) diff --git "a/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" "b/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" index eda46902..852de8fc 100644 --- "a/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" +++ "b/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" @@ -14,6 +14,8 @@ class File_Helper { "File_Helper.save": "将内容[text]命名为[name]并保存", "File_Helper.delete": "删除键[name]", "File_Helper.segmentation": "将[text]按[s]分割", + "File_Helper.encrypt": "base64加密[text]", + "File_Helper.decrypt": "base64解密[text]", }, en: { "File_Helper.name": "File Helper", @@ -22,6 +24,8 @@ class File_Helper { "File_Helper.save": "Save content [text] with [name]", "File_Helper.delete": "delete value [name]", "File_Helper.segmentation": "Split [text] by [s]", + "File_Helper.encrypt": "base64 encrypt[text]", + "File_Helper.decrypt": "base64 decrypt[text]", } }) } @@ -57,6 +61,10 @@ class File_Helper { type: "string", defaultValue: 'wit_cat.txt', }, + s: { + type: "string", + defaultValue: '', + }, }, }, { @@ -110,20 +118,48 @@ class File_Helper { defaultValue: 'i', }, }, + }, + { + opcode: "encrypt", + blockType: "reporter", + text: this.formatMessage("File_Helper.encrypt"), + arguments: { + text: { + type: "string", + defaultValue: 'i love中国', + }, + }, + }, + { + opcode: "decrypt", + blockType: "reporter", + text: this.formatMessage("File_Helper.decrypt"), + arguments: { + text: { + type: "string", + defaultValue: 'aSUyMGxvdmUlRTQlQjglQUQlRTUlOUIlQkQ=', + }, + }, }, ] }; } //下载文件 download(args){ + var text = args.text; const filename = args.name; var s = args.s; - var a = args.text.split(s); - var h = a[0]; - for(j = 1; j < a.length; j++) { - var h = h + ` + if(s != ""){ + var a = text.split(s); + var h = a[0]; + for(j = 1; j < a.length; j++) { + var h = h + ` ` + a[j]; - } + } + } + else{ + var h = text; + } const content = h; // 创建隐藏的可下载链接 var eleLink = document.createElement('a'); @@ -223,14 +259,27 @@ class File_Helper { } localStorage.removeItem(h+name); } - + //字符串分割 segmentation(args){ var text = args.text; var s = args.s; var array = text.split(s); return array; } - + //加密 + encrypt(args){ + var str = args.text; + var jiaMi = encodeURIComponent(str); + var jiaM = btoa(jiaMi); + return jiaM; + } + //解密 + decrypt(args){ + var jiaM = args.text; + var jieMi = atob(jiaM); + var jieM = decodeURIComponent(jieMi); + return jieM; + } } window.tempExt = { @@ -257,4 +306,4 @@ window.tempExt = { } }; -/* vim: set expandtab tabstop=2 shiftwidth=2: */ \ No newline at end of file +/* vim: set expandtab tabstop=2 shiftwidth=2: */ From c959d199aa51455193e99c640447d52dafb83f51 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 25 Sep 2022 20:24:50 +0800 Subject: [PATCH 141/357] =?UTF-8?q?Delete=20=E6=96=87=E4=BB=B6=E5=8A=A9?= =?UTF-8?q?=E6=89=8B.js?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...07\344\273\266\345\212\251\346\211\213.js" | 309 ------------------ 1 file changed, 309 deletions(-) delete mode 100644 "wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" diff --git "a/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" "b/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" deleted file mode 100644 index 852de8fc..00000000 --- "a/wit_cat/\346\226\207\344\273\266\345\212\251\346\211\213.js" +++ /dev/null @@ -1,309 +0,0 @@ -// import Cast from "cast.js"; - -let _picture = ""; - -let _icon = ""; - -class File_Helper { - constructor(runtime) { - this._formatMessage = runtime.getFormatMessage({ - "zh-cn": { - "File_Helper.name": "文件助手", - "File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", - "File_Helper.upload": "获取键[name]的值", - "File_Helper.save": "将内容[text]命名为[name]并保存", - "File_Helper.delete": "删除键[name]", - "File_Helper.segmentation": "将[text]按[s]分割", - "File_Helper.encrypt": "base64加密[text]", - "File_Helper.decrypt": "base64解密[text]", - }, - en: { - "File_Helper.name": "File Helper", - "File_Helper.download": "Download split content [text] by [S] named [name]", - "File_Helper.upload": "Get value [name]", - "File_Helper.save": "Save content [text] with [name]", - "File_Helper.delete": "delete value [name]", - "File_Helper.segmentation": "Split [text] by [s]", - "File_Helper.encrypt": "base64 encrypt[text]", - "File_Helper.decrypt": "base64 decrypt[text]", - } - }) - } - - formatMessage(id) { - return this._formatMessage({ - id, - default: id, - description: id - }); - } - - getInfo() { - return { - id: "File_Helper", // 拓展id - name: this.formatMessage("File_Helper.name"), // 拓展名 - // docsURI: "https://www.baidu.com", - blockIconURI: _icon, - menuIconURI: _icon, - color1: "#60D6F4", - color2: "#55a7f7", - blocks: [ - { - opcode: "download", - blockType: "command", - text: this.formatMessage("File_Helper.download"), - arguments: { - text: { - type: "string", - defaultValue: 'awa!!!', - }, - name: { - type: "string", - defaultValue: 'wit_cat.txt', - }, - s: { - type: "string", - defaultValue: '', - }, - }, - }, - { - opcode: "upload", - blockType: "reporter", - text: this.formatMessage("File_Helper.upload"), - arguments: { - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "save", - blockType: "command", - text: this.formatMessage("File_Helper.save"), - arguments: { - text: { - type: "string", - defaultValue: '0', - }, - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "segmentation", - blockType: "reporter", - text: this.formatMessage("File_Helper.segmentation"), - arguments: { - text: { - type: "string", - defaultValue: 'wow/!!!', - }, - s: { - type: "string", - defaultValue: '/', - }, - }, - }, - { - opcode: "delete", - blockType: "command", - text: this.formatMessage("File_Helper.delete"), - arguments: { - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "encrypt", - blockType: "reporter", - text: this.formatMessage("File_Helper.encrypt"), - arguments: { - text: { - type: "string", - defaultValue: 'i love中国', - }, - }, - }, - { - opcode: "decrypt", - blockType: "reporter", - text: this.formatMessage("File_Helper.decrypt"), - arguments: { - text: { - type: "string", - defaultValue: 'aSUyMGxvdmUlRTQlQjglQUQlRTUlOUIlQkQ=', - }, - }, - }, - ] - }; - } - //下载文件 - download(args){ - var text = args.text; - const filename = args.name; - var s = args.s; - if(s != ""){ - var a = text.split(s); - var h = a[0]; - for(j = 1; j < a.length; j++) { - var h = h + ` -` + a[j]; - } - } - else{ - var h = text; - } - const content = h; - // 创建隐藏的可下载链接 - var eleLink = document.createElement('a'); - eleLink.download = filename; - eleLink.style.display = 'none'; - // 字符内容转变成blob地址 - var blob = new Blob([content]); - eleLink.href = URL.createObjectURL(blob); - // 触发点击 - document.body.appendChild(eleLink); - eleLink.click(); - // 然后移除 - document.body.removeChild(eleLink); - } - //读取本地变量 - upload(args){ - const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ - var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - return localStorage.getItem(h+name); - } - //保存本地变量 - save(args){ - const text = args.text; - const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ - var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - localStorage.setItem(h+name, text); - } - //删除本地变量 - delete(args){ - const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ - var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } - else{ - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - localStorage.removeItem(h+name); - } - //字符串分割 - segmentation(args){ - var text = args.text; - var s = args.s; - var array = text.split(s); - return array; - } - //加密 - encrypt(args){ - var str = args.text; - var jiaMi = encodeURIComponent(str); - var jiaM = btoa(jiaMi); - return jiaM; - } - //解密 - decrypt(args){ - var jiaM = args.text; - var jieMi = atob(jiaM); - var jieM = decodeURIComponent(jieMi); - return jieM; - } -} - -window.tempExt = { - Extension: File_Helper, - info: { - name: "File_Helper.name", - description: "File_Helper.descp", - extensionId: "File_Helper", - iconURL: _picture, - insetIconURL: _icon, - featured: true, - disabled: false, - collaborator: "白猫 x CCW" - }, - l10n: { - "zh-cn": { - "File_Helper.name": "文件助手", - "File_Helper.descp": "处理本地数据" - }, - en: { - "File_Helper.name": "File Helper", - "File_Helper.descp": "Handling local data" - } - } -}; - -/* vim: set expandtab tabstop=2 shiftwidth=2: */ From 12977e25ed707156e1cca63a9cd87865793ccfe9 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 25 Sep 2022 20:26:53 +0800 Subject: [PATCH 142/357] File Helper you can use this to handling local data --- wit_cat/File Helper.js | 309 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 309 insertions(+) create mode 100644 wit_cat/File Helper.js diff --git a/wit_cat/File Helper.js b/wit_cat/File Helper.js new file mode 100644 index 00000000..083a47db --- /dev/null +++ b/wit_cat/File Helper.js @@ -0,0 +1,309 @@ +// import Cast from "cast.js"; + +let _picture = ""; + +let _icon = ""; + +class File_Helper { + constructor(runtime) { + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "File_Helper.name": "文件助手", + "File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", + "File_Helper.upload": "获取键[name]的值", + "File_Helper.save": "将内容[text]命名为[name]并保存", + "File_Helper.delete": "删除键[name]", + "File_Helper.segmentation": "将[text]按[s]分割", + "File_Helper.encrypt": "base64加密[text]", + "File_Helper.decrypt": "base64解密[text]", + }, + en: { + "File_Helper.name": "File Helper", + "File_Helper.download": "Download split content [text] by [S] named [name]", + "File_Helper.upload": "Get value [name]", + "File_Helper.save": "Save content [text] with [name]", + "File_Helper.delete": "delete value [name]", + "File_Helper.segmentation": "Split [text] by [s]", + "File_Helper.encrypt": "base64 encrypt[text]", + "File_Helper.decrypt": "base64 decrypt[text]", + } + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: "Wit_Cat_File_Helper", // 拓展id + name: this.formatMessage("File_Helper.name"), // 拓展名 + // docsURI: "https://www.baidu.com", + blockIconURI: _icon, + menuIconURI: _icon, + color1: "#60D6F4", + color2: "#55a7f7", + blocks: [ + { + opcode: "download", + blockType: "command", + text: this.formatMessage("File_Helper.download"), + arguments: { + text: { + type: "string", + defaultValue: 'awa!!!', + }, + name: { + type: "string", + defaultValue: 'wit_cat.txt', + }, + s: { + type: "string", + defaultValue: '', + }, + }, + }, + { + opcode: "upload", + blockType: "reporter", + text: this.formatMessage("File_Helper.upload"), + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "save", + blockType: "command", + text: this.formatMessage("File_Helper.save"), + arguments: { + text: { + type: "string", + defaultValue: '0', + }, + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "segmentation", + blockType: "reporter", + text: this.formatMessage("File_Helper.segmentation"), + arguments: { + text: { + type: "string", + defaultValue: 'wow/!!!', + }, + s: { + type: "string", + defaultValue: '/', + }, + }, + }, + { + opcode: "delete", + blockType: "command", + text: this.formatMessage("File_Helper.delete"), + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "encrypt", + blockType: "reporter", + text: this.formatMessage("File_Helper.encrypt"), + arguments: { + text: { + type: "string", + defaultValue: 'i love中国', + }, + }, + }, + { + opcode: "decrypt", + blockType: "reporter", + text: this.formatMessage("File_Helper.decrypt"), + arguments: { + text: { + type: "string", + defaultValue: 'aSUyMGxvdmUlRTQlQjglQUQlRTUlOUIlQkQ=', + }, + }, + }, + ] + }; + } + //下载文件 + download(args){ + var text = args.text; + const filename = args.name; + var s = args.s; + if(s != ""){ + var a = text.split(s); + var h = a[0]; + for(j = 1; j < a.length; j++) { + var h = h + ` +` + a[j]; + } + } + else{ + var h = text; + } + const content = h; + // 创建隐藏的可下载链接 + var eleLink = document.createElement('a'); + eleLink.download = filename; + eleLink.style.display = 'none'; + // 字符内容转变成blob地址 + var blob = new Blob([content]); + eleLink.href = URL.createObjectURL(blob); + // 触发点击 + document.body.appendChild(eleLink); + eleLink.click(); + // 然后移除 + document.body.removeChild(eleLink); + } + //读取本地变量 + upload(args){ + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + return localStorage.getItem(h+name); + } + //保存本地变量 + save(args){ + const text = args.text; + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + localStorage.setItem(h+name, text); + } + //删除本地变量 + delete(args){ + const name = args.name; + var h = window.location.href; + console.log(h); + var array = h.split("/"); + var indexOf4 = array.findIndex((item) => item == 'detail'); + if(indexOf4 == -1){ + var indexOf4 = array.findIndex((item) => item == 'project'); + if(indexOf4 == -1){ + var array = h.split("?"); + var h = array[1]; + var array = h.split("="); + var h = array[1]; + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + } + else{ + var h = array[indexOf4 + 1]; + var array = h.split("?"); + var h = array[0]; + } + localStorage.removeItem(h+name); + } + //字符串分割 + segmentation(args){ + var text = args.text; + var s = args.s; + var array = text.split(s); + return array; + } + //加密 + encrypt(args){ + var str = args.text; + var jiaMi = encodeURIComponent(str); + var jiaM = btoa(jiaMi); + return jiaM; + } + //解密 + decrypt(args){ + var jiaM = args.text; + var jieMi = atob(jiaM); + var jieM = decodeURIComponent(jieMi); + return jieM; + } +} + +window.tempExt = { + Extension: File_Helper, + info: { + name: "File_Helper.name", + description: "File_Helper.descp", + extensionId: "File_Helper", + iconURL: _picture, + insetIconURL: _icon, + featured: true, + disabled: false, + collaborator: "白猫 @ CCW" + }, + l10n: { + "zh-cn": { + "File_Helper.name": "文件助手", + "File_Helper.descp": "处理本地数据" + }, + en: { + "File_Helper.name": "File Helper", + "File_Helper.descp": "Handling local data" + } + } +}; + +/* vim: set expandtab tabstop=2 shiftwidth=2: */ \ No newline at end of file From 7c8badf973dcd7305627e3e0ab9e53f85f509c00 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 25 Sep 2022 23:14:12 +0800 Subject: [PATCH 143/357] Update File Helper.js --- wit_cat/File Helper.js | 386 ++++++++++++++++++++--------------------- 1 file changed, 189 insertions(+), 197 deletions(-) diff --git a/wit_cat/File Helper.js b/wit_cat/File Helper.js index 083a47db..a638425e 100644 --- a/wit_cat/File Helper.js +++ b/wit_cat/File Helper.js @@ -4,160 +4,158 @@ let _picture = " let _icon = ""; -class File_Helper { - constructor(runtime) { - this._formatMessage = runtime.getFormatMessage({ - "zh-cn": { - "File_Helper.name": "文件助手", - "File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", - "File_Helper.upload": "获取键[name]的值", - "File_Helper.save": "将内容[text]命名为[name]并保存", - "File_Helper.delete": "删除键[name]", - "File_Helper.segmentation": "将[text]按[s]分割", - "File_Helper.encrypt": "base64加密[text]", - "File_Helper.decrypt": "base64解密[text]", - }, - en: { - "File_Helper.name": "File Helper", - "File_Helper.download": "Download split content [text] by [S] named [name]", - "File_Helper.upload": "Get value [name]", - "File_Helper.save": "Save content [text] with [name]", - "File_Helper.delete": "delete value [name]", - "File_Helper.segmentation": "Split [text] by [s]", - "File_Helper.encrypt": "base64 encrypt[text]", - "File_Helper.decrypt": "base64 decrypt[text]", - } - }) - } +class Wit_Cat_File_Helper { + constructor(runtime) { + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "File_Helper.name": "文件助手", + "File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", + "File_Helper.upload": "获取键[name]的值", + "File_Helper.save": "将内容[text]命名为[name]并保存", + "File_Helper.delete": "删除键[name]", + "File_Helper.segmentation": "将[text]按[s]分割", + "File_Helper.encrypt": "base64加密[text]", + "File_Helper.decrypt": "base64解密[text]", + }, + en: { + "File_Helper.name": "File Helper", + "File_Helper.download": "Download split content [text] by [S] named [name]", + "File_Helper.upload": "Get value [name]", + "File_Helper.save": "Save content [text] with [name]", + "File_Helper.delete": "delete value [name]", + "File_Helper.segmentation": "Split [text] by [s]", + "File_Helper.encrypt": "base64 encrypt[text]", + "File_Helper.decrypt": "base64 decrypt[text]", + } + }) + } - formatMessage(id) { - return this._formatMessage({ - id, - default: id, - description: id - }); - } + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } - getInfo() { - return { - id: "Wit_Cat_File_Helper", // 拓展id - name: this.formatMessage("File_Helper.name"), // 拓展名 - // docsURI: "https://www.baidu.com", - blockIconURI: _icon, - menuIconURI: _icon, - color1: "#60D6F4", - color2: "#55a7f7", - blocks: [ - { - opcode: "download", - blockType: "command", - text: this.formatMessage("File_Helper.download"), - arguments: { - text: { - type: "string", - defaultValue: 'awa!!!', - }, - name: { - type: "string", - defaultValue: 'wit_cat.txt', - }, - s: { - type: "string", - defaultValue: '', - }, - }, - }, - { - opcode: "upload", - blockType: "reporter", - text: this.formatMessage("File_Helper.upload"), - arguments: { - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "save", - blockType: "command", - text: this.formatMessage("File_Helper.save"), - arguments: { - text: { - type: "string", - defaultValue: '0', - }, - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "segmentation", - blockType: "reporter", - text: this.formatMessage("File_Helper.segmentation"), - arguments: { - text: { - type: "string", - defaultValue: 'wow/!!!', - }, - s: { - type: "string", - defaultValue: '/', - }, - }, - }, - { - opcode: "delete", - blockType: "command", - text: this.formatMessage("File_Helper.delete"), - arguments: { - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "encrypt", - blockType: "reporter", - text: this.formatMessage("File_Helper.encrypt"), - arguments: { - text: { - type: "string", - defaultValue: 'i love中国', - }, - }, - }, - { - opcode: "decrypt", - blockType: "reporter", - text: this.formatMessage("File_Helper.decrypt"), - arguments: { - text: { - type: "string", - defaultValue: 'aSUyMGxvdmUlRTQlQjglQUQlRTUlOUIlQkQ=', - }, - }, - }, - ] - }; - } + getInfo() { + return { + id: "Wit_Cat_File_Helper", // 拓展id + name: this.formatMessage("File_Helper.name"), // 拓展名 + // docsURI: "https://www.baidu.com", + blockIconURI: _icon, + menuIconURI: _icon, + color1: "#60D6F4", + color2: "#55a7f7", + blocks: [{ + opcode: "download", + blockType: "command", + text: this.formatMessage("File_Helper.download"), + arguments: { + text: { + type: "string", + defaultValue: 'awa!!!', + }, + name: { + type: "string", + defaultValue: 'wit_cat.txt', + }, + s: { + type: "string", + defaultValue: '', + }, + }, + }, + { + opcode: "upload", + blockType: "reporter", + text: this.formatMessage("File_Helper.upload"), + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "save", + blockType: "command", + text: this.formatMessage("File_Helper.save"), + arguments: { + text: { + type: "string", + defaultValue: '0', + }, + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "segmentation", + blockType: "reporter", + text: this.formatMessage("File_Helper.segmentation"), + arguments: { + text: { + type: "string", + defaultValue: 'wow/!!!', + }, + s: { + type: "string", + defaultValue: '/', + }, + }, + }, + { + opcode: "delete", + blockType: "command", + text: this.formatMessage("File_Helper.delete"), + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "encrypt", + blockType: "reporter", + text: this.formatMessage("File_Helper.encrypt"), + arguments: { + text: { + type: "string", + defaultValue: 'i love中国', + }, + }, + }, + { + opcode: "decrypt", + blockType: "reporter", + text: this.formatMessage("File_Helper.decrypt"), + arguments: { + text: { + type: "string", + defaultValue: 'aSUyMGxvdmUlRTQlQjglQUQlRTUlOUIlQkQ=', + }, + }, + }, + ] + }; + } //下载文件 - download(args){ + download(args) { var text = args.text; const filename = args.name; var s = args.s; - if(s != ""){ + if (s != "") { var a = text.split(s); var h = a[0]; - for(j = 1; j < a.length; j++) { + for (j = 1; j < a.length; j++) { var h = h + ` -` + a[j]; - } - } - else{ +` + a[j]; //不能对齐格式,因为用了``来台行 + } + } else { var h = text; } const content = h; @@ -175,106 +173,100 @@ class File_Helper { document.body.removeChild(eleLink); } //读取本地变量 - upload(args){ + upload(args) { const name = args.name; var h = window.location.href; console.log(h); var array = h.split("/"); var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ + if (indexOf4 == -1) { var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ + if (indexOf4 == -1) { var array = h.split("?"); var h = array[1]; var array = h.split("="); var h = array[1]; - } - else{ + } else { var h = array[indexOf4 + 1]; var array = h.split("?"); var h = array[0]; - } - } - else{ + } + } else { var h = array[indexOf4 + 1]; var array = h.split("?"); var h = array[0]; } - return localStorage.getItem(h+name); + return localStorage.getItem(h + name); } //保存本地变量 - save(args){ + save(args) { const text = args.text; const name = args.name; var h = window.location.href; console.log(h); var array = h.split("/"); var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ + if (indexOf4 == -1) { var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ + if (indexOf4 == -1) { var array = h.split("?"); var h = array[1]; var array = h.split("="); var h = array[1]; - } - else{ + } else { var h = array[indexOf4 + 1]; var array = h.split("?"); var h = array[0]; - } - } - else{ + } + } else { var h = array[indexOf4 + 1]; var array = h.split("?"); var h = array[0]; } - localStorage.setItem(h+name, text); + localStorage.setItem(h + name, text); } //删除本地变量 - delete(args){ + delete(args) { const name = args.name; var h = window.location.href; console.log(h); var array = h.split("/"); var indexOf4 = array.findIndex((item) => item == 'detail'); - if(indexOf4 == -1){ + if (indexOf4 == -1) { var indexOf4 = array.findIndex((item) => item == 'project'); - if(indexOf4 == -1){ + if (indexOf4 == -1) { var array = h.split("?"); var h = array[1]; var array = h.split("="); var h = array[1]; - } - else{ + } else { var h = array[indexOf4 + 1]; var array = h.split("?"); var h = array[0]; - } - } - else{ + } + } else { var h = array[indexOf4 + 1]; var array = h.split("?"); var h = array[0]; } - localStorage.removeItem(h+name); + localStorage.removeItem(h + name); } //字符串分割 - segmentation(args){ + segmentation(args) { var text = args.text; var s = args.s; var array = text.split(s); return array; } //加密 - encrypt(args){ + encrypt(args) { var str = args.text; var jiaMi = encodeURIComponent(str); var jiaM = btoa(jiaMi); return jiaM; } //解密 - decrypt(args){ + decrypt(args) { var jiaM = args.text; var jieMi = atob(jiaM); var jieM = decodeURIComponent(jieMi); @@ -283,27 +275,27 @@ class File_Helper { } window.tempExt = { - Extension: File_Helper, - info: { - name: "File_Helper.name", - description: "File_Helper.descp", - extensionId: "File_Helper", - iconURL: _picture, - insetIconURL: _icon, - featured: true, - disabled: false, - collaborator: "白猫 @ CCW" - }, - l10n: { - "zh-cn": { - "File_Helper.name": "文件助手", - "File_Helper.descp": "处理本地数据" - }, - en: { - "File_Helper.name": "File Helper", - "File_Helper.descp": "Handling local data" - } - } + Extension: File_Helper, + info: { + name: "File_Helper.name", + description: "File_Helper.descp", + extensionId: "File_Helper", + iconURL: _picture, + insetIconURL: _icon, + featured: true, + disabled: false, + collaborator: "白猫 @ CCW" + }, + l10n: { + "zh-cn": { + "File_Helper.name": "文件助手", + "File_Helper.descp": "处理本地数据" + }, + en: { + "File_Helper.name": "File Helper", + "File_Helper.descp": "Handling local data" + } + } }; -/* vim: set expandtab tabstop=2 shiftwidth=2: */ \ No newline at end of file +/* vim: set expandtab tabstop=2 shiftwidth=2: */ From b61ae415484afb82bae9fcd699e2a8e3dfa16fc4 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 27 Sep 2022 14:34:31 +0800 Subject: [PATCH 144/357] Update File Helper.js --- wit_cat/File Helper.js | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/wit_cat/File Helper.js b/wit_cat/File Helper.js index a638425e..2de84745 100644 --- a/wit_cat/File Helper.js +++ b/wit_cat/File Helper.js @@ -152,8 +152,7 @@ class Wit_Cat_File_Helper { var a = text.split(s); var h = a[0]; for (j = 1; j < a.length; j++) { - var h = h + ` -` + a[j]; //不能对齐格式,因为用了``来台行 + var h = h + "\n" + a[j]; } } else { var h = text; From c504d1207addc0a207b7333ad5bbb7ab8be8f908 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 27 Sep 2022 17:01:27 +0800 Subject: [PATCH 145/357] Update File Helper.js --- wit_cat/File Helper.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wit_cat/File Helper.js b/wit_cat/File Helper.js index 2de84745..31b7a5d7 100644 --- a/wit_cat/File Helper.js +++ b/wit_cat/File Helper.js @@ -274,7 +274,7 @@ class Wit_Cat_File_Helper { } window.tempExt = { - Extension: File_Helper, + Extension: Wit_Cat_File_Helper, info: { name: "File_Helper.name", description: "File_Helper.descp", From 1265a297ee4d5a40bc7344153c270cfb14bc763f Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 27 Sep 2022 21:06:02 +0800 Subject: [PATCH 146/357] Update and rename File Helper.js to File_Helper.js --- wit_cat/{File Helper.js => File_Helper.js} | 68 +++++++++++----------- 1 file changed, 34 insertions(+), 34 deletions(-) rename wit_cat/{File Helper.js => File_Helper.js} (92%) diff --git a/wit_cat/File Helper.js b/wit_cat/File_Helper.js similarity index 92% rename from wit_cat/File Helper.js rename to wit_cat/File_Helper.js index 31b7a5d7..bd92bf3c 100644 --- a/wit_cat/File Helper.js +++ b/wit_cat/File_Helper.js @@ -4,28 +4,28 @@ let _picture = " let _icon = ""; -class Wit_Cat_File_Helper { +class Wit_Cat_Wit_Cat_File_Helper { constructor(runtime) { this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "File_Helper.name": "文件助手", - "File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", - "File_Helper.upload": "获取键[name]的值", - "File_Helper.save": "将内容[text]命名为[name]并保存", - "File_Helper.delete": "删除键[name]", - "File_Helper.segmentation": "将[text]按[s]分割", - "File_Helper.encrypt": "base64加密[text]", - "File_Helper.decrypt": "base64解密[text]", + "Wit_Cat_File_Helper.name": "文件助手", + "Wit_Cat_File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", + "Wit_Cat_File_Helper.upload": "获取键[name]的值", + "Wit_Cat_File_Helper.save": "将内容[text]命名为[name]并保存", + "Wit_Cat_File_Helper.delete": "删除键[name]", + "Wit_Cat_File_Helper.segmentation": "将[text]按[s]分割", + "Wit_Cat_File_Helper.encrypt": "base64加密[text]", + "Wit_Cat_File_Helper.decrypt": "base64解密[text]", }, en: { - "File_Helper.name": "File Helper", - "File_Helper.download": "Download split content [text] by [S] named [name]", - "File_Helper.upload": "Get value [name]", - "File_Helper.save": "Save content [text] with [name]", - "File_Helper.delete": "delete value [name]", - "File_Helper.segmentation": "Split [text] by [s]", - "File_Helper.encrypt": "base64 encrypt[text]", - "File_Helper.decrypt": "base64 decrypt[text]", + "Wit_Cat_File_Helper.name": "File Helper", + "Wit_Cat_File_Helper.download": "Download split content [text] by [S] named [name]", + "Wit_Cat_File_Helper.upload": "Get value [name]", + "Wit_Cat_File_Helper.save": "Save content [text] with [name]", + "Wit_Cat_File_Helper.delete": "delete value [name]", + "Wit_Cat_File_Helper.segmentation": "Split [text] by [s]", + "Wit_Cat_File_Helper.encrypt": "base64 encrypt[text]", + "Wit_Cat_File_Helper.decrypt": "base64 decrypt[text]", } }) } @@ -40,8 +40,8 @@ class Wit_Cat_File_Helper { getInfo() { return { - id: "Wit_Cat_File_Helper", // 拓展id - name: this.formatMessage("File_Helper.name"), // 拓展名 + id: "Wit_Cat_Wit_Cat_File_Helper", // 拓展id + name: this.formatMessage("Wit_Cat_File_Helper.name"), // 拓展名 // docsURI: "https://www.baidu.com", blockIconURI: _icon, menuIconURI: _icon, @@ -50,7 +50,7 @@ class Wit_Cat_File_Helper { blocks: [{ opcode: "download", blockType: "command", - text: this.formatMessage("File_Helper.download"), + text: this.formatMessage("Wit_Cat_File_Helper.download"), arguments: { text: { type: "string", @@ -69,7 +69,7 @@ class Wit_Cat_File_Helper { { opcode: "upload", blockType: "reporter", - text: this.formatMessage("File_Helper.upload"), + text: this.formatMessage("Wit_Cat_File_Helper.upload"), arguments: { name: { type: "string", @@ -80,7 +80,7 @@ class Wit_Cat_File_Helper { { opcode: "save", blockType: "command", - text: this.formatMessage("File_Helper.save"), + text: this.formatMessage("Wit_Cat_File_Helper.save"), arguments: { text: { type: "string", @@ -95,7 +95,7 @@ class Wit_Cat_File_Helper { { opcode: "segmentation", blockType: "reporter", - text: this.formatMessage("File_Helper.segmentation"), + text: this.formatMessage("Wit_Cat_File_Helper.segmentation"), arguments: { text: { type: "string", @@ -110,7 +110,7 @@ class Wit_Cat_File_Helper { { opcode: "delete", blockType: "command", - text: this.formatMessage("File_Helper.delete"), + text: this.formatMessage("Wit_Cat_File_Helper.delete"), arguments: { name: { type: "string", @@ -121,7 +121,7 @@ class Wit_Cat_File_Helper { { opcode: "encrypt", blockType: "reporter", - text: this.formatMessage("File_Helper.encrypt"), + text: this.formatMessage("Wit_Cat_File_Helper.encrypt"), arguments: { text: { type: "string", @@ -132,7 +132,7 @@ class Wit_Cat_File_Helper { { opcode: "decrypt", blockType: "reporter", - text: this.formatMessage("File_Helper.decrypt"), + text: this.formatMessage("Wit_Cat_File_Helper.decrypt"), arguments: { text: { type: "string", @@ -274,11 +274,11 @@ class Wit_Cat_File_Helper { } window.tempExt = { - Extension: Wit_Cat_File_Helper, + Extension: Wit_Cat_Wit_Cat_File_Helper, info: { - name: "File_Helper.name", - description: "File_Helper.descp", - extensionId: "File_Helper", + name: "Wit_Cat_File_Helper.name", + description: "Wit_Cat_File_Helper.descp", + extensionId: "Wit_Cat_File_Helper", iconURL: _picture, insetIconURL: _icon, featured: true, @@ -287,12 +287,12 @@ window.tempExt = { }, l10n: { "zh-cn": { - "File_Helper.name": "文件助手", - "File_Helper.descp": "处理本地数据" + "Wit_Cat_File_Helper.name": "文件助手", + "Wit_Cat_File_Helper.descp": "处理本地数据" }, en: { - "File_Helper.name": "File Helper", - "File_Helper.descp": "Handling local data" + "Wit_Cat_File_Helper.name": "File Helper", + "Wit_Cat_File_Helper.descp": "Handling local data" } } }; From 6ee4c4a22966fa8ba7244dd04ea4ecea2006a286 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 27 Sep 2022 22:01:22 +0800 Subject: [PATCH 147/357] Update File_Helper.js --- wit_cat/File_Helper.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index bd92bf3c..971085e1 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -4,7 +4,7 @@ let _picture = " let _icon = ""; -class Wit_Cat_Wit_Cat_File_Helper { +class Wit_Cat_File_Helper { constructor(runtime) { this._formatMessage = runtime.getFormatMessage({ "zh-cn": { @@ -274,7 +274,7 @@ class Wit_Cat_Wit_Cat_File_Helper { } window.tempExt = { - Extension: Wit_Cat_Wit_Cat_File_Helper, + Extension: Wit_Cat_File_Helper, info: { name: "Wit_Cat_File_Helper.name", description: "Wit_Cat_File_Helper.descp", From 148e8a8e9e140edffe47fca433bfb50ea1b3b8a4 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 27 Sep 2022 22:16:26 +0800 Subject: [PATCH 148/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E4=BA=86ID=E7=9A=84B?= =?UTF-8?q?UG=E3=80=81=E3=80=81=E3=80=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- wit_cat/File_Helper.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 971085e1..8612a9e4 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -40,7 +40,7 @@ class Wit_Cat_File_Helper { getInfo() { return { - id: "Wit_Cat_Wit_Cat_File_Helper", // 拓展id + id: "Wit_Cat_File_Helper", // 拓展id name: this.formatMessage("Wit_Cat_File_Helper.name"), // 拓展名 // docsURI: "https://www.baidu.com", blockIconURI: _icon, From 12573464eb3ec48e5c0f38ba6f4415e4dfe65a5f Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 27 Sep 2022 23:09:12 +0800 Subject: [PATCH 149/357] =?UTF-8?q?=E6=9B=B4=E6=96=B0=E6=8B=93=E5=B1=95ID?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- wit_cat/File_Helper.js | 70 ++++++++++++++++++++++-------------------- 1 file changed, 36 insertions(+), 34 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 8612a9e4..8d1ae63e 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -4,28 +4,30 @@ let _picture = " let _icon = ""; -class Wit_Cat_File_Helper { +const extensionId = "WitCatFileHelper"; + +class WitCatFileHelper { constructor(runtime) { this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "Wit_Cat_File_Helper.name": "文件助手", - "Wit_Cat_File_Helper.download": "将内容[text]按[s]分割后命名为[name]并下载", - "Wit_Cat_File_Helper.upload": "获取键[name]的值", - "Wit_Cat_File_Helper.save": "将内容[text]命名为[name]并保存", - "Wit_Cat_File_Helper.delete": "删除键[name]", - "Wit_Cat_File_Helper.segmentation": "将[text]按[s]分割", - "Wit_Cat_File_Helper.encrypt": "base64加密[text]", - "Wit_Cat_File_Helper.decrypt": "base64解密[text]", + "WitCatFileHelper.name": "文件助手", + "WitCatFileHelper.download": "将内容[text]按[s]分割后命名为[name]并下载", + "WitCatFileHelper.upload": "获取键[name]的值", + "WitCatFileHelper.save": "将内容[text]命名为[name]并保存", + "WitCatFileHelper.delete": "删除键[name]", + "WitCatFileHelper.segmentation": "将[text]按[s]分割", + "WitCatFileHelper.encrypt": "base64加密[text]", + "WitCatFileHelper.decrypt": "base64解密[text]", }, en: { - "Wit_Cat_File_Helper.name": "File Helper", - "Wit_Cat_File_Helper.download": "Download split content [text] by [S] named [name]", - "Wit_Cat_File_Helper.upload": "Get value [name]", - "Wit_Cat_File_Helper.save": "Save content [text] with [name]", - "Wit_Cat_File_Helper.delete": "delete value [name]", - "Wit_Cat_File_Helper.segmentation": "Split [text] by [s]", - "Wit_Cat_File_Helper.encrypt": "base64 encrypt[text]", - "Wit_Cat_File_Helper.decrypt": "base64 decrypt[text]", + "WitCatFileHelper.name": "File Helper", + "WitCatFileHelper.download": "Download split content [text] by [S] named [name]", + "WitCatFileHelper.upload": "Get value [name]", + "WitCatFileHelper.save": "Save content [text] with [name]", + "WitCatFileHelper.delete": "delete value [name]", + "WitCatFileHelper.segmentation": "Split [text] by [s]", + "WitCatFileHelper.encrypt": "base64 encrypt[text]", + "WitCatFileHelper.decrypt": "base64 decrypt[text]", } }) } @@ -40,8 +42,8 @@ class Wit_Cat_File_Helper { getInfo() { return { - id: "Wit_Cat_File_Helper", // 拓展id - name: this.formatMessage("Wit_Cat_File_Helper.name"), // 拓展名 + id: extensionId, // 拓展id + name: this.formatMessage("WitCatFileHelper.name"), // 拓展名 // docsURI: "https://www.baidu.com", blockIconURI: _icon, menuIconURI: _icon, @@ -50,7 +52,7 @@ class Wit_Cat_File_Helper { blocks: [{ opcode: "download", blockType: "command", - text: this.formatMessage("Wit_Cat_File_Helper.download"), + text: this.formatMessage("WitCatFileHelper.download"), arguments: { text: { type: "string", @@ -69,7 +71,7 @@ class Wit_Cat_File_Helper { { opcode: "upload", blockType: "reporter", - text: this.formatMessage("Wit_Cat_File_Helper.upload"), + text: this.formatMessage("WitCatFileHelper.upload"), arguments: { name: { type: "string", @@ -80,7 +82,7 @@ class Wit_Cat_File_Helper { { opcode: "save", blockType: "command", - text: this.formatMessage("Wit_Cat_File_Helper.save"), + text: this.formatMessage("WitCatFileHelper.save"), arguments: { text: { type: "string", @@ -95,7 +97,7 @@ class Wit_Cat_File_Helper { { opcode: "segmentation", blockType: "reporter", - text: this.formatMessage("Wit_Cat_File_Helper.segmentation"), + text: this.formatMessage("WitCatFileHelper.segmentation"), arguments: { text: { type: "string", @@ -110,7 +112,7 @@ class Wit_Cat_File_Helper { { opcode: "delete", blockType: "command", - text: this.formatMessage("Wit_Cat_File_Helper.delete"), + text: this.formatMessage("WitCatFileHelper.delete"), arguments: { name: { type: "string", @@ -121,7 +123,7 @@ class Wit_Cat_File_Helper { { opcode: "encrypt", blockType: "reporter", - text: this.formatMessage("Wit_Cat_File_Helper.encrypt"), + text: this.formatMessage("WitCatFileHelper.encrypt"), arguments: { text: { type: "string", @@ -132,7 +134,7 @@ class Wit_Cat_File_Helper { { opcode: "decrypt", blockType: "reporter", - text: this.formatMessage("Wit_Cat_File_Helper.decrypt"), + text: this.formatMessage("WitCatFileHelper.decrypt"), arguments: { text: { type: "string", @@ -274,11 +276,11 @@ class Wit_Cat_File_Helper { } window.tempExt = { - Extension: Wit_Cat_File_Helper, + Extension: WitCatFileHelper, info: { - name: "Wit_Cat_File_Helper.name", - description: "Wit_Cat_File_Helper.descp", - extensionId: "Wit_Cat_File_Helper", + name: "WitCatFileHelper.name", + description: "WitCatFileHelper.descp", + extensionId: extensionId, iconURL: _picture, insetIconURL: _icon, featured: true, @@ -287,12 +289,12 @@ window.tempExt = { }, l10n: { "zh-cn": { - "Wit_Cat_File_Helper.name": "文件助手", - "Wit_Cat_File_Helper.descp": "处理本地数据" + "WitCatFileHelper.name": "文件助手", + "WitCatFileHelper.descp": "处理本地数据" }, en: { - "Wit_Cat_File_Helper.name": "File Helper", - "Wit_Cat_File_Helper.descp": "Handling local data" + "WitCatFileHelper.name": "File Helper", + "WitCatFileHelper.descp": "Handling local data" } } }; From 47dbeeb2a90dacfb179bf0fba29b6d68200deb6b Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Wed, 28 Sep 2022 00:13:49 +0800 Subject: [PATCH 150/357] =?UTF-8?q?=E4=BC=98=E5=8C=96+=E4=BF=AE=E5=A4=8D?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 修复下载的BUG,优化保存的流程,修复分割的BUG --- wit_cat/File_Helper.js | 62 ++++++++++++++++++++++++++++++++++-------- 1 file changed, 51 insertions(+), 11 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 8d1ae63e..c5bdc698 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -11,9 +11,10 @@ class WitCatFileHelper { this._formatMessage = runtime.getFormatMessage({ "zh-cn": { "WitCatFileHelper.name": "文件助手", - "WitCatFileHelper.download": "将内容[text]按[s]分割后命名为[name]并下载", + "WitCatFileHelper.downloads": "将内容[text]按[s]分割后命名为[name]并下载多行文本", + "WitCatFileHelper.download": "将内容[text]命名为[name]并下载", + "WitCatFileHelper.save": "设置键[name]为[text]并储存到本地", "WitCatFileHelper.upload": "获取键[name]的值", - "WitCatFileHelper.save": "将内容[text]命名为[name]并保存", "WitCatFileHelper.delete": "删除键[name]", "WitCatFileHelper.segmentation": "将[text]按[s]分割", "WitCatFileHelper.encrypt": "base64加密[text]", @@ -21,9 +22,10 @@ class WitCatFileHelper { }, en: { "WitCatFileHelper.name": "File Helper", - "WitCatFileHelper.download": "Download split content [text] by [S] named [name]", + "WitCatFileHelper.downloads": "Download split content [text] by [S] named [name]", + "WitCatFileHelper.download": "Download content [text] named [name]", + "WitCatFileHelper.save": "Save content [text] with [name] on computer", "WitCatFileHelper.upload": "Get value [name]", - "WitCatFileHelper.save": "Save content [text] with [name]", "WitCatFileHelper.delete": "delete value [name]", "WitCatFileHelper.segmentation": "Split [text] by [s]", "WitCatFileHelper.encrypt": "base64 encrypt[text]", @@ -50,13 +52,13 @@ class WitCatFileHelper { color1: "#60D6F4", color2: "#55a7f7", blocks: [{ - opcode: "download", + opcode: "downloads", blockType: "command", - text: this.formatMessage("WitCatFileHelper.download"), + text: this.formatMessage("WitCatFileHelper.downloads"), arguments: { text: { type: "string", - defaultValue: 'awa!!!', + defaultValue: 'awa!!!|awa!!!', }, name: { type: "string", @@ -64,7 +66,22 @@ class WitCatFileHelper { }, s: { type: "string", - defaultValue: '', + defaultValue: '|', + }, + }, + }, + { + opcode: "download", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.download"), + arguments: { + text: { + type: "string", + defaultValue: 'awa!!!', + }, + name: { + type: "string", + defaultValue: 'wit_cat.txt', }, }, }, @@ -145,11 +162,12 @@ class WitCatFileHelper { ] }; } - //下载文件 - download(args) { + //下载多行文件 + downloads(args) { var text = args.text; const filename = args.name; var s = args.s; + var j = 0; if (s != "") { var a = text.split(s); var h = a[0]; @@ -173,6 +191,23 @@ class WitCatFileHelper { // 然后移除 document.body.removeChild(eleLink); } + //下载文件 + download(args) { + const filename = args.name; + const content = args.text; + // 创建隐藏的可下载链接 + var eleLink = document.createElement('a'); + eleLink.download = filename; + eleLink.style.display = 'none'; + // 字符内容转变成blob地址 + var blob = new Blob([content]); + eleLink.href = URL.createObjectURL(blob); + // 触发点击 + document.body.appendChild(eleLink); + eleLink.click(); + // 然后移除 + document.body.removeChild(eleLink); + } //读取本地变量 upload(args) { const name = args.name; @@ -257,7 +292,12 @@ class WitCatFileHelper { var text = args.text; var s = args.s; var array = text.split(s); - return array; + const a = `","`; + const b = `["`; + const c = `"]`; + var str = array.join(a); + var r = b + str + c; + return r; } //加密 encrypt(args) { From 6331044475b163de320e9cf425d7ae018007102a Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 11 Oct 2022 14:03:09 +0800 Subject: [PATCH 151/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8DBUG?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- wit_cat/File_Helper.js | 149 +++++++++++++++++------------------------ 1 file changed, 61 insertions(+), 88 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index c5bdc698..a885f7fa 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -1,13 +1,14 @@ // import Cast from "cast.js"; -let _picture = ""; +const _picture = ""; -let _icon = ""; +const _icon = ""; const extensionId = "WitCatFileHelper"; class WitCatFileHelper { constructor(runtime) { + this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { "WitCatFileHelper.name": "文件助手", @@ -19,6 +20,7 @@ class WitCatFileHelper { "WitCatFileHelper.segmentation": "将[text]按[s]分割", "WitCatFileHelper.encrypt": "base64加密[text]", "WitCatFileHelper.decrypt": "base64解密[text]", + "WitCatFileHelper.openfile": "打卡文件" }, en: { "WitCatFileHelper.name": "File Helper", @@ -30,6 +32,7 @@ class WitCatFileHelper { "WitCatFileHelper.segmentation": "Split [text] by [s]", "WitCatFileHelper.encrypt": "base64 encrypt[text]", "WitCatFileHelper.decrypt": "base64 decrypt[text]", + "WitCatFileHelper.openfile": "openfile" } }) } @@ -46,7 +49,7 @@ class WitCatFileHelper { return { id: extensionId, // 拓展id name: this.formatMessage("WitCatFileHelper.name"), // 拓展名 - // docsURI: "https://www.baidu.com", + docsURI: "https://www.ccw.site/post/d6d96e80-3f58-4a19-b7e6-c567d3a6a583", blockIconURI: _icon, menuIconURI: _icon, color1: "#60D6F4", @@ -159,31 +162,37 @@ class WitCatFileHelper { }, }, }, + { + opcode: "openfile", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.openfile"), + arguments: {}, + }, ] }; } //下载多行文件 downloads(args) { - var text = args.text; + let text = args.text; const filename = args.name; - var s = args.s; - var j = 0; + let s = args.s; + let j = 0; if (s != "") { - var a = text.split(s); - var h = a[0]; + let a = text.split(s); + let h = a[0]; for (j = 1; j < a.length; j++) { - var h = h + "\n" + a[j]; + let h = h + "\n" + a[j]; } } else { - var h = text; + let h = text; } const content = h; // 创建隐藏的可下载链接 - var eleLink = document.createElement('a'); + let eleLink = document.createElement('a'); eleLink.download = filename; eleLink.style.display = 'none'; // 字符内容转变成blob地址 - var blob = new Blob([content]); + let blob = new Blob([content]); eleLink.href = URL.createObjectURL(blob); // 触发点击 document.body.appendChild(eleLink); @@ -196,11 +205,11 @@ class WitCatFileHelper { const filename = args.name; const content = args.text; // 创建隐藏的可下载链接 - var eleLink = document.createElement('a'); + let eleLink = document.createElement('a'); eleLink.download = filename; eleLink.style.display = 'none'; // 字符内容转变成blob地址 - var blob = new Blob([content]); + let blob = new Blob([content]); eleLink.href = URL.createObjectURL(blob); // 触发点击 document.body.appendChild(eleLink); @@ -211,108 +220,72 @@ class WitCatFileHelper { //读取本地变量 upload(args) { const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if (indexOf4 == -1) { - var indexOf4 = array.findIndex((item) => item == 'project'); - if (indexOf4 == -1) { - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } else { - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } else { - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } + let h = this.runtime.ccwAPI.getProjectUUID(); return localStorage.getItem(h + name); } //保存本地变量 save(args) { const text = args.text; const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if (indexOf4 == -1) { - var indexOf4 = array.findIndex((item) => item == 'project'); - if (indexOf4 == -1) { - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } else { - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } else { - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; + let h = this.runtime.ccwAPI.getProjectUUID(); + if(h == ""){ + alert("请先保存作品"); + } else{ + localStorage.setItem(h + name, text); } - localStorage.setItem(h + name, text); } //删除本地变量 delete(args) { const name = args.name; - var h = window.location.href; - console.log(h); - var array = h.split("/"); - var indexOf4 = array.findIndex((item) => item == 'detail'); - if (indexOf4 == -1) { - var indexOf4 = array.findIndex((item) => item == 'project'); - if (indexOf4 == -1) { - var array = h.split("?"); - var h = array[1]; - var array = h.split("="); - var h = array[1]; - } else { - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } - } else { - var h = array[indexOf4 + 1]; - var array = h.split("?"); - var h = array[0]; - } + let h = this.runtime.ccwAPI.getProjectUUID(); localStorage.removeItem(h + name); } //字符串分割 segmentation(args) { - var text = args.text; - var s = args.s; - var array = text.split(s); + let text = args.text; + let s = args.s; + let array = text.split(s); const a = `","`; const b = `["`; const c = `"]`; - var str = array.join(a); - var r = b + str + c; + let str = array.join(a); + let r = b + str + c; return r; } //加密 encrypt(args) { - var str = args.text; - var jiaMi = encodeURIComponent(str); - var jiaM = btoa(jiaMi); + let str = args.text; + let jiaMi = encodeURIComponent(str); + let jiaM = btoa(jiaMi); return jiaM; } //解密 decrypt(args) { - var jiaM = args.text; - var jieMi = atob(jiaM); - var jieM = decodeURIComponent(jieMi); + let jiaM = args.text; + let jieMi = atob(jiaM); + let jieM = decodeURIComponent(jieMi); return jieM; } + //打开文件 + openfile(args){ + return new Promise(resolve => { + const input = document.createElement("input"); + input.type = "file"; + input.onchange = () => { + const reader = new FileReader(); + const file = input.files[0]; + reader.onload = (e) => { + resolve(e.target.result); + }; + reader.onerror = (e) => { + console.error(e); + resolve(); + }; + reader.readAsText(file); + } + input.click(); + }); + } } window.tempExt = { From 66b968c417da6020eb61cb0550a52822b4158079 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 11 Oct 2022 19:13:58 +0800 Subject: [PATCH 152/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8D?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 修了下载的BUG,修复一个错别字 --- wit_cat/File_Helper.js | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index a885f7fa..57693743 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -20,7 +20,7 @@ class WitCatFileHelper { "WitCatFileHelper.segmentation": "将[text]按[s]分割", "WitCatFileHelper.encrypt": "base64加密[text]", "WitCatFileHelper.decrypt": "base64解密[text]", - "WitCatFileHelper.openfile": "打卡文件" + "WitCatFileHelper.openfile": "打开文件" }, en: { "WitCatFileHelper.name": "File Helper", @@ -173,18 +173,19 @@ class WitCatFileHelper { } //下载多行文件 downloads(args) { + let h = args.text; let text = args.text; const filename = args.name; let s = args.s; let j = 0; if (s != "") { let a = text.split(s); - let h = a[0]; + h = a[0]; for (j = 1; j < a.length; j++) { - let h = h + "\n" + a[j]; + h = h + "\n" + a[j]; } } else { - let h = text; + h = text; } const content = h; // 创建隐藏的可下载链接 From 75e455cd0622422aa793e4a47785138af8b8aa46 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sat, 1 Oct 2022 17:36:18 +0800 Subject: [PATCH 153/357] up up --- danny/scCOM.js | 1254 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 1254 insertions(+) create mode 100644 danny/scCOM.js diff --git a/danny/scCOM.js b/danny/scCOM.js new file mode 100644 index 00000000..e7fcf227 --- /dev/null +++ b/danny/scCOM.js @@ -0,0 +1,1254 @@ +const Cast = require('../../util/cast'); +const ArgumentType = require('../../extension-support/argument-type'); +const BlockType = require('../../extension-support/block-type'); +// var dgram = require("dgram"); +//感谢Arkos开源的扩展代码,作为参照编写框架 +//欸,别把作者忘了。Danny欸 +class AuEx_communication { + constructor(runtime) { + this.runtime = runtime + this._formatMessage = runtime.getFormatMessage({ + 'zh-cn': { + 'AusCOM.extensionName': 'AUSの通讯模块', + 'AusCOM.tit1': 'sock通讯', + 'AusCOM.tit2': '数据处理', + 'AusCOM.tit3': 'JSON', + 'AusCOM.http': 'HTTP以[g_way]方式访问[site]发送[body]', + 'AusCOM.wsocket_create': 'ws.创建websocket,协议[yi],并返回ccid', + 'AusCOM.wsocket_connect': 'ws.id[id]socket连接[host]', + 'AusCOM.wsocket_send': 'ws.发送id[id]数据[text]', + 'AusCOM.wsocket_recv': 'ws.接收数据id[id]', + 'AusCOM.wsocket_conclose': 'ws.连接关闭id[id]', + 'AusCOM.wsocket_close': 'ws.销毁websocket id[id]', + 'AusCOM.socket_create': 'sock.创建[yi]方式[method]并返回ccid', + 'AusCOM.socket_create_chk': 'sock.创建是否成功[ccid]并返回id', + 'AusCOM.socket_recv': 'sock.收到数据id [id]', + 'AusCOM.socket_send': 'sock.发送数据id [id] ip[ip]数据[TEXT]', + 'AusCOM.socket_conclose': 'sock.断开连接id[id]', + 'AusCOM.socket_connect': 'sock.连接/启动服务 id[id] ip [ip] 接收长度[buff]', + 'AusCOM.socket_breakev': 'sock.断开连接id[id]ip[ip]', + 'AusCOM.socket_close': 'sock.销毁socket对象 id[id]', + 'AusCOM.socket_parse': 'sock.解析返回的数据data[data]属性[type]', + 'AusCOM.socket_opchk': 'sock.操作是否成功[cid]', + 'AusCOM.code_RC4': 'RC4加解密 密钥[KEY] 文本[TEXT]', + 'AusCOM.code_base64_en': 'base64编码文本[TEXT]', + 'AusCOM.code_base64_de': 'base64解码文本[TEXT]', + 'AusCOM.code_base58_en': 'base58编码文本[TEXT]', + 'AusCOM.code_base58_de': 'base58解码码文本[TEXT]', + 'AusCOM.code_base_custom_en': '自定义base加密 字符集[ALPHABET] 文本[TEXT]', + 'AusCOM.code_base_custom_de': '自定义base解密 字符集[ALPHABET] 文本[TEXT]', + 'AusCOM.str': '字符串', + 'AusCOM.int': '整数', + 'AusCOM.float': '浮点数', + 'AusCOM.list': '列表', + 'AusCOM.logic': '逻辑', + 'AusCOM.json_create': '创建json 载入[json_] 并返回id', + 'AusCOM.json_load': '载入json id[id]json[json_]', + 'AusCOM.json_get_all_item': '获取json id[id]中所有属性名', + 'AusCOM.json_add_from_id': '将id[_id]对应的json加入json/改变值 id[id] 属性名[name]', + 'AusCOM.json_add': '加入/改变对象到json id[id]类型[type]属性名[name]内容[TEXT]', + 'AusCOM.json_get_item': '获取json id [id]中某属性名为[name]项的值', + 'AusCOM.json_del_item': '删除json id [id]中某属性名为[name]项的值', + 'AusCOM.json_get': '获取json id[id]', + 'AusCOM.json_destroy': '销毁json id[id]', + 'AusCOM.help': '使用帮助', + }, + en: { + 'AusCOM.extensionName': 'AUSのcommunication', + 'AusCOM.tit1': 'sock communication', + 'AusCOM.tit2': 'solve data', + 'AusCOM.tit3': 'JSON', + 'AusCOM.http': 'http access[g_way]to[site]send[body]', + 'AusCOM.wsocket_create': 'ws.create websocket,protocol', 'AusCOM.wsocket_connect': 'id[id]websocket connect[host]', + 'AusCOM.wsocket_send': 'ws.send id[id] data[text]', + 'AusCOM.wsocket_recv': 'ws.recv id [id]', + 'AusCOM.wsocket_conclose': 'ws.When connection closed id[id]', + 'AusCOM.wsocket_close': 'ws.destroy socket id[id]', + 'AusCOM.socket_ sreate ': 'sock. Create [yi] method [method] and return ID', + 'AusCOM.socket_create_chk': 'sock.create successfully[ccid]?', + 'AusCOM.socket_ recv': 'sock. Received data ID [id] ', + 'AusCOM.socket_ send': 'sock. Send data ID [id] data [text] ', + 'AusCOM.socket_ close ': 'sock. Connection closing ID [id]', + 'AusCOM.socket_ connect': 'sock. Connect to/start server ID [id] IP [ip] ,buff[buff]', + 'AusCOM.socket_breakev': 'sock.break connection off id[id]ip[ip]', + 'AusCOM.socket_ close': 'sock. Destroy the socket object ID [id] ', + 'AusCOM.socket_ parse ': 'sock. Parse the returned data [data] attribute [type]', + 'AusCOM.socket_opchk': 'sock.run operation successfully[cid]?', + 'AusCOM.code_RC4': 'RC4 KEY[KEY] TEXT[TEXT]', + 'AusCOM.code_base64_en': 'base64 encoding[TEXT]', + 'AusCOM.code_base64_de': 'base64 decoding[TEXT]', + 'AusCOM.code_base58_en': 'base58 encoding[TEXT]', + 'AusCOM.code_base58_de': 'base58 decoding[TEXT]', + 'AusCOM.code_base_custom_en': 'Custom base encryption,ALPHABET[ALPHABET] TEXT[TEXT]', + 'AusCOM.code_base_custom_de': 'Custom base decryption,ALPHABET[ALPHABET] TEXT[TEXT]', + 'AusCOM.str': 'string', + 'AusCOM.int': 'int', + 'AusCOM.float': 'float', + 'AusCOM.list': 'list', + 'AusCOM.logic': 'logic', + 'AusCOM.json_create': 'create json,load[json_] and return id', + 'AusCOM.json_load': ' load json id[id]json[json_]', + 'AusCOM.json_get_all_item': 'get all attribute name of json id[id]', + 'AusCOM.json_add_from_id': 'Add the JSON corresponding to ID [_id] to JSON/change a value ID [id] attribute name [name]', + 'AusCOM.json_add': 'Add object/change a value to JSON ID [id], type [type], attribute name [name], content [TEXT]', + 'AusCOM.json_get_item': 'Get the value of an attribute named [name] in JSON ID [id]', + 'AusCOM.json_del_item': 'Delete the value of an attribute named [name] in JSON ID [ID]', + 'AusCOM.json_get': 'get json id[id]', + 'AusCOM.json_destroy': 'destroy json id[id]', + 'AusCOM.help': 'help', + }, + }) + this.aulink = new Worker('./aurora_link.js') + window.AusCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [] } + + } + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id, + }) + } + getInfo() { + return { + id: 'DAdevNet', // 拓展id + name: this.formatMessage('AusCOM.extensionName'), // 拓展名 + color1: '#FF8383', + menuIconURI: "", + blockIconURI: "", + blocks: [ + "---" + this.formatMessage("AusCOM.tit1"), + { + opcode: 'http', + blockType: 'reporter', + text: this.formatMessage('AusCOM.http'), + arguments: { + g_way: { + type: 'string', + menu: 'HTTP_method' + }, + site: { + type: 'string', + defaultValue: '/' + }, + body: { + type: 'string', + defaultValue: ' ' + }, + } + }, + + { + opcode: 'ws_cre_sock', + blockType: 'reporter', + text: this.formatMessage('AusCOM.wsocket_create'), + arguments: { + yi: { + type: 'string', + menu: 'ws_method' + }, + } + }, + { + opcode: 'ws_conn_sock', + blockType: 'Boolean', + text: this.formatMessage('AusCOM.wsocket_connect'), + arguments: { + id: { + type: 'string', + defaultValue: '0' + }, + host: { + type: 'string', + defaultValue: '0.0.0.0:1212/' + }, + } + }, + { + opcode: 'ws_recv', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.wsocket_recv'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: '0' + } + }, + }, + { + opcode: 'ws_close', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.wsocket_conclose'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: '0' + } + }, + }, + { + opcode: 'ws_destroy', + blockType: BlockType.COMMAND, + text: this.formatMessage('AusCOM.wsocket_close'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: '0' + } + }, + }, + { + opcode: 'ws_send', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.wsocket_send'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: '0' + }, + text: { + type: ArgumentType.STRING, + defaultValue: ' ' + } + }, + }, + // { + // opcode: 'sock_cre', + // blockType: BlockType.REPORTER, + // text: this.formatMessage('AusCOM.socket_create'), + // arguments: { + // + // yi: { + // type: ArgumentType.STRING, + // menu: 'sock_yi' + // }, + // + // method: { + // type: ArgumentType.STRING, + // menu: 'sock_method' + // }, + // buff: { + // type: ArgumentType.STRING, + // defaultValue: ' ' + // } + // }, + //}, + "---" + this.formatMessage("AusCOM.tit2"), + { + opcode: 'RC4_coding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_RC4'), + arguments: { + KEY: { + type: ArgumentType.STRING, + defaultValue: 'Danny' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '我来了' + } + }, + }, + { + opcode: 'b64encoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base64_en'), + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: 'Danny' + }, + }, + + }, + { + opcode: 'b64decoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base64_de'), + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: 'RGFubnk=' + }, + }, + + }, + { + opcode: 'b58encoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base58_en'), + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: 'Danny' + }, + }, + + }, + { + opcode: 'b58decoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base58_de'), + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: '8iTbLRE' + }, + }, + + }, + { + opcode: 'bcostomencoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base_custom_en'), + arguments: { + ALPHABET: { + type: ArgumentType.STRING, + defaultValue: '!@#$%^&*()' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: 'Danny' + }, + }, + + }, + { + opcode: 'bcostomdecoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base_custom_de'), + arguments: { + ALPHABET: { + type: ArgumentType.STRING, + defaultValue: '!@#$%^&*()' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '#)$&)#%!$$#@' + }, + }, + + }, + + "---" + this.formatMessage("AusCOM.tit3"), + { + opcode: 'json_create', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.json_create'), + arguments: { + json_: { + type: ArgumentType.STRING, + defaultValue: '{}' + }, + }, + }, + { + opcode: 'json_load', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_load'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + json_: { + type: ArgumentType.STRING, + defaultValue: '{}' + }, + }, + }, + { + opcode: 'json_get_all_item', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.json_get_all_item'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + }, + }, + { + opcode: 'json_add_from_id', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_add_from_id'), + arguments: { + _id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + name: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + }, + }, + { + opcode: 'json_add', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_add'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + type: { + type: ArgumentType.STRING, + menu: 'JSON_add' + }, + name: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + }, + }, + { + opcode: 'json_get_item', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.json_get_item'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + name: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + + }, + }, + { + opcode: 'json_del_item', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_del_item'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + name: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + + }, + }, + { + opcode: 'GETjson', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.json_get'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + } + }, + { + opcode: 'DESTROYjson', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_destroy'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + }, + }, + + { + opcode: 'help', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.help') + 'ver 1.00C1', + }, + + ], + menus: { + HTTP_method: [ + { + text: 'GET', + value: 'GET' + }, + { + text: 'POST', + value: 'POST' + }, + { + text: 'HEAD', + value: 'HEAD' + }, + { + text: 'PUT', + value: 'PUT' + }, + { + text: 'DELETE', + value: 'DELETE' + }, + { + text: 'OPTIONS', + value: 'OPTIONS' + }, + ], + ws_method: [ + { + text: 'WS', + value: 'WS' + }, + { + text: 'WSS', + value: 'WSS' + }, + ], + JSON_add: [ + { + text: this.formatMessage('AusCOM.str'), + value: '0' + }, + { + text: this.formatMessage('AusCOM.int'), + value: '1' + }, + { + text: this.formatMessage('AusCOM.float'), + value: '2' + }, + { + text: this.formatMessage('AusCOM.list'), + value: '3' + }, + { + text: this.formatMessage('AusCOM.logic'), + value: '4' + }, + ], + + sock_yi: [ + { + text: 'TCP', + value: '0' + }, + { + text: 'UDP', + value: '1' + }, + ], + sock_method: [ + { + text: 'client', + value: '0' + }, + { + text: 'server', + value: '1' + }, + ], + } + } + } + httpGet(theUrl, body, method) { + try { + var xmlHttp = new XMLHttpRequest(); + xmlHttp.open(method, theUrl, false); // false 为同步请求 + xmlHttp.send(body); + return xmlHttp.responseText; + } + catch (error) { + return error + } + } + + http(args) { + const { g_way, site, body } = args + return this.httpGet(site, body, g_way); + } + + ws_cre_sock(args) { + const { yi } = args + if (yi == 'WS') { + id = Date.now() + window.AusCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WS' }) + return String(id) + } + if (yi == 'WSS') { + id = Date.now() + window.AusCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WSS' }) + return String(id) + } + return '未知协议' + } + ws_conn_sock(args) { + const { id, host } = args + console.log(id, host) + for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { + if (String(id) == window.AusCOM.wsock[i]["id"]) { + for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { + delete window.AusCOM.wsnotalive_[String(id)] + } + } + + try { + window.AusCOM.wsock[i]['obj'] = new WebSocket(window.AusCOM.wsock[i]["way"] + '://' + host) + console.log(window.AusCOM.wsock) + window.AusCOM.wsock[i]['obj'].onmessage = function (evt) { + + window.abc = evt.data + window.AusCOM.wsrecv_.push({ 'id': window.AusCOM.wsock[i]['id'], 'nr': evt.data }) + } + window.AusCOM.wsock[i]['obj'].onclose = function (evt) { + window.AusCOM.wsnotalive_.push(window.AusCOM.wsock[i]['id']) + } + return true + } + catch (error) { + + return false + } + } + } + + return false + } + ws_recv(args) { + const { id } = args + console.log(window.AusCOM.wsrecv_) + for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { + console.log(window.AusCOM.wsrecv_) + try { + var a = String(window.AusCOM.wsrecv_[i]["nr"]) + window.AusCOM.wsrecv_.splice(i, 1); + return a + } + catch (error) { + + return '' + } + } + } + return false + } + ws_close(args) { + const { id } = args + for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { + window.AusCOM.wsnotalive_.splice(i, 1) + return true + } + } + return false + } + ws_destroy(args) { + const { id } = args + for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { + console.log(window.AusCOM.wsock[i]) + if (String(id) == window.AusCOM.wsock[i]["id"]) { + try { + if (window.AusCOM.wsock[i]["obj"] == null) { } + else { window.AusCOM.wsock[i]["obj"].close() } + window.AusCOM.wsock.splice(i, 1); + // alert(window.AusCOM.wsock) + return '' + } + catch (error) { console.log(error) } + } + } + } + ws_send(args) { + const { id, text } = args + for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { + if (String(id) == window.AusCOM.wsock[i]["id"]) { + console.log(window.AusCOM.wsrecv_) + console.log(window.AusCOM.wsock[i]) + + try { + if (window.AusCOM.wsock[i]["obj"] == null) { return false } else { window.AusCOM.wsock[i]["obj"].send(text); return true } + } + catch (error) { + + return false + } + } + + } + + return false + } + help() { + return "http访问出现fail to load XX可能是域名/ip不存在,也有可能是出现跨域问题,可以联系本站站住修改HTTP协议头或者自行修改浏览器跨域设置。如果ws无法发送数据,请F12打开开发者工具查看是否报错,如果报错则是无法连接,目前浏览器还不支持侦察此类报错。更多帮助请进入 https://gitee.com/ausx/scCOM" + } + RC4_coding(args) { + var key = '' + var data = '' + const { KEY, TEXT } = args + key = String(KEY) + data = String(TEXT)//对js不识别字符串进行处理 + var seq = Array(256); //如果有机会编译成离线可执行程序,会考虑支持AVX512 + var das = Array(data.length); + for (var i = 0; i < 256; i++) { + seq[i] = i; + var j = (j + seq[i] + key.charCodeAt(i % key.length)) % 256; + var temp = seq[i]; + seq[i] = seq[j]; + seq[j] = temp; + } + for (var i = 0; i < data.length; i++) { + das[i] = data.charCodeAt(i) + } + for (var x = 0; x < das.length; x++) { + var i = (i + 1) % 256; + var j = (j + seq[i]) % 256; + var temp = seq[i]; + seq[i] = seq[j]; + seq[j] = temp; + var k = (seq[i] + (seq[j] % 256)) % 256; + das[x] = String.fromCharCode(das[x] ^ seq[k]); + } + return das.join(''); + } + b64encoding(args) { + const { TEXT } = args + return window.btoa(encodeURIComponent(String(TEXT))); // 编码 + } + b64decoding(args) { + const { TEXT } = args + try { + return decodeURIComponent(window.atob(String(TEXT))); // 解码 + } + catch (error) { + return error + } + } + b58encoding(args) { + const { TEXT } = args + var ALPHABET = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'; + var ALPHABET_MAP = {}; + var BASE = 58; + for (var i = 0; i < ALPHABET.length; i++) { + ALPHABET_MAP[ALPHABET.charAt(i)] = i; + } + function ToUTF8(str) { + var result = new Array(); + + var k = 0; + for (var i = 0; i < str.length; i++) { + var j = encodeURI(str[i]); + if (j.length == 1) { + // 未转换的字符 + result[k++] = j.charCodeAt(0); + } else { + // 转换成%XX形式的字符 + var bytes = j.split("%"); + for (var l = 1; l < bytes.length; l++) { + result[k++] = parseInt("0x" + bytes[l]); + } + } + } + + return result; + } + + + // 如果有特殊需求,要转成utf16,可以用以下函数 + function ToUTF16(str) { + var result = new Array(); + + var k = 0; + for (var i = 0; i < str.length; i++) { + var j = str[i].charCodeAt(0); + result[k++] = j & 0xFF; + result[k++] = j >> 8; + } + + return result; + } + // 传进已经转成字节的数组 -->buffer(utf8格式) + function encode(buffer) { + if (buffer.length === 0) return ''; + var i, + j, + digits = [0]; + for (i = 0; i < buffer.length; i++) { + for (j = 0; j < digits.length; j++) { + digits[j] <<= 8; + } + digits[0] += buffer[i]; + var carry = 0; + for (j = 0; j < digits.length; ++j) { + digits[j] += carry; + carry = (digits[j] / BASE) | 0; + digits[j] %= BASE; + } + while (carry) { + digits.push(carry % BASE); + carry = (carry / BASE) | 0; + } + } + // deal with leading zeros + for (i = 0; buffer[i] === 0 && i < buffer.length - 1; i++) digits.push(0); + return digits + .reverse() + .map(function (digit) { + return ALPHABET[digit]; + }) + .join(''); + } + + + return encode(ToUTF8(String(TEXT))) + } + b58decoding(args) { + const { TEXT } = args + var ALPHABET = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'; + var ALPHABET_MAP = {}; + var BASE = 58; + for (var i = 0; i < ALPHABET.length; i++) { + ALPHABET_MAP[ALPHABET.charAt(i)] = i; + } + function decode(string) { + if (string.length === 0) return []; + var i, + j, + bytes = [0]; + for (i = 0; i < string.length; i++) { + var c = string[i]; + // c是不是ALPHABET_MAP的key + if (!(c in ALPHABET_MAP)) throw new Error('Non-base58 character'); + for (j = 0; j < bytes.length; j++) bytes[j] *= BASE; + bytes[0] += ALPHABET_MAP[c]; + var carry = 0; + for (j = 0; j < bytes.length; ++j) { + bytes[j] += carry; + carry = bytes[j] >> 8; + // 0xff --> 11111111 + bytes[j] &= 0xff; + } + while (carry) { + bytes.push(carry & 0xff); + carry >>= 8; + } + } + for (i = 0; string[i] === '1' && i < string.length - 1; i++) bytes.push(0); + return bytes.reverse(); + } + + function byteToString(arr) { + if (typeof arr === 'string') { + return arr; + } + var str = '', + _arr = arr; + for (var i = 0; i < _arr.length; i++) { + // 数组中每个数字转为二进制 匹配出开头为1的直到0的字符 + // eg:123-->"1111011"-->{0:"1111",groups: undefined, index: 0, input: "1111011"} + var one = _arr[i].toString(2), + v = one.match(/^1+?(?=0)/); + if (v && one.length == 8) { + var bytesLength = v[0].length; + var store = _arr[i].toString(2).slice(7 - bytesLength); + for (var st = 1; st < bytesLength; st++) { + store += _arr[st + i].toString(2).slice(2); + } + str += String.fromCharCode(parseInt(store, 2)); + i += bytesLength - 1; + } else { + str += String.fromCharCode(_arr[i]); + } + } + return str; + } + return byteToString(decode(String(TEXT))) + } + bcostomencoding(args) { + const { ALPHABET, TEXT } = args + var ALPHABET_MAP = {}; + var BASE = ALPHABET.length; + for (var i = 0; i < ALPHABET.length; i++) { + ALPHABET_MAP[ALPHABET.charAt(i)] = i; + } + function ToUTF8(str) { + var result = new Array(); + + var k = 0; + for (var i = 0; i < str.length; i++) { + var j = encodeURI(str[i]); + if (j.length == 1) { + // 未转换的字符 + result[k++] = j.charCodeAt(0); + } else { + // 转换成%XX形式的字符 + var bytes = j.split("%"); + for (var l = 1; l < bytes.length; l++) { + result[k++] = parseInt("0x" + bytes[l]); + } + } + } + + return result; + } + + + // 如果有特殊需求,要转成utf16,可以用以下函数 + function ToUTF16(str) { + var result = new Array(); + + var k = 0; + for (var i = 0; i < str.length; i++) { + var j = str[i].charCodeAt(0); + result[k++] = j & 0xFF; + result[k++] = j >> 8; + } + + return result; + } + // 传进已经转成字节的数组 -->buffer(utf8格式) + function encode(buffer) { + if (buffer.length === 0) return ''; + var i, + j, + digits = [0]; + for (i = 0; i < buffer.length; i++) { + for (j = 0; j < digits.length; j++) { + digits[j] <<= 8; + } + digits[0] += buffer[i]; + var carry = 0; + for (j = 0; j < digits.length; ++j) { + digits[j] += carry; + carry = (digits[j] / BASE) | 0; + digits[j] %= BASE; + } + while (carry) { + digits.push(carry % BASE); + carry = (carry / BASE) | 0; + } + } + // deal with leading zeros + for (i = 0; buffer[i] === 0 && i < buffer.length - 1; i++) digits.push(0); + return digits + .reverse() + .map(function (digit) { + return ALPHABET[digit]; + }) + .join(''); + } + + + return encode(ToUTF8(String(TEXT))) + } + bcostomdecoding(args) { + const { ALPHABET, TEXT } = args + var ALPHABET_MAP = {}; + var BASE = ALPHABET.length; + for (var i = 0; i < ALPHABET.length; i++) { + ALPHABET_MAP[ALPHABET.charAt(i)] = i; + } + function decode(string) { + if (string.length === 0) return []; + var i, + j, + bytes = [0]; + for (i = 0; i < string.length; i++) { + var c = string[i]; + // c是不是ALPHABET_MAP的key + if (!(c in ALPHABET_MAP)) throw new Error('Non-base58 character'); + for (j = 0; j < bytes.length; j++) bytes[j] *= BASE; + bytes[0] += ALPHABET_MAP[c]; + var carry = 0; + for (j = 0; j < bytes.length; ++j) { + bytes[j] += carry; + carry = bytes[j] >> 8; + // 0xff --> 11111111 + bytes[j] &= 0xff; + } + while (carry) { + bytes.push(carry & 0xff); + carry >>= 8; + } + } + for (i = 0; string[i] === '1' && i < string.length - 1; i++) bytes.push(0); + return bytes.reverse(); + } + + function byteToString(arr) { + if (typeof arr === 'string') { + return arr; + } + var str = '', + _arr = arr; + for (var i = 0; i < _arr.length; i++) { + // 数组中每个数字转为二进制 匹配出开头为1的直到0的字符 + // eg:123-->"1111011"-->{0:"1111",groups: undefined, index: 0, input: "1111011"} + var one = _arr[i].toString(2), + v = one.match(/^1+?(?=0)/); + if (v && one.length == 8) { + var bytesLength = v[0].length; + var store = _arr[i].toString(2).slice(7 - bytesLength); + for (var st = 1; st < bytesLength; st++) { + store += _arr[st + i].toString(2).slice(2); + } + str += String.fromCharCode(parseInt(store, 2)); + i += bytesLength - 1; + } else { + str += String.fromCharCode(_arr[i]); + } + } + return str; + } + return byteToString(decode(String(TEXT))) + } + json_create(args) { + const { json_ } = args + function create(j) { + id = Date.now() + a = {} + try { + a = JSON.parse(j) + } + catch (error) { } + window.AusCOM.json_list.push({ 'id': String(id), 'json': a }) + return id + } + return create(json_) + + } + json_load(args) { + const { id, json_ } = args + function load(id, j) { + var a = {} + try { + a = JSON.parse(j) + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + window.AusCOM.json_list[i]['json'] = a + return true + } + } + } + catch (error) { return false } + return false + + } + return load(id, json_) + } + json_get_all_item(args) { + const { id } = args + var b = [] + try { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + b = Object.keys(window.AusCOM.json_list[i]['json']) + } + } + } + catch (error) { b = [] } + var d = '' + for (let c of b) { + if (c == b[-1]) { + d += c + } + else { + d += c + ' ' + } + } + + return d + } + check(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return true + } + } + return false + } + getjson(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return window.AusCOM.json_list[i]['json'] + } + } + return {} + } + json_add_from_id(args) { + const { _id, id, name } = args + function check(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return true + } + } + return false + } + function getjson(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return window.AusCOM.json_list[i]['json'] + } + } + return {} + } + function add_item_from_id(id, id1, name) { + if (check(id) & check(id1)) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + window.AusCOM.json_list[i]['json'][name] = getjson(id1) + return true + } + } + } + else { return false } + return false + } + return add_item_from_id(id, _id, name) + } + json_add(args) { + const { id, type, name, TEXT } = args + function check(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return true + } + } + return false + } + function getjson(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return window.AusCOM.json_list[i]['json'] + } + } + return {} + } + function add_item(id, name, type, content) { + if (check(id)) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + if (type == '0') {//字符串 + window.AusCOM.json_list[i]['json'][String(name)] = String(content) + return true + } + if (type == '1') {//整数 + window.AusCOM.json_list[i]['json'][String(name)] = parseInt(content) + return true + } + if (type == '2') {//浮点数 + window.AusCOM.json_list[i]['json'][String(name)] = parseFloat(content) + return true + } + if (type == '3') {//列表 + window.AusCOM.json_list[i]['json'][String(name)] = content.split(' ') + return true + } + if (type == '4') {//逻辑 + if (String(content) == 'true') { + window.AusCOM.json_list[i]['json'][String(name)] = true + return true + } + else { + window.AusCOM.json_list[i]['json'][String(name)] = false + return true + } + } + + } + } + + } + return false + } + return add_item(id, name, type, TEXT) + } + GETjson(args) { + const { id } = args + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return JSON.stringify(window.AusCOM.json_list[i]['json']) + } + } + return '' + } + DESTROYjson(args) { + const { id } = args + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + window.AusCOM.json_list.splice(i, 1); + return true + } + } + return false + } + json_get_item(args) { // + const { id, name } = args + try { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'string') { //id_0 字符串 + return window.AusCOM.json_list[i]['json'][name] + } + if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'number') { //id_1&2 整数&浮点 + return window.AusCOM.json_list[i]['json'][name] + } + if (window.AusCOM.json_list[i]['json'][name] instanceof Array) { //id_4列表 + var a = '' + for (let i_ = 0, len = window.AusCOM.json_list[i]['json'][name].length; i_ < len; i_++) { + if (i_ == window.AusCOM.json_list[i]['json'][name].length) { + a += window.AusCOM.json_list[i]['json'][name][i] + } + else { + a += window.AusCOM.json_list[i]['json'][name][i] + ' ' + } + } + } + if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'boolean') { // id_4 逻辑 + return window.AusCOM.json_list[i]['json'][name] + } + + } + } + } catch { } + return '' + } + json_del_item(args) { //删除项 + const { id, name } = args + try { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + Reflect.deleteProperty(window.AusCOM.json_list[i]['json'], String(name)) + return true + } + } + } catch { } + return false + } + _ws(host, data) { + id = this.ws_conn_sock({ 'id': 'WS', 'host': host }) + this.ws_send({ 'id': id, 'text': data }) + recv = this.ws_recv({ 'id': id }) + for (; ;) { if (recv != '') { break } } + return recv + } + + sock_cre(args) { + const { yi, method } = args + j = JSON.parse(this._ws('127.0.0.1:23089/socket/create', JSON.stringify({ 'yi': yi, 'method': method }))) + return j['id'] + + } +} +module.exports = AuEx_communication; From de7e28cf869c762f2b2af5ab420970b503875f6d Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sat, 1 Oct 2022 18:21:05 +0800 Subject: [PATCH 154/357] Delete scCOM.js --- danny/scCOM.js | 1254 ------------------------------------------------ 1 file changed, 1254 deletions(-) delete mode 100644 danny/scCOM.js diff --git a/danny/scCOM.js b/danny/scCOM.js deleted file mode 100644 index e7fcf227..00000000 --- a/danny/scCOM.js +++ /dev/null @@ -1,1254 +0,0 @@ -const Cast = require('../../util/cast'); -const ArgumentType = require('../../extension-support/argument-type'); -const BlockType = require('../../extension-support/block-type'); -// var dgram = require("dgram"); -//感谢Arkos开源的扩展代码,作为参照编写框架 -//欸,别把作者忘了。Danny欸 -class AuEx_communication { - constructor(runtime) { - this.runtime = runtime - this._formatMessage = runtime.getFormatMessage({ - 'zh-cn': { - 'AusCOM.extensionName': 'AUSの通讯模块', - 'AusCOM.tit1': 'sock通讯', - 'AusCOM.tit2': '数据处理', - 'AusCOM.tit3': 'JSON', - 'AusCOM.http': 'HTTP以[g_way]方式访问[site]发送[body]', - 'AusCOM.wsocket_create': 'ws.创建websocket,协议[yi],并返回ccid', - 'AusCOM.wsocket_connect': 'ws.id[id]socket连接[host]', - 'AusCOM.wsocket_send': 'ws.发送id[id]数据[text]', - 'AusCOM.wsocket_recv': 'ws.接收数据id[id]', - 'AusCOM.wsocket_conclose': 'ws.连接关闭id[id]', - 'AusCOM.wsocket_close': 'ws.销毁websocket id[id]', - 'AusCOM.socket_create': 'sock.创建[yi]方式[method]并返回ccid', - 'AusCOM.socket_create_chk': 'sock.创建是否成功[ccid]并返回id', - 'AusCOM.socket_recv': 'sock.收到数据id [id]', - 'AusCOM.socket_send': 'sock.发送数据id [id] ip[ip]数据[TEXT]', - 'AusCOM.socket_conclose': 'sock.断开连接id[id]', - 'AusCOM.socket_connect': 'sock.连接/启动服务 id[id] ip [ip] 接收长度[buff]', - 'AusCOM.socket_breakev': 'sock.断开连接id[id]ip[ip]', - 'AusCOM.socket_close': 'sock.销毁socket对象 id[id]', - 'AusCOM.socket_parse': 'sock.解析返回的数据data[data]属性[type]', - 'AusCOM.socket_opchk': 'sock.操作是否成功[cid]', - 'AusCOM.code_RC4': 'RC4加解密 密钥[KEY] 文本[TEXT]', - 'AusCOM.code_base64_en': 'base64编码文本[TEXT]', - 'AusCOM.code_base64_de': 'base64解码文本[TEXT]', - 'AusCOM.code_base58_en': 'base58编码文本[TEXT]', - 'AusCOM.code_base58_de': 'base58解码码文本[TEXT]', - 'AusCOM.code_base_custom_en': '自定义base加密 字符集[ALPHABET] 文本[TEXT]', - 'AusCOM.code_base_custom_de': '自定义base解密 字符集[ALPHABET] 文本[TEXT]', - 'AusCOM.str': '字符串', - 'AusCOM.int': '整数', - 'AusCOM.float': '浮点数', - 'AusCOM.list': '列表', - 'AusCOM.logic': '逻辑', - 'AusCOM.json_create': '创建json 载入[json_] 并返回id', - 'AusCOM.json_load': '载入json id[id]json[json_]', - 'AusCOM.json_get_all_item': '获取json id[id]中所有属性名', - 'AusCOM.json_add_from_id': '将id[_id]对应的json加入json/改变值 id[id] 属性名[name]', - 'AusCOM.json_add': '加入/改变对象到json id[id]类型[type]属性名[name]内容[TEXT]', - 'AusCOM.json_get_item': '获取json id [id]中某属性名为[name]项的值', - 'AusCOM.json_del_item': '删除json id [id]中某属性名为[name]项的值', - 'AusCOM.json_get': '获取json id[id]', - 'AusCOM.json_destroy': '销毁json id[id]', - 'AusCOM.help': '使用帮助', - }, - en: { - 'AusCOM.extensionName': 'AUSのcommunication', - 'AusCOM.tit1': 'sock communication', - 'AusCOM.tit2': 'solve data', - 'AusCOM.tit3': 'JSON', - 'AusCOM.http': 'http access[g_way]to[site]send[body]', - 'AusCOM.wsocket_create': 'ws.create websocket,protocol', 'AusCOM.wsocket_connect': 'id[id]websocket connect[host]', - 'AusCOM.wsocket_send': 'ws.send id[id] data[text]', - 'AusCOM.wsocket_recv': 'ws.recv id [id]', - 'AusCOM.wsocket_conclose': 'ws.When connection closed id[id]', - 'AusCOM.wsocket_close': 'ws.destroy socket id[id]', - 'AusCOM.socket_ sreate ': 'sock. Create [yi] method [method] and return ID', - 'AusCOM.socket_create_chk': 'sock.create successfully[ccid]?', - 'AusCOM.socket_ recv': 'sock. Received data ID [id] ', - 'AusCOM.socket_ send': 'sock. Send data ID [id] data [text] ', - 'AusCOM.socket_ close ': 'sock. Connection closing ID [id]', - 'AusCOM.socket_ connect': 'sock. Connect to/start server ID [id] IP [ip] ,buff[buff]', - 'AusCOM.socket_breakev': 'sock.break connection off id[id]ip[ip]', - 'AusCOM.socket_ close': 'sock. Destroy the socket object ID [id] ', - 'AusCOM.socket_ parse ': 'sock. Parse the returned data [data] attribute [type]', - 'AusCOM.socket_opchk': 'sock.run operation successfully[cid]?', - 'AusCOM.code_RC4': 'RC4 KEY[KEY] TEXT[TEXT]', - 'AusCOM.code_base64_en': 'base64 encoding[TEXT]', - 'AusCOM.code_base64_de': 'base64 decoding[TEXT]', - 'AusCOM.code_base58_en': 'base58 encoding[TEXT]', - 'AusCOM.code_base58_de': 'base58 decoding[TEXT]', - 'AusCOM.code_base_custom_en': 'Custom base encryption,ALPHABET[ALPHABET] TEXT[TEXT]', - 'AusCOM.code_base_custom_de': 'Custom base decryption,ALPHABET[ALPHABET] TEXT[TEXT]', - 'AusCOM.str': 'string', - 'AusCOM.int': 'int', - 'AusCOM.float': 'float', - 'AusCOM.list': 'list', - 'AusCOM.logic': 'logic', - 'AusCOM.json_create': 'create json,load[json_] and return id', - 'AusCOM.json_load': ' load json id[id]json[json_]', - 'AusCOM.json_get_all_item': 'get all attribute name of json id[id]', - 'AusCOM.json_add_from_id': 'Add the JSON corresponding to ID [_id] to JSON/change a value ID [id] attribute name [name]', - 'AusCOM.json_add': 'Add object/change a value to JSON ID [id], type [type], attribute name [name], content [TEXT]', - 'AusCOM.json_get_item': 'Get the value of an attribute named [name] in JSON ID [id]', - 'AusCOM.json_del_item': 'Delete the value of an attribute named [name] in JSON ID [ID]', - 'AusCOM.json_get': 'get json id[id]', - 'AusCOM.json_destroy': 'destroy json id[id]', - 'AusCOM.help': 'help', - }, - }) - this.aulink = new Worker('./aurora_link.js') - window.AusCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [] } - - } - formatMessage(id) { - return this._formatMessage({ - id, - default: id, - description: id, - }) - } - getInfo() { - return { - id: 'DAdevNet', // 拓展id - name: this.formatMessage('AusCOM.extensionName'), // 拓展名 - color1: '#FF8383', - menuIconURI: "", - blockIconURI: "", - blocks: [ - "---" + this.formatMessage("AusCOM.tit1"), - { - opcode: 'http', - blockType: 'reporter', - text: this.formatMessage('AusCOM.http'), - arguments: { - g_way: { - type: 'string', - menu: 'HTTP_method' - }, - site: { - type: 'string', - defaultValue: '/' - }, - body: { - type: 'string', - defaultValue: ' ' - }, - } - }, - - { - opcode: 'ws_cre_sock', - blockType: 'reporter', - text: this.formatMessage('AusCOM.wsocket_create'), - arguments: { - yi: { - type: 'string', - menu: 'ws_method' - }, - } - }, - { - opcode: 'ws_conn_sock', - blockType: 'Boolean', - text: this.formatMessage('AusCOM.wsocket_connect'), - arguments: { - id: { - type: 'string', - defaultValue: '0' - }, - host: { - type: 'string', - defaultValue: '0.0.0.0:1212/' - }, - } - }, - { - opcode: 'ws_recv', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.wsocket_recv'), - arguments: { - id: { - type: ArgumentType.STRING, - defaultValue: '0' - } - }, - }, - { - opcode: 'ws_close', - blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.wsocket_conclose'), - arguments: { - id: { - type: ArgumentType.STRING, - defaultValue: '0' - } - }, - }, - { - opcode: 'ws_destroy', - blockType: BlockType.COMMAND, - text: this.formatMessage('AusCOM.wsocket_close'), - arguments: { - id: { - type: ArgumentType.STRING, - defaultValue: '0' - } - }, - }, - { - opcode: 'ws_send', - blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.wsocket_send'), - arguments: { - id: { - type: ArgumentType.STRING, - defaultValue: '0' - }, - text: { - type: ArgumentType.STRING, - defaultValue: ' ' - } - }, - }, - // { - // opcode: 'sock_cre', - // blockType: BlockType.REPORTER, - // text: this.formatMessage('AusCOM.socket_create'), - // arguments: { - // - // yi: { - // type: ArgumentType.STRING, - // menu: 'sock_yi' - // }, - // - // method: { - // type: ArgumentType.STRING, - // menu: 'sock_method' - // }, - // buff: { - // type: ArgumentType.STRING, - // defaultValue: ' ' - // } - // }, - //}, - "---" + this.formatMessage("AusCOM.tit2"), - { - opcode: 'RC4_coding', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_RC4'), - arguments: { - KEY: { - type: ArgumentType.STRING, - defaultValue: 'Danny' - }, - TEXT: { - type: ArgumentType.STRING, - defaultValue: '我来了' - } - }, - }, - { - opcode: 'b64encoding', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base64_en'), - arguments: { - TEXT: { - type: ArgumentType.STRING, - defaultValue: 'Danny' - }, - }, - - }, - { - opcode: 'b64decoding', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base64_de'), - arguments: { - TEXT: { - type: ArgumentType.STRING, - defaultValue: 'RGFubnk=' - }, - }, - - }, - { - opcode: 'b58encoding', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base58_en'), - arguments: { - TEXT: { - type: ArgumentType.STRING, - defaultValue: 'Danny' - }, - }, - - }, - { - opcode: 'b58decoding', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base58_de'), - arguments: { - TEXT: { - type: ArgumentType.STRING, - defaultValue: '8iTbLRE' - }, - }, - - }, - { - opcode: 'bcostomencoding', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base_custom_en'), - arguments: { - ALPHABET: { - type: ArgumentType.STRING, - defaultValue: '!@#$%^&*()' - }, - TEXT: { - type: ArgumentType.STRING, - defaultValue: 'Danny' - }, - }, - - }, - { - opcode: 'bcostomdecoding', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base_custom_de'), - arguments: { - ALPHABET: { - type: ArgumentType.STRING, - defaultValue: '!@#$%^&*()' - }, - TEXT: { - type: ArgumentType.STRING, - defaultValue: '#)$&)#%!$$#@' - }, - }, - - }, - - "---" + this.formatMessage("AusCOM.tit3"), - { - opcode: 'json_create', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.json_create'), - arguments: { - json_: { - type: ArgumentType.STRING, - defaultValue: '{}' - }, - }, - }, - { - opcode: 'json_load', - blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_load'), - arguments: { - id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - json_: { - type: ArgumentType.STRING, - defaultValue: '{}' - }, - }, - }, - { - opcode: 'json_get_all_item', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.json_get_all_item'), - arguments: { - id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - }, - }, - { - opcode: 'json_add_from_id', - blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_add_from_id'), - arguments: { - _id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - name: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - }, - }, - { - opcode: 'json_add', - blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_add'), - arguments: { - - id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - type: { - type: ArgumentType.STRING, - menu: 'JSON_add' - }, - name: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - TEXT: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - }, - }, - { - opcode: 'json_get_item', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.json_get_item'), - arguments: { - - id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - name: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - - }, - }, - { - opcode: 'json_del_item', - blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_del_item'), - arguments: { - - id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - name: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - - }, - }, - { - opcode: 'GETjson', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.json_get'), - arguments: { - - id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - } - }, - { - opcode: 'DESTROYjson', - blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_destroy'), - arguments: { - - id: { - type: ArgumentType.STRING, - defaultValue: ' ' - }, - }, - }, - - { - opcode: 'help', - blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.help') + 'ver 1.00C1', - }, - - ], - menus: { - HTTP_method: [ - { - text: 'GET', - value: 'GET' - }, - { - text: 'POST', - value: 'POST' - }, - { - text: 'HEAD', - value: 'HEAD' - }, - { - text: 'PUT', - value: 'PUT' - }, - { - text: 'DELETE', - value: 'DELETE' - }, - { - text: 'OPTIONS', - value: 'OPTIONS' - }, - ], - ws_method: [ - { - text: 'WS', - value: 'WS' - }, - { - text: 'WSS', - value: 'WSS' - }, - ], - JSON_add: [ - { - text: this.formatMessage('AusCOM.str'), - value: '0' - }, - { - text: this.formatMessage('AusCOM.int'), - value: '1' - }, - { - text: this.formatMessage('AusCOM.float'), - value: '2' - }, - { - text: this.formatMessage('AusCOM.list'), - value: '3' - }, - { - text: this.formatMessage('AusCOM.logic'), - value: '4' - }, - ], - - sock_yi: [ - { - text: 'TCP', - value: '0' - }, - { - text: 'UDP', - value: '1' - }, - ], - sock_method: [ - { - text: 'client', - value: '0' - }, - { - text: 'server', - value: '1' - }, - ], - } - } - } - httpGet(theUrl, body, method) { - try { - var xmlHttp = new XMLHttpRequest(); - xmlHttp.open(method, theUrl, false); // false 为同步请求 - xmlHttp.send(body); - return xmlHttp.responseText; - } - catch (error) { - return error - } - } - - http(args) { - const { g_way, site, body } = args - return this.httpGet(site, body, g_way); - } - - ws_cre_sock(args) { - const { yi } = args - if (yi == 'WS') { - id = Date.now() - window.AusCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WS' }) - return String(id) - } - if (yi == 'WSS') { - id = Date.now() - window.AusCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WSS' }) - return String(id) - } - return '未知协议' - } - ws_conn_sock(args) { - const { id, host } = args - console.log(id, host) - for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { - if (String(id) == window.AusCOM.wsock[i]["id"]) { - for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { - if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { - delete window.AusCOM.wsnotalive_[String(id)] - } - } - - try { - window.AusCOM.wsock[i]['obj'] = new WebSocket(window.AusCOM.wsock[i]["way"] + '://' + host) - console.log(window.AusCOM.wsock) - window.AusCOM.wsock[i]['obj'].onmessage = function (evt) { - - window.abc = evt.data - window.AusCOM.wsrecv_.push({ 'id': window.AusCOM.wsock[i]['id'], 'nr': evt.data }) - } - window.AusCOM.wsock[i]['obj'].onclose = function (evt) { - window.AusCOM.wsnotalive_.push(window.AusCOM.wsock[i]['id']) - } - return true - } - catch (error) { - - return false - } - } - } - - return false - } - ws_recv(args) { - const { id } = args - console.log(window.AusCOM.wsrecv_) - for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { - if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { - console.log(window.AusCOM.wsrecv_) - try { - var a = String(window.AusCOM.wsrecv_[i]["nr"]) - window.AusCOM.wsrecv_.splice(i, 1); - return a - } - catch (error) { - - return '' - } - } - } - return false - } - ws_close(args) { - const { id } = args - for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { - if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { - window.AusCOM.wsnotalive_.splice(i, 1) - return true - } - } - return false - } - ws_destroy(args) { - const { id } = args - for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { - console.log(window.AusCOM.wsock[i]) - if (String(id) == window.AusCOM.wsock[i]["id"]) { - try { - if (window.AusCOM.wsock[i]["obj"] == null) { } - else { window.AusCOM.wsock[i]["obj"].close() } - window.AusCOM.wsock.splice(i, 1); - // alert(window.AusCOM.wsock) - return '' - } - catch (error) { console.log(error) } - } - } - } - ws_send(args) { - const { id, text } = args - for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { - if (String(id) == window.AusCOM.wsock[i]["id"]) { - console.log(window.AusCOM.wsrecv_) - console.log(window.AusCOM.wsock[i]) - - try { - if (window.AusCOM.wsock[i]["obj"] == null) { return false } else { window.AusCOM.wsock[i]["obj"].send(text); return true } - } - catch (error) { - - return false - } - } - - } - - return false - } - help() { - return "http访问出现fail to load XX可能是域名/ip不存在,也有可能是出现跨域问题,可以联系本站站住修改HTTP协议头或者自行修改浏览器跨域设置。如果ws无法发送数据,请F12打开开发者工具查看是否报错,如果报错则是无法连接,目前浏览器还不支持侦察此类报错。更多帮助请进入 https://gitee.com/ausx/scCOM" - } - RC4_coding(args) { - var key = '' - var data = '' - const { KEY, TEXT } = args - key = String(KEY) - data = String(TEXT)//对js不识别字符串进行处理 - var seq = Array(256); //如果有机会编译成离线可执行程序,会考虑支持AVX512 - var das = Array(data.length); - for (var i = 0; i < 256; i++) { - seq[i] = i; - var j = (j + seq[i] + key.charCodeAt(i % key.length)) % 256; - var temp = seq[i]; - seq[i] = seq[j]; - seq[j] = temp; - } - for (var i = 0; i < data.length; i++) { - das[i] = data.charCodeAt(i) - } - for (var x = 0; x < das.length; x++) { - var i = (i + 1) % 256; - var j = (j + seq[i]) % 256; - var temp = seq[i]; - seq[i] = seq[j]; - seq[j] = temp; - var k = (seq[i] + (seq[j] % 256)) % 256; - das[x] = String.fromCharCode(das[x] ^ seq[k]); - } - return das.join(''); - } - b64encoding(args) { - const { TEXT } = args - return window.btoa(encodeURIComponent(String(TEXT))); // 编码 - } - b64decoding(args) { - const { TEXT } = args - try { - return decodeURIComponent(window.atob(String(TEXT))); // 解码 - } - catch (error) { - return error - } - } - b58encoding(args) { - const { TEXT } = args - var ALPHABET = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'; - var ALPHABET_MAP = {}; - var BASE = 58; - for (var i = 0; i < ALPHABET.length; i++) { - ALPHABET_MAP[ALPHABET.charAt(i)] = i; - } - function ToUTF8(str) { - var result = new Array(); - - var k = 0; - for (var i = 0; i < str.length; i++) { - var j = encodeURI(str[i]); - if (j.length == 1) { - // 未转换的字符 - result[k++] = j.charCodeAt(0); - } else { - // 转换成%XX形式的字符 - var bytes = j.split("%"); - for (var l = 1; l < bytes.length; l++) { - result[k++] = parseInt("0x" + bytes[l]); - } - } - } - - return result; - } - - - // 如果有特殊需求,要转成utf16,可以用以下函数 - function ToUTF16(str) { - var result = new Array(); - - var k = 0; - for (var i = 0; i < str.length; i++) { - var j = str[i].charCodeAt(0); - result[k++] = j & 0xFF; - result[k++] = j >> 8; - } - - return result; - } - // 传进已经转成字节的数组 -->buffer(utf8格式) - function encode(buffer) { - if (buffer.length === 0) return ''; - var i, - j, - digits = [0]; - for (i = 0; i < buffer.length; i++) { - for (j = 0; j < digits.length; j++) { - digits[j] <<= 8; - } - digits[0] += buffer[i]; - var carry = 0; - for (j = 0; j < digits.length; ++j) { - digits[j] += carry; - carry = (digits[j] / BASE) | 0; - digits[j] %= BASE; - } - while (carry) { - digits.push(carry % BASE); - carry = (carry / BASE) | 0; - } - } - // deal with leading zeros - for (i = 0; buffer[i] === 0 && i < buffer.length - 1; i++) digits.push(0); - return digits - .reverse() - .map(function (digit) { - return ALPHABET[digit]; - }) - .join(''); - } - - - return encode(ToUTF8(String(TEXT))) - } - b58decoding(args) { - const { TEXT } = args - var ALPHABET = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'; - var ALPHABET_MAP = {}; - var BASE = 58; - for (var i = 0; i < ALPHABET.length; i++) { - ALPHABET_MAP[ALPHABET.charAt(i)] = i; - } - function decode(string) { - if (string.length === 0) return []; - var i, - j, - bytes = [0]; - for (i = 0; i < string.length; i++) { - var c = string[i]; - // c是不是ALPHABET_MAP的key - if (!(c in ALPHABET_MAP)) throw new Error('Non-base58 character'); - for (j = 0; j < bytes.length; j++) bytes[j] *= BASE; - bytes[0] += ALPHABET_MAP[c]; - var carry = 0; - for (j = 0; j < bytes.length; ++j) { - bytes[j] += carry; - carry = bytes[j] >> 8; - // 0xff --> 11111111 - bytes[j] &= 0xff; - } - while (carry) { - bytes.push(carry & 0xff); - carry >>= 8; - } - } - for (i = 0; string[i] === '1' && i < string.length - 1; i++) bytes.push(0); - return bytes.reverse(); - } - - function byteToString(arr) { - if (typeof arr === 'string') { - return arr; - } - var str = '', - _arr = arr; - for (var i = 0; i < _arr.length; i++) { - // 数组中每个数字转为二进制 匹配出开头为1的直到0的字符 - // eg:123-->"1111011"-->{0:"1111",groups: undefined, index: 0, input: "1111011"} - var one = _arr[i].toString(2), - v = one.match(/^1+?(?=0)/); - if (v && one.length == 8) { - var bytesLength = v[0].length; - var store = _arr[i].toString(2).slice(7 - bytesLength); - for (var st = 1; st < bytesLength; st++) { - store += _arr[st + i].toString(2).slice(2); - } - str += String.fromCharCode(parseInt(store, 2)); - i += bytesLength - 1; - } else { - str += String.fromCharCode(_arr[i]); - } - } - return str; - } - return byteToString(decode(String(TEXT))) - } - bcostomencoding(args) { - const { ALPHABET, TEXT } = args - var ALPHABET_MAP = {}; - var BASE = ALPHABET.length; - for (var i = 0; i < ALPHABET.length; i++) { - ALPHABET_MAP[ALPHABET.charAt(i)] = i; - } - function ToUTF8(str) { - var result = new Array(); - - var k = 0; - for (var i = 0; i < str.length; i++) { - var j = encodeURI(str[i]); - if (j.length == 1) { - // 未转换的字符 - result[k++] = j.charCodeAt(0); - } else { - // 转换成%XX形式的字符 - var bytes = j.split("%"); - for (var l = 1; l < bytes.length; l++) { - result[k++] = parseInt("0x" + bytes[l]); - } - } - } - - return result; - } - - - // 如果有特殊需求,要转成utf16,可以用以下函数 - function ToUTF16(str) { - var result = new Array(); - - var k = 0; - for (var i = 0; i < str.length; i++) { - var j = str[i].charCodeAt(0); - result[k++] = j & 0xFF; - result[k++] = j >> 8; - } - - return result; - } - // 传进已经转成字节的数组 -->buffer(utf8格式) - function encode(buffer) { - if (buffer.length === 0) return ''; - var i, - j, - digits = [0]; - for (i = 0; i < buffer.length; i++) { - for (j = 0; j < digits.length; j++) { - digits[j] <<= 8; - } - digits[0] += buffer[i]; - var carry = 0; - for (j = 0; j < digits.length; ++j) { - digits[j] += carry; - carry = (digits[j] / BASE) | 0; - digits[j] %= BASE; - } - while (carry) { - digits.push(carry % BASE); - carry = (carry / BASE) | 0; - } - } - // deal with leading zeros - for (i = 0; buffer[i] === 0 && i < buffer.length - 1; i++) digits.push(0); - return digits - .reverse() - .map(function (digit) { - return ALPHABET[digit]; - }) - .join(''); - } - - - return encode(ToUTF8(String(TEXT))) - } - bcostomdecoding(args) { - const { ALPHABET, TEXT } = args - var ALPHABET_MAP = {}; - var BASE = ALPHABET.length; - for (var i = 0; i < ALPHABET.length; i++) { - ALPHABET_MAP[ALPHABET.charAt(i)] = i; - } - function decode(string) { - if (string.length === 0) return []; - var i, - j, - bytes = [0]; - for (i = 0; i < string.length; i++) { - var c = string[i]; - // c是不是ALPHABET_MAP的key - if (!(c in ALPHABET_MAP)) throw new Error('Non-base58 character'); - for (j = 0; j < bytes.length; j++) bytes[j] *= BASE; - bytes[0] += ALPHABET_MAP[c]; - var carry = 0; - for (j = 0; j < bytes.length; ++j) { - bytes[j] += carry; - carry = bytes[j] >> 8; - // 0xff --> 11111111 - bytes[j] &= 0xff; - } - while (carry) { - bytes.push(carry & 0xff); - carry >>= 8; - } - } - for (i = 0; string[i] === '1' && i < string.length - 1; i++) bytes.push(0); - return bytes.reverse(); - } - - function byteToString(arr) { - if (typeof arr === 'string') { - return arr; - } - var str = '', - _arr = arr; - for (var i = 0; i < _arr.length; i++) { - // 数组中每个数字转为二进制 匹配出开头为1的直到0的字符 - // eg:123-->"1111011"-->{0:"1111",groups: undefined, index: 0, input: "1111011"} - var one = _arr[i].toString(2), - v = one.match(/^1+?(?=0)/); - if (v && one.length == 8) { - var bytesLength = v[0].length; - var store = _arr[i].toString(2).slice(7 - bytesLength); - for (var st = 1; st < bytesLength; st++) { - store += _arr[st + i].toString(2).slice(2); - } - str += String.fromCharCode(parseInt(store, 2)); - i += bytesLength - 1; - } else { - str += String.fromCharCode(_arr[i]); - } - } - return str; - } - return byteToString(decode(String(TEXT))) - } - json_create(args) { - const { json_ } = args - function create(j) { - id = Date.now() - a = {} - try { - a = JSON.parse(j) - } - catch (error) { } - window.AusCOM.json_list.push({ 'id': String(id), 'json': a }) - return id - } - return create(json_) - - } - json_load(args) { - const { id, json_ } = args - function load(id, j) { - var a = {} - try { - a = JSON.parse(j) - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - window.AusCOM.json_list[i]['json'] = a - return true - } - } - } - catch (error) { return false } - return false - - } - return load(id, json_) - } - json_get_all_item(args) { - const { id } = args - var b = [] - try { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - b = Object.keys(window.AusCOM.json_list[i]['json']) - } - } - } - catch (error) { b = [] } - var d = '' - for (let c of b) { - if (c == b[-1]) { - d += c - } - else { - d += c + ' ' - } - } - - return d - } - check(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return true - } - } - return false - } - getjson(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return window.AusCOM.json_list[i]['json'] - } - } - return {} - } - json_add_from_id(args) { - const { _id, id, name } = args - function check(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return true - } - } - return false - } - function getjson(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return window.AusCOM.json_list[i]['json'] - } - } - return {} - } - function add_item_from_id(id, id1, name) { - if (check(id) & check(id1)) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - window.AusCOM.json_list[i]['json'][name] = getjson(id1) - return true - } - } - } - else { return false } - return false - } - return add_item_from_id(id, _id, name) - } - json_add(args) { - const { id, type, name, TEXT } = args - function check(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return true - } - } - return false - } - function getjson(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return window.AusCOM.json_list[i]['json'] - } - } - return {} - } - function add_item(id, name, type, content) { - if (check(id)) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - if (type == '0') {//字符串 - window.AusCOM.json_list[i]['json'][String(name)] = String(content) - return true - } - if (type == '1') {//整数 - window.AusCOM.json_list[i]['json'][String(name)] = parseInt(content) - return true - } - if (type == '2') {//浮点数 - window.AusCOM.json_list[i]['json'][String(name)] = parseFloat(content) - return true - } - if (type == '3') {//列表 - window.AusCOM.json_list[i]['json'][String(name)] = content.split(' ') - return true - } - if (type == '4') {//逻辑 - if (String(content) == 'true') { - window.AusCOM.json_list[i]['json'][String(name)] = true - return true - } - else { - window.AusCOM.json_list[i]['json'][String(name)] = false - return true - } - } - - } - } - - } - return false - } - return add_item(id, name, type, TEXT) - } - GETjson(args) { - const { id } = args - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return JSON.stringify(window.AusCOM.json_list[i]['json']) - } - } - return '' - } - DESTROYjson(args) { - const { id } = args - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - window.AusCOM.json_list.splice(i, 1); - return true - } - } - return false - } - json_get_item(args) { // - const { id, name } = args - try { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'string') { //id_0 字符串 - return window.AusCOM.json_list[i]['json'][name] - } - if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'number') { //id_1&2 整数&浮点 - return window.AusCOM.json_list[i]['json'][name] - } - if (window.AusCOM.json_list[i]['json'][name] instanceof Array) { //id_4列表 - var a = '' - for (let i_ = 0, len = window.AusCOM.json_list[i]['json'][name].length; i_ < len; i_++) { - if (i_ == window.AusCOM.json_list[i]['json'][name].length) { - a += window.AusCOM.json_list[i]['json'][name][i] - } - else { - a += window.AusCOM.json_list[i]['json'][name][i] + ' ' - } - } - } - if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'boolean') { // id_4 逻辑 - return window.AusCOM.json_list[i]['json'][name] - } - - } - } - } catch { } - return '' - } - json_del_item(args) { //删除项 - const { id, name } = args - try { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - Reflect.deleteProperty(window.AusCOM.json_list[i]['json'], String(name)) - return true - } - } - } catch { } - return false - } - _ws(host, data) { - id = this.ws_conn_sock({ 'id': 'WS', 'host': host }) - this.ws_send({ 'id': id, 'text': data }) - recv = this.ws_recv({ 'id': id }) - for (; ;) { if (recv != '') { break } } - return recv - } - - sock_cre(args) { - const { yi, method } = args - j = JSON.parse(this._ws('127.0.0.1:23089/socket/create', JSON.stringify({ 'yi': yi, 'method': method }))) - return j['id'] - - } -} -module.exports = AuEx_communication; From 24326a8c94bed3b035fa9661ddaac2a81f6b45a9 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sat, 1 Oct 2022 18:21:44 +0800 Subject: [PATCH 155/357] Create scCOM.js --- dannydev/scCOM.js | 1254 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 1254 insertions(+) create mode 100644 dannydev/scCOM.js diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js new file mode 100644 index 00000000..7d68d64d --- /dev/null +++ b/dannydev/scCOM.js @@ -0,0 +1,1254 @@ +const Cast = require('../../util/cast'); +const ArgumentType = require('../../extension-support/argument-type'); +const BlockType = require('../../extension-support/block-type'); +// var dgram = require("dgram"); +//感谢Arkos开源的扩展代码,作为参照编写框架 +//欸,别把作者忘了。Danny欸 +class AuEx_communication { + constructor(runtime) { + this.runtime = runtime + this._formatMessage = runtime.getFormatMessage({ + 'zh-cn': { + 'AusCOM.extensionName': 'AUSの通讯模块', + 'AusCOM.tit1': 'sock通讯', + 'AusCOM.tit2': '数据处理', + 'AusCOM.tit3': 'JSON', + 'AusCOM.http': 'HTTP以[g_way]方式访问[site]发送[body]', + 'AusCOM.wsocket_create': 'ws.创建websocket,协议[yi],并返回ccid', + 'AusCOM.wsocket_connect': 'ws.id[id]socket连接[host]', + 'AusCOM.wsocket_send': 'ws.发送id[id]数据[text]', + 'AusCOM.wsocket_recv': 'ws.接收数据id[id]', + 'AusCOM.wsocket_conclose': 'ws.连接关闭id[id]', + 'AusCOM.wsocket_close': 'ws.销毁websocket id[id]', + 'AusCOM.socket_create': 'sock.创建[yi]方式[method]并返回ccid', + 'AusCOM.socket_create_chk': 'sock.创建是否成功[ccid]并返回id', + 'AusCOM.socket_recv': 'sock.收到数据id [id]', + 'AusCOM.socket_send': 'sock.发送数据id [id] ip[ip]数据[TEXT]', + 'AusCOM.socket_conclose': 'sock.断开连接id[id]', + 'AusCOM.socket_connect': 'sock.连接/启动服务 id[id] ip [ip] 接收长度[buff]', + 'AusCOM.socket_breakev': 'sock.断开连接id[id]ip[ip]', + 'AusCOM.socket_close': 'sock.销毁socket对象 id[id]', + 'AusCOM.socket_parse': 'sock.解析返回的数据data[data]属性[type]', + 'AusCOM.socket_opchk': 'sock.操作是否成功[cid]', + 'AusCOM.code_RC4': 'RC4加解密 密钥[KEY] 文本[TEXT]', + 'AusCOM.code_base64_en': 'base64编码文本[TEXT]', + 'AusCOM.code_base64_de': 'base64解码文本[TEXT]', + 'AusCOM.code_base58_en': 'base58编码文本[TEXT]', + 'AusCOM.code_base58_de': 'base58解码码文本[TEXT]', + 'AusCOM.code_base_custom_en': '自定义base加密 字符集[ALPHABET] 文本[TEXT]', + 'AusCOM.code_base_custom_de': '自定义base解密 字符集[ALPHABET] 文本[TEXT]', + 'AusCOM.str': '字符串', + 'AusCOM.int': '整数', + 'AusCOM.float': '浮点数', + 'AusCOM.list': '列表', + 'AusCOM.logic': '逻辑', + 'AusCOM.json_create': '创建json 载入[json_] 并返回id', + 'AusCOM.json_load': '载入json id[id]json[json_]', + 'AusCOM.json_get_all_item': '获取json id[id]中所有属性名', + 'AusCOM.json_add_from_id': '将id[_id]对应的json加入json/改变值 id[id] 属性名[name]', + 'AusCOM.json_add': '加入/改变对象到json id[id]类型[type]属性名[name]内容[TEXT]', + 'AusCOM.json_get_item': '获取json id [id]中某属性名为[name]项的值', + 'AusCOM.json_del_item': '删除json id [id]中某属性名为[name]项的值', + 'AusCOM.json_get': '获取json id[id]', + 'AusCOM.json_destroy': '销毁json id[id]', + 'AusCOM.help': '使用帮助', + }, + en: { + 'AusCOM.extensionName': 'AUSのcommunication', + 'AusCOM.tit1': 'sock communication', + 'AusCOM.tit2': 'solve data', + 'AusCOM.tit3': 'JSON', + 'AusCOM.http': 'http access[g_way]to[site]send[body]', + 'AusCOM.wsocket_create': 'ws.create websocket,protocol', 'AusCOM.wsocket_connect': 'id[id]websocket connect[host]', + 'AusCOM.wsocket_send': 'ws.send id[id] data[text]', + 'AusCOM.wsocket_recv': 'ws.recv id [id]', + 'AusCOM.wsocket_conclose': 'ws.When connection closed id[id]', + 'AusCOM.wsocket_close': 'ws.destroy socket id[id]', + 'AusCOM.socket_ sreate ': 'sock. Create [yi] method [method] and return ID', + 'AusCOM.socket_create_chk': 'sock.create successfully[ccid]?', + 'AusCOM.socket_ recv': 'sock. Received data ID [id] ', + 'AusCOM.socket_ send': 'sock. Send data ID [id] data [text] ', + 'AusCOM.socket_ close ': 'sock. Connection closing ID [id]', + 'AusCOM.socket_ connect': 'sock. Connect to/start server ID [id] IP [ip] ,buff[buff]', + 'AusCOM.socket_breakev': 'sock.break connection off id[id]ip[ip]', + 'AusCOM.socket_ close': 'sock. Destroy the socket object ID [id] ', + 'AusCOM.socket_ parse ': 'sock. Parse the returned data [data] attribute [type]', + 'AusCOM.socket_opchk': 'sock.run operation successfully[cid]?', + 'AusCOM.code_RC4': 'RC4 KEY[KEY] TEXT[TEXT]', + 'AusCOM.code_base64_en': 'base64 encoding[TEXT]', + 'AusCOM.code_base64_de': 'base64 decoding[TEXT]', + 'AusCOM.code_base58_en': 'base58 encoding[TEXT]', + 'AusCOM.code_base58_de': 'base58 decoding[TEXT]', + 'AusCOM.code_base_custom_en': 'Custom base encryption,ALPHABET[ALPHABET] TEXT[TEXT]', + 'AusCOM.code_base_custom_de': 'Custom base decryption,ALPHABET[ALPHABET] TEXT[TEXT]', + 'AusCOM.str': 'string', + 'AusCOM.int': 'int', + 'AusCOM.float': 'float', + 'AusCOM.list': 'list', + 'AusCOM.logic': 'logic', + 'AusCOM.json_create': 'create json,load[json_] and return id', + 'AusCOM.json_load': ' load json id[id]json[json_]', + 'AusCOM.json_get_all_item': 'get all attribute name of json id[id]', + 'AusCOM.json_add_from_id': 'Add the JSON corresponding to ID [_id] to JSON/change a value ID [id] attribute name [name]', + 'AusCOM.json_add': 'Add object/change a value to JSON ID [id], type [type], attribute name [name], content [TEXT]', + 'AusCOM.json_get_item': 'Get the value of an attribute named [name] in JSON ID [id]', + 'AusCOM.json_del_item': 'Delete the value of an attribute named [name] in JSON ID [ID]', + 'AusCOM.json_get': 'get json id[id]', + 'AusCOM.json_destroy': 'destroy json id[id]', + 'AusCOM.help': 'help', + }, + }) + this.aulink = new Worker('./aurora_link.js') + window.AusCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [] } + this.icon="" + } + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id, + }) + } + getInfo() { + return { + id: 'DAdevNet', // 拓展id + name: this.formatMessage('AusCOM.extensionName'), // 拓展名 + color1: '#FF8383', + blockIconURI: this.icon, + menuIconURI: this.icon, + blocks: [ + "---" + this.formatMessage("AusCOM.tit1"), + { + opcode: 'http', + blockType: 'reporter', + text: this.formatMessage('AusCOM.http'), + arguments: { + g_way: { + type: 'string', + menu: 'HTTP_method' + }, + site: { + type: 'string', + defaultValue: '/' + }, + body: { + type: 'string', + defaultValue: ' ' + }, + } + }, + + { + opcode: 'ws_cre_sock', + blockType: 'reporter', + text: this.formatMessage('AusCOM.wsocket_create'), + arguments: { + yi: { + type: 'string', + menu: 'ws_method' + }, + } + }, + { + opcode: 'ws_conn_sock', + blockType: 'Boolean', + text: this.formatMessage('AusCOM.wsocket_connect'), + arguments: { + id: { + type: 'string', + defaultValue: '0' + }, + host: { + type: 'string', + defaultValue: '0.0.0.0:1212/' + }, + } + }, + { + opcode: 'ws_recv', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.wsocket_recv'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: '0' + } + }, + }, + { + opcode: 'ws_close', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.wsocket_conclose'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: '0' + } + }, + }, + { + opcode: 'ws_destroy', + blockType: BlockType.COMMAND, + text: this.formatMessage('AusCOM.wsocket_close'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: '0' + } + }, + }, + { + opcode: 'ws_send', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.wsocket_send'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: '0' + }, + text: { + type: ArgumentType.STRING, + defaultValue: ' ' + } + }, + }, + // { + // opcode: 'sock_cre', + // blockType: BlockType.REPORTER, + // text: this.formatMessage('AusCOM.socket_create'), + // arguments: { + // + // yi: { + // type: ArgumentType.STRING, + // menu: 'sock_yi' + // }, + // + // method: { + // type: ArgumentType.STRING, + // menu: 'sock_method' + // }, + // buff: { + // type: ArgumentType.STRING, + // defaultValue: ' ' + // } + // }, + //}, + "---" + this.formatMessage("AusCOM.tit2"), + { + opcode: 'RC4_coding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_RC4'), + arguments: { + KEY: { + type: ArgumentType.STRING, + defaultValue: 'Danny' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '我来了' + } + }, + }, + { + opcode: 'b64encoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base64_en'), + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: 'Danny' + }, + }, + + }, + { + opcode: 'b64decoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base64_de'), + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: 'RGFubnk=' + }, + }, + + }, + { + opcode: 'b58encoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base58_en'), + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: 'Danny' + }, + }, + + }, + { + opcode: 'b58decoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base58_de'), + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: '8iTbLRE' + }, + }, + + }, + { + opcode: 'bcostomencoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base_custom_en'), + arguments: { + ALPHABET: { + type: ArgumentType.STRING, + defaultValue: '!@#$%^&*()' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: 'Danny' + }, + }, + + }, + { + opcode: 'bcostomdecoding', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.code_base_custom_de'), + arguments: { + ALPHABET: { + type: ArgumentType.STRING, + defaultValue: '!@#$%^&*()' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '#)$&)#%!$$#@' + }, + }, + + }, + + "---" + this.formatMessage("AusCOM.tit3"), + { + opcode: 'json_create', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.json_create'), + arguments: { + json_: { + type: ArgumentType.STRING, + defaultValue: '{}' + }, + }, + }, + { + opcode: 'json_load', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_load'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + json_: { + type: ArgumentType.STRING, + defaultValue: '{}' + }, + }, + }, + { + opcode: 'json_get_all_item', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.json_get_all_item'), + arguments: { + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + }, + }, + { + opcode: 'json_add_from_id', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_add_from_id'), + arguments: { + _id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + name: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + }, + }, + { + opcode: 'json_add', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_add'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + type: { + type: ArgumentType.STRING, + menu: 'JSON_add' + }, + name: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + }, + }, + { + opcode: 'json_get_item', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.json_get_item'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + name: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + + }, + }, + { + opcode: 'json_del_item', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_del_item'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + name: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + + }, + }, + { + opcode: 'GETjson', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.json_get'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + } + }, + { + opcode: 'DESTROYjson', + blockType: BlockType.BOOLEAN, + text: this.formatMessage('AusCOM.json_destroy'), + arguments: { + + id: { + type: ArgumentType.STRING, + defaultValue: ' ' + }, + }, + }, + + { + opcode: 'help', + blockType: BlockType.REPORTER, + text: this.formatMessage('AusCOM.help') + 'ver 1.00C1', + }, + + ], + menus: { + HTTP_method: [ + { + text: 'GET', + value: 'GET' + }, + { + text: 'POST', + value: 'POST' + }, + { + text: 'HEAD', + value: 'HEAD' + }, + { + text: 'PUT', + value: 'PUT' + }, + { + text: 'DELETE', + value: 'DELETE' + }, + { + text: 'OPTIONS', + value: 'OPTIONS' + }, + ], + ws_method: [ + { + text: 'WS', + value: 'WS' + }, + { + text: 'WSS', + value: 'WSS' + }, + ], + JSON_add: [ + { + text: this.formatMessage('AusCOM.str'), + value: '0' + }, + { + text: this.formatMessage('AusCOM.int'), + value: '1' + }, + { + text: this.formatMessage('AusCOM.float'), + value: '2' + }, + { + text: this.formatMessage('AusCOM.list'), + value: '3' + }, + { + text: this.formatMessage('AusCOM.logic'), + value: '4' + }, + ], + + sock_yi: [ + { + text: 'TCP', + value: '0' + }, + { + text: 'UDP', + value: '1' + }, + ], + sock_method: [ + { + text: 'client', + value: '0' + }, + { + text: 'server', + value: '1' + }, + ], + } + } + } + httpGet(theUrl, body, method) { + try { + var xmlHttp = new XMLHttpRequest(); + xmlHttp.open(method, theUrl, false); // false 为同步请求 + xmlHttp.send(body); + return xmlHttp.responseText; + } + catch (error) { + return error + } + } + + http(args) { + const { g_way, site, body } = args + return this.httpGet(site, body, g_way); + } + + ws_cre_sock(args) { + const { yi } = args + if (yi == 'WS') { + id = Date.now() + window.AusCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WS' }) + return String(id) + } + if (yi == 'WSS') { + id = Date.now() + window.AusCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WSS' }) + return String(id) + } + return '未知协议' + } + ws_conn_sock(args) { + const { id, host } = args + console.log(id, host) + for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { + if (String(id) == window.AusCOM.wsock[i]["id"]) { + for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { + delete window.AusCOM.wsnotalive_[String(id)] + } + } + + try { + window.AusCOM.wsock[i]['obj'] = new WebSocket(window.AusCOM.wsock[i]["way"] + '://' + host) + console.log(window.AusCOM.wsock) + window.AusCOM.wsock[i]['obj'].onmessage = function (evt) { + + window.abc = evt.data + window.AusCOM.wsrecv_.push({ 'id': window.AusCOM.wsock[i]['id'], 'nr': evt.data }) + } + window.AusCOM.wsock[i]['obj'].onclose = function (evt) { + window.AusCOM.wsnotalive_.push(window.AusCOM.wsock[i]['id']) + } + return true + } + catch (error) { + + return false + } + } + } + + return false + } + ws_recv(args) { + const { id } = args + console.log(window.AusCOM.wsrecv_) + for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { + console.log(window.AusCOM.wsrecv_) + try { + var a = String(window.AusCOM.wsrecv_[i]["nr"]) + window.AusCOM.wsrecv_.splice(i, 1); + return a + } + catch (error) { + + return '' + } + } + } + return false + } + ws_close(args) { + const { id } = args + for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { + window.AusCOM.wsnotalive_.splice(i, 1) + return true + } + } + return false + } + ws_destroy(args) { + const { id } = args + for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { + console.log(window.AusCOM.wsock[i]) + if (String(id) == window.AusCOM.wsock[i]["id"]) { + try { + if (window.AusCOM.wsock[i]["obj"] == null) { } + else { window.AusCOM.wsock[i]["obj"].close() } + window.AusCOM.wsock.splice(i, 1); + // alert(window.AusCOM.wsock) + return '' + } + catch (error) { console.log(error) } + } + } + } + ws_send(args) { + const { id, text } = args + for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { + if (String(id) == window.AusCOM.wsock[i]["id"]) { + console.log(window.AusCOM.wsrecv_) + console.log(window.AusCOM.wsock[i]) + + try { + if (window.AusCOM.wsock[i]["obj"] == null) { return false } else { window.AusCOM.wsock[i]["obj"].send(text); return true } + } + catch (error) { + + return false + } + } + + } + + return false + } + help() { + return "http访问出现fail to load XX可能是域名/ip不存在,也有可能是出现跨域问题,可以联系本站站住修改HTTP协议头或者自行修改浏览器跨域设置。如果ws无法发送数据,请F12打开开发者工具查看是否报错,如果报错则是无法连接,目前浏览器还不支持侦察此类报错。更多帮助请进入 https://gitee.com/ausx/scCOM" + } + RC4_coding(args) { + var key = '' + var data = '' + const { KEY, TEXT } = args + key = String(KEY) + data = String(TEXT)//对js不识别字符串进行处理 + var seq = Array(256); //如果有机会编译成离线可执行程序,会考虑支持AVX512 + var das = Array(data.length); + for (var i = 0; i < 256; i++) { + seq[i] = i; + var j = (j + seq[i] + key.charCodeAt(i % key.length)) % 256; + var temp = seq[i]; + seq[i] = seq[j]; + seq[j] = temp; + } + for (var i = 0; i < data.length; i++) { + das[i] = data.charCodeAt(i) + } + for (var x = 0; x < das.length; x++) { + var i = (i + 1) % 256; + var j = (j + seq[i]) % 256; + var temp = seq[i]; + seq[i] = seq[j]; + seq[j] = temp; + var k = (seq[i] + (seq[j] % 256)) % 256; + das[x] = String.fromCharCode(das[x] ^ seq[k]); + } + return das.join(''); + } + b64encoding(args) { + const { TEXT } = args + return window.btoa(encodeURIComponent(String(TEXT))); // 编码 + } + b64decoding(args) { + const { TEXT } = args + try { + return decodeURIComponent(window.atob(String(TEXT))); // 解码 + } + catch (error) { + return error + } + } + b58encoding(args) { + const { TEXT } = args + var ALPHABET = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'; + var ALPHABET_MAP = {}; + var BASE = 58; + for (var i = 0; i < ALPHABET.length; i++) { + ALPHABET_MAP[ALPHABET.charAt(i)] = i; + } + function ToUTF8(str) { + var result = new Array(); + + var k = 0; + for (var i = 0; i < str.length; i++) { + var j = encodeURI(str[i]); + if (j.length == 1) { + // 未转换的字符 + result[k++] = j.charCodeAt(0); + } else { + // 转换成%XX形式的字符 + var bytes = j.split("%"); + for (var l = 1; l < bytes.length; l++) { + result[k++] = parseInt("0x" + bytes[l]); + } + } + } + + return result; + } + + + // 如果有特殊需求,要转成utf16,可以用以下函数 + function ToUTF16(str) { + var result = new Array(); + + var k = 0; + for (var i = 0; i < str.length; i++) { + var j = str[i].charCodeAt(0); + result[k++] = j & 0xFF; + result[k++] = j >> 8; + } + + return result; + } + // 传进已经转成字节的数组 -->buffer(utf8格式) + function encode(buffer) { + if (buffer.length === 0) return ''; + var i, + j, + digits = [0]; + for (i = 0; i < buffer.length; i++) { + for (j = 0; j < digits.length; j++) { + digits[j] <<= 8; + } + digits[0] += buffer[i]; + var carry = 0; + for (j = 0; j < digits.length; ++j) { + digits[j] += carry; + carry = (digits[j] / BASE) | 0; + digits[j] %= BASE; + } + while (carry) { + digits.push(carry % BASE); + carry = (carry / BASE) | 0; + } + } + // deal with leading zeros + for (i = 0; buffer[i] === 0 && i < buffer.length - 1; i++) digits.push(0); + return digits + .reverse() + .map(function (digit) { + return ALPHABET[digit]; + }) + .join(''); + } + + + return encode(ToUTF8(String(TEXT))) + } + b58decoding(args) { + const { TEXT } = args + var ALPHABET = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'; + var ALPHABET_MAP = {}; + var BASE = 58; + for (var i = 0; i < ALPHABET.length; i++) { + ALPHABET_MAP[ALPHABET.charAt(i)] = i; + } + function decode(string) { + if (string.length === 0) return []; + var i, + j, + bytes = [0]; + for (i = 0; i < string.length; i++) { + var c = string[i]; + // c是不是ALPHABET_MAP的key + if (!(c in ALPHABET_MAP)) throw new Error('Non-base58 character'); + for (j = 0; j < bytes.length; j++) bytes[j] *= BASE; + bytes[0] += ALPHABET_MAP[c]; + var carry = 0; + for (j = 0; j < bytes.length; ++j) { + bytes[j] += carry; + carry = bytes[j] >> 8; + // 0xff --> 11111111 + bytes[j] &= 0xff; + } + while (carry) { + bytes.push(carry & 0xff); + carry >>= 8; + } + } + for (i = 0; string[i] === '1' && i < string.length - 1; i++) bytes.push(0); + return bytes.reverse(); + } + + function byteToString(arr) { + if (typeof arr === 'string') { + return arr; + } + var str = '', + _arr = arr; + for (var i = 0; i < _arr.length; i++) { + // 数组中每个数字转为二进制 匹配出开头为1的直到0的字符 + // eg:123-->"1111011"-->{0:"1111",groups: undefined, index: 0, input: "1111011"} + var one = _arr[i].toString(2), + v = one.match(/^1+?(?=0)/); + if (v && one.length == 8) { + var bytesLength = v[0].length; + var store = _arr[i].toString(2).slice(7 - bytesLength); + for (var st = 1; st < bytesLength; st++) { + store += _arr[st + i].toString(2).slice(2); + } + str += String.fromCharCode(parseInt(store, 2)); + i += bytesLength - 1; + } else { + str += String.fromCharCode(_arr[i]); + } + } + return str; + } + return byteToString(decode(String(TEXT))) + } + bcostomencoding(args) { + const { ALPHABET, TEXT } = args + var ALPHABET_MAP = {}; + var BASE = ALPHABET.length; + for (var i = 0; i < ALPHABET.length; i++) { + ALPHABET_MAP[ALPHABET.charAt(i)] = i; + } + function ToUTF8(str) { + var result = new Array(); + + var k = 0; + for (var i = 0; i < str.length; i++) { + var j = encodeURI(str[i]); + if (j.length == 1) { + // 未转换的字符 + result[k++] = j.charCodeAt(0); + } else { + // 转换成%XX形式的字符 + var bytes = j.split("%"); + for (var l = 1; l < bytes.length; l++) { + result[k++] = parseInt("0x" + bytes[l]); + } + } + } + + return result; + } + + + // 如果有特殊需求,要转成utf16,可以用以下函数 + function ToUTF16(str) { + var result = new Array(); + + var k = 0; + for (var i = 0; i < str.length; i++) { + var j = str[i].charCodeAt(0); + result[k++] = j & 0xFF; + result[k++] = j >> 8; + } + + return result; + } + // 传进已经转成字节的数组 -->buffer(utf8格式) + function encode(buffer) { + if (buffer.length === 0) return ''; + var i, + j, + digits = [0]; + for (i = 0; i < buffer.length; i++) { + for (j = 0; j < digits.length; j++) { + digits[j] <<= 8; + } + digits[0] += buffer[i]; + var carry = 0; + for (j = 0; j < digits.length; ++j) { + digits[j] += carry; + carry = (digits[j] / BASE) | 0; + digits[j] %= BASE; + } + while (carry) { + digits.push(carry % BASE); + carry = (carry / BASE) | 0; + } + } + // deal with leading zeros + for (i = 0; buffer[i] === 0 && i < buffer.length - 1; i++) digits.push(0); + return digits + .reverse() + .map(function (digit) { + return ALPHABET[digit]; + }) + .join(''); + } + + + return encode(ToUTF8(String(TEXT))) + } + bcostomdecoding(args) { + const { ALPHABET, TEXT } = args + var ALPHABET_MAP = {}; + var BASE = ALPHABET.length; + for (var i = 0; i < ALPHABET.length; i++) { + ALPHABET_MAP[ALPHABET.charAt(i)] = i; + } + function decode(string) { + if (string.length === 0) return []; + var i, + j, + bytes = [0]; + for (i = 0; i < string.length; i++) { + var c = string[i]; + // c是不是ALPHABET_MAP的key + if (!(c in ALPHABET_MAP)) throw new Error('Non-base58 character'); + for (j = 0; j < bytes.length; j++) bytes[j] *= BASE; + bytes[0] += ALPHABET_MAP[c]; + var carry = 0; + for (j = 0; j < bytes.length; ++j) { + bytes[j] += carry; + carry = bytes[j] >> 8; + // 0xff --> 11111111 + bytes[j] &= 0xff; + } + while (carry) { + bytes.push(carry & 0xff); + carry >>= 8; + } + } + for (i = 0; string[i] === '1' && i < string.length - 1; i++) bytes.push(0); + return bytes.reverse(); + } + + function byteToString(arr) { + if (typeof arr === 'string') { + return arr; + } + var str = '', + _arr = arr; + for (var i = 0; i < _arr.length; i++) { + // 数组中每个数字转为二进制 匹配出开头为1的直到0的字符 + // eg:123-->"1111011"-->{0:"1111",groups: undefined, index: 0, input: "1111011"} + var one = _arr[i].toString(2), + v = one.match(/^1+?(?=0)/); + if (v && one.length == 8) { + var bytesLength = v[0].length; + var store = _arr[i].toString(2).slice(7 - bytesLength); + for (var st = 1; st < bytesLength; st++) { + store += _arr[st + i].toString(2).slice(2); + } + str += String.fromCharCode(parseInt(store, 2)); + i += bytesLength - 1; + } else { + str += String.fromCharCode(_arr[i]); + } + } + return str; + } + return byteToString(decode(String(TEXT))) + } + json_create(args) { + const { json_ } = args + function create(j) { + id = Date.now() + a = {} + try { + a = JSON.parse(j) + } + catch (error) { } + window.AusCOM.json_list.push({ 'id': String(id), 'json': a }) + return id + } + return create(json_) + + } + json_load(args) { + const { id, json_ } = args + function load(id, j) { + var a = {} + try { + a = JSON.parse(j) + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + window.AusCOM.json_list[i]['json'] = a + return true + } + } + } + catch (error) { return false } + return false + + } + return load(id, json_) + } + json_get_all_item(args) { + const { id } = args + var b = [] + try { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + b = Object.keys(window.AusCOM.json_list[i]['json']) + } + } + } + catch (error) { b = [] } + var d = '' + for (let c of b) { + if (c == b[-1]) { + d += c + } + else { + d += c + ' ' + } + } + + return d + } + check(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return true + } + } + return false + } + getjson(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return window.AusCOM.json_list[i]['json'] + } + } + return {} + } + json_add_from_id(args) { + const { _id, id, name } = args + function check(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return true + } + } + return false + } + function getjson(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return window.AusCOM.json_list[i]['json'] + } + } + return {} + } + function add_item_from_id(id, id1, name) { + if (check(id) & check(id1)) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + window.AusCOM.json_list[i]['json'][name] = getjson(id1) + return true + } + } + } + else { return false } + return false + } + return add_item_from_id(id, _id, name) + } + json_add(args) { + const { id, type, name, TEXT } = args + function check(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return true + } + } + return false + } + function getjson(id) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return window.AusCOM.json_list[i]['json'] + } + } + return {} + } + function add_item(id, name, type, content) { + if (check(id)) { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + if (type == '0') {//字符串 + window.AusCOM.json_list[i]['json'][String(name)] = String(content) + return true + } + if (type == '1') {//整数 + window.AusCOM.json_list[i]['json'][String(name)] = parseInt(content) + return true + } + if (type == '2') {//浮点数 + window.AusCOM.json_list[i]['json'][String(name)] = parseFloat(content) + return true + } + if (type == '3') {//列表 + window.AusCOM.json_list[i]['json'][String(name)] = content.split(' ') + return true + } + if (type == '4') {//逻辑 + if (String(content) == 'true') { + window.AusCOM.json_list[i]['json'][String(name)] = true + return true + } + else { + window.AusCOM.json_list[i]['json'][String(name)] = false + return true + } + } + + } + } + + } + return false + } + return add_item(id, name, type, TEXT) + } + GETjson(args) { + const { id } = args + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + return JSON.stringify(window.AusCOM.json_list[i]['json']) + } + } + return '' + } + DESTROYjson(args) { + const { id } = args + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + window.AusCOM.json_list.splice(i, 1); + return true + } + } + return false + } + json_get_item(args) { // + const { id, name } = args + try { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'string') { //id_0 字符串 + return window.AusCOM.json_list[i]['json'][name] + } + if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'number') { //id_1&2 整数&浮点 + return window.AusCOM.json_list[i]['json'][name] + } + if (window.AusCOM.json_list[i]['json'][name] instanceof Array) { //id_4列表 + var a = '' + for (let i_ = 0, len = window.AusCOM.json_list[i]['json'][name].length; i_ < len; i_++) { + if (i_ == window.AusCOM.json_list[i]['json'][name].length) { + a += window.AusCOM.json_list[i]['json'][name][i] + } + else { + a += window.AusCOM.json_list[i]['json'][name][i] + ' ' + } + } + } + if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'boolean') { // id_4 逻辑 + return window.AusCOM.json_list[i]['json'][name] + } + + } + } + } catch { } + return '' + } + json_del_item(args) { //删除项 + const { id, name } = args + try { + for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { + if (window.AusCOM.json_list[i]['id'] == String(id)) { + Reflect.deleteProperty(window.AusCOM.json_list[i]['json'], String(name)) + return true + } + } + } catch { } + return false + } + _ws(host, data) { + id = this.ws_conn_sock({ 'id': 'WS', 'host': host }) + this.ws_send({ 'id': id, 'text': data }) + recv = this.ws_recv({ 'id': id }) + for (; ;) { if (recv != '') { break } } + return recv + } + + sock_cre(args) { + const { yi, method } = args + j = JSON.parse(this._ws('127.0.0.1:23089/socket/create', JSON.stringify({ 'yi': yi, 'method': method }))) + return j['id'] + + } +} +module.exports = AuEx_communication; From 2f6c5245de1a4f2a18d1444c754ba0af0d96b6b0 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sun, 2 Oct 2022 08:14:45 +0800 Subject: [PATCH 156/357] Update scCOM.js --- dannydev/scCOM.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index 7d68d64d..e95b2442 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -111,7 +111,7 @@ class AuEx_communication { } getInfo() { return { - id: 'DAdevNet', // 拓展id + id: 'AUdevNet', // 拓展id name: this.formatMessage('AusCOM.extensionName'), // 拓展名 color1: '#FF8383', blockIconURI: this.icon, From 118e86d753a897296cc7b0c8e66f3a816c5660c0 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sun, 2 Oct 2022 22:06:52 +0800 Subject: [PATCH 157/357] Update scCOM.js --- dannydev/scCOM.js | 116 ++++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 106 insertions(+), 10 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index e95b2442..7f0c3368 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1,6 +1,7 @@ const Cast = require('../../util/cast'); const ArgumentType = require('../../extension-support/argument-type'); const BlockType = require('../../extension-support/block-type'); +const { string } = require('prop-types'); // var dgram = require("dgram"); //感谢Arkos开源的扩展代码,作为参照编写框架 //欸,别把作者忘了。Danny欸 @@ -10,10 +11,14 @@ class AuEx_communication { this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { 'AusCOM.extensionName': 'AUSの通讯模块', - 'AusCOM.tit1': 'sock通讯', + 'AusCOM.tit1': '通讯', + 'AusCOM.tit1.1': 'http', + 'AusCOM.tit1.2': 'WebSocket', 'AusCOM.tit2': '数据处理', 'AusCOM.tit3': 'JSON', - 'AusCOM.http': 'HTTP以[g_way]方式访问[site]发送[body]', + 'AusCOM.http': 'HTTP以[g_way]方式访问[site]参数[p]发送[body]', + 'AusCOM.http_pm': '网址参数构造 id[id](创建/修改)名称[name]内容[text]', + 'AusCOM.http_pm_des': '销毁网址参数构造[id]', 'AusCOM.wsocket_create': 'ws.创建websocket,协议[yi],并返回ccid', 'AusCOM.wsocket_connect': 'ws.id[id]socket连接[host]', 'AusCOM.wsocket_send': 'ws.发送id[id]数据[text]', @@ -55,10 +60,15 @@ class AuEx_communication { }, en: { 'AusCOM.extensionName': 'AUSのcommunication', - 'AusCOM.tit1': 'sock communication', + 'AusCOM.tit1': 'communication', + 'AusCOM.tit1.1': 'http', + 'AusCOM.tit1.2': 'WebSocket', + 'AusCOM.tit2': '数据处理', 'AusCOM.tit2': 'solve data', 'AusCOM.tit3': 'JSON', - 'AusCOM.http': 'http access[g_way]to[site]send[body]', + 'AusCOM.http': 'http access[g_way]to[site]parameter[p]send[body]', + 'AusCOM.http_pm': 'URL Parameter Construction id [id] (Create/Modify) Name [name] Content [text]', + 'AusCOM.http_pm_des': 'Destroy URL parameter structure [id]', 'AusCOM.wsocket_create': 'ws.create websocket,protocol', 'AusCOM.wsocket_connect': 'id[id]websocket connect[host]', 'AusCOM.wsocket_send': 'ws.send id[id] data[text]', 'AusCOM.wsocket_recv': 'ws.recv id [id]', @@ -98,9 +108,8 @@ class AuEx_communication { 'AusCOM.help': 'help', }, }) - this.aulink = new Worker('./aurora_link.js') - window.AusCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [] } - this.icon="" + window.AusCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [] } + this.icon = "" } formatMessage(id) { return this._formatMessage({ @@ -118,6 +127,7 @@ class AuEx_communication { menuIconURI: this.icon, blocks: [ "---" + this.formatMessage("AusCOM.tit1"), + "---" + this.formatMessage("AusCOM.tit1.1"), { opcode: 'http', blockType: 'reporter', @@ -131,13 +141,47 @@ class AuEx_communication { type: 'string', defaultValue: '/' }, + p: { + type: 'string', + defaultValue: 'None' + }, body: { type: 'string', defaultValue: ' ' }, } }, - + { + opcode: 'http_pm_cm', + blockType: BlockType.COMMAND, + text: this.formatMessage('AusCOM.http_pm'), + arguments: { + id: { + type: 'string', + defaultValue: '1' + }, + name: { + type: 'string', + defaultValue: 'a' + }, + text: { + type: 'string', + defaultValue: 'b' + }, + } + }, + { + opcode: 'http_pm_des', + blockType: BlockType.COMMAND, + text: this.formatMessage('AusCOM.http_pm_des'), + arguments: { + id: { + type: 'string', + defaultValue: '1' + } + }, + }, + "---" + this.formatMessage("AusCOM.tit1.2"), { opcode: 'ws_cre_sock', blockType: 'reporter', @@ -573,8 +617,28 @@ class AuEx_communication { } http(args) { - const { g_way, site, body } = args - return this.httpGet(site, body, g_way); + function get_p(p) { + pt='?' + for (let i = 0, len = window.AusCOM.http_pm_.length; i < len; i++) { + if (window.AusCOM.http_pm_[i]['id'] == String(p)) { + for (let i_ = 0, len_ = window.AusCOM.http_pm_[i_].p.length; i_ < len_; i_++) { + if(i_==0){ + pt+=window.AusCOM.http_pm_[i].p[i_].name+'='+window.AusCOM.http_pm_[i].p[i_].value + } + else{ + pt+='&'+window.AusCOM.http_pm_[i].p[i_].name+'='+window.AusCOM.http_pm_[i].p[i_].value + } + } + } + } + return pt + } + const { g_way, site, p, body } = args + if (p=='None'){ + + + return this.httpGet(site, body, g_way);} + else{return this.httpGet(site+get_p(p), body, g_way);} } ws_cre_sock(args) { @@ -1250,5 +1314,37 @@ class AuEx_communication { return j['id'] } + http_pm_cm(args) { + const { id, name, text } = args + let a = false + for (let i = 0, len = window.AusCOM.http_pm_.length; i < len; i++) { + if (window.AusCOM.http_pm_[i]['id'] == String(id)) { + a = true + } + } + if (a) { } else { window.AusCOM.http_pm_.push({ 'id': String(id), 'p': [] }) } + for (let i = 0, len = window.AusCOM.http_pm_.length; i < len; i++) { + if (window.AusCOM.http_pm_[i]['id'] == String(id)) { + for (let i_ = 0, len_ = window.AusCOM.http_pm_[i]['p'].length; i_ < len_; i_++) { + if (window.AusCOM.http_pm_[i]['p'][i_].name==name){ + window.AusCOM.http_pm_[i]['p'][i_].value=text + return '' + } + } + window.AusCOM.http_pm_[i]['p'].push({"name":name,'value':text}) + } + } + } + http_pm_des(args) { + const { id } = args + try{ + for (let i = 0, len = window.AusCOM.http_pm_.length; i < len; i++) { + if (window.AusCOM.http_pm_[i]['id'] == String(id)) { + window.AusCOM.http_pm_.splice(i,1) + return + } + }} + catch(error){} + } } module.exports = AuEx_communication; From cdae95ba6e713c150e3f0c5284fc04e070002a80 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Mon, 3 Oct 2022 09:47:08 +0800 Subject: [PATCH 158/357] Update scCOM.js --- dannydev/scCOM.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index 7f0c3368..b949d930 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -517,7 +517,7 @@ class AuEx_communication { { opcode: 'help', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.help') + 'ver 1.00C1', + text: this.formatMessage('AusCOM.help') + 'ver 1.00C2', }, ], From 1e18dcde4a06b1a469c1cc9157dc6657c468a2f8 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sat, 15 Oct 2022 10:54:18 +0800 Subject: [PATCH 159/357] Update scCOM.js --- dannydev/scCOM.js | 518 +++++++++++++++++++++++----------------------- 1 file changed, 262 insertions(+), 256 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index b949d930..f18afab6 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1,3 +1,4 @@ + const Cast = require('../../util/cast'); const ArgumentType = require('../../extension-support/argument-type'); const BlockType = require('../../extension-support/block-type'); @@ -5,110 +6,112 @@ const { string } = require('prop-types'); // var dgram = require("dgram"); //感谢Arkos开源的扩展代码,作为参照编写框架 //欸,别把作者忘了。Danny欸 -class AuEx_communication { +class AuEx_Communication { constructor(runtime) { this.runtime = runtime this._formatMessage = runtime.getFormatMessage({ 'zh-cn': { - 'AusCOM.extensionName': 'AUSの通讯模块', - 'AusCOM.tit1': '通讯', - 'AusCOM.tit1.1': 'http', - 'AusCOM.tit1.2': 'WebSocket', - 'AusCOM.tit2': '数据处理', - 'AusCOM.tit3': 'JSON', - 'AusCOM.http': 'HTTP以[g_way]方式访问[site]参数[p]发送[body]', - 'AusCOM.http_pm': '网址参数构造 id[id](创建/修改)名称[name]内容[text]', - 'AusCOM.http_pm_des': '销毁网址参数构造[id]', - 'AusCOM.wsocket_create': 'ws.创建websocket,协议[yi],并返回ccid', - 'AusCOM.wsocket_connect': 'ws.id[id]socket连接[host]', - 'AusCOM.wsocket_send': 'ws.发送id[id]数据[text]', - 'AusCOM.wsocket_recv': 'ws.接收数据id[id]', - 'AusCOM.wsocket_conclose': 'ws.连接关闭id[id]', - 'AusCOM.wsocket_close': 'ws.销毁websocket id[id]', - 'AusCOM.socket_create': 'sock.创建[yi]方式[method]并返回ccid', - 'AusCOM.socket_create_chk': 'sock.创建是否成功[ccid]并返回id', - 'AusCOM.socket_recv': 'sock.收到数据id [id]', - 'AusCOM.socket_send': 'sock.发送数据id [id] ip[ip]数据[TEXT]', - 'AusCOM.socket_conclose': 'sock.断开连接id[id]', - 'AusCOM.socket_connect': 'sock.连接/启动服务 id[id] ip [ip] 接收长度[buff]', - 'AusCOM.socket_breakev': 'sock.断开连接id[id]ip[ip]', - 'AusCOM.socket_close': 'sock.销毁socket对象 id[id]', - 'AusCOM.socket_parse': 'sock.解析返回的数据data[data]属性[type]', - 'AusCOM.socket_opchk': 'sock.操作是否成功[cid]', - 'AusCOM.code_RC4': 'RC4加解密 密钥[KEY] 文本[TEXT]', - 'AusCOM.code_base64_en': 'base64编码文本[TEXT]', - 'AusCOM.code_base64_de': 'base64解码文本[TEXT]', - 'AusCOM.code_base58_en': 'base58编码文本[TEXT]', - 'AusCOM.code_base58_de': 'base58解码码文本[TEXT]', - 'AusCOM.code_base_custom_en': '自定义base加密 字符集[ALPHABET] 文本[TEXT]', - 'AusCOM.code_base_custom_de': '自定义base解密 字符集[ALPHABET] 文本[TEXT]', - 'AusCOM.str': '字符串', - 'AusCOM.int': '整数', - 'AusCOM.float': '浮点数', - 'AusCOM.list': '列表', - 'AusCOM.logic': '逻辑', - 'AusCOM.json_create': '创建json 载入[json_] 并返回id', - 'AusCOM.json_load': '载入json id[id]json[json_]', - 'AusCOM.json_get_all_item': '获取json id[id]中所有属性名', - 'AusCOM.json_add_from_id': '将id[_id]对应的json加入json/改变值 id[id] 属性名[name]', - 'AusCOM.json_add': '加入/改变对象到json id[id]类型[type]属性名[name]内容[TEXT]', - 'AusCOM.json_get_item': '获取json id [id]中某属性名为[name]项的值', - 'AusCOM.json_del_item': '删除json id [id]中某属性名为[name]项的值', - 'AusCOM.json_get': '获取json id[id]', - 'AusCOM.json_destroy': '销毁json id[id]', - 'AusCOM.help': '使用帮助', + 'DannyDevCOM.extensionName': 'AUSの通讯模块', + 'DannyDevCOM.tit1': '通讯', + 'DannyDevCOM.tit1.1': 'http', + 'DannyDevCOM.tit1.2': 'WebSocket', + 'DannyDevCOM.tit2': '数据处理', + 'DannyDevCOM.tit3': 'JSON', + 'DannyDevCOM.http': 'HTTP以[g_way]方式访问[site]参数[p]发送[body]', + 'DannyDevCOM.http_pm': '网址参数构造 id[id](创建/修改)名称[name]内容[text]', + 'DannyDevCOM.http_pm_des': '销毁网址参数构造[id]', + 'DannyDevCOM.wsocket_create': 'ws.创建websocket,协议[yi],并返回cid', + 'DannyDevCOM.wsocket_connect': 'ws.id[id]socket连接[host]', + 'DannyDevCOM.wsocket_send': 'ws.发送id[id]数据[text]', + 'DannyDevCOM.wsocket_recv': 'ws.接收数据id[id]', + 'DannyDevCOM.wsocket_conclose': 'ws.连接关闭id[id]', + 'DannyDevCOM.wsocket_close': 'ws.销毁websocket id[id]', + 'DannyDevCOM.socket_create': 'sock.创建[yi]方式[method]并返回ccid', + 'DannyDevCOM.socket_create_chk': 'sock.创建是否成功[ccid]并返回id', + 'DannyDevCOM.socket_recv': 'sock.收到数据id [id]', + 'DannyDevCOM.socket_send': 'sock.发送数据id [id] ip[ip]数据[TEXT]', + 'DannyDevCOM.socket_conclose': 'sock.断开连接id[id]', + 'DannyDevCOM.socket_connect': 'sock.连接/启动服务 id[id] ip [ip] 接收长度[buff]', + 'DannyDevCOM.socket_breakev': 'sock.断开连接id[id]ip[ip]', + 'DannyDevCOM.socket_close': 'sock.销毁socket对象 id[id]', + 'DannyDevCOM.socket_parse': 'sock.解析返回的数据data[data]属性[type]', + 'DannyDevCOM.socket_opchk': 'sock.操作是否成功[cid]', + 'DannyDevCOM.code_RC4': 'RC4加解密 密钥[KEY] 文本[TEXT]', + 'DannyDevCOM.code_base64_en': 'base64编码文本[TEXT]', + 'DannyDevCOM.code_base64_de': 'base64解码文本[TEXT]', + 'DannyDevCOM.code_base58_en': 'base58编码文本[TEXT]', + 'DannyDevCOM.code_base58_de': 'base58解码码文本[TEXT]', + 'DannyDevCOM.code_base_custom_en': '自定义base加密 字符集[ALPHABET] 文本[TEXT]', + 'DannyDevCOM.code_base_custom_de': '自定义base解密 字符集[ALPHABET] 文本[TEXT]', + 'DannyDevCOM.str': '字符串', + 'DannyDevCOM.int': '整数', + 'DannyDevCOM.float': '浮点数', + 'DannyDevCOM.list': '列表', + 'DannyDevCOM.logic': '逻辑', + 'DannyDevCOM.json_create': '创建json 载入[json_] 并返回id', + 'DannyDevCOM.json_load': '载入json id[id]json[json_]', + 'DannyDevCOM.json_get_all_item': '获取json id[id]中所有属性名', + 'DannyDevCOM.json_add_from_id': '将id[_id]对应的json加入json/改变值 id[id] 属性名[name]', + 'DannyDevCOM.json_add': '加入/改变对象到json id[id]类型[type]属性名[name]内容[TEXT]', + 'DannyDevCOM.json_get_item': '获取json id [id]中某属性名为[name]项的值', + 'DannyDevCOM.json_del_item': '删除json id [id]中某属性名为[name]项的值', + 'DannyDevCOM.json_get': '获取json id[id]', + 'DannyDevCOM.json_destroy': '销毁json id[id]', + 'DannyDevCOM.help': '使用帮助', + 'DannyDevCOM.help.t': 'http访问出现fail to load XX可能是域名/ip不存在,也有可能是出现跨域问题,可以联系本站站住修改HTTP协议头或者自行修改浏览器跨域设置。如果ws无法发送数据,请F12打开开发者工具查看是否报错,如果报错则是无法连接,目前浏览器还不支持侦察此类报错。更多帮助请进入 https://gitee.com/ausx/scCOM', }, en: { - 'AusCOM.extensionName': 'AUSのcommunication', - 'AusCOM.tit1': 'communication', - 'AusCOM.tit1.1': 'http', - 'AusCOM.tit1.2': 'WebSocket', - 'AusCOM.tit2': '数据处理', - 'AusCOM.tit2': 'solve data', - 'AusCOM.tit3': 'JSON', - 'AusCOM.http': 'http access[g_way]to[site]parameter[p]send[body]', - 'AusCOM.http_pm': 'URL Parameter Construction id [id] (Create/Modify) Name [name] Content [text]', - 'AusCOM.http_pm_des': 'Destroy URL parameter structure [id]', - 'AusCOM.wsocket_create': 'ws.create websocket,protocol', 'AusCOM.wsocket_connect': 'id[id]websocket connect[host]', - 'AusCOM.wsocket_send': 'ws.send id[id] data[text]', - 'AusCOM.wsocket_recv': 'ws.recv id [id]', - 'AusCOM.wsocket_conclose': 'ws.When connection closed id[id]', - 'AusCOM.wsocket_close': 'ws.destroy socket id[id]', - 'AusCOM.socket_ sreate ': 'sock. Create [yi] method [method] and return ID', - 'AusCOM.socket_create_chk': 'sock.create successfully[ccid]?', - 'AusCOM.socket_ recv': 'sock. Received data ID [id] ', - 'AusCOM.socket_ send': 'sock. Send data ID [id] data [text] ', - 'AusCOM.socket_ close ': 'sock. Connection closing ID [id]', - 'AusCOM.socket_ connect': 'sock. Connect to/start server ID [id] IP [ip] ,buff[buff]', - 'AusCOM.socket_breakev': 'sock.break connection off id[id]ip[ip]', - 'AusCOM.socket_ close': 'sock. Destroy the socket object ID [id] ', - 'AusCOM.socket_ parse ': 'sock. Parse the returned data [data] attribute [type]', - 'AusCOM.socket_opchk': 'sock.run operation successfully[cid]?', - 'AusCOM.code_RC4': 'RC4 KEY[KEY] TEXT[TEXT]', - 'AusCOM.code_base64_en': 'base64 encoding[TEXT]', - 'AusCOM.code_base64_de': 'base64 decoding[TEXT]', - 'AusCOM.code_base58_en': 'base58 encoding[TEXT]', - 'AusCOM.code_base58_de': 'base58 decoding[TEXT]', - 'AusCOM.code_base_custom_en': 'Custom base encryption,ALPHABET[ALPHABET] TEXT[TEXT]', - 'AusCOM.code_base_custom_de': 'Custom base decryption,ALPHABET[ALPHABET] TEXT[TEXT]', - 'AusCOM.str': 'string', - 'AusCOM.int': 'int', - 'AusCOM.float': 'float', - 'AusCOM.list': 'list', - 'AusCOM.logic': 'logic', - 'AusCOM.json_create': 'create json,load[json_] and return id', - 'AusCOM.json_load': ' load json id[id]json[json_]', - 'AusCOM.json_get_all_item': 'get all attribute name of json id[id]', - 'AusCOM.json_add_from_id': 'Add the JSON corresponding to ID [_id] to JSON/change a value ID [id] attribute name [name]', - 'AusCOM.json_add': 'Add object/change a value to JSON ID [id], type [type], attribute name [name], content [TEXT]', - 'AusCOM.json_get_item': 'Get the value of an attribute named [name] in JSON ID [id]', - 'AusCOM.json_del_item': 'Delete the value of an attribute named [name] in JSON ID [ID]', - 'AusCOM.json_get': 'get json id[id]', - 'AusCOM.json_destroy': 'destroy json id[id]', - 'AusCOM.help': 'help', + 'DannyDevCOM.extensionName': 'AUSのcommunication', + 'DannyDevCOM.tit1': 'communication', + 'DannyDevCOM.tit1.1': 'http', + 'DannyDevCOM.tit1.2': 'WebSocket', + 'DannyDevCOM.tit2': '数据处理', + 'DannyDevCOM.tit2': 'solve data', + 'DannyDevCOM.tit3': 'JSON', + 'DannyDevCOM.http': 'http access[g_way]to[site]parameter[p]send[body]', + 'DannyDevCOM.http_pm': 'URL Parameter Construction id [id] (Create/Modify) Name [name] Content [text]', + 'DannyDevCOM.http_pm_des': 'Destroy URL parameter structure [id]', + 'DannyDevCOM.wsocket_create': 'ws.create websocket,protocol', 'DannyDevCOM.wsocket_connect': 'id[id]websocket connect[host]', + 'DannyDevCOM.wsocket_send': 'ws.send id[id] data[text]', + 'DannyDevCOM.wsocket_recv': 'ws.recv id [id]', + 'DannyDevCOM.wsocket_conclose': 'ws.When connection closed id[id]', + 'DannyDevCOM.wsocket_close': 'ws.destroy socket id[id]', + 'DannyDevCOM.socket_ sreate ': 'sock. Create [yi] method [method] and return ID', + 'DannyDevCOM.socket_create_chk': 'sock.create successfully[ccid]?', + 'DannyDevCOM.socket_ recv': 'sock. Received data ID [id] ', + 'DannyDevCOM.socket_ send': 'sock. Send data ID [id] data [text] ', + 'DannyDevCOM.socket_ close ': 'sock. Connection closing ID [id]', + 'DannyDevCOM.socket_ connect': 'sock. Connect to/start server ID [id] IP [ip] ,buff[buff]', + 'DannyDevCOM.socket_breakev': 'sock.break connection off id[id]ip[ip]', + 'DannyDevCOM.socket_ close': 'sock. Destroy the socket object ID [id] ', + 'DannyDevCOM.socket_ parse ': 'sock. Parse the returned data [data] attribute [type]', + 'DannyDevCOM.socket_opchk': 'sock.run operation successfully[cid]?', + 'DannyDevCOM.code_RC4': 'RC4 KEY[KEY] TEXT[TEXT]', + 'DannyDevCOM.code_base64_en': 'base64 encoding[TEXT]', + 'DannyDevCOM.code_base64_de': 'base64 decoding[TEXT]', + 'DannyDevCOM.code_base58_en': 'base58 encoding[TEXT]', + 'DannyDevCOM.code_base58_de': 'base58 decoding[TEXT]', + 'DannyDevCOM.code_base_custom_en': 'Custom base encryption,ALPHABET[ALPHABET] TEXT[TEXT]', + 'DannyDevCOM.code_base_custom_de': 'Custom base decryption,ALPHABET[ALPHABET] TEXT[TEXT]', + 'DannyDevCOM.str': 'string', + 'DannyDevCOM.int': 'int', + 'DannyDevCOM.float': 'float', + 'DannyDevCOM.list': 'list', + 'DannyDevCOM.logic': 'logic', + 'DannyDevCOM.json_create': 'create json,load[json_] and return id', + 'DannyDevCOM.json_load': ' load json id[id]json[json_]', + 'DannyDevCOM.json_get_all_item': 'get all attribute name of json id[id]', + 'DannyDevCOM.json_add_from_id': 'Add the JSON corresponding to ID [_id] to JSON/change a value ID [id] attribute name [name]', + 'DannyDevCOM.json_add': 'Add object/change a value to JSON ID [id], type [type], attribute name [name], content [TEXT]', + 'DannyDevCOM.json_get_item': 'Get the value of an attribute named [name] in JSON ID [id]', + 'DannyDevCOM.json_del_item': 'Delete the value of an attribute named [name] in JSON ID [ID]', + 'DannyDevCOM.json_get': 'get json id[id]', + 'DannyDevCOM.json_destroy': 'destroy json id[id]', + 'DannyDevCOM.help': 'help', + 'DannyDevCOM.help.t': "If the http access fails to load XX, it may be that the domain name/ip does not exist, or it may be a cross domain problem. You can contact our website to modify the HTTP protocol header or modify the browser's cross domain settings. If ws is unable to send data, please F12 open the developer tool to check whether an error is reported. If an error is reported, the connection cannot be made. At present, the browser does not support detecting such errors. For more help, please go to https://gitee.com/ausx/scCOM", }, }) - window.AusCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [] } + window.DannyDevCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [] } this.icon = "" } formatMessage(id) { @@ -120,18 +123,18 @@ class AuEx_communication { } getInfo() { return { - id: 'AUdevNet', // 拓展id - name: this.formatMessage('AusCOM.extensionName'), // 拓展名 + id: 'DannyDevNet', // 拓展id + name: this.formatMessage('DannyDevCOM.extensionName'), // 拓展名 color1: '#FF8383', blockIconURI: this.icon, menuIconURI: this.icon, blocks: [ - "---" + this.formatMessage("AusCOM.tit1"), - "---" + this.formatMessage("AusCOM.tit1.1"), + "---" + this.formatMessage("DannyDevCOM.tit1"), + "---" + this.formatMessage("DannyDevCOM.tit1.1"), { opcode: 'http', blockType: 'reporter', - text: this.formatMessage('AusCOM.http'), + text: this.formatMessage('DannyDevCOM.http'), arguments: { g_way: { type: 'string', @@ -154,7 +157,7 @@ class AuEx_communication { { opcode: 'http_pm_cm', blockType: BlockType.COMMAND, - text: this.formatMessage('AusCOM.http_pm'), + text: this.formatMessage('DannyDevCOM.http_pm'), arguments: { id: { type: 'string', @@ -173,7 +176,7 @@ class AuEx_communication { { opcode: 'http_pm_des', blockType: BlockType.COMMAND, - text: this.formatMessage('AusCOM.http_pm_des'), + text: this.formatMessage('DannyDevCOM.http_pm_des'), arguments: { id: { type: 'string', @@ -181,11 +184,11 @@ class AuEx_communication { } }, }, - "---" + this.formatMessage("AusCOM.tit1.2"), + "---" + this.formatMessage("DannyDevCOM.tit1.2"), { opcode: 'ws_cre_sock', blockType: 'reporter', - text: this.formatMessage('AusCOM.wsocket_create'), + text: this.formatMessage('DannyDevCOM.wsocket_create'), arguments: { yi: { type: 'string', @@ -196,7 +199,7 @@ class AuEx_communication { { opcode: 'ws_conn_sock', blockType: 'Boolean', - text: this.formatMessage('AusCOM.wsocket_connect'), + text: this.formatMessage('DannyDevCOM.wsocket_connect'), arguments: { id: { type: 'string', @@ -211,7 +214,7 @@ class AuEx_communication { { opcode: 'ws_recv', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.wsocket_recv'), + text: this.formatMessage('DannyDevCOM.wsocket_recv'), arguments: { id: { type: ArgumentType.STRING, @@ -222,7 +225,7 @@ class AuEx_communication { { opcode: 'ws_close', blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.wsocket_conclose'), + text: this.formatMessage('DannyDevCOM.wsocket_conclose'), arguments: { id: { type: ArgumentType.STRING, @@ -233,7 +236,7 @@ class AuEx_communication { { opcode: 'ws_destroy', blockType: BlockType.COMMAND, - text: this.formatMessage('AusCOM.wsocket_close'), + text: this.formatMessage('DannyDevCOM.wsocket_close'), arguments: { id: { type: ArgumentType.STRING, @@ -244,7 +247,7 @@ class AuEx_communication { { opcode: 'ws_send', blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.wsocket_send'), + text: this.formatMessage('DannyDevCOM.wsocket_send'), arguments: { id: { type: ArgumentType.STRING, @@ -259,7 +262,7 @@ class AuEx_communication { // { // opcode: 'sock_cre', // blockType: BlockType.REPORTER, - // text: this.formatMessage('AusCOM.socket_create'), + // text: this.formatMessage('DannyDevCOM.socket_create'), // arguments: { // // yi: { @@ -277,11 +280,11 @@ class AuEx_communication { // } // }, //}, - "---" + this.formatMessage("AusCOM.tit2"), + "---" + this.formatMessage("DannyDevCOM.tit2"), { opcode: 'RC4_coding', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_RC4'), + text: this.formatMessage('DannyDevCOM.code_RC4'), arguments: { KEY: { type: ArgumentType.STRING, @@ -296,7 +299,7 @@ class AuEx_communication { { opcode: 'b64encoding', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base64_en'), + text: this.formatMessage('DannyDevCOM.code_base64_en'), arguments: { TEXT: { type: ArgumentType.STRING, @@ -308,7 +311,7 @@ class AuEx_communication { { opcode: 'b64decoding', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base64_de'), + text: this.formatMessage('DannyDevCOM.code_base64_de'), arguments: { TEXT: { type: ArgumentType.STRING, @@ -320,7 +323,7 @@ class AuEx_communication { { opcode: 'b58encoding', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base58_en'), + text: this.formatMessage('DannyDevCOM.code_base58_en'), arguments: { TEXT: { type: ArgumentType.STRING, @@ -332,7 +335,7 @@ class AuEx_communication { { opcode: 'b58decoding', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base58_de'), + text: this.formatMessage('DannyDevCOM.code_base58_de'), arguments: { TEXT: { type: ArgumentType.STRING, @@ -344,7 +347,7 @@ class AuEx_communication { { opcode: 'bcostomencoding', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base_custom_en'), + text: this.formatMessage('DannyDevCOM.code_base_custom_en'), arguments: { ALPHABET: { type: ArgumentType.STRING, @@ -360,7 +363,7 @@ class AuEx_communication { { opcode: 'bcostomdecoding', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.code_base_custom_de'), + text: this.formatMessage('DannyDevCOM.code_base_custom_de'), arguments: { ALPHABET: { type: ArgumentType.STRING, @@ -374,11 +377,11 @@ class AuEx_communication { }, - "---" + this.formatMessage("AusCOM.tit3"), + "---" + this.formatMessage("DannyDevCOM.tit3"), { opcode: 'json_create', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.json_create'), + text: this.formatMessage('DannyDevCOM.json_create'), arguments: { json_: { type: ArgumentType.STRING, @@ -389,7 +392,7 @@ class AuEx_communication { { opcode: 'json_load', blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_load'), + text: this.formatMessage('DannyDevCOM.json_load'), arguments: { id: { type: ArgumentType.STRING, @@ -404,7 +407,7 @@ class AuEx_communication { { opcode: 'json_get_all_item', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.json_get_all_item'), + text: this.formatMessage('DannyDevCOM.json_get_all_item'), arguments: { id: { type: ArgumentType.STRING, @@ -415,7 +418,7 @@ class AuEx_communication { { opcode: 'json_add_from_id', blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_add_from_id'), + text: this.formatMessage('DannyDevCOM.json_add_from_id'), arguments: { _id: { type: ArgumentType.STRING, @@ -434,7 +437,7 @@ class AuEx_communication { { opcode: 'json_add', blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_add'), + text: this.formatMessage('DannyDevCOM.json_add'), arguments: { id: { @@ -458,7 +461,7 @@ class AuEx_communication { { opcode: 'json_get_item', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.json_get_item'), + text: this.formatMessage('DannyDevCOM.json_get_item'), arguments: { id: { @@ -475,7 +478,7 @@ class AuEx_communication { { opcode: 'json_del_item', blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_del_item'), + text: this.formatMessage('DannyDevCOM.json_del_item'), arguments: { id: { @@ -492,7 +495,7 @@ class AuEx_communication { { opcode: 'GETjson', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.json_get'), + text: this.formatMessage('DannyDevCOM.json_get'), arguments: { id: { @@ -504,7 +507,7 @@ class AuEx_communication { { opcode: 'DESTROYjson', blockType: BlockType.BOOLEAN, - text: this.formatMessage('AusCOM.json_destroy'), + text: this.formatMessage('DannyDevCOM.json_destroy'), arguments: { id: { @@ -517,7 +520,7 @@ class AuEx_communication { { opcode: 'help', blockType: BlockType.REPORTER, - text: this.formatMessage('AusCOM.help') + 'ver 1.00C2', + text: this.formatMessage('DannyDevCOM.help') + 'ver 1.00B1', }, ], @@ -560,23 +563,23 @@ class AuEx_communication { ], JSON_add: [ { - text: this.formatMessage('AusCOM.str'), + text: this.formatMessage('DannyDevCOM.str'), value: '0' }, { - text: this.formatMessage('AusCOM.int'), + text: this.formatMessage('DannyDevCOM.int'), value: '1' }, { - text: this.formatMessage('AusCOM.float'), + text: this.formatMessage('DannyDevCOM.float'), value: '2' }, { - text: this.formatMessage('AusCOM.list'), + text: this.formatMessage('DannyDevCOM.list'), value: '3' }, { - text: this.formatMessage('AusCOM.logic'), + text: this.formatMessage('DannyDevCOM.logic'), value: '4' }, ], @@ -618,15 +621,15 @@ class AuEx_communication { http(args) { function get_p(p) { - pt='?' - for (let i = 0, len = window.AusCOM.http_pm_.length; i < len; i++) { - if (window.AusCOM.http_pm_[i]['id'] == String(p)) { - for (let i_ = 0, len_ = window.AusCOM.http_pm_[i_].p.length; i_ < len_; i_++) { - if(i_==0){ - pt+=window.AusCOM.http_pm_[i].p[i_].name+'='+window.AusCOM.http_pm_[i].p[i_].value + pt = '?' + for (let i = 0, len = window.DannyDevCOM.http_pm_.length; i < len; i++) { + if (window.DannyDevCOM.http_pm_[i]['id'] == String(p)) { + for (let i_ = 0, len_ = window.DannyDevCOM.http_pm_[i_].p.length; i_ < len_; i_++) { + if (i_ == 0) { + pt += window.DannyDevCOM.http_pm_[i].p[i_].name + '=' + window.DannyDevCOM.http_pm_[i].p[i_].value } - else{ - pt+='&'+window.AusCOM.http_pm_[i].p[i_].name+'='+window.AusCOM.http_pm_[i].p[i_].value + else { + pt += '&' + window.DannyDevCOM.http_pm_[i].p[i_].name + '=' + window.DannyDevCOM.http_pm_[i].p[i_].value } } } @@ -634,23 +637,25 @@ class AuEx_communication { return pt } const { g_way, site, p, body } = args - if (p=='None'){ + if (p == 'None') { + return this.httpGet(site, body, g_way); + } + else { + return this.httpGet(site + get_p(p), body, g_way); + } - - return this.httpGet(site, body, g_way);} - else{return this.httpGet(site+get_p(p), body, g_way);} } ws_cre_sock(args) { const { yi } = args if (yi == 'WS') { id = Date.now() - window.AusCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WS' }) + window.DannyDevCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WS' }) return String(id) } if (yi == 'WSS') { id = Date.now() - window.AusCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WSS' }) + window.DannyDevCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WSS' }) return String(id) } return '未知协议' @@ -658,24 +663,24 @@ class AuEx_communication { ws_conn_sock(args) { const { id, host } = args console.log(id, host) - for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { - if (String(id) == window.AusCOM.wsock[i]["id"]) { - for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { - if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { - delete window.AusCOM.wsnotalive_[String(id)] + for (let i = 0, len = window.DannyDevCOM.wsock.length; i < len; i++) { + if (String(id) == window.DannyDevCOM.wsock[i]["id"]) { + for (let i = 0, len = window.DannyDevCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.DannyDevCOM.wsrecv_[i]["id"]) { + delete window.DannyDevCOM.wsnotalive_[String(id)] } } try { - window.AusCOM.wsock[i]['obj'] = new WebSocket(window.AusCOM.wsock[i]["way"] + '://' + host) - console.log(window.AusCOM.wsock) - window.AusCOM.wsock[i]['obj'].onmessage = function (evt) { + window.DannyDevCOM.wsock[i]['obj'] = new WebSocket(window.DannyDevCOM.wsock[i]["way"] + '://' + host) + console.log(window.DannyDevCOM.wsock) + window.DannyDevCOM.wsock[i]['obj'].onmessage = function (evt) { window.abc = evt.data - window.AusCOM.wsrecv_.push({ 'id': window.AusCOM.wsock[i]['id'], 'nr': evt.data }) + window.DannyDevCOM.wsrecv_.push({ 'id': window.DannyDevCOM.wsock[i]['id'], 'nr': evt.data }) } - window.AusCOM.wsock[i]['obj'].onclose = function (evt) { - window.AusCOM.wsnotalive_.push(window.AusCOM.wsock[i]['id']) + window.DannyDevCOM.wsock[i]['obj'].onclose = function (evt) { + window.DannyDevCOM.wsnotalive_.push(window.DannyDevCOM.wsock[i]['id']) } return true } @@ -690,13 +695,13 @@ class AuEx_communication { } ws_recv(args) { const { id } = args - console.log(window.AusCOM.wsrecv_) - for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { - if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { - console.log(window.AusCOM.wsrecv_) + console.log(window.DannyDevCOM.wsrecv_) + for (let i = 0, len = window.DannyDevCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.DannyDevCOM.wsrecv_[i]["id"]) { + console.log(window.DannyDevCOM.wsrecv_) try { - var a = String(window.AusCOM.wsrecv_[i]["nr"]) - window.AusCOM.wsrecv_.splice(i, 1); + var a = String(window.DannyDevCOM.wsrecv_[i]["nr"]) + window.DannyDevCOM.wsrecv_.splice(i, 1); return a } catch (error) { @@ -709,9 +714,9 @@ class AuEx_communication { } ws_close(args) { const { id } = args - for (let i = 0, len = window.AusCOM.wsrecv_.length; i < len; i++) { - if (String(id) == window.AusCOM.wsrecv_[i]["id"]) { - window.AusCOM.wsnotalive_.splice(i, 1) + for (let i = 0, len = window.DannyDevCOM.wsrecv_.length; i < len; i++) { + if (String(id) == window.DannyDevCOM.wsrecv_[i]["id"]) { + window.DannyDevCOM.wsnotalive_.splice(i, 1) return true } } @@ -719,14 +724,14 @@ class AuEx_communication { } ws_destroy(args) { const { id } = args - for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { - console.log(window.AusCOM.wsock[i]) - if (String(id) == window.AusCOM.wsock[i]["id"]) { + for (let i = 0, len = window.DannyDevCOM.wsock.length; i < len; i++) { + console.log(window.DannyDevCOM.wsock[i]) + if (String(id) == window.DannyDevCOM.wsock[i]["id"]) { try { - if (window.AusCOM.wsock[i]["obj"] == null) { } - else { window.AusCOM.wsock[i]["obj"].close() } - window.AusCOM.wsock.splice(i, 1); - // alert(window.AusCOM.wsock) + if (window.DannyDevCOM.wsock[i]["obj"] == null) { } + else { window.DannyDevCOM.wsock[i]["obj"].close() } + window.DannyDevCOM.wsock.splice(i, 1); + // alert(window.DannyDevCOM.wsock) return '' } catch (error) { console.log(error) } @@ -735,13 +740,13 @@ class AuEx_communication { } ws_send(args) { const { id, text } = args - for (let i = 0, len = window.AusCOM.wsock.length; i < len; i++) { - if (String(id) == window.AusCOM.wsock[i]["id"]) { - console.log(window.AusCOM.wsrecv_) - console.log(window.AusCOM.wsock[i]) + for (let i = 0, len = window.DannyDevCOM.wsock.length; i < len; i++) { + if (String(id) == window.DannyDevCOM.wsock[i]["id"]) { + console.log(window.DannyDevCOM.wsrecv_) + console.log(window.DannyDevCOM.wsock[i]) try { - if (window.AusCOM.wsock[i]["obj"] == null) { return false } else { window.AusCOM.wsock[i]["obj"].send(text); return true } + if (window.DannyDevCOM.wsock[i]["obj"] == null) { return false } else { window.DannyDevCOM.wsock[i]["obj"].send(text); return true } } catch (error) { @@ -754,7 +759,7 @@ class AuEx_communication { return false } help() { - return "http访问出现fail to load XX可能是域名/ip不存在,也有可能是出现跨域问题,可以联系本站站住修改HTTP协议头或者自行修改浏览器跨域设置。如果ws无法发送数据,请F12打开开发者工具查看是否报错,如果报错则是无法连接,目前浏览器还不支持侦察此类报错。更多帮助请进入 https://gitee.com/ausx/scCOM" + return this.formatMessage('DannyDevCOM.help.t') } RC4_coding(args) { var key = '' @@ -1085,7 +1090,7 @@ class AuEx_communication { a = JSON.parse(j) } catch (error) { } - window.AusCOM.json_list.push({ 'id': String(id), 'json': a }) + window.DannyDevCOM.json_list.push({ 'id': String(id), 'json': a }) return id } return create(json_) @@ -1097,9 +1102,9 @@ class AuEx_communication { var a = {} try { a = JSON.parse(j) - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - window.AusCOM.json_list[i]['json'] = a + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + window.DannyDevCOM.json_list[i]['json'] = a return true } } @@ -1114,9 +1119,9 @@ class AuEx_communication { const { id } = args var b = [] try { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - b = Object.keys(window.AusCOM.json_list[i]['json']) + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + b = Object.keys(window.DannyDevCOM.json_list[i]['json']) } } } @@ -1134,17 +1139,17 @@ class AuEx_communication { return d } check(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { return true } } return false } getjson(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return window.AusCOM.json_list[i]['json'] + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + return window.DannyDevCOM.json_list[i]['json'] } } return {} @@ -1152,26 +1157,26 @@ class AuEx_communication { json_add_from_id(args) { const { _id, id, name } = args function check(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { return true } } return false } function getjson(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return window.AusCOM.json_list[i]['json'] + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + return window.DannyDevCOM.json_list[i]['json'] } } return {} } function add_item_from_id(id, id1, name) { if (check(id) & check(id1)) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - window.AusCOM.json_list[i]['json'][name] = getjson(id1) + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + window.DannyDevCOM.json_list[i]['json'][name] = getjson(id1) return true } } @@ -1184,48 +1189,48 @@ class AuEx_communication { json_add(args) { const { id, type, name, TEXT } = args function check(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { return true } } return false } function getjson(id) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return window.AusCOM.json_list[i]['json'] + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + return window.DannyDevCOM.json_list[i]['json'] } } return {} } function add_item(id, name, type, content) { if (check(id)) { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { if (type == '0') {//字符串 - window.AusCOM.json_list[i]['json'][String(name)] = String(content) + window.DannyDevCOM.json_list[i]['json'][String(name)] = String(content) return true } if (type == '1') {//整数 - window.AusCOM.json_list[i]['json'][String(name)] = parseInt(content) + window.DannyDevCOM.json_list[i]['json'][String(name)] = parseInt(content) return true } if (type == '2') {//浮点数 - window.AusCOM.json_list[i]['json'][String(name)] = parseFloat(content) + window.DannyDevCOM.json_list[i]['json'][String(name)] = parseFloat(content) return true } if (type == '3') {//列表 - window.AusCOM.json_list[i]['json'][String(name)] = content.split(' ') + window.DannyDevCOM.json_list[i]['json'][String(name)] = content.split(' ') return true } if (type == '4') {//逻辑 if (String(content) == 'true') { - window.AusCOM.json_list[i]['json'][String(name)] = true + window.DannyDevCOM.json_list[i]['json'][String(name)] = true return true } else { - window.AusCOM.json_list[i]['json'][String(name)] = false + window.DannyDevCOM.json_list[i]['json'][String(name)] = false return true } } @@ -1240,18 +1245,18 @@ class AuEx_communication { } GETjson(args) { const { id } = args - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - return JSON.stringify(window.AusCOM.json_list[i]['json']) + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + return JSON.stringify(window.DannyDevCOM.json_list[i]['json']) } } return '' } DESTROYjson(args) { const { id } = args - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - window.AusCOM.json_list.splice(i, 1); + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + window.DannyDevCOM.json_list.splice(i, 1); return true } } @@ -1260,27 +1265,27 @@ class AuEx_communication { json_get_item(args) { // const { id, name } = args try { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'string') { //id_0 字符串 - return window.AusCOM.json_list[i]['json'][name] + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + if (typeof (window.DannyDevCOM.json_list[i]['json'][name]) == 'string') { //id_0 字符串 + return window.DannyDevCOM.json_list[i]['json'][name] } - if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'number') { //id_1&2 整数&浮点 - return window.AusCOM.json_list[i]['json'][name] + if (typeof (window.DannyDevCOM.json_list[i]['json'][name]) == 'number') { //id_1&2 整数&浮点 + return window.DannyDevCOM.json_list[i]['json'][name] } - if (window.AusCOM.json_list[i]['json'][name] instanceof Array) { //id_4列表 + if (window.DannyDevCOM.json_list[i]['json'][name] instanceof Array) { //id_4列表 var a = '' - for (let i_ = 0, len = window.AusCOM.json_list[i]['json'][name].length; i_ < len; i_++) { - if (i_ == window.AusCOM.json_list[i]['json'][name].length) { - a += window.AusCOM.json_list[i]['json'][name][i] + for (let i_ = 0, len = window.DannyDevCOM.json_list[i]['json'][name].length; i_ < len; i_++) { + if (i_ == window.DannyDevCOM.json_list[i]['json'][name].length) { + a += window.DannyDevCOM.json_list[i]['json'][name][i] } else { - a += window.AusCOM.json_list[i]['json'][name][i] + ' ' + a += window.DannyDevCOM.json_list[i]['json'][name][i] + ' ' } } } - if (typeof (window.AusCOM.json_list[i]['json'][name]) == 'boolean') { // id_4 逻辑 - return window.AusCOM.json_list[i]['json'][name] + if (typeof (window.DannyDevCOM.json_list[i]['json'][name]) == 'boolean') { // id_4 逻辑 + return window.DannyDevCOM.json_list[i]['json'][name] } } @@ -1291,9 +1296,9 @@ class AuEx_communication { json_del_item(args) { //删除项 const { id, name } = args try { - for (let i = 0, len = window.AusCOM.json_list.length; i < len; i++) { - if (window.AusCOM.json_list[i]['id'] == String(id)) { - Reflect.deleteProperty(window.AusCOM.json_list[i]['json'], String(name)) + for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { + if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { + Reflect.deleteProperty(window.DannyDevCOM.json_list[i]['json'], String(name)) return true } } @@ -1317,34 +1322,35 @@ class AuEx_communication { http_pm_cm(args) { const { id, name, text } = args let a = false - for (let i = 0, len = window.AusCOM.http_pm_.length; i < len; i++) { - if (window.AusCOM.http_pm_[i]['id'] == String(id)) { + for (let i = 0, len = window.DannyDevCOM.http_pm_.length; i < len; i++) { + if (window.DannyDevCOM.http_pm_[i]['id'] == String(id)) { a = true } } - if (a) { } else { window.AusCOM.http_pm_.push({ 'id': String(id), 'p': [] }) } - for (let i = 0, len = window.AusCOM.http_pm_.length; i < len; i++) { - if (window.AusCOM.http_pm_[i]['id'] == String(id)) { - for (let i_ = 0, len_ = window.AusCOM.http_pm_[i]['p'].length; i_ < len_; i_++) { - if (window.AusCOM.http_pm_[i]['p'][i_].name==name){ - window.AusCOM.http_pm_[i]['p'][i_].value=text - return '' - } + if (a) { } else { window.DannyDevCOM.http_pm_.push({ 'id': String(id), 'p': [] }) } + for (let i = 0, len = window.DannyDevCOM.http_pm_.length; i < len; i++) { + if (window.DannyDevCOM.http_pm_[i]['id'] == String(id)) { + for (let i_ = 0, len_ = window.DannyDevCOM.http_pm_[i]['p'].length; i_ < len_; i_++) { + if (window.DannyDevCOM.http_pm_[i]['p'][i_].name == name) { + window.DannyDevCOM.http_pm_[i]['p'][i_].value = text + return '' } - window.AusCOM.http_pm_[i]['p'].push({"name":name,'value':text}) + } + window.DannyDevCOM.http_pm_[i]['p'].push({ "name": name, 'value': text }) } } } http_pm_des(args) { const { id } = args - try{ - for (let i = 0, len = window.AusCOM.http_pm_.length; i < len; i++) { - if (window.AusCOM.http_pm_[i]['id'] == String(id)) { - window.AusCOM.http_pm_.splice(i,1) - return + try { + for (let i = 0, len = window.DannyDevCOM.http_pm_.length; i < len; i++) { + if (window.DannyDevCOM.http_pm_[i]['id'] == String(id)) { + window.DannyDevCOM.http_pm_.splice(i, 1) + return + } } - }} - catch(error){} + } + catch (error) { } } } -module.exports = AuEx_communication; +module.exports = AuEx_Communication; From c2b341940e91288ecb32dfdda3547a6b569d0d61 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sat, 15 Oct 2022 12:22:05 +0800 Subject: [PATCH 160/357] Update scCOM.js --- dannydev/scCOM.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index f18afab6..a328efa1 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1353,4 +1353,4 @@ class AuEx_Communication { catch (error) { } } } -module.exports = AuEx_Communication; +scratch.extensions.register(new AuEx_Communication()) From 3327a46b66a785146947e4e0c0305b4061cbf02d Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sat, 15 Oct 2022 14:52:01 +0800 Subject: [PATCH 161/357] Update scCOM.js --- dannydev/scCOM.js | 32 +++++++++++++++++++++++++------- 1 file changed, 25 insertions(+), 7 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index a328efa1..432e6196 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1,4 +1,3 @@ - const Cast = require('../../util/cast'); const ArgumentType = require('../../extension-support/argument-type'); const BlockType = require('../../extension-support/block-type'); @@ -111,9 +110,7 @@ class AuEx_Communication { 'DannyDevCOM.help.t': "If the http access fails to load XX, it may be that the domain name/ip does not exist, or it may be a cross domain problem. You can contact our website to modify the HTTP protocol header or modify the browser's cross domain settings. If ws is unable to send data, please F12 open the developer tool to check whether an error is reported. If an error is reported, the connection cannot be made. At present, the browser does not support detecting such errors. For more help, please go to https://gitee.com/ausx/scCOM", }, }) - window.DannyDevCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [] } - this.icon = "" - } + window.DannyDevCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [] } } formatMessage(id) { return this._formatMessage({ id, @@ -126,8 +123,6 @@ class AuEx_Communication { id: 'DannyDevNet', // 拓展id name: this.formatMessage('DannyDevCOM.extensionName'), // 拓展名 color1: '#FF8383', - blockIconURI: this.icon, - menuIconURI: this.icon, blocks: [ "---" + this.formatMessage("DannyDevCOM.tit1"), "---" + this.formatMessage("DannyDevCOM.tit1.1"), @@ -1353,4 +1348,27 @@ class AuEx_Communication { catch (error) { } } } -scratch.extensions.register(new AuEx_Communication()) + +window.tempExt = { + Extension: AuEx_Communication, + info: { + name: 'DannyDevNet.extensionName', + description: 'DannyDevNet.description', + extensionId: 'DannyDevNet', + iconURL: "", + insetIconURL: "", + featured: true, + disabled: false, + collaborator: 'only for hcn test', + }, + l10n: { + 'zh-cn': { + 'DannyDevNet.extensionName': 'AUSの通讯模块', + 'DannyDevNet.description': 'AUSの通讯与编解码', + }, + en: { + 'DannyDevNet.extensionName': 'AUSのcommunication', + 'DannyDevNet.description': 'AUSのcommunication and en/decode', + }, + }, + } From 31db8e4990e70651a831dd1e36597d1afddc9d72 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sat, 15 Oct 2022 22:42:48 +0800 Subject: [PATCH 162/357] Update scCOM.js --- dannydev/scCOM.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index 432e6196..2761338f 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1355,7 +1355,7 @@ window.tempExt = { name: 'DannyDevNet.extensionName', description: 'DannyDevNet.description', extensionId: 'DannyDevNet', - iconURL: "", + iconURL: "" insetIconURL: "", featured: true, disabled: false, From 0db75b4ca264e88136e17e29aa2b7e3a9d63d385 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sun, 16 Oct 2022 00:10:04 +0800 Subject: [PATCH 163/357] Update scCOM.js --- dannydev/scCOM.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index 2761338f..a9d80200 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1356,7 +1356,7 @@ window.tempExt = { description: 'DannyDevNet.description', extensionId: 'DannyDevNet', iconURL: "" - insetIconURL: "", + //insetIconURL: "", featured: true, disabled: false, collaborator: 'only for hcn test', From 2c4d020e73cfc6a47097eac3675cc36aaf8bc05a Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sun, 16 Oct 2022 00:23:33 +0800 Subject: [PATCH 164/357] Update scCOM.js MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 修复 --- dannydev/scCOM.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index a9d80200..c87bf759 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1355,8 +1355,8 @@ window.tempExt = { name: 'DannyDevNet.extensionName', description: 'DannyDevNet.description', extensionId: 'DannyDevNet', - iconURL: "" - //insetIconURL: "", + iconURL: "", + insetIconURL: "", featured: true, disabled: false, collaborator: 'only for hcn test', From bf06b06d3541da45d3b7bd24cbf9c39daaa09f06 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sun, 16 Oct 2022 11:37:36 +0800 Subject: [PATCH 165/357] Update scCOM.js MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 修bug --- dannydev/scCOM.js | 37 +++++++++++++++++++------------------ 1 file changed, 19 insertions(+), 18 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index c87bf759..8719efa7 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -110,7 +110,8 @@ class AuEx_Communication { 'DannyDevCOM.help.t': "If the http access fails to load XX, it may be that the domain name/ip does not exist, or it may be a cross domain problem. You can contact our website to modify the HTTP protocol header or modify the browser's cross domain settings. If ws is unable to send data, please F12 open the developer tool to check whether an error is reported. If an error is reported, the connection cannot be made. At present, the browser does not support detecting such errors. For more help, please go to https://gitee.com/ausx/scCOM", }, }) - window.DannyDevCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [] } } + window.DannyDevCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [] } + } formatMessage(id) { return this._formatMessage({ id, @@ -1352,23 +1353,23 @@ class AuEx_Communication { window.tempExt = { Extension: AuEx_Communication, info: { - name: 'DannyDevNet.extensionName', - description: 'DannyDevNet.description', - extensionId: 'DannyDevNet', - iconURL: "", - insetIconURL: "", - featured: true, - disabled: false, - collaborator: 'only for hcn test', + name: 'DannyDevNet.extensionName', + description: 'DannyDevNet.description', + extensionId: 'DannyDevNet', + iconURL: "", + insetIconURL: "", + featured: true, + disabled: false, + collaborator: 'only for hcn test', }, l10n: { - 'zh-cn': { - 'DannyDevNet.extensionName': 'AUSの通讯模块', - 'DannyDevNet.description': 'AUSの通讯与编解码', - }, - en: { - 'DannyDevNet.extensionName': 'AUSのcommunication', - 'DannyDevNet.description': 'AUSのcommunication and en/decode', - }, + 'zh-cn': { + 'DannyDevNet.extensionName': 'AUSの通讯模块', + 'DannyDevNet.description': 'AUSの通讯与编解码', + }, + en: { + 'DannyDevNet.extensionName': 'AUSのcommunication', + 'DannyDevNet.description': 'AUSのcommunication and en/decode', + }, }, - } +} From cacbc81d06394a4f0c8d451db9402e2571e91de5 Mon Sep 17 00:00:00 2001 From: nickhu <314484063@qq.com> Date: Mon, 17 Oct 2022 13:54:34 +0800 Subject: [PATCH 166/357] add git ignore --- .gitignore | 1 + 1 file changed, 1 insertion(+) create mode 100644 .gitignore diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..3c3629e6 --- /dev/null +++ b/.gitignore @@ -0,0 +1 @@ +node_modules From fcb686b65adeca3624d9c7dd59b3173a379468c0 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Mon, 17 Oct 2022 20:47:24 +0800 Subject: [PATCH 167/357] =?UTF-8?q?=E9=80=82=E9=85=8D=E6=B5=8F=E8=A7=88?= =?UTF-8?q?=E5=99=A8=EF=BC=8C=EF=BC=88nodejs=E5=87=BD=E6=95=B0=E5=85=BC?= =?UTF-8?q?=E5=AE=B9=EF=BC=89?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- dannydev/scCOM.js | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index 8719efa7..935b34b5 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1,7 +1,5 @@ -const Cast = require('../../util/cast'); -const ArgumentType = require('../../extension-support/argument-type'); -const BlockType = require('../../extension-support/block-type'); -const { string } = require('prop-types'); +import ArgumentType from '../../extension-support/argument-type' +import BlockType from'../../extension-support/block-type' // var dgram = require("dgram"); //感谢Arkos开源的扩展代码,作为参照编写框架 //欸,别把作者忘了。Danny欸 From d522c3d4494df48b1a2b10d5f8eafe87fc50612f Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Sat, 22 Oct 2022 09:34:09 +0800 Subject: [PATCH 168/357] Update scCOM.js --- dannydev/scCOM.js | 122 +++++++++++++++++++++++----------------------- 1 file changed, 61 insertions(+), 61 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index 935b34b5..a2572aa3 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1,5 +1,5 @@ -import ArgumentType from '../../extension-support/argument-type' -import BlockType from'../../extension-support/block-type' +//import ArgumentType from '../../extension-support/argument-type' +//import BlockType from'../../extension-support/block-type' // var dgram = require("dgram"); //感谢Arkos开源的扩展代码,作为参照编写框架 //欸,别把作者忘了。Danny欸 @@ -150,7 +150,7 @@ class AuEx_Communication { }, { opcode: 'http_pm_cm', - blockType: BlockType.COMMAND, + blockType: 'command', text: this.formatMessage('DannyDevCOM.http_pm'), arguments: { id: { @@ -169,7 +169,7 @@ class AuEx_Communication { }, { opcode: 'http_pm_des', - blockType: BlockType.COMMAND, + blockType: 'command', text: this.formatMessage('DannyDevCOM.http_pm_des'), arguments: { id: { @@ -207,69 +207,69 @@ class AuEx_Communication { }, { opcode: 'ws_recv', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.wsocket_recv'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '0' } }, }, { opcode: 'ws_close', - blockType: BlockType.BOOLEAN, + blockType: 'Boolean', text: this.formatMessage('DannyDevCOM.wsocket_conclose'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '0' } }, }, { opcode: 'ws_destroy', - blockType: BlockType.COMMAND, + blockType: 'command', text: this.formatMessage('DannyDevCOM.wsocket_close'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '0' } }, }, { opcode: 'ws_send', - blockType: BlockType.BOOLEAN, + blockType: 'Boolean', text: this.formatMessage('DannyDevCOM.wsocket_send'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '0' }, text: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' } }, }, // { // opcode: 'sock_cre', - // blockType: BlockType.REPORTER, + // blockType: 'reporter', // text: this.formatMessage('DannyDevCOM.socket_create'), // arguments: { // // yi: { - // type: ArgumentType.STRING, + // type: 'string', // menu: 'sock_yi' // }, // // method: { - // type: ArgumentType.STRING, + // type: 'string', // menu: 'sock_method' // }, // buff: { - // type: ArgumentType.STRING, + // type: 'string', // defaultValue: ' ' // } // }, @@ -277,26 +277,26 @@ class AuEx_Communication { "---" + this.formatMessage("DannyDevCOM.tit2"), { opcode: 'RC4_coding', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.code_RC4'), arguments: { KEY: { - type: ArgumentType.STRING, + type: 'string', defaultValue: 'Danny' }, TEXT: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '我来了' } }, }, { opcode: 'b64encoding', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.code_base64_en'), arguments: { TEXT: { - type: ArgumentType.STRING, + type: 'string', defaultValue: 'Danny' }, }, @@ -304,11 +304,11 @@ class AuEx_Communication { }, { opcode: 'b64decoding', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.code_base64_de'), arguments: { TEXT: { - type: ArgumentType.STRING, + type: 'string', defaultValue: 'RGFubnk=' }, }, @@ -316,11 +316,11 @@ class AuEx_Communication { }, { opcode: 'b58encoding', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.code_base58_en'), arguments: { TEXT: { - type: ArgumentType.STRING, + type: 'string', defaultValue: 'Danny' }, }, @@ -328,11 +328,11 @@ class AuEx_Communication { }, { opcode: 'b58decoding', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.code_base58_de'), arguments: { TEXT: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '8iTbLRE' }, }, @@ -340,15 +340,15 @@ class AuEx_Communication { }, { opcode: 'bcostomencoding', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.code_base_custom_en'), arguments: { ALPHABET: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '!@#$%^&*()' }, TEXT: { - type: ArgumentType.STRING, + type: 'string', defaultValue: 'Danny' }, }, @@ -356,15 +356,15 @@ class AuEx_Communication { }, { opcode: 'bcostomdecoding', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.code_base_custom_de'), arguments: { ALPHABET: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '!@#$%^&*()' }, TEXT: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '#)$&)#%!$$#@' }, }, @@ -374,96 +374,96 @@ class AuEx_Communication { "---" + this.formatMessage("DannyDevCOM.tit3"), { opcode: 'json_create', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.json_create'), arguments: { json_: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '{}' }, }, }, { opcode: 'json_load', - blockType: BlockType.BOOLEAN, + blockType: 'Boolean', text: this.formatMessage('DannyDevCOM.json_load'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, json_: { - type: ArgumentType.STRING, + type: 'string', defaultValue: '{}' }, }, }, { opcode: 'json_get_all_item', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.json_get_all_item'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, }, }, { opcode: 'json_add_from_id', - blockType: BlockType.BOOLEAN, + blockType: 'Boolean', text: this.formatMessage('DannyDevCOM.json_add_from_id'), arguments: { _id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, name: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, }, }, { opcode: 'json_add', - blockType: BlockType.BOOLEAN, + blockType: 'Boolean', text: this.formatMessage('DannyDevCOM.json_add'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, type: { - type: ArgumentType.STRING, + type: 'string', menu: 'JSON_add' }, name: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, TEXT: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, }, }, { opcode: 'json_get_item', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.json_get_item'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, name: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, @@ -471,16 +471,16 @@ class AuEx_Communication { }, { opcode: 'json_del_item', - blockType: BlockType.BOOLEAN, + blockType: 'Boolean', text: this.formatMessage('DannyDevCOM.json_del_item'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, name: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, @@ -488,24 +488,24 @@ class AuEx_Communication { }, { opcode: 'GETjson', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.json_get'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, } }, { opcode: 'DESTROYjson', - blockType: BlockType.BOOLEAN, + blockType: 'Boolean', text: this.formatMessage('DannyDevCOM.json_destroy'), arguments: { id: { - type: ArgumentType.STRING, + type: 'string', defaultValue: ' ' }, }, @@ -513,7 +513,7 @@ class AuEx_Communication { { opcode: 'help', - blockType: BlockType.REPORTER, + blockType: 'reporter', text: this.formatMessage('DannyDevCOM.help') + 'ver 1.00B1', }, From 4967b6888b3ba3975f8d0f498ef3b371bfdf27cd Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Tue, 25 Oct 2022 20:53:52 +0800 Subject: [PATCH 169/357] Update scCOM.js MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 修修value bug' --- dannydev/scCOM.js | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index a2572aa3..b6a121d5 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -615,7 +615,7 @@ class AuEx_Communication { http(args) { function get_p(p) { - pt = '?' + let pt = '?' for (let i = 0, len = window.DannyDevCOM.http_pm_.length; i < len; i++) { if (window.DannyDevCOM.http_pm_[i]['id'] == String(p)) { for (let i_ = 0, len_ = window.DannyDevCOM.http_pm_[i_].p.length; i_ < len_; i_++) { @@ -643,12 +643,12 @@ class AuEx_Communication { ws_cre_sock(args) { const { yi } = args if (yi == 'WS') { - id = Date.now() + let id = Date.now() window.DannyDevCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WS' }) return String(id) } if (yi == 'WSS') { - id = Date.now() + let id = Date.now() window.DannyDevCOM.wsock.push({ 'id': String(id), 'ip': null, 'port': null, 'obj': null, 'way': 'WSS' }) return String(id) } @@ -1078,8 +1078,8 @@ class AuEx_Communication { json_create(args) { const { json_ } = args function create(j) { - id = Date.now() - a = {} + let id = Date.now() + let a = {} try { a = JSON.parse(j) } @@ -1354,7 +1354,7 @@ window.tempExt = { name: 'DannyDevNet.extensionName', description: 'DannyDevNet.description', extensionId: 'DannyDevNet', - iconURL: "", + iconURL: "", insetIconURL: "", featured: true, disabled: false, From 9e0799d210405a72d0379a550d9cf5edb030cb2b Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Fri, 4 Nov 2022 21:57:02 +0800 Subject: [PATCH 170/357] =?UTF-8?q?=E5=AE=89=E5=85=A8=E9=99=90=E5=88=B6?= =?UTF-8?q?=EF=BC=8C=E6=B5=81=E9=87=8F=E9=99=90=E5=88=B6=EF=BC=8C=E9=A2=91?= =?UTF-8?q?=E7=8E=87=E9=99=90=E5=88=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- dannydev/scCOM.js | 68 +++++++++++++++++++---------------------------- 1 file changed, 27 insertions(+), 41 deletions(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index b6a121d5..33e6f7c0 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1,6 +1,5 @@ -//import ArgumentType from '../../extension-support/argument-type' -//import BlockType from'../../extension-support/block-type' -// var dgram = require("dgram"); +//import ArgumentType from '../../extension-support/argument-type'本地支持 +//import BlockType from'../../extension-support/block-type' 本地支持 //感谢Arkos开源的扩展代码,作为参照编写框架 //欸,别把作者忘了。Danny欸 class AuEx_Communication { @@ -108,7 +107,7 @@ class AuEx_Communication { 'DannyDevCOM.help.t': "If the http access fails to load XX, it may be that the domain name/ip does not exist, or it may be a cross domain problem. You can contact our website to modify the HTTP protocol header or modify the browser's cross domain settings. If ws is unable to send data, please F12 open the developer tool to check whether an error is reported. If an error is reported, the connection cannot be made. At present, the browser does not support detecting such errors. For more help, please go to https://gitee.com/ausx/scCOM", }, }) - window.DannyDevCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [] } + window.DannyDevCOM = { 'wsrecv_': [], 'wsnotalive_': [], 'json_list': [], 'wsock': [], 'http_pm_': [],'cl':[0,0]} } formatMessage(id) { return this._formatMessage({ @@ -605,7 +604,7 @@ class AuEx_Communication { try { var xmlHttp = new XMLHttpRequest(); xmlHttp.open(method, theUrl, false); // false 为同步请求 - xmlHttp.send(body); + xmlHttp.send(window.btoa(encodeURIComponent(String(body)))); //强制base64 return xmlHttp.responseText; } catch (error) { @@ -615,6 +614,13 @@ class AuEx_Communication { http(args) { function get_p(p) { + let d1 = new Date() + if (window.DannyDevCOM.cl[0]==0){} + else{ + if (parseInt(d1 - window.DannyDevCOM.cl[0])<500) { + setTimeout('console.log("限流延时机制调用中")', d1 - window.DannyDevCOM.cl[0]) + }} + window.DannyDevCOM.cl[1]= new Date() let pt = '?' for (let i = 0, len = window.DannyDevCOM.http_pm_.length; i < len; i++) { if (window.DannyDevCOM.http_pm_[i]['id'] == String(p)) { @@ -631,6 +637,7 @@ class AuEx_Communication { return pt } const { g_way, site, p, body } = args + if(String(body).length >1024*10){return false} if (p == 'None') { return this.httpGet(site, body, g_way); } @@ -740,7 +747,15 @@ class AuEx_Communication { console.log(window.DannyDevCOM.wsock[i]) try { - if (window.DannyDevCOM.wsock[i]["obj"] == null) { return false } else { window.DannyDevCOM.wsock[i]["obj"].send(text); return true } + let d1 = new Date() + if (window.DannyDevCOM.cl[1]==0){} + else{ + if (parseInt(d1 - window.DannyDevCOM.cl[1])<500) { + setTimeout('console.log("限流延时机制调用中")', d1 - window.DannyDevCOM.cl[1]) + }} + window.DannyDevCOM.cl[1]= new Date() + if(String(text).length >1024*10){return false} + if (window.DannyDevCOM.wsock[i]["obj"] == null) { return false } else { window.DannyDevCOM.wsock[i]["obj"].send(window.btoa(encodeURIComponent(String(text)))); return true }// 强制base64 } catch (error) { @@ -1150,27 +1165,12 @@ class AuEx_Communication { } json_add_from_id(args) { const { _id, id, name } = args - function check(id) { - for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { - if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { - return true - } - } - return false - } - function getjson(id) { - for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { - if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { - return window.DannyDevCOM.json_list[i]['json'] - } - } - return {} - } + function add_item_from_id(id, id1, name) { - if (check(id) & check(id1)) { + if (this.check(id) & this.check(id1)) { for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { - window.DannyDevCOM.json_list[i]['json'][name] = getjson(id1) + window.DannyDevCOM.json_list[i]['json'][name] = this.getjson(id1) return true } } @@ -1182,24 +1182,9 @@ class AuEx_Communication { } json_add(args) { const { id, type, name, TEXT } = args - function check(id) { - for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { - if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { - return true - } - } - return false - } - function getjson(id) { - for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { - if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { - return window.DannyDevCOM.json_list[i]['json'] - } - } - return {} - } + function add_item(id, name, type, content) { - if (check(id)) { + if (this.check(id)) { for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { if (window.DannyDevCOM.json_list[i]['id'] == String(id)) { if (type == '0') {//字符串 @@ -1237,6 +1222,7 @@ class AuEx_Communication { } return add_item(id, name, type, TEXT) } + GETjson(args) { const { id } = args for (let i = 0, len = window.DannyDevCOM.json_list.length; i < len; i++) { From 9dd056c7402402c5442ad162613cde61dc99f191 Mon Sep 17 00:00:00 2001 From: DannyDevPro <105956898+DannyDevPro@users.noreply.github.com> Date: Fri, 4 Nov 2022 22:03:20 +0800 Subject: [PATCH 171/357] Update scCOM.js --- dannydev/scCOM.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dannydev/scCOM.js b/dannydev/scCOM.js index 33e6f7c0..cc626fc5 100644 --- a/dannydev/scCOM.js +++ b/dannydev/scCOM.js @@ -1344,7 +1344,7 @@ window.tempExt = { insetIconURL: "", featured: true, disabled: false, - collaborator: 'only for hcn test', + collaborator: 'Danny @ CCW', }, l10n: { 'zh-cn': { From 1135c4dd27f11f318347a415287f9db60c489b7d Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 13 Nov 2022 11:37:47 +0800 Subject: [PATCH 172/357] Update File_Helper.js --- wit_cat/File_Helper.js | 190 +++++++++++++++++++++++++++++++++++++---- 1 file changed, 175 insertions(+), 15 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 57693743..f0d5c8c5 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -11,7 +11,9 @@ class WitCatFileHelper { this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "WitCatFileHelper.name": "文件助手", + "WitCatFileHelper.name": "wit_catの拓展", + "WitCatFileHelper.filehelper": "文件处理", + "WitCatFileHelper.inputmanagement": "文本框", "WitCatFileHelper.downloads": "将内容[text]按[s]分割后命名为[name]并下载多行文本", "WitCatFileHelper.download": "将内容[text]命名为[name]并下载", "WitCatFileHelper.save": "设置键[name]为[text]并储存到本地", @@ -20,11 +22,18 @@ class WitCatFileHelper { "WitCatFileHelper.segmentation": "将[text]按[s]分割", "WitCatFileHelper.encrypt": "base64加密[text]", "WitCatFileHelper.decrypt": "base64解密[text]", - "WitCatFileHelper.openfile": "打开文件" + "WitCatFileHelper.openfile": "打开文件", + "WitCatFileHelper.createinput": "设置或创建ID为[id]的文本框的宽[width]高[height]内容[text]", + "WitCatFileHelper.deleteinput": "删除ID为[id]的文本框", + "WitCatFileHelper.getinput": "获得ID为[id]的文本框内容", + "WitCatFileHelper.isinput": "焦点是否在ID为[id]的文本框上", + "WitCatFileHelper.nowinput": "将焦点聚焦在ID为[id]的文本框上", }, en: { - "WitCatFileHelper.name": "File Helper", - "WitCatFileHelper.downloads": "Download split content [text] by [S] named [name]", + "WitCatFileHelper.name": "wit_catのextension", + "WitCatFileHelper.filehelper": "file helper", + "WitCatFileHelper.inputmanagement": "input", + "WitCatFileHelper.downloads": "Download split content [text] by [s] named [name]", "WitCatFileHelper.download": "Download content [text] named [name]", "WitCatFileHelper.save": "Save content [text] with [name] on computer", "WitCatFileHelper.upload": "Get value [name]", @@ -32,7 +41,12 @@ class WitCatFileHelper { "WitCatFileHelper.segmentation": "Split [text] by [s]", "WitCatFileHelper.encrypt": "base64 encrypt[text]", "WitCatFileHelper.decrypt": "base64 decrypt[text]", - "WitCatFileHelper.openfile": "openfile" + "WitCatFileHelper.openfile": "openfile", + "WitCatFileHelper.createinput": "Set or create an input with ID[id]width[width]height[height]content[text]", + "WitCatFileHelper.deleteinput": "delete an input with ID[id]", + "WitCatFileHelper.getinput": "get an input with ID[id]", + "WitCatFileHelper.isinput": "is the focus on the input with ID[id]?", + "WitCatFileHelper.nowinput": "let teh focus on the input with ID[id]", } }) } @@ -54,7 +68,9 @@ class WitCatFileHelper { menuIconURI: _icon, color1: "#60D6F4", color2: "#55a7f7", - blocks: [{ + blocks: [ + "---"+this.formatMessage("WitCatFileHelper.filehelper"), + { opcode: "downloads", blockType: "command", text: this.formatMessage("WitCatFileHelper.downloads"), @@ -147,7 +163,7 @@ class WitCatFileHelper { arguments: { text: { type: "string", - defaultValue: 'i love中国', + defaultValue: 'i love china', }, }, }, @@ -158,7 +174,7 @@ class WitCatFileHelper { arguments: { text: { type: "string", - defaultValue: 'aSUyMGxvdmUlRTQlQjglQUQlRTUlOUIlQkQ=', + defaultValue: 'aSUyMGxvdmUlMjBjaGluYQ==', }, }, }, @@ -168,6 +184,74 @@ class WitCatFileHelper { text: this.formatMessage("WitCatFileHelper.openfile"), arguments: {}, }, + "---"+this.formatMessage("WitCatFileHelper.inputmanagement"), + { + opcode: "createinput", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.createinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + }, + width:{ + type:"number", + defaultValue:"100", + }, + height:{ + type:"number", + defaultValue:"20", + }, + text:{ + type:"string", + defaultValue:"hello word!", + } + }, + }, + { + opcode: "deleteinput", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.deleteinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + } + }, + }, + { + opcode: "getinput", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.getinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + } + }, + }, + { + opcode: "isinput", + blockType: "bool", + text: this.formatMessage("WitCatFileHelper.isinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + } + }, + }, + { + opcode: "nowinput", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.nowinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + } + }, + }, ] }; } @@ -175,7 +259,11 @@ class WitCatFileHelper { downloads(args) { let h = args.text; let text = args.text; - const filename = args.name; + let filenames = args.name; + if(filenames == ""){ + filenames = "none.txt" + } + const filename = filenames; let s = args.s; let j = 0; if (s != "") { @@ -203,8 +291,12 @@ class WitCatFileHelper { } //下载文件 download(args) { - const filename = args.name; const content = args.text; + let filenames = args.name; + if(filenames == ""){ + filenames = "none.txt" + } + const filename = filenames; // 创建隐藏的可下载链接 let eleLink = document.createElement('a'); eleLink.download = filename; @@ -272,6 +364,7 @@ class WitCatFileHelper { return new Promise(resolve => { const input = document.createElement("input"); input.type = "file"; + input.click(); input.onchange = () => { const reader = new FileReader(); const file = input.files[0]; @@ -284,9 +377,76 @@ class WitCatFileHelper { }; reader.readAsText(file); } - input.click(); + // 当点击取消 + window.addEventListener('focus',() => { + setTimeout(() => { + if (fileCancle) { + resolve(); + } + }, 100) + }, + { + once: true + } + ) }); } + //设置或创建文本框 + createinput(args){ + let dom = `position:absolute; left:`+ args.x + `px; top:` + args.y + `px; width:` + args.width + `px; height:` + args.height + `px`; + let search = document.getElementsByClassName("WitCatInput" + args.id); + if(search.length != 0){ + search[0].style = dom; + scratch[0].value = args.text; + } + else{ + let eleLink = document.createElement('input'); + eleLink.type = "text"; + eleLink.style = dom; + eleLink.className = "WitCatInput" + args.id; + eleLink.value = args.text; + document.body.appendChild(eleLink); + } + } + //删除文本框 + deleteinput(args){ + let search = document.getElementsByClassName("WitCatInput" + args.id); + if(search.length != 0){ + document.body.removeChild(scratch[0]); + } + } + //获取文本框内容 + getinput(args){ + let search = document.getElementsByClassName("WitCatInput" + args.id); + if(search.length != 0){ + return(scratch[0].value); + } + else{ + return(""); + } + } + //焦点判断 + isinput(args){ + let search = document.getElementsByClassName("WitCatInput" + args.id); + if(search.length != 0){ + if(document.hasFocus()){ + return(true); + } + else{ + return(false); + } + } + else{ + return(false); + } + } + //焦点获取 + nowinput(args){ + let search = document.getElementsByClassName("WitCatInput" + args.id); + if(search.length != 0){ + search[0].focus(); + } + } } window.tempExt = { @@ -303,12 +463,12 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatFileHelper.name": "文件助手", - "WitCatFileHelper.descp": "处理本地数据" + "WitCatFileHelper.name": "wit_catの拓展", + "WitCatFileHelper.descp": "一堆没用的小玩意" }, en: { - "WitCatFileHelper.name": "File Helper", - "WitCatFileHelper.descp": "Handling local data" + "WitCatFileHelper.name": "wit_catのextension", + "WitCatFileHelper.descp": "some useless block" } } }; From 424a0e34db8b8aec1afaf7cf0b65462848bf0060 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 13 Nov 2022 12:58:23 +0800 Subject: [PATCH 173/357] Update File_Helper.js --- wit_cat/File_Helper.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index f0d5c8c5..88e83ce8 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -393,7 +393,7 @@ class WitCatFileHelper { } //设置或创建文本框 createinput(args){ - let dom = `position:absolute; left:`+ args.x + `px; top:` + args.y + `px; width:` + args.width + `px; height:` + args.height + `px`; + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ args.x + `px; top:` + args.y + `px; width:` + args.width + `px; height:` + args.height + `px`; let search = document.getElementsByClassName("WitCatInput" + args.id); if(search.length != 0){ search[0].style = dom; From e51f65983bd5778f3b3d5e6abf00a51142209e39 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 13 Nov 2022 21:47:47 +0800 Subject: [PATCH 174/357] Update File_Helper.js --- wit_cat/File_Helper.js | 30 +++++++++++++++++------------- 1 file changed, 17 insertions(+), 13 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 88e83ce8..33138649 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -23,7 +23,7 @@ class WitCatFileHelper { "WitCatFileHelper.encrypt": "base64加密[text]", "WitCatFileHelper.decrypt": "base64解密[text]", "WitCatFileHelper.openfile": "打开文件", - "WitCatFileHelper.createinput": "设置或创建ID为[id]的文本框的宽[width]高[height]内容[text]", + "WitCatFileHelper.createinput": "设置或创建ID为[id]的文本框的X[x]Y[y]宽[width]高[height]内容[text]", "WitCatFileHelper.deleteinput": "删除ID为[id]的文本框", "WitCatFileHelper.getinput": "获得ID为[id]的文本框内容", "WitCatFileHelper.isinput": "焦点是否在ID为[id]的文本框上", @@ -42,7 +42,7 @@ class WitCatFileHelper { "WitCatFileHelper.encrypt": "base64 encrypt[text]", "WitCatFileHelper.decrypt": "base64 decrypt[text]", "WitCatFileHelper.openfile": "openfile", - "WitCatFileHelper.createinput": "Set or create an input with ID[id]width[width]height[height]content[text]", + "WitCatFileHelper.createinput": "Set or create an input with ID[id]X[x]Y[y]width[width]height[height]content[text]", "WitCatFileHelper.deleteinput": "delete an input with ID[id]", "WitCatFileHelper.getinput": "get an input with ID[id]", "WitCatFileHelper.isinput": "is the focus on the input with ID[id]?", @@ -194,6 +194,14 @@ class WitCatFileHelper { type:"string", defaultValue:"i", }, + x:{ + type:"number", + defaultValue:"0", + }, + y:{ + type:"number", + defaultValue:"0", + }, width:{ type:"number", defaultValue:"100", @@ -232,7 +240,7 @@ class WitCatFileHelper { }, { opcode: "isinput", - blockType: "bool", + blockType: "Boolean", text: this.formatMessage("WitCatFileHelper.isinput"), arguments: { id:{ @@ -362,6 +370,7 @@ class WitCatFileHelper { //打开文件 openfile(args){ return new Promise(resolve => { + let a = 0; const input = document.createElement("input"); input.type = "file"; input.click(); @@ -378,17 +387,12 @@ class WitCatFileHelper { reader.readAsText(file); } // 当点击取消 - window.addEventListener('focus',() => { - setTimeout(() => { - if (fileCancle) { - resolve(); - } - }, 100) - }, - { - once: true + input.onfocus = function () { + a++; + if (a == 2) { + resolve(); } - ) + }, }); } //设置或创建文本框 From 27a436ba08dddbb3945e4f1b85a20b5904195cf6 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 13 Nov 2022 22:39:12 +0800 Subject: [PATCH 175/357] Update File_Helper.js --- wit_cat/File_Helper.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 33138649..6f62c019 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -392,7 +392,7 @@ class WitCatFileHelper { if (a == 2) { resolve(); } - }, + } }); } //设置或创建文本框 From eb360899a4d7146897e6df1cc46326b54eef5a28 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 14 Nov 2022 13:00:48 +0800 Subject: [PATCH 176/357] Update File_Helper.js --- wit_cat/File_Helper.js | 32 ++++++++++++++++---------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 6f62c019..b334e9ae 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -398,32 +398,32 @@ class WitCatFileHelper { //设置或创建文本框 createinput(args){ let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ args.x + `px; top:` + args.y + `px; width:` + args.width + `px; height:` + args.height + `px`; - let search = document.getElementsByClassName("WitCatInput" + args.id); - if(search.length != 0){ - search[0].style = dom; - scratch[0].value = args.text; + let search = document.getElementById("WitCatInput" + args.id); + if(search != null){ + search.style = dom; + search.value = args.text; } else{ let eleLink = document.createElement('input'); eleLink.type = "text"; eleLink.style = dom; - eleLink.className = "WitCatInput" + args.id; + eleLink.id = "WitCatInput" + args.id; eleLink.value = args.text; document.body.appendChild(eleLink); } } //删除文本框 deleteinput(args){ - let search = document.getElementsByClassName("WitCatInput" + args.id); - if(search.length != 0){ - document.body.removeChild(scratch[0]); + let search = document.getElementById("WitCatInput" + args.id); + if(search != 0){ + document.body.removeChild(search); } } //获取文本框内容 getinput(args){ - let search = document.getElementsByClassName("WitCatInput" + args.id); - if(search.length != 0){ - return(scratch[0].value); + let search = document.getElementById("WitCatInput" + args.id); + if(search != null){ + return(search.value); } else{ return(""); @@ -431,8 +431,8 @@ class WitCatFileHelper { } //焦点判断 isinput(args){ - let search = document.getElementsByClassName("WitCatInput" + args.id); - if(search.length != 0){ + let search = document.getElementById("WitCatInput" + args.id); + if(search != null){ if(document.hasFocus()){ return(true); } @@ -446,9 +446,9 @@ class WitCatFileHelper { } //焦点获取 nowinput(args){ - let search = document.getElementsByClassName("WitCatInput" + args.id); - if(search.length != 0){ - search[0].focus(); + let search = document.getElementById("WitCatInput" + args.id); + if(search != null){ + search.focus(); } } } From eea77f3a09906c22f9b7968f14de7521ca6210bc Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 20 Nov 2022 14:15:22 +0800 Subject: [PATCH 177/357] Update File_Helper.js --- wit_cat/File_Helper.js | 580 ++++++++++++++++++++++++++++++++++++++--- 1 file changed, 542 insertions(+), 38 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index b334e9ae..ff30d1d4 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -14,39 +14,69 @@ class WitCatFileHelper { "WitCatFileHelper.name": "wit_catの拓展", "WitCatFileHelper.filehelper": "文件处理", "WitCatFileHelper.inputmanagement": "文本框", + "WitCatFileHelper.Multiplelinestext": "多行文本", "WitCatFileHelper.downloads": "将内容[text]按[s]分割后命名为[name]并下载多行文本", "WitCatFileHelper.download": "将内容[text]命名为[name]并下载", "WitCatFileHelper.save": "设置键[name]为[text]并储存到本地", "WitCatFileHelper.upload": "获取键[name]的值", "WitCatFileHelper.delete": "删除键[name]", - "WitCatFileHelper.segmentation": "将[text]按[s]分割", + "WitCatFileHelper.segmentation": "将[text]按[s]分割为[thing]", "WitCatFileHelper.encrypt": "base64加密[text]", "WitCatFileHelper.decrypt": "base64解密[text]", "WitCatFileHelper.openfile": "打开文件", - "WitCatFileHelper.createinput": "设置或创建ID为[id]的文本框的X[x]Y[y]宽[width]高[height]内容[text]", + "WitCatFileHelper.createinput": "设置或创建ID为[id]的文本框的X[x]Y[y]宽[width]高[height]内容[text]提示[texts]", "WitCatFileHelper.deleteinput": "删除ID为[id]的文本框", "WitCatFileHelper.getinput": "获得ID为[id]的文本框内容", "WitCatFileHelper.isinput": "焦点是否在ID为[id]的文本框上", "WitCatFileHelper.nowinput": "将焦点聚焦在ID为[id]的文本框上", + "WitCatFileHelper.deleteallinput": "删除所有文本框", + "WitCatFileHelper.showvar": "设置键[name]的状态为[show]", + "WitCatFileHelper.saveother": "设置作品ID[id]的键[name]为[text]", + "WitCatFileHelper.uploadother": "获取作品[id]的键[name]的值", + "WitCatFileHelper.other": "作品[id]的键[name]的状态", + "WitCatFileHelper.showon": "只读", + "WitCatFileHelper.showoff": "私有", + "WitCatFileHelper.showall": "公开", + "WitCatFileHelper.deleteMultiplelinestext": "删除[text]的第[num]行", + "WitCatFileHelper.addMultiplelinestext": "将[text]加入[texts]的第[num]行", + "WitCatFileHelper.whatMultiplelinestext": "[text]的第[num]行", + "WitCatFileHelper.numMultiplelinestext": "[text]的行数", + "WitCatFileHelper.thing.1": "数组", + "WitCatFileHelper.thing.2": "多行文本", }, en: { "WitCatFileHelper.name": "wit_catのextension", "WitCatFileHelper.filehelper": "file helper", "WitCatFileHelper.inputmanagement": "input", + "WitCatFileHelper.Multiplelinestext": "Multiple lines of text", "WitCatFileHelper.downloads": "Download split content [text] by [s] named [name]", "WitCatFileHelper.download": "Download content [text] named [name]", "WitCatFileHelper.save": "Save content [text] with [name] on computer", "WitCatFileHelper.upload": "Get value [name]", "WitCatFileHelper.delete": "delete value [name]", - "WitCatFileHelper.segmentation": "Split [text] by [s]", + "WitCatFileHelper.segmentation": "Split [text] by [s] to [thing]", "WitCatFileHelper.encrypt": "base64 encrypt[text]", "WitCatFileHelper.decrypt": "base64 decrypt[text]", "WitCatFileHelper.openfile": "openfile", - "WitCatFileHelper.createinput": "Set or create an input with ID[id]X[x]Y[y]width[width]height[height]content[text]", + "WitCatFileHelper.createinput": "Set or create an input with ID[id]X[x]Y[y]width[width]height[height]content[text]prompt[texts]", "WitCatFileHelper.deleteinput": "delete an input with ID[id]", "WitCatFileHelper.getinput": "get an input with ID[id]", "WitCatFileHelper.isinput": "is the focus on the input with ID[id]?", "WitCatFileHelper.nowinput": "let teh focus on the input with ID[id]", + "WitCatFileHelper.deleteallinput": "delete all input", + "WitCatFileHelper.showvar": "set value[name]`s state[show]", + "WitCatFileHelper.saveother": "save ID[id]`s content name[name]with[text]", + "WitCatFileHelper.uploadother": "get ID[id]`s value[name]", + "WitCatFileHelper.other": "get ID[id]`s value[name]`s state", + "WitCatFileHelper.showon": "View", + "WitCatFileHelper.showoff": "private", + "WitCatFileHelper.showall": "public", + "WitCatFileHelper.deleteMultiplelinestext": "delete[text]the[num]line of text", + "WitCatFileHelper.addMultiplelinestext": "let[text]add with[texts]the[num]line of text", + "WitCatFileHelper.whatMultiplelinestext": "[text]`s[num]line of text", + "WitCatFileHelper.numMultiplelinestext": "[text]Number of rows", + "WitCatFileHelper.thing.1": "array", + "WitCatFileHelper.thing.2": "Multiple lines of text", } }) } @@ -104,6 +134,12 @@ class WitCatFileHelper { }, }, }, + { + opcode: "openfile", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.openfile"), + arguments: {}, + }, { opcode: "upload", blockType: "reporter", @@ -131,28 +167,77 @@ class WitCatFileHelper { }, }, { - opcode: "segmentation", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.segmentation"), + opcode: "delete", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.delete"), arguments: { - text: { + name: { type: "string", - defaultValue: 'wow/!!!', + defaultValue: 'i', }, - s: { - type: "string", - defaultValue: '/', + }, + }, + { + opcode: "showvar", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.showvar"), + arguments: { + name:{ + type:"string", + defaultValue:"i", + }, + show:{ + type:"string", + menu:"setvariable", }, }, }, { - opcode: "delete", + opcode: "saveother", blockType: "command", - text: this.formatMessage("WitCatFileHelper.delete"), + text: this.formatMessage("WitCatFileHelper.saveother"), arguments: { - name: { - type: "string", - defaultValue: 'i', + id:{ + type:"string", + defaultValue:"6373950041d21d2d2cd0da9b", + }, + name:{ + type:"string", + defaultValue:"i", + }, + text:{ + type:"string", + defaultValue:"wit_cat!", + }, + }, + }, + { + opcode: "uploadother", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.uploadother"), + arguments: { + id:{ + type:"string", + defaultValue:"6373950041d21d2d2cd0da9b", + }, + name:{ + type:"string", + defaultValue:"i", + }, + }, + }, + { + opcode: "other", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.other"), + arguments: { + id:{ + type:"string", + defaultValue:"6373950041d21d2d2cd0da9b", + }, + name:{ + type:"string", + defaultValue:"i", }, }, }, @@ -178,12 +263,6 @@ class WitCatFileHelper { }, }, }, - { - opcode: "openfile", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.openfile"), - arguments: {}, - }, "---"+this.formatMessage("WitCatFileHelper.inputmanagement"), { opcode: "createinput", @@ -213,6 +292,10 @@ class WitCatFileHelper { text:{ type:"string", defaultValue:"hello word!", + }, + texts:{ + type:"string", + defaultValue:"hello word!", } }, }, @@ -260,7 +343,119 @@ class WitCatFileHelper { } }, }, - ] + { + opcode: "deleteallinput", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.deleteallinput"), + arguments: {}, + }, + "---"+this.formatMessage("WitCatFileHelper.Multiplelinestext"), + { + opcode: "segmentation", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.segmentation"), + arguments: { + text: { + type: "string", + defaultValue: 'wow/!!!', + }, + s: { + type: "string", + defaultValue: '/', + }, + thing:{ + type: "string", + menu: 'thing', + } + }, + }, + { + opcode: "deleteMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.deleteMultiplelinestext"), + arguments: { + text:{ + type:"string", + defaultValue:"wit_cat!!!\nwit_cat!!!", + }, + num:{ + type:"number", + defaultValue:"1", + } + }, + }, + { + opcode: "addMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.addMultiplelinestext"), + arguments: { + text:{ + type:"string", + defaultValue:"wit_cat!!!", + }, + texts:{ + type:"string", + defaultValue:"wit_cat!!!\nwit_cat!!!", + }, + num:{ + type:"string", + defaultValue:"last", + } + }, + }, + { + opcode: "whatMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.whatMultiplelinestext"), + arguments: { + text:{ + type:"string", + defaultValue:"wit_cat!!!", + }, + num:{ + type:"string", + defaultValue:"1", + } + }, + }, + { + opcode: "numMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.numMultiplelinestext"), + arguments: { + text:{ + type:"string", + defaultValue:"wit_cat!!!\nwit_cat!!!", + }, + }, + }, + ], + menus:{ + setvariable: [ + { + text: this.formatMessage('WitCatFileHelper.showall'), + value: '#' + }, + { + text: this.formatMessage('WitCatFileHelper.showon'), + value: '$' + }, + { + text: this.formatMessage('WitCatFileHelper.showoff'), + value: '' + }, + ], + thing: [ + { + text: this.formatMessage('WitCatFileHelper.thing.2'), + value: 'false' + }, + { + text: this.formatMessage('WitCatFileHelper.thing.1'), + value: 'true' + }, + ], + } }; } //下载多行文件 @@ -322,35 +517,97 @@ class WitCatFileHelper { upload(args) { const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); - return localStorage.getItem(h + name); + //寻找状态 + let show = ""; + let value = localStorage.getItem(h + name); + if (value == null) { + value = localStorage.getItem("#" + h + name); + if (value == null) { + value = localStorage.getItem("$" + h + name); + if (value != null) { + show = "$"; + } + } + else { + show = "#"; + } + } + + return localStorage.getItem(show + h + name); } //保存本地变量 save(args) { const text = args.text; const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); + //寻找状态 + let show = ""; + let value = localStorage.getItem(h + name); + if (value == null) { + value = localStorage.getItem("#" + h + name); + if (value == null) { + value = localStorage.getItem("$" + h + name); + if (value != null) { + show = "$"; + } + } + else { + show = "#"; + } + } + if(h == ""){ alert("请先保存作品"); } else{ - localStorage.setItem(h + name, text); + localStorage.setItem(show + h + name, text); } } //删除本地变量 delete(args) { const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); - localStorage.removeItem(h + name); + //寻找状态 + let show = ""; + let value = localStorage.getItem(h + name); + if (value == null) { + value = localStorage.getItem("#" + h + name); + if (value == null) { + value = localStorage.getItem("$" + h + name); + if (value != null) { + show = "$"; + } + } + else { + show = "#"; + } + } + + localStorage.removeItem(show + h + name); } //字符串分割 segmentation(args) { let text = args.text; let s = args.s; let array = text.split(s); - const a = `","`; - const b = `["`; - const c = `"]`; - let str = array.join(a); - let r = b + str + c; + let r = ""; + if(args.thing == "true"){ + const a = `","`; + const b = `["`; + const c = `"]`; + let str = array.join(a); + r = b + str + c; + } + else{ + let i; + for(i = 0;i < array.length;i++){ + if(r == ""){ + r = array[i]; + } + else{ + r = r + "\n" + array[i]; + } + } + } return r; } //加密 @@ -381,7 +638,6 @@ class WitCatFileHelper { resolve(e.target.result); }; reader.onerror = (e) => { - console.error(e); resolve(); }; reader.readAsText(file); @@ -397,11 +653,56 @@ class WitCatFileHelper { } //设置或创建文本框 createinput(args){ - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ args.x + `px; top:` + args.y + `px; width:` + args.width + `px; height:` + args.height + `px`; + let x = args.x; + let y = args.y; + let width = args.width; + let height = args.height; + if(args.x > this.runtime.stageWidth){ + x = this.runtime.stageWidth; + } + if(args.x < 0){ + x = 0; + } + if(args.y > this.runtime.stageHeight){ + y = this.runtime.stageHeight; + } + if(args.y < 0){ + y = 0; + } + if(args.x + args.width > this.runtime.stageWidth){ + width = this.runtime.stageWidth - x; + } + if(args.width < 0){ + width = 0; + } + if(args.y + args.height > this.runtime.stageHeight){ + height = this.runtime.stageHeight - y; + } + if(args.height < 0){ + height = 0; + } + x = (x / this.runtime.stageWidth) * 100; + y = (y / this.runtime.stageHeight) * 100; + width = (width / this.runtime.stageWidth) * 100; + height = (height / this.runtime.stageHeight) * 100; + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;`; let search = document.getElementById("WitCatInput" + args.id); + //找渲染div + let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 + if(div == null){ + div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 + if(div == null){ + div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 + if(div == null){ + alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); + return; + } + } + } if(search != null){ search.style = dom; search.value = args.text; + search.placeholder = args.texts; } else{ let eleLink = document.createElement('input'); @@ -409,14 +710,17 @@ class WitCatFileHelper { eleLink.style = dom; eleLink.id = "WitCatInput" + args.id; eleLink.value = args.text; - document.body.appendChild(eleLink); + eleLink.className = "WitCatInput"; + eleLink.placeholder = args.texts; + div.appendChild(eleLink); } } //删除文本框 deleteinput(args){ let search = document.getElementById("WitCatInput" + args.id); - if(search != 0){ - document.body.removeChild(search); + if(search != null){ + let div = search.parentNode; + div.removeChild(search); } } //获取文本框内容 @@ -433,7 +737,7 @@ class WitCatFileHelper { isinput(args){ let search = document.getElementById("WitCatInput" + args.id); if(search != null){ - if(document.hasFocus()){ + if(search == document.activeElement){ return(true); } else{ @@ -451,6 +755,206 @@ class WitCatFileHelper { search.focus(); } } + //删除所有文本框 + deleteallinput(args){ + let search = document.getElementsByClassName("WitCatInput"); + let i = 0; + let div; + for(i = search.length - 1;i >= 0;i--){ + div = search[i].parentNode; + div.removeChild(search[i]); + } + } + //设置状态 + showvar(args){ + const name = args.name; + let h = this.runtime.ccwAPI.getProjectUUID(); + //寻找状态 + let show = ""; + let value = localStorage.getItem(h + name); + if (value == null) { + value = localStorage.getItem("#" + h + name); + if (value == null) { + value = localStorage.getItem("$" + h + name); + if (value != null) { + show = "$"; + } + } + else { + show = "#"; + } + } + + let text = localStorage.getItem(show + h + name); + localStorage.removeItem(show + h + name); + if(h == ""){ + alert("请先保存作品"); + } else{ + localStorage.setItem(args.show + h + name, text); + } + } + //修改别人的键 + saveother(args){ + let name = args.name; + let h = args.id; + let text = args.text; + //寻找状态 + let show = ""; + let value = localStorage.getItem(h + name); + if (value == null) { + value = localStorage.getItem("#" + h + name); + if (value == null) { + value = localStorage.getItem("$" + h + name); + if (value == null) { + show = "null"; + } + else{ + show = "$"; + } + } + else { + show = "#"; + } + } + + if(show == "#" || show == "null"){ + localStorage.setItem("#" + h + name, text); + } + } + //获取别人的键 + uploadother(args){ + let name = args.name; + let h = args.id; + //寻找状态 + let show = ""; + let value = localStorage.getItem(h + name); + if (value == null) { + value = localStorage.getItem("#" + h + name); + if (value == null) { + value = localStorage.getItem("$" + h + name); + if (value != null) { + show = "$"; + } + } + else { + show = "#"; + } + } + + if(show != ""){ + return localStorage.getItem(show + h + name); + } + } + //获取键状态 + other(args){ + let name = args.name; + let h = args.id; + //寻找状态 + let show = ""; + let value = localStorage.getItem(h + name); + if (value == null) { + value = localStorage.getItem("#" + h + name); + if (value == null) { + value = localStorage.getItem("$" + h + name); + if (value == null) { + show = "null"; + } + else{ + show = "$"; + } + } + else { + show = "#"; + } + } + + if(show == "#"){ + return "公开"; + } + if(show == "$"){ + return "只读"; + } + if(show == ""){ + return "私有"; + } + return "键不存在"; + } + //删除多行文本行 + deleteMultiplelinestext(args){ + let text = args.text; + let texts = text.split('\n'); + let num = args.num; + if(args.num == "last"){ + num = texts.length + 1; + } + if(args.num == "first"){ + num = 1; + } + let a = ""; + let i; + for(i = 0;i < texts.length;i++){ + if(i + 1 != num){ + if(a == ""){ + a = texts[i]; + } + else{ + a = a + "\n" + texts[i]; + } + } + } + return a; + } + //插入多行文本 + addMultiplelinestext(args){ + let text = args.text; + let texts = args.texts.split("\n"); + let num = args.num; + if(args.num == "last"){ + num = texts.length + 1; + } + if(args.num == "first"){ + num = 1; + } + let a = ""; + let i ,j = 0; + for(i = 0;i <= texts.length;i++){ + if(i + 1 != num){ + if(a == ""){ + a = texts[i + j]; + } + else{ + a = a + "\n" + texts[i + j]; + } + } + else{ + if(a == ""){ + a = text; + } + else{ + a = a + "\n" + text; + } + j = -1; + } + } + return a; + } + //多行文本第几行 + whatMultiplelinestext(args){ + let text = args.text.split("\n"); + let num = args.num; + if(args.num == "last"){ + num = text.length + 1; + } + if(args.num == "first"){ + num = 1; + } + return text[num + 1]; + } + //多行文本行数 + numMultiplelinestext(args){ + let text = args.text.split("\n"); + return text.length; + } } window.tempExt = { From 7be9a49b505fb9e4523eb1e4320ae1852ebb6def Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 20 Nov 2022 16:04:06 +0800 Subject: [PATCH 178/357] Update File_Helper.js --- wit_cat/File_Helper.js | 44 +++++++++++++++++++++++++++++++++++++----- 1 file changed, 39 insertions(+), 5 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index ff30d1d4..01ac29c0 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -24,7 +24,7 @@ class WitCatFileHelper { "WitCatFileHelper.encrypt": "base64加密[text]", "WitCatFileHelper.decrypt": "base64解密[text]", "WitCatFileHelper.openfile": "打开文件", - "WitCatFileHelper.createinput": "设置或创建ID为[id]的文本框的X[x]Y[y]宽[width]高[height]内容[text]提示[texts]", + "WitCatFileHelper.createinput": "设置或创建ID为[id]的文本框的X[x]Y[y]宽[width]高[height]内容[text]提示[texts]字体大小[size]", "WitCatFileHelper.deleteinput": "删除ID为[id]的文本框", "WitCatFileHelper.getinput": "获得ID为[id]的文本框内容", "WitCatFileHelper.isinput": "焦点是否在ID为[id]的文本框上", @@ -43,6 +43,7 @@ class WitCatFileHelper { "WitCatFileHelper.numMultiplelinestext": "[text]的行数", "WitCatFileHelper.thing.1": "数组", "WitCatFileHelper.thing.2": "多行文本", + "WitCatFileHelper.compute": "当前分辨率下高[size]的字体大小", }, en: { "WitCatFileHelper.name": "wit_catのextension", @@ -58,7 +59,7 @@ class WitCatFileHelper { "WitCatFileHelper.encrypt": "base64 encrypt[text]", "WitCatFileHelper.decrypt": "base64 decrypt[text]", "WitCatFileHelper.openfile": "openfile", - "WitCatFileHelper.createinput": "Set or create an input with ID[id]X[x]Y[y]width[width]height[height]content[text]prompt[texts]", + "WitCatFileHelper.createinput": "Set or create an input with ID[id]X[x]Y[y]width[width]height[height]content[text]prompt[texts]font-size[size]", "WitCatFileHelper.deleteinput": "delete an input with ID[id]", "WitCatFileHelper.getinput": "get an input with ID[id]", "WitCatFileHelper.isinput": "is the focus on the input with ID[id]?", @@ -77,6 +78,7 @@ class WitCatFileHelper { "WitCatFileHelper.numMultiplelinestext": "[text]Number of rows", "WitCatFileHelper.thing.1": "array", "WitCatFileHelper.thing.2": "Multiple lines of text", + "WitCatFileHelper.compute": "The font size at the current resolution that is high [size]", } }) } @@ -296,6 +298,21 @@ class WitCatFileHelper { texts:{ type:"string", defaultValue:"hello word!", + }, + size:{ + type:"number", + defaultValue:"16", + } + }, + }, + { + opcode: "compute", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.compute"), + arguments: { + size:{ + type:"number", + defaultValue:"16", } }, }, @@ -685,7 +702,7 @@ class WitCatFileHelper { y = (y / this.runtime.stageHeight) * 100; width = (width / this.runtime.stageWidth) * 100; height = (height / this.runtime.stageHeight) * 100; - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;`; + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;`; let search = document.getElementById("WitCatInput" + args.id); //找渲染div let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 @@ -943,18 +960,35 @@ class WitCatFileHelper { let text = args.text.split("\n"); let num = args.num; if(args.num == "last"){ - num = text.length + 1; + num = text.length; } if(args.num == "first"){ num = 1; } - return text[num + 1]; + return text[num - 1]; } //多行文本行数 numMultiplelinestext(args){ let text = args.text.split("\n"); return text.length; } + //计算坐标 + compute(args){ + //找渲染div + let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 + if(div == null){ + div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 + if(div == null){ + div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 + if(div == null){ + alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); + return; + } + } + } + console.log(div.style.width) + return (div.style.width.split("px")[0] / 360) * args.size; + } } window.tempExt = { From c5837e632a09ff8773485b2d9298676ec4aa8ebc Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 22 Nov 2022 13:32:04 +0800 Subject: [PATCH 179/357] Update File_Helper.js --- wit_cat/File_Helper.js | 243 ++--------------------------------------- 1 file changed, 8 insertions(+), 235 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 01ac29c0..82f4e28c 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -24,12 +24,6 @@ class WitCatFileHelper { "WitCatFileHelper.encrypt": "base64加密[text]", "WitCatFileHelper.decrypt": "base64解密[text]", "WitCatFileHelper.openfile": "打开文件", - "WitCatFileHelper.createinput": "设置或创建ID为[id]的文本框的X[x]Y[y]宽[width]高[height]内容[text]提示[texts]字体大小[size]", - "WitCatFileHelper.deleteinput": "删除ID为[id]的文本框", - "WitCatFileHelper.getinput": "获得ID为[id]的文本框内容", - "WitCatFileHelper.isinput": "焦点是否在ID为[id]的文本框上", - "WitCatFileHelper.nowinput": "将焦点聚焦在ID为[id]的文本框上", - "WitCatFileHelper.deleteallinput": "删除所有文本框", "WitCatFileHelper.showvar": "设置键[name]的状态为[show]", "WitCatFileHelper.saveother": "设置作品ID[id]的键[name]为[text]", "WitCatFileHelper.uploadother": "获取作品[id]的键[name]的值", @@ -59,12 +53,6 @@ class WitCatFileHelper { "WitCatFileHelper.encrypt": "base64 encrypt[text]", "WitCatFileHelper.decrypt": "base64 decrypt[text]", "WitCatFileHelper.openfile": "openfile", - "WitCatFileHelper.createinput": "Set or create an input with ID[id]X[x]Y[y]width[width]height[height]content[text]prompt[texts]font-size[size]", - "WitCatFileHelper.deleteinput": "delete an input with ID[id]", - "WitCatFileHelper.getinput": "get an input with ID[id]", - "WitCatFileHelper.isinput": "is the focus on the input with ID[id]?", - "WitCatFileHelper.nowinput": "let teh focus on the input with ID[id]", - "WitCatFileHelper.deleteallinput": "delete all input", "WitCatFileHelper.showvar": "set value[name]`s state[show]", "WitCatFileHelper.saveother": "save ID[id]`s content name[name]with[text]", "WitCatFileHelper.uploadother": "get ID[id]`s value[name]", @@ -265,107 +253,6 @@ class WitCatFileHelper { }, }, }, - "---"+this.formatMessage("WitCatFileHelper.inputmanagement"), - { - opcode: "createinput", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.createinput"), - arguments: { - id:{ - type:"string", - defaultValue:"i", - }, - x:{ - type:"number", - defaultValue:"0", - }, - y:{ - type:"number", - defaultValue:"0", - }, - width:{ - type:"number", - defaultValue:"100", - }, - height:{ - type:"number", - defaultValue:"20", - }, - text:{ - type:"string", - defaultValue:"hello word!", - }, - texts:{ - type:"string", - defaultValue:"hello word!", - }, - size:{ - type:"number", - defaultValue:"16", - } - }, - }, - { - opcode: "compute", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.compute"), - arguments: { - size:{ - type:"number", - defaultValue:"16", - } - }, - }, - { - opcode: "deleteinput", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.deleteinput"), - arguments: { - id:{ - type:"string", - defaultValue:"i", - } - }, - }, - { - opcode: "getinput", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.getinput"), - arguments: { - id:{ - type:"string", - defaultValue:"i", - } - }, - }, - { - opcode: "isinput", - blockType: "Boolean", - text: this.formatMessage("WitCatFileHelper.isinput"), - arguments: { - id:{ - type:"string", - defaultValue:"i", - } - }, - }, - { - opcode: "nowinput", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.nowinput"), - arguments: { - id:{ - type:"string", - defaultValue:"i", - } - }, - }, - { - opcode: "deleteallinput", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.deleteallinput"), - arguments: {}, - }, "---"+this.formatMessage("WitCatFileHelper.Multiplelinestext"), { opcode: "segmentation", @@ -463,14 +350,14 @@ class WitCatFileHelper { }, ], thing: [ - { - text: this.formatMessage('WitCatFileHelper.thing.2'), - value: 'false' - }, { text: this.formatMessage('WitCatFileHelper.thing.1'), value: 'true' }, + { + text: this.formatMessage('WitCatFileHelper.thing.2'), + value: 'false' + }, ], } }; @@ -668,120 +555,6 @@ class WitCatFileHelper { } }); } - //设置或创建文本框 - createinput(args){ - let x = args.x; - let y = args.y; - let width = args.width; - let height = args.height; - if(args.x > this.runtime.stageWidth){ - x = this.runtime.stageWidth; - } - if(args.x < 0){ - x = 0; - } - if(args.y > this.runtime.stageHeight){ - y = this.runtime.stageHeight; - } - if(args.y < 0){ - y = 0; - } - if(args.x + args.width > this.runtime.stageWidth){ - width = this.runtime.stageWidth - x; - } - if(args.width < 0){ - width = 0; - } - if(args.y + args.height > this.runtime.stageHeight){ - height = this.runtime.stageHeight - y; - } - if(args.height < 0){ - height = 0; - } - x = (x / this.runtime.stageWidth) * 100; - y = (y / this.runtime.stageHeight) * 100; - width = (width / this.runtime.stageWidth) * 100; - height = (height / this.runtime.stageHeight) * 100; - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;`; - let search = document.getElementById("WitCatInput" + args.id); - //找渲染div - let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 - if(div == null){ - div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 - if(div == null){ - div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 - if(div == null){ - alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); - return; - } - } - } - if(search != null){ - search.style = dom; - search.value = args.text; - search.placeholder = args.texts; - } - else{ - let eleLink = document.createElement('input'); - eleLink.type = "text"; - eleLink.style = dom; - eleLink.id = "WitCatInput" + args.id; - eleLink.value = args.text; - eleLink.className = "WitCatInput"; - eleLink.placeholder = args.texts; - div.appendChild(eleLink); - } - } - //删除文本框 - deleteinput(args){ - let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ - let div = search.parentNode; - div.removeChild(search); - } - } - //获取文本框内容 - getinput(args){ - let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ - return(search.value); - } - else{ - return(""); - } - } - //焦点判断 - isinput(args){ - let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ - if(search == document.activeElement){ - return(true); - } - else{ - return(false); - } - } - else{ - return(false); - } - } - //焦点获取 - nowinput(args){ - let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ - search.focus(); - } - } - //删除所有文本框 - deleteallinput(args){ - let search = document.getElementsByClassName("WitCatInput"); - let i = 0; - let div; - for(i = search.length - 1;i >= 0;i--){ - div = search[i].parentNode; - div.removeChild(search[i]); - } - } //设置状态 showvar(args){ const name = args.name; @@ -1005,12 +778,12 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatFileHelper.name": "wit_catの拓展", - "WitCatFileHelper.descp": "一堆没用的小玩意" + "WitCatFileHelper.name": "文件助手", + "WitCatFileHelper.descp": "读取/处理本地数据" }, en: { - "WitCatFileHelper.name": "wit_catのextension", - "WitCatFileHelper.descp": "some useless block" + "WitCatFileHelper.name": "File Helper", + "WitCatFileHelper.descp": "Handling local data" } } }; From 9f833909ef1ad6639c0d95fa4796ef5ff080805f Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 22 Nov 2022 13:32:38 +0800 Subject: [PATCH 180/357] Add files via upload --- wit_cat/input.js | 314 ++++++++++++++++++++++++++++++++++++++++++ wit_cat/more mouse.js | 184 +++++++++++++++++++++++++ 2 files changed, 498 insertions(+) create mode 100644 wit_cat/input.js create mode 100644 wit_cat/more mouse.js diff --git a/wit_cat/input.js b/wit_cat/input.js new file mode 100644 index 00000000..1da69b34 --- /dev/null +++ b/wit_cat/input.js @@ -0,0 +1,314 @@ +// import Cast from "cast.js"; + +const _picture = ""; + +const _icon = ""; + +const extensionId = "WitCatInput"; + +class WitCatInput { + constructor(runtime) { + this.runtime = runtime; + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "WitCatInput.name": "文本输入框", + "WitCatInput.createinput": "设置或创建ID为[id]的文本框的X[x]Y[y]宽[width]高[height]内容[text]提示[texts]字体大小[size]", + "WitCatInput.deleteinput": "删除ID为[id]的文本框", + "WitCatInput.getinput": "获得ID为[id]的文本框内容", + "WitCatInput.isinput": "焦点是否在ID为[id]的文本框上", + "WitCatInput.nowinput": "将焦点聚焦在ID为[id]的文本框上", + "WitCatInput.deleteallinput": "删除所有文本框", + "WitCatInput.compute": "当前分辨率下高[size]的字体大小", + }, + en: { + "WitCatInput.name": "input", + "WitCatInput.createinput": "Set or create an input with ID[id]X[x]Y[y]width[width]height[height]content[text]prompt[texts]font-size[size]", + "WitCatInput.deleteinput": "delete an input with ID[id]", + "WitCatInput.getinput": "get an input with ID[id]", + "WitCatInput.isinput": "is the focus on the input with ID[id]?", + "WitCatInput.nowinput": "let teh focus on the input with ID[id]", + "WitCatInput.deleteallinput": "delete all input", + "WitCatInput.compute": "The font size at the current resolution that is high [size]", + } + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: extensionId, // 拓展id + name: this.formatMessage("WitCatInput.name"), // 拓展名 + docsURI: "https://www.ccw.site/post/d6d96e80-3f58-4a19-b7e6-c567d3a6a583", + blockIconURI: _icon, + menuIconURI: _icon, + color1: "#52baba", + color2: "#ffffff", + blocks: [ + { + opcode: "createinput", + blockType: "command", + text: this.formatMessage("WitCatInput.createinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + }, + x:{ + type:"number", + defaultValue:"0", + }, + y:{ + type:"number", + defaultValue:"0", + }, + width:{ + type:"number", + defaultValue:"100", + }, + height:{ + type:"number", + defaultValue:"20", + }, + text:{ + type:"string", + defaultValue:"hello word!", + }, + texts:{ + type:"string", + defaultValue:"hello word!", + }, + size:{ + type:"number", + defaultValue:"16", + } + }, + }, + { + opcode: "compute", + blockType: "reporter", + text: this.formatMessage("WitCatInput.compute"), + arguments: { + size:{ + type:"number", + defaultValue:"16", + } + }, + }, + { + opcode: "deleteinput", + blockType: "command", + text: this.formatMessage("WitCatInput.deleteinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + } + }, + }, + { + opcode: "getinput", + blockType: "reporter", + text: this.formatMessage("WitCatInput.getinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + } + }, + }, + { + opcode: "isinput", + blockType: "Boolean", + text: this.formatMessage("WitCatInput.isinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + } + }, + }, + { + opcode: "nowinput", + blockType: "command", + text: this.formatMessage("WitCatInput.nowinput"), + arguments: { + id:{ + type:"string", + defaultValue:"i", + } + }, + }, + { + opcode: "deleteallinput", + blockType: "command", + text: this.formatMessage("WitCatInput.deleteallinput"), + arguments: {}, + }, + ] + }; + } + //设置或创建文本框 + createinput(args){ + let x = args.x; + let y = args.y; + let width = args.width; + let height = args.height; + if(args.x > this.runtime.stageWidth){ + x = this.runtime.stageWidth; + } + if(args.x < 0){ + x = 0; + } + if(args.y > this.runtime.stageHeight){ + y = this.runtime.stageHeight; + } + if(args.y < 0){ + y = 0; + } + if(args.x + args.width > this.runtime.stageWidth){ + width = this.runtime.stageWidth - x; + } + if(args.width < 0){ + width = 0; + } + if(args.y + args.height > this.runtime.stageHeight){ + height = this.runtime.stageHeight - y; + } + if(args.height < 0){ + height = 0; + } + x = (x / this.runtime.stageWidth) * 100; + y = (y / this.runtime.stageHeight) * 100; + width = (width / this.runtime.stageWidth) * 100; + height = (height / this.runtime.stageHeight) * 100; + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;`; + let search = document.getElementById("WitCatInput" + args.id); + //找渲染div + let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 + if(div == null){ + div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 + if(div == null){ + div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 + if(div == null){ + alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); + return; + } + } + } + if(search != null){ + search.style = dom; + search.value = args.text; + search.placeholder = args.texts; + } + else{ + let eleLink = document.createElement('input'); + eleLink.type = "text"; + eleLink.style = dom; + eleLink.id = "WitCatInput" + args.id; + eleLink.value = args.text; + eleLink.className = "WitCatInput"; + eleLink.placeholder = args.texts; + div.appendChild(eleLink); + } + } + //删除文本框 + deleteinput(args){ + let search = document.getElementById("WitCatInput" + args.id); + if(search != null){ + let div = search.parentNode; + div.removeChild(search); + } + } + //获取文本框内容 + getinput(args){ + let search = document.getElementById("WitCatInput" + args.id); + if(search != null){ + return(search.value); + } + else{ + return(""); + } + } + //焦点判断 + isinput(args){ + let search = document.getElementById("WitCatInput" + args.id); + if(search != null){ + if(search == document.activeElement){ + return(true); + } + else{ + return(false); + } + } + else{ + return(false); + } + } + //焦点获取 + nowinput(args){ + let search = document.getElementById("WitCatInput" + args.id); + if(search != null){ + search.focus(); + } + } + //删除所有文本框 + deleteallinput(args){ + let search = document.getElementsByClassName("WitCatInput"); + let i = 0; + let div; + for(i = search.length - 1;i >= 0;i--){ + div = search[i].parentNode; + div.removeChild(search[i]); + } + } + //计算坐标 + compute(args){ + //找渲染div + let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 + if(div == null){ + div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 + if(div == null){ + div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 + if(div == null){ + alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); + return; + } + } + } + console.log(div.style.width) + return (div.style.width.split("px")[0] / 360) * args.size; + } +} + +window.tempExt = { + Extension: WitCatInput, + info: { + name: "WitCatInput.name", + description: "WitCatInput.descp", + extensionId: extensionId, + iconURL: _picture, + insetIconURL: _icon, + featured: true, + disabled: false, + collaborator: "白猫 @ CCW" + }, + l10n: { + "zh-cn": { + "WitCatInput.name": "文本框", + "WitCatInput.descp": "全新的输入框!" + }, + en: { + "WitCatInput.name": "input", + "WitCatInput.descp": "what a nice input!" + } + } +}; + +/* vim: set expandtab tabstop=2 shiftwidth=2: */ diff --git a/wit_cat/more mouse.js b/wit_cat/more mouse.js new file mode 100644 index 00000000..dc0b7a3a --- /dev/null +++ b/wit_cat/more mouse.js @@ -0,0 +1,184 @@ +// import Cast from "cast.js"; + +const _picture = ""; + +const _icon = ""; + +const extensionId = "WitCatMouse"; +let button = ["up","up","up","up","up"]; + +class WitCatMouse { + constructor(runtime) { + this.runtime = runtime; + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "WitCatMouse.name": "高级鼠标", + "WitCatMouse.set": "[set]右键菜单", + "WitCatMouse.set.1": "启用", + "WitCatMouse.set.2": "禁用", + "WitCatMouse.when": "鼠标[key]被按下", + "WitCatMouse.key.1": "左键", + "WitCatMouse.key.2": "中键", + "WitCatMouse.key.3": "右键", + "WitCatMouse.key.4": "前侧键", + "WitCatMouse.key.5": "后侧键", + }, + en: { + "WitCatMouse.name": "More Mouse", + "WitCatMouse.set": "[set]Right-click menu", + "WitCatMouse.set.1": "allow", + "WitCatMouse.set.2": "forbid", + "WitCatMouse.when": "mouse[key]down?", + "WitCatMouse.key.1": "left", + "WitCatMouse.key.2": "middle", + "WitCatMouse.key.3": "right", + "WitCatMouse.key.4": "front", + "WitCatMouse.key.5": "back", + } + }) + } + + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: extensionId, // 拓展id + name: this.formatMessage("WitCatMouse.name"), // 拓展名 + docsURI: "https://www.ccw.site/post/d6d96e80-3f58-4a19-b7e6-c567d3a6a583", + blockIconURI: _icon, + menuIconURI: _icon, + color1: "#8eace1", + color2: "#ffffff", + blocks: [ + { + opcode: 'set', + blockType: "command", + text: this.formatMessage("WitCatMouse.set"), + arguments: { + set:{ + type:"bool", + menu:"set", + }, + } + }, + { + opcode: "when", + blockType: "Boolean", + text: this.formatMessage("WitCatMouse.when"), + arguments: { + key:{ + type:"string", + menu:"key", + }, + }, + }, + ], + menus:{ + key: [ + { + text: this.formatMessage('WitCatMouse.key.1'), + value: '0' + }, + { + text: this.formatMessage('WitCatMouse.key.2'), + value: '1' + }, + { + text: this.formatMessage('WitCatMouse.key.3'), + value: '2' + }, + { + text: this.formatMessage('WitCatMouse.key.4'), + value: '3' + }, + { + text: this.formatMessage('WitCatMouse.key.5'), + value: '4' + }, + ], + set: [ + { + text: this.formatMessage('WitCatMouse.set.1'), + value: "true" + }, + { + text: this.formatMessage('WitCatMouse.set.2'), + value: "false" + }, + ], + } + }; + } + //右键菜单 + set(args){ + //找渲染div + let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 + if(div == null){ + div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 + if(div == null){ + div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 + if(div == null){ + alert("当前页面不支持禁用右键菜单,请前往作品详情页体验完整作品!"); + return; + } + } + } + history.pushState(null, null, null); + div.oncontextmenu = function (){ + if(args.set == "true"){ + return true; + } + else{ + return false; + } + } + } + //按下判断 + when(args){ + if(button[args.key] == "down"){ + return true; + } + else{ + return false; + } + } +} + +window.tempExt = { + Extension: WitCatMouse, + info: { + name: "WitCatMouse.name", + description: "WitCatMouse.descp", + extensionId: extensionId, + iconURL: _picture, + insetIconURL: _icon, + featured: true, + disabled: false, + collaborator: "白猫 @ CCW" + }, + l10n: { + "zh-cn": { + "WitCatMouse.name": "高级鼠标", + "WitCatMouse.descp": "更精准的控制鼠标!" + }, + en: { + "WitCatMouse.name": "More Mouse", + "WitCatMouse.descp": "More precise mouse control!" + } + } +}; + +/* vim: set expandtab tabstop=2 shiftwidth=2: */ + +document.onmousedown = function(event) { + button[event.button] = "down"; +} +document.onmouseup = function(event) { + button[event.button] = "up"; +} \ No newline at end of file From 2f97415f7750fe65fc4f5b693638ded38c6e8f4a Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sat, 26 Nov 2022 10:49:28 +0800 Subject: [PATCH 181/357] Create input.js --- wit_cat/input.js | 58 ++++++++++++++++++++++++++++++++++++++++-------- 1 file changed, 49 insertions(+), 9 deletions(-) diff --git a/wit_cat/input.js b/wit_cat/input.js index 1da69b34..127c7595 100644 --- a/wit_cat/input.js +++ b/wit_cat/input.js @@ -12,23 +12,27 @@ class WitCatInput { this._formatMessage = runtime.getFormatMessage({ "zh-cn": { "WitCatInput.name": "文本输入框", - "WitCatInput.createinput": "设置或创建ID为[id]的文本框的X[x]Y[y]宽[width]高[height]内容[text]提示[texts]字体大小[size]", + "WitCatInput.createinput": "设置或创建ID为[id]的[type]文本框的X[x]Y[y]宽[width]高[height]内容[text]提示[texts]字体大小[size]", "WitCatInput.deleteinput": "删除ID为[id]的文本框", "WitCatInput.getinput": "获得ID为[id]的文本框内容", "WitCatInput.isinput": "焦点是否在ID为[id]的文本框上", "WitCatInput.nowinput": "将焦点聚焦在ID为[id]的文本框上", "WitCatInput.deleteallinput": "删除所有文本框", "WitCatInput.compute": "当前分辨率下高[size]的字体大小", + "WitCatInput.type.1": "单行文本", + "WitCatInput.type.2": "多行文本", }, en: { "WitCatInput.name": "input", - "WitCatInput.createinput": "Set or create an input with ID[id]X[x]Y[y]width[width]height[height]content[text]prompt[texts]font-size[size]", + "WitCatInput.createinput": "Set or create an[type]input with ID[id]X[x]Y[y]width[width]height[height]content[text]prompt[texts]font-size[size]", "WitCatInput.deleteinput": "delete an input with ID[id]", "WitCatInput.getinput": "get an input with ID[id]", "WitCatInput.isinput": "is the focus on the input with ID[id]?", "WitCatInput.nowinput": "let teh focus on the input with ID[id]", "WitCatInput.deleteallinput": "delete all input", "WitCatInput.compute": "The font size at the current resolution that is high [size]", + "WitCatInput.type.1": "Single line", + "WitCatInput.type.2": "Multiple lines", } }) } @@ -60,6 +64,10 @@ class WitCatInput { type:"string", defaultValue:"i", }, + type:{ + type:"string", + menu:"type", + }, x:{ type:"number", defaultValue:"0", @@ -151,7 +159,19 @@ class WitCatInput { text: this.formatMessage("WitCatInput.deleteallinput"), arguments: {}, }, - ] + ], + menus:{ + type: [ + { + text: this.formatMessage('WitCatInput.type.1'), + value: 'input' + }, + { + text: this.formatMessage('WitCatInput.type.2'), + value: 'Textarea' + }, + ], + } }; } //设置或创建文本框 @@ -188,7 +208,7 @@ class WitCatInput { y = (y / this.runtime.stageHeight) * 100; width = (width / this.runtime.stageWidth) * 100; height = (height / this.runtime.stageHeight) * 100; - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;`; + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none`; let search = document.getElementById("WitCatInput" + args.id); //找渲染div let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 @@ -203,17 +223,37 @@ class WitCatInput { } } if(search != null){ - search.style = dom; - search.value = args.text; - search.placeholder = args.texts; + if(search.name == args.type){ + search.style = dom; + search.value = args.text; + search.placeholder = args.texts; + } + else{ + let div = search.parentNode; + div.removeChild(search); + let eleLink = document.createElement(args.type); + if(args.type == "input"){ + eleLink.type = "text"; + } + eleLink.style = dom; + eleLink.id = "WitCatInput" + args.id; + eleLink.value = args.text; + eleLink.className = "WitCatInput"; + eleLink.name = args.type; + eleLink.placeholder = args.texts; + div.appendChild(eleLink); + } } else{ - let eleLink = document.createElement('input'); - eleLink.type = "text"; + let eleLink = document.createElement(args.type); + if(args.type == "input"){ + eleLink.type = "text"; + } eleLink.style = dom; eleLink.id = "WitCatInput" + args.id; eleLink.value = args.text; eleLink.className = "WitCatInput"; + eleLink.name = args.type; eleLink.placeholder = args.texts; div.appendChild(eleLink); } From 2ba51866be9e1263c90068b50053d482d65a0ef0 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sat, 26 Nov 2022 10:50:18 +0800 Subject: [PATCH 182/357] Update more mouse.js --- wit_cat/more mouse.js | 103 +++++++++++++++++++++++++++++++++++++++++- 1 file changed, 102 insertions(+), 1 deletion(-) diff --git a/wit_cat/more mouse.js b/wit_cat/more mouse.js index dc0b7a3a..cd98e3df 100644 --- a/wit_cat/more mouse.js +++ b/wit_cat/more mouse.js @@ -6,6 +6,10 @@ const _icon = " const extensionId = "WitCatMouse"; let button = ["up","up","up","up","up"]; +let xMouse = 0; +let yMouse = 0; +let isMove = false,timer = null; + class WitCatMouse { constructor(runtime) { @@ -22,6 +26,12 @@ class WitCatMouse { "WitCatMouse.key.3": "右键", "WitCatMouse.key.4": "前侧键", "WitCatMouse.key.5": "后侧键", + "WitCatMouse.mouseuse": "[type]鼠标", + "WitCatMouse.type.1": "锁定", + "WitCatMouse.type.2": "释放", + "WitCatMouse.acceleration": "鼠标[way]加速度", + "WitCatMouse.way.1": "X", + "WitCatMouse.way.2": "Y", }, en: { "WitCatMouse.name": "More Mouse", @@ -34,6 +44,12 @@ class WitCatMouse { "WitCatMouse.key.3": "right", "WitCatMouse.key.4": "front", "WitCatMouse.key.5": "back", + "WitCatMouse.mouseuse": "[type]mouse", + "WitCatMouse.type.1": "lock", + "WitCatMouse.type.2": "release", + "WitCatMouse.acceleration": "mouse[way]acceleration", + "WitCatMouse.way.1": "X", + "WitCatMouse.way.2": "Y", } }) } @@ -78,6 +94,28 @@ class WitCatMouse { }, }, }, + { + opcode: "mouseuse", + blockType: "command", + text: this.formatMessage("WitCatMouse.mouseuse"), + arguments: { + type:{ + type:"string", + menu:"type", + }, + }, + }, + { + opcode: "acceleration", + blockType: "reporter", + text: this.formatMessage("WitCatMouse.acceleration"), + arguments: { + way:{ + type:"string", + menu:"way", + }, + }, + }, ], menus:{ key: [ @@ -112,6 +150,26 @@ class WitCatMouse { value: "false" }, ], + type: [ + { + text: this.formatMessage('WitCatMouse.type.1'), + value: "false" + }, + { + text: this.formatMessage('WitCatMouse.type.2'), + value: "true" + }, + ], + way: [ + { + text: this.formatMessage('WitCatMouse.way.1'), + value: "x" + }, + { + text: this.formatMessage('WitCatMouse.way.2'), + value: "y" + }, + ], } }; } @@ -148,6 +206,38 @@ class WitCatMouse { return false; } } + //控制鼠标 + mouseuse(args){ + //找渲染div + let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 + if(div == null){ + div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 + if(div == null){ + div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 + if(div == null){ + alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); + return; + } + } + } + if(args.type){ + div.addEventListener('click', function () { + div.requestPointerLock(); + }); + } + else{ + document.exitPointerLock(); + } + } + //鼠标移动量 + acceleration(args){ + if(args.way == "x"){ + return xMouse; + } + else{ + return -yMouse; + } + } } window.tempExt = { @@ -181,4 +271,15 @@ document.onmousedown = function(event) { } document.onmouseup = function(event) { button[event.button] = "up"; -} \ No newline at end of file +} +document.addEventListener("mousemove", ev => { + xMouse = ev.movementX; // 获得鼠标指针的x移动量 + yMouse = ev.movementY; // 获得鼠标指针的y移动量 + isMove = true; + clearTimeout(timer); + timer = setTimeout(function(){ + isMove = false; + xMouse = 0; + yMouse = 0; + },30); +}); From bd071049fbea7db83696d706d029b67a73626953 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sat, 26 Nov 2022 10:56:30 +0800 Subject: [PATCH 183/357] Update more mouse.js --- wit_cat/more mouse.js | 34 ++++++---------------------------- 1 file changed, 6 insertions(+), 28 deletions(-) diff --git a/wit_cat/more mouse.js b/wit_cat/more mouse.js index cd98e3df..ee02a591 100644 --- a/wit_cat/more mouse.js +++ b/wit_cat/more mouse.js @@ -26,9 +26,7 @@ class WitCatMouse { "WitCatMouse.key.3": "右键", "WitCatMouse.key.4": "前侧键", "WitCatMouse.key.5": "后侧键", - "WitCatMouse.mouseuse": "[type]鼠标", - "WitCatMouse.type.1": "锁定", - "WitCatMouse.type.2": "释放", + "WitCatMouse.mouseuse": "锁定鼠标", "WitCatMouse.acceleration": "鼠标[way]加速度", "WitCatMouse.way.1": "X", "WitCatMouse.way.2": "Y", @@ -98,12 +96,7 @@ class WitCatMouse { opcode: "mouseuse", blockType: "command", text: this.formatMessage("WitCatMouse.mouseuse"), - arguments: { - type:{ - type:"string", - menu:"type", - }, - }, + arguments: {}, }, { opcode: "acceleration", @@ -150,16 +143,6 @@ class WitCatMouse { value: "false" }, ], - type: [ - { - text: this.formatMessage('WitCatMouse.type.1'), - value: "false" - }, - { - text: this.formatMessage('WitCatMouse.type.2'), - value: "true" - }, - ], way: [ { text: this.formatMessage('WitCatMouse.way.1'), @@ -207,7 +190,7 @@ class WitCatMouse { } } //控制鼠标 - mouseuse(args){ + mouseuse(){ //找渲染div let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 if(div == null){ @@ -220,14 +203,9 @@ class WitCatMouse { } } } - if(args.type){ - div.addEventListener('click', function () { - div.requestPointerLock(); - }); - } - else{ - document.exitPointerLock(); - } + div.addEventListener('click', function () { + div.requestPointerLock(); + }); } //鼠标移动量 acceleration(args){ From 378e843abbce303735914cb4f0a224fe527a1a4a Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sat, 26 Nov 2022 11:42:53 +0800 Subject: [PATCH 184/357] Update more mouse.js --- wit_cat/more mouse.js | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/wit_cat/more mouse.js b/wit_cat/more mouse.js index ee02a591..c8bcbcf8 100644 --- a/wit_cat/more mouse.js +++ b/wit_cat/more mouse.js @@ -42,9 +42,7 @@ class WitCatMouse { "WitCatMouse.key.3": "right", "WitCatMouse.key.4": "front", "WitCatMouse.key.5": "back", - "WitCatMouse.mouseuse": "[type]mouse", - "WitCatMouse.type.1": "lock", - "WitCatMouse.type.2": "release", + "WitCatMouse.mouseuse": "lock mouse", "WitCatMouse.acceleration": "mouse[way]acceleration", "WitCatMouse.way.1": "X", "WitCatMouse.way.2": "Y", @@ -203,9 +201,7 @@ class WitCatMouse { } } } - div.addEventListener('click', function () { - div.requestPointerLock(); - }); + div.requestPointerLock(); } //鼠标移动量 acceleration(args){ From 9ba91a853069e341f7210d5e28870f558c0a08b5 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 28 Nov 2022 18:07:39 +0800 Subject: [PATCH 185/357] Update File_Helper.js --- wit_cat/File_Helper.js | 23 +++++++++++++++++------ 1 file changed, 17 insertions(+), 6 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 82f4e28c..67d2cb94 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -5,6 +5,7 @@ const _picture = " const _icon = ""; const extensionId = "WitCatFileHelper"; +let FLAG = 0; class WitCatFileHelper { constructor(runtime) { @@ -530,28 +531,37 @@ class WitCatFileHelper { } //打开文件 openfile(args){ + FLAG = 1; + console.log("aaa"); return new Promise(resolve => { let a = 0; const input = document.createElement("input"); input.type = "file"; + input.style = "display:none;"; input.click(); input.onchange = () => { const reader = new FileReader(); const file = input.files[0]; reader.onload = (e) => { + FLAG = 0; resolve(e.target.result); }; reader.onerror = (e) => { + FLAG = 0; resolve(); }; reader.readAsText(file); } - // 当点击取消 - input.onfocus = function () { - a++; - if (a == 2) { - resolve(); - } + //取消检测 + window.onfocus = function() { + console.log("az") + setTimeout(e => { + if (FLAG == 1) { + FLAG = 0; + console.log('取消'); + resolve(""); + } + }, 1000); } }); } @@ -789,3 +799,4 @@ window.tempExt = { }; /* vim: set expandtab tabstop=2 shiftwidth=2: */ + From 225708d00047f929f23d28a33ea874145e1dfd7e Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 28 Nov 2022 18:09:31 +0800 Subject: [PATCH 186/357] Update File_Helper.js --- wit_cat/File_Helper.js | 3 --- 1 file changed, 3 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 67d2cb94..d9ef0aa1 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -532,7 +532,6 @@ class WitCatFileHelper { //打开文件 openfile(args){ FLAG = 1; - console.log("aaa"); return new Promise(resolve => { let a = 0; const input = document.createElement("input"); @@ -554,11 +553,9 @@ class WitCatFileHelper { } //取消检测 window.onfocus = function() { - console.log("az") setTimeout(e => { if (FLAG == 1) { FLAG = 0; - console.log('取消'); resolve(""); } }, 1000); From 2009501cfa3f8b398baf44b77e0a28a5015f848b Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Wed, 30 Nov 2022 13:41:11 +0800 Subject: [PATCH 187/357] Rename more mouse.js to More_Mouse.js --- wit_cat/{more mouse.js => More_Mouse.js} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename wit_cat/{more mouse.js => More_Mouse.js} (100%) diff --git a/wit_cat/more mouse.js b/wit_cat/More_Mouse.js similarity index 100% rename from wit_cat/more mouse.js rename to wit_cat/More_Mouse.js From 94eacc08172e5683cfd429402a49873cce9a1145 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Wed, 30 Nov 2022 13:41:39 +0800 Subject: [PATCH 188/357] Rename input.js to Input.js --- wit_cat/{input.js => Input.js} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename wit_cat/{input.js => Input.js} (100%) diff --git a/wit_cat/input.js b/wit_cat/Input.js similarity index 100% rename from wit_cat/input.js rename to wit_cat/Input.js From cb8aaedea78fb864978a2aebd5210a1e5eff8345 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 13 Dec 2022 12:02:55 +0800 Subject: [PATCH 189/357] Update File_Helper.js --- wit_cat/File_Helper.js | 27 ++++----------------------- 1 file changed, 4 insertions(+), 23 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index d9ef0aa1..a97b3fc4 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -12,7 +12,7 @@ class WitCatFileHelper { this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "WitCatFileHelper.name": "wit_catの拓展", + "WitCatFileHelper.name": "文件助手", "WitCatFileHelper.filehelper": "文件处理", "WitCatFileHelper.inputmanagement": "文本框", "WitCatFileHelper.Multiplelinestext": "多行文本", @@ -38,11 +38,10 @@ class WitCatFileHelper { "WitCatFileHelper.numMultiplelinestext": "[text]的行数", "WitCatFileHelper.thing.1": "数组", "WitCatFileHelper.thing.2": "多行文本", - "WitCatFileHelper.compute": "当前分辨率下高[size]的字体大小", }, en: { - "WitCatFileHelper.name": "wit_catのextension", - "WitCatFileHelper.filehelper": "file helper", + "WitCatFileHelper.name": "file helper", + "WitCatFileHelper.filehelper": "file", "WitCatFileHelper.inputmanagement": "input", "WitCatFileHelper.Multiplelinestext": "Multiple lines of text", "WitCatFileHelper.downloads": "Download split content [text] by [s] named [name]", @@ -67,7 +66,6 @@ class WitCatFileHelper { "WitCatFileHelper.numMultiplelinestext": "[text]Number of rows", "WitCatFileHelper.thing.1": "array", "WitCatFileHelper.thing.2": "Multiple lines of text", - "WitCatFileHelper.compute": "The font size at the current resolution that is high [size]", } }) } @@ -551,8 +549,8 @@ class WitCatFileHelper { }; reader.readAsText(file); } - //取消检测 window.onfocus = function() { + // 开始计时或者播放 setTimeout(e => { if (FLAG == 1) { FLAG = 0; @@ -752,23 +750,6 @@ class WitCatFileHelper { let text = args.text.split("\n"); return text.length; } - //计算坐标 - compute(args){ - //找渲染div - let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 - if(div == null){ - div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 - if(div == null){ - div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 - if(div == null){ - alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); - return; - } - } - } - console.log(div.style.width) - return (div.style.width.split("px")[0] / 360) * args.size; - } } window.tempExt = { From a99be24ce0af864cb3dfad9330f745121d3d1a8d Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 13 Dec 2022 12:11:01 +0800 Subject: [PATCH 190/357] Update More_Mouse.js --- wit_cat/More_Mouse.js | 110 ++++++++++++++++++++++++++++++------------ 1 file changed, 80 insertions(+), 30 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index c8bcbcf8..725e5950 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -11,6 +11,12 @@ let yMouse = 0; let isMove = false,timer = null; +//找渲染cvs +let cvs = document.getElementsByTagName("canvas")[0]; +if(cvs == null){ + alert("当前页面不支持多指触控,请前往作品详情页体验完整作品!"); +} + class WitCatMouse { constructor(runtime) { this.runtime = runtime; @@ -30,6 +36,10 @@ class WitCatMouse { "WitCatMouse.acceleration": "鼠标[way]加速度", "WitCatMouse.way.1": "X", "WitCatMouse.way.2": "Y", + "WitCatTouch.down": "按下的手指数量", + "WitCatTouch.num": "第[num]个手指的[type]", + "WitCatTouch.type.1": "X", + "WitCatTouch.type.2": "Y", }, en: { "WitCatMouse.name": "More Mouse", @@ -46,6 +56,11 @@ class WitCatMouse { "WitCatMouse.acceleration": "mouse[way]acceleration", "WitCatMouse.way.1": "X", "WitCatMouse.way.2": "Y", + "WitCatTouch.name": "Touch", + "WitCatTouch.down": "Fingers Number", + "WitCatTouch.num": "the[num]finger`s[type]", + "WitCatTouch.type.1": "X", + "WitCatTouch.type.2": "Y", } }) } @@ -107,6 +122,27 @@ class WitCatMouse { }, }, }, + { + opcode: 'down', + blockType: "reporter", + text: this.formatMessage("WitCatTouch.down"), + arguments: {} + }, + { + opcode: "num", + blockType: "reporter", + text: this.formatMessage("WitCatTouch.num"), + arguments: { + num:{ + type:"number", + defaultValue:"1", + }, + type:{ + type:"string", + menu:"type", + }, + }, + }, ], menus:{ key: [ @@ -151,25 +187,23 @@ class WitCatMouse { value: "y" }, ], + type: [ + { + text: this.formatMessage('WitCatTouch.type.1'), + value: "x" + }, + { + text: this.formatMessage('WitCatTouch.type.2'), + value: "y" + }, + ], } }; } //右键菜单 set(args){ - //找渲染div - let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 - if(div == null){ - div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 - if(div == null){ - div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 - if(div == null){ - alert("当前页面不支持禁用右键菜单,请前往作品详情页体验完整作品!"); - return; - } - } - } history.pushState(null, null, null); - div.oncontextmenu = function (){ + cvs.parentNode.oncontextmenu = function (){ if(args.set == "true"){ return true; } @@ -189,19 +223,7 @@ class WitCatMouse { } //控制鼠标 mouseuse(){ - //找渲染div - let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 - if(div == null){ - div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 - if(div == null){ - div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 - if(div == null){ - alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); - return; - } - } - } - div.requestPointerLock(); + cvs.parentNode.requestPointerLock(); } //鼠标移动量 acceleration(args){ @@ -212,6 +234,25 @@ class WitCatMouse { return -yMouse; } } + //数量 + down(){ + return touch.length; + } + //坐标 + num(args){ + if(args.num > 0 && args.num <= touch.length){ + if(args.type == "x"){ + return this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.style.width.split("px")[0]); + } + else{ + console.log(cvs.style.height); + return this.runtime.stageHeight * ((touch[args.num - 1].clientY - cvs.getBoundingClientRect().top) / cvs.style.height.split("px")[0]); + } + } + else{ + return null; + } + } } window.tempExt = { @@ -228,12 +269,12 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatMouse.name": "高级鼠标", - "WitCatMouse.descp": "更精准的控制鼠标!" + "WitCatMouse.name": "高级鼠标/触控", + "WitCatMouse.descp": "更精准的控制鼠标/触屏!" }, en: { - "WitCatMouse.name": "More Mouse", - "WitCatMouse.descp": "More precise mouse control!" + "WitCatMouse.name": "More Mouse/Touch", + "WitCatMouse.descp": "More precise mouse/touch control!" } } }; @@ -257,3 +298,12 @@ document.addEventListener("mousemove", ev => { yMouse = 0; },30); }); +cvs.addEventListener('touchstart',function(e){ + touch = e.targetTouches; +}) +cvs.addEventListener('touchmove',function(e){ + touch = e.targetTouches; +}) +cvs.addEventListener('touchend',function(e){ + touch = e.targetTouches; +}) From faa148aef5d2b53da98d5933a09a51dabb89b8a0 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 13 Dec 2022 12:12:35 +0800 Subject: [PATCH 191/357] Update Input.js --- wit_cat/Input.js | 51 +++++++++++++----------------------------------- 1 file changed, 14 insertions(+), 37 deletions(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 127c7595..941a85fb 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -6,6 +6,12 @@ const _icon = " const extensionId = "WitCatInput"; +//找渲染cvs +let cvs = document.getElementsByTagName("canvas")[0]; +if(cvs == null){ + alert("当前页面不支持多指触控,请前往作品详情页体验完整作品!"); +} + class WitCatInput { constructor(runtime) { this.runtime = runtime; @@ -86,11 +92,11 @@ class WitCatInput { }, text:{ type:"string", - defaultValue:"hello word!", + defaultValue:"hello world!", }, texts:{ type:"string", - defaultValue:"hello word!", + defaultValue:"hello world!", }, size:{ type:"number", @@ -210,18 +216,6 @@ class WitCatInput { height = (height / this.runtime.stageHeight) * 100; let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none`; let search = document.getElementById("WitCatInput" + args.id); - //找渲染div - let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 - if(div == null){ - div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 - if(div == null){ - div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 - if(div == null){ - alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); - return; - } - } - } if(search != null){ if(search.name == args.type){ search.style = dom; @@ -229,8 +223,7 @@ class WitCatInput { search.placeholder = args.texts; } else{ - let div = search.parentNode; - div.removeChild(search); + cvs.parentNode.removeChild(search); let eleLink = document.createElement(args.type); if(args.type == "input"){ eleLink.type = "text"; @@ -241,7 +234,7 @@ class WitCatInput { eleLink.className = "WitCatInput"; eleLink.name = args.type; eleLink.placeholder = args.texts; - div.appendChild(eleLink); + cvs.parentNode.appendChild(eleLink); } } else{ @@ -255,15 +248,14 @@ class WitCatInput { eleLink.className = "WitCatInput"; eleLink.name = args.type; eleLink.placeholder = args.texts; - div.appendChild(eleLink); + cvs.parentNode.appendChild(eleLink); } } //删除文本框 deleteinput(args){ let search = document.getElementById("WitCatInput" + args.id); if(search != null){ - let div = search.parentNode; - div.removeChild(search); + cvs.parentNode.removeChild(search); } } //获取文本框内容 @@ -302,28 +294,13 @@ class WitCatInput { deleteallinput(args){ let search = document.getElementsByClassName("WitCatInput"); let i = 0; - let div; for(i = search.length - 1;i >= 0;i--){ - div = search[i].parentNode; - div.removeChild(search[i]); + search[i].parentNode.removeChild(search[i]); } } //计算坐标 compute(args){ - //找渲染div - let div = document.getElementsByClassName("gandi_stage_stage_1fD7k ccw-stage-wrapper")[0]; //gandi编辑器 - if(div == null){ - div = document.getElementsByClassName("stage_stage_1fD7k ccw-stage-wrapper")[0]; //传统编辑器 - if(div == null){ - div = document.getElementsByClassName("gandi_stage-wrapper_stage-canvas-wrapper_3ewmd")[0]; //作品展示页 - if(div == null){ - alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); - return; - } - } - } - console.log(div.style.width) - return (div.style.width.split("px")[0] / 360) * args.size; + return (cvs.style.width.split("px")[0] / 360) * args.size; } } From f860b0fe5ac93e15858e149a9e28601c87547e6a Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 13 Dec 2022 12:53:21 +0800 Subject: [PATCH 192/357] Update More_Mouse.js --- wit_cat/More_Mouse.js | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 725e5950..64a4be6f 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -9,6 +9,7 @@ let button = ["up","up","up","up","up"]; let xMouse = 0; let yMouse = 0; let isMove = false,timer = null; +let touch = []; //找渲染cvs @@ -228,10 +229,10 @@ class WitCatMouse { //鼠标移动量 acceleration(args){ if(args.way == "x"){ - return xMouse; + return xMouse * (document.body.clientHeight / cvs.offsetHeight); } else{ - return -yMouse; + return -yMouse * (document.body.clientHeight / cvs.offsetHeight); } } //数量 @@ -242,11 +243,11 @@ class WitCatMouse { num(args){ if(args.num > 0 && args.num <= touch.length){ if(args.type == "x"){ - return this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.style.width.split("px")[0]); + return (this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.offsetWidth)) - (this.runtime.stageWidth / 2); + } else{ - console.log(cvs.style.height); - return this.runtime.stageHeight * ((touch[args.num - 1].clientY - cvs.getBoundingClientRect().top) / cvs.style.height.split("px")[0]); + return (this.runtime.stageHeight / 2) - this.runtime.stageHeight * ((touch[args.num - 1].clientY - cvs.getBoundingClientRect().top) / cvs.offsetHeight); } } else{ From 6ea9bf2f52f066e8bef1b8e27771e65de8cb24c3 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Wed, 21 Dec 2022 15:55:03 +0800 Subject: [PATCH 193/357] Update More_Mouse.js --- wit_cat/More_Mouse.js | 284 ++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 272 insertions(+), 12 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 64a4be6f..58c76737 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -10,7 +10,11 @@ let xMouse = 0; let yMouse = 0; let isMove = false,timer = null; let touch = []; - +let fill = 0; +let filln = 0; +let w = 0,h = 0; +let div,divs,divv,divvs,news,background,Operatinginstructions,bug,url,discord; +let wv = 0,hv = 0; //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; @@ -18,12 +22,28 @@ if(cvs == null){ alert("当前页面不支持多指触控,请前往作品详情页体验完整作品!"); } + +//添加监听器 +const config = { attributes: true, childList: true, subtree: true, attributeFilter:['style'] }; +const callback = function(mutationsList, observer) { + if(fill == 1){ + observer.disconnect(); + fills(); + observer.observe(cvs, config); + } +}; +const observer = new MutationObserver(callback); +observer.observe(cvs, config); + + class WitCatMouse { constructor(runtime) { this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "WitCatMouse.name": "高级鼠标", + "WitCatMouse.name": "高级操作", + "WitCatMouse.name.1": "高级鼠标", + "WitCatMouse.name.2": "多指触控", "WitCatMouse.set": "[set]右键菜单", "WitCatMouse.set.1": "启用", "WitCatMouse.set.2": "禁用", @@ -41,9 +61,16 @@ class WitCatMouse { "WitCatTouch.num": "第[num]个手指的[type]", "WitCatTouch.type.1": "X", "WitCatTouch.type.2": "Y", + "WitCatMouse.fill": "[set]沉浸式全屏", + "WitCatMouse.fillask.1": "作品请求沉浸式全屏,是否同意?\n", + "WitCatMouse.fillask.2": "/3次连续拒绝后将不再提示\n您仍可以使用 ctrl+shift+alt 切换沉浸式全屏状态", + "WitCatMouse.setfill": "⚠️(危)设置分辨率高设为[num]", + "WitCatMouse.resolution": "当前分辨率高", }, en: { - "WitCatMouse.name": "More Mouse", + "WitCatMouse.name": "More operate", + "WitCatMouse.name.1": "more mouse", + "WitCatMouse.name.2": "more touch", "WitCatMouse.set": "[set]Right-click menu", "WitCatMouse.set.1": "allow", "WitCatMouse.set.2": "forbid", @@ -62,6 +89,11 @@ class WitCatMouse { "WitCatTouch.num": "the[num]finger`s[type]", "WitCatTouch.type.1": "X", "WitCatTouch.type.2": "Y", + "WitCatMouse.fill": "[set]immersive full-screen", + "WitCatMouse.fillask.1": "The project requests to turn on immersive full-screen, agree or not?\nWill stop asking if you keep on to reject for ", + "WitCatMouse.fillask.2": "/3 times\nYou can also use Ctrl+Shift+Alt to toggle immersive full-screen later.", + "WitCatMouse.setfill": "⚠️(danger)Set resolution height to[num]", + "WitCatMouse.resolution": "Current high resolution", } }) } @@ -84,6 +116,35 @@ class WitCatMouse { color1: "#8eace1", color2: "#ffffff", blocks: [ + { + opcode: 'setfill', + blockType: "command", + text: this.formatMessage("WitCatMouse.setfill"), + arguments: { + num:{ + type:"number", + defaultValue:"360", + }, + } + }, + { + opcode: 'fill', + blockType: "command", + text: this.formatMessage("WitCatMouse.fill"), + arguments: { + set:{ + type:"bool", + menu:"set", + }, + } + }, + { + opcode: 'resolution', + blockType: "reporter", + text: this.formatMessage("WitCatMouse.resolution"), + arguments: {} + }, + "---"+this.formatMessage("WitCatMouse.name.1"), { opcode: 'set', blockType: "command", @@ -123,6 +184,7 @@ class WitCatMouse { }, }, }, + "---"+this.formatMessage("WitCatMouse.name.2"), { opcode: 'down', blockType: "reporter", @@ -229,10 +291,10 @@ class WitCatMouse { //鼠标移动量 acceleration(args){ if(args.way == "x"){ - return xMouse * (document.body.clientHeight / cvs.offsetHeight); + return xMouse; } else{ - return -yMouse * (document.body.clientHeight / cvs.offsetHeight); + return -yMouse; } } //数量 @@ -243,17 +305,47 @@ class WitCatMouse { num(args){ if(args.num > 0 && args.num <= touch.length){ if(args.type == "x"){ - return (this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.offsetWidth)) - (this.runtime.stageWidth / 2); - + return this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.offsetWidth); } else{ - return (this.runtime.stageHeight / 2) - this.runtime.stageHeight * ((touch[args.num - 1].clientY - cvs.getBoundingClientRect().top) / cvs.offsetHeight); + return this.runtime.stageHeight * ((touch[args.num - 1].clientY - cvs.getBoundingClientRect().top) / cvs.offsetHeight); } } else{ return null; } } + //全屏 + fill(args){ + if(args.set == "true"){ + if(filln < 3){ + if(fill == 0){ + let a = confirm(this.formatMessage('WitCatMouse.fillask.1') + filln + this.formatMessage('WitCatMouse.fillask.2')); + if(a == true){ + filln = 0; + fillin(); + } + else{ + filln += 1; + } + } + } + } + else{ + outoffill(); + } + } + //设置分辨率 + setfill(args){ + w = args.num / this.runtime.stageHeight * this.runtime.stageWidth; + h = args.num; + cvs.width = args.num / this.runtime.stageHeight * this.runtime.stageWidth; + cvs.height = args.num; + } + //当前分辨率 + resolution(args){ + return cvs.height; + } } window.tempExt = { @@ -270,12 +362,12 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatMouse.name": "高级鼠标/触控", - "WitCatMouse.descp": "更精准的控制鼠标/触屏!" + "WitCatMouse.name": "高级操控", + "WitCatMouse.descp": "更精准的控制鼠标/触屏/全屏!" }, en: { - "WitCatMouse.name": "More Mouse/Touch", - "WitCatMouse.descp": "More precise mouse/touch control!" + "WitCatMouse.name": "More operate", + "WitCatMouse.descp": "More precise mouse/touch/full screen control!" } } }; @@ -299,6 +391,7 @@ document.addEventListener("mousemove", ev => { yMouse = 0; },30); }); +//多指触控 cvs.addEventListener('touchstart',function(e){ touch = e.targetTouches; }) @@ -308,3 +401,170 @@ cvs.addEventListener('touchmove',function(e){ cvs.addEventListener('touchend',function(e){ touch = e.targetTouches; }) + + + +//大小改变刷新 +window.onresize=function(){ + if(fill == 1){ + fills(); + } +} +//开启全屏 +function fillin(){ + fill = 1; + + var mo=function(e){e.preventDefault();}; + document.body.style.overflow='hidden'; + document.addEventListener("touchmove",mo,false);//禁止页面滑动 + if(window.location.href.split("/")[2] != "cocrea.world"){ + div = document.getElementById("root").getElementsByTagName('div')[0]; + divs = document.getElementById("root"); + url = window.location.href.split("/")[3].split("?")[0]; + if(url != "scratch-player"){ + divs.removeChild(div); + } + } + + news = document.body.getElementsByClassName("convention-1wIbd")[0]; + if(typeof(news) != "undefined"){ + news.style = "display:none;"; + } + divv = document.body.getElementsByClassName("actions-2lk9z")[0]; + if(typeof(divv) != "undefined"){ + divv.style = "display:none;"; + } + Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; + if(typeof(Operatinginstructions) != "undefined"){ + Operatinginstructions.style = "display:none;"; + } + bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; + if(typeof(bug) != "undefined"){ + bug.style = "display:none;"; + } + discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; + if(typeof(discord) != "undefined"){ + discord.style = "display:none"; + } + document.body.scrollTop = 0; + document.documentElement.scrollTop = 0; + scrollTo(0,0); + setTimeout(function() { + fills(); + }, 50); +} +//关闭全屏 +function outoffill(){ + //恢复头 + fill = 0; + var mo=function(e){e.preventDefault();}; + document.body.style.overflow='auto'; + document.addEventListener("touchmove",mo,true);//允许页面滑动 + if(window.location.href.split("/")[2] != "cocrea.world"){ + divs = document.getElementById("root"); + url = window.location.href.split("/")[3].split("?")[0]; + if(url != "scratch-player"){ + divs.insertBefore(div, divs.children[0]); + } + } + document.body.removeChild(background); + news = document.body.getElementsByClassName("convention-1wIbd")[0]; + if(typeof(news) != "undefined"){ + news.style = ""; + } + divv = document.body.getElementsByClassName("actions-2lk9z")[0]; + if(typeof(divv) != "undefined"){ + divv.style = ""; + } + divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; + if(typeof(divvs) != "undefined"){ + divvs.style = ""; + } + Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; + if(typeof(Operatinginstructions) != "undefined"){ + Operatinginstructions.style = ""; + } + bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; + if(typeof(bug) != "undefined"){ + bug.style = ""; + } + discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; + if(typeof(discord) != "undefined"){ + discord.style = ""; + } + cvs.parentNode.parentNode.parentNode.parentNode.style = "width:100%; height:100%;z-index: 1000000;"; + cvs.parentNode.parentNode.style = "height: 100%; width: 100%;"; + cvs.parentNode.style = ""; + cvs.style = "height: 100%; width: 100%;"; +} +//循环检测 +function fills(){ + if(w != 0 && h != 0){ + cvs.width = w; + cvs.height = h; + } + let ws = 0,hs = 0; + cvs = document.getElementsByTagName("canvas")[0]; + if((cvs.width / cvs.height) >= (document.documentElement.clientWidth / document.documentElement.clientHeight)){ + hs = (document.documentElement.clientWidth / cvs.width) * cvs.height; + ws = document.documentElement.clientWidth; + wv = 0; + hv = (document.documentElement.clientHeight - hs) / 2; + } + else{ + hs = document.documentElement.clientHeight; + ws = (document.documentElement.clientHeight / cvs.height) * cvs.width; + wv = (document.documentElement.clientWidth - ws) / 2; + hv = 0; + } + divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; + if(typeof(divvs) != "undefined"){ + divvs.style = "display:none;"; + } + cvs.parentNode.parentNode.parentNode.parentNode.style = "position:fixed; left:" + wv + "px; top:" + hv + "px; width:" + ws + "px; height:" + hs + "px;z-index: 1000000;border-radius:0px"; + cvs.parentNode.parentNode.parentNode.style = "width:" + ws + "px; height:" + hs + "px;border-radius:0px"; + cvs.parentNode.parentNode.style = "width:100%; height:100%;border-radius:0px"; + cvs.parentNode.style = "width:100%; height:100%;border-radius:0px"; + if(isInPage(cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild)){ + cvs.parentNode.parentNode.nextElementSibling.style="width: 100%; height: 100%;"; + cvs.parentNode.parentNode.nextElementSibling.firstChild.style="width: 100%; height: 100%;"; + cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.firstChild.style="width: 100%; height: 100%;"; + cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.style="width: 100%;"; + } + cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("div")[0].style = "width:100%; height:100%;" + let canvass = cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("canvas"); + for(let i = 0;i Date: Wed, 21 Dec 2022 17:35:11 +0800 Subject: [PATCH 194/357] Update More_Mouse.js --- wit_cat/More_Mouse.js | 17 ++++++++++------- 1 file changed, 10 insertions(+), 7 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 58c76737..b0c9e640 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -19,9 +19,16 @@ let wv = 0,hv = 0; //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; if(cvs == null){ - alert("当前页面不支持多指触控,请前往作品详情页体验完整作品!"); + alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); +} +else{ + for(let i = 1;cvs.className != "" && i <= document.getElementsByTagName("canvas").length;i++){ + cvs = document.getElementsByTagName("canvas")[i]; + } + if(cvs == null){ + alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); + } } - //添加监听器 const config = { attributes: true, childList: true, subtree: true, attributeFilter:['style'] }; @@ -504,7 +511,6 @@ function fills(){ cvs.height = h; } let ws = 0,hs = 0; - cvs = document.getElementsByTagName("canvas")[0]; if((cvs.width / cvs.height) >= (document.documentElement.clientWidth / document.documentElement.clientHeight)){ hs = (document.documentElement.clientWidth / cvs.width) * cvs.height; ws = document.documentElement.clientWidth; @@ -532,10 +538,7 @@ function fills(){ cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.style="width: 100%;"; } cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("div")[0].style = "width:100%; height:100%;" - let canvass = cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("canvas"); - for(let i = 0;i Date: Wed, 21 Dec 2022 17:35:38 +0800 Subject: [PATCH 195/357] Update Input.js --- wit_cat/Input.js | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 941a85fb..b00121fd 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -9,7 +9,15 @@ const extensionId = "WitCatInput"; //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; if(cvs == null){ - alert("当前页面不支持多指触控,请前往作品详情页体验完整作品!"); + alert("当前页面不支持输入框,请前往作品详情页体验完整作品!"); +} +else{ + for(let i = 1;cvs.className != "" && i <= document.getElementsByTagName("canvas").length;i++){ + cvs = document.getElementsByTagName("canvas")[i]; + } + if(cvs == null){ + alert("当前页面不支持输入框,请前往作品详情页体验完整作品!"); + } } class WitCatInput { From 73ac6685ccd58e247ab88a43307ad7be4fb08b3d Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Thu, 22 Dec 2022 11:33:43 +0800 Subject: [PATCH 196/357] Update More_Mouse.js --- wit_cat/More_Mouse.js | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index b0c9e640..4b9b7119 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -339,7 +339,9 @@ class WitCatMouse { } } else{ - outoffill(); + if(fill == 1){ + outoffill(); + } } } //设置分辨率 From c1eb58de3c220af218ebdbe73495296ec62d400a Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Thu, 22 Dec 2022 16:36:35 +0800 Subject: [PATCH 197/357] Update Input.js --- wit_cat/Input.js | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index b00121fd..2d0338b6 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -26,7 +26,7 @@ class WitCatInput { this._formatMessage = runtime.getFormatMessage({ "zh-cn": { "WitCatInput.name": "文本输入框", - "WitCatInput.createinput": "设置或创建ID为[id]的[type]文本框的X[x]Y[y]宽[width]高[height]内容[text]提示[texts]字体大小[size]", + "WitCatInput.createinput": "设置或创建ID为[id]的[type]文本框的X[x]Y[y]宽[width]高[height]内容[text]颜色[color]提示[texts]字体大小[size]", "WitCatInput.deleteinput": "删除ID为[id]的文本框", "WitCatInput.getinput": "获得ID为[id]的文本框内容", "WitCatInput.isinput": "焦点是否在ID为[id]的文本框上", @@ -38,13 +38,13 @@ class WitCatInput { }, en: { "WitCatInput.name": "input", - "WitCatInput.createinput": "Set or create an[type]input with ID[id]X[x]Y[y]width[width]height[height]content[text]prompt[texts]font-size[size]", + "WitCatInput.createinput": "Set or create an[type]input with ID[id]X[x]Y[y]width[width]height[height]content[text]color[color]prompt[texts]font-size[size]", "WitCatInput.deleteinput": "delete an input with ID[id]", "WitCatInput.getinput": "get an input with ID[id]", "WitCatInput.isinput": "is the focus on the input with ID[id]?", "WitCatInput.nowinput": "let teh focus on the input with ID[id]", "WitCatInput.deleteallinput": "delete all input", - "WitCatInput.compute": "The font size at the current resolution that is high [size]", + "WitCatInput.compute": "Font size of now screen[size]", "WitCatInput.type.1": "Single line", "WitCatInput.type.2": "Multiple lines", } @@ -102,6 +102,10 @@ class WitCatInput { type:"string", defaultValue:"hello world!", }, + color:{ + type:"string", + defaultValue:"#000000", + }, texts:{ type:"string", defaultValue:"hello world!", @@ -222,7 +226,7 @@ class WitCatInput { y = (y / this.runtime.stageHeight) * 100; width = (width / this.runtime.stageWidth) * 100; height = (height / this.runtime.stageHeight) * 100; - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none`; + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none;color:` + args.color + `;`; let search = document.getElementById("WitCatInput" + args.id); if(search != null){ if(search.name == args.type){ From 8433f0ad7550b5ba565c595a164f72e5205f0af3 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sat, 31 Dec 2022 21:29:51 +0800 Subject: [PATCH 198/357] Update Input.js --- wit_cat/Input.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 2d0338b6..59ef6cae 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -63,7 +63,7 @@ class WitCatInput { return { id: extensionId, // 拓展id name: this.formatMessage("WitCatInput.name"), // 拓展名 - docsURI: "https://www.ccw.site/post/d6d96e80-3f58-4a19-b7e6-c567d3a6a583", + docsURI: "https://www.ccw.site/post/6153a7a6-05fb-462e-b785-b97700b12bc2", blockIconURI: _icon, menuIconURI: _icon, color1: "#52baba", From cf4484ce6680e841604267fecc257529dbf7df20 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sat, 31 Dec 2022 21:31:54 +0800 Subject: [PATCH 199/357] Update More_Mouse.js --- wit_cat/More_Mouse.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 4b9b7119..5596f387 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -48,7 +48,7 @@ class WitCatMouse { this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "WitCatMouse.name": "高级操作", + "WitCatMouse.name": "高级工具", "WitCatMouse.name.1": "高级鼠标", "WitCatMouse.name.2": "多指触控", "WitCatMouse.set": "[set]右键菜单", @@ -75,7 +75,7 @@ class WitCatMouse { "WitCatMouse.resolution": "当前分辨率高", }, en: { - "WitCatMouse.name": "More operate", + "WitCatMouse.name": "More tools", "WitCatMouse.name.1": "more mouse", "WitCatMouse.name.2": "more touch", "WitCatMouse.set": "[set]Right-click menu", @@ -117,7 +117,7 @@ class WitCatMouse { return { id: extensionId, // 拓展id name: this.formatMessage("WitCatMouse.name"), // 拓展名 - docsURI: "https://www.ccw.site/post/d6d96e80-3f58-4a19-b7e6-c567d3a6a583", + docsURI: "https://www.ccw.site/post/c36aa805-b29d-48da-aba1-468a6cf80bfa", blockIconURI: _icon, menuIconURI: _icon, color1: "#8eace1", @@ -371,11 +371,11 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatMouse.name": "高级操控", + "WitCatMouse.name": "高级工具", "WitCatMouse.descp": "更精准的控制鼠标/触屏/全屏!" }, en: { - "WitCatMouse.name": "More operate", + "WitCatMouse.name": "More tools", "WitCatMouse.descp": "More precise mouse/touch/full screen control!" } } From 4c91706999deba4f4d8dcf550cb828000ac2dd5d Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 2 Jan 2023 15:33:35 +0800 Subject: [PATCH 200/357] Update Input.js --- wit_cat/Input.js | 393 +++++++++++++++++++++++++++++++++++------------ 1 file changed, 299 insertions(+), 94 deletions(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 59ef6cae..c73317f8 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -8,14 +8,14 @@ const extensionId = "WitCatInput"; //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; -if(cvs == null){ +if (cvs == null) { alert("当前页面不支持输入框,请前往作品详情页体验完整作品!"); } -else{ - for(let i = 1;cvs.className != "" && i <= document.getElementsByTagName("canvas").length;i++){ +else { + for (let i = 1; cvs.className != "" && i <= document.getElementsByTagName("canvas").length; i++) { cvs = document.getElementsByTagName("canvas")[i]; } - if(cvs == null){ + if (cvs == null) { alert("当前页面不支持输入框,请前往作品详情页体验完整作品!"); } } @@ -26,27 +26,53 @@ class WitCatInput { this._formatMessage = runtime.getFormatMessage({ "zh-cn": { "WitCatInput.name": "文本输入框", - "WitCatInput.createinput": "设置或创建ID为[id]的[type]文本框的X[x]Y[y]宽[width]高[height]内容[text]颜色[color]提示[texts]字体大小[size]", + "WitCatInput.createinput": "ID为[id]的[type]框X[x]Y[y]宽[width]高[height]内容[text]颜色[color]提示[texts]字体大小[size]", "WitCatInput.deleteinput": "删除ID为[id]的文本框", - "WitCatInput.getinput": "获得ID为[id]的文本框内容", + "WitCatInput.getinput": "获得ID为[id]的文本框的[type]", "WitCatInput.isinput": "焦点是否在ID为[id]的文本框上", + "WitCatInput.whatinput": "焦点位置", "WitCatInput.nowinput": "将焦点聚焦在ID为[id]的文本框上", "WitCatInput.deleteallinput": "删除所有文本框", "WitCatInput.compute": "当前分辨率下高[size]的字体大小", "WitCatInput.type.1": "单行文本", "WitCatInput.type.2": "多行文本", + "WitCatInput.number": "第[num]个文本框的[type]", + "WitCatInput.numbers": "文本框的数量", + "WitCatInput.number.1": "X", + "WitCatInput.number.2": "Y", + "WitCatInput.number.3": "宽", + "WitCatInput.number.4": "高", + "WitCatInput.number.5": "内容", + "WitCatInput.number.6": "颜色", + "WitCatInput.number.7": "提示", + "WitCatInput.number.8": "字体大小", + "WitCatInput.number.9": "所有(json)", + "WitCatInput.number.10": "ID", }, en: { "WitCatInput.name": "input", - "WitCatInput.createinput": "Set or create an[type]input with ID[id]X[x]Y[y]width[width]height[height]content[text]color[color]prompt[texts]font-size[size]", + "WitCatInput.createinput": "[type]input with ID[id]X[x]Y[y]width[width]height[height]content[text]color[color]prompt[texts]font-size[size]", "WitCatInput.deleteinput": "delete an input with ID[id]", - "WitCatInput.getinput": "get an input with ID[id]", + "WitCatInput.getinput": "get [type] with ID[id]", "WitCatInput.isinput": "is the focus on the input with ID[id]?", + "WitCatInput.whatinput": "Focal position", "WitCatInput.nowinput": "let teh focus on the input with ID[id]", "WitCatInput.deleteallinput": "delete all input", "WitCatInput.compute": "Font size of now screen[size]", "WitCatInput.type.1": "Single line", "WitCatInput.type.2": "Multiple lines", + "WitCatInput.number": "[type]of[num]input", + "WitCatInput.numbers": "input number", + "WitCatInput.number.1": "X", + "WitCatInput.number.2": "Y", + "WitCatInput.number.3": "width", + "WitCatInput.number.4": "height", + "WitCatInput.number.5": "content", + "WitCatInput.number.6": "color", + "WitCatInput.number.7": "prompt", + "WitCatInput.number.8": "font-size", + "WitCatInput.number.9": "all(json)", + "WitCatInput.number.10": "ID", } }) } @@ -74,45 +100,45 @@ class WitCatInput { blockType: "command", text: this.formatMessage("WitCatInput.createinput"), arguments: { - id:{ - type:"string", - defaultValue:"i", + id: { + type: "string", + defaultValue: "i", }, - type:{ - type:"string", - menu:"type", + type: { + type: "string", + menu: "type", }, - x:{ - type:"number", - defaultValue:"0", + x: { + type: "number", + defaultValue: "0", }, - y:{ - type:"number", - defaultValue:"0", + y: { + type: "number", + defaultValue: "0", }, - width:{ - type:"number", - defaultValue:"100", + width: { + type: "number", + defaultValue: "100", }, - height:{ - type:"number", - defaultValue:"20", + height: { + type: "number", + defaultValue: "20", }, - text:{ - type:"string", - defaultValue:"hello world!", + text: { + type: "string", + defaultValue: "hello world!", }, - color:{ - type:"string", - defaultValue:"#000000", + color: { + type: "string", + defaultValue: "#000000", }, - texts:{ - type:"string", - defaultValue:"hello world!", + texts: { + type: "string", + defaultValue: "hello world!", }, - size:{ - type:"number", - defaultValue:"16", + size: { + type: "number", + defaultValue: "16", } }, }, @@ -121,9 +147,9 @@ class WitCatInput { blockType: "reporter", text: this.formatMessage("WitCatInput.compute"), arguments: { - size:{ - type:"number", - defaultValue:"16", + size: { + type: "number", + defaultValue: "16", } }, }, @@ -132,9 +158,9 @@ class WitCatInput { blockType: "command", text: this.formatMessage("WitCatInput.deleteinput"), arguments: { - id:{ - type:"string", - defaultValue:"i", + id: { + type: "string", + defaultValue: "i", } }, }, @@ -143,31 +169,62 @@ class WitCatInput { blockType: "reporter", text: this.formatMessage("WitCatInput.getinput"), arguments: { - id:{ - type:"string", - defaultValue:"i", + id: { + type: "string", + defaultValue: "i", + }, + type: { + type: "string", + menu: "types", } }, }, + { + opcode: "number", + blockType: "reporter", + text: this.formatMessage("WitCatInput.number"), + arguments: { + num: { + type: "number", + defaultValue: "1", + }, + type: { + type: "string", + menu: "types", + } + }, + }, + { + opcode: "numbers", + blockType: "reporter", + text: this.formatMessage("WitCatInput.numbers"), + arguments: {}, + }, { opcode: "isinput", blockType: "Boolean", text: this.formatMessage("WitCatInput.isinput"), arguments: { - id:{ - type:"string", - defaultValue:"i", + id: { + type: "string", + defaultValue: "i", } }, }, + { + opcode: "whatinput", + blockType: "reporter", + text: this.formatMessage("WitCatInput.whatinput"), + arguments: {}, + }, { opcode: "nowinput", blockType: "command", text: this.formatMessage("WitCatInput.nowinput"), arguments: { - id:{ - type:"string", - defaultValue:"i", + id: { + type: "string", + defaultValue: "i", } }, }, @@ -178,66 +235,108 @@ class WitCatInput { arguments: {}, }, ], - menus:{ + menus: { type: [ { - text: this.formatMessage('WitCatInput.type.1'), - value: 'input' + text: this.formatMessage('WitCatInput.type.1'), + value: 'input' }, { - text: this.formatMessage('WitCatInput.type.2'), - value: 'Textarea' + text: this.formatMessage('WitCatInput.type.2'), + value: 'Textarea' }, ], + types: [ + { + text: this.formatMessage('WitCatInput.number.10'), + value: 'ID' + }, + { + text: this.formatMessage('WitCatInput.number.1'), + value: 'X' + }, + { + text: this.formatMessage('WitCatInput.number.2'), + value: 'Y' + }, + { + text: this.formatMessage('WitCatInput.number.3'), + value: 'width' + }, + { + text: this.formatMessage('WitCatInput.number.4'), + value: 'height' + }, + { + text: this.formatMessage('WitCatInput.number.5'), + value: 'content' + }, + { + text: this.formatMessage('WitCatInput.number.6'), + value: 'color' + }, + { + text: this.formatMessage('WitCatInput.number.7'), + value: 'prompt' + }, + { + text: this.formatMessage('WitCatInput.number.8'), + value: 'font-size' + }, + { + text: this.formatMessage('WitCatInput.number.9'), + value: 'json' + }, + ] } }; } //设置或创建文本框 - createinput(args){ + createinput(args) { let x = args.x; let y = args.y; let width = args.width; let height = args.height; - if(args.x > this.runtime.stageWidth){ + if (args.x > this.runtime.stageWidth) { x = this.runtime.stageWidth; } - if(args.x < 0){ + if (args.x < 0) { x = 0; } - if(args.y > this.runtime.stageHeight){ + if (args.y > this.runtime.stageHeight) { y = this.runtime.stageHeight; } - if(args.y < 0){ + if (args.y < 0) { y = 0; } - if(args.x + args.width > this.runtime.stageWidth){ + if (Number(args.x) + Number(args.width) > this.runtime.stageWidth) { width = this.runtime.stageWidth - x; } - if(args.width < 0){ + if (args.width < 0) { width = 0; } - if(args.y + args.height > this.runtime.stageHeight){ + if (Number(args.y) + Number(args.height) > this.runtime.stageHeight) { height = this.runtime.stageHeight - y; } - if(args.height < 0){ + if (args.height < 0) { height = 0; } x = (x / this.runtime.stageWidth) * 100; y = (y / this.runtime.stageHeight) * 100; width = (width / this.runtime.stageWidth) * 100; height = (height / this.runtime.stageHeight) * 100; - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:`+ x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none;color:` + args.color + `;`; + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none;color:` + args.color + `;`; let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ - if(search.name == args.type){ + if (search != null) { + if (search.name == args.type) { search.style = dom; search.value = args.text; search.placeholder = args.texts; } - else{ + else { cvs.parentNode.removeChild(search); let eleLink = document.createElement(args.type); - if(args.type == "input"){ + if (args.type == "input") { eleLink.type = "text"; } eleLink.style = dom; @@ -249,9 +348,9 @@ class WitCatInput { cvs.parentNode.appendChild(eleLink); } } - else{ + else { let eleLink = document.createElement(args.type); - if(args.type == "input"){ + if (args.type == "input") { eleLink.type = "text"; } eleLink.style = dom; @@ -264,56 +363,140 @@ class WitCatInput { } } //删除文本框 - deleteinput(args){ + deleteinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ + if (search != null) { cvs.parentNode.removeChild(search); } } //获取文本框内容 - getinput(args){ + getinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ - return(search.value); + if (search != null) { + if (args.type == "X") + return (search.style.left.split("%")[0] / 100) * this.runtime.stageWidth; + else if (args.type == "Y") + return (search.style.top.split("%")[0] / 100) * this.runtime.stageHeight; + else if (args.type == "width") + return (search.style.width.split("%")[0] / 100) * this.runtime.stageWidth; + else if (args.type == "height") + return (search.style.height.split("%")[0] / 100) * this.runtime.stageHeight; + else if (args.type == "content") + return search.value; + else if (args.type == "color") + return search.style.color.colorHex(); + else if (args.type == "prompt") + return search.placeholder; + else if (args.type == "font-size") + return search.style.fontSize.split("px")[0]; + else if (args.type == "ID") + return search.id.split("WitCatInput")[1]; + else { + return ( + "\{\"" + "X" + "\":\"" + ((search.style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + + "\"" + "Y" + "\":\"" + ((search.style.top.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + + "\"" + "width" + "\":\"" + ((search.style.width.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + + "\"" + "height" + "\":\"" + ((search.style.height.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + + "\"" + "content" + "\":" + JSON.stringify(search.value) + "," + + "\"" + "color" + "\":\"" + (search.style.color.colorHex()) + "\"," + + "\"" + "prompt" + "\":\"" + (search.placeholder) + "\"," + + "\"" + "font-size" + "\":\"" + (search.style.fontSize.split("px")[0]) + "\"," + + "\"" + "ID" + "\":\"" + (search.id.split("WitCatInput")[1]) + "\"\}" + ) + } } - else{ - return(""); + else { + return (""); } } //焦点判断 - isinput(args){ + isinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ - if(search == document.activeElement){ - return(true); + if (search != null) { + if (search == document.activeElement) { + return (true); } - else{ - return(false); + else { + return (false); } } - else{ - return(false); + else { + return (false); + } + } + //焦点位置 + whatinput() { + if (document.activeElement.className == "WitCatInput") { + return document.activeElement.id.split("WitCatInput"); + } + else { + return ""; } } //焦点获取 - nowinput(args){ + nowinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if(search != null){ + if (search != null) { search.focus(); } } //删除所有文本框 - deleteallinput(args){ + deleteallinput(args) { let search = document.getElementsByClassName("WitCatInput"); let i = 0; - for(i = search.length - 1;i >= 0;i--){ + for (i = search.length - 1; i >= 0; i--) { search[i].parentNode.removeChild(search[i]); } } //计算坐标 - compute(args){ + compute(args) { return (cvs.style.width.split("px")[0] / 360) * args.size; } + //获取状态 + number(args) { + let search = document.getElementsByClassName("WitCatInput"); + if (search.length >= args.num) { + if (args.type == "X") + return (search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth; + else if (args.type == "Y") + return (search[args.num - 1].style.top.split("%")[0] / 100) * this.runtime.stageHeight; + else if (args.type == "width") + return (search[args.num - 1].style.width.split("%")[0] / 100) * this.runtime.stageWidth; + else if (args.type == "height") + return (search[args.num - 1].style.height.split("%")[0] / 100) * this.runtime.stageHeight; + else if (args.type == "content") + return search[args.num - 1].value; + else if (args.type == "color") + return search[args.num - 1].style.color.colorHex(); + else if (args.type == "prompt") + return search[args.num - 1].placeholder; + else if (args.type == "font-size") + return search[args.num - 1].style.fontSize.split("px")[0]; + else if (args.type == "ID") + return search[args.num - 1].id.split("WitCatInput")[1]; + else { + return ( + "\{\"" + "X" + "\":\"" + ((search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + + "\"" + "Y" + "\":\"" + ((search[args.num - 1].style.top.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + + "\"" + "width" + "\":\"" + ((search[args.num - 1].style.width.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + + "\"" + "height" + "\":\"" + ((search[args.num - 1].style.height.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + + "\"" + "content" + "\":" + JSON.stringify(search[args.num - 1].value) + "," + + "\"" + "color" + "\":\"" + (search[args.num - 1].style.color.colorHex()) + "\"," + + "\"" + "prompt" + "\":\"" + (search[args.num - 1].placeholder) + "\"," + + "\"" + "font-size" + "\":\"" + (search[args.num - 1].style.fontSize.split("px")[0]) + "\"," + + "\"" + "ID" + "\":\"" + (search[args.num - 1].id.split("WitCatInput")[1]) + "\"\}" + ) + } + } + else { + return ""; + } + } + //文本框数量 + numbers() { + let search = document.getElementsByClassName("WitCatInput"); + return search.length; + } } window.tempExt = { @@ -341,3 +524,25 @@ window.tempExt = { }; /* vim: set expandtab tabstop=2 shiftwidth=2: */ +//颜色转换 +String.prototype.colorHex = function () { + // RGB颜色值的正则 + var reg = /^(rgb|RGB)/; + var color = this; + if (reg.test(color)) { + var strHex = "#"; + // 把RGB的3个数值变成数组 + var colorArr = color.replace(/(?:\(|\)|rgb|RGB)*/g, "").split(","); + // 转成16进制 + for (var i = 0; i < colorArr.length; i++) { + var hex = Number(colorArr[i]).toString(16); + if (hex === "0") { + hex += hex; + } + strHex += hex; + } + return strHex; + } else { + return String(color); + } +}; From 28cef85cae76edc0e7a2a1928192350ab0fd27e8 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 2 Jan 2023 15:54:50 +0800 Subject: [PATCH 201/357] Update File_Helper.js --- wit_cat/File_Helper.js | 237 +++++++++++++++++++++-------------------- 1 file changed, 120 insertions(+), 117 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index a97b3fc4..8d06f392 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -88,7 +88,7 @@ class WitCatFileHelper { color1: "#60D6F4", color2: "#55a7f7", blocks: [ - "---"+this.formatMessage("WitCatFileHelper.filehelper"), + "---" + this.formatMessage("WitCatFileHelper.filehelper"), { opcode: "downloads", blockType: "command", @@ -171,13 +171,13 @@ class WitCatFileHelper { blockType: "command", text: this.formatMessage("WitCatFileHelper.showvar"), arguments: { - name:{ - type:"string", - defaultValue:"i", + name: { + type: "string", + defaultValue: "i", }, - show:{ - type:"string", - menu:"setvariable", + show: { + type: "string", + menu: "setvariable", }, }, }, @@ -186,17 +186,17 @@ class WitCatFileHelper { blockType: "command", text: this.formatMessage("WitCatFileHelper.saveother"), arguments: { - id:{ - type:"string", - defaultValue:"6373950041d21d2d2cd0da9b", + id: { + type: "string", + defaultValue: "6373950041d21d2d2cd0da9b", }, - name:{ - type:"string", - defaultValue:"i", + name: { + type: "string", + defaultValue: "i", }, - text:{ - type:"string", - defaultValue:"wit_cat!", + text: { + type: "string", + defaultValue: "wit_cat!", }, }, }, @@ -205,13 +205,13 @@ class WitCatFileHelper { blockType: "reporter", text: this.formatMessage("WitCatFileHelper.uploadother"), arguments: { - id:{ - type:"string", - defaultValue:"6373950041d21d2d2cd0da9b", + id: { + type: "string", + defaultValue: "6373950041d21d2d2cd0da9b", }, - name:{ - type:"string", - defaultValue:"i", + name: { + type: "string", + defaultValue: "i", }, }, }, @@ -220,13 +220,13 @@ class WitCatFileHelper { blockType: "reporter", text: this.formatMessage("WitCatFileHelper.other"), arguments: { - id:{ - type:"string", - defaultValue:"6373950041d21d2d2cd0da9b", + id: { + type: "string", + defaultValue: "6373950041d21d2d2cd0da9b", }, - name:{ - type:"string", - defaultValue:"i", + name: { + type: "string", + defaultValue: "i", }, }, }, @@ -252,7 +252,7 @@ class WitCatFileHelper { }, }, }, - "---"+this.formatMessage("WitCatFileHelper.Multiplelinestext"), + "---" + this.formatMessage("WitCatFileHelper.Multiplelinestext"), { opcode: "segmentation", blockType: "reporter", @@ -266,7 +266,7 @@ class WitCatFileHelper { type: "string", defaultValue: '/', }, - thing:{ + thing: { type: "string", menu: 'thing', } @@ -277,13 +277,13 @@ class WitCatFileHelper { blockType: "reporter", text: this.formatMessage("WitCatFileHelper.deleteMultiplelinestext"), arguments: { - text:{ - type:"string", - defaultValue:"wit_cat!!!\nwit_cat!!!", + text: { + type: "string", + defaultValue: "wit_cat!!!\nwit_cat!!!", }, - num:{ - type:"number", - defaultValue:"1", + num: { + type: "number", + defaultValue: "1", } }, }, @@ -292,17 +292,17 @@ class WitCatFileHelper { blockType: "reporter", text: this.formatMessage("WitCatFileHelper.addMultiplelinestext"), arguments: { - text:{ - type:"string", - defaultValue:"wit_cat!!!", + text: { + type: "string", + defaultValue: "wit_cat!!!", }, - texts:{ - type:"string", - defaultValue:"wit_cat!!!\nwit_cat!!!", + texts: { + type: "string", + defaultValue: "wit_cat!!!\nwit_cat!!!", }, - num:{ - type:"string", - defaultValue:"last", + num: { + type: "string", + defaultValue: "last", } }, }, @@ -311,13 +311,13 @@ class WitCatFileHelper { blockType: "reporter", text: this.formatMessage("WitCatFileHelper.whatMultiplelinestext"), arguments: { - text:{ - type:"string", - defaultValue:"wit_cat!!!", + text: { + type: "string", + defaultValue: "wit_cat!!!", }, - num:{ - type:"string", - defaultValue:"1", + num: { + type: "string", + defaultValue: "1", } }, }, @@ -326,36 +326,36 @@ class WitCatFileHelper { blockType: "reporter", text: this.formatMessage("WitCatFileHelper.numMultiplelinestext"), arguments: { - text:{ - type:"string", - defaultValue:"wit_cat!!!\nwit_cat!!!", + text: { + type: "string", + defaultValue: "wit_cat!!!\nwit_cat!!!", }, }, }, ], - menus:{ + menus: { setvariable: [ { - text: this.formatMessage('WitCatFileHelper.showall'), - value: '#' + text: this.formatMessage('WitCatFileHelper.showall'), + value: '#' }, { - text: this.formatMessage('WitCatFileHelper.showon'), - value: '$' + text: this.formatMessage('WitCatFileHelper.showon'), + value: '$' }, { - text: this.formatMessage('WitCatFileHelper.showoff'), - value: '' + text: this.formatMessage('WitCatFileHelper.showoff'), + value: '' }, ], thing: [ { - text: this.formatMessage('WitCatFileHelper.thing.1'), - value: 'true' + text: this.formatMessage('WitCatFileHelper.thing.1'), + value: 'true' }, { - text: this.formatMessage('WitCatFileHelper.thing.2'), - value: 'false' + text: this.formatMessage('WitCatFileHelper.thing.2'), + value: 'false' }, ], } @@ -366,7 +366,7 @@ class WitCatFileHelper { let h = args.text; let text = args.text; let filenames = args.name; - if(filenames == ""){ + if (filenames == "") { filenames = "none.txt" } const filename = filenames; @@ -399,7 +399,7 @@ class WitCatFileHelper { download(args) { const content = args.text; let filenames = args.name; - if(filenames == ""){ + if (filenames == "") { filenames = "none.txt" } const filename = filenames; @@ -459,9 +459,9 @@ class WitCatFileHelper { } } - if(h == ""){ + if (h == "") { alert("请先保存作品"); - } else{ + } else { localStorage.setItem(show + h + name, text); } } @@ -493,20 +493,23 @@ class WitCatFileHelper { let s = args.s; let array = text.split(s); let r = ""; - if(args.thing == "true"){ + if (args.thing == "true") { const a = `","`; const b = `["`; const c = `"]`; - let str = array.join(a); + let str = ""; + array.forEach(e => { + str = str + JSON.stringify(e).slice(1, - 1) + a; + }); r = b + str + c; } - else{ + else { let i; - for(i = 0;i < array.length;i++){ - if(r == ""){ + for (i = 0; i < array.length; i++) { + if (r == "") { r = array[i]; } - else{ + else { r = r + "\n" + array[i]; } } @@ -528,7 +531,7 @@ class WitCatFileHelper { return jieM; } //打开文件 - openfile(args){ + openfile(args) { FLAG = 1; return new Promise(resolve => { let a = 0; @@ -549,19 +552,19 @@ class WitCatFileHelper { }; reader.readAsText(file); } - window.onfocus = function() { + window.onfocus = function () { // 开始计时或者播放 setTimeout(e => { - if (FLAG == 1) { - FLAG = 0; - resolve(""); - } + if (FLAG == 1) { + FLAG = 0; + resolve(""); + } }, 1000); } }); } //设置状态 - showvar(args){ + showvar(args) { const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 @@ -582,14 +585,14 @@ class WitCatFileHelper { let text = localStorage.getItem(show + h + name); localStorage.removeItem(show + h + name); - if(h == ""){ + if (h == "") { alert("请先保存作品"); - } else{ + } else { localStorage.setItem(args.show + h + name, text); } } //修改别人的键 - saveother(args){ + saveother(args) { let name = args.name; let h = args.id; let text = args.text; @@ -603,7 +606,7 @@ class WitCatFileHelper { if (value == null) { show = "null"; } - else{ + else { show = "$"; } } @@ -612,12 +615,12 @@ class WitCatFileHelper { } } - if(show == "#" || show == "null"){ + if (show == "#" || show == "null") { localStorage.setItem("#" + h + name, text); } } //获取别人的键 - uploadother(args){ + uploadother(args) { let name = args.name; let h = args.id; //寻找状态 @@ -636,12 +639,12 @@ class WitCatFileHelper { } } - if(show != ""){ - return localStorage.getItem(show + h + name); + if (show != "") { + return localStorage.getItem(show + h + name); } } //获取键状态 - other(args){ + other(args) { let name = args.name; let h = args.id; //寻找状态 @@ -654,7 +657,7 @@ class WitCatFileHelper { if (value == null) { show = "null"; } - else{ + else { show = "$"; } } @@ -663,36 +666,36 @@ class WitCatFileHelper { } } - if(show == "#"){ + if (show == "#") { return "公开"; } - if(show == "$"){ + if (show == "$") { return "只读"; } - if(show == ""){ + if (show == "") { return "私有"; } return "键不存在"; } //删除多行文本行 - deleteMultiplelinestext(args){ + deleteMultiplelinestext(args) { let text = args.text; let texts = text.split('\n'); let num = args.num; - if(args.num == "last"){ + if (args.num == "last") { num = texts.length + 1; } - if(args.num == "first"){ + if (args.num == "first") { num = 1; } let a = ""; let i; - for(i = 0;i < texts.length;i++){ - if(i + 1 != num){ - if(a == ""){ + for (i = 0; i < texts.length; i++) { + if (i + 1 != num) { + if (a == "") { a = texts[i]; } - else{ + else { a = a + "\n" + texts[i]; } } @@ -700,32 +703,32 @@ class WitCatFileHelper { return a; } //插入多行文本 - addMultiplelinestext(args){ + addMultiplelinestext(args) { let text = args.text; let texts = args.texts.split("\n"); let num = args.num; - if(args.num == "last"){ + if (args.num == "last") { num = texts.length + 1; } - if(args.num == "first"){ + if (args.num == "first") { num = 1; } let a = ""; - let i ,j = 0; - for(i = 0;i <= texts.length;i++){ - if(i + 1 != num){ - if(a == ""){ + let i, j = 0; + for (i = 0; i <= texts.length; i++) { + if (i + 1 != num) { + if (a == "") { a = texts[i + j]; } - else{ + else { a = a + "\n" + texts[i + j]; } } - else{ - if(a == ""){ + else { + if (a == "") { a = text; } - else{ + else { a = a + "\n" + text; } j = -1; @@ -734,19 +737,19 @@ class WitCatFileHelper { return a; } //多行文本第几行 - whatMultiplelinestext(args){ + whatMultiplelinestext(args) { let text = args.text.split("\n"); let num = args.num; - if(args.num == "last"){ + if (args.num == "last") { num = text.length; } - if(args.num == "first"){ + if (args.num == "first") { num = 1; } return text[num - 1]; } //多行文本行数 - numMultiplelinestext(args){ + numMultiplelinestext(args) { let text = args.text.split("\n"); return text.length; } From 9e6e654eb62aacd1e34d732a23043b5a2363e81a Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 2 Jan 2023 19:08:07 +0800 Subject: [PATCH 202/357] Update Input.js --- wit_cat/Input.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index c73317f8..f7fcad29 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -455,7 +455,7 @@ class WitCatInput { //获取状态 number(args) { let search = document.getElementsByClassName("WitCatInput"); - if (search.length >= args.num) { + if (search.length >= args.num && args.num > 0) { if (args.type == "X") return (search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth; else if (args.type == "Y") From 26cab5a7caec25aac64bc2359cd564a00c5e0d21 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 3 Jan 2023 10:45:41 +0800 Subject: [PATCH 203/357] Update File_Helper.js --- wit_cat/File_Helper.js | 245 ++++++++++++++++++++++++++++++----------- 1 file changed, 179 insertions(+), 66 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 8d06f392..57f3531e 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -38,6 +38,10 @@ class WitCatFileHelper { "WitCatFileHelper.numMultiplelinestext": "[text]的行数", "WitCatFileHelper.thing.1": "数组", "WitCatFileHelper.thing.2": "多行文本", + "WitCatFileHelper.number": "第[num]个键的[type]", + "WitCatFileHelper.numbers": "键数量", + "WitCatFileHelper.number.1": "键名", + "WitCatFileHelper.number.2": "键值", }, en: { "WitCatFileHelper.name": "file helper", @@ -181,6 +185,27 @@ class WitCatFileHelper { }, }, }, + { + opcode: "number", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.number"), + arguments: { + num: { + type: "number", + defaultValue: "i", + }, + type: { + type: "string", + menu: "type", + }, + }, + }, + { + opcode: "numbers", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.numbers"), + arguments: {}, + }, { opcode: "saveother", blockType: "command", @@ -337,15 +362,15 @@ class WitCatFileHelper { setvariable: [ { text: this.formatMessage('WitCatFileHelper.showall'), - value: '#' + value: '#witcat' }, { text: this.formatMessage('WitCatFileHelper.showon'), - value: '$' + value: '$witcat' }, { text: this.formatMessage('WitCatFileHelper.showoff'), - value: '' + value: '@witcat' }, ], thing: [ @@ -358,6 +383,16 @@ class WitCatFileHelper { value: 'false' }, ], + type: [ + { + text: this.formatMessage('WitCatFileHelper.number.1'), + value: 'name' + }, + { + text: this.formatMessage('WitCatFileHelper.number.2'), + value: 'content' + }, + ], } }; } @@ -421,22 +456,30 @@ class WitCatFileHelper { const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 - let show = ""; + let show = "@witcat"; let value = localStorage.getItem(h + name); if (value == null) { - value = localStorage.getItem("#" + h + name); + let value = localStorage.getItem("@witcat" + h + "©" + name); if (value == null) { - value = localStorage.getItem("$" + h + name); - if (value != null) { - show = "$"; + value = localStorage.getItem("#witcat" + h + "©" + name); + if (value == null) { + value = localStorage.getItem("$witcat" + h + "©" + name); + if (value != null) { + show = "$witcat"; + } + } + else { + show = "#witcat"; } - } - else { - show = "#"; } } - - return localStorage.getItem(show + h + name); + else { + show = ""; + } + if (show == "") { + return localStorage.getItem(show + h + name); + } + return localStorage.getItem(show + h + "©" + name); } //保存本地变量 save(args) { @@ -444,25 +487,31 @@ class WitCatFileHelper { const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 - let show = ""; + let show = "@witcat"; let value = localStorage.getItem(h + name); if (value == null) { - value = localStorage.getItem("#" + h + name); + let value = localStorage.getItem("@witcat" + h + "©" + name); if (value == null) { - value = localStorage.getItem("$" + h + name); - if (value != null) { - show = "$"; + value = localStorage.getItem("#witcat" + h + "©" + name); + if (value == null) { + value = localStorage.getItem("$witcat" + h + "©" + name); + if (value != null) { + show = "$witcat"; + } + } + else { + show = "#witcat"; } - } - else { - show = "#"; } } + else { + localStorage.removeItem(h + name); + } if (h == "") { alert("请先保存作品"); } else { - localStorage.setItem(show + h + name, text); + localStorage.setItem(show + h + "©" + name, text); } } //删除本地变量 @@ -470,22 +519,31 @@ class WitCatFileHelper { const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 - let show = ""; + let show = "@witcat"; let value = localStorage.getItem(h + name); if (value == null) { - value = localStorage.getItem("#" + h + name); + let value = localStorage.getItem("@witcat" + h + "©" + name); if (value == null) { - value = localStorage.getItem("$" + h + name); - if (value != null) { - show = "$"; + value = localStorage.getItem("#witcat" + h + "©" + name); + if (value == null) { + value = localStorage.getItem("$witcat" + h + "©" + name); + if (value != null) { + show = "$witcat"; + } + } + else { + show = "#witcat"; } } - else { - show = "#"; - } + } + else { + show = ""; + } + if (show == "") { + localStorage.removeItem(show + h + name); } - localStorage.removeItem(show + h + name); + localStorage.removeItem(show + h + "©" + name); } //字符串分割 segmentation(args) { @@ -568,27 +626,36 @@ class WitCatFileHelper { const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 - let show = ""; + let show = "@witcat"; let value = localStorage.getItem(h + name); if (value == null) { - value = localStorage.getItem("#" + h + name); + let value = localStorage.getItem("@witcat" + h + "©" + name); if (value == null) { - value = localStorage.getItem("$" + h + name); - if (value != null) { - show = "$"; + value = localStorage.getItem("#witcat" + h + "©" + name); + if (value == null) { + value = localStorage.getItem("$witcat" + h + "©" + name); + if (value != null) { + show = "$witcat"; + } + } + else { + show = "#witcat"; } - } - else { - show = "#"; } } - - let text = localStorage.getItem(show + h + name); - localStorage.removeItem(show + h + name); + else { + show = ""; + } + let text = localStorage.getItem(show + h + "©" + name);; + if (show == "") { + text = localStorage.getItem(show + h + name); + localStorage.removeItem(show + h + name); + } + localStorage.removeItem(show + h + "©" + name); if (h == "") { alert("请先保存作品"); } else { - localStorage.setItem(args.show + h + name, text); + localStorage.setItem(args.show + h + "©" + name, text); } } //修改别人的键 @@ -600,23 +667,30 @@ class WitCatFileHelper { let show = ""; let value = localStorage.getItem(h + name); if (value == null) { - value = localStorage.getItem("#" + h + name); + let value = localStorage.getItem("@witcat" + h + "©" + name); if (value == null) { - value = localStorage.getItem("$" + h + name); + value = localStorage.getItem("#witcat" + h + "©" + name); if (value == null) { - show = "null"; + value = localStorage.getItem("$witcat" + h + "©" + name); + if (value != null) { + show = "$witcat"; + } } else { - show = "$"; + show = "#witcat"; } } else { - show = "#"; + show = "@witcat"; } } + else { + localStorage.removeItem(h + name); + show = "@witcat"; + } - if (show == "#" || show == "null") { - localStorage.setItem("#" + h + name, text); + if (show == "#witcat" || show == "") { + localStorage.setItem("#witcat" + h + "©" + name, text); } } //获取别人的键 @@ -627,21 +701,25 @@ class WitCatFileHelper { let show = ""; let value = localStorage.getItem(h + name); if (value == null) { - value = localStorage.getItem("#" + h + name); + let value = localStorage.getItem("@witcat" + h + "©" + name); if (value == null) { - value = localStorage.getItem("$" + h + name); - if (value != null) { - show = "$"; + value = localStorage.getItem("#witcat" + h + "©" + name); + if (value == null) { + value = localStorage.getItem("$witcat" + h + "©" + name); + if (value != null) { + show = "$witcat"; + } + } + else { + show = "#witcat"; } - } - else { - show = "#"; } } if (show != "") { - return localStorage.getItem(show + h + name); + return localStorage.getItem(show + h + "©" + name); } + return ""; } //获取键状态 other(args) { @@ -651,28 +729,34 @@ class WitCatFileHelper { let show = ""; let value = localStorage.getItem(h + name); if (value == null) { - value = localStorage.getItem("#" + h + name); + let value = localStorage.getItem("@witcat" + h + "©" + name); if (value == null) { - value = localStorage.getItem("$" + h + name); + value = localStorage.getItem("#witcat" + h + "©" + name); if (value == null) { - show = "null"; + value = localStorage.getItem("$witcat" + h + "©" + name); + if (value != null) { + show = "$witcat"; + } } else { - show = "$"; + show = "#witcat"; } } else { - show = "#"; + show = "@witcat"; } } + else { + show = "@witcat"; + } - if (show == "#") { + if (show == "#witcat") { return "公开"; } - if (show == "$") { + if (show == "$witcat") { return "只读"; } - if (show == "") { + if (show == "@witcat") { return "私有"; } return "键不存在"; @@ -753,6 +837,35 @@ class WitCatFileHelper { let text = args.text.split("\n"); return text.length; } + //键值对数量 + numbers() { + let a = 0; + for (let i = 0; i < localStorage.length; i++) { + if (localStorage.key(i).slice(1, 7) == "witcat") { + a++; + } + } + return a; + } + //键值对内容 + number(args) { + let a = 0; + let i = 0; + for (i = 0; a < args.num && i < localStorage.length; i++) { + if (localStorage.key(i).slice(1, 7) == "witcat") { + a++; + } + } + if (a == args.num) { + if (args.type == "name") { + return localStorage.key(i - 1).slice(localStorage.key(i - 1).indexOf("©", 7) + 1, localStorage.key(i - 1).length); + } + else { + return localStorage.getItem(localStorage.key(i - 1)); + } + } + return ""; + } } window.tempExt = { From 7bc5ac636d5b2a73e89bec6d84f1f859c80ca0b3 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 3 Jan 2023 11:49:14 +0800 Subject: [PATCH 204/357] Update File_Helper.js --- wit_cat/File_Helper.js | 25 +++++++++++++++++-------- 1 file changed, 17 insertions(+), 8 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 57f3531e..6603761d 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -453,6 +453,7 @@ class WitCatFileHelper { } //读取本地变量 upload(args) { + console.warn("读取键值对:积木已下线,请更换");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 @@ -479,10 +480,11 @@ class WitCatFileHelper { if (show == "") { return localStorage.getItem(show + h + name); } - return localStorage.getItem(show + h + "©" + name); + return localStorage.getItem(show + h + "©" + name);*/ } //保存本地变量 save(args) { + console.warn("保存键值对:积木已下线,请更换");/* const text = args.text; const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); @@ -512,10 +514,11 @@ class WitCatFileHelper { alert("请先保存作品"); } else { localStorage.setItem(show + h + "©" + name, text); - } + }*/ } //删除本地变量 delete(args) { + console.warn("删除键值对:积木已下线,请更换");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 @@ -543,7 +546,7 @@ class WitCatFileHelper { localStorage.removeItem(show + h + name); } - localStorage.removeItem(show + h + "©" + name); + localStorage.removeItem(show + h + "©" + name);*/ } //字符串分割 segmentation(args) { @@ -660,6 +663,7 @@ class WitCatFileHelper { } //修改别人的键 saveother(args) { + console.warn("修改别人的键:积木已下线,请更换");/* let name = args.name; let h = args.id; let text = args.text; @@ -691,10 +695,11 @@ class WitCatFileHelper { if (show == "#witcat" || show == "") { localStorage.setItem("#witcat" + h + "©" + name, text); - } + }*/ } //获取别人的键 uploadother(args) { + console.warn("获取别人的键:积木已下线,请更换");/* let name = args.name; let h = args.id; //寻找状态 @@ -719,10 +724,11 @@ class WitCatFileHelper { if (show != "") { return localStorage.getItem(show + h + "©" + name); } - return ""; + return "";*/ } //获取键状态 other(args) { + console.warn("获取键状态:积木已下线,请更换");/* let name = args.name; let h = args.id; //寻找状态 @@ -759,7 +765,7 @@ class WitCatFileHelper { if (show == "@witcat") { return "私有"; } - return "键不存在"; + return "键不存在";*/ } //删除多行文本行 deleteMultiplelinestext(args) { @@ -839,16 +845,19 @@ class WitCatFileHelper { } //键值对数量 numbers() { + console.warn("获取键值对数量:积木已下线,请更换");/* let a = 0; for (let i = 0; i < localStorage.length; i++) { if (localStorage.key(i).slice(1, 7) == "witcat") { a++; } } - return a; + return a;*/ } //键值对内容 number(args) { + console.warn("获取键值对内容:积木已下线,请更换"); + /* let a = 0; let i = 0; for (i = 0; a < args.num && i < localStorage.length; i++) { @@ -864,7 +873,7 @@ class WitCatFileHelper { return localStorage.getItem(localStorage.key(i - 1)); } } - return ""; + return "";*/ } } From c28ab07cf39c57e90c7ea3f23c3ef1f4e246abdd Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 3 Jan 2023 11:56:59 +0800 Subject: [PATCH 205/357] Update File_Helper.js --- wit_cat/File_Helper.js | 30 ++++++++++++++++++++++-------- 1 file changed, 22 insertions(+), 8 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 6603761d..c06fb99b 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -70,6 +70,10 @@ class WitCatFileHelper { "WitCatFileHelper.numMultiplelinestext": "[text]Number of rows", "WitCatFileHelper.thing.1": "array", "WitCatFileHelper.thing.2": "Multiple lines of text", + "WitCatFileHelper.number": "第[num]个键的[type]", + "WitCatFileHelper.numbers": "键数量", + "WitCatFileHelper.number.1": "键名", + "WitCatFileHelper.number.2": "键值", } }) } @@ -137,6 +141,7 @@ class WitCatFileHelper { opcode: "upload", blockType: "reporter", text: this.formatMessage("WitCatFileHelper.upload"), + hideFromPalette: true, arguments: { name: { type: "string", @@ -148,6 +153,7 @@ class WitCatFileHelper { opcode: "save", blockType: "command", text: this.formatMessage("WitCatFileHelper.save"), + hideFromPalette: true, arguments: { text: { type: "string", @@ -163,6 +169,7 @@ class WitCatFileHelper { opcode: "delete", blockType: "command", text: this.formatMessage("WitCatFileHelper.delete"), + hideFromPalette: true, arguments: { name: { type: "string", @@ -174,6 +181,7 @@ class WitCatFileHelper { opcode: "showvar", blockType: "command", text: this.formatMessage("WitCatFileHelper.showvar"), + hideFromPalette: true, arguments: { name: { type: "string", @@ -189,6 +197,7 @@ class WitCatFileHelper { opcode: "number", blockType: "reporter", text: this.formatMessage("WitCatFileHelper.number"), + hideFromPalette: true, arguments: { num: { type: "number", @@ -204,12 +213,14 @@ class WitCatFileHelper { opcode: "numbers", blockType: "reporter", text: this.formatMessage("WitCatFileHelper.numbers"), + hideFromPalette: true, arguments: {}, }, { opcode: "saveother", blockType: "command", text: this.formatMessage("WitCatFileHelper.saveother"), + hideFromPalette: true, arguments: { id: { type: "string", @@ -229,6 +240,7 @@ class WitCatFileHelper { opcode: "uploadother", blockType: "reporter", text: this.formatMessage("WitCatFileHelper.uploadother"), + hideFromPalette: true, arguments: { id: { type: "string", @@ -244,6 +256,7 @@ class WitCatFileHelper { opcode: "other", blockType: "reporter", text: this.formatMessage("WitCatFileHelper.other"), + hideFromPalette: true, arguments: { id: { type: "string", @@ -255,6 +268,7 @@ class WitCatFileHelper { }, }, }, + "---" + "base64", { opcode: "encrypt", blockType: "reporter", @@ -453,7 +467,7 @@ class WitCatFileHelper { } //读取本地变量 upload(args) { - console.warn("读取键值对:积木已下线,请更换");/* + console.warn("文件助手——读取键值对:积木已下线,请更换");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 @@ -484,7 +498,7 @@ class WitCatFileHelper { } //保存本地变量 save(args) { - console.warn("保存键值对:积木已下线,请更换");/* + console.warn("文件助手——保存键值对:积木已下线,请更换");/* const text = args.text; const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); @@ -518,7 +532,7 @@ class WitCatFileHelper { } //删除本地变量 delete(args) { - console.warn("删除键值对:积木已下线,请更换");/* + console.warn("文件助手——删除键值对:积木已下线,请更换");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 @@ -663,7 +677,7 @@ class WitCatFileHelper { } //修改别人的键 saveother(args) { - console.warn("修改别人的键:积木已下线,请更换");/* + console.warn("文件助手——修改别人的键:积木已下线,请更换");/* let name = args.name; let h = args.id; let text = args.text; @@ -699,7 +713,7 @@ class WitCatFileHelper { } //获取别人的键 uploadother(args) { - console.warn("获取别人的键:积木已下线,请更换");/* + console.warn("文件助手——获取别人的键:积木已下线,请更换");/* let name = args.name; let h = args.id; //寻找状态 @@ -728,7 +742,7 @@ class WitCatFileHelper { } //获取键状态 other(args) { - console.warn("获取键状态:积木已下线,请更换");/* + console.warn("文件助手——获取键状态:积木已下线,请更换");/* let name = args.name; let h = args.id; //寻找状态 @@ -845,7 +859,7 @@ class WitCatFileHelper { } //键值对数量 numbers() { - console.warn("获取键值对数量:积木已下线,请更换");/* + console.warn("文件助手——获取键值对数量:积木已下线,请更换");/* let a = 0; for (let i = 0; i < localStorage.length; i++) { if (localStorage.key(i).slice(1, 7) == "witcat") { @@ -856,7 +870,7 @@ class WitCatFileHelper { } //键值对内容 number(args) { - console.warn("获取键值对内容:积木已下线,请更换"); + console.warn("文件助手——获取键值对内容:积木已下线,请更换"); /* let a = 0; let i = 0; From c357798d129584a9c80d65ee9f02f520c12f187a Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 3 Jan 2023 12:09:43 +0800 Subject: [PATCH 206/357] Update File_Helper.js --- wit_cat/File_Helper.js | 24 ++++++++++++------------ 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index c06fb99b..ffac4273 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -70,10 +70,10 @@ class WitCatFileHelper { "WitCatFileHelper.numMultiplelinestext": "[text]Number of rows", "WitCatFileHelper.thing.1": "array", "WitCatFileHelper.thing.2": "Multiple lines of text", - "WitCatFileHelper.number": "第[num]个键的[type]", - "WitCatFileHelper.numbers": "键数量", - "WitCatFileHelper.number.1": "键名", - "WitCatFileHelper.number.2": "键值", + "WitCatFileHelper.number": "[type]of[num]key", + "WitCatFileHelper.numbers": "key number", + "WitCatFileHelper.number.1": "name", + "WitCatFileHelper.number.2": "content", } }) } @@ -467,7 +467,7 @@ class WitCatFileHelper { } //读取本地变量 upload(args) { - console.warn("文件助手——读取键值对:积木已下线,请更换");/* + console.warn("文件助手——读取键值对:积木已下线,请更换\nfile_helper-Read key value pair: Block is offline, please replace");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 @@ -498,7 +498,7 @@ class WitCatFileHelper { } //保存本地变量 save(args) { - console.warn("文件助手——保存键值对:积木已下线,请更换");/* + console.warn("文件助手——保存键值对:积木已下线,请更换\nfile_helper-Save the key pair: Block is offline, please replace");/* const text = args.text; const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); @@ -532,7 +532,7 @@ class WitCatFileHelper { } //删除本地变量 delete(args) { - console.warn("文件助手——删除键值对:积木已下线,请更换");/* + console.warn("文件助手——删除键值对:积木已下线,请更换\nfile_helper-Delet the key pair: Block is offline, please replace");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 @@ -677,7 +677,7 @@ class WitCatFileHelper { } //修改别人的键 saveother(args) { - console.warn("文件助手——修改别人的键:积木已下线,请更换");/* + console.warn("文件助手——修改别人的键:积木已下线,请更换\nfile_helper-Modify other people's keys: Block is offline, please replace");/* let name = args.name; let h = args.id; let text = args.text; @@ -713,7 +713,7 @@ class WitCatFileHelper { } //获取别人的键 uploadother(args) { - console.warn("文件助手——获取别人的键:积木已下线,请更换");/* + console.warn("文件助手——获取别人的键:积木已下线,请更换\nfile_helper-Get other people's keys: Block is offline, please replace");/* let name = args.name; let h = args.id; //寻找状态 @@ -742,7 +742,7 @@ class WitCatFileHelper { } //获取键状态 other(args) { - console.warn("文件助手——获取键状态:积木已下线,请更换");/* + console.warn("文件助手——获取键状态:积木已下线,请更换\nfile_helper-Get key state: Block is offline, please replace");/* let name = args.name; let h = args.id; //寻找状态 @@ -859,7 +859,7 @@ class WitCatFileHelper { } //键值对数量 numbers() { - console.warn("文件助手——获取键值对数量:积木已下线,请更换");/* + console.warn("文件助手——获取键值对数量:积木已下线,请更换\nfile_helper-number of keys: Block is offline, please replace");/* let a = 0; for (let i = 0; i < localStorage.length; i++) { if (localStorage.key(i).slice(1, 7) == "witcat") { @@ -870,7 +870,7 @@ class WitCatFileHelper { } //键值对内容 number(args) { - console.warn("文件助手——获取键值对内容:积木已下线,请更换"); + console.warn("文件助手——获取键值对内容:积木已下线,请更换\nfile_helper-Get the key pair: Block is offline, please replace"); /* let a = 0; let i = 0; From 75214baa3084257d23ec01f073b01662600bb363 Mon Sep 17 00:00:00 2001 From: nickhu <314484063@qq.com> Date: Wed, 4 Jan 2023 16:05:34 +0800 Subject: [PATCH 207/357] add ignore --- .gitignore | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitignore b/.gitignore index 3c3629e6..76add878 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,2 @@ node_modules +dist \ No newline at end of file From f279b027284277f25dc11141fff65feeda4ac476 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 9 Jan 2023 19:27:26 +0800 Subject: [PATCH 208/357] =?UTF-8?q?=E4=BF=AE=E5=A4=8DBUG=EF=BC=88=E6=96=87?= =?UTF-8?q?=E6=9C=AC=E6=A1=86=EF=BC=8C=E6=96=87=E4=BB=B6=E5=8A=A9=E6=89=8B?= =?UTF-8?q?=EF=BC=89=EF=BC=8C=E5=A2=9E=E5=8A=A0=E5=8A=9F=E8=83=BD=EF=BC=88?= =?UTF-8?q?=E6=96=87=E6=9C=AC=E6=A1=86=EF=BC=89=20(#57)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Update Input.js * Update File_Helper.js * Update More_Mouse.js * 修复BG * 修复多行文本不适配\r的BUG --- wit_cat/File_Helper.js | 20 +- wit_cat/Input.js | 223 ++++++++++++++++++- wit_cat/More_Mouse.js | 494 ++++++++++++++++++++--------------------- 3 files changed, 476 insertions(+), 261 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index ffac4273..a03392ed 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -135,6 +135,7 @@ class WitCatFileHelper { opcode: "openfile", blockType: "reporter", text: this.formatMessage("WitCatFileHelper.openfile"), + disableMonitor: true, arguments: {}, }, { @@ -569,14 +570,7 @@ class WitCatFileHelper { let array = text.split(s); let r = ""; if (args.thing == "true") { - const a = `","`; - const b = `["`; - const c = `"]`; - let str = ""; - array.forEach(e => { - str = str + JSON.stringify(e).slice(1, - 1) + a; - }); - r = b + str + c; + r = JSON.stringify(array); } else { let i; @@ -797,10 +791,10 @@ class WitCatFileHelper { for (i = 0; i < texts.length; i++) { if (i + 1 != num) { if (a == "") { - a = texts[i]; + a = texts[i].split("\r")[0]; } else { - a = a + "\n" + texts[i]; + a = a + "\n" + texts[i].split("\r")[0]; } } } @@ -822,10 +816,10 @@ class WitCatFileHelper { for (i = 0; i <= texts.length; i++) { if (i + 1 != num) { if (a == "") { - a = texts[i + j]; + a = texts[i + j].split("\r")[0]; } else { - a = a + "\n" + texts[i + j]; + a = a + "\n" + texts[i + j].split("\r")[0]; } } else { @@ -850,7 +844,7 @@ class WitCatFileHelper { if (args.num == "first") { num = 1; } - return text[num - 1]; + return text[num - 1].split("\r")[0]; } //多行文本行数 numMultiplelinestext(args) { diff --git a/wit_cat/Input.js b/wit_cat/Input.js index f7fcad29..44c70f9f 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -6,6 +6,10 @@ const _icon = " const extensionId = "WitCatInput"; +let keypress = {}; +let lastKey = "", MouseWheel = 0; +let timer; + //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; if (cvs == null) { @@ -48,6 +52,11 @@ class WitCatInput { "WitCatInput.number.8": "字体大小", "WitCatInput.number.9": "所有(json)", "WitCatInput.number.10": "ID", + "WitCatInput.key": "按下按键[type]?", + "WitCatInput.keys": "按下按键[type]?", + "WitCatInput.lastkey": "上次按下的键", + "WitCatInput.mousewheel": "鼠标滚轮", + "WitCatInput.setinput": "设置ID为[id]的文本框的[type]为[text]", }, en: { "WitCatInput.name": "input", @@ -73,6 +82,11 @@ class WitCatInput { "WitCatInput.number.8": "font-size", "WitCatInput.number.9": "all(json)", "WitCatInput.number.10": "ID", + "WitCatInput.key": "Press the key[type]?", + "WitCatInput.keys": "Press the key [type]?", + "WitCatInput.lastkey": "last key pressed", + "WitCatInput.mousewheel": "MouseWheel", + "WitCatInput.setinput": "Set[type]of input whose ID is[id]to[text]", } }) } @@ -142,6 +156,25 @@ class WitCatInput { } }, }, + { + opcode: "setinput", + blockType: "command", + text: this.formatMessage("WitCatInput.setinput"), + arguments: { + id: { + type: "string", + defaultValue: "i", + }, + type: { + type: "string", + menu: "typess", + }, + text: { + type: "string", + defaultValue: "i", + }, + }, + }, { opcode: "compute", blockType: "reporter", @@ -228,6 +261,42 @@ class WitCatInput { } }, }, + { + opcode: "key", + blockType: "Boolean", + text: this.formatMessage("WitCatInput.key"), + arguments: { + type: { + type: "string", + defaultValue: "a", + } + }, + }, + { + opcode: "mousewheel", + blockType: "reporter", + text: this.formatMessage("WitCatInput.mousewheel"), + arguments: {}, + }, + { + opcode: "keys", + blockType: "hat", + text: this.formatMessage("WitCatInput.keys"), + func: false, + arguments: { + type: { + type: "string", + defaultValue: "a", + } + }, + }, + { + opcode: "lastkey", + blockType: "reporter", + text: this.formatMessage("WitCatInput.lastkey"), + func: false, + arguments: {}, + }, { opcode: "deleteallinput", blockType: "command", @@ -287,6 +356,40 @@ class WitCatInput { text: this.formatMessage('WitCatInput.number.9'), value: 'json' }, + ], + typess: [ + { + text: this.formatMessage('WitCatInput.number.1'), + value: 'X' + }, + { + text: this.formatMessage('WitCatInput.number.2'), + value: 'Y' + }, + { + text: this.formatMessage('WitCatInput.number.3'), + value: 'width' + }, + { + text: this.formatMessage('WitCatInput.number.4'), + value: 'height' + }, + { + text: this.formatMessage('WitCatInput.number.5'), + value: 'content' + }, + { + text: this.formatMessage('WitCatInput.number.6'), + value: 'color' + }, + { + text: this.formatMessage('WitCatInput.number.7'), + value: 'prompt' + }, + { + text: this.formatMessage('WitCatInput.number.8'), + value: 'font-size' + }, ] } }; @@ -427,7 +530,7 @@ class WitCatInput { //焦点位置 whatinput() { if (document.activeElement.className == "WitCatInput") { - return document.activeElement.id.split("WitCatInput"); + return document.activeElement.id.split("WitCatInput")[0]; } else { return ""; @@ -439,6 +542,9 @@ class WitCatInput { if (search != null) { search.focus(); } + else if (document.activeElement.className == "WitCatInput") { + document.activeElement.blur(); + } } //删除所有文本框 deleteallinput(args) { @@ -497,6 +603,94 @@ class WitCatInput { let search = document.getElementsByClassName("WitCatInput"); return search.length; } + //按键检测 + key(args) { + return (args.type in keypress); + } + //按键检测 + keys(args) { + return (args.type in keypress); + } + //上次按下的键 + lastkey() { + return lastKey; + } + //鼠标滚轮 + mousewheel() { + return MouseWheel; + } + //设置文本框 + setinput(args) { + let search = document.getElementById("WitCatInput" + args.id); + if (search != null) { + let x = search.style.left.split("%")[0]; + let y = search.style.top.split("%")[0]; + let width = search.style.width.split("%")[0]; + let height = search.style.height.split("%")[0]; + let content = search.value; + let prompt = search.placeholder; + let color = search.style.color.colorHex() + let size = search.style.fontSize.split("px")[0]; + if (args.type == "X") { + x = args.text; + if (args.text > this.runtime.stageWidth) { + x = this.runtime.stageWidth; + } + if (args.text < 0) { + x = 0; + } + x = (x / this.runtime.stageWidth) * 100; + } + else if (args.type == "Y") { + y = args.text; + if (args.text > this.runtime.stageHeight) { + y = this.runtime.stageHeight; + } + if (args.text < 0) { + y = 0; + } + y = (y / this.runtime.stageHeight) * 100; + } + else if (args.type == "width") { + width = args.text; + if (Number(x) + Number(args.text) > this.runtime.stageWidth) { + width = this.runtime.stageWidth - x; + } + if (args.text < 0) { + width = 0; + } + width = (width / this.runtime.stageWidth) * 100; + } + else if (args.type == "height") { + height = args.text; + if (Number(y) + Number(args.text) > this.runtime.stageHeight) { + height = this.runtime.stageHeight - y; + } + if (args.text < 0) { + height = 0; + } + height = (height / this.runtime.stageHeight) * 100; + } + else if (args.type == "content") { + content = args.text; + } + else if (args.type == "prompt") { + prompt = args.text; + } + else if (args.type == "color") { + color = args.text; + } + else if (args.type == "font-size") { + size = args.text; + } + + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + size + `px;resize:none;color:` + color + `;`; + + search.style = dom; + search.value = content; + search.placeholder = prompt; + } + } } window.tempExt = { @@ -546,3 +740,30 @@ String.prototype.colorHex = function () { return String(color); } }; +//键盘事件监听 +document.onkeydown = function (event) { + keypress[event.key.toLowerCase()] = true; + lastKey = event.key.toLowerCase(); +} +document.onkeyup = function (event) { + delete keypress[event.key.toLowerCase()]; +} +//滚轮事件监听 +var scrollFunc = function (e) { + e = e || window.event; + if (e.wheelDelta) { //判断浏览器IE,谷歌滑轮事件 + MouseWheel = e.wheelDelta; + } else if (e.detail) { //Firefox滑轮事件 + MouseWheel = e.detail; + } + clearTimeout(timer); + timer = setTimeout(function () { + MouseWheel = 0; + }, 30); +}; +//给页面绑定滑轮滚动事件 +if (document.addEventListener) { //火狐使用DOMMouseScroll绑定 + document.addEventListener('DOMMouseScroll', scrollFunc, false); +} +//其他浏览器直接绑定滚动事件 +window.onmousewheel = document.onmousewheel = scrollFunc; diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 5596f387..1e552737 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -5,39 +5,39 @@ const _picture = " const _icon = ""; const extensionId = "WitCatMouse"; -let button = ["up","up","up","up","up"]; +let button = ["up", "up", "up", "up", "up"]; let xMouse = 0; let yMouse = 0; -let isMove = false,timer = null; +let isMove = false, timer = null; let touch = []; let fill = 0; let filln = 0; -let w = 0,h = 0; -let div,divs,divv,divvs,news,background,Operatinginstructions,bug,url,discord; -let wv = 0,hv = 0; +let w = 0, h = 0; +let div, divs, divv, divvs, news, background, Operatinginstructions, bug, url, discord; +let wv = 0, hv = 0; //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; -if(cvs == null){ +if (cvs == null) { alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); } -else{ - for(let i = 1;cvs.className != "" && i <= document.getElementsByTagName("canvas").length;i++){ +else { + for (let i = 1; cvs.className != "" && i <= document.getElementsByTagName("canvas").length; i++) { cvs = document.getElementsByTagName("canvas")[i]; } - if(cvs == null){ + if (cvs == null) { alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); } } //添加监听器 -const config = { attributes: true, childList: true, subtree: true, attributeFilter:['style'] }; -const callback = function(mutationsList, observer) { - if(fill == 1){ - observer.disconnect(); - fills(); - observer.observe(cvs, config); - } +const config = { attributes: true, childList: true, subtree: true, attributeFilter: ['style'] }; +const callback = function (mutationsList, observer) { + if (fill == 1) { + observer.disconnect(); + fills(); + observer.observe(cvs, config); + } }; const observer = new MutationObserver(callback); observer.observe(cvs, config); @@ -48,10 +48,10 @@ class WitCatMouse { this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "WitCatMouse.name": "高级工具", + "WitCatMouse.name": "高级操作", "WitCatMouse.name.1": "高级鼠标", "WitCatMouse.name.2": "多指触控", - "WitCatMouse.set": "[set]右键菜单", + "WitCatMouse.set": "[set]右键菜单", "WitCatMouse.set.1": "启用", "WitCatMouse.set.2": "禁用", "WitCatMouse.when": "鼠标[key]被按下", @@ -75,10 +75,10 @@ class WitCatMouse { "WitCatMouse.resolution": "当前分辨率高", }, en: { - "WitCatMouse.name": "More tools", + "WitCatMouse.name": "More operate", "WitCatMouse.name.1": "more mouse", "WitCatMouse.name.2": "more touch", - "WitCatMouse.set": "[set]Right-click menu", + "WitCatMouse.set": "[set]Right-click menu", "WitCatMouse.set.1": "allow", "WitCatMouse.set.2": "forbid", "WitCatMouse.when": "mouse[key]down?", @@ -128,9 +128,9 @@ class WitCatMouse { blockType: "command", text: this.formatMessage("WitCatMouse.setfill"), arguments: { - num:{ - type:"number", - defaultValue:"360", + num: { + type: "number", + defaultValue: "360", }, } }, @@ -139,9 +139,9 @@ class WitCatMouse { blockType: "command", text: this.formatMessage("WitCatMouse.fill"), arguments: { - set:{ - type:"bool", - menu:"set", + set: { + type: "bool", + menu: "set", }, } }, @@ -151,15 +151,15 @@ class WitCatMouse { text: this.formatMessage("WitCatMouse.resolution"), arguments: {} }, - "---"+this.formatMessage("WitCatMouse.name.1"), + "---" + this.formatMessage("WitCatMouse.name.1"), { opcode: 'set', blockType: "command", text: this.formatMessage("WitCatMouse.set"), arguments: { - set:{ - type:"bool", - menu:"set", + set: { + type: "bool", + menu: "set", }, } }, @@ -168,9 +168,9 @@ class WitCatMouse { blockType: "Boolean", text: this.formatMessage("WitCatMouse.when"), arguments: { - key:{ - type:"string", - menu:"key", + key: { + type: "string", + menu: "key", }, }, }, @@ -185,13 +185,13 @@ class WitCatMouse { blockType: "reporter", text: this.formatMessage("WitCatMouse.acceleration"), arguments: { - way:{ - type:"string", - menu:"way", + way: { + type: "string", + menu: "way", }, }, }, - "---"+this.formatMessage("WitCatMouse.name.2"), + "---" + this.formatMessage("WitCatMouse.name.2"), { opcode: 'down', blockType: "reporter", @@ -203,156 +203,156 @@ class WitCatMouse { blockType: "reporter", text: this.formatMessage("WitCatTouch.num"), arguments: { - num:{ - type:"number", - defaultValue:"1", + num: { + type: "number", + defaultValue: "1", }, - type:{ - type:"string", - menu:"type", + type: { + type: "string", + menu: "type", }, }, }, ], - menus:{ + menus: { key: [ { - text: this.formatMessage('WitCatMouse.key.1'), - value: '0' + text: this.formatMessage('WitCatMouse.key.1'), + value: '0' }, { - text: this.formatMessage('WitCatMouse.key.2'), - value: '1' + text: this.formatMessage('WitCatMouse.key.2'), + value: '1' }, { - text: this.formatMessage('WitCatMouse.key.3'), - value: '2' + text: this.formatMessage('WitCatMouse.key.3'), + value: '2' }, { - text: this.formatMessage('WitCatMouse.key.4'), - value: '3' + text: this.formatMessage('WitCatMouse.key.4'), + value: '3' }, { - text: this.formatMessage('WitCatMouse.key.5'), - value: '4' + text: this.formatMessage('WitCatMouse.key.5'), + value: '4' }, ], set: [ { - text: this.formatMessage('WitCatMouse.set.1'), - value: "true" + text: this.formatMessage('WitCatMouse.set.1'), + value: "true" }, { - text: this.formatMessage('WitCatMouse.set.2'), - value: "false" + text: this.formatMessage('WitCatMouse.set.2'), + value: "false" }, ], way: [ { - text: this.formatMessage('WitCatMouse.way.1'), - value: "x" + text: this.formatMessage('WitCatMouse.way.1'), + value: "x" }, { - text: this.formatMessage('WitCatMouse.way.2'), - value: "y" + text: this.formatMessage('WitCatMouse.way.2'), + value: "y" }, ], type: [ { - text: this.formatMessage('WitCatTouch.type.1'), - value: "x" + text: this.formatMessage('WitCatTouch.type.1'), + value: "x" }, { - text: this.formatMessage('WitCatTouch.type.2'), - value: "y" + text: this.formatMessage('WitCatTouch.type.2'), + value: "y" }, ], } }; } //右键菜单 - set(args){ + set(args) { history.pushState(null, null, null); - cvs.parentNode.oncontextmenu = function (){ - if(args.set == "true"){ + cvs.parentNode.oncontextmenu = function () { + if (args.set == "true") { return true; } - else{ + else { return false; } } } //按下判断 - when(args){ - if(button[args.key] == "down"){ + when(args) { + if (button[args.key] == "down") { return true; } - else{ + else { return false; } } //控制鼠标 - mouseuse(){ + mouseuse() { cvs.parentNode.requestPointerLock(); } //鼠标移动量 - acceleration(args){ - if(args.way == "x"){ + acceleration(args) { + if (args.way == "x") { return xMouse; } - else{ + else { return -yMouse; } } //数量 - down(){ + down() { return touch.length; } //坐标 - num(args){ - if(args.num > 0 && args.num <= touch.length){ - if(args.type == "x"){ + num(args) { + if (args.num > 0 && args.num <= touch.length) { + if (args.type == "x") { return this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.offsetWidth); } - else{ + else { return this.runtime.stageHeight * ((touch[args.num - 1].clientY - cvs.getBoundingClientRect().top) / cvs.offsetHeight); } } - else{ + else { return null; } } //全屏 - fill(args){ - if(args.set == "true"){ - if(filln < 3){ - if(fill == 0){ + fill(args) { + if (args.set == "true") { + if (filln < 3) { + if (fill == 0) { let a = confirm(this.formatMessage('WitCatMouse.fillask.1') + filln + this.formatMessage('WitCatMouse.fillask.2')); - if(a == true){ + if (a == true) { filln = 0; fillin(); } - else{ + else { filln += 1; } } } } - else{ - if(fill == 1){ + else { + if (fill == 1) { outoffill(); } } } //设置分辨率 - setfill(args){ + setfill(args) { w = args.num / this.runtime.stageHeight * this.runtime.stageWidth; h = args.num; - cvs.width = args.num / this.runtime.stageHeight * this.runtime.stageWidth; + cvs.width = args.num / this.runtime.stageHeight * this.runtime.stageWidth; cvs.height = args.num; } //当前分辨率 - resolution(args){ + resolution(args) { return cvs.height; } } @@ -371,205 +371,205 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatMouse.name": "高级工具", + "WitCatMouse.name": "高级鼠标", "WitCatMouse.descp": "更精准的控制鼠标/触屏/全屏!" }, en: { - "WitCatMouse.name": "More tools", + "WitCatMouse.name": "More Mouse", "WitCatMouse.descp": "More precise mouse/touch/full screen control!" } } }; /* vim: set expandtab tabstop=2 shiftwidth=2: */ - -document.onmousedown = function(event) { + +document.onmousedown = function (event) { button[event.button] = "down"; } -document.onmouseup = function(event) { +document.onmouseup = function (event) { button[event.button] = "up"; } document.addEventListener("mousemove", ev => { - xMouse = ev.movementX; // 获得鼠标指针的x移动量 - yMouse = ev.movementY; // 获得鼠标指针的y移动量 + xMouse = ev.movementX; // 获得鼠标指针的x移动量 + yMouse = ev.movementY; // 获得鼠标指针的y移动量 isMove = true; clearTimeout(timer); - timer = setTimeout(function(){ + timer = setTimeout(function () { isMove = false; xMouse = 0; yMouse = 0; - },30); + }, 30); }); //多指触控 -cvs.addEventListener('touchstart',function(e){ +cvs.addEventListener('touchstart', function (e) { touch = e.targetTouches; }) -cvs.addEventListener('touchmove',function(e){ +cvs.addEventListener('touchmove', function (e) { touch = e.targetTouches; }) -cvs.addEventListener('touchend',function(e){ +cvs.addEventListener('touchend', function (e) { touch = e.targetTouches; }) //大小改变刷新 -window.onresize=function(){ - if(fill == 1){ - fills(); - } +window.onresize = function () { + if (fill == 1) { + fills(); + } } //开启全屏 -function fillin(){ - fill = 1; +function fillin() { + fill = 1; - var mo=function(e){e.preventDefault();}; - document.body.style.overflow='hidden'; - document.addEventListener("touchmove",mo,false);//禁止页面滑动 - if(window.location.href.split("/")[2] != "cocrea.world"){ - div = document.getElementById("root").getElementsByTagName('div')[0]; - divs = document.getElementById("root"); - url = window.location.href.split("/")[3].split("?")[0]; - if(url != "scratch-player"){ - divs.removeChild(div); - } - } + var mo = function (e) { e.preventDefault(); }; + document.body.style.overflow = 'hidden'; + document.addEventListener("touchmove", mo, false);//禁止页面滑动 + if (window.location.href.split("/")[2] != "cocrea.world") { + div = document.getElementById("root").getElementsByTagName('div')[0]; + divs = document.getElementById("root"); + url = window.location.href.split("/")[3].split("?")[0]; + if (url != "scratch-player") { + divs.removeChild(div); + } + } - news = document.body.getElementsByClassName("convention-1wIbd")[0]; - if(typeof(news) != "undefined"){ - news.style = "display:none;"; - } - divv = document.body.getElementsByClassName("actions-2lk9z")[0]; - if(typeof(divv) != "undefined"){ - divv.style = "display:none;"; - } - Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; - if(typeof(Operatinginstructions) != "undefined"){ - Operatinginstructions.style = "display:none;"; - } - bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; - if(typeof(bug) != "undefined"){ - bug.style = "display:none;"; - } - discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; - if(typeof(discord) != "undefined"){ - discord.style = "display:none"; - } - document.body.scrollTop = 0; - document.documentElement.scrollTop = 0; - scrollTo(0,0); - setTimeout(function() { - fills(); - }, 50); + news = document.body.getElementsByClassName("convention-1wIbd")[0]; + if (typeof (news) != "undefined") { + news.style = "display:none;"; + } + divv = document.body.getElementsByClassName("actions-2lk9z")[0]; + if (typeof (divv) != "undefined") { + divv.style = "display:none;"; + } + Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; + if (typeof (Operatinginstructions) != "undefined") { + Operatinginstructions.style = "display:none;"; + } + bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; + if (typeof (bug) != "undefined") { + bug.style = "display:none;"; + } + discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; + if (typeof (discord) != "undefined") { + discord.style = "display:none"; + } + document.body.scrollTop = 0; + document.documentElement.scrollTop = 0; + scrollTo(0, 0); + setTimeout(function () { + fills(); + }, 50); } //关闭全屏 -function outoffill(){ - //恢复头 - fill = 0; - var mo=function(e){e.preventDefault();}; - document.body.style.overflow='auto'; - document.addEventListener("touchmove",mo,true);//允许页面滑动 - if(window.location.href.split("/")[2] != "cocrea.world"){ - divs = document.getElementById("root"); - url = window.location.href.split("/")[3].split("?")[0]; - if(url != "scratch-player"){ - divs.insertBefore(div, divs.children[0]); - } - } - document.body.removeChild(background); - news = document.body.getElementsByClassName("convention-1wIbd")[0]; - if(typeof(news) != "undefined"){ - news.style = ""; - } - divv = document.body.getElementsByClassName("actions-2lk9z")[0]; - if(typeof(divv) != "undefined"){ - divv.style = ""; - } - divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; - if(typeof(divvs) != "undefined"){ - divvs.style = ""; - } - Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; - if(typeof(Operatinginstructions) != "undefined"){ - Operatinginstructions.style = ""; - } - bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; - if(typeof(bug) != "undefined"){ - bug.style = ""; - } - discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; - if(typeof(discord) != "undefined"){ - discord.style = ""; - } - cvs.parentNode.parentNode.parentNode.parentNode.style = "width:100%; height:100%;z-index: 1000000;"; +function outoffill() { + //恢复头 + fill = 0; + var mo = function (e) { e.preventDefault(); }; + document.body.style.overflow = 'auto'; + document.addEventListener("touchmove", mo, true);//允许页面滑动 + if (window.location.href.split("/")[2] != "cocrea.world") { + divs = document.getElementById("root"); + url = window.location.href.split("/")[3].split("?")[0]; + if (url != "scratch-player") { + divs.insertBefore(div, divs.children[0]); + } + } + document.body.removeChild(background); + news = document.body.getElementsByClassName("convention-1wIbd")[0]; + if (typeof (news) != "undefined") { + news.style = ""; + } + divv = document.body.getElementsByClassName("actions-2lk9z")[0]; + if (typeof (divv) != "undefined") { + divv.style = ""; + } + divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; + if (typeof (divvs) != "undefined") { + divvs.style = ""; + } + Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; + if (typeof (Operatinginstructions) != "undefined") { + Operatinginstructions.style = ""; + } + bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; + if (typeof (bug) != "undefined") { + bug.style = ""; + } + discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; + if (typeof (discord) != "undefined") { + discord.style = ""; + } + cvs.parentNode.parentNode.parentNode.parentNode.style = "width:100%; height:100%;z-index: 1000000;"; cvs.parentNode.parentNode.style = "height: 100%; width: 100%;"; cvs.parentNode.style = ""; cvs.style = "height: 100%; width: 100%;"; } //循环检测 -function fills(){ - if(w != 0 && h != 0){ - cvs.width = w; - cvs.height = h; - } - let ws = 0,hs = 0; - if((cvs.width / cvs.height) >= (document.documentElement.clientWidth / document.documentElement.clientHeight)){ - hs = (document.documentElement.clientWidth / cvs.width) * cvs.height; - ws = document.documentElement.clientWidth; - wv = 0; - hv = (document.documentElement.clientHeight - hs) / 2; - } - else{ - hs = document.documentElement.clientHeight; - ws = (document.documentElement.clientHeight / cvs.height) * cvs.width; - wv = (document.documentElement.clientWidth - ws) / 2; - hv = 0; - } - divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; - if(typeof(divvs) != "undefined"){ - divvs.style = "display:none;"; - } - cvs.parentNode.parentNode.parentNode.parentNode.style = "position:fixed; left:" + wv + "px; top:" + hv + "px; width:" + ws + "px; height:" + hs + "px;z-index: 1000000;border-radius:0px"; - cvs.parentNode.parentNode.parentNode.style = "width:" + ws + "px; height:" + hs + "px;border-radius:0px"; - cvs.parentNode.parentNode.style = "width:100%; height:100%;border-radius:0px"; - cvs.parentNode.style = "width:100%; height:100%;border-radius:0px"; - if(isInPage(cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild)){ - cvs.parentNode.parentNode.nextElementSibling.style="width: 100%; height: 100%;"; - cvs.parentNode.parentNode.nextElementSibling.firstChild.style="width: 100%; height: 100%;"; - cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.firstChild.style="width: 100%; height: 100%;"; - cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.style="width: 100%;"; - } - cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("div")[0].style = "width:100%; height:100%;" - cvs.style = "width:100%; height:100%;"; - setTimeout(function() { - let a = cvs.offsetWidth / document.documentElement.clientWidth; - cvs.parentNode.parentNode.getElementsByTagName("div")[1].style="transform:scale(" + a + ");transform-origin:0% 0% 0"; - }, 20); - background = document.getElementsByClassName("witcatbackground")[0]; - if(background == null){ - background = document.createElement("div"); - background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; - background.className = "witcatbackground"; - document.body.insertBefore(background, document.body.children[0]); - } - else{ - background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; - } +function fills() { + if (w != 0 && h != 0) { + cvs.width = w; + cvs.height = h; + } + let ws = 0, hs = 0; + if ((cvs.width / cvs.height) >= (document.documentElement.clientWidth / document.documentElement.clientHeight)) { + hs = (document.documentElement.clientWidth / cvs.width) * cvs.height; + ws = document.documentElement.clientWidth; + wv = 0; + hv = (document.documentElement.clientHeight - hs) / 2; + } + else { + hs = document.documentElement.clientHeight; + ws = (document.documentElement.clientHeight / cvs.height) * cvs.width; + wv = (document.documentElement.clientWidth - ws) / 2; + hv = 0; + } + divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; + if (typeof (divvs) != "undefined") { + divvs.style = "display:none;"; + } + cvs.parentNode.parentNode.parentNode.parentNode.style = "position:fixed; left:" + wv + "px; top:" + hv + "px; width:" + ws + "px; height:" + hs + "px;z-index: 1000000;border-radius:0px"; + cvs.parentNode.parentNode.parentNode.style = "width:" + ws + "px; height:" + hs + "px;border-radius:0px"; + cvs.parentNode.parentNode.style = "width:100%; height:100%;border-radius:0px"; + cvs.parentNode.style = "width:100%; height:100%;border-radius:0px"; + if (isInPage(cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild)) { + cvs.parentNode.parentNode.nextElementSibling.style = "width: 100%; height: 100%;"; + cvs.parentNode.parentNode.nextElementSibling.firstChild.style = "width: 100%; height: 100%;"; + cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.firstChild.style = "width: 100%; height: 100%;"; + cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.style = "width: 100%;"; + } + cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("div")[0].style = "width:100%; height:100%;" + cvs.style = "width:100%; height:100%;"; + setTimeout(function () { + let a = cvs.offsetWidth / document.documentElement.clientWidth; + cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(" + a + ");transform-origin:0% 0% 0"; + }, 20); + background = document.getElementsByClassName("witcatbackground")[0]; + if (background == null) { + background = document.createElement("div"); + background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; + background.className = "witcatbackground"; + document.body.insertBefore(background, document.body.children[0]); + } + else { + background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; + } } //检测div存在 function isInPage(node) { - return (node === document.body) ? false : document.body.contains(node); + return (node === document.body) ? false : document.body.contains(node); } - + //按键检测 -document.onkeydown = function(e) { - var evt = e || window.event; - if (evt.ctrlKey && evt.shiftKey && evt.altKey){ - if(fill == 0){ - fillin(); - } - else{ - outoffill(); - } +document.onkeydown = function (e) { + var evt = e || window.event; + if (evt.ctrlKey && evt.shiftKey && evt.altKey) { + if (fill == 0) { + fillin(); + } + else { + outoffill(); + } } } From a0eab52319d5e2297a45cd150a70b792d90bc0c0 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Fri, 13 Jan 2023 13:20:42 +0800 Subject: [PATCH 209/357] =?UTF-8?q?=E6=9B=B4=E6=96=B0=EF=BC=8C=E4=BC=98?= =?UTF-8?q?=E5=8C=96=20(#58)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * 更新下载限制,更新三个积木,优化效率 feat(compile):增加下载文件限制 feat(block):增加多行文本与数组转换积木,增加可下载数量积木 perf(block):优化多行文本处理效率 style:将"=="更改为"===" * 优化事件绑定 style:将"=="改为"===" perf(compile):滑轮事件绑定至cvs.parentNode * 更新检测设备支持积木,抽象化检测模式 feat(block):更新设备检测,设备支持积木 perf(block):抽象检测模式,以便开发者进行开发 style:将"=="改为"===" * 修复按键检测无效的BUG,优化鼠标滚轮检测范围 fix(block):修复按键检测无效的BUG perf(compile):优化鼠标滚轮检测范围,从dom到scratch舞台 * 优化了事件绑定位置 perf(compile):优化了事件绑定至dom,改为AddListener --- wit_cat/File_Helper.js | 322 +++++++++++++++++++++++------------------ wit_cat/Input.js | 105 +++++++------- wit_cat/More_Mouse.js | 131 ++++++++++++----- 3 files changed, 332 insertions(+), 226 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index a03392ed..1fe82a4a 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -5,7 +5,13 @@ const _picture = " const _icon = ""; const extensionId = "WitCatFileHelper"; -let FLAG = 0; +let FLAG = 0, download = 0; + +setInterval(function () { + if (download > 0) { + download--; + } +}, 1000) class WitCatFileHelper { constructor(runtime) { @@ -42,6 +48,9 @@ class WitCatFileHelper { "WitCatFileHelper.numbers": "键数量", "WitCatFileHelper.number.1": "键名", "WitCatFileHelper.number.2": "键值", + "WitCatFileHelper.turnMultiplelinestext": "将多行文本[text]转化为数组", + "WitCatFileHelper.turnsMultiplelinestext": "将数组[text]转化为多行文本", + "WitCatFileHelper.downloadnum": "可下载文件数量", }, en: { "WitCatFileHelper.name": "file helper", @@ -74,6 +83,9 @@ class WitCatFileHelper { "WitCatFileHelper.numbers": "key number", "WitCatFileHelper.number.1": "name", "WitCatFileHelper.number.2": "content", + "WitCatFileHelper.turnMultiplelinestext": "Converts multiline[text]text to an array", + "WitCatFileHelper.turnsMultiplelinestext": "Converts the array[text]to multiline text", + "WitCatFileHelper.downloadnum": "Number of downloadable files", } }) } @@ -131,6 +143,12 @@ class WitCatFileHelper { }, }, }, + { + opcode: "downloadnum", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.downloadnum"), + arguments: {}, + }, { opcode: "openfile", blockType: "reporter", @@ -372,6 +390,28 @@ class WitCatFileHelper { }, }, }, + { + opcode: "turnMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.turnMultiplelinestext"), + arguments: { + text: { + type: "string", + defaultValue: "wit_cat!!!\nwit_cat!!!", + }, + }, + }, + { + opcode: "turnsMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.turnsMultiplelinestext"), + arguments: { + text: { + type: "string", + defaultValue: "[\"wit_cat!!!\",\"wit_cat!!!\"]", + }, + }, + }, ], menus: { setvariable: [ @@ -413,58 +453,70 @@ class WitCatFileHelper { } //下载多行文件 downloads(args) { - let h = args.text; - let text = args.text; - let filenames = args.name; - if (filenames == "") { - filenames = "none.txt" - } - const filename = filenames; - let s = args.s; - let j = 0; - if (s != "") { - let a = text.split(s); - h = a[0]; - for (j = 1; j < a.length; j++) { - h = h + "\n" + a[j]; + download += 1; + if (download < 3) { + let h = args.text; + let text = args.text; + let filenames = args.name; + if (filenames === "") { + filenames = "none.txt" } - } else { - h = text; - } - const content = h; - // 创建隐藏的可下载链接 - let eleLink = document.createElement('a'); - eleLink.download = filename; - eleLink.style.display = 'none'; - // 字符内容转变成blob地址 - let blob = new Blob([content]); - eleLink.href = URL.createObjectURL(blob); - // 触发点击 - document.body.appendChild(eleLink); - eleLink.click(); - // 然后移除 - document.body.removeChild(eleLink); + const filename = filenames; + let s = args.s; + let j = 0; + if (s != "") { + let a = text.split(s); + h = a[0]; + for (j = 1; j < a.length; j++) { + h = h + "\n" + a[j]; + } + } else { + h = text; + } + const content = h; + // 创建隐藏的可下载链接 + let eleLink = document.createElement('a'); + eleLink.download = filename; + eleLink.style.display = 'none'; + // 字符内容转变成blob地址 + let blob = new Blob([content]); + eleLink.href = URL.createObjectURL(blob); + // 触发点击 + document.body.appendChild(eleLink); + eleLink.click(); + // 然后移除 + document.body.removeChild(eleLink); + } + else { + console.warn("下载太频繁!\nToo many downloads!"); + } } //下载文件 download(args) { - const content = args.text; - let filenames = args.name; - if (filenames == "") { - filenames = "none.txt" - } - const filename = filenames; - // 创建隐藏的可下载链接 - let eleLink = document.createElement('a'); - eleLink.download = filename; - eleLink.style.display = 'none'; - // 字符内容转变成blob地址 - let blob = new Blob([content]); - eleLink.href = URL.createObjectURL(blob); - // 触发点击 - document.body.appendChild(eleLink); - eleLink.click(); - // 然后移除 - document.body.removeChild(eleLink); + download += 1; + if (download < 3) { + const content = args.text; + let filenames = args.name; + if (filenames === "") { + filenames = "none.txt" + } + const filename = filenames; + // 创建隐藏的可下载链接 + let eleLink = document.createElement('a'); + eleLink.download = filename; + eleLink.style.display = 'none'; + // 字符内容转变成blob地址 + let blob = new Blob([content]); + eleLink.href = URL.createObjectURL(blob); + // 触发点击 + document.body.appendChild(eleLink); + eleLink.click(); + // 然后移除 + document.body.removeChild(eleLink); + } + else { + console.warn("下载太频繁!\nToo many downloads!"); + } } //读取本地变量 upload(args) { @@ -474,11 +526,11 @@ class WitCatFileHelper { //寻找状态 let show = "@witcat"; let value = localStorage.getItem(h + name); - if (value == null) { + if (value === null) { let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("#witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("$witcat" + h + "©" + name); if (value != null) { show = "$witcat"; @@ -492,7 +544,7 @@ class WitCatFileHelper { else { show = ""; } - if (show == "") { + if (show === "") { return localStorage.getItem(show + h + name); } return localStorage.getItem(show + h + "©" + name);*/ @@ -506,11 +558,11 @@ class WitCatFileHelper { //寻找状态 let show = "@witcat"; let value = localStorage.getItem(h + name); - if (value == null) { + if (value === null) { let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("#witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("$witcat" + h + "©" + name); if (value != null) { show = "$witcat"; @@ -525,7 +577,7 @@ class WitCatFileHelper { localStorage.removeItem(h + name); } - if (h == "") { + if (h === "") { alert("请先保存作品"); } else { localStorage.setItem(show + h + "©" + name, text); @@ -539,11 +591,11 @@ class WitCatFileHelper { //寻找状态 let show = "@witcat"; let value = localStorage.getItem(h + name); - if (value == null) { + if (value === null) { let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("#witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("$witcat" + h + "©" + name); if (value != null) { show = "$witcat"; @@ -557,7 +609,7 @@ class WitCatFileHelper { else { show = ""; } - if (show == "") { + if (show === "") { localStorage.removeItem(show + h + name); } @@ -569,19 +621,11 @@ class WitCatFileHelper { let s = args.s; let array = text.split(s); let r = ""; - if (args.thing == "true") { + if (args.thing === "true") { r = JSON.stringify(array); } else { - let i; - for (i = 0; i < array.length; i++) { - if (r == "") { - r = array[i]; - } - else { - r = r + "\n" + array[i]; - } - } + r = array.join("\n"); } return r; } @@ -624,7 +668,7 @@ class WitCatFileHelper { window.onfocus = function () { // 开始计时或者播放 setTimeout(e => { - if (FLAG == 1) { + if (FLAG === 1) { FLAG = 0; resolve(""); } @@ -634,16 +678,17 @@ class WitCatFileHelper { } //设置状态 showvar(args) { + console.warn("文件助手——设置键状态:积木已下线,请更换\nfile_helper-Set key state: Block is offline, please replace");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); //寻找状态 let show = "@witcat"; let value = localStorage.getItem(h + name); - if (value == null) { + if (value === null) { let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("#witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("$witcat" + h + "©" + name); if (value != null) { show = "$witcat"; @@ -658,16 +703,16 @@ class WitCatFileHelper { show = ""; } let text = localStorage.getItem(show + h + "©" + name);; - if (show == "") { + if (show === "") { text = localStorage.getItem(show + h + name); localStorage.removeItem(show + h + name); } localStorage.removeItem(show + h + "©" + name); - if (h == "") { + if (h === "") { alert("请先保存作品"); } else { localStorage.setItem(args.show + h + "©" + name, text); - } + }*/ } //修改别人的键 saveother(args) { @@ -678,11 +723,11 @@ class WitCatFileHelper { //寻找状态 let show = ""; let value = localStorage.getItem(h + name); - if (value == null) { + if (value === null) { let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("#witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("$witcat" + h + "©" + name); if (value != null) { show = "$witcat"; @@ -701,7 +746,7 @@ class WitCatFileHelper { show = "@witcat"; } - if (show == "#witcat" || show == "") { + if (show === "#witcat" || show === "") { localStorage.setItem("#witcat" + h + "©" + name, text); }*/ } @@ -713,11 +758,11 @@ class WitCatFileHelper { //寻找状态 let show = ""; let value = localStorage.getItem(h + name); - if (value == null) { + if (value === null) { let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("#witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("$witcat" + h + "©" + name); if (value != null) { show = "$witcat"; @@ -742,11 +787,11 @@ class WitCatFileHelper { //寻找状态 let show = ""; let value = localStorage.getItem(h + name); - if (value == null) { + if (value === null) { let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("#witcat" + h + "©" + name); - if (value == null) { + if (value === null) { value = localStorage.getItem("$witcat" + h + "©" + name); if (value != null) { show = "$witcat"; @@ -764,84 +809,51 @@ class WitCatFileHelper { show = "@witcat"; } - if (show == "#witcat") { + if (show === "#witcat") { return "公开"; } - if (show == "$witcat") { + if (show === "$witcat") { return "只读"; } - if (show == "@witcat") { + if (show === "@witcat") { return "私有"; } return "键不存在";*/ } //删除多行文本行 deleteMultiplelinestext(args) { - let text = args.text; - let texts = text.split('\n'); + let texts = multipleText(args.text); let num = args.num; - if (args.num == "last") { - num = texts.length + 1; + if (args.num === "last") { + num = texts.length; } - if (args.num == "first") { + if (args.num === "first") { num = 1; } - let a = ""; - let i; - for (i = 0; i < texts.length; i++) { - if (i + 1 != num) { - if (a == "") { - a = texts[i].split("\r")[0]; - } - else { - a = a + "\n" + texts[i].split("\r")[0]; - } - } - } - return a; + texts.splice(num - 1, 1); + return texts.join("\n"); } //插入多行文本 addMultiplelinestext(args) { - let text = args.text; - let texts = args.texts.split("\n"); + let texts = multipleText(args.texts); let num = args.num; - if (args.num == "last") { + if (args.num === "last") { num = texts.length + 1; } - if (args.num == "first") { + if (args.num === "first") { num = 1; } - let a = ""; - let i, j = 0; - for (i = 0; i <= texts.length; i++) { - if (i + 1 != num) { - if (a == "") { - a = texts[i + j].split("\r")[0]; - } - else { - a = a + "\n" + texts[i + j].split("\r")[0]; - } - } - else { - if (a == "") { - a = text; - } - else { - a = a + "\n" + text; - } - j = -1; - } - } - return a; + texts.splice(num - 1, 0, args.text); + return texts.join("\n"); } //多行文本第几行 whatMultiplelinestext(args) { let text = args.text.split("\n"); let num = args.num; - if (args.num == "last") { + if (args.num === "last") { num = text.length; } - if (args.num == "first") { + if (args.num === "first") { num = 1; } return text[num - 1].split("\r")[0]; @@ -851,12 +863,22 @@ class WitCatFileHelper { let text = args.text.split("\n"); return text.length; } + //多行文本转数组 + turnMultiplelinestext(args) { + let texts = multipleText(args.text); + return JSON.stringify(texts); + } + //数组转多行文本 + turnsMultiplelinestext(args) { + let texts = JSON.parse(args.text); + return texts.join("\n"); + } //键值对数量 numbers() { console.warn("文件助手——获取键值对数量:积木已下线,请更换\nfile_helper-number of keys: Block is offline, please replace");/* let a = 0; for (let i = 0; i < localStorage.length; i++) { - if (localStorage.key(i).slice(1, 7) == "witcat") { + if (localStorage.key(i).slice(1, 7) === "witcat") { a++; } } @@ -869,12 +891,12 @@ class WitCatFileHelper { let a = 0; let i = 0; for (i = 0; a < args.num && i < localStorage.length; i++) { - if (localStorage.key(i).slice(1, 7) == "witcat") { + if (localStorage.key(i).slice(1, 7) === "witcat") { a++; } } - if (a == args.num) { - if (args.type == "name") { + if (a === args.num) { + if (args.type === "name") { return localStorage.key(i - 1).slice(localStorage.key(i - 1).indexOf("©", 7) + 1, localStorage.key(i - 1).length); } else { @@ -883,6 +905,10 @@ class WitCatFileHelper { } return "";*/ } + //可下载文本数量 + downloadnum(args) { + return 3 - download; + } } window.tempExt = { @@ -911,3 +937,17 @@ window.tempExt = { /* vim: set expandtab tabstop=2 shiftwidth=2: */ +//多行文本解码 +function multipleText(text) { + let texts = text.split("\n"); + let a = []; + if (text.indexOf("\r") !== -1) { + texts.forEach(e => { + a.push(e.split("\r")[0]); + }); + } + else { + a = texts; + } + return a; +} diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 44c70f9f..86006bb7 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -10,16 +10,17 @@ let keypress = {}; let lastKey = "", MouseWheel = 0; let timer; + //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; -if (cvs == null) { +if (cvs === null) { alert("当前页面不支持输入框,请前往作品详情页体验完整作品!"); } else { - for (let i = 1; cvs.className != "" && i <= document.getElementsByTagName("canvas").length; i++) { + for (let i = 1; cvs.className !== "" && i <= document.getElementsByTagName("canvas").length; i++) { cvs = document.getElementsByTagName("canvas")[i]; } - if (cvs == null) { + if (cvs === null) { alert("当前页面不支持输入框,请前往作品详情页体验完整作品!"); } } @@ -268,7 +269,7 @@ class WitCatInput { arguments: { type: { type: "string", - defaultValue: "a", + defaultValue: "KeyA", } }, }, @@ -286,7 +287,7 @@ class WitCatInput { arguments: { type: { type: "string", - defaultValue: "a", + defaultValue: "KeyA", } }, }, @@ -430,8 +431,8 @@ class WitCatInput { height = (height / this.runtime.stageHeight) * 100; let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none;color:` + args.color + `;`; let search = document.getElementById("WitCatInput" + args.id); - if (search != null) { - if (search.name == args.type) { + if (search !== null) { + if (search.name === args.type) { search.style = dom; search.value = args.text; search.placeholder = args.texts; @@ -439,7 +440,7 @@ class WitCatInput { else { cvs.parentNode.removeChild(search); let eleLink = document.createElement(args.type); - if (args.type == "input") { + if (args.type === "input") { eleLink.type = "text"; } eleLink.style = dom; @@ -453,7 +454,7 @@ class WitCatInput { } else { let eleLink = document.createElement(args.type); - if (args.type == "input") { + if (args.type === "input") { eleLink.type = "text"; } eleLink.style = dom; @@ -468,31 +469,31 @@ class WitCatInput { //删除文本框 deleteinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if (search != null) { + if (search !== null) { cvs.parentNode.removeChild(search); } } //获取文本框内容 getinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if (search != null) { - if (args.type == "X") + if (search !== null) { + if (args.type === "X") return (search.style.left.split("%")[0] / 100) * this.runtime.stageWidth; - else if (args.type == "Y") + else if (args.type === "Y") return (search.style.top.split("%")[0] / 100) * this.runtime.stageHeight; - else if (args.type == "width") + else if (args.type === "width") return (search.style.width.split("%")[0] / 100) * this.runtime.stageWidth; - else if (args.type == "height") + else if (args.type === "height") return (search.style.height.split("%")[0] / 100) * this.runtime.stageHeight; - else if (args.type == "content") + else if (args.type === "content") return search.value; - else if (args.type == "color") + else if (args.type === "color") return search.style.color.colorHex(); - else if (args.type == "prompt") + else if (args.type === "prompt") return search.placeholder; - else if (args.type == "font-size") + else if (args.type === "font-size") return search.style.fontSize.split("px")[0]; - else if (args.type == "ID") + else if (args.type === "ID") return search.id.split("WitCatInput")[1]; else { return ( @@ -515,8 +516,8 @@ class WitCatInput { //焦点判断 isinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if (search != null) { - if (search == document.activeElement) { + if (search !== null) { + if (search === document.activeElement) { return (true); } else { @@ -529,7 +530,7 @@ class WitCatInput { } //焦点位置 whatinput() { - if (document.activeElement.className == "WitCatInput") { + if (document.activeElement.className === "WitCatInput") { return document.activeElement.id.split("WitCatInput")[0]; } else { @@ -539,10 +540,10 @@ class WitCatInput { //焦点获取 nowinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if (search != null) { + if (search !== null) { search.focus(); } - else if (document.activeElement.className == "WitCatInput") { + else if (document.activeElement.className === "WitCatInput") { document.activeElement.blur(); } } @@ -562,23 +563,23 @@ class WitCatInput { number(args) { let search = document.getElementsByClassName("WitCatInput"); if (search.length >= args.num && args.num > 0) { - if (args.type == "X") + if (args.type === "X") return (search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth; - else if (args.type == "Y") + else if (args.type === "Y") return (search[args.num - 1].style.top.split("%")[0] / 100) * this.runtime.stageHeight; - else if (args.type == "width") + else if (args.type === "width") return (search[args.num - 1].style.width.split("%")[0] / 100) * this.runtime.stageWidth; - else if (args.type == "height") + else if (args.type === "height") return (search[args.num - 1].style.height.split("%")[0] / 100) * this.runtime.stageHeight; - else if (args.type == "content") + else if (args.type === "content") return search[args.num - 1].value; - else if (args.type == "color") + else if (args.type === "color") return search[args.num - 1].style.color.colorHex(); - else if (args.type == "prompt") + else if (args.type === "prompt") return search[args.num - 1].placeholder; - else if (args.type == "font-size") + else if (args.type === "font-size") return search[args.num - 1].style.fontSize.split("px")[0]; - else if (args.type == "ID") + else if (args.type === "ID") return search[args.num - 1].id.split("WitCatInput")[1]; else { return ( @@ -622,7 +623,7 @@ class WitCatInput { //设置文本框 setinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if (search != null) { + if (search !== null) { let x = search.style.left.split("%")[0]; let y = search.style.top.split("%")[0]; let width = search.style.width.split("%")[0]; @@ -631,7 +632,7 @@ class WitCatInput { let prompt = search.placeholder; let color = search.style.color.colorHex() let size = search.style.fontSize.split("px")[0]; - if (args.type == "X") { + if (args.type === "X") { x = args.text; if (args.text > this.runtime.stageWidth) { x = this.runtime.stageWidth; @@ -641,7 +642,7 @@ class WitCatInput { } x = (x / this.runtime.stageWidth) * 100; } - else if (args.type == "Y") { + else if (args.type === "Y") { y = args.text; if (args.text > this.runtime.stageHeight) { y = this.runtime.stageHeight; @@ -651,7 +652,7 @@ class WitCatInput { } y = (y / this.runtime.stageHeight) * 100; } - else if (args.type == "width") { + else if (args.type === "width") { width = args.text; if (Number(x) + Number(args.text) > this.runtime.stageWidth) { width = this.runtime.stageWidth - x; @@ -661,7 +662,7 @@ class WitCatInput { } width = (width / this.runtime.stageWidth) * 100; } - else if (args.type == "height") { + else if (args.type === "height") { height = args.text; if (Number(y) + Number(args.text) > this.runtime.stageHeight) { height = this.runtime.stageHeight - y; @@ -671,16 +672,16 @@ class WitCatInput { } height = (height / this.runtime.stageHeight) * 100; } - else if (args.type == "content") { + else if (args.type === "content") { content = args.text; } - else if (args.type == "prompt") { + else if (args.type === "prompt") { prompt = args.text; } - else if (args.type == "color") { + else if (args.type === "color") { color = args.text; } - else if (args.type == "font-size") { + else if (args.type === "font-size") { size = args.text; } @@ -741,12 +742,14 @@ String.prototype.colorHex = function () { } }; //键盘事件监听 -document.onkeydown = function (event) { - keypress[event.key.toLowerCase()] = true; - lastKey = event.key.toLowerCase(); +document.addEventListener("keydown", keydown); +document.addEventListener("keyup", keyup); +function keydown(event) { + keypress[event.code] = true; + lastKey = event.code; } -document.onkeyup = function (event) { - delete keypress[event.key.toLowerCase()]; +function keyup(event) { + delete keypress[event.code]; } //滚轮事件监听 var scrollFunc = function (e) { @@ -762,8 +765,8 @@ var scrollFunc = function (e) { }, 30); }; //给页面绑定滑轮滚动事件 -if (document.addEventListener) { //火狐使用DOMMouseScroll绑定 - document.addEventListener('DOMMouseScroll', scrollFunc, false); +if (cvs.parentNode.addEventListener) { //火狐使用DOMMouseScroll绑定 + cvs.parentNode.addEventListener('DOMMouseScroll', scrollFunc, false); } //其他浏览器直接绑定滚动事件 -window.onmousewheel = document.onmousewheel = scrollFunc; +cvs.parentNode.onmousewheel = scrollFunc; diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 1e552737..7de6a40c 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -15,17 +15,18 @@ let filln = 0; let w = 0, h = 0; let div, divs, divv, divvs, news, background, Operatinginstructions, bug, url, discord; let wv = 0, hv = 0; +let isTouchDevice = 'ontouchstart' in document.documentElement; //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; -if (cvs == null) { +if (cvs === null) { alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); } else { - for (let i = 1; cvs.className != "" && i <= document.getElementsByTagName("canvas").length; i++) { + for (let i = 1; cvs.className !== "" && i <= document.getElementsByTagName("canvas").length; i++) { cvs = document.getElementsByTagName("canvas")[i]; } - if (cvs == null) { + if (cvs === null) { alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); } } @@ -33,7 +34,7 @@ else { //添加监听器 const config = { attributes: true, childList: true, subtree: true, attributeFilter: ['style'] }; const callback = function (mutationsList, observer) { - if (fill == 1) { + if (fill === 1) { observer.disconnect(); fills(); observer.observe(cvs, config); @@ -73,6 +74,10 @@ class WitCatMouse { "WitCatMouse.fillask.2": "/3次连续拒绝后将不再提示\n您仍可以使用 ctrl+shift+alt 切换沉浸式全屏状态", "WitCatMouse.setfill": "⚠️(危)设置分辨率高设为[num]", "WitCatMouse.resolution": "当前分辨率高", + "WitCatMouse.cantouch": "设备支持[type]?", + "WitCatMouse.types.1": "触屏", + "WitCatMouse.types.2": "鼠标", + "WitCatMouse.IsMobile": "移动设备?", }, en: { "WitCatMouse.name": "More operate", @@ -101,6 +106,10 @@ class WitCatMouse { "WitCatMouse.fillask.2": "/3 times\nYou can also use Ctrl+Shift+Alt to toggle immersive full-screen later.", "WitCatMouse.setfill": "⚠️(danger)Set resolution height to[num]", "WitCatMouse.resolution": "Current high resolution", + "WitCatMouse.cantouch": "Device support[type]?", + "WitCatMouse.types.1": "Touch screen", + "WitCatMouse.types.2": "mouse", + "WitCatMouse.IsMobile": "Mobile devices?", } }) } @@ -213,6 +222,23 @@ class WitCatMouse { }, }, }, + { + opcode: "cantouch", + blockType: "Boolean", + text: this.formatMessage("WitCatMouse.cantouch"), + arguments: { + type: { + type: "string", + menu: "types", + }, + }, + }, + { + opcode: "IsMobile", + blockType: "Boolean", + text: this.formatMessage("WitCatMouse.IsMobile"), + arguments: {}, + }, ], menus: { key: [ @@ -267,6 +293,16 @@ class WitCatMouse { value: "y" }, ], + types: [ + { + text: this.formatMessage('WitCatMouse.types.1'), + value: "ontouchstart" + }, + { + text: this.formatMessage('WitCatMouse.types.2'), + value: "onmousedown" + }, + ], } }; } @@ -274,7 +310,7 @@ class WitCatMouse { set(args) { history.pushState(null, null, null); cvs.parentNode.oncontextmenu = function () { - if (args.set == "true") { + if (args.set === "true") { return true; } else { @@ -284,7 +320,7 @@ class WitCatMouse { } //按下判断 when(args) { - if (button[args.key] == "down") { + if (button[args.key] === "down") { return true; } else { @@ -297,7 +333,7 @@ class WitCatMouse { } //鼠标移动量 acceleration(args) { - if (args.way == "x") { + if (args.way === "x") { return xMouse; } else { @@ -311,7 +347,7 @@ class WitCatMouse { //坐标 num(args) { if (args.num > 0 && args.num <= touch.length) { - if (args.type == "x") { + if (args.type === "x") { return this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.offsetWidth); } else { @@ -324,11 +360,11 @@ class WitCatMouse { } //全屏 fill(args) { - if (args.set == "true") { + if (args.set === "true") { if (filln < 3) { - if (fill == 0) { + if (fill === 0) { let a = confirm(this.formatMessage('WitCatMouse.fillask.1') + filln + this.formatMessage('WitCatMouse.fillask.2')); - if (a == true) { + if (a === true) { filln = 0; fillin(); } @@ -339,7 +375,7 @@ class WitCatMouse { } } else { - if (fill == 1) { + if (fill === 1) { outoffill(); } } @@ -355,6 +391,14 @@ class WitCatMouse { resolution(args) { return cvs.height; } + //设备是否支持触屏 + cantouch(args) { + return (args.type in document.documentElement); + } + //是否是手机 + IsMobile() { + return /Android|iPhone|iPad|iPod|BlackBerry|webOS|Windows Phone|SymbianOS|IEMobile|Opera Mini/i.test(navigator.userAgent); + } } window.tempExt = { @@ -382,14 +426,21 @@ window.tempExt = { }; /* vim: set expandtab tabstop=2 shiftwidth=2: */ - +//鼠标 document.onmousedown = function (event) { button[event.button] = "down"; } document.onmouseup = function (event) { button[event.button] = "up"; + touch = []; } document.addEventListener("mousemove", ev => { + if (button[0] === "down") { + touch = JSON.parse("[{\"clientX\":\"" + ev.clientX + "\",\"clientY\":\"" + ev.clientY + "\"}]"); + } + else { + touch = []; + } xMouse = ev.movementX; // 获得鼠标指针的x移动量 yMouse = ev.movementY; // 获得鼠标指针的y移动量 isMove = true; @@ -403,19 +454,30 @@ document.addEventListener("mousemove", ev => { //多指触控 cvs.addEventListener('touchstart', function (e) { touch = e.targetTouches; + button[0] = "down"; }) cvs.addEventListener('touchmove', function (e) { + xMouse = e.targetTouches[0].clientX - touch[0].clientX; // 获得手指的x移动量 + yMouse = e.targetTouches[0].clientY - touch[0].clientY; // 获得手指的y移动量 + isMove = true; + clearTimeout(timer); + timer = setTimeout(function () { + isMove = false; + xMouse = 0; + yMouse = 0; + }, 30); touch = e.targetTouches; }) cvs.addEventListener('touchend', function (e) { touch = e.targetTouches; + button[0] = "up"; }) //大小改变刷新 window.onresize = function () { - if (fill == 1) { + if (fill === 1) { fills(); } } @@ -426,33 +488,33 @@ function fillin() { var mo = function (e) { e.preventDefault(); }; document.body.style.overflow = 'hidden'; document.addEventListener("touchmove", mo, false);//禁止页面滑动 - if (window.location.href.split("/")[2] != "cocrea.world") { + if (window.location.href.split("/")[2] !== "cocrea.world") { div = document.getElementById("root").getElementsByTagName('div')[0]; divs = document.getElementById("root"); url = window.location.href.split("/")[3].split("?")[0]; - if (url != "scratch-player") { + if (url !== "scratch-player") { divs.removeChild(div); } } news = document.body.getElementsByClassName("convention-1wIbd")[0]; - if (typeof (news) != "undefined") { + if (typeof (news) !== "undefined") { news.style = "display:none;"; } divv = document.body.getElementsByClassName("actions-2lk9z")[0]; - if (typeof (divv) != "undefined") { + if (typeof (divv) !== "undefined") { divv.style = "display:none;"; } Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; - if (typeof (Operatinginstructions) != "undefined") { + if (typeof (Operatinginstructions) !== "undefined") { Operatinginstructions.style = "display:none;"; } bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; - if (typeof (bug) != "undefined") { + if (typeof (bug) !== "undefined") { bug.style = "display:none;"; } discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; - if (typeof (discord) != "undefined") { + if (typeof (discord) !== "undefined") { discord.style = "display:none"; } document.body.scrollTop = 0; @@ -469,36 +531,36 @@ function outoffill() { var mo = function (e) { e.preventDefault(); }; document.body.style.overflow = 'auto'; document.addEventListener("touchmove", mo, true);//允许页面滑动 - if (window.location.href.split("/")[2] != "cocrea.world") { + if (window.location.href.split("/")[2] !== "cocrea.world") { divs = document.getElementById("root"); url = window.location.href.split("/")[3].split("?")[0]; - if (url != "scratch-player") { + if (url !== "scratch-player") { divs.insertBefore(div, divs.children[0]); } } document.body.removeChild(background); news = document.body.getElementsByClassName("convention-1wIbd")[0]; - if (typeof (news) != "undefined") { + if (typeof (news) !== "undefined") { news.style = ""; } divv = document.body.getElementsByClassName("actions-2lk9z")[0]; - if (typeof (divv) != "undefined") { + if (typeof (divv) !== "undefined") { divv.style = ""; } divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; - if (typeof (divvs) != "undefined") { + if (typeof (divvs) !== "undefined") { divvs.style = ""; } Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; - if (typeof (Operatinginstructions) != "undefined") { + if (typeof (Operatinginstructions) !== "undefined") { Operatinginstructions.style = ""; } bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; - if (typeof (bug) != "undefined") { + if (typeof (bug) !== "undefined") { bug.style = ""; } discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; - if (typeof (discord) != "undefined") { + if (typeof (discord) !== "undefined") { discord.style = ""; } cvs.parentNode.parentNode.parentNode.parentNode.style = "width:100%; height:100%;z-index: 1000000;"; @@ -508,7 +570,7 @@ function outoffill() { } //循环检测 function fills() { - if (w != 0 && h != 0) { + if (w !== 0 && h !== 0) { cvs.width = w; cvs.height = h; } @@ -526,7 +588,7 @@ function fills() { hv = 0; } divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; - if (typeof (divvs) != "undefined") { + if (typeof (divvs) !== "undefined") { divvs.style = "display:none;"; } cvs.parentNode.parentNode.parentNode.parentNode.style = "position:fixed; left:" + wv + "px; top:" + hv + "px; width:" + ws + "px; height:" + hs + "px;z-index: 1000000;border-radius:0px"; @@ -546,7 +608,7 @@ function fills() { cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(" + a + ");transform-origin:0% 0% 0"; }, 20); background = document.getElementsByClassName("witcatbackground")[0]; - if (background == null) { + if (background === null) { background = document.createElement("div"); background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; background.className = "witcatbackground"; @@ -562,10 +624,11 @@ function isInPage(node) { } //按键检测 -document.onkeydown = function (e) { +document.addEventListener("keydown", keydown); +function keydown(e) { var evt = e || window.event; if (evt.ctrlKey && evt.shiftKey && evt.altKey) { - if (fill == 0) { + if (fill === 0) { fillin(); } else { From cc330b39d4163076b8bff8aaed6ef71a0193fee6 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 16 Jan 2023 14:33:06 +0800 Subject: [PATCH 210/357] =?UTF-8?q?=E4=BC=98=E5=8C=96=E4=B8=8B=E8=BD=BD?= =?UTF-8?q?=E5=A4=9A=E8=A1=8C=E6=96=87=E6=9C=AC=E9=80=9F=E5=BA=A6=EF=BC=8C?= =?UTF-8?q?=E6=96=B0=E5=A2=9E=E4=B8=8B=E8=BD=BD=E6=96=87=E4=BB=B6=E9=99=90?= =?UTF-8?q?=E5=88=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit perf(compile):优化下载多行文本速度 feat(compile):新增下载文件限制,限制下载*.cmd、*.bat、*.vbs、*.ps1、*.sh --- wit_cat/File_Helper.js | 23 +++++++++++++++++------ 1 file changed, 17 insertions(+), 6 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 1fe82a4a..bdfe3403 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -51,6 +51,7 @@ class WitCatFileHelper { "WitCatFileHelper.turnMultiplelinestext": "将多行文本[text]转化为数组", "WitCatFileHelper.turnsMultiplelinestext": "将数组[text]转化为多行文本", "WitCatFileHelper.downloadnum": "可下载文件数量", + "WitCatFileHelper.downloadask": "作品企图下载疑似会威胁电脑的文件,是否继续?\n代码如下:", }, en: { "WitCatFileHelper.name": "file helper", @@ -86,6 +87,7 @@ class WitCatFileHelper { "WitCatFileHelper.turnMultiplelinestext": "Converts multiline[text]text to an array", "WitCatFileHelper.turnsMultiplelinestext": "Converts the array[text]to multiline text", "WitCatFileHelper.downloadnum": "Number of downloadable files", + "WitCatFileHelper.downloadask": "works will attempt to download the suspected threat to computer files, whether or not to continue? \n code is as follows: ", } }) } @@ -463,16 +465,18 @@ class WitCatFileHelper { } const filename = filenames; let s = args.s; - let j = 0; if (s != "") { - let a = text.split(s); - h = a[0]; - for (j = 1; j < a.length; j++) { - h = h + "\n" + a[j]; - } + h = text.split(s).join("\n"); } else { h = text; } + let SuffixName = filename.split(".")[filename.split(".").length - 1]; + if (SuffixName === "bat" || SuffixName === "cmd" || SuffixName === "vbs" || SuffixName === "ps1" || SuffixName === "sh") { + let a = confirm(this.formatMessage('WitCatFileHelper.downloadask') + SuffixName + ":\n" + h); + if (a === false) { + return; + } + } const content = h; // 创建隐藏的可下载链接 let eleLink = document.createElement('a'); @@ -500,6 +504,13 @@ class WitCatFileHelper { if (filenames === "") { filenames = "none.txt" } + let SuffixName = filenames.split(".")[filenames.split(".").length - 1]; + if (SuffixName === "bat" || SuffixName === "cmd" || SuffixName === "vbs" || SuffixName === "ps1" || SuffixName === "sh") { + let a = confirm(this.formatMessage('WitCatFileHelper.downloadask') + SuffixName + ":\n" + content); + if (a === false) { + return; + } + } const filename = filenames; // 创建隐藏的可下载链接 let eleLink = document.createElement('a'); From 8a2dc3d3c6f257bff98454344aae60bec7fe2ee0 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 16 Jan 2023 14:34:44 +0800 Subject: [PATCH 211/357] =?UTF-8?q?=E6=9B=B4=E6=AD=A3=E7=A7=AF=E6=9C=A8?= =?UTF-8?q?=E9=BB=98=E8=AE=A4=E5=80=BC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit perf(block):更正积木默认值 --- wit_cat/Input.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 86006bb7..e0f696bf 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -172,7 +172,7 @@ class WitCatInput { }, text: { type: "string", - defaultValue: "i", + defaultValue: "10", }, }, }, From a48aaf1a4ae846d26456085f6b178d6301c892c4 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 16 Jan 2023 14:43:17 +0800 Subject: [PATCH 212/357] =?UTF-8?q?=E6=9B=B4=E6=96=B0=E9=99=90=E5=88=B6?= =?UTF-8?q?=E6=9C=BA=E5=88=B6=EF=BC=8C=E4=BC=98=E5=8C=96=E5=A4=9A=E8=A1=8C?= =?UTF-8?q?=E6=96=87=E6=9C=AC=E4=B8=8B=E8=BD=BD=EF=BC=8C=E4=BF=AE=E6=94=B9?= =?UTF-8?q?=E6=96=87=E6=9C=AC=E6=A1=86=E9=BB=98=E8=AE=A4=E5=80=BC=20(#59)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * 优化下载多行文本速度,新增下载文件限制 perf(compile):优化下载多行文本速度 feat(compile):新增下载文件限制,限制下载*.cmd、*.bat、*.vbs、*.ps1、*.sh * 更正积木默认值 perf(block):更正积木默认值 --- wit_cat/File_Helper.js | 23 +++++++++++++++++------ wit_cat/Input.js | 2 +- 2 files changed, 18 insertions(+), 7 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 1fe82a4a..bdfe3403 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -51,6 +51,7 @@ class WitCatFileHelper { "WitCatFileHelper.turnMultiplelinestext": "将多行文本[text]转化为数组", "WitCatFileHelper.turnsMultiplelinestext": "将数组[text]转化为多行文本", "WitCatFileHelper.downloadnum": "可下载文件数量", + "WitCatFileHelper.downloadask": "作品企图下载疑似会威胁电脑的文件,是否继续?\n代码如下:", }, en: { "WitCatFileHelper.name": "file helper", @@ -86,6 +87,7 @@ class WitCatFileHelper { "WitCatFileHelper.turnMultiplelinestext": "Converts multiline[text]text to an array", "WitCatFileHelper.turnsMultiplelinestext": "Converts the array[text]to multiline text", "WitCatFileHelper.downloadnum": "Number of downloadable files", + "WitCatFileHelper.downloadask": "works will attempt to download the suspected threat to computer files, whether or not to continue? \n code is as follows: ", } }) } @@ -463,16 +465,18 @@ class WitCatFileHelper { } const filename = filenames; let s = args.s; - let j = 0; if (s != "") { - let a = text.split(s); - h = a[0]; - for (j = 1; j < a.length; j++) { - h = h + "\n" + a[j]; - } + h = text.split(s).join("\n"); } else { h = text; } + let SuffixName = filename.split(".")[filename.split(".").length - 1]; + if (SuffixName === "bat" || SuffixName === "cmd" || SuffixName === "vbs" || SuffixName === "ps1" || SuffixName === "sh") { + let a = confirm(this.formatMessage('WitCatFileHelper.downloadask') + SuffixName + ":\n" + h); + if (a === false) { + return; + } + } const content = h; // 创建隐藏的可下载链接 let eleLink = document.createElement('a'); @@ -500,6 +504,13 @@ class WitCatFileHelper { if (filenames === "") { filenames = "none.txt" } + let SuffixName = filenames.split(".")[filenames.split(".").length - 1]; + if (SuffixName === "bat" || SuffixName === "cmd" || SuffixName === "vbs" || SuffixName === "ps1" || SuffixName === "sh") { + let a = confirm(this.formatMessage('WitCatFileHelper.downloadask') + SuffixName + ":\n" + content); + if (a === false) { + return; + } + } const filename = filenames; // 创建隐藏的可下载链接 let eleLink = document.createElement('a'); diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 86006bb7..e0f696bf 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -172,7 +172,7 @@ class WitCatInput { }, text: { type: "string", - defaultValue: "i", + defaultValue: "10", }, }, }, From 24af1654434dea25b41c51cc6a0a7bbade4bb68d Mon Sep 17 00:00:00 2001 From: nickhu <314484063@qq.com> Date: Mon, 16 Jan 2023 16:29:13 +0800 Subject: [PATCH 213/357] chore: chang ext name --- wit_cat/File_Helper.js | 4 ++-- wit_cat/Input.js | 4 ++-- wit_cat/More_Mouse.js | 4 ++-- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index bdfe3403..4b1d6088 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -18,7 +18,7 @@ class WitCatFileHelper { this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "WitCatFileHelper.name": "文件助手", + "WitCatFileHelper.name": "[beta]白猫的文件助手", "WitCatFileHelper.filehelper": "文件处理", "WitCatFileHelper.inputmanagement": "文本框", "WitCatFileHelper.Multiplelinestext": "多行文本", @@ -54,7 +54,7 @@ class WitCatFileHelper { "WitCatFileHelper.downloadask": "作品企图下载疑似会威胁电脑的文件,是否继续?\n代码如下:", }, en: { - "WitCatFileHelper.name": "file helper", + "WitCatFileHelper.name": "[beta]WitCat’s File Helper", "WitCatFileHelper.filehelper": "file", "WitCatFileHelper.inputmanagement": "input", "WitCatFileHelper.Multiplelinestext": "Multiple lines of text", diff --git a/wit_cat/Input.js b/wit_cat/Input.js index e0f696bf..882ee981 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -30,7 +30,7 @@ class WitCatInput { this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "WitCatInput.name": "文本输入框", + "WitCatInput.name": "[beta]白猫的输入框", "WitCatInput.createinput": "ID为[id]的[type]框X[x]Y[y]宽[width]高[height]内容[text]颜色[color]提示[texts]字体大小[size]", "WitCatInput.deleteinput": "删除ID为[id]的文本框", "WitCatInput.getinput": "获得ID为[id]的文本框的[type]", @@ -60,7 +60,7 @@ class WitCatInput { "WitCatInput.setinput": "设置ID为[id]的文本框的[type]为[text]", }, en: { - "WitCatInput.name": "input", + "WitCatInput.name": "[beta]WitCat‘s Input", "WitCatInput.createinput": "[type]input with ID[id]X[x]Y[y]width[width]height[height]content[text]color[color]prompt[texts]font-size[size]", "WitCatInput.deleteinput": "delete an input with ID[id]", "WitCatInput.getinput": "get [type] with ID[id]", diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 7de6a40c..ff647452 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -49,7 +49,7 @@ class WitCatMouse { this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { - "WitCatMouse.name": "高级操作", + "WitCatMouse.name": "[beta]白猫的高级鼠标", "WitCatMouse.name.1": "高级鼠标", "WitCatMouse.name.2": "多指触控", "WitCatMouse.set": "[set]右键菜单", @@ -80,7 +80,7 @@ class WitCatMouse { "WitCatMouse.IsMobile": "移动设备?", }, en: { - "WitCatMouse.name": "More operate", + "WitCatMouse.name": "[beta]WitCat’s Mouse", "WitCatMouse.name.1": "more mouse", "WitCatMouse.name.2": "more touch", "WitCatMouse.set": "[set]Right-click menu", From 530c994676d35297454381941a2c04dff4cbd08f Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 16 Jan 2023 19:23:51 +0800 Subject: [PATCH 214/357] =?UTF-8?q?=E4=BC=98=E5=8C=96=E6=A0=BC=E5=BC=8F?= =?UTF-8?q?=EF=BC=8C=E4=BF=AE=E5=A4=8D=E5=85=A8=E5=B1=8FBUG=EF=BC=8C?= =?UTF-8?q?=E4=BC=98=E5=8C=96=E5=85=A8=E5=B1=8F=E4=BD=93=E9=AA=8C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit style:将原来的settimeout匿名函数套函数修复为直接函数 fix(compile):修复全屏创建背景时报错的BUG perf(browser):修复全屏后变量大小显示错位的BUG --- wit_cat/More_Mouse.js | 19 ++++++++----------- 1 file changed, 8 insertions(+), 11 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index ff647452..5a04eb3e 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -415,11 +415,11 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatMouse.name": "高级鼠标", + "WitCatMouse.name": "[beta]白猫的高级鼠标", "WitCatMouse.descp": "更精准的控制鼠标/触屏/全屏!" }, en: { - "WitCatMouse.name": "More Mouse", + "WitCatMouse.name": "[beta]WitCat’s Mouse", "WitCatMouse.descp": "More precise mouse/touch/full screen control!" } } @@ -484,7 +484,6 @@ window.onresize = function () { //开启全屏 function fillin() { fill = 1; - var mo = function (e) { e.preventDefault(); }; document.body.style.overflow = 'hidden'; document.addEventListener("touchmove", mo, false);//禁止页面滑动 @@ -520,9 +519,8 @@ function fillin() { document.body.scrollTop = 0; document.documentElement.scrollTop = 0; scrollTo(0, 0); - setTimeout(function () { - fills(); - }, 50); + + setTimeout(fills(), 50); } //关闭全屏 function outoffill() { @@ -567,6 +565,7 @@ function outoffill() { cvs.parentNode.parentNode.style = "height: 100%; width: 100%;"; cvs.parentNode.style = ""; cvs.style = "height: 100%; width: 100%;"; + cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(1);transform-origin:0% 0% 0"; } //循环检测 function fills() { @@ -603,12 +602,10 @@ function fills() { } cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("div")[0].style = "width:100%; height:100%;" cvs.style = "width:100%; height:100%;"; - setTimeout(function () { - let a = cvs.offsetWidth / document.documentElement.clientWidth; - cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(" + a + ");transform-origin:0% 0% 0"; - }, 20); + let a = document.documentElement.clientWidth / cvs.parentNode.parentNode.getElementsByTagName("div")[1].clientWidth; + cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(" + a + ");transform-origin:0% 0% 0"; background = document.getElementsByClassName("witcatbackground")[0]; - if (background === null) { + if (!background) { background = document.createElement("div"); background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; background.className = "witcatbackground"; From 558a1289ca15e4b865186f306d486b15c1439392 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 16 Jan 2023 19:27:26 +0800 Subject: [PATCH 215/357] =?UTF-8?q?=E4=BC=98=E5=8C=96=E6=A0=BC=E5=BC=8F?= =?UTF-8?q?=EF=BC=8C=E4=BF=AE=E5=A4=8D=E5=85=A8=E5=B1=8FBUG=EF=BC=8C?= =?UTF-8?q?=E4=BC=98=E5=8C=96=E5=85=A8=E5=B1=8F=E4=BD=93=E9=AA=8C=20(#60)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * 优化下载多行文本速度,新增下载文件限制 perf(compile):优化下载多行文本速度 feat(compile):新增下载文件限制,限制下载*.cmd、*.bat、*.vbs、*.ps1、*.sh * 更正积木默认值 perf(block):更正积木默认值 * 优化格式,修复全屏BUG,优化全屏体验 style:将原来的settimeout匿名函数套函数修复为直接函数 fix(compile):修复全屏创建背景时报错的BUG perf(browser):修复全屏后变量大小显示错位的BUG --- wit_cat/More_Mouse.js | 19 ++++++++----------- 1 file changed, 8 insertions(+), 11 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index ff647452..5a04eb3e 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -415,11 +415,11 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatMouse.name": "高级鼠标", + "WitCatMouse.name": "[beta]白猫的高级鼠标", "WitCatMouse.descp": "更精准的控制鼠标/触屏/全屏!" }, en: { - "WitCatMouse.name": "More Mouse", + "WitCatMouse.name": "[beta]WitCat’s Mouse", "WitCatMouse.descp": "More precise mouse/touch/full screen control!" } } @@ -484,7 +484,6 @@ window.onresize = function () { //开启全屏 function fillin() { fill = 1; - var mo = function (e) { e.preventDefault(); }; document.body.style.overflow = 'hidden'; document.addEventListener("touchmove", mo, false);//禁止页面滑动 @@ -520,9 +519,8 @@ function fillin() { document.body.scrollTop = 0; document.documentElement.scrollTop = 0; scrollTo(0, 0); - setTimeout(function () { - fills(); - }, 50); + + setTimeout(fills(), 50); } //关闭全屏 function outoffill() { @@ -567,6 +565,7 @@ function outoffill() { cvs.parentNode.parentNode.style = "height: 100%; width: 100%;"; cvs.parentNode.style = ""; cvs.style = "height: 100%; width: 100%;"; + cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(1);transform-origin:0% 0% 0"; } //循环检测 function fills() { @@ -603,12 +602,10 @@ function fills() { } cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("div")[0].style = "width:100%; height:100%;" cvs.style = "width:100%; height:100%;"; - setTimeout(function () { - let a = cvs.offsetWidth / document.documentElement.clientWidth; - cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(" + a + ");transform-origin:0% 0% 0"; - }, 20); + let a = document.documentElement.clientWidth / cvs.parentNode.parentNode.getElementsByTagName("div")[1].clientWidth; + cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(" + a + ");transform-origin:0% 0% 0"; background = document.getElementsByClassName("witcatbackground")[0]; - if (background === null) { + if (!background) { background = document.createElement("div"); background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; background.className = "witcatbackground"; From ae23508f89146fa9a0d98cb9c9a5a0efb7cd14d6 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Mon, 16 Jan 2023 20:33:42 +0800 Subject: [PATCH 216/357] =?UTF-8?q?=E4=B8=8B=E7=BA=BF=E5=85=A8=E5=B1=8F?= =?UTF-8?q?=E7=9B=B8=E5=85=B3=E5=8A=9F=E8=83=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit fix(block):下线全屏积木,下线修改分辨率积木 --- wit_cat/More_Mouse.js | 188 +----------------------------------------- 1 file changed, 4 insertions(+), 184 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 5a04eb3e..ba84a14d 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -136,6 +136,7 @@ class WitCatMouse { opcode: 'setfill', blockType: "command", text: this.formatMessage("WitCatMouse.setfill"), + hideFromPalette: true, arguments: { num: { type: "number", @@ -147,6 +148,7 @@ class WitCatMouse { opcode: 'fill', blockType: "command", text: this.formatMessage("WitCatMouse.fill"), + hideFromPalette: true, arguments: { set: { type: "bool", @@ -360,32 +362,11 @@ class WitCatMouse { } //全屏 fill(args) { - if (args.set === "true") { - if (filln < 3) { - if (fill === 0) { - let a = confirm(this.formatMessage('WitCatMouse.fillask.1') + filln + this.formatMessage('WitCatMouse.fillask.2')); - if (a === true) { - filln = 0; - fillin(); - } - else { - filln += 1; - } - } - } - } - else { - if (fill === 1) { - outoffill(); - } - } + console.warn("全屏因浏览器兼容问题已下线,在未来修复后将会重新上线\nFull screen has been taken offline due to browser compatibility issues. It will be back online after a future fix"); } //设置分辨率 setfill(args) { - w = args.num / this.runtime.stageHeight * this.runtime.stageWidth; - h = args.num; - cvs.width = args.num / this.runtime.stageHeight * this.runtime.stageWidth; - cvs.height = args.num; + console.warn("全屏因浏览器兼容问题已下线,在未来修复后将会重新上线\nFull screen has been taken offline due to browser compatibility issues. It will be back online after a future fix"); } //当前分辨率 resolution(args) { @@ -472,164 +453,3 @@ cvs.addEventListener('touchend', function (e) { touch = e.targetTouches; button[0] = "up"; }) - - - -//大小改变刷新 -window.onresize = function () { - if (fill === 1) { - fills(); - } -} -//开启全屏 -function fillin() { - fill = 1; - var mo = function (e) { e.preventDefault(); }; - document.body.style.overflow = 'hidden'; - document.addEventListener("touchmove", mo, false);//禁止页面滑动 - if (window.location.href.split("/")[2] !== "cocrea.world") { - div = document.getElementById("root").getElementsByTagName('div')[0]; - divs = document.getElementById("root"); - url = window.location.href.split("/")[3].split("?")[0]; - if (url !== "scratch-player") { - divs.removeChild(div); - } - } - - news = document.body.getElementsByClassName("convention-1wIbd")[0]; - if (typeof (news) !== "undefined") { - news.style = "display:none;"; - } - divv = document.body.getElementsByClassName("actions-2lk9z")[0]; - if (typeof (divv) !== "undefined") { - divv.style = "display:none;"; - } - Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; - if (typeof (Operatinginstructions) !== "undefined") { - Operatinginstructions.style = "display:none;"; - } - bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; - if (typeof (bug) !== "undefined") { - bug.style = "display:none;"; - } - discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; - if (typeof (discord) !== "undefined") { - discord.style = "display:none"; - } - document.body.scrollTop = 0; - document.documentElement.scrollTop = 0; - scrollTo(0, 0); - - setTimeout(fills(), 50); -} -//关闭全屏 -function outoffill() { - //恢复头 - fill = 0; - var mo = function (e) { e.preventDefault(); }; - document.body.style.overflow = 'auto'; - document.addEventListener("touchmove", mo, true);//允许页面滑动 - if (window.location.href.split("/")[2] !== "cocrea.world") { - divs = document.getElementById("root"); - url = window.location.href.split("/")[3].split("?")[0]; - if (url !== "scratch-player") { - divs.insertBefore(div, divs.children[0]); - } - } - document.body.removeChild(background); - news = document.body.getElementsByClassName("convention-1wIbd")[0]; - if (typeof (news) !== "undefined") { - news.style = ""; - } - divv = document.body.getElementsByClassName("actions-2lk9z")[0]; - if (typeof (divv) !== "undefined") { - divv.style = ""; - } - divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; - if (typeof (divvs) !== "undefined") { - divvs.style = ""; - } - Operatinginstructions = document.body.getElementsByClassName("showWorksDesc-1iD-M")[0]; - if (typeof (Operatinginstructions) !== "undefined") { - Operatinginstructions.style = ""; - } - bug = document.body.getElementsByClassName("bug-report-1CfBK")[0]; - if (typeof (bug) !== "undefined") { - bug.style = ""; - } - discord = document.body.getElementsByClassName("style_bugReport__gJk19 MuiBox-root css-0")[0]; - if (typeof (discord) !== "undefined") { - discord.style = ""; - } - cvs.parentNode.parentNode.parentNode.parentNode.style = "width:100%; height:100%;z-index: 1000000;"; - cvs.parentNode.parentNode.style = "height: 100%; width: 100%;"; - cvs.parentNode.style = ""; - cvs.style = "height: 100%; width: 100%;"; - cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(1);transform-origin:0% 0% 0"; -} -//循环检测 -function fills() { - if (w !== 0 && h !== 0) { - cvs.width = w; - cvs.height = h; - } - let ws = 0, hs = 0; - if ((cvs.width / cvs.height) >= (document.documentElement.clientWidth / document.documentElement.clientHeight)) { - hs = (document.documentElement.clientWidth / cvs.width) * cvs.height; - ws = document.documentElement.clientWidth; - wv = 0; - hv = (document.documentElement.clientHeight - hs) / 2; - } - else { - hs = document.documentElement.clientHeight; - ws = (document.documentElement.clientHeight / cvs.height) * cvs.width; - wv = (document.documentElement.clientWidth - ws) / 2; - hv = 0; - } - divvs = document.body.getElementsByClassName("arrowTop-2Fru_")[0]; - if (typeof (divvs) !== "undefined") { - divvs.style = "display:none;"; - } - cvs.parentNode.parentNode.parentNode.parentNode.style = "position:fixed; left:" + wv + "px; top:" + hv + "px; width:" + ws + "px; height:" + hs + "px;z-index: 1000000;border-radius:0px"; - cvs.parentNode.parentNode.parentNode.style = "width:" + ws + "px; height:" + hs + "px;border-radius:0px"; - cvs.parentNode.parentNode.style = "width:100%; height:100%;border-radius:0px"; - cvs.parentNode.style = "width:100%; height:100%;border-radius:0px"; - if (isInPage(cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild)) { - cvs.parentNode.parentNode.nextElementSibling.style = "width: 100%; height: 100%;"; - cvs.parentNode.parentNode.nextElementSibling.firstChild.style = "width: 100%; height: 100%;"; - cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.firstChild.style = "width: 100%; height: 100%;"; - cvs.parentNode.parentNode.nextElementSibling.firstChild.firstChild.style = "width: 100%;"; - } - cvs.parentNode.parentNode.parentNode.parentNode.getElementsByTagName("div")[0].style = "width:100%; height:100%;" - cvs.style = "width:100%; height:100%;"; - let a = document.documentElement.clientWidth / cvs.parentNode.parentNode.getElementsByTagName("div")[1].clientWidth; - cvs.parentNode.parentNode.getElementsByTagName("div")[1].style = "transform:scale(" + a + ");transform-origin:0% 0% 0"; - background = document.getElementsByClassName("witcatbackground")[0]; - if (!background) { - background = document.createElement("div"); - background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; - background.className = "witcatbackground"; - document.body.insertBefore(background, document.body.children[0]); - } - else { - background.style = "width:" + document.documentElement.clientWidth + "px;height:" + document.documentElement.clientHeight + "px;color:black;"; - } -} -//检测div存在 -function isInPage(node) { - return (node === document.body) ? false : document.body.contains(node); -} - -//按键检测 -document.addEventListener("keydown", keydown); -function keydown(e) { - var evt = e || window.event; - if (evt.ctrlKey && evt.shiftKey && evt.altKey) { - if (fill === 0) { - fillin(); - } - else { - outoffill(); - } - } -} From 9dc43d979bfcfc502377f6dadb7b2b628f03efa6 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 17 Jan 2023 11:26:44 +0800 Subject: [PATCH 217/357] =?UTF-8?q?witcat`s=20ext=201.1=20=E4=BF=AE?= =?UTF-8?q?=E5=A4=8D=E6=96=87=E4=BB=B6=E5=8A=A9=E6=89=8B=E7=9A=84BUG=20(#6?= =?UTF-8?q?2)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * 修复多行文本获取第几行报错的BUG fix(block):修复多行文本获取第几行报错的BUG * 修复了文本框焦点位置始终为空的BUG fix(block):修复了文本框焦点位置始终为空的BUG --- wit_cat/File_Helper.js | 11 ++++++++--- wit_cat/Input.js | 6 +++--- 2 files changed, 11 insertions(+), 6 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index 4b1d6088..ab59bd6d 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -867,7 +867,12 @@ class WitCatFileHelper { if (args.num === "first") { num = 1; } - return text[num - 1].split("\r")[0]; + if (text.length >= num) { + return text[num - 1].split("\r")[0]; + } + else { + return ""; + } } //多行文本行数 numMultiplelinestext(args) { @@ -936,11 +941,11 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatFileHelper.name": "文件助手", + "WitCatFileHelper.name": "[beta]白猫的文件助手", "WitCatFileHelper.descp": "读取/处理本地数据" }, en: { - "WitCatFileHelper.name": "File Helper", + "WitCatFileHelper.name": "[beta]WitCat’s File Helper", "WitCatFileHelper.descp": "Handling local data" } } diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 882ee981..7043c908 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -531,7 +531,7 @@ class WitCatInput { //焦点位置 whatinput() { if (document.activeElement.className === "WitCatInput") { - return document.activeElement.id.split("WitCatInput")[0]; + return document.activeElement.id.split("WitCatInput")[1]; } else { return ""; @@ -708,11 +708,11 @@ window.tempExt = { }, l10n: { "zh-cn": { - "WitCatInput.name": "文本框", + "WitCatInput.name": "[beta]白猫的输入框", "WitCatInput.descp": "全新的输入框!" }, en: { - "WitCatInput.name": "input", + "WitCatInput.name": "[beta]WitCat‘s Input", "WitCatInput.descp": "what a nice input!" } } From d9d1b0446809b46998034d7937b23c98f1d7322c Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 17 Jan 2023 15:08:42 +0800 Subject: [PATCH 218/357] =?UTF-8?q?File=5FHelper=202.3=20=E4=BF=AE?= =?UTF-8?q?=E5=A4=8D=E8=8E=B7=E5=8F=96=E8=A1=8C=E6=95=B0BUG=EF=BC=8C?= =?UTF-8?q?=E4=BF=AE=E5=A4=8DESLint=E6=8A=A5=E9=94=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit fix(block):修复了获取行数为负数时报错的BUG refactor(compile):修复了ESLint报错的部分 --- wit_cat/File_Helper.js | 31 +++++++++++++++---------------- 1 file changed, 15 insertions(+), 16 deletions(-) diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index ab59bd6d..b4f395d2 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -7,7 +7,7 @@ const _icon = " const extensionId = "WitCatFileHelper"; let FLAG = 0, download = 0; -setInterval(function () { +setInterval(() => { if (download > 0) { download--; } @@ -530,7 +530,7 @@ class WitCatFileHelper { } } //读取本地变量 - upload(args) { + upload() { console.warn("文件助手——读取键值对:积木已下线,请更换\nfile_helper-Read key value pair: Block is offline, please replace");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); @@ -561,7 +561,7 @@ class WitCatFileHelper { return localStorage.getItem(show + h + "©" + name);*/ } //保存本地变量 - save(args) { + save() { console.warn("文件助手——保存键值对:积木已下线,请更换\nfile_helper-Save the key pair: Block is offline, please replace");/* const text = args.text; const name = args.name; @@ -595,7 +595,7 @@ class WitCatFileHelper { }*/ } //删除本地变量 - delete(args) { + delete() { console.warn("文件助手——删除键值对:积木已下线,请更换\nfile_helper-Delet the key pair: Block is offline, please replace");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); @@ -655,10 +655,9 @@ class WitCatFileHelper { return jieM; } //打开文件 - openfile(args) { + openfile() { FLAG = 1; return new Promise(resolve => { - let a = 0; const input = document.createElement("input"); input.type = "file"; input.style = "display:none;"; @@ -670,15 +669,15 @@ class WitCatFileHelper { FLAG = 0; resolve(e.target.result); }; - reader.onerror = (e) => { + reader.onerror = () => { FLAG = 0; resolve(); }; reader.readAsText(file); } - window.onfocus = function () { + window.onfocus = () => { // 开始计时或者播放 - setTimeout(e => { + setTimeout(() => { if (FLAG === 1) { FLAG = 0; resolve(""); @@ -688,7 +687,7 @@ class WitCatFileHelper { }); } //设置状态 - showvar(args) { + showvar() { console.warn("文件助手——设置键状态:积木已下线,请更换\nfile_helper-Set key state: Block is offline, please replace");/* const name = args.name; let h = this.runtime.ccwAPI.getProjectUUID(); @@ -726,7 +725,7 @@ class WitCatFileHelper { }*/ } //修改别人的键 - saveother(args) { + saveother() { console.warn("文件助手——修改别人的键:积木已下线,请更换\nfile_helper-Modify other people's keys: Block is offline, please replace");/* let name = args.name; let h = args.id; @@ -762,7 +761,7 @@ class WitCatFileHelper { }*/ } //获取别人的键 - uploadother(args) { + uploadother() { console.warn("文件助手——获取别人的键:积木已下线,请更换\nfile_helper-Get other people's keys: Block is offline, please replace");/* let name = args.name; let h = args.id; @@ -791,7 +790,7 @@ class WitCatFileHelper { return "";*/ } //获取键状态 - other(args) { + other() { console.warn("文件助手——获取键状态:积木已下线,请更换\nfile_helper-Get key state: Block is offline, please replace");/* let name = args.name; let h = args.id; @@ -867,7 +866,7 @@ class WitCatFileHelper { if (args.num === "first") { num = 1; } - if (text.length >= num) { + if (text.length >= num && num > 0) { return text[num - 1].split("\r")[0]; } else { @@ -901,7 +900,7 @@ class WitCatFileHelper { return a;*/ } //键值对内容 - number(args) { + number() { console.warn("文件助手——获取键值对内容:积木已下线,请更换\nfile_helper-Get the key pair: Block is offline, please replace"); /* let a = 0; @@ -922,7 +921,7 @@ class WitCatFileHelper { return "";*/ } //可下载文本数量 - downloadnum(args) { + downloadnum() { return 3 - download; } } From 9bf81994c55435e4b60dd0331f40e50ceb325c89 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 17 Jan 2023 15:10:36 +0800 Subject: [PATCH 219/357] =?UTF-8?q?Input=201.1=20=E4=BF=AE=E5=A4=8DESLint?= =?UTF-8?q?=E6=8A=A5=E9=94=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit refactor(compile):修复了ESLint报错的部分 --- wit_cat/Input.js | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 7043c908..93391ada 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -497,7 +497,7 @@ class WitCatInput { return search.id.split("WitCatInput")[1]; else { return ( - "\{\"" + "X" + "\":\"" + ((search.style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + + "{\"" + "X" + "\":\"" + ((search.style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + "\"" + "Y" + "\":\"" + ((search.style.top.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + "\"" + "width" + "\":\"" + ((search.style.width.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + "\"" + "height" + "\":\"" + ((search.style.height.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + @@ -505,7 +505,7 @@ class WitCatInput { "\"" + "color" + "\":\"" + (search.style.color.colorHex()) + "\"," + "\"" + "prompt" + "\":\"" + (search.placeholder) + "\"," + "\"" + "font-size" + "\":\"" + (search.style.fontSize.split("px")[0]) + "\"," + - "\"" + "ID" + "\":\"" + (search.id.split("WitCatInput")[1]) + "\"\}" + "\"" + "ID" + "\":\"" + (search.id.split("WitCatInput")[1]) + "\"}" ) } } @@ -548,7 +548,7 @@ class WitCatInput { } } //删除所有文本框 - deleteallinput(args) { + deleteallinput() { let search = document.getElementsByClassName("WitCatInput"); let i = 0; for (i = search.length - 1; i >= 0; i--) { @@ -583,7 +583,7 @@ class WitCatInput { return search[args.num - 1].id.split("WitCatInput")[1]; else { return ( - "\{\"" + "X" + "\":\"" + ((search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + + "{\"" + "X" + "\":\"" + ((search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + "\"" + "Y" + "\":\"" + ((search[args.num - 1].style.top.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + "\"" + "width" + "\":\"" + ((search[args.num - 1].style.width.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + "\"" + "height" + "\":\"" + ((search[args.num - 1].style.height.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + @@ -591,7 +591,7 @@ class WitCatInput { "\"" + "color" + "\":\"" + (search[args.num - 1].style.color.colorHex()) + "\"," + "\"" + "prompt" + "\":\"" + (search[args.num - 1].placeholder) + "\"," + "\"" + "font-size" + "\":\"" + (search[args.num - 1].style.fontSize.split("px")[0]) + "\"," + - "\"" + "ID" + "\":\"" + (search[args.num - 1].id.split("WitCatInput")[1]) + "\"\}" + "\"" + "ID" + "\":\"" + (search[args.num - 1].id.split("WitCatInput")[1]) + "\"}" ) } } @@ -720,7 +720,7 @@ window.tempExt = { /* vim: set expandtab tabstop=2 shiftwidth=2: */ //颜色转换 -String.prototype.colorHex = function () { +String.prototype.colorHex = () => { // RGB颜色值的正则 var reg = /^(rgb|RGB)/; var color = this; @@ -752,7 +752,7 @@ function keyup(event) { delete keypress[event.code]; } //滚轮事件监听 -var scrollFunc = function (e) { +var scrollFunc = e => { e = e || window.event; if (e.wheelDelta) { //判断浏览器IE,谷歌滑轮事件 MouseWheel = e.wheelDelta; @@ -760,7 +760,7 @@ var scrollFunc = function (e) { MouseWheel = e.detail; } clearTimeout(timer); - timer = setTimeout(function () { + timer = setTimeout(() => { MouseWheel = 0; }, 30); }; From 9baf0f9ba3b954c927c040ee34d3ec2dd63255d8 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Tue, 17 Jan 2023 15:14:37 +0800 Subject: [PATCH 220/357] =?UTF-8?q?More=5FMouse=201.1=20=E4=BF=AE=E5=A4=8D?= =?UTF-8?q?ESLint=E6=8A=A5=E9=94=99?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit refactor(compile):修复了ESLint报错的部分 --- wit_cat/More_Mouse.js | 51 ++++++++++++------------------------------- 1 file changed, 14 insertions(+), 37 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 2fe471ef..216d41d3 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -8,14 +8,8 @@ const extensionId = "WitCatMouse"; let button = ["up", "up", "up", "up", "up"]; let xMouse = 0; let yMouse = 0; -let isMove = false, timer = null; +let timer = null; let touch = []; -let fill = 0; -let filln = 0; -let w = 0, h = 0; -let div, divs, divv, divvs, news, background, Operatinginstructions, bug, url, discord; -let wv = 0, hv = 0; -let isTouchDevice = 'ontouchstart' in document.documentElement; //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; @@ -31,19 +25,6 @@ else { } } -//添加监听器 -const config = { attributes: true, childList: true, subtree: true, attributeFilter: ['style'] }; -const callback = function (mutationsList, observer) { - if (fill === 1) { - observer.disconnect(); - fills(); - observer.observe(cvs, config); - } -}; -const observer = new MutationObserver(callback); -observer.observe(cvs, config); - - class WitCatMouse { constructor(runtime) { this.runtime = runtime; @@ -147,7 +128,7 @@ class WitCatMouse { { opcode: 'fill', blockType: "command", - text: this.formatMessage("WitCatMouse.fill") + text: this.formatMessage("WitCatMouse.fill"), hideFromPalette: true, arguments: { set: { @@ -311,7 +292,7 @@ class WitCatMouse { //右键菜单 set(args) { history.pushState(null, null, null); - cvs.parentNode.oncontextmenu = function () { + cvs.parentNode.oncontextmenu = () => { if (args.set === "true") { return true; } @@ -361,15 +342,15 @@ class WitCatMouse { } } //全屏 - fill(args) { + fill() { console.warn("全屏因浏览器兼容问题已下线,在未来修复后将会重新上线\nFull screen has been taken offline due to browser compatibility issues. It will be back online after a future fix"); } //设置分辨率 - setfill(args) { + setfill() { console.warn("全屏因浏览器兼容问题已下线,在未来修复后将会重新上线\nFull screen has been taken offline due to browser compatibility issues. It will be back online after a future fix"); } //当前分辨率 - resolution(args) { + resolution() { return cvs.height; } //设备是否支持触屏 @@ -408,10 +389,10 @@ window.tempExt = { /* vim: set expandtab tabstop=2 shiftwidth=2: */ //鼠标 -document.onmousedown = function (event) { +document.onmousedown = event => { button[event.button] = "down"; } -document.onmouseup = function (event) { +document.onmouseup = event => { button[event.button] = "up"; touch = []; } @@ -424,32 +405,28 @@ document.addEventListener("mousemove", ev => { } xMouse = ev.movementX; // 获得鼠标指针的x移动量 yMouse = ev.movementY; // 获得鼠标指针的y移动量 - isMove = true; clearTimeout(timer); - timer = setTimeout(function () { - isMove = false; + timer = setTimeout(() => { xMouse = 0; yMouse = 0; }, 30); }); //多指触控 -cvs.addEventListener('touchstart', function (e) { +cvs.addEventListener('touchstart', e => { touch = e.targetTouches; button[0] = "down"; }) -cvs.addEventListener('touchmove', function (e) { +cvs.addEventListener('touchmove', e => { xMouse = e.targetTouches[0].clientX - touch[0].clientX; // 获得手指的x移动量 yMouse = e.targetTouches[0].clientY - touch[0].clientY; // 获得手指的y移动量 - isMove = true; clearTimeout(timer); - timer = setTimeout(function () { - isMove = false; + timer = setTimeout(() => { xMouse = 0; yMouse = 0; }, 30); touch = e.targetTouches; }) -cvs.addEventListener('touchend', function (e) { +cvs.addEventListener('touchend', e => { touch = e.targetTouches; button[0] = "up"; -}) \ No newline at end of file +}) From f680ca4a6e62bad12e8f0c2199630c4bdfdff070 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Thu, 19 Jan 2023 19:47:29 +0800 Subject: [PATCH 221/357] =?UTF-8?q?witcat=20ext=201.1=20=E4=BF=AE=E5=A4=8D?= =?UTF-8?q?=E6=96=87=E6=9C=AC=E6=A1=86=E8=8E=B7=E5=8F=96=E9=A2=9C=E8=89=B2?= =?UTF-8?q?=E7=9A=84BUG=20(#64)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * 修复获取颜色的BUG fix(block):修复修改为箭头函数导致的undefined的BUG --- wit_cat/Input.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 93391ada..be49f57b 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -10,7 +10,6 @@ let keypress = {}; let lastKey = "", MouseWheel = 0; let timer; - //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; if (cvs === null) { @@ -720,7 +719,7 @@ window.tempExt = { /* vim: set expandtab tabstop=2 shiftwidth=2: */ //颜色转换 -String.prototype.colorHex = () => { +String.prototype.colorHex = function () { // RGB颜色值的正则 var reg = /^(rgb|RGB)/; var color = this; @@ -741,6 +740,7 @@ String.prototype.colorHex = () => { return String(color); } }; + //键盘事件监听 document.addEventListener("keydown", keydown); document.addEventListener("keyup", keyup); From 47e5f0a511c5ec3bea64597ba2f8222bef6cd543 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sat, 11 Feb 2023 15:39:43 +0800 Subject: [PATCH 222/357] =?UTF-8?q?witcat=20ext=202.0=20=E6=96=B0=E5=A2=9E?= =?UTF-8?q?=E7=A7=AF=E6=9C=A8=EF=BC=8C=E5=8A=9F=E8=83=BD=EF=BC=8C=E4=BC=98?= =?UTF-8?q?=E5=8C=96=E5=8A=9F=E8=83=BD=EF=BC=8C=E4=BF=AE=E5=A4=8D=E9=94=99?= =?UTF-8?q?=E5=88=AB=E5=AD=97=20(#65)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * More_Mouse 2.0 新增积木,优化功能 feat(block):修改鼠标指针样式系列积木 perf(block):锁定鼠标可以选择释放 fix(block):电脑端模拟触摸屏按下无响应的问题 * Input 2.0 新增功能,修复错别字 feat(block):获取,设置文本框透明度 feat(block):获取,设置文本框的滚动位置 feat(block):获取文本框文本高度 feat(block):获取,设置文本框的光标位置 fix(block):修复英文the写成teh的错误 * Input 2.1 新增设置状态 feat(block):新增可以设置文本框的状态为不可编辑 * More_Mouse 2.0 将doc事件绑定改为监听 fix:将doc事件绑定改为监听 * More_Mouse 2.0 漏打了一个单词 fix:漏打了一个单词 --- wit_cat/Input.js | 139 ++++++++++++++++++++- wit_cat/More_Mouse.js | 273 +++++++++++++++++++++++++++++++++++++++--- 2 files changed, 392 insertions(+), 20 deletions(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index be49f57b..4245be60 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -52,11 +52,18 @@ class WitCatInput { "WitCatInput.number.8": "字体大小", "WitCatInput.number.9": "所有(json)", "WitCatInput.number.10": "ID", + "WitCatInput.number.11": "滚动位置", + "WitCatInput.number.12": "文本高度", + "WitCatInput.number.13": "光标位置", + "WitCatInput.number.14": "透明度", "WitCatInput.key": "按下按键[type]?", "WitCatInput.keys": "按下按键[type]?", "WitCatInput.lastkey": "上次按下的键", "WitCatInput.mousewheel": "鼠标滚轮", "WitCatInput.setinput": "设置ID为[id]的文本框的[type]为[text]", + "WitCatInput.setread": "设置ID为[id]的文本框为[read]", + "WitCatInput.read.1": "可编辑", + "WitCatInput.read.2": "不可编辑", }, en: { "WitCatInput.name": "[beta]WitCat‘s Input", @@ -65,7 +72,7 @@ class WitCatInput { "WitCatInput.getinput": "get [type] with ID[id]", "WitCatInput.isinput": "is the focus on the input with ID[id]?", "WitCatInput.whatinput": "Focal position", - "WitCatInput.nowinput": "let teh focus on the input with ID[id]", + "WitCatInput.nowinput": "let the focus on the input with ID[id]", "WitCatInput.deleteallinput": "delete all input", "WitCatInput.compute": "Font size of now screen[size]", "WitCatInput.type.1": "Single line", @@ -82,11 +89,18 @@ class WitCatInput { "WitCatInput.number.8": "font-size", "WitCatInput.number.9": "all(json)", "WitCatInput.number.10": "ID", + "WitCatInput.number.11": "Rolling position", + "WitCatInput.number.12": "Text height", + "WitCatInput.number.13": "cursor position ", + "WitCatInput.number.14": "transparency", "WitCatInput.key": "Press the key[type]?", "WitCatInput.keys": "Press the key [type]?", "WitCatInput.lastkey": "last key pressed", "WitCatInput.mousewheel": "MouseWheel", "WitCatInput.setinput": "Set[type]of input whose ID is[id]to[text]", + "WitCatInput.setread": "Set the text box with ID[id]to[read]", + "WitCatInput.read.1": "editable", + "WitCatInput.read.2": "uneditable", } }) } @@ -175,6 +189,21 @@ class WitCatInput { }, }, }, + { + opcode: "setread", + blockType: "command", + text: this.formatMessage("WitCatInput.setread"), + arguments: { + id: { + type: "string", + defaultValue: "i", + }, + read: { + type: "string", + menu: "read", + }, + }, + }, { opcode: "compute", blockType: "reporter", @@ -352,6 +381,22 @@ class WitCatInput { text: this.formatMessage('WitCatInput.number.8'), value: 'font-size' }, + { + text: this.formatMessage('WitCatInput.number.11'), + value: 'rp' + }, + { + text: this.formatMessage('WitCatInput.number.12'), + value: 'th' + }, + { + text: this.formatMessage('WitCatInput.number.13'), + value: 'cp' + }, + { + text: this.formatMessage('WitCatInput.number.14'), + value: 'op' + }, { text: this.formatMessage('WitCatInput.number.9'), value: 'json' @@ -390,6 +435,28 @@ class WitCatInput { text: this.formatMessage('WitCatInput.number.8'), value: 'font-size' }, + { + text: this.formatMessage('WitCatInput.number.11'), + value: 'rp' + }, + { + text: this.formatMessage('WitCatInput.number.13'), + value: 'cp' + }, + { + text: this.formatMessage('WitCatInput.number.14'), + value: 'op' + }, + ], + read: [ + { + text: this.formatMessage('WitCatInput.read.1'), + value: 'eb' + }, + { + text: this.formatMessage('WitCatInput.read.2'), + value: 'ue' + }, ] } }; @@ -428,7 +495,7 @@ class WitCatInput { y = (y / this.runtime.stageHeight) * 100; width = (width / this.runtime.stageWidth) * 100; height = (height / this.runtime.stageHeight) * 100; - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none;color:` + args.color + `;`; + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none;color:` + args.color + `;opacity:1;`; let search = document.getElementById("WitCatInput" + args.id); if (search !== null) { if (search.name === args.type) { @@ -494,6 +561,14 @@ class WitCatInput { return search.style.fontSize.split("px")[0]; else if (args.type === "ID") return search.id.split("WitCatInput")[1]; + else if (args.type === "rp") + return search.scrollTop; + else if (args.type === "th") + return search.scrollHeight; + else if (args.type === "cp") + return JSON.stringify([search.selectionStart, search.selectionEnd]); + else if (args.type === "op") + return 100 - (search.style.opacity * 100); else { return ( "{\"" + "X" + "\":\"" + ((search.style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + @@ -504,7 +579,10 @@ class WitCatInput { "\"" + "color" + "\":\"" + (search.style.color.colorHex()) + "\"," + "\"" + "prompt" + "\":\"" + (search.placeholder) + "\"," + "\"" + "font-size" + "\":\"" + (search.style.fontSize.split("px")[0]) + "\"," + - "\"" + "ID" + "\":\"" + (search.id.split("WitCatInput")[1]) + "\"}" + "\"" + "ID" + "\":\"" + (search.id.split("WitCatInput")[1]) + "\"," + + "\"" + "Rolling position" + "\":\"" + (search.scrollTop) + "\"," + + "\"" + "Text height" + "\":\"" + (search.scrollHeight) + "\"," + + "\"" + "cursor position" + "\":\"" + (JSON.stringify([search.selectionStart, search.selectionEnd])) + "\"}" ) } } @@ -580,6 +658,14 @@ class WitCatInput { return search[args.num - 1].style.fontSize.split("px")[0]; else if (args.type === "ID") return search[args.num - 1].id.split("WitCatInput")[1]; + else if (args.type === "rp") + return search[args.num - 1].scrollTop; + else if (args.type === "th") + return search[args.num - 1].scrollHeight; + else if (args.type === "cp") + return JSON.stringify([search[args.num - 1].selectionStart, search[args.num - 1].selectionEnd]); + else if (args.type === "op") + return 100 - (search[args.num - 1].style.opacity * 100); else { return ( "{\"" + "X" + "\":\"" + ((search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + @@ -590,7 +676,10 @@ class WitCatInput { "\"" + "color" + "\":\"" + (search[args.num - 1].style.color.colorHex()) + "\"," + "\"" + "prompt" + "\":\"" + (search[args.num - 1].placeholder) + "\"," + "\"" + "font-size" + "\":\"" + (search[args.num - 1].style.fontSize.split("px")[0]) + "\"," + - "\"" + "ID" + "\":\"" + (search[args.num - 1].id.split("WitCatInput")[1]) + "\"}" + "\"" + "ID" + "\":\"" + (search[args.num - 1].id.split("WitCatInput")[1]) + "\"," + + "\"" + "Rolling position" + "\":\"" + (search[args.num - 1].scrollTop) + "\"," + + "\"" + "Text height" + "\":\"" + (search[args.num - 1].scrollHeight) + "\"," + + "\"" + "cursor position" + "\":\"" + (JSON.stringify([search[args.num - 1].selectionStart, search[args.num - 1].selectionEnd])) + "\"}" ) } } @@ -631,6 +720,8 @@ class WitCatInput { let prompt = search.placeholder; let color = search.style.color.colorHex() let size = search.style.fontSize.split("px")[0]; + let scrolltop = search.scrollTop; + let opacity = search.style.opacity; if (args.type === "X") { x = args.text; if (args.text > this.runtime.stageWidth) { @@ -683,12 +774,48 @@ class WitCatInput { else if (args.type === "font-size") { size = args.text; } - - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + size + `px;resize:none;color:` + color + `;`; + else if (args.type === "rp") { + scrolltop = args.text; + } + else if (args.type === "op") { + if (!isNaN(args.text)) { + opacity = 1 - (args.text / 100); + } + else { + opacity = 1; + } + } + else if (args.type === "cp") { + try { + if (JSON.parse(args.text).length >= 2) { + search.setSelectionRange(JSON.parse(args.text)[0], JSON.parse(args.text)[1]); + } + else { + search.setSelectionRange(args.text, args.text); + } + } + catch { + return; + } + } + let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + size + `px;resize:none;color:` + color + `;opacity:` + opacity + `;`; search.style = dom; search.value = content; search.placeholder = prompt; + search.scrollTop = scrolltop; + } + } + //设置状态 + setread(args) { + let search = document.getElementById("WitCatInput" + args.id); + if (search !== null) { + if (args.read === "eb") { + search.disabled = false; + } + else { + search.disabled = true; + } } } } diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 216d41d3..36c954c7 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -11,6 +11,35 @@ let yMouse = 0; let timer = null; let touch = []; + +//base64转blob +function base64ImgtoFile(dataurl, filename = 'file') { + const arr = dataurl.split(',') + const mime = arr[0].match(/:(.*?);/)[1] + const suffix = mime.split('/')[1] + const bstr = atob(arr[1]) + let n = bstr.length + const u8arr = new Uint8Array(n) + while (n--) { + u8arr[n] = bstr.charCodeAt(n) + } + return new File([u8arr], `${filename}.${suffix}`, { + type: mime + }) +} + +//检测是不是ico的base64 +function isBase64(str) { + let a = "^([A-Za-z0-9+/]{4})*([A-Za-z0-9+/]{4}|[A-Za-z0-9+/]{3}=|[A-Za-z0-9+/]{2}==)$"; + if (str.match(a) == null) { + return true; + } + else { + console.warn("请使用ico格式的base64文本\nPlease use base64 text in ico format"); + return false; + } +} + //找渲染cvs let cvs = document.getElementsByTagName("canvas")[0]; if (cvs === null) { @@ -42,7 +71,9 @@ class WitCatMouse { "WitCatMouse.key.3": "右键", "WitCatMouse.key.4": "前侧键", "WitCatMouse.key.5": "后侧键", - "WitCatMouse.mouseuse": "锁定鼠标", + "WitCatMouse.mouseuse": "[mouseuse]鼠标", + "WitCatMouse.mouseuse.1": "锁定", + "WitCatMouse.mouseuse.2": "释放", "WitCatMouse.acceleration": "鼠标[way]加速度", "WitCatMouse.way.1": "X", "WitCatMouse.way.2": "Y", @@ -50,6 +81,7 @@ class WitCatMouse { "WitCatTouch.num": "第[num]个手指的[type]", "WitCatTouch.type.1": "X", "WitCatTouch.type.2": "Y", + "WitCatTouch.type.3": "ID", "WitCatMouse.fill": "[set]沉浸式全屏", "WitCatMouse.fillask.1": "作品请求沉浸式全屏,是否同意?\n", "WitCatMouse.fillask.2": "/3次连续拒绝后将不再提示\n您仍可以使用 ctrl+shift+alt 切换沉浸式全屏状态", @@ -59,6 +91,21 @@ class WitCatMouse { "WitCatMouse.types.1": "触屏", "WitCatMouse.types.2": "鼠标", "WitCatMouse.IsMobile": "移动设备?", + "WitCatMouse.cursor": "更改鼠标的样式为[cursor]", + "WitCatMouse.cursor.1": "默认", + "WitCatMouse.cursor.2": "光标", + "WitCatMouse.cursor.3": "十字移动", + "WitCatMouse.cursor.4": "上下移动", + "WitCatMouse.cursor.5": "左右移动", + "WitCatMouse.cursor.6": "禁止", + "WitCatMouse.cursor.7": "选择", + "WitCatMouse.cursor.8": "加载", + "WitCatMouse.cursor.9": "缓慢加载", + "WitCatMouse.cursor.10": "帮助", + "WitCatMouse.cursor.11": "横向光标", + "WitCatMouse.cursor.12": "精准选择", + "WitCatMouse.cursorurl": "更改鼠标的样式为X[x]Y[y]base64[text]", + "WitCatMouse.url": "上传ico并获得base64", }, en: { "WitCatMouse.name": "[beta]WitCat’s Mouse", @@ -73,7 +120,9 @@ class WitCatMouse { "WitCatMouse.key.3": "right", "WitCatMouse.key.4": "front", "WitCatMouse.key.5": "back", - "WitCatMouse.mouseuse": "lock mouse", + "WitCatMouse.mouseuse": "[mouseuse]mouse", + "WitCatMouse.mouseuse.1": "Lock", + "WitCatMouse.mouseuse.2": "Release", "WitCatMouse.acceleration": "mouse[way]acceleration", "WitCatMouse.way.1": "X", "WitCatMouse.way.2": "Y", @@ -82,6 +131,7 @@ class WitCatMouse { "WitCatTouch.num": "the[num]finger`s[type]", "WitCatTouch.type.1": "X", "WitCatTouch.type.2": "Y", + "WitCatTouch.type.3": "ID", "WitCatMouse.fill": "[set]immersive full-screen", "WitCatMouse.fillask.1": "The project requests to turn on immersive full-screen, agree or not?\nWill stop asking if you keep on to reject for ", "WitCatMouse.fillask.2": "/3 times\nYou can also use Ctrl+Shift+Alt to toggle immersive full-screen later.", @@ -91,6 +141,21 @@ class WitCatMouse { "WitCatMouse.types.1": "Touch screen", "WitCatMouse.types.2": "mouse", "WitCatMouse.IsMobile": "Mobile devices?", + "WitCatMouse.cursor": "Change mouse style to[cursor]", + "WitCatMouse.cursor.1": "Default", + "WitCatMouse.cursor.2": "Cursor", + "WitCatMouse.cursor.3": "Cross Move", + "WitCatMouse.cursor.4": "Move up and down", + "WitCatMouse.cursor.5": "Move left and right", + "WitCatMouse.cursor.6": "Prohibit", + "WitCatMouse.cursor.7": "Select", + "WitCatMouse.cursor.8": "Load", + "WitCatMouse.cursor.9": "Slow loading", + "WitCatMouse.cursor.10": "Help", + "WitCatMouse.cursor.11": "Landscape cursor", + "WitCatMouse.cursor.12": "Precise selection", + "WitCatMouse.cursorurl": "Change the style of the mouse to X[x]Y[y]base64[text]", + "WitCatMouse.url": "Upload ico and get base64", } }) } @@ -111,7 +176,7 @@ class WitCatMouse { blockIconURI: _icon, menuIconURI: _icon, color1: "#8eace1", - color2: "#ffffff", + color2: "#86a2d4", blocks: [ { opcode: 'setfill', @@ -166,11 +231,51 @@ class WitCatMouse { }, }, }, + { + blockType: "button", + text: this.formatMessage('WitCatMouse.url'), + onClick: this.url, + }, + { + opcode: "cursor", + blockType: "command", + text: this.formatMessage("WitCatMouse.cursor"), + arguments: { + cursor: { + type: "string", + menu: "cursor", + }, + }, + }, + { + opcode: "cursorurl", + blockType: "command", + text: this.formatMessage("WitCatMouse.cursorurl"), + arguments: { + x: { + type: "string", + defaultValue: "0", + }, + y: { + type: "string", + defaultValue: "0", + }, + text: { + type: "string", + defaultValue: "base64:ico", + }, + }, + }, { opcode: "mouseuse", blockType: "command", text: this.formatMessage("WitCatMouse.mouseuse"), - arguments: {}, + arguments: { + mouseuse: { + type: "string", + menu: "mouseuse", + }, + }, }, { opcode: "acceleration", @@ -246,6 +351,59 @@ class WitCatMouse { value: '4' }, ], + cursor: { + acceptReporters: true, + items: [ + { + text: this.formatMessage('WitCatMouse.cursor.1'), + value: 'default' + }, + { + text: this.formatMessage('WitCatMouse.cursor.2'), + value: 'text' + }, + { + text: this.formatMessage('WitCatMouse.cursor.3'), + value: 'move' + }, + { + text: this.formatMessage('WitCatMouse.cursor.4'), + value: 'n-resize' + }, + { + text: this.formatMessage('WitCatMouse.cursor.5'), + value: 'e-resize' + }, + { + text: this.formatMessage('WitCatMouse.cursor.6'), + value: 'not-allowed' + }, + { + text: this.formatMessage('WitCatMouse.cursor.7'), + value: 'pointer' + }, + { + text: this.formatMessage('WitCatMouse.cursor.8'), + value: 'progress' + }, + { + text: this.formatMessage('WitCatMouse.cursor.9'), + value: 'wait' + }, + { + text: this.formatMessage('WitCatMouse.cursor.10'), + value: 'help' + }, + { + text: this.formatMessage('WitCatMouse.cursor.11'), + value: 'vertical-text' + }, + { + text: this.formatMessage('WitCatMouse.cursor.12'), + value: 'crosshair' + }, + ], + }, set: [ { text: this.formatMessage('WitCatMouse.set.1'), @@ -275,6 +433,10 @@ class WitCatMouse { text: this.formatMessage('WitCatTouch.type.2'), value: "y" }, + { + text: this.formatMessage('WitCatTouch.type.3'), + value: "ID" + }, ], types: [ { @@ -286,6 +448,16 @@ class WitCatMouse { value: "onmousedown" }, ], + mouseuse: [ + { + text: this.formatMessage('WitCatMouse.mouseuse.1'), + value: "lock" + }, + { + text: this.formatMessage('WitCatMouse.mouseuse.2'), + value: "release" + }, + ], } }; } @@ -311,8 +483,13 @@ class WitCatMouse { } } //控制鼠标 - mouseuse() { - cvs.parentNode.requestPointerLock(); + mouseuse(args) { + if (args.mouseuse === "release") { + document.exitPointerLock(); + } + else { + cvs.parentNode.requestPointerLock(); + } } //鼠标移动量 acceleration(args) { @@ -333,9 +510,12 @@ class WitCatMouse { if (args.type === "x") { return this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.offsetWidth); } - else { + else if (args.type === "y") { return this.runtime.stageHeight * ((touch[args.num - 1].clientY - cvs.getBoundingClientRect().top) / cvs.offsetHeight); } + else { + return touch[args.num - 1].identifier; + } } else { return null; @@ -361,6 +541,68 @@ class WitCatMouse { IsMobile() { return /Android|iPhone|iPad|iPod|BlackBerry|webOS|Windows Phone|SymbianOS|IEMobile|Opera Mini/i.test(navigator.userAgent); } + //设置光标 + cursor(args) { + cvs.parentNode.parentNode.parentNode.style.cursor = args.cursor; + } + //设置光标为url + cursorurl(args) { + if (isBase64(args.text)) { + const img = args.text; + let file = base64ImgtoFile(img); // 得到File对象 + let imgUrl = window.webkitURL.createObjectURL(file) || window.URL.createObjectURL(file) // imgUrl图片网络路径 + cvs.parentNode.parentNode.parentNode.style.cursor = "url(" + imgUrl + ")" + args.x + " " + args.y + ",auto"; + return; + } + } + //打开ico文件 + url() { + return new Promise(resolve => { + const input = document.createElement("input"); + input.type = "file"; + input.style = "display:none;"; + input.accept = ".ico"; + input.click(); + input.onchange = () => { + const reader = new FileReader(); + const readers = new FileReader(); + const file = input.files[0]; + reader.onload = (e) => { + navigator.clipboard.writeText(e.currentTarget.result); + alert("base64代码已经被复制到剪切板,可以粘贴以使用\nThe base64 code has been copied to the clipboard and can be pasted for use"); + resolve(e.target.result); + }; + reader.onerror = () => { + resolve(); + }; + readers.readAsArrayBuffer(file); + + readers.onload = (e) => { + if (file.name.split('.')[file.name.split('.').length - 1] == "ico") { + var uri = e.target.result; + console.log(uri.byteLength / 1024 + " KB"); + if (uri.byteLength / 1024 <= 10) { + reader.readAsDataURL(file); + } + else { + console.warn("文件过大,可能导致工程文件崩溃!!!\nThe file is too large, may cause the project file crash!!!"); + alert("文件过大,可能导致工程文件崩溃!!!\nThe file is too large, may cause the project file crash!!!"); + } + } + else { + console.warn("请选择*.ico文件\nPlease select the *.ico file"); + alert("请选择*.ico文件\nPlease select the *.ico file"); + } + }; + } + window.onfocus = () => { + // 开始计时或者播放 + setTimeout(() => { + resolve(""); + }, 1000); + } + }); + } } window.tempExt = { @@ -389,16 +631,19 @@ window.tempExt = { /* vim: set expandtab tabstop=2 shiftwidth=2: */ //鼠标 -document.onmousedown = event => { - button[event.button] = "down"; -} -document.onmouseup = event => { - button[event.button] = "up"; +document.addEventListener('mousedown', e => { + button[e.button] = "down"; + if (button[0] === "down") { + touch = JSON.parse("[{\"clientX\":\"" + e.clientX + "\",\"clientY\":\"" + e.clientY + "\",\"identifier\":\"mouse\"}]"); + } +}) +document.addEventListener('mouseup', e => { + button[e.button] = "up"; touch = []; -} +}) document.addEventListener("mousemove", ev => { if (button[0] === "down") { - touch = JSON.parse("[{\"clientX\":\"" + ev.clientX + "\",\"clientY\":\"" + ev.clientY + "\"}]"); + touch = JSON.parse("[{\"clientX\":\"" + ev.clientX + "\",\"clientY\":\"" + ev.clientY + "\",\"identifier\":\"mouse\"}]"); } else { touch = []; From 01543e8905a5cf4b223dd6ce90e3b98a7805480b Mon Sep 17 00:00:00 2001 From: Nick Date: Wed, 15 Feb 2023 14:48:42 +0800 Subject: [PATCH 223/357] Create LICENSE (#68) --- LICENSE | 504 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 504 insertions(+) create mode 100644 LICENSE diff --git a/LICENSE b/LICENSE new file mode 100644 index 00000000..8000a6fa --- /dev/null +++ b/LICENSE @@ -0,0 +1,504 @@ + GNU LESSER GENERAL PUBLIC LICENSE + Version 2.1, February 1999 + + Copyright (C) 1991, 1999 Free Software Foundation, Inc. + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + +[This is the first released version of the Lesser GPL. It also counts + as the successor of the GNU Library Public License, version 2, hence + the version number 2.1.] + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +Licenses are intended to guarantee your freedom to share and change +free software--to make sure the software is free for all its users. + + This license, the Lesser General Public License, applies to some +specially designated software packages--typically libraries--of the +Free Software Foundation and other authors who decide to use it. You +can use it too, but we suggest you first think carefully about whether +this license or the ordinary General Public License is the better +strategy to use in any particular case, based on the explanations below. + + When we speak of free software, we are referring to freedom of use, +not price. Our General Public Licenses are designed to make sure that +you have the freedom to distribute copies of free software (and charge +for this service if you wish); that you receive source code or can get +it if you want it; that you can change the software and use pieces of +it in new free programs; and that you are informed that you can do +these things. + + To protect your rights, we need to make restrictions that forbid +distributors to deny you these rights or to ask you to surrender these +rights. These restrictions translate to certain responsibilities for +you if you distribute copies of the library or if you modify it. + + For example, if you distribute copies of the library, whether gratis +or for a fee, you must give the recipients all the rights that we gave +you. You must make sure that they, too, receive or can get the source +code. If you link other code with the library, you must provide +complete object files to the recipients, so that they can relink them +with the library after making changes to the library and recompiling +it. And you must show them these terms so they know their rights. + + We protect your rights with a two-step method: (1) we copyright the +library, and (2) we offer you this license, which gives you legal +permission to copy, distribute and/or modify the library. + + To protect each distributor, we want to make it very clear that +there is no warranty for the free library. Also, if the library is +modified by someone else and passed on, the recipients should know +that what they have is not the original version, so that the original +author's reputation will not be affected by problems that might be +introduced by others. + + Finally, software patents pose a constant threat to the existence of +any free program. We wish to make sure that a company cannot +effectively restrict the users of a free program by obtaining a +restrictive license from a patent holder. Therefore, we insist that +any patent license obtained for a version of the library must be +consistent with the full freedom of use specified in this license. + + Most GNU software, including some libraries, is covered by the +ordinary GNU General Public License. This license, the GNU Lesser +General Public License, applies to certain designated libraries, and +is quite different from the ordinary General Public License. We use +this license for certain libraries in order to permit linking those +libraries into non-free programs. + + When a program is linked with a library, whether statically or using +a shared library, the combination of the two is legally speaking a +combined work, a derivative of the original library. The ordinary +General Public License therefore permits such linking only if the +entire combination fits its criteria of freedom. The Lesser General +Public License permits more lax criteria for linking other code with +the library. + + We call this license the "Lesser" General Public License because it +does Less to protect the user's freedom than the ordinary General +Public License. It also provides other free software developers Less +of an advantage over competing non-free programs. These disadvantages +are the reason we use the ordinary General Public License for many +libraries. However, the Lesser license provides advantages in certain +special circumstances. + + For example, on rare occasions, there may be a special need to +encourage the widest possible use of a certain library, so that it becomes +a de-facto standard. To achieve this, non-free programs must be +allowed to use the library. A more frequent case is that a free +library does the same job as widely used non-free libraries. In this +case, there is little to gain by limiting the free library to free +software only, so we use the Lesser General Public License. + + In other cases, permission to use a particular library in non-free +programs enables a greater number of people to use a large body of +free software. For example, permission to use the GNU C Library in +non-free programs enables many more people to use the whole GNU +operating system, as well as its variant, the GNU/Linux operating +system. + + Although the Lesser General Public License is Less protective of the +users' freedom, it does ensure that the user of a program that is +linked with the Library has the freedom and the wherewithal to run +that program using a modified version of the Library. + + The precise terms and conditions for copying, distribution and +modification follow. Pay close attention to the difference between a +"work based on the library" and a "work that uses the library". The +former contains code derived from the library, whereas the latter must +be combined with the library in order to run. + + GNU LESSER GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License Agreement applies to any software library or other +program which contains a notice placed by the copyright holder or +other authorized party saying it may be distributed under the terms of +this Lesser General Public License (also called "this License"). +Each licensee is addressed as "you". + + A "library" means a collection of software functions and/or data +prepared so as to be conveniently linked with application programs +(which use some of those functions and data) to form executables. + + The "Library", below, refers to any such software library or work +which has been distributed under these terms. A "work based on the +Library" means either the Library or any derivative work under +copyright law: that is to say, a work containing the Library or a +portion of it, either verbatim or with modifications and/or translated +straightforwardly into another language. (Hereinafter, translation is +included without limitation in the term "modification".) + + "Source code" for a work means the preferred form of the work for +making modifications to it. For a library, complete source code means +all the source code for all modules it contains, plus any associated +interface definition files, plus the scripts used to control compilation +and installation of the library. + + Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running a program using the Library is not restricted, and output from +such a program is covered only if its contents constitute a work based +on the Library (independent of the use of the Library in a tool for +writing it). Whether that is true depends on what the Library does +and what the program that uses the Library does. + + 1. You may copy and distribute verbatim copies of the Library's +complete source code as you receive it, in any medium, provided that +you conspicuously and appropriately publish on each copy an +appropriate copyright notice and disclaimer of warranty; keep intact +all the notices that refer to this License and to the absence of any +warranty; and distribute a copy of this License along with the +Library. + + You may charge a fee for the physical act of transferring a copy, +and you may at your option offer warranty protection in exchange for a +fee. + + 2. You may modify your copy or copies of the Library or any portion +of it, thus forming a work based on the Library, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) The modified work must itself be a software library. + + b) You must cause the files modified to carry prominent notices + stating that you changed the files and the date of any change. + + c) You must cause the whole of the work to be licensed at no + charge to all third parties under the terms of this License. + + d) If a facility in the modified Library refers to a function or a + table of data to be supplied by an application program that uses + the facility, other than as an argument passed when the facility + is invoked, then you must make a good faith effort to ensure that, + in the event an application does not supply such function or + table, the facility still operates, and performs whatever part of + its purpose remains meaningful. + + (For example, a function in a library to compute square roots has + a purpose that is entirely well-defined independent of the + application. Therefore, Subsection 2d requires that any + application-supplied function or table used by this function must + be optional: if the application does not supply it, the square + root function must still compute square roots.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Library, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Library, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote +it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Library. + +In addition, mere aggregation of another work not based on the Library +with the Library (or with a work based on the Library) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may opt to apply the terms of the ordinary GNU General Public +License instead of this License to a given copy of the Library. To do +this, you must alter all the notices that refer to this License, so +that they refer to the ordinary GNU General Public License, version 2, +instead of to this License. (If a newer version than version 2 of the +ordinary GNU General Public License has appeared, then you can specify +that version instead if you wish.) Do not make any other change in +these notices. + + Once this change is made in a given copy, it is irreversible for +that copy, so the ordinary GNU General Public License applies to all +subsequent copies and derivative works made from that copy. + + This option is useful when you wish to copy part of the code of +the Library into a program that is not a library. + + 4. You may copy and distribute the Library (or a portion or +derivative of it, under Section 2) in object code or executable form +under the terms of Sections 1 and 2 above provided that you accompany +it with the complete corresponding machine-readable source code, which +must be distributed under the terms of Sections 1 and 2 above on a +medium customarily used for software interchange. + + If distribution of object code is made by offering access to copy +from a designated place, then offering equivalent access to copy the +source code from the same place satisfies the requirement to +distribute the source code, even though third parties are not +compelled to copy the source along with the object code. + + 5. A program that contains no derivative of any portion of the +Library, but is designed to work with the Library by being compiled or +linked with it, is called a "work that uses the Library". Such a +work, in isolation, is not a derivative work of the Library, and +therefore falls outside the scope of this License. + + However, linking a "work that uses the Library" with the Library +creates an executable that is a derivative of the Library (because it +contains portions of the Library), rather than a "work that uses the +library". The executable is therefore covered by this License. +Section 6 states terms for distribution of such executables. + + When a "work that uses the Library" uses material from a header file +that is part of the Library, the object code for the work may be a +derivative work of the Library even though the source code is not. +Whether this is true is especially significant if the work can be +linked without the Library, or if the work is itself a library. The +threshold for this to be true is not precisely defined by law. + + If such an object file uses only numerical parameters, data +structure layouts and accessors, and small macros and small inline +functions (ten lines or less in length), then the use of the object +file is unrestricted, regardless of whether it is legally a derivative +work. (Executables containing this object code plus portions of the +Library will still fall under Section 6.) + + Otherwise, if the work is a derivative of the Library, you may +distribute the object code for the work under the terms of Section 6. +Any executables containing that work also fall under Section 6, +whether or not they are linked directly with the Library itself. + + 6. As an exception to the Sections above, you may also combine or +link a "work that uses the Library" with the Library to produce a +work containing portions of the Library, and distribute that work +under terms of your choice, provided that the terms permit +modification of the work for the customer's own use and reverse +engineering for debugging such modifications. + + You must give prominent notice with each copy of the work that the +Library is used in it and that the Library and its use are covered by +this License. You must supply a copy of this License. If the work +during execution displays copyright notices, you must include the +copyright notice for the Library among them, as well as a reference +directing the user to the copy of this License. Also, you must do one +of these things: + + a) Accompany the work with the complete corresponding + machine-readable source code for the Library including whatever + changes were used in the work (which must be distributed under + Sections 1 and 2 above); and, if the work is an executable linked + with the Library, with the complete machine-readable "work that + uses the Library", as object code and/or source code, so that the + user can modify the Library and then relink to produce a modified + executable containing the modified Library. (It is understood + that the user who changes the contents of definitions files in the + Library will not necessarily be able to recompile the application + to use the modified definitions.) + + b) Use a suitable shared library mechanism for linking with the + Library. A suitable mechanism is one that (1) uses at run time a + copy of the library already present on the user's computer system, + rather than copying library functions into the executable, and (2) + will operate properly with a modified version of the library, if + the user installs one, as long as the modified version is + interface-compatible with the version that the work was made with. + + c) Accompany the work with a written offer, valid for at + least three years, to give the same user the materials + specified in Subsection 6a, above, for a charge no more + than the cost of performing this distribution. + + d) If distribution of the work is made by offering access to copy + from a designated place, offer equivalent access to copy the above + specified materials from the same place. + + e) Verify that the user has already received a copy of these + materials or that you have already sent this user a copy. + + For an executable, the required form of the "work that uses the +Library" must include any data and utility programs needed for +reproducing the executable from it. However, as a special exception, +the materials to be distributed need not include anything that is +normally distributed (in either source or binary form) with the major +components (compiler, kernel, and so on) of the operating system on +which the executable runs, unless that component itself accompanies +the executable. + + It may happen that this requirement contradicts the license +restrictions of other proprietary libraries that do not normally +accompany the operating system. Such a contradiction means you cannot +use both them and the Library together in an executable that you +distribute. + + 7. You may place library facilities that are a work based on the +Library side-by-side in a single library together with other library +facilities not covered by this License, and distribute such a combined +library, provided that the separate distribution of the work based on +the Library and of the other library facilities is otherwise +permitted, and provided that you do these two things: + + a) Accompany the combined library with a copy of the same work + based on the Library, uncombined with any other library + facilities. This must be distributed under the terms of the + Sections above. + + b) Give prominent notice with the combined library of the fact + that part of it is a work based on the Library, and explaining + where to find the accompanying uncombined form of the same work. + + 8. You may not copy, modify, sublicense, link with, or distribute +the Library except as expressly provided under this License. Any +attempt otherwise to copy, modify, sublicense, link with, or +distribute the Library is void, and will automatically terminate your +rights under this License. However, parties who have received copies, +or rights, from you under this License will not have their licenses +terminated so long as such parties remain in full compliance. + + 9. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Library or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Library (or any work based on the +Library), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Library or works based on it. + + 10. Each time you redistribute the Library (or any work based on the +Library), the recipient automatically receives a license from the +original licensor to copy, distribute, link with or modify the Library +subject to these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties with +this License. + + 11. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Library at all. For example, if a patent +license would not permit royalty-free redistribution of the Library by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Library. + +If any portion of this section is held invalid or unenforceable under any +particular circumstance, the balance of the section is intended to apply, +and the section as a whole is intended to apply in other circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 12. If the distribution and/or use of the Library is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Library under this License may add +an explicit geographical distribution limitation excluding those countries, +so that distribution is permitted only in or among countries not thus +excluded. In such case, this License incorporates the limitation as if +written in the body of this License. + + 13. The Free Software Foundation may publish revised and/or new +versions of the Lesser General Public License from time to time. +Such new versions will be similar in spirit to the present version, +but may differ in detail to address new problems or concerns. + +Each version is given a distinguishing version number. If the Library +specifies a version number of this License which applies to it and +"any later version", you have the option of following the terms and +conditions either of that version or of any later version published by +the Free Software Foundation. If the Library does not specify a +license version number, you may choose any version ever published by +the Free Software Foundation. + + 14. If you wish to incorporate parts of the Library into other free +programs whose distribution conditions are incompatible with these, +write to the author to ask for permission. For software which is +copyrighted by the Free Software Foundation, write to the Free +Software Foundation; we sometimes make exceptions for this. Our +decision will be guided by the two goals of preserving the free status +of all derivatives of our free software and of promoting the sharing +and reuse of software generally. + + NO WARRANTY + + 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO +WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. +EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR +OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY +KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE +LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME +THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN +WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY +AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU +FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR +CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE +LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING +RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A +FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF +SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH +DAMAGES. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Libraries + + If you develop a new library, and you want it to be of the greatest +possible use to the public, we recommend making it free software that +everyone can redistribute and change. You can do so by permitting +redistribution under these terms (or, alternatively, under the terms of the +ordinary General Public License). + + To apply these terms, attach the following notices to the library. It is +safest to attach them to the start of each source file to most effectively +convey the exclusion of warranty; and each file should have at least the +"copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with this library; if not, write to the Free Software + Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 + USA + +Also add information on how to contact you by electronic and paper mail. + +You should also get your employer (if you work as a programmer) or your +school, if any, to sign a "copyright disclaimer" for the library, if +necessary. Here is a sample; alter the names: + + Yoyodyne, Inc., hereby disclaims all copyright interest in the + library `Frob' (a library for tweaking knobs) written by James Random + Hacker. + + , 1 April 1990 + Ty Coon, President of Vice + +That's all there is to it! From 3a7fecbc20c0ca15ed7631a5239ad3b6a968eb38 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=87=8C?= Date: Wed, 1 Mar 2023 10:07:09 +0800 Subject: [PATCH 224/357] =?UTF-8?q?:sparkles:=20=E6=B7=BB=E5=8A=A0=20Furry?= =?UTF-8?q?R/not.js=20(#69)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- FurryR/README.md | 9 + FurryR/not.js.js | 434 +++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 443 insertions(+) create mode 100644 FurryR/README.md create mode 100644 FurryR/not.js.js diff --git a/FurryR/README.md b/FurryR/README.md new file mode 100644 index 00000000..e90416ba --- /dev/null +++ b/FurryR/README.md @@ -0,0 +1,9 @@ +# FurryR 的插件仓库 + +目前包含的插件: + +- not.js (MIT): [GitHub](https://github.com/FurryR/not.js) + +--- + +_所有插件均遵循源仓库的开源协议开源。_ diff --git a/FurryR/not.js.js b/FurryR/not.js.js new file mode 100644 index 00000000..e824c011 --- /dev/null +++ b/FurryR/not.js.js @@ -0,0 +1,434 @@ +const iconURL = '', insetIconURL = ''; +class NotJS { + constructor(runtime) { + const fmt = runtime.getFormatMessage({ + 'zh-cn': { + 'notjs.extensionName': 'Not.js', + 'notjs.getHelp': '打开帮助文档', + 'notjs.title.parse': '解析', + 'notjs.parseJSON': '解析 [json]', + 'notjs.fromString': '解析字符串 [str] 为 JSON', + 'notjs.fromBoolean': '解析布尔值 [bool] 为 JSON', + 'notjs.title.type': '类型', + 'notjs.asString': '[json] 作为字符串', + 'notjs.asBoolean': '[json] 作为布尔值', + 'notjs.getType': '[json] 的类型', + 'notjs.title.member': '成员', + 'notjs.getMember': '[json] 的成员 [member]', + 'notjs.setMember': '设定 [json] 的成员 [member] 为 [value]', + 'notjs.removeMember': '删除 [json] 的成员 [member]', + 'notjs.exists': '[json] 存在成员 [member]?', + 'notjs.length': '[json] 的长度', + 'notjs.keys': '[json] 的全部键', + 'notjs.values': '[json] 的全部值' + }, + en: { + 'notjs.extensionName': 'Not.js', + 'notjs.getHelp': 'Open Documentation', + 'notjs.title.parse': 'Parse', + 'notjs.parseJSON': 'parse [json]', + 'notjs.fromString': 'parse string [str] as JSON', + 'notjs.fromBoolean': 'parse boolean [bool] as JSON', + 'notjs.title.type': 'Type', + 'notjs.asString': '[json] as string', + 'notjs.asBoolean': '[json] as boolean', + 'notjs.getType': 'type of [json]', + 'notjs.title.member': 'Member', + 'notjs.getMember': 'get member [member] of [json]', + 'notjs.setMember': 'set member [member] of [json] to [value]', + 'notjs.removeMember': 'remove member [member] of [json]', + 'notjs.exists': 'member [member] exists in [json]?', + 'notjs.length': 'length of [json]', + 'notjs.keys': 'keys of [json]', + 'notjs.values': 'values of [json]' + } + }); + this.formatMessage = (id) => fmt({ id, default: id, description: id }); + } + getInfo() { + return { + id: 'FurryR.NotJS', + name: this.formatMessage('notjs.extensionName'), + color1: '#8A8A8A', + menuIconURI: insetIconURL, + blockIconURI: insetIconURL, + blocks: [ + { + blockType: 'button', + text: this.formatMessage('notjs.getHelp'), + onClick: this.getHelp + }, + `---${this.formatMessage('notjs.title.parse')}`, + { + opcode: 'parseJSON', + blockType: 'reporter', + text: this.formatMessage('notjs.parseJSON'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + } + } + }, + { + opcode: 'fromString', + blockType: 'reporter', + text: this.formatMessage('notjs.fromString'), + arguments: { + str: { + type: 'string', + defaultValue: 'Hello World' + } + } + }, + { + opcode: 'fromBoolean', + blockType: 'reporter', + text: this.formatMessage('notjs.fromBoolean'), + arguments: { + bool: { + type: 'Boolean' + } + } + }, + `---${this.formatMessage('notjs.title.type')}`, + { + opcode: 'asString', + blockType: 'reporter', + text: this.formatMessage('notjs.asString'), + arguments: { + json: { + type: 'string', + defaultValue: '""' + } + } + }, + { + opcode: 'asBoolean', + blockType: 'Boolean', + text: this.formatMessage('notjs.asBoolean'), + arguments: { + json: { + type: 'string', + defaultValue: 'true' + } + } + }, + { + opcode: 'getType', + blockType: 'reporter', + text: this.formatMessage('notjs.getType'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + } + } + }, + `---${this.formatMessage('notjs.title.member')}`, + { + opcode: 'getMember', + blockType: 'reporter', + text: this.formatMessage('notjs.getMember'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + }, + member: { + type: 'string', + defaultValue: 'a' + } + } + }, + { + opcode: 'setMember', + blockType: 'reporter', + text: this.formatMessage('notjs.setMember'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + }, + member: { + type: 'string', + defaultValue: 'a' + }, + value: { + type: 'string', + defaultValue: '{}' + } + } + }, + { + opcode: 'removeMember', + blockType: 'reporter', + text: this.formatMessage('notjs.removeMember'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + }, + member: { + type: 'string', + defaultValue: 'a' + } + } + }, + { + opcode: 'exists', + blockType: 'Boolean', + text: this.formatMessage('notjs.exists'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + }, + member: { + type: 'string', + defaultValue: 'a' + } + } + }, + { + opcode: 'length', + blockType: 'reporter', + text: this.formatMessage('notjs.length'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + } + } + }, + { + opcode: 'keys', + blockType: 'reporter', + text: this.formatMessage('notjs.keys'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + } + } + }, + { + opcode: 'values', + blockType: 'reporter', + text: this.formatMessage('notjs.values'), + arguments: { + json: { + type: 'string', + defaultValue: '{}' + } + } + } + ] + }; + } + _parseJSON(json) { + try { + return JSON.parse(json); + } + catch (_) { + return undefined; + } + } + getHelp() { + window.open('https://www.ccw.site/post/9cb88c31-c4eb-4244-a412-625e33ca6882'); + } + parseJSON({ json }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + return JSON.stringify(v); + } + fromString({ str }) { + return JSON.stringify(str); + } + fromBoolean({ bool }) { + return bool == undefined ? undefined : JSON.stringify(bool); + } + asString({ json }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (v == null || + typeof v == 'string' || + typeof v == 'boolean' || + typeof v == 'number') { + return String(v); + } + else + return JSON.stringify(v); + } + asBoolean({ json }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (typeof v == 'boolean' || typeof v == 'number') { + return Boolean(v); + } + else if (v == null) { + return false; + } + else if (typeof v == 'string' || v instanceof Array) + return v.length != 0; + else + return Object.keys(v).length != 0; + } + getType({ json }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (v == null) { + return 'null'; + } + else if (typeof v == 'string' || + typeof v == 'boolean' || + typeof v == 'number') { + return typeof v; + } + else if (v instanceof Array) { + return 'array'; + } + else + return 'object'; + } + getMember({ json, member }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (v instanceof Array || typeof v == 'string') { + const idx = parseInt(member); + if (v[idx] === undefined) + return 'null'; + return JSON.stringify(v[idx]); + } + else if (v instanceof Object) { + const x = v; + if (x[member] === undefined) + return 'null'; + return JSON.stringify(x[member]); + } + else { + return 'null'; + } + } + setMember({ json, member, value }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (v instanceof Array) { + const idx = parseInt(member); + if (isNaN(idx) || idx < 0) + return JSON.stringify(v); + const c = this._parseJSON(value); + if (c === undefined) + return; + v[idx] = c; + } + else if (v instanceof Object) { + const c = this._parseJSON(value); + if (c === undefined) + return; + v[member] = c; + } + else { + return JSON.stringify(v); + } + return JSON.stringify(v); + } + removeMember({ json, member }) { + let v = this._parseJSON(json); + if (v === undefined) + return; + if (v instanceof Array) { + const idx = parseInt(member); + if (v[idx] !== undefined) { + if (idx == v.length - 1) { + v = v.slice(0, -1); + } + else if (idx == 0) { + v = v.slice(1); + } + else { + v[idx] = null; + } + } + } + else if (v instanceof Object) { + delete v[member]; + } + else { + return JSON.stringify(v); + } + return JSON.stringify(v); + } + exists({ json, member }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (v instanceof Array) { + return v[parseInt(member)] !== undefined; + } + else if (v instanceof Object) { + return v[member] !== undefined; + } + return false; + } + length({ json }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (v instanceof Array || typeof v == 'string') { + return v.length; + } + else if (v instanceof Object) { + return Object.keys(v).length; + } + return; + } + keys({ json }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (typeof v == 'string' || v instanceof Object) { + return JSON.stringify(Object.keys(v)); + } + return '[]'; + } + values({ json }) { + const v = this._parseJSON(json); + if (v === undefined) + return; + if (v instanceof Object || typeof v == 'string') { + return JSON.stringify(Object.values(v)); + } + return '[]'; + } +} +void (window.tempExt = { + Extension: NotJS, + info: { + name: 'notjs.extensionName', + description: 'notjs.description', + extensionId: 'FurryR.NotJS', + iconURL, + insetIconURL, + featured: true, + disabled: false, + collaborator: 'FurryR @ Simplicity Studio', + doc: 'https://www.ccw.site/post/9cb88c31-c4eb-4244-a412-625e33ca6882' + }, + l10n: { + 'zh-cn': { + 'notjs.extensionName': 'Not.js', + 'notjs.description': '次世代的 Gandi JSON 处理器。' + }, + en: { + 'notjs.extensionName': 'Not.js', + 'notjs.description': 'Next-generation JSON processor for Gandi IDE.' + } + } +}); From 8cd383adf149ac05a93a2f8df4224abd2cb48a67 Mon Sep 17 00:00:00 2001 From: CatRanger Date: Thu, 11 May 2023 13:42:43 +0800 Subject: [PATCH 225/357] Update README.md --- README.md | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/README.md b/README.md index e6d200e6..a8400b6e 100644 --- a/README.md +++ b/README.md @@ -35,8 +35,7 @@ Extensions in this repo will have full access to vm and runtime utilities allows When your complete testing,you can apply for publish extensions to CCW Extension Library. All CCW Creator can use it in their project. ### Apply for CCW collaborator - -contact [wangjiaming@ccw.site](mailto:wangjiaming@ccw.site) +- fork this repo and submit your PR。 ## How to develop ### Define a extension From b8dc464bc47d7bb87b8c7229c35fce4c48bb1386 Mon Sep 17 00:00:00 2001 From: Arkos123 <101853866+Arkos123@users.noreply.github.com> Date: Sun, 28 May 2023 15:07:38 +0800 Subject: [PATCH 226/357] =?UTF-8?q?Arkos=E6=8B=93=E5=B1=95=20v1.5=20?= =?UTF-8?q?=E6=96=B0=E5=A2=9E=E7=A7=AF=E6=9C=A8=EF=BC=8C=E4=BF=AE=E5=A4=8D?= =?UTF-8?q?bug=20(#77)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * feat:增加积木,获取缩放值 fix:加回废弃积木,以兼容之前的版本 fix:排序表传入的排序值cast转为数字(避免出现用字符串排序) fix:优化积木名称,把强制移动、强制设置大小积木的⚠️标志去掉。 * feat:加了新积木:JSON处理积木、最大值、克隆体数量等 perf:返回值转布尔积木,增加下拉菜单,原积木隐藏 --- Arkos/projectWith30.js | 650 +++++++++++++++++++++++++++++++++++++++-- 1 file changed, 630 insertions(+), 20 deletions(-) diff --git a/Arkos/projectWith30.js b/Arkos/projectWith30.js index 8aa6c5ed..8f80bc8b 100644 --- a/Arkos/projectWith30.js +++ b/Arkos/projectWith30.js @@ -4,12 +4,11 @@ import Color from '../utils/color.js' // import icon from './assets/icon.svg' //鸣谢:-6 优化代码和修复了一些 bug;_30 提供了部分拓展积木 -console.log(Cast.toNumber('123')) -console.log(Cast.toNumber('aab')) class ArkosExtensions { constructor(runtime) { this.runtime = runtime this.tempData = {} + this.lastKeyPressed={} //记录上一帧按下的键状态 this.sortedTable = { list1: { order: 'desc', @@ -26,6 +25,7 @@ class ArkosExtensions { 'ArkosExt.info1': '🚶 坐标和方向', 'ArkosExt.info2': '🔠 字符串处理', 'ArkosExt.info3': '🛠 实用积木', + 'ArkosExt.info3.5': '⚙️ JSON工具', 'ArkosExt.info4': '📄 信息获取', 'ArkosExt.info5': '📊 排序表', 'ArkosExt.info6': '🗂️ 临时数据', @@ -53,17 +53,39 @@ class ArkosExtensions { 'ArkosExt.isHiding': '角色隐藏?', 'ArkosExt.getRotationStyle': '当前旋转方式', 'ArkosExt.getWidthOrHeight': '获取当前造型的[t]', - 'ArkosExt.setSize': '⚠️强行将大小设为[size](无视限制)', + 'ArkosExt.setSize': '强行将大小设为[size](无视限制)', 'ArkosExt.width': '宽', 'ArkosExt.height': '高', - - 'ArkosExt.setXY': '⚠️强行移到x:[x]y:[y](无视边界)', + 'ArkosExt.true': '成立', + 'ArkosExt.false': '不成立', + 'ArkosExt.probability': '概率[p]', + 'ArkosExt.getKeyDown': '按下[key],且上次检测未按下', + 'ArkosExt.dataChanged': '当值[c]发生变化', + 'ArkosExt.isNum': '是数字', + 'ArkosExt.isInt': '是整数', + 'ArkosExt.sgn': '[c]的符号', + 'ArkosExt.defaultValue': '[c],默认值=[d]', + 'ArkosExt.reporterToBoolean': '[t][type]', + 'ArkosExt.max': '最大值', + 'ArkosExt.min': '最小值', + 'ArkosExt.diff': '差', + 'ArkosExt.sumOfSqu': '平方和', + 'ArkosExt.sqrtSumOfSqu': '平方和开根号', + 'ArkosExt.contain': '[list]包含[c]?(以[ch]分隔)', + 'ArkosExt.lenOfJSONList': '列表JSON[list]的长度', + 'ArkosExt.JSONListContains': '列表JSON[list]包含[c]?', + 'ArkosExt.editJSONList': '列表JSON[list]将[c][type]', + 'ArkosExt.JSONm1': '加到末尾', + 'ArkosExt.JSONm2': '从中删除', + + 'ArkosExt.setXY': '强行移到x:[x]y:[y](无视边界)', 'ArkosExt.getBoundaryCoord': '获取角色的[t]', 'ArkosExt.top': '上边缘y', 'ArkosExt.bottom': '下边缘y', 'ArkosExt.left': '左边缘x', 'ArkosExt.right': '右边缘x', 'ArkosExt.isOutOfSight': '角色移到舞台区外?', + 'ArkosExt.cloneCount': '当前克隆体数量', 'ArkosExt.and': '且', 'ArkosExt.or': '或', @@ -88,6 +110,7 @@ class ArkosExtensions { 'ArkosExt.deleteAllTempData': '🗂️清空所有临时数据', 'ArkosExt.getCountOfTempData': '🗂️临时数据量', 'ArkosExt.delTempData': '🗂️删除名为[data]的临时数据', + 'ArkosExt.ifTempDataExist': '🗂️存在名为[data]的临时数据?', 'ArkosExt.setTempVar': '🗂️将临时变量[var]设为[t]', 'ArkosExt.addTempVar': '🗂️将临时变量[var]增加[t]', @@ -103,6 +126,8 @@ class ArkosExtensions { 'ArkosExt.delItemOfTempList': '🗂️删除临时列表[list]第[n]项', 'ArkosExt.getItemOfTempList': '🗂️临时列表[list]第[n]项', 'ArkosExt.lengthOfTempList': '🗂️临时列表[list]长度', + 'ArkosExt.ifListItemExist': '🗂️临时列表[list]包含[c]?', + 'ArkosExt.getListItemIdx': '🗂️临时列表[list]中第一个[c]的编号', 'ArkosExt.clearTempCon': '🗂️创建或清空临时容器[con]', 'ArkosExt.initTempCon': '🗂️临时容器[con]内容设为[t]', @@ -115,14 +140,20 @@ class ArkosExtensions { 'ArkosExt.conInfo1': '名称', 'ArkosExt.conInfo2': '内容', 'ArkosExt.lengthOfTempCon': '🗂️临时容器[con]中内容数', + 'ArkosExt.ifConItemExist': '🗂️临时容器[con]包含[c]?', '30Ext.info': '✨ 以下扩展由_30提供', '30Ext.info.1': '🔮 定向缩放操作', + '30Ext.block.mirrorSprite': '(❌废弃,请使用新积木)[mirrorMethod]当前角色', + '30Ext.block.clearMirror': '(❌废弃,请使用新积木)清除角色镜像变换', '30Ext.block.scaleSpriteX': '将角色水平缩放比例设为[input](倍)', '30Ext.block.scaleSpriteY': '将角色垂直缩放比例设为[input](倍)', '30Ext.info.2': '图层操作', '30Ext.block.getLayer': '角色当前图层序数', '30Ext.block.setLayer': '将角色移到第[input]图层', + '30Ext.block.getScale': '当前角色的[input]缩放(倍)', + '30Ext.block.hor': '水平', + '30Ext.block.ver': '垂直', }, en: { @@ -146,17 +177,39 @@ class ArkosExtensions { 'ArkosExt.isHiding': 'is hiding?', 'ArkosExt.getRotationStyle': 'rotation style', 'ArkosExt.getWidthOrHeight': 'get [t] of the current costume', - 'ArkosExt.setSize': '⚠️force the size to [size] % (regardless of limitation) ', + 'ArkosExt.setSize': 'force the size to [size] % (regardless of limitation) ', 'ArkosExt.width': 'width', 'ArkosExt.height': 'height', - - 'ArkosExt.setXY': '⚠️force to x:[x]y:[y] (regardless of the boundary)', + 'ArkosExt.defaultValue': '[c],default=[d]', + 'ArkosExt.max': 'max', + 'ArkosExt.min': 'min', + 'ArkosExt.diff': 'difference between', + 'ArkosExt.sumOfSqu': 'square sum', + 'ArkosExt.sqrtSumOfSqu': 'sqrt square sum', + 'ArkosExt.contain': '[list]contains[c](separated by[ch])', + 'ArkosExt.lenOfJSONList': 'length of JSON[list]', + 'ArkosExt.JSONListContains': 'list JSON[list]contains key[c]?', + 'ArkosExt.editJSONList': '[c][type]list JSON[list]', + 'ArkosExt.JSONm1': 'add to', + 'ArkosExt.JSONm2': 'delete from', + 'ArkosExt.true': 'is', + 'ArkosExt.false': 'not', + 'ArkosExt.reporterToBoolean': '[type][t]', + 'ArkosExt.probability': 'probability[p]', + 'ArkosExt.getKeyDown': 'key[key]pressed, and not pressed last time', + 'ArkosExt.dataChanged': 'value[c]changed', + 'ArkosExt.isNum': 'is a number', + 'ArkosExt.isInt': 'is an integer', + 'ArkosExt.sgn': 'sign of[c]', + + 'ArkosExt.setXY': 'force to x:[x]y:[y] (regardless of the boundary)', 'ArkosExt.getBoundaryCoord': 'get [t] of the sprite', 'ArkosExt.top': 'top y', 'ArkosExt.bottom': 'bottom y', 'ArkosExt.left': 'left x', 'ArkosExt.right': 'right x', 'ArkosExt.isOutOfSight': 'is out of stage?', + 'ArkosExt.cloneCount': 'the number of clones', 'ArkosExt.and': 'and', 'ArkosExt.or': 'or', @@ -181,6 +234,7 @@ class ArkosExtensions { 'ArkosExt.info1': '🚶 Coordinate and Direction', 'ArkosExt.info2': '🔠 String Processing', 'ArkosExt.info3': '🛠 Utilities', + 'ArkosExt.info3.5': '⚙️ JSON utils', 'ArkosExt.info4': '📄 Information', 'ArkosExt.info5': '📊 Sorted Table', 'ArkosExt.info6': '🗂️ Temporary Data', @@ -191,6 +245,7 @@ class ArkosExtensions { 'ArkosExt.deleteAllTempData': '🗂️clear all temporary data', 'ArkosExt.getCountOfTempData': '🗂️count of temporary data', 'ArkosExt.delTempData': '🗂️delete temporary data[data]', + 'ArkosExt.ifTempDataExist': '🗂️temporary data[data]exists', 'ArkosExt.setTempVar': '🗂️set temp var[var] to [t]', 'ArkosExt.addTempVar': '🗂️change temp var[var] by [t]', @@ -207,6 +262,8 @@ class ArkosExtensions { 'ArkosExt.delItemOfTempList': '🗂️delete [n]of temp list[list]', 'ArkosExt.getItemOfTempList': '🗂️item[n]of temp list[list]', 'ArkosExt.lengthOfTempList': '🗂️length of temp list[list]', + 'ArkosExt.ifListItemExist': '🗂️temp list[list]contains[c]?', + 'ArkosExt.getListItemIdx': '🗂️item # of[c]in temp list[list]', 'ArkosExt.clearTempCon': '🗂️create or clear temp container[con]', 'ArkosExt.initTempCon': '🗂️set temp container[con]to[t]', @@ -219,14 +276,20 @@ class ArkosExtensions { 'ArkosExt.conInfo1': 'name', 'ArkosExt.conInfo2': 'content', 'ArkosExt.lengthOfTempCon': '🗂️count of contents in temp container[con]', + 'ArkosExt.ifConItemExist': '🗂️temp container[con]contains[c]?', '30Ext.info': '✨ Contributed by _30', '30Ext.info.1': '🔮 Directional scale', + '30Ext.block.mirrorSprite': '(❌abandoned, use new block instead)[mirrorMethod] current sprite', + '30Ext.block.clearMirror': '(❌abandoned, use new block instead)Clear the mirror transform', '30Ext.block.scaleSpriteX': 'Set the horizontal scaling of the sprite to [input] (Times)', '30Ext.block.scaleSpriteY': 'Set the vertical scaling of the sprite to [input] (Times)', '30Ext.info.2': 'Layer Manage', '30Ext.block.getLayer': 'Current layer of the sprite', '30Ext.block.setLayer': 'Move the sprite to layer [input]', + '30Ext.block.getScale': '[input]scaling of the sprite (Times)', + '30Ext.block.hor': 'horizontal', + '30Ext.block.ver': 'vertical', }, }) } @@ -412,11 +475,12 @@ class ArkosExtensions { }, }, }, - //返回值转bool积木 + //(隐藏)返回值转bool积木 { opcode: 'reporterToBoolean', blockType: 'Boolean', text: '[t]', + hideFromPalette: true, arguments: { t: { type: 'string', @@ -424,6 +488,142 @@ class ArkosExtensions { } }, }, + //返回值转bool积木2 + { + opcode: 'reporterToBoolean2', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.reporterToBoolean'), + arguments: { + t: { + type: 'string', + defaultValue: '1', + }, + type: { + type: 'string', + menu: 'TorF2', + } + }, + }, + //概率 + { + opcode: 'probability', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.probability'), + arguments: { + p: { + type: 'number', + defaultValue: '0.5', + } + }, + }, + //(有bug暂时隐藏)按下x键且上次没按 + // { + // opcode: 'getKeyDown', + // blockType: 'Boolean', + // text: this.formatMessage('ArkosExt.getKeyDown'), + // arguments: { + // key: { + // type: 'string', + // defaultValue: 'a', + // } + // }, + // }, + //(暂时不知如何实现)检测值变化 + // { + // opcode: 'dataChanged', + // blockType: 'Boolean', + // text: this.formatMessage('ArkosExt.dataChanged'), + // arguments: { + // c: { + // type: 'string', + // defaultValue: '变量', + // } + // }, + // }, + //判断是否是数字 + { + opcode: 'isNum', + blockType: 'Boolean', + text: '[type][c]', + arguments: { + c: { + type: 'number', + defaultValue: '0.2', + }, + type: { + type: 'string', + menu: 'isNumMenu', + }, + }, + }, + //符号 + { + opcode: 'sgn', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.sgn'), + arguments: { + c: { + type: 'number', + defaultValue: '-5', + } + }, + }, + //默认值 + { + opcode: 'defaultValue', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.defaultValue'), + arguments: { + c: { + type: 'string', + defaultValue: '', + }, + d: { + type: 'string', + defaultValue: '10', + } + }, + }, + //max min 差.. + { + opcode: 'binaryCal', + blockType: 'reporter', + text: '[cal][a][b]', + arguments: { + cal: { + type: 'string', + menu: 'biCalMenu', + }, + a: { + type: 'string', + defaultValue: '1', + }, + b: { + type: 'string', + defaultValue: '99', + } + }, + }, + //xx,xx,xx包含xx? + { + opcode: 'contain', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.contain'), + arguments: { + list: { + type: 'string', + defaultValue: '苹果,香蕉,橘子,菠萝', + }, + ch: { + type: 'string', + defaultValue: ',', + }, + c: { + type: 'string', + defaultValue: '苹果', + } + }, + }, //形如 a≤b≤c { opcode: 'compareTwoSides', @@ -529,6 +729,55 @@ class ArkosExtensions { }, filter: ['sprite'] }, + "---" + this.formatMessage("ArkosExt.info3.5"), //🔧JSON积木 + //JSON列表长度 + { + opcode: 'lenOfJSONList', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.lenOfJSONList'), + arguments: { + list: { + type: 'string', + defaultValue: '[1,2,"apple"]', + } + }, + }, + //JSON列表包含XX + { + opcode: 'JSONListContains', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.JSONListContains'), + arguments: { + list: { + type: 'string', + defaultValue: '[1,2,"apple"]', + }, + c: { + type: 'string', + defaultValue: 'apple', + } + }, + }, + //JSON列表加入/删除 + { + opcode: 'editJSONList', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.editJSONList'), + arguments: { + list: { + type: 'string', + defaultValue: '[1,2,"apple"]', + }, + c: { + type: 'string', + defaultValue: 'apple', + }, + type: { + type: 'string', + menu: 'JSONm', + } + }, + }, "---" + this.formatMessage("ArkosExt.info4"), //📄数据获取 //获取特效值 { @@ -588,6 +837,12 @@ class ArkosExtensions { text: this.formatMessage('ArkosExt.isOutOfSight'), filter: ['sprite'] }, + //克隆体数量 + { + opcode: 'cloneCount', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.cloneCount') + }, "---" + this.formatMessage("ArkosExt.info5"), //📊排序表 //📊清空排序表 { @@ -737,6 +992,18 @@ class ArkosExtensions { }, }, }, + //判断数据存在 + { + opcode: 'ifTempDataExist', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.ifTempDataExist'), + arguments: { + data: { + type: 'string', + defaultValue: 'i', + }, + }, + }, "---" + this.formatMessage("ArkosExt.info7"), //临时变量 //设置临时数据 { @@ -895,6 +1162,38 @@ class ArkosExtensions { }, }, }, + //临时列表包含xx? + { + opcode: 'ifListItemExist', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.ifListItemExist'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + c: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, + //获取列表第一个xx的索引 + { + opcode: 'getListItemIdx', + blockType: 'reporter', + text: this.formatMessage('ArkosExt.getListItemIdx'), + arguments: { + list: { + type: 'string', + defaultValue: 'list', + }, + c: { + type: 'string', + defaultValue: 'thing', + }, + }, + }, "---" + this.formatMessage("ArkosExt.info9"), //临时容器 //创建或清空临时容器 { @@ -1013,11 +1312,59 @@ class ArkosExtensions { }, }, }, + //ifConItemExist + { + opcode: 'ifConItemExist', + blockType: 'Boolean', + text: this.formatMessage('ArkosExt.ifConItemExist'), + arguments: { + con: { + type: 'string', + defaultValue: 'con1', + }, + c: { + type: 'string', + defaultValue: 'coins', + } + }, + }, // "---" + this.formatMessage("30Ext.info"), //感谢30提供的扩展 "---" + this.formatMessage("30Ext.info.1"), //定向缩放 + { + opcode: 'mirrorSprite', + blockType: 'command', + text: this.formatMessage('30Ext.block.mirrorSprite'), + hideFromPalette: true, + arguments: { + mirrorMethod: { + type: 'string', + defaultValue: '' + } + } + }, + // 清除镜像 + { + opcode: 'clearMirror', + blockType: 'command', + hideFromPalette: true, + text: this.formatMessage('30Ext.block.clearMirror') + }, + //获取缩放 + { + opcode: 'getScale', + blockType: 'reporter', + text: this.formatMessage('30Ext.block.getScale'), + arguments: { + input: { + type: 'string', + menu: 'HVMenu', + } + }, + filter: ['sprite'] + }, // x向缩放 { opcode: 'scaleSpriteX', @@ -1066,6 +1413,72 @@ class ArkosExtensions { }, ], menus: { + isNumMenu:[{ + text: this.formatMessage('ArkosExt.isNum'), + value: '1' + }, + { + text: this.formatMessage('ArkosExt.isInt'), + value: '2' + } + ], + biCalMenu: [{ + text: this.formatMessage('ArkosExt.max'), + value: '1' + }, + { + text: this.formatMessage('ArkosExt.min'), + value: '2' + }, + { + text: this.formatMessage('ArkosExt.diff'), + value: '3' + }, + { + text: this.formatMessage('ArkosExt.sumOfSqu'), + value: '4' + }, + { + text: this.formatMessage('ArkosExt.sqrtSumOfSqu'), + value: '5' + } + ], + TorF: [{ + text: 'true', + value: '1' + }, + { + text: 'false', + value: '2' + }, + ], + TorF2: [{ + text: this.formatMessage('ArkosExt.true'), + value: '1' + }, + { + text: this.formatMessage('ArkosExt.false'), + value: '0' + }, + ], + JSONm: [{ + text: this.formatMessage('ArkosExt.JSONm1'), //加入 + value: '1' + }, + { + text: this.formatMessage('ArkosExt.JSONm2'), //删除 + value: '2' + }, + ], + HVMenu: [{ + text: this.formatMessage('30Ext.block.hor'), //水平 + value: 'h' + }, + { + text: this.formatMessage('30Ext.block.ver'), //垂直 + value: 'v' + }, + ], conInfoMenu: [{ text: this.formatMessage('ArkosExt.conInfo1'), //名称 value: '1' @@ -1307,10 +1720,6 @@ class ArkosExtensions { return !util.target.visible; } - //获取图层(逝一逝) - // getLayer (args, util) { - // return util.target.layer; - // } //获取当前角色的旋转方式 getRotationStyle(args, util) { @@ -1408,14 +1817,30 @@ class ArkosExtensions { return false; } - //形如:<() > + cloneCount(){ + return this.runtime._cloneCounter; + } + + //(废弃)形如:<() > reporterToBoolean(args) { - if(Cast.toString(args.t) - .toLowerCase() === 'false') return false; - if(args.t === '0') return false; + const t = Cast.toString(args.t).toLowerCase() + if(t === 'false'||t === '0'||t === 'undefined'||t === 'null'||t === '') return false; return (args.t) ? true : false; } + //形如:<()成立/不成立 > + reporterToBoolean2(args) { + const t = Cast.toString(args.t).toLowerCase() + let b + if(t === 'false'||t === '0'||t === 'undefined'||t === 'null'||t === '') b = false; + else b = (args.t) ? true : false; + return (args.type === '1')? b : (!b); + } + + trueOrFalse(args) { + return (args.type === '1')? true : false; + } + compare(a, b, op) { switch (op) { case '<': @@ -1435,6 +1860,125 @@ class ArkosExtensions { } } + binaryCal(args){ + let a = Cast.toNumber(args.a) + let b = Cast.toNumber(args.b) + switch (args.cal) { + case '1': + return (Cast.compare(args.a, args.b) > 0) ? args.a : args.b; //max + case '2': + return (Cast.compare(args.a, args.b) > 0) ? args.b : args.a; //min + case '3': + return Math.abs(a-b);//差 + case '4': + return a*a+b*b;//平方和 + default: + return Math.sqrt(a*a+b*b);//平方和开方 + } + } + + defaultValue(args){ + return (args.c === '')? args.d : args.c; + } + + isNum(args){ + if(args.type === '1') + { + return !isNaN(args.c); + } + if(args.type === '2') + { + if(isNaN(args.c)) return false; + return Cast.isInt(args.c); + } + return false; + } + + //取符号。负数为-1,0和正数为1 + sgn(args){ + let c = Cast.toNumber(args.c) + return c<0 ? -1 : 1; + } + + //概率 + probability(args){ + let p = Cast.toNumber(args.p) + if(p===1) return true; + if(p===0) return false; + return (Math.random() < p)? true : false; + } + + //积木暂时隐藏,不上线 + getKeyDown (args, util) { + let flag = false + let pressed = util.ioQuery('keyboard', 'getKeyIsDown', [args.key]); + if(!this.lastKeyPressed[args.key] && pressed) flag = true; //这一帧按下,且上一帧未按下 + this.lastKeyPressed[args.key] = pressed + return flag; + } + + //暂时不知如何实现 + dataChanged(args, util){ + // let cached = util.target.blocks._cache._executeCached + // console.log(Object.keys(cached)[0]) + // console.log(util.target.blocks._cache) + console.log(util.target.blocks) + return false; + } + + //xxx,xx,xx 包含xx? + contain(args) { + let list = Cast.toString(args.list).split(Cast.toString(args.ch)) + return this._ifListItemExist(list, Cast.toString(args.c)) + } + + lenOfJSONList(args) { + try { + let list = JSON.parse(Cast.toString(args.list)) + if(typeof(list) === 'object' && list !== null) { + return Object.keys(list).length; + } + return 0; + } catch (e) { + return 0; + } + } + + JSONListContains(args) { + try { + let list = JSON.parse(Cast.toString(args.list)) + if(Array.isArray(list)) { + return this._ifListItemExist(list, Cast.toString(args.c)); + } + return false; + } catch (e) { + return false; + } + } + + //加入/从JSON列表删除 + editJSONList(args) { + try { + let list = JSON.parse(Cast.toString(args.list)) + if(Array.isArray(list)) { + const item = this._anythingToNumberString(args.c) + if(args.type === '1') //加入列表 + { + list.push(item); + } + if(args.type === '2') //从列表删除 + { + const idx = this._getListItemIdx(list, item) - 1 + if(idx >= 0) list.splice(idx, 1); + } + return JSON.stringify(list); + } + return ''; + } catch (e) { + return ''; + } + } + //形如:a≤b≤c op1,op2 compareTwoSides(args) { return this.compare(args.a, args.b, args.op1) && this.compare(args.b, args.c, args.op2) @@ -1538,7 +2082,7 @@ class ArkosExtensions { this.sortedTable[args.list].list, this.sortedTable[args.list].order, { name: args.name, - rankValue: args.value, + rankValue: Cast.toNumber(args.value), extra: args.extra }); } @@ -1640,6 +2184,10 @@ class ArkosExtensions { delete this.tempData[Cast.toString(args.data)]; } + ifTempDataExist(args) { + return this.tempData.hasOwnProperty(Cast.toString(args.data)) + } + setTempVar(args) { this.tempData[Cast.toString(args.var)] = args.t; } @@ -1721,6 +2269,48 @@ class ArkosExtensions { return list.length; } + //检查list是否包含item + _ifListItemExist(list, item) { + if (list.indexOf(item) >= 0) { + return true; + } + // Try using Scratch comparison operator on each item. + // (Scratch considers the string '123' equal to the number 123). + for (let i = 0; i < list.length; i++) { + if (Cast.compare(list[i], item) === 0) { + return true; + } + } + return false; + } + + ifListItemExist(args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return false; + const item = Cast.toString(args.c) + + return this._ifListItemExist(list, item) + } + + //获取list中item索引 + _getListItemIdx(list, item) { + for (let i = 0; i < list.length; i++) { + if (Cast.compare(list[i], item) === 0) { + return i + 1; + } + } + return 0; + } + + getListItemIdx(args) { + let list = this.tempData[Cast.toString(args.list)] + if(!Array.isArray(list)) return 0; + const item = Cast.toString(args.c) + + return this._getListItemIdx(list, item) + + } + //容器 clearTempCon(args) { this.tempData[Cast.toString(args.con)] = {}; @@ -1784,8 +2374,13 @@ class ArkosExtensions { lengthOfTempCon(args) { let con = this.tempData[Cast.toString(args.con)] if(!(typeof(con) === 'object' && con !== null)) return 0; - return Object.keys(con) - .length; + return Object.keys(con).length; + } + + ifConItemExist(args) { + let con = this.tempData[Cast.toString(args.con)] + if(!(typeof(con) === 'object' && con !== null)) return false; + return con.hasOwnProperty(Cast.toString(args.c)); } @@ -1808,6 +2403,21 @@ class ArkosExtensions { // //角色造型操作 // + clearMirror(){ + console.warn("镜像积木已下线,请使用新积木\nMirror block is offline, please use new blocks."); + } + + mirrorSprite(){ + console.warn("镜像积木已下线,请使用新积木\nMirror block is offline, please use new blocks."); + } + + getScale(args, util) { + let drawable = this.runtime.renderer._allDrawables[util.target.drawableID] + if(!drawable.ext30_scale) return 1 + else if(args.input === 'v') return drawable.ext30_scale[1] + else return drawable.ext30_scale[0] + } + scaleSprite(index, value, util) { let target = util.target; let drawable = this.runtime.renderer._allDrawables[target.drawableID]; From a6beaad2a0456f3ba2ac59fc028b571ca2935ecb Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 28 May 2023 15:57:37 +0800 Subject: [PATCH 227/357] FPS 1.1 (#80) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit feat(block):计算帧率浮动 --- wit_cat/FPS.js | 351 +++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 351 insertions(+) create mode 100644 wit_cat/FPS.js diff --git a/wit_cat/FPS.js b/wit_cat/FPS.js new file mode 100644 index 00000000..8f95a6a4 --- /dev/null +++ b/wit_cat/FPS.js @@ -0,0 +1,351 @@ + +const witcat_fps_picture = ""; + +const witcat_fps_icon = ""; + +const witcat_fps_extensionId = "WitCatFPS"; + +/** @typedef {string|number|boolean} SCarg 来自Scratch圆形框的参数,虽然这个框可能只能输入数字,但是可以放入变量,因此有可能获得数字、布尔和文本(极端情况下还有 null 或 undefined,需要同时处理 */ + +class WitCatFPS { + constructor(runtime) { + this.runtime = runtime; + /** Scratch fps 开关 */ + this.scfpson = false; + /** Scratch fps(每秒更新) */ + this.scfps = 0; + /** Scratch fps(随时更新) */ + this.scfpsn = 0; + /** Scratch fps 计数 */ + this.scfpscnt = 0; + /** Scratch fps 每秒 setInterval 计时器 */ + this.scfpsinterval = -1; + /** Scratch fps 上一帧时间(计算瞬时fps) */ + this.scfpstime = 0; + + /** 浏览器 fps 开关 */ + this.webfpson = false; + /** 浏览器 fps(每秒更新) */ + this.webfps = 0; + /** 浏览器 fps(随时更新) */ + this.webfpsn = 0; + /** 浏览器 fps 计数 */ + this.webfpscnt = 0; + /** 浏览器 fps 计数函数已经关闭? */ + this.webfpstick_end = true; + /** 浏览器 fps 上一帧时间(计算瞬时fps) */ + this.webfpstime = 0; + /** 浏览器 fps 每秒更新 setInterval 计时器 */ + this.webfpsinterval = -1; + + + /** 被挂钩的 runtime._step 函数 */ + this.step = this.runtime._step; + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "WitCatFPS.name": "[beta]白猫的FPS", + "WitCatFPS.webs": "浏览器", + "WitCatFPS.scratchs": "舞台", + "WitCatFPS.more": "更多", + "WitCatFPS.scratch": "舞台FPS", + "WitCatFPS.web": "浏览器FPS", + "WitCatFPS.nscratch": "瞬时舞台FPS", + "WitCatFPS.nweb": "瞬时浏览器FPS", + "WitCatFPS.turnon": "[type]舞台FPS检测", + "WitCatFPS.webturnon": "[type]浏览器FPS检测", + "WitCatFPS.type.1": "开启", + "WitCatFPS.type.2": "关闭", + "WitCatFPS.docs": "📖拓展教程", + "WitCatFPS.compute": "在帧率[fps]下的[num]", + }, + en: { + "WitCatFPS.name": "[beta]WitCat’s FPS", + "WitCatFPS.webs": "web", + "WitCatFPS.scratchs": "stage", + "WitCatFPS.more": "more", + "WitCatFPS.scratch": "stage FPS", + "WitCatFPS.web": "web FPS", + "WitCatFPS.nscratch": "stage current FPS", + "WitCatFPS.nweb": "web current FPS", + "WitCatFPS.turnon": "[type]stage FPS detection", + "WitCatFPS.webturnon": "[type]web FPS detection", + "WitCatFPS.type.1": "enable", + "WitCatFPS.type.2": "disable", + "WitCatFPS.docs": "📖 Tutorial", + "WitCatFPS.compute": "the number[num]from[fps]", + } + }) + } + + /** + * 翻译 + * @param {string} id + * @return {string} + */ + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: witcat_fps_extensionId, // 拓展id + name: this.formatMessage("WitCatFPS.name"), // 拓展名 + blockIconURI: witcat_fps_icon, + menuIconURI: witcat_fps_icon, + color1: "#EC3838", + color2: "#B32B2B", + blocks: [ + { + blockType: "button", + text: this.formatMessage('WitCatFPS.docs'), + onClick: this.docs, + }, + "---" + this.formatMessage("WitCatFPS.scratchs"), + { + opcode: "scratch", + blockType: "reporter", + text: this.formatMessage("WitCatFPS.scratch"), + arguments: {}, + }, + { + opcode: "nscratch", + blockType: "reporter", + text: this.formatMessage("WitCatFPS.nscratch"), + arguments: {}, + }, + { + opcode: "turnon", + blockType: "command", + text: this.formatMessage("WitCatFPS.turnon"), + arguments: { + type: { + type: "string", + menu: 'type', + } + }, + }, + "---" + this.formatMessage("WitCatFPS.webs"), + { + opcode: "web", + blockType: "reporter", + text: this.formatMessage("WitCatFPS.web"), + arguments: {}, + }, + { + opcode: "nweb", + blockType: "reporter", + text: this.formatMessage("WitCatFPS.nweb"), + arguments: {}, + }, + { + opcode: "webturnon", + blockType: "command", + text: this.formatMessage("WitCatFPS.webturnon"), + arguments: { + type: { + type: "string", + menu: 'type', + } + }, + }, + "---" + this.formatMessage("WitCatFPS.more"), + { + opcode: "compute", + blockType: "reporter", + text: this.formatMessage("WitCatFPS.compute"), + arguments: { + fps: { + type: "number", + defaultValue: '30', + }, + num: { + type: "number", + defaultValue: '5', + } + }, + }, + ], + menus: { + type: [ + { + text: this.formatMessage('WitCatFPS.type.1'), + value: 'true' + }, + { + text: this.formatMessage('WitCatFPS.type.2'), + value: 'false' + }, + ], + }, + }; + } + + /** 打开教程 */ + docs() { + let a = document.createElement('a'); + a.href = "https://www.ccw.site/post/d6d96e80-3f58-4a19-b7e6-c567d3a6a583"; + a.rel = "noopener noreferrer"; + a.target = "_blank"; + a.click(); + } + + /** + * 获取浏览器帧率用的回调函数 + * @param {DOMHighResTimeStamp} timestamp + * @returns {void} + */ + _webfpstick(timestamp) { + // 存在多次开关功能导致该函数大量触发的情况。 + // 这样子设计,每帧只触发一次 + // 这里判断 webfpson 是否为真来判断是否要继续更新帧数, + // 这样才能正常关闭。 + if (this.webfpson) { + this.webfpstick_end = false; + this.webfpscnt++; + this.webfpsn = 1000 / (timestamp - this.webfpstime); + this.webfpstime = timestamp; + // 这里的 bind(this) 非常重要,让回调的时候 this 指向当前的 this。 + requestAnimationFrame(this._webfpstick.bind(this)); + } + else { + this.webfpstick_end = true; + } + } + + /** + * 舞台的帧率 + * @return {number} + */ + scratch() { + return this.scfps; + } + + /** + * 舞台的瞬时帧率 + * @return {number} + */ + nscratch() { + return this.scfpsn; + } + + /** + * 浏览器的帧率 + * @return {number} + */ + web() { + return this.webfps; + } + + /** + * 浏览器的瞬时帧率 + * @return {number} + */ + nweb() { + return this.webfpsn; + } + + /** + * 设置舞台帧率检测 + * @param {object} args + * @param {SCarg|"true"|"false"} args.type + * @return {void} + */ + turnon(args) { + if (args.type == "true") { + if (this.scfpson == false) { + this.scfpson = true; + this.scfpstime = Date.now(); + let that = this; + this.runtime._step = function () { + if (that.scfpson) { + that.step.call(this); + let timestamp = Date.now(); + that.scfpsn = 1000 / (timestamp - that.scfpstime); + that.scfpstime = timestamp; + that.scfpscnt++; + } + } + this.scfpsinterval = setInterval(() => { + this.scfps = this.scfpscnt; + this.scfpscnt = 0; + }, 1000) + } + } + else { + if (this.scfpson == true) { + this.scfpson = false; + this.runtime._step = this.step; + clearInterval(this.scfpsinterval); + this.scfps = 0; + this.scfpsn = 0; + } + } + } + + /** + * 设置浏览器帧率检测 + * @param {object} args + * @param {SCarg|"true"|"false"} args.type + * @return {void} + */ + webturnon(args) { + if (args.type == "true") { + if (this.webfpson == false) { + this.webfpson = true; + this.webfpstime = Date.now(); + if (this.webfpstick_end) { + // 这里的 bind(this) 非常重要,让回调的时候 this 指向当前的 this。 + requestAnimationFrame(this._webfpstick.bind(this)); + } + this.webfpsinterval = setInterval(() => { + this.webfps = this.webfpscnt; + this.webfpscnt = 0; + }, 1000) + } + } + else { + if (this.webfpson == true) { + this.webfpson = false; + clearInterval(this.webfpsinterval); + this.webfps = 0; + this.webfpsn = 0; + } + } + } + /** + * @param {number} fps 对照帧率 + * @param {number} num 需要计算的值 + * @return {number} + */ + compute(args) { + return Number(args.num) / (this.scfpsn / Number(args.fps)); + } +} + +window.tempExt = { + Extension: WitCatFPS, + info: { + name: "WitCatFPS.name", + description: "WitCatFPS.descp", + extensionId: witcat_fps_extensionId, + iconURL: witcat_fps_picture, + insetIconURL: witcat_fps_icon, + featured: true, + disabled: false, + collaborator: "白猫 @ CCW" + }, + l10n: { + "zh-cn": { + "WitCatFPS.name": "[beta]白猫的FPS", + "WitCatFPS.descp": "运行速度!!!" + }, + en: { + "WitCatFPS.name": "[beta]WitCat’s FPS", + "WitCatFPS.descp": "Detect running speed!!!" + } + } +}; From 69bd73e6a0f24b08e92a8bd3ff0fbb00e9dc2e9b Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 28 May 2023 16:05:41 +0800 Subject: [PATCH 228/357] File_Helper.js 2.7 (#79) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Update File_Helper.js * 添加依赖 fix:添加依赖和说明 --- wit_cat/File_Helper.js | 2162 ++++++++++++++++++++++------------------ wit_cat/README.md | 1 + wit_cat/base64.js | 300 ++++++ 3 files changed, 1509 insertions(+), 954 deletions(-) create mode 100644 wit_cat/README.md create mode 100644 wit_cat/base64.js diff --git a/wit_cat/File_Helper.js b/wit_cat/File_Helper.js index b4f395d2..ba0b9eeb 100644 --- a/wit_cat/File_Helper.js +++ b/wit_cat/File_Helper.js @@ -1,968 +1,1222 @@ // import Cast from "cast.js"; -const _picture = ""; +import { Base64 } from "./base64.js"; -const _icon = ""; +const witcat_file_helper_picture = ""; -const extensionId = "WitCatFileHelper"; -let FLAG = 0, download = 0; +const witcat_file_helper_icon = ""; -setInterval(() => { - if (download > 0) { - download--; - } -}, 1000) +const witcat_file_helper_extensionId = "WitCatFileHelper"; + +/** @typedef {string|number|boolean} SCarg 来自Scratch圆形框的参数,虽然这个框可能只能输入数字,但是可以放入变量,因此有可能获得数字、布尔和文本(极端情况下还有 null 或 undefined,需要同时处理 */ class WitCatFileHelper { - constructor(runtime) { - this.runtime = runtime; - this._formatMessage = runtime.getFormatMessage({ - "zh-cn": { - "WitCatFileHelper.name": "[beta]白猫的文件助手", - "WitCatFileHelper.filehelper": "文件处理", - "WitCatFileHelper.inputmanagement": "文本框", - "WitCatFileHelper.Multiplelinestext": "多行文本", - "WitCatFileHelper.downloads": "将内容[text]按[s]分割后命名为[name]并下载多行文本", - "WitCatFileHelper.download": "将内容[text]命名为[name]并下载", - "WitCatFileHelper.save": "设置键[name]为[text]并储存到本地", - "WitCatFileHelper.upload": "获取键[name]的值", - "WitCatFileHelper.delete": "删除键[name]", - "WitCatFileHelper.segmentation": "将[text]按[s]分割为[thing]", - "WitCatFileHelper.encrypt": "base64加密[text]", - "WitCatFileHelper.decrypt": "base64解密[text]", - "WitCatFileHelper.openfile": "打开文件", - "WitCatFileHelper.showvar": "设置键[name]的状态为[show]", - "WitCatFileHelper.saveother": "设置作品ID[id]的键[name]为[text]", - "WitCatFileHelper.uploadother": "获取作品[id]的键[name]的值", - "WitCatFileHelper.other": "作品[id]的键[name]的状态", - "WitCatFileHelper.showon": "只读", - "WitCatFileHelper.showoff": "私有", - "WitCatFileHelper.showall": "公开", - "WitCatFileHelper.deleteMultiplelinestext": "删除[text]的第[num]行", - "WitCatFileHelper.addMultiplelinestext": "将[text]加入[texts]的第[num]行", - "WitCatFileHelper.whatMultiplelinestext": "[text]的第[num]行", - "WitCatFileHelper.numMultiplelinestext": "[text]的行数", - "WitCatFileHelper.thing.1": "数组", - "WitCatFileHelper.thing.2": "多行文本", - "WitCatFileHelper.number": "第[num]个键的[type]", - "WitCatFileHelper.numbers": "键数量", - "WitCatFileHelper.number.1": "键名", - "WitCatFileHelper.number.2": "键值", - "WitCatFileHelper.turnMultiplelinestext": "将多行文本[text]转化为数组", - "WitCatFileHelper.turnsMultiplelinestext": "将数组[text]转化为多行文本", - "WitCatFileHelper.downloadnum": "可下载文件数量", - "WitCatFileHelper.downloadask": "作品企图下载疑似会威胁电脑的文件,是否继续?\n代码如下:", - }, - en: { - "WitCatFileHelper.name": "[beta]WitCat’s File Helper", - "WitCatFileHelper.filehelper": "file", - "WitCatFileHelper.inputmanagement": "input", - "WitCatFileHelper.Multiplelinestext": "Multiple lines of text", - "WitCatFileHelper.downloads": "Download split content [text] by [s] named [name]", - "WitCatFileHelper.download": "Download content [text] named [name]", - "WitCatFileHelper.save": "Save content [text] with [name] on computer", - "WitCatFileHelper.upload": "Get value [name]", - "WitCatFileHelper.delete": "delete value [name]", - "WitCatFileHelper.segmentation": "Split [text] by [s] to [thing]", - "WitCatFileHelper.encrypt": "base64 encrypt[text]", - "WitCatFileHelper.decrypt": "base64 decrypt[text]", - "WitCatFileHelper.openfile": "openfile", - "WitCatFileHelper.showvar": "set value[name]`s state[show]", - "WitCatFileHelper.saveother": "save ID[id]`s content name[name]with[text]", - "WitCatFileHelper.uploadother": "get ID[id]`s value[name]", - "WitCatFileHelper.other": "get ID[id]`s value[name]`s state", - "WitCatFileHelper.showon": "View", - "WitCatFileHelper.showoff": "private", - "WitCatFileHelper.showall": "public", - "WitCatFileHelper.deleteMultiplelinestext": "delete[text]the[num]line of text", - "WitCatFileHelper.addMultiplelinestext": "let[text]add with[texts]the[num]line of text", - "WitCatFileHelper.whatMultiplelinestext": "[text]`s[num]line of text", - "WitCatFileHelper.numMultiplelinestext": "[text]Number of rows", - "WitCatFileHelper.thing.1": "array", - "WitCatFileHelper.thing.2": "Multiple lines of text", - "WitCatFileHelper.number": "[type]of[num]key", - "WitCatFileHelper.numbers": "key number", - "WitCatFileHelper.number.1": "name", - "WitCatFileHelper.number.2": "content", - "WitCatFileHelper.turnMultiplelinestext": "Converts multiline[text]text to an array", - "WitCatFileHelper.turnsMultiplelinestext": "Converts the array[text]to multiline text", - "WitCatFileHelper.downloadnum": "Number of downloadable files", - "WitCatFileHelper.downloadask": "works will attempt to download the suspected threat to computer files, whether or not to continue? \n code is as follows: ", - } - }) - } - - formatMessage(id) { - return this._formatMessage({ - id, - default: id, - description: id - }); - } - - getInfo() { - return { - id: extensionId, // 拓展id - name: this.formatMessage("WitCatFileHelper.name"), // 拓展名 - docsURI: "https://www.ccw.site/post/d6d96e80-3f58-4a19-b7e6-c567d3a6a583", - blockIconURI: _icon, - menuIconURI: _icon, - color1: "#60D6F4", - color2: "#55a7f7", - blocks: [ - "---" + this.formatMessage("WitCatFileHelper.filehelper"), - { - opcode: "downloads", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.downloads"), - arguments: { - text: { - type: "string", - defaultValue: 'awa!!!|awa!!!', - }, - name: { - type: "string", - defaultValue: 'wit_cat.txt', - }, - s: { - type: "string", - defaultValue: '|', - }, - }, - }, - { - opcode: "download", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.download"), - arguments: { - text: { - type: "string", - defaultValue: 'awa!!!', - }, - name: { - type: "string", - defaultValue: 'wit_cat.txt', - }, - }, - }, - { - opcode: "downloadnum", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.downloadnum"), - arguments: {}, - }, - { - opcode: "openfile", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.openfile"), - disableMonitor: true, - arguments: {}, - }, - { - opcode: "upload", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.upload"), - hideFromPalette: true, - arguments: { - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "save", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.save"), - hideFromPalette: true, - arguments: { - text: { - type: "string", - defaultValue: '0', - }, - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "delete", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.delete"), - hideFromPalette: true, - arguments: { - name: { - type: "string", - defaultValue: 'i', - }, - }, - }, - { - opcode: "showvar", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.showvar"), - hideFromPalette: true, - arguments: { - name: { - type: "string", - defaultValue: "i", - }, - show: { - type: "string", - menu: "setvariable", - }, - }, - }, - { - opcode: "number", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.number"), - hideFromPalette: true, - arguments: { - num: { - type: "number", - defaultValue: "i", - }, - type: { - type: "string", - menu: "type", - }, - }, - }, - { - opcode: "numbers", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.numbers"), - hideFromPalette: true, - arguments: {}, - }, - { - opcode: "saveother", - blockType: "command", - text: this.formatMessage("WitCatFileHelper.saveother"), - hideFromPalette: true, - arguments: { - id: { - type: "string", - defaultValue: "6373950041d21d2d2cd0da9b", - }, - name: { - type: "string", - defaultValue: "i", - }, - text: { - type: "string", - defaultValue: "wit_cat!", - }, - }, - }, - { - opcode: "uploadother", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.uploadother"), - hideFromPalette: true, - arguments: { - id: { - type: "string", - defaultValue: "6373950041d21d2d2cd0da9b", - }, - name: { - type: "string", - defaultValue: "i", - }, - }, - }, - { - opcode: "other", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.other"), - hideFromPalette: true, - arguments: { - id: { - type: "string", - defaultValue: "6373950041d21d2d2cd0da9b", - }, - name: { - type: "string", - defaultValue: "i", - }, - }, - }, - "---" + "base64", - { - opcode: "encrypt", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.encrypt"), - arguments: { - text: { - type: "string", - defaultValue: 'i love china', - }, - }, - }, - { - opcode: "decrypt", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.decrypt"), - arguments: { - text: { - type: "string", - defaultValue: 'aSUyMGxvdmUlMjBjaGluYQ==', - }, - }, - }, - "---" + this.formatMessage("WitCatFileHelper.Multiplelinestext"), - { - opcode: "segmentation", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.segmentation"), - arguments: { - text: { - type: "string", - defaultValue: 'wow/!!!', - }, - s: { - type: "string", - defaultValue: '/', - }, - thing: { - type: "string", - menu: 'thing', - } - }, - }, - { - opcode: "deleteMultiplelinestext", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.deleteMultiplelinestext"), - arguments: { - text: { - type: "string", - defaultValue: "wit_cat!!!\nwit_cat!!!", - }, - num: { - type: "number", - defaultValue: "1", - } - }, - }, - { - opcode: "addMultiplelinestext", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.addMultiplelinestext"), - arguments: { - text: { - type: "string", - defaultValue: "wit_cat!!!", - }, - texts: { - type: "string", - defaultValue: "wit_cat!!!\nwit_cat!!!", - }, - num: { - type: "string", - defaultValue: "last", - } - }, - }, - { - opcode: "whatMultiplelinestext", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.whatMultiplelinestext"), - arguments: { - text: { - type: "string", - defaultValue: "wit_cat!!!", - }, - num: { - type: "string", - defaultValue: "1", - } - }, - }, - { - opcode: "numMultiplelinestext", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.numMultiplelinestext"), - arguments: { - text: { - type: "string", - defaultValue: "wit_cat!!!\nwit_cat!!!", - }, - }, - }, - { - opcode: "turnMultiplelinestext", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.turnMultiplelinestext"), - arguments: { - text: { - type: "string", - defaultValue: "wit_cat!!!\nwit_cat!!!", - }, - }, - }, - { - opcode: "turnsMultiplelinestext", - blockType: "reporter", - text: this.formatMessage("WitCatFileHelper.turnsMultiplelinestext"), - arguments: { - text: { - type: "string", - defaultValue: "[\"wit_cat!!!\",\"wit_cat!!!\"]", - }, - }, - }, - ], - menus: { - setvariable: [ - { - text: this.formatMessage('WitCatFileHelper.showall'), - value: '#witcat' - }, - { - text: this.formatMessage('WitCatFileHelper.showon'), - value: '$witcat' - }, - { - text: this.formatMessage('WitCatFileHelper.showoff'), - value: '@witcat' - }, - ], - thing: [ - { - text: this.formatMessage('WitCatFileHelper.thing.1'), - value: 'true' - }, - { - text: this.formatMessage('WitCatFileHelper.thing.2'), - value: 'false' - }, - ], - type: [ - { - text: this.formatMessage('WitCatFileHelper.number.1'), - value: 'name' - }, - { - text: this.formatMessage('WitCatFileHelper.number.2'), - value: 'content' - }, - ], - } - }; - } - //下载多行文件 - downloads(args) { - download += 1; - if (download < 3) { - let h = args.text; - let text = args.text; - let filenames = args.name; - if (filenames === "") { - filenames = "none.txt" - } - const filename = filenames; - let s = args.s; - if (s != "") { - h = text.split(s).join("\n"); - } else { - h = text; - } - let SuffixName = filename.split(".")[filename.split(".").length - 1]; - if (SuffixName === "bat" || SuffixName === "cmd" || SuffixName === "vbs" || SuffixName === "ps1" || SuffixName === "sh") { - let a = confirm(this.formatMessage('WitCatFileHelper.downloadask') + SuffixName + ":\n" + h); - if (a === false) { - return; - } - } - const content = h; - // 创建隐藏的可下载链接 - let eleLink = document.createElement('a'); - eleLink.download = filename; - eleLink.style.display = 'none'; - // 字符内容转变成blob地址 - let blob = new Blob([content]); - eleLink.href = URL.createObjectURL(blob); - // 触发点击 - document.body.appendChild(eleLink); - eleLink.click(); - // 然后移除 - document.body.removeChild(eleLink); - } - else { - console.warn("下载太频繁!\nToo many downloads!"); - } - } - //下载文件 - download(args) { - download += 1; - if (download < 3) { - const content = args.text; - let filenames = args.name; - if (filenames === "") { - filenames = "none.txt" - } - let SuffixName = filenames.split(".")[filenames.split(".").length - 1]; - if (SuffixName === "bat" || SuffixName === "cmd" || SuffixName === "vbs" || SuffixName === "ps1" || SuffixName === "sh") { - let a = confirm(this.formatMessage('WitCatFileHelper.downloadask') + SuffixName + ":\n" + content); - if (a === false) { - return; - } - } - const filename = filenames; - // 创建隐藏的可下载链接 - let eleLink = document.createElement('a'); - eleLink.download = filename; - eleLink.style.display = 'none'; - // 字符内容转变成blob地址 - let blob = new Blob([content]); - eleLink.href = URL.createObjectURL(blob); - // 触发点击 - document.body.appendChild(eleLink); - eleLink.click(); - // 然后移除 - document.body.removeChild(eleLink); - } - else { - console.warn("下载太频繁!\nToo many downloads!"); - } - } - //读取本地变量 - upload() { - console.warn("文件助手——读取键值对:积木已下线,请更换\nfile_helper-Read key value pair: Block is offline, please replace");/* - const name = args.name; - let h = this.runtime.ccwAPI.getProjectUUID(); - //寻找状态 - let show = "@witcat"; - let value = localStorage.getItem(h + name); - if (value === null) { - let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("#witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("$witcat" + h + "©" + name); - if (value != null) { - show = "$witcat"; - } - } - else { - show = "#witcat"; - } - } - } - else { - show = ""; - } - if (show === "") { - return localStorage.getItem(show + h + name); - } - return localStorage.getItem(show + h + "©" + name);*/ - } - //保存本地变量 - save() { - console.warn("文件助手——保存键值对:积木已下线,请更换\nfile_helper-Save the key pair: Block is offline, please replace");/* - const text = args.text; - const name = args.name; - let h = this.runtime.ccwAPI.getProjectUUID(); - //寻找状态 - let show = "@witcat"; - let value = localStorage.getItem(h + name); - if (value === null) { - let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("#witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("$witcat" + h + "©" + name); - if (value != null) { - show = "$witcat"; - } - } - else { - show = "#witcat"; - } - } - } - else { - localStorage.removeItem(h + name); - } - - if (h === "") { - alert("请先保存作品"); - } else { - localStorage.setItem(show + h + "©" + name, text); - }*/ - } - //删除本地变量 - delete() { - console.warn("文件助手——删除键值对:积木已下线,请更换\nfile_helper-Delet the key pair: Block is offline, please replace");/* - const name = args.name; - let h = this.runtime.ccwAPI.getProjectUUID(); - //寻找状态 - let show = "@witcat"; - let value = localStorage.getItem(h + name); - if (value === null) { - let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("#witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("$witcat" + h + "©" + name); - if (value != null) { - show = "$witcat"; - } - } - else { - show = "#witcat"; - } - } - } - else { - show = ""; - } - if (show === "") { - localStorage.removeItem(show + h + name); - } - - localStorage.removeItem(show + h + "©" + name);*/ - } - //字符串分割 - segmentation(args) { - let text = args.text; - let s = args.s; - let array = text.split(s); - let r = ""; - if (args.thing === "true") { - r = JSON.stringify(array); - } - else { - r = array.join("\n"); - } - return r; - } - //加密 - encrypt(args) { - let str = args.text; - let jiaMi = encodeURIComponent(str); - let jiaM = btoa(jiaMi); - return jiaM; - } - //解密 - decrypt(args) { - let jiaM = args.text; - let jieMi = atob(jiaM); - let jieM = decodeURIComponent(jieMi); - return jieM; - } - //打开文件 - openfile() { - FLAG = 1; - return new Promise(resolve => { - const input = document.createElement("input"); - input.type = "file"; - input.style = "display:none;"; - input.click(); - input.onchange = () => { - const reader = new FileReader(); - const file = input.files[0]; - reader.onload = (e) => { - FLAG = 0; - resolve(e.target.result); - }; - reader.onerror = () => { - FLAG = 0; - resolve(); - }; - reader.readAsText(file); - } - window.onfocus = () => { - // 开始计时或者播放 - setTimeout(() => { - if (FLAG === 1) { - FLAG = 0; - resolve(""); - } - }, 1000); - } - }); - } - //设置状态 - showvar() { - console.warn("文件助手——设置键状态:积木已下线,请更换\nfile_helper-Set key state: Block is offline, please replace");/* - const name = args.name; - let h = this.runtime.ccwAPI.getProjectUUID(); - //寻找状态 - let show = "@witcat"; - let value = localStorage.getItem(h + name); - if (value === null) { - let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("#witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("$witcat" + h + "©" + name); - if (value != null) { - show = "$witcat"; - } - } - else { - show = "#witcat"; - } - } - } - else { - show = ""; - } - let text = localStorage.getItem(show + h + "©" + name);; - if (show === "") { - text = localStorage.getItem(show + h + name); - localStorage.removeItem(show + h + name); - } - localStorage.removeItem(show + h + "©" + name); - if (h === "") { - alert("请先保存作品"); - } else { - localStorage.setItem(args.show + h + "©" + name, text); - }*/ - } - //修改别人的键 - saveother() { - console.warn("文件助手——修改别人的键:积木已下线,请更换\nfile_helper-Modify other people's keys: Block is offline, please replace");/* - let name = args.name; - let h = args.id; - let text = args.text; - //寻找状态 - let show = ""; - let value = localStorage.getItem(h + name); - if (value === null) { - let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("#witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("$witcat" + h + "©" + name); - if (value != null) { - show = "$witcat"; - } - } - else { - show = "#witcat"; - } - } - else { - show = "@witcat"; - } - } - else { - localStorage.removeItem(h + name); - show = "@witcat"; - } - - if (show === "#witcat" || show === "") { - localStorage.setItem("#witcat" + h + "©" + name, text); - }*/ - } - //获取别人的键 - uploadother() { - console.warn("文件助手——获取别人的键:积木已下线,请更换\nfile_helper-Get other people's keys: Block is offline, please replace");/* - let name = args.name; - let h = args.id; - //寻找状态 - let show = ""; - let value = localStorage.getItem(h + name); - if (value === null) { - let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("#witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("$witcat" + h + "©" + name); - if (value != null) { - show = "$witcat"; - } - } - else { - show = "#witcat"; - } - } - } - - if (show != "") { - return localStorage.getItem(show + h + "©" + name); - } - return "";*/ - } - //获取键状态 - other() { - console.warn("文件助手——获取键状态:积木已下线,请更换\nfile_helper-Get key state: Block is offline, please replace");/* - let name = args.name; - let h = args.id; - //寻找状态 - let show = ""; - let value = localStorage.getItem(h + name); - if (value === null) { - let value = localStorage.getItem("@witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("#witcat" + h + "©" + name); - if (value === null) { - value = localStorage.getItem("$witcat" + h + "©" + name); - if (value != null) { - show = "$witcat"; - } - } - else { - show = "#witcat"; - } - } - else { - show = "@witcat"; - } - } - else { - show = "@witcat"; - } - - if (show === "#witcat") { - return "公开"; - } - if (show === "$witcat") { - return "只读"; - } - if (show === "@witcat") { - return "私有"; - } - return "键不存在";*/ - } - //删除多行文本行 - deleteMultiplelinestext(args) { - let texts = multipleText(args.text); - let num = args.num; - if (args.num === "last") { - num = texts.length; - } - if (args.num === "first") { - num = 1; - } - texts.splice(num - 1, 1); - return texts.join("\n"); - } - //插入多行文本 - addMultiplelinestext(args) { - let texts = multipleText(args.texts); - let num = args.num; - if (args.num === "last") { - num = texts.length + 1; - } - if (args.num === "first") { - num = 1; - } - texts.splice(num - 1, 0, args.text); - return texts.join("\n"); - } - //多行文本第几行 - whatMultiplelinestext(args) { - let text = args.text.split("\n"); - let num = args.num; - if (args.num === "last") { - num = text.length; - } - if (args.num === "first") { - num = 1; - } - if (text.length >= num && num > 0) { - return text[num - 1].split("\r")[0]; - } - else { - return ""; - } - } - //多行文本行数 - numMultiplelinestext(args) { - let text = args.text.split("\n"); - return text.length; - } - //多行文本转数组 - turnMultiplelinestext(args) { - let texts = multipleText(args.text); - return JSON.stringify(texts); - } - //数组转多行文本 - turnsMultiplelinestext(args) { - let texts = JSON.parse(args.text); - return texts.join("\n"); - } - //键值对数量 - numbers() { - console.warn("文件助手——获取键值对数量:积木已下线,请更换\nfile_helper-number of keys: Block is offline, please replace");/* - let a = 0; - for (let i = 0; i < localStorage.length; i++) { - if (localStorage.key(i).slice(1, 7) === "witcat") { - a++; - } - } - return a;*/ - } - //键值对内容 - number() { - console.warn("文件助手——获取键值对内容:积木已下线,请更换\nfile_helper-Get the key pair: Block is offline, please replace"); - /* - let a = 0; - let i = 0; - for (i = 0; a < args.num && i < localStorage.length; i++) { - if (localStorage.key(i).slice(1, 7) === "witcat") { - a++; - } - } - if (a === args.num) { - if (args.type === "name") { - return localStorage.key(i - 1).slice(localStorage.key(i - 1).indexOf("©", 7) + 1, localStorage.key(i - 1).length); - } - else { - return localStorage.getItem(localStorage.key(i - 1)); - } - } - return "";*/ - } - //可下载文本数量 - downloadnum() { - return 3 - download; - } + constructor(runtime) { + /** + * 被打开的文件列表(类似input.files) + * @type {File[]} + */ + this.filelist = []; + + /** + * 预读取的文件内容 + * @type {string[]} + */ + this.filecontent = []; + + /** + * 短时间内连续下载数量 + */ + this.downloadcnt = 0; + + setInterval(() => { + if (this.downloadcnt > 0) { + this.downloadcnt--; + } + }, 1000) + + this.runtime = runtime; + this._formatMessage = runtime.getFormatMessage({ + "zh-cn": { + "WitCatFileHelper.name": "[beta]白猫的文件助手", + "WitCatFileHelper.filehelper": "文件处理", + "WitCatFileHelper.inputmanagement": "键值对", + "WitCatFileHelper.imageprocess": "图片", + "WitCatFileHelper.Multiplelinestext": "多行文本", + "WitCatFileHelper.downloads": "将内容[text]按[s]分割后命名为[name]并下载多行文本", + "WitCatFileHelper.download": "将内容[text]命名为[name]并下载", + "WitCatFileHelper.save": "设置键[name]的值为[text]", + "WitCatFileHelper.upload": "获取键[name]的值", + "WitCatFileHelper.delete": "删除键[name]", + "WitCatFileHelper.segmentation": "将[text]按[s]分割为[thing]", + "WitCatFileHelper.encrypt": "base64编码[text]", + "WitCatFileHelper.decrypt": "base64解码[text]", + "WitCatFileHelper.openfile": "打开文件", + "WitCatFileHelper.showvar": "设置其他作品[show]键[name]的值", + "WitCatFileHelper.saveother": "设置作品ID[id]的键[name]的值为[text]", + "WitCatFileHelper.uploadother": "获取作品[id]的键[name]的值", + "WitCatFileHelper.other": "作品[id]的键[name]的状态", + "WitCatFileHelper.showon": "只读", + "WitCatFileHelper.showoff": "私有", + "WitCatFileHelper.showall": "公开", + "WitCatFileHelper.deleteMultiplelinestext": "删除[text]的第[num]行", + "WitCatFileHelper.addMultiplelinestext": "将[text]插入到[texts]的第[num]行之前", + "WitCatFileHelper.whatMultiplelinestext": "[text]的第[num]行", + "WitCatFileHelper.numMultiplelinestext": "[text]的行数", + "WitCatFileHelper.thing.1": "数组", + "WitCatFileHelper.thing.2": "多行文本", + "WitCatFileHelper.number": "第[num]个键值对的[type]", + "WitCatFileHelper.numbers": "键值对数量", + "WitCatFileHelper.number.1": "键", + "WitCatFileHelper.number.2": "值", + "WitCatFileHelper.turnMultiplelinestext": "将多行文本[text]转化为数组", + "WitCatFileHelper.turnsMultiplelinestext": "将数组[text]转化为多行文本", + "WitCatFileHelper.downloadnum": "可下载文件数量", + "WitCatFileHelper.downloadask": "作品企图下载疑似会威胁电脑的文件,是否继续?\n内容如下:", + "WitCatFileHelper.openfiless": "打开[name]的[nums]并读取为[type]", + "WitCatFileHelper.openfiless.1": "单文件", + "WitCatFileHelper.openfiless.2": "多文件", + "WitCatFileHelper.openfile.1": "UTF-8(文本)", + "WitCatFileHelper.file": "上次打开第[num]个文件的[type]", + "WitCatFileHelper.file.1": "文件名", + "WitCatFileHelper.file.2": "文件后缀", + "WitCatFileHelper.file.3": "文件大小(KB)", + "WitCatFileHelper.file.4": "文件内容", + "WitCatFileHelper.file.5": "文件大小(字节数)", + "WitCatFileHelper.file.6": "文件大小(自动单位)", + "WitCatFileHelper.docs": "📖拓展教程", + "WitCatFileHelper.arrayjoin": "用[s]作为分隔符合并数组[text]", + "WitCatFileHelper.asks": "文件大小太大,可能导致浏览器崩溃,确定继续?", + }, + en: { + "WitCatFileHelper.name": "[beta]WitCat’s File Helper", + "WitCatFileHelper.filehelper": "File", + "WitCatFileHelper.inputmanagement": "Key-value pair", + "WitCatFileHelper.imageprocess": "Image", + "WitCatFileHelper.Multiplelinestext": "Multi-line text", + "WitCatFileHelper.downloads": "Download split text [text] by [s] named [name]", + "WitCatFileHelper.download": "Download text [text] named [name]", + "WitCatFileHelper.save": "Set value of key [name] to [text]", + "WitCatFileHelper.upload": "value of key [name]", + "WitCatFileHelper.delete": "Delete key [name]", + "WitCatFileHelper.segmentation": "Split [text] by [s] to [thing]", + "WitCatFileHelper.encrypt": "base64 encode[text]", + "WitCatFileHelper.decrypt": "base64 decode[text]", + "WitCatFileHelper.openfile": "load file", + "WitCatFileHelper.showvar": "Other project [show] value of key [name]", + "WitCatFileHelper.saveother": "Set value of key [name] of project [id] to [text]", + "WitCatFileHelper.uploadother": "value of key [name] of project [id]", + "WitCatFileHelper.other": "get permission of value [name] of project [id]", + // Is value [name] of project [id] readable? + // Is value [name] of project [id] modifiable? + "WitCatFileHelper.showon": "can read", + "WitCatFileHelper.showoff": "can't read", + "WitCatFileHelper.showall": "can read and modify", + "WitCatFileHelper.deleteMultiplelinestext": "delete the [num] line of [text]", + "WitCatFileHelper.addMultiplelinestext": "insert [text] before line [num] of [texts]", + "WitCatFileHelper.whatMultiplelinestext": "line [num] of [text]", + "WitCatFileHelper.numMultiplelinestext": "number of lines in [text]", + "WitCatFileHelper.thing.1": "array", + "WitCatFileHelper.thing.2": "multi-line text", + "WitCatFileHelper.number": "[type] of key [num]", + "WitCatFileHelper.numbers": "key count", + "WitCatFileHelper.number.1": "key", + "WitCatFileHelper.number.2": "value", + "WitCatFileHelper.turnMultiplelinestext": "convert multi-line [text] to array", + "WitCatFileHelper.turnsMultiplelinestext": "convert array[text] to multi-line text", + "WitCatFileHelper.downloadnum": "number of downloadable files", + "WitCatFileHelper.downloadask": "The project attempts to download a suspicious file, continue? \n File content: ", + "WitCatFileHelper.openfiless": "load[name]`s[nums]file read as[type]", + "WitCatFileHelper.openfiless.1": "Single", + "WitCatFileHelper.openfiless.2": "multi", + "WitCatFileHelper.openfile.1": "UTF-8(text)", + "WitCatFileHelper.file": "[type] of [num] file opened", + "WitCatFileHelper.file.1": "name", + "WitCatFileHelper.file.2": "extension", + "WitCatFileHelper.file.3": "size(KB)", + "WitCatFileHelper.file.4": "content", + "WitCatFileHelper.file.5": "size(bytes)", + "WitCatFileHelper.file.6": "size(auto-unit)", + "WitCatFileHelper.docs": "📖 Tutorial", + "WitCatFileHelper.arrayjoin": "join array [text], seperating by [s]", + "WitCatFileHelper.asks": "The file size is too large and may cause the browser to crash, are you sure to continue?", + } + }) + } + + /** + * 翻译 + * @param {string} id + * @returns {string} + */ + formatMessage(id) { + return this._formatMessage({ + id, + default: id, + description: id + }); + } + + getInfo() { + return { + id: witcat_file_helper_extensionId, // 拓展id + name: this.formatMessage("WitCatFileHelper.name"), // 拓展名 + blockIconURI: witcat_file_helper_icon, + menuIconURI: witcat_file_helper_icon, + color1: "#60D6F4", + color2: "#55a7f7", + blocks: [ + { + blockType: "button", + text: this.formatMessage('WitCatFileHelper.docs'), + onClick: this.docs, + }, + "---" + this.formatMessage("WitCatFileHelper.filehelper"), + { + opcode: "downloads", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.downloads"), + arguments: { + text: { + type: "string", + defaultValue: 'awa!!!|awa!!!', + }, + name: { + type: "string", + defaultValue: 'wit_cat.txt', + }, + s: { + type: "string", + defaultValue: '|', + }, + }, + }, + { + opcode: "download", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.download"), + arguments: { + text: { + type: "string", + defaultValue: 'awa!!!', + }, + name: { + type: "string", + defaultValue: 'wit_cat.txt', + }, + }, + }, + { + opcode: "downloadnum", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.downloadnum"), + arguments: {}, + }, + { + opcode: "openfile", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.openfile"), + disableMonitor: true, + arguments: {}, + }, + { + opcode: "openfiless", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.openfiless"), + disableMonitor: true, + arguments: { + name: { + type: "string", + defaultValue: ".txt,.cpp", + }, + nums: { + type: "string", + menu: "openfiless", + }, + type: { + type: "string", + menu: "openfile", + } + }, + }, + { + opcode: "file", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.file"), + disableMonitor: true, + arguments: { + num: { + type: "number", + defaultValue: '1', + }, + type: { + type: "string", + menu: 'file', + }, + }, + }, + //键值对(已下线) + { + opcode: "upload", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.upload"), + hideFromPalette: true, + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "save", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.save"), + hideFromPalette: true, + arguments: { + text: { + type: "string", + defaultValue: '0', + }, + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "delete", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.delete"), + hideFromPalette: true, + arguments: { + name: { + type: "string", + defaultValue: 'i', + }, + }, + }, + { + opcode: "showvar", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.showvar"), + hideFromPalette: true, + arguments: { + name: { + type: "string", + defaultValue: "i", + }, + show: { + type: "string", + menu: "setvariable", + }, + }, + }, + { + opcode: "number", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.number"), + hideFromPalette: true, + arguments: { + num: { + type: "number", + defaultValue: "1", + }, + type: { + type: "string", + menu: "type", + }, + }, + }, + { + opcode: "numbers", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.numbers"), + hideFromPalette: true, + arguments: {}, + }, + { + opcode: "saveother", + blockType: "command", + text: this.formatMessage("WitCatFileHelper.saveother"), + hideFromPalette: true, + arguments: { + id: { + type: "string", + defaultValue: "6373950041d21d2d2cd0da9b", + }, + name: { + type: "string", + defaultValue: "i", + }, + text: { + type: "string", + defaultValue: "wit_cat!", + }, + }, + }, + { + opcode: "uploadother", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.uploadother"), + hideFromPalette: true, + arguments: { + id: { + type: "string", + defaultValue: "6373950041d21d2d2cd0da9b", + }, + name: { + type: "string", + defaultValue: "i", + }, + }, + }, + { + opcode: "other", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.other"), + hideFromPalette: true, + arguments: { + id: { + type: "string", + defaultValue: "6373950041d21d2d2cd0da9b", + }, + name: { + type: "string", + defaultValue: "i", + }, + }, + }, + "---" + "Base64", + { + opcode: "encrypt", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.encrypt"), + arguments: { + text: { + type: "string", + defaultValue: 'i love china', + }, + }, + }, + { + opcode: "decrypt", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.decrypt"), + arguments: { + text: { + type: "string", + defaultValue: 'aSBsb3ZlIGNoaW5h', + }, + }, + }, + "---" + this.formatMessage("WitCatFileHelper.Multiplelinestext"), + { + opcode: "arrayjoin", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.arrayjoin"), + arguments: { + text: { + type: "string", + defaultValue: '["wow","!!!"]', + }, + s: { + type: "string", + defaultValue: '/', + }, + }, + }, + { + opcode: "segmentation", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.segmentation"), + arguments: { + text: { + type: "string", + defaultValue: 'wow/!!!', + }, + s: { + type: "string", + defaultValue: '/', + }, + thing: { + type: "string", + menu: 'thing', + } + }, + }, + { + opcode: "deleteMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.deleteMultiplelinestext"), + arguments: { + text: { + type: "string", + defaultValue: "wit_cat!!!\nwit_cat!!!", + }, + num: { + type: "number", + defaultValue: "1", + } + }, + }, + { + opcode: "addMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.addMultiplelinestext"), + arguments: { + text: { + type: "string", + defaultValue: "wit_cat!!!", + }, + texts: { + type: "string", + defaultValue: "wit_cat!!!\nwit_cat!!!", + }, + num: { + type: "string", + defaultValue: "last", + } + }, + }, + { + opcode: "whatMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.whatMultiplelinestext"), + arguments: { + text: { + type: "string", + defaultValue: "wit_cat!!!", + }, + num: { + type: "string", + defaultValue: "1", + } + }, + }, + { + opcode: "numMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.numMultiplelinestext"), + arguments: { + text: { + type: "string", + defaultValue: "wit_cat!!!\nwit_cat!!!", + }, + }, + }, + { + opcode: "turnMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.turnMultiplelinestext"), + arguments: { + text: { + type: "string", + defaultValue: "wit_cat!!!\nwit_cat!!!", + }, + }, + }, + { + opcode: "turnsMultiplelinestext", + blockType: "reporter", + text: this.formatMessage("WitCatFileHelper.turnsMultiplelinestext"), + arguments: { + text: { + type: "string", + defaultValue: "[\"wit_cat!!!\",\"wit_cat!!!\"]", + }, + }, + }, + ], + menus: { + setvariable: [ + { + text: this.formatMessage('WitCatFileHelper.showall'), + value: '#witcat' + }, + { + text: this.formatMessage('WitCatFileHelper.showon'), + value: '$witcat' + }, + { + text: this.formatMessage('WitCatFileHelper.showoff'), + value: '@witcat' + }, + ], + thing: [ + { + text: this.formatMessage('WitCatFileHelper.thing.1'), + value: 'true' + }, + { + text: this.formatMessage('WitCatFileHelper.thing.2'), + value: 'false' + }, + ], + type: [ + { + text: this.formatMessage('WitCatFileHelper.number.1'), + value: 'name' + }, + { + text: this.formatMessage('WitCatFileHelper.number.2'), + value: 'content' + }, + ], + file: [ + { + text: this.formatMessage('WitCatFileHelper.file.1'), + value: 'name' + }, + { + text: this.formatMessage('WitCatFileHelper.file.2'), + value: 'suffix' + }, + { + text: this.formatMessage('WitCatFileHelper.file.3'), + value: 'size' + }, + { + text: this.formatMessage('WitCatFileHelper.file.5'), + value: 'sizeB' + }, + { + text: this.formatMessage('WitCatFileHelper.file.6'), + value: 'sizeH' + }, + { + text: this.formatMessage('WitCatFileHelper.file.4'), + value: 'content' + }, + ], + openfile: [ + { + text: this.formatMessage('WitCatFileHelper.openfile.1'), + value: 'utf-8' + }, + ], + openfiless: [ + { + text: this.formatMessage('WitCatFileHelper.openfiless.1'), + value: 'Single' + }, + { + text: this.formatMessage('WitCatFileHelper.openfiless.2'), + value: 'multiple' + }, + ], + } + }; + } + /** + * 打开教程 + * @returns {void} + */ + docs() { + let a = document.createElement('a'); + a.href = "https://www.ccw.site/post/d6d96e80-3f58-4a19-b7e6-c567d3a6a583"; + a.rel = "noopener noreferrer"; + a.target = "_blank"; + a.click(); + } + + /** 判断是否允许下载文件 + * @param {()=>void} callback 可以下载则调用 + */ + _downloadable(callback) { + if (this.downloadcnt < 3) { + this.downloadcnt += 1; + callback(); + } + else { + console.warn("下载太频繁!\nToo many downloads!"); + } + } + + /** + * 下载方法 + * @param {string} text 下载的文本 + * @param {string} name 下载后文件名 + * @return {void} + */ + _downloadText(text, name) { + // 字符内容转变成blob地址 + const blob = new Blob([text]); + const url = URL.createObjectURL(blob); + let SuffixName = ""; + if (name.includes(".")) { + SuffixName = name.substring(name.lastIndexOf(".") + 1); + } + // 疑似危险的文件后缀 + const maliext = ["bat", "cmd", "vbs", "ps1", "sh"]; + if (maliext.includes(SuffixName.toLowerCase())) { + const ask = this.formatMessage("WitCatFileHelper.downloadask"); + // 如果生成大量文字,则截短文字 + let showtext = text; + const max_length = 1000; + const short_length = 20; + if (showtext.length > max_length) { + showtext = `${ + showtext.slice(0, max_length - short_length) + }(... 省略 ${ + showtext.length - max_length + short_length + } 字符)`; + } + let a = confirm(ask + "\n\n" + showtext); + if (!a) { + return; + } + } + const a = document.createElement("a"); + a.href = url; + a.download = name; + a.target = "_blank"; + a.click(); + } + + /** + * 下载多行文件 + * @param {Object} args + * @param {SCarg} args.text 文本 + * @param {SCarg} args.name 文件名 + * @param {SCarg} args.s 分隔符 + * @returns {void} + */ + downloads(args) { + this._downloadable(() => { + let h = String(args.text); + + let s = String(args.s); + if (s != "") { + h = h.split(s).join("\n"); + } + + this._downloadText(h, String(args.name)); + }); + } + /** + * 下载文本文件 + * @param {object} args + * @param {SCarg} args.text 文本 + * @param {SCarg} args.name 文件名 + * @returns {void} + */ + download(args) { + this._downloadable(() => { + const content = String(args.text); + this._downloadText(content, String(args.name)); + }); + } + /** + * 读取本地变量 + * @deprecated + */ + upload() { + console.warn("文件助手:读取本地变量积木已下线\nFile Helper: read key value pair block has been offline"); + } + /** + * 保存本地变量 + * @deprecated + */ + save() { + console.warn("文件助手:保存本地变量积木已下线\nFile Helper: save key value pair block has been offline"); + } + /** + * 删除本地变量 + * @deprecated + */ + delete() { + console.warn("文件助手:删除本地变量积木已下线\nFile Helper: delete key value pair block has been offline"); + } + /** + * 字符串分割 + * @param {object} args + * @param {SCarg} args.text 文本 + * @param {SCarg} args.s 分隔符 + * @param {SCarg|"true"|"false"} args.thing 分割为数组? + * @returns {string} 分割结果 + */ + segmentation(args) { + let text = String(args.text); + let s = String(args.s); + let array = text.split(s); + let r = ""; + if (args.thing === "true") { + r = JSON.stringify(array); + } + else { + r = array.join("\n"); + } + return r; + } + /** + * base64编码 + * @param {object} args + * @param {SCarg} args.text 要编码的文本 + * @returns {string} 编码结果 + */ + encrypt(args) { + try { + return Base64.encode(String(args.text)) + } + catch { + return ""; + } + } + /** + * base64解码 + * @param {object} args + * @param {SCarg} args.text 要解码的文本 + * @returns {string} 解码结果 + */ + decrypt(args) { + try { + return Base64.decode(String(args.text)) + } + catch { + return ""; + } + } + + /** + * 打开文件选择框 + * @param {string} accept 接受的文件扩展名 + * @param {boolean} multiple 接受多个文件 + * @return {Promise} [异步地]返回选择后的文件列表input.files转换成的数组(可能没有文件) + */ + _inputfileclick(accept, multiple) { + return new Promise((resolve, reject) => { + const input = document.createElement("input"); + input.type = "file"; + input.accept = accept; + input.style.display = "none"; + input.multiple = multiple; + input.click(); + input.addEventListener("change", () => { + if (input.files === null) { + reject(new Error("不应该看到这个")); + } else { + // 返回了关键的 input.files,而不是整个 input。 + // 之后如果要考虑“读取素材库文件”,“拖动导入文件”等 + // 只能获得 Blob/File 的情况,可以方便适配 + // 这里加 Array.from 是因为 input.files 是 FileList, + // 不是 File[],一些数组拥有的功能它没有。虽然一般情况下 + // 不会注意到区别,但是类型检查会把这种情况查出来。 + resolve(Array.from(input.files)); + } + }, {once: true}); // 只触发一次 + window.addEventListener("focus", () => { + setTimeout(() => { + if (input.files === null) { + reject(new Error("不应该看到这个")); + } else { + resolve(Array.from(input.files)); + } + }, 1000); + }, {once: true}); // 只触发一次 + }); + } + + /** + * 读取文件 + * @param {File|Blob} file File 或者 Blob + * @param {"arraybuffer"|"dataurl"|"text"} mode 读取模式 + * @return {Promise} [异步地]返回读取后的内容 + */ + _readerasync(file, mode) { + return new Promise((resolve, reject) => { + const reader = new FileReader(); + reader.onload = () => { + resolve(reader.result); + }; + reader.onerror = (e) => { + reject(e); + }; + switch (mode) { + case "arraybuffer": + reader.readAsArrayBuffer(file); + break; + case "dataurl": + reader.readAsDataURL(file); + break; + case "text": + reader.readAsText(file); + break; + default: + reject(new Error("mode 错误: 应该是 arraybuffer, dataurl 或者 text")); + return; + } + }); + } + + /** + * 提前加载所有已经打开的文件,注意是 async 的 + */ + async _preloadcontent() { + this.filecontent = []; + for (let file of this.filelist) { + this.filecontent.push(String(await this._readerasync(file, "text"))); + } + } + + /** + * 打开文件 + * @returns {Promise} 文件内容 + */ + async openfile() { + // 因为可以返回 Promise,所以这里直接用 async + const filelist = await this._inputfileclick("", false); + // 首先,检查所有打开的文件大小 + if (filelist.some((file) => file.size > 50 * 1024 * 1024)) { // 50M + let usercheck = confirm(this.formatMessage("WitCatFileHelper.asks")); + if (!usercheck) { + console.error("文件过大\nfile is too lage."); + return ""; + } + } + + // 检查通过后,才把文件列表更新到类里。 + // 注意读取后的 File[] 保存到了 this.filelist + this.filelist = filelist; + // 提前加载所有文件的内容,这样就不需要担心后面读取文件内容的时候要异步了 + this._preloadcontent(); + + const file = this.filelist[0]; + if (file === undefined) { + // 记住这是 async + return ""; + } + return String(await this._readerasync(file, "text")); + } + + /** + * 打开任意文件 + * @param {object} args + * @param {SCarg} args.name 文件类型 + * @param {SCarg|"Single"|"multiple"} args.nums 单选/多选 + * @param {SCarg|"base64"|"utf-8"} args.type 返回值类型 + * @returns {Promise} 文件内容 + */ + async openfiless(args) { + try { + // 因为可以返回 Promise,所以这里直接用 async + // 注意读取后的 File[] 保存到了 this.filelist + let accepttype = String(args.name); + let multiple = args.nums === "multiple"; + + const filelist = await this._inputfileclick(accepttype, multiple); + // 首先,检查所有打开的文件大小 + if (filelist.some((file) => file.size > 50 * 1024 * 1024)) { // 50M + let usercheck = confirm(this.formatMessage("WitCatFileHelper.asks")); + if (!usercheck) { + console.error("文件过大\nfile is too lage."); + return ""; + } + } + + // 检查通过后,才把文件列表更新到类里。 + this.filelist = filelist; + // 提前加载所有文件的内容,这样就不需要担心后面读取文件内容的时候要异步了 + this._preloadcontent(); + + // 返回第一个文件的内容 + if (this.filelist.length === 0) { + console.warn("File_Helper.js: 没有选择文件") + // 这是 async 函数 + return ""; + } + + const file = this.filelist[0]; + + /** @type {"text"|"arraybuffer"|"dataurl"} */ + let mode = "text"; + switch (args.type) { + case "base64": + mode = "dataurl"; + break; + case "utf-8": + mode = "text"; + break; + } + return String(await this._readerasync(file, mode)); + } + catch (e) { + console.error("witcat open any file error:", e); + return ""; + } + } + + /** + * 打开文件的信息 + * @param {object} args + * @param {SCarg|"name"|"suffix"|"size"|"content"} args.type + * @param {SCarg} args.num + * @returns {string} + */ + file(args) { + try { + const file = this.filelist[Number(args.num) - 1]; + switch (args.type) { + case "name": + return file.name; + case "suffix": + let n = file.name; + if (n.includes(".")) { + return n.substring(n.lastIndexOf(".") + 1); + } else { + return ""; + } + case "size": + return file.size / 1024 + "KB"; + case "sizeH": + return HumanSize(file.size); + case "sizeB": + return file.size; + case "content": + // 直接读取预加载的文件内容 + return this.filecontent[Number(args.num) - 1]; + default: + return ""; + } + } + catch { + return ""; + } + } + /** + * 设置状态 + * @deprecated + */ + showvar() { + console.warn("文件助手:判断键值对积木已下线\nFile Helper: Determine the key value pair block has been offline"); + } + /** + * 修改别人的键 + * @deprecated + */ + saveother() { + console.warn("文件助手:判断键值对积木已下线\nFile Helper: Determine the key value pair block has been offline"); + } + /** + * 获取别人的键 + * @deprecated + */ + uploadother() { + console.warn("文件助手:判断键值对积木已下线\nFile Helper: Determine the key value pair block has been offline"); + } + /** + * 获取键状态 + * @deprecated + */ + other() { + console.warn("文件助手:判断键值对积木已下线\nFile Helper: Determine the key value pair block has been offline"); + } + /** + * 删除多行文本行 + * @param {object} args + * @param {SCarg} args.text 多行文本 + * @param {SCarg} args.num 删除的文本行数(1开始) + * @returns {string} 结果 + */ + deleteMultiplelinestext(args) { + let texts = multipleText(String(args.text)); + let num = 0; + // 已经是 Scratch 3.0 了,这种遗老就不要了吧 + switch (args.num) { + case "first": + num = 1; + break; + case "last": + num = texts.length; + break; + default: + num = Number(args.num); + } + texts.splice(num - 1, 1); + return texts.join("\n"); + } + /** + * 插入多行文本 + * @param {object} args + * @param {SCarg} args.texts 多行文本 + * @param {SCarg} args.num 插入文本位置 + * @param {SCarg} args.text 要插入的文本 + * @returns {string} 结果 + */ + addMultiplelinestext(args) { + let texts = multipleText(String(args.texts)); + let num = 0; + switch (args.num) { + case "first": + num = 1; + break; + case "last": + num = texts.length + 1; + break; + default: + num = Number(args.num); + } + texts.splice(num - 1, 0, String(args.text)); + return texts.join("\n"); + } + /** + * 多行文本第几行 + * @param {object} args + * @param {SCarg} args.text 多行文本 + * @param {SCarg} args.num 获取的行数 + * @returns {string} 结果 + */ + whatMultiplelinestext(args) { + let text = String(args.text).split("\n"); + let num = 0; + switch (args.num) { + case "first": + num = 1; + break; + case "last": + num = text.length; + break; + default: + num = Number(args.num); + } + if (text.length >= num && num > 0) { + // 万一是小数呢? + return text[num - 1].split("\r")[0]; + } + else { + return ""; + } + } + /** + * 多行文本行数 + * @param {object} args + * @param {SCarg} args.text 多行文本 + * @returns {number} 行数 + */ + numMultiplelinestext(args) { + let text = String(args.text).split("\n"); + return text.length; + } + /** + * 多行文本转数组 + * @param {object} args + * @param {SCarg} args.text 多行文本 + * @returns {string} 数组的JSON + */ + turnMultiplelinestext(args) { + let texts = multipleText(String(args.text)); + return JSON.stringify(texts); + } + /** + * 数组转多行文本 + * @param {object} args + * @param {SCarg} args.text 数组 + * @returns {Promise} 多行文本 + */ + turnsMultiplelinestext(args) { + let texts = JSON.parse(String(args.text)); + return texts.join("\n"); + } + /** + * 判断键值对 + * @deprecated + */ + number() { + console.warn("文件助手:判断键值对积木已下线\nFile Helper: Determine the key value pair block has been offline");/* + return new Promise(resolve => { + console.log(args.num) + read(args.num, e => { + return e.value; + }); + });*/ + } + /** + * 键值对内容 + * @deprecated + */ + numbers() { + console.warn("文件助手:键值对数量积木已下线\nFile Helper: number of key value pair block has been offline");/* + return new Promise(resolve => { + read("witcat" + h + "#" + name, e => { + return e.value; + }); + });*/ + } + /** + * 可下载文本数量 + * @returns {number} + */ + downloadnum() { + return 3 - this.downloadcnt; + } + /** + * 数组合并成文本 + * @param {object} args + * @param {SCarg} args.text 要插入的文本 + * @param {SCarg} args.s 分隔符 + * @returns {Promise} 结果 + */ + arrayjoin(args) { + return JSON.parse(String(args.text)).join(args.s); + } } window.tempExt = { - Extension: WitCatFileHelper, - info: { - name: "WitCatFileHelper.name", - description: "WitCatFileHelper.descp", - extensionId: extensionId, - iconURL: _picture, - insetIconURL: _icon, - featured: true, - disabled: false, - collaborator: "白猫 @ CCW" - }, - l10n: { - "zh-cn": { - "WitCatFileHelper.name": "[beta]白猫的文件助手", - "WitCatFileHelper.descp": "读取/处理本地数据" - }, - en: { - "WitCatFileHelper.name": "[beta]WitCat’s File Helper", - "WitCatFileHelper.descp": "Handling local data" - } - } + Extension: WitCatFileHelper, + info: { + name: "WitCatFileHelper.name", + description: "WitCatFileHelper.descp", + extensionId: witcat_file_helper_extensionId, + iconURL: witcat_file_helper_picture, + insetIconURL: witcat_file_helper_icon, + featured: true, + disabled: false, + collaborator: "白猫 @ CCW" + }, + l10n: { + "zh-cn": { + "WitCatFileHelper.name": "[beta]白猫的文件助手", + "WitCatFileHelper.descp": "读取/处理本地数据" + }, + en: { + "WitCatFileHelper.name": "[beta]WitCat’s File Helper", + "WitCatFileHelper.descp": "Handling local data" + } + } }; -/* vim: set expandtab tabstop=2 shiftwidth=2: */ - -//多行文本解码 +/** + * 多行文本解码 + * @param {string} text + * @returns {string[]} + */ function multipleText(text) { - let texts = text.split("\n"); - let a = []; - if (text.indexOf("\r") !== -1) { - texts.forEach(e => { - a.push(e.split("\r")[0]); - }); - } - else { - a = texts; - } - return a; + // 最好用正则处理\r,\n,\r\n三种换行符 + // let texts = text.split(/\r\n?|\n/g); + + let texts = text.split("\n"); + // 如果末尾是 \r,那么去掉。 + let a = texts.map(line => line.slice(-1) === "\r" ? line.slice(0, -1) : line); + return a; +} + +/** + * 把文件大小转换成合适的单位 (K, M, G, T, P) + * @param {number} size 文件大小 + * @returns {string} + */ +function HumanSize(size) { + const units = ["", "K", "M", "G", "T", "P"]; + let i = 0; + let sizen = size; + while (i < units.length - 1 && sizen > 1024) { + i++; + sizen /= 1024; + } + if (i === 0) { // 没有单位 + return String(sizen); + } else { + // 保留两位小数 + return sizen.toFixed(2) + units[i]; + } } diff --git a/wit_cat/README.md b/wit_cat/README.md new file mode 100644 index 00000000..6d6d2c3f --- /dev/null +++ b/wit_cat/README.md @@ -0,0 +1 @@ +文件助手扩展使用了来自 [js-base64](https://github.com/dankogai/js-base64) 的代码 (js-base64 使用 [BSD 3-Clause 协议](https://github.com/dankogai/js-base64/blob/main/LICENSE.md)) diff --git a/wit_cat/base64.js b/wit_cat/base64.js new file mode 100644 index 00000000..19d2917e --- /dev/null +++ b/wit_cat/base64.js @@ -0,0 +1,300 @@ +// 此处代码来自 https://github.com/dankogai/js-base64/blob/main/base64.mjs +// 使用 BSD 3-Clause 协议 +// 协议地址: https://github.com/dankogai/js-base64/blob/main/LICENSE.md + +/** + * base64.ts + * + * Licensed under the BSD 3-Clause License. + * http://opensource.org/licenses/BSD-3-Clause + * + * References: + * http://en.wikipedia.org/wiki/Base64 + * + * @author Dan Kogai (https://github.com/dankogai) + */ +const version = '3.7.5'; +/** + * @deprecated use lowercase `version`. + */ +const VERSION = version; +const _hasatob = typeof atob === 'function'; +const _hasbtoa = typeof btoa === 'function'; +const _hasBuffer = typeof Buffer === 'function'; +const _TD = typeof TextDecoder === 'function' ? new TextDecoder() : undefined; +const _TE = typeof TextEncoder === 'function' ? new TextEncoder() : undefined; +const b64ch = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='; +const b64chs = Array.prototype.slice.call(b64ch); +const b64tab = ((a) => { + let tab = {}; + a.forEach((c, i) => tab[c] = i); + return tab; +})(b64chs); +const b64re = /^(?:[A-Za-z\d+\/]{4})*?(?:[A-Za-z\d+\/]{2}(?:==)?|[A-Za-z\d+\/]{3}=?)?$/; +const _fromCC = String.fromCharCode.bind(String); +const _U8Afrom = typeof Uint8Array.from === 'function' + ? Uint8Array.from.bind(Uint8Array) + : (it) => new Uint8Array(Array.prototype.slice.call(it, 0)); +const _mkUriSafe = (src) => src + .replace(/=/g, '').replace(/[+\/]/g, (m0) => m0 == '+' ? '-' : '_'); +const _tidyB64 = (s) => s.replace(/[^A-Za-z0-9\+\/]/g, ''); +/** + * polyfill version of `btoa` + */ +const btoaPolyfill = (bin) => { + // console.log('polyfilled'); + let u32, c0, c1, c2, asc = ''; + const pad = bin.length % 3; + for (let i = 0; i < bin.length;) { + if ((c0 = bin.charCodeAt(i++)) > 255 || + (c1 = bin.charCodeAt(i++)) > 255 || + (c2 = bin.charCodeAt(i++)) > 255) + throw new TypeError('invalid character found'); + u32 = (c0 << 16) | (c1 << 8) | c2; + asc += b64chs[u32 >> 18 & 63] + + b64chs[u32 >> 12 & 63] + + b64chs[u32 >> 6 & 63] + + b64chs[u32 & 63]; + } + return pad ? asc.slice(0, pad - 3) + "===".substring(pad) : asc; +}; +/** + * does what `window.btoa` of web browsers do. + * @param {String} bin binary string + * @returns {string} Base64-encoded string + */ +const _btoa = _hasbtoa ? (bin) => btoa(bin) + : _hasBuffer ? (bin) => Buffer.from(bin, 'binary').toString('base64') + : btoaPolyfill; +const _fromUint8Array = _hasBuffer + ? (u8a) => Buffer.from(u8a).toString('base64') + : (u8a) => { + // cf. https://stackoverflow.com/questions/12710001/how-to-convert-uint8-array-to-base64-encoded-string/12713326#12713326 + const maxargs = 0x1000; + let strs = []; + for (let i = 0, l = u8a.length; i < l; i += maxargs) { + strs.push(_fromCC.apply(null, u8a.subarray(i, i + maxargs))); + } + return _btoa(strs.join('')); + }; +/** + * converts a Uint8Array to a Base64 string. + * @param {boolean} [urlsafe] URL-and-filename-safe a la RFC4648 §5 + * @returns {string} Base64 string + */ +const fromUint8Array = (u8a, urlsafe = false) => urlsafe ? _mkUriSafe(_fromUint8Array(u8a)) : _fromUint8Array(u8a); +// This trick is found broken https://github.com/dankogai/js-base64/issues/130 +// const utob = (src: string) => unescape(encodeURIComponent(src)); +// reverting good old fationed regexp +const cb_utob = (c) => { + if (c.length < 2) { + var cc = c.charCodeAt(0); + return cc < 0x80 ? c + : cc < 0x800 ? (_fromCC(0xc0 | (cc >>> 6)) + + _fromCC(0x80 | (cc & 0x3f))) + : (_fromCC(0xe0 | ((cc >>> 12) & 0x0f)) + + _fromCC(0x80 | ((cc >>> 6) & 0x3f)) + + _fromCC(0x80 | (cc & 0x3f))); + } + else { + var cc = 0x10000 + + (c.charCodeAt(0) - 0xD800) * 0x400 + + (c.charCodeAt(1) - 0xDC00); + return (_fromCC(0xf0 | ((cc >>> 18) & 0x07)) + + _fromCC(0x80 | ((cc >>> 12) & 0x3f)) + + _fromCC(0x80 | ((cc >>> 6) & 0x3f)) + + _fromCC(0x80 | (cc & 0x3f))); + } +}; +const re_utob = /[\uD800-\uDBFF][\uDC00-\uDFFFF]|[^\x00-\x7F]/g; +/** + * @deprecated should have been internal use only. + * @param {string} src UTF-8 string + * @returns {string} UTF-16 string + */ +const utob = (u) => u.replace(re_utob, cb_utob); +// +const _encode = _hasBuffer + ? (s) => Buffer.from(s, 'utf8').toString('base64') + : _TE + ? (s) => _fromUint8Array(_TE.encode(s)) + : (s) => _btoa(utob(s)); +/** + * converts a UTF-8-encoded string to a Base64 string. + * @param {boolean} [urlsafe] if `true` make the result URL-safe + * @returns {string} Base64 string + */ +const encode = (src, urlsafe = false) => urlsafe + ? _mkUriSafe(_encode(src)) + : _encode(src); +/** + * converts a UTF-8-encoded string to URL-safe Base64 RFC4648 §5. + * @returns {string} Base64 string + */ +const encodeURI = (src) => encode(src, true); +// This trick is found broken https://github.com/dankogai/js-base64/issues/130 +// const btou = (src: string) => decodeURIComponent(escape(src)); +// reverting good old fationed regexp +const re_btou = /[\xC0-\xDF][\x80-\xBF]|[\xE0-\xEF][\x80-\xBF]{2}|[\xF0-\xF7][\x80-\xBF]{3}/g; +const cb_btou = (cccc) => { + switch (cccc.length) { + case 4: + var cp = ((0x07 & cccc.charCodeAt(0)) << 18) + | ((0x3f & cccc.charCodeAt(1)) << 12) + | ((0x3f & cccc.charCodeAt(2)) << 6) + | (0x3f & cccc.charCodeAt(3)), offset = cp - 0x10000; + return (_fromCC((offset >>> 10) + 0xD800) + + _fromCC((offset & 0x3FF) + 0xDC00)); + case 3: + return _fromCC(((0x0f & cccc.charCodeAt(0)) << 12) + | ((0x3f & cccc.charCodeAt(1)) << 6) + | (0x3f & cccc.charCodeAt(2))); + default: + return _fromCC(((0x1f & cccc.charCodeAt(0)) << 6) + | (0x3f & cccc.charCodeAt(1))); + } +}; +/** + * @deprecated should have been internal use only. + * @param {string} src UTF-16 string + * @returns {string} UTF-8 string + */ +const btou = (b) => b.replace(re_btou, cb_btou); +/** + * polyfill version of `atob` + */ +const atobPolyfill = (asc) => { + // console.log('polyfilled'); + asc = asc.replace(/\s+/g, ''); + if (!b64re.test(asc)) + throw new TypeError('malformed base64.'); + asc += '=='.slice(2 - (asc.length & 3)); + let u24, bin = '', r1, r2; + for (let i = 0; i < asc.length;) { + u24 = b64tab[asc.charAt(i++)] << 18 + | b64tab[asc.charAt(i++)] << 12 + | (r1 = b64tab[asc.charAt(i++)]) << 6 + | (r2 = b64tab[asc.charAt(i++)]); + bin += r1 === 64 ? _fromCC(u24 >> 16 & 255) + : r2 === 64 ? _fromCC(u24 >> 16 & 255, u24 >> 8 & 255) + : _fromCC(u24 >> 16 & 255, u24 >> 8 & 255, u24 & 255); + } + return bin; +}; +/** + * does what `window.atob` of web browsers do. + * @param {String} asc Base64-encoded string + * @returns {string} binary string + */ +const _atob = _hasatob ? (asc) => atob(_tidyB64(asc)) + : _hasBuffer ? (asc) => Buffer.from(asc, 'base64').toString('binary') + : atobPolyfill; +// +const _toUint8Array = _hasBuffer + ? (a) => _U8Afrom(Buffer.from(a, 'base64')) + : (a) => _U8Afrom(_atob(a).split('').map(c => c.charCodeAt(0))); +/** + * converts a Base64 string to a Uint8Array. + */ +const toUint8Array = (a) => _toUint8Array(_unURI(a)); +// +const _decode = _hasBuffer + ? (a) => Buffer.from(a, 'base64').toString('utf8') + : _TD + ? (a) => _TD.decode(_toUint8Array(a)) + : (a) => btou(_atob(a)); +const _unURI = (a) => _tidyB64(a.replace(/[-_]/g, (m0) => m0 == '-' ? '+' : '/')); +/** + * converts a Base64 string to a UTF-8 string. + * @param {String} src Base64 string. Both normal and URL-safe are supported + * @returns {string} UTF-8 string + */ +const decode = (src) => _decode(_unURI(src)); +/** + * check if a value is a valid Base64 string + * @param {String} src a value to check + */ +const isValid = (src) => { + if (typeof src !== 'string') + return false; + const s = src.replace(/\s+/g, '').replace(/={0,2}$/, ''); + return !/[^\s0-9a-zA-Z\+/]/.test(s) || !/[^\s0-9a-zA-Z\-_]/.test(s); +}; +// +const _noEnum = (v) => { + return { + value: v, enumerable: false, writable: true, configurable: true + }; +}; +/** + * extend String.prototype with relevant methods + */ +const extendString = function () { + const _add = (name, body) => Object.defineProperty(String.prototype, name, _noEnum(body)); + _add('fromBase64', function () { return decode(this); }); + _add('toBase64', function (urlsafe) { return encode(this, urlsafe); }); + _add('toBase64URI', function () { return encode(this, true); }); + _add('toBase64URL', function () { return encode(this, true); }); + _add('toUint8Array', function () { return toUint8Array(this); }); +}; +/** + * extend Uint8Array.prototype with relevant methods + */ +const extendUint8Array = function () { + const _add = (name, body) => Object.defineProperty(Uint8Array.prototype, name, _noEnum(body)); + _add('toBase64', function (urlsafe) { return fromUint8Array(this, urlsafe); }); + _add('toBase64URI', function () { return fromUint8Array(this, true); }); + _add('toBase64URL', function () { return fromUint8Array(this, true); }); +}; +/** + * extend Builtin prototypes with relevant methods + */ +const extendBuiltins = () => { + extendString(); + extendUint8Array(); +}; +const gBase64 = { + version: version, + VERSION: VERSION, + atob: _atob, + atobPolyfill: atobPolyfill, + btoa: _btoa, + btoaPolyfill: btoaPolyfill, + fromBase64: decode, + toBase64: encode, + encode: encode, + encodeURI: encodeURI, + encodeURL: encodeURI, + utob: utob, + btou: btou, + decode: decode, + isValid: isValid, + fromUint8Array: fromUint8Array, + toUint8Array: toUint8Array, + extendString: extendString, + extendUint8Array: extendUint8Array, + extendBuiltins: extendBuiltins, +}; +// makecjs:CUT // +export { version }; +export { VERSION }; +export { _atob as atob }; +export { atobPolyfill }; +export { _btoa as btoa }; +export { btoaPolyfill }; +export { decode as fromBase64 }; +export { encode as toBase64 }; +export { utob }; +export { encode }; +export { encodeURI }; +export { encodeURI as encodeURL }; +export { btou }; +export { decode }; +export { isValid }; +export { fromUint8Array }; +export { toUint8Array }; +export { extendString }; +export { extendUint8Array }; +export { extendBuiltins }; +// and finally, +export { gBase64 as Base64 }; From 8fbfb765fa378d3911083f8b9218749defeac4f5 Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 28 May 2023 16:15:13 +0800 Subject: [PATCH 229/357] More_Mouse.js 2.7 (#83) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * More_Mouse.js 2.7 feat(block):增加判断鼠标点击,双击 fix(compile):修复ico转base64概率报错 fix(compile):修复鼠标设置ico图片时错误base64循环报错的BUG feat(block):新增按下几秒系列积木(boolean,reporter,hat) feat(block):新增鼠标滚轮积木 * More_Mouse.js 2.7 fix(block):修复上传文件转ico后无法设置为鼠标样式的BUG * More_Mouse.js 2.7 修复 fix(commit):修复切换语言后拓展失效的BUG --- wit_cat/More_Mouse.js | 858 ++++++++++++++++++++++++++++++++---------- 1 file changed, 668 insertions(+), 190 deletions(-) diff --git a/wit_cat/More_Mouse.js b/wit_cat/More_Mouse.js index 36c954c7..7eb59fc3 100644 --- a/wit_cat/More_Mouse.js +++ b/wit_cat/More_Mouse.js @@ -1,61 +1,85 @@ -// import Cast from "cast.js"; - -const _picture = ""; - -const _icon = ""; - -const extensionId = "WitCatMouse"; -let button = ["up", "up", "up", "up", "up"]; -let xMouse = 0; -let yMouse = 0; -let timer = null; -let touch = []; - - -//base64转blob -function base64ImgtoFile(dataurl, filename = 'file') { - const arr = dataurl.split(',') - const mime = arr[0].match(/:(.*?);/)[1] - const suffix = mime.split('/')[1] - const bstr = atob(arr[1]) - let n = bstr.length - const u8arr = new Uint8Array(n) - while (n--) { - u8arr[n] = bstr.charCodeAt(n) - } - return new File([u8arr], `${filename}.${suffix}`, { - type: mime - }) -} +const witcat_more_mouse_picture = ""; -//检测是不是ico的base64 -function isBase64(str) { - let a = "^([A-Za-z0-9+/]{4})*([A-Za-z0-9+/]{4}|[A-Za-z0-9+/]{3}=|[A-Za-z0-9+/]{2}==)$"; - if (str.match(a) == null) { - return true; - } - else { - console.warn("请使用ico格式的base64文本\nPlease use base64 text in ico format"); - return false; - } -} +const witcat_more_mouse_icon = ""; -//找渲染cvs -let cvs = document.getElementsByTagName("canvas")[0]; -if (cvs === null) { - alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); -} -else { - for (let i = 1; cvs.className !== "" && i <= document.getElementsByTagName("canvas").length; i++) { - cvs = document.getElementsByTagName("canvas")[i]; - } - if (cvs === null) { - alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); - } -} +const witcat_more_mouse_extensionId = "WitCatMouse"; + +/** @typedef {string|number} SCarg 来自Scratch圆形框的参数,虽然这个框可能只能输入数字,但是可以放入变量,因此有可能获得数字和文本,需要同时处理 */ class WitCatMouse { constructor(runtime) { + /** + * 鼠标键是否按下 + * @type {("up"|"down")[]} + */ + this.button = ["up", "up", "up", "up", "up"]; + + /** + * 鼠标x移动速度 + */ + this.xMouse = 0; + + /** + * 鼠标y移动速度 + */ + this.yMouse = 0; + + /** + * 鼠标滚轮速度 + */ + this.MouseWheel = 0; + + /** + * 鼠标速度重置计时器 + * @type {null|number} + */ + this.timer = null; + + /** + * 手指列表 + * @type {{identifier: number|"mouse", clientX: number, clientY: number}[]} + */ + this.touch = []; + + /** + * 点击事件计时器 + * @type {false|number} + */ + this.click = false; + + /** + * 双击事件计时器 + * @type {false|number} + */ + this.dclick = false; + + /** + * 鼠标按下计时器(记录按下鼠标的时刻) + * @type {(""|number)[]} + */ + this.mousetdlist = ["", "", "", "", ""]; + + /** + * Scratch 所使用的 canvas,获取不到返回 null + * @return {HTMLCanvasElement | null} + */ + this.canvas = () => { + try { + const canvas = runtime.renderer.canvas; + if (canvas instanceof HTMLCanvasElement) { + return canvas; + } + } catch (err) { + return null; + } + }; + + if (this.canvas() === null) { + alert("当前页面不支持多指触控/全屏,请前往作品详情页体验完整作品!"); + // 注意:在提示之后,扩展仍然在运行。需要在后面引用 Canvas 的部分进行判断。 + } + this._addevent(); + this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { @@ -74,7 +98,7 @@ class WitCatMouse { "WitCatMouse.mouseuse": "[mouseuse]鼠标", "WitCatMouse.mouseuse.1": "锁定", "WitCatMouse.mouseuse.2": "释放", - "WitCatMouse.acceleration": "鼠标[way]加速度", + "WitCatMouse.acceleration": "鼠标[way]速度",// 速度和加速度的区别 "WitCatMouse.way.1": "X", "WitCatMouse.way.2": "Y", "WitCatTouch.down": "按下的手指数量", @@ -93,42 +117,49 @@ class WitCatMouse { "WitCatMouse.IsMobile": "移动设备?", "WitCatMouse.cursor": "更改鼠标的样式为[cursor]", "WitCatMouse.cursor.1": "默认", - "WitCatMouse.cursor.2": "光标", + "WitCatMouse.cursor.2": "文本", "WitCatMouse.cursor.3": "十字移动", "WitCatMouse.cursor.4": "上下移动", "WitCatMouse.cursor.5": "左右移动", "WitCatMouse.cursor.6": "禁止", - "WitCatMouse.cursor.7": "选择", + "WitCatMouse.cursor.7": "手指", "WitCatMouse.cursor.8": "加载", - "WitCatMouse.cursor.9": "缓慢加载", + "WitCatMouse.cursor.9": "等待", "WitCatMouse.cursor.10": "帮助", - "WitCatMouse.cursor.11": "横向光标", - "WitCatMouse.cursor.12": "精准选择", - "WitCatMouse.cursorurl": "更改鼠标的样式为X[x]Y[y]base64[text]", - "WitCatMouse.url": "上传ico并获得base64", + "WitCatMouse.cursor.11": "竖向文本", + "WitCatMouse.cursor.12": "十字准星", + "WitCatMouse.cursorurl": "更改鼠标的样式为X[x]Y[y]data url[text]", + "WitCatMouse.url": "上传ico并获得data url", + "WitCatMouse.click": "点击", + "WitCatMouse.dclick": "双击", + "WitCatMouse.mouse": "鼠标被[way]?", + "WitCatMouse.mousetd": "当鼠标[key]按住[time]秒", + "WitCatMouse.mouset": "鼠标[key]按下时长(秒)", + "WitCatMouse.docs": "📖拓展教程", + "WitCatMouse.mousewheel": "鼠标滚轮速度", }, en: { "WitCatMouse.name": "[beta]WitCat’s Mouse", "WitCatMouse.name.1": "more mouse", "WitCatMouse.name.2": "more touch", - "WitCatMouse.set": "[set]Right-click menu", - "WitCatMouse.set.1": "allow", - "WitCatMouse.set.2": "forbid", + "WitCatMouse.set": "[set]right-click menu", + "WitCatMouse.set.1": "Enable", + "WitCatMouse.set.2": "Disable", "WitCatMouse.when": "mouse[key]down?", - "WitCatMouse.key.1": "left", - "WitCatMouse.key.2": "middle", - "WitCatMouse.key.3": "right", - "WitCatMouse.key.4": "front", - "WitCatMouse.key.5": "back", + "WitCatMouse.key.1": "left button", + "WitCatMouse.key.2": "middle button", + "WitCatMouse.key.3": "right button", + "WitCatMouse.key.4": "first side button", + "WitCatMouse.key.5": "second side button", "WitCatMouse.mouseuse": "[mouseuse]mouse", "WitCatMouse.mouseuse.1": "Lock", "WitCatMouse.mouseuse.2": "Release", - "WitCatMouse.acceleration": "mouse[way]acceleration", + "WitCatMouse.acceleration": "mouse[way]speed", "WitCatMouse.way.1": "X", "WitCatMouse.way.2": "Y", "WitCatTouch.name": "Touch", - "WitCatTouch.down": "Fingers Number", - "WitCatTouch.num": "the[num]finger`s[type]", + "WitCatTouch.down": "Finger count", + "WitCatTouch.num": "[type]of finger[num]", "WitCatTouch.type.1": "X", "WitCatTouch.type.2": "Y", "WitCatTouch.type.3": "ID", @@ -138,28 +169,40 @@ class WitCatMouse { "WitCatMouse.setfill": "⚠️(danger)Set resolution height to[num]", "WitCatMouse.resolution": "Current high resolution", "WitCatMouse.cantouch": "Device support[type]?", - "WitCatMouse.types.1": "Touch screen", + "WitCatMouse.types.1": "touch screen", "WitCatMouse.types.2": "mouse", - "WitCatMouse.IsMobile": "Mobile devices?", - "WitCatMouse.cursor": "Change mouse style to[cursor]", - "WitCatMouse.cursor.1": "Default", - "WitCatMouse.cursor.2": "Cursor", - "WitCatMouse.cursor.3": "Cross Move", - "WitCatMouse.cursor.4": "Move up and down", - "WitCatMouse.cursor.5": "Move left and right", - "WitCatMouse.cursor.6": "Prohibit", - "WitCatMouse.cursor.7": "Select", - "WitCatMouse.cursor.8": "Load", - "WitCatMouse.cursor.9": "Slow loading", - "WitCatMouse.cursor.10": "Help", - "WitCatMouse.cursor.11": "Landscape cursor", - "WitCatMouse.cursor.12": "Precise selection", + "WitCatMouse.IsMobile": "Mobile device?", + "WitCatMouse.cursor": "Change cursor style to[cursor]", + "WitCatMouse.cursor.1": "default", + "WitCatMouse.cursor.2": "text", + "WitCatMouse.cursor.3": "cross move", + "WitCatMouse.cursor.4": "move up and down", + "WitCatMouse.cursor.5": "move left and right", + "WitCatMouse.cursor.6": "not allowed", + "WitCatMouse.cursor.7": "pointer", + "WitCatMouse.cursor.8": "progress", + "WitCatMouse.cursor.9": "wait", + "WitCatMouse.cursor.10": "help", + "WitCatMouse.cursor.11": "vertical text", + "WitCatMouse.cursor.12": "crosshair", "WitCatMouse.cursorurl": "Change the style of the mouse to X[x]Y[y]base64[text]", - "WitCatMouse.url": "Upload ico and get base64", + "WitCatMouse.url": "upload ico and get base64", + "WitCatMouse.click": "click", + "WitCatMouse.dclick": "double-click", + "WitCatMouse.mouse": "mouse[way]?", + "WitCatMouse.mousetd": "When mouse[key]pressed[time]seconds", + "WitCatMouse.mouset": "mouse[key]hold time(sec)", + "WitCatMouse.docs": "📖Tutorials", + "WitCatInput.mousewheel": "mouse wheel speed", } }) } + /** + * 翻译 + * @param {string} id + * @returns {string} + */ formatMessage(id) { return this._formatMessage({ id, @@ -170,14 +213,18 @@ class WitCatMouse { getInfo() { return { - id: extensionId, // 拓展id + id: witcat_more_mouse_extensionId, // 拓展id name: this.formatMessage("WitCatMouse.name"), // 拓展名 - docsURI: "https://www.ccw.site/post/c36aa805-b29d-48da-aba1-468a6cf80bfa", - blockIconURI: _icon, - menuIconURI: _icon, + blockIconURI: witcat_more_mouse_icon, + menuIconURI: witcat_more_mouse_icon, color1: "#8eace1", color2: "#86a2d4", blocks: [ + { + blockType: "button", + text: this.formatMessage('WitCatMouse.docs'), + onClick: this.docs, + }, { opcode: 'setfill', blockType: "command", @@ -231,10 +278,80 @@ class WitCatMouse { }, }, }, + { + opcode: "mouses", + blockType: "Boolean", + text: this.formatMessage("WitCatMouse.mouse"), + arguments: { + way: { + type: "string", + menu: "ways", + }, + }, + }, + { + opcode: "mousewheel", + blockType: "reporter", + text: this.formatMessage("WitCatMouse.mousewheel"), + arguments: {}, + }, + { + opcode: "mouse", + blockType: "hat", + text: this.formatMessage("WitCatMouse.mouse"), + arguments: { + way: { + type: "string", + menu: "ways", + }, + }, + }, + { + opcode: "mousetd", + blockType: "Boolean", + text: this.formatMessage("WitCatMouse.mousetd"), + arguments: { + key: { + type: "number", + menu: "key", + }, + time: { + type: "number", + defaultValue: "1", + }, + }, + }, + { + opcode: "mousetds", + blockType: "hat", + text: this.formatMessage("WitCatMouse.mousetd"), + arguments: { + key: { + type: "number", + menu: "key", + }, + time: { + type: "number", + defaultValue: "1", + }, + }, + }, + { + opcode: "mouset", + blockType: "reporter", + text: this.formatMessage("WitCatMouse.mouset"), + arguments: { + key: { + type: "number", + menu: "key", + }, + }, + }, { blockType: "button", text: this.formatMessage('WitCatMouse.url'), - onClick: this.url, + // 这里的 bind 必须,因为 this.url 里面引用了 this + onClick: this.url.bind(this), }, { opcode: "cursor", @@ -424,6 +541,16 @@ class WitCatMouse { value: "y" }, ], + ways: [ + { + text: this.formatMessage('WitCatMouse.click'), + value: "click" + }, + { + text: this.formatMessage('WitCatMouse.dclick'), + value: "dclick" + }, + ], type: [ { text: this.formatMessage('WitCatTouch.type.1'), @@ -461,103 +588,266 @@ class WitCatMouse { } }; } - //右键菜单 + + /** + * 启用右键菜单? + * @param {object} args + * @param {SCarg} args.set + */ set(args) { - history.pushState(null, null, null); - cvs.parentNode.oncontextmenu = () => { - if (args.set === "true") { - return true; - } - else { - return false; - } + if (this.canvas() === null) { + return; + } + // 在把自己的方法设为给其他事件/函数的回调时加上 bind(this) 是很有必要的, + // 确保回调函数触发时,里面的 this 指向自己。 + // 不过回调函数里面没有用到 this,所以是否有这个 bind 没有区别。 + // 如果加上了 bind,因为每次 bind() 都会生成一个新函数, + // removeEventListener 的时候就会因为函数不一致导致 remove 不掉, + // 需要提前把 bind 过的函数设为某类内变量。 + if (args.set === "false") { + this.canvas().addEventListener("contextmenu", this._nocontextmenu); + } else { + this.canvas().removeEventListener("contextmenu", this._nocontextmenu); } } - //按下判断 + + /** + * 禁用右键菜单用的事件捕获函数 + * @param {Event} event + */ + _nocontextmenu(event) { + // 阻止右键菜单 + event.preventDefault(); + } + + /** + * 按下判断 + * @param {object} args + * @param {SCarg} args.key + */ when(args) { - if (button[args.key] === "down") { - return true; - } - else { - return false; - } + return this.button[Number(args.key)] === "down"; } - //控制鼠标 + + /** + * 控制鼠标 + * @param {object} args + * @param {SCarg} args.mouseuse + */ mouseuse(args) { if (args.mouseuse === "release") { document.exitPointerLock(); } else { - cvs.parentNode.requestPointerLock(); + document.body.requestPointerLock(); } } - //鼠标移动量 + + /** + * 鼠标移动量 + * @param {object} args + * @param {SCarg} args.way + * @returns {number} + */ acceleration(args) { if (args.way === "x") { - return xMouse; + return this.xMouse; } else { - return -yMouse; + return -this.yMouse; } } - //数量 + + /** + * 手指数量 + * @returns {number} + */ down() { - return touch.length; + return this.touch.length; } - //坐标 + + /** + * 坐标 + * @param {object} args + * @param {SCarg} args.num 手指编号 + * @param {SCarg} args.type 数据类型 "x"|"y"|"identifier" + * @returns {number|string} + */ num(args) { - if (args.num > 0 && args.num <= touch.length) { + if (this.canvas() === null) { + return 0; + } + const touch1 = this.touch[Number(args.num) - 1]; + if (touch1 !== undefined) { if (args.type === "x") { - return this.runtime.stageWidth * ((touch[args.num - 1].clientX - cvs.getBoundingClientRect().left) / cvs.offsetWidth); + return this.runtime.stageWidth * ((touch1.clientX - this.canvas().getBoundingClientRect().left) / this.canvas().offsetWidth); } else if (args.type === "y") { - return this.runtime.stageHeight * ((touch[args.num - 1].clientY - cvs.getBoundingClientRect().top) / cvs.offsetHeight); + return this.runtime.stageHeight * ((touch1.clientY - this.canvas().getBoundingClientRect().top) / this.canvas().offsetHeight); } else { - return touch[args.num - 1].identifier; + return touch1.identifier; } } else { - return null; + return 0; } } - //全屏 + + /** + * 全屏 + * @deprecated + */ fill() { console.warn("全屏因浏览器兼容问题已下线,在未来修复后将会重新上线\nFull screen has been taken offline due to browser compatibility issues. It will be back online after a future fix"); } - //设置分辨率 + + /** + * 设置分辨率 + * @deprecated + */ setfill() { console.warn("全屏因浏览器兼容问题已下线,在未来修复后将会重新上线\nFull screen has been taken offline due to browser compatibility issues. It will be back online after a future fix"); } - //当前分辨率 + + /** + * 当前分辨率 + * @returns {number} + */ resolution() { - return cvs.height; + if (this.canvas() === null) { + return 0; + } + return this.canvas().height; } - //设备是否支持触屏 + + /** + * 设备是否支持触屏/鼠标 + * @param {object} args + * @param {SCarg} args.type + * @returns {boolean} + */ cantouch(args) { return (args.type in document.documentElement); } - //是否是手机 + + /** + * 是否是手机 + * @returns {boolean} + */ IsMobile() { return /Android|iPhone|iPad|iPod|BlackBerry|webOS|Windows Phone|SymbianOS|IEMobile|Opera Mini/i.test(navigator.userAgent); } - //设置光标 + + /** + * 设置光标样式 + * @param {object} args + * @param {SCarg} args.cursor 样式 + */ cursor(args) { - cvs.parentNode.parentNode.parentNode.style.cursor = args.cursor; + if (this.canvas() === null) { + return; + } + this.canvas().parentNode.parentNode.parentNode.style.cursor = String(args.cursor); } - //设置光标为url + + /** + * 设置光标为url + * @param {object} args + * @param {SCarg} args.text 样式 + * @param {SCarg} args.x x偏移 + * @param {SCarg} args.y y偏移 + */ cursorurl(args) { - if (isBase64(args.text)) { - const img = args.text; - let file = base64ImgtoFile(img); // 得到File对象 - let imgUrl = window.webkitURL.createObjectURL(file) || window.URL.createObjectURL(file) // imgUrl图片网络路径 - cvs.parentNode.parentNode.parentNode.style.cursor = "url(" + imgUrl + ")" + args.x + " " + args.y + ",auto"; + if (this.canvas() === null) { return; } + let url = String(args.text); + const x = Number(args.x); + const y = Number(args.y); + // 针对 url() 里的语法,先转义回去,再完整地转义回来。 + url = this.base64ToBlob(url); + // 实际上 cursorurl 处可以直接使用 正常的 url 和 data url。 + // 不需要特地转换。 + this.canvas().parentNode.parentNode.parentNode.style.cursor = `url("${url}") ${x} ${y}, auto`; } - //打开ico文件 - url() { - return new Promise(resolve => { + + /** + * 打开文件选择框 + * @param {string} accept 接受的文件扩展名 + * @param {boolean} multiple 接受多个文件 + * @return {Promise} [异步地]返回选择后的文件列表input.files转换成的数组(可能没有文件) + */ + _inputfileclick(accept, multiple) { + return new Promise((resolve, reject) => { + const input = document.createElement("input"); + input.type = "file"; + input.accept = accept; + input.style.display = "none"; + input.multiple = multiple; + input.click(); + input.addEventListener("change", () => { + if (input.files === null) { + reject(new Error("不应该看到这个")); + } else { + // 返回了关键的 input.files,而不是整个 input。 + // 之后如果要考虑“读取素材库文件”,“拖动导入文件”等 + // 只能获得 Blob/File 的情况,可以方便适配 + // 这里加 Array.from 是因为 input.files 是 FileList, + // 不是 File[],一些数组拥有的功能它没有。虽然一般情况下 + // 不会注意到区别,但是类型检查会把这种情况查出来。 + resolve(Array.from(input.files)); + } + }, { once: true }); // 只触发一次 + window.addEventListener("focus", () => { + setTimeout(() => { + if (input.files === null) { + reject(new Error("不应该看到这个")); + } else { + resolve(Array.from(input.files)); + } + }, 1000); + }, { once: true }); // 只触发一次 + }); + } + + /** + * 读取文件 + * @param {File|Blob} file File 或者 Blob + * @param {"arraybuffer"|"dataurl"|"text"} mode 读取模式 + * @return {Promise} [异步地]返回读取后的内容 + */ + _readerasync(file, mode) { + return new Promise((resolve, reject) => { + const reader = new FileReader(); + reader.onload = () => { + resolve(reader.result); + }; + reader.onerror = (e) => { + reject(e); + }; + switch (mode) { + case "arraybuffer": + reader.readAsArrayBuffer(file); + break; + case "dataurl": + reader.readAsDataURL(file); + break; + case "text": + reader.readAsText(file); + break; + default: + reject(new Error("mode 错误: 应该是 arraybuffer, dataurl 或者 text")); + return; + } + }); + } + + /** + * 打开ico文件 + */ + async url() { + new Promise(resolve => { const input = document.createElement("input"); input.type = "file"; input.style = "display:none;"; @@ -568,8 +858,7 @@ class WitCatMouse { const readers = new FileReader(); const file = input.files[0]; reader.onload = (e) => { - navigator.clipboard.writeText(e.currentTarget.result); - alert("base64代码已经被复制到剪切板,可以粘贴以使用\nThe base64 code has been copied to the clipboard and can be pasted for use"); + prompt("请复制以下代码:", e.currentTarget.result); resolve(e.target.result); }; reader.onerror = () => { @@ -603,6 +892,241 @@ class WitCatMouse { } }); } + + /** + * 打开教程 + */ + docs() { + let a = document.createElement('a'); + a.href = "https://www.ccw.site/post/c36aa805-b29d-48da-aba1-468a6cf80bfa"; + a.rel = "noopener noreferrer"; + a.target = "_blank"; + a.click(); + } + + /** + * 鼠标点击/双击 + * @param {object} args + * @param {SCarg} args.way 点击/双击 + * @returns {boolean} + */ + mouse(args) { + if (args.way === "click") { + return this.click !== false; + } + if (args.way === "dclick") { + return this.dclick !== false; + } + return false; + } + /** + * 鼠标滚轮速度 + * @returns {number} + */ + mousewheel() { + clearTimeout(this.timer); + let a = this.MouseWheel; + this.MouseWheel = 0; + return a; + } + + /** + * 鼠标点击/双击(帽子积木) + * @param {object} args + * @param {SCarg} args.way 点击/双击 + * @returns {boolean} + */ + mouses(args) { + return this.mouse(args); + } + + /** + * 判断鼠标键按下时长 + * @param {object} args + * @param {SCarg} args.key 按键编号 + * @param {SCarg} args.time 按键时长 + * @returns {boolean} + */ + mousetd(args) { + const mousetdkey = this.mousetdlist[Number(args.key)]; + if (mousetdkey !== undefined && mousetdkey != "") { + let time = Date.now() - (Number(args.time) * 1000 + mousetdkey); + if (-50 <= time && time <= 50) { + return true; + } + } + return false; + } + + /** + * 判断鼠标键按下时长(帽子积木) + * @param {object} args + * @param {SCarg} args.key 按键编号 + * @param {SCarg} args.time 按键时长 + * @returns {boolean} + */ + mousetds(args) { + return this.mousetd(args); + } + + /** + * 鼠标被按下的时间 + * @param {object} args + * @param {SCarg} args.key 按键编号 + * @returns {number} + */ + mouset(args) { + const mousetdkey = this.mousetdlist[Number(args.key)]; + if (mousetdkey !== undefined && mousetdkey != "") { + return (Date.now() - mousetdkey) / 1000; + } + return 0; + } + + /** + * 复制触摸点数 + * @param {TouchList} touches + */ + _copytouch(touches) { + this.touch = Array.from(touches).map((touch) => { + return { + clientX: touch.clientX, + clientY: touch.clientY, + identifier: touch.identifier + } + }); + } + + /** 添加事件触发器 */ + _addevent() { + if (this.canvas() === null) { + return; + } + //鼠标 + this.canvas().addEventListener('mousedown', e => { + this.button[e.button] = "down"; + this.mousetdlist[e.button] = Date.now(); + if (this.button[0] === "down") { + this.touch = [{ + clientX: e.clientX, + clientY: e.clientY, + identifier: "mouse" + }]; + } + }) + document.addEventListener('mouseup', e => { + this.button[e.button] = "up"; + this.mousetdlist[e.button] = ""; + this.touch = []; + }) + document.addEventListener("mousemove", ev => { + if (this.button[0] === "down") { + this.touch = [{ + clientX: ev.clientX, + clientY: ev.clientY, + identifier: "mouse" + }]; + } + else { + this.touch = []; + } + this.xMouse = ev.movementX; // 获得鼠标指针的x移动量 + this.yMouse = ev.movementY; // 获得鼠标指针的y移动量 + if (this.timer !== null) { + clearTimeout(this.timer); + } + this.timer = setTimeout(() => { + this.xMouse = 0; + this.yMouse = 0; + }, 30); + }); + //多指触控 + this.canvas().addEventListener('touchstart', e => { + // e.targetTouches 会随着时间改变,必须复制一份。 + this._copytouch(e.targetTouches); + this.button[0] = "down"; + this.mousetdlist[0] = Date.now(); + }) + this.canvas().addEventListener('touchmove', e => { + if (e.targetTouches[0] !== undefined && this.touch[0] !== undefined) { + this.xMouse = e.targetTouches[0].clientX - this.touch[0].clientX; // 获得手指的x移动量 + this.yMouse = e.targetTouches[0].clientY - this.touch[0].clientY; // 获得手指的y移动量 + } + if (this.timer !== null) { + clearTimeout(this.timer); + } + this.timer = setTimeout(() => { + this.xMouse = 0; + this.yMouse = 0; + }, 30); + // e.targetTouches 会随着时间改变,必须复制一份。 + this._copytouch(e.targetTouches); + }) + this.canvas().addEventListener('touchend', e => { + // e.targetTouches 会随着时间改变,必须复制一份。 + this._copytouch(e.targetTouches); + this.mousetdlist[0] = ""; + this.button[0] = "up"; + }) + this.canvas().addEventListener('click', () => { + if (this.click !== false) { + clearTimeout(this.click); + } + this.click = setTimeout(() => { + this.click = false; + }, 50); + }); + this.canvas().addEventListener('dblclick', () => { + if (this.dclick !== false) { + clearTimeout(this.dclick); + } + this.dclick = setTimeout(() => { + this.dclick = false; + }, 50); + }); + //给页面绑定滑轮滚动事件 + this.canvas().addEventListener('wheel', (e) => { + // 注意这个负数…… + // 目前的标准用法是使用 deltaY,但是 deltaY 的符号和 WheelDeltaY 相反。 + // 为了和原有的行为一致,乘上 -3 + // 在我的浏览器中 deltaY = WheelDeltaY / -3 + this.MouseWheel = e.WheelDelta; + clearTimeout(this.timer); + this.timer = setTimeout(() => { + this.MouseWheel = 0; + }, 30); + }, { capture: true }); + } + /** + * base64转blob + * @param {string} base64 传入base64 + * @returns {string} + */ + base64ToBlob(base64) { + try { + let audioSrc = base64; // 拼接最终的base64 + + let arr = audioSrc.split(','); + let array = arr[0].match(/:(.*?);/); + let mime = (array && array.length > 1 ? array[1] : type) || type; + // 去掉url的头,并转化为byte + let bytes = window.atob(arr[1]); + // 处理异常,将ascii码小于0的转换为大于0 + let ab = new ArrayBuffer(bytes.length); + // 生成视图(直接针对内存):8位无符号整数,长度1个字节 + let ia = new Uint8Array(ab); + for (let i = 0; i < bytes.length; i++) { + ia[i] = bytes.charCodeAt(i); + } + return URL.createObjectURL(new Blob([ab], { + type: mime + })) + } + catch { + return undefined; + } + } + } window.tempExt = { @@ -610,9 +1134,9 @@ window.tempExt = { info: { name: "WitCatMouse.name", description: "WitCatMouse.descp", - extensionId: extensionId, - iconURL: _picture, - insetIconURL: _icon, + extensionId: witcat_more_mouse_extensionId, + iconURL: witcat_more_mouse_picture, + insetIconURL: witcat_more_mouse_icon, featured: true, disabled: false, collaborator: "白猫 @ CCW" @@ -629,49 +1153,3 @@ window.tempExt = { } }; -/* vim: set expandtab tabstop=2 shiftwidth=2: */ -//鼠标 -document.addEventListener('mousedown', e => { - button[e.button] = "down"; - if (button[0] === "down") { - touch = JSON.parse("[{\"clientX\":\"" + e.clientX + "\",\"clientY\":\"" + e.clientY + "\",\"identifier\":\"mouse\"}]"); - } -}) -document.addEventListener('mouseup', e => { - button[e.button] = "up"; - touch = []; -}) -document.addEventListener("mousemove", ev => { - if (button[0] === "down") { - touch = JSON.parse("[{\"clientX\":\"" + ev.clientX + "\",\"clientY\":\"" + ev.clientY + "\",\"identifier\":\"mouse\"}]"); - } - else { - touch = []; - } - xMouse = ev.movementX; // 获得鼠标指针的x移动量 - yMouse = ev.movementY; // 获得鼠标指针的y移动量 - clearTimeout(timer); - timer = setTimeout(() => { - xMouse = 0; - yMouse = 0; - }, 30); -}); -//多指触控 -cvs.addEventListener('touchstart', e => { - touch = e.targetTouches; - button[0] = "down"; -}) -cvs.addEventListener('touchmove', e => { - xMouse = e.targetTouches[0].clientX - touch[0].clientX; // 获得手指的x移动量 - yMouse = e.targetTouches[0].clientY - touch[0].clientY; // 获得手指的y移动量 - clearTimeout(timer); - timer = setTimeout(() => { - xMouse = 0; - yMouse = 0; - }, 30); - touch = e.targetTouches; -}) -cvs.addEventListener('touchend', e => { - touch = e.targetTouches; - button[0] = "up"; -}) From 22b0066f74a159e8891d4340a46df66b9b3b4fcc Mon Sep 17 00:00:00 2001 From: wit_cat <96361142+little-starts@users.noreply.github.com> Date: Sun, 28 May 2023 16:15:55 +0800 Subject: [PATCH 230/357] Input.js 2.7 (#82) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Input.js 2.7 重新提交pr feat(block):设置,获取背景图片 fix(compile):修复通过颜色修改css的BUG fix(compile):修复通过数字进行css自定义的BUG feat(block):设置文本框为密码框或者输入框 feat(block):设置文本框的css feat(block):从网络获取字体文件并加载为文本框字体 feat(block):设置文本框的更多属性(行高,字体,字体粗细等) feat(block):正在按下的按键 * Input.js 2.7 修复 fix(commit):修复切换语言后拓展失效的BUG --- wit_cat/Input.js | 1424 +++++++++++++++++++++++++++++++++------------- 1 file changed, 1043 insertions(+), 381 deletions(-) diff --git a/wit_cat/Input.js b/wit_cat/Input.js index 4245be60..89fa2c2d 100644 --- a/wit_cat/Input.js +++ b/wit_cat/Input.js @@ -1,45 +1,104 @@ -// import Cast from "cast.js"; +const witcat_input_picture = ""; -const _picture = ""; +const witcat_input_icon = ""; -const _icon = ""; +const witcat_input_extensionId = "WitCatInput"; -const extensionId = "WitCatInput"; - -let keypress = {}; -let lastKey = "", MouseWheel = 0; -let timer; - -//找渲染cvs -let cvs = document.getElementsByTagName("canvas")[0]; -if (cvs === null) { - alert("当前页面不支持输入框,请前往作品详情页体验完整作品!"); -} -else { - for (let i = 1; cvs.className !== "" && i <= document.getElementsByTagName("canvas").length; i++) { - cvs = document.getElementsByTagName("canvas")[i]; - } - if (cvs === null) { - alert("当前页面不支持输入框,请前往作品详情页体验完整作品!"); - } -} +/** @typedef {string|number|boolean} SCarg 来自Scratch圆形框的参数,虽然这个框可能只能输入数字,但是可以放入变量,因此有可能获得数字和文本,需要同时处理 */ class WitCatInput { constructor(runtime) { + /** + * 按下的按键 + * @type {{[key: string]: true}} + */ + this.keypresslist = {}; + + /** + * 最后一次按下的按键 + */ + this.lastKey = ""; + + /** + * 鼠标滚轮速度 + */ + this.MouseWheel = 0; + + /** + * 鼠标速度复位为0的计时器 + * @type {number|undefined} + */ + this.timer = undefined; + + /** + * 保存输入框文本大小,当舞台大小变化时,同比修改输入框文本大小 + * @type {{[key: string]: number}} + */ + this.inputFontSize = {}; + + /** + * 是否开启同比修改输入框文本大小的功能 + */ + this.adaptive = false; + + /** + * 监控舞台大小的变化 + * @type {MutationObserver | null} + */ + this.observer = null; + + /** + * Scratch 所使用的 canvas,获取不到返回 null + * @return {HTMLCanvasElement | null} + */ + this.canvas = () => { + try { + const canvas = runtime.renderer.canvas; + if (canvas instanceof HTMLCanvasElement) { + return canvas; + } + } catch (err) { + return null; + } + }; + + /** + * 所有输入框所在的父角色,目前设为 canvas 的父角色。 + * 获取不到返回 null + * @return {HTMLElement | null} + */ + this.inputParent = () => { + try { + const canvas = runtime.renderer.canvas; + if (canvas instanceof HTMLCanvasElement) { + return canvas.parentElement; + } + } catch (err) { + console.error(err); + return null; + } + }; + + if (this.canvas() === null || this.inputParent() === null) { + alert("当前页面不支持文本框,请前往作品详情页体验完整作品!"); + // 注意:在提示之后,扩展仍然在运行。需要在后面引用 Canvas 的部分进行判断。 + } + this._addevent(); + this.runtime = runtime; this._formatMessage = runtime.getFormatMessage({ "zh-cn": { "WitCatInput.name": "[beta]白猫的输入框", - "WitCatInput.createinput": "ID为[id]的[type]框X[x]Y[y]宽[width]高[height]内容[text]颜色[color]提示[texts]字体大小[size]", - "WitCatInput.deleteinput": "删除ID为[id]的文本框", - "WitCatInput.getinput": "获得ID为[id]的文本框的[type]", - "WitCatInput.isinput": "焦点是否在ID为[id]的文本框上", + "WitCatInput.createinput": "创建或修改[type]文本框并命名为[id],X[x]Y[y]宽[width]高[height]内容[text]颜色[color]提示[texts]字体大小[size]", + "WitCatInput.deleteinput": "删除文本框[id]", + "WitCatInput.getinput": "获得文本框[id]的[type]", + "WitCatInput.isinput": "焦点是否在文本框[id]上", "WitCatInput.whatinput": "焦点位置", - "WitCatInput.nowinput": "将焦点聚焦在ID为[id]的文本框上", + "WitCatInput.nowinput": "将焦点聚焦在文本框[id]上", "WitCatInput.deleteallinput": "删除所有文本框", "WitCatInput.compute": "当前分辨率下高[size]的字体大小", - "WitCatInput.type.1": "单行文本", - "WitCatInput.type.2": "多行文本", + "WitCatInput.type.1": "单行", + "WitCatInput.type.2": "多行", "WitCatInput.number": "第[num]个文本框的[type]", "WitCatInput.numbers": "文本框的数量", "WitCatInput.number.1": "X", @@ -56,29 +115,54 @@ class WitCatInput { "WitCatInput.number.12": "文本高度", "WitCatInput.number.13": "光标位置", "WitCatInput.number.14": "透明度", + "WitCatInput.number.15": "背景图片", + "WitCatInput.number.16": "字体", + "WitCatInput.number.17": "行高", + "WitCatInput.number.18": "字体粗细", + "WitCatInput.number.19": "阴影", + "WitCatInput.fontweight": "设置文本框[id]的字体粗细为[text]", // 在文档中提示用户字体兼容性问题,并推荐合适的字体 + "WitCatInput.normal": "常规", + "WitCatInput.bold": "粗体", + "WitCatInput.thin": "细体", "WitCatInput.key": "按下按键[type]?", - "WitCatInput.keys": "按下按键[type]?", + "WitCatInput.keys": "当按下按键[type]时", + "WitCatInput.keypress": "按下的按键", "WitCatInput.lastkey": "上次按下的键", - "WitCatInput.mousewheel": "鼠标滚轮", - "WitCatInput.setinput": "设置ID为[id]的文本框的[type]为[text]", - "WitCatInput.setread": "设置ID为[id]的文本框为[read]", + "WitCatInput.mousewheel": "鼠标滚轮速度", // 这个积木应该在 More_Mouse 里? + "WitCatInput.setinput": "设置文本框[id]的[type]为[text]", + "WitCatInput.setread": "设置文本框[id]为[read]", // 重复使用三次? + "WitCatInput.setfontfamily": "设置文本框[id]字体从[text]加载字体名[name]", // 分成“加载”和“使用” "WitCatInput.read.1": "可编辑", "WitCatInput.read.2": "不可编辑", + "WitCatInput.password.1": "文本框", + "WitCatInput.password.2": "密码框", + "WitCatInput.textalign.1": "左对齐", + "WitCatInput.textalign.2": "中对齐", + "WitCatInput.textalign.3": "右对齐", + "WitCatInput.docs": "📖拓展教程", + "WitCatInput.input": "输入框", + "WitCatInput.focal": "焦点", + "WitCatInput.keyboard": "键盘", + "WitCatInput.shadow": "阴影x[x]y[y]宽[width]颜色[color]", + "WitCatInput.shadows": "合并阴影[first]和[last]", + "WitCatInput.fontadaptive": "字体大小自适应[type]", + "WitCatInput.set.1": "启用", + "WitCatInput.set.2": "禁用", }, en: { "WitCatInput.name": "[beta]WitCat‘s Input", - "WitCatInput.createinput": "[type]input with ID[id]X[x]Y[y]width[width]height[height]content[text]color[color]prompt[texts]font-size[size]", - "WitCatInput.deleteinput": "delete an input with ID[id]", - "WitCatInput.getinput": "get [type] with ID[id]", - "WitCatInput.isinput": "is the focus on the input with ID[id]?", - "WitCatInput.whatinput": "Focal position", - "WitCatInput.nowinput": "let the focus on the input with ID[id]", - "WitCatInput.deleteallinput": "delete all input", + "WitCatInput.createinput": "Create or modify [type]input ID[id]X[x]Y[y]width[width]height[height]content[text]color[color]prompt[texts]fontSize[size]", + "WitCatInput.deleteinput": "Delete input[id]", + "WitCatInput.getinput": "[type] of input [id]", + "WitCatInput.isinput": "input [id] received focus?", + "WitCatInput.whatinput": "focused input id", + "WitCatInput.nowinput": "Focus on input [id]", + "WitCatInput.deleteallinput": "Delete all inputs", "WitCatInput.compute": "Font size of now screen[size]", - "WitCatInput.type.1": "Single line", - "WitCatInput.type.2": "Multiple lines", - "WitCatInput.number": "[type]of[num]input", - "WitCatInput.numbers": "input number", + "WitCatInput.type.1": "single-line", + "WitCatInput.type.2": "multi-line", + "WitCatInput.number": "[type]of input[num]", + "WitCatInput.numbers": "input count", "WitCatInput.number.1": "X", "WitCatInput.number.2": "Y", "WitCatInput.number.3": "width", @@ -86,25 +170,55 @@ class WitCatInput { "WitCatInput.number.5": "content", "WitCatInput.number.6": "color", "WitCatInput.number.7": "prompt", - "WitCatInput.number.8": "font-size", + "WitCatInput.number.8": "font size", "WitCatInput.number.9": "all(json)", "WitCatInput.number.10": "ID", - "WitCatInput.number.11": "Rolling position", - "WitCatInput.number.12": "Text height", + "WitCatInput.number.11": "scroll position", + "WitCatInput.number.12": "text height", "WitCatInput.number.13": "cursor position ", "WitCatInput.number.14": "transparency", - "WitCatInput.key": "Press the key[type]?", - "WitCatInput.keys": "Press the key [type]?", + "WitCatInput.number.15": "background", + "WitCatInput.number.16": "font family", + "WitCatInput.number.17": "line height", + "WitCatInput.number.18": "font weight", + "WitCatInput.number.19": "shadow", + "WitCatInput.fontweight": "Set fontweight of input[id] to[text]", + "WitCatInput.normal": "normal", + "WitCatInput.bold": "bold", + "WitCatInput.thin": "thin", + "WitCatInput.key": "key[type]pressed?", + "WitCatInput.keys": "When key [type] pressed", + "WitCatInput.keypress": "pressed keys", "WitCatInput.lastkey": "last key pressed", - "WitCatInput.mousewheel": "MouseWheel", - "WitCatInput.setinput": "Set[type]of input whose ID is[id]to[text]", - "WitCatInput.setread": "Set the text box with ID[id]to[read]", + "WitCatInput.mousewheel": "mouse wheel speed", + "WitCatInput.setinput": "Set[type]of input[id]to[text]", + "WitCatInput.setread": "Set input[id]to[read]", + "WitCatInput.setfontfamily": "ID[id]`s font family url[text] name [name]", "WitCatInput.read.1": "editable", "WitCatInput.read.2": "uneditable", + "WitCatInput.password.1": "text", + "WitCatInput.password.2": "password", + "WitCatInput.textalign.1": "left", + "WitCatInput.textalign.2": "center", + "WitCatInput.textalign.3": "right", + "WitCatInput.docs": "📖Tutorials", + "WitCatInput.input": "text area", + "WitCatInput.focal": "focal", + "WitCatInput.keyboard": "keyboard", + "WitCatInput.shadow": "shadow x[x]y[y]weight[width]color[color]", + "WitCatInput.shadows": "Merge shadows[first]with[last]", + "WitCatInput.fontadaptive": "[type]font size adaptation", + "WitCatInput.set.1": "Enable", + "WitCatInput.set.2": "Disable", } }) } + /** + * 翻译 + * @param {string} id + * @returns {string} + */ formatMessage(id) { return this._formatMessage({ id, @@ -115,14 +229,19 @@ class WitCatInput { getInfo() { return { - id: extensionId, // 拓展id + id: witcat_input_extensionId, // 拓展id name: this.formatMessage("WitCatInput.name"), // 拓展名 - docsURI: "https://www.ccw.site/post/6153a7a6-05fb-462e-b785-b97700b12bc2", - blockIconURI: _icon, - menuIconURI: _icon, + blockIconURI: witcat_input_icon, + menuIconURI: witcat_input_icon, color1: "#52baba", - color2: "#ffffff", + color2: "#459c9c", blocks: [ + { + blockType: "button", + text: this.formatMessage('WitCatInput.docs'), + onClick: this.docs, + }, + "---" + this.formatMessage('WitCatInput.input'), { opcode: "createinput", blockType: "command", @@ -189,6 +308,59 @@ class WitCatInput { }, }, }, + { + opcode: "shadow", + blockType: "reporter", + text: this.formatMessage("WitCatInput.shadow"), + arguments: { + x: { + type: "number", + defaultValue: "0", + }, + y: { + type: "number", + defaultValue: "0", + }, + width: { + type: "number", + defaultValue: "3", + }, + color: { + type: "string", + defaultValue: "#000000", + }, + }, + }, + { + opcode: "shadows", + blockType: "reporter", + text: this.formatMessage("WitCatInput.shadows"), + arguments: { + first: { + type: "string", + defaultValue: "0px 0px 3px #000000", + }, + last: { + type: "string", + defaultValue: "0px 0px 3px #000000", + }, + }, + }, + { + opcode: "fontweight", + blockType: "command", + text: this.formatMessage("WitCatInput.fontweight"), + arguments: { + id: { + type: "string", + defaultValue: "i", + }, + text: { + type: "string", + menu: "fontweight", + }, + }, + }, { opcode: "setread", blockType: "command", @@ -205,24 +377,62 @@ class WitCatInput { }, }, { - opcode: "compute", - blockType: "reporter", - text: this.formatMessage("WitCatInput.compute"), + opcode: "password", + blockType: "command", + text: this.formatMessage("WitCatInput.setread"), arguments: { - size: { - type: "number", - defaultValue: "16", - } + id: { + type: "string", + defaultValue: "i", + }, + read: { + type: "string", + menu: "password", + }, }, }, { - opcode: "deleteinput", + opcode: "textalign", blockType: "command", - text: this.formatMessage("WitCatInput.deleteinput"), + text: this.formatMessage("WitCatInput.setread"), + arguments: { + id: { + type: "string", + defaultValue: "i", + }, + read: { + type: "string", + menu: "textalign", + }, + }, + }, + { + opcode: "setfont", + blockType: "command", + text: this.formatMessage("WitCatInput.setfontfamily"), arguments: { id: { type: "string", defaultValue: "i", + }, + text: { + type: "string", + defaultValue: "url", + }, + name: { + type: "string", + defaultValue: "arial", + }, + }, + }, + { + opcode: "compute", + blockType: "reporter", + text: this.formatMessage("WitCatInput.compute"), + arguments: { + size: { + type: "number", + defaultValue: "16", } }, }, @@ -262,6 +472,35 @@ class WitCatInput { text: this.formatMessage("WitCatInput.numbers"), arguments: {}, }, + { + opcode: "deleteinput", + blockType: "command", + text: this.formatMessage("WitCatInput.deleteinput"), + arguments: { + id: { + type: "string", + defaultValue: "i", + } + }, + }, + { + opcode: "deleteallinput", + blockType: "command", + text: this.formatMessage("WitCatInput.deleteallinput"), + arguments: {}, + }, + { + opcode: "fontadaptive", + blockType: "command", + text: this.formatMessage("WitCatInput.fontadaptive"), + arguments: { + type: { + type: "Boolean", + menu: "set", + } + }, + }, + "---" + this.formatMessage('WitCatInput.focal'), { opcode: "isinput", blockType: "Boolean", @@ -290,6 +529,7 @@ class WitCatInput { } }, }, + "---" + this.formatMessage('WitCatInput.keyboard'), { opcode: "key", blockType: "Boolean", @@ -327,9 +567,10 @@ class WitCatInput { arguments: {}, }, { - opcode: "deleteallinput", - blockType: "command", - text: this.formatMessage("WitCatInput.deleteallinput"), + opcode: "keypress", + blockType: "reporter", + text: this.formatMessage("WitCatInput.keypress"), + func: false, arguments: {}, }, ], @@ -397,6 +638,26 @@ class WitCatInput { text: this.formatMessage('WitCatInput.number.14'), value: 'op' }, + { + text: this.formatMessage('WitCatInput.number.15'), + value: 'bg' + }, + { + text: this.formatMessage('WitCatInput.number.16'), + value: 'ff' + }, + { + text: this.formatMessage('WitCatInput.number.17'), + value: 'lh' + }, + { + text: this.formatMessage('WitCatInput.number.18'), + value: 'fw' + }, + { + text: this.formatMessage('WitCatInput.number.19'), + value: 'ts' + }, { text: this.formatMessage('WitCatInput.number.9'), value: 'json' @@ -447,6 +708,22 @@ class WitCatInput { text: this.formatMessage('WitCatInput.number.14'), value: 'op' }, + { + text: this.formatMessage('WitCatInput.number.15'), + value: 'bg' + }, + { + text: this.formatMessage('WitCatInput.number.17'), + value: 'lh' + }, + { + text: this.formatMessage('WitCatInput.number.19'), + value: 'ts' + }, + { + text: '⚠css', + value: 'css' + }, ], read: [ { @@ -457,358 +734,566 @@ class WitCatInput { text: this.formatMessage('WitCatInput.read.2'), value: 'ue' }, - ] + ], + password: [ + { + text: this.formatMessage('WitCatInput.password.1'), + value: 'text' + }, + { + text: this.formatMessage('WitCatInput.password.2'), + value: 'password' + }, + ], + textalign: [ + { + text: this.formatMessage('WitCatInput.textalign.1'), + value: 'left' + }, + { + text: this.formatMessage('WitCatInput.textalign.2'), + value: 'center' + }, + { + text: this.formatMessage('WitCatInput.textalign.3'), + value: 'right' + }, + ], + fontweight: { + acceptReporters: true, + items: [ + { + text: `100(${this.formatMessage('WitCatInput.thin')})`, + value: '100' + }, + { + text: '200', + value: '200' + }, + { + text: '300', + value: '300' + }, + { + text: `400(${this.formatMessage('WitCatInput.normal')})`, + value: '400' + }, + { + text: '500', + value: '500' + }, + { + text: '600', + value: '600' + }, + { + text: `700(${this.formatMessage('WitCatInput.bold')})`, + value: '700' + }, + { + text: '800', + value: '800' + }, + { + text: '900', + value: '900' + }, + ], + }, + set: [ + { + text: this.formatMessage('WitCatInput.set.1'), + value: 'true' + }, + { + text: this.formatMessage('WitCatInput.set.2'), + value: 'false' + }, + ], } }; } - //设置或创建文本框 + + /** + * 打开教程 + * @returns {void} + */ + docs() { + let a = document.createElement('a'); + a.href = "https://www.ccw.site/post/6153a7a6-05fb-462e-b785-b97700b12bc2"; + a.rel = "noopener noreferrer"; + a.target = "_blank"; + a.click(); + } + + /** + * 限制值的范围,如果值是NaN,返回最小值 + * @param {number} x 数值 + * @param {number} min 最小值 + * @param {number} max 最大值 + * @return {number} + */ + _clamp(x, min, max) { + return isNaN(x) ? min : x < min ? min : x > max ? max : x; + // return isNaN(x) ? min : Math.min(max, Math.max(min, x)); + } + + /** + * 设置或创建文本框 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + * @param {SCarg} args.type 文本框类型 "input"|"textarea" + * @param {SCarg} args.x 左上角x + * @param {SCarg} args.y 左上角y + * @param {SCarg} args.width 宽度 + * @param {SCarg} args.height 高度 + * @param {SCarg} args.text 初始文本 + * @param {SCarg} args.color 文字颜色 + * @param {SCarg} args.texts 提示文本 + * @param {SCarg} args.size 文字大小 + */ createinput(args) { - let x = args.x; - let y = args.y; - let width = args.width; - let height = args.height; - if (args.x > this.runtime.stageWidth) { - x = this.runtime.stageWidth; - } - if (args.x < 0) { - x = 0; - } - if (args.y > this.runtime.stageHeight) { - y = this.runtime.stageHeight; - } - if (args.y < 0) { - y = 0; - } - if (Number(args.x) + Number(args.width) > this.runtime.stageWidth) { - width = this.runtime.stageWidth - x; - } - if (args.width < 0) { - width = 0; - } - if (Number(args.y) + Number(args.height) > this.runtime.stageHeight) { - height = this.runtime.stageHeight - y; - } - if (args.height < 0) { - height = 0; + if (this.canvas() === null || this.inputParent() === null) { + return; } + let x = Number(args.x); + let y = Number(args.y); + let width = Number(args.width); + let height = Number(args.height); + x = this._clamp(x, 0, this.runtime.stageWidth); + y = this._clamp(y, 0, this.runtime.stageHeight); + width = this._clamp(width, 0, this.runtime.stageWidth - x); + height = this._clamp(height, 0, this.runtime.stageHeight - y); x = (x / this.runtime.stageWidth) * 100; y = (y / this.runtime.stageHeight) * 100; width = (width / this.runtime.stageWidth) * 100; height = (height / this.runtime.stageHeight) * 100; - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + args.size + `px;resize:none;color:` + args.color + `;opacity:1;`; - let search = document.getElementById("WitCatInput" + args.id); - if (search !== null) { - if (search.name === args.type) { - search.style = dom; - search.value = args.text; - search.placeholder = args.texts; - } - else { - cvs.parentNode.removeChild(search); - let eleLink = document.createElement(args.type); - if (args.type === "input") { - eleLink.type = "text"; - } - eleLink.style = dom; - eleLink.id = "WitCatInput" + args.id; - eleLink.value = args.text; - eleLink.className = "WitCatInput"; - eleLink.name = args.type; - eleLink.placeholder = args.texts; - cvs.parentNode.appendChild(eleLink); - } + + /** @type {HTMLInputElement|HTMLTextAreaElement|null} */ + let search = null; + let search_1 = document.getElementById("WitCatInput" + args.id); + if (search_1 instanceof HTMLInputElement || search_1 instanceof HTMLTextAreaElement) { + search = search_1; } - else { - let eleLink = document.createElement(args.type); - if (args.type === "input") { - eleLink.type = "text"; + // 这里通过“如果不符合,就删除;如果不存在,就建立”的方式, + // 避免后面大量复制粘贴样式操作。 + // 大段的复制粘贴往往意味着之后会犯错(只改一半) + if (search !== null && search.tagName !== args.type) { + this.inputParent().removeChild(search); + search = null; + } + if (search === null) { + // 标准下 input 和 textarea 都应该是小写, + // 但是菜单中的 textarea 是大写的。 + // 为了兼容不能修改菜单数值,只能转换。 + const argstype = String(args.type).toLowerCase(); + if (argstype !== "input" && argstype !== "textarea") { + // 防止修改 JSON 注入 + console.warn("Input.js: 类型应该是 input 或者 textarea"); + return; + } + search = document.createElement(argstype); + // 只有 input 才有 type 属性,textarea 没有 + if (search instanceof HTMLInputElement) { + search.type = "text"; } - eleLink.style = dom; - eleLink.id = "WitCatInput" + args.id; - eleLink.value = args.text; - eleLink.className = "WitCatInput"; - eleLink.name = args.type; - eleLink.placeholder = args.texts; - cvs.parentNode.appendChild(eleLink); + search.id = "WitCatInput" + args.id; + search.value = String(args.text); + search.className = "WitCatInput"; + search.name = String(args.type); + search.placeholder = String(args.texts); + this.inputParent().appendChild(search); } + + // 现在直接通过style的属性修改样式表,不需要担心“分号注入”问题了 + let sstyle = search.style; + sstyle.backgroundColor = "transparent"; + sstyle.border = "0px"; + sstyle.textShadow = "0 0 0 #000"; + sstyle.outline = "none"; + sstyle.position = "absolute"; + sstyle.left = `${x}%`; + sstyle.top = `${y}%`; + sstyle.width = `${width}%`; + sstyle.height = `${height}%`; + sstyle.fontSize = `${this.adaptive ? (parseFloat(this.canvas().style.width) / 360) * Number(args.size) : Number(args.size)}px`; + sstyle.resize = "none"; + sstyle.color = String(args.color); + sstyle.opacity = "1"; + sstyle.backgroundSize = "100% 100%"; + search.value = String(args.text); + search.placeholder = String(args.texts); + + this.inputFontSize[String(args.id)] = Number(args.size); } - //删除文本框 + + /** + * 删除文本框 + * @param {object} args + * @param {SCarg} args.id 文本框id + */ deleteinput(args) { + if (this.inputParent() === null) { + return; + } let search = document.getElementById("WitCatInput" + args.id); if (search !== null) { - cvs.parentNode.removeChild(search); + this.inputParent().removeChild(search); } } - //获取文本框内容 + + /** + * 获取文本框内容 + * @param {object} args + * @param {SCarg} args.id 文本框id + * @param {SCarg} args.type 内容类型 + * @return {SCarg} + */ getinput(args) { let search = document.getElementById("WitCatInput" + args.id); - if (search !== null) { - if (args.type === "X") - return (search.style.left.split("%")[0] / 100) * this.runtime.stageWidth; - else if (args.type === "Y") - return (search.style.top.split("%")[0] / 100) * this.runtime.stageHeight; - else if (args.type === "width") - return (search.style.width.split("%")[0] / 100) * this.runtime.stageWidth; - else if (args.type === "height") - return (search.style.height.split("%")[0] / 100) * this.runtime.stageHeight; - else if (args.type === "content") - return search.value; - else if (args.type === "color") - return search.style.color.colorHex(); - else if (args.type === "prompt") - return search.placeholder; - else if (args.type === "font-size") - return search.style.fontSize.split("px")[0]; - else if (args.type === "ID") - return search.id.split("WitCatInput")[1]; - else if (args.type === "rp") - return search.scrollTop; - else if (args.type === "th") - return search.scrollHeight; - else if (args.type === "cp") - return JSON.stringify([search.selectionStart, search.selectionEnd]); - else if (args.type === "op") - return 100 - (search.style.opacity * 100); - else { - return ( - "{\"" + "X" + "\":\"" + ((search.style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + - "\"" + "Y" + "\":\"" + ((search.style.top.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + - "\"" + "width" + "\":\"" + ((search.style.width.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + - "\"" + "height" + "\":\"" + ((search.style.height.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + - "\"" + "content" + "\":" + JSON.stringify(search.value) + "," + - "\"" + "color" + "\":\"" + (search.style.color.colorHex()) + "\"," + - "\"" + "prompt" + "\":\"" + (search.placeholder) + "\"," + - "\"" + "font-size" + "\":\"" + (search.style.fontSize.split("px")[0]) + "\"," + - "\"" + "ID" + "\":\"" + (search.id.split("WitCatInput")[1]) + "\"," + - "\"" + "Rolling position" + "\":\"" + (search.scrollTop) + "\"," + - "\"" + "Text height" + "\":\"" + (search.scrollHeight) + "\"," + - "\"" + "cursor position" + "\":\"" + (JSON.stringify([search.selectionStart, search.selectionEnd])) + "\"}" - ) - } - } - else { - return (""); - } + return search === null ? "" : this._getattrib(search, String(args.type)); } - //焦点判断 + + /** + * 焦点判断 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + * @returns {boolean} + */ isinput(args) { let search = document.getElementById("WitCatInput" + args.id); if (search !== null) { if (search === document.activeElement) { - return (true); + return true; } else { - return (false); + return false; } } else { - return (false); + return false; } } - //焦点位置 + + /** + * 焦点位置 + * @returns {string} 文本框 ID + */ whatinput() { - if (document.activeElement.className === "WitCatInput") { - return document.activeElement.id.split("WitCatInput")[1]; + if (document.activeElement !== null && document.activeElement.className === "WitCatInput") { + return this._getWitCatID(document.activeElement); } else { return ""; } } - //焦点获取 + + /** + * 焦点获取 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + */ nowinput(args) { let search = document.getElementById("WitCatInput" + args.id); if (search !== null) { search.focus(); } - else if (document.activeElement.className === "WitCatInput") { - document.activeElement.blur(); + else { + const active = document.activeElement; + if (active !== null && active.className === "WitCatInput") { + if (active instanceof HTMLInputElement || active instanceof HTMLTextAreaElement) { + active.blur(); + } + } } } - //删除所有文本框 + + /** + * 删除所有文本框 + */ deleteallinput() { + if (this.inputParent() === null) { + return; + } let search = document.getElementsByClassName("WitCatInput"); - let i = 0; - for (i = search.length - 1; i >= 0; i--) { - search[i].parentNode.removeChild(search[i]); + for (const item of Array.from(search)) { + this.inputParent().removeChild(item); } } - //计算坐标 + + /** + * 计算文字大小 + * @param {object} args + * @param {SCarg} args.size Scratch 文字大小 + * @returns {number} + */ compute(args) { - return (cvs.style.width.split("px")[0] / 360) * args.size; + if (this.canvas() === null) { + return 0; + } + return parseFloat(this.canvas().style.width) / 360 * Number(args.size); } - //获取状态 + + /** + * 获取指定编号的文本框属性 + * @param {object} args + * @param {SCarg} args.num 文本框序号 + * @param {SCarg} args.type 属性类型 + * @returns {SCarg} + */ number(args) { - let search = document.getElementsByClassName("WitCatInput"); - if (search.length >= args.num && args.num > 0) { - if (args.type === "X") - return (search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth; - else if (args.type === "Y") - return (search[args.num - 1].style.top.split("%")[0] / 100) * this.runtime.stageHeight; - else if (args.type === "width") - return (search[args.num - 1].style.width.split("%")[0] / 100) * this.runtime.stageWidth; - else if (args.type === "height") - return (search[args.num - 1].style.height.split("%")[0] / 100) * this.runtime.stageHeight; - else if (args.type === "content") - return search[args.num - 1].value; - else if (args.type === "color") - return search[args.num - 1].style.color.colorHex(); - else if (args.type === "prompt") - return search[args.num - 1].placeholder; - else if (args.type === "font-size") - return search[args.num - 1].style.fontSize.split("px")[0]; - else if (args.type === "ID") - return search[args.num - 1].id.split("WitCatInput")[1]; - else if (args.type === "rp") - return search[args.num - 1].scrollTop; - else if (args.type === "th") - return search[args.num - 1].scrollHeight; - else if (args.type === "cp") - return JSON.stringify([search[args.num - 1].selectionStart, search[args.num - 1].selectionEnd]); - else if (args.type === "op") - return 100 - (search[args.num - 1].style.opacity * 100); - else { - return ( - "{\"" + "X" + "\":\"" + ((search[args.num - 1].style.left.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + - "\"" + "Y" + "\":\"" + ((search[args.num - 1].style.top.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + - "\"" + "width" + "\":\"" + ((search[args.num - 1].style.width.split("%")[0] / 100) * this.runtime.stageWidth) + "\"," + - "\"" + "height" + "\":\"" + ((search[args.num - 1].style.height.split("%")[0] / 100) * this.runtime.stageHeight) + "\"," + - "\"" + "content" + "\":" + JSON.stringify(search[args.num - 1].value) + "," + - "\"" + "color" + "\":\"" + (search[args.num - 1].style.color.colorHex()) + "\"," + - "\"" + "prompt" + "\":\"" + (search[args.num - 1].placeholder) + "\"," + - "\"" + "font-size" + "\":\"" + (search[args.num - 1].style.fontSize.split("px")[0]) + "\"," + - "\"" + "ID" + "\":\"" + (search[args.num - 1].id.split("WitCatInput")[1]) + "\"," + - "\"" + "Rolling position" + "\":\"" + (search[args.num - 1].scrollTop) + "\"," + - "\"" + "Text height" + "\":\"" + (search[args.num - 1].scrollHeight) + "\"," + - "\"" + "cursor position" + "\":\"" + (JSON.stringify([search[args.num - 1].selectionStart, search[args.num - 1].selectionEnd])) + "\"}" - ) - } + let searchall = document.getElementsByClassName("WitCatInput"); + let index = Number(args.num); + let search = searchall[index - 1]; + if (search !== undefined) { + return this._getattrib(search, String(args.type)); + } else { + return ""; } - else { + } + + /** + * 获取文本框的属性 + * @param {Element} element 文本框元素 + * @param {string} type 属性类型 + * @returns {number|string} + */ + _getattrib(element, type) { + // 用于通过类型检查,确保不出错 + if (!(element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement)) { + console.warn("Input.js: 获取到的元素的类型不正确: ", element); return ""; } + switch (type) { + case "X": + return parseFloat(element.style.left) / 100 * this.runtime.stageWidth; + // 理论上需要加break;,但是前面已经return了 + case "Y": + return parseFloat(element.style.top) / 100 * this.runtime.stageHeight; + case "width": + return parseFloat(element.style.width) / 100 * this.runtime.stageWidth; + case "height": + return parseFloat(element.style.height) / 100 * this.runtime.stageHeight; + case "content": + return element.value; + case "color": + return string_colorHex(element.style.color); + case "prompt": + return element.placeholder; + case "font-size": + return parseFloat(element.style.fontSize); + case "ID": + // 直接上正则,可以处理类似“WitCatInput123WitCatInput456”这样包含“WitCatInput”的奇葩ID + { + let match = /^WitCatInput(.*)$/.exec(element.id); + return match === null || match[1] === undefined ? "" : match[1]; + } + case "rp": + return element.scrollTop; + case "th": + return element.scrollHeight; + case "cp": + return JSON.stringify([element.selectionStart, element.selectionEnd]); + case "op": + return 100 - (Number(element.style.opacity) * 100); + case "bg": + // 不适合split的地方,直接上正则 + // (注:一开始backgroundImage的值可能是空的或者别的什么东西……) + { + // 打花括号之后就可以在里面声明变量了 + let match = /^url\("(.*)"\)$/.exec(element.style.backgroundImage); + if (match !== null && match[1] !== undefined) { + return decodeURI(match[1]); + } else { + // 正则匹配失败 + return ""; + } + } + case "ff": + return element.style.fontFamily; + case "lh": + return parseFloat(element.style.lineHeight); + case "fw": + return element.style.fontWeight; + case "ts": + return element.style.textShadow; + case "json": + // 直接把整个东西转成 JSON 对象,再拼接 + return JSON.stringify( + { + X: this._getattrib(element, "X"), + Y: this._getattrib(element, "Y"), + width: this._getattrib(element, "width"), + height: this._getattrib(element, "height"), + content: this._getattrib(element, "content"), + color: this._getattrib(element, "color"), + prompt: this._getattrib(element, "prompt"), + "font-size": this._getattrib(element, "font-size"), + ID: this._getattrib(element, "ID"), + "Rolling position": this._getattrib(element, "rp"), + "Text height": this._getattrib(element, "th"), + "cursor position": JSON.parse(String(this._getattrib(element, "cp"))) + // 这里看起来缺了一些东西,如果没有合并复制粘贴的代码以及进 + // 行优化,将需要修改两次大段内容,现在修改就简单了。 + } + ) + default: + return ""; + } } - //文本框数量 + + /** + * 文本框数量 + * @returns {number} + */ numbers() { let search = document.getElementsByClassName("WitCatInput"); return search.length; } - //按键检测 + + /** + * 按键检测 + * @param {object} args + * @param {SCarg} args.type 按键类型,用逗号分隔 + * @returns {boolean} + */ key(args) { - return (args.type in keypress); + let key = String(args.type).split(","); + for (const item of key) { + if (!Object.keys(this.keypresslist).includes(item)) { + return false; + } + } + return true; } - //按键检测 + + /** + * 按键检测(帽子积木) + * @param {object} args + * @param {SCarg} args.type 按键类型,用逗号分隔 + * @returns {boolean} + */ keys(args) { - return (args.type in keypress); + return this.key(args); } - //上次按下的键 + + /** + * 上次按下的键 + * @returns {string} + */ lastkey() { - return lastKey; + return this.lastKey; } - //鼠标滚轮 + + /** + * 鼠标滚轮速度 + * @returns {number} + */ mousewheel() { - return MouseWheel; + return this.MouseWheel; } - //设置文本框 + + /** + * 设置文本框 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + * @param {SCarg} args.type 属性类型 + * @param {SCarg} args.text 属性值 + */ setinput(args) { - let search = document.getElementById("WitCatInput" + args.id); + let search = this._findWitCatInput(String(args.id)); if (search !== null) { - let x = search.style.left.split("%")[0]; - let y = search.style.top.split("%")[0]; - let width = search.style.width.split("%")[0]; - let height = search.style.height.split("%")[0]; - let content = search.value; - let prompt = search.placeholder; - let color = search.style.color.colorHex() - let size = search.style.fontSize.split("px")[0]; - let scrolltop = search.scrollTop; - let opacity = search.style.opacity; - if (args.type === "X") { - x = args.text; - if (args.text > this.runtime.stageWidth) { - x = this.runtime.stageWidth; - } - if (args.text < 0) { - x = 0; - } - x = (x / this.runtime.stageWidth) * 100; - } - else if (args.type === "Y") { - y = args.text; - if (args.text > this.runtime.stageHeight) { - y = this.runtime.stageHeight; - } - if (args.text < 0) { - y = 0; - } - y = (y / this.runtime.stageHeight) * 100; - } - else if (args.type === "width") { - width = args.text; - if (Number(x) + Number(args.text) > this.runtime.stageWidth) { - width = this.runtime.stageWidth - x; - } - if (args.text < 0) { - width = 0; - } - width = (width / this.runtime.stageWidth) * 100; - } - else if (args.type === "height") { - height = args.text; - if (Number(y) + Number(args.text) > this.runtime.stageHeight) { - height = this.runtime.stageHeight - y; - } - if (args.text < 0) { - height = 0; - } - height = (height / this.runtime.stageHeight) * 100; - } - else if (args.type === "content") { - content = args.text; - } - else if (args.type === "prompt") { - prompt = args.text; - } - else if (args.type === "color") { - color = args.text; - } - else if (args.type === "font-size") { - size = args.text; - } - else if (args.type === "rp") { - scrolltop = args.text; - } - else if (args.type === "op") { - if (!isNaN(args.text)) { - opacity = 1 - (args.text / 100); - } - else { - opacity = 1; - } - } - else if (args.type === "cp") { - try { - if (JSON.parse(args.text).length >= 2) { - search.setSelectionRange(JSON.parse(args.text)[0], JSON.parse(args.text)[1]); + let sstyle = search.style; + let x, y, width, height, opacity; + switch (args.type) { + case "X": + x = this._clamp(Number(args.text), 0, this.runtime.stageWidth); + x = (x / this.runtime.stageWidth) * 100; + sstyle.left = x + "%"; + break; + case "Y": + y = this._clamp(Number(args.text), 0, this.runtime.stageHeight); + y = (y / this.runtime.stageHeight) * 100; + sstyle.top = y + "%"; + break; + case "width": + x = parseFloat(sstyle.left) / 100 * this.runtime.stageWidth; + width = this._clamp(Number(args.text), 0, this.runtime.stageWidth - x); + width = (width / this.runtime.stageWidth) * 100; + sstyle.width = Number(width) + "%"; + break; + case "height": + y = parseFloat(sstyle.top) / 100 * this.runtime.stageHeight; + height = this._clamp(Number(args.text), 0, this.runtime.stageHeight - y); + height = (height / this.runtime.stageHeight) * 100; + sstyle.height = Number(height) + "%"; + break; + case "content": + search.value = String(args.text); + break; + case "prompt": + search.placeholder = String(args.text); + break; + case "color": + sstyle.color = String(args.text); + break; + case "font-size": + sstyle.fontSize = Number(args.text) + "px"; + this.inputFontSize[String(args.id)] = Number(args.text); + break; + case "rp": + search.scrollTop = Number(args.text); + break; + case "op": + opacity = this._clamp(Number(args.text), 0, 100); + opacity = 1 - (opacity / 100); + sstyle.opacity = String(opacity); + break; + case "cp": + try { + let selection = JSON.parse(String(args.text)); + if (selection instanceof Array && selection.length === 2) { + search.setSelectionRange(selection[0], selection[1]); + } + else if (typeof (selection) === "number") { + search.setSelectionRange(selection, selection); + } + } + catch { + return; + } + break; + case "bg": + if (String(args.text).startsWith("https://m.ccw.site/") || String(args.text).startsWith("https://m.xiguacity.com/")) { + sstyle.backgroundImage = 'url("' + encodeURI(String(args.text)) + '")'; + sstyle.backgroundSize = "100% 100%"; } else { - search.setSelectionRange(args.text, args.text); + console.warn("禁止的链接/图片\nBanned links/pictures"); } - } - catch { - return; - } + break; + case "lh": + sstyle.lineHeight = Number(args.text) + "px"; + break; + case "ts": + sstyle.textShadow = String(args.text); + break; + case "css": + // https://www.cnblogs.com/ndos/p/9706646.html + search.setAttribute("style", String(args.text)); + break; } - let dom = `background-color: transparent;border:0px;text-shadow: 0 0 0 #000;outline: none;position:absolute; left:` + x + `%; top:` + y + `%; width:` + width + `%; height:` + height + `%;font-size: ` + size + `px;resize:none;color:` + color + `;opacity:` + opacity + `;`; - - search.style = dom; - search.value = content; - search.placeholder = prompt; - search.scrollTop = scrolltop; } } - //设置状态 + + /** + * 设置是否可修改 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + * @param {SCarg} args.read 能否修改 + */ setread(args) { - let search = document.getElementById("WitCatInput" + args.id); + let search = this._findWitCatInput(String(args.id)); if (search !== null) { if (args.read === "eb") { search.disabled = false; @@ -818,6 +1303,211 @@ class WitCatInput { } } } + + /** + * 设置文本框是否为密码框 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + * @param {SCarg} args.read 是密码框 "test"|"password" + */ + password(args) { + let search = this._findWitCatInput(String(args.id)); + if (search !== null) { + if (search instanceof HTMLTextAreaElement) { + console.warn("Input.js: 多行文本框无法设为密码框"); + return; + } + search.type = String(args.read); + } + } + + /** + * 获取按下的按键 + * @returns {string} + */ + keypress() { + return JSON.stringify(Object.keys(this.keypresslist)); + } + + /** + * 设置字体 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + * @param {SCarg} args.name 字体名 + * @param {SCarg} args.text 字体链接 + */ + setfont(args) { + const search = document.getElementById("WitCatInput" + args.id); + if (search !== null) { + const xhr = new XMLHttpRequest(); // 定义一个异步对象 + xhr.open('GET', String(args.text), true); // 异步GET方式加载字体 + xhr.responseType = "arraybuffer"; //把异步获取类型改为arraybuffer二进制类型 + xhr.onload = function () { + // 这里做了一个判断:如果浏览器支持FontFace方法执行 + if (typeof FontFace != 'undefined') { + document.fonts.add(new FontFace(String(args.name), this.response)); // 将字体对象添加到页面中 + search.style.fontFamily = `"${args.name}"`; + } else { + search.innerHTML = `@font-face{font-family:"${args.name}";src:url("${args.text}") `; + } + } + xhr.send(); + } + } + + /** + * 设置对齐方式 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + * @param {SCarg} args.read 对齐方式 "left"|"center"|"right" + */ + textalign(args) { + let search = document.getElementById("WitCatInput" + args.id); + if (search !== null) { + search.style.textAlign = String(args.read); + } + } + + /** + * 设置文本框字体粗细 + * @param {object} args + * @param {SCarg} args.id 文本框 ID + * @param {SCarg} args.text 字体粗细 + */ + fontweight(args) { + let search = document.getElementById("WitCatInput" + args.id); + if (search !== null) { + search.style.fontWeight = String(args.text); + } + } + + /** + * 创建阴影 + * @param {object} args + * @param {SCarg} args.x 偏移x + * @param {SCarg} args.y 偏移y + * @param {SCarg} args.width 宽度 + * @param {SCarg} args.color 颜色 + * @returns {string} + */ + shadow(args) { + return `${args.x}px ${args.y}px ${args.width}px ${args.color}` + } + /** + * 合并阴影 + * @param {object} args + * @param {SCarg} args.first 第一个阴影 + * @param {SCarg} args.last 第二个阴影 + * @returns {string} + */ + shadows(args) { + return `${args.first},${args.last}` + } + + /** + * 设置字体自适应 + * @param {object} args + * @param {SCarg} args.type + */ + fontadaptive(args) { + if (this.canvas() === null) { + return; + } + if (args.type == "true") { + if (!this.adaptive) { + let search = document.getElementsByClassName("WitCatInput"); + const config = { attributes: true, childList: true, subtree: true, attributeFilter: ['style'] }; + const callback = () => { + if (this.canvas() === null) { + return; + } + for (let searchi of Array.from(search)) { + const searchid = this._getWitCatID(searchi); + const fontsize = this.inputFontSize[searchid]; + if (fontsize === undefined) { + continue; + } + searchi.style.fontSize = parseFloat(this.canvas().style.width) / 360 * fontsize + "px"; + } + }; + this.observer = new MutationObserver(callback); + this.observer.observe(this.canvas(), config); + this.adaptive = true; + } + } + else { + if (this.adaptive) { + if (this.observer !== null) { + this.observer.disconnect(); + } + this.adaptive = false; + } + } + } + + /** + * 添加键盘鼠标事件 + */ + _addevent() { + if (this.canvas() === null || this.inputParent() === null) { + return; + } + //键盘事件监听 + document.addEventListener("keydown", (event) => { + this.keypresslist[event.code] = true; + this.lastKey = event.code; + }); + document.addEventListener("keyup", (event) => { + delete this.keypresslist[event.code]; + }); + + //给页面绑定滑轮滚动事件 + this.canvas().addEventListener('wheel', (e) => { + // 注意这个负数…… + // 目前的标准用法是使用 deltaY,但是 deltaY 的符号和 WheelDeltaY 相反。 + // 为了和原有的行为一致,乘上 -3 + // 在我的浏览器中 deltaY = WheelDeltaY / -3 + this.MouseWheel = e.deltaY * -3; + clearTimeout(this.timer); + this.timer = setTimeout(() => { + this.MouseWheel = 0; + }, 30); + }, { capture: true }); + } + + /** + * 获取指定元素的 WitCatInput ID + * @param {Element} element 元素 + * @returns {string} 元素 ID 去掉 WitCatInput 后的部分,如果没有,返回 "" + */ + _getWitCatID(element) { + const match = /^WitCatInput(.*)$/.exec(element.id); + if (match === null || match[1] === undefined) { + console.warn("Input.js: 无法获取 WitCatInput ID: ", element); + return ""; + } + return match[1]; + } + + /** + * 获取指定 WitCatInput ID 的元素 + * @param {string} witcatID 元素 + * @returns {HTMLInputElement | HTMLTextAreaElement | null} 获取的元素,或者 null + */ + _findWitCatInput(witcatID) { + const search = document.getElementById("WitCatInput" + witcatID); + if (search === null) { + console.warn("Input.js: 找不到 ID", witcatID); + return null; + } + if (search instanceof HTMLInputElement || search instanceof HTMLTextAreaElement) { + return search; + } + console.warn("Input.js: 元素不是 或者