├── .gitignore ├── CHANGELOG.md ├── README.md ├── build.py ├── config.py ├── dist └── public │ ├── StashDB Userscripts Bundle.user.js │ └── scene.css ├── images └── allow-cors-tamper-monkey.png └── src ├── StashDBUserscriptLibrary.js ├── body ├── StashDB Copy Scene Name.js ├── StashDB Copy StashID.user.js └── StashDB Scene Filter.user.js ├── header ├── StashDB Copy Scene Name.js ├── StashDB Copy StashID.user.js └── StashDB Scene Filter.user.js └── scene.css /.gitignore: -------------------------------------------------------------------------------- 1 | dist/local 2 | __pycache__ 3 | stashdb-userscripts.code-workspace -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Changelog 2 | 3 | ## 0.5.1 4 | * Fixed to work with stashdb.org updates 5 | 6 | ## 0.5.0 7 | * Updated to work with Stash v0.24.0 -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # StashDB Userscripts 2 | 3 | ## [INSTALL USERSCRIPT](dist/public/StashDB%20Userscripts%20Bundle.user.js?raw=1) 4 | 5 | Installation requires a browser extension such as [Violentmonkey](https://violentmonkey.github.io/) / [Tampermonkey](https://www.tampermonkey.net/) / [Greasemonkey](https://www.greasespot.net/). 6 | 7 | > You may remove any unwanted userscripts from the bundle by removing the line that starts with `// @require` that corresponds to the userscript you wish to remove. 8 | 9 | ![Allow cors - Tamper Monkey](images/allow-cors-tamper-monkey.png?raw=true "Allow cors - Tamper Monkey") 10 | 11 | *Known issues: If username/password access is enabled in stash, Firefox + Tampermonkey does not work, but Firefox + Violentmonkey works. Both work in Chrome* 12 | 13 | ## Developing 14 | 15 | Each userscript source is split into two files: 16 | * `src/header` - Folder with userscript metadata blocks 17 | * `src/body` - Folder with main script code 18 | 19 | Execute `py build.py` to combine source files and generate: 20 | * a userscript bundle to `dist\local` for local development 21 | * individual userscripts and a bundle to `dist\public` for release 22 | 23 | Build output directories: 24 | * `dist\local` - A userscript bundle with `@require` headers that load the script code from local files (`src/body`) 25 | * `dist\public` - Userscripts with `@require` headers that load the script code from this github repo -------------------------------------------------------------------------------- /build.py: -------------------------------------------------------------------------------- 1 | import os 2 | import config 3 | import shutil 4 | from pathlib import Path 5 | 6 | def get_active_branch_name(): 7 | head_dir = Path(".") / ".git" / "HEAD" 8 | with head_dir.open("r") as f: content = f.read().splitlines() 9 | 10 | for line in content: 11 | if line[0:4] == "ref:": 12 | return line.partition("refs/heads/")[2] 13 | 14 | def build(): 15 | ROOTDIR = Path(__file__).parent.resolve() 16 | LIBFILE = "StashDBUserscriptLibrary.js" 17 | GIT_BRANCH = get_active_branch_name() 18 | GITHUB_ROOT_URL = config.GITHUB_ROOT_URL.replace('%%BRANCH%%', GIT_BRANCH) 19 | print('git branch', GIT_BRANCH) 20 | 21 | localbodyfiles = [] 22 | distbodyfiles = [] 23 | distlibfile = os.path.join(GITHUB_ROOT_URL, 'src', LIBFILE) 24 | for file in os.listdir('src/header'): 25 | # headerpath = os.path.join('src/header', file) 26 | # bodypath = os.path.join('src/body', file) 27 | # distpublicpath = os.path.join('dist/public', file) 28 | # header = open(headerpath, 'r').read() 29 | # body = open(bodypath, 'r').read() 30 | 31 | localbodyfiles.append("file://" + os.path.join(ROOTDIR, 'src/body', file)) 32 | distbodyfiles.append(os.path.join(GITHUB_ROOT_URL, 'src/body', file)) 33 | 34 | # header = header.replace("%NAMESPACE%", config.NAMESPACE) \ 35 | # .replace("%LIBRARYPATH%", distlibfile) \ 36 | # .replace("%MATCHURL%", f"{config.SERVER_URL}/*") \ 37 | # .replace("// @require %FILEPATH%\n", "") 38 | # distscript = header + "\n\n" + body 39 | # with open(distpublicpath, 'w') as f: 40 | # f.write(distscript) 41 | # print(distpublicpath) 42 | 43 | localpath = 'dist/local/StashDB Userscripts Development Bundle.user.js' 44 | locallibfile = "file://" + os.path.join(ROOTDIR, 'src', LIBFILE) 45 | with open(localpath, 'w') as f: 46 | f.write(f"""// ==UserScript== 47 | // @name StashDB Userscripts Development Bundle 48 | // @namespace {config.NAMESPACE} 49 | // @description StashDB Userscripts Development Bundle 50 | // @version {config.BUNDLE_VERSION} 51 | // @author 7dJx1qP 52 | // @match {config.SERVER_URL}/* 53 | // @resource IMPORTED_CSS file://{os.path.join(ROOTDIR, 'src')}\scene.css 54 | // @grant unsafeWindow 55 | // @grant GM_setClipboard 56 | // @grant GM_getResourceText 57 | // @grant GM_addStyle 58 | // @grant GM.getValue 59 | // @grant GM.setValue 60 | // @grant GM.listValues 61 | // @grant GM.xmlHttpRequest 62 | // @require {locallibfile} 63 | // 64 | // ************************************************************************************************** 65 | // * YOU MAY REMOVE ANY OF THE @require LINES BELOW FOR SCRIPTS YOU DO NOT WANT * 66 | // ************************************************************************************************** 67 | //\n""") 68 | for localbodyfile in localbodyfiles: 69 | f.write(f"// @require {localbodyfile}\n") 70 | f.write("\n// ==/UserScript==\n") 71 | print(localpath) 72 | 73 | distpath = 'dist/public/StashDB Userscripts Bundle.user.js' 74 | with open(distpath, 'w') as f: 75 | f.write(f"""// ==UserScript== 76 | // @name StashDB Userscripts Bundle 77 | // @namespace {config.NAMESPACE} 78 | // @description StashDB Userscripts Bundle 79 | // @version {config.BUNDLE_VERSION} 80 | // @author 7dJx1qP 81 | // @match {config.SERVER_URL}/* 82 | // @resource IMPORTED_CSS https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/{GIT_BRANCH}/dist/public/scene.css 83 | // @grant unsafeWindow 84 | // @grant GM_setClipboard 85 | // @grant GM_getResourceText 86 | // @grant GM_addStyle 87 | // @grant GM.getValue 88 | // @grant GM.setValue 89 | // @grant GM.listValues 90 | // @grant GM.xmlHttpRequest 91 | // @require {distlibfile} 92 | // 93 | // ************************************************************************************************** 94 | // * YOU MAY REMOVE ANY OF THE @require LINES BELOW FOR SCRIPTS YOU DO NOT WANT * 95 | // ************************************************************************************************** 96 | //\n""") 97 | for distbodyfile in distbodyfiles: 98 | f.write(f"// @require {distbodyfile}\n") 99 | f.write("\n// ==/UserScript==\n") 100 | print(distpath) 101 | 102 | shutil.copyfile('src/scene.css', 'dist/public/scene.css') 103 | 104 | build() -------------------------------------------------------------------------------- /config.py: -------------------------------------------------------------------------------- 1 | GITHUB_ROOT_URL = r"https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/%%BRANCH%%/" 2 | BUNDLE_VERSION = "0.5.1" 3 | SERVER_URL = "https://stashdb.org" 4 | NAMESPACE = "https://github.com/7dJx1qP/stashdb-userscripts" -------------------------------------------------------------------------------- /dist/public/StashDB Userscripts Bundle.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name StashDB Userscripts Bundle 3 | // @namespace https://github.com/7dJx1qP/stashdb-userscripts 4 | // @description StashDB Userscripts Bundle 5 | // @version 0.5.1 6 | // @author 7dJx1qP 7 | // @match https://stashdb.org/* 8 | // @resource IMPORTED_CSS https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/master/dist/public/scene.css 9 | // @grant unsafeWindow 10 | // @grant GM_setClipboard 11 | // @grant GM_getResourceText 12 | // @grant GM_addStyle 13 | // @grant GM.getValue 14 | // @grant GM.setValue 15 | // @grant GM.listValues 16 | // @grant GM.xmlHttpRequest 17 | // @require https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/master/src\StashDBUserscriptLibrary.js 18 | // 19 | // ************************************************************************************************** 20 | // * YOU MAY REMOVE ANY OF THE @require LINES BELOW FOR SCRIPTS YOU DO NOT WANT * 21 | // ************************************************************************************************** 22 | // 23 | // @require https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/master/src/body\StashDB Copy Scene Name.js 24 | // @require https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/master/src/body\StashDB Copy StashID.user.js 25 | // @require https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/master/src/body\StashDB Scene Filter.user.js 26 | 27 | // ==/UserScript== 28 | -------------------------------------------------------------------------------- /dist/public/scene.css: -------------------------------------------------------------------------------- 1 | .svg-inline--fa { 2 | display: var(--fa-display, inline-block); 3 | height: 1em; 4 | overflow: visible; 5 | vertical-align: -0.125em; 6 | } 7 | 8 | .nav-link .fa-gear { 9 | width: 24px; 10 | height: 24px; 11 | } 12 | 13 | .stash_id_match a:hover { 14 | background-color: rgba(0, 0, 0, 0.541); 15 | color: #fff !important; 16 | } 17 | 18 | .stash_id_match.search_match { 19 | position: absolute; 20 | top: 10px; 21 | right: 10px; 22 | align-self: center; 23 | } 24 | 25 | .stash_id_match.scene_match { 26 | position: relative; 27 | margin-left: 10px; 28 | cursor: pointer; 29 | align-self: center; 30 | display: inline; 31 | } 32 | 33 | .match-yes { 34 | color: green; 35 | } 36 | 37 | .match-no { 38 | color: red; 39 | } 40 | 41 | .stash_id_ignored .match-no { 42 | color: yellow; 43 | } 44 | 45 | .stash_id_wanted .match-no { 46 | color: gold; 47 | } 48 | 49 | .stash_id_match svg { 50 | height: 24px; 51 | width: 24px; 52 | } 53 | 54 | .stash-performer-link img, 55 | .stash-scene-link img, 56 | .stash-studio-link img { 57 | width: 2rem; 58 | padding-left: 0.5rem; 59 | } 60 | 61 | .scene-performers .stash-performer-link { 62 | padding-right: 0.25rem; 63 | } 64 | 65 | .SearchPage .stash-performer-link img { 66 | width: 2rem; 67 | padding-left: 0rem; 68 | margin-right: 10px; 69 | } 70 | 71 | .stash_id_ignored, 72 | .stash_id_ignored > .card { 73 | background-color: rgba(48, 64, 77, 0.25) !important; 74 | } 75 | 76 | .stash_id_ignored img { 77 | opacity: 0.25; 78 | } 79 | 80 | .settings-box { 81 | padding: 1rem; 82 | margin-bottom: 0; 83 | position: absolute; 84 | right: 0; 85 | z-index: 999; 86 | background-color: inherit; 87 | } -------------------------------------------------------------------------------- /images/allow-cors-tamper-monkey.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/314ac382ddfb7e9a7726424baccfd1ffb0dfeea4/images/allow-cors-tamper-monkey.png -------------------------------------------------------------------------------- /src/StashDBUserscriptLibrary.js: -------------------------------------------------------------------------------- 1 | // StashDB Userscript Library 2 | // Exports utility functions and a StashDB class that emits events whenenever a page navigation change is detected 3 | // version 0.2.0 4 | 5 | (function () { 6 | 'use strict'; 7 | 8 | const css = GM_getResourceText("IMPORTED_CSS"); 9 | GM_addStyle(css); 10 | 11 | const STASH_IMAGE = ''; 12 | 13 | const stashdb = function () { 14 | 15 | const { fetch: originalFetch } = window; 16 | const stashdbListener = new EventTarget(); 17 | 18 | unsafeWindow.fetch = async (...args) => { 19 | let [resource, config ] = args; 20 | // request interceptor here 21 | const response = await originalFetch(resource, config); 22 | // response interceptor here 23 | const contentType = response.headers.get("content-type"); 24 | if (contentType && contentType.indexOf("application/json") !== -1) { 25 | const data = await response.clone().json(); 26 | stashdbListener.dispatchEvent(new CustomEvent('response', { 'detail': data })); 27 | } 28 | return response; 29 | }; 30 | 31 | class Logger { 32 | constructor(enabled) { 33 | this.enabled = enabled; 34 | } 35 | debug() { 36 | if (!this.enabled) return; 37 | console.debug(...arguments); 38 | } 39 | } 40 | 41 | function waitForElementId(elementId, callBack, time) { 42 | time = (typeof time !== 'undefined') ? time : 100; 43 | window.setTimeout(() => { 44 | const element = document.getElementById(elementId); 45 | if (element) { 46 | callBack(elementId, element); 47 | } else { 48 | waitForElementId(elementId, callBack); 49 | } 50 | }, time); 51 | } 52 | 53 | function waitForElementClass(elementId, callBack, time) { 54 | time = (typeof time !== 'undefined') ? time : 100; 55 | window.setTimeout(() => { 56 | const element = document.getElementsByClassName(elementId); 57 | if (element.length > 0) { 58 | callBack(elementId, element); 59 | } else { 60 | waitForElementClass(elementId, callBack); 61 | } 62 | }, time); 63 | } 64 | 65 | function waitForElementByXpath(xpath, callBack, time) { 66 | time = (typeof time !== 'undefined') ? time : 100; 67 | window.setTimeout(() => { 68 | const element = getElementByXpath(xpath); 69 | if (element) { 70 | callBack(xpath, element); 71 | } else { 72 | waitForElementByXpath(xpath, callBack); 73 | } 74 | }, time); 75 | } 76 | 77 | function getElementByXpath(xpath, contextNode) { 78 | return document.evaluate(xpath, contextNode || document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; 79 | } 80 | 81 | function getElementsByXpath(xpath, contextNode) { 82 | return document.evaluate(xpath, contextNode || document, null, XPathResult.ORDERED_NODE_ITERATOR_TYPE, null); 83 | } 84 | 85 | function getClosestAncestor(el, selector, stopSelector) { 86 | let retval = null; 87 | while (el) { 88 | if (el.matches(selector)) { 89 | retval = el; 90 | break 91 | } else if (stopSelector && el.matches(stopSelector)) { 92 | break 93 | } 94 | el = el.parentElement; 95 | } 96 | return retval; 97 | } 98 | 99 | function insertAfter(newNode, existingNode) { 100 | existingNode.parentNode.insertBefore(newNode, existingNode.nextSibling); 101 | } 102 | 103 | function createElementFromHTML(htmlString) { 104 | const div = document.createElement('div'); 105 | div.innerHTML = htmlString.trim(); 106 | 107 | // Change this to div.childNodes to support multiple top-level nodes. 108 | return div.firstChild; 109 | } 110 | 111 | 112 | function setNativeValue(element, value) { 113 | const valueSetter = Object.getOwnPropertyDescriptor(element, 'value').set; 114 | const prototype = Object.getPrototypeOf(element); 115 | const prototypeValueSetter = Object.getOwnPropertyDescriptor(prototype, 'value').set; 116 | 117 | if (valueSetter && valueSetter !== prototypeValueSetter) { 118 | prototypeValueSetter.call(element, value); 119 | } else { 120 | valueSetter.call(element, value); 121 | } 122 | } 123 | 124 | function updateTextInput(element, value) { 125 | setNativeValue(element, value); 126 | element.dispatchEvent(new Event('input', { bubbles: true })); 127 | } 128 | 129 | function concatRegexp(reg, exp) { 130 | let flags = reg.flags + exp.flags; 131 | flags = Array.from(new Set(flags.split(''))).join(); 132 | return new RegExp(reg.source + exp.source, flags); 133 | } 134 | 135 | function sortElementChildren(node) { 136 | const items = node.childNodes; 137 | const itemsArr = []; 138 | for (const i in items) { 139 | if (items[i].nodeType == Node.ELEMENT_NODE) { // get rid of the whitespace text nodes 140 | itemsArr.push(items[i]); 141 | } 142 | } 143 | 144 | itemsArr.sort((a, b) => { 145 | return a.innerHTML == b.innerHTML 146 | ? 0 147 | : (a.innerHTML > b.innerHTML ? 1 : -1); 148 | }); 149 | 150 | for (let i = 0; i < itemsArr.length; i++) { 151 | node.appendChild(itemsArr[i]); 152 | } 153 | } 154 | 155 | function xPathResultToArray(result) { 156 | let node = null; 157 | const nodes = []; 158 | while (node = result.iterateNext()) { 159 | nodes.push(node); 160 | } 161 | return nodes; 162 | } 163 | 164 | const reloadImg = url => 165 | fetch(url, { cache: 'reload', mode: 'no-cors' }) 166 | .then(() => document.body.querySelectorAll(`img[src='${url}']`) 167 | .forEach(img => img.src = url)); 168 | 169 | function isInViewport(element) { 170 | const rect = element.getBoundingClientRect(); 171 | return ( 172 | rect.top >= 0 && 173 | rect.left >= 0 && 174 | rect.bottom <= (window.innerHeight || document.documentElement.clientHeight) && 175 | rect.right <= (window.innerWidth || document.documentElement.clientWidth) 176 | ); 177 | } 178 | 179 | function toHHMMSS(i) { 180 | const sec_num = parseInt(i, 10); // don't forget the second param 181 | const hours = Math.floor(sec_num / 3600); 182 | const minutes = Math.floor((sec_num - (hours * 3600)) / 60); 183 | const seconds = sec_num - (hours * 3600) - (minutes * 60); 184 | 185 | const parts = []; 186 | 187 | if (hours > 0) { 188 | parts.push(hours); 189 | } 190 | parts.push(String(minutes).padStart(2, '0')); 191 | parts.push(String(seconds).padStart(2, '0')); 192 | 193 | return parts.join(':'); 194 | } 195 | 196 | const checkLabel = ''; 197 | const timesLabel = ''; 198 | const clearLabel = ''; 199 | const starLabel = ''; 200 | 201 | class StashDB extends EventTarget { 202 | constructor({ pageUrlCheckInterval = 50, logging = false } = {}) { 203 | super(); 204 | this.stashUrl = 'http://localhost:9999'; 205 | this.loggedIn = false; 206 | this.userName = null; 207 | this.log = new Logger(logging); 208 | this._pageUrlCheckInterval = pageUrlCheckInterval; 209 | this.fireOnHashChangesToo = true; 210 | this.pageURLCheckTimer = setInterval(() => { 211 | // Loop every 500ms 212 | if (this.lastPathStr !== location.pathname || this.lastQueryStr !== location.search || (this.fireOnHashChangesToo && this.lastHashStr !== location.hash)) { 213 | this.lastPathStr = location.pathname; 214 | this.lastQueryStr = location.search; 215 | this.lastHashStr = location.hash; 216 | this.gmMain(); 217 | } 218 | }, this._pageUrlCheckInterval); 219 | stashdbListener.addEventListener('response', (evt) => { 220 | //this.processScenes(evt.detail); 221 | //this.processPerformers(evt.detail); 222 | this.dispatchEvent(new CustomEvent('stashdb:response', { 'detail': evt.detail })); 223 | }); 224 | } 225 | async callGQL(reqData) { 226 | const options = { 227 | method: 'POST', 228 | body: JSON.stringify(reqData), 229 | headers: { 230 | 'Content-Type': 'application/json' 231 | } 232 | } 233 | if (this.stashApiKey) { 234 | options.headers.ApiKey = this.stashApiKey; 235 | } 236 | 237 | return new Promise((resolve, reject) => { 238 | GM.xmlHttpRequest({ 239 | method: "POST", 240 | url: this.stashUrl + '/graphql', 241 | data: JSON.stringify(reqData), 242 | headers: { 243 | "Content-Type": "application/json" 244 | }, 245 | onload: response => { 246 | resolve(JSON.parse(response.response)); 247 | }, 248 | onerror: reject 249 | }); 250 | }); 251 | } 252 | async callStashDbGQL(reqData) { 253 | const options = { 254 | method: 'POST', 255 | body: JSON.stringify(reqData), 256 | headers: { 257 | 'Content-Type': 'application/json' 258 | } 259 | } 260 | 261 | try { 262 | const res = await unsafeWindow.fetch('/graphql', options); 263 | this.log.debug(res); 264 | return res.json(); 265 | } 266 | catch (err) { 267 | console.error(err); 268 | } 269 | } 270 | async findSceneByStashId(id) { 271 | const reqData = { 272 | "variables": { 273 | "scene_filter": { 274 | "stash_id_endpoint": { 275 | "endpoint": "", 276 | "stash_id": id, 277 | "modifier": "EQUALS" 278 | } 279 | } 280 | }, 281 | "query": `query FindSceneByStashId($scene_filter: SceneFilterType) { 282 | findScenes(scene_filter: $scene_filter) { 283 | scenes { 284 | title 285 | stash_ids { 286 | endpoint 287 | stash_id 288 | } 289 | id 290 | } 291 | } 292 | }` 293 | } 294 | return this.callGQL(reqData); 295 | } 296 | async findStudioByStashId(id) { 297 | const reqData = { 298 | "variables": { 299 | "studio_filter": { 300 | "stash_id_endpoint": { 301 | "endpoint": "", 302 | "stash_id": id, 303 | "modifier": "EQUALS" 304 | } 305 | } 306 | }, 307 | "query": `query FindStudioByStashId($studio_filter: StudioFilterType) { 308 | findStudios(studio_filter: $studio_filter) { 309 | studios { 310 | stash_ids { 311 | endpoint 312 | stash_id 313 | } 314 | id 315 | } 316 | } 317 | }` 318 | } 319 | return this.callGQL(reqData); 320 | } 321 | async findPerformerByStashId(id) { 322 | const reqData = { 323 | "variables": { 324 | "performer_filter": { 325 | "stash_id_endpoint": { 326 | "endpoint": "", 327 | "stash_id": id, 328 | "modifier": "EQUALS" 329 | } 330 | } 331 | }, 332 | "query": `query FindPerformers($filter: FindFilterType, $performer_filter: PerformerFilterType) { 333 | findPerformers(filter: $filter, performer_filter: $performer_filter) { 334 | count 335 | performers { 336 | id 337 | } 338 | } 339 | }` 340 | } 341 | return this.callGQL(reqData); 342 | } 343 | /*async processScenes(data) { 344 | if (data?.data?.queryScenes?.scenes) { 345 | return Promise.all(data?.data?.queryScenes?.scenes.map(scene => this.processListScene(scene.id))); 346 | } 347 | }*/ 348 | async processListScene(stashId, sceneEl) { 349 | const data = await this.findSceneByStashId(stashId); 350 | const localId = data?.data?.findScenes?.scenes[0]?.id; 351 | waitForElementByXpath(`//div[@class='card-footer']//a[contains(@href,'${stashId}')]`, async (xpath, el) => { 352 | await this.addSceneMarker(stashId, localId, el.parentElement, sceneEl); 353 | }); 354 | } 355 | async processPageScene(stashId) { 356 | const data = await this.findSceneByStashId(stashId); 357 | const localId = data?.data?.findScenes?.scenes[0]?.id; 358 | waitForElementByXpath(`//div[contains(@class,'scene-info')]/div[@class='card-header']/div[@class='float-end']`, async (xpath, el) => { 359 | await this.addSceneMarker(stashId, localId, el, getClosestAncestor(el, '.scene-info')); 360 | }); 361 | } 362 | async processSearchScene(stashId, sceneEl) { 363 | const data = await this.findSceneByStashId(stashId); 364 | const localId = data?.data?.findScenes?.scenes[0]?.id; 365 | waitForElementByXpath(`//a[contains(@href,'${stashId}')]//h5`, async (xpath, el) => { 366 | el.classList.add('d-flex'); 367 | const markerEl = await this.addSceneMarker(stashId, localId, el, sceneEl); 368 | markerEl.classList.add('ms-auto'); 369 | }); 370 | } 371 | async addSceneMarker(stashId, localId, parentElement, sceneEl) { 372 | let markerEl = parentElement.querySelector('.stash_id_match'); 373 | if (!markerEl) { 374 | let label = localId ? checkLabel : timesLabel; 375 | 376 | const sceneState = JSON.parse(await GM.getValue(stashId, '{"ignored":false,"wanted":false}')); 377 | if (sceneState.ignored) { 378 | sceneEl.classList.add('stash_id_ignored'); 379 | label = clearLabel; 380 | } 381 | else if (sceneState.wanted) { 382 | sceneEl.classList.add('stash_id_wanted'); 383 | label = starLabel; 384 | } 385 | 386 | markerEl = createElementFromHTML(`
${label}
`); 387 | markerEl.classList.add(localId ? 'match-yes' : 'match-no'); 388 | 389 | let dropdownEl; 390 | markerEl.addEventListener('mouseenter', evt => { 391 | dropdownEl = createElementFromHTML(``); 392 | markerEl.appendChild(dropdownEl); 393 | 394 | const rect = document.body.getBoundingClientRect(); 395 | const rect2 = evt.currentTarget.getBoundingClientRect(); 396 | const x = rect2.left;// - rect.left; 397 | const y = rect2.top;// - rect.top; 398 | dropdownEl.style.left = `${x}px`; 399 | dropdownEl.style.top = `${y}px`; 400 | 401 | dropdownEl.addEventListener("click", evt => { 402 | evt.preventDefault(); 403 | evt.stopImmediatePropagation(); 404 | }); 405 | 406 | const menuEl = createElementFromHTML(``); 407 | dropdownEl.appendChild(menuEl); 408 | 409 | if (localId) { 410 | const localLink = this.stashUrl + '/scenes/' + localId; 411 | const gotoSceneEl = createElementFromHTML(`Go to Scene`); 412 | gotoSceneEl.addEventListener("click", evt => { 413 | evt.preventDefault(); 414 | evt.stopImmediatePropagation(); 415 | window.open( 416 | localLink, 417 | '_blank' 418 | ); 419 | }); 420 | menuEl.appendChild(gotoSceneEl); 421 | } 422 | 423 | const ignoreEl = createElementFromHTML(``); 424 | const wishlistEl = createElementFromHTML(`>`); 425 | 426 | if (sceneState.ignored) { 427 | ignoreEl.innerText = 'Clear Ignore'; 428 | 429 | ignoreEl.addEventListener("click", async evt => { 430 | evt.preventDefault(); 431 | evt.stopImmediatePropagation(); 432 | sceneState.ignored = false; 433 | sceneEl.classList.remove('stash_id_ignored'); 434 | markerEl.querySelector('a').innerHTML = localId ? checkLabel : timesLabel; 435 | menuEl.remove(); 436 | await GM.setValue(stashId, JSON.stringify(sceneState)); 437 | }); 438 | 439 | menuEl.append(ignoreEl); 440 | } 441 | else if (sceneState.wanted) { 442 | wishlistEl.innerText = 'Remove From Wishlist'; 443 | 444 | wishlistEl.addEventListener("click", async evt => { 445 | evt.preventDefault(); 446 | evt.stopImmediatePropagation(); 447 | sceneState.wanted = false; 448 | sceneEl.classList.remove('stash_id_wanted'); 449 | markerEl.querySelector('a').innerHTML = localId ? checkLabel : timesLabel; 450 | menuEl.remove(); 451 | await GM.setValue(stashId, JSON.stringify(sceneState)); 452 | }); 453 | 454 | menuEl.append(wishlistEl); 455 | } 456 | else if (!localId) { 457 | wishlistEl.innerText = 'Add to Wishlist'; 458 | ignoreEl.innerText = 'Ignore Scene'; 459 | 460 | wishlistEl.addEventListener("click", async evt => { 461 | evt.preventDefault(); 462 | evt.stopImmediatePropagation(); 463 | sceneState.wanted = true; 464 | sceneEl.classList.add('stash_id_wanted'); 465 | markerEl.querySelector('a').innerHTML = starLabel; 466 | menuEl.remove(); 467 | if (!sceneState.data) { 468 | const data = (await this.findStashboxSceneByStashId(stashId))?.data?.findScene; 469 | const { title = '', release_date = '', duration } = data; 470 | const studioName = data?.studio?.name || ''; 471 | const studioId = data?.studio?.id || ''; 472 | const cover = data?.images[0]?.url; 473 | sceneState.data = { 474 | title, 475 | release_date, 476 | duration, 477 | studioName, 478 | studioId, 479 | cover 480 | } 481 | } 482 | await GM.setValue(stashId, JSON.stringify(sceneState)); 483 | }); 484 | 485 | ignoreEl.addEventListener("click", async evt => { 486 | evt.preventDefault(); 487 | evt.stopImmediatePropagation(); 488 | sceneState.ignored = true; 489 | sceneEl.classList.add('stash_id_ignored'); 490 | markerEl.querySelector('a').innerHTML = clearLabel; 491 | menuEl.remove(); 492 | await GM.setValue(stashId, JSON.stringify(sceneState)); 493 | }); 494 | 495 | menuEl.append(wishlistEl); 496 | menuEl.append(ignoreEl); 497 | } 498 | 499 | if(!isInViewport(menuEl)) { 500 | dropdownEl.style.left = `${x-150}px`; 501 | dropdownEl.style.top = `${y-80}px`; 502 | } 503 | 504 | }); 505 | markerEl.addEventListener('mouseleave', () => { 506 | dropdownEl.remove(); 507 | }); 508 | parentElement.appendChild(markerEl); 509 | } 510 | this.dispatchEvent(new CustomEvent('scenecard', { 'detail': { sceneEl } })); 511 | return markerEl; 512 | } 513 | async createStashPerformerLink(stashId, callback) { 514 | const reqData = { 515 | "variables": { 516 | "performer_filter": { 517 | "stash_id_endpoint": { 518 | "endpoint": "", 519 | "stash_id": stashId, 520 | "modifier": "EQUALS" 521 | } 522 | } 523 | }, 524 | "query": `query FindPerformers($filter: FindFilterType, $performer_filter: PerformerFilterType) { 525 | findPerformers(filter: $filter, performer_filter: $performer_filter) { 526 | count 527 | performers { 528 | id 529 | } 530 | } 531 | }` 532 | } 533 | const results = await this.callGQL(reqData); 534 | if (results.data.findPerformers.count === 0) return; 535 | const performerId = results.data.findPerformers.performers[0].id; 536 | const performerUrl = `${this.stashUrl}/performers/${performerId}`; 537 | const performerLink = document.createElement('a'); 538 | performerLink.classList.add('stash-performer-link'); 539 | performerLink.href = performerUrl; 540 | const stashIcon = document.createElement('img'); 541 | stashIcon.src = STASH_IMAGE; 542 | performerLink.appendChild(stashIcon); 543 | performerLink.setAttribute('target', '_blank'); 544 | callback(performerLink); 545 | } 546 | addStashPerformerLinks() { 547 | if (!document.querySelector('.stash-performer-link')) { 548 | for (const searchPerformer of document.querySelectorAll('div.PerformerCard a')) { 549 | const url = new URL(searchPerformer.href); 550 | const stashId = url.pathname.replace('/performers/', ''); 551 | const searchPerformerHeader = searchPerformer.querySelector('div.card-footer > h5'); 552 | this.createStashPerformerLink(stashId, function (performerLink) { 553 | searchPerformerHeader.appendChild(performerLink); 554 | performerLink.addEventListener('click', function (event) { 555 | event.preventDefault(); 556 | window.open(performerLink.href, '_blank'); 557 | }); 558 | }); 559 | } 560 | } 561 | } 562 | addStashPerformerLink() { 563 | if (!document.querySelector('.stash-performer-link')) { 564 | const header = document.querySelector('.card-header h3'); 565 | const stashId = window.location.pathname.replace('/performers/', ''); 566 | this.createStashPerformerLink(stashId, function (performerLink) { 567 | header.appendChild(performerLink); 568 | }); 569 | } 570 | } 571 | addStashScenePerformerLink() { 572 | if (!document.querySelector('.stash-performer-link')) { 573 | const header = document.querySelector('.scene-performers'); 574 | for (const scenePerformer of document.querySelectorAll('a.scene-performer')) { 575 | const url = new URL(scenePerformer.href); 576 | const stashId = url.pathname.replace('/performers/', ''); 577 | this.createStashPerformerLink(stashId, function (performerLink) { 578 | header.insertBefore(performerLink, scenePerformer); 579 | }); 580 | } 581 | } 582 | } 583 | addStashSearchPerformerLink() { 584 | if (!document.querySelector('.stash-performer-link')) { 585 | for (const searchPerformer of document.querySelectorAll('a.SearchPage-performer')) { 586 | const url = new URL(searchPerformer.href); 587 | const stashId = url.pathname.replace('/performers/', ''); 588 | const searchPerformerHeader = searchPerformer.querySelector('div.card > div.ms-3 > h4 > span'); 589 | this.createStashPerformerLink(stashId, function (performerLink) { 590 | searchPerformerHeader.parentElement.insertBefore(performerLink, searchPerformerHeader); 591 | performerLink.addEventListener('click', function (event) { 592 | event.preventDefault(); 593 | window.open(performerLink.href, '_blank'); 594 | }); 595 | }); 596 | } 597 | } 598 | } 599 | processEdits() { 600 | for (const sceneEdit of document.querySelectorAll('.EditCard')) { 601 | this.processEdit(sceneEdit); 602 | } 603 | } 604 | addStashLink(node, localId, stashType) { 605 | if (!localId) return; 606 | const localLink = `${this.stashUrl}/${stashType}/${localId}`; 607 | const link = document.createElement('a'); 608 | link.classList.add(`stash-${stashType.slice(0, -1)}-link`); 609 | link.href = localLink; 610 | const stashIcon = document.createElement('img'); 611 | stashIcon.src = STASH_IMAGE; 612 | link.appendChild(stashIcon); 613 | link.setAttribute('target', '_blank'); 614 | insertAfter(link, node); 615 | } 616 | processEdit(sceneEdit) { 617 | for (const anchor of sceneEdit.querySelectorAll('a')) { 618 | if (anchor.classList.contains('checked-stash')) continue; 619 | anchor.classList.add('checked-stash'); 620 | const url = new URL(anchor.href); 621 | if (url.pathname.startsWith('/scenes/')) { 622 | const stashId = url.pathname.replace('/scenes/', ''); 623 | this.findSceneByStashId(stashId).then(data => { 624 | const localId = data?.data?.findScenes?.scenes[0]?.id; 625 | this.addStashLink(anchor, localId, 'scenes'); 626 | }); 627 | } 628 | else if (url.pathname.startsWith('/performers/')) { 629 | const stashId = url.pathname.replace('/performers/', ''); 630 | this.findPerformerByStashId(stashId).then(data => { 631 | const localId = data?.data?.findPerformers?.performers[0]?.id; 632 | this.addStashLink(anchor, localId, 'performers'); 633 | }); 634 | } 635 | else if (url.pathname.startsWith('/studios/')) { 636 | const stashId = url.pathname.replace('/studios/', ''); 637 | this.findStudioByStashId(stashId).then(data => { 638 | const localId = data?.data?.findStudios?.studios[0]?.id; 639 | this.addStashLink(anchor, localId, 'studios'); 640 | }); 641 | } 642 | } 643 | } 644 | matchUrl(location, fragment) { 645 | const regexp = concatRegexp(new RegExp(location.origin), fragment); 646 | this.log.debug(regexp, location.href.match(regexp)); 647 | return location.href.match(regexp) != null; 648 | } 649 | gmMain() { 650 | const location = window.location; 651 | this.log.debug(URL, window.location); 652 | 653 | waitForElementByXpath('//div[contains(@class, "align-items-center") and contains(@class, "navbar-nav")]//a', async (xpath, el) => { 654 | this.loggedIn = el.tagName === 'A'; 655 | this.userName = this.loggedIn ? el.innerText : null; 656 | 657 | if (this.loggedIn && !document.querySelector('.settings-box')) { 658 | const gearIcon = ``; 659 | const settingsEl = createElementFromHTML(`${gearIcon}`); 660 | el.parentElement.appendChild(settingsEl); 661 | const settingsMenuEl = createElementFromHTML(``); 676 | settingsEl.appendChild(settingsMenuEl); 677 | 678 | settingsEl.addEventListener('click', evt => { 679 | if (settingsMenuEl.style.display === 'none') { 680 | settingsMenuEl.style.display = 'block'; 681 | } 682 | else { 683 | settingsMenuEl.style.display = 'none'; 684 | } 685 | }); 686 | 687 | settingsMenuEl.addEventListener('click', evt => { 688 | evt.stopPropagation(); 689 | }); 690 | 691 | this.stashUrl = await GM.getValue('stashAddress', 'http://localhost:9999'); 692 | const stashAddress = document.getElementById('address'); 693 | stashAddress.value = this.stashUrl; 694 | stashAddress.addEventListener('change', async () => { 695 | await GM.setValue('stashAddress', stashAddress.value || 'http://localhost:9999'); 696 | }); 697 | 698 | this.stashApiKey = await GM.getValue('stashApiKey', ''); 699 | const stashApiKey = document.getElementById('apiKey'); 700 | stashApiKey.value = this.stashApiKey; 701 | stashApiKey.addEventListener('change', async () => { 702 | await GM.setValue('stashApiKey', stashApiKey.value || ''); 703 | }); 704 | } 705 | 706 | const [_, stashType, stashId, action] = location.pathname.split('/'); 707 | if (location.pathname === '/' || 708 | (stashType === 'scenes' && !stashId) || 709 | (stashType === 'performers' && stashId && !action) || 710 | (stashType === 'studios' && stashId && !action) || 711 | (stashType === 'tags' && stashId && !action) || 712 | (stashType === 'wishlist' && !stashId)) { 713 | waitForElementByXpath('(//div[contains(@class, "HomePage-scenes")]/div[@class="col"]|//div[@class="scenes-list"]/div[@class="row"]/div[@class="col-3"])/div[contains(@class, "SceneCard")]', (xpath, el) => { 714 | const sceneCards = document.querySelectorAll('.row .SceneCard'); 715 | for (const sceneCard of sceneCards) { 716 | const stashId = getElementByXpath("./div[@class='card-footer']//a/@href", sceneCard).value.replace('/scenes/', ''); 717 | this.processListScene(stashId, sceneCard); 718 | } 719 | }); 720 | } 721 | else if (stashType === 'scenes' && stashId && !action) { 722 | this.processPageScene(stashId); 723 | } 724 | else if (stashType === 'search' && stashId && !action) { 725 | waitForElementByXpath('//div[@class="SearchPage"]/div[@class="row"]/div[@class="col-6"]/h3[text()="Scenes"]', (xpath, el) => { 726 | const sceneCards = document.querySelectorAll('.SearchPage-scene'); 727 | for (const sceneCard of sceneCards) { 728 | const stashId = sceneCard.href.split('/').pop(); 729 | this.processSearchScene(stashId, sceneCard); 730 | } 731 | }); 732 | } 733 | 734 | if (stashType === 'performers' && !stashId) { 735 | waitForElementClass('PerformerCard', (className, el) => { 736 | this.addStashPerformerLinks(); 737 | }); 738 | } 739 | if (stashType === 'performers' && stashId && !action) { 740 | waitForElementClass('PerformerInfo', (className, el) => { 741 | this.addStashPerformerLink(); 742 | }); 743 | } 744 | else if (stashType === 'scenes' && stashId && !action) { 745 | waitForElementClass('scene-performers', (className, el) => { 746 | this.addStashScenePerformerLink(); 747 | }); 748 | } 749 | else if (stashType === 'search' && stashId && !action) { 750 | waitForElementClass('SearchPage-performer', (className, el) => { 751 | this.addStashSearchPerformerLink(); 752 | }); 753 | } 754 | else if (stashType === 'users' && stashId && action === 'edits') { 755 | waitForElementByXpath("//div[contains(@class, 'EditCard')]|//h4[text()='No results']", (xpath, el) => { 756 | this.processEdits(); 757 | }); 758 | } 759 | else if (stashType === 'edits') { 760 | if (stashId) { 761 | waitForElementByXpath("//div[contains(@class, 'EditCard')]|//h4[text()='No results']", (xpath, el) => { 762 | this.processEdits(); 763 | }); 764 | } 765 | else { 766 | waitForElementByXpath("//div[contains(@class, 'EditCard')]|//h4[text()='No results']", (xpath, el) => { 767 | this.processEdits(); 768 | }); 769 | } 770 | } 771 | 772 | if (location.pathname === '/') { 773 | this.log.debug('[Navigation] Home Page'); 774 | } 775 | 776 | if (stashType === 'wishlist') { 777 | this.viewWishlist(); 778 | } 779 | else { 780 | if (document.getElementById('wishlist')) { 781 | document.getElementById('wishlist').remove(); 782 | document.getElementById('wishlist-header').remove(); 783 | document.getElementById('wishlist-pagination').remove(); 784 | } 785 | } 786 | 787 | waitForElementByXpath("//div[contains(@class, 'navbar-nav')]", (xpath, el) => { 788 | if (!document.getElementById('nav-wishlist')) { 789 | const navWishlist = createElementFromHTML(`Wishlist`); 790 | el.appendChild(navWishlist); 791 | } 792 | if (stashType === 'wishlist') { 793 | document.getElementById('nav-wishlist').classList.add('active'); 794 | document.getElementById('nav-wishlist').setAttribute('aria-current', 'page'); 795 | } 796 | else { 797 | document.getElementById('nav-wishlist').classList.remove('active'); 798 | document.getElementById('nav-wishlist').removeAttribute('aria-current'); 799 | } 800 | }); 801 | 802 | this.dispatchEvent(new CustomEvent('page', { 'detail': { stashType, stashId, action } })); 803 | }); 804 | } 805 | createPaginationButtons(maxPages, page, paginationEl) { 806 | let start = Math.max(page - 2, 1); 807 | let end = Math.min(start + 4, maxPages); 808 | if (maxPages <= 5) { 809 | start = 1; 810 | end = maxPages; 811 | } 812 | else if (end - start < 4) { 813 | start = Math.max(end - 4, 1); 814 | } 815 | 816 | const paginationButtonHandler = evt => { 817 | const page = getClosestAncestor(evt.target, '.page-item').querySelector('.page-link').dataset.page; 818 | evt.preventDefault(); 819 | evt.stopPropagation(); 820 | let url = window.location.href.split('?')[0]; 821 | if (page > 1) { 822 | url += `?page=${page}`; 823 | } 824 | window.location = url; 825 | } 826 | 827 | let pageEl; 828 | if (start > 1) { 829 | pageEl = createElementFromHTML(`
  • 830 | First 831 |
  • `); 832 | pageEl.addEventListener('click', paginationButtonHandler); 833 | paginationEl.appendChild(pageEl); 834 | } 835 | pageEl = createElementFromHTML(`
  • 836 | 837 | 838 | Previous 839 | 840 |
  • `); 841 | if (page !== 1) { 842 | pageEl.addEventListener('click', paginationButtonHandler); 843 | } 844 | paginationEl.appendChild(pageEl); 845 | const spanCurrent = ` (current)`; 846 | for (let i = start; i <= end; i++) { 847 | pageEl = createElementFromHTML(`
  • 848 | ${i}${page === i ? spanCurrent : ''} 849 |
  • `); 850 | pageEl.addEventListener('click', paginationButtonHandler); 851 | paginationEl.appendChild(pageEl); 852 | } 853 | pageEl = createElementFromHTML(`
  • 854 | 855 | 856 | Next 857 | 858 |
  • `); 859 | if (page !== maxPages) { 860 | pageEl.addEventListener('click', paginationButtonHandler); 861 | } 862 | paginationEl.appendChild(pageEl); 863 | if (end < maxPages) { 864 | pageEl = createElementFromHTML(`
  • 865 | Last 866 |
  • `); 867 | pageEl.addEventListener('click', paginationButtonHandler); 868 | paginationEl.appendChild(pageEl); 869 | } 870 | } 871 | async viewWishlist() { 872 | waitForElementClass('NarrowPage', async (className, el) => { 873 | 874 | const keys = await GM.listValues(); 875 | let stashIds = []; 876 | const sceneStates = {}; 877 | for (const key of keys) { 878 | if (key.startsWith('stash')) continue; 879 | const data = JSON.parse(await GM.getValue(key)); 880 | if (data.wanted) { 881 | stashIds.push(key); 882 | sceneStates[key] = data; 883 | } 884 | } 885 | 886 | el[0].appendChild(createElementFromHTML(`

    Wishlist

    `)); 887 | 888 | const pagination = createElementFromHTML(`
    889 |
    890 | ${stashIds.length.toLocaleString()} results 891 | 893 |
    894 |
    `); 895 | el[0].appendChild(pagination); 896 | 897 | const searchParams = new URLSearchParams(window.location.search); 898 | let page = parseInt(searchParams.get('page') || 1); 899 | const pageSize = 20; 900 | const maxPages = Math.ceil(stashIds.length / pageSize); 901 | if (isNaN(page) || page < 1) page = 1; 902 | if (page > maxPages) page = maxPages; 903 | this.createPaginationButtons(maxPages, page, pagination.querySelector('ul.pagination')); 904 | const startIndex = (page - 1) * pageSize; 905 | const endIndex = page * pageSize; 906 | stashIds = stashIds.slice(startIndex, endIndex); 907 | 908 | const scenesList = createElementFromHTML(`
    `); 909 | el[0].appendChild(scenesList); 910 | const scenesRow = scenesList.firstChild; 911 | 912 | for (const stashId of stashIds) { 913 | const sceneState = sceneStates[stashId]; 914 | let title, release_date, duration, studioName, studioId, cover; 915 | if (sceneState.data) { 916 | const data = sceneState.data; 917 | title = data.title; 918 | release_date = data.release_date; 919 | duration = data.duration; 920 | studioName = data.studioName; 921 | studioId = data.studioId; 922 | cover = data.cover; 923 | } 924 | else { 925 | const data = (await this.findStashboxSceneByStashId(stashId))?.data?.findScene; 926 | title = data.title; 927 | release_date = data.release_date; 928 | duration = data.duration; 929 | studioName = data?.studio?.name || ''; 930 | studioId = data?.studio?.id || ''; 931 | cover = data?.images[0]?.url; 932 | sceneState.data = { 933 | title, 934 | release_date, 935 | duration, 936 | studioName, 937 | studioId, 938 | cover 939 | } 940 | await GM.setValue(stashId, JSON.stringify(sceneState)); 941 | } 942 | const scene = createElementFromHTML(`
    943 |
    944 |
    945 | 946 | 947 | 948 |
    949 | 966 |
    967 |
    `); 968 | scenesRow.appendChild(scene); 969 | } 970 | }); 971 | } 972 | async findStashboxSceneByStashId(stashId) { 973 | const reqData = { 974 | "operationName": "Scene", 975 | "variables": { 976 | "id": stashId 977 | }, 978 | "query": `query Scene($id: ID!) { 979 | findScene(id: $id) { 980 | id 981 | release_date 982 | title 983 | deleted 984 | duration 985 | images { 986 | id 987 | url 988 | width 989 | height 990 | } 991 | studio { 992 | id 993 | name 994 | } 995 | } 996 | }` 997 | }; 998 | return this.callStashDbGQL(reqData); 999 | } 1000 | } 1001 | 1002 | return { 1003 | stashdb: new StashDB({ logging: false }), 1004 | StashDB, 1005 | waitForElementId, 1006 | waitForElementClass, 1007 | waitForElementByXpath, 1008 | getElementByXpath, 1009 | getElementsByXpath, 1010 | getClosestAncestor, 1011 | insertAfter, 1012 | createElementFromHTML, 1013 | setNativeValue, 1014 | updateTextInput, 1015 | sortElementChildren, 1016 | xPathResultToArray, 1017 | reloadImg, 1018 | Logger, 1019 | }; 1020 | }; 1021 | 1022 | if (!unsafeWindow.stashdb) { 1023 | unsafeWindow.stashdb = stashdb(); 1024 | } 1025 | })(); -------------------------------------------------------------------------------- /src/body/StashDB Copy Scene Name.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | 'use strict'; 3 | 4 | const { 5 | stashdb, 6 | StashDB, 7 | waitForElementId, 8 | waitForElementClass, 9 | waitForElementByXpath, 10 | getElementByXpath, 11 | sortElementChildren, 12 | createElementFromHTML, 13 | } = unsafeWindow.stashdb; 14 | 15 | function createTooltipElement() { 16 | const copyTooltip = document.createElement('span'); 17 | copyTooltip.setAttribute('id', 'copy-tooltip'); 18 | copyTooltip.innerText = 'Copied!'; 19 | copyTooltip.classList.add('fade', 'hide'); 20 | copyTooltip.style.position = "absolute"; 21 | copyTooltip.style.left = '0px'; 22 | copyTooltip.style.top = '0px'; 23 | copyTooltip.style.marginLeft = '40px'; 24 | copyTooltip.style.padding = '5px 12px'; 25 | copyTooltip.style.backgroundColor = '#000000df'; 26 | copyTooltip.style.borderRadius = '4px'; 27 | copyTooltip.style.color = '#fff'; 28 | document.body.appendChild(copyTooltip); 29 | return copyTooltip; 30 | } 31 | 32 | function createCopyButton() { 33 | const copyBtn = document.createElement('button'); 34 | copyBtn.setAttribute('id', 'copy-scene-name'); 35 | copyBtn.title = 'Copy to clipboard'; 36 | copyBtn.innerHTML = `Copy Scene Name`; 37 | copyBtn.classList.add('btn', 'btn-secondary', 'btn-sm', 'minimal', 'ml-1'); 38 | copyBtn.addEventListener('click', evt => { 39 | const title = document.querySelector('.card-header > h3').innerText; 40 | const studio = document.querySelector('.card-header > h6 > a').innerText.replaceAll(' ', ''); 41 | const datestring = document.querySelector('.card-header > h6').childNodes[2].nodeValue.replaceAll('-', '.'); 42 | const performers = [...document.querySelectorAll('.scene-performer > svg[data-icon=venus] + span')].map(node => node.innerText); 43 | GM_setClipboard(`[${studio}] ${performers.join(', ')} - ${title} (${datestring})`); 44 | const copyTooltip = createTooltipElement(); 45 | const rect = document.body.getBoundingClientRect(); 46 | const rect2 = evt.currentTarget.getBoundingClientRect(); 47 | const x = rect2.left - rect.left; 48 | const y = rect2.top - rect.top; 49 | copyTooltip.classList.add('show'); 50 | copyTooltip.style.left = `${x}px`; 51 | copyTooltip.style.top = `${y}px`; 52 | setTimeout(() => { 53 | copyTooltip.remove(); 54 | }, 500); 55 | }); 56 | return copyBtn; 57 | } 58 | 59 | stashdb.addEventListener('page', evt => { 60 | const { stashType, stashId, action } = evt.detail; 61 | 62 | waitForElementByXpath("//div[contains(@class, 'card-header')]", (xpath, el) => { 63 | if ((stashType === 'scenes' && stashId && !action)) { 64 | if (!document.getElementById('copy-scene-name')) { 65 | el.appendChild(createCopyButton()); 66 | } 67 | else { 68 | document.getElementById('copy-scene-name').style.display = 'inline-block'; 69 | } 70 | } 71 | else if (document.getElementById('copy-scene-name')) { 72 | document.getElementById('copy-scene-name').style.display = 'none'; 73 | } 74 | }); 75 | 76 | 77 | }); 78 | 79 | 80 | 81 | 82 | })(); -------------------------------------------------------------------------------- /src/body/StashDB Copy StashID.user.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | 'use strict'; 3 | 4 | const { 5 | stashdb, 6 | StashDB, 7 | waitForElementId, 8 | waitForElementClass, 9 | waitForElementByXpath, 10 | getElementByXpath, 11 | sortElementChildren, 12 | createElementFromHTML, 13 | } = unsafeWindow.stashdb; 14 | 15 | function createTooltipElement() { 16 | const copyTooltip = document.createElement('span'); 17 | copyTooltip.setAttribute('id', 'copy-tooltip'); 18 | copyTooltip.innerText = 'Copied!'; 19 | copyTooltip.classList.add('fade', 'hide'); 20 | copyTooltip.style.position = "absolute"; 21 | copyTooltip.style.left = '0px'; 22 | copyTooltip.style.top = '0px'; 23 | copyTooltip.style.marginLeft = '40px'; 24 | copyTooltip.style.padding = '5px 12px'; 25 | copyTooltip.style.backgroundColor = '#000000df'; 26 | copyTooltip.style.borderRadius = '4px'; 27 | copyTooltip.style.color = '#fff'; 28 | document.body.appendChild(copyTooltip); 29 | return copyTooltip; 30 | } 31 | 32 | function createCopyButton() { 33 | const copyBtn = document.createElement('button'); 34 | copyBtn.setAttribute('id', 'copy-stashid'); 35 | copyBtn.title = 'Copy to clipboard'; 36 | copyBtn.innerHTML = `Copy StashID`; 37 | copyBtn.classList.add('btn', 'btn-secondary', 'btn-sm', 'minimal', 'ml-1'); 38 | copyBtn.addEventListener('click', evt => { 39 | GM_setClipboard(window.location.pathname.split('/').pop()); 40 | const copyTooltip = createTooltipElement(); 41 | const rect = document.body.getBoundingClientRect(); 42 | const rect2 = evt.currentTarget.getBoundingClientRect(); 43 | const x = rect2.left - rect.left; 44 | const y = rect2.top - rect.top; 45 | copyTooltip.classList.add('show'); 46 | copyTooltip.style.left = `${x}px`; 47 | copyTooltip.style.top = `${y}px`; 48 | setTimeout(() => { 49 | copyTooltip.remove(); 50 | }, 500); 51 | }); 52 | return copyBtn; 53 | } 54 | 55 | stashdb.addEventListener('page', evt => { 56 | const { stashType, stashId, action } = evt.detail; 57 | 58 | waitForElementByXpath("//div[contains(@class, 'navbar-nav')]", (xpath, el) => { 59 | if ((stashType === 'scenes' && stashId && !action) || 60 | (stashType === 'performers' && stashId && !action) || 61 | (stashType === 'studios' && stashId && !action)) { 62 | if (!document.getElementById('copy-stashid')) { 63 | el.appendChild(createCopyButton()); 64 | } 65 | else { 66 | document.getElementById('copy-stashid').style.display = 'inline-block'; 67 | } 68 | } 69 | else if (document.getElementById('copy-stashid')) { 70 | document.getElementById('copy-stashid').style.display = 'none'; 71 | } 72 | }); 73 | 74 | 75 | }); 76 | 77 | 78 | 79 | 80 | })(); -------------------------------------------------------------------------------- /src/body/StashDB Scene Filter.user.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | 'use strict'; 3 | 4 | const { 5 | stashdb, 6 | StashDB, 7 | waitForElementId, 8 | waitForElementClass, 9 | waitForElementByXpath, 10 | getElementByXpath, 11 | sortElementChildren, 12 | createElementFromHTML, 13 | insertAfter, 14 | getClosestAncestor, 15 | } = unsafeWindow.stashdb; 16 | 17 | function updateVisibility(dropdown) { 18 | for (const sceneCard of document.querySelectorAll('.SceneCard')) { 19 | sceneCard.parentElement.classList.remove('d-none'); 20 | } 21 | if (dropdown.value === 'OWNED') { 22 | for (const node of document.querySelectorAll('.match-no')) { 23 | const sceneCard = getClosestAncestor(node, '.SceneCard'); 24 | sceneCard.parentElement.classList.add('d-none'); 25 | } 26 | } 27 | else if (dropdown.value === 'MISSING') { 28 | for (const node of document.querySelectorAll('.match-yes')) { 29 | const sceneCard = getClosestAncestor(node, '.SceneCard'); 30 | sceneCard.parentElement.classList.add('d-none'); 31 | } 32 | } 33 | } 34 | 35 | stashdb.addEventListener('page', evt => { 36 | const { stashType, stashId, action } = evt.detail; 37 | 38 | waitForElementByXpath("//div[contains(@class, 'navbar-nav')]", (xpath, el) => { 39 | if ((stashType === 'scenes' && !stashId && !action) || 40 | (stashType === 'performers' && stashId && !action) || 41 | (stashType === 'studios' && stashId && !action)) { 42 | waitForElementClass('scene-sort', (className, el) => { 43 | if (!document.querySelector('.visible-filter')) { 44 | const dropdownContainer = createElementFromHTML(`
    45 | 50 |
    `); 51 | insertAfter(dropdownContainer, el[0].parentElement); 52 | 53 | const dropdown = document.querySelector('.visible-filter select'); 54 | dropdown.addEventListener('change', evt => { 55 | updateVisibility(dropdown); 56 | }) 57 | } 58 | }); 59 | } 60 | }); 61 | 62 | 63 | }); 64 | 65 | stashdb.addEventListener('scenecard', evt => { 66 | const { sceneEl } = evt.detail; 67 | const dropdown = document.querySelector('.visible-filter select'); 68 | if (!dropdown) return; 69 | sceneEl.parentElement.classList.remove('d-none'); 70 | if (dropdown.value === 'OWNED' && sceneEl.querySelector('.match-no')) { 71 | sceneEl.parentElement.classList.add('d-none'); 72 | } 73 | else if (dropdown.value === 'MISSING' && sceneEl.querySelector('.match-yes')) { 74 | sceneEl.parentElement.classList.add('d-none'); 75 | } 76 | }); 77 | 78 | 79 | 80 | 81 | })(); -------------------------------------------------------------------------------- /src/header/StashDB Copy Scene Name.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name StashDB Copy Scene Name 3 | // @namespace https://github.com/7dJx1qP/stashdb-userscripts 4 | // @description StashDB Copy Scene Name 5 | // @version 0.2.0 6 | // @author 7dJx1qP 7 | // @match https://stashdb.org/* 8 | // @resource IMPORTED_CSS https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/dist/public/scene.css 9 | // @grant unsafeWindow 10 | // @grant GM_setClipboard 11 | // @grant GM_getResourceText 12 | // @grant GM_addStyle 13 | // @grant GM.getValue 14 | // @grant GM.setValue 15 | // @grant GM.listValues 16 | // @grant GM.xmlHttpRequest 17 | // @require %LIBRARYPATH% 18 | // @require %FILEPATH% 19 | // @run-at document-start 20 | 21 | // ==/UserScript== -------------------------------------------------------------------------------- /src/header/StashDB Copy StashID.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name StashDB Copy StashID 3 | // @namespace https://github.com/7dJx1qP/stashdb-userscripts 4 | // @description StashDB Copy StashID 5 | // @version 0.2.0 6 | // @author 7dJx1qP 7 | // @match https://stashdb.org/* 8 | // @resource IMPORTED_CSS https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/dist/public/scene.css 9 | // @grant unsafeWindow 10 | // @grant GM_setClipboard 11 | // @grant GM_getResourceText 12 | // @grant GM_addStyle 13 | // @grant GM.getValue 14 | // @grant GM.setValue 15 | // @grant GM.listValues 16 | // @grant GM.xmlHttpRequest 17 | // @require %LIBRARYPATH% 18 | // @require %FILEPATH% 19 | // @run-at document-start 20 | 21 | // ==/UserScript== -------------------------------------------------------------------------------- /src/header/StashDB Scene Filter.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name StashDB Scene Filter 3 | // @namespace https://github.com/7dJx1qP/stashdb-userscripts 4 | // @description StashDB Scene Filter 5 | // @version 0.2.0 6 | // @author 7dJx1qP 7 | // @match https://stashdb.org/* 8 | // @resource IMPORTED_CSS https://raw.githubusercontent.com/7dJx1qP/stashdb-userscripts/dist/public/scene.css 9 | // @grant unsafeWindow 10 | // @grant GM_setClipboard 11 | // @grant GM_getResourceText 12 | // @grant GM_addStyle 13 | // @grant GM.getValue 14 | // @grant GM.setValue 15 | // @grant GM.listValues 16 | // @grant GM.xmlHttpRequest 17 | // @require %LIBRARYPATH% 18 | // @require %FILEPATH% 19 | // @run-at document-start 20 | 21 | // ==/UserScript== -------------------------------------------------------------------------------- /src/scene.css: -------------------------------------------------------------------------------- 1 | .svg-inline--fa { 2 | display: var(--fa-display, inline-block); 3 | height: 1em; 4 | overflow: visible; 5 | vertical-align: -0.125em; 6 | } 7 | 8 | .nav-link .fa-gear { 9 | width: 24px; 10 | height: 24px; 11 | } 12 | 13 | .stash_id_match a:hover { 14 | background-color: rgba(0, 0, 0, 0.541); 15 | color: #fff !important; 16 | } 17 | 18 | .stash_id_match.search_match { 19 | position: absolute; 20 | top: 10px; 21 | right: 10px; 22 | align-self: center; 23 | } 24 | 25 | .stash_id_match.scene_match { 26 | position: relative; 27 | margin-left: 10px; 28 | cursor: pointer; 29 | align-self: center; 30 | display: inline; 31 | } 32 | 33 | .match-yes { 34 | color: green; 35 | } 36 | 37 | .match-no { 38 | color: red; 39 | } 40 | 41 | .stash_id_ignored .match-no { 42 | color: yellow; 43 | } 44 | 45 | .stash_id_wanted .match-no { 46 | color: gold; 47 | } 48 | 49 | .stash_id_match svg { 50 | height: 24px; 51 | width: 24px; 52 | } 53 | 54 | .stash-performer-link img, 55 | .stash-scene-link img, 56 | .stash-studio-link img { 57 | width: 2rem; 58 | padding-left: 0.5rem; 59 | } 60 | 61 | .scene-performers .stash-performer-link { 62 | padding-right: 0.25rem; 63 | } 64 | 65 | .SearchPage .stash-performer-link img { 66 | width: 2rem; 67 | padding-left: 0rem; 68 | margin-right: 10px; 69 | } 70 | 71 | .stash_id_ignored, 72 | .stash_id_ignored > .card { 73 | background-color: rgba(48, 64, 77, 0.25) !important; 74 | } 75 | 76 | .stash_id_ignored img { 77 | opacity: 0.25; 78 | } 79 | 80 | .settings-box { 81 | padding: 1rem; 82 | margin-bottom: 0; 83 | position: absolute; 84 | right: 0; 85 | z-index: 999; 86 | background-color: inherit; 87 | } --------------------------------------------------------------------------------