0.0.7 #11
| 
						 | 
				
			
			@ -61,6 +61,48 @@ jobs:
 | 
			
		|||
        name: firefox-addon
 | 
			
		||||
        path: dist/
 | 
			
		||||
 | 
			
		||||
  build-for-safari:
 | 
			
		||||
    name: 构建 Safari 扩展程序
 | 
			
		||||
    runs-on: macos
 | 
			
		||||
    env:
 | 
			
		||||
      VITE_RUN_ID: ${{ gitea.run_number }}
 | 
			
		||||
      VITE_HASH_ID: ${{ gitea.sha }}
 | 
			
		||||
    
 | 
			
		||||
    steps:
 | 
			
		||||
    - uses: actions/checkout@v3
 | 
			
		||||
      name: 检出代码
 | 
			
		||||
    
 | 
			
		||||
    - name: 设置 Node.js
 | 
			
		||||
      uses: actions/setup-node@v3
 | 
			
		||||
      with:
 | 
			
		||||
        node-version: '22'
 | 
			
		||||
        
 | 
			
		||||
    - name: 安装依赖
 | 
			
		||||
      run: npm install
 | 
			
		||||
      
 | 
			
		||||
    - name: 构建扩展程序
 | 
			
		||||
      run: npm run build:safari
 | 
			
		||||
      
 | 
			
		||||
    - name: 设置 Xcode
 | 
			
		||||
      uses: maxim-lobanov/setup-xcode@v1
 | 
			
		||||
      with:
 | 
			
		||||
        xcode-version: latest-stable
 | 
			
		||||
        
 | 
			
		||||
    - name: 创建 Safari 扩展项目
 | 
			
		||||
      run: |
 | 
			
		||||
        xcrun safari-web-extension-converter dist --project-location safari-extension --app-name "MSR Mod" --bundle-identifier "moe.astrian.ext-msrmod" --swift --no-open
 | 
			
		||||
        
 | 
			
		||||
    - name: 构建 Safari 扩展
 | 
			
		||||
      run: |
 | 
			
		||||
        cd "safari-extension/MSR Mod"
 | 
			
		||||
        xcodebuild -project "MSR Mod.xcodeproj" -scheme "MSR Mod (macOS)" -configuration Release -destination "generic/platform=macOS" build
 | 
			
		||||
        
 | 
			
		||||
    - name: 上传构建工件
 | 
			
		||||
      uses: actions/upload-artifact@v3
 | 
			
		||||
      with:
 | 
			
		||||
        name: safari-extension
 | 
			
		||||
        path: safari-extension/
 | 
			
		||||
 | 
			
		||||
  publish-to-chrome-webstore: 
 | 
			
		||||
    name: 发布至 Chrome 应用商店
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										29
									
								
								CLAUDE.md
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								CLAUDE.md
									
									
									
									
									
								
							| 
						 | 
				
			
			@ -19,6 +19,7 @@ npm i                       # Install dependencies
 | 
			
		|||
```bash
 | 
			
		||||
npm run build:chrome        # Build for Chrome/Chromium browsers
 | 
			
		||||
npm run build:firefox       # Build for Firefox
 | 
			
		||||
npm run build:safari        # Build for Safari (uses background.html)
 | 
			
		||||
npm run build              # Default build (Chrome)
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -71,6 +72,7 @@ npm run qc                 # Alias for quality-check
 | 
			
		|||
### Browser Compatibility
 | 
			
		||||
- **Chrome**: Uses service worker, full CSP support
 | 
			
		||||
- **Firefox**: Uses background scripts, modified CSP, specific gecko settings
 | 
			
		||||
- **Safari**: Uses background page (background.html) instead of service worker
 | 
			
		||||
- **Prebuild Scripts**: Automatically modify manifest.json and HTML for each platform
 | 
			
		||||
 | 
			
		||||
### Storage Strategy
 | 
			
		||||
| 
						 | 
				
			
			@ -99,6 +101,7 @@ npm run qc                 # Alias for quality-check
 | 
			
		|||
### `/scripts/`
 | 
			
		||||
- **prebuild-chrome.js**: Removes localhost dev configs for production
 | 
			
		||||
- **prebuild-firefox.js**: Adapts manifest for Firefox compatibility
 | 
			
		||||
- **prebuild-safari.js**: Creates background.html and adapts manifest for Safari
 | 
			
		||||
 | 
			
		||||
### `/public/`
 | 
			
		||||
- **manifest.json**: Extension manifest (modified by prebuild scripts)
 | 
			
		||||
| 
						 | 
				
			
			@ -128,3 +131,29 @@ npm run qc                 # Alias for quality-check
 | 
			
		|||
- Graceful fallbacks for storage API unavailability
 | 
			
		||||
- Resource URL rotation handling with automatic refresh
 | 
			
		||||
- Cross-browser compatibility with feature detection
 | 
			
		||||
 | 
			
		||||
## Safari Extension Considerations
 | 
			
		||||
 | 
			
		||||
### Background Script Handling
 | 
			
		||||
Safari Web Extensions have different requirements for background scripts:
 | 
			
		||||
 | 
			
		||||
1. **Background Page vs Service Worker**: Safari uses `background.page` instead of `service_worker`
 | 
			
		||||
2. **Background HTML**: The prebuild script creates `background.html` that loads `background.js`
 | 
			
		||||
3. **Manifest Configuration**: Uses `"background": { "page": "background.html", "persistent": false }`
 | 
			
		||||
 | 
			
		||||
### Auto-redirect Functionality
 | 
			
		||||
The auto-redirect feature in Safari may require special handling due to:
 | 
			
		||||
- Different WebKit extension APIs
 | 
			
		||||
- Safari's stricter security policies
 | 
			
		||||
- Tab management differences from Chromium
 | 
			
		||||
 | 
			
		||||
### Building for Safari
 | 
			
		||||
```bash
 | 
			
		||||
npm run build:safari    # Creates background.html and Safari-specific manifest
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
The Safari build process:
 | 
			
		||||
1. Removes localhost development configurations
 | 
			
		||||
2. Converts `service_worker` to `background.page`
 | 
			
		||||
3. Creates `background.html` wrapper for `background.js`
 | 
			
		||||
4. Adds Safari-specific browser settings
 | 
			
		||||
| 
						 | 
				
			
			@ -8,6 +8,7 @@
 | 
			
		|||
		"build": "echo 'No platform specified, will build for Chromium.' && npm run build-chrome",
 | 
			
		||||
		"build:chrome": "npm run prebuild:chrome && vue-tsc -b && vite build && cp -r public/* dist/",
 | 
			
		||||
		"build:firefox": "npm run prebuild:firefox && vue-tsc -b && vite build && cp -r public/* dist/",
 | 
			
		||||
		"build:safari": "npm run prebuild:safari && vue-tsc -b && vite build && cp -r public/* dist/",
 | 
			
		||||
		"dev:refresh": "vue-tsc -b && vite build && cp -r public/* dist/",
 | 
			
		||||
		"build:watch": "vite build --watch",
 | 
			
		||||
		"preview": "vite preview",
 | 
			
		||||
| 
						 | 
				
			
			@ -15,7 +16,8 @@
 | 
			
		|||
		"quality-check": "biome ci",
 | 
			
		||||
		"qc": "npm run quality-check",
 | 
			
		||||
		"prebuild:chrome": "node scripts/prebuild-chrome.js",
 | 
			
		||||
		"prebuild:firefox": "node scripts/prebuild-firefox.js"
 | 
			
		||||
		"prebuild:firefox": "node scripts/prebuild-firefox.js",
 | 
			
		||||
		"prebuild:safari": "node scripts/prebuild-safari.js"
 | 
			
		||||
	},
 | 
			
		||||
	"dependencies": {
 | 
			
		||||
		"@tailwindcss/vite": "^4.1.7",
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										359
									
								
								scripts/prebuild-safari.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										359
									
								
								scripts/prebuild-safari.js
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,359 @@
 | 
			
		|||
import fs from 'fs';
 | 
			
		||||
import path from 'path';
 | 
			
		||||
import { fileURLToPath } from 'url';
 | 
			
		||||
 | 
			
		||||
const __filename = fileURLToPath(import.meta.url);
 | 
			
		||||
const __dirname = path.dirname(__filename);
 | 
			
		||||
 | 
			
		||||
// 处理 manifest.json for Safari
 | 
			
		||||
function processManifest() {
 | 
			
		||||
  const manifestPath = path.join(__dirname, '../public/manifest.json');
 | 
			
		||||
  const manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf8'));
 | 
			
		||||
 | 
			
		||||
  // 移除本地调试相关的配置
 | 
			
		||||
  if (manifest.host_permissions) {
 | 
			
		||||
    manifest.host_permissions = manifest.host_permissions.filter(
 | 
			
		||||
      permission => !permission.includes('localhost')
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (manifest.content_security_policy && manifest.content_security_policy.extension_pages) {
 | 
			
		||||
    // 移除 CSP 中的本地开发相关配置
 | 
			
		||||
    manifest.content_security_policy.extension_pages = manifest.content_security_policy.extension_pages
 | 
			
		||||
      .replace(/script-src 'self' http:\/\/localhost:5173;\s*/g, '')
 | 
			
		||||
      .replace(/\s*http:\/\/localhost:5173\s*/g, ' ')
 | 
			
		||||
      .replace(/\s*ws:\/\/localhost:5173\s*/g, ' ')
 | 
			
		||||
      .replace(/;\s+/g, '; ') // 标准化分号后的空格
 | 
			
		||||
      .replace(/\s+/g, ' ') // 合并多个空格为一个
 | 
			
		||||
      .trim();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Safari 特殊处理:添加 appShell.html 到 content scripts 匹配
 | 
			
		||||
  if (manifest.content_scripts && manifest.content_scripts[0]) {
 | 
			
		||||
    // 添加 appShell.html 的匹配规则
 | 
			
		||||
    const existingMatches = manifest.content_scripts[0].matches;
 | 
			
		||||
    if (!existingMatches.includes("https://monster-siren.hypergryph.com/")) {
 | 
			
		||||
      existingMatches.push("https://monster-siren.hypergryph.com/");
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Safari 特殊处理:使用 background.page 而不是 service_worker
 | 
			
		||||
  if (manifest.background && manifest.background.service_worker) {
 | 
			
		||||
    // Safari 扩展在 Manifest V3 中必须使用 persistent: false
 | 
			
		||||
    // 但为了调试,我们暂时设为 true 来确保页面加载
 | 
			
		||||
    manifest.background = {
 | 
			
		||||
      page: "background.html",
 | 
			
		||||
      persistent: true
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // 创建 background.html 文件用于 Safari
 | 
			
		||||
  const backgroundHtmlPath = path.join(__dirname, '../public/background.html');
 | 
			
		||||
  const backgroundHtmlContent = `<!DOCTYPE html>
 | 
			
		||||
<html>
 | 
			
		||||
<head>
 | 
			
		||||
  <meta charset="utf-8">
 | 
			
		||||
  <title>MSR Mod Background</title>
 | 
			
		||||
</head>
 | 
			
		||||
<body>
 | 
			
		||||
  <h1>MSR Mod Background Page</h1>
 | 
			
		||||
  <p>If you can see this page, the background page is loaded!</p>
 | 
			
		||||
  <div id="log"></div>
 | 
			
		||||
  
 | 
			
		||||
  <script>
 | 
			
		||||
    // 创建日志函数,同时显示在页面和控制台
 | 
			
		||||
    function log(message) {
 | 
			
		||||
      console.log(message);
 | 
			
		||||
      const logDiv = document.getElementById('log');
 | 
			
		||||
      if (logDiv) {
 | 
			
		||||
        logDiv.innerHTML += '<div>' + message + '</div>';
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    log('=== SAFARI BACKGROUND PAGE LOADED ===');
 | 
			
		||||
    log('Document ready state: ' + document.readyState);
 | 
			
		||||
    log('Location: ' + location.href);
 | 
			
		||||
    log('Time: ' + new Date().toISOString());
 | 
			
		||||
    
 | 
			
		||||
    // 确保在 Safari 中正确加载脚本
 | 
			
		||||
    try {
 | 
			
		||||
      log('Safari extension context: ' + JSON.stringify({
 | 
			
		||||
        chrome: typeof chrome,
 | 
			
		||||
        browser: typeof browser,
 | 
			
		||||
        safari: typeof safari
 | 
			
		||||
      }));
 | 
			
		||||
    } catch (e) {
 | 
			
		||||
      log('Error in background.html: ' + e.message);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    // 监听事件
 | 
			
		||||
    document.addEventListener('DOMContentLoaded', function() {
 | 
			
		||||
      log('=== DOMContentLoaded fired ===');
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
    window.addEventListener('load', function() {
 | 
			
		||||
      log('=== Window load fired ===');
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
    log('About to load background.js...');
 | 
			
		||||
  </script>
 | 
			
		||||
  <script src="background.js" onload="log('background.js loaded successfully')" onerror="log('Failed to load background.js')"></script>
 | 
			
		||||
  <script>
 | 
			
		||||
    log('=== After background.js script tag ===');
 | 
			
		||||
  </script>
 | 
			
		||||
</body>
 | 
			
		||||
</html>`;
 | 
			
		||||
  fs.writeFileSync(backgroundHtmlPath, backgroundHtmlContent);
 | 
			
		||||
 | 
			
		||||
  // 创建 Safari 兼容的 background.js
 | 
			
		||||
  const backgroundJsPath = path.join(__dirname, '../public/background.js');
 | 
			
		||||
  let backgroundJsContent = fs.readFileSync(backgroundJsPath, 'utf8');
 | 
			
		||||
  
 | 
			
		||||
  // 检查是否已经添加过 Safari 代码,避免重复
 | 
			
		||||
  if (backgroundJsContent.includes('=== Safari background.js starting ===')) {
 | 
			
		||||
    console.log('Safari background.js already processed, skipping...');
 | 
			
		||||
  } else {
 | 
			
		||||
    // 在开头添加 Safari 调试信息(只添加一次)
 | 
			
		||||
    const safariDebugCode = `
 | 
			
		||||
console.log("=== Safari background.js starting ===");
 | 
			
		||||
console.log("Available APIs:", {
 | 
			
		||||
  chrome: typeof chrome,
 | 
			
		||||
  browser: typeof browser,
 | 
			
		||||
  safari: typeof safari
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
// Safari 特殊处理
 | 
			
		||||
if (typeof chrome === 'undefined' && typeof browser === 'undefined') {
 | 
			
		||||
  console.log("No extension APIs available in Safari");
 | 
			
		||||
  // 如果没有扩展 API,创建一个空的对象避免错误
 | 
			
		||||
  window.chrome = {
 | 
			
		||||
    webRequest: { onBeforeRequest: { addListener: () => {} } },
 | 
			
		||||
    storage: { sync: { get: () => Promise.resolve({}) } },
 | 
			
		||||
    tabs: { create: () => {}, remove: () => {}, update: () => {} },
 | 
			
		||||
    runtime: { 
 | 
			
		||||
      getURL: (path) => path,
 | 
			
		||||
      onMessage: { addListener: () => {} }
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Safari 消息监听器:处理来自 content script 的重定向请求
 | 
			
		||||
if (typeof chrome !== 'undefined' && chrome.runtime && chrome.runtime.onMessage) {
 | 
			
		||||
  chrome.runtime.onMessage.addListener((message, sender, sendResponse) => {
 | 
			
		||||
    console.log('Background received message:', message);
 | 
			
		||||
    
 | 
			
		||||
    if (message.action === 'redirect_to_extension') {
 | 
			
		||||
      console.log('Processing redirect request from content script');
 | 
			
		||||
      
 | 
			
		||||
      try {
 | 
			
		||||
        // 创建新标签页并打开扩展
 | 
			
		||||
        const extensionUrl = chrome.runtime.getURL('index.html');
 | 
			
		||||
        chrome.tabs.create({ url: extensionUrl }, (newTab) => {
 | 
			
		||||
          console.log('New extension tab created:', newTab.id);
 | 
			
		||||
          
 | 
			
		||||
          // 关闭原始标签页
 | 
			
		||||
          if (sender.tab && sender.tab.id) {
 | 
			
		||||
            chrome.tabs.remove(sender.tab.id);
 | 
			
		||||
          }
 | 
			
		||||
          
 | 
			
		||||
          sendResponse({ success: true, url: extensionUrl });
 | 
			
		||||
        });
 | 
			
		||||
      } catch (error) {
 | 
			
		||||
        console.error('Failed to redirect:', error);
 | 
			
		||||
        sendResponse({ success: false, error: error.message });
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
      return true; // 保持消息通道开放
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
`;
 | 
			
		||||
    
 | 
			
		||||
    // 替换 Safari 的重定向 URL 监听
 | 
			
		||||
    backgroundJsContent = backgroundJsContent.replace(
 | 
			
		||||
      /{ urls: \['https:\/\/monster-siren\.hypergryph\.com\/api\/fontset', 'https:\/\/monster-siren\.hypergryph\.com\/manifest\.json'\] }/g,
 | 
			
		||||
      "{ urls: ['https://monster-siren.hypergryph.com/api/fontset', 'https://monster-siren.hypergryph.com/manifest.json', 'https://monster-siren.hypergryph.com/'] }"
 | 
			
		||||
    );
 | 
			
		||||
    
 | 
			
		||||
    // 替换 Safari 的重定向判断逻辑
 | 
			
		||||
    backgroundJsContent = backgroundJsContent.replace(
 | 
			
		||||
      /details\.url === 'https:\/\/monster-siren\.hypergryph\.com\/manifest\.json'/g,
 | 
			
		||||
      "(details.url === 'https://monster-siren.hypergryph.com/manifest.json' || details.url === 'https://monster-siren.hypergryph.com/')"
 | 
			
		||||
    );
 | 
			
		||||
    
 | 
			
		||||
    // 清理可能的重复条件
 | 
			
		||||
    backgroundJsContent = backgroundJsContent.replace(
 | 
			
		||||
      /\(\(details\.url === 'https:\/\/monster-siren\.hypergryph\.com\/manifest\.json' \|\| details\.url === 'https:\/\/monster-siren\.hypergryph\.com\/'\) \|\| details\.url === 'https:\/\/monster-siren\.hypergryph\.com\/'\)/g,
 | 
			
		||||
      "(details.url === 'https://monster-siren.hypergryph.com/manifest.json' || details.url === 'https://monster-siren.hypergryph.com/')"
 | 
			
		||||
    );
 | 
			
		||||
    
 | 
			
		||||
    backgroundJsContent = safariDebugCode + backgroundJsContent;
 | 
			
		||||
  }
 | 
			
		||||
  fs.writeFileSync(backgroundJsPath, backgroundJsContent);
 | 
			
		||||
  console.log('✅ Safari-compatible background.js created');
 | 
			
		||||
 | 
			
		||||
  // 创建 Safari 专用的 content.js
 | 
			
		||||
  const contentJsPath = path.join(__dirname, '../public/content.js');
 | 
			
		||||
  
 | 
			
		||||
  // 检查是否已经处理过 content.js
 | 
			
		||||
  const existingContentJs = fs.existsSync(contentJsPath) ? fs.readFileSync(contentJsPath, 'utf8') : '';
 | 
			
		||||
  if (existingContentJs.includes('checkRedirectPreference')) {
 | 
			
		||||
    console.log('Safari content.js already processed, skipping...');
 | 
			
		||||
  } else {
 | 
			
		||||
    const contentJsContent = `
 | 
			
		||||
// Safari 扩展 content script for redirect
 | 
			
		||||
console.log('MSR Mod content script loaded on:', window.location.href);
 | 
			
		||||
 | 
			
		||||
// 兼容 Safari 的浏览器 API
 | 
			
		||||
const browserAPI = typeof browser !== 'undefined' ? browser : chrome;
 | 
			
		||||
 | 
			
		||||
// 异步函数:检查重定向偏好设置
 | 
			
		||||
async function checkRedirectPreference() {
 | 
			
		||||
  try {
 | 
			
		||||
    console.log('Checking redirect preferences...');
 | 
			
		||||
    
 | 
			
		||||
    // 读取偏好设置
 | 
			
		||||
    const pref = await browserAPI.storage.sync.get('preferences');
 | 
			
		||||
    console.log('Retrieved preferences:', pref);
 | 
			
		||||
    
 | 
			
		||||
    // 检查自动重定向设置(默认为 true)
 | 
			
		||||
    const shouldRedirect = pref === undefined || 
 | 
			
		||||
                          pref.preferences === undefined || 
 | 
			
		||||
                          pref.preferences.autoRedirect === undefined || 
 | 
			
		||||
                          pref.preferences.autoRedirect === true;
 | 
			
		||||
    
 | 
			
		||||
    console.log('Should redirect:', shouldRedirect);
 | 
			
		||||
    return shouldRedirect;
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    console.error('Error reading preferences:', error);
 | 
			
		||||
    // 如果读取偏好设置失败,默认重定向
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// 执行重定向的函数
 | 
			
		||||
function performRedirect() {
 | 
			
		||||
  console.log('Performing redirect to extension...');
 | 
			
		||||
  
 | 
			
		||||
  try {
 | 
			
		||||
    // 对于 Safari,我们需要使用消息传递来请求重定向
 | 
			
		||||
    // 因为 content script 无法直接访问 chrome.runtime.getURL
 | 
			
		||||
    
 | 
			
		||||
    // 方案1:尝试通过消息传递
 | 
			
		||||
    if (typeof chrome !== 'undefined' && chrome.runtime) {
 | 
			
		||||
      chrome.runtime.sendMessage({action: 'redirect_to_extension'}, (response) => {
 | 
			
		||||
        if (chrome.runtime.lastError) {
 | 
			
		||||
          console.log('Message sending failed, trying direct redirect...');
 | 
			
		||||
          // 方案2:尝试直接重定向(可能在某些情况下有效)
 | 
			
		||||
          window.location.href = 'safari-web-extension://[extension-id]/index.html';
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    } else {
 | 
			
		||||
      console.log('Chrome runtime not available, trying alternative redirect...');
 | 
			
		||||
      // 方案3:显示提示让用户手动打开扩展
 | 
			
		||||
      document.body.innerHTML = \`
 | 
			
		||||
        <div style="
 | 
			
		||||
          position: fixed; 
 | 
			
		||||
          top: 50%; 
 | 
			
		||||
          left: 50%; 
 | 
			
		||||
          transform: translate(-50%, -50%);
 | 
			
		||||
          background: #1a1a1a; 
 | 
			
		||||
          color: white; 
 | 
			
		||||
          padding: 20px; 
 | 
			
		||||
          border-radius: 8px;
 | 
			
		||||
          text-align: center;
 | 
			
		||||
          font-family: -apple-system, BlinkMacSystemFont, sans-serif;
 | 
			
		||||
          z-index: 10000;
 | 
			
		||||
        ">
 | 
			
		||||
          <h2>MSR Mod Extension Detected</h2>
 | 
			
		||||
          <p>Please click the MSR Mod extension icon in your Safari toolbar to open the app.</p>
 | 
			
		||||
          <button onclick="window.close()" style="
 | 
			
		||||
            background: #007AFF; 
 | 
			
		||||
            color: white; 
 | 
			
		||||
            border: none; 
 | 
			
		||||
            padding: 10px 20px; 
 | 
			
		||||
            border-radius: 4px; 
 | 
			
		||||
            cursor: pointer;
 | 
			
		||||
            margin-top: 10px;
 | 
			
		||||
          ">Close Tab</button>
 | 
			
		||||
        </div>
 | 
			
		||||
      \`;
 | 
			
		||||
    }
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    console.error('Redirect failed:', error);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// 主逻辑:检查页面并根据偏好设置决定是否重定向
 | 
			
		||||
async function main() {
 | 
			
		||||
  // 检查是否是目标页面
 | 
			
		||||
  if (window.location.pathname === '/' || window.location.href.includes('appShell.html')) {
 | 
			
		||||
    console.log('Detected target page, checking preferences...');
 | 
			
		||||
    
 | 
			
		||||
    // 检查偏好设置
 | 
			
		||||
    const shouldRedirect = await checkRedirectPreference();
 | 
			
		||||
    
 | 
			
		||||
    if (shouldRedirect) {
 | 
			
		||||
      console.log('Auto-redirect is enabled, proceeding with redirect...');
 | 
			
		||||
      performRedirect();
 | 
			
		||||
    } else {
 | 
			
		||||
      console.log('Auto-redirect is disabled, skipping redirect.');
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// 执行主逻辑
 | 
			
		||||
main().catch(error => {
 | 
			
		||||
  console.error('Error in main function:', error);
 | 
			
		||||
});
 | 
			
		||||
`;
 | 
			
		||||
    
 | 
			
		||||
    fs.writeFileSync(contentJsPath, contentJsContent);
 | 
			
		||||
  }
 | 
			
		||||
  console.log('✅ Safari-compatible content.js created');
 | 
			
		||||
 | 
			
		||||
  // Safari 可能需要额外的权限
 | 
			
		||||
  if (!manifest.permissions.includes('activeTab')) {
 | 
			
		||||
    manifest.permissions.push('activeTab');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // 添加 Safari 特有配置
 | 
			
		||||
  manifest.browser_specific_settings = {
 | 
			
		||||
    safari: {
 | 
			
		||||
      minimum_version: "14.0"
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  fs.writeFileSync(manifestPath, JSON.stringify(manifest, null, 2));
 | 
			
		||||
  console.log('✅ Safari Manifest.json processed');
 | 
			
		||||
  console.log('✅ Background.html created for Safari');
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// 处理 index.html
 | 
			
		||||
function processIndexHtml() {
 | 
			
		||||
  const indexPath = path.join(__dirname, '../index.html');
 | 
			
		||||
  let content = fs.readFileSync(indexPath, 'utf8');
 | 
			
		||||
 | 
			
		||||
  // 替换脚本地址
 | 
			
		||||
  content = content.replace(
 | 
			
		||||
    /src="[^"]*\/src\/main\.ts"/g,
 | 
			
		||||
    'src="./src/main.ts"'
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  // 移除 crossorigin 属性
 | 
			
		||||
  content = content.replace(/\s+crossorigin/g, '');
 | 
			
		||||
 | 
			
		||||
  fs.writeFileSync(indexPath, content);
 | 
			
		||||
  console.log('✅ Index.html processed for Safari');
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// 执行处理
 | 
			
		||||
try {
 | 
			
		||||
  processManifest();
 | 
			
		||||
  processIndexHtml();
 | 
			
		||||
  console.log('🎉 Safari build preparation completed!');
 | 
			
		||||
} catch (error) {
 | 
			
		||||
  console.error('❌ Error during Safari build preparation:', error);
 | 
			
		||||
  process.exit(1);
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										15
									
								
								src/assets/icons/soundwave.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								src/assets/icons/soundwave.vue
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,15 @@
 | 
			
		|||
<script setup lang="ts">
 | 
			
		||||
defineProps<{
 | 
			
		||||
	size: number
 | 
			
		||||
}>()
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
	<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="currentColor" :class="`w-${size} h-${size}`">
 | 
			
		||||
		<rect x="3" y="9" width="2" height="6" rx="1"></rect>
 | 
			
		||||
		<rect x="7" y="5" width="2" height="14" rx="1"></rect>
 | 
			
		||||
		<rect x="11" y="7" width="2" height="10" rx="1"></rect>
 | 
			
		||||
		<rect x="15" y="4" width="2" height="16" rx="1"></rect>
 | 
			
		||||
		<rect x="19" y="10" width="2" height="4" rx="1"></rect>
 | 
			
		||||
	</svg>
 | 
			
		||||
</template>
 | 
			
		||||
| 
						 | 
				
			
			@ -1,10 +1,11 @@
 | 
			
		|||
<script setup lang="ts">
 | 
			
		||||
import { usePlayQueueStore } from '../stores/usePlayQueueStore'
 | 
			
		||||
import { artistsOrganize } from '../utils'
 | 
			
		||||
import { artistsOrganize, supportsWebAudioVisualization } from '../utils'
 | 
			
		||||
 | 
			
		||||
import XIcon from '../assets/icons/x.vue'
 | 
			
		||||
import UpHyphenIcon from '../assets/icons/uphypen.vue'
 | 
			
		||||
import DownHyphenIcon from '../assets/icons/downhyphen.vue'
 | 
			
		||||
import SoundwaveIcon from '../assets/icons/soundwave.vue'
 | 
			
		||||
 | 
			
		||||
import { ref } from 'vue'
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -18,6 +19,9 @@ const playQueueStore = usePlayQueueStore()
 | 
			
		|||
 | 
			
		||||
const hover = ref(false)
 | 
			
		||||
 | 
			
		||||
// 检查浏览器是否支持音频可视化
 | 
			
		||||
const isAudioVisualizationSupported = supportsWebAudioVisualization()
 | 
			
		||||
 | 
			
		||||
function moveUp() {
 | 
			
		||||
	if (props.index === 0) return
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -156,12 +160,14 @@ function removeItem() {
 | 
			
		|||
				<img :src="queueItem.album?.coverUrl" />
 | 
			
		||||
				<div class="w-full h-full absolute top-0 left-0 bg-neutral-900/75 flex justify-center items-center"
 | 
			
		||||
					v-if="isCurrent">
 | 
			
		||||
					<div style="height: 1rem;" class="flex justify-center items-center gap-[.125rem]">
 | 
			
		||||
					<!-- 在支持的浏览器上显示可视化,否则显示音波图标 -->
 | 
			
		||||
					<div v-if="isAudioVisualizationSupported" style="height: 1rem;" class="flex justify-center items-center gap-[.125rem]">
 | 
			
		||||
						<div class="bg-white w-[.125rem] rounded-full" v-for="(bar, index) in playQueueStore.visualizer"
 | 
			
		||||
							:key="index" :style="{
 | 
			
		||||
								height: `${Math.max(10, bar)}%`
 | 
			
		||||
							}" />
 | 
			
		||||
					</div>
 | 
			
		||||
					<SoundwaveIcon v-else :size="6" class="text-white animate-pulse" />
 | 
			
		||||
				</div>
 | 
			
		||||
			</div>
 | 
			
		||||
			<div class="flex flex-col text-left flex-auto w-0">
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,13 +1,14 @@
 | 
			
		|||
<!-- Player.vue - 添加预加载功能 -->
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import { computed, nextTick, useTemplateRef, watch } from 'vue'
 | 
			
		||||
import { computed, nextTick, ref, useTemplateRef, watch } from 'vue'
 | 
			
		||||
import { useRoute } from 'vue-router'
 | 
			
		||||
import { useFavourites } from '../stores/useFavourites'
 | 
			
		||||
import { usePlayQueueStore } from '../stores/usePlayQueueStore'
 | 
			
		||||
 | 
			
		||||
import LoadingIndicator from '../assets/icons/loadingindicator.vue'
 | 
			
		||||
import PlayIcon from '../assets/icons/play.vue'
 | 
			
		||||
import { audioVisualizer, checkAndRefreshSongResource } from '../utils'
 | 
			
		||||
import PauseIcon from '../assets/icons/pause.vue'
 | 
			
		||||
import { audioVisualizer, checkAndRefreshSongResource, supportsWebAudioVisualization } from '../utils'
 | 
			
		||||
 | 
			
		||||
const playQueueStore = usePlayQueueStore()
 | 
			
		||||
const favourites = useFavourites()
 | 
			
		||||
| 
						 | 
				
			
			@ -263,21 +264,40 @@ function updateCurrentTime() {
 | 
			
		|||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
console.log('[Player] 初始化 audioVisualizer')
 | 
			
		||||
const { barHeights, connectAudio, isAnalyzing, error } = audioVisualizer({
 | 
			
		||||
	sensitivity: 1.5,
 | 
			
		||||
	barCount: 6,
 | 
			
		||||
	maxDecibels: -10,
 | 
			
		||||
	bassBoost: 0.8,
 | 
			
		||||
	midBoost: 1.2,
 | 
			
		||||
	trebleBoost: 1.4,
 | 
			
		||||
	threshold: 0,
 | 
			
		||||
})
 | 
			
		||||
// 检查浏览器是否支持音频可视化
 | 
			
		||||
const isAudioVisualizationSupported = supportsWebAudioVisualization()
 | 
			
		||||
console.log('[Player] 音频可视化支持状态:', isAudioVisualizationSupported)
 | 
			
		||||
 | 
			
		||||
console.log('[Player] audioVisualizer 返回值:', {
 | 
			
		||||
	barHeights: barHeights.value,
 | 
			
		||||
	isAnalyzing: isAnalyzing.value,
 | 
			
		||||
})
 | 
			
		||||
// 只在支持的浏览器上初始化音频可视化
 | 
			
		||||
let barHeights = ref<number[]>([0, 0, 0, 0, 0, 0])
 | 
			
		||||
let connectAudio = (_audio: HTMLAudioElement) => {}
 | 
			
		||||
let isAnalyzing = ref(false)
 | 
			
		||||
let error = ref<string | null>(null)
 | 
			
		||||
 | 
			
		||||
if (isAudioVisualizationSupported) {
 | 
			
		||||
	console.log('[Player] 初始化 audioVisualizer')
 | 
			
		||||
	const visualizer = audioVisualizer({
 | 
			
		||||
		sensitivity: 1.5,
 | 
			
		||||
		barCount: 6,
 | 
			
		||||
		maxDecibels: -10,
 | 
			
		||||
		bassBoost: 0.8,
 | 
			
		||||
		midBoost: 1.2,
 | 
			
		||||
		trebleBoost: 1.4,
 | 
			
		||||
		threshold: 0,
 | 
			
		||||
	})
 | 
			
		||||
	
 | 
			
		||||
	barHeights = visualizer.barHeights
 | 
			
		||||
	connectAudio = visualizer.connectAudio
 | 
			
		||||
	isAnalyzing = visualizer.isAnalyzing
 | 
			
		||||
	error = visualizer.error
 | 
			
		||||
	
 | 
			
		||||
	console.log('[Player] audioVisualizer 返回值:', {
 | 
			
		||||
		barHeights: barHeights.value,
 | 
			
		||||
		isAnalyzing: isAnalyzing.value,
 | 
			
		||||
	})
 | 
			
		||||
} else {
 | 
			
		||||
	console.log('[Player] 音频可视化被禁用(Safari 或不支持的浏览器)')
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// 监听播放列表变化
 | 
			
		||||
watch(
 | 
			
		||||
| 
						 | 
				
			
			@ -293,13 +313,17 @@ watch(
 | 
			
		|||
		await nextTick()
 | 
			
		||||
 | 
			
		||||
		if (player.value) {
 | 
			
		||||
			console.log('[Player] 连接音频元素到可视化器')
 | 
			
		||||
			console.log('[Player] 音频元素状态:', {
 | 
			
		||||
				src: player.value.src?.substring(0, 50) + '...',
 | 
			
		||||
				readyState: player.value.readyState,
 | 
			
		||||
				paused: player.value.paused,
 | 
			
		||||
			})
 | 
			
		||||
			connectAudio(player.value)
 | 
			
		||||
			if (isAudioVisualizationSupported) {
 | 
			
		||||
				console.log('[Player] 连接音频元素到可视化器')
 | 
			
		||||
				console.log('[Player] 音频元素状态:', {
 | 
			
		||||
					src: player.value.src?.substring(0, 50) + '...',
 | 
			
		||||
					readyState: player.value.readyState,
 | 
			
		||||
					paused: player.value.paused,
 | 
			
		||||
				})
 | 
			
		||||
				connectAudio(player.value)
 | 
			
		||||
			} else {
 | 
			
		||||
				console.log('[Player] 跳过音频可视化连接(不支持的浏览器)')
 | 
			
		||||
			}
 | 
			
		||||
		} else {
 | 
			
		||||
			console.log('[Player] ❌ 音频元素不存在')
 | 
			
		||||
		}
 | 
			
		||||
| 
						 | 
				
			
			@ -322,7 +346,7 @@ watch(
 | 
			
		|||
watch(
 | 
			
		||||
	() => player.value,
 | 
			
		||||
	(audioElement) => {
 | 
			
		||||
		if (audioElement && playQueueStore.list.length > 0) {
 | 
			
		||||
		if (audioElement && playQueueStore.list.length > 0 && isAudioVisualizationSupported) {
 | 
			
		||||
			connectAudio(audioElement)
 | 
			
		||||
		}
 | 
			
		||||
	},
 | 
			
		||||
| 
						 | 
				
			
			@ -495,12 +519,14 @@ setInterval(syncVolumeFromStorage, 100)
 | 
			
		|||
			}">
 | 
			
		||||
				<div v-if="playQueueStore.isPlaying">
 | 
			
		||||
					<LoadingIndicator v-if="playQueueStore.isBuffering === true" :size="4" />
 | 
			
		||||
					<div v-else class="h-4 flex justify-center items-center gap-[.125rem]">
 | 
			
		||||
					<!-- 在支持的浏览器上显示可视化,否则显示暂停图标 -->
 | 
			
		||||
					<div v-else-if="isAudioVisualizationSupported" class="h-4 flex justify-center items-center gap-[.125rem]">
 | 
			
		||||
						<div class="bg-white/75 w-[.125rem] rounded-full" v-for="(bar, index) in playQueueStore.visualizer"
 | 
			
		||||
							:key="index" :style="{
 | 
			
		||||
								height: `${Math.max(10, bar)}%`
 | 
			
		||||
							}" />
 | 
			
		||||
					</div>
 | 
			
		||||
					<PauseIcon v-else :size="4" />
 | 
			
		||||
				</div>
 | 
			
		||||
				<PlayIcon v-else :size="4" />
 | 
			
		||||
			</button>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										98
									
								
								src/utils/browserDetection.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								src/utils/browserDetection.ts
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,98 @@
 | 
			
		|||
/**
 | 
			
		||||
 * 浏览器检测工具
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 检测是否为 Safari 浏览器
 | 
			
		||||
 * @returns {boolean} 如果是 Safari 返回 true,否则返回 false
 | 
			
		||||
 */
 | 
			
		||||
export function isSafari(): boolean {
 | 
			
		||||
	const ua = navigator.userAgent.toLowerCase()
 | 
			
		||||
	
 | 
			
		||||
	// 检测 Safari 浏览器(包括 iOS 和 macOS)
 | 
			
		||||
	// Safari 的 User Agent 包含 'safari' 但不包含 'chrome' 或 'chromium'
 | 
			
		||||
	const isSafariBrowser = ua.includes('safari') && 
 | 
			
		||||
		!ua.includes('chrome') && 
 | 
			
		||||
		!ua.includes('chromium') &&
 | 
			
		||||
		!ua.includes('android')
 | 
			
		||||
	
 | 
			
		||||
	// 额外检查:使用 Safari 特有的 API
 | 
			
		||||
	const isSafariByFeature = 'safari' in window || 
 | 
			
		||||
		/^((?!chrome|android).)*safari/i.test(navigator.userAgent)
 | 
			
		||||
	
 | 
			
		||||
	return isSafariBrowser || isSafariByFeature
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 检测是否为移动版 Safari
 | 
			
		||||
 * @returns {boolean} 如果是移动版 Safari 返回 true,否则返回 false
 | 
			
		||||
 */
 | 
			
		||||
export function isMobileSafari(): boolean {
 | 
			
		||||
	return /iPad|iPhone|iPod/.test(navigator.userAgent) && !(window as any).MSStream
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 检测是否支持 Web Audio API 的完整功能
 | 
			
		||||
 * @returns {boolean} 如果支持返回 true,否则返回 false
 | 
			
		||||
 */
 | 
			
		||||
export function supportsWebAudioVisualization(): boolean {
 | 
			
		||||
	// Safari 在某些情况下对 AudioContext 的支持有限制
 | 
			
		||||
	// 特别是在处理跨域音频资源时
 | 
			
		||||
	if (isSafari()) {
 | 
			
		||||
		console.log('[BrowserDetection] Safari detected, audio visualization disabled')
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
	
 | 
			
		||||
	// 检查基本的 Web Audio API 支持
 | 
			
		||||
	const hasAudioContext = 'AudioContext' in window || 'webkitAudioContext' in window
 | 
			
		||||
	const hasAnalyserNode = hasAudioContext && (
 | 
			
		||||
		'AnalyserNode' in window || 
 | 
			
		||||
		((window as any).AudioContext && 'createAnalyser' in (window as any).AudioContext.prototype)
 | 
			
		||||
	)
 | 
			
		||||
	
 | 
			
		||||
	return hasAudioContext && hasAnalyserNode
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 获取浏览器信息
 | 
			
		||||
 * @returns {object} 包含浏览器类型和版本信息的对象
 | 
			
		||||
 */
 | 
			
		||||
export function getBrowserInfo() {
 | 
			
		||||
	const ua = navigator.userAgent
 | 
			
		||||
	let browserName = 'Unknown'
 | 
			
		||||
	let browserVersion = 'Unknown'
 | 
			
		||||
	
 | 
			
		||||
	if (isSafari()) {
 | 
			
		||||
		browserName = 'Safari'
 | 
			
		||||
		const versionMatch = ua.match(/Version\/(\d+\.\d+)/)
 | 
			
		||||
		if (versionMatch) {
 | 
			
		||||
			browserVersion = versionMatch[1]
 | 
			
		||||
		}
 | 
			
		||||
	} else if (ua.includes('Chrome')) {
 | 
			
		||||
		browserName = 'Chrome'
 | 
			
		||||
		const versionMatch = ua.match(/Chrome\/(\d+\.\d+)/)
 | 
			
		||||
		if (versionMatch) {
 | 
			
		||||
			browserVersion = versionMatch[1]
 | 
			
		||||
		}
 | 
			
		||||
	} else if (ua.includes('Firefox')) {
 | 
			
		||||
		browserName = 'Firefox'
 | 
			
		||||
		const versionMatch = ua.match(/Firefox\/(\d+\.\d+)/)
 | 
			
		||||
		if (versionMatch) {
 | 
			
		||||
			browserVersion = versionMatch[1]
 | 
			
		||||
		}
 | 
			
		||||
	} else if (ua.includes('Edge')) {
 | 
			
		||||
		browserName = 'Edge'
 | 
			
		||||
		const versionMatch = ua.match(/Edge\/(\d+\.\d+)/)
 | 
			
		||||
		if (versionMatch) {
 | 
			
		||||
			browserVersion = versionMatch[1]
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	
 | 
			
		||||
	return {
 | 
			
		||||
		name: browserName,
 | 
			
		||||
		version: browserVersion,
 | 
			
		||||
		isSafari: isSafari(),
 | 
			
		||||
		isMobileSafari: isMobileSafari(),
 | 
			
		||||
		supportsAudioVisualization: supportsWebAudioVisualization()
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -2,5 +2,16 @@ import artistsOrganize from "./artistsOrganize"
 | 
			
		|||
import { audioVisualizer } from "./audioVisualizer"
 | 
			
		||||
import cicdInfo from "./cicdInfo"
 | 
			
		||||
import { checkAndRefreshSongResource, checkAndRefreshMultipleSongs } from "./songResourceChecker"
 | 
			
		||||
import { isSafari, isMobileSafari, supportsWebAudioVisualization, getBrowserInfo } from "./browserDetection"
 | 
			
		||||
 | 
			
		||||
export { artistsOrganize, audioVisualizer, cicdInfo, checkAndRefreshSongResource, checkAndRefreshMultipleSongs }
 | 
			
		||||
export { 
 | 
			
		||||
	artistsOrganize, 
 | 
			
		||||
	audioVisualizer, 
 | 
			
		||||
	cicdInfo, 
 | 
			
		||||
	checkAndRefreshSongResource, 
 | 
			
		||||
	checkAndRefreshMultipleSongs,
 | 
			
		||||
	isSafari,
 | 
			
		||||
	isMobileSafari,
 | 
			
		||||
	supportsWebAudioVisualization,
 | 
			
		||||
	getBrowserInfo
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in New Issue
	
	Block a user