Improved citation finder for MathOverflow.
// ==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)">×</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>< 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!");
});
});})();