// ==UserScript==
// @name History of the Seen
// @namespace https://github.com/theoky/HistoryOfTheSeen
// @description Script to implement a history of the seen approach for some news sites. Details at https://github.com/theoky/HistoryOfTheSeen
// @author Theoky
// @version 0.4192
// @lastchanges workaround for bug in GreaseMonkey 3.2
// @license GNU GPL version 3
// @released 2014-02-20
// @updated 2014-06-10
// @homepageURL https://github.com/theoky/HistoryOfTheSeen
//
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_registerMenuCommand
// @grant GM_listValues
// @grant GM_addStyle
//
// for testing purposes (set FireFox greasemonkey.fileIsGreaseable)
// @include file://*testhistory.html
//
// @include http*://*.derstandard.at/*
// @include http*://*.faz.net/*
// @include http*://*.golem.de/*
// @include http*://*.handelsblatt.com/*
// @include http*://*.heise.de/newsticker/*
// @include http*://*.kleinezeitung.at/*
// @include http*://*.nachrichten.at/*
// @include http*://*.oe24.at/*
// @include http*://*.orf.at/*
// @include http*://orf.at/*
// @include http*://*.reddit.com/*
// @include http*://*.spiegel.de/*
// @include http*://*.sueddeutsche.de/*
// @include http*://*.welt.de/*
// @include http*://*.wirtschaftsblatt.at/*
// @include http*://*.zeit.de/*
// @include http*://dastandard.at/*
// @include http*://derstandard.at/*
// @include http*://diepresse.com/*
// @include http*://diestandard.at/*
// @include http*://kurier.at/*
// @include http*://slashdot.org/*
// @include http*://taz.de/*
// @include http*://notalwaysright.com/*
// @include http*://www.nytimes.com/*
// @require http://code.jquery.com/jquery-2.1.1.min.js
// @require http://code.jquery.com/ui/1.11.2/jquery-ui.js
// @require https://greasyfork.org/scripts/130-portable-md5-function/code/Portable%20MD5%20Function.js?version=10066
// was require md5.js
// was require http://crypto-js.googlecode.com/svn/tags/3.1.2/build/rollups/md5.js
// ==/UserScript==
// Copyright (C) 2015 T. Kopetzky - theoky
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
//
// Tested with FireFox 34 and GreaseMonkey 2.3
//-------------------------------------------------
//Functions
(function(){
var defaultSettings = {
ageOfUrl: 5, // age in days after a url is deleted from the store
// < 0 erases all dates (disables history)
targetOpacity: 0.3,
targetOpacity4Dim: 0.85,
steps: 10,
dimInterval: 30000,
expireAllDomains: true, // On fast machines this can be true and expires
// all domains in the database with each call. If false,
// only the urls of the current domain are expired which
// is slightly faster.
cleanOnlyDaily: true,
considerViewPort: true,
dbOpsPerRun: 5
};
var UNDEF = 'undefined';
var DEFAULT_TAG = 'a';
var defaultGetContentFct = function(elem) {
if ((typeof elem != UNDEF) && (typeof elem.href != UNDEF)) {
return elem.href;
}
return UNDEF;
};
var AFTER_SCROLL_DELAY = 750;
var DO_DEBUG = false;
var DEBUG_LVL_ERROR = 1;
var DEBUG_LVL_WARN = 2;
var DEBUG_LVL_INFO = 4;
var perUrlSettings = [
{
url : ['.*\.?slashdot\.org' ],
tag : 'article',
upTrigger: "../article",
getContent: function(elem) {
if ((typeof elem != UNDEF) && (typeof elem.id != UNDEF)) {
return elem.id;
}
return UNDEF;
},
parentHints : [ ]
},
{
url : ['.*\.?derstandard\.at', '.*\.?diestandard\.at', '.*\.?dastandard\.at' ],
upTrigger: "../a",
parentHints : [
"ancestor::div[contains(concat(' ', @class, ' '), ' text ')]",
"ancestor::ul[@class='stories']" ]
},
{
url : ['notalwaysright\.com'],
upTrigger: "../a[@rel='bookmark']",
parentHints : [ "ancestor::div[contains(concat(' ', @class, ' '), ' post ')]" ]
},
{
url : ['.*\.?golem.de'],
upTrigger: "../a",
parentHints : [ "ancestor::li",
"ancestor::section[@id='index-promo']",
"ancestor::section[contains(concat(' ', @class, ' '), ' promo ')]" ]
},
{
url : ['.*\.?reddit.com'],
// class="title may-blank srTagged imgScanned"
upTrigger: "../a[contains(@class, 'title') and contains(@class, 'may-blank')]",
parentHints : [ "ancestor::div[contains(concat(' ', @class, ' '), ' thing ')]" ]
},
{
url : ['nytimes\.com'],
upTrigger: "../a",
parentHints : [
// "ancestor::li[contains(concat(' ', @class, ' '), ' portal-post ')]",
"ancestor::div[contains(concat(' ', @class, ' '), ' collection ')]"
]
}
];
var dimMap = {};
var countDownTimer = defaultSettings.steps;
var theHRefs = null;
var curSettings = null;
var KEY_LAST_EXPIRE_OP = "lastExpire";
var timeOutAfterLastScroll = UNDEF;
var tag2Process = null;
var getContentFct = null;
var theDomain = null;
var progressbar;
var progressLabel;
// Styling
var progressBarStyle =
".ui-widget {" +
" font-family: Verdana,Arial,sans-serif !important;" +
" font-size: 1.1em !important;" +
"}" +
".ui-widget-content {" +
" border: 1px solid #aaaaaa !important;" +
" color: #222222 !important;" +
"}" +
".ui-widget-header {" +
" border: 1px solid #aaaaaa !important;" +
" background: #cccccc !important;" +
" color: #222222 !important;" +
" font-weight: bold !important;" +
"}" +
".ui-progressbar {" +
" height: 2em !important;" +
" text-align: left !important;" +
" overflow: hidden !important;" +
"}" +
".ui-progressbar .ui-progressbar-value {" +
" margin: -1px !important;" +
" height: 100% !important;" +
"}" +
".ui-progressbar .ui-progressbar-overlay {" +
" background: url('') !important;" +
" height: 100% !important;" +
" filter: alpha(opacity=25) !important; /* support: IE8 */" +
" opacity: 0.25 !important;" +
"}" +
// ".ui-progressbar-indeterminate .ui-progressbar-value {" +
// " background-image: none !important;" +
// "}" +
".ui-progressbar {" +
" height: 2em !important;" +
" text-align: left !important;" +
" overflow: hidden !important;" +
" position: absolute !important;" +
" left: 20% !important;" +
" top: 4px !important;" +
" width: 60% !important;" +
" z-index: 255 !important;" +
"}" +
".progress-label {" +
" position: absolute !important;" +
" left: 5% !important;" +
" top: 4px !important;" +
" font-weight: bold !important;" +
" text-shadow: 1px 1px 0 #fff !important;" +
" z-index: 256 !important;" +
"}";
// Debugging
function debuglog(msg) {
if (DO_DEBUG) {
console.log(msg);
}
}
function debugLogLvl(lvl, msg) {
if (lvl & DEBUG_LVL_ERROR) {
console.log("error: " + msg);
}
if (DO_DEBUG) {
if (lvl & DEBUG_LVL_WARN) {
console.log("warn:" + msg);
}
if (lvl & DEBUG_LVL_INFO) {
console.log(msg);
}
}
}
var g_index;
var g_keys;
var g_lengthOfKeysArray;
var g_workInProgress = false;
var g_par1 = UNDEF;
var g_par2 = UNDEF;
var g_workerFctDefault = function(key, par1, par2) {
GM_deleteValue(key);
};
var g_workerFct = g_workerFctDefault;
var g_finishFct_Default = function() {
document.location.reload(true);
};
var g_finishFct = g_finishFct_Default;
var g_label;
function appendProgressBar() {
$("body").append ( '\
<div id="progressbar" class="ui-progressbar ui-progressbar-indeterminate"><div class="progress-label">History of the Seen: Resetting DB for current domain...</div></div>');
}
function removeProgressBar(reload) {
$("#progressbar").remove();
if (reload) {
document.location.reload(true);
}
}
/*
* Init function for "threading"
*/
function initThreadingLoop()
{
if (g_workInProgress) {
debugLogLvl(DEBUG_LVL_ERROR, "initThreading with already threading in progress.");
return;
}
g_workInProgress = true;
g_index = 0;
g_keys = GM_listValues();
if (!g_keys) {
debugLogLvl(DEBUG_LVL_WARN, "g_keys empty?");
return;
}
g_lengthOfKeysArray = g_keys.length;
appendProgressBar();
progressbar = $("#progressbar");
progressLabel = $(".progress-label");
progressbar.progressbar({
value : false,
change : function() {
progressLabel.text(g_label + progressbar.progressbar("value").toFixed(2) + "% ");
},
complete : function() {
progressLabel.text(" History of the Seen: Operation Complete! ");
}
});
progressbar.progressbar("value", 0);
setTimeout(doThreadWork, 1);
}
/*
* Worker method
*/
function doThreadWork()
{
if (!g_workInProgress) {
return;
}
var i = 0;
var currentKey = null;
currentKey = g_keys[g_index];
while (i < defaultSettings.dbOpsPerRun && currentKey) {
g_workerFct(currentKey, g_par1, g_par2);
g_index ++;
i++;
currentKey = g_keys[g_index];
}
progressbar.progressbar("value", g_index * 100 / g_lengthOfKeysArray);
if (currentKey) {
setTimeout(doThreadWork, 10);
} else
{
removeProgressBar(false);
if (g_finishFct !== UNDEF) {
g_finishFct();
}
g_workInProgress = false;
}
}
// Resetting section
function resetAllUrls() {
if (!g_workInProgress && confirm('Are you sure you want to erase the complete seen history?')) {
g_label = " History of the Seen: Cleaning DB, done ";
g_par1 = UNDEF;
g_par2 = UNDEF;
g_workerFct = g_workerFctDefault;
g_finishFct = g_finishFct_Default;
initThreadingLoop();
}
}
function resetUrlsForCurrentHelper(dKey, domainOrUri) {
if (confirm('Are you sure you want to erase the seen history for ' +
domainOrUri + '?')) {
g_label = " History of the Seen: Cleaning DB, done ";
g_par1 = dKey;
g_par2 = domainOrUri;
g_workerFct = function(key, dKey, domainOrUri) {
if (key == KEY_LAST_EXPIRE_OP){
return;
}
try {
var val = GM_getValue(key, "{}");
var dict = JSON.parse(val);
if(dict) {
if (dict[dKey] == domainOrUri) {
GM_deleteValue(key);
}
}
} catch (e) {
console.log(e);
}
};
g_finishFct = g_finishFct_Default;
initThreadingLoop();
}
}
function resetUrlsForCurrentDomain() {
resetUrlsForCurrentHelper("domain", document.domain);
}
function resetUrlsForCurrentSite() {
resetUrlsForCurrentHelper("base", document.baseURI);
}
function expireUrls() {
debugLogLvl(DEBUG_LVL_INFO, "expireUrls");
if (defaultSettings.cleanOnlyDaily) {
var lastExpireDate = new Date(GM_getValue(KEY_LAST_EXPIRE_OP, nDaysOlderFromNow(2)));
var diff = Math.abs((new Date()) - lastExpireDate);
if (diff / 1000 / 3600 / 24 < 1) {
// less than one day -> no DB cleaning
return;
}
}
/*
var val = GM_getValue(KEY_EXPIRE_OP_INPROGRESS);
if (typeof val !== UNDEF) {
// expire in progress
return;
}
GM_setValue(KEY_EXPIRE_OP_INPROGRESS, True);
*/
// cutOffDate
g_label = " History of the Seen: Expiring old URLs for this site, done ";
g_par1 = nDaysOlderFromNow(defaultSettings.ageOfUrl);
debuglog("cutOffDate" + g_par1);
g_par2 = UNDEF;
g_workerFct = function(key, cutOffDate, par2) {
if (key == KEY_LAST_EXPIRE_OP){
return;
}
var dict = JSON.parse(GM_getValue(key, "{}"));
if(dict) {
try {
debuglog(dict["domain"], cutOffDate.getTime(), dict["date"]);
if (cutOffDate.getTime() > dict["date"]) {
if (defaultSettings.expireAllDomains ||
(dict["domain"] == document.domain))
{
GM_deleteValue(key);
}
}
} catch (e) {
console.log(e);
}
}
else {
console.log('Error! JSON.parse failed - dict is likely to be corrupted. Probably best to completely clean DB.');
}
};
g_finishFct = function() {
GM_setValue(KEY_LAST_EXPIRE_OP, new Date());
}
initThreadingLoop();
}
function nDaysOlderFromNow(age, aDate, zeroHour) {
var aDate = typeof aDate !== UNDEF ? aDate : new Date();
var zeroHour = typeof zeroHour !== UNDEF ? zeroHour : true;
var dateStore = new Date(aDate.getTime());
var workDate = aDate;
if (age >= 0) {
workDate.setDate(dateStore.getDate() - age);
if (zeroHour) {
workDate.setHours(0,0,0,0);
}
}
return workDate;
}
/*
* Find the settings for a given URL
*/
function findPerUrlSettings(theSettings, aDomain) {
debugLogLvl(DEBUG_LVL_INFO, "findPerUrlSettings");
for (var i=0; i < theSettings.length; ++i) {
for (var j = 0; j < theSettings[i].url.length; ++j) {
var myRegExp = new RegExp(theSettings[i].url[j], 'i');
if (aDomain.match(myRegExp)) {
return theSettings[i];
}
}
}
}
/*
* Find the parent element as specified in the settings.
*/
function locateParentElem(curSettings, aDomain, aRoot) {
if (!curSettings) {
return null;
}
// console.log("locateParentElem 1", curSettings.url);
var res = null;
for (var xpath = 0; xpath < curSettings.parentHints.length; ++xpath) {
// console.log("locateParentElem 2", curSettings.parentHints[xpath], aRoot);
res = document.evaluate(curSettings.parentHints[xpath], aRoot, null, 9, null).singleNodeValue;
if (res) {
// console.log("locateParentElem found something");
return res;
}
}
return res;
}
/*
* Check if the current node qualifies for looking up the hierarchy.
*/
function goUp(curSettings, aRoot) {
if (!curSettings) {
return false;
}
var res = null;
if (curSettings.upTrigger !== "") {
res = document.evaluate(curSettings.upTrigger, aRoot, null, 9, null).singleNodeValue;
}
return res !== null;
}
/*
* Set the opacity for specified links
*/
function dimLinks() {
var interval = (1 - defaultSettings.targetOpacity4Dim)/defaultSettings.steps;
var countDownTimer = countDownTimer - 1;
var curOpacity = defaultSettings.targetOpacity4Dim + interval*countDownTimer;
// TODO: Better iterate over dimmap
for(var i = 0; i < theHRefs.length; i++)
{
var hash = 'm' + hex_md5(theHRefs[i].href);
if (hash in dimMap) {
theHRefs[i].style.opacity = curOpacity;
}
}
if (countDownTimer > 0) {
var to = setTimeout(dimLinks, defaultSettings.dimInterval);
}
}
/*
* Check if an element is fully drawn on the viewport
* from http://stackoverflow.com/questions/487073/check-if-element-is-visible-after-scrolling?lq=1
*/
function isFullyInView(elem)
{
debugLogLvl(DEBUG_LVL_INFO, "isFullyInView");
var docViewTop = $(window).scrollTop();
var docViewBottom = docViewTop + $(window).height();
var elemTop = $(elem).offset().top;
var elemBottom = elemTop + $(elem).height();
return ((elemBottom <= docViewBottom) && (elemTop >= docViewTop));
// is really fully in view
// return ((elemBottom >= docViewTop) && (elemTop <= docViewBottom)
// && (elemBottom <= docViewBottom) && (elemTop >= docViewTop) );
}
/*
* Called after scrolling finished for defined time
*/
function evaluateElems() {
debugLogLvl(DEBUG_LVL_INFO, "evaluate all");
processElements(false);
timeOutAfterLastScroll = UNDEF;
}
/*
* Wait for scrolling to end
*/
function onScroll()
{
if (timeOutAfterLastScroll !== UNDEF) {
window.clearTimeout(timeOutAfterLastScroll)
}
timeOutAfterLastScroll = setTimeout(evaluateElems, AFTER_SCROLL_DELAY);
}
/*
* Process all elements
*/
function processElements(firstCall) {
debugLogLvl(DEBUG_LVL_INFO, "processElements");
var allTagElems = document.getElementsByTagName(tag2Process);
var elemMap = {};
var theBase = document.baseURI;
// Change the DOM
// First loop: gather all new links and make already seen opaque.
for(var i = 0; i < allTagElems.length; i++)
{
var hash = 'm' + hex_md5(getContentFct(allTagElems[i]));
// setValue needs letter in the beginning, thus use of 'm'
debugLogLvl(DEBUG_LVL_INFO, "hash: " + hash);
var key = GM_getValue(hash);
if (typeof key !== UNDEF && key !== null) {
// workaround for issue https://github.com/greasemonkey/greasemonkey/issues/2156
// key found -> loaded this reference already
debugLogLvl(DEBUG_LVL_INFO, "key found");
if (firstCall) {
var done = false;
if(goUp(curSettings, allTagElems[i])) {
var pe = locateParentElem(curSettings, theDomain, allTagElems[i])
// console.log("locate parent done", pe);
if (pe) {
pe.style.opacity = defaultSettings.targetOpacity;
done = true;
}
}
if (!done) {
// change display
allTagElems[i].style.opacity = defaultSettings.targetOpacity;
debugLogLvl(DEBUG_LVL_INFO, "changing opacity");
}
}
} else {
//check if element is fully visible
debugLogLvl(DEBUG_LVL_INFO, "key not found");
if (isFullyInView(allTagElems[i])) {
debuglog(allTagElems[i] + " is in view");
// key not found, store it with current date
elemMap[hash] = {"domain":theDomain, "date":(new Date()).getTime(), "base":theBase};
dimMap[hash] = allTagElems[i];
}
}
}
// remember all new urls to hide the next time
for (var e2 in elemMap) {
GM_setValue(e2, JSON.stringify(elemMap[e2]));
}
theHRefs = allTagElems;
if (firstCall) {
var to = setTimeout(dimLinks, defaultSettings.dimInterval);
}
}
// Menus
GM_registerMenuCommand("Remove the seen history for this site.", resetUrlsForCurrentSite);
GM_registerMenuCommand("Remove the seen history for this domain.", resetUrlsForCurrentDomain);
GM_registerMenuCommand("Remove all seen history (for all sites)!", resetAllUrls);
GM_addStyle(progressBarStyle);
// Main part
function run_script() {
debugLogLvl(DEBUG_LVL_INFO, "run");
dimMap = {};
theDomain = document.domain;
curSettings = findPerUrlSettings(perUrlSettings, theDomain);
tag2Process = DEFAULT_TAG;
getContentFct = defaultGetContentFct;
if (typeof curSettings != UNDEF) {
if (typeof curSettings.tag != UNDEF) {
tag2Process = curSettings.tag;
}
if (typeof curSettings.getContent != UNDEF) {
getContentFct = curSettings.getContent;
}
}
expireUrls();
processElements(true);
window.addEventListener("scroll", onScroll, false);
}
run_script();
})();