Font Customizer

Customize fonts for any website through the Tampermonkey menu

// ==UserScript==
// @name         Font Customizer
// @namespace    http://tampermonkey.net/
// @version      1.1
// @description  Customize fonts for any website through the Tampermonkey menu
// @author       Cursor, claude-3.7, and me(qooo).
// @license      MIT
// @match        *://*/*
// @icon         
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// @grant        GM_addStyle
// @grant        GM_xmlhttpRequest
// @grant        GM_setValue
// @grant        GM_getValue
// @connect      fonts.googleapis.com
// @connect      fonts.gstatic.com
// ==/UserScript==

(function () {
  "use strict";

  // Storage keys
  const STORAGE_KEY_PREFIX = "fontCustomizer_";
  const ENABLED_SUFFIX = "_enabled";
  const FONT_SUFFIX = "_font";
  const FONT_LIST_KEY = "fontCustomizer_globalFonts";

  // Common web fonts from Google Fonts
  const WEB_FONTS = [
    {
      name: "Roboto",
      family: "Roboto, sans-serif",
      url: "https://fonts.googleapis.com/css2?family=Roboto:wght@300;400;500;700&display=swap",
    },
    {
      name: "Open Sans",
      family: "'Open Sans', sans-serif",
      url: "https://fonts.googleapis.com/css2?family=Open+Sans:wght@300;400;500;600;700&display=swap",
    },
    {
      name: "Lato",
      family: "Lato, sans-serif",
      url: "https://fonts.googleapis.com/css2?family=Lato:wght@300;400;700&display=swap",
    },
    {
      name: "Montserrat",
      family: "Montserrat, sans-serif",
      url: "https://fonts.googleapis.com/css2?family=Montserrat:wght@300;400;500;600;700&display=swap",
    },
    {
      name: "Poppins",
      family: "Poppins, sans-serif",
      url: "https://fonts.googleapis.com/css2?family=Poppins:wght@300;400;500;600;700&display=swap",
    },
    {
      name: "Source Sans Pro",
      family: "'Source Sans Pro', sans-serif",
      url: "https://fonts.googleapis.com/css2?family=Source+Sans+Pro:wght@300;400;600;700&display=swap",
    },
    {
      name: "Ubuntu",
      family: "Ubuntu, sans-serif",
      url: "https://fonts.googleapis.com/css2?family=Ubuntu:wght@300;400;500;700&display=swap",
    },
    {
      name: "Nunito",
      family: "Nunito, sans-serif",
      url: "https://fonts.googleapis.com/css2?family=Nunito:wght@300;400;600;700&display=swap",
    },
    {
      name: "Playfair Display",
      family: "'Playfair Display', serif",
      url: "https://fonts.googleapis.com/css2?family=Playfair+Display:wght@400;500;600;700&display=swap",
    },
    {
      name: "Merriweather",
      family: "Merriweather, serif",
      url: "https://fonts.googleapis.com/css2?family=Merriweather:wght@300;400;700&display=swap",
    },
  ];

  // Load a web font
  function loadWebFont(fontUrl) {
    GM_xmlhttpRequest({
      method: "GET",
      url: fontUrl,
      onload: function (response) {
        const style = document.createElement("style");
        style.textContent = response.responseText;
        document.head.appendChild(style);
      },
    });
  }

  // Get saved fonts or initialize with empty array
  function getSavedFonts() {
    return GM_getValue(FONT_LIST_KEY, []);
  }

  // Save a font to the list if it doesn't exist already
  function saveFontToList(font) {
    const fonts = getSavedFonts();
    if (!fonts.includes(font)) {
      fonts.push(font);
      GM_setValue(FONT_LIST_KEY, fonts);
    }
  }

  // Remove a font from the saved list
  function removeFontFromList(font) {
    const fonts = getSavedFonts();
    const index = fonts.indexOf(font);
    if (index !== -1) {
      fonts.splice(index, 1);
      GM_setValue(FONT_LIST_KEY, fonts);
    }
  }

  // Get current hostname
  const hostname = window.location.hostname;

  // Storage helper functions
  function getStorageKey(suffix) {
    return STORAGE_KEY_PREFIX + hostname + suffix;
  }

  function isEnabledForSite() {
    return localStorage.getItem(getStorageKey(ENABLED_SUFFIX)) === "true";
  }

  function setEnabledForSite(enabled) {
    localStorage.setItem(getStorageKey(ENABLED_SUFFIX), enabled.toString());
  }

  function getFontForSite() {
    return localStorage.getItem(getStorageKey(FONT_SUFFIX)) || "";
  }

  function setFontForSite(font) {
    localStorage.setItem(getStorageKey(FONT_SUFFIX), font);
  }

  // Apply font to the website
  function applyFont() {
    if (isEnabledForSite()) {
      const font = getFontForSite();
      GM_addStyle(`
        * {
          font-family: ${font} !important;
        }
      `);
    }
  }

  // Remove applied font styles
  function removeAppliedFont() {
    const styleId = "font-customizer-styles";
    const existingStyle = document.getElementById(styleId);
    if (existingStyle) {
      existingStyle.remove();
    }
    applyStyles();
  }

  // Apply all necessary styles
  function applyStyles() {
    if (isEnabledForSite()) {
      const font = getFontForSite();
      const styleElement = document.createElement("style");
      styleElement.id = "font-customizer-styles";
      styleElement.textContent = `
        * {
          font-family: ${font} !important;
        }
      `;
      document.head.appendChild(styleElement);
    }
  }

  // Menu command IDs
  let toggleCommandId = null;
  let fontCommandId = null;

  // Register menu commands
  function registerMenuCommands() {
    if (toggleCommandId !== null) {
      GM_unregisterMenuCommand(toggleCommandId);
    }
    if (fontCommandId !== null) {
      GM_unregisterMenuCommand(fontCommandId);
    }

    const enabled = isEnabledForSite();
    const toggleText = enabled
      ? "🟢 Enabled on the site"
      : "🔴 Disabled on the site";

    toggleCommandId = GM_registerMenuCommand(toggleText, function () {
      const newEnabledState = !enabled;
      setEnabledForSite(newEnabledState);

      if (newEnabledState) {
        applyStyles();
      } else {
        removeAppliedFont();
      }

      registerMenuCommands();
    });

    const currentFont = getFontForSite();
    // Truncate the font name if it's too long (more than x characters)
    const truncatedFont =
      currentFont && currentFont.length > 10
        ? currentFont.substring(0, 10) + "..."
        : currentFont || "None";

    fontCommandId = GM_registerMenuCommand(
      `✨ Select Font (Current: ${truncatedFont})`,
      showFontSelector
    );
  }

  // Create and show font selector popup
  function showFontSelector() {
    // Remove existing popup if any
    const existingPopup = document.getElementById("font-customizer-popup");
    if (existingPopup) {
      existingPopup.remove();
    }

    // Create popup container
    const popup = document.createElement("div");
    popup.id = "font-customizer-popup";

    // Add styles for the popup
    GM_addStyle(`
      #font-customizer-popup {
        position: fixed;
        top: 50%;
        left: 50%;
        transform: translate(-50%, -50%);
        background-color: var(--popup-bg, #ffffff);
        color: var(--popup-text, #000000);
        border: 1px solid var(--popup-border, #cccccc);
        border-radius: 12px;
        padding: 24px;
        z-index: 9999;
        min-width: 380px;
        width: 400px;
        height: fit-content;
        overflow-y: auto;
        box-shadow: 0 8px 24px rgba(0, 0, 0, 0.2);
        font-family: system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
        animation: popup-fade-in 0.2s ease-out;
      }

      @keyframes popup-fade-in {
        from { opacity: 0; transform: translate(-50%, -48%); }
        to { opacity: 1; transform: translate(-50%, -50%); }
      }

      #font-customizer-popup h2 {
        margin-top: 0;
        margin-bottom: 20px;
        font-size: 20px;
        font-weight: 600;
        text-align: center;
        color: var(--popup-title, inherit);
      }

      #font-customizer-popup .font-input-container {
        margin-bottom: 16px;
      }

      #font-customizer-popup .font-input {
        width: 100%;
        padding: 12px;
        border: 1px solid var(--popup-border, #cccccc);
        border-radius: 8px;
        box-sizing: border-box;
        font-size: 14px;
        transition: border-color 0.2s;
        margin-bottom: 8px;
      }

      #font-customizer-popup .font-input:focus {
        border-color: var(--popup-button, #4a86e8);
        outline: none;
        box-shadow: 0 0 0 2px rgba(74, 134, 232, 0.2);
      }

      #font-customizer-popup .add-font-button {
        display: block;
        width: 100%;
        padding: 8px 16px;
        background-color: var(--popup-button, #4a86e8);
        color: white;
        border: none;
        border-radius: 8px;
        cursor: pointer;
        font-weight: 600;
        font-size: 14px;
        transition: background-color 0.2s, transform 0.1s;
      }

      #font-customizer-popup .add-font-button:hover {
        background-color: var(--popup-button-hover, #3b78e7);
      }

      #font-customizer-popup .add-font-button:active {
        transform: scale(0.98);
      }

      #font-customizer-popup .section-title {
        font-size: 16px;
        font-weight: 600;
        margin: 16px 0 8px 0;
        color: var(--popup-title, inherit);
        display: flex;
        align-items: center;
        justify-content: space-between;
      }

      #font-customizer-popup .section-title .title-with-info {
        display: flex;
        align-items: center;
        gap: 8px;
      }
      #font-customizer-popup .toggle-section{
        cursor: pointer;
        border: 1px solid var(--popup-border, #cccccc);
        border-radius: 8px;
        padding: 4px 8px;
        font-size: 12px;
        font-weight: 600;
        color: var(--popup-text-secondary);
      }
      #font-customizer-popup .toggle-section:hover {
        background-color: var(--popup-hover, #f5f5f5);
      }

      #font-customizer-popup .info-icon {
        cursor: help;
        color: var(--popup-text-secondary);
        font-size: 14px;
        position: relative;
      }

      #font-customizer-popup .info-tooltip {
        visibility: hidden;
        position: absolute;
        left: 50%;
        transform: translateX(-50%);
        bottom: calc(100% + 8px);
        color: var(--popup-tooltip-text);
        padding: 8px 12px;
        background-color: var(--popup-tooltip-bg);
        border-radius: 6px;
        font-size: 12px;
        font-weight: normal;
        white-space: normal;
        text-wrap: wrap;
        z-index: 10000;
        opacity: 0;
        transition: opacity 0.2s, visibility 0.2s;
        box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);
        width: 250px;
        line-height: 1.4;
      }

      #font-customizer-popup .info-tooltip::after {
        content: '';
        position: absolute;
        top: 100%;
        left: 50%;
        transform: translateX(-50%);
        border-width: 5px;
        border-style: solid;
        border-color: var(--popup-tooltip-bg) transparent transparent transparent;
      }

      #font-customizer-popup .info-icon:hover .info-tooltip {
        visibility: visible;
        opacity: 1;
      }

      #font-customizer-popup .no-fonts-message {
        color: var(--popup-text-secondary, #666666);
        font-style: italic;
        text-align: center;
        padding: 16px 0;
      }

      #font-customizer-popup ul {
        list-style: none;
        padding: 0;
        margin: 0 0 16px 0;
        max-height: 200px;
        overflow-y: auto;
        border-radius: 8px;
        border: 1px solid var(--popup-border, #eaeaea);
      }

      #font-customizer-popup ul:empty {
        display: none;
      }

      #font-customizer-popup ul::-webkit-scrollbar {
        width: 8px;
      }

      #font-customizer-popup ul::-webkit-scrollbar-track {
        background: var(--popup-scrollbar-track, #f1f1f1);
        border-radius: 0 8px 8px 0;
      }

      #font-customizer-popup ul::-webkit-scrollbar-thumb {
        background: var(--popup-scrollbar-thumb, #c1c1c1);
        border-radius: 4px;
      }

      #font-customizer-popup ul::-webkit-scrollbar-thumb:hover {
        background: var(--popup-scrollbar-thumb-hover, #a1a1a1);
      }

      #font-customizer-popup li {
        padding: 10px 16px;
        cursor: pointer;
        transition: all 0.15s ease;
        border-bottom: 1px solid var(--popup-border, #eaeaea);
        display: flex;
        align-items: center;
        justify-content: space-between;
      }

      #font-customizer-popup li:last-child {
        border-bottom: none;
      }

      #font-customizer-popup li:hover {
        background-color: var(--popup-hover, #f5f5f5);
      }

      #font-customizer-popup li.selected {
        background-color: var(--popup-selected, #e8f0fe);
        font-weight: 500;
      }

      #font-customizer-popup li.selected .font-name::before {
        content: "✓";
        margin-right: 8px;
        color: var(--popup-check, #4a86e8);
        font-weight: bold;
      }

      #font-customizer-popup li:not(.selected) .font-name {
        padding-left: 24px;
      }

      #font-customizer-popup .font-actions {
        display: flex;
        opacity: 0;
        transition: opacity 0.2s;
      }

      #font-customizer-popup li:hover .font-actions {
        opacity: 1;
      }

      #font-customizer-popup .delete-font {
        color: var(--popup-delete, #e53935);
        cursor: pointer;
        font-size: 16px;
        padding: 4px;
        border-radius: 4px;
        transition: background-color 0.2s;
      }

      #font-customizer-popup .delete-font:hover {
        background-color: var(--popup-delete-hover, rgba(229, 57, 53, 0.1));
      }

      #font-customizer-popup .close-button {
        display: block;
        width: 100%;
        margin: 16px auto 0;
        padding: 12px 16px;
        background-color: var(--popup-button-secondary, #757575);
        color: white;
        border: none;
        border-radius: 8px;
        cursor: pointer;
        font-weight: 600;
        font-size: 15px;
        transition: background-color 0.2s, transform 0.1s;
      }

      #font-customizer-popup .close-button:hover {
        background-color: var(--popup-button-secondary-hover, #616161);
      }

      #font-customizer-popup .close-button:active {
        transform: scale(0.98);
      }

      #font-customizer-popup .web-font-item {
        font-family: inherit;
      }

      #font-customizer-popup .web-font-item .font-name {
        font-family: inherit;
      }

      /* Dark mode detection and styles */
      @media (prefers-color-scheme: dark) {
        #font-customizer-popup {
          --popup-bg: #222222;
          --popup-text: #ffffff;
          --popup-text-secondary: #aaaaaa;
          --popup-title: #ffffff;
          --popup-border: #444444;
          --popup-hover: #333333;
          --popup-selected: #2c3e50;
          --popup-check: #64b5f6;
          --popup-button: #4a86e8;
          --popup-button-hover: #3b78e7;
          --popup-button-secondary: #616161;
          --popup-button-secondary-hover: #757575;
          --popup-delete: #f44336;
          --popup-delete-hover: rgba(244, 67, 54, 0.2);
          --popup-scrollbar-track: #333333;
          --popup-scrollbar-thumb: #555555;
          --popup-scrollbar-thumb-hover: #666666;
          --popup-tooltip-bg: #4a4a4a;
          --popup-tooltip-text: #ffffff;
        }
      }

      /* Light mode styles */
      @media (prefers-color-scheme: light) {
        #font-customizer-popup {
          --popup-bg: #ffffff;
          --popup-text: #333333;
          --popup-text-secondary: #666666;
          --popup-title: #222222;
          --popup-border: #eaeaea;
          --popup-hover: #f5f5f5;
          --popup-selected: #e8f0fe;
          --popup-check: #4a86e8;
          --popup-button: #4a86e8;
          --popup-button-hover: #3b78e7;
          --popup-button-secondary: #757575;
          --popup-button-secondary-hover: #616161;
          --popup-delete: #e53935;
          --popup-delete-hover: rgba(229, 57, 53, 0.1);
          --popup-scrollbar-track: #f1f1f1;
          --popup-scrollbar-thumb: #c1c1c1;
          --popup-scrollbar-thumb-hover: #a1a1a1;
          --popup-tooltip-bg: #ffffff;
          --popup-tooltip-text: #333333;
        }
      }

      /* Overlay to prevent clicking outside */
      #font-customizer-overlay {
        position: fixed;
        top: 0;
        left: 0;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0, 0.5);
        z-index: 9998;
        animation: overlay-fade-in 0.2s ease-out;
      }

      @keyframes overlay-fade-in {
        from { opacity: 0; }
        to { opacity: 1; }
      }
    `);

    // Create overlay to prevent clicking outside
    const overlay = document.createElement("div");
    overlay.id = "font-customizer-overlay";
    document.body.appendChild(overlay);

    // Add click event to overlay to close popup
    overlay.addEventListener("click", closePopup);

    // Create popup content
    popup.innerHTML = `
      <h2>Font Customizer</h2>
      <div class="font-input-container">
        <input type="text" id="new-font-input" class="font-input" placeholder="Enter font name (e.g., Arial, sans-serif)">
        <button id="add-font-button" class="add-font-button">Add & Apply Font</button>
      </div>
      <div class="section-title">
        <div class="title-with-info">
          <span>Your Saved Fonts</span>
          <span class="info-icon">ℹ️
            <span class="info-tooltip">Custom fonts will only work if they are installed on your system. Use exact font names for best results.</span>
          </span>
        </div>
        <span class="toggle-section" id="toggle-saved">Hide</span>
      </div>
      <ul id="saved-fonts-list"></ul>
      <div id="no-saved-fonts" class="no-fonts-message">No saved fonts or Hiden yet. Add one above!</div>
      <div class="section-title">
        <div class="title-with-info">
          <span>Web Fonts</span>
          <span class="info-icon">ℹ️
            <span class="info-tooltip">These fonts will be loaded from Google Fonts when selected. They work on any system but require internet connection.</span>
          </span>
        </div>
        <span class="toggle-section" id="toggle-web">Show</span>
      </div>
      <ul id="web-fonts-list" style="display: none;"></ul>
      <button class="close-button" id="close-popup">Close</button>
    `;

    document.body.appendChild(popup);

    // Get current font and saved fonts
    const currentFont = getFontForSite();
    const savedFonts = getSavedFonts();

    // Populate saved fonts list
    const savedFontsList = document.getElementById("saved-fonts-list");
    const noSavedFonts = document.getElementById("no-saved-fonts");

    // Show/hide no fonts message
    if (savedFonts.length === 0) {
      noSavedFonts.style.display = "block";
    } else {
      noSavedFonts.style.display = "none";
    }

    // Add saved fonts to the list
    savedFonts.forEach((font) => {
      addFontToList(font, savedFontsList, true);
    });

    // Add web fonts to the list
    const webFontsList = document.getElementById("web-fonts-list");
    WEB_FONTS.forEach((font) => {
      addFontToList(font.name, webFontsList, false, font);
    });

    // Toggle sections
    document
      .getElementById("toggle-saved")
      .addEventListener("click", function () {
        const savedList = document.getElementById("saved-fonts-list");
        const noSaved = document.getElementById("no-saved-fonts");
        const isHidden = savedList.style.display === "none";

        savedList.style.display = isHidden ? "block" : "none";
        noSaved.style.display =
          isHidden && getSavedFonts().length === 0 ? "block" : "none";
        this.textContent = isHidden ? "Hide" : "Show";
      });

    document
      .getElementById("toggle-web")
      .addEventListener("click", function () {
        const webList = document.getElementById("web-fonts-list");
        const isHidden = webList.style.display === "none";

        webList.style.display = isHidden ? "block" : "none";
        this.textContent = isHidden ? "Hide" : "Show";
      });

    // Handle new font input
    const newFontInput = document.getElementById("new-font-input");
    const addFontButton = document.getElementById("add-font-button");

    // Focus the input field
    newFontInput.focus();

    // Add font when button is clicked
    addFontButton.addEventListener("click", addNewFont);

    // Add font when Enter is pressed
    newFontInput.addEventListener("keydown", (e) => {
      if (e.key === "Enter") {
        addNewFont();
      }
    });

    // Function to add a font to the list
    function addFontToList(font, listElement, isSaved = false, webFont = null) {
      const li = document.createElement("li");
      if (webFont) {
        li.classList.add("web-font-item");
        li.style.fontFamily = webFont.family;
      }

      li.innerHTML = `
        <span class="font-name">${font}</span>
        ${
          isSaved
            ? `
          <div class="font-actions">
            <span class="delete-font" title="Remove font">🗑️</span>
          </div>
        `
            : ""
        }
      `;

      if (font === currentFont) {
        li.classList.add("selected");
      }

      // Select font when clicked
      li.addEventListener("click", (e) => {
        if (e.target.classList.contains("delete-font")) {
          return;
        }

        // Remove selected class from all items
        document
          .querySelectorAll("#font-customizer-popup li")
          .forEach((item) => {
            item.classList.remove("selected");
          });

        // Add selected class to clicked item
        li.classList.add("selected");

        // Set the selected font
        if (webFont) {
          setFontForSite(webFont.family);
          loadWebFont(webFont.url);
        } else {
          setFontForSite(font);
        }

        // Apply the font if enabled
        if (isEnabledForSite()) {
          removeAppliedFont();
          applyStyles();
        }

        // Update menu commands
        registerMenuCommands();
      });

      // Delete font when delete button is clicked (only for saved fonts)
      if (isSaved) {
        const deleteButton = li.querySelector(".delete-font");
        deleteButton.addEventListener("click", (e) => {
          e.stopPropagation();

          removeFontFromList(font);
          li.remove();

          if (font === currentFont) {
            const remainingFonts = getSavedFonts();
            if (remainingFonts.length > 0) {
              setFontForSite(remainingFonts[0]);
              const firstFont = document.querySelector("#saved-fonts-list li");
              if (firstFont) {
                firstFont.classList.add("selected");
              }
            } else {
              setFontForSite("");
            }

            if (isEnabledForSite()) {
              removeAppliedFont();
              applyStyles();
            }

            registerMenuCommands();
          }

          if (savedFontsList.children.length === 0) {
            noSavedFonts.style.display = "block";
          }
        });
      }

      listElement.appendChild(li);
    }

    // Function to add a new font
    function addNewFont() {
      const fontName = newFontInput.value.trim();
      if (fontName) {
        saveFontToList(fontName);
        newFontInput.value = "";
        noSavedFonts.style.display = "none";
        addFontToList(fontName, savedFontsList, true);
        setFontForSite(fontName);

        document
          .querySelectorAll("#font-customizer-popup li")
          .forEach((item) => {
            item.classList.remove("selected");
          });

        const newFontElement = Array.from(
          document.querySelectorAll("#saved-fonts-list li")
        ).find((li) => li.querySelector(".font-name").textContent === fontName);
        if (newFontElement) {
          newFontElement.classList.add("selected");
        }

        if (isEnabledForSite()) {
          removeAppliedFont();
          applyStyles();
        }

        registerMenuCommands();
      }
    }

    // Function to close the popup
    function closePopup() {
      const popup = document.getElementById("font-customizer-popup");
      const overlay = document.getElementById("font-customizer-overlay");
      if (popup) popup.remove();
      if (overlay) overlay.remove();
    }

    // Close button functionality
    document
      .getElementById("close-popup")
      .addEventListener("click", closePopup);

    // Prevent closing when clicking on the popup itself
    popup.addEventListener("click", (e) => {
      e.stopPropagation();
    });

    // Allow Escape key to close the popup
    document.addEventListener("keydown", function handleEscape(e) {
      if (e.key === "Escape") {
        closePopup();
        document.removeEventListener("keydown", handleEscape);
      }
    });
  }

  // Initialize
  function init() {
    registerMenuCommands();
    applyStyles();

    const observer = new MutationObserver(function (mutations) {
      if (!isEnabledForSite()) return;

      const styleElement = document.getElementById("font-customizer-styles");
      if (!styleElement) {
        applyStyles();
      }
    });

    observer.observe(document.documentElement, {
      childList: true,
      subtree: true,
    });
  }

  // Run the script
  init();
})();