Kemono Tweaks & Player

Fetches post title for accuracy, features an expandable title header for long names, and plays .wav/.mp3 files in a feature-rich audio player modal. No need to download WAVs or MP3s to play them anymore.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, Greasemonkey alebo Violentmonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, % alebo Violentmonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, % alebo Violentmonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey alebo Userscripts.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie, ako napríklad Tampermonkey.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie správcu používateľských skriptov.

(Už mám správcu používateľských skriptov, nechajte ma ho nainštalovať!)

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

(Už mám správcu používateľských štýlov, nechajte ma ho nainštalovať!)

// ==UserScript==
// @name         Kemono Tweaks & Player
// @namespace    http://tampermonkey.net/
// @version      3.1
// @description  Fetches post title for accuracy, features an expandable title header for long names, and plays .wav/.mp3 files in a feature-rich audio player modal. No need to download WAVs or MP3s to play them anymore.
// @match        https://kemono.cr/*
// @author       medy17
// @run-at       document-start
// @grant        GM_xmlhttpRequest
// @license      MIT
// ==/UserScript==

(function () {
  "use strict";

  try {
    const style = document.createElement("style");
    style.textContent = `
    :root {
        --c-primary: #3a86ff;
        --c-secondary: #007bff;
        --brilliant-white: #ffffff;
    }
    .post-card { position: relative !important; }
    .post-card__header { padding: 5px !important; z-index: 1 !important; color: #fff !important; white-space: nowrap !important; overflow: hidden !important; text-overflow: ellipsis !important; max-width: 100% !important; display: block !important; position: relative !important; }
    .post-card:hover .post-card__header { white-space: normal !important; overflow: visible !important; background: #2e1905 !important; color: #fff !important; padding: 4px 6px !important; z-index: 9999 !important; position: absolute !important; width: auto !important; max-width: 300px !important; border-radius: 6px !important; }

    .modal-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.7); z-index: 10000; display: flex; align-items: center; justify-content: center; opacity: 0; transition: opacity 0.3s ease; backdrop-filter: blur(8px); -webkit-backdrop-filter: blur(8px); }
    .modal-overlay.show { opacity: 1; }
    #audio-player-container { position: relative; z-index: 1; background-color: #1e1e1e; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif; border-radius: 16px; overflow: hidden; box-shadow: 0 20px 60px rgba(0, 0, 0, 0.4); border: 1px solid rgba(255, 255, 255, 0.1); width: 90%; max-width: 600px; transform: scale(0.95) translateY(20px); transition: transform 0.3s cubic-bezier(0.4, 0, 0.2, 1); padding: 25px; }
    .modal-overlay.show #audio-player-container { transform: scale(1) translateY(0); }
    #audio-element { display: none; }
    #close-audio-btn { position: absolute; top: 1rem; right: 1rem; z-index: 20; background: rgba(24, 24, 27, 0.5); backdrop-filter: blur(4px); border: none; font-size: 1.5rem; color: var(--brilliant-white); cursor: pointer; padding: 0.5rem; border-radius: 50%; width: 40px; height: 40px; display: flex; align-items: center; justify-content: center; transition: opacity 0.3s ease, background 0.2s ease, transform 0.2s ease; }
    #close-audio-btn:hover { background: linear-gradient(135deg, rgba(245, 169, 184, 0.15), rgba(91, 206, 250, 0.1)), rgba(24, 24, 27, 0.7); background-blend-mode: overlay; color: var(--c-primary); transform: rotate(90deg); }
    .audio-title-container { display: flex; align-items: center; gap: 10px; cursor: pointer; user-select: none; margin: 0 40px 20px 0; }
    .audio-title { flex: 1; min-width: 0; font-size: 1.3em; font-weight: 500; color: #fff; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
    .expand-arrow { flex-shrink: 0; font-size: 1.2em; color: #aaa; transition: transform 0.3s ease; }
    .audio-title-container.expanded .audio-title { white-space: normal; overflow-wrap: break-word; }
    .audio-title-container.expanded .expand-arrow { transform: rotate(180deg); }
    .audio-controls-container { color: #fff; user-select: none; }
    .audio-controls-container button { background: none; border: none; color: #fff; padding: 0; cursor: pointer; opacity: 0.85; transition: opacity 0.2s, transform 0.2s; }
    .audio-controls-container button:hover { opacity: 1; transform: scale(1.1); }
    .audio-controls-container button svg { width: 24px; height: 24px; display: block; stroke-width: 2; }
    .controls { display: flex; justify-content: space-between; align-items: center; padding: 0.5rem 0.2rem; }
    .controls-left, .controls-right { display: flex; align-items: center; gap: 0.75rem; }
    .play-pause-btn .play-icon { display: block; fill: currentColor; }
    .play-pause-btn .pause-icon { display: none; fill: currentColor; }
    #audio-player-container:not(.paused) .play-pause-btn .play-icon { display: none; }
    #audio-player-container:not(.paused) .play-pause-btn .pause-icon { display: block; }
    .volume-container { display: flex; align-items: center; }
    .volume-container:hover .volume-slider { width: 80px; transform: scaleX(1); }
    .volume-slider { width: 0; transform: scaleX(0); transform-origin: left; transition: width 0.2s ease-in-out, transform 0.2s ease-in-out; cursor: pointer; }
    .volume-btn svg { stroke-width: 0; fill: currentColor; }
    .volume-btn .low-volume-icon, .volume-btn .muted-icon { display: none; }
    .time-container { font-size: 0.9rem; min-width: 95px; white-space: nowrap; text-align: center; padding: 0 0.25rem; }
    .timeline-container { padding: 0.5rem 0.2rem; cursor: pointer; }
    .timeline-container:hover .timeline { height: 8px; }
    .timeline-container:hover .timeline .progress-bar::after { transform: translateY(-50%) scale(1); }
    .timeline { height: 5px; width: 100%; background-color: rgba(255, 255, 255, 0.3); border-radius: 3px; position: relative; }
    .timeline .progress-bar { height: 100%; width: 0%; background: linear-gradient(90deg, var(--c-primary), var(--c-secondary)); border-radius: 3px; position: relative; transition: width 0.1s linear; }
    .timeline .progress-bar::after { content: ''; position: absolute; right: -6px; top: 50%; transform: translateY(-50%) scale(0); width: 12px; height: 12px; border-radius: 50%; background-color: var(--brilliant-white); transition: transform 0.2s; }
    .timeline .buffered-bar, .timeline .hover-indicator { position: absolute; top: 0; left: 0; height: 100%; width: 0; background-color: rgba(255, 255, 255, 0.5); border-radius: 3px; }
    input[type=range].volume-slider { -webkit-appearance: none; appearance: none; background: #fff0; }
    input[type=range].volume-slider::-webkit-slider-runnable-track { height: 5px; background: rgba(255, 255, 255, 0.3); border-radius: 3px; }
    input[type=range].volume-slider::-webkit-slider-thumb { -webkit-appearance: none; appearance: none; margin-top: -3.5px; background-color: #fff; height: 12px; width: 12px; border-radius: 50%; }
    .audio-loader { margin-bottom: 5px; }
    .audio-progress-bar { width: 100%; background-color: #333; border-radius: 4px; overflow: hidden; height: 8px; }
    .audio-progress-fill { width: 0%; height: 100%; background-image: linear-gradient(45deg, var(--c-secondary), var(--c-primary)); transition: width 0.1s linear; }
    .audio-progress-text { font-size: 0.8em; color: #aaa; text-align: center; margin-top: 8px; }
    `;
    (document.head || document.documentElement).appendChild(style);
  } catch (e) {
  }

  const shim = function () {
    try {
      const S = String.prototype;
      const origSlice = S.slice;
      const origConcat = S.concat;
      let lastSliceValue = null;
      let lastSliceSource = null;
      Object.defineProperty(S, "slice", {
        configurable: true,
        writable: true,
        value: function (start, end) {
          const src = String(this);
          const out = origSlice.call(src, start, end);
          if (
            start === 0 &&
            end === 50 &&
            typeof out === "string" &&
            src.length > 50
          ) {
            lastSliceValue = out;
            lastSliceSource = src;
          } else {
            lastSliceValue = null;
            lastSliceSource = null;
          }
          return out;
        },
      });
      Object.defineProperty(S, "concat", {
        configurable: true,
        writable: true,
        value: function (...args) {
          try {
            if (
              (this === "" || String(this) === "") &&
              args.length === 2 &&
              args[1] === "..." &&
              typeof args[0] === "string" &&
              lastSliceValue !== null &&
              args[0] === lastSliceValue
            ) {
              return lastSliceSource;
            }
          } catch (e) {
          }
          return origConcat.apply(this, args);
        },
      });
    } catch (e) {
    }
  };

  try {
    const s = document.createElement("script");
    s.textContent = `(${shim})();`;
    (document.head || document.documentElement).appendChild(s);
    s.remove();
  } catch (e) {
  }

  const audioPlayer = (() => {
    let isInitialized = false;
    let modalOverlay,
      playerContainer,
      audio,
      closeBtn,
      titleContainer,
      titleEl,
      loaderContainer,
      progressFill,
      progressText,
      controlsContainer,
      playPauseBtn,
      volumeBtn,
      volumeSlider,
      currentTimeEl,
      totalTimeEl,
      timelineContainer,
      progressBar,
      bufferedBar,
      hoverIndicator,
      downloadBtn;

    let currentAudioUrl = null;
    let currentFileName = "";
    let activeRequest = null;
    let lastVolume = 1;

    function init() {
      if (isInitialized) return;

      const playerTemplate = `
        <div class="modal-overlay" id="audioModal" style="display: none;">
            <div id="audio-player-container">
                <button id="close-audio-btn" aria-label="Close Audio Player"><svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" fill="currentColor" viewBox="0 0 16 16"><path d="M2.146 2.854a.5.5 0 1 1 .708-.708L8 7.293l5.146-5.147a.5.5 0 0 1 .708.708L8.707 8l5.147 5.146a.5.5 0 0 1-.708.708L8 8.707l-5.146 5.147a.5.5 0 0 1-.708-.708L7.293 8 2.146 2.854Z"/></svg></button>
                <div class="audio-title-container">
                    <span class="expand-arrow">▼</span>
                    <h3 class="audio-title"></h3>
                </div>
                <div class="audio-loader">
                    <div class="audio-progress-bar"><div class="audio-progress-fill"></div></div>
                    <div class="audio-progress-text">Initializing...</div>
                </div>
                <audio id="audio-element" preload="metadata"></audio>
                <div class="audio-controls-container" style="display:none;">
                    <div class="timeline-container">
                        <div class="timeline">
                            <div class="hover-indicator"></div><div class="buffered-bar"></div><div class="progress-bar"></div>
                        </div>
                    </div>
                    <div class="controls">
                        <div class="controls-left">
                            <button class="play-pause-btn" aria-label="Play/Pause">
                                <svg class="play-icon" viewBox="0 0 24 24"><polygon points="5 3 19 12 5 21 5 3"></polygon></svg>
                                <svg class="pause-icon" viewBox="0 0 24 24"><rect x="6" y="4" width="4" height="16"></rect><rect x="14" y="4" width="4" height="16"></rect></svg>
                            </button>
                            <div class="volume-container">
                                <button class="volume-btn" aria-label="Mute/Unmute">
                                    <svg class="high-volume-icon" viewBox="0 0 24 24"><path d="M14,3.23V5.29C16.89,6.15 19,8.83 19,12C19,15.17 16.89,17.84 14,18.7V20.77C18,19.86 21,16.28 21,12C21,7.72 18,4.14 14,3.23M16.5,12C16.5,10.23 15.5,8.71 14,7.97V16C15.5,15.29 16.5,13.76 16.5,12M3,9V15H7L12,20V4L7,9H3Z" /></svg>
                                    <svg class="low-volume-icon" viewBox="0 0 24 24"><path d="M5,9V15H9L14,20V4L9,9M18.5,12C18.5,10.23 17.5,8.71 16,7.97V16C17.5,15.29 18.5,13.76 18.5,12Z" /></svg>
                                    <svg class="muted-icon" viewBox="0 0 24 24"><path d="M12,4L9.91,6.09L12,8.18M4.27,3L3,4.27L7.73,9H3V15H7L12,20V13.27L16.25,17.53C15.58,18.04 14.83,18.46 14,18.7V20.77C15.38,20.45 16.63,19.82 17.68,18.96L19.73,21L21,19.73L12,10.73M19,12C19,12.94 18.8,13.82 18.46,14.64L19.97,16.15C20.62,14.91 21,13.5 21,12C21,7.72 18,4.14 14,3.23V5.29C16.89,6.15 19,8.83 19,12M16.5,12C16.5,12.22 16.47,12.43 16.43,12.64L14,10.21V7.97C15.5,8.71 16.5,10.23 16.5,12Z" /></svg>
                                </button>
                                <input class="volume-slider" type="range" min="0" max="1" step="any" value="1">
                            </div>
                            <div class="time-container"><span class="current-time">0:00</span> / <span class="total-time">0:00</span></div>
                        </div>
                        <div class="controls-right"><button class="download-btn" aria-label="Download Audio"><svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2"><path d="M3 15v4c0 1.1.9 2 2 2h14a2 2 0 0 0 2-2v-4M17 9l-5 5-5-5M12 12.8V2.5"/></svg></button></div>
                    </div>
                </div>
            </div>
        </div>
    `;
      const tempContainer = document.createElement("div");
      tempContainer.innerHTML = playerTemplate.trim();
      document.body.appendChild(tempContainer.firstChild);

      modalOverlay = document.getElementById("audioModal");
      playerContainer = document.getElementById("audio-player-container");
      audio = document.getElementById("audio-element");
      closeBtn = document.getElementById("close-audio-btn");
      titleContainer = playerContainer.querySelector(".audio-title-container");
      titleEl = titleContainer.querySelector(".audio-title");
      loaderContainer = playerContainer.querySelector(".audio-loader");
      progressFill = playerContainer.querySelector(".audio-progress-fill");
      progressText = playerContainer.querySelector(".audio-progress-text");
      controlsContainer = playerContainer.querySelector(
        ".audio-controls-container",
      );
      playPauseBtn = playerContainer.querySelector(".play-pause-btn");
      volumeBtn = playerContainer.querySelector(".volume-btn");
      volumeSlider = playerContainer.querySelector(".volume-slider");
      currentTimeEl = playerContainer.querySelector(".current-time");
      totalTimeEl = playerContainer.querySelector(".total-time");
      timelineContainer = playerContainer.querySelector(
        ".timeline-container",
      );
      progressBar = playerContainer.querySelector(".progress-bar");
      bufferedBar = playerContainer.querySelector(".buffered-bar");
      hoverIndicator = playerContainer.querySelector(".hover-indicator");
      downloadBtn = playerContainer.querySelector(".download-btn");

      bindEvents();
      isInitialized = true;
    }

    const formatBytes = (bytes, d = 2) =>
      bytes === 0
        ? "0 Bytes"
        : `${parseFloat((bytes / Math.pow(1024, Math.floor(Math.log(bytes) / Math.log(1024)))).toFixed(d < 0 ? 0 : d))} ${["Bytes", "KB", "MB", "GB", "TB"][Math.floor(Math.log(bytes) / Math.log(1024))]}`;

    function bindEvents() {
      closeBtn.addEventListener("click", close);
      modalOverlay.addEventListener(
        "click",
        (e) => e.target === modalOverlay && close(),
      );
      titleContainer.addEventListener("click", () =>
        titleContainer.classList.toggle("expanded"),
      );
      playPauseBtn.addEventListener("click", togglePlay);
      audio.addEventListener("play", () =>
        playerContainer.classList.remove("paused"),
      );
      audio.addEventListener("pause", () =>
        playerContainer.classList.add("paused"),
      );
      audio.addEventListener("loadedmetadata", handleMetadataLoaded);
      audio.addEventListener("timeupdate", handleTimeUpdate);
      audio.addEventListener("progress", handleBufferUpdate);
      audio.addEventListener("volumechange", updateVolumeUI);
      volumeBtn.addEventListener("click", toggleMute);
      volumeSlider.addEventListener(
        "input",
        (e) => (audio.volume = e.target.value),
      );
      downloadBtn.addEventListener("click", downloadAudio);
      timelineContainer.addEventListener("mousemove", handleTimelineHover);
      timelineContainer.addEventListener("click", handleTimelineSeek);
    }
    function handleMetadataLoaded() {
      playerContainer.classList.add("paused");
      totalTimeEl.textContent = formatTime(audio.duration);
      audio.volume = volumeSlider.value;
      updateVolumeUI();
      audio.play().catch((e) => console.error("Autoplay prevented:", e));
    }
    function togglePlay() {
      audio.paused ? audio.play() : audio.pause();
    }
    function toggleMute() {
      audio.volume > 0
        ? ((lastVolume = audio.volume), (audio.volume = 0))
        : (audio.volume = lastVolume);
    }
    function handleTimeUpdate() {
      currentTimeEl.textContent = formatTime(audio.currentTime);
      progressBar.style.width = `${(audio.currentTime / audio.duration) * 100}%`;
    }
    function handleBufferUpdate() {
      if (audio.duration > 0)
        for (let i = 0; i < audio.buffered.length; i++)
          if (
            audio.buffered.start(i) <= audio.currentTime &&
            audio.currentTime <= audio.buffered.end(i)
          ) {
            bufferedBar.style.width = `${(audio.buffered.end(i) / audio.duration) * 100}%`;
            break;
          }
    }
    function handleTimelineHover(e) {
      const rect = timelineContainer.getBoundingClientRect();
      hoverIndicator.style.width = `${(Math.min(Math.max(0, e.x - rect.x), rect.width) / rect.width) * 100}%`;
    }
    function handleTimelineSeek(e) {
      const rect = timelineContainer.getBoundingClientRect();
      audio.currentTime =
        (Math.min(Math.max(0, e.x - rect.x), rect.width) / rect.width) *
        audio.duration;
    }
    function updateVolumeUI() {
      volumeSlider.value = audio.volume;
      const icons = [
        ".high-volume-icon",
        ".low-volume-icon",
        ".muted-icon",
      ].map((s) => volumeBtn.querySelector(s));
      icons.forEach((i) => (i.style.display = "none"));
      if (audio.volume === 0 || audio.muted) icons[2].style.display = "block";
      else if (audio.volume < 0.5) icons[1].style.display = "block";
      else icons[0].style.display = "block";
    }
    function formatTime(t) {
      const r = new Date(t * 1000).toISOString().substr(11, 8);
      return r.startsWith("00:") ? r.substr(3) : r;
    }
    function downloadAudio() {
      if (!audio.src) return;
      const link = document.createElement("a");
      link.href = audio.src;
      link.download = currentFileName || "kemono-audio";
      document.body.appendChild(link);
      link.click();
      document.body.removeChild(link);
    }
    function handleKeyboardShortcuts(e) {
      if (document.activeElement.tagName.toLowerCase() === "input") return;
      switch (e.key.toLowerCase()) {
        case "escape":
          close();
          break;
        case " ":
          if (document.activeElement.tagName.toLowerCase() !== "button") {
            e.preventDefault();
            togglePlay();
          }
          break;
        case "m":
          toggleMute();
          break;
        case "arrowright":
          audio.currentTime = Math.min(
            audio.duration,
            audio.currentTime + 5,
          );
          break;
        case "arrowleft":
          audio.currentTime = Math.max(0, audio.currentTime - 5);
          break;
      }
    }

    function open(url, fileName) {
      if (currentAudioUrl) URL.revokeObjectURL(currentAudioUrl);
      currentFileName = fileName;
      titleContainer.classList.remove("expanded");
      titleEl.textContent = fileName;
      loaderContainer.style.display = "block";
      controlsContainer.style.display = "none";
      progressFill.style.width = "0%";
      progressText.textContent = "Initializing...";
      modalOverlay.style.display = "flex";
      setTimeout(() => modalOverlay.classList.add("show"), 10);
      document.body.style.overflow = "hidden";
      document.addEventListener("keydown", handleKeyboardShortcuts);

      activeRequest = GM_xmlhttpRequest({
        method: "GET",
        url,
        responseType: "blob",
        onprogress: (p) => {
          if (p.lengthComputable) {
            const percent = Math.round((p.loaded / p.total) * 100);
            progressFill.style.width = `${percent}%`;
            progressText.textContent = `Downloading... ${percent}% (${formatBytes(p.loaded)} / ${formatBytes(p.total)})`;
          }
        },
        onload: (res) => {
          activeRequest = null;
          currentAudioUrl = URL.createObjectURL(res.response);
          audio.src = currentAudioUrl;
          audio.load();
          loaderContainer.style.display = "none";
          controlsContainer.style.display = "block";
        },
        onerror: () => {
          activeRequest = null;
          progressText.textContent = "Error: Could not load audio file.";
        },
        onabort: () => (activeRequest = null),
      });
    }

    function close() {
      if (activeRequest) activeRequest.abort();
      audio.pause();
      modalOverlay.classList.remove("show");
      setTimeout(() => {
        modalOverlay.style.display = "none";
        if (currentAudioUrl) {
          URL.revokeObjectURL(currentAudioUrl);
          currentAudioUrl = null;
          currentFileName = "";
          audio.removeAttribute("src");
          audio.load();
        }
      }, 300);
      document.body.style.overflow = "";
      document.removeEventListener("keydown", handleKeyboardShortcuts);
    }

    return { init, open };
  })();

  function initializeScript() {
    audioPlayer.init();
    document.body.addEventListener("click", (e) => {
      const link = e.target.closest(
        'a[href*=".wav?f="], a[href*=".mp3?f="]',
      );
      if (link) {
        e.preventDefault();
        e.stopPropagation();

        let title = "Audio Player";

        const titleElement = document.querySelector("h1.post__title");
        if (titleElement) {
          title = titleElement.textContent.trim();
        }
        else {
          try {
            const urlParams = new URLSearchParams(link.search);
            title = decodeURIComponent(urlParams.get("f")) || title;
          } catch {
          }
        }

        audioPlayer.open(link.href, title);
      }
    });
  }

  if (document.body) {
    initializeScript();
  } else {
    new MutationObserver((mutations, observer) => {
      if (document.body) {
        initializeScript();
        observer.disconnect();
      }
    }).observe(document.documentElement, { childList: true });
  }
})();