- // ==UserScript==
- // @name Bionic Reading Neo
- // @namespace http://tampermonkey.net/
- // @version 0.1.3
- // @description Bionic Reading User Script ⌘ + B
- // @author RoCry
- // @match *://*/*
- // @icon 
- // @exclude /\.(js|java|c|cpp|h|py|css|less|scss|json|yaml|yml|xml)(?:\?.+)$/
- // @license MIT
- // @run-at document-end
- // @grant GM_getValue
- // @grant GM_setValue
- // @grant GM_registerMenuCommand
- // @grant GM_addStyle
- // ==/UserScript==
-
- const defaultConfig = {
- // Whether to automatically apply bionic reading when page loads
- autoBionic: false,
-
- // Whether to skip processing text inside <a> tags that contain URLs
- skipLinks: true,
-
- // The ratio of characters to bold at the start of each word (0.0 to 1.0)
- scale: 0.5,
-
- // Maximum number of characters to bold in any word (null for no limit)
- maxBionicLength: null,
-
- // Opacity of the bold text (0.0 to 1.0)
- opacity: 1,
-
- // Number of words to skip between processed words (0 for no skipping)
- // Higher values create a "saccade" effect, mimicking natural eye movement
- saccade: 0,
-
- // Whether to use special Unicode characters instead of bold text
- symbolMode: false,
-
- // Whether to skip processing words in the excludeWords list
- skipWords: true,
- // List of common words to skip when skipWords is true
- excludeWords: ['is','and','as','if','the','of','to','be','for','this'],
-
- // Minimum word length to apply bionic reading (shorter words are skipped)
- minWordLength: 3,
-
- // Minimum ratio of ASCII characters required to consider content as English
- // (0.0 to 1.0) - Higher values mean stricter English detection
- minAsciiRatio: 0.9,
- // Number of characters to analyze for language detection
- charsToCheck: 300,
-
- // 400: Normal
- // 500: Medium
- // 600: Semi-bold
- // 700: Bold
- // 800: Extra bold
- // 900: Black (maximum boldness)
- fontWeight: 700,
- };
-
- let config = defaultConfig;
- try {
- config = (_=>{
- const _config = GM_getValue('config');
- if(!_config) return defaultConfig;
-
- for(let key in defaultConfig){
- if(_config[key] === undefined) _config[key] = defaultConfig[key];
- }
- return _config;
- })();
-
- GM_setValue('config',config);
- } catch(e) {
- console.log('Failed to read default config')
- }
-
- let isBionic = false;
- let body = document.body;
-
- const styleEl = document.createElement('style');
- styleEl.textContent = `bbb{font-weight:${config.fontWeight};opacity:${config.opacity}}html[data-site="greasyfork"] a bionic{pointer-events:none}`;
-
- document.documentElement.setAttribute('data-site',location.hostname.replace(/\.\w+$|www\./ig,''))
-
- const excludeNodeNames = [
- 'script','style','xmp',
- 'input','textarea','select',
- 'pre','code',
- 'h1','h2',
- 'b','strong',
- 'svg','embed',
- 'img','audio','video',
- 'canvas',
- ];
-
- const excludeClasses = [
- 'highlight',
- 'katex',
- 'editor',
- ]
-
- const excludeClassesRegexi = new RegExp(excludeClasses.join('|'),'i');
- const linkRegex = /^https?:\/\//;
-
- function isEnglishContent() {
- try {
- const title = document.title || '';
- const firstParagraphs = Array.from(document.getElementsByTagName('p'))
- .slice(0, 3)
- .map(p => p.textContent)
- .join(' ');
-
- const textToAnalyze = (title + ' ' + firstParagraphs)
- .slice(0, config.charsToCheck)
- .replace(/\s+/g, ' ')
- .trim();
-
- if (!textToAnalyze) return true;
-
- const asciiChars = textToAnalyze.replace(/[\s\.,\-_'"!?()]/g, '')
- .split('')
- .filter(char => char.charCodeAt(0) <= 127).length;
-
- const totalChars = textToAnalyze.replace(/[\s\.,\-_'"!?()]/g, '').length;
-
- if (totalChars === 0) return true;
-
- const asciiRatio = asciiChars / totalChars;
- console.log('🈂️ ASCII Ratio:', asciiRatio.toFixed(2));
-
- return asciiRatio >= config.minAsciiRatio;
- } catch (e) {
- console.error('Error checking content language:', e);
- return true;
- }
- }
-
- const gather = el=>{
- let textEls = [];
- el.childNodes.forEach(el=>{
- if(el.isEnB) return;
- if(el.originEl) return;
-
- if(el.nodeType === 3){
- textEls.push(el);
- }else if(el.childNodes){
- const nodeName = el.nodeName.toLowerCase();
- if(excludeNodeNames.includes(nodeName)) return;
- if(config.skipLinks){
- if(nodeName === 'a'){
- if(linkRegex.test(el.textContent)) return;
- }
- }
- if(el.getAttribute){
- if(el.getAttribute('class') && excludeClassesRegexi.test(el.getAttribute('class'))) return;
- if(el.getAttribute('contentEditable') === 'true') return;
- }
-
- textEls = textEls.concat(gather(el))
- }
- })
- return textEls;
- };
-
- const engRegex = /[a-zA-Z][a-z]+/;
- const engRegexg = new RegExp(engRegex,'g');
-
- const getHalfLength = word=>{
- if (word.length < config.minWordLength) return 0;
-
- let halfLength;
- if(/ing$/.test(word)){
- halfLength = word.length - 3;
- }else if(word.length<5){
- halfLength = Math.floor(word.length * config.scale);
- }else{
- halfLength = Math.ceil(word.length * config.scale);
- }
-
- if(config.maxBionicLength){
- halfLength = Math.min(halfLength, config.maxBionicLength)
- }
- return halfLength;
- }
-
- let count = 0;
- const saccadeRound = config.saccade + 1;
- const saccadeCounter = _=>{
- return ++count % saccadeRound === 0;
- };
-
- const replaceTextByEl = el=>{
- const text = el.data;
- if(!engRegex.test(text))return;
-
- if(!el.replaceEl){
- const spanEl = document.createElement('bionic');
- spanEl.isEnB = true;
- spanEl.innerHTML = text.replace(/[\u00A0-\u9999<>\&]/g, w=>'&#'+w.charCodeAt(0)+';')
- .replace(engRegexg,word=>{
- if(word.length < config.minWordLength) return word;
- if(config.skipWords && config.excludeWords.includes(word)) return word;
- if(config.saccade && !saccadeCounter()) return word;
-
- const halfLength = getHalfLength(word);
- if (halfLength === 0) return word;
- return '<bbb>'+word.substr(0,halfLength)+'</bbb>'+word.substr(halfLength);
- });
- spanEl.originEl = el;
- el.replaceEl = spanEl;
- }
-
- el.after(el.replaceEl);
- el.remove();
- };
-
- const replaceTextSymbolModeByEl = el=>{
- const text = el.data;
- if(!engRegex.test(text))return;
-
- // For symbol mode, we can still use textContent since we're not creating HTML
- el.data = text.replace(engRegexg,word=>{
- if(word.length < config.minWordLength) return word;
- if(config.skipWords && config.excludeWords.includes(word)) return word;
- if(config.saccade && !saccadeCounter()) return word;
-
- const halfLength = getHalfLength(word);
- if (halfLength === 0) return word;
- const a = word.substr(0,halfLength).
- replace(/[a-z]/g,w=>String.fromCharCode(55349,w.charCodeAt(0)+56717)).
- replace(/[A-Z]/g,w=>String.fromCharCode(55349,w.charCodeAt(0)+56723));
- const b = word.substr(halfLength).
- replace(/[a-z]/g,w=> String.fromCharCode(55349,w.charCodeAt(0)+56665)).
- replace(/[A-Z]/g,w=> String.fromCharCode(55349,w.charCodeAt(0)+56671));
- return a + b;
- })
- }
-
- const bionic = (checkIsEnglish = true) => {
- if (checkIsEnglish && !isEnglishContent()) {
- console.log('🈂️ Non-English content detected, skipping bionic reading');
- return;
- }
-
- const textEls = gather(body);
- isBionic = true;
- count = 0;
-
- let replaceFunc = config.symbolMode ? replaceTextSymbolModeByEl : replaceTextByEl;
- textEls.forEach(replaceFunc);
- document.head.appendChild(styleEl);
- }
-
- const lazy = (func,ms = 15)=> {
- return _=>{
- clearTimeout(func.T)
- func.T = setTimeout(func,ms)
- }
- };
-
- const listenerFunc = lazy(_=>{
- if(!isBionic) return;
- // Don't recheck language for updates when bionic is already enabled
- bionic(false);
- });
-
- if(window.MutationObserver){
- (new MutationObserver(listenerFunc)).observe(body,{
- childList: true,
- subtree: true,
- attributes: true,
- });
- }else{
- const {open,send} = XMLHttpRequest.prototype;
- XMLHttpRequest.prototype.open = function(){
- this.addEventListener('load',listenerFunc);
- return open.apply(this,arguments);
- };
- document.addEventListener('DOMContentLoaded',listenerFunc);
- document.addEventListener('DOMNodeInserted',listenerFunc);
- }
-
- if(config.autoBionic){
- window.addEventListener('load', () => bionic(true));
- }
-
- const revoke = _=>{
- const els = [...document.querySelectorAll('bionic')];
- els.forEach(el=>{
- const {originEl} = el;
- if(!originEl) return;
- el.after(originEl);
- el.remove();
- })
- isBionic = false;
- };
-
- document.addEventListener('keydown',e=>{
- const { ctrlKey , metaKey, key } = e;
- if( ctrlKey || metaKey ){
- if(key === 'b'){
- if(isBionic){
- revoke();
- // Save autoBionic as false when manually disabled
- config.autoBionic = false;
- GM_setValue('config', config);
- }else{
- bionic(false);
- // Save autoBionic as true when manually enabled
- config.autoBionic = true;
- GM_setValue('config', config);
- }
- }
- }
- })
-
- GM_addStyle(`
- .bionic-config-dialog {
- position: fixed;
- top: 50%;
- left: 50%;
- transform: translate(-50%, -50%);
- background: white;
- padding: 20px;
- border-radius: 8px;
- box-shadow: 0 0 10px rgba(0,0,0,0.3);
- z-index: 999999;
- max-width: 500px;
- width: 90%;
- }
- .bionic-config-dialog label {
- display: block;
- margin: 10px 0;
- }
- .bionic-config-dialog input[type="number"] {
- width: 60px;
- }
- `);
-
- function showConfigDialog() {
- const dialog = document.createElement('div');
- dialog.className = 'bionic-config-dialog';
- dialog.innerHTML = `
- <h3>Bionic Reading Settings</h3>
- <label>
- <input type="checkbox" id="autoBionic" ${config.autoBionic ? 'checked' : ''}>
- Auto-enable on page load
- </label>
- <label>
- <input type="checkbox" id="skipLinks" ${config.skipLinks ? 'checked' : ''}>
- Skip URL links
- </label>
- <label>
- Bold ratio (0.0-1.0):
- <input type="number" id="scale" min="0" max="1" step="0.1" value="${config.scale}">
- </label>
- <label>
- Bold opacity (0.0-1.0):
- <input type="number" id="opacity" min="0" max="1" step="0.1" value="${config.opacity}">
- </label>
- <label>
- Minimum word length:
- <input type="number" id="minWordLength" min="1" max="10" value="${config.minWordLength}">
- </label>
- <label>
- <input type="checkbox" id="symbolMode" ${config.symbolMode ? 'checked' : ''}>
- Use symbols instead of bold
- </label>
- <label>
- Font Weight (400-900):
- <input type="number" id="fontWeight" min="400" max="900" step="100" value="${config.fontWeight}">
- </label>
- <div>
- <button id="closeBtn">Close</button>
- <button id="saveBtn">Save</button>
- </div>
- `;
-
- // Add event listeners after creating the dialog
- dialog.querySelector('#closeBtn').addEventListener('click', () => dialog.remove());
- dialog.querySelector('#saveBtn').addEventListener('click', () => {
- const newConfig = {
- ...config,
- autoBionic: dialog.querySelector('#autoBionic').checked,
- skipLinks: dialog.querySelector('#skipLinks').checked,
- scale: parseFloat(dialog.querySelector('#scale').value),
- opacity: parseFloat(dialog.querySelector('#opacity').value),
- minWordLength: parseInt(dialog.querySelector('#minWordLength').value),
- symbolMode: dialog.querySelector('#symbolMode').checked,
- fontWeight: parseInt(dialog.querySelector('#fontWeight').value),
- };
-
- config = newConfig;
- GM_setValue('config', config);
-
- // Update style with new font weight
- styleEl.textContent = `bbb{font-weight:${config.fontWeight};opacity:${config.opacity}}html[data-site="greasyfork"] a bionic{pointer-events:none}`;
-
- if (isBionic) {
- revoke();
- bionic(false);
- }
-
- dialog.remove();
- });
-
- document.body.appendChild(dialog);
- }
-
- // Register the menu command
- GM_registerMenuCommand('Bionic Reading Settings', showConfigDialog);