NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name FaceCards // @namespace https://openuserjs.org/scripts/hklene/FaceCards // @description Gesichter lernen mit Karteikarten // @copyright 2018, hklene (https://openuserjs.org/users/hklene) // @license GPL-3.0-or-later; http://www.gnu.org/licenses/gpl-3.0.txt // @version 2020-05-18 // @include https://*/browsepeople.action* // @include https://*/dopeopledirectorysearch.action* // @icon  // @grant none // ==/UserScript== // ==OpenUserJS== // @author hklene // ==OpenUserJS== // Der Inhalt dieser umschließenden Funktion wird komplett in die Seite injiziert (benötigt keinen Zugriff auf besondere GreaseMonkey-API). function injectContent() { // BEGIN injectContent() function debug(message) { if (false) { alert(message); } } // http://de.selfhtml.org/xml/darstellung/xpathsyntax.htm function viaXpath(path, node, ordered) { return document.evaluate(path, node ? node : document, null, ordered ? XPathResult.ORDERED_NODE_SNAPSHOT_TYPE : XPathResult.UNORDERED_NODE_SNAPSHOT_TYPE, null); } // https://developer.mozilla.org/en-US/docs/Introduction_to_using_XPath_in_JavaScript#First_Node function viaXpath0(path, node, ordered) { return document.evaluate(path, node ? node : document, null, ordered ? XPathResult.FIRST_ORDERED_NODE_TYPE : XPathResult.ANY_UNORDERED_NODE_TYPE, null).singleNodeValue; } // Lightbox nach Denis Potschien: http://www.drweb.de/magazin/css3-lightbox-komplett-ohne-javascript function lightboxStyle() { /* DIV.lightbox { overflow: hidden; position: fixed; width: 0; height: 0; left: 0; top: 0; opacity: 0; background: rgba(127,127,127, 0.95); -moz-transition: opacity 0.5s; -o-transition: opacity 0.5s; -webkit-transition: opacity 0.5s; } DIV.lightbox:target { width: 100%; height: 100%; opacity: 1; z-index: 100; } */} function lightboxContent() { /* <TABLE width="100%" height="100%" border="0" style="padding:10px;"><TR> <TD><A id="auswertung" style="visibility:hidden" href="javascript:auswertung();" title="Auswertung"><H1>A</H1></A></TD> <TD><CENTER id="FaceCardsNumber"></CENTER></TD> <TD><A href="#" title="Schließen"><H1 style="text-align:right;">X</H1></A> </TD></TR><TR><TD colspan=3 align="center" height="100%"> <IMG id="FaceCardsImg" style="height:100%; max-height:512px;"> </TD></TR><TR><TD colspan=3 align="center"> <DIV id="FaceCardsQuestion"> <H1>Wer ist das?</H1> <BUTTON style="width:15%;" onclick="solve();">Lösung</BUTTON> </DIV> <DIV id="FaceCardsResult"><H1 style="font-weight:bold;"></H1><H2></H2><BR> <H2>Bewerte, wie gut Du Dich an dieses Gesicht erinnern konntest:</H2><BR> <BUTTON style="width:15%; background:#F77;" onclick="showRandomEntry(1);">Keine Ahnung</BUTTON> <BUTTON style="width:15%; background:#FB7;" onclick="showRandomEntry(2);">Lange überlegt</BUTTON> <BUTTON style="width:15%; background:#FF7;" onclick="showRandomEntry(3);">Beinahe</BUTTON> <BUTTON style="width:15%; background:#BF7;" onclick="showRandomEntry(6);">Zuversichtlich</BUTTON> <BUTTON style="width:15%; background:#7F7;" onclick="showRandomEntry(9);">Sicher erkannt</BUTTON> </DIV> </TD></TR> </TABLE> */} // https://developer.mozilla.org/en-US/docs/Web/API/Web_Storage_API/Using_the_Web_Storage_API var storage, prefix = "FaceCards_"; try { // detect availability and ensure functionality storage = window.localStorage; storage.setItem(prefix, prefix); storage.removeItem(prefix); } catch(e) { storage = null; } var FaceCardsNumber, FaceCardsImg, FaceCardsQuestion, FaceCardsResult; function solve() { FaceCardsQuestion.style.visibility = "hidden"; FaceCardsQuestion.style.height = "0px"; FaceCardsResult .style.visibility = "visible"; FaceCardsResult .style.height = null; viaXpath0("BUTTON[3]", FaceCardsResult).focus(); } function Entry(src) { this.src = src; this.name = ""; this.email = ""; this.confidence = 0; this.hash = null; this.load = function() { // Falls lokaler Web Speicher verfügbar, versuche alten Lernfortschritt zu lesen if (storage) { this.hash = prefix + MD5_hexhash(src); this.confidence = parseInt(storage.getItem(this.hash)); if (!this.confidence) { this.confidence = 0; } } } this.store = function() { if (storage && this.confidence > 0) { storage.setItem(this.hash, this.confidence); } } this.toString = function() { return this.name + "\n" + this.confidence + "\n" + this.email + "\n" + this.src; } } var entries = null, lesson = null, showing = false, rawCount = 0, enoughForToday = 0, enoughTotal = 0, hotspot = new Array(), hotspotSize = 20, confidenceThreshold = 9, confidenceModulo = 3; // in zufälliger Reihenfolge in Hotspot einfügen function addToHotspot(index) { if (hotspot.length < hotspotSize && hotspot.indexOf(index) < 0) { if (0 == hotspot.length) { hotspot.push(index); } else { var i = (Math.random() * hotspot.length) | 0; hotspot.push(hotspot[i]); hotspot[i] = index; } // debug("inserted i:" + index + " in h:" + hotspot.join()); } } function countSaved(location) { var saved = 0, expected = 0; if (storage) { for(var propertyName in storage) { if (propertyName.substr(0,prefix.length) == prefix) { saved++; } } for (var i = 0; i < entries.length; i++) { if (0 < entries[i].confidence) { expected++; } } if (saved != expected) { debug("Abweichung: " + saved + " != " + expected + "\nin: " + location); } } return saved; } function cleanup() { if (waitOnNavigation) { return; } var before = countSaved("clean before"); // normalisiere auf das Minimum var min = confidenceThreshold; for (var i = 0; i < entries.length; i++) { if (0 == entries[i].confidence) { continue; } min = Math.min(min, entries[i].confidence); if (min < confidenceModulo) { break; } } if (storage) { // Alle Werte aus dem Speicher löschen, auch die, zu denen es keinen Eintrag mehr gibt. for(var propertyName in storage) { if (propertyName.substr(0,prefix.length) == prefix) { storage.removeItem(propertyName); } } } if (confidenceModulo < min) { min--; min = min - min % confidenceModulo; } else { min = 0; } var foundNew = 0; for (var i = 0; i < entries.length; i++) { if (0 == entries[i].confidence) { foundNew++; continue; } if (entries[i].confidence == min){ debug("Killing " + entries[i]); } entries[i].confidence -= min; entries[i].store(); if (entries[i].confidence < confidenceThreshold) { // bevorzuge "alte" unbekannte vor neuen Gesichtern addToHotspot(i); } } if (foundNew && hotspot.length < hotspotSize) { // fülle den Hotspot am Ende mit unbekannten Gesichtern auf // wer schon die meisten kennt, kommt so schneller an die wenigen neuen Kollegen foundNew = hotspot.length; for (var i = 0; i < entries.length; i++) { if (0 == entries[i].confidence) { addToHotspot(i); if (hotspot.length == hotspotSize) { break; } } } debug("Hotspot aufgefüllt mit " + (hotspot.length - foundNew) + " neuen Kollegen!"); } var after = countSaved("clean after"); if (before != after) { debug("Anzahl " + before + " -> " + after); } } function padding(zahl, length) { var string = "" + zahl; if (!length) { length = 2; } while (string.length < length) { // 2 Leerzeichen sind so breit wie eine Ziffer string = " " + string; } return string; } function auswertung() { cleanup(); // Liste alle Werte auf var min = entries[0].confidence, max = min; var w = "Werte: " + entries.length + "\n", map = new Object(); for (var i = 0; i < entries.length; i++) { min = Math.min(min, entries[i].confidence); max = Math.max(max, entries[i].confidence); w += padding(entries[i].confidence ? entries[i].confidence : '_') + (i % 10 == 9 ? ",\n" : ", "); map[entries[i].confidence] = (map[entries[i].confidence] ? map[entries[i].confidence] + 1 : 1); } // Verteilung zwischen min und max var v = "", c = 0, d = 0; for (var k in map) { v += padding(k) + "=" + padding(map[k]) + (++c % 5 == 0 ? ",\n" : ", "); d += map[k]; } v = "\nVerteilung: " + c + " Gruppen " + d + " Werte\n" + v; var h = "\nHotspot: " + hotspot.length + "\n"; for (var i = 0; i < hotspot.length; i++) { h += padding(hotspot[i]) + (i % 10 == 9 ? ",\n" : ", "); } alert("Auswertung\nAngemeldet: " + rawCount + "\nmit Foto: " + entries.length + "\ngespeichert: " + countSaved("auswertung") + "\nmin:" + min + " max:" + max + v.replace(/,\s$/, "\n") + w + h); } function showRandomEntry(lessonLearned) { // summiere den aktuellen Lernfortschritt lesson.confidence += lessonLearned; if (confidenceThreshold < lesson.confidence && lesson.confidence % confidenceModulo != 0) { lesson.confidence = lesson.confidence - lesson.confidence % confidenceModulo; } lesson.store(); // einer der linken beiden Buttons // oder der mittlere Button und Gesamtsicherheit ist noch zu gering // -> versuche es im Hotspot unterzubringen var old = entries.indexOf(lesson); if (0 < lessonLearned) { if (lessonLearned < confidenceModulo || lessonLearned == confidenceModulo && lesson.confidence <= confidenceThreshold) { addToHotspot(old); enoughForToday--; } else { // aus Hotspot entfernen var hi = hotspot.indexOf(old); if (0 <= hi) { hotspot.splice(hi, 1); } enoughForToday++; } enoughTotal++; if (hotspotSize < Math.abs(enoughForToday) || 3 * hotspotSize < enoughTotal) { cleanup(); if (confirm("Genug für heute?")) { window.location.hash = "#"; } else { enoughForToday = enoughTotal = 0; } } } var j, category = '', hr = hotspot.length; if (0 < hr // 20% für letzten Eintrag bis 80% wenn hotspot voll && (hr - 1) / (hotspotSize - 1) * 3 > 5 * Math.random() - 1 && hotspot[hr = hr * Math.random() | 0] != old) { j = hotspot[hr]; lesson = entries[j]; category = 'Hotspot'; } else { // wähle zufällig die nächste Herausforderung, möglichst mit geringerem Lernfortschritt for (var i = 0; ; i++) { j = (Math.random() * entries.length) | 0; var candidate = entries[j]; if (candidate.confidence < lesson.confidence || i == confidenceThreshold) { // wähle geringeren Lernfortschritt oder akzeptiere den letzten Versuch lesson = candidate; category = 0 <= hotspot.indexOf(j) ? 'hotspot' : 'random'; break; } } } FaceCardsNumber.innerHTML = "Punkte: " + lesson.confidence + ", Foto: " + j + " von " + entries.length + ", Auswahl: " + category + " bei " + hotspot.length + " " + enoughForToday + " " + enoughTotal; FaceCardsImg.src = lesson.src; FaceCardsResult.style.visibility = "hidden"; FaceCardsResult.style.height = "0px"; FaceCardsResult.firstChild.innerHTML = lesson.name; viaXpath0("H2", FaceCardsResult).innerHTML = lesson.email ? lesson.email : ""; FaceCardsQuestion.style.visibility = "visible"; FaceCardsQuestion.style.height = null; viaXpath0("BUTTON", FaceCardsQuestion).focus(); showing = true; } function collectData(myBody) { var img = viaXpath("//DIV[@class='profile-macro']", myBody); rawCount += img.snapshotLength; for (var i = 0; i < img.snapshotLength; i++) { var parent = img.snapshotItem(i); var current = viaXpath0(".//A/IMG[@class='userLogo logo']", parent); if (current.src.indexOf('/default.svg') >= 0 || current.src.indexOf('/default.png') >= 0) { continue; } // parent.style.border = "5px solid red"; var entry = new Entry(current.src); entry.load(); current.setAttribute("test", i); current = viaXpath0(".//DIV[@class='values']/H4/A[contains(@class, 'confluence-userlink')]", parent); entry.name = current.firstChild.data; current = viaXpath0(".//DIV[@class='values']/A[@class='email']", parent); if (current && current.firstChild) { entry.email = current.firstChild.data; } entries.push(entry); if (0 < entry.confidence && entry.confidence < confidenceThreshold) { addToHotspot(entries.length - 1); } } // debug("collectData " + img.snapshotLength + " " + entries.length + " " + lesson); } function loadNextPage(link, direction) { // debug("loadNextPage " + link + " " + direction); var childDoc = new XMLHttpRequest(); childDoc.open("GET", link, false); childDoc.send(); childDoc = childDoc.responseText; childDoc = childDoc.substring(childDoc.indexOf("<body"), childDoc.lastIndexOf("</body>")).replace(/^.*>/, ""); // weil eine HTML-Seite und kein XML abgefragt wird, weise den Inhalt einem dummy-Knoten zu var childBody = document.createElement("DIV"); childBody.innerHTML = childDoc; collectData(childBody); navigateRecursive(childBody, direction); } var waitOnNavigation = 0; function navigateRecursive(myBody, direction) { var link = viaXpath0("//OL[@class]/LI[@class='" + direction + "']/A[@href]", myBody); if (!link) { waitOnNavigation--; if (!waitOnNavigation) { document.getElementById("auswertung").removeAttribute("style"); cleanup(); } return; } if (showing || entries.length == 0) { // erstes Foto wird bereits angezeigt oder noch keines gefunden -> suche synchron auf der nächsten Seite loadNextPage(link.href, direction); } else { // es wurde bereits ein Foto gefunden -> breche hier ab um es sofort anzuzeigen und lade die weiteren Seiten erst im Anschluss window.setTimeout("loadNextPage('" + link.href + "', '" + direction + "');", 0); } } function initialize() { // definiere CSS für die Lightbox var lightbox = document.createElement("STYLE"); lightbox.type = "text/css"; lightboxStyle = lightboxStyle.toString(); lightboxStyle = lightboxStyle.substring(lightboxStyle.indexOf("/*") + 2, lightboxStyle.lastIndexOf("*/") - 1); lightbox.innerHTML = lightboxStyle; document.head.appendChild(lightbox); // füge die Lightbox selbst unten an das Dokument an (wird dank CSS später die anderen Inhalte überdecken) lightbox = document.createElement("DIV"); lightbox.id = "FaceCards"; lightbox.setAttribute("class", "lightbox"); lightboxContent = lightboxContent.toString(); lightboxContent = lightboxContent.substring(lightboxContent.indexOf("/*") + 2, lightboxContent.lastIndexOf("*/") - 1); lightbox.innerHTML = lightboxContent; document.body.appendChild(lightbox); // Esc key action $(lightbox).keyup(function(e) { if (e.keyCode == 27) { window.location.hash = "#"; } }); FaceCardsNumber = document.getElementById("FaceCardsNumber"); FaceCardsImg = document.getElementById("FaceCardsImg"); FaceCardsQuestion = document.getElementById("FaceCardsQuestion"); FaceCardsResult = document.getElementById("FaceCardsResult"); // durchsuche die Original-Seite nach Fotos und sammle die Informationen collectData(document.body); waitOnNavigation = 2; navigateRecursive(document.body, "aui-nav-previous"); navigateRecursive(document.body, "aui-nav-next"); } function show() { enoughForToday = enoughTotal = 0; // faule Initialisierung der Lightbox if (!entries) { entries = new Array(); initialize(); } if (entries.length == 0) { alert("Es wurden keine Fotos auf dieser Seite gefunden! " + rawCount); return; } // rufe die Lightbox auf window.location.hash = "#FaceCards"; lesson = entries[0]; showRandomEntry(0); } // Füge einen Link zum Aufruf der Initialisierung hinzu var nav = viaXpath0("//DIV[@class='aui-navgroup-inner']/UL[@class='aui-nav']"); var link = document.createElement("li"); link.innerHTML = "<A href='javascript:show();' title='Zeige Karteikarten der Fotos'>Namen lernen</A>"; nav.appendChild(link); // TODO Debug: rufe die Lightbox sofort auf und warte nicht auf den Klick // initialize(); // show(); // END injectContent() } // -------------------------------------------------------------------------------------------------------------------- // Message Digest MD5 - Hash Funktion zur Verschlüsselung personenbezogener Daten vor der Speicherung lokal im Browser // http://www.onicos.com/staff/iz/amuse/javascript/expert/md5.txt function injectMD5() { // BEGIN injectMD5() /* md5.js - MD5 Message-Digest * Copyright (C) 1999,2002 Masanao Izumo <iz@onicos.co.jp> * Version: 2.0.0 * LastModified: May 13 2002 * * This program is free software. You can redistribute it and/or modify * it without any warranty. This library calculates the MD5 based on RFC1321. * See RFC1321 for more information and algorism. */ /* Interface: * md5_128bits = MD5_hash(data); * md5_hexstr = MD5_hexhash(data); */ /* ChangeLog * 2002/05/13: Version 2.0.0 released * NOTICE: API is changed. * 2002/04/15: Bug fix about MD5 length. */ // md5_T[i] = parseInt(Math.abs(Math.sin(i)) * 4294967296.0); var MD5_T = new Array(0x00000000, 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee, 0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501, 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be, 0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821, 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa, 0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8, 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed, 0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a, 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c, 0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70, 0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05, 0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665, 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039, 0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1, 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1, 0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391); var MD5_round1 = new Array(new Array( 0, 7, 1), new Array( 1,12, 2), new Array( 2,17, 3), new Array( 3,22, 4), new Array( 4, 7, 5), new Array( 5,12, 6), new Array( 6,17, 7), new Array( 7,22, 8), new Array( 8, 7, 9), new Array( 9,12,10), new Array(10,17,11), new Array(11,22,12), new Array(12, 7,13), new Array(13,12,14), new Array(14,17,15), new Array(15,22,16)); var MD5_round2 = new Array(new Array( 1, 5,17), new Array( 6, 9,18), new Array(11,14,19), new Array( 0,20,20), new Array( 5, 5,21), new Array(10, 9,22), new Array(15,14,23), new Array( 4,20,24), new Array( 9, 5,25), new Array(14, 9,26), new Array( 3,14,27), new Array( 8,20,28), new Array(13, 5,29), new Array( 2, 9,30), new Array( 7,14,31), new Array(12,20,32)); var MD5_round3 = new Array(new Array( 5, 4,33), new Array( 8,11,34), new Array(11,16,35), new Array(14,23,36), new Array( 1, 4,37), new Array( 4,11,38), new Array( 7,16,39), new Array(10,23,40), new Array(13, 4,41), new Array( 0,11,42), new Array( 3,16,43), new Array( 6,23,44), new Array( 9, 4,45), new Array(12,11,46), new Array(15,16,47), new Array( 2,23,48)); var MD5_round4 = new Array(new Array( 0, 6,49), new Array( 7,10,50), new Array(14,15,51), new Array( 5,21,52), new Array(12, 6,53), new Array( 3,10,54), new Array(10,15,55), new Array( 1,21,56), new Array( 8, 6,57), new Array(15,10,58), new Array( 6,15,59), new Array(13,21,60), new Array( 4, 6,61), new Array(11,10,62), new Array( 2,15,63), new Array( 9,21,64)); function MD5_F(x, y, z) { return (x & y) | (~x & z); } function MD5_G(x, y, z) { return (x & z) | (y & ~z); } function MD5_H(x, y, z) { return x ^ y ^ z; } function MD5_I(x, y, z) { return y ^ (x | ~z); } var MD5_round = new Array(new Array(MD5_F, MD5_round1), new Array(MD5_G, MD5_round2), new Array(MD5_H, MD5_round3), new Array(MD5_I, MD5_round4)); function MD5_pack(n32) { return String.fromCharCode(n32 & 0xff) + String.fromCharCode((n32 >>> 8) & 0xff) + String.fromCharCode((n32 >>> 16) & 0xff) + String.fromCharCode((n32 >>> 24) & 0xff); } function MD5_unpack(s4) { return s4.charCodeAt(0) | (s4.charCodeAt(1) << 8) | (s4.charCodeAt(2) << 16) | (s4.charCodeAt(3) << 24); } function MD5_number(n) { while (n < 0) n += 4294967296; while (n > 4294967295) n -= 4294967296; return n; } function MD5_apply_round(x, s, f, abcd, r) { var a, b, c, d; var kk, ss, ii; var t, u; a = abcd[0]; b = abcd[1]; c = abcd[2]; d = abcd[3]; kk = r[0]; ss = r[1]; ii = r[2]; u = f(s[b], s[c], s[d]); t = s[a] + u + x[kk] + MD5_T[ii]; t = MD5_number(t); t = ((t<<ss) | (t>>>(32-ss))); t += s[b]; s[a] = MD5_number(t); } function MD5_hash(data) { var abcd, x, state, s; var len, index, padLen, f, r; var i, j, k; var tmp; state = new Array(0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476); len = data.length; index = len & 0x3f; padLen = (index < 56) ? (56 - index) : (120 - index); if(padLen > 0) { data += "\x80"; for(i = 0; i < padLen - 1; i++) data += "\x00"; } data += MD5_pack(len * 8); data += MD5_pack(0); len += padLen + 8; abcd = new Array(0, 1, 2, 3); x = new Array(16); s = new Array(4); for(k = 0; k < len; k += 64) { for(i = 0, j = k; i < 16; i++, j += 4) { x[i] = data.charCodeAt(j) | (data.charCodeAt(j + 1) << 8) | (data.charCodeAt(j + 2) << 16) | (data.charCodeAt(j + 3) << 24); } for(i = 0; i < 4; i++) s[i] = state[i]; for(i = 0; i < 4; i++) { f = MD5_round[i][0]; r = MD5_round[i][1]; for(j = 0; j < 16; j++) { MD5_apply_round(x, s, f, abcd, r[j]); tmp = abcd[0]; abcd[0] = abcd[3]; abcd[3] = abcd[2]; abcd[2] = abcd[1]; abcd[1] = tmp; } } for(i = 0; i < 4; i++) { state[i] += s[i]; state[i] = MD5_number(state[i]); } } return MD5_pack(state[0]) + MD5_pack(state[1]) + MD5_pack(state[2]) + MD5_pack(state[3]); } function MD5_hexhash(data) { var i, out, c; var bit128; bit128 = MD5_hash(data); out = ""; for(i = 0; i < 16; i++) { c = bit128.charCodeAt(i); out += "0123456789abcdef".charAt((c>>4) & 0xf); out += "0123456789abcdef".charAt(c & 0xf); } return out; } // END injectMD5() } // -------------------------------------------------------------------------------------------------------------------- // Wandle die Funktion in eine Zeichenkette und hänge sie an den Dokumenten-Kopf injectMD5 = injectMD5.toString(); injectMD5 = injectMD5.substring(injectMD5.indexOf("{") + 1, injectMD5.lastIndexOf("}") - 1); injectContent = injectContent.toString(); injectContent = injectContent.substring(injectContent.indexOf("{") + 1, injectContent.lastIndexOf("}") - 1); var inPageScript = document.createElement("SCRIPT"); inPageScript.type = "text/javascript"; inPageScript.innerHTML = injectMD5 + "\n" + injectContent; document.head.appendChild(inPageScript);