Add artist search

This commit is contained in:
DrunkOatmeal 2023-07-19 18:10:32 +07:00
parent 7b31cd163b
commit a25f0519bb
9 changed files with 353 additions and 1241 deletions

View File

@ -5,6 +5,10 @@ An image search userscript designed specifically for [Old Twiter Layout Extensio
### Installation
Grab a compiled `otis.user.js` file from `/dist` and install it on your userscript manager.
### TODOs
- ~~Manual artist search button~~
- Selective image upload
### 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

1072
dist/otis.user.js vendored
View File

@ -1,7 +1,7 @@
// ==UserScript==
// @name Old Twitter Image Search
// @namespace vite-plugin-monkey
// @version 1.0.0
// @version 1.1.0
// @author SoberOatmeal
// @description otis
// @license MIT
@ -19,1074 +19,12 @@
// @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)}} ");
(i=>{const t=document.createElement("style");t.dataset.source="vite-plugin-monkey",t.textContent=i,document.head.append(t)})(' .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}.danbo-artist-name{font-size:13px;bottom:3px;position:relative;font-weight:700;display:inline-block;cursor:pointer}.danbo-artist-name>a{color:var(--light-gray)}.danbo-artist-banned{color:#f91880!important;text-decoration:line-through}.danbo-artist-name>img:hover{opacity:.8}.danbo-artist-name-tweetpage{display:block;width:fit-content;bottom:unset}.tweet-time:after{color:var(--light-gray)!important;content:"\xB7";margin:0 3px}@keyframes icon-spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}} ');
(function () {
'use strict';
'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(`<hr>`), _tmpl$2$1 = /* @__PURE__ */ template(`<h1>Old Twitter Image Search Settings`), _tmpl$3$1 = /* @__PURE__ */ template(`<br>`), _tmpl$4$1 = /* @__PURE__ */ template(`<span> Userscript made by SoberOatmeal `), _tmpl$5$1 = /* @__PURE__ */ template(`<span> SauceNAO API Key `), _tmpl$6$1 = /* @__PURE__ */ template(`<span>Can be acquired by signing up to SauceNAO then go to <b>Account > api </b> and then grab the generated api key.`), _tmpl$7$1 = /* @__PURE__ */ template(`<div class="setting"><input type="text" placeholder="Put SauceNAO API key here">`);
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(`<img class="otis-icon" title="Tweet Search">`), _tmpl$2 = /* @__PURE__ */ template(`<img class="otis-icon" title="Danbooru IQDB Search">`), _tmpl$3 = /* @__PURE__ */ template(`<img class="otis-icon" title="SauceNAO Search">`), _tmpl$4 = /* @__PURE__ */ template(`<img class="otis-danbo-icon">`), _tmpl$5 = /* @__PURE__ */ template(`<a rel="noopener" target="_blank" class="danbo-text-link">`), _tmpl$6 = /* @__PURE__ */ template(`<span>`), _tmpl$7 = /* @__PURE__ */ template(`<div class="dropdown-menu danbo-more-results">`), _tmpl$8 = /* @__PURE__ */ template(`<span class="danbo-item"><img width="80px"><div><div> #<!> </div><div> <!> </div><div> <!> `);
const clickOutside = click_out_directive;
const GrapeCatIcon = ({
onClick
}) => (() => {
const _el$ = _tmpl$();
addEventListener(_el$, "click", onClick, true);
setAttribute(_el$, "src", grape_icon);
_el$.style.setProperty("width", "20px");
_el$.style.setProperty("margin-bottom", "2px");
return _el$;
})();
const IqdbIcon = ({
onClick
}) => (() => {
const _el$2 = _tmpl$2();
addEventListener(_el$2, "click", onClick, true);
setAttribute(_el$2, "src", iqdb_icon);
return _el$2;
})();
const SauceIcon = ({
onClick
}) => (() => {
const _el$3 = _tmpl$3();
addEventListener(_el$3, "click", onClick, true);
setAttribute(_el$3, "src", sauce_icon);
_el$3.style.setProperty("margin-right", "unset");
return _el$3;
})();
const DanboIcon = ({
onClick
}) => (() => {
const _el$4 = _tmpl$4();
addEventListener(_el$4, "click", onClick, true);
setAttribute(_el$4, "src", danbo_icon);
return _el$4;
})();
async function main() {
const settings_page = document.URL.endsWith("/old/settings");
if (settings_page)
return OTISSettingsPage();
const GrapeSearch = ({
media_url,
id,
username
}) => {
const [showTooltip, setShowTooltip] = createSignal(false);
const [ShowDanboResult, setShowDanboResult] = createSignal(false);
createSignal(0);
const [posts, setPosts] = createSignal([]);
const booru_check = async (e) => {
const ico = e.target;
ico.classList.add("otis-icon-loading");
ico.classList.remove("otis-icon-noresult");
const res = await tweet_search(id, username);
ico.classList.remove("otis-icon-loading");
if (res.length == 0) {
ico.classList.add("otis-icon-noresult");
ico.title = "Can't find any post based on Tweet ID.";
return void 0;
}
setPosts(res);
if (res.length === 1)
setShowDanboResult(true);
else
setShowTooltip(true);
};
const iqdb_check = async (e) => {
const ico = e.target;
ico.classList.add("otis-icon-loading");
ico.classList.remove("otis-icon-noresult");
const res = await iqdb_search(media_url[0]);
ico.classList.remove("otis-icon-loading");
if (res.length == 0) {
ico.classList.add("otis-icon-noresult");
ico.title = "No relevant image search results from IQDB";
return void 0;
}
setPosts(res);
if (res.length === 1)
setShowDanboResult(true);
else
setShowTooltip(true);
};
const sauce_check = async (e) => {
const ico = e.target;
ico.classList.add("otis-icon-loading");
ico.classList.remove("otis-icon-noresult");
const res = await saucenao_search(media_url[0]);
ico.classList.remove("otis-icon-loading");
if (res.length == 0) {
ico.classList.add("otis-icon-noresult");
ico.title = "No relevant image search results from IQDB";
return void 0;
}
setPosts(res);
if (res.length === 1)
setShowDanboResult(true);
else
setShowTooltip(true);
};
const DanboPostsResult = () => [createComponent(DanboIcon, {}), (() => {
const _el$5 = _tmpl$5();
_el$5.$$mouseover = () => setShowTooltip(true);
insert(_el$5, () => {
var _a;
return `#${(_a = posts()[0]) == null ? void 0 : _a.id}`;
});
createRenderEffect(() => {
var _a;
return setAttribute(_el$5, "href", `${DANBOORU_API}/posts/${(_a = posts()[0]) == null ? void 0 : _a.id}`);
});
return _el$5;
})()];
return [(() => {
const _el$6 = _tmpl$6();
insert(_el$6, createComponent(Show, {
get when() {
return !ShowDanboResult();
},
get fallback() {
return createComponent(DanboPostsResult, {});
},
get children() {
return [createComponent(GrapeCatIcon, {
onClick: booru_check
}), createComponent(IqdbIcon, {
onClick: iqdb_check
}), createComponent(SauceIcon, {
onClick: sauce_check
})];
}
}));
return _el$6;
})(), (() => {
const _el$7 = _tmpl$7();
use(clickOutside, _el$7, () => () => setShowTooltip(false));
insert(_el$7, createComponent(For, {
get each() {
return posts();
},
children: (post) => (() => {
const _el$8 = _tmpl$8(), _el$9 = _el$8.firstChild, _el$10 = _el$9.nextSibling, _el$11 = _el$10.firstChild, _el$12 = _el$11.firstChild, _el$14 = _el$12.nextSibling;
_el$14.nextSibling;
const _el$15 = _el$11.nextSibling, _el$16 = _el$15.firstChild, _el$18 = _el$16.nextSibling;
_el$18.nextSibling;
const _el$19 = _el$15.nextSibling, _el$20 = _el$19.firstChild, _el$22 = _el$20.nextSibling;
_el$22.nextSibling;
_el$8.$$click = () => window.open(`${DANBOORU_API}/posts/${post.id}`, "_blank");
insert(_el$11, () => post.id, _el$14);
insert(_el$15, () => post.source_domain, _el$18);
insert(_el$19, () => post.img_width >= 0 ? `${post.img_width} x ${post.img_height}` : "", _el$22);
createRenderEffect(() => setAttribute(_el$9, "src", post.thumbnail));
return _el$8;
})()
}));
createRenderEffect(() => _el$7.hidden = !showTooltip());
return _el$7;
})()];
};
const callback = (mutationList, observer) => {
for (const mutation of mutationList) {
if (mutation.type !== "childList")
return;
mutation.addedNodes.forEach((node) => {
const tweet = node;
const media = tweet.getElementsByClassName("tweet-media").item(0);
if (!media)
return;
const imgs = media.getElementsByTagName("img");
if (!imgs)
return;
const twt_interact = tweet.getElementsByClassName("tweet-interact").item(0);
const twt_avatar = tweet.getElementsByClassName("tweet-avatar-link").item(0);
const twt_child_elipsis = twt_interact == null ? void 0 : twt_interact.getElementsByClassName("tweet-interact-more").item(0);
const tweet_id = tweet.dataset.tweetId || "0";
const tweet_user = twt_avatar.href.split("/").pop() || "*";
let img_urls = [];
for (const img of imgs) {
img_urls.push(img.src);
}
const grape = createComponent(GrapeSearch, {
media_url: img_urls,
id: tweet_id,
username: tweet_user
});
twt_child_elipsis == null ? void 0 : twt_child_elipsis.before(...grape);
});
}
};
const tweet_observer = new MutationObserver(callback);
const timeline = document.getElementById("timeline");
if (!timeline)
return;
const config = {
subtree: false,
childList: true,
attributes: false
};
tweet_observer.observe(timeline, config);
}
const element = document.getElementById("center-cell");
if (element) {
main();
}
delegateEvents(["click", "mouseover"]);
const yA=(A,t)=>A===t,TA=Symbol("solid-track"),E={equals:yA};let VA=uA;const O=1,k=2,rA={owned:null,cleanups:null,context:null,owner:null};var z=null;let Q=null,m=null,p=null,X=null,F=0;function B(A,t){const e=m,n=z,s=A.length===0,o=s?rA:{owned:null,cleanups:null,context:null,owner:t===void 0?n:t},l=s?A:()=>A(()=>U(()=>C(o)));z=o,m=null;try{return W(l,!0)}finally{m=e,z=n;}}function L(A,t){t=t?Object.assign({},E,t):E;const e={value:A,observers:null,observerSlots:null,comparator:t.equals||void 0},n=s=>(typeof s=="function"&&(s=s(e.value)),fA(e,s));return [lA.bind(e),n]}function N(A,t,e){const n=aA(A,t,!1,O);G(n);}function _(A,t,e){e=e?Object.assign({},E,e):E;const n=aA(A,t,!0,0);return n.observers=null,n.observerSlots=null,n.comparator=e.equals||void 0,G(n),lA.bind(n)}function U(A){if(m===null)return A();const t=m;m=null;try{return A()}finally{m=t;}}function oA(A){return z===null||(z.cleanups===null?z.cleanups=[A]:z.cleanups.push(A)),A}function lA(){if(this.sources&&this.state)if(this.state===O)G(this);else {const A=p;p=null,W(()=>I(this),!1),p=A;}if(m){const A=this.observers?this.observers.length:0;m.sources?(m.sources.push(this),m.sourceSlots.push(A)):(m.sources=[this],m.sourceSlots=[A]),this.observers?(this.observers.push(m),this.observerSlots.push(m.sources.length-1)):(this.observers=[m],this.observerSlots=[m.sources.length-1]);}return this.value}function fA(A,t,e){let n=A.value;return (!A.comparator||!A.comparator(n,t))&&(A.value=t,A.observers&&A.observers.length&&W(()=>{for(let s=0;s<A.observers.length;s+=1){const o=A.observers[s],l=Q&&Q.running;l&&Q.disposed.has(o),(l?!o.tState:!o.state)&&(o.pure?p.push(o):X.push(o),o.observers&&vA(o)),l||(o.state=O);}if(p.length>1e6)throw p=[],new Error},!1)),t}function G(A){if(!A.fn)return;C(A);const t=z,e=m,n=F;m=z=A,xA(A,A.value,n),m=e,z=t;}function xA(A,t,e){let n;try{n=A.fn(t);}catch(s){return A.pure&&(A.state=O,A.owned&&A.owned.forEach(C),A.owned=null),A.updatedAt=e+1,PA(s)}(!A.updatedAt||A.updatedAt<=e)&&(A.updatedAt!=null&&"observers"in A?fA(A,n):A.value=n,A.updatedAt=e);}function aA(A,t,e,n=O,s){const o={fn:A,state:n,updatedAt:null,owned:null,sources:null,sourceSlots:null,cleanups:null,value:t,owner:z,context:null,pure:e};return z===null||z!==rA&&(z.owned?z.owned.push(o):z.owned=[o]),o}function cA(A){if(A.state===0)return;if(A.state===k)return I(A);if(A.suspense&&U(A.suspense.inFallback))return A.suspense.effects.push(A);const t=[A];for(;(A=A.owner)&&(!A.updatedAt||A.updatedAt<F);)A.state&&t.push(A);for(let e=t.length-1;e>=0;e--)if(A=t[e],A.state===O)G(A);else if(A.state===k){const n=p;p=null,W(()=>I(A,t[0]),!1),p=n;}}function W(A,t){if(p)return A();let e=!1;t||(p=[]),X?e=!0:X=[],F++;try{const n=A();return wA(e),n}catch(n){e||(X=null),p=null,PA(n);}}function wA(A){if(p&&(uA(p),p=null),A)return;const t=X;X=null,t.length&&W(()=>VA(t),!1);}function uA(A){for(let t=0;t<A.length;t++)cA(A[t]);}function I(A,t){A.state=0;for(let e=0;e<A.sources.length;e+=1){const n=A.sources[e];if(n.sources){const s=n.state;s===O?n!==t&&(!n.updatedAt||n.updatedAt<F)&&cA(n):s===k&&I(n,t);}}}function vA(A){for(let t=0;t<A.observers.length;t+=1){const e=A.observers[t];e.state||(e.state=k,e.pure?p.push(e):X.push(e),e.observers&&vA(e));}}function C(A){let t;if(A.sources)for(;A.sources.length;){const e=A.sources.pop(),n=A.sourceSlots.pop(),s=e.observers;if(s&&s.length){const o=s.pop(),l=e.observerSlots.pop();n<s.length&&(o.sourceSlots[l]=n,s[n]=o,e.observerSlots[n]=l);}}if(A.owned){for(t=A.owned.length-1;t>=0;t--)C(A.owned[t]);A.owned=null;}if(A.cleanups){for(t=A.cleanups.length-1;t>=0;t--)A.cleanups[t]();A.cleanups=null;}A.state=0,A.context=null;}function SA(A){return A instanceof Error?A:new Error(typeof A=="string"?A:"Unknown error",{cause:A})}function PA(A,t=z){throw SA(A)}const NA=Symbol("fallback");function nA(A){for(let t=0;t<A.length;t++)A[t]();}function jA(A,t,e={}){let n=[],s=[],o=[],l=0,i=t.length>1?[]:null;return oA(()=>nA(o)),()=>{let a=A()||[],f,r;return a[TA],U(()=>{let P=a.length,u,v,d,w,V,h,g,y,x;if(P===0)l!==0&&(nA(o),o=[],n=[],s=[],l=0,i&&(i=[])),e.fallback&&(n=[NA],s[0]=B(Y=>(o[0]=Y,e.fallback())),l=1);else if(l===0){for(s=new Array(P),r=0;r<P;r++)n[r]=a[r],s[r]=B(c);l=P;}else {for(d=new Array(P),w=new Array(P),i&&(V=new Array(P)),h=0,g=Math.min(l,P);h<g&&n[h]===a[h];h++);for(g=l-1,y=P-1;g>=h&&y>=h&&n[g]===a[y];g--,y--)d[y]=s[g],w[y]=o[g],i&&(V[y]=i[g]);for(u=new Map,v=new Array(y+1),r=y;r>=h;r--)x=a[r],f=u.get(x),v[r]=f===void 0?-1:f,u.set(x,r);for(f=h;f<=g;f++)x=n[f],r=u.get(x),r!==void 0&&r!==-1?(d[r]=s[f],w[r]=o[f],i&&(V[r]=i[f]),r=v[r],u.set(x,r)):o[f]();for(r=h;r<P;r++)r in d?(s[r]=d[r],o[r]=w[r],i&&(i[r]=V[r],i[r](r))):s[r]=B(c);s=s.slice(0,l=P),n=a.slice(0);}return s});function c(P){if(o[r]=P,i){const[u,v]=L(r);return i[r]=v,t(a[r],u)}return t(a[r])}}}function T(A,t){return U(()=>A(t||{}))}const UA=A=>`Stale read from <${A}>.`;function dA(A){const t="fallback"in A&&{fallback:()=>A.fallback};return _(jA(()=>A.each,A.children,t||void 0))}function mA(A){const t=A.keyed,e=_(()=>A.when,void 0,{equals:(n,s)=>t?n===s:!n==!s});return _(()=>{const n=e();if(n){const s=A.children;return typeof s=="function"&&s.length>0?U(()=>s(t?n:()=>{if(!U(e))throw UA("Show");return A.when})):s}return A.fallback},void 0,void 0)}function XA(A,t,e){let n=e.length,s=t.length,o=n,l=0,i=0,a=t[s-1].nextSibling,f=null;for(;l<s||i<o;){if(t[l]===e[i]){l++,i++;continue}for(;t[s-1]===e[o-1];)s--,o--;if(s===l){const r=o<n?i?e[i-1].nextSibling:e[o-i]:a;for(;i<o;)A.insertBefore(e[i++],r);}else if(o===i)for(;l<s;)(!f||!f.has(t[l]))&&t[l].remove(),l++;else if(t[l]===e[o-1]&&e[i]===t[s-1]){const r=t[--s].nextSibling;A.insertBefore(e[i++],t[l++].nextSibling),A.insertBefore(e[--o],r),t[s]=e[o];}else {if(!f){f=new Map;let c=i;for(;c<o;)f.set(e[c],c++);}const r=f.get(t[l]);if(r!=null)if(i<r&&r<o){let c=l,P=1,u;for(;++c<s&&c<o&&!((u=f.get(t[c]))==null||u!==r+P);)P++;if(P>r-i){const v=t[l];for(;i<r;)A.insertBefore(e[i++],v);}else A.replaceChild(e[i++],t[l++]);}else l++;else t[l++].remove();}}}const sA="_$DX_DELEGATE";function b(A,t,e){let n;const s=()=>{const l=document.createElement("template");return l.innerHTML=A,e?l.content.firstChild.firstChild:l.content.firstChild},o=t?()=>U(()=>document.importNode(n||(n=s()),!0)):()=>(n||(n=s())).cloneNode(!0);return o.cloneNode=o,o}function J(A,t=window.document){const e=t[sA]||(t[sA]=new Set);for(let n=0,s=A.length;n<s;n++){const o=A[n];e.has(o)||(e.add(o),t.addEventListener(o,DA));}}function j(A,t,e){e==null?A.removeAttribute(t):A.setAttribute(t,e);}function q(A,t,e,n){if(n)Array.isArray(e)?(A[`$$${t}`]=e[0],A[`$$${t}Data`]=e[1]):A[`$$${t}`]=e;else if(Array.isArray(e)){const s=e[0];A.addEventListener(t,e[0]=o=>s.call(A,e[1],o));}else A.addEventListener(t,e);}function AA(A,t,e){return U(()=>A(t,e))}function S(A,t,e,n){if(e!==void 0&&!n&&(n=[]),typeof t!="function")return Z(A,t,n,e);N(s=>Z(A,t(),s,e),n);}function DA(A){const t=`$$${A.type}`;let e=A.composedPath&&A.composedPath()[0]||A.target;for(A.target!==e&&Object.defineProperty(A,"target",{configurable:!0,value:e}),Object.defineProperty(A,"currentTarget",{configurable:!0,get(){return e||document}});e;){const n=e[t];if(n&&!e.disabled){const s=e[`${t}Data`];if(s!==void 0?n.call(e,s,A):n.call(e,A),A.cancelBubble)return}e=e._$host||e.parentNode||e.host;}}function Z(A,t,e,n,s){for(;typeof e=="function";)e=e();if(t===e)return e;const o=typeof t,l=n!==void 0;if(A=l&&e[0]&&e[0].parentNode||A,o==="string"||o==="number")if(o==="number"&&(t=t.toString()),l){let i=e[0];i&&i.nodeType===3?i.data=t:i=document.createTextNode(t),e=R(A,e,n,i);}else e!==""&&typeof e=="string"?e=A.firstChild.data=t:e=A.textContent=t;else if(t==null||o==="boolean")e=R(A,e,n);else {if(o==="function")return N(()=>{let i=t();for(;typeof i=="function";)i=i();e=Z(A,i,e,n);}),()=>e;if(Array.isArray(t)){const i=[],a=e&&Array.isArray(e);if($(i,t,e,s))return N(()=>e=Z(A,i,e,n,!0)),()=>e;if(i.length===0){if(e=R(A,e,n),l)return e}else a?e.length===0?iA(A,i,n):XA(A,e,i):(e&&R(A),iA(A,i));e=i;}else if(t.nodeType){if(Array.isArray(e)){if(l)return e=R(A,e,n,t);R(A,e,null,t);}else e==null||e===""||!A.firstChild?A.appendChild(t):A.replaceChild(t,A.firstChild);e=t;}}return e}function $(A,t,e,n){let s=!1;for(let o=0,l=t.length;o<l;o++){let i=t[o],a=e&&e[o],f;if(!(i==null||i===!0||i===!1))if((f=typeof i)=="object"&&i.nodeType)A.push(i);else if(Array.isArray(i))s=$(A,i,a)||s;else if(f==="function")if(n){for(;typeof i=="function";)i=i();s=$(A,Array.isArray(i)?i:[i],Array.isArray(a)?a:[a])||s;}else A.push(i),s=!0;else {const r=String(i);a&&a.nodeType===3&&a.data===r?A.push(a):A.push(document.createTextNode(r));}}return s}function iA(A,t,e=null){for(let n=0,s=t.length;n<s;n++)A.insertBefore(t[n],e);}function R(A,t,e,n){if(e===void 0)return A.textContent="";const s=n||document.createTextNode("");if(t.length){let o=!1;for(let l=t.length-1;l>=0;l--){const i=t[l];if(s!==i){const a=i.parentNode===A;!o&&!l?a?A.replaceChild(s,i):A.insertBefore(s,e):a&&i.remove();}else o=!0;}}else A.insertBefore(s,e);return [s]}var bA=(()=>typeof GM_getValue<"u"?GM_getValue:void 0)(),OA=(()=>typeof GM_setValue<"u"?GM_setValue:void 0)(),K=(()=>typeof GM_xmlhttpRequest<"u"?GM_xmlhttpRequest:void 0)();const YA=b("<hr>"),RA=b("<h1>Old Twitter Image Search Settings"),H=b("<br>"),HA=b("<span> Userscript made by SoberOatmeal "),WA=b("<span> SauceNAO API Key "),MA=b("<span>Can be acquired by signing up to SauceNAO then go to <b>Account > api </b> and then grab the generated api key."),EA=b('<div class="setting"><input type="text" placeholder="Put SauceNAO API key here">');function kA(){const t=document.getElementById("settings").getElementsByTagName("hr").item(0),e=()=>{const[n,s]=L(bA("DunkOatmeal_SNKey",""));return [YA(),RA(),H(),HA(),H(),H(),WA(),H(),MA(),(()=>{const o=EA(),l=o.firstChild;return l.$$input=i=>OA("DunkOatmeal_SNKey",i.target.value),l.style.setProperty("height","25px"),l.style.setProperty("width","550px"),N(()=>l.value=n()),o})(),H()]};t.before(...T(e,{}));}J(["input"]);const D="https://danbooru.donmai.us";async function IA(A,t){const e=`${D}/posts.json`,n=`https://twitter.com/${t}/status/${A}`,s=new URL(e);s.searchParams.append("tags",`status:any source:${n}`),s.searchParams.append("limit","5");const l=await new Promise((f,r)=>{K({url:s.toString(),method:"GET",responseType:"json",anonymous:!0,onload:c=>f(c.responseText),onerror:c=>r(c)});});return JSON.parse(l).map(f=>{var P;const r=(P=f.media_asset.variants)==null?void 0:P.find(u=>u.type=="preview"||u.type=="180x180"),c=new URL(f.source).hostname;return {id:f.id,timestamp:f.created_at,source_domain:c,thumbnail:r==null?void 0:r.url,img_width:f.image_width,img_height:f.image_height}})}async function ZA(A){const t=`${D}/iqdb_queries.json`,e=new URL(t);e.searchParams.append("url",A),e.searchParams.append("similarity","80"),e.searchParams.append("limit","5");const s=await new Promise((i,a)=>{K({url:e.toString(),method:"GET",responseType:"json",anonymous:!0,onload:f=>i(f.responseText),onerror:f=>a(f)});});return JSON.parse(s).map(i=>{var c;const a=i.post,f=(c=a.media_asset.variants)==null?void 0:c.find(P=>P.type=="preview"||P.type=="180x180"),r=new URL(a.source).hostname;return {id:a.id,timestamp:a.created_at,source_domain:r,thumbnail:f==null?void 0:f.url,img_width:a.image_width,img_height:a.image_height}})}async function FA(A){const t=bA("DunkOatmeal_SNKey","");if(!t)return alert(`SauceNAO API required.
Go to Profile Menu > Settings to add it.`),[];const n=512,s=80,o="http://saucenao.com/search.php",l=new URL(o);l.searchParams.append("numres","5"),l.searchParams.append("output_type","2"),l.searchParams.append("dbmask",String(n)),l.searchParams.append("api_key",t),l.searchParams.append("url",A);const a=await new Promise((c,P)=>{K({url:l.toString(),method:"GET",responseType:"json",anonymous:!0,onload:u=>c(u.responseText),onerror:u=>P(u)});});return JSON.parse(a).results.filter(c=>Number(c.header.similarity)>=s).map(c=>{const P=c.data,u=c.header.thumbnail,v=new URL(P.source).hostname;return {id:P.danbooru_id,timestamp:"",source_domain:v,thumbnail:u,img_width:-1,img_height:-1}})}async function GA(A){const t=`${D}/artists.json`,e=`https://twitter.com/${A}`,n=new URL(t);n.searchParams.append("search[url_matches]",e),n.searchParams.append("only","id,name,urls,is_banned");const o=await new Promise((a,f)=>{K({url:n.toString(),method:"GET",responseType:"json",anonymous:!0,onload:r=>a(r.responseText),onerror:r=>f(r)});});let i=JSON.parse(o).shift();return i&&(i.urls=i.urls.filter(a=>a.is_active&&!(a.url.includes("twitter.com/intent/")||a.url.includes("pixiv.net/stacc/")))),i}const CA="",JA="",qA="",KA="",QA=b('<img class="otis-icon" title="Tweet Search">'),BA=b('<img class="otis-icon" title="Danbooru IQDB Search">'),_A=b('<img class="otis-icon" title="SauceNAO Search">'),$A=b('<img class="otis-danbo-icon">'),Ae=({onClick:A})=>(()=>{const t=QA();return q(t,"click",A,!0),j(t,"src",qA),t.style.setProperty("width","20px"),t.style.setProperty("margin-bottom","2px"),t})(),ee=({onClick:A})=>(()=>{const t=BA();return q(t,"click",A,!0),j(t,"src",CA),t})(),te=({onClick:A})=>(()=>{const t=_A();return q(t,"click",A,!0),j(t,"src",JA),t.style.setProperty("margin-right","unset"),t})(),hA=({onClick:A,title:t="",ref:e})=>(()=>{const n=$A(),s=e;return typeof s=="function"?AA(s,n):e=n,q(n,"click",A,!0),j(n,"src",KA),j(n,"title",t),n})();J(["click"]);function zA(A,t){const e=n=>{var s;return !A.contains(n.target)&&((s=t())==null?void 0:s())};document.body.addEventListener("click",e),oA(()=>document.body.removeEventListener("click",e));}const ne=b('<a rel="noopener" target="_blank" class="danbo-text-link">'),se=b("<span>"),ie=b('<div class="dropdown-menu danbo-more-results">'),re=b('<span class="danbo-item"><img width="80px"><div><div> #<!> </div><div> <!> </div><div> <!> '),oe=zA;function le({media_url:A,id:t,username:e}){const[n,s]=L(!1),[o,l]=L(!1);L(0);const[i,a]=L([]),f=async u=>{const v=u.target;v.classList.add("otis-icon-loading"),v.classList.remove("otis-icon-noresult");const d=await IA(t,e);if(v.classList.remove("otis-icon-loading"),d.length==0){v.classList.add("otis-icon-noresult"),v.title="Can't find any post based on Tweet ID.";return}a(d),d.length===1?l(!0):s(!0);},r=async u=>{const v=u.target;v.classList.add("otis-icon-loading"),v.classList.remove("otis-icon-noresult");const d=await ZA(A[0]);if(v.classList.remove("otis-icon-loading"),d.length==0){v.classList.add("otis-icon-noresult"),v.title="No relevant image search results from IQDB";return}a(d),d.length===1?l(!0):s(!0);},c=async u=>{const v=u.target;v.classList.add("otis-icon-loading"),v.classList.remove("otis-icon-noresult");const d=await FA(A[0]);if(v.classList.remove("otis-icon-loading"),d.length==0){v.classList.add("otis-icon-noresult"),v.title="No relevant image search results from IQDB";return}a(d),d.length===1?l(!0):s(!0);},P=()=>[T(hA,{}),(()=>{const u=ne();return u.$$mouseover=()=>s(!0),S(u,()=>{var v;return `#${(v=i()[0])==null?void 0:v.id}`}),N(()=>{var v;return j(u,"href",`${D}/posts/${(v=i()[0])==null?void 0:v.id}`)}),u})()];return [(()=>{const u=se();return S(u,T(mA,{get when(){return !o()},get fallback(){return T(P,{})},get children(){return [T(Ae,{onClick:f}),T(ee,{onClick:r}),T(te,{onClick:c})]}})),u})(),(()=>{const u=ie();return AA(oe,u,()=>()=>s(!1)),S(u,T(dA,{get each(){return i()},children:v=>(()=>{const d=re(),w=d.firstChild,V=w.nextSibling,h=V.firstChild,g=h.firstChild,y=g.nextSibling;y.nextSibling;const x=h.nextSibling,Y=x.firstChild,M=Y.nextSibling;M.nextSibling;const eA=x.nextSibling,gA=eA.firstChild,tA=gA.nextSibling;return tA.nextSibling,d.$$click=()=>window.open(`${D}/posts/${v.id}`,"_blank"),S(h,()=>v.id,y),S(x,()=>v.source_domain,M),S(eA,()=>v.img_width>=0?`${v.img_width} x ${v.img_height}`:"",tA),N(()=>j(w,"src",v.thumbnail)),d})()})),N(()=>u.hidden=!n()),u})()]}J(["mouseover","click"]);const fe=b('<a target="_blank">'),ae=b('<div class="dropdown-menu">'),ce=b('<span class="danbo-artist-name">'),ue=b('<a target="_blank"> <!> '),ve=zA;function Pe({username:A,user_id:t,is_tweet_page:e}){let n;const[s,o]=L(),[l,i]=L(!1),a=async f=>{const r=n;r.classList.add("otis-icon-loading"),r.classList.remove("otis-icon-noresult");const c=await GA(A);if(r.classList.remove("otis-icon-loading"),!c){r.classList.add("otis-icon-noresult"),r.title="No artist tag found on Danbooru";return}o(c),c.urls.length>=1&&i(!0);};return (()=>{const f=ce();return f.$$click=a,f.classList.toggle("danbo-artist-name-tweetpage",!!e),S(f,T(hA,{ref(r){const c=n;typeof c=="function"?c(r):n=r;},title:"Perform manual artist search on Danbooru"}),null),S(f,T(mA,{get when(){return s()},fallback:e?"Artist Search":"",get children(){return [(()=>{const r=fe();return r.$$mouseover=()=>i(!0),S(r,()=>s().name),N(c=>{const P=s().is_banned?`${D}/artists/${s().id}`:`${D}/posts?tags=${s().name}`,u=!!s().is_banned;return P!==c._v$&&j(r,"href",c._v$=P),u!==c._v$2&&r.classList.toggle("danbo-artist-banned",c._v$2=u),c},{_v$:void 0,_v$2:void 0}),r})(),(()=>{const r=ae();return AA(ve,r,()=>()=>i(!1)),r.addEventListener("mouseleave",()=>i(!1)),S(r,T(dA,{get each(){return s().urls},children:({url:c})=>(()=>{const P=ue(),u=P.firstChild,v=u.nextSibling;return v.nextSibling,j(P,"href",c),S(P,c,v),P})()})),N(()=>r.hidden=!l()),r})()]}}),null),f})()}async function de(){if(document.URL.endsWith("/old/settings"))return kA();const t=(o,l)=>{for(const i of o){if(i.type!=="childList")return;i.addedNodes.forEach(a=>{const f=a,r=f.getElementsByClassName("tweet-media").item(0);if(!r)return;const c=r.getElementsByTagName("img");if(!c||r.getElementsByTagName("video").length)return;const u=f.getElementsByClassName("tweet-interact").item(0),v=f.getElementsByClassName("tweet-avatar-link").item(0),d=u==null?void 0:u.getElementsByClassName("tweet-interact-more").item(0),w=f.getElementsByClassName("tweet-time").item(0),V=f.getElementsByClassName("tweet-body-main").item(0),h=f.dataset.tweetId||"0",g=v.href.split("/").pop()||"*";let y=[];for(const M of c)y.push(M.src);const x=()=>T(le,{media_url:y,id:h,username:g}),Y=()=>T(Pe,{username:g,user_id:h,get is_tweet_page(){return !!V}});V?V==null||V.prepend(Y):w==null||w.after(Y()),d==null||d.before(...x());});}},e=new MutationObserver(t),n=document.getElementById("timeline");if(!n)return;const s={subtree:!1,childList:!0,attributes:!1};e.observe(n,s);}const pA=async(A=0)=>A>=5?void 0:document.getElementById("center-cell")?de():(await delay(1e3),pA(A+=1));pA();J(["click","mouseover"]);
})();

View File

@ -1,13 +1,13 @@
{
"name": "otis",
"description": "Old Twitter Image Search",
"version": "1.0.0",
"version": "1.1.0",
"type": "module",
"author": "SoberOatmeal",
"license": "MIT",
"scripts": {
"dev": "vite",
"build": "vite build",
"build": "vite build --mode production",
"preview": "vite preview"
},
"devDependencies": {

134
src/GrapeSearch.tsx Normal file
View File

@ -0,0 +1,134 @@
import { Component, createSignal, For, Show } from "solid-js";
import { DANBOORU_API, booru_post, iqdb_search, saucenao_search, tweet_search } from "./api";
import { DanboIcon, IqdbIcon, SauceIcon, GrapeCatIcon } from './Icons';
import { click_out_directive } from "./helper";
type GrapeSearchParams = {
media_url:string[],
id:string,
username: string
}
// Typescript strips this directive if we don't
// force call it in this module
const clickOutside = click_out_directive
export function GrapeSearch({media_url,id, username}:GrapeSearchParams) {
const [showTooltip, setShowTooltip] = createSignal(false);
const [ShowDanboResult, setShowDanboResult] = createSignal(false);
const [danboPostID, setPostID ] = createSignal(0);
const [posts, setPosts] = createSignal<Array<booru_post>>([])
const booru_check = async (e:MouseEvent) => {
//console.debug({id,username})
//console.debug({event:e});
const ico = e.target as HTMLImageElement;
ico.classList.add('otis-icon-loading');
ico.classList.remove('otis-icon-noresult');
const res = await tweet_search(id,username)
ico.classList.remove('otis-icon-loading');
if ( res.length == 0 ) {
ico.classList.add('otis-icon-noresult');
ico.title = "Can't find any post based on Tweet ID."
return console.debug("grape: none")
}
setPosts(res)
if ( res.length === 1 ) setShowDanboResult(true)
else setShowTooltip(true)
}
const iqdb_check = async (e:MouseEvent) => {
const ico = e.target as HTMLImageElement;
ico.classList.add('otis-icon-loading');
ico.classList.remove('otis-icon-noresult');
// TODO: handle which image to check
const res = await iqdb_search(media_url[0])
ico.classList.remove('otis-icon-loading');
if ( res.length == 0 ) {
ico.classList.add('otis-icon-noresult');
ico.title = "No relevant image search results from IQDB"
return console.debug("iqdb: none");
}
setPosts(res)
if ( res.length === 1 ) setShowDanboResult(true)
else setShowTooltip(true)
}
const sauce_check = async (e:MouseEvent) => {
const ico = e.target as HTMLImageElement;
ico.classList.add('otis-icon-loading');
ico.classList.remove('otis-icon-noresult');
// TODO: handle which image to check
const res = await saucenao_search(media_url[0]);
ico.classList.remove('otis-icon-loading');
if ( res.length == 0 ) {
ico.classList.add('otis-icon-noresult');
ico.title = "No relevant image search results from IQDB"
return console.debug("sauce: none");
}
setPosts(res)
if ( res.length === 1 ) setShowDanboResult(true)
else setShowTooltip(true)
}
const DanboPostsResult = () => (
<>
<DanboIcon />
<a
rel="noopener"
target="_blank"
class={"danbo-text-link"}
href={`${DANBOORU_API}/posts/${posts()[0]?.id}`}
onmouseover={() => setShowTooltip(true)}>
{`#${posts()[0]?.id}`}
</a>
</>
)
return ( <>
<span>
<Show
when={!ShowDanboResult()}
fallback={<DanboPostsResult /> }>
<GrapeCatIcon onClick={booru_check} />
<IqdbIcon onClick={iqdb_check} />
<SauceIcon onClick={sauce_check} />
</Show>
</span>
<div class={"dropdown-menu danbo-more-results"} use:clickOutside={() => setShowTooltip(false)} hidden={!showTooltip()}>
<For each={posts()}>
{ (post) =>
<span
class={"danbo-item"}
onClick={() => window.open(`${DANBOORU_API}/posts/${post.id}`,'_blank')}>
<img src={post.thumbnail} width="80px"/>
<div>
<div> #{post.id} </div>
<div> {post.source_domain} </div>
<div> { post.img_width >= 0 ? `${post.img_width} x ${post.img_height}` : ""} </div>
</div>
</span>
}
</For>
</div>
</>
)
}

9
src/Icons.tsx Normal file
View File

@ -0,0 +1,9 @@
import iqdb_icon from "./assets/iqdb.ico";
import sauce_icon from './assets/saucenao.ico';
import grape_icon from './assets/grapecat.png';
import danbo_icon from './assets/danbo.png';
export const GrapeCatIcon = ({onClick}:any) => <img onClick={onClick} src={grape_icon} class={"otis-icon"} title="Tweet Search" style={{"width":"20px", "margin-bottom":"2px"}} />
export const IqdbIcon = ({onClick}:any) => <img onClick={onClick} src={iqdb_icon } class={"otis-icon"} title="Danbooru IQDB Search"/>
export const SauceIcon = ({onClick}:any) => <img onClick={onClick} src={sauce_icon} class={"otis-icon"} title="SauceNAO Search" style={{"margin-right":"unset"}} />
export const DanboIcon = ({onClick,title="", ref}:any) => <img onClick={onClick} src={danbo_icon} class={"otis-danbo-icon"} title={title} ref={ref}/>

View File

@ -166,3 +166,45 @@ export async function saucenao_search(media_url:string) {
})
return result;
}
export type danbo_artist = {
id: number,
name: string,
is_banned: bool,
urls: Array<danbo_artist_url>
}
type danbo_artist_url = {
url:string,
is_active:bool
}
export async function danbo_artist_search(tweet_username:string) : Promise<danbo_artist> {
const artist_api = `${DANBOORU_API}/artists.json`
const tweet_url = `https://twitter.com/${tweet_username}`;
const artists_url = new URL(artist_api);
artists_url.searchParams.append("search[url_matches]",tweet_url);
artists_url.searchParams.append("only","id,name,urls,is_banned");
const booru_request = new Promise<string>((res,rej) => {
GM_xmlhttpRequest({
url: artists_url.toString(),
method:'GET',
responseType: 'json',
anonymous:true,
onload: (response) => res(response.responseText),
onerror: (error) => rej(error)
})
})
const response = await booru_request
const artists : danbo_artist[] = JSON.parse(response);
let artist = artists.shift();
if ( artist ) artist.urls = artist.urls.filter(x =>
x.is_active &&
!( x.url.includes("twitter.com/intent/") || x.url.includes("pixiv.net/stacc/")));
return artist
}

View File

@ -1,30 +1,82 @@
import { Component, createSignal, For, Show } from "solid-js";
import { click_out_directive } from "./helper";
import { DANBOORU_API, booru_post, iqdb_search, saucenao_search, tweet_search } from "./api";
import { OTISSettingsPage } from "./settings";
import iqdb_icon from "./assets/iqdb.ico";
import sauce_icon from './assets/saucenao.ico';
import grape_icon from './assets/grapecat.png';
import danbo_icon from './assets/danbo.png';
import { GrapeSearch } from "./GrapeSearch";
import "./style.css";
// Typescript strips this directive if we don't
// force call it in this module
import { DanboIcon } from "./Icons";
import { DANBOORU_API, danbo_artist, danbo_artist_search } from "./api";
import { createSignal, Show, For } from "solid-js";
import { click_out_directive } from "./helper";
const clickOutside = click_out_directive
type GrapeSearchParams = {
media_url:string[],
id:string,
username: string
type UserSearchParams = {
username: string,
user_id: string,
is_tweet_page: bool
}
const GrapeCatIcon = ({onClick}:any) => <img onClick={onClick} src={grape_icon} class={"otis-icon"} title="Tweet Search" style={{"width":"20px", "margin-bottom":"2px"}} />
const IqdbIcon = ({onClick}:any) => <img onClick={onClick} src={iqdb_icon } class={"otis-icon"} title="Danbooru IQDB Search" />
const SauceIcon = ({onClick}:any) => <img onClick={onClick} src={sauce_icon} class={"otis-icon"} title="SauceNAO Search" style={{"margin-right":"unset"}} />
const DanboIcon = ({onClick}:any) => <img onClick={onClick} src={danbo_icon} class={"otis-danbo-icon"} />
function ArtistSearch({username,user_id, is_tweet_page}:UserSearchParams) {
//const DanboIcon = ({onClick}:any) => <img onClick={onClick} src={danbo_icon} class={"otis-danbo-icon"} />
let danboIcon;
const [Artist,setArtist] = createSignal<danbo_artist>();
const [ShowUrls, setShowUrls] = createSignal<boolean>(false);
const ArtistSearchClicked = async (e:MouseEvent) => {
const ico = danboIcon as HTMLImageElement;
ico.classList.add('otis-icon-loading');
ico.classList.remove('otis-icon-noresult');
const res = await danbo_artist_search(username)
console.debug(res || "not found")
ico.classList.remove('otis-icon-loading');
if ( !res ) {
ico.classList.add('otis-icon-noresult');
ico.title = "No artist tag found on Danbooru"
return console.debug("artist: none")
}
setArtist(res);
if ( res.urls.length >= 1 ) setShowUrls(true)
}
return (
<span class={"danbo-artist-name"} onClick={ArtistSearchClicked} classList={{"danbo-artist-name-tweetpage": is_tweet_page}}>
<DanboIcon
ref={danboIcon}
title="Perform manual artist search on Danbooru"
/>
<Show
when={Artist()}
fallback={ is_tweet_page ? "Artist Search": ""} >
<a
onmouseover={() => setShowUrls(true)}
href={ !(Artist().is_banned)
? `${DANBOORU_API}/posts?tags=${Artist().name}`
: `${DANBOORU_API}/artists/${Artist().id}`}
classList={{"danbo-artist-banned": Artist().is_banned}}
target="_blank" >
{Artist().name}
</a>
<div
class={"dropdown-menu"}
hidden={!ShowUrls()}
onmouseleave={() => setShowUrls(false)}
use:clickOutside={() => setShowUrls(false)}>
<For each={Artist().urls} >
{ ({url}) =>
<a href={url} target="_blank"> {url} </a>
}
</For>
</div>
</Show>
</span>
)
}
async function main() {
@ -32,141 +84,30 @@ async function main() {
const settings_page = document.URL.endsWith("/old/settings");
if ( settings_page ) return OTISSettingsPage()
const GrapeSearch : Component<GrapeSearchParams> = ({media_url,id, username}) => {
const [showTooltip, setShowTooltip] = createSignal(false);
const [ShowDanboResult, setShowDanboResult] = createSignal(false);
const [danboPostID, setPostID ] = createSignal(0);
const [posts, setPosts] = createSignal<Array<booru_post>>([])
const booru_check = async (e:MouseEvent) => {
//console.debug({id,username})
//console.debug({event:e});
const ico = e.target as HTMLImageElement;
ico.classList.add('otis-icon-loading');
ico.classList.remove('otis-icon-noresult');
const res = await tweet_search(id,username)
ico.classList.remove('otis-icon-loading');
if ( res.length == 0 ) {
ico.classList.add('otis-icon-noresult');
ico.title = "Can't find any post based on Tweet ID."
return console.debug("grape: none")
}
setPosts(res)
if ( res.length === 1 ) setShowDanboResult(true)
else setShowTooltip(true)
}
const iqdb_check = async (e:MouseEvent) => {
const ico = e.target as HTMLImageElement;
ico.classList.add('otis-icon-loading');
ico.classList.remove('otis-icon-noresult');
// TODO: handle which image to check
const res = await iqdb_search(media_url[0])
ico.classList.remove('otis-icon-loading');
if ( res.length == 0 ) {
ico.classList.add('otis-icon-noresult');
ico.title = "No relevant image search results from IQDB"
return console.debug("iqdb: none");
}
setPosts(res)
if ( res.length === 1 ) setShowDanboResult(true)
else setShowTooltip(true)
}
const sauce_check = async (e:MouseEvent) => {
const ico = e.target as HTMLImageElement;
ico.classList.add('otis-icon-loading');
ico.classList.remove('otis-icon-noresult');
// TODO: handle which image to check
const res = await saucenao_search(media_url[0]);
ico.classList.remove('otis-icon-loading');
if ( res.length == 0 ) {
ico.classList.add('otis-icon-noresult');
ico.title = "No relevant image search results from IQDB"
return console.debug("sauce: none");
}
setPosts(res)
if ( res.length === 1 ) setShowDanboResult(true)
else setShowTooltip(true)
}
const DanboPostsResult = () => (
<>
<DanboIcon />
<a
rel="noopener"
target="_blank"
class={"danbo-text-link"}
href={`${DANBOORU_API}/posts/${posts()[0]?.id}`}
onmouseover={() => setShowTooltip(true)}>
{`#${posts()[0]?.id}`}
</a>
</>
)
return ( <>
<span>
<Show
when={!ShowDanboResult()}
fallback={<DanboPostsResult /> }>
<GrapeCatIcon onClick={booru_check} />
<IqdbIcon onClick={iqdb_check} />
<SauceIcon onClick={sauce_check} />
</Show>
</span>
<div class={"dropdown-menu danbo-more-results"} use:clickOutside={() => setShowTooltip(false)} hidden={!showTooltip()}>
<For each={posts()}>
{ (post) =>
<span
class={"danbo-item"}
onClick={() => window.open(`${DANBOORU_API}/posts/${post.id}`,'_blank')}>
<img src={post.thumbnail} width="80px"/>
<div>
<div> #{post.id} </div>
<div> {post.source_domain} </div>
<div> { post.img_width >= 0 ? `${post.img_width} x ${post.img_height}` : ""} </div>
</div>
</span>
}
</For>
</div>
</>
)
}
const callback : MutationCallback = (mutationList, observer) => {
for (const mutation of mutationList) {
if (mutation.type !== "childList") return;
mutation.addedNodes.forEach( (node) => {
const tweet = node as HTMLElement;
const media = tweet.getElementsByClassName('tweet-media').item(0);
const tweet = node as HTMLElement;
const media = tweet.getElementsByClassName('tweet-media').item(0);
if ( !media ) return;
const imgs = media.getElementsByTagName('img');
if ( !imgs ) return;
const vids = media.getElementsByTagName('video');
console.debug(vids);
if ( vids.length ) return;
const twt_interact = tweet.getElementsByClassName('tweet-interact').item(0);
const twt_avatar = tweet.getElementsByClassName('tweet-avatar-link').item(0) as HTMLAnchorElement;
const twt_child_elipsis = twt_interact?.getElementsByClassName('tweet-interact-more').item(0);
const twt_time = tweet.getElementsByClassName('tweet-time').item(0);
const twt_body = tweet.getElementsByClassName('tweet-body-main').item(0);
const tweet_id = tweet.dataset.tweetId || "0"
const tweet_user = twt_avatar.href.split('/').pop() || "*"
@ -179,8 +120,13 @@ async function main() {
img_urls.push(img.src)
}
//console.debug({media:img_urls, id:tweet_id, user:tweet_user});
const grape = <GrapeSearch media_url={img_urls} id={tweet_id} username={tweet_user}/>;
twt_child_elipsis?.before(...grape as any);
const grape = () => <GrapeSearch media_url={img_urls} id={tweet_id} username={tweet_user}/>;
const artist = () => <ArtistSearch username={tweet_user} user_id={tweet_id} is_tweet_page={Boolean(twt_body)}/>
if ( !twt_body ) twt_time?.after(artist());
else twt_body?.prepend(artist);
twt_child_elipsis?.before(...grape());
// TODO: Implement OTIS buttons
})
@ -201,32 +147,33 @@ async function main() {
}
// const init:any = async (limit=0) => {
// /* We need to do this hack because Old Twitter Extension
// would just dump entire Twitter's page including userscripts
// and replace it with OT's custom page instead of replacing
// Twitter's Document Body page
// */
const init:any = async (limit=0) => {
/* We need to do this hack because Old Twitter Extension
would just dump entire Twitter's page including userscripts
and replace it with OT's custom page instead of replacing
Twitter's Document Body page
*/
// if ( limit >= 10 ) return console.error("OTIS: Old Twitter Extension not detected, userscript won't run")
// const element = document.getElementById('center-cell');
// if (element) {
// console.debug("Old Twitter detected");
// return main();
// }
// else {
// console.debug("can't find it");
// await delay(1000);
// return init(limit += 1);
// }
// }
//init()
if ( limit >= 5 ) return console.error("OTIS: Old Twitter Extension not detected, userscript won't run")
const element = document.getElementById('center-cell');
if (element) {
console.debug("Old Twitter detected");
return main();
}
else {
console.debug("can't find it");
await delay(1000);
return init(limit += 1);
}
}
const element = document.getElementById('center-cell');
if (element) {
console.debug("Old Twitter Extension detected");
main();
}
else {
console.debug("can't find it");
}
init()
// const element = document.getElementById('center-cell');
// if (element) {
// console.debug("Old Twitter Extension detected");
// main();
// }
// else {
// console.debug("can't find it");
// }

View File

@ -48,6 +48,40 @@
}
.danbo-artist-name {
font-size:13px;
bottom:3px;
position:relative;
font-weight:700;
display:inline-block;
cursor: pointer;
}
.danbo-artist-name > a {
color:var(--light-gray);
}
.danbo-artist-banned {
color: rgb(249, 24, 128) !important;
text-decoration: line-through;
}
.danbo-artist-name > img:hover {
opacity:0.8;
}
.danbo-artist-name-tweetpage {
display:block;
width: fit-content;
bottom:unset;
}
.tweet-time::after {
color: var(--light-gray) !important;
content: "\00b7";
margin: 0px 3px;
}
@keyframes icon-spin {
from {
transform:rotate(0deg);

View File

@ -3,10 +3,14 @@ import solidPlugin from 'vite-plugin-solid';
import Monkey from 'vite-plugin-monkey';
import { name, description, version } from './package.json'
export default defineConfig((config) => {
export default defineConfig(({mode}) => {
return {
esbuild: {
drop: ['console', 'debugger'],
pure: mode === 'production' ? ['console.debug'] : [],
drop: ['console']
},
build: {
minify: mode === 'production'
},
plugins: [
solidPlugin(),