MathOverflow/MSE Citation Helper

Improved citation finder for MathOverflow.

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 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!");
        });
    });})();