commit 7b31cd163be946e2c8db91180faaea62f3e5de3a Author: DrunkOatmeal Date: Thu Jul 13 13:28:59 2023 +0700 Initial Commit diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..1e430d8 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +node_modules +docs diff --git a/README.md b/README.md new file mode 100644 index 0000000..20fb3e1 --- /dev/null +++ b/README.md @@ -0,0 +1,12 @@ +## Old Twitter Image Search + +An image search userscript designed specifically for [Old Twiter Layout Extension](https://github.com/dimdenGD/OldTwitter) + +### Installation +Grab a compiled `otis.user.js` file from `/dist` and install it on your userscript manager. + +### Special thanks +- [NTISAS](https://github.com/BrokenEagle/JavaScripts) for the original inspiration +- [Old Twiter Layout Extension](https://github.com/dimdenGD/OldTwitter) for lightweight twitter web app +- [vite-plugin-monkey](https://github.com/lisonge/vite-plugin-monkey) to make it possible to build userscript with modern framework +- [SolidJS](https://www.solidjs.com/) for a robust JS framework diff --git a/dist/otis.user.js b/dist/otis.user.js new file mode 100644 index 0000000..1cb5721 --- /dev/null +++ b/dist/otis.user.js @@ -0,0 +1,1092 @@ +// ==UserScript== +// @name Old Twitter Image Search +// @namespace vite-plugin-monkey +// @version 1.0.0 +// @author SoberOatmeal +// @description otis +// @license MIT +// @match https://twitter.com/* +// @connect donmai.us +// @connect saucenao.com +// @connect twimg.com +// @connect api.twitter.com +// @grant GM.getValue +// @grant GM.setValue +// @grant GM.xmlHttpRequest +// @grant GM_getValue +// @grant GM_setValue +// @grant GM_xmlhttpRequest +// @run-at document-end +// ==/UserScript== + +(t=>{const i=document.createElement("style");i.dataset.source="vite-plugin-monkey",i.textContent=t,document.head.append(i)})(" .otis-icon{width:16px;vertical-align:middle;margin-right:20px;margin-left:6px;margin-bottom:4px}.otis-icon:hover{opacity:.8}.otis-icon-loading{opacity:.6!important;animation-name:icon-spin;animation-duration:2s;animation-iteration-count:infinite;animation-timing-function:linear}.otis-icon-noresult{opacity:.3!important}.otis-danbo-icon{width:18px;vertical-align:middle;margin-bottom:2px}.danbo-text-link{color:var(--light-gray);font-weight:700;font-size:12px}.danbo-more-results{margin-left:250px}.danbo-item>img,.danbo-item>div{vertical-align:middle;display:inline-block}.danbo-item>div{margin-left:10px}@keyframes icon-spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}} "); + +(function () { + 'use strict'; + + const equalFn = (a, b) => a === b; + const $TRACK = Symbol("solid-track"); + const signalOptions = { + equals: equalFn + }; + let runEffects = runQueue; + const STALE = 1; + const PENDING = 2; + const UNOWNED = { + owned: null, + cleanups: null, + context: null, + owner: null + }; + var Owner = null; + let Transition = null; + let Listener = null; + let Updates = null; + let Effects = null; + let ExecCount = 0; + function createRoot(fn, detachedOwner) { + const listener = Listener, owner = Owner, unowned = fn.length === 0, root = unowned ? UNOWNED : { + owned: null, + cleanups: null, + context: null, + owner: detachedOwner === void 0 ? owner : detachedOwner + }, updateFn = unowned ? fn : () => fn(() => untrack(() => cleanNode(root))); + Owner = root; + Listener = null; + try { + return runUpdates(updateFn, true); + } finally { + Listener = listener; + Owner = owner; + } + } + function createSignal(value, options) { + options = options ? Object.assign({}, signalOptions, options) : signalOptions; + const s = { + value, + observers: null, + observerSlots: null, + comparator: options.equals || void 0 + }; + const setter = (value2) => { + if (typeof value2 === "function") { + value2 = value2(s.value); + } + return writeSignal(s, value2); + }; + return [readSignal.bind(s), setter]; + } + function createRenderEffect(fn, value, options) { + const c = createComputation(fn, value, false, STALE); + updateComputation(c); + } + function createMemo(fn, value, options) { + options = options ? Object.assign({}, signalOptions, options) : signalOptions; + const c = createComputation(fn, value, true, 0); + c.observers = null; + c.observerSlots = null; + c.comparator = options.equals || void 0; + updateComputation(c); + return readSignal.bind(c); + } + function untrack(fn) { + if (Listener === null) + return fn(); + const listener = Listener; + Listener = null; + try { + return fn(); + } finally { + Listener = listener; + } + } + function onCleanup(fn) { + if (Owner === null) + ; + else if (Owner.cleanups === null) + Owner.cleanups = [fn]; + else + Owner.cleanups.push(fn); + return fn; + } + function readSignal() { + if (this.sources && this.state) { + if (this.state === STALE) + updateComputation(this); + else { + const updates = Updates; + Updates = null; + runUpdates(() => lookUpstream(this), false); + Updates = updates; + } + } + if (Listener) { + const sSlot = this.observers ? this.observers.length : 0; + if (!Listener.sources) { + Listener.sources = [this]; + Listener.sourceSlots = [sSlot]; + } else { + Listener.sources.push(this); + Listener.sourceSlots.push(sSlot); + } + if (!this.observers) { + this.observers = [Listener]; + this.observerSlots = [Listener.sources.length - 1]; + } else { + this.observers.push(Listener); + this.observerSlots.push(Listener.sources.length - 1); + } + } + return this.value; + } + function writeSignal(node, value, isComp) { + let current = node.value; + if (!node.comparator || !node.comparator(current, value)) { + node.value = value; + if (node.observers && node.observers.length) { + runUpdates(() => { + for (let i = 0; i < node.observers.length; i += 1) { + const o = node.observers[i]; + const TransitionRunning = Transition && Transition.running; + if (TransitionRunning && Transition.disposed.has(o)) + ; + if (TransitionRunning ? !o.tState : !o.state) { + if (o.pure) + Updates.push(o); + else + Effects.push(o); + if (o.observers) + markDownstream(o); + } + if (!TransitionRunning) + o.state = STALE; + } + if (Updates.length > 1e6) { + Updates = []; + if (false) + ; + throw new Error(); + } + }, false); + } + } + return value; + } + function updateComputation(node) { + if (!node.fn) + return; + cleanNode(node); + const owner = Owner, listener = Listener, time = ExecCount; + Listener = Owner = node; + runComputation(node, node.value, time); + Listener = listener; + Owner = owner; + } + function runComputation(node, value, time) { + let nextValue; + try { + nextValue = node.fn(value); + } catch (err) { + if (node.pure) { + { + node.state = STALE; + node.owned && node.owned.forEach(cleanNode); + node.owned = null; + } + } + node.updatedAt = time + 1; + return handleError(err); + } + if (!node.updatedAt || node.updatedAt <= time) { + if (node.updatedAt != null && "observers" in node) { + writeSignal(node, nextValue); + } else + node.value = nextValue; + node.updatedAt = time; + } + } + function createComputation(fn, init, pure, state = STALE, options) { + const c = { + fn, + state, + updatedAt: null, + owned: null, + sources: null, + sourceSlots: null, + cleanups: null, + value: init, + owner: Owner, + context: null, + pure + }; + if (Owner === null) + ; + else if (Owner !== UNOWNED) { + { + if (!Owner.owned) + Owner.owned = [c]; + else + Owner.owned.push(c); + } + } + return c; + } + function runTop(node) { + if (node.state === 0) + return; + if (node.state === PENDING) + return lookUpstream(node); + if (node.suspense && untrack(node.suspense.inFallback)) + return node.suspense.effects.push(node); + const ancestors = [node]; + while ((node = node.owner) && (!node.updatedAt || node.updatedAt < ExecCount)) { + if (node.state) + ancestors.push(node); + } + for (let i = ancestors.length - 1; i >= 0; i--) { + node = ancestors[i]; + if (node.state === STALE) { + updateComputation(node); + } else if (node.state === PENDING) { + const updates = Updates; + Updates = null; + runUpdates(() => lookUpstream(node, ancestors[0]), false); + Updates = updates; + } + } + } + function runUpdates(fn, init) { + if (Updates) + return fn(); + let wait = false; + if (!init) + Updates = []; + if (Effects) + wait = true; + else + Effects = []; + ExecCount++; + try { + const res = fn(); + completeUpdates(wait); + return res; + } catch (err) { + if (!wait) + Effects = null; + Updates = null; + handleError(err); + } + } + function completeUpdates(wait) { + if (Updates) { + runQueue(Updates); + Updates = null; + } + if (wait) + return; + const e = Effects; + Effects = null; + if (e.length) + runUpdates(() => runEffects(e), false); + } + function runQueue(queue) { + for (let i = 0; i < queue.length; i++) + runTop(queue[i]); + } + function lookUpstream(node, ignore) { + node.state = 0; + for (let i = 0; i < node.sources.length; i += 1) { + const source = node.sources[i]; + if (source.sources) { + const state = source.state; + if (state === STALE) { + if (source !== ignore && (!source.updatedAt || source.updatedAt < ExecCount)) + runTop(source); + } else if (state === PENDING) + lookUpstream(source, ignore); + } + } + } + function markDownstream(node) { + for (let i = 0; i < node.observers.length; i += 1) { + const o = node.observers[i]; + if (!o.state) { + o.state = PENDING; + if (o.pure) + Updates.push(o); + else + Effects.push(o); + o.observers && markDownstream(o); + } + } + } + function cleanNode(node) { + let i; + if (node.sources) { + while (node.sources.length) { + const source = node.sources.pop(), index = node.sourceSlots.pop(), obs = source.observers; + if (obs && obs.length) { + const n = obs.pop(), s = source.observerSlots.pop(); + if (index < obs.length) { + n.sourceSlots[s] = index; + obs[index] = n; + source.observerSlots[index] = s; + } + } + } + } + if (node.owned) { + for (i = node.owned.length - 1; i >= 0; i--) + cleanNode(node.owned[i]); + node.owned = null; + } + if (node.cleanups) { + for (i = node.cleanups.length - 1; i >= 0; i--) + node.cleanups[i](); + node.cleanups = null; + } + node.state = 0; + node.context = null; + } + function castError(err) { + if (err instanceof Error) + return err; + return new Error(typeof err === "string" ? err : "Unknown error", { + cause: err + }); + } + function handleError(err, owner = Owner) { + const error = castError(err); + throw error; + } + const FALLBACK = Symbol("fallback"); + function dispose(d) { + for (let i = 0; i < d.length; i++) + d[i](); + } + function mapArray(list, mapFn, options = {}) { + let items = [], mapped = [], disposers = [], len = 0, indexes = mapFn.length > 1 ? [] : null; + onCleanup(() => dispose(disposers)); + return () => { + let newItems = list() || [], i, j; + newItems[$TRACK]; + return untrack(() => { + let newLen = newItems.length, newIndices, newIndicesNext, temp, tempdisposers, tempIndexes, start, end, newEnd, item; + if (newLen === 0) { + if (len !== 0) { + dispose(disposers); + disposers = []; + items = []; + mapped = []; + len = 0; + indexes && (indexes = []); + } + if (options.fallback) { + items = [FALLBACK]; + mapped[0] = createRoot((disposer) => { + disposers[0] = disposer; + return options.fallback(); + }); + len = 1; + } + } else if (len === 0) { + mapped = new Array(newLen); + for (j = 0; j < newLen; j++) { + items[j] = newItems[j]; + mapped[j] = createRoot(mapper); + } + len = newLen; + } else { + temp = new Array(newLen); + tempdisposers = new Array(newLen); + indexes && (tempIndexes = new Array(newLen)); + for (start = 0, end = Math.min(len, newLen); start < end && items[start] === newItems[start]; start++) + ; + for (end = len - 1, newEnd = newLen - 1; end >= start && newEnd >= start && items[end] === newItems[newEnd]; end--, newEnd--) { + temp[newEnd] = mapped[end]; + tempdisposers[newEnd] = disposers[end]; + indexes && (tempIndexes[newEnd] = indexes[end]); + } + newIndices = /* @__PURE__ */ new Map(); + newIndicesNext = new Array(newEnd + 1); + for (j = newEnd; j >= start; j--) { + item = newItems[j]; + i = newIndices.get(item); + newIndicesNext[j] = i === void 0 ? -1 : i; + newIndices.set(item, j); + } + for (i = start; i <= end; i++) { + item = items[i]; + j = newIndices.get(item); + if (j !== void 0 && j !== -1) { + temp[j] = mapped[i]; + tempdisposers[j] = disposers[i]; + indexes && (tempIndexes[j] = indexes[i]); + j = newIndicesNext[j]; + newIndices.set(item, j); + } else + disposers[i](); + } + for (j = start; j < newLen; j++) { + if (j in temp) { + mapped[j] = temp[j]; + disposers[j] = tempdisposers[j]; + if (indexes) { + indexes[j] = tempIndexes[j]; + indexes[j](j); + } + } else + mapped[j] = createRoot(mapper); + } + mapped = mapped.slice(0, len = newLen); + items = newItems.slice(0); + } + return mapped; + }); + function mapper(disposer) { + disposers[j] = disposer; + if (indexes) { + const [s, set] = createSignal(j); + indexes[j] = set; + return mapFn(newItems[j], s); + } + return mapFn(newItems[j]); + } + }; + } + function createComponent(Comp, props) { + return untrack(() => Comp(props || {})); + } + const narrowedError = (name) => `Stale read from <${name}>.`; + function For(props) { + const fallback = "fallback" in props && { + fallback: () => props.fallback + }; + return createMemo(mapArray(() => props.each, props.children, fallback || void 0)); + } + function Show(props) { + const keyed = props.keyed; + const condition = createMemo(() => props.when, void 0, { + equals: (a, b) => keyed ? a === b : !a === !b + }); + return createMemo(() => { + const c = condition(); + if (c) { + const child = props.children; + const fn = typeof child === "function" && child.length > 0; + return fn ? untrack(() => child(keyed ? c : () => { + if (!untrack(condition)) + throw narrowedError("Show"); + return props.when; + })) : child; + } + return props.fallback; + }, void 0, void 0); + } + function reconcileArrays(parentNode, a, b) { + let bLength = b.length, aEnd = a.length, bEnd = bLength, aStart = 0, bStart = 0, after = a[aEnd - 1].nextSibling, map = null; + while (aStart < aEnd || bStart < bEnd) { + if (a[aStart] === b[bStart]) { + aStart++; + bStart++; + continue; + } + while (a[aEnd - 1] === b[bEnd - 1]) { + aEnd--; + bEnd--; + } + if (aEnd === aStart) { + const node = bEnd < bLength ? bStart ? b[bStart - 1].nextSibling : b[bEnd - bStart] : after; + while (bStart < bEnd) + parentNode.insertBefore(b[bStart++], node); + } else if (bEnd === bStart) { + while (aStart < aEnd) { + if (!map || !map.has(a[aStart])) + a[aStart].remove(); + aStart++; + } + } else if (a[aStart] === b[bEnd - 1] && b[bStart] === a[aEnd - 1]) { + const node = a[--aEnd].nextSibling; + parentNode.insertBefore(b[bStart++], a[aStart++].nextSibling); + parentNode.insertBefore(b[--bEnd], node); + a[aEnd] = b[bEnd]; + } else { + if (!map) { + map = /* @__PURE__ */ new Map(); + let i = bStart; + while (i < bEnd) + map.set(b[i], i++); + } + const index = map.get(a[aStart]); + if (index != null) { + if (bStart < index && index < bEnd) { + let i = aStart, sequence = 1, t; + while (++i < aEnd && i < bEnd) { + if ((t = map.get(a[i])) == null || t !== index + sequence) + break; + sequence++; + } + if (sequence > index - bStart) { + const node = a[aStart]; + while (bStart < index) + parentNode.insertBefore(b[bStart++], node); + } else + parentNode.replaceChild(b[bStart++], a[aStart++]); + } else + aStart++; + } else + a[aStart++].remove(); + } + } + } + const $$EVENTS = "_$DX_DELEGATE"; + function template(html, isCE, isSVG) { + let node; + const create = () => { + const t = document.createElement("template"); + t.innerHTML = html; + return isSVG ? t.content.firstChild.firstChild : t.content.firstChild; + }; + const fn = isCE ? () => untrack(() => document.importNode(node || (node = create()), true)) : () => (node || (node = create())).cloneNode(true); + fn.cloneNode = fn; + return fn; + } + function delegateEvents(eventNames, document2 = window.document) { + const e = document2[$$EVENTS] || (document2[$$EVENTS] = /* @__PURE__ */ new Set()); + for (let i = 0, l = eventNames.length; i < l; i++) { + const name = eventNames[i]; + if (!e.has(name)) { + e.add(name); + document2.addEventListener(name, eventHandler); + } + } + } + function setAttribute(node, name, value) { + if (value == null) + node.removeAttribute(name); + else + node.setAttribute(name, value); + } + function addEventListener(node, name, handler, delegate) { + if (delegate) { + if (Array.isArray(handler)) { + node[`$$${name}`] = handler[0]; + node[`$$${name}Data`] = handler[1]; + } else + node[`$$${name}`] = handler; + } else if (Array.isArray(handler)) { + const handlerFn = handler[0]; + node.addEventListener(name, handler[0] = (e) => handlerFn.call(node, handler[1], e)); + } else + node.addEventListener(name, handler); + } + function use(fn, element2, arg) { + return untrack(() => fn(element2, arg)); + } + function insert(parent, accessor, marker, initial) { + if (marker !== void 0 && !initial) + initial = []; + if (typeof accessor !== "function") + return insertExpression(parent, accessor, initial, marker); + createRenderEffect((current) => insertExpression(parent, accessor(), current, marker), initial); + } + function eventHandler(e) { + const key = `$$${e.type}`; + let node = e.composedPath && e.composedPath()[0] || e.target; + if (e.target !== node) { + Object.defineProperty(e, "target", { + configurable: true, + value: node + }); + } + Object.defineProperty(e, "currentTarget", { + configurable: true, + get() { + return node || document; + } + }); + while (node) { + const handler = node[key]; + if (handler && !node.disabled) { + const data = node[`${key}Data`]; + data !== void 0 ? handler.call(node, data, e) : handler.call(node, e); + if (e.cancelBubble) + return; + } + node = node._$host || node.parentNode || node.host; + } + } + function insertExpression(parent, value, current, marker, unwrapArray) { + while (typeof current === "function") + current = current(); + if (value === current) + return current; + const t = typeof value, multi = marker !== void 0; + parent = multi && current[0] && current[0].parentNode || parent; + if (t === "string" || t === "number") { + if (t === "number") + value = value.toString(); + if (multi) { + let node = current[0]; + if (node && node.nodeType === 3) { + node.data = value; + } else + node = document.createTextNode(value); + current = cleanChildren(parent, current, marker, node); + } else { + if (current !== "" && typeof current === "string") { + current = parent.firstChild.data = value; + } else + current = parent.textContent = value; + } + } else if (value == null || t === "boolean") { + current = cleanChildren(parent, current, marker); + } else if (t === "function") { + createRenderEffect(() => { + let v = value(); + while (typeof v === "function") + v = v(); + current = insertExpression(parent, v, current, marker); + }); + return () => current; + } else if (Array.isArray(value)) { + const array = []; + const currentArray = current && Array.isArray(current); + if (normalizeIncomingArray(array, value, current, unwrapArray)) { + createRenderEffect(() => current = insertExpression(parent, array, current, marker, true)); + return () => current; + } + if (array.length === 0) { + current = cleanChildren(parent, current, marker); + if (multi) + return current; + } else if (currentArray) { + if (current.length === 0) { + appendNodes(parent, array, marker); + } else + reconcileArrays(parent, current, array); + } else { + current && cleanChildren(parent); + appendNodes(parent, array); + } + current = array; + } else if (value.nodeType) { + if (Array.isArray(current)) { + if (multi) + return current = cleanChildren(parent, current, marker, value); + cleanChildren(parent, current, null, value); + } else if (current == null || current === "" || !parent.firstChild) { + parent.appendChild(value); + } else + parent.replaceChild(value, parent.firstChild); + current = value; + } else + ; + return current; + } + function normalizeIncomingArray(normalized, array, current, unwrap) { + let dynamic = false; + for (let i = 0, len = array.length; i < len; i++) { + let item = array[i], prev = current && current[i], t; + if (item == null || item === true || item === false) + ; + else if ((t = typeof item) === "object" && item.nodeType) { + normalized.push(item); + } else if (Array.isArray(item)) { + dynamic = normalizeIncomingArray(normalized, item, prev) || dynamic; + } else if (t === "function") { + if (unwrap) { + while (typeof item === "function") + item = item(); + dynamic = normalizeIncomingArray(normalized, Array.isArray(item) ? item : [item], Array.isArray(prev) ? prev : [prev]) || dynamic; + } else { + normalized.push(item); + dynamic = true; + } + } else { + const value = String(item); + if (prev && prev.nodeType === 3 && prev.data === value) + normalized.push(prev); + else + normalized.push(document.createTextNode(value)); + } + } + return dynamic; + } + function appendNodes(parent, array, marker = null) { + for (let i = 0, len = array.length; i < len; i++) + parent.insertBefore(array[i], marker); + } + function cleanChildren(parent, current, marker, replacement) { + if (marker === void 0) + return parent.textContent = ""; + const node = replacement || document.createTextNode(""); + if (current.length) { + let inserted = false; + for (let i = current.length - 1; i >= 0; i--) { + const el = current[i]; + if (node !== el) { + const isParent = el.parentNode === parent; + if (!inserted && !i) + isParent ? parent.replaceChild(node, el) : parent.insertBefore(node, marker); + else + isParent && el.remove(); + } else + inserted = true; + } + } else + parent.insertBefore(node, marker); + return [node]; + } + function click_out_directive(el, accessor) { + const onClick = (e) => { + var _a; + return !el.contains(e.target) && ((_a = accessor()) == null ? void 0 : _a()); + }; + document.body.addEventListener("click", onClick); + onCleanup(() => document.body.removeEventListener("click", onClick)); + } + var _GM_getValue = /* @__PURE__ */ (() => typeof GM_getValue != "undefined" ? GM_getValue : void 0)(); + var _GM_setValue = /* @__PURE__ */ (() => typeof GM_setValue != "undefined" ? GM_setValue : void 0)(); + var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : void 0)(); + const DANBOORU_API = "https://danbooru.donmai.us"; + async function tweet_search(id, username) { + const posts_api = `${DANBOORU_API}/posts.json`; + const tweet_url = `https://twitter.com/${username}/status/${id}`; + const posts_url = new URL(posts_api); + posts_url.searchParams.append("tags", `status:any source:${tweet_url}`); + posts_url.searchParams.append("limit", `5`); + const booru_request = new Promise((res, rej) => { + _GM_xmlhttpRequest({ + url: posts_url.toString(), + method: "GET", + responseType: "json", + anonymous: true, + onload: (response2) => res(response2.responseText), + onerror: (error) => rej(error) + }); + }); + const response = await booru_request; + const posts = JSON.parse(response); + const result = posts.map((post) => { + var _a; + const preview = (_a = post.media_asset.variants) == null ? void 0 : _a.find((v) => v.type == "preview" || v.type == "180x180"); + const source_domain = new URL(post.source).hostname; + return { + id: post.id, + timestamp: post.created_at, + source_domain, + thumbnail: preview == null ? void 0 : preview.url, + img_width: post.image_width, + img_height: post.image_height + }; + }); + return result; + } + async function iqdb_search(media_url) { + const iqdb_api = `${DANBOORU_API}/iqdb_queries.json`; + const iqdb_url = new URL(iqdb_api); + iqdb_url.searchParams.append("url", media_url); + iqdb_url.searchParams.append("similarity", "80"); + iqdb_url.searchParams.append("limit", "5"); + const booru_request = new Promise((res, rej) => { + _GM_xmlhttpRequest({ + url: iqdb_url.toString(), + method: "GET", + responseType: "json", + anonymous: true, + onload: (response2) => res(response2.responseText), + onerror: (error) => rej(error) + }); + }); + const response = await booru_request; + const iqdb_res = JSON.parse(response); + const result = iqdb_res.map((i) => { + var _a; + const post = i.post; + const preview = (_a = post.media_asset.variants) == null ? void 0 : _a.find((v) => v.type == "preview" || v.type == "180x180"); + const source_domain = new URL(post.source).hostname; + return { + id: post.id, + timestamp: post.created_at, + source_domain, + thumbnail: preview == null ? void 0 : preview.url, + img_width: post.image_width, + img_height: post.image_height + }; + }); + return result; + } + async function saucenao_search(media_url) { + const saucenao_key = _GM_getValue("DunkOatmeal_SNKey", ""); + if (!saucenao_key) { + alert("SauceNAO API required. \nGo to Profile Menu > Settings to add it."); + return []; + } + const danbo_bm = 512; + const bitmask = danbo_bm; + const min_similarity = 80; + const sauce_api = `http://saucenao.com/search.php`; + const sauce_url = new URL(sauce_api); + sauce_url.searchParams.append("numres", "5"); + sauce_url.searchParams.append("output_type", "2"); + sauce_url.searchParams.append("dbmask", String(bitmask)); + sauce_url.searchParams.append("api_key", saucenao_key); + sauce_url.searchParams.append("url", media_url); + const booru_request = new Promise((res, rej) => { + _GM_xmlhttpRequest({ + url: sauce_url.toString(), + method: "GET", + responseType: "json", + anonymous: true, + onload: (response2) => res(response2.responseText), + onerror: (error) => rej(error) + }); + }); + const response = await booru_request; + const sauce_res = JSON.parse(response); + const result = sauce_res.results.filter((x) => Number(x.header.similarity) >= min_similarity).map((s) => { + const danbo = s.data; + const preview = s.header.thumbnail; + const source_domain = new URL(danbo.source).hostname; + return { + id: danbo.danbooru_id, + timestamp: "", + source_domain, + thumbnail: preview, + img_width: -1, + img_height: -1 + }; + }); + return result; + } + const _tmpl$$1 = /* @__PURE__ */ template(`
`), _tmpl$2$1 = /* @__PURE__ */ template(`

Old Twitter Image Search Settings`), _tmpl$3$1 = /* @__PURE__ */ template(`
`), _tmpl$4$1 = /* @__PURE__ */ template(` Userscript made by SoberOatmeal `), _tmpl$5$1 = /* @__PURE__ */ template(` SauceNAO API Key `), _tmpl$6$1 = /* @__PURE__ */ template(`Can be acquired by signing up to SauceNAO then go to Account > api and then grab the generated api key.`), _tmpl$7$1 = /* @__PURE__ */ template(`
`); + function OTISSettingsPage() { + const settingsElement = document.getElementById("settings"); + const ot_settings = settingsElement.getElementsByTagName("hr").item(0); + const OTISSettings = () => { + const [SauceKey, setSauceKey] = createSignal(_GM_getValue("DunkOatmeal_SNKey", "")); + return [_tmpl$$1(), _tmpl$2$1(), _tmpl$3$1(), _tmpl$4$1(), _tmpl$3$1(), _tmpl$3$1(), _tmpl$5$1(), _tmpl$3$1(), _tmpl$6$1(), (() => { + const _el$10 = _tmpl$7$1(), _el$11 = _el$10.firstChild; + _el$11.$$input = (i) => _GM_setValue("DunkOatmeal_SNKey", i.target.value); + _el$11.style.setProperty("height", "25px"); + _el$11.style.setProperty("width", "550px"); + createRenderEffect(() => _el$11.value = SauceKey()); + return _el$10; + })(), _tmpl$3$1()]; + }; + ot_settings.before(...createComponent(OTISSettings, {})); + } + delegateEvents(["input"]); + const iqdb_icon = ""; + const sauce_icon = ""; + const grape_icon = ""; + const danbo_icon = ""; + const _tmpl$ = /* @__PURE__ */ template(``), _tmpl$2 = /* @__PURE__ */ template(``), _tmpl$3 = /* @__PURE__ */ template(``), _tmpl$4 = /* @__PURE__ */ template(``), _tmpl$5 = /* @__PURE__ */ template(``), _tmpl$6 = /* @__PURE__ */ template(``), _tmpl$7 = /* @__PURE__ */ template(`