Skip to content

Instantly share code, notes, and snippets.

Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save joeytwiddle/1079652 to your computer and use it in GitHub Desktop.
Save joeytwiddle/1079652 to your computer and use it in GitHub Desktop.
Delicious search results on Google
// ==UserScript==
// @name Delicious Search Results on Google
// @namespace gaillard
// @description Shows results from Delicious on Google search pages
// @include http://www.google.*/search?*q=*
// @include http://www.google.*/*
// @date 2009-03-07
// @version 0.1
// @GM_version 0.8.20080609.0
//
// 99% of the code for this user script comes form markcarey's userscript 'Twitter Search Results on Google' (http://userscripts.org/scripts/show/43451)
// ==/UserScript==
// This wasn't working for me: @require http://hwi.ath.cx/javascript/xhr_via_json/xss_xhr_via_jsonp.js
// DONE: Don't keep spamming Delicious when we go forwards and backwards to Google again.
// Cache and re-use the results if they are recent.
// TODO: Sometimes you click on a Google result before these results have
// appeared, but the browser applies the click *after* it appeared. How can we
// work around this?
// 1) After parsing and constructing the results (a slightly heavy task),
// pause a bit (with a setTimout closure) before presenting, to allow the
// browser a chance to process any click events.
// 2) Add the results immediately but with their click actions disabled.
// Enable them later, after a timeout.
//// Lookalike GM_xhR for this script only, actually requests JSONP directly from delicious.
// We always replace Google Chrome's GM_xmlhttpRequest because it is not cross-site.
if (!this.GM_xmlhttpRequest || window.navigator.vendor.match(/Google/)) {
GM_xmlhttpRequest = function(details) {
// Insert a callback into the root window, anonymised by a random key.
var callbackName = "xss_xhr_via_jsonp_callback_" + parseInt(Math.random()*987654321);
// Where should we place the callback? Chrome Userscripts need to obtain unsafeWindow.
var target = window;
var weAreInUserscriptScope = (typeof GM_log != 'undefined');
if (window.navigator.vendor.match(/Google/) && weAreInUserscriptScope) {
var div = document.createElement("div");
div.setAttribute("onclick", "return window;");
unsafeWindow = div.onclick();
target = unsafeWindow;
}
var script = document.createElement("script");
var callbackFunction = function(dataObj) {
var responseDetails = {};
responseDetails.responseText = JSON.stringify(dataObj);
details.onload(responseDetails);
// Cleanup artifacts: script and callback function
delete target[callbackName];
document.getElementsByTagName("head")[0].removeChild(script);
};
target[callbackName] = callbackFunction;
// Request a JSONP response from delicious, which should return some javascript to call the callback.
script.src = details.url + "?callback="+callbackName;
document.getElementsByTagName("head")[0].appendChild(script);
};
}
// DONE: scripts and functions added to the DOM should be removed after use
if (!this.GM_getValue) {
GM_getValue = function(name,def) {
return def;
};
}
GM_DUR = {
un : "",
init : function()
{
var href = document.location.href;
try {
GM_DUR.un = href.match(/[&?]q=([^&]*)(?:&|$)/)[1];
} catch (e) {
var child = (window.top == window ? '' : 'child frame ' );
GM_log("Failed to find search string in "+child+""+href+" - "+e);
}
if( GM_DUR.un != "" )
{
// Can we re-use cached results?
var cached_un = GM_getValue("cached_un");
if (cached_un == GM_DUR.un) {
GM_log("Re-using cached Delicious response.");
var response = ({});
response.responseText = GM_getValue("cached_responseText");
GM_DUR.handle(response);
} else {
GM_log("Making a fresh request to "+GM_DUR.un);
GM_xmlhttpRequest({
method:"GET",
url:"http://feeds.delicious.com/v2/json/tag/"+GM_DUR.un,
headers:{
"User-Agent":"Mozilla/5.0",
"Accept":"text/json"
},
onload:GM_DUR.handle
});
}
}
},
handle : function(response)
{
// GM_log("Delicious response text: "+response.responseText.substring(0,100));
GM_log("Delicious responded with "+response.responseText.length+" bytes of data for "+GM_DUR.un);
var results = eval("("+response.responseText+")");
GM_setValue("cached_un",GM_DUR.un);
GM_setValue("cached_responseText",response.responseText);
var ds = document.createElement("ol");
var deliciousImageHTML = '<img src="" alt="" height="42" width="43">';
if(results.length > 0) {
//// Breaks in FF3.5:
// ds.style = "position: absolute; right: 10px;";
//// Works but not well with Google Preview Pane:
// ds.style.position = 'absolute';
// ds.style.right = '10px';
var il, h;
var query = unescape(GM_DUR.un).replace(/\+/g, ' ');
h = ds.appendChild(document.createElement("li"));
h.className = "g";
var h3 = h.appendChild(document.createElement("h3"));
h3.className = "r";
h3.innerHTML = "<a href='http://delicious.com/search?p=" + GM_DUR.un +"'>Delicious results for <em>"+ query +"</em></a>";
var t = h.appendChild(document.createElement("table"));
t.className = "ts";
var tb = t.appendChild(document.createElement("tbody"));
var row = tb.appendChild(document.createElement("tr"));
var newHTML = '<td style="padding-top: 5px; padding-right: 10px; font-size: 78%; line-height: normal; width: 43px; text-align: center;" valign="top">'+deliciousImageHTML+'</td><td style="padding-top: 3px;" valign="top">';
for( var i=0; i < 5; i++ ) {
if (i >= results.length) {
continue;
}
// var fixedDate = results[i].dt.replace(/-/g,'/').replace(/Z$/,'').replace(/T/,' ');
// // GM_log("fixedDate="+fixedDate);
// var ageMonths = (new Date() - new Date(fixedDate)) / 1000 / 60 / 60 / 24 / 30;
// // GM_log("ageMonths="+ageMonths);
// /* My approximation of popularity, max 2400 */
// var pop = parseInt(2400 / Math.floor(1,ageMonths));
// // GM_log("pop="+pop);
// ageMonths = parseInt(ageMonths*10)/10;
// var pop = 100;
var tit = ( results[i].n ? "title='"+results[i].n.replace(/'/,'&apos;','g')+"'" : "" );
il = "<div class='s'>"+
// "<div style='position:relative; background-color=#00cccc; z-index:-100; height=10px; width="+pop+"px;'/>"+
// ageMonths+"m "+
"<a "+tit+" href='"+results[i].u+"' class='l'>"+
results[i].d+"</a> "+
' <span class="f"><FONT size="-1">'+ (""+results[i].t).replace(/,/g,', ') +'</FONT></span></div>';
newHTML += il;
}
newHTML += '</td>';
row.innerHTML = newHTML;
} else {
// ds.innerHTML = deliciousImageHTML;
ds.appendChild(document.createTextNode("No Delicious results"));
}
var results_section = document.getElementById("res");
results_section.insertBefore(ds, results_section.firstChild);
},
};
//// This appears to work in Konqueror but not Firefox O_o
// document.addEventListener('DOMContentLoaded',GM_DUR.init,false);
GM_DUR.init();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment