Greasy Fork

Greasy Fork is available in English.

网页字体修改器

导入本地字体更换网页字体,支持悬浮球控制和自定义字体颜色,新增字体清晰度调整和网页编辑模式

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name        网页字体修改器
// @namespace   greasyfork.org
// @version     4.1
// @description 导入本地字体更换网页字体,支持悬浮球控制和自定义字体颜色,新增字体清晰度调整和网页编辑模式
// @author      ^o^
// @run-at      document-start
// @match       *://*/*
// @grant       GM_registerMenuCommand
// @grant       GM_setValue
// @grant       GM_getValue
// @grant       GM_deleteValue
// @grant       GM_addStyle
// @icon        
// ==/UserScript==

(() => {
'use strict';
const DEFAULT_COLOR='#333333',FAB_SIZE=50,DEFAULT_CLARITY=0;
let editMode = false; // 编辑模式状态变量

const main=()=>{
const defaultFont={name:'system-ui(默认)',fontFamily:'system-ui',isDefault:true};
let fontData=GM_getValue('fontData',{fonts:[defaultFont],currentFont:defaultFont.name,fontColor:DEFAULT_COLOR,fabPosition:null,clarityValue:DEFAULT_CLARITY});
if(fontData.clarityValue===undefined){
fontData.clarityValue=DEFAULT_CLARITY;
GM_setValue('fontData',fontData);
}
const cachedFontBlobUrls={};
let fab=null,panel=null,overlay=null,isFabVisible=GM_getValue('fabVisible',true);
GM_addStyle(`
#via-font-fab{position:fixed;width:${FAB_SIZE}px;height:${FAB_SIZE}px;background:linear-gradient(45deg,#2196F3,#9C27B0);color:white;border-radius:50%;text-align:center;line-height:${FAB_SIZE}px;font-size:24px;font-weight:bold;box-shadow:0 4px 8px rgba(0,0,0,0.3);z-index:999999;touch-action:none;user-select:none;transition:left 0.2s,top 0.2s,transform 0.2s ease,opacity 0.2s ease;opacity:0.7;transform:scale(0.8);cursor:pointer;}
#via-font-overlay{position:fixed;top:0;left:0;right:0;bottom:0;background:rgba(0,0,0,0.3);z-index:999997;display:none;opacity:0;transition:opacity 0.3s ease;backdrop-filter:blur(3px);-webkit-backdrop-filter:blur(3px);}
#via-font-panel{position:fixed;top:50%;left:50%;transform:translate(-50%,-50%) scale(0.8);background:rgba(255,255,255,0.85);border-radius:16px;box-shadow:0 12px 30px rgba(0,0,0,0.25);padding:20px;max-height:80vh;overflow-y:auto;z-index:999998;width:90%;max-width:420px;opacity:0;transition:all 0.3s cubic-bezier(0.175,0.885,0.32,1.275);display:none;backdrop-filter:blur(20px) saturate(180%);-webkit-backdrop-filter:blur(20px) saturate(180%);border:1px solid rgba(255,255,255,0.3);}
.panel-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:15px;padding-bottom:12px;border-bottom:1px solid rgba(0,0,0,0.1);}
.panel-title{margin:0;font-weight:600;color:#333;font-size:18px;}
.close-btn{background:none;border:none;font-size:24px;cursor:pointer;color:#777;transition:color 0.2s;width:36px;height:36px;display:flex;align-items:center;justify-content:center;border-radius:50%;background:rgba(0,0,0,0.05);}
.close-btn:hover{background:rgba(0,0,0,0.08);color:#333;}
.setting-group{margin-bottom:15px;padding:12px;background:rgba(255,255,255,0.5);border-radius:12px;box-shadow:0 2px 8px rgba(0,0,0,0.05);}
.setting-label{display:block;margin-bottom:6px;font-weight:500;color:#444;font-size:14px;}
.font-select{width:100%;padding:10px 12px;border-radius:8px;border:1px solid rgba(0,0,0,0.1);font-size:14px;background:rgba(255,255,255,0.8);box-shadow:0 1px 4px rgba(0,0,0,0.05);transition:border-color 0.2s;}
.font-select:focus{border-color:#2196F3;outline:none;box-shadow:0 0 0 2px rgba(33,150,243,0.2);}
.color-controls{display:flex;align-items:center;gap:8px;}
.color-picker{width:40px;height:32px;padding:1px;border-radius:6px;border:1px solid rgba(0,0,0,0.1);background:rgba(255,255,255,0.8);cursor:pointer;}
.color-input{flex:1;padding:8px 12px;border-radius:8px;border:1px solid rgba(0,0,0,0.1);font-size:13px;background:rgba(255,255,255,0.8);box-shadow:0 1px 4px rgba(0,0,0,0.05);}
.color-reset-btn{padding:6px 12px;background:rgba(240,240,240,0.7);border:1px solid rgba(0,0,0,0.1);border-radius:6px;cursor:pointer;transition:all 0.2s;font-size:13px;}
.color-reset-btn:hover{background:rgba(224,224,224,0.7);}
.upload-area{margin:20px 0;padding:15px 12px;background:rgba(249,249,249,0.5);border-radius:10px;border:2px dashed rgba(0,0,0,0.1);text-align:center;transition:background 0.2s;}
.upload-area:hover{background:rgba(240,240,240,0.6);}
.installed-fonts-title{margin-bottom:12px;font-size:16px;color:#444;}
.fonts-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(150px,1fr));gap:12px;}
.font-item{background:rgba(245,245,245,0.7);padding:12px;border-radius:10px;text-align:center;transition:transform 0.2s,box-shadow 0.2s;box-shadow:0 1px 4px rgba(0,0,0,0.05);}
.font-item:hover{transform:translateY(-3px);box-shadow:0 3px 12px rgba(0,0,0,0.1);background:rgba(255,255,255,0.8);}
.font-name{font-size:14px;margin-bottom:10px;color:#333;font-weight:500;}
.delete-btn{padding:7px 12px;background:rgba(255,77,79,0.9);color:white;border:none;border-radius:6px;cursor:pointer;width:100%;transition:background 0.2s;font-size:13px;}
.delete-btn:hover{background:rgba(255,51,54,0.9);}
.clarity-controls{display:flex;flex-direction:column;gap:8px;margin-top:12px;padding:12px;background:rgba(255,255,255,0.5);border-radius:10px;box-shadow:0 1px 4px rgba(0,0,0,0.05);}
.clarity-slider{width:100%;height:6px;-webkit-appearance:none;appearance:none;background:#e0e0e0;border-radius:3px;outline:none;}
.clarity-slider::-webkit-slider-thumb{-webkit-appearance:none;appearance:none;width:18px;height:18px;border-radius:50%;background:#2196F3;cursor:pointer;box-shadow:0 1px 3px rgba(0,0,0,0.2);border:2px solid white;}
.clarity-value-display{text-align:center;font-size:13px;color:#444;font-weight:500;}
.clarity-description{font-size:12px;color:#666;margin-top:3px;text-align:center;}
.edit-mode-indicator{position:fixed;top:10px;right:10px;background:rgba(255,0,0,0.7);color:white;padding:5px 10px;border-radius:4px;font-size:12px;z-index:1000000;display:none;}
@media (max-width:500px){#via-font-panel{width:95%;padding:15px 10px;}.fonts-grid{grid-template-columns:repeat(auto-fill,minmax(130px,1fr));}}`);

// 创建编辑模式指示器
const createEditIndicator = () => {
  const indicator = document.createElement('div');
  indicator.className = 'edit-mode-indicator';
  indicator.textContent = '编辑模式已启用';
  indicator.id = 'edit-mode-indicator';
  document.body.appendChild(indicator);
  return indicator;
};

// 切换编辑模式
const toggleEditMode = () => {
  if (editMode) {
    exitEditMode();
  } else {
    enterEditMode();
  }
  editMode = !editMode;
};

// 进入编辑模式
const enterEditMode = () => {
  document.body.contentEditable = 'true';
  document.getElementById('edit-mode-indicator').style.display = 'block';
  GM_registerMenuCommand('🛑 退出编辑模式', toggleEditMode);
};

// 退出编辑模式
const exitEditMode = () => {
  document.body.contentEditable = 'false';
  document.getElementById('edit-mode-indicator').style.display = 'none';
  GM_registerMenuCommand('✏️ 进入编辑模式', toggleEditMode);
};

const createFAB=()=>{
fab=document.createElement('div');
fab.id='via-font-fab';
if(fontData.fabPosition){
fab.style.left=`${fontData.fabPosition.x}px`;
fab.style.top=`${fontData.fabPosition.y}px`;
}else{
fab.style.right='20px';
fab.style.bottom='30px';
}
fab.innerHTML='Aa';
document.body.appendChild(fab);
if(!isFabVisible){fab.style.display='none';}
return fab;};

const createPanel=()=>{
overlay=document.createElement('div');
overlay.id='via-font-overlay';
document.body.appendChild(overlay);
panel=document.createElement('div');
panel.id='via-font-panel';
panel.innerHTML=`<div class="panel-header"><h3 class="panel-title">字体设置</h3><button class="close-btn">×</button></div><div class="panel-content"></div>`;
document.body.appendChild(panel);
return panel;};

const setupDrag=()=>{
let startX,startY,initialX,initialY,dragging=false,fabTimer=null,edgeTimer=null;
const onTouchStart=e=>{
if(e.touches[0]){
const touch=e.touches[0];
startX=touch.clientX;
startY=touch.clientY;
initialX=fab.offsetLeft;
initialY=fab.offsetTop;
fab.style.transition='none';
fab.style.opacity='1';
fab.style.transform='scale(1)';
clearTimeout(fabTimer);
clearTimeout(edgeTimer);
document.addEventListener('touchmove',onTouchMove);
document.addEventListener('touchend',onTouchEnd);}};
const onTouchMove=e=>{
if(e.touches[0]){
const touch=e.touches[0];
const diffX=touch.clientX-startX;
const diffY=touch.clientY-startY;
if(Math.abs(diffX)>5||Math.abs(diffY)>5){dragging=true;}
let newX=initialX+diffX;
let newY=initialY+diffY;
const maxX=window.innerWidth-fab.offsetWidth;
const maxY=window.innerHeight-fab.offsetHeight;
newX=Math.max(0,Math.min(newX,maxX));
newY=Math.max(0,Math.min(newY,maxY));
fab.style.left=`${newX}px`;
fab.style.top=`${newY}px`;
fab.style.right='auto';
fab.style.bottom='auto';}};
const onTouchEnd=()=>{
document.removeEventListener('touchmove',onTouchMove);
document.removeEventListener('touchend',onTouchEnd);
fab.style.transition='left 0.2s, top 0.2s, transform 0.2s ease, opacity 0.2s ease';
if(dragging){
fontData.fabPosition={x:fab.offsetLeft,y:fab.offsetTop};
GM_setValue('fontData',fontData);
dragging=false;}
clearTimeout(fabTimer);
fabTimer=setTimeout(()=>{
fab.style.opacity='0.7';
fab.style.transform='scale(0.8)';
edgeTimer=setTimeout(()=>checkEdge(),100);},800);};
fab.addEventListener('touchstart',onTouchStart);};

const checkEdge=()=>{
if(!fab)return;
const fabRect=fab.getBoundingClientRect();
const windowWidth=window.innerWidth;
const edgeThreshold=10;
if(fabRect.left<edgeThreshold){
fab.style.transform='scale(0.8) translateX(-40%)';
fab.style.opacity='0.5';
}else if(windowWidth-fabRect.right<edgeThreshold){
fab.style.transform='scale(0.8) translateX(40%)';
fab.style.opacity='0.5';
}else{
fab.style.transform='scale(0.8)';
fab.style.opacity='0.7';}};

const createStyleElement=(elementId,highPriority=false)=>{
let styleElement=document.getElementById(elementId);
if(!styleElement){
styleElement=document.createElement('style');
styleElement.id=elementId;
if(highPriority){document.head.insertBefore(styleElement,document.head.firstChild);}
else{document.head.appendChild(styleElement);}}
return styleElement;};

const fontFaceStyleElement=createStyleElement('font-face-style',true);
const commonStyleElement=createStyleElement('font-common-style',true);
const colorStyleElement=createStyleElement('font-color-style');
const clarityStyleElement=createStyleElement('font-clarity-style');

const updateCommonStyles=()=>{
const selectedFont=fontData.fonts.find(font=>font.name===fontData.currentFont);
if(!selectedFont)return;
const cssRules=`html *:not(i):not(em):not(:empty) { font-family: "${selectedFont.fontFamily}" !important; }`;
commonStyleElement.textContent=cssRules;};

const applyColor=()=>{
if(fontData.fontColor===DEFAULT_COLOR){
colorStyleElement.textContent='';
return;}
colorStyleElement.textContent=`body, body * { color: ${fontData.fontColor} !important; }`;};

const updateFontFaces=selectedFont=>{
if(!selectedFont||!selectedFont.storageKey){
fontFaceStyleElement.textContent='';
updateCommonStyles();
return;}
const fontBlobUrl=cachedFontBlobUrls[selectedFont.storageKey]||'';
if(fontBlobUrl){
const fontFaceCss=`@font-face { font-family: "${selectedFont.fontFamily}"; src: url(${fontBlobUrl}) format('${selectedFont.format}'); }`;
fontFaceStyleElement.textContent=fontFaceCss;
updateCommonStyles();
return;}
const fontChunks=GM_getValue(`font_${selectedFont.storageKey}_chunks`,[]);
const totalChunks=GM_getValue(`font_${selectedFont.storageKey}_total`,0);
if(fontChunks.length===totalChunks){
Promise.all(fontChunks.map(index=>GM_getValue(`font_${selectedFont.storageKey}_chunk_${index}`)))
.then(base64Chunks=>{
const base64Data=base64Chunks.join('');
const blob=base64ToBlob(base64Data,selectedFont.mimeType);
const fontBlobUrl=URL.createObjectURL(blob);
cachedFontBlobUrls[selectedFont.storageKey]=fontBlobUrl;
const fontFaceCss=`@font-face { font-family: "${selectedFont.fontFamily}"; src: url(${fontBlobUrl}) format('${selectedFont.format}'); }`;
fontFaceStyleElement.textContent=fontFaceCss;
updateCommonStyles();});}};

const updateClarityStyle=()=>{
const value=fontData.clarityValue;
clarityStyleElement.textContent=`*:not(pre):not(code):not(tt):not(kbd):not(samp) {-webkit-text-stroke:${value}px !important;text-stroke:${value}px !important;}`;};

const togglePanel=()=>{
if(panel.style.display==='none'||panel.style.display===''){
overlay.style.display='block';
panel.style.display='block';
setTimeout(()=>{
overlay.style.opacity='1';
panel.style.opacity='1';
panel.style.transform='translate(-50%,-50%) scale(1)';},10);
refreshPanel();
}else{
overlay.style.opacity='0';
panel.style.opacity='0';
panel.style.transform='translate(-50%,-50%) scale(0.8)';
setTimeout(()=>{
overlay.style.display='none';
panel.style.display='none';},300);}};

const refreshPanel=()=>{
const content=panel.querySelector('.panel-content');
if(!content)return;
content.innerHTML=`
<div class="setting-group">
<label class="setting-label">当前字体</label>
<select class="font-select">
${fontData.fonts.map(font=>`<option value="${font.name}"${fontData.currentFont===font.name?' selected':''}>${font.name}</option>`).join('')}
</select></div>
<div class="setting-group">
<label class="setting-label">字体颜色</label>
<div class="color-controls">
<input type="color" class="color-picker" value="${fontData.fontColor}">
<input type="text" class="color-input" value="${fontData.fontColor}">
<button class="color-reset-btn">重置</button></div></div>
<div class="setting-group">
<label class="setting-label">字体清晰度</label>
<div class="clarity-controls">
<input type="range" min="0" max="0.8" step="0.05" value="${fontData.clarityValue}" class="clarity-slider">
<div class="clarity-value-display">当前值:${fontData.clarityValue.toFixed(2)}</div>
<div class="clarity-description">调整字体边缘清晰度(0为默认,0.8为最清晰)</div></div></div>
<div class="upload-area">
<label class="setting-label">上传字体文件(.ttf,.otf,.woff,.woff2)</label>
<input type="file" accept=".ttf,.otf,.woff,.woff2" multiple style="width:100%;padding:8px"></div>
<div class="setting-group">
<h4 class="installed-fonts-title">已安装字体(${fontData.fonts.length})</h4>
<div class="fonts-grid">
${fontData.fonts.filter(f=>!f.isDefault).map(font=>`<div class="font-item"><div class="font-name">${font.name}</div><button data-font="${font.name}" class="delete-btn">删除</button></div>`).join('')}</div></div>`;
setupPanelEvents();};

const setupPanelEvents=()=>{
panel.querySelector('.font-select').addEventListener('change',e=>{
fontData.currentFont=e.target.value;
const selectedFont=fontData.fonts.find(f=>f.name===fontData.currentFont);
if(selectedFont){
updateFontFaces(selectedFont);
GM_setValue('fontData',fontData);}});
const colorPicker=panel.querySelector('.color-picker');
const colorInput=panel.querySelector('.color-input');
const colorResetBtn=panel.querySelector('.color-reset-btn');
colorPicker.addEventListener('input',e=>{
fontData.fontColor=e.target.value;
colorInput.value=e.target.value;
applyColor();
GM_setValue('fontData',fontData);});
colorInput.addEventListener('input',e=>{
const value=e.target.value;
if(/^#([A-Fa-f0-9]{6}|[A-Fa-f0-9]{3})$/.test(value)){
fontData.fontColor=value;
colorPicker.value=value;
applyColor();
GM_setValue('fontData',fontData);}});
colorResetBtn.addEventListener('click',()=>{
fontData.fontColor=DEFAULT_COLOR;
colorPicker.value=DEFAULT_COLOR;
colorInput.value=DEFAULT_COLOR;
applyColor();
GM_setValue('fontData',fontData);});
panel.querySelector('input[type="file"]').addEventListener('change',e=>{
handleFontUpload(Array.from(e.target.files));});
panel.querySelectorAll('.delete-btn').forEach(btn=>{
btn.addEventListener('click',()=>{
const fontName=btn.dataset.font;
handleDeleteFont(fontName);});});
panel.querySelector('.close-btn').addEventListener('click',togglePanel);
overlay.addEventListener('click',e=>{if(e.target===overlay)togglePanel();});
const claritySlider=panel.querySelector('.clarity-slider');
if(claritySlider){
claritySlider.addEventListener('input',e=>{
const value=parseFloat(e.target.value);
fontData.clarityValue=value;
panel.querySelector('.clarity-value-display').textContent=`当前值:${value.toFixed(2)}`;
updateClarityStyle();
GM_setValue('fontData',fontData);});}};

const handleFontUpload=async files=>{
for(const file of files){await processFontFile(file);}
refreshPanel();};

const processFontFile=file=>{
return new Promise(resolve=>{
const originalName=file.name.replace(/\.[^/.]+$/,"");
const extension=file.name.slice(file.name.lastIndexOf('.'));
let newName=originalName;
let index=2;
while(fontData.fonts.some(f=>f.name===newName)){
newName=`${originalName}(${index})`;
index++;}
const reader=new FileReader();
reader.onload=()=>{
const result=reader.result;
const base64Data=result.split(',')[1];
const mimeType=result.split(',')[0].split(':')[1];
const storageKey='font_'+Date.now();
const format=getFontFormat(file.name);
const chunkSize=500000;
const chunks=[];
for(let i=0;i<base64Data.length;i+=chunkSize){
const chunk=base64Data.substring(i,i+chunkSize);
GM_setValue(`font_${storageKey}_chunk_${chunks.length}`,chunk);
chunks.push(chunks.length);}
GM_setValue(`font_${storageKey}_chunks`,chunks);
GM_setValue(`font_${storageKey}_total`,chunks.length);
fontData.fonts.push({
name:newName,
fontFamily:newName,
originalFileName:file.name,
mimeType:mimeType,
storageKey:storageKey,
format:format,
fileSize:file.size});
fontData.currentFont=newName;
GM_setValue('fontData',fontData);
const selectedFont=fontData.fonts.find(f=>f.name===newName);
if(selectedFont){updateFontFaces(selectedFont);}
resolve();};
reader.readAsDataURL(file);});};

const handleDeleteFont=fontName=>{
if(!confirm(`确定要删除字体"${fontName}"吗?`))return;
const fontIndex=fontData.fonts.findIndex(f=>f.name===fontName);
if(fontIndex===-1)return;
const font=fontData.fonts[fontIndex];
if(font.storageKey){
const fontChunks=GM_getValue(`font_${font.storageKey}_chunks`,[]);
fontChunks.forEach((_,i)=>GM_deleteValue(`font_${font.storageKey}_chunk_${i}`));
GM_deleteValue(`font_${font.storageKey}_chunks`);
GM_deleteValue(`font_${font.storageKey}_total`);
if(cachedFontBlobUrls[font.storageKey]){
URL.revokeObjectURL(cachedFontBlobUrls[font.storageKey]);
delete cachedFontBlobUrls[font.storageKey];}}
fontData.fonts.splice(fontIndex,1);
if(fontData.currentFont===fontName){
fontData.currentFont=fontData.fonts[0].name;}
GM_setValue('fontData',fontData);
const selectedFont=fontData.fonts.find(f=>f.name===fontData.currentFont);
if(selectedFont){updateFontFaces(selectedFont);}
refreshPanel();};

const base64ToBlob=(base64String,mimeType)=>{
const byteCharacters=atob(base64String);
const byteArrays=[];
for(let i=0;i<byteCharacters.length;i+=512){
const slice=byteCharacters.slice(i,i+512);
const byteNumbers=new Array(slice.length);
for(let j=0;j<slice.length;j++){byteNumbers[j]=slice.charCodeAt(j);}
byteArrays.push(new Uint8Array(byteNumbers));}
return new Blob(byteArrays,{type:mimeType});};

const getFontFormat=fileName=>{
const ext=fileName.split('.').pop().toLowerCase();
return{'ttf':'truetype','otf':'opentype','woff':'woff','woff2':'woff2'}[ext]||'truetype';};

fab=createFAB();
createPanel();
setupDrag();
createEditIndicator(); // 创建编辑模式指示器
window.addEventListener('resize',checkEdge);
checkEdge();
fab.addEventListener('click',togglePanel);
const selectedFont=fontData.fonts.find(font=>font.name===fontData.currentFont);
if(selectedFont){updateFontFaces(selectedFont);}
applyColor();
updateClarityStyle();
GM_registerMenuCommand('🎨 打开字体设置',togglePanel);
GM_registerMenuCommand('🔄 切换悬浮球显示',()=>{
isFabVisible=!isFabVisible;
fab.style.display=isFabVisible?'block':'none';
GM_setValue('fabVisible',isFabVisible);});
// 初始化编辑模式菜单
GM_registerMenuCommand('✏️ 进入编辑模式', toggleEditMode);};

if(document.readyState==='loading'){
document.addEventListener('DOMContentLoaded',main);
}else{main();}
})();