您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Unwraps Gemini JSON, soft-fixes \\n, preserves highlight
- // ==UserScript==
- // @name Gemini JSON Wrapper – Highlight-Safe,
- // @namespace http://tampermonkey.net/
- // @version 3.0
- // @description Unwraps Gemini JSON, soft-fixes \\n, preserves highlight
- // @author copypaste
- // @match *://gemini.google.com/*
- // @grant none
- // ==/UserScript==
- (function() {
- 'use strict';
- const CONFIG = {
- wrapStyles: {
- whiteSpace: 'pre-wrap',
- wordBreak: 'break-word',
- overflowWrap: 'break-word',
- overflowX: 'auto',
- maxWidth: '100%',
- fontFamily: 'monospace',
- fontSize: '13px',
- lineHeight: '1.4'
- },
- targetSelectors: [
- '[data-text-generation-response] pre',
- '[data-text-generation-response] code',
- '.text-generation-response pre',
- '.text-generation-response code',
- '.response-output pre',
- '.response-output code',
- 'pre',
- 'code'
- ],
- checkInterval: 2000,
- minJsonLength: 100,
- dataAttribute: 'data-json-wrapped'
- };
- function looksLikeJSON(str) {
- const trimmed = str.trim();
- if (!trimmed.startsWith('{') && !trimmed.startsWith('[')) return false;
- try {
- JSON.parse(trimmed);
- return true;
- } catch {
- return false;
- }
- }
- function softUnescape(str) {
- return str.replace(/\\n/g, '\n').replace(/\\t/g, '\t');
- }
- function processCodeElements() {
- let changes = 0;
- CONFIG.targetSelectors.forEach(selector => {
- document.querySelectorAll(selector).forEach(el => {
- if (el.hasAttribute(CONFIG.dataAttribute)) return;
- const hasSpans = el.innerHTML.includes('<span');
- if (!hasSpans) {
- const txt = el.textContent.trim();
- if (txt.length >= CONFIG.minJsonLength && looksLikeJSON(txt)) {
- el.textContent = softUnescape(txt);
- changes++;
- }
- }
- Object.assign(el.style, CONFIG.wrapStyles);
- el.setAttribute(CONFIG.dataAttribute, 'true');
- });
- });
- if (changes > 0) console.debug(`[JSON Wrapper] Updated ${changes} elements`);
- }
- function debounce(func, delay) {
- let timeoutId;
- return function(...args) {
- clearTimeout(timeoutId);
- timeoutId = setTimeout(() => func.apply(this, args), delay);
- };
- }
- const debouncedProcess = debounce(processCodeElements, 500);
- function initObserver() {
- const observer = new MutationObserver((mutations) => {
- let shouldProcess = false;
- for (const mutation of mutations) {
- if (
- mutation.type === 'childList' && mutation.addedNodes.length > 0 ||
- (mutation.type === 'attributes' &&
- CONFIG.targetSelectors.some(sel => sel.includes(mutation.attributeName)))
- ) {
- shouldProcess = true;
- break;
- }
- }
- if (shouldProcess) debouncedProcess();
- });
- const containers = document.querySelectorAll('.response-output, [data-text-generation-response]');
- if (containers.length > 0) {
- containers.forEach(container => {
- observer.observe(container, {
- childList: true,
- subtree: true,
- attributes: true,
- attributeFilter: ['class', 'data-text-generation-response']
- });
- });
- } else {
- observer.observe(document.body, {
- childList: true,
- subtree: true
- });
- }
- }
- function init() {
- processCodeElements(); // Initial run
- initObserver(); // Watch for dynamic loads
- setInterval(() => {
- debouncedProcess(); // Lazy interval refresh in case observer misses something
- }, CONFIG.checkInterval);
- }
- if (document.readyState === 'loading') {
- document.addEventListener('DOMContentLoaded', init);
- } else {
- init();
- }
- })();