Jump to content

MediaWiki:Common.js: Difference between revisions

From Grantha
No edit summary
No edit summary
Line 1: Line 1:
(function () {
/* MediaWiki:Common.js — grantha.io
*
* Responsibilities:
*  1. Transliteration engine (Devanāgarī → IAST / Kannada / Tamil).
*  2. Tag all transliteratable text nodes once per page load.
*  3. Apply the script stored in localStorage('grantha_reader_script')
*    so every page opens in the user's preferred script automatically.
*  4. Listen for the 'gr-script-change' CustomEvent dispatched by the
*    ReaderToolbar dropdown and re-apply the new script immediately.
*  5. TOC active-item highlight via MutationObserver (unchanged).
*
* The script-switcher bar/buttons previously built here are removed —
* the ReaderToolbar extension now owns the dropdown UI.
*
* localStorage key shared with ReaderToolbar: 'grantha_reader_script'
*/


( function () {
  var LS_SCRIPT_KEY = 'grantha_reader_script';
   var currentScript = 'deva';
   var currentScript = 'deva';


   // ── IAST transliteration (inline, no external library) ─────────
   // ── IAST transliteration ────────────────────────────────────────
   function devanagariToIAST(text) {
   function devanagariToIAST( text ) {
     var CONSONANTS = {
     var CONSONANTS = {
       'क':'k','ख':'kh','ग':'g','घ':'gh','ङ':'ṅ',
       'क':'k','ख':'kh','ग':'g','घ':'gh','ङ':'ṅ',
Line 28: Line 46:
     };
     };
     var HALANTA = '्';
     var HALANTA = '्';
     var chars = Array.from(text);
     var chars = Array.from( text );
     var result = '';
     var result = '';
     var i = 0;
     var i = 0;
     while (i < chars.length) {
     while ( i < chars.length ) {
       var ch = chars[i];
       var ch   = chars[ i ];
       if (CONSONANTS[ch]) {
      var next = chars[ i + 1 ];
         var base = CONSONANTS[ch];
       if ( CONSONANTS[ ch ] ) {
        var next = chars[i + 1];
         var base = CONSONANTS[ ch ];
         if (next === HALANTA) {
         if ( next === HALANTA )         { result += base;               i += 2; }
          result += base; i += 2;
         else if ( DIACRITICS[ next ] )   { result += base + DIACRITICS[ next ]; i += 2; }
         } else if (DIACRITICS[next]) {
         else if ( next === 'ं' || next === 'ः' ) { result += base + 'a' + MISC[ next ]; i += 2; }
          result += base + DIACRITICS[next]; i += 2;
         else                             { result += base + 'a';         i++;   }
         } else if (next === 'ं' || next === 'ः') {
       } else if ( VOWELS[ ch ] )         { result += VOWELS[ ch ];       i++; }
          result += base + 'a' + MISC[next]; i += 2;
       else if ( DIACRITICS[ ch ] )       { result += DIACRITICS[ ch ];   i++; }
         } else {
       else if ( MISC[ ch ] )             { result += MISC[ ch ];         i++; }
          result += base + 'a'; i++;
       else                               { result += ch;                 i++; }
        }
       } else if (VOWELS[ch])     { result += VOWELS[ch];     i++; }
       else if (DIACRITICS[ch])   { result += DIACRITICS[ch]; i++; }
       else if (MISC[ch])         { result += MISC[ch];       i++; }
       else                       { result += ch;             i++; }
     }
     }
     return result;
     return result;
Line 90: Line 103:


   var PRE = [
   var PRE = [
     [/ङ्क/g,'ंक'],[/ङ्ख/g,'ंख'],[/ङ्ग/g,'ंग'],[/ङ्घ/g,'ंघ'],
     [ /ङ्क/g, 'ंक' ], [ /ङ्ख/g, 'ंख' ], [ /ङ्ग/g, 'ंग' ], [ /ङ्घ/g, 'ंघ' ],
     [/ञ्च/g,'ंच'],[/ञ्ज/g,'ंज'],[/ण्ट/g,'ंट'],[/ण्ड/g,'ंड'],
     [ /ञ्च/g, 'ंच' ], [ /ञ्ज/g, 'ंज' ], [ /ण्ट/g, 'ंट' ], [ /ण्ड/g, 'ंड' ],
     [/न्त/g,'ंत'],[/न्द/g,'ंद'],[/म्ब/g,'ंब'],[/म्भ/g,'ंभ']
     [ /न्त/g, 'ंत' ], [ /न्द/g, 'ंद' ], [ /म्ब/g, 'ंब' ], [ /म्भ/g, 'ंभ' ]
   ];
   ];


   function transliterateText(text, script) {
   function transliterateText( text, script ) {
     if (script === 'en') return devanagariToIAST(text);
     if ( script === 'en' ) return devanagariToIAST( text );
     var map = SCRIPT_MAP[script];
     var map = SCRIPT_MAP[ script ];
     if (!map) return text;
     if ( !map ) return text;
     var t = text;
     var t = text;
     PRE.forEach(function (p) { t = t.replace(p[0], p[1]); });
     PRE.forEach( function ( p ) { t = t.replace( p[ 0 ], p[ 1 ] ); } );
     return Array.from(t).map(function (ch) {
     return Array.from( t ).map( function ( ch ) {
       return map[ch] !== undefined ? map[ch] : ch;
       return map[ ch ] !== undefined ? map[ ch ] : ch;
     }).join('');
     } ).join( '' );
   }
   }


   // ── Collect all transliteratable text nodes once ────────────────
   // ── Tag all transliteratable text nodes once per page ───────────
   // Store originals as data-deva on a wrapper span.
   // Wraps each text node in a <span data-deva="original"> so that
   // On script switch, update textContent of those spans — never
   // applyScript() can flip the content without touching innerHTML.
  // touch innerHTML of the whole content block.
   var translatableSpans = [];
   var translatableSpans = [];


   function tagTextNodes() {
   function tagTextNodes() {
     var content = document.querySelector('.mw-parser-output');
     var content = document.querySelector( '.mw-parser-output' );
     if (!content) return;
     if ( !content ) return;


     var walker = document.createTreeWalker(content, NodeFilter.SHOW_TEXT);
     var walker = document.createTreeWalker( content, NodeFilter.SHOW_TEXT );
     var nodes = [];
     var nodes = [];
     while (walker.nextNode()) nodes.push(walker.currentNode);
     while ( walker.nextNode() ) nodes.push( walker.currentNode );


     nodes.forEach(function (node) {
     nodes.forEach( function ( node ) {
       var p = node.parentNode;
       var p = node.parentNode;
       if (!p) return;
       if ( !p ) return;
      // Skip already tagged
       if ( p.hasAttribute && p.hasAttribute( 'data-deva' ) ) return;
       if (p.hasAttribute && p.hasAttribute('data-deva')) return;
       if ( p.closest ) {
      // Skip UI elements
         if ( p.closest( '.gr-script-bar' ) || // legacy bar (harmless guard)
       if (p.closest) {
            p.closest( '.gr-controls' )  ||  // toolbar controls
         if (p.closest('.gr-script-bar') ||
            p.closest( '.vector-toc' )   ||
            p.closest('.vector-toc') ||
            p.closest( '#toc' )           ||
            p.closest('#toc') ||
            p.closest( '.mw-editsection' ) ) return;
            p.closest('.mw-editsection')) return;
       }
       }
       var orig = node.textContent;
       var orig = node.textContent;
       if (!orig.trim()) return;
       if ( !orig.trim() ) return;
      // Wrap in span with original stored
       var span = document.createElement( 'span' );
       var span = document.createElement('span');
       span.setAttribute( 'data-deva', orig );
       span.setAttribute('data-deva', orig);
       span.textContent = orig;
       span.textContent = orig;
       p.replaceChild(span, node);
       p.replaceChild( span, node );
       translatableSpans.push(span);
       translatableSpans.push( span );
     });
     } );
   }
   }


   function applyScript(script) {
  // ── Apply a script to all tagged spans ─────────────────────────
   function applyScript( script ) {
     currentScript = script;
     currentScript = script;
     translatableSpans.forEach(function (span) {
     translatableSpans.forEach( function ( span ) {
       if (!span.parentNode) return; // detached
       if ( !span.parentNode ) return; // detached node
       var orig = span.getAttribute('data-deva');
       var orig = span.getAttribute( 'data-deva' );
       if (script === 'deva') {
       span.textContent = ( script === 'deva' )
         span.textContent = orig;
         ? orig
      } else {
         : transliterateText( orig, script );
         span.textContent = transliterateText(orig, script);
     } );
      }
    });
    // Update active button
    var bar = document.querySelector('.gr-script-bar');
    if (bar) {
      bar.querySelectorAll('.gr-script-btn').forEach(function (b) {
        b.classList.toggle('active', b.getAttribute('data-script') === script);
      });
    }
  }
 
  // ── Build switcher bar ─────────────────────────────────────────
  function buildBar() {
    var title = document.querySelector('.gr-doc-title');
    if (!title) return;
    if (document.querySelector('.gr-script-bar')) return;
 
    var bar = document.createElement('div');
    bar.className = 'gr-script-bar';
    bar.innerHTML =
      '<span class="gr-script-label">change script to</span>' +
      '<a class="gr-script-btn active" data-script="deva">देवनागरी</a>' +
      '<a class="gr-script-btn" data-script="kn">ಕನ್ನಡ</a>' +
      '<a class="gr-script-btn" data-script="ta">தமிழ்</a>' +
      '<a class="gr-script-btn" data-script="en">English</a>';
 
    title.after(bar);
 
    bar.querySelectorAll('.gr-script-btn').forEach(function (btn) {
      btn.addEventListener('click', function (e) {
        e.preventDefault();
        applyScript(btn.getAttribute('data-script'));
      });
     });
   }
   }


   // ── TOC active highlight (CSS-driven, MutationObserver backup) ──
   // ── TOC active-item highlight ───────────────────────────────────
  // MutationObserver watches Vector 2022 TOC class changes and colours
  // the active link orange.  This block is not modified.
   function watchTocActive() {
   function watchTocActive() {
     var toc = document.querySelector('.vector-toc');
     var toc = document.querySelector( '.vector-toc' );
     if (!toc || toc._grObserved) return;
     if ( !toc || toc._grObserved ) return;
     toc._grObserved = true;
     toc._grObserved = true;


     var observer = new MutationObserver(function (mutations) {
     var observer = new MutationObserver( function ( mutations ) {
       mutations.forEach(function (m) {
       mutations.forEach( function ( m ) {
         if (m.attributeName !== 'class') return;
         if ( m.attributeName !== 'class' ) return;
         var li = m.target;
         var li   = m.target;
         var link = li.querySelector(':scope > .vector-toc-link');
         var link = li.querySelector( ':scope > .vector-toc-link' );
         if (!link) return;
         if ( !link ) return;
         if (li.classList.contains('vector-toc-list-item-active')) {
         if ( li.classList.contains( 'vector-toc-list-item-active' ) ) {
           link.style.color = '#f57c00';
           link.style.color     = '#f57c00';
           link.style.fontWeight = '700';
           link.style.fontWeight = '700';
         } else {
         } else {
           link.style.color = '';
           link.style.color     = '';
           link.style.fontWeight = '';
           link.style.fontWeight = '';
         }
         }
       });
       } );
     });
     } );


     toc.querySelectorAll('.vector-toc-list-item').forEach(function (li) {
     toc.querySelectorAll( '.vector-toc-list-item' ).forEach( function ( li ) {
       observer.observe(li, { attributes: true, attributeFilter: ['class'] });
       observer.observe( li, { attributes: true, attributeFilter: [ 'class' ] } );
     });
     } );
   }
   }


   // ── Init ───────────────────────────────────────────────────────
   // ── Init ────────────────────────────────────────────────────────
   function init() {
   function init() {
     var content = document.querySelector('.mw-parser-output');
     var content     = document.querySelector( '.mw-parser-output' );
     var alreadyTagged = content && content.querySelector('[data-deva]');
     var alreadyTagged = content && content.querySelector( '[data-deva]' );
     if (!alreadyTagged) {
     if ( !alreadyTagged ) {
       translatableSpans = [];
       translatableSpans = [];
       tagTextNodes();
       tagTextNodes();
     }
     }
     buildBar();
 
     // Apply the globally persisted script preference immediately
    var saved = ( function () {
      try { return localStorage.getItem( LS_SCRIPT_KEY ); } catch ( e ) { return null; }
    }() );
    if ( saved && saved !== 'deva' ) {
      applyScript( saved );
    } else {
      currentScript = 'deva';
    }
 
     watchTocActive();
     watchTocActive();
   }
   }


   if (window.mw) {
   // ── React to toolbar dropdown changes on the same page ─────────
    mw.hook('wikipage.content').add(function () {
  // The ReaderToolbar dispatches 'gr-script-change' when the user
      setTimeout(function () {
  // picks a new script. We just call applyScript() — no need to
        var oldBar = document.querySelector('.gr-script-bar');
  // touch localStorage again (the toolbar already wrote it).
        if (oldBar) oldBar.remove();
  window.addEventListener( 'gr-script-change', function ( e ) {
 
    var script = e && e.detail && e.detail.script;
        // Check if content is already tagged from a previous hook fire
    if ( script ) applyScript( script );
        var content = document.querySelector('.mw-parser-output');
  } );
        var alreadyTagged = content && content.querySelector('[data-deva]');


         if (!alreadyTagged) {
  // ── MediaWiki hook (SPA-style navigation support) ───────────────
  if ( window.mw ) {
    mw.hook( 'wikipage.content' ).add( function () {
      setTimeout( function () {
        var content      = document.querySelector( '.mw-parser-output' );
        var alreadyTagged = content && content.querySelector( '[data-deva]' );
         if ( !alreadyTagged ) {
           translatableSpans = [];
           translatableSpans = [];
           tagTextNodes();
           tagTextNodes();
         }
         }
 
        // Re-apply current script to newly loaded content
         buildBar();
         if ( currentScript !== 'deva' ) {
          applyScript( currentScript );
        }
         watchTocActive();
         watchTocActive();
       }, 150);
       }, 150 );
     });
     } );
   }
   }


   // Fallback for non-MW environments
   // ── Fallback for non-MW environments ───────────────────────────
   if (document.readyState === 'loading') {
   if ( document.readyState === 'loading' ) {
     document.addEventListener('DOMContentLoaded', init);
     document.addEventListener( 'DOMContentLoaded', init );
   } else {
   } else {
     init();
     init();
   }
   }


})();
}() );

Revision as of 09:58, 15 April 2026

/* MediaWiki:Common.js — grantha.io
 *
 * Responsibilities:
 *  1. Transliteration engine (Devanāgarī → IAST / Kannada / Tamil).
 *  2. Tag all transliteratable text nodes once per page load.
 *  3. Apply the script stored in localStorage('grantha_reader_script')
 *     so every page opens in the user's preferred script automatically.
 *  4. Listen for the 'gr-script-change' CustomEvent dispatched by the
 *     ReaderToolbar dropdown and re-apply the new script immediately.
 *  5. TOC active-item highlight via MutationObserver (unchanged).
 *
 * The script-switcher bar/buttons previously built here are removed —
 * the ReaderToolbar extension now owns the dropdown UI.
 *
 * localStorage key shared with ReaderToolbar: 'grantha_reader_script'
 */

( function () {

  var LS_SCRIPT_KEY = 'grantha_reader_script';
  var currentScript = 'deva';

  // ── IAST transliteration ────────────────────────────────────────
  function devanagariToIAST( text ) {
    var CONSONANTS = {
      'क':'k','ख':'kh','ग':'g','घ':'gh','ङ':'ṅ',
      'च':'c','छ':'ch','ज':'j','झ':'jh','ञ':'ñ',
      'ट':'ṭ','ठ':'ṭh','ड':'ḍ','ढ':'ḍh','ण':'ṇ',
      'त':'t','थ':'th','द':'d','ध':'dh','न':'n',
      'प':'p','फ':'ph','ब':'b','भ':'bh','म':'m',
      'य':'y','र':'r','ल':'l','ळ':'ḷ','व':'v',
      'श':'ś','ष':'ṣ','स':'s','ह':'h'
    };
    var DIACRITICS = {
      'ा':'ā','ि':'i','ी':'ī','ु':'u','ू':'ū',
      'ृ':'ṛ','ॄ':'ṝ','े':'e','ै':'ai','ो':'o','ौ':'au'
    };
    var VOWELS = {
      'अ':'a','आ':'ā','इ':'i','ई':'ī','उ':'u','ऊ':'ū',
      'ऋ':'ṛ','ॠ':'ṝ','ए':'e','ऐ':'ai','ओ':'o','औ':'au','ऽ':"'"
    };
    var MISC = {
      'ं':'ṃ','ः':'ḥ','ँ':'m̐','ॐ':'oṃ',
      '०':'0','१':'1','२':'2','३':'3','४':'4',
      '५':'5','६':'6','७':'7','८':'8','९':'9'
    };
    var HALANTA = '्';
    var chars = Array.from( text );
    var result = '';
    var i = 0;
    while ( i < chars.length ) {
      var ch   = chars[ i ];
      var next = chars[ i + 1 ];
      if ( CONSONANTS[ ch ] ) {
        var base = CONSONANTS[ ch ];
        if ( next === HALANTA )          { result += base;               i += 2; }
        else if ( DIACRITICS[ next ] )   { result += base + DIACRITICS[ next ]; i += 2; }
        else if ( next === 'ं' || next === 'ः' ) { result += base + 'a' + MISC[ next ]; i += 2; }
        else                             { result += base + 'a';         i++;    }
      } else if ( VOWELS[ ch ] )         { result += VOWELS[ ch ];       i++; }
      else if ( DIACRITICS[ ch ] )       { result += DIACRITICS[ ch ];   i++; }
      else if ( MISC[ ch ] )             { result += MISC[ ch ];         i++; }
      else                               { result += ch;                 i++; }
    }
    return result;
  }

  // ── Character maps for Kannada / Tamil ─────────────────────────
  var SCRIPT_MAP = {
    kn: {
      'अ':'ಅ','आ':'ಆ','इ':'ಇ','ई':'ಈ','उ':'ಉ','ऊ':'ಊ','ऋ':'ಋ',
      'ए':'ಏ','ऐ':'ಐ','ओ':'ಓ','औ':'ಔ','ऽ':'ಽ',
      'क':'ಕ','ख':'ಖ','ग':'ಗ','घ':'ಘ','ङ':'ಙ',
      'च':'ಚ','छ':'ಛ','ज':'ಜ','झ':'ಝ','ञ':'ಞ',
      'ट':'ಟ','ठ':'ಠ','ड':'ಡ','ढ':'ಢ','ण':'ಣ',
      'त':'ತ','थ':'ಥ','द':'ದ','ध':'ಧ','न':'ನ',
      'प':'ಪ','फ':'ಫ','ब':'ಬ','भ':'ಭ','म':'ಮ',
      'य':'ಯ','र':'ರ','ल':'ಲ','व':'ವ',
      'श':'ಶ','ष':'ಷ','स':'ಸ','ह':'ಹ',
      'ा':'ಾ','ि':'ಿ','ी':'ೀ','ु':'ು','ू':'ೂ',
      'ृ':'ೃ','े':'ೇ','ै':'ೈ','ो':'ೋ','ौ':'ೌ',
      'ं':'ಂ','ः':'ಃ','्':'್',
      '०':'೦','१':'೧','२':'೨','३':'೩','४':'೪',
      '५':'೫','६':'೬','७':'೭','८':'೮','९':'೯'
    },
    ta: {
      'अ':'அ','आ':'ஆ','इ':'இ','ई':'ஈ','उ':'உ','ऊ':'ஊ',
      'ए':'ஏ','ऐ':'ஐ','ओ':'ஓ','औ':'ஔ',
      'क':'க','ख':'க','ग':'க','घ':'க','ङ':'ங',
      'च':'ச','छ':'ச','ज':'ஜ','झ':'ஜ','ञ':'ஞ',
      'ट':'ட','ठ':'ட','ड':'ட','ढ':'ட','ण':'ண',
      'त':'த','थ':'த','द':'த','ध':'த','न':'ந',
      'प':'ப','फ':'ப','ब':'ப','भ':'ப','म':'ம',
      'य':'ய','र':'ர','ல':'ல','व':'வ',
      'श':'ஶ','ष':'ஷ','स':'ஸ','ह':'ஹ',
      'ा':'ா','ि':'ி','ी':'ீ','ु':'ு','ू':'ூ',
      'े':'ே','ै':'ை','ो':'ோ','ौ':'ௌ',
      'ं':'ம்','ः':':','्':'்','ॐ':'ௐ',
      '०':'0','१':'1','२':'2','३':'3','४':'4',
      '५':'5','६':'6','७':'7','८':'8','९':'9'
    }
  };

  var PRE = [
    [ /ङ्क/g, 'ंक' ], [ /ङ्ख/g, 'ंख' ], [ /ङ्ग/g, 'ंग' ], [ /ङ्घ/g, 'ंघ' ],
    [ /ञ्च/g, 'ंच' ], [ /ञ्ज/g, 'ंज' ], [ /ण्ट/g, 'ंट' ], [ /ण्ड/g, 'ंड' ],
    [ /न्त/g, 'ंत' ], [ /न्द/g, 'ंद' ], [ /म्ब/g, 'ंब' ], [ /म्भ/g, 'ंभ' ]
  ];

  function transliterateText( text, script ) {
    if ( script === 'en' ) return devanagariToIAST( text );
    var map = SCRIPT_MAP[ script ];
    if ( !map ) return text;
    var t = text;
    PRE.forEach( function ( p ) { t = t.replace( p[ 0 ], p[ 1 ] ); } );
    return Array.from( t ).map( function ( ch ) {
      return map[ ch ] !== undefined ? map[ ch ] : ch;
    } ).join( '' );
  }

  // ── Tag all transliteratable text nodes once per page ───────────
  // Wraps each text node in a <span data-deva="original"> so that
  // applyScript() can flip the content without touching innerHTML.
  var translatableSpans = [];

  function tagTextNodes() {
    var content = document.querySelector( '.mw-parser-output' );
    if ( !content ) return;

    var walker = document.createTreeWalker( content, NodeFilter.SHOW_TEXT );
    var nodes  = [];
    while ( walker.nextNode() ) nodes.push( walker.currentNode );

    nodes.forEach( function ( node ) {
      var p = node.parentNode;
      if ( !p ) return;
      if ( p.hasAttribute && p.hasAttribute( 'data-deva' ) ) return;
      if ( p.closest ) {
        if ( p.closest( '.gr-script-bar' ) ||  // legacy bar (harmless guard)
             p.closest( '.gr-controls' )   ||  // toolbar controls
             p.closest( '.vector-toc' )    ||
             p.closest( '#toc' )           ||
             p.closest( '.mw-editsection' ) ) return;
      }
      var orig = node.textContent;
      if ( !orig.trim() ) return;
      var span = document.createElement( 'span' );
      span.setAttribute( 'data-deva', orig );
      span.textContent = orig;
      p.replaceChild( span, node );
      translatableSpans.push( span );
    } );
  }

  // ── Apply a script to all tagged spans ─────────────────────────
  function applyScript( script ) {
    currentScript = script;
    translatableSpans.forEach( function ( span ) {
      if ( !span.parentNode ) return; // detached node
      var orig = span.getAttribute( 'data-deva' );
      span.textContent = ( script === 'deva' )
        ? orig
        : transliterateText( orig, script );
    } );
  }

  // ── TOC active-item highlight ───────────────────────────────────
  // MutationObserver watches Vector 2022 TOC class changes and colours
  // the active link orange.  This block is not modified.
  function watchTocActive() {
    var toc = document.querySelector( '.vector-toc' );
    if ( !toc || toc._grObserved ) return;
    toc._grObserved = true;

    var observer = new MutationObserver( function ( mutations ) {
      mutations.forEach( function ( m ) {
        if ( m.attributeName !== 'class' ) return;
        var li   = m.target;
        var link = li.querySelector( ':scope > .vector-toc-link' );
        if ( !link ) return;
        if ( li.classList.contains( 'vector-toc-list-item-active' ) ) {
          link.style.color      = '#f57c00';
          link.style.fontWeight = '700';
        } else {
          link.style.color      = '';
          link.style.fontWeight = '';
        }
      } );
    } );

    toc.querySelectorAll( '.vector-toc-list-item' ).forEach( function ( li ) {
      observer.observe( li, { attributes: true, attributeFilter: [ 'class' ] } );
    } );
  }

  // ── Init ────────────────────────────────────────────────────────
  function init() {
    var content      = document.querySelector( '.mw-parser-output' );
    var alreadyTagged = content && content.querySelector( '[data-deva]' );
    if ( !alreadyTagged ) {
      translatableSpans = [];
      tagTextNodes();
    }

    // Apply the globally persisted script preference immediately
    var saved = ( function () {
      try { return localStorage.getItem( LS_SCRIPT_KEY ); } catch ( e ) { return null; }
    }() );
    if ( saved && saved !== 'deva' ) {
      applyScript( saved );
    } else {
      currentScript = 'deva';
    }

    watchTocActive();
  }

  // ── React to toolbar dropdown changes on the same page ─────────
  // The ReaderToolbar dispatches 'gr-script-change' when the user
  // picks a new script.  We just call applyScript() — no need to
  // touch localStorage again (the toolbar already wrote it).
  window.addEventListener( 'gr-script-change', function ( e ) {
    var script = e && e.detail && e.detail.script;
    if ( script ) applyScript( script );
  } );

  // ── MediaWiki hook (SPA-style navigation support) ───────────────
  if ( window.mw ) {
    mw.hook( 'wikipage.content' ).add( function () {
      setTimeout( function () {
        var content      = document.querySelector( '.mw-parser-output' );
        var alreadyTagged = content && content.querySelector( '[data-deva]' );
        if ( !alreadyTagged ) {
          translatableSpans = [];
          tagTextNodes();
        }
        // Re-apply current script to newly loaded content
        if ( currentScript !== 'deva' ) {
          applyScript( currentScript );
        }
        watchTocActive();
      }, 150 );
    } );
  }

  // ── Fallback for non-MW environments ───────────────────────────
  if ( document.readyState === 'loading' ) {
    document.addEventListener( 'DOMContentLoaded', init );
  } else {
    init();
  }

}() );