MediaWiki:Gadget-GrAnnotations.js: Difference between revisions

No edit summary
No edit summary
 
(50 intermediate revisions by 2 users not shown)
Line 1: Line 1:
/**
/**
  * gr_annotations.js  —  grantha.io inline Comments + Bookmarks
  * gr_annotations.js  —  grantha.io inline Notes + Bookmarks + Feedback (v6 + Strategy B)
* ══════════════════════════════════════════════════════════════════════
*
  * BEHAVIOUR (mirrors Google Docs)
* ────────────────────────────────
* 1. User selects text anywhere in mw-content-text.
* 2. A floating action strip (FAB) appears to the right of the selection
*    with two icon buttons: Comment and Bookmark.
* 3. Clicking Comment:
*    - Opens an inline composer card anchored to the selection position.
*    - On submit: wraps the selected text in a yellow highlight span,
*      saves the comment to <PageTitle>/Comments via MW edit API,
*      posts a notification to the admin user talk page,
*      and adds a card to the Comments tab of the right panel.
* 4. Clicking Bookmark:
*    - Opens a small composer to name the bookmark.
*    - On submit: wraps selected text in a blue highlight span,
*      saves bookmark in localStorage (per-user, per-page),
*      and adds a card to the Bookmarks tab.
*    - Clicking a bookmark card scrolls to and flashes that highlight.
* 5. Right panel:
*    - Slides in from the right as an overlay (no layout shift).
*    - Tab 1: Comments  (loads from /Comments wiki page)
*    - Tab 2: Bookmarks (loads from localStorage)
*    - Panel can be closed via ✕ or clicking the backdrop.
* 6. Clicking any comment/bookmark highlight in the text:
*    - Opens the panel to the correct tab and scrolls to the card.
*
* STORAGE
* ───────
* Comments  → MediaWiki page: Talk:<PageTitle>/GrComments
*            Talk namespace is excluded from XML/PDF exports and never
*            shown in the content editor — comments cannot affect exports.
*            Format: {{GrComment|id=…|author=…|timestamp=…|quote=…|text=…}}
* Bookmarks → localStorage (per-user, per-page — not shared across users)
*            Key: grantha_bm_<pageName>
*
* ADMIN NOTIFICATION
* ──────────────────
* On new comment: sends email via MW EmailUser API (action=emailuser).
* Email contains the comment text and a direct anchor link to the passage.
* Fallback: if admin has no email set, posts to User_talk:ADMIN_USER instead.
* Requires $wgEnableEmail = true and $wgEnableUserEmail = true (MW defaults).
*
* DEPENDENCIES
* ────────────
* • jQuery (loaded by MediaWiki)
* • mw.Api  (MediaWiki core)
* • gr_annotations.css  (companion stylesheet)
* • /images/commentary.svg, /images/bookmark.svg  (Hostinger)
*
* DEPLOY
* ──────
* Add both files to MediaWiki:Common.css / Common.js, or register as
* a gadget in MediaWiki:Gadgets-definition:
*  GrAnnotations[ResourceLoader|default]|gr_annotations.js|gr_annotations.css
* ══════════════════════════════════════════════════════════════════════
  */
  */


Line 63: Line 7:
   'use strict';
   'use strict';


  // ── Configuration ────────────────────────────────────────────────────
   var CONTENT_SEL   = '#mw-content-text';
   var ADMIN_USER  = 'GranthaGate';   // MW username for talk-page fallback
   var BM_LS_KEY     = 'grantha_bm_'  + ( ( window.mw && mw.config.get( 'wgPageName' ) ) || '' );
  var ADMIN_EMAIL = 'admin@grantha.io'; // ← set your email here directly
   var NT_LS_KEY    = 'grantha_nt_' + ( ( window.mw && mw.config.get( 'wgPageName' ) ) || '' );
  var CONTENT_SEL = '#mw-content-text';
   var pageTitle     = ( window.mw && mw.config.get( 'wgPageName' ) ) || '';
   var BM_LS_KEY   = 'grantha_bm_'  + ( ( window.mw && mw.config.get( 'wgPageName' ) ) || '' );
   var currentUser   = ( window.mw && mw.config.get( 'wgUserName' ) ) || '';
   var CMT_LS_KEY  = 'grantha_cmt_' + ( ( window.mw && mw.config.get( 'wgPageName' ) ) || '' );
   var userInitial   = currentUser ? currentUser.charAt( 0 ).toUpperCase() : '?';
   var pageTitle   = ( window.mw && mw.config.get( 'wgPageName' ) ) || '';
  var currentUserEmail = '';
  /* Comments stored on Talk:<PageTitle>/GrComments — talk namespace is
 
  * excluded from XML/PDF exports and never shown in the content editor,
  if ( currentUser && window.mw ) {
  * so comments cannot tamper with document content or affect exports. */
    new mw.Api().get({ action: 'query', meta: 'userinfo', uiprop: 'email', formatversion: 2 })
   var commentsPage = 'Talk:' + pageTitle + '/GrComments';
      .then( function (data) {
   var currentUser  = ( window.mw && mw.config.get( 'wgUserName' ) ) || '';
        var info = data && data.query && data.query.userinfo;
   var userInitial = currentUser ? currentUser.charAt( 0 ).toUpperCase() : '?';
        if ( info && info.email ) currentUserEmail = info.email;
      } ).catch( function () {} );
  }


  // Only run on content namespaces, not on the /Comments page itself
   if ( window.mw ) {
   if ( window.mw ) {
     var ns = mw.config.get( 'wgNamespaceNumber' );
     var ns = mw.config.get( 'wgNamespaceNumber' );
     if ( ns < 0 ) return;
     if ( ns < 0 ) return;
    if ( /\/Comments$/.test( pageTitle ) ) return;
   }
   }


  // ── State ────────────────────────────────────────────────────────────
   var _selRange  = null;
  var _sel        = null;  // current Selection snapshot
   var _selText    = '';
   var _selRange  = null;   // saved Range for wrapping
   var _selRect    = null;
   var _selText    = '';     // selected text string
   var _notes      = [];
   var _selRect    = null;   // bounding rect of selection
   var _bookmarks  = [];
   var _comments  = [];     // [{id, anchor, author, ts, quote, text}]
   var _activeTab = 'notes';
   var _bookmarks  = [];     // [{id, name, quote, anchorHtml, ts}]
  var _selVersion = 0;
   var _cmtLoaded = false;
   var _fabSelVer = -1;
   var _activeTab = 'comments';  // 'comments' | 'bookmarks'
  var _mobile    = window.innerWidth < 768 || 'ontouchstart' in window;
  var _fabTouched = false;  // flag to prevent hideActions when tapping fab


  // ── Helpers ──────────────────────────────────────────────────────────
   function uid() { return 'gra_' + Date.now() + '_' + Math.random().toString(36).slice(2,7); }
   function uid() {
  function esc(s) {
    return 'gra_' + Date.now() + '_' + Math.random().toString(36).slice(2,7);
    return String(s||'').replace(/&/g,'&amp;').replace(/</g,'&lt;')
                        .replace(/>/g,'&gt;').replace(/"/g,'&quot;');
   }
   }
  function esc( s ) {
   function nowIso() { return new Date().toISOString().replace(/\.\d{3}Z$/,'Z'); }
    return String( s || '' )
   function fmtTs(ts) {
      .replace(/&/g,'&amp;').replace(/</g,'&lt;')
      .replace(/>/g,'&gt;').replace(/"/g,'&quot;');
  }
   function nowIso() {
    return new Date().toISOString().replace(/\.\d{3}Z$/,'Z');
  }
   function fmtTs( ts ) {
     try {
     try {
       var d = new Date( ts );
       var d = new Date(ts);
       return d.toLocaleDateString('en-IN',{day:'numeric',month:'short',year:'numeric'})
       return d.toLocaleDateString('en-IN',{day:'numeric',month:'short',year:'numeric'})
           + ' ' + d.toLocaleTimeString('en-IN',{hour:'2-digit',minute:'2-digit',hour12:false});
           + ' ' + d.toLocaleTimeString('en-IN',{hour:'2-digit',minute:'2-digit',hour12:false});
     } catch(e){ return ts; }
     } catch(e){ return ts; }
   }
   }
   function clamp( val, min, max ) { return Math.max(min, Math.min(max, val)); }
   function clamp(v,lo,hi){ return Math.max(lo,Math.min(hi,v)); }
  function isMobile() { return _mobile; }


  // ── DOM references (populated in boot) ───────────────────────────────
   var $fab, $mobileBar, $panel, $backdrop;
   var $fab, $panel, $backdrop, $panelBody;
   var $ntComposer, $ntInput, $ntSubmit;
   var $cmpComposer, $cmpInput, $cmpSubmit;
   var $bmComposer, $bmInput, $bmSubmit;
   var $bmComposer, $bmInput, $bmSubmit;
   var $tabComments, $tabBookmarks;
   var $fbComposer, $fbIssueType, $fbText, $fbEmail, $fbSubmit, $fbQuote;
   var $paneComments, $paneBookmarks;
   var $tabNotes, $tabBookmarks, $paneNotes, $paneBookmarks;
 
  // ════════════════════════════════════════════════════════════════════
  // DOM BUILDER
  // ════════════════════════════════════════════════════════════════════


   function buildDom() {
   function buildDom() {
    // ── Floating action strip ────────────────────────────────────────
     $fab = $( [
     $fab = $( [
       '<div id="gra-fab">',
       '<div id="gra-fab" role="toolbar" aria-label="Feedback / Notes / Bookmark">',
       '  <button class="gra-fab-btn" id="gra-fab-comment" type="button">',
       '  <button class="gra-fab-btn" id="gra-fab-note" type="button" aria-label="Note">',
       '    <span class="gra-icon gra-icon-comment"></span>',
       '    <span class="gra-icon gra-icon-note" aria-hidden="true"></span>',
       '    <span class="gra-fab-tooltip">Add comment</span>',
       '    <span class="gra-fab-btn-label">Note</span>',
      '  </button>',
      '  <button class="gra-fab-btn" id="gra-fab-bookmark" type="button" aria-label="Mark">',
      '    <span class="gra-icon gra-icon-bookmark" aria-hidden="true"></span>',
      '    <span class="gra-fab-btn-label">Mark</span>',
      '  </button>',
      '  <button class="gra-fab-btn" id="gra-fab-feedback" type="button" aria-label="Feedback">',
      '    <span class="gra-icon gra-icon-feedback" aria-hidden="true"></span>',
      '    <span class="gra-fab-btn-label">Feedback</span>',
       '  </button>',
       '  </button>',
       '  <button class="gra-fab-btn" id="gra-fab-bookmark" type="button">',
       '  <button class="gra-fab-btn" id="gra-fab-search" type="button" aria-label="Search">',
       '    <span class="gra-icon gra-icon-bookmark"></span>',
       '    <span class="gra-icon gra-icon-search" aria-hidden="true"></span>',
       '    <span class="gra-fab-tooltip">Bookmark</span>',
       '    <span class="gra-fab-btn-label">Search</span>',
      '  </button>',
      '  <button class="gra-fab-btn gra-fab-btn-dismiss" id="gra-fab-dismiss" type="button" aria-label="Dismiss">',
      '    <span class="gra-icon gra-icon-dismiss" aria-hidden="true"></span>',
      '    <span class="gra-fab-btn-label">Close</span>',
       '  </button>',
       '  </button>',
       '</div>',
       '</div>',
     ].join('') );
     ].join('') );
     $( 'body' ).append( $fab );
     $('body').append($fab);


     // ── Comment composer card ────────────────────────────────────────
     $mobileBar = $('<div id="gra-mobile-bar"></div>');
     $cmpComposer = $( [
    $('body').append($mobileBar);
       '<div class="gra-composer" id="gra-cmp-composer">',
 
    $fbComposer = $( [
      '<div class="gra-composer" id="gra-fb-composer" role="dialog" aria-label="Send feedback">',
      '  <div class="gra-composer-header">',
      '    <span class="gra-icon gra-icon-feedback" aria-hidden="true"></span>',
      '    <strong>Feedback</strong>',
      '    <button class="gra-btn-x" id="gra-fb-close" title="Close">✕</button>',
      '  </div>',
      '  <div class="gra-fb-quote-label">Selected text:</div>',
      '  <div class="gra-fb-quote" id="gra-fb-quote"></div>',
      '  <div class="gra-fb-field-label">Issue type</div>',
      '  <select class="gra-fb-select" id="gra-fb-issue">',
      '    <option value="">— Choose —</option>',
      '    <option value="wrong_text">Formatting error</option>',
      '    <option value="reference_issue">Reference issue</option>',
      '    <option value="spelling_mistake">Spelling mistake</option>',
      '    <option value="other">Other</option>',
      '  </select>',
      '  <div class="gra-fb-field-label">Details (optional)</div>',
      '  <textarea class="gra-composer-input" id="gra-fb-text" placeholder="Describe the issue…" rows="3"></textarea>',
      '  <div class="gra-fb-field-label">Your email (optional)</div>',
      '  <input class="gra-composer-input gra-fb-email-input" id="gra-fb-email" type="email" placeholder="you@example.com" autocomplete="email">',
      '  <div class="gra-composer-actions">',
      '    <button class="gra-btn-cancel" id="gra-fb-cancel">Cancel</button>',
      '    <button class="gra-btn-submit" id="gra-fb-submit" disabled>Send</button>',
      '  </div>',
      '  <div class="gra-fb-status" id="gra-fb-status"></div>',
      '</div>',
    ].join('') );
     $('body').append($fbComposer);
 
    $ntComposer = $( [
       '<div class="gra-composer" id="gra-nt-composer" role="dialog" aria-label="Add note">',
       '  <div class="gra-composer-user">',
       '  <div class="gra-composer-user">',
       '    <div class="gra-avatar" id="gra-cmp-avatar">' + esc(userInitial) + '</div>',
       '    <div class="gra-avatar">' + esc(currentUser ? userInitial : '✎') + '</div>',
       '    <div class="gra-composer-name" id="gra-cmp-name">' + esc(currentUser || 'Anonymous') + '</div>',
       '    <div class="gra-composer-uname">' + esc(currentUser || 'Notes') + '</div>',
      '  </div>',
      '  <div class="gra-quick-chips" id="gra-quick-chips">',
      '    <button class="gra-chip" data-val="Spelling mistake" type="button">Spelling mistake</button>',
      '    <button class="gra-chip" data-val="Reference error" type="button">Reference error</button>',
      '    <button class="gra-chip" data-val="Others" type="button">Others ▾</button>',
       '  </div>',
       '  </div>',
       '  <textarea class="gra-composer-input" id="gra-cmp-input"',
       '  <textarea class="gra-composer-input" id="gra-nt-input" placeholder="Write a note…" rows="3"></textarea>',
      '    placeholder="Describe the issue…" rows="3"',
      '    style="display:none"></textarea>',
       '  <div class="gra-composer-actions">',
       '  <div class="gra-composer-actions">',
       '    <button class="gra-btn-cancel" id="gra-cmp-cancel">Cancel</button>',
       '    <button class="gra-btn-cancel" id="gra-nt-cancel">Cancel</button>',
       '    <button class="gra-btn-submit" id="gra-cmp-submit" disabled>Comment</button>',
       '    <button class="gra-btn-submit" id="gra-nt-submit" disabled>Save Note</button>',
       '  </div>',
       '  </div>',
       '</div>',
       '</div>',
     ].join('') );
     ].join('') );
    $('body').append($ntComposer);


    // ── Bookmark composer card ────────────────────────────────────────
     $bmComposer = $( [
     $bmComposer = $( [
       '<div class="gra-bm-composer" id="gra-bm-composer">',
       '<div class="gra-bm-composer" id="gra-bm-composer" role="dialog" aria-label="Bookmark">',
       '  <div class="gra-bm-composer-label">',
       '  <div class="gra-bm-composer-label">',
       '    <span class="gra-icon gra-icon-bookmark"></span>',
       '    <span class="gra-icon gra-icon-bookmark" aria-hidden="true"></span>',
       '    Save bookmark',
       '    Save bookmark',
       '  </div>',
       '  </div>',
       '  <input class="gra-composer-input" id="gra-bm-input"',
       '  <input class="gra-composer-input" id="gra-bm-input" type="text" placeholder="Name this bookmark…" autocomplete="off">',
      '    type="text" placeholder="Bookmark name…" autocomplete="off">',
       '  <div class="gra-composer-actions">',
       '  <div class="gra-composer-actions">',
       '    <button class="gra-btn-cancel" id="gra-bm-cancel">Cancel</button>',
       '    <button class="gra-btn-cancel" id="gra-bm-cancel">Cancel</button>',
Line 181: Line 149:
       '</div>',
       '</div>',
     ].join('') );
     ].join('') );
     $( 'body' ).append( $bmComposer );
     $('body').append($bmComposer);


    // ── Right panel ───────────────────────────────────────────────────
     $panel = $( [
     $panel = $( [
       '<div id="gra-panel">',
       '<div id="gra-panel" role="complementary" aria-label="Notes">',
       '  <div id="gra-panel-head">',
       '  <div id="gra-panel-head">',
       '    <div></div>',
       '    <div id="gra-panel-title"></div>',
       '    <button id="gra-panel-close" title="Close panel">✕</button>',
       '    <button id="gra-panel-close" title="Close">✕</button>',
       '  </div>',
       '  </div>',
       '  <div id="gra-tabs">',
       '  <div id="gra-tabs">',
       '    <button class="gra-tab gra-tab-active" id="gra-tab-comments">',
       '    <button class="gra-tab gra-tab-active" id="gra-tab-notes">',
       '      <span class="gra-icon gra-icon-comment"></span> Comments',
       '      <span class="gra-icon gra-icon-note" aria-hidden="true"></span> Notes',
       '    </button>',
       '    </button>',
       '    <button class="gra-tab" id="gra-tab-bookmarks">',
       '    <button class="gra-tab" id="gra-tab-bookmarks">',
       '      <span class="gra-icon gra-icon-bookmark"></span> Bookmarks',
       '      <span class="gra-icon gra-icon-bookmark" aria-hidden="true"></span> Bookmarks',
       '    </button>',
       '    </button>',
       '  </div>',
       '  </div>',
       '  <div id="gra-panel-body">',
       '  <div id="gra-panel-body">',
       '    <div class="gra-pane gra-pane-active" id="gra-pane-comments"></div>',
       '    <div class="gra-pane gra-pane-active" id="gra-pane-notes"></div>',
       '    <div class="gra-pane"             id="gra-pane-bookmarks"></div>',
       '    <div class="gra-pane" id="gra-pane-bookmarks"></div>',
       '  </div>',
       '  </div>',
       '</div>',
       '</div>',
     ].join('') );
     ].join('') );
     $( 'body' ).append( $panel );
     $('body').append($panel);


    // Backdrop
     $backdrop = $('<div id="gra-backdrop" aria-hidden="true"></div>');
     $backdrop = $('<div id="gra-backdrop"></div>');
     $('body').append($backdrop);
     $( 'body' ).append( $backdrop );


    // Persistent toggle button — always visible, opens the panel
     var $toggle = $( [
     var $toggle = $( [
       '<button id="gra-toggle" title="Comments &amp; Bookmarks">',
       '<button id="gra-toggle" aria-label="Notes">',
       '  <span class="gra-icon gra-icon-comment" id="gra-toggle-icon"></span>',
       '  <span class="gra-icon gra-icon-note" id="gra-toggle-icon" aria-hidden="true"></span>',
       '  <span id="gra-toggle-badge"></span>',
       '  <span id="gra-toggle-badge" aria-live="polite"></span>',
       '</button>',
       '</button>',
     ].join('') );
     ].join('') );
     $( 'body' ).append( $toggle );
     $('body').append($toggle);
     $toggle.on( 'click', function() {
     $toggle.on('click', function() {
       if ( $panel.hasClass('gra-panel-open') ) {
       $panel.hasClass('gra-panel-open') ? closePanel() : openPanel(_activeTab);
        closePanel();
     });
      } else {
        openPanel( _activeTab );
      }
     } );


    // Cache references
     $('#gra-panel-title').text(pageTitle.replace(/_/g,' ').split('/')[0].slice(0,30));
     $panelBody    = $( '#gra-panel-body' );
     $tabNotes    = $('#gra-tab-notes');
    // Set panel title to page name
     $tabBookmarks = $('#gra-tab-bookmarks');
    $( '#gra-panel-head div' ).first()
     $paneNotes    = $('#gra-pane-notes');
      .text( pageTitle.replace(/_/g,' ').split('/')[0].slice(0,30) )
     $paneBookmarks= $('#gra-pane-bookmarks');
      .css({ 'font-size':'13px', 'font-weight':'600', 'color':'#5a3a00' });
     $ntInput     = $('#gra-nt-input');
     $tabComments  = $( '#gra-tab-comments' );
     $ntSubmit     = $('#gra-nt-submit');
     $tabBookmarks = $( '#gra-tab-bookmarks' );
     $bmInput     = $('#gra-bm-input');
     $paneComments  = $( '#gra-pane-comments' );
     $bmSubmit     = $('#gra-bm-submit');
     $paneBookmarks = $( '#gra-pane-bookmarks' );
     $fbIssueType  = $('#gra-fb-issue');
     $cmpInput     = $( '#gra-cmp-input' );
    $fbText       = $('#gra-fb-text');
     $cmpSubmit     = $( '#gra-cmp-submit' );
    $fbEmail      = $('#gra-fb-email');
     $bmInput       = $( '#gra-bm-input' );
    $fbSubmit    = $('#gra-fb-submit');
     $bmSubmit     = $( '#gra-bm-submit' );
    $fbQuote      = $('#gra-fb-quote');
 
     // Disable comment submit when not logged in
    if ( !currentUser ) {
       $cmpSubmit.prop( 'disabled', true );
      $cmpComposer.find( '.gra-composer-actions' )
        .prepend( '<a href="' + esc( mw ? mw.util.getUrl('Special:UserLogin') : '/wiki/Special:UserLogin' ) +
                  '" style="font-size:11px;color:#999;margin-right:auto">Log in to comment</a>' );
    }
   }
   }
  // ════════════════════════════════════════════════════════════════════
  // FLOATING ACTION STRIP — position & show/hide
  // ════════════════════════════════════════════════════════════════════
  function showFab( rect ) {
    if ( !rect ) return;
    // FAB uses position:fixed — coords are viewport-relative (no scroll offset needed).
    // Place strip to the right of the selection; if it would go off-screen, place to the left.
    var fabW = 46; var fabH = 84;
    var top  = rect.top + ( rect.height / 2 ) - ( fabH / 2 );
    var left = rect.right + 10;
    // If too close to right edge, flip to left of selection
    if ( left + fabW > window.innerWidth - 8 ) {
      left = rect.left - fabW - 10;
    }
    // Clamp vertically within viewport
    top = clamp( top, 8, window.innerHeight - fabH - 8 );
    // Clamp horizontally
    left = clamp( left, 8, window.innerWidth - fabW - 8 );
    $fab.css({ top: top + 'px', left: left + 'px' }).addClass('gra-fab-visible');
  }
  function hideFab() {
    $fab.removeClass('gra-fab-visible');
  }
  // ════════════════════════════════════════════════════════════════════
  // SELECTION HANDLING
  // ════════════════════════════════════════════════════════════════════


   function captureSelection() {
   function captureSelection() {
     var sel = window.getSelection();
     var sel = window.getSelection();
     if ( !sel || sel.isCollapsed || !sel.rangeCount ) return false;
     if (!sel || sel.isCollapsed || !sel.rangeCount) return false;
 
     var range = sel.getRangeAt(0);
     var range   = sel.getRangeAt(0);
     var text = sel.toString().trim();
     var text     = sel.toString().trim();
     if (!text || text.length < 2) return false;
     if ( !text || text.length < 2 ) return false;
     var contentEl = document.querySelector(CONTENT_SEL);
 
     if (!contentEl) return false;
    // Must be inside mw-content-text
     var ancestor = range.commonAncestorContainer;
     var contentEl = document.querySelector( CONTENT_SEL );
     if (ancestor.nodeType === 3) ancestor = ancestor.parentNode;
     if ( !contentEl ) return false;
     if (!ancestor || !contentEl.contains(ancestor)) return false;
     var startEl = range.commonAncestorContainer;
    var _editorEl = document.getElementById('se-surface') ||
     if ( startEl.nodeType === 3 ) startEl = startEl.parentNode;
                    document.querySelector('.se-outer');
     if ( !contentEl.contains( startEl ) ) return false;
    if ( _editorEl && _editorEl.contains(ancestor) ) return false;
 
     _selText  = text;
     _selText  = text;
    _selRange = range.cloneRange();
     _selRect  = range.getBoundingClientRect();
     _selRect  = range.getBoundingClientRect();
    try { _selRange = range.cloneRange(); }
    catch(e){ _selRange = null; }
     return true;
     return true;
   }
   }


   // ════════════════════════════════════════════════════════════════════
   function reCaptureFromDOM() {
  // COMPOSER POSITIONING
    if (!_selText) return false;
   // ════════════════════════════════════════════════════════════════════
    var contentEl = document.querySelector(CONTENT_SEL);
    if (!contentEl) return false;
    var found = findTextInContent(contentEl, _selText.slice(0,80).replace(/…$/,'').trim());
    if (!found) return false;
    _selRange = found;
    return true;
   }


   function positionComposer( $el ) {
   function tryShowActions() {
     if ( !_selRect ) return;
     if ($fbComposer && $fbComposer.hasClass('gra-composer-visible')) return;
     // position:fixed — viewport coords only
     if ($ntComposer && $ntComposer.hasClass('gra-composer-visible')) return;
     var top  = _selRect.bottom + 8;
     if ($bmComposer && $bmComposer.hasClass('gra-composer-visible')) return;
     var left = _selRect.left;
     if (!captureSelection()) { hideActions(); return; }
     // Keep composer within viewport
    _fabSelVer = _selVersion;
     var composerW = 308;
    showFab(_selRect);
     if ( left + composerW > window.innerWidth - 8 ) {
  }
       left = window.innerWidth - composerW - 8;
 
  function showFab(rect) {
     if (!rect) return;
     var fabW, fabH, top, left;
     if (_mobile) {
      /* Docked as a fixed bar below the reader toolbar.
        All positioning is handled by CSS via .gra-fab-mobile-docked,
        so it never collides with the native selection menu and never
        clips at screen edges or causes horizontal scroll. */
       $fab.css({ position: '', top: '', left: '', visibility: '' })
          .addClass('gra-fab-visible gra-fab-mobile-docked');
      return;
     }
     }
     left = Math.max( left, 8 );
     fabW = 46; fabH = 126;
     // If composer would appear below viewport, show above selection instead
    top  = rect.top + (rect.height / 2) - (fabH / 2);
     if ( top + 160 > window.innerHeight ) {
     left = rect.right + 10;
      top = _selRect.top - 170;
     if (left + fabW > window.innerWidth - 8) left = rect.left - fabW - 10;
     }
    top = clamp(top,  8, window.innerHeight - fabH - 8);
    top = Math.max( top, 8 );
     left = clamp(left, 8, window.innerWidth  - fabW - 8);
     $el.css({ top: top + 'px', left: left + 'px' });
     $fab.css({ top: top + 'px', left: left + 'px' }).addClass('gra-fab-visible');
   }
   }


   // ════════════════════════════════════════════════════════════════════
   function hideFab() { $fab.removeClass('gra-fab-visible gra-fab-mobile-docked'); }
   // WRAP SELECTION IN HIGHLIGHT SPAN
   function hideActions() { hideFab(); }
  // ════════════════════════════════════════════════════════════════════


   function wrapSelection( id, cssClass ) {
   function wrapSelection(id, cssClass) {
     if ( !_selRange ) return null;
    var range = _selRange;
    _selRange = null;
     if (!range) return null;
     try {
     try {
       var span = document.createElement('span');
       if (!document.contains(range.startContainer) ||
      span.className = cssClass;
          !document.contains(range.endContainer)) return null;
      span.setAttribute('data-gra-id', id);
     } catch(e) { return null; }
      _selRange.surroundContents( span );
    function makeSpan() {
      return span;
      var sp = document.createElement('span');
     } catch ( e ) {
      sp.className = cssClass;
      // surroundContents fails for multi-element selections;
      sp.setAttribute('data-gra-id', id);
      // extract and wrap the fragment
       return sp;
      try {
        var frag = _selRange.extractContents();
        var span2 = document.createElement('span');
        span2.className = cssClass;
        span2.setAttribute('data-gra-id', id);
        span2.appendChild( frag );
        _selRange.insertNode( span2 );
        return span2;
       } catch(e2) { return null; }
     }
     }
    try {
      var span = makeSpan();
      range.surroundContents(span);
      if (span.parentNode) return span;
    } catch(e) {}
    try {
      var frag = range.extractContents();
      var sp2  = makeSpan();
      sp2.appendChild(frag);
      range.insertNode(sp2);
      if (sp2 && sp2.parentNode) return sp2;
    } catch(e2) {}
    return null;
   }
   }


   // ════════════════════════════════════════════════════════════════════
   function openFeedbackComposer() {
  // COMMENT FLOW
    hideActions();
   // ════════════════════════════════════════════════════════════════════
    $fbQuote.text(_selText.slice(0,200) + (_selText.length > 200 ? '…' : ''));
    $fbIssueType.val('');
    $fbText.val('');
    $fbSubmit.prop('disabled', true);
    $('#gra-fb-status').text('').removeClass('gra-fb-ok gra-fb-err');
    if (currentUserEmail) $fbEmail.val(currentUserEmail);
    else $fbEmail.val('');
    if (!_mobile) $fbComposer.css({top:'', left:'', transform:''});
    $fbComposer.addClass('gra-composer-visible');
    $backdrop.addClass('gra-backdrop-visible');
    setTimeout(function(){ $fbIssueType.focus(); }, isMobile() ? 300 : 0);
   }


   function openCommentComposer() {
   function closeFeedbackComposer() {
     hideFab();
     $fbComposer.removeClass('gra-composer-visible');
    positionComposer( $cmpComposer );
     $backdrop.removeClass('gra-backdrop-visible');
    $cmpComposer.addClass('gra-composer-visible');
    _selRange = null; _selText = ''; _selRect = null;
    // Don't focus the textarea — it's hidden until "Others" chip is selected.
    // Focus the composer container so Escape key works.
     $cmpComposer[0].focus && $cmpComposer[0].setAttribute('tabindex', '-1');
   }
   }


   function closeCommentComposer() {
   function submitFeedback() {
     $cmpComposer.removeClass('gra-composer-visible');
     var issueType = $fbIssueType.val();
     $( '#gra-quick-chips .gra-chip' ).removeClass('gra-chip-active');
    var details  = $fbText.val().trim();
     $cmpInput.hide().val('');
    var email    = $fbEmail.val().trim();
     $cmpSubmit.prop('disabled', true);
    var quote    = $fbQuote.text();
     _selRange = null;
    if (!issueType) return;
     _selText  = '';
    $fbSubmit.prop('disabled', true).text('Sending…');
     _selRect  = null;
     $('#gra-fb-status').text('').removeClass('gra-fb-ok gra-fb-err');
    var issueLabels = {
      wrong_text: 'Formatting error', reference_issue: 'Reference issue',
      spelling_mistake: 'Spelling mistake', other: 'Other'
    };
    var payload = new FormData();
    payload.append('issue_type',    issueLabels[issueType] || issueType);
    payload.append('page',          pageTitle.replace(/_/g,' '));
     payload.append('url',          window.location.href);
    payload.append('selected_text', quote);
     payload.append('details',       details || '');
     payload.append('user_email',    email || currentUserEmail || '');
     payload.append('wiki_user',    currentUser || 'anonymous');
     fetch('/feedback.php', {method:'POST', body:payload})
      .then(function(r){ return r.json(); })
      .then(function(data){
        if (data && data.ok) showFeedbackSuccess();
        else showFeedbackError(data && data.error ? data.error : 'Could not send.');
      })
      .catch(function(){ showFeedbackError('Network error. Please try again.'); });
   }
   }


   function submitComment() {
   function showFeedbackSuccess() {
     var text  = $cmpInput.val().trim();
     $fbSubmit.prop('disabled', false).text('Send');
     if ( !text || !currentUser ) return;
     $('#gra-fb-status').text('✓ Feedback sent. Thank you!').addClass('gra-fb-ok');
     var id    = uid();
     setTimeout(closeFeedbackComposer, 2500);
    var ts    = nowIso();
  }
     var quote = _selText.slice(0, 120) + ( _selText.length > 120 ? '' : '' );
  function showFeedbackError(msg) {
     $fbSubmit.prop('disabled', false).text('Send');
    $('#gra-fb-status').text('' + msg).addClass('gra-fb-err');
  }


    // Wrap text in highlight span
  function openNoteComposer() {
     var span = wrapSelection( id, 'gra-comment-highlight' );
     hideActions();
     if ( span ) span.setAttribute('data-gra-quote', quote);
     if (!_mobile) $ntComposer.css({ top: '', left: '', transform: '' });
 
     $ntComposer.addClass('gra-composer-visible');
    // Save locally
     $backdrop.addClass('gra-backdrop-visible');
    var entry = { id:id, author:currentUser, ts:ts, quote:quote, text:text };
     setTimeout(function(){ $ntInput.focus(); }, isMobile() ? 300 : 0);
     _comments.push( entry );
     // Persist highlight anchor so it survives page refresh
    persistCommentHighlight( id, quote );
 
     // Persist to wiki
    saveCommentToWiki( id, quote, text, ts );
    notifyAdmin( _activeId || '', quote, text, ts );
 
    // Update panel
    renderCommentCards();
    closeCommentComposer();
    openPanel('comments');
   }
   }


   function saveCommentToWiki( id, quote, text, ts ) {
   function closeNoteComposer() {
     if ( !window.mw ) return;
     $ntComposer.removeClass('gra-composer-visible');
    var api = new mw.Api();
     $backdrop.removeClass('gra-backdrop-visible');
     api.get({
     $ntInput.val('');
      action:'query', prop:'revisions', titles:commentsPage,
     $ntSubmit.prop('disabled', true);
      rvprop:'content', rvslots:'main', formatversion:2,
    _selRange = null; _selText = ''; _selRect = null;
     }).then(function(data){
      var page = (data.query.pages||[])[0]||{};
      var existing = '';
      if (page.revisions && page.revisions[0]) {
        var rev = page.revisions[0];
        existing = rev.slots ? rev.slots.main.content : rev['*'] || '';
      }
      var entry = '{{GrComment\n'
                + '| id        = ' + id          + '\n'
                + '| author    = ' + currentUser + '\n'
                + '| timestamp = ' + ts          + '\n'
                + '| quote     = ' + quote.replace(/\n/g,' ') + '\n'
                + '| text      = ' + text.replace(/\n/g,' ') + '\n'
                + '}}\n';
      var updated = (existing.trim() ? existing.trim() + '\n\n' : '') + entry;
      api.postWithEditToken({
        action:'edit', title:commentsPage, text:updated,
        summary:'New comment by ' + currentUser + ' on [[' + pageTitle + ']] (stored off-page)',
        bot:0,
      }).catch(function(){});
    }).catch(function(){});
   }
   }


   function notifyAdmin( anchorId, quote, commentText, ts ) {
   function submitNote() {
     /* Send notification email directly using MW's EmailUser API, targeting
     var text = $ntInput.val().trim();
    * ADMIN_EMAIL via the ADMIN_USER account — no server-side SMTP config needed.
     if (!text) return;
    * ADMIN_EMAIL is set at the top of this file; change it to any address. */
     var id    = uid();
     if ( !window.mw ) return;
    var ts    = nowIso();
 
     var quote = _selText.slice(0,120) + (_selText.length > 120 ? '' : '');
     var articlePath = ( mw.config.get('wgArticlePath') || '/wiki/$1' )
     if (!_selRange && _selText) reCaptureFromDOM();
                        .replace( '$1', pageTitle );
     var span  = wrapSelection(id, 'gra-note-highlight');
     var anchorLink  = window.location.origin + articlePath
     if (span) span.setAttribute('data-gra-quote', quote);
                    + ( anchorId ? '#' + anchorId : '' );
     _notes.push({id:id, ts:ts, quote:quote, text:text});
     var pageDisplay = pageTitle.replace( /_/g, ' ' );
    persistNotes();
     var subject    = '[Grantha] New comment on "' + pageDisplay + '"';
    persistNoteHighlight(id, quote);
     var body        = 'Page    : ' + pageDisplay + '\n'
    renderNoteCards();
                    + 'By      : ' + ( currentUser || 'Anonymous' ) + '\n'
    closeNoteComposer();
                    + 'Time    : ' + ts + '\n'
    openPanel('notes');
                    + 'Passage : "' + quote + '"\n\n'
                    + 'Comment :\n' + commentText + '\n\n'
                    + 'Link    : ' + anchorLink + '\n';
 
     /* Primary: send via MW EmailUser API to ADMIN_USER account.
    * The admin account must have ADMIN_EMAIL set in Special:Preferences,
    * OR you can hardcode the target as any confirmed MW user. */
    new mw.Api().post({
      action: 'emailuser',
      target: ADMIN_USER,
      subject: subject,
      text:   body,
      token:  mw.user.tokens.get( 'csrfToken' ),
    }).catch(function(){
      /* Fallback: post a section to admin talk page for Echo notification */
      if ( !currentUser ) return;
      var wikimsg = '== New comment on [[' + pageDisplay + ']] ==\n'
                  + '; By: ' + ( currentUser || 'Anonymous' ) + '\n'
                  + '; Passage: //' + quote + '//\n'
                  + '; Link: ' + anchorLink + '\n\n'
                  + commentText.slice(0,500)
                  + ( commentText.length > 500 ? '\n...' : '' )
                  + '\n\n[[User:Chandrashekars|Chandrashekars]] ([[User talk:Chandrashekars|talk]]) 18:31, 25 April 2026 (UTC)';
      new mw.Api().postWithEditToken({
        action:'edit', title:'User_talk:' + ADMIN_USER, section:'new',
        sectiontitle:'Comment on ' + pageDisplay,
        text:wikimsg,
        summary:'Comment notification from [[' + pageDisplay + ']]',
        bot:0,
      }).catch(function(){});
    });
   }
   }


  // ── Load comments from /Comments wiki page ───────────────────────
   function persistNotes() {
   function loadComments( cb ) {
     try { localStorage.setItem(NT_LS_KEY, JSON.stringify(_notes)); } catch(e){}
    if ( _cmtLoaded ) { if (cb) cb(); return; }
     if ( !window.mw ) { _cmtLoaded=true; if(cb)cb(); return; }
    new mw.Api().get({
      action:'query', prop:'revisions', titles:commentsPage,
      rvprop:'content', rvslots:'main', formatversion:2,
    }).then(function(data){
      var page = (data.query.pages||[])[0]||{};
      var wt = '';
      if (page.revisions && page.revisions[0]) {
        var rev = page.revisions[0];
        wt = rev.slots ? rev.slots.main.content : rev['*'] || '';
      }
      _comments = parseCommentsWt(wt);
      _cmtLoaded = true;
      if (cb) cb();
    }).catch(function(){ _cmtLoaded=true; if(cb)cb(); });
   }
   }
 
   function loadNotes() {
   function parseCommentsWt( wt ) {
     try { var r = localStorage.getItem(NT_LS_KEY); if (r) _notes = JSON.parse(r)||[]; } catch(e){}
     var out = [];
    var re = /\{\{GrComment\s*\n([\s\S]*?)\}\}/g;
    var m;
    while ((m = re.exec(wt)) !== null) {
      var block = m[1];
      var f = {};
      var lr = /\|\s*([\w_]+)\s*=\s*([\s\S]*?)(?=\n\||\n\}\}|$)/g;
      var lm;
      while ((lm = lr.exec(block)) !== null) {
        f[lm[1].trim()] = lm[2].trim();
      }
      if (f.id) out.push({
        id:f.id, author:f.author||'Anonymous',
        ts:f.timestamp||'', quote:f.quote||'', text:f.text||''
      });
    }
    return out;
   }
   }
  // ════════════════════════════════════════════════════════════════════
  // BOOKMARK FLOW
  // ════════════════════════════════════════════════════════════════════


   function openBookmarkComposer() {
   function openBookmarkComposer() {
     hideFab();
     hideActions();
     positionComposer( $bmComposer );
     if (!_mobile) $bmComposer.css({ top: '', left: '', transform: '' });
     $bmComposer.addClass('gra-composer-visible');
     $bmComposer.addClass('gra-composer-visible');
     $bmInput.val('').focus();
     $backdrop.addClass('gra-backdrop-visible');
    setTimeout(function(){ $bmInput.focus(); }, isMobile() ? 300 : 0);
   }
   }


   function closeBookmarkComposer() {
   function closeBookmarkComposer() {
     $bmComposer.removeClass('gra-composer-visible');
     $bmComposer.removeClass('gra-composer-visible');
    $backdrop.removeClass('gra-backdrop-visible');
     $bmInput.val('');
     $bmInput.val('');
     _selRange = null;
     _selRange = null; _selText = ''; _selRect = null;
    _selText = '';
    _selRect = null;
   }
   }


   function submitBookmark() {
   function submitBookmark() {
     var name  = $bmInput.val().trim() || ( 'Bookmark ' + (_bookmarks.length+1) );
     var name  = $bmInput.val().trim() || ('Bookmark ' + (_bookmarks.length+1));
     var id    = uid();
     var id    = uid();
    var ts    = nowIso();
     var quote = _selText.slice(0,120) + (_selText.length > 120 ? '…' : '');
     var quote = _selText.slice(0,120) + (_selText.length>120?'…':'');
     if (!_selRange && _selText) reCaptureFromDOM();
 
     var span = wrapSelection(id, 'gra-bookmark-highlight');
     // Wrap text in bookmark highlight span
     if (span) { span.setAttribute('data-gra-id', id); span.setAttribute('data-gra-name', name); }
     var span = wrapSelection( id, 'gra-bookmark-highlight' );
     _bookmarks.push({id:id, name:name, quote:quote, ts:nowIso()});
     if ( span ) {
      span.setAttribute('data-gra-id', id);
      span.setAttribute('data-gra-name', name);
    }
 
     var entry = { id:id, name:name, quote:quote, ts:ts };
    _bookmarks.push( entry );
     persistBookmarks();
     persistBookmarks();
     renderBookmarkCards();
     renderBookmarkCards();
Line 560: Line 425:
   }
   }


   function deleteBookmark( id ) {
   function deleteBookmark(id) {
     _bookmarks = _bookmarks.filter(function(b){ return b.id !== id; });
     _bookmarks = _bookmarks.filter(function(b){ return b.id !== id; });
    // Remove highlight span, replace with its text content
     var span = document.querySelector('[data-gra-id="'+id+'"].gra-bookmark-highlight');
     var span = document.querySelector('[data-gra-id="' + id + '"].gra-bookmark-highlight');
     if (span && span.parentNode) {
     if (span) {
       var p = span.parentNode;
       var parent = span.parentNode;
       while (span.firstChild) p.insertBefore(span.firstChild, span);
       while (span.firstChild) parent.insertBefore(span.firstChild, span);
       p.removeChild(span);
       parent.removeChild(span);
     }
     }
     persistBookmarks();
     persistBookmarks(); renderBookmarkCards();
    renderBookmarkCards();
   }
   }


Line 576: Line 439:
     try { localStorage.setItem(BM_LS_KEY, JSON.stringify(_bookmarks)); } catch(e){}
     try { localStorage.setItem(BM_LS_KEY, JSON.stringify(_bookmarks)); } catch(e){}
   }
   }
   function loadBookmarks() {
   function loadBookmarks() {
     try {
     try { var r = localStorage.getItem(BM_LS_KEY); if (r) _bookmarks = JSON.parse(r)||[]; } catch(e){}
      var raw = localStorage.getItem(BM_LS_KEY);
      if (raw) _bookmarks = JSON.parse(raw) || [];
    } catch(e){ _bookmarks = []; }
   }
   }


  // ════════════════════════════════════════════════════════════════════
   function openPanel(tab) {
  // PANEL — open / close / tabs
  // ════════════════════════════════════════════════════════════════════
 
   function openPanel( tab ) {
     _activeTab = tab || _activeTab;
     _activeTab = tab || _activeTab;
     switchTab( _activeTab );
     switchTab(_activeTab);
     $panel.addClass('gra-panel-open');
     $panel.addClass('gra-panel-open');
     $backdrop.addClass('gra-backdrop-visible');
     $backdrop.addClass('gra-backdrop-visible');
   }
   }
   function closePanel() {
   function closePanel() {
     $panel.removeClass('gra-panel-open');
     $panel.removeClass('gra-panel-open');
     $backdrop.removeClass('gra-backdrop-visible');
     $backdrop.removeClass('gra-backdrop-visible');
   }
   }
 
   function switchTab(tab) {
   function switchTab( tab ) {
     _activeTab = tab;
     _activeTab = tab;
     $tabComments.toggleClass('gra-tab-active', tab==='comments');
     $tabNotes.toggleClass('gra-tab-active', tab==='notes');
     $tabBookmarks.toggleClass('gra-tab-active', tab==='bookmarks');
     $tabBookmarks.toggleClass('gra-tab-active', tab==='bookmarks');
     $paneComments.toggleClass('gra-pane-active', tab==='comments');
     $paneNotes.toggleClass('gra-pane-active', tab==='notes');
     $paneBookmarks.toggleClass('gra-pane-active', tab==='bookmarks');
     $paneBookmarks.toggleClass('gra-pane-active', tab==='bookmarks');
     if (tab==='comments') {
     if (tab==='notes') renderNoteCards();
      loadComments(function(){ renderCommentCards(); });
     else renderBookmarkCards();
     } else {
      renderBookmarkCards();
    }
   }
   }


  // ════════════════════════════════════════════════════════════════════
   function renderNoteCards() {
  // RENDER CARDS
     if (!_notes.length) {
  // ════════════════════════════════════════════════════════════════════
       $paneNotes.html('<div class="gra-empty-state">No notes yet.<br>Select text and tap ✎ to add one.</div>');
 
   function renderCommentCards() {
     if ( _comments.length === 0 ) {
       $paneComments.html('<div class="gra-empty-state">No comments yet.<br>Select text and click 💬 to add one.</div>');
       return;
       return;
     }
     }
     var html = '';
     var html = '';
     _comments.slice().reverse().forEach(function(c){
     _notes.slice().reverse().forEach(function(n){
       html += '<div class="gra-comment-card" data-gra-id="' + esc(c.id) + '">'
       html += '<div class="gra-note-card" data-gra-id="'+esc(n.id)+'">'
             + '<div class="gra-card-header">'
             + '<div class="gra-card-header">'
             + '<div class="gra-avatar">' + esc((c.author||'?').charAt(0).toUpperCase()) + '</div>'
             + '<span class="gra-icon gra-icon-note" aria-hidden="true"></span>'
             + '<div class="gra-card-meta">'
             + '<div class="gra-card-meta">'
            + '<div class="gra-card-author">' + esc(c.author) + '</div>'
             + (n.ts ? '<div class="gra-card-ts">'+esc(fmtTs(n.ts))+'</div>' : '')
             + (c.ts ? '<div class="gra-card-ts">' + esc(fmtTs(c.ts)) + '</div>' : '')
             + '</div>'
             + '</div>'
            + '<button class="gra-note-del" data-del-id="'+esc(n.id)+'" title="Delete">×</button>'
             + '</div>'
             + '</div>'
             + (c.quote ? '<div class="gra-card-quote">' + esc(c.quote) + '</div>' : '')
             + (n.quote ? '<div class="gra-card-quote">'+esc(n.quote)+'</div>' : '')
             + '<div class="gra-card-text">' + esc(c.text) + '</div>'
             + '<div class="gra-card-text">'+esc(n.text)+'</div>'
             + '</div>';
             + '</div>';
     });
     });
     $paneComments.html(html);
     $paneNotes.html(html);
  }
 
  function deleteNote(id) {
    _notes = _notes.filter(function(n){ return n.id !== id; });
    var span = document.querySelector('[data-gra-id="'+id+'"].gra-note-highlight');
    if (span && span.parentNode) {
      var p = span.parentNode;
      while (span.firstChild) p.insertBefore(span.firstChild, span);
      p.removeChild(span);
    }
    try {
      var s = JSON.parse(localStorage.getItem(NT_LS_KEY+'_hl')||'[]');
      s = s.filter(function(h){ return h.id !== id; });
      localStorage.setItem(NT_LS_KEY+'_hl', JSON.stringify(s));
    } catch(e){}
    persistNotes(); renderNoteCards();
   }
   }


   function renderBookmarkCards() {
   function renderBookmarkCards() {
     if ( _bookmarks.length === 0 ) {
     if (!_bookmarks.length) {
       $paneBookmarks.html('<div class="gra-empty-state">No bookmarks yet.<br>Select text and click 🔖 to save a bookmark.</div>');
       $paneBookmarks.html('<div class="gra-empty-state">No bookmarks yet.<br>Select text and tap 🔖 to save one.</div>');
       return;
       return;
     }
     }
     var html = '';
     var html = '';
     _bookmarks.slice().reverse().forEach(function(b){
     _bookmarks.slice().reverse().forEach(function(b){
       html += '<div class="gra-bookmark-card" data-gra-id="' + esc(b.id) + '">'
       html += '<div class="gra-bookmark-card" data-gra-id="'+esc(b.id)+'">'
             + '<span class="gra-icon gra-icon-bookmark"></span>'
             + '<span class="gra-icon gra-icon-bookmark" aria-hidden="true"></span>'
             + '<div class="gra-bookmark-info">'
             + '<div class="gra-bookmark-info">'
             + '<div class="gra-bookmark-name">' + esc(b.name) + '</div>'
             + '<div class="gra-bookmark-name">'+esc(b.name)+'</div>'
             + (b.quote ? '<div class="gra-bookmark-quote">' + esc(b.quote) + '</div>' : '')
             + (b.quote ? '<div class="gra-bookmark-quote">'+esc(b.quote)+'</div>' : '')
             + '</div>'
             + '</div>'
             + '<button class="gra-bookmark-del" data-del-id="' + esc(b.id) + '" title="Remove bookmark">×</button>'
             + '<button class="gra-bookmark-del" data-del-id="'+esc(b.id)+'" title="Remove">×</button>'
             + '</div>';
             + '</div>';
     });
     });
Line 658: Line 520:
   }
   }


  // ════════════════════════════════════════════════════════════════════
   function scrollToHighlight(id) {
  // SCROLL TO HIGHLIGHT
     var el = document.querySelector('[data-gra-id="'+id+'"]');
  // ════════════════════════════════════════════════════════════════════
 
   function scrollToHighlight( id ) {
     var el = document.querySelector('[data-gra-id="' + id + '"]');
     if (!el) return;
     if (!el) return;
     el.scrollIntoView({ behavior:'smooth', block:'center' });
     el.scrollIntoView({behavior:'smooth', block:'center'});
     el.classList.add('gra-hl-active');
     el.classList.add('gra-hl-active');
     setTimeout(function(){ el.classList.remove('gra-hl-active'); }, 2000);
     setTimeout(function(){ el.classList.remove('gra-hl-active'); }, 2000);
   }
   }


   // ════════════════════════════════════════════════════════════════════
   function wireEvents() {
  // EVENT WIRING
  // ════════════════════════════════════════════════════════════════════


  function wireEvents() {
    /* Suppress native context menu inside article content (Android/desktop) */
    document.addEventListener('contextmenu', function(e) {
      var tag = e.target.tagName;
      if (tag === 'INPUT' || tag === 'TEXTAREA' || tag === 'SELECT') return;
      var c = document.querySelector(CONTENT_SEL);
      if (c && c.contains(e.target)) e.preventDefault();
    }, { passive: false });


     // ── Selection → show FAB ──────────────────────────────────────
     /* Desktop mouseup */
    // On mouseup: capture the selection and show the FAB strip.
     $(document).on('mouseup', function(e){
    // We use mouseup (not click) so the selection is fully made.
       if (e.button !== 0) return;
     $( document ).on('mouseup', function(e){
       if (_mobile) return;
      // Ignore mouseup on FAB/composer — those are handled separately
       setTimeout(tryShowActions, 20);
       if ( $(e.target).closest('#gra-fab, .gra-composer, .gra-bm-composer').length ) return;
      // Ignore if a composer is already open
       if ( $cmpComposer.hasClass('gra-composer-visible') ||
          $bmComposer.hasClass('gra-composer-visible') ) return;
       setTimeout(function(){
        if ( captureSelection() ) {
          showFab( _selRect );
        } else {
          hideFab();
        }
      }, 10);
     });
     });


     // ── FAB mousedown: block selection collapse ───────────────────
     /* Separate timers so mobile + desktop never clobber each other */
     // Browsers collapse text selection on mousedown. preventDefault
    var _selTimer    = null;  /* desktop debounce */
     // keeps the selection alive so the click handler can wrap it.
     var _mobShowTimer = null; /* mobile show-on-touchend */
     $fab.on('mousedown', function(e){
 
       e.preventDefault();
    /* Mobile: show fab quickly after finger lifts (selection settled).
      180ms feels instant while still letting the range stabilise. */
    document.addEventListener('touchend', function(e) {
      if (!_mobile) return;
      if ($fab[0] && $fab[0].contains(e.target)) return;
      clearTimeout(_mobShowTimer);
      _mobShowTimer = setTimeout(function() {
        var sel = window.getSelection();
        if (!sel || sel.isCollapsed || !sel.toString().trim()) return;
        tryShowActions();
      }, 180);
    }, { passive: true });
 
     /* Mobile: only HIDE the fab when selection is cleared while it's visible.
      (Reposition isn't needed now that the bar is docked, and re-running
      showFab here was causing the lag/flicker.) */
     document.addEventListener('selectionchange', function() {
       if (!_mobile) return;
      if (!$fab.hasClass('gra-fab-visible')) return;
      var sel = window.getSelection();
      if (!sel || sel.isCollapsed || !sel.toString().trim()) {
        clearTimeout(_mobShowTimer);
        hideActions();
      }
     });
     });


     // ── Mousedown outside: hide FAB ───────────────────────────────
     /* selectionchange debounced (desktop only) */
     $( document ).on('mousedown', function(e){
     document.addEventListener('selectionchange', function() {
       var $t = $(e.target);
      if (_mobile) return;
      if ( !$t.closest('#gra-fab, .gra-composer, .gra-bm-composer, #gra-toggle, #gra-panel').length ) {
      _selVersion++;
         hideFab();
      clearTimeout(_selTimer);
       }
       var v = _selVersion;
      _selTimer = setTimeout(function(){
        if (v !== _selVersion) return;
        if (_fabSelVer === v) return;
         tryShowActions();
       }, 600);
     });
     });


     // ── FAB: Comment button ───────────────────────────────────────
     /* ── KEY FIX: fab touchstart sets flag to prevent hideActions ── */
     $( '#gra-fab-comment' ).on('click', function(e){
     $fab[0].addEventListener('touchstart', function(e) {
      _fabTouched = true;
      /* Don't propagate to document handler */
       e.stopPropagation();
       e.stopPropagation();
       openCommentComposer();
    }, { passive: true });
 
    /* Click outside → hide actions (blocked if fab was touched) */
    $(document).on('mousedown touchstart', function(e){
       if (_fabTouched) { _fabTouched = false; return; }
      var t = e.target;
      if ($fab[0]        && $fab[0].contains(t))        return;
      if ($fbComposer[0] && $fbComposer[0].contains(t)) return;
      if ($ntComposer[0] && $ntComposer[0].contains(t)) return;
      if ($bmComposer[0] && $bmComposer[0].contains(t)) return;
      hideActions();
     });
     });


     // ── FAB: Bookmark button ──────────────────────────────────────
     /* ── FAB buttons — use touchend for mobile, click for desktop ── */
     $( '#gra-fab-bookmark' ).on('click', function(e){
    function fabAction(btnId, action) {
       e.stopPropagation();
      var el = document.getElementById(btnId);
       openBookmarkComposer();
      if (!el) return;
      /* touchend: fires before document touchstart clears _selRange */
      el.addEventListener('touchend', function(e) {
        e.preventDefault();
        e.stopPropagation();
        if (!_selRange && !reCaptureFromDOM()) return;
        action();
      }, { passive: false });
      /* click: for desktop */
      el.addEventListener('click', function(e) {
        e.preventDefault();
        e.stopPropagation();
        if (!_selRange && !reCaptureFromDOM()) return;
        action();
      });
    }
 
    fabAction('gra-fab-note',     openNoteComposer);
    fabAction('gra-fab-bookmark', openBookmarkComposer);
    fabAction('gra-fab-feedback', openFeedbackComposer);
 
    document.getElementById('gra-fab-search').addEventListener('touchend', function(e) {
      e.preventDefault(); e.stopPropagation();
      var q = _selText;
      hideActions();
      _selRange = null; _selText = ''; _selRect = null;
      if (q && window.showSearchDialog) { window.showSearchDialog(q); }
    }, { passive: false });
    document.getElementById('gra-fab-search').addEventListener('click', function(e) {
       e.preventDefault(); e.stopPropagation();
       var q = _selText;
      hideActions();
      _selRange = null; _selText = ''; _selRect = null;
      if (q && window.showSearchDialog) { window.showSearchDialog(q); }
      else if (q) { $(document).trigger($.Event('keydown', {ctrlKey:true, key:'k', keyCode:75})); }
     });
     });


     // ── Comment composer ──────────────────────────────────────────
     /* ── Dismiss button: hide toolbar + clear selection (mobile) ── */
    /* Quick-select chips: clicking a chip selects it and sets comment text.
     (function () {
    * 'Others' chip shows the textarea for free-form input.
       var dismissEl = document.getElementById('gra-fab-dismiss');
    * Any other chip sets the text directly and enables submit. */
       if (!dismissEl) return;
     $( '#gra-cmp-composer' ).on( 'click', '.gra-chip', function() {
       function doDismiss(e) {
       var $chip = $( this );
         e.preventDefault(); e.stopPropagation();
      var val  = $chip.attr('data-val');
        hideActions();
      // Toggle active state — allow deselecting
         _selRange = null; _selText = ''; _selRect = null;
      var wasActive = $chip.hasClass('gra-chip-active');
        if (window.getSelection) {
      $( '#gra-quick-chips .gra-chip' ).removeClass('gra-chip-active');
          var s = window.getSelection();
       if ( !wasActive ) $chip.addClass('gra-chip-active');
          if (s && s.removeAllRanges) s.removeAllRanges();
       if ( val === 'Others' && !wasActive ) {
         }
         // Show textarea for free-form input
        $cmpInput.show().focus().val('');
         $cmpSubmit.prop('disabled', true);
      } else if ( !wasActive ) {
        // Pre-fill and hide textarea — chip text is the comment
        $cmpInput.hide().val(val);
        $cmpSubmit.prop('disabled', !currentUser);
      } else {
        // Deselected — clear
        $cmpInput.hide().val('');
         $cmpSubmit.prop('disabled', true);
       }
       }
     } );
      dismissEl.addEventListener('touchend', doDismiss, { passive: false });
     $cmpInput.on('input', function(){
      dismissEl.addEventListener('click', doDismiss);
      $cmpSubmit.prop('disabled', !$(this).val().trim() || !currentUser);
     }());
    });
 
     $( '#gra-cmp-cancel' ).on('click', function(){
    /* Feedback composer */
      closeCommentComposer();
     $fbIssueType.on('change', function(){ $fbSubmit.prop('disabled', !$(this).val()); });
      hideFab();
     $('#gra-fb-cancel, #gra-fb-close').on('click', closeFeedbackComposer);
     });
    $fbSubmit.on('click', submitFeedback);
     $cmpSubmit.on('click', submitComment);
    $fbText.on('keydown', function(e){ if(e.key==='Escape') closeFeedbackComposer(); });
     $cmpInput.on('keydown', function(e){
 
      // Ctrl+Enter or Cmd+Enter submits
    /* Note composer */
       if ((e.ctrlKey||e.metaKey) && e.key==='Enter') { submitComment(); }
    $ntInput.on('input', function(){ $ntSubmit.prop('disabled', !$(this).val().trim()); });
       if (e.key==='Escape') { closeCommentComposer(); hideFab(); }
     $('#gra-nt-cancel').on('click', closeNoteComposer);
     $ntSubmit.on('click', submitNote);
     $ntInput.on('keydown', function(e){
       if ((e.ctrlKey||e.metaKey) && e.key==='Enter') submitNote();
       if (e.key==='Escape') closeNoteComposer();
     });
     });


     // ── Bookmark composer ─────────────────────────────────────────
     /* Bookmark composer */
     $( '#gra-bm-cancel' ).on('click', function(){
     $('#gra-bm-cancel').on('click', closeBookmarkComposer);
      closeBookmarkComposer();
      hideFab();
    });
     $bmSubmit.on('click', submitBookmark);
     $bmSubmit.on('click', submitBookmark);
     $bmInput.on('keydown', function(e){
     $bmInput.on('keydown', function(e){
       if (e.key==='Enter') submitBookmark();
       if (e.key==='Enter') submitBookmark();
       if (e.key==='Escape'){ closeBookmarkComposer(); hideFab(); }
       if (e.key==='Escape') closeBookmarkComposer();
     });
     });


     // ── Panel close ───────────────────────────────────────────────
     /* Panel */
     $( '#gra-panel-close' ).on('click', closePanel);
     $('#gra-panel-close').on('click', closePanel);
     $backdrop.on('click', closePanel);
     $backdrop.on('click touchend', function(e){
 
      e.preventDefault();
     // ── Tab switching ─────────────────────────────────────────────
      if ($fbComposer.hasClass('gra-composer-visible')) closeFeedbackComposer();
     $tabComments.on('click', function(){ switchTab('comments'); });
      else if ($ntComposer.hasClass('gra-composer-visible')) closeNoteComposer();
      else if ($bmComposer.hasClass('gra-composer-visible')) closeBookmarkComposer();
      else closePanel();
     });
     $tabNotes.on('click', function(){ switchTab('notes'); });
     $tabBookmarks.on('click', function(){ switchTab('bookmarks'); });
     $tabBookmarks.on('click', function(){ switchTab('bookmarks'); });


    // ── Panel: click comment card → scroll to highlight ───────────
     $paneNotes.on('click', '.gra-note-card', function(e){
     $paneComments.on('click', '.gra-comment-card', function(){
      if ($(e.target).hasClass('gra-note-del')) return;
       var id = $(this).attr('data-gra-id');
       var id = $(this).attr('data-gra-id');
       if (id) scrollToHighlight(id);
       if (id) { closePanel(); scrollToHighlight(id); }
    });
    $paneNotes.on('click', '.gra-note-del', function(e){
      e.stopPropagation();
      var id = $(this).attr('data-del-id');
      if (id) deleteNote(id);
     });
     });
    // ── Panel: click bookmark card → scroll to highlight ──────────
     $paneBookmarks.on('click', '.gra-bookmark-card', function(e){
     $paneBookmarks.on('click', '.gra-bookmark-card', function(e){
       if ($(e.target).hasClass('gra-bookmark-del')) return;
       if ($(e.target).hasClass('gra-bookmark-del')) return;
       var id = $(this).attr('data-gra-id');
       var id = $(this).attr('data-gra-id');
       if (id) scrollToHighlight(id);
       if (id) { closePanel(); scrollToHighlight(id); }
     });
     });
    // ── Panel: delete bookmark ────────────────────────────────────
     $paneBookmarks.on('click', '.gra-bookmark-del', function(e){
     $paneBookmarks.on('click', '.gra-bookmark-del', function(e){
       e.stopPropagation();
       e.stopPropagation();
Line 799: Line 719:
     });
     });


    // ── Click on highlight in text → open panel ───────────────────
     $(CONTENT_SEL).on('click', '.gra-note-highlight', function(){
     $( CONTENT_SEL ).on('click', '.gra-comment-highlight', function(){
       var id = $(this).attr('data-gra-id');
       var id = $(this).attr('data-gra-id');
       openPanel('comments');
       openPanel('notes');
      // After render, scroll to matching card
       setTimeout(function(){
       setTimeout(function(){
         var $card = $paneComments.find('[data-gra-id="'+id+'"]');
         var $card = $paneNotes.find('[data-gra-id="'+id+'"]');
         if ($card.length) {
         if ($card.length) {
           $card.addClass('gra-card-active');
           $card.addClass('gra-card-active');
           $card[0].scrollIntoView({behavior:'smooth',block:'nearest'});
           $card[0].scrollIntoView({behavior:'smooth', block:'nearest'});
           setTimeout(function(){ $card.removeClass('gra-card-active'); }, 2000);
           setTimeout(function(){ $card.removeClass('gra-card-active'); }, 2000);
         }
         }
       }, 100);
       }, 100);
     });
     });
 
     $(CONTENT_SEL).on('click', '.gra-bookmark-highlight', function(){
     $( CONTENT_SEL ).on('click', '.gra-bookmark-highlight', function(){
       var id = $(this).attr('data-gra-id');
       var id = $(this).attr('data-gra-id');
       openPanel('bookmarks');
       openPanel('bookmarks');
       setTimeout(function(){
       setTimeout(function(){
         var $card = $paneBookmarks.find('[data-gra-id="'+id+'"]');
         var $card = $paneBookmarks.find('[data-gra-id="'+id+'"]');
         if ($card.length) $card[0].scrollIntoView({behavior:'smooth',block:'nearest'});
         if ($card.length) $card[0].scrollIntoView({behavior:'smooth', block:'nearest'});
       }, 100);
       }, 100);
     });
     });


    // ── Escape key closes panels / composers ─────────────────────
     $(document).on('keydown', function(e){
     $( document ).on('keydown', function(e){
       if (e.key !== 'Escape') return;
       if (e.key==='Escape'){
      if ($fbComposer.hasClass('gra-composer-visible')) closeFeedbackComposer();
        if ($cmpComposer.hasClass('gra-composer-visible')) { closeCommentComposer(); hideFab(); }
      else if ($ntComposer.hasClass('gra-composer-visible')) closeNoteComposer();
        else if ($bmComposer.hasClass('gra-composer-visible')) { closeBookmarkComposer(); hideFab(); }
      else if ($bmComposer.hasClass('gra-composer-visible')) closeBookmarkComposer();
        else closePanel();
      else closePanel();
      }
     });
     });
   }
   }


  // ════════════════════════════════════════════════════════════════════
   function persistNoteHighlight(id, quote) {
  // RESTORE BOOKMARK HIGHLIGHTS from localStorage on page load
  // ════════════════════════════════════════════════════════════════════
  // Bookmarks store the quote text. We do a best-effort text search
  // in the content to re-wrap the same span after page reload.
  // (Comments are server-stored and we only re-render cards, not re-wrap.)
 
  // ── Persist comment highlight anchors to localStorage ────────────────
  // We only store {id, quote} — the full comment data lives on the wiki.
  // On reload, restoreCommentHighlights re-wraps the quote text in the page
  // so the yellow highlight appears again.
 
   function persistCommentHighlight( id, quote ) {
     try {
     try {
       var stored = JSON.parse( localStorage.getItem( CMT_LS_KEY ) || '[]' );
       var s = JSON.parse(localStorage.getItem(NT_LS_KEY+'_hl')||'[]');
       // Deduplicate
       s = s.filter(function(h){ return h.id !== id; });
      stored = stored.filter( function(h){ return h.id !== id; } );
       s.push({id:id, quote:quote});
       stored.push( { id: id, quote: quote } );
       localStorage.setItem(NT_LS_KEY+'_hl', JSON.stringify(s));
       localStorage.setItem( CMT_LS_KEY, JSON.stringify( stored ) );
     } catch(e){}
     } catch(e){}
   }
   }


   function restoreCommentHighlights() {
   function restoreNoteHighlights() {
     var stored = [];
     var s = [];
     try { stored = JSON.parse( localStorage.getItem( CMT_LS_KEY ) || '[]' ); } catch(e){}
     try { s = JSON.parse(localStorage.getItem(NT_LS_KEY+'_hl')||'[]'); } catch(e){}
     stored.forEach( function(h) {
     s.forEach(function(h){
       if ( !h.quote || !h.id ) return;
       if (!h.quote || !h.id) return;
       if ( document.querySelector( '[data-gra-id="' + h.id + '"].gra-comment-highlight' ) ) return;
       if (document.querySelector('[data-gra-id="'+h.id+'"].gra-note-highlight')) return;
       var needle = h.quote.replace(/…$/, '').trim().slice(0, 80);
       var needle = h.quote.replace(/…$/,'').trim().slice(0,80);
       if ( !needle ) return;
       if (!needle) return;
       var range = findTextInContent( document.querySelector(CONTENT_SEL), needle );
       var range = findTextInContent(document.querySelector(CONTENT_SEL), needle);
       if ( range ) {
       if (!range) return;
        var span = document.createElement('span');
      var sp = document.createElement('span');
        span.className = 'gra-comment-highlight';
      sp.className = 'gra-note-highlight';
        span.setAttribute('data-gra-id', h.id);
      sp.setAttribute('data-gra-id', h.id);
        try { range.surroundContents(span); } catch(e){}
      try { range.surroundContents(sp); } catch(e){}
      }
     });
     });
   }
   }
Line 875: Line 777:
   function restoreBookmarkHighlights() {
   function restoreBookmarkHighlights() {
     _bookmarks.forEach(function(b){
     _bookmarks.forEach(function(b){
       if ( !b.quote ) return;
       if (!b.quote) return;
      // Already highlighted from this session
       if (document.querySelector('[data-gra-id="'+b.id+'"].gra-bookmark-highlight')) return;
       if ( document.querySelector('[data-gra-id="'+b.id+'"].gra-bookmark-highlight') ) return;
      // Text search — find first occurrence of quote in content
      var contentEl = document.querySelector( CONTENT_SEL );
      if (!contentEl) return;
       var needle = b.quote.replace(/…$/,'').trim().slice(0,60);
       var needle = b.quote.replace(/…$/,'').trim().slice(0,60);
       if (!needle) return;
       if (!needle) return;
      // Walk text nodes to find a match
       var found = findTextInContent(document.querySelector(CONTENT_SEL), needle);
       var found = findTextInContent( contentEl, needle );
       if (!found) return;
       if (found) {
      var sp = document.createElement('span');
        var span = document.createElement('span');
      sp.className = 'gra-bookmark-highlight';
        span.className = 'gra-bookmark-highlight';
      sp.setAttribute('data-gra-id', b.id);
        span.setAttribute('data-gra-id', b.id);
      sp.setAttribute('data-gra-name', b.name);
        span.setAttribute('data-gra-name', b.name);
      try { found.surroundContents(sp); } catch(e){}
        try {
          found.surroundContents(span);
        } catch(e){}
      }
     });
     });
   }
   }


   function findTextInContent( root, needle ) {
   function findTextInContent(root, needle) {
     // Returns a Range covering the first occurrence of needle in root's text
     if (!root || !needle) return null;
     var text = root.textContent || '';
     var text = root.textContent || '';
     var idx  = text.indexOf(needle);
     var idx  = text.indexOf(needle);
     if (idx < 0) return null;
     if (idx < 0) return null;
    // Walk to find the exact text nodes
     var iter = document.createNodeIterator(root, NodeFilter.SHOW_TEXT, null, false);
     var iter = document.createNodeIterator(root, NodeFilter.SHOW_TEXT, null, false);
     var pos = 0;
     var pos = 0, node, startNode, startOffset, endNode, endOffset;
    var node, startNode, startOffset, endNode, endOffset;
     while ((node = iter.nextNode())) {
     while ((node = iter.nextNode())) {
       var len = node.nodeValue.length;
       var len = node.nodeValue.length;
       if (!startNode && pos+len > idx) {
       if (!startNode && pos + len > idx) { startNode = node; startOffset = idx - pos; }
        startNode   = node;
        startOffset = idx - pos;
      }
       var endIdx = idx + needle.length;
       var endIdx = idx + needle.length;
       if (startNode && pos+len >= endIdx) {
       if (startNode && pos + len >= endIdx) { endNode = node; endOffset = endIdx - pos; break; }
        endNode   = node;
        endOffset = endIdx - pos;
        break;
      }
       pos += len;
       pos += len;
     }
     }
     if (!startNode || !endNode) return null;
     if (!startNode || !endNode) return null;
     var range = document.createRange();
     try {
    range.setStart(startNode, startOffset);
      var r = document.createRange();
    range.setEnd(endNode, endOffset);
      r.setStart(startNode, startOffset);
     return range;
      r.setEnd(endNode, endOffset);
      return r;
     } catch(e){ return null; }
   }
   }


   // ════════════════════════════════════════════════════════════════════
   $(function() {
  // BOOT
    _mobile = window.innerWidth < 768 || 'ontouchstart' in window;
  // ════════════════════════════════════════════════════════════════════
    window.addEventListener('resize', function(){
 
      _mobile = window.innerWidth < 768 || 'ontouchstart' in window;
  $( function () {
    });
     buildDom();
     buildDom();
     wireEvents();
     wireEvents();
    loadNotes();
     loadBookmarks();
     loadBookmarks();
     restoreBookmarkHighlights();
     setTimeout(function(){
    restoreCommentHighlights();
       try { restoreNoteHighlights(); } catch(e){}
    // Pre-load comment count in background
      try { restoreBookmarkHighlights(); } catch(e){}
    loadComments(function(){
     }, 500);
       // Update tab label with count
   });
      if (_comments.length > 0) {
        $tabComments.find('span.gra-icon').after(
          ' <span style="background:#e53935;color:#fff;border-radius:9px;' +
          'font-size:10px;padding:0 5px;margin-left:2px;">' + _comments.length + '</span>'
        );
        // Update toggle badge
        var $badge = $( '#gra-toggle-badge' );
        if ( $badge.length ) $badge.text( _comments.length ).css('display','flex');
      }
     });
   } );


}() );
}() );