您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
adds a simple way of searching for functions when selected on a > block - this is meant to be used with my scratch extended script as without it it won't do anything.
// ==UserScript== // @name scratch extended autocomplete // @version 0.2 // @description adds a simple way of searching for functions when selected on a > block - this is meant to be used with my scratch extended script as without it it won't do anything. // @run-at document-load // @author rssaromeo // @match *://scratch.mit.edu/* // @icon  // @grant none // @license GNU GPLv3 // @namespace https://greasyfork.org/users/1184528 // @tag unused // ==/UserScript== function foreachobj( //array obj, //object cb, //function ) { return Reflect.ownKeys(obj).map((e, i) => { return cb(e, obj[e], i) }) } function qs( //element text, //string parent = document, //element|string ) { //querySelector if (gettype(parent) == "string") parent = qs(parent) return parent.querySelector(text) } function gettype( //string a, //any ) { var thing = a var type = "string" if (typeof thing == "undefined") return "undefined" if (typeof thing == "symbol") return "symbol" if (ignore(() => thing.addEventListener) !== undefined) type = "element" if (typeof thing == "function") type = "function" if (ignore(() => thing.reverse)) type = "array" if (!!thing) if (Object.keys(thing).length !== undefined && thing.length === undefined) type = "object" if (typeof thing !== "object") if ( isNaN(thing) && String(Number(thing)) === String(thing) && thing.length === undefined && String(thing) === "NaN" && JSON.stringify(thing) === "null" ) type = "NaN" if (typeof thing !== "object") if (String(!!a) === String(a)) type = "boolean" if (typeof thing !== "object") if (String(Number(thing)) === String(thing) && a[0] === undefined) type = "number" return type } function ignore( //any func, //func senderr, //boolean ) { try { func() } catch (e) { return senderr ? e.message : undefined } } function createelem( //element elem, //string data, //object parent, //string|element ) { elem = document.createElement(elem) if (data.style) data.style.split(" ").forEach((e) => { elem.classList.add(e) }) Object.assign(elem.style, data) Object.assign(elem, data) if (typeof parent == "string") parent = a(parent).qs().val a(() => parent.appendChild(elem)).ignore() return elem } function listen( //[element, string, function] elem, //element type, //string cb, //function ) { elem.addEventListener(type, cb) return [elem, type, cb] } if (!qs("#scratchextendedstyles")) createelem( "style", { innerHTML: `.hide{ display:none; } z:hover{ background-color:#999; }`, id: "scratchextendedstyles", }, document.head, ) if (!window.g) window.g = createelem( "div", { innerHTML: "asd", position: "absolute", color: "#777", backgroundColor: "#777", maxHeight: "200px", overflow: "scroll", border: "2px solid black", }, document.body, ) listen(window, "mousemove", update) listen(window, "scroll", update) listen(window, "keyup", update) listen(window, "keypress", update) var lastf, lastfhtml log("asdasdsd") function update() { var f = document.querySelectorAll( "g.blocklyDraggable.blocklySelected > * > .blocklyEditableText > text", )[0] if (f == lastf && f?.innerHTML == lastfhtml) return if (!f?.innerHTML) return g.classList.add("hide") lastf = f lastfhtml = f.innerHTML if (!f) return g.classList.add("hide") g.classList.remove("hide") var temp if (!(temp = document.querySelector("g.blocklyDraggable.blocklySelected"))) return g.classList.add("hide") temp = temp["dataset"].id temp = vm.editingTarget.blocks._blocks[temp] if (!temp) return if (temp.opcode !== "operator_gt") return g.classList.add("hide") temp = temp.inputs.OPERAND1.block var r = rect(f) Object.assign(g.style, { top: r.y + r.h * 2 + "px", left: r.x + "px" }) g.innerHTML = "" var projectid = location.href.match(/(?<=\/)[0-9]+(?=\/)/)?.[0] || "local" var allow = [ ...JSON.parse(localStorage["allowed:" + projectid] ?? "[]"), ...JSON.parse(localStorage["scratch:defaultperms"] ?? "[]"), ] var deny = JSON.parse(localStorage["denied:" + projectid] ?? "[]") search( f.innerHTML, Object.keys(__scratchextended) .filter((e) => !["inputtype", "eventdata", "thread"].includes(e)) .sort(), ).forEach((e) => { createelem( "z", { f, class: "selectables", padding: "2px", innerHTML: e, color: allow.includes(e) ? "#0a0" : deny.includes(e) ? "#a00" : "black", onclick(e) { var temp if ( !(temp = document.querySelector( "g.blocklyDraggable.blocklySelected", )) ) return temp = vm.editingTarget.blocks._blocks[temp["dataset"].id].inputs.OPERAND1 .block vm.editingTarget.blocks._blocks[temp].fields.TEXT.value = this.innerHTML g.classList.add("hide") vm.emitWorkspaceUpdate() }, }, g, ) createelem("br", {}, g) }) } function rect(e) { var { x, y, width, height } = e.getBoundingClientRect().toJSON() return { x, y, w: width, h: height } } function search(q, o) { var arr = [] o.forEach((e) => { var text = e.trim() var s = "" q.split(" ").forEach((e, i) => { s += `(?:^| ).*?${e}.*?` }) if (new RegExp(s, "gi").test(text)) arr.push(e) }) return arr }