MathOverflow/MSE Citation Helper

Improved citation finder for MathOverflow.

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         MathOverflow/MSE Citation Helper
// @version      2.2.2
// @description  Improved citation finder for MathOverflow.
// @author       Asaf Karagila, aspen138
// @match        https://mathoverflow.net/*
// @match        https://math.stackexchange.com/*
// @grant        none
// @inject-into  page
// @icon         
// @namespace https://greasyfork.org/users/1177387
// ==/UserScript==

const $ = window.jQuery;
const StackExchange = window.StackExchange;
const MathJax = window.MathJax;
const CH = StackExchange.citationHelper;

(function() {
    'use strict';

    StackExchange.citationHelper = (function () {

        var currentResult = false;

        return { init: init }; // Hoisting!

        function thingFinder(markdown, textareaContent, sentinel, thingName, things) {
            return markdown;
        }

        function init() {
            if (StackExchange.externalEditor) {
                StackExchange.externalEditor.init({
                    thingName: 'cite',
                    thingFinder: thingFinder,
                    buttonTooltip: 'Insert Citation 2.0',
                    buttonImageUrl: '/content/Sites/mathoverflow/img/cite-icon.svg',
                    onShow: function (editorClosed) {
                        $('#search-text').on('blur', runSearch).on('keypress', runSearchKey).focus();
                        $('#backlink').on('click', goBack);
                        currentResult = false;
                        $('#popup-cite .popup-submit').on('click', function () { if (currentResult) { listenMessage(currentResult); } });
                        window.addCitationHelper = editorClosed;

                        if (window.addEventListener) {
                            window.addEventListener('message', listenMessage, false);
                        } else {
                            window.attachEvent('onmessage', listenMessage);
                        }
                    },
                    onRemove: function () {
                        window.addCitationHelper = null;
                        try {
                            delete window.addCitationHelper;
                        } catch (e) { } // IE doesn't allow deleting from window
                    },
                    getDivContent: function (oldContent) {
                        return searchDialog();
                    }
                });
            }
        }

        // Prepare the search dialog
        function searchDialog() {

            if ($('.popup-cite').length > 0) { return; } // Abort if dialog already exists

            // Updated HTML with BibTeX container
            var popupHTML = '<div id="popup-cite" class="popup"><div class="popup-close"><a title="close this popup (or hit Esc)" href="javascript:void(0)">&times;</a></div><h2 class="popup-title-container handle"> <span class="popup-breadcrumbs"></span><span class="popup-title">Insert citation 2.0</span></h2><div id="pane-main" class="popup-pane popup-active-pane close-as-duplicate-pane" data-title="Insert Citation 2.0" data-breadcrumb="Cite"><input id="search-text" class="js-duplicate-search-field" type="text" style="width: 740px; z-index: 1; position: relative;"><div class="search-errors search-spinner"></div> <div class="original-display"> <div id="previewbox" style="display:none"><div><a href="javascript:void(0)" id=backlink>&lt; Back to results</a></div><div class="preview" ></div><div class="bibtex-container" style="margin-top:15px;"><h3 style="margin-bottom:8px;">BibTeX:</h3><textarea id="bibtex-output" readonly style="width:100%;min-height:120px;font-family:Consolas,Monaco,monospace;font-size:11px;padding:10px;border:1px solid #ccc;background-color:#f9f9f9;resize:vertical;"></textarea><button id="copy-bibtex" style="margin-top:8px;padding:6px 12px;cursor:pointer;background:#0095ff;color:white;border:none;border-radius:3px;">Copy BibTeX</button></div></div> <div class="list-container"> <div class="list-originals" id="results"> </div> </div> </div></div><div class="popup-actions"><input type="submit" id="cite-submit" class="popup-submit disabled-button" value="Insert Citation" disabled="disabled" style="cursor: default;"></div></div>';

            return popupHTML;
        }

        // The event handler for the message
        function listenMessage(msg) {
            window.addCitationHelper(getCitationHtml(msg), '\n\n' + getCitationHtml(msg));
            StackExchange.MarkdownEditor.refreshAllPreviews();
            StackExchange.helpers.closePopups();
        }

        function runSearchKey(e) {
            var key = (e.keyCode ? e.keyCode : e.which);
            if (key == 13) {
                runSearch();
            }
        }

        // Run a search
        function runSearch() {
            goBack();
            $('#popup-cite .search-spinner').removeSpinner().addSpinner();
            $.getJSON('https://zbmath.org/citationmatching/mathoverflow/v2', { 'q': $('#search-text').val() }, fetchCallback);
        }

        // Callback to run when search completes
        function fetchCallback(response) {
            var html = $('<div class="list">');
            response.results.forEach(res => {
                var zbl = res.zbl_link;

                var doi = res.external_ids.doi && res.external_ids.doi[0] ? "https://doi.org/"+res.external_ids.doi[0] : "";
                var arxiv = res.external_ids.arxiv && res.external_ids.arxiv[0] ? "https://arxiv.org/abs/"+res.external_ids.arxiv[0] : "";
                var authors = sanitizeForDisplay(res.authors.reduce((acc,val,i,arr) => acc + fixName(val) + (i+1 == arr.length ? "" : i+2 == arr.length ? " and " : ", "),""));
                var title = sanitizeForDisplay(res.title).replaceAll(/\\[\(\)\[\]]/gm,"$");
                var serial = res.serial ? res.serial : null;
                var source = serial ? sanitizeForDisplay("<i>"+(serial.short_title ? serial.short_title : serial.title)+"</i> <b>"+serial.volume+"</b>" + (serial.issue ? " ("+serial.issue+")" : "") + ", " +res.pagination + " (" + res.year+")") : res.source;
                var citationHtml = sanitizeForDisplay(source);

                var outputJSON = {
                    authors: authors,
                    title: title,
                    source: source,
                    zbl: zbl,
                    doi: doi,
                    arxiv: arxiv,
                    zbl_id: res.zbl_id,
                    // Store original data for BibTeX generation
                    originalData: res
                };

                var renderResult = $('<div class="item" style="float:none;padding:5px">')
                .html($('<div class="summary post-link" style="float:none;width:auto;font-weight:bold;">')
                      .text(title))
                .append('<br/>')
                .append($('<span class="body-summary" style="float:none"></span>')
                        .append(authors + '<br/>' + citationHtml + '<br/> Preview (opens in new tab): ')
                        .append(renderOptionalLink(doi, 'article'))
                        .append(renderOptionalLink(zbl, 'zbmath'))
                        .append(renderOptionalLink(arxiv, 'arxiv'))
                       )
                .click(loadResultCallback(doi, outputJSON))
                .hover(function () { $(this).css('background-color', '#e6e6e6') }, function () { $(this).css('background-color', '#fff') });

                html.append(renderResult);
                renderResult.find('a').on('click', function (e) { e.stopPropagation(); });
            })

            $('#results').html('').append(html);
            MathJax.Hub.Queue(['Typeset', MathJax.Hub, 'results']);
            $('#popup-cite .search-spinner').removeSpinner();
        }

        function fixName(name) {
            return name.split(",").reverse().reduce((a,v,i) => a+(i > 0 ? " " : "")+v.trim(),"")
        }

        function sanitizeForDisplay(html) {
            return StackExchange.MarkdownEditor.sanitizeHtml(html);
        }

        function renderOptionalLink(href, text) {
            if (href) {
                return $(sanitizeForDisplay($('<a>').attr('href', href).text(text + ' ').prop('outerHTML'))).attr('target', '_blank');
            } else {
                return '';
            }
        }

        function loadResultCallback(href, result) {
            return function (e) { e.preventDefault(); e.stopPropagation(); loadResult(e, href, result); return false; }
        }

        function loadResult(e, href, result) {
            $('#popup-cite .popup-submit').enable();
            currentResult = result;
            $('.list-container').hide();
            $('#popup-cite #previewbox').show();
            $('#popup-cite .preview').html($(e.target).closest('.item').html());

            // Generate and display BibTeX
            generateBibtex(result);
        }

        function generateBibtex(result) {
            var bibtex = '';
            var data = result.originalData;

            // Generate citation key from first author and year
            var firstAuthor = data.authors && data.authors[0] ? data.authors[0].split(',')[0].trim() : 'Unknown';
            var year = data.year || 'n.d.';
            var citeKey = firstAuthor.replace(/[^a-zA-Z]/g, '') + year;

            // Determine entry type
            var entryType = '@article';
            if (data.document_type) {
                if (data.document_type.toLowerCase().includes('book')) {
                    entryType = '@book';
                } else if (data.document_type.toLowerCase().includes('proceeding')) {
                    entryType = '@inproceedings';
                }
            }

            bibtex += entryType + '{' + citeKey + ',\n';

            // Author
            if (data.authors && data.authors.length > 0) {
                var authorList = data.authors.map(function(a) {
                    return a; // Authors are already in "Last, First" format
                }).join(' and ');
                bibtex += '  author = {' + authorList + '},\n';
            }

            // Title
            if (data.title) {
                bibtex += '  title = {' + data.title.replace(/\$/g, '$') + '},\n';
            }

            // Journal/Serial info
            if (data.serial) {
                if (data.serial.title) {
                    bibtex += '  journal = {' + data.serial.title + '},\n';
                }
                if (data.serial.volume) {
                    bibtex += '  volume = {' + data.serial.volume + '},\n';
                }
                if (data.serial.issue) {
                    bibtex += '  number = {' + data.serial.issue + '},\n';
                }
            }

            // Pages
            if (data.pagination) {
                bibtex += '  pages = {' + data.pagination + '},\n';
            }

            // Year
            if (data.year) {
                bibtex += '  year = {' + data.year + '},\n';
            }

            // DOI
            if (result.doi) {
                var doiStr = result.doi.replace('https://doi.org/', '');
                bibtex += '  doi = {' + doiStr + '},\n';
            }

            // arXiv
            if (result.arxiv) {
                var arxivId = result.arxiv.replace('https://arxiv.org/abs/', '');
                bibtex += '  eprint = {' + arxivId + '},\n';
                bibtex += '  archivePrefix = {arXiv},\n';
            }

            // Zbl number
            bibtex += '  zbl = {' + result.zbl_id + '},\n';
            bibtex += '  url = {' + result.zbl + '}\n';
            bibtex += '}';

            $('#bibtex-output').val(bibtex);

            // Setup copy button
            $('#copy-bibtex').off('click').on('click', function() {
                var $textarea = $('#bibtex-output');
                $textarea[0].select();
                $textarea[0].setSelectionRange(0, 99999); // For mobile

                try {
                    document.execCommand('copy');
                    var $btn = $(this);
                    var originalText = $btn.text();
                    $btn.text('Copied!').css('background', '#5eba7d');
                    setTimeout(function() {
                        $btn.text(originalText).css('background', '#0095ff');
                    }, 2000);
                } catch (err) {
                    alert('Failed to copy. Please select and copy manually.');
                }
            });
        }

        function goBack() {
            $('#popup-cite .search-spinner').removeSpinner();
            $('.list-container').show();
            $('#popup-cite #previewbox').hide();
            $('#popup-cite .popup-submit').disable();
        }

        function getCitationHtml(json) {
            var link = json.doi ? json.doi : json.arxiv ? json.arxiv : json.zbl;
            var cite = $('<cite>').attr('authors', json.authors)
            .append('_' + json.authors + '_, ')
            .append('[**' + json.title + '**](' + encodeURI(link) + ')' + (".,;".includes(json.title.slice(-1)) ? " " : ". "))
            .append(json.source + ' ([ZBL' + json.zbl_id + ']('+encodeURI(json.zbl) + ')' + (json.arxiv ? ', [arXiv:'+json.arxiv.substring(22)+']('+encodeURI(json.arxiv)+')' : '')+')')
            .append('.');

            var citeContainer = $('<span></span>').append(cite).html();

            return citeContainer;
        }
    })(); //end function call

    StackExchange.using('editor', function (){
        StackExchange.using('externalEditor', function (){
            if (CH == undefined) StackExchange.citationHelper.init();
            console.log("Citation Helper v2 is loaded!");
        });
    });})();