diff --git a/plugins/cjCardTweaks/README.md b/plugins/cjCardTweaks/README.md new file mode 100644 index 00000000..934daf31 --- /dev/null +++ b/plugins/cjCardTweaks/README.md @@ -0,0 +1,19 @@ +# CJ's Card Tweaks +This plugin contains the various tweaks I've made to my Stash cards for anyone who may be interested. Each tweak will be toggleable, so users have the option to subscribe to some without subscribing to all. + +## Tweaks + +### File Count on Cards +![Screenshot 2024-07-24 173921](https://github.com/user-attachments/assets/8eaf0dce-a6c2-4d92-aa78-7ddc2322392a) + +Scenes, Galleries, or Images with a file count greater than one will have a badge similar to the badge present in the file count tab when more than one file is present. This badge will be located at the top right on the card where the studio logo used to live. ATM, the CSS that relocates the studio logo to the left of the title card, is not included as a toggleable tweak, but I plan to extract that out of my SCSS theme project and incorporate it here as a toggleable tweak. Until then, users who aren't using any other plugins that reposition the studio logo can tweak the CSS to reposition the file count to a new location. + +### 3D rating banner +![Screenshot 2024-07-29 131937](https://github.com/user-attachments/assets/64d03cd7-6e31-4373-b831-e99a942216cf) + +Adds an additional dimension to the rating banners. + +### Performer profile cards +![unnamed_2](https://github.com/user-attachments/assets/f505417d-ed0c-40c4-9c78-647081a41307) + +Modify the performer cards to use a traditional profile design diff --git a/plugins/cjCardTweaks/cjCardTweaks.js b/plugins/cjCardTweaks/cjCardTweaks.js new file mode 100644 index 00000000..7c32a10e --- /dev/null +++ b/plugins/cjCardTweaks/cjCardTweaks.js @@ -0,0 +1,272 @@ +(async () => { + "use strict"; + + const userSettings = await csLib.getConfiguration("cjCardTweaks", {}); + const SETTINGS = parseSettings(userSettings ?? ""); + const CARD_KEYS = { + galleries: "gallery", + images: "image", + movies: "movie", + performers: "performer", + scenes: "scene", + studios: "studio", + }; + + const CARDS = Object.entries(CARD_KEYS).reduce((acc, [plural, singular]) => { + acc[singular] = { + class: `${singular}-card`, + data: stash[plural], + isContentCard: ["scene", "gallery", "image"].includes(singular), + }; + return acc; + }, {}); + + function parseSettings(settings) { + return Object.keys(settings).reduce((acc, key) => { + if ( + key === "fileCount" || + key === "addBannerDimension" || + key === "performerProfileCards" + ) { + acc[key] = settings[key]; + } else { + // does nothing for now + } + return acc; + }, {}); + } + + const FILE_COUNT_STYLE = + "span.file-count.badge.badge-pill.badge-info{position: absolute;top: 0.3rem;right: 0.5rem;border-radius: 50%;width: 1.7rem;height: 1.7rem;padding: 5px 8px;font-size: 100%;box-shadow: 1px 3px 4px rgba(0, 0, 0, 0.5)}.grid-card:hover .file-count.badge{opacity: 0;transition: opacity 0.5s}"; + const PERFORMER_PROFILE_CARD_STYLE = + ".performer-card:hover img.performer-card-image{box-shadow: 0 0 0 rgb(0 0 0 / 20%), 0 0 6px rgb(0 0 0 / 90%);transition: box-shadow .5s .5s}.performer-card:hover .thumbnail-section button.btn.favorite-button.not-favorite, .performer-card:hover .thumbnail-section button.btn.favorite-button.favorite{filter: drop-shadow(0 0 2px rgba(0, 0, 0, .9));transition: filter .5s .5s}.performer-card .thumbnail-section button.btn.favorite-button.not-favorite, .performer-card .thumbnail-section button.btn.favorite-button.favorite{top: 10px;filter: drop-shadow(0 2px 2px rgba(0, 0, 0, .9))}.item-list-container .performer-card__age,.recommendation-row .performer-card__age,.item-list-container .performer-card .card-section-title,.recommendation-row .performer-card .card-section-title,.item-list-container .performer-card .thumbnail-section,.recommendation-row .performer-card .thumbnail-section{display: flex;align-content: center;justify-content: center}.item-list-container .performer-card .thumbnail-section a,.recommendation-row .performer-card .thumbnail-section a{display: contents}.item-list-container .performer-card-image,.recommendation-row .performer-card-image{aspect-ratio: 1 / 1;display: flex;object-fit: cover;border: 3px solid var(--plex-yelow);border-radius: 50%;min-width: unset;position: relative;width: 58%;margin: auto;z-index: 1;margin-top: 1.5rem;box-shadow:0 13px 26px rgb(0 0 0 / 20%),0 3px 6px rgb(0 0 0 / 90%);object-position: center}.item-list-container .performer-card hr,.recommendation-row .performer-card hr{width: 90%}.item-list-container .performer-card .fi,.recommendation-row .performer-card .fi{position: absolute;top: 81.5%;left: 69%;border-radius: 50% !important;background-size: cover;margin-left: -1px;height: 1.5rem;width: 1.5rem;z-index: 10;border: solid 2px #252525;box-shadow: unset}.item-list-container .performer-card .card-popovers .btn,.recommendation-row .performer-card .card-popovers .btn{font-size: 0.9rem}"; + const RATING_BANNER_3D_STYLE = + ".grid-card{overflow: unset}.detail-group .rating-banner-3d{display: none}.grid-card:hover .rating-banner-3d{opacity: 0;transition: opacity .5s}.rating-banner{display: none}.rating-banner-3d{height: 110px;left: -6px;overflow: hidden;position: absolute;top: -6px;width: 110px}.rating-banner-3d span{box-shadow: 0 5px 4px rgb(0 0 0 / 50%);position: absolute;display: block;width: 170px;padding: 10px 0;padding-right: 5px;background-color: #ff6a07;color: #fff;font: 700 18px / 1 'Lato', sans-serif;text-shadow: 0 1px 1px rgba(0, 0, 0, .2);text-transform: uppercase;text-align: center;font-size: 1rem;font-weight: 700;letter-spacing: 1px;right: -20px;top: 24px;transform: rotate(-45deg)}.rating-banner-3d::before{top: 0;right: 0;position: absolute;z-index: -1;content: '';display: block;border: 5px solid #a34405;border-top-color: transparent;border-left-color: transparent}.rating-banner-3d::after{bottom: 0;left: 0;position: absolute;z-index: -1;content: '';display: block;border: 5px solid #963e04}"; + + /** + * Element to inject custom CSS styles. + */ + const styleElement = document.createElement("style"); + document.head.appendChild(styleElement); + + if (SETTINGS.fileCount) styleElement.innerHTML += FILE_COUNT_STYLE; + if (SETTINGS.addBannerDimension) + styleElement.innerHTML += RATING_BANNER_3D_STYLE; + if (SETTINGS.performerProfileCards) + styleElement.innerHTML += PERFORMER_PROFILE_CARD_STYLE; + + function createElementFromHTML(htmlString) { + const div = document.createElement("div"); + div.innerHTML = htmlString.trim(); + return div.firstChild; + } + + // Mapping of configuration keys to functions + const cardsHandlers = { + gallery: handleGalleriesCards, + image: handleImagesCards, + movie: handleMoviesCards, + performer: handlePerformersCards, + scene: handleScenesCards, + studio: handleStudiosCards, + }; + + // Handle home cards + handleHomeHotCards(); + + for (const [key, card] of Object.entries(CARDS)) { + if (cardsHandlers[key]) { + cardsHandlers[key](); + } + } + + /** + * Add cards on home page. + */ + function handleHomeHotCards() { + const pattern = /^(\/)?$/; + registerPathChangeListener(pattern, () => { + setTimeout(() => { + for (const card of Object.values(CARDS)) handleCards(card, true); + }, 3000); + }); + } + + /** + * Handles gallery cards to specific paths in Stash. + * + * The supported paths are: + * - /galleries + * - /performers/{id}/galleries + * - /studios/{id}/galleries + * - /tags/{id}/galleries + * - /scenes/{id} + */ + function handleGalleriesCards() { + const pattern = + /^\/(galleries|(performers|studios|tags)\/\d+\/galleries|scenes\/\d+)$/; + tweakCards(pattern, CARDS.gallery); + } + + /** + * Handles image cards to specific paths in Stash. + * + * The supported paths are: + * - /images + * - /performers/{id}/images + * - /studios/{id}/images + * - /tags/{id}/images + * - /galleries/{id} + */ + function handleImagesCards() { + const pattern = + /^\/(images|(performers|studios|tags)\/\d+\/images|galleries\/\d+)$/; + tweakCards(pattern, CARDS.image); + } + + /** + * Handles movie cards to specific paths in Stash. + * + * The supported paths are: + * - /movies + * - /performers/{id}/movies + * - /studios/{id}/movies + * - /tags/{id}/movies + * - /scenes/{id} + */ + function handleMoviesCards() { + const pattern = + /^\/(movies|(performers|studios|tags)\/\d+\/movies|scenes\/\d+)$/; + tweakCards(pattern, CARDS.movie); + } + + /** + * Handles performer cards to specific paths in Stash. + * + * The supported paths are: + * - /performers + * - /performers/{id}/appearswith + * - /studios/{id}/performers + * - /tags/{id}/performers + * - /scenes/{id} + * - /galleries/{id} + * - /images/{id} + */ + function handlePerformersCards() { + const pattern = + /^\/(performers(?:\/\d+\/appearswith)?|(performers|studios|tags)\/\d+\/performers|(scenes|galleries|images)\/\d+)$/; + tweakCards(pattern, CARDS.performer); + } + + /** + * Handles scene cards to specific paths in Stash. + * + * The supported paths are: + * - /scenes + * - /performers/{id}/scenes + * - /studios/{id}/scenes + * - /tags/{id}/scenes + * - /movies/{id} + * - /galleries/{id} + */ + function handleScenesCards() { + const pattern = + /^\/(scenes|(performers|studios|tags|movies)\/\d+\/scenes|(movies|galleries)\/\d+)$/; + tweakCards(pattern, CARDS.scene); + } + + /** + * Handles studio cards to specific paths in Stash. + * + * The supported paths are: + * - /studios + * - /studios/{id}/childstudios + * - /tags/{id}/studios + */ + function handleStudiosCards() { + const pattern = + /^\/(studios|(studios\/\d+\/childstudios)|(tags\/\d+\/studios))$/; + tweakCards(pattern, CARDS.studio); + } + + function tweakCards(pattern, card) { + registerPathChangeListener(pattern, () => { + handleCards(card); + }); + } + + function handleCards(card, isHome = false) { + waitForClass(card.class, () => { + executeTweaks(card.data, card.class, card.isContentCard); + }); + } + + function executeTweaks(stashData, cardClass, isContentCard) { + const cards = document.querySelectorAll(`.${cardClass}`); + + cards.forEach((card) => { + maybeAddFileCount(card, stashData, isContentCard); + maybeAddDimensionToBanner(card); + }); + } + + /** + * Add badge with file count on cards with more than 1 associated file + * + * @param {Element} card - Card element cards list. + * @param {Object} stashData - Data fetched from the GraphQL interceptor. e.g. stash.performers. + * @param {boolean} isContentCard - Flag indicating if card is a content card. + */ + function maybeAddFileCount(card, stashData, isContentCard) { + if (!SETTINGS.fileCount || !isContentCard) return; + + // verify this function was not run twice on the same card for some strange reason + const fileCountBadge = card.querySelector(".file-count"); + if (fileCountBadge) return; + + const link = card.querySelector(".thumbnail-section > a"); + const id = new URL(link.href).pathname.split("/").pop(); + const data = stashData[id]; + + if (!data || data.files.length <= 1) return; + + const el = createElementFromHTML( + `` + + data?.files.length + + `` + ); + link.parentElement.appendChild(el); + } + + /** + * Add additional dimention to rating banner + * + * @param {Element} card - Card element cards list. + */ + function maybeAddDimensionToBanner(card) { + if (!SETTINGS.addBannerDimension) return; + + const oldBanner = card.querySelector(".rating-banner"); + if (!oldBanner) return; + + const link = card.querySelector(".thumbnail-section > a"); + + const rating = oldBanner.textContent; + const color = window.getComputedStyle(oldBanner).backgroundColor; + const colorClass = + oldBanner.className.replace("rating-banner", "").trim() + "-3d"; + + if (!styleElement.innerHTML.includes(colorClass)) { + styleElement.innerHTML += `.${colorClass} span {background-color: ${color};}`; + styleElement.innerHTML += `.rating-banner-3d.${colorClass}:before {border: 5px solid ${color}; filter: brightness(0.9);}`; + styleElement.innerHTML += `.rating-banner-3d.${colorClass}:after {border: 5px solid ${color}; filter: brightness(0.9);}`; + } + const el = createElementFromHTML( + `
` + ); + const span = el.querySelector("span"); + span.style.backgroundColor = color; + link.parentElement.appendChild(el); + oldBanner.remove(); + } +})(); diff --git a/plugins/cjCardTweaks/cjCardTweaks.yml b/plugins/cjCardTweaks/cjCardTweaks.yml new file mode 100644 index 00000000..25fab599 --- /dev/null +++ b/plugins/cjCardTweaks/cjCardTweaks.yml @@ -0,0 +1,26 @@ +name: CJ's Card Tweaks. +description: Provides various tweaks for the Stash Cards. +version: 1.0 +# requires: CommunityScriptsUILibrary +ui: + requires: + - CommunityScriptsUILibrary + javascript: + - https://cdn.jsdelivr.net/gh/HandyRandyx/stash-plugins@main/utils/fetchInterceptor.js + - https://cdn.jsdelivr.net/gh/HandyRandyx/stash-plugins@main/utils/stashHandler.js + - https://cdn.jsdelivr.net/gh/HandyRandyx/stash-plugins@main/utils/registerPathChangeListener.js + - https://cdn.jsdelivr.net/gh/HandyRandyx/stash-plugins@main/utils/waitForClass.js + - cjCardTweaks.js +settings: + addBannerDimension: + displayName: 3D rating banner + description: "Adds additional dimension to the rating banners." + type: BOOLEAN + fileCount: + displayName: Enable for file count + description: "Displays file count on scene, gallery, and image cards." + type: BOOLEAN + performerProfileCards: + displayName: Performer profile cards + description: "Tweaks performer cards to use a traditional profile design." + type: BOOLEAN