MediaWiki:Common.js: Difference between revisions

No edit summary
No edit summary
Line 1: Line 1:
/* MediaWiki:Common.js — grantha.io  (v7)
/* MediaWiki:Common.js — grantha.io  (v7) */
*
* Changes vs v6:
*  1. BUG FIX: Main IIFE was never properly closed — About link block was
*    pasted inside it without the closing }() ); which broke the module
*    boundary and killed the by-Author toggle. Each block is now its
*    own self-contained IIFE.
*  2. TOC: "Beginning" link removed.
*  3. TOC: Heading label → "विषयसूची".
*  4. TOC: मूल / उल्लेख nav links injected at top of TOC panel.
*  5. TOC: All subsections expanded by default.
*  6. TOC: Only the active item is bold; all others are normal weight.
*  7. About link URL derived from wgArticlePath — works under any
*    sub-path install (e.g. /My_wiki/). Points to "My_wiki:About".
*  8. TOC customisations skipped on Main_Page and About pages.
*  9. Uses Vector's native TOC with IntersectionObserver-based active highlight.
*/


( function () {
( function () {
Line 22: Line 6:
   var currentScript = 'deva';
   var currentScript = 'deva';


  // ── IAST transliteration ────────────────────────────────────────
   function devanagariToIAST( text ) {
   function devanagariToIAST( text ) {
     var CONSONANTS = {
     var CONSONANTS = {
Line 67: Line 50:
   }
   }


  // ── Character maps for Kannada / Tamil ─────────────────────────
   var SCRIPT_MAP = {
   var SCRIPT_MAP = {
     kn: {
     kn: {
Line 91: Line 73:
       'क':'க','ख':'க','ग':'க','घ':'க','ङ':'ங',
       'क':'க','ख':'க','ग':'க','घ':'க','ङ':'ங',
       'च':'ச','छ':'ச','ज':'ஜ','झ':'ஜ','ञ':'ஞ',
       'च':'ச','छ':'ச','ज':'ஜ','झ':'ஜ','ञ':'ஞ',
       '':'ட','ठ':'ட','ड':'ட','ढ':'ட','':'ண',
       '':'ட','ठ':'ட','ड':'ட','ढ':'ட','':'ண',
       'त':'த','थ':'த','द':'த','ध':'த','न':'ந',
       'त':'த','थ':'த','द':'த','ध':'த','न':'ந',
       'प':'ப','फ':'ப','ब':'ப','भ':'ப','म':'ம',
       'प':'ப','फ':'ப','ब':'ப','भ':'ப','म':'ம',
Line 98: Line 80:
       'ा':'ா','ि':'ி','ी':'ீ','ु':'ு','ू':'ூ',
       'ा':'ா','ि':'ி','ी':'ீ','ु':'ு','ू':'ூ',
       'ृ':'ு','ॄ':'ூ',
       'ृ':'ு','ॄ':'ூ',
       'े':'ே','':'ை','ो':'ோ','ौ':'ௌ',
       'े':'ே','':'ை','ो':'ோ','ौ':'ௌ',
       'ं':'ம்','ः':':','ँ':'ம்','्':'்','ॐ':'ௐ','ऽ':'ௗ',
       'ं':'ம்','ः':':','ँ':'ம்','्':'்','ॐ':'ௐ','ऽ':'ௗ',
       '०':'0','१':'1','२':'2','३':'3','४':'4',
       '०':'0','१':'1','२':'2','३':'3','४':'4',
Line 122: Line 104:
   }
   }


  // ── Tag all transliteratable text nodes once per page ───────────
   var translatableSpans = [];
   var translatableSpans = [];


Line 131: Line 112:
       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;
Line 149: Line 129:
       } );
       } );
     }
     }
     document.querySelectorAll( '.vector-toc .vector-toc-text' ).forEach( function ( span ) {
     document.querySelectorAll( '.vector-toc .vector-toc-text' ).forEach( function ( span ) {
       if ( span.hasAttribute( 'data-deva' ) ) return;
       if ( span.hasAttribute( 'data-deva' ) ) return;
Line 159: Line 138:
   }
   }


  // ── Apply a script to all tagged spans ─────────────────────────
   function applyScript( script ) {
   function applyScript( script ) {
     currentScript = script;
     currentScript = script;
Line 172: Line 150:
   }
   }


  // ── Pages where sidebar TOC should not be modified ──────────────
   function _isNoTocPage() {
   function _isNoTocPage() {
     var pn = ( window.mw && mw.config && mw.config.get( 'wgPageName' ) ) || '';
     var pn = ( window.mw && mw.config && mw.config.get( 'wgPageName' ) ) || '';
Line 178: Line 155:
   }
   }


   // ── TOC: rename "Contents" → "विषयसूची" ────────────────────────
   function renameTocTitle() {
function renameTocTitle() {
    if ( _isNoTocPage() ) return;
  if ( _isNoTocPage() ) return;
    var toc = document.querySelector('.vector-toc');
 
    if ( !toc ) return;
  var toc = document.querySelector('.vector-toc');
    var titleEl = toc.querySelector('.vector-toc-title') || toc.querySelector('.vector-pinnable-header-label');
  if ( !toc ) return;
    if ( !titleEl ) return;
 
    var LABEL = 'विषयसूची';
  var titleEl =
    var span = titleEl.querySelector('.gr-toc-title');
    toc.querySelector('.vector-toc-title') ||
    if ( !span ) {
    toc.querySelector('.vector-pinnable-header-label');
      titleEl.innerHTML = '';
 
      span = document.createElement('span');
  if ( !titleEl ) return;
      span.className = 'gr-toc-title';
 
      span.setAttribute('data-deva', LABEL);
  var LABEL = 'विषयसूची';
      titleEl.appendChild(span);
 
      translatableSpans.push(span);
  var span = titleEl.querySelector('.gr-toc-title');
    }
 
    span.textContent = currentScript === 'deva' ? LABEL : transliterateText(LABEL, currentScript);
  if ( !span ) {
    titleEl.innerHTML = '';
 
    span = document.createElement('span');
    span.className = 'gr-toc-title';
    span.setAttribute('data-deva', LABEL);
 
    titleEl.appendChild(span);
    translatableSpans.push(span);
   }
   }


  span.textContent =
    currentScript === 'deva'
      ? LABEL
      : transliterateText(LABEL, currentScript);
}
  // ── TOC: Remove the "Beginning" / top-of-page link ─────────────
   function removeTocBeginning() {
   function removeTocBeginning() {
     if ( _isNoTocPage() ) return;
     if ( _isNoTocPage() ) return;
Line 230: Line 192:
   }
   }


  // ── TOC: Expand all collapsed subsections on load ───────────────
   function expandTocSections() {
   function expandTocSections() {
     if ( _isNoTocPage() ) return;
     if ( _isNoTocPage() ) return;
Line 240: Line 201:
   }
   }


  // ── TOC: Inject मूल / उल्लेख nav links ─────────────────────────
   function injectTocDocNav() {
   function injectTocDocNav() {
     if ( _isNoTocPage() ) return;
     if ( _isNoTocPage() ) return;
Line 246: Line 206:
     if ( !toc ) return;
     if ( !toc ) return;
     if ( document.getElementById( 'gr-toc-doc-nav' ) ) return;
     if ( document.getElementById( 'gr-toc-doc-nav' ) ) return;
     var artPath  = ( window.mw && mw.config && mw.config.get( 'wgArticlePath' ) ) || '/wiki/$1';
     var artPath  = ( window.mw && mw.config && mw.config.get( 'wgArticlePath' ) ) || '/wiki/$1';
     var pageTitle = ( window.mw && mw.config && mw.config.get( 'wgPageName'  ) ) || '';
     var pageTitle = ( window.mw && mw.config && mw.config.get( 'wgPageName'  ) ) || '';
     var teekaPage  = document.querySelector( '.gr-teeka-page' );
     var teekaPage  = document.querySelector( '.gr-teeka-page' );
     var primarySlug = teekaPage ? ( teekaPage.getAttribute( 'data-primary' ) || '' ) : '';
     var primarySlug = teekaPage ? ( teekaPage.getAttribute( 'data-primary' ) || '' ) : '';
    var docSlug    = teekaPage ? ( teekaPage.getAttribute( 'data-slug'    ) || '' ) : '';
     if ( !primarySlug ) { primarySlug = pageTitle.split( '/' )[ 0 ]; }
 
     if ( !primarySlug ) {
      primarySlug = pageTitle.split( '/' )[ 0 ];
      docSlug    = primarySlug;
    }
     if ( !primarySlug ) return;
     if ( !primarySlug ) return;
     function wikiUrl( slug ) {
     function wikiUrl( slug ) {
       if ( window.mw && mw.util && mw.util.getUrl ) return mw.util.getUrl( slug );
       if ( window.mw && mw.util && mw.util.getUrl ) return mw.util.getUrl( slug );
       return artPath.replace( '$1', encodeURIComponent( slug ).replace( /%2F/g, '/' ) );
       return artPath.replace( '$1', encodeURIComponent( slug ).replace( /%2F/g, '/' ) );
     }
     }
 
     var docTitleEl     = document.querySelector( '.gr-doc-title' );
     var moolaUrl  = wikiUrl( primarySlug );
     var ullekhaUrl = wikiUrl( primarySlug + '/Ullekha' );
 
    var docTitleEl  = document.querySelector( '.gr-doc-title' );
     var hasMoolaPage  = docTitleEl && docTitleEl.getAttribute( 'data-has-moola' )  === '1';
     var hasMoolaPage  = docTitleEl && docTitleEl.getAttribute( 'data-has-moola' )  === '1';
     var hasUllekhaPage = docTitleEl && docTitleEl.getAttribute( 'data-has-ullekha' ) === '1';
     var hasUllekhaPage = docTitleEl && docTitleEl.getAttribute( 'data-has-ullekha' ) === '1';
    var moolaPageUrl = wikiUrl( primarySlug + '/Moola' );
     var showMoolaPage  = !teekaPage && hasMoolaPage;
     var showMoolaPage  = !teekaPage && hasMoolaPage;
     var showMoolaBack  = !!teekaPage;
     var showMoolaBack  = !!teekaPage;
     var showUllekha    = hasUllekhaPage || !!teekaPage;
     var showUllekha    = hasUllekhaPage || !!teekaPage;
     if ( !showMoolaPage && !showMoolaBack && !showUllekha ) return;
     if ( !showMoolaPage && !showMoolaBack && !showUllekha ) return;
     var nav = document.createElement( 'div' );
     var nav = document.createElement( 'div' );
     nav.id = 'gr-toc-doc-nav';
     nav.id = 'gr-toc-doc-nav';
     nav.setAttribute( 'class', 'toc-main-links');
     nav.setAttribute( 'class', 'toc-main-links' );
 
     function makeBtn( href, label ) {
     function makeBtn( href, label ) {
       var a = document.createElement( 'a' );
       var a = document.createElement( 'a' );
       a.href = href;
       a.href = href;
       a.setAttribute( 'class', 'toc-main-link-item');
       a.setAttribute( 'class', 'toc-main-link-item' );
       var lspan = document.createElement( 'span' );
       var lspan = document.createElement( 'span' );
       lspan.setAttribute( 'data-deva', label );
       lspan.setAttribute( 'data-deva', label );
       lspan.textContent = ( currentScript && currentScript !== 'deva' )
       lspan.textContent = ( currentScript && currentScript !== 'deva' ) ? transliterateText( label, currentScript ) : label;
        ? transliterateText( label, currentScript )
        : label;
       translatableSpans.push( lspan );
       translatableSpans.push( lspan );
       a.appendChild( lspan );
       a.appendChild( lspan );
Line 299: Line 239:
       return a;
       return a;
     }
     }
 
     if ( showMoolaPage ) nav.appendChild( makeBtn( wikiUrl( primarySlug + '/Moola' ), 'मूलम्' ) );
     if ( showMoolaPage ) nav.appendChild( makeBtn( moolaPageUrl, 'मूलम्' ) );
     if ( showMoolaBack ) nav.appendChild( makeBtn( wikiUrl( primarySlug ),           'मूल'   ) );
     if ( showMoolaBack ) nav.appendChild( makeBtn( moolaUrl,     'मूल' ) );
     if ( showUllekha  ) nav.appendChild( makeBtn( wikiUrl( primarySlug + '/Ullekha'), 'उल्लेख') );
     if ( showUllekha  ) nav.appendChild( makeBtn( ullekhaUrl,   'उल्लेख') );
 
     var tocContents = toc.querySelector( '.vector-toc-contents' );
     var tocContents = toc.querySelector( '.vector-toc-contents' );
     if ( tocContents ) toc.insertBefore( nav, tocContents );
     if ( tocContents ) toc.insertBefore( nav, tocContents );
Line 309: Line 247:
   }
   }


  // ── TOC active-item highlight ────────────────────────────────────
  function watchTocActive() {
    if ( _isNoTocPage() ) return;
    var toc = document.querySelector( '.vector-toc' );
    if ( !toc ) return;
    if ( !window.MutationObserver ) return;
    if ( toc._grObserved ) {
      toc.querySelectorAll( '.vector-toc-list-item' ).forEach( attachHighlight );
      return;
    }
    toc._grObserved = true;
    var ACTIVE_COLOR = '#f57c00';
    function setActive( li, on ) {
      var link = li.querySelector( '.vector-toc-link' ) || li.querySelector( 'a' );
      if ( !link ) return;
      var hasActiveChild = !!li.querySelector(
        '.vector-toc-list-item .vector-toc-list-item-active'
      );
      var shouldHighlight = on && !hasActiveChild;
      if ( shouldHighlight ) {
        link.style.setProperty( 'color',      ACTIVE_COLOR, 'important' );
        link.style.setProperty( 'font-weight', '700',        'important' );
        link.querySelectorAll( '*' ).forEach( function ( el ) {
          el.style.setProperty( 'color',      ACTIVE_COLOR, 'important' );
          el.style.setProperty( 'font-weight', '700',        'important' );
        } );
      } else {
        link.style.removeProperty( 'color' );
        link.style.setProperty( 'font-weight', '400', 'important' );
        link.querySelectorAll( '*' ).forEach( function ( el ) {
          el.style.removeProperty( 'color' );
          el.style.setProperty( 'font-weight', '400', 'important' );
        } );
      }
    }
    function normaliseAll() {
      toc.querySelectorAll( '.vector-toc-list-item' ).forEach( function ( li ) {
        setActive( li, li.classList.contains( 'vector-toc-list-item-active' ) );
      } );
    }
    function attachHighlight( li ) {
      if ( li._grHighlightAttached ) return;
      li._grHighlightAttached = true;
      liObs.observe( li, { attributes: true, attributeFilter: [ 'class' ] } );
      setActive( li, li.classList.contains( 'vector-toc-list-item-active' ) );
    }
    var liObs = new MutationObserver( function ( mutations ) {
      mutations.forEach( function ( m ) {
        if ( m.attributeName !== 'class' ) return;
        var li    = m.target;
        var active = li.classList.contains( 'vector-toc-list-item-active' );
        setActive( li, active );
        if ( active ) {
          var anc = li.parentNode;
          while ( anc && anc !== document.body ) {
            if ( anc.classList &&
                anc.classList.contains( 'vector-toc-list-item' ) &&
                anc.classList.contains( 'vector-toc-list-item-collapsed' ) ) {
              anc.classList.remove( 'vector-toc-list-item-collapsed' );
            }
            anc = anc.parentNode;
          }
          var container = document.querySelector( '.vector-sticky-pinned-container' );
          if ( container ) {
            var vis = container.offsetHeight > 0 &&
                      container.offsetParent !== null &&
                      window.getComputedStyle( container ).display !== 'none' &&
                      window.getComputedStyle( container ).visibility !== 'hidden';
            if ( vis ) {
              var lr = li.getBoundingClientRect();
              var cr = container.getBoundingClientRect();
              if ( lr.top < cr.top + 4 || lr.bottom > cr.bottom - 4 ) {
                var sh = null, node = li.parentNode;
                while ( node && node !== document.body ) {
                  var ov = window.getComputedStyle( node ).overflowY;
                  if ( ( ov === 'auto' || ov === 'scroll' ) &&
                      node.scrollHeight > node.clientHeight ) { sh = node; break; }
                  node = node.parentNode;
                }
                if ( !sh && container.scrollHeight > container.clientHeight ) sh = container;
                if ( sh ) {
                  var hr  = sh.getBoundingClientRect();
                  var top = lr.top - hr.top + sh.scrollTop;
                  sh.scrollTop = Math.max( 0, top - sh.clientHeight / 2 + li.offsetHeight / 2 );
                }
              }
            }
          }
        }
      } );
    } );
    var structObs = new MutationObserver( function ( mutations ) {
      mutations.forEach( function ( m ) {
        if ( m.type !== 'childList' ) return;
        m.addedNodes.forEach( function ( n ) {
          if ( n.nodeType !== 1 ) return;
          if ( n.classList && n.classList.contains( 'vector-toc-list-item' ) ) attachHighlight( n );
          if ( n.querySelectorAll ) n.querySelectorAll( '.vector-toc-list-item' ).forEach( attachHighlight );
          var newSpans = [];
          if ( n.classList && n.classList.contains( 'vector-toc-text' ) ) newSpans.push( n );
          if ( n.querySelectorAll ) n.querySelectorAll( '.vector-toc-text' ).forEach( function ( s ) { newSpans.push( s ); } );
          newSpans.forEach( function ( span ) {
            if ( span.hasAttribute( 'data-deva' ) ) return;
            var orig = span.textContent;
            if ( !orig.trim() ) return;
            span.setAttribute( 'data-deva', orig );
            if ( currentScript !== 'deva' ) span.textContent = transliterateText( orig, currentScript );
            translatableSpans.push( span );
          } );
        } );
      } );
    } );
    toc.querySelectorAll( '.vector-toc-list-item' ).forEach( attachHighlight );
    structObs.observe( toc, { childList: true, subtree: true } );
    setTimeout( function () {
      normaliseAll();
      var active = toc.querySelector( '.vector-toc-list-item-active' );
      if ( active ) setActive( active, true );
    }, 300 );
  }
  // ── TOC: Use Vector's native TOC, add highlight + expand + rename ──────
  function setupToc() {
    if ( _isNoTocPage() ) return;
    var toc = document.querySelector( '.vector-toc' );
    if ( !toc ) return;
    removeTocBeginning();
    renameTocTitle();
    expandTocSections();
    injectTocDocNav();
    attachHeadingObserver();
  }
  // ── IntersectionObserver on actual heading elements ──────────────
   var _headingObserver = null;
   var _headingObserver = null;


Line 460: Line 252:
     if ( _isNoTocPage() ) return;
     if ( _isNoTocPage() ) return;
     if ( _headingObserver ) return;
     if ( _headingObserver ) return;
     if ( !window.IntersectionObserver ) {
     if ( !window.IntersectionObserver ) return;
      return; /* IntersectionObserver not available — skip active highlight */
    }
 
     var ACTIVE_COLOR = '#f57c00';
     var ACTIVE_COLOR = '#f57c00';
     var _activeId    = null;
     var _activeId    = null;
     var content  = document.querySelector( '.mw-parser-output' );
     var content  = document.querySelector( '.mw-parser-output' );
     if ( !content ) return;
     if ( !content ) return;
     var headings = Array.from(
     var headings = Array.from( content.querySelectorAll( 'h1[id],h2[id],h3[id],h4[id],h5[id],h6[id]' ) );
      content.querySelectorAll( 'h1[id],h2[id],h3[id],h4[id],h5[id],h6[id]' )
    );
     if ( !headings.length ) return;
     if ( !headings.length ) return;
     var toc = document.querySelector( '.vector-toc' );
     var toc = document.querySelector( '.vector-toc' );
     if ( !toc ) return;
     if ( !toc ) return;
 
     function getTocLink( id ) { return toc.querySelector( 'a[href="#' + CSS.escape( id ) + '"]' ); }
     function getTocLink( id ) {
     function getTocLi( id ) { var a = getTocLink( id ); return a ? a.closest( '.vector-toc-list-item' ) : null; }
      return toc.querySelector( 'a[href="#' + CSS.escape( id ) + '"]' );
    }
 
     function getTocLi( id ) {
      var a = getTocLink( id );
      return a ? a.closest( '.vector-toc-list-item' ) : null;
    }
 
     function clearActive() {
     function clearActive() {
      /* Clear ALL toc items — prevents multiple items staying highlighted simultaneously */
       toc.querySelectorAll( '.vector-toc-list-item' ).forEach( function ( li ) {
       toc.querySelectorAll( '.vector-toc-list-item' ).forEach( function ( li ) {
         li.classList.remove( 'vector-toc-list-item-active' );
         li.classList.remove( 'vector-toc-list-item-active' );
Line 494: Line 270:
         lnk.style.removeProperty( 'color' );
         lnk.style.removeProperty( 'color' );
         lnk.style.setProperty( 'font-weight', '400', 'important' );
         lnk.style.setProperty( 'font-weight', '400', 'important' );
         lnk.querySelectorAll( '*' ).forEach( function ( el ) {
         lnk.querySelectorAll( '*' ).forEach( function ( el ) { el.style.removeProperty( 'color' ); el.style.setProperty( 'font-weight', '400', 'important' ); } );
          el.style.removeProperty( 'color' );
          el.style.setProperty( 'font-weight', '400', 'important' );
        } );
       } );
       } );
     }
     }
     function setActive( id ) {
     function setActive( id ) {
       if ( _activeId === id ) return;
       if ( _activeId === id ) return;
Line 506: Line 278:
       clearActive();
       clearActive();
       if ( !id ) return;
       if ( !id ) return;
       var li = getTocLi( id );
       var li = getTocLi( id );
       if ( !li ) return;
       if ( !li ) return;
       li.classList.add( 'vector-toc-list-item-active' );
       li.classList.add( 'vector-toc-list-item-active' );
 
       var hasActiveChild = !!li.querySelector( '.vector-toc-list-item .vector-toc-list-item-active' );
       var hasActiveChild = !!li.querySelector(
        '.vector-toc-list-item .vector-toc-list-item-active'
      );
       if ( !hasActiveChild ) {
       if ( !hasActiveChild ) {
         var lnk = li.querySelector( '.vector-toc-link' );
         var lnk = li.querySelector( '.vector-toc-link' );
         if ( lnk ) {
         if ( lnk ) {
           lnk.style.setProperty( 'color',       ACTIVE_COLOR, 'important' );
           lnk.style.setProperty( 'color', ACTIVE_COLOR, 'important' );
           lnk.style.setProperty( 'font-weight', '700',       'important' );
           lnk.style.setProperty( 'font-weight', '700', 'important' );
           lnk.querySelectorAll( '*' ).forEach( function ( el ) {
           lnk.querySelectorAll( '*' ).forEach( function ( el ) { el.style.setProperty( 'color', ACTIVE_COLOR, 'important' ); el.style.setProperty( 'font-weight', '700', 'important' ); } );
            el.style.setProperty( 'color',       ACTIVE_COLOR, 'important' );
            el.style.setProperty( 'font-weight', '700',       'important' );
          } );
         }
         }
       }
       }
      // Expand ALL ancestor sections — remove collapsed, force display
       var anc = li.parentNode;
       var anc = li.parentNode;
       while ( anc && anc !== toc ) {
       while ( anc && anc !== toc ) {
         if ( anc.classList ) {
         if ( anc.classList ) anc.classList.remove( 'vector-toc-list-item-collapsed' );
          anc.classList.remove( 'vector-toc-list-item-collapsed' );
         if ( anc.tagName === 'UL' || anc.tagName === 'LI' ) anc.style.removeProperty( 'display' );
        }
         if ( anc.tagName === 'UL' || anc.tagName === 'LI' ) {
          anc.style.removeProperty( 'display' );
        }
         anc = anc.parentNode;
         anc = anc.parentNode;
       }
       }
      // Scroll TOC item into view within sidebar
       var sticky = document.querySelector( '.vector-sticky-pinned-container' );
       var sticky = document.querySelector( '.vector-sticky-pinned-container' );
       var scrollEl = sticky || toc;
       var scrollEl = sticky || toc;
Line 550: Line 306:
       }
       }
     }
     }
     var _visible = new Set();
     var _visible = new Set();
     _headingObserver = new IntersectionObserver( function ( entries ) {
     _headingObserver = new IntersectionObserver( function ( entries ) {
       entries.forEach( function ( entry ) {
       entries.forEach( function ( entry ) {
         if ( entry.isIntersecting ) {
         if ( entry.isIntersecting ) _visible.add( entry.target.id );
          _visible.add( entry.target.id );
         else _visible.delete( entry.target.id );
         } else {
          _visible.delete( entry.target.id );
        }
       } );
       } );
 
       var topId = null, topY = Infinity;
       var topId = null;
      var topY = Infinity;
       _visible.forEach( function ( id ) {
       _visible.forEach( function ( id ) {
         var el = document.getElementById( id );
         var el = document.getElementById( id );
         if ( el ) {
         if ( el ) { var y = el.getBoundingClientRect().top; if ( y >= 0 && y < topY ) { topY = y; topId = id; } }
          var y = el.getBoundingClientRect().top;
          if ( y >= 0 && y < topY ) { topY = y; topId = id; }
        }
       } );
       } );
       if ( !topId ) {
       if ( !topId ) {
         var bestY = -Infinity;
         var bestY = -Infinity;
         headings.forEach( function ( h ) {
         headings.forEach( function ( h ) { var y = h.getBoundingClientRect().top; if ( y < 0 && y > bestY ) { bestY = y; topId = h.id; } } );
          var y = h.getBoundingClientRect().top;
          if ( y < 0 && y > bestY ) { bestY = y; topId = h.id; }
        } );
       }
       }
       setActive( topId || null );
       setActive( topId || null );
     }, {
     }, { rootMargin: '-60px 0px -65% 0px', threshold: 0 } );
      rootMargin: '-60px 0px -65% 0px',
     headings.forEach( function ( h ) { _headingObserver.observe( h ); } );
      threshold: 0
  }
     } );


     headings.forEach( function ( h ) { _headingObserver.observe( h ); } );
  function setupToc() {
     if ( _isNoTocPage() ) return;
    var toc = document.querySelector( '.vector-toc' );
    if ( !toc ) return;
    removeTocBeginning();
    renameTocTitle();
    expandTocSections();
    injectTocDocNav();
    attachHeadingObserver();
   }
   }


  // ── Init ────────────────────────────────────────────────────────
   function init() {
   function init() {
     var HIDE_IDS = [
     var HIDE_IDS = [ 'vector-appearance', 'vector-appearance-pinned-container', 'vector-appearance-unpinned-container' ];
      'vector-appearance',
      'vector-appearance-pinned-container',
      'vector-appearance-unpinned-container'
    ];
     function removeHiddenEls() {
     function removeHiddenEls() {
       HIDE_IDS.forEach( function ( id ) {
       HIDE_IDS.forEach( function ( id ) { var el = document.getElementById( id ); if ( el && el.parentNode ) el.parentNode.removeChild( el ); } );
        var el = document.getElementById( id );
       var pt = document.getElementById( 'vector-page-tools' ) || document.querySelector( '.vector-page-tools-pinned-container' );
        if ( el && el.parentNode ) el.parentNode.removeChild( el );
       if ( pt ) pt.querySelectorAll( '[aria-controls="vector-appearance"]' ).forEach( function ( el ) { if ( el.parentNode ) el.parentNode.removeChild( el ); } );
      } );
       var pt = document.getElementById( 'vector-page-tools' ) ||
              document.querySelector( '.vector-page-tools-pinned-container' );
       if ( pt ) {
        pt.querySelectorAll( '[aria-controls="vector-appearance"]' )
          .forEach( function ( el ) { if ( el.parentNode ) el.parentNode.removeChild( el ); } );
      }
     }
     }
     removeHiddenEls();
     removeHiddenEls();
     ( function detectTeekaMode() {
     ( function detectTeekaMode() {
       var tp = document.querySelector( '.gr-teeka-page' );
       var tp = document.querySelector( '.gr-teeka-page' );
Line 617: Line 352:
       var mainUrl = artPath.replace( '$1', primary );
       var mainUrl = artPath.replace( '$1', primary );
       var refParam = window.location.search.match( /[?&]ref=([01])/ );
       var refParam = window.location.search.match( /[?&]ref=([01])/ );
       if ( refParam ) {
       if ( refParam ) { document.body.classList.add( refParam[1] === '1' ? 'gr-ref-mode' : 'gr-standalone' ); return; }
        document.body.classList.add( refParam[1] === '1' ? 'gr-ref-mode' : 'gr-standalone' );
        return;
      }
       var ref = document.referrer || '';
       var ref = document.referrer || '';
       document.body.classList.add(
       document.body.classList.add( ( ref && primary && ref.indexOf( mainUrl ) !== -1 ) ? 'gr-ref-mode' : 'gr-standalone' );
        ( ref && primary && ref.indexOf( mainUrl ) !== -1 ) ? 'gr-ref-mode' : 'gr-standalone'
      );
     }() );
     }() );
     if ( window.MutationObserver ) {
     if ( window.MutationObserver ) {
       var hideObs = new MutationObserver( function ( mutations ) {
       var hideObs = new MutationObserver( function ( mutations ) {
Line 636: Line 365:
       setTimeout( function () { hideObs.disconnect(); }, 6000 );
       setTimeout( function () { hideObs.disconnect(); }, 6000 );
     }
     }
 
     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 = []; tagTextNodes(); }
      translatableSpans = [];
     else {
      tagTextNodes();
     } else {
       document.querySelectorAll( '.vector-toc .vector-toc-text:not([data-deva])' ).forEach( function ( span ) {
       document.querySelectorAll( '.vector-toc .vector-toc-text:not([data-deva])' ).forEach( function ( span ) {
         var orig = span.textContent;
         var orig = span.textContent; if ( !orig.trim() ) return;
        if ( !orig.trim() ) return;
         span.setAttribute( 'data-deva', orig ); translatableSpans.push( span );
         span.setAttribute( 'data-deva', orig );
        translatableSpans.push( span );
       } );
       } );
     }
     }
 
     var saved = ( function () { try { return localStorage.getItem( LS_SCRIPT_KEY ); } catch ( e ) { return null; } }() );
     var saved = ( function () {
     if ( saved && saved !== 'deva' ) { applyScript( saved ); } else { currentScript = 'deva'; }
      try { return localStorage.getItem( LS_SCRIPT_KEY ); } catch ( e ) { return null; }
    }() );
     if ( saved && saved !== 'deva' ) { applyScript( saved ); }
    else { currentScript = 'deva'; }
 
     setTimeout( setupToc, 200 );
     setTimeout( setupToc, 200 );
   }
   }


  // ── React to toolbar script-change events ──────────────────────
   window.addEventListener( 'gr-script-change', function ( e ) {
   window.addEventListener( 'gr-script-change', function ( e ) {
     var script = e && e.detail && e.detail.script;
     var script = e && e.detail && e.detail.script;
Line 666: Line 384:
   } );
   } );


  // ── React to gr-new-content (siteNav panel rendered new items) ──
   window.addEventListener( 'gr-new-content', function ( e ) {
   window.addEventListener( 'gr-new-content', function ( e ) {
     var container = e && e.detail && e.detail.container;
     var container = e && e.detail && e.detail.container;
Line 686: Line 403:
   } );
   } );


  // ── React to script changes from OTHER tabs (BroadcastChannel) ──
   try {
   try {
     var _grBC = new BroadcastChannel( 'gr-script' );
     var _grBC = new BroadcastChannel( 'gr-script' );
     _grBC.onmessage = function ( e ) {
     _grBC.onmessage = function ( e ) {
       var script = e && e.data && e.data.script;
       var script = e && e.data && e.data.script;
       if ( script ) {
       if ( script ) { currentScript = script; var sel = document.querySelector( '.gr-script-sel' ); if ( sel ) sel.value = script; applyScript( script ); }
        currentScript = script;
        var sel = document.querySelector( '.gr-script-sel' );
        if ( sel ) sel.value = script;
        applyScript( script );
      }
     };
     };
   } catch ( e ) {}
   } catch ( e ) {}


  // ── MediaWiki SPA-style navigation ──────────────────────────────
   if ( window.mw ) {
   if ( window.mw ) {
     mw.hook( 'wikipage.content' ).add( function () {
     mw.hook( 'wikipage.content' ).add( function () {
       setTimeout( function () {
       setTimeout( function () {
         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 = []; tagTextNodes(); }
          translatableSpans = [];
         else {
          tagTextNodes();
         } else {
           document.querySelectorAll( '.vector-toc .vector-toc-text:not([data-deva])' ).forEach( function ( span ) {
           document.querySelectorAll( '.vector-toc .vector-toc-text:not([data-deva])' ).forEach( function ( span ) {
             var orig = span.textContent;
             var orig = span.textContent; if ( !orig.trim() ) return;
            if ( !orig.trim() ) return;
             span.setAttribute( 'data-deva', orig ); translatableSpans.push( span );
             span.setAttribute( 'data-deva', orig );
            translatableSpans.push( span );
           } );
           } );
         }
         }
Line 723: Line 429:
   }
   }


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


}() );  /* ← end of main IIFE */
}() );  /* ← end of main IIFE */
Line 739: Line 442:
     return ap.replace( '$1', title );
     return ap.replace( '$1', title );
   }
   }
 
   var linkStyle = [ 'color:rgba(255,255,255,0.88)','font-size:0.88em','font-family:system-ui,sans-serif','font-weight:500','text-decoration:none','padding:4px 10px','border-radius:4px','margin-right:4px','transition:color 0.15s,background 0.15s','white-space:nowrap' ].join( ';' );
   var linkStyle = [
    'color:rgba(255,255,255,0.88)', 'font-size:0.88em',
    'font-family:system-ui,sans-serif', 'font-weight:500',
    'text-decoration:none', 'padding:4px 10px', 'border-radius:4px',
    'margin-right:4px', 'transition:color 0.15s,background 0.15s',
    'white-space:nowrap',
  ].join( ';' );
 
   function makeHeaderLink( id, href, label ) {
   function makeHeaderLink( id, href, label ) {
     var a = document.createElement( 'a' );
     var a = document.createElement( 'a' );
     a.id = id; a.href = href; a.textContent = label;
     a.id = id; a.href = href; a.textContent = label; a.style.cssText = linkStyle;
    a.style.cssText = linkStyle;
     a.addEventListener( 'mouseover', function () { this.style.color = '#fff'; this.style.background = 'rgba(255,255,255,0.12)'; } );
     a.addEventListener( 'mouseover', function () {
     a.addEventListener( 'mouseout', function () { this.style.color = 'rgba(255,255,255,0.88)'; this.style.background = 'transparent'; } );
      this.style.color = '#fff'; this.style.background = 'rgba(255,255,255,0.12)';
    } );
     a.addEventListener( 'mouseout', function () {
      this.style.color = 'rgba(255,255,255,0.88)'; this.style.background = 'transparent';
    } );
     return a;
     return a;
   }
   }
   function injectHeaderLinks() {
   function injectHeaderLinks() {
     if ( document.getElementById( 'gr-about-link' ) ) return;
     if ( document.getElementById( 'gr-about-link' ) ) return;
 
     var headerEnd = document.querySelector( '.vector-header-end' ) || document.querySelector( '#vector-user-links' ) || document.querySelector( '.mw-header' );
     var headerEnd = document.querySelector( '.vector-header-end' ) ||
                    document.querySelector( '#vector-user-links' ) ||
                    document.querySelector( '.mw-header' );
     if ( !headerEnd ) return;
     if ( !headerEnd ) return;
     var helpLink  = makeHeaderLink( 'gr-help-link',  wikiHref( 'My_wiki:Help'  ), 'Help'  );
     var helpLink  = makeHeaderLink( 'gr-help-link',  wikiHref( 'My_wiki:Help'  ), 'Help'  );
     var aboutLink = makeHeaderLink( 'gr-about-link', wikiHref( 'My_wiki:About' ), 'About' );
     var aboutLink = makeHeaderLink( 'gr-about-link', wikiHref( 'My_wiki:About' ), 'About' );
 
     var ul = document.querySelector( '.vector-user-links' ) || document.querySelector( '#pt-userpage' );
     var ul = document.querySelector( '.vector-user-links' ) ||
     if ( ul && ul.parentNode === headerEnd ) { headerEnd.insertBefore( aboutLink, ul ); headerEnd.insertBefore( helpLink, aboutLink ); }
            document.querySelector( '#pt-userpage' );
     else { headerEnd.appendChild( helpLink ); headerEnd.appendChild( aboutLink ); }
     if ( ul && ul.parentNode === headerEnd ) {
      headerEnd.insertBefore( aboutLink, ul );
      headerEnd.insertBefore( helpLink, aboutLink );
     } else {
      headerEnd.appendChild( helpLink );
      headerEnd.appendChild( aboutLink );
    }
   }
   }
   if ( document.readyState === 'loading' ) document.addEventListener( 'DOMContentLoaded', injectHeaderLinks );
   if ( document.readyState === 'loading' ) document.addEventListener( 'DOMContentLoaded', injectHeaderLinks );
   else injectHeaderLinks();
   else injectHeaderLinks();
Line 802: Line 479:
     try { localStorage.setItem( 'gr_home_view', v ); } catch ( e ) {}
     try { localStorage.setItem( 'gr_home_view', v ); } catch ( e ) {}
   }
   }
   function initHomeToggle() {
   function initHomeToggle() {
     var gBtn = document.getElementById( 'gr-toggle-grantha' );
     var gBtn = document.getElementById( 'gr-toggle-grantha' );
Line 809: Line 485:
     gBtn.addEventListener( 'click', function () { grHomeView( 'grantha' ); } );
     gBtn.addEventListener( 'click', function () { grHomeView( 'grantha' ); } );
     aBtn.addEventListener( 'click', function () { grHomeView( 'author' );  } );
     aBtn.addEventListener( 'click', function () { grHomeView( 'author' );  } );
     [ gBtn, aBtn ].forEach( function ( btn ) {
     [ gBtn, aBtn ].forEach( function ( btn ) { btn.addEventListener( 'keydown', function ( e ) { if ( e.key === 'Enter' || e.key === ' ' ) btn.click(); } ); } );
      btn.addEventListener( 'keydown', function ( e ) {
     var saved; try { saved = localStorage.getItem( 'gr_home_view' ); } catch ( e ) {}
        if ( e.key === 'Enter' || e.key === ' ' ) btn.click();
      } );
    } );
     var saved;
    try { saved = localStorage.getItem( 'gr_home_view' ); } catch ( e ) {}
     if ( saved === 'author' ) grHomeView( 'author' );
     if ( saved === 'author' ) grHomeView( 'author' );
   }
   }
   if ( document.readyState === 'loading' ) document.addEventListener( 'DOMContentLoaded', initHomeToggle );
   if ( document.readyState === 'loading' ) document.addEventListener( 'DOMContentLoaded', initHomeToggle );
   else initHomeToggle();
   else initHomeToggle();
Line 826: Line 496:
// ── Ullekha reference link handler ─────────────────────────────────
// ── Ullekha reference link handler ─────────────────────────────────
( function () {
( function () {
   function highlightOnArrival() {
   function highlightOnArrival() {
     var search = window.location.search;
     var search = window.location.search;
Line 835: Line 504:
     try { needle = decodeURIComponent( m[ 1 ] ); } catch ( e ) { return; }
     try { needle = decodeURIComponent( m[ 1 ] ); } catch ( e ) { return; }
     if ( !needle || needle.length < 4 ) return;
     if ( !needle || needle.length < 4 ) return;
     function doHighlight() {
     function doHighlight() {
       var content = document.querySelector( '.mw-parser-output' );
       var content = document.querySelector( '.mw-parser-output' );
       if ( !content ) return;
       if ( !content ) return;
       var snippet = needle.slice( 0, 40 );
       var snippet = needle.slice( 0, 40 );
       var found  = false;
       var found  = false;
       var spans = content.querySelectorAll( '[data-deva]' );
       var spans = content.querySelectorAll( '[data-deva]' );
       for ( var i = 0; i < spans.length && !found; i++ ) {
       for ( var i = 0; i < spans.length && !found; i++ ) {
Line 848: Line 514:
         var orig  = spanEl.getAttribute( 'data-deva' ) || '';
         var orig  = spanEl.getAttribute( 'data-deva' ) || '';
         if ( orig.indexOf( snippet ) === -1 ) continue;
         if ( orig.indexOf( snippet ) === -1 ) continue;
         var idx    = orig.indexOf( snippet );
         var idx    = orig.indexOf( snippet );
         var hlText = orig.slice( idx, Math.min( idx + needle.length, orig.length ) );
         var hlText = orig.slice( idx, Math.min( idx + needle.length, orig.length ) );
         var mark  = document.createElement( 'mark' );
         var mark  = document.createElement( 'mark' );
         mark.className = 'gr-ullekha-highlight';
         mark.className = 'gr-ullekha-highlight';
         mark.style.cssText = 'background:#fff176;border-radius:2px;padding:0 2px;';
         mark.style.cssText = 'background:#fff176;border-radius:2px;padding:0 2px;';
         mark.textContent   = hlText;
         mark.textContent = hlText;
 
         var parent = spanEl.parentNode;
         var parent = spanEl.parentNode;
         if ( !parent ) continue;
         if ( !parent ) continue;
         var before = document.createTextNode( orig.slice( 0, idx ) );
         var before = document.createTextNode( orig.slice( 0, idx ) );
         var after  = document.createTextNode( orig.slice( idx + hlText.length ) );
         var after  = document.createTextNode( orig.slice( idx + hlText.length ) );
         parent.insertBefore( before, spanEl );
         parent.insertBefore( before, spanEl ); parent.insertBefore( mark, spanEl ); parent.insertBefore( after, spanEl ); parent.removeChild( spanEl );
        parent.insertBefore( mark,   spanEl );
         setTimeout( function () { mark.scrollIntoView( { behavior: 'smooth', block: 'center' } ); }, 100 );
        parent.insertBefore( after, spanEl );
        parent.removeChild( spanEl );
 
         setTimeout( function () {
          mark.scrollIntoView( { behavior: 'smooth', block: 'center' } );
        }, 100 );
         found = true;
         found = true;
       }
       }
       if ( !found ) {
       if ( !found ) {
         var walker = document.createTreeWalker( content, NodeFilter.SHOW_TEXT );
         var walker = document.createTreeWalker( content, NodeFilter.SHOW_TEXT );
Line 878: Line 534:
           var txt  = node.textContent || '';
           var txt  = node.textContent || '';
           if ( txt.indexOf( snippet ) === -1 ) continue;
           if ( txt.indexOf( snippet ) === -1 ) continue;
           var idx2   = txt.indexOf( snippet );
           var idx2 = txt.indexOf( snippet );
           var mark2 = document.createElement( 'mark' );
           var mark2 = document.createElement( 'mark' );
           mark2.className   = 'gr-ullekha-highlight';
           mark2.className = 'gr-ullekha-highlight';
           mark2.style.cssText = 'background:#fff176;border-radius:2px;padding:0 2px;';
           mark2.style.cssText = 'background:#fff176;border-radius:2px;padding:0 2px;';
           mark2.textContent = txt.slice( idx2, Math.min( idx2 + needle.length, txt.length ) );
           mark2.textContent = txt.slice( idx2, Math.min( idx2 + needle.length, txt.length ) );
           var p = node.parentNode;
           var p = node.parentNode;
           p.insertBefore( document.createTextNode( txt.slice( 0, idx2 ) ), node );
           p.insertBefore( document.createTextNode( txt.slice( 0, idx2 ) ), node );
Line 888: Line 544:
           p.insertBefore( document.createTextNode( txt.slice( idx2 + mark2.textContent.length ) ), node );
           p.insertBefore( document.createTextNode( txt.slice( idx2 + mark2.textContent.length ) ), node );
           p.removeChild( node );
           p.removeChild( node );
           setTimeout( function () {
           setTimeout( function () { mark2.scrollIntoView( { behavior: 'smooth', block: 'center' } ); }, 100 );
            mark2.scrollIntoView( { behavior: 'smooth', block: 'center' } );
          }, 100 );
           found = true;
           found = true;
         }
         }
       }
       }
     }
     }
     doHighlight();
     doHighlight();
   }
   }
   function wireUllekhaLinks() {
   function wireUllekhaLinks() {
     document.querySelectorAll( '.gr-ullekha-ref-link' ).forEach( function ( wrap ) {
     document.querySelectorAll( '.gr-ullekha-ref-link' ).forEach( function ( wrap ) {
Line 905: Line 557:
       var a = wrap.querySelector( 'a' );
       var a = wrap.querySelector( 'a' );
       if ( !a ) return;
       if ( !a ) return;
       var base = a.href.split( '#' )[ 0 ];
       var base = a.href.split( '#' )[ 0 ];
       var encoded = encodeURIComponent( hl );
       var encoded = encodeURIComponent( hl );
       a.href = base + ( hl ? '?hlUllekha=' + encoded : '' )
       a.href = base + ( hl ? '?hlUllekha=' + encoded : '' ) + ( anchor ? '#' + anchor : '' );
                    + ( anchor ? '#' + anchor : '' );
     } );
     } );
   }
   }
  if ( document.readyState === 'loading' ) { document.addEventListener( 'DOMContentLoaded', function () { highlightOnArrival(); wireUllekhaLinks(); } ); }
  else { highlightOnArrival(); wireUllekhaLinks(); }
}() );


  if ( document.readyState === 'loading' ) {
    document.addEventListener( 'DOMContentLoaded', function () {
      highlightOnArrival();
      wireUllekhaLinks();
    } );
  } else {
    highlightOnArrival();
    wireUllekhaLinks();
  }
}() );
/* ── Search result highlight ──────────────────────────────────── */
/* ── Search result highlight ──────────────────────────────────── */
/* Replace the ENTIRE search highlight IIFE in MediaWiki:Common.js
/* Replace the ENTIRE search highlight IIFE in MediaWiki:Common.js
  * (from the line "( function () {" just before "function storeQueryForLink"
  * (from "( function () {" before "function storeQueryForLink"
  *  to its closing "}() );" )
  *  to its closing "}() );" ) with this block.
* with this block.
  *
  *
  * FIXES vs previous version:
  * Changes vs previous version:
  * 1. Path comparison now also stores wgPageName (title) as fallback,
  * - storeQueryForLink: also stores wgPageName as title fallback
  *   avoiding any pathname encoding mismatch.
  * - applyHighlight: path check uses both pathname AND title fallback
  * 2. sessionStorage key is NOT removed until after highlight succeeds
  * - sessionStorage key cleared AFTER highlight runs, not before
  *   (or max retries exhausted) — prevents race with wikipage.content hook.
  * - wrapMatches: unchanged from original working version
  * 3. Retry logic waits for .mw-parser-output to actually exist in DOM.
  * - showDismissBar mobile: full-width terracotta bar (not pill)
* 4. Mobile bar is full-width terracotta (matches annotation bar style).
  */
  */
( function () {
( function () {


  /* Called by readerToolbar.js before navigating to a result page */
   function storeQueryForLink( url, query ) {
   function storeQueryForLink( url, query ) {
     try {
     try {
Line 945: Line 587:
       sessionStorage.setItem( 'gr_search_hl', JSON.stringify({
       sessionStorage.setItem( 'gr_search_hl', JSON.stringify({
         query:    query,
         query:    query,
         pathname: a.pathname,
         pathname: a.pathname
        /* Also store the decoded title as extra fallback */
        title:    decodeURIComponent( a.pathname.replace( /^\//, '' ) ).replace( /_/g, ' ' )
       }) );
       }) );
     } catch(e) {}
     } catch(e) {}
Line 959: Line 599:
     if ( !stored || !stored.query ) return;
     if ( !stored || !stored.query ) return;


    /* ── Path / title check ────────────────────────────────────────
     var currentPath = window.location.pathname;
    * Match if:
     var storedPath = stored.pathname || '';
    *  (a) pathnames match after normalisation, OR
    *  (b) stored title matches current wgPageName (most reliable)
    * ──────────────────────────────────────────────────────────── */
     var currentPath = window.location.pathname;
     var storedPath   = stored.pathname || '';


     function normPath(p) {
     function normPath(p) {
       return decodeURIComponent(p).replace(/\/+$/, '').replace(/_/g, ' ').toLowerCase();
       return decodeURIComponent(p).replace(/\/+$/, '').replace(/_/g, ' ');
     }
     }


    /* Primary: pathname match. Fallback: stored path's last segment matches
    * wgPageName (handles /index.php?title= vs short-URL discrepancy) */
     var pathMatch  = storedPath && normPath(storedPath) === normPath(currentPath);
     var pathMatch  = storedPath && normPath(storedPath) === normPath(currentPath);
 
     var pageTitle  = (window.mw && mw.config && mw.config.get('wgPageName') || '').replace(/_/g,' ');
    /* wgPageName fallback — most reliable on all URL schemes */
     var storedPage = normPath(storedPath).replace(/^\//, '');
     var currentTitle = ( window.mw && mw.config && mw.config.get('wgPageName') || '' )
     var titleMatch = storedPage && pageTitle && pageTitle.toLowerCase() === storedPage.toLowerCase();
                        .replace(/_/g, ' ');
     var storedTitle  = stored.title || normPath(storedPath).replace(/^\//, '');
     var titleMatch   = storedTitle && currentTitle &&
                      currentTitle.toLowerCase() === storedTitle.toLowerCase();


     if ( storedPath && !pathMatch && !titleMatch ) {
     if ( storedPath && !pathMatch && !titleMatch ) {
      /* Not the right page — clear and bail */
       try { sessionStorage.removeItem( 'gr_search_hl' ); } catch(e) {}
       try { sessionStorage.removeItem( 'gr_search_hl' ); } catch(e) {}
       return;
       return;
Line 989: Line 621:
     if ( !query ) return;
     if ( !query ) return;


     /* ── Retry until .mw-parser-output is in the DOM ──────────────
     /* Remove key now — committed to running on this page */
    * Don't clear sessionStorage until highlight actually ran —
    try { sessionStorage.removeItem( 'gr_search_hl' ); } catch(e) {}
    * the wikipage.content hook may fire before DOMContentLoaded
 
    * on some MW setups.
    /* Retry until .mw-parser-output is present (Minerva lazy-loads sections) */
    * ──────────────────────────────────────────────────────────── */
     var attempts = 0;
     var attempts = 0;
     var MAX_ATTEMPTS = 8;
     var delays = [ 0, 200, 500, 1000, 2000 ];
 
     delays.forEach( function(ms) {
     function tryHighlight() {
       setTimeout( function() {
       attempts++;
        if ( document.querySelector( '.gr-search-hl' ) ) return; /* already done */
      var content = document.querySelector( '#mw-content-text .mw-parser-output' );
        highlightText( query );
      if ( !content && attempts < MAX_ATTEMPTS ) {
       }, ms );
        setTimeout( tryHighlight, attempts * 200 );
     } );
        return;
      }
 
      /* Clear key now — we're committed to running */
      try { sessionStorage.removeItem( 'gr_search_hl' ); } catch(e) {}
 
       if ( !content ) return;
      highlightText( query );
     }
 
    /* First attempt immediately, then back off */
    tryHighlight();
   }
   }


Line 1,023: Line 642:
     if ( !raw ) return;
     if ( !raw ) return;


     /* Build patterns: full phrase first, then individual words */
     var patterns = [];
     var patterns = [ escapeRegex( raw ) ];
     patterns.push( escapeRegex( raw ) );
     raw.split( /\s+/ ).forEach( function(w) {
     raw.split( /\s+/ ).forEach( function(w) {
       if ( w.length >= 2 ) patterns.push( escapeRegex( w ) );
       if ( w.length >= 2 ) patterns.push( escapeRegex( w ) );
Line 1,034: Line 653:
       try { re = new RegExp( '(' + patterns[pi] + ')', 'gi' ); }
       try { re = new RegExp( '(' + patterns[pi] + ')', 'gi' ); }
       catch(e) { continue; }
       catch(e) { continue; }
       var count = wrapMatches( content, re );
       var count = wrapMatches( content, re );
       if ( count > 0 ) { matched = true; break; }
       if ( count > 0 ) { matched = true; break; }
Line 1,042: Line 662:
     var first = document.querySelector( '.gr-search-hl' );
     var first = document.querySelector( '.gr-search-hl' );
     if ( first ) {
     if ( first ) {
       /* Small delay so page layout is stable before scrolling */
       first.scrollIntoView({ behavior: 'smooth', block: 'center' });
      first.classList.add( 'gr-search-hl-pulse' );
       setTimeout( function() {
       setTimeout( function() {
         first.scrollIntoView({ behavior: 'smooth', block: 'center' });
         first.classList.remove( 'gr-search-hl-pulse' );
        first.classList.add( 'gr-search-hl-pulse' );
      }, 2000 );
        setTimeout( function() {
          first.classList.remove( 'gr-search-hl-pulse' );
        }, 2000 );
      }, 150 );
     }
     }


Line 1,059: Line 676:
   }
   }


  /* Original working wrapMatches — unchanged */
   function wrapMatches( root, re ) {
   function wrapMatches( root, re ) {
     var count = 0;
     var count = 0;
Line 1,069: Line 687:
           if ( tag === 'SCRIPT' || tag === 'STYLE' || tag === 'NOSCRIPT' ) return NodeFilter.FILTER_REJECT;
           if ( tag === 'SCRIPT' || tag === 'STYLE' || tag === 'NOSCRIPT' ) return NodeFilter.FILTER_REJECT;
           if ( p.classList.contains( 'gr-search-hl' ) ) return NodeFilter.FILTER_REJECT;
           if ( p.classList.contains( 'gr-search-hl' ) ) return NodeFilter.FILTER_REJECT;
          /* Skip nodes inside the static bar / UI chrome */
          if ( p.closest && p.closest( '#gr-static-bar, .gr-controls, #gra-fab, #gra-mobile-bar' ) ) return NodeFilter.FILTER_REJECT;
           return NodeFilter.FILTER_ACCEPT;
           return NodeFilter.FILTER_ACCEPT;
         }
         }
Line 1,102: Line 718:
         frag.appendChild( document.createTextNode( val.slice( last ) ) );
         frag.appendChild( document.createTextNode( val.slice( last ) ) );
       }
       }
       if ( textNode.parentNode ) {
       textNode.parentNode.replaceChild( frag, textNode );
        textNode.parentNode.replaceChild( frag, textNode );
      }
     } );
     } );


Line 1,111: Line 725:


   function showDismissBar( query ) {
   function showDismissBar( query ) {
    /* Remove any existing bar */
     var existing = document.getElementById( 'gr-hl-bar' );
     var existing = document.getElementById( 'gr-hl-bar' );
     if ( existing ) existing.remove();
     if ( existing ) existing.remove();


     var isMob = window.innerWidth < 768 ||
     var isMob = window.innerWidth < 768 || !!document.getElementById( 'mw-mf-viewport' );
                !!document.getElementById( 'mw-mf-viewport' );
 
     var bar = document.createElement( 'div' );
     var bar = document.createElement( 'div' );
     bar.id = 'gr-hl-bar';
     bar.id = 'gr-hl-bar';


     var count = document.querySelectorAll( '.gr-search-hl' ).length;
     var count = document.querySelectorAll( '.gr-search-hl' ).length;
    if ( !count ) return; /* Nothing to navigate */


     if ( isMob ) {
     if ( isMob ) {
      /* ── Mobile: full-width terracotta bottom bar ── */
       bar.style.cssText = [
       bar.style.cssText = [
         'position:fixed', 'bottom:0', 'left:0', 'right:0',
         'position:fixed', 'bottom:0', 'left:0', 'right:0',
Line 1,137: Line 746:
       ].join(';');
       ].join(';');


       var mobBtnStyle = 'flex:1;display:flex;flex-direction:column;align-items:center;' +
       var mbs = 'flex:1;display:flex;flex-direction:column;align-items:center;' +
         'justify-content:center;background:none;border:none;color:#fff;cursor:pointer;' +
         'justify-content:center;background:none;border:none;color:#fff;cursor:pointer;' +
         'padding:6px 4px;font-family:system-ui,sans-serif;font-size:11px;font-weight:500;' +
         'padding:6px 4px;font-family:system-ui,sans-serif;font-size:11px;font-weight:500;' +
Line 1,143: Line 752:


       bar.innerHTML =
       bar.innerHTML =
         '<button id="gr-hl-prev" style="' + mobBtnStyle + '">' +
         '<button id="gr-hl-prev" style="' + mbs + '">' +
           '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><polyline points="15 18 9 12 15 6"/></svg>' +
           '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><polyline points="15 18 9 12 15 6"/></svg>' +
           '<span>Prev</span>' +
           '<span>Prev</span></button>' +
        '</button>' +
         '<button id="gr-hl-next" style="' + mbs + '">' +
         '<button id="gr-hl-next" style="' + mobBtnStyle + '">' +
           '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><polyline points="9 18 15 12 9 6"/></svg>' +
           '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><polyline points="9 18 15 12 9 6"/></svg>' +
           '<span>Next</span>' +
           '<span>Next</span></button>' +
        '</button>' +
         '<button id="gr-hl-results" style="' + mbs + 'flex:1.6;border-left:1px solid rgba(255,255,255,0.2);border-right:1px solid rgba(255,255,255,0.2);">' +
         '<button id="gr-hl-results" style="' + mobBtnStyle + 'flex:1.6;border-left:1px solid rgba(255,255,255,0.2);border-right:1px solid rgba(255,255,255,0.2);">' +
           '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><circle cx="11" cy="11" r="7"/><line x1="16.5" y1="16.5" x2="22" y2="22"/></svg>' +
           '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><circle cx="11" cy="11" r="7"/><line x1="16.5" y1="16.5" x2="22" y2="22"/></svg>' +
           '<span>' + count + ' match' + ( count === 1 ? '' : 'es' ) + '</span>' +
           '<span>' + count + ' match' + ( count === 1 ? '' : 'es' ) + '</span></button>' +
        '</button>' +
         '<button id="gr-hl-dismiss" style="' + mbs + '">' +
         '<button id="gr-hl-dismiss" style="' + mobBtnStyle + '">' +
           '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg>' +
           '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round" stroke-linejoin="round"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg>' +
           '<span>Close</span>' +
           '<span>Close</span></button>';
        '</button>';
 
     } else {
     } else {
      /* ── Desktop: bottom banner ── */
       bar.style.cssText = [
       bar.style.cssText = [
         'position:fixed', 'bottom:0', 'left:0', 'right:0', 'z-index:10200',
         'position:fixed', 'bottom:0', 'left:0', 'right:0', 'z-index:10200',
Line 1,179: Line 782:
       bar.appendChild( nav );
       bar.appendChild( nav );


       var dismissBtn = document.createElement( 'button' );
       var db = document.createElement( 'button' );
       dismissBtn.id = 'gr-hl-dismiss';
       db.id = 'gr-hl-dismiss'; db.textContent = '✕ Close';
      dismissBtn.textContent = '✕ Close';
       db.style.cssText = 'background:rgba(255,255,255,0.15);border:none;color:#fff;padding:4px 12px;border-radius:4px;cursor:pointer;font-size:13px;min-height:32px;';
       dismissBtn.style.cssText = 'background:rgba(255,255,255,0.15);border:none;color:#fff;padding:4px 12px;border-radius:4px;cursor:pointer;font-size:13px;min-height:32px;';
       bar.appendChild( db );
       bar.appendChild( dismissBtn );
     }
     }


     document.body.appendChild( bar );
     document.body.appendChild( bar );


    /* ── Navigation ── */
     var hlEls = Array.from( document.querySelectorAll( '.gr-search-hl' ) );
     var hlEls   = Array.from( document.querySelectorAll( '.gr-search-hl' ) );
     var current = 0;
     var current = 0;


Line 1,195: Line 796:
       hlEls.forEach( function(el) { el.classList.remove( 'gr-search-hl-current' ); } );
       hlEls.forEach( function(el) { el.classList.remove( 'gr-search-hl-current' ); } );
       current = ( ( idx % hlEls.length ) + hlEls.length ) % hlEls.length;
       current = ( ( idx % hlEls.length ) + hlEls.length ) % hlEls.length;
       var el = hlEls[ current ];
       hlEls[ current ].classList.add( 'gr-search-hl-current' );
      el.classList.add( 'gr-search-hl-current' );
       hlEls[ current ].scrollIntoView({ behavior: 'smooth', block: 'center' });
       el.scrollIntoView({ behavior: 'smooth', block: 'center' });
    }
 
    function dismiss() {
      clearHighlights();
      bar.remove();
     }
     }


     var nextBtn    = document.getElementById( 'gr-hl-next' );
     function dismiss() { clearHighlights(); bar.remove(); }
    var prevBtn    = document.getElementById( 'gr-hl-prev' );
    var resultsBtn = document.getElementById( 'gr-hl-results' );
    var dismissEl  = document.getElementById( 'gr-hl-dismiss' );


     if ( nextBtn )   nextBtn.onclick   = function() { goTo( current + 1 ); };
    var nb = document.getElementById( 'gr-hl-next' );
     if ( prevBtn )   prevBtn.onclick   = function() { goTo( current - 1 ); };
    var pb = document.getElementById( 'gr-hl-prev' );
     if ( dismissEl ) dismissEl.onclick = dismiss;
    var rb = document.getElementById( 'gr-hl-results' );
     if ( resultsBtn ) resultsBtn.onclick = function() {
    var db2 = document.getElementById( 'gr-hl-dismiss' );
       bar.remove();
     if ( nb ) nb.onclick = function() { goTo( current + 1 ); };
      clearHighlights();
     if ( pb ) pb.onclick = function() { goTo( current - 1 ); };
     if ( db2 ) db2.onclick = dismiss;
     if ( rb ) rb.onclick = function() {
       bar.remove(); clearHighlights();
       if ( window.showSearchDialog ) window.showSearchDialog( query );
       if ( window.showSearchDialog ) window.showSearchDialog( query );
     };
     };
Line 1,222: Line 817:
   function clearHighlights() {
   function clearHighlights() {
     document.querySelectorAll( '.gr-search-hl' ).forEach( function( span ) {
     document.querySelectorAll( '.gr-search-hl' ).forEach( function( span ) {
       var parent = span.parentNode;
       var p = span.parentNode; if ( !p ) return;
      if ( !parent ) return;
       while ( span.firstChild ) p.insertBefore( span.firstChild, span );
       while ( span.firstChild ) parent.insertBefore( span.firstChild, span );
       p.removeChild( span );
       parent.removeChild( span );
     } );
     } );
   }
   }
Line 1,238: Line 832:
     s.id = 'gr-hl-css';
     s.id = 'gr-hl-css';
     s.textContent = [
     s.textContent = [
       '.gr-search-hl{background:#fff176;color:#1a1a1a;border-radius:2px;',
       '.gr-search-hl{',
       'padding:0 1px;box-shadow:0 0 0 1px rgba(181,69,27,0.25);}',
      '  background:#fff176;color:#1a1a1a;',
       '.gr-search-hl-current{background:#ffb300!important;',
      '  border-radius:2px;padding:0 1px;',
       'box-shadow:0 0 0 2px #b5451b!important;}',
       ' box-shadow:0 0 0 1px rgba(181,69,27,0.25);',
      '}',
       '.gr-search-hl-current{',
      '  background:#ffb300!important;',
       ' box-shadow:0 0 0 2px #b5451b!important;',
      '}',
       '@keyframes gr-hl-pulse{',
       '@keyframes gr-hl-pulse{',
       '0%{background:#ffb300;}50%{background:#fff176;}100%{background:#fff176;}}',
       ' 0%{background:#ffb300;}',
      '  50%{background:#fff176;}',
      '  100%{background:#fff176;}',
      '}',
       '.gr-search-hl-pulse{animation:gr-hl-pulse 1.2s ease 2;}',
       '.gr-search-hl-pulse{animation:gr-hl-pulse 1.2s ease 2;}',
     ].join('');
     ].join('');
Line 1,250: Line 852:


   injectHighlightCSS();
   injectHighlightCSS();
  /* Run at DOMContentLoaded AND on wikipage.content hook (SPA navigation) */
  function run() {
    /* Small delay so mw.config is populated and DOM is ready */
    setTimeout( applyHighlight, 50 );
  }


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


  /* Hook for MW SPA navigation */
   if ( window.mw ) {
   if ( window.mw ) {
     mw.hook( 'wikipage.content' ).add( function() {
     mw.hook( 'wikipage.content' ).add( function() {
Line 1,273: Line 868:


}() );
}() );
/* ═══════════════════════════════════════════════════════════════
/* ═══════════════════════════════════════════════════════════════
   Mobile addon — paste at bottom of MediaWiki:Common.js
   Mobile addon — only runs on Minerva (mobile) skin
  Only runs on Minerva (mobile) skin
   ═══════════════════════════════════════════════════════════════ */
   ═══════════════════════════════════════════════════════════════ */
( function () {
( function () {
   if ( !document.body.classList.contains( 'skin-minerva' ) ) return;
   if ( !document.body.classList.contains( 'skin-minerva' ) ) return;


  /* ── 1. CSS injected into <head> ── */
   function injectCSS() {
   function injectCSS() {
     if ( document.getElementById( 'gr-mob-css' ) ) return;
     if ( document.getElementById( 'gr-mob-css' ) ) return;
Line 1,291: Line 885:
       '.minerva-header{background:#b5451b!important;min-height:54px!important;}' +
       '.minerva-header{background:#b5451b!important;min-height:54px!important;}' +
       '.minerva-header .search-toggle,.minerva-header .minerva-user-notifications{display:none!important;}' +
       '.minerva-header .search-toggle,.minerva-header .minerva-user-notifications{display:none!important;}' +
       '.branding-box a{display:flex!important;align-items:center!important;' +
       '.branding-box a{display:flex!important;align-items:center!important;text-decoration:none!important;max-width:calc(100vw - 80px)!important;}' +
        'text-decoration:none!important;max-width:calc(100vw - 80px)!important;}' +
       '.branding-box a::before{content:"";display:block;width:30px;height:30px;flex-shrink:0;background:url("/favicon.png") center/contain no-repeat;margin-right:8px;}' +
       '.branding-box a::before{content:"";display:block;width:30px;height:30px;flex-shrink:0;' +
       '.branding-box a span{color:#fff!important;font-size:16px!important;font-weight:700!important;font-family:system-ui,sans-serif!important;line-height:1.2!important;flex:1 1 auto!important;min-width:0!important;}' +
        'background:url("/favicon.png") center/contain no-repeat;' +
        'margin-right:8px;}' +
       '.branding-box a span{color:#fff!important;font-size:16px!important;font-weight:700!important;' +
        'font-family:system-ui,sans-serif!important;line-height:1.2!important;' +
        'flex:1 1 auto!important;min-width:0!important;}' +
       '.minerva-header svg path,.minerva-header svg rect,.minerva-header svg circle{fill:#fff!important;}' +
       '.minerva-header svg path,.minerva-header svg rect,.minerva-header svg circle{fill:#fff!important;}' +
       '.minerva-header label{color:#fff!important;}' +
       '.minerva-header label{color:#fff!important;}' +
       '.minerva-tabs,.mw-portlet-associated-pages,.page-actions-menu,' +
       '.minerva-tabs,.mw-portlet-associated-pages,.page-actions-menu,#page-secondary-actions,.last-modified-bar,.minerva-anon-talk-link{display:none!important;}' +
        '#page-secondary-actions,.last-modified-bar,.minerva-anon-talk-link{display:none!important;}' +
       '#gr-mob-menu-items{display:block!important;}' +
       '#gr-mob-menu-items{display:block!important;}' +
       '.mw-footer.minerva-footer,.footer-places,.footer-info,' +
       '.mw-footer.minerva-footer,.footer-places,.footer-info,.minerva-footer-logo,#footer-places-about,#footer-places-disclaimers,#footer-places-privacy{display:none!important;}' +
        '.minerva-footer-logo,#footer-places-about,' +
        '#footer-places-disclaimers,#footer-places-privacy{display:none!important;}' +
       '#gr-static-bar{position:sticky!important;top:54px!important;z-index:200!important;}' +
       '#gr-static-bar{position:sticky!important;top:54px!important;z-index:200!important;}' +
       '.mf-section-0,.mf-section-1,.mf-section-2,.mf-section-3,.mf-section-4,' +
       '.mf-section-0,.mf-section-1,.mf-section-2,.mf-section-3,.mf-section-4,.mf-section-5,.mf-section-6,.mf-section-7,.mf-section-8,.mf-section-9,.mf-section-10{display:block!important;visibility:visible!important;}' +
        '.mf-section-5,.mf-section-6,.mf-section-7,.mf-section-8,.mf-section-9,' +
        '.mf-section-10{display:block!important;visibility:visible!important;}' +
       '.collapsible-block{display:block!important;}' +
       '.collapsible-block{display:block!important;}' +
       '.section-heading .indicator,.collapsible-heading .indicator{display:none!important;}' +
       '.section-heading .indicator,.collapsible-heading .indicator{display:none!important;}' +
       '.section-heading,.collapsible-heading{pointer-events:none!important;}' +
       '.section-heading,.collapsible-heading{pointer-events:none!important;}' +
       '.gr-home-grid{flex-direction:column!important;flex-wrap:nowrap!important;' +
       '.gr-home-grid{flex-direction:column!important;flex-wrap:nowrap!important;gap:12px!important;width:100%!important;}' +
        'gap:12px!important;width:100%!important;}' +
       '.gr-home-card{width:100%!important;max-width:100%!important;min-width:unset!important;box-sizing:border-box!important;flex:none!important;}' +
       '.gr-home-card{width:100%!important;max-width:100%!important;' +
        'min-width:unset!important;box-sizing:border-box!important;flex:none!important;}' +
       '.gr-home-toggle{flex-wrap:wrap!important;}' +
       '.gr-home-toggle{flex-wrap:wrap!important;}' +
       '.mw-parser-output{font-size:18px!important;line-height:1.8!important;}' +
       '.mw-parser-output{font-size:18px!important;line-height:1.8!important;}' +
Line 1,323: Line 905:
       '.bhashyam-block{margin-left:8px!important;}' +
       '.bhashyam-block{margin-left:8px!important;}' +
       '#footer,.mw-footer,.catlinks,#catlinks{display:none!important;}' +
       '#footer,.mw-footer,.catlinks,#catlinks{display:none!important;}' +
       '.gr-mob-toc-panel{position:fixed!important;top:0!important;left:0!important;' +
       '.gr-mob-toc-panel{position:fixed!important;top:0!important;left:0!important;bottom:0!important;width:82vw!important;max-width:340px!important;background:#fff!important;z-index:10400!important;box-shadow:4px 0 28px rgba(0,0,0,0.22)!important;overflow-y:auto!important;padding:0 0 40px!important;transform:translateX(-110%)!important;transition:transform 0.26s cubic-bezier(0.4,0,0.2,1)!important;display:block!important;}' +
        'bottom:0!important;width:82vw!important;max-width:340px!important;' +
        'background:#fff!important;z-index:10400!important;' +
        'box-shadow:4px 0 28px rgba(0,0,0,0.22)!important;overflow-y:auto!important;' +
        'padding:0 0 40px!important;transform:translateX(-110%)!important;' +
        'transition:transform 0.26s cubic-bezier(0.4,0,0.2,1)!important;display:block!important;}' +
       '.gr-mob-toc-panel.open{transform:translateX(0)!important;}' +
       '.gr-mob-toc-panel.open{transform:translateX(0)!important;}' +
       '.gr-mob-toc-backdrop{display:none!important;position:fixed!important;inset:0!important;' +
       '.gr-mob-toc-backdrop{display:none!important;position:fixed!important;inset:0!important;background:rgba(0,0,0,0.4)!important;z-index:10399!important;}' +
        'background:rgba(0,0,0,0.4)!important;z-index:10399!important;}' +
       '.gr-mob-toc-backdrop.open{display:block!important;}' +
       '.gr-mob-toc-backdrop.open{display:block!important;}' +
       '.gr-mob-toc-header{position:sticky!important;top:0!important;background:#fff!important;' +
       '.gr-mob-toc-header{position:sticky!important;top:0!important;background:#fff!important;display:flex!important;align-items:center!important;justify-content:space-between!important;padding:16px 16px 12px!important;border-bottom:1px solid #f0ebe6!important;z-index:1!important;}' +
        'display:flex!important;align-items:center!important;justify-content:space-between!important;' +
       '.gr-mob-toc-title{font-size:13px!important;font-weight:700!important;text-transform:uppercase!important;letter-spacing:0.08em!important;color:#b5451b!important;font-family:system-ui,sans-serif!important;}' +
        'padding:16px 16px 12px!important;border-bottom:1px solid #f0ebe6!important;z-index:1!important;}' +
       '.gr-mob-toc-close{background:none!important;border:none!important;font-size:22px!important;color:#999!important;cursor:pointer!important;padding:4px 8px!important;}' +
       '.gr-mob-toc-title{font-size:13px!important;font-weight:700!important;text-transform:uppercase!important;' +
        'letter-spacing:0.08em!important;color:#b5451b!important;font-family:system-ui,sans-serif!important;}' +
       '.gr-mob-toc-close{background:none!important;border:none!important;' +
        'font-size:22px!important;color:#999!important;cursor:pointer!important;padding:4px 8px!important;}' +
       '.gr-mob-toc-body{padding:12px 16px!important;}' +
       '.gr-mob-toc-body{padding:12px 16px!important;}' +
       '.gr-mob-toc-body a{display:block!important;font-size:16px!important;line-height:1.6!important;' +
       '.gr-mob-toc-body a{display:block!important;font-size:16px!important;line-height:1.6!important;color:#2c1810!important;text-decoration:none!important;padding:8px 0!important;border-bottom:1px solid #f5f0ed!important;}';
        'color:#2c1810!important;text-decoration:none!important;padding:8px 0!important;' +
        'border-bottom:1px solid #f5f0ed!important;}';
     document.head.appendChild( s );
     document.head.appendChild( s );
   }
   }


  /* ── 2. Expand hidden sections ── */
   function expandSections() {
   function expandSections() {
     document.querySelectorAll( '[class*="mf-section-"], .collapsible-block' )
     document.querySelectorAll( '[class*="mf-section-"], .collapsible-block' ).forEach( function ( el ) {
      .forEach( function ( el ) {
      el.removeAttribute( 'hidden' );
        el.removeAttribute( 'hidden' );
      el.style.setProperty( 'display', 'block', 'important' );
        el.style.setProperty( 'display', 'block', 'important' );
      el.style.setProperty( 'visibility', 'visible', 'important' );
        el.style.setProperty( 'visibility', 'visible', 'important' );
      el.removeAttribute( 'aria-hidden' );
        el.removeAttribute( 'aria-hidden' );
    } );
      } );
     document.querySelectorAll( '.section-heading, .collapsible-heading' ).forEach( function ( el ) {
     document.querySelectorAll( '.section-heading, .collapsible-heading' )
      el.setAttribute( 'aria-expanded', 'true' );
      .forEach( function ( el ) {
      el.style.setProperty( 'pointer-events', 'none', 'important' );
        el.setAttribute( 'aria-expanded', 'true' );
    } );
        el.style.setProperty( 'pointer-events', 'none', 'important' );
      } );
   }
   }


Line 1,366: Line 933:
     var t = null;
     var t = null;
     var obs = new MutationObserver( function ( ms ) {
     var obs = new MutationObserver( function ( ms ) {
       if ( ms.some( function(m){ return m.attributeName === 'hidden'; } ) ) {
       if ( ms.some( function(m){ return m.attributeName === 'hidden'; } ) ) { clearTimeout(t); t = setTimeout( expandSections, 30 ); }
        clearTimeout(t); t = setTimeout( expandSections, 30 );
      }
     } );
     } );
     var root = document.querySelector( '#mw-content-text' ) || document.body;
     var root = document.querySelector( '#mw-content-text' ) || document.body;
Line 1,374: Line 939:
   }
   }


  /* ── 3. Body padding observer ── */
   function watchBodyPadding() {
   function watchBodyPadding() {
     new MutationObserver( function () {
     new MutationObserver( function () {
Line 1,383: Line 947:
   }
   }


  /* ── 4. Inject custom links into drawer ── */
   function injectMenuLinks() {
   function injectMenuLinks() {
     if ( document.getElementById( 'gr-mob-menu-items' ) ) return;
     if ( document.getElementById( 'gr-mob-menu-items' ) ) return;
     var navDrawer = document.querySelector( '.navigation-drawer' );
     var navDrawer = document.querySelector( '.navigation-drawer' );
     if ( !navDrawer ) return;
     if ( !navDrawer ) return;
     var wrap = document.createElement( 'div' );
     var wrap = document.createElement( 'div' );
     wrap.id = 'gr-mob-menu-items';
     wrap.id = 'gr-mob-menu-items';
     wrap.style.cssText = 'width:100%;background:#fff;margin-top:8px;';
     wrap.style.cssText = 'width:100%;background:#fff;margin-top:8px;';
 
     var itemStyle = 'display:flex;align-items:center;gap:14px;padding:15px 20px;font-size:16px;color:#2c1810;text-decoration:none;font-family:system-ui,sans-serif;border-bottom:1px solid #f0ebe6;background:#fff;';
     var itemStyle = 'display:flex;align-items:center;gap:14px;padding:15px 20px;' +
      'font-size:16px;color:#2c1810;text-decoration:none;' +
      'font-family:system-ui,sans-serif;border-bottom:1px solid #f0ebe6;background:#fff;';
 
     function makeItem( href, emoji, label ) {
     function makeItem( href, emoji, label ) {
       var a = document.createElement( 'a' );
       var a = document.createElement( 'a' ); a.href = href; a.style.cssText = itemStyle;
      a.href = href;
       a.innerHTML = '<span>' + label + '</span>'; return a;
      a.style.cssText = itemStyle;
       a.innerHTML =
        '<span>' + label + '</span>';
      return a;
     }
     }
     wrap.appendChild( makeItem( '/Main_Page', '', 'Home' ) );
     wrap.appendChild( makeItem( '/Main_Page', '', 'Home' ) );
     wrap.appendChild( makeItem( '/My_wiki:Help', '', 'Help' ) );
     wrap.appendChild( makeItem( '/My_wiki:Help', '', 'Help' ) );
     wrap.appendChild( makeItem( '/My_wiki:About', '', 'About' ) );
     wrap.appendChild( makeItem( '/My_wiki:About', '', 'About' ) );
     var userName = window.mw ? mw.config.get( 'wgUserName' ) : null;
     var userName = window.mw ? mw.config.get( 'wgUserName' ) : null;
     if ( userName ) {
     if ( userName ) {
       var la = document.querySelector( 'a[href*="action=logout"]' );
       var la = document.querySelector( 'a[href*="action=logout"]' );
       wrap.appendChild( makeItem(
       wrap.appendChild( makeItem( la ? la.href : '/index.php?title=Special:UserLogout', '', 'Log out' ) );
        la ? la.href : '/index.php?title=Special:UserLogout', '', 'Log out'
      ) );
     } else {
     } else {
       wrap.appendChild( makeItem( '/index.php?title=Special:UserLogin', '', 'Log in' ) );
       wrap.appendChild( makeItem( '/index.php?title=Special:UserLogin', '', 'Log in' ) );
     }
     }
     var pageLeft = document.getElementById( 'mw-mf-page-left' );
     var pageLeft = document.getElementById( 'mw-mf-page-left' );
     if ( pageLeft ) {
     if ( pageLeft ) { while ( pageLeft.firstChild ) pageLeft.removeChild( pageLeft.firstChild ); pageLeft.style.removeProperty( 'display' ); pageLeft.appendChild( wrap ); }
      while ( pageLeft.firstChild ) pageLeft.removeChild( pageLeft.firstChild );
     else navDrawer.appendChild( wrap );
      pageLeft.style.removeProperty( 'display' );
     document.querySelectorAll( '.mw-footer.minerva-footer,.footer-places,.footer-info,.minerva-footer-logo,#footer-places-about,#footer-places-disclaimers,#footer-places-privacy' ).forEach( function(el) { el.style.setProperty('display','none','important'); } );
      pageLeft.appendChild( wrap );
     } else {
      navDrawer.appendChild( wrap );
    }
 
     document.querySelectorAll(
      '.mw-footer.minerva-footer,.footer-places,.footer-info,' +
      '.minerva-footer-logo,#footer-places-about,' +
      '#footer-places-disclaimers,#footer-places-privacy'
    ).forEach( function(el) { el.style.setProperty('display','none','important'); } );
   }
   }


  /* ── 5. Mobile TOC overlay ── */
   var _tocDone = false;
   var _tocDone = false;
   function initToc() {
   function initToc() {
     if ( _tocDone ) return;
     if ( _tocDone ) return;
     var tocList = document.querySelector( '.vector-toc-contents, .vector-toc .vector-toc-list' );
     var tocList = document.querySelector( '.vector-toc-contents, .vector-toc .vector-toc-list' );
     if ( !tocList ) return;
     if ( !tocList || !tocList.querySelector( 'li' ) ) return;
    if ( !tocList.querySelector( 'li' ) ) return;
     _tocDone = true;
     _tocDone = true;
 
     var bd = document.createElement( 'div' ); bd.className = 'gr-mob-toc-backdrop'; document.body.appendChild( bd );
     var bd = document.createElement( 'div' );
     var panel = document.createElement( 'div' ); panel.className = 'gr-mob-toc-panel';
    bd.className = 'gr-mob-toc-backdrop';
     var hdr = document.createElement( 'div' ); hdr.className = 'gr-mob-toc-header';
    document.body.appendChild( bd );
     var ttl = document.createElement( 'div' ); ttl.className = 'gr-mob-toc-title'; ttl.textContent = 'विषयसूची';
 
     var cls = document.createElement( 'button' ); cls.className = 'gr-mob-toc-close'; cls.textContent = '✕';
     var panel = document.createElement( 'div' );
     hdr.appendChild( ttl ); hdr.appendChild( cls ); panel.appendChild( hdr );
    panel.className = 'gr-mob-toc-panel';
     var body = document.createElement( 'div' ); body.className = 'gr-mob-toc-body';
 
     body.appendChild( tocList.cloneNode( true ) ); panel.appendChild( body ); document.body.appendChild( panel );
     var hdr = document.createElement( 'div' );
     var btn = document.createElement( 'button' ); btn.id = 'gr-mob-toc-btn';
    hdr.className = 'gr-mob-toc-header';
     var ttl = document.createElement( 'div' );
    ttl.className = 'gr-mob-toc-title';
    ttl.textContent = 'विषयसूची';
     var cls = document.createElement( 'button' );
    cls.className = 'gr-mob-toc-close';
    cls.textContent = '✕';
     hdr.appendChild( ttl ); hdr.appendChild( cls );
    panel.appendChild( hdr );
 
     var body = document.createElement( 'div' );
    body.className = 'gr-mob-toc-body';
     body.appendChild( tocList.cloneNode( true ) );
    panel.appendChild( body );
    document.body.appendChild( panel );
 
     var btn = document.createElement( 'button' );
    btn.id = 'gr-mob-toc-btn';
     btn.innerHTML = '☰ &nbsp;Contents';
     btn.innerHTML = '☰ &nbsp;Contents';
     btn.style.cssText = 'position:fixed;bottom:148px;left:16px;z-index:9100;' +
     btn.style.cssText = 'position:fixed;bottom:148px;left:16px;z-index:9100;background:#fff;border:1.5px solid #b5451b;border-radius:24px;padding:10px 16px;font-size:15px;font-family:system-ui,sans-serif;color:#b5451b;font-weight:600;box-shadow:0 3px 14px rgba(0,0,0,0.15);cursor:pointer;';
      'background:#fff;border:1.5px solid #b5451b;border-radius:24px;' +
      'padding:10px 16px;font-size:15px;font-family:system-ui,sans-serif;' +
      'color:#b5451b;font-weight:600;box-shadow:0 3px 14px rgba(0,0,0,0.15);cursor:pointer;';
     document.body.appendChild( btn );
     document.body.appendChild( btn );
     function open()  { panel.classList.add('open'); bd.classList.add('open'); document.body.style.overflow='hidden'; }
     function open()  { panel.classList.add('open'); bd.classList.add('open'); document.body.style.overflow='hidden'; }
     function close() { panel.classList.remove('open'); bd.classList.remove('open'); document.body.style.overflow=''; }
     function close() { panel.classList.remove('open'); bd.classList.remove('open'); document.body.style.overflow=''; }
     btn.onclick = open; cls.onclick = close; bd.onclick = close;
     btn.onclick = open; cls.onclick = close; bd.onclick = close;
     body.querySelectorAll('a').forEach(function(a){ a.onclick = close; });
     body.querySelectorAll('a').forEach(function(a){ a.onclick = close; });
   }
   }


  /* ── 6. Moolam / Ullekha links below page title (mobile only) ── */
   function injectMoolaUllekhaLinks() {
   function injectMoolaUllekhaLinks() {
     if ( document.getElementById( 'gr-mob-doc-nav' ) ) return;
     if ( document.getElementById( 'gr-mob-doc-nav' ) ) return;
     var pageName = ( window.mw && mw.config && mw.config.get( 'wgPageName' ) ) || '';
     var pageName = ( window.mw && mw.config && mw.config.get( 'wgPageName' ) ) || '';
     if ( pageName === 'Main_Page' || !pageName ) return;
     if ( pageName === 'Main_Page' || !pageName ) return;
     function wikiUrl( slug ) {
     function wikiUrl( slug ) {
       if ( window.mw && mw.util && mw.util.getUrl ) return mw.util.getUrl( slug );
       if ( window.mw && mw.util && mw.util.getUrl ) return mw.util.getUrl( slug );
Line 1,497: Line 1,008:
       return ap.replace( '$1', encodeURIComponent( slug ).replace( /%2F/g, '/' ) );
       return ap.replace( '$1', encodeURIComponent( slug ).replace( /%2F/g, '/' ) );
     }
     }
 
     var teekaPage   = document.querySelector( '.gr-teeka-page' );
     var teekaPage     = document.querySelector( '.gr-teeka-page' );
     var primarySlug = teekaPage ? ( teekaPage.getAttribute( 'data-primary' ) || pageName.split('/')[0] ) : pageName.split('/')[0];
     var primarySlug   = teekaPage
     var docTitleEl   = document.querySelector( '.gr-doc-title' );
      ? ( teekaPage.getAttribute( 'data-primary' ) || pageName.split('/')[0] )
      : pageName.split('/')[0];
     var docTitleEl     = document.querySelector( '.gr-doc-title' );
     var hasMoolaPage  = docTitleEl && docTitleEl.getAttribute( 'data-has-moola' )  === '1';
     var hasMoolaPage  = docTitleEl && docTitleEl.getAttribute( 'data-has-moola' )  === '1';
     var hasUllekhaPage = docTitleEl && docTitleEl.getAttribute( 'data-has-ullekha' ) === '1';
     var hasUllekhaPage = docTitleEl && docTitleEl.getAttribute( 'data-has-ullekha' ) === '1';
     var showMoolam  = !!teekaPage || hasMoolaPage;
     var showMoolam  = !!teekaPage || hasMoolaPage;
     var showUllekha = hasUllekhaPage || !!teekaPage;
     var showUllekha = hasUllekhaPage || !!teekaPage;
     if ( !showMoolam && !showUllekha ) return;
     if ( !showMoolam && !showUllekha ) return;
 
     var nav = document.createElement( 'div' ); nav.id = 'gr-mob-doc-nav';
     var nav = document.createElement( 'div' );
     nav.style.cssText = 'display:flex;gap:10px;padding:10px 16px 8px;background:#fdf8f5;border-bottom:1px solid #f0e0d6;font-family:system-ui,sans-serif;';
    nav.id = 'gr-mob-doc-nav';
     nav.style.cssText = 'display:flex;gap:10px;padding:10px 16px 8px;' +
      'background:#fdf8f5;border-bottom:1px solid #f0e0d6;font-family:system-ui,sans-serif;';
 
     function makeLink( href, label ) {
     function makeLink( href, label ) {
       var a = document.createElement( 'a' );
       var a = document.createElement( 'a' ); a.href = href; a.textContent = label;
      a.href = href; a.textContent = label;
       a.style.cssText = 'display:inline-flex;align-items:center;padding:5px 16px;border-radius:20px;background:#fff;border:1.5px solid #e8cfc4;color:#b5451b;font-size:14px;font-weight:600;text-decoration:none;';
       a.style.cssText = 'display:inline-flex;align-items:center;padding:5px 16px;' +
        'border-radius:20px;background:#fff;border:1.5px solid #e8cfc4;' +
        'color:#b5451b;font-size:14px;font-weight:600;text-decoration:none;';
       return a;
       return a;
     }
     }
     if ( teekaPage )        nav.appendChild( makeLink( wikiUrl( primarySlug ),              'मूल' ) );
     if ( teekaPage )        nav.appendChild( makeLink( wikiUrl( primarySlug ),              'मूल' ) );
     else if ( hasMoolaPage ) nav.appendChild( makeLink( wikiUrl( primarySlug + '/Moola' ),  'मूलम्' ) );
     else if ( hasMoolaPage ) nav.appendChild( makeLink( wikiUrl( primarySlug + '/Moola' ),  'मूलम्' ) );
     if ( showUllekha )      nav.appendChild( makeLink( wikiUrl( primarySlug + '/Ullekha' ), 'उल्लेख' ) );
     if ( showUllekha )      nav.appendChild( makeLink( wikiUrl( primarySlug + '/Ullekha' ), 'उल्लेख' ) );
 
     var h1 = document.getElementById( 'firstHeading' ) || document.querySelector( '.page-heading, h1.firstHeading, .mw-first-heading' );
    /* Insert below the page h1 */
     if ( h1 && h1.parentNode ) h1.parentNode.insertBefore( nav, h1.nextSibling );
     var h1 = document.getElementById( 'firstHeading' ) ||
     else { var ct = document.getElementById( 'mw-content-text' ); if ( ct ) ct.insertBefore( nav, ct.firstChild ); }
            document.querySelector( '.page-heading, h1.firstHeading, .mw-first-heading' );
     if ( h1 && h1.parentNode ) {
      h1.parentNode.insertBefore( nav, h1.nextSibling );
     } else {
      var ct = document.getElementById( 'mw-content-text' );
      if ( ct ) ct.insertBefore( nav, ct.firstChild );
    }
   }
   }


  /* ── Boot ── */
   injectCSS();
   injectCSS();
   watchBodyPadding();
   watchBodyPadding();
Line 1,552: Line 1,043:
   }
   }


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


   if ( window.mw ) {
   if ( window.mw ) {
     mw.hook( 'wikipage.content' ).add(function() {
     mw.hook( 'wikipage.content' ).add(function() {
       setTimeout(function(){
       setTimeout(function(){ expandSections(); injectMenuLinks(); injectMoolaUllekhaLinks(); initToc(); }, 300);
        expandSections(); injectMenuLinks(); injectMoolaUllekhaLinks(); initToc();
      }, 300);
     });
     });
   }
   }


}() );
}() );
/**
 
* grantha-mobile-fixes.js
*/


( function () {
( function () {
Line 1,578: Line 1,062:
       var blocks = document.querySelectorAll( '.collapsible-block, .toggle-list' );
       var blocks = document.querySelectorAll( '.collapsible-block, .toggle-list' );
       Array.prototype.forEach.call( blocks, function ( el ) {
       Array.prototype.forEach.call( blocks, function ( el ) {
         if ( !el.parentNode ) {
         if ( !el.parentNode ) { try { el.remove(); } catch(e) {} }
          try { el.remove(); } catch(e) {}
        }
       } );
       } );
     }, 0 );
     }, 0 );
Line 1,599: Line 1,081:
     var bar = document.getElementById( 'gr-static-bar' );
     var bar = document.getElementById( 'gr-static-bar' );
     if ( !bar ) return;
     if ( !bar ) return;
     var barRect = bar.getBoundingClientRect();
     var barRect = bar.getBoundingClientRect();
     var barBottom = Math.round( barRect.bottom );
     var barBottom = Math.round( barRect.bottom );
     var homeEl = document.getElementById( 'gr-home' );
     var homeEl = document.getElementById( 'gr-home' );
     var toggleEl = document.getElementById( 'gr-home-toggle' );
     var toggleEl = document.getElementById( 'gr-home-toggle' );
     var firstChild = document.querySelector(
     var firstChild = document.querySelector( '#mw-content-text .mw-parser-output > .gr-home, #mw-content-text .mw-parser-output > *:first-child' );
      '#mw-content-text .mw-parser-output > .gr-home, ' +
     [ homeEl, toggleEl, firstChild ].forEach( function ( el ) { if ( el ) el.style.scrollMarginTop = ( barBottom + 4 ) + 'px'; } );
      '#mw-content-text .mw-parser-output > *:first-child'
    );
 
     [ homeEl, toggleEl, firstChild ].forEach( function ( el ) {
      if ( el ) el.style.scrollMarginTop = ( barBottom + 4 ) + 'px';
    } );
 
     var contentText = document.getElementById( 'mw-content-text' );
     var contentText = document.getElementById( 'mw-content-text' );
     if ( contentText ) {
     if ( contentText ) {
       var isMob = window.innerWidth < 768 || !!document.getElementById( 'mw-mf-viewport' );
       var isMob = window.innerWidth < 768 || !!document.getElementById( 'mw-mf-viewport' );
       if ( isMob ) {
       if ( isMob && toggleEl ) {
         if ( toggleEl ) {
         var toggleRect = toggleEl.getBoundingClientRect();
          var toggleRect = toggleEl.getBoundingClientRect();
        if ( toggleRect.top < barBottom ) {
          if ( toggleRect.top < barBottom ) {
          var currentPT = parseInt( window.getComputedStyle( document.body ).paddingTop, 10 ) || 0;
            var currentPT = parseInt( window.getComputedStyle( document.body ).paddingTop, 10 ) || 0;
          document.body.style.paddingTop = ( currentPT + ( barBottom - toggleRect.top ) + 4 ) + 'px';
            var needed    = currentPT + ( barBottom - toggleRect.top ) + 4;
            document.body.style.paddingTop = needed + 'px';
          }
         }
         }
       }
       }
Line 1,631: Line 1,101:


   $( function () {
   $( function () {
     var $toggle   = $( '#gr-home-toggle' );
     var $toggle = $( '#gr-home-toggle' ), $viewG = $( '#gr-view-grantha' ), $viewA = $( '#gr-view-author' );
    var $viewG   = $( '#gr-view-grantha' );
     var $btnG = $( '#gr-toggle-grantha' ), $btnA = $( '#gr-toggle-author' );
    var $viewA   = $( '#gr-view-author' );
     var $btnG     = $( '#gr-toggle-grantha' );
    var $btnA     = $( '#gr-toggle-author' );
 
     if ( !$toggle.length || !$viewG.length || !$viewA.length ) return;
     if ( !$toggle.length || !$viewG.length || !$viewA.length ) return;
     if ( $toggle.data( 'gr-wired' ) ) return;
     if ( $toggle.data( 'gr-wired' ) ) return;
     $toggle.data( 'gr-wired', true );
     $toggle.data( 'gr-wired', true );
     function showView( which ) {
     function showView( which ) {
       if ( which === 'grantha' ) {
       if ( which === 'grantha' ) { $viewG.show(); $viewA.hide(); $btnG.addClass( 'gr-toggle-active' ); $btnA.removeClass( 'gr-toggle-active' ); }
        $viewG.show(); $viewA.hide();
       else { $viewA.show(); $viewG.hide(); $btnA.addClass( 'gr-toggle-active' ); $btnG.removeClass( 'gr-toggle-active' ); }
        $btnG.addClass( 'gr-toggle-active' );
        $btnA.removeClass( 'gr-toggle-active' );
       } else {
        $viewA.show(); $viewG.hide();
        $btnA.addClass( 'gr-toggle-active' );
        $btnG.removeClass( 'gr-toggle-active' );
      }
       try { localStorage.setItem( 'grantha_home_tab', which ); } catch(e){}
       try { localStorage.setItem( 'grantha_home_tab', which ); } catch(e){}
     }
     }
 
     $btnG.on( 'click keydown', function(e){ if ( e.type === 'keydown' && e.key !== 'Enter' && e.key !== ' ' ) return; showView( 'grantha' ); });
     $btnG.on( 'click keydown', function(e){
     $btnA.on( 'click keydown', function(e){ if ( e.type === 'keydown' && e.key !== 'Enter' && e.key !== ' ' ) return; showView( 'author' ); });
      if ( e.type === 'keydown' && e.key !== 'Enter' && e.key !== ' ' ) return;
     try { var saved = localStorage.getItem( 'grantha_home_tab' ); if ( saved === 'author' ) showView( 'author' ); else showView( 'grantha' ); } catch(e) { showView( 'grantha' ); }
      showView( 'grantha' );
    });
     $btnA.on( 'click keydown', function(e){
      if ( e.type === 'keydown' && e.key !== 'Enter' && e.key !== ' ' ) return;
      showView( 'author' );
    });
 
     try {
      var saved = localStorage.getItem( 'grantha_home_tab' );
      if ( saved === 'author' ) showView( 'author' );
      else showView( 'grantha' );
    } catch(e) { showView( 'grantha' ); }
   } );
   } );


}() );
}() );