Greasy Fork is available in English.

Login reminder popup remover (developement ceased - not up to date)

Removes the nagging login popups and banners from mobile and desktop versions of Facebook, Instagram, Reddit, Twitter, Quora, VK, Pinterest, Tumblr, Twitch and from the mobile versions of Youtube and TikTok.

// ==UserScript==
// @name        Login reminder popup remover (developement ceased - not up to date)
// @name:it  Rimuovi i popup di richiesta d'accesso (non aggiornato - sviluppo terminato)
// @namespace   StephenP
// @description Removes the nagging login popups and banners from mobile and desktop versions of Facebook, Instagram, Reddit, Twitter, Quora, VK, Pinterest, Tumblr, Twitch and from the mobile versions of Youtube and TikTok.
// @description:it Rimuovi i fastidiosi banner e popup di richiesta d'accesso dalle versioni mobile e desktop di Facebook, Instagram, Reddit, Twitter, Quora, VK, Pinterest, Tumblr, Twitch e dalla versione mobile di Youtube e TikTok.
// @match     https://*.facebook.com/*
// @exclude   https://developers.facebook.com/*
// @exclude   https://www.instagram.com/privacy/checks/*
// @exclude   https://twitter.com/intent/tweet?*
// @exclude   https://mobile.twitter.com/intent/tweet?*
// @exclude   https://x.com/intent/tweet?*
// @exclude   https://mobile.x.com/intent/tweet?*
// @exclude   https://vk.com/im?*
// @match     https://www.instagram.com/*
// @match     https://m.youtube.com/*
// @match     https://www.youtube.com/*
// @match     https://www.reddit.com/*
// @match     https://twitter.com/*
// @match     https://mobile.twitter.com/*
// @match     https://x.com/*
// @match     https://mobile.x.com/*
// @match     https://*.quora.com/*
// @match     https://vk.com/*
// @match     https://m.vk.com/*
// @match     https://*.tumblr.com/*
// @match     https://www.twitch.tv/*
// @match     https://www.tiktok.com/*/video/*
// @include   https://www.pinterest.tld/*
// @version   2.7.21
// @grant     GM.getValue
// @grant     GM.setValue
// @grant     GM.registerMenuCommand
// @grant     GM.unsafeWindow
// @license   CC-BY-NC-SA-4.0
// @icon      
// ==/UserScript==

//A ROUGH SCHEME OF HOW ALL THIS CODE WORKS:
//First, it checks the configuration (language of the browser and sites where the user disabled the execution of this script).
//Then, if it has just been loaded, the script proceeds to add specific CSS code for the site, and then some specific JS code if needed for elements that are hard to select using CSS rules.
//Otherwise it constantly checks if the page has changed (happens when pages are not loaded normally, but an AJAX request is used to fetch the new content of the page), and if it has changed, applies again the necessary CSS and JS code to remove the popups.

//Various intervals used by the timers in the funcions (mainly to check the page until the needed items appeare, without relying on the cumbersome mutation observers)
var interval_0;
var interval_1;
var interval_2;
var interval_3;
var interval_4;
var mutationObserver_0;
var url;
//list of disabled sites (synced with the "disabledSites" list saved with GM.setValue())
var disabledSites=[];
var l;
//localizations in JSON format
const jsonLocalization=JSON.parse(`
{
    "en": {
        "disableFor": "Disable for ",
        "enableFor": "Enable again for ",
		"disabledSites": "Disabled sites: ",
		"isDisabled": " is currently disabled.",
		"isNotDisabled": " isn't currently disabled.",
        "igBanNote": "If the page doesn't load anymore, your IP may have been temporarily limited from anonymous Instagram browsing. If you still need to view the page, you can try using one of the external services listed below:"
	},
    "it": {
        "disableFor": "Disabilita per ",
        "enableFor": "Abilita nuovamente per ",
		"disabledSites": "Siti disabilitati: ",
		"isDisabled": " è attualmente disabilitato.",
		"isNotDisabled": " non è attualmente disabilitato.",
        "igBanNote": "Se la pagina non carica oltre, il tuo indirizzo IP potrebbe essere stato temporaneamente limitato dalla navigazione anonima su Instagram. Se hai comunque bisogno di vedere la pagina, puoi provare usando uno dei servizi esterni elencati qui sotto:"
    }
}
`);

//Program start: checks and sets the language, looks for which sites the execution has been disabled and launches the function to add the specific css and js for each site.
(async function(){
  let lang=getLang();
  if(lang.startsWith("it")){
    l=jsonLocalization.it;
  }
  else{
    l=jsonLocalization.en;
  }
  disabledSites=await getDisabledSites();
  if(!disabledSites.includes(document.location.hostname)){
    GM.registerMenuCommand(l.disableFor+document.location.hostname, changeForThisSite, "D");
    applyUserCss();
    setInterval(checkPageChanged,1000);
  }
  else{
    GM.registerMenuCommand(l.enableFor+document.location.hostname, changeForThisSite, "E");
  }
})();
function getLang() {
  try{
    if (navigator.languages != undefined)
      return navigator.languages[0];
    return navigator.language;
  }
  catch(e){
    return "en"
  }
}
async function getDisabledSites(){
  var dS=await GM.getValue("disabledSites");
  console.log(l.disabledSites+dS);
  if(dS){
		return dS.split(',');
  }
  else{
    return [];
  }
}
//this function switches the setting of a site from enabled to disabled and viceversa
function changeForThisSite(){
  let tS=document.location.hostname;
  if(disabledSites.includes(tS)){
    console.log(tS+l.isDisabled);
    var filtered = disabledSites.filter(function(value){
      return value !== document.location.hostname;
    });
    console.log(filtered);
    disabledSites=filtered;
    GM.setValue("disabledSites",filtered.join(","));
    document.location.reload();
  }
  else if(!disabledSites.includes(tS)){
    console.log(tS+l.isNotDisabled);
    disabledSites.push(tS);
    GM.setValue("disabledSites",disabledSites.join(","));
    document.location.reload();
  }
}
//when a page is loaded through an AJAX request, the script is not automatically reloaded. As tracking the AJAX requests is a pointless effort, this function just compares if the url of the page changed since the last check (once per second, can be changed in the main function)
function checkPageChanged(){
 if(document.location!=url){
   console.log("Changed url from "+url+" to "+document.location.toString());
 	interval_4=setInterval(checkPageReady,500);
  url=document.location.toString();
 }
}
//this function applies for every site a specific set of css rules to remove most of the annoyances.
//some of them are very hard, if not impossible, to be defined with css rules, hence the checkPageReady() function that waits for the page to load and uses specific js functions to remove these annoyances through the means of js.
function applyUserCss(){//this function adds the css styling for removing popups to the head of the document, as soon as possible when the page is loaded. Other popups that can't be removed just with css selectors are removed later with the check() function
  let st=document.createElement("STYLE");
  st.id="LPRStyle";
  if(document.location.href.includes("instagram.com")){
    st.textContent=".u7YqG{z-index: 1} .xUdfV{z-index: 2} ._3sb-{z-index: 3} .G_hoz.LcKDX, ._7zNgw.GLdVF{z-index: 2} .FFVAD{z-index: 1} .tWeCl{z-index: 1} .v5DqJ, .RnEpo.Yx5HN, .RnEpo._Yhr4, .xZ2Xk, .tHaIX, nav.kqZqN.e6AxN.JIFNi, .mray0b6k, ._aa9j, ._ab8q._ab8w, .h4m39qi9, [class=x1uhb9sk]{display: none !important} body{overflow-y: scroll !important} .CzVzU>div{z-index: 4} #scrollview{scrollbar-width: unset} html{scrollbar-width: none}";
  }
  else if(document.location.href.includes("twitter.com")||document.location.href.includes("//x.com")||document.location.href.includes(".x.com")){
    //IF EDITING THIS LIST OF POPUPS, KEEP IT IN SYNC WITH THE POPUPS LISTED blockBannerTw
    st.textContent="#layers>.css-1dbjc4n.r-aqfbo4.r-1p0dtai.r-1d2f490.r-12vffkv.r-1xcajam.r-zchlnj>.css-1dbjc4n.r-12vffkv>.css-1dbjc4n.r-12vffkv>.css-1dbjc4n.r-l5o3uw, .css-1dbjc4n.r-1awozwy.r-14lw9ot.r-1dgieki.r-1efd50x.r-5kkj8d.r-18u37iz.r-16y2uox.r-1a1dyw.r-1swwhx3.r-1j3t67a.r-1qxgc49, .css-1dbjc4n.r-1awozwy.r-1kihuf0.r-18u37iz.r-1pi2tsx.r-1777fci.r-1pjcn9w.r-1xcajam.r-ipm5af.r-g6jmlv, .css-1dbjc4n.r-aqfbo4.r-1d2f490.r-12vffkv.r-1xcajam.r-zchlnj.r-ipm5af, .css-1dbjc4n.r-aqfbo4.r-1p0dtai.r-1d2f490{display: none !important}";
  }
  else if(document.location.href.includes("reddit.com")){
    st.textContent="* {filter: none !important} .XPromoNSFWBlocking__warning, .XPromoNSFWBlockingModal, .xPromoChoiceBanner, .XPromoPill, .XPromoPopup, .GetAppFooter, .XPromoInFeed, .XPromoBlockingModal, .PreviewDrawer, [data-testid=bottom-cell-wrapper], shreddit-experience-tree, shreddit-comments-page-ad, xpromo-untagged-content-blocking-modal, xpromo-nsfw-blocking-modal, xpromo-new-nsfw-blocking-modal, .XPromoBottomBar, [bundlename=bottom_bar_xpromo], [bundlename=desktop_rpl_nsfw_blocking_modal], .XPromoPopupRpl, .XPromoBlockingModalRpl, xpromo-nsfw-blocking-modal-desktop, body>div[style*='inset: 0'], [bundlename=app_selector], [paint-group=xpromo], inline-auth-landing-experience-xpromo-shell {display: none !important} .scroll-disabled {overflow-y: scroll !important; position: static !important} .NavFrame, body,.scroll-is-blocked {overflow-y: scroll !important} body{pointer-events: inherit !important} shreddit-app>div.fixed{position: inherit !important}";
  }
  else if(document.location.href.includes("quora.com/")){
    st.textContent="*, #page_wrapper {filter: none !important} .new_signup_dialog, .qu-zIndex--blocking_wall{display: none !important} body, .q-platform--mobile{overflow-y: scroll !important; overflow-x: hidden !important;} .q-sticky{position: inherit !important}";
  }
  else if(document.location.href.includes("vk.com/")){
    st.textContent="body{overflow-y: scroll !important} .PageBottomBanner--unauth{display: none !important}";
  }
  else if(document.location.href.includes("//m.youtube.com")){
  	st.textContent=".upsell-dialog-lightbox{display: none !important} [modal-open-body]{position: inherit !important}";
  }
  else if(document.location.href.includes("//www.youtube.com")){
  	st.textContent="ytd-guide-signin-promo-renderer{display: none !important}";
  }
  else if(document.location.href.includes("facebook.com")){
    st.textContent=".asf1osic.k4urcfbm.j9ispegn.poy2od1o.tw6a2znq.m7u2wfa4.d1544ag0.i3j981x3.rlt63pii.i09qtzwb.hybvsw6c, .tlfeazxf, [role=main]+[data-nosnippet], .xixxii4.xh8yej3.x1ey2m1c {display: none !important}";
    if((!document.location.href.includes("m.facebook.com"))&&(!document.location.href.includes("iphone.facebook.com"))&&(!document.location.href.includes("x.facebook.com"))&&(!document.location.href.includes("touch.facebook.com"))){
      st.textContent+="#pagelet_growth_expanding_cta{display: none !important}";
    }
    if(document.location.href.includes("/posts/")){
      st.textContent+="#headerArea{display: none !important}";
    }
    else if(document.location.href.includes("touch.facebook.com")){
      st.textContent+="._4b-b{display: none !important}";
    }
  }
  else if(document.location.href.includes("quora.com")){
    st.textContent=".signup_wall_wrapper>.BaseSignupForm._DialogSignupForm{display: none !important}";
  }
  else if(document.location.href.includes("https://www.pinterest.")){
    st.textContent="[data-test-id=bottom-right-upsell], [data-test-id=giftWrap], [data-test-id=fullPageSignupModal], [data-test-id=floating-app-upsell], .FNs.XiG.zI7.iyn.Hsu{display: none !important} .article-row{-ms-overflow-style: auto !important;scrollbar-width: auto !important;} ::-webkit-scrollbar {display: block !important;} ::-webkit-scrollbar-thumb {background: grey;}";
  }
  else if(document.location.href.includes("tumblr.com")){
    st.textContent="div.IvzMP{display:none} body{overflow-y: scroll !important}";
  }
  else if(document.location.href.includes("twitch.tv")){
    st.textContent="#twilight-sticky-footer-root{display:none}";
  }
  else if(document.location.href.includes("tiktok.com")){
    st.textContent="[class*=DivTopBannerAB][type=bottom], div.box-border>.z-2000, .matrix-smart-wrapper{display:none}";
  }
  document.getElementsByTagName("HEAD")[0].appendChild(st);
  //some sites are evil and try to delete the css that this script adds. The following tries to restore it in the very moment it is removed.
  const observer = new MutationObserver(function(mutations_list) {
    mutations_list.forEach(function(mutation) {
      mutation.removedNodes.forEach(function(removed_node) {
        if(removed_node.id == 'LPRStyle') {
          console.log('#child has been removed');
          observer.disconnect();
          applyUserCss();
        }
      });
    });
  });

	observer.observe(document.getElementsByTagName("HEAD")[0], { subtree: false, childList: true });


}
//checkPageReady() checks for specific elements of the page that appear when the page is ready enough to operate the functions on it. Once a page is ready, specific functions are executed to remove popups, banners, etc.
function checkPageReady(){
  if((document.getElementsByTagName("BUTTON").length>0)||(document.getElementsByTagName("FORM").length>0)||(document.getElementsByClassName("signup_login_buttons").length>0)||(document.getElementsByTagName("shreddit-post").length>0)||(document.getElementById("layers")!==null)||(document.getElementById("mcont")!==null)||(document.getElementsByTagName("VIDEO").length>0)){//If buttons are there, the page should have loaded. At least that's what I see...
    console.log("Page is ready");
    clearInterval(interval_4);
    if(document.location.href.includes("facebook.com")){
      interval_0=setInterval(blockPopupFB,500);
    }
    else if((document.location.href.includes("instagram.com"))&&(!document.location.href.includes("/accounts/login/?next="))){
      clearInterval(interval_2);
      //The following observer checks if the popup is a login popup or a suggestion to use the app: if that's not the case, the popup is displayed.
      const igPopupObserverConfig = { attributes: false, childList: true, subtree: false };
      const igPopupObserverCallback = function(mutationsList, observeIgPopups) {
          for(const mutation of mutationsList) {
            if(mutation.addedNodes.length>0){
              //console.log(mutation);
              if(mutation.addedNodes[0].classList.contains("RnEpo")){
                //console.log(!mutation.addedNodes[0].querySelector("[name='username']"));
                if((!mutation.addedNodes[0].querySelector("[name='username']"))&&(!mutation.addedNodes[0].querySelector("[href*='/accounts/login/']"))){
                  let style="";
                  if(mutation.addedNodes[0].hasAttribute("style")){
                    style=mutation.addedNodes[0].getAttribute("style");
                  }
                   mutation.addedNodes[0].setAttribute("style","display: block !important;"+style);
                }
              }
            }
          }
      };
      const observeIgPopups = new MutationObserver(igPopupObserverCallback);
      observeIgPopups.observe(document.body, igPopupObserverConfig);
      if(!document.querySelector("[href='/direct/inbox/']")){
        igLoadOutsideNote();
      }
      //interval_1=setInterval(blockBannerIG,200);
      if((document.location.href.includes('instagram.com/p/'))||(document.location.href.includes('instagram.com/tv/'))||(document.location.href.includes('/reel/'))){
        if((document.location.href.includes('/tv/'))){
          let igtvVideo=document.getElementsByTagName("VIDEO")[0];
          let cln=igtvVideo.cloneNode(true);
          insertAfter(cln,igtvVideo);
          igtvVideo.remove();
        }
        interval_0=setInterval(allowVideoReplayStarterIG,500);
      }
      else if(document.location.href.includes('/channel')){
        if(document.getElementsByClassName("coreSpriteSearchIcon").length>0){//only desktop version shows login popup on igtv
           interval_2=setInterval(removePicturePopupIGTV,1000);
        }
      }
      else if((document.location.href.includes('/reels/'))||
              (document.location.href.includes('/guides/'))||
              (document.location.href.includes('/guide/'))){
        //do nothing;
      }
      else{//this should ideally only include the main page of the profile
        //interval_2=setInterval(blockPopupIG,200);
        let sectionLinks=document.getElementsByClassName("_9VEo1");
        for(let sectionLink of sectionLinks){
        	if(sectionLink.href.includes("%2Ffeed%2F")){
            let tmpSectionLink=sectionLink.cloneNode(true);
            tmpSectionLink.href=document.location.href+"feed/";
            sectionLink=sectionLink.parentNode.replaceChild(tmpSectionLink,sectionLink);
            break;
          }
        }
        interval_3=setInterval(removeRelatedAccounts,200);
        interval_2=setInterval(removePicturePopup,500);
      }
    }
    else if(document.location.href.includes('/accounts/login/?next=')){
      igLoadOutsideNote();
    }
    else if(document.location.href.includes("reddit.com")){//reddit has a wide range of different login reminders for installing the app or logging in when browsing from a phone
      if(document.getElementsByTagName("shreddit-app").length>0){
      	redditPatch(true);//isMobile
      }
      else{
        redditPatch(false);
      }
    }
    else if(document.location.href.includes("quora.com/")){
      quoraPatchObserver();
    }
    else if(document.location.href.includes("/twitter.com")||document.location.href.includes("//x.com")||(document.location.href.includes(".x.com")&&!document.location.href.includes("mobile.x.com"))){
      blockBannerTW("d");
    }
    else if(document.location.href.includes("mobile.x.com")){
      blockBannerTW("m");
    }
    else if(document.location.href.includes("https://vk.com/")){
      checkVKLoginPopup("d");
    }
    else if(document.location.href.includes("https://m.vk.com/")){
      checkVKLoginPopup("m");
    }
    else if(document.location.href.includes("https://www.pinterest.")){
      document.body.setAttribute("style", "overflow-y: auto !important");
    }
    else if(document.location.href.includes("tumblr.com")){
      tumblrDeClassify();
      tumblrFreeScroll();
    }
    else if(document.location.href.includes("tiktok.com")){
      interval_0=setInterval(tiktok_waitReady,500);
      interval_1=setInterval(tiktok_clickList,500);
    }
  }
  else{
    console.log("still non buttons");
  }
}
//
//
//THE FOLLOWING SECTION INCLUDES ALL THE FUNCTIONS THAT ARE SPECIFICAL FOR EACH WEBSITE.
//Some may be not even needed anymore, some are not updated and some should be added, but all of them should have been named better.
//
//
function blockPopupFB(){
  var popupFB;
  var loc=document.location.href;
  if((loc.includes("m.facebook.com"))||(loc.includes("iphone.facebook.com"))||(loc.includes("x.facebook.com"))){
    popupFB=document.getElementById("popup_xout").parentNode.parentNode.parentNode;
  }
  if(popupFB==undefined){
    popupFB=document.getElementsByClassName("_5hn6")[0];
    if(popupFB==undefined){
      popupFB=document.getElementById("login_popup_cta_form").parentNode.parentNode.parentNode.parentNode;
      let scrollview=document.getElementById("scrollview");
      if(scrollview){
        let closePopupButton=document.getElementsByClassName("cypi58rs pmk7jnqg fcg2cn6m tkr6xdv7");
        if(closePopupButton.length>0){
          closePopupButton[0].firstChild.click();
        }
        else{
          scrollview.parentNode.lastChild.remove();
          scrollview.style.scrollbarWidth="auto";
        }
      }
      try{
      	document.getElementsByClassName("asf1osic k4urcfbm j9ispegn poy2od1o tw6a2znq m7u2wfa4 d1544ag0 i3j981x3 rlt63pii i09qtzwb hybvsw6c")[0].remove();
      }
      catch(e){console.log("Can't remove element with CSS Selector .asf1osic.k4urcfbm.j9ispegn.poy2od1o.tw6a2znq.m7u2wfa4.d1544ag0.i3j981x3.rlt63pii.i09qtzwb.hybvsw6c")}
    }
  }
  if(popupFB!==undefined){
    //popupFB.parentNode.removeChild(popupFB);
    clearInterval(interval_0);
  }

}
function removeRelatedAccounts(){
  let morePostsBtn=document.getElementsByClassName('tCibT qq7_A  z4xUb w5S7h')[0];
  if(morePostsBtn!==undefined){
    morePostsBtn.click();
    clearInterval(interval_3);
  }
  else{
    if(document.querySelector("a[href*=\"/related_profiles/\"]")){
      let relatedAccountsSection=document.querySelector("a[href*=\"/related_profiles/\"]").parentNode.parentNode;
      var closeButtons=relatedAccountsSection.querySelectorAll("button[aria-label][alt]");
      if(closeButtons.length>0){
        for(var i=0;i<closeButtons.length;i++){
          closeButtons[i].click();
        }
        clearInterval(interval_3);
      }
    }
    else{
      clearInterval(interval_3);
    }
  }
}
function removePicturePopup(){
  if(!document.location.href.includes("/p/")){
    var photoLinks=document.getElementsByClassName("_aabd _aa8k _aanf");
    for(var i=0;i<photoLinks.length;i++){
      if(photoLinks[i].getAttribute("class").includes("repaired")===false){
        if(photoLinks[i].getElementsByTagName("IMG")[0].getAttribute("src")!==null){
          var cln = photoLinks[i].children[0].cloneNode(true);
          photoLinks[i].removeChild(photoLinks[i].children[0]);
          photoLinks[i].appendChild(cln);
          photoLinks[i].setAttribute("class",(photoLinks[i].getAttribute("class")+" repaired"));
          photoLinks[i].children[0].setAttribute("target","_blank");
          photoLinks[i].children[0].setAttribute("rel","noopener noreferrer");
          photoLinks[i].children[0].children[0].children[0].children[0].style.visibility="visible";
        }
      }
    }
  }
}
function removePicturePopupIGTV(){
  console.log("removing picture popup from igtv");
  if(!document.location.href.includes("/tv/")){
    var videoLinks=document.getElementsByClassName("_bz0w");
    for(var i=0;i<videoLinks.length;i++){
      if(videoLinks[i].getAttribute("class").includes("repaired")===false){
        if(videoLinks[i].getElementsByClassName("lVhHa RNL1l")[0].style.backgroundSrc!==null){
          var cln = videoLinks[i].cloneNode(true);
          cln.setAttribute("class",(videoLinks[i].getAttribute("class")+" repaired"));
          insertAfter(cln,videoLinks[i]);
          videoLinks[i].remove();
        }
      }
    }
  }
}
//this function creates a note that suggests the usage of external services to view Instagram pages, as Instagram is devoting all the possible effort to stop people from watching its content without registering.
//Current websites listed as external services have been chosen just because they work reliably enough to be used ()
function igLoadOutsideNote(){
  let note=document.createElement("DIV");
  note.id="extSvcNote";
  let noteText=document.createElement("SPAN");
  let link_1=document.createElement("A");
  let link_2=document.createElement("A");
  noteText.textContent=l.igBanNote;
  note.appendChild(noteText);
  var username="";
  var authorId="";
  var currentURL=window.location.toString();
  if(document.querySelector("[role=tablist]")){ //if one of the main profile pages
    if(currentURL.lastIndexOf("\/")==currentURL.length-1){
      currentURL=currentURL.substring(0,currentURL.length-1);
    }
    username=currentURL.slice(currentURL.indexOf("instagram.com/")+14);
    link_1.href="https://www.picnob.com/profile/"+username;
    link_1.textContent=link_1.href;
    link_2.href="https://imginn.com/"+username;
    link_2.textContent=link_2.href;
    link_2.textContent=link_2.href;
  }
  else if(document.location.href.includes('/accounts/login/?next=')){ //if it's the forced login page
    link_1.href="https://www.picnob.com/";
    link_2.href="https://imginn.com/";
    link_1.textContent=link_1.href;
    link_2.textContent=link_2.href;
    currentURL=currentURL.substring(currentURL.indexOf("next=")+5,currentURL.length);
    if(currentURL.match(/[/]/g).length==2){
      username=currentURL.slice(1,currentURL.length-1);
      link_1.href=link_1.href+"u/"+username;
      link_1.textContent=link_1.href;
      link_2.href=link_2.href+username;
      link_2.textContent=link_2.href;
    }
  }
  else{
    link_1.href="https://www.picnob.com/";
    link_1.textContent=link_1.href;
    link_2.href="https://imginn.com";
    link_2.textContent=link_2.href;
  }
  note.appendChild(document.createElement("BR"));
  note.appendChild(link_1);
  note.appendChild(document.createElement("BR"));
  note.appendChild(link_2);
  console.log(note);
  const footer=document.body.getElementsByTagName("footer")[0];
  footer.insertBefore(note,footer.firstChild);
  try{
    note.setAttribute("style","border: 4px solid #E79; border-radius: 1em; padding: 1em; margin: 2em auto 2em auto; max-width: 935px"); //style doesn't apply and I don't know why
  }
  catch(e){
    console.log(e)
  }
}
function insertAfter(newNode, existingNode) {
    existingNode.parentNode.insertBefore(newNode, existingNode.nextSibling);
}

/*function blockPopupIG(){ //kept for reference
  if(document.getElementsByClassName("RnEpo")[0]!==undefined){
    setScrollable();
  }
}*/
/*function setScrollable(){ //kept for reference
  document.body.style.overflow = "scroll !important";
}*/
function blockBannerTW(s){
  /*if(s=='d'){
  	document.getElementsByClassName("css-1dbjc4n r-16y2uox r-1n0xq6e")[0].parentNode.parentNode.parentNode.parentNode.remove();
  }
  else{
    var banner=document.getElementsByClassName("css-1dbjc4n r-1awozwy r-1pz39u2 r-18u37iz r-16y2uox")[0].parentNode;
    if(banner!=='undefined'){
      banner.parentNode.style.height="53px";
      banner.remove();
    }
    banner=document.getElementsByClassName("css-1dbjc4n r-urgr8i r-97e31f")[0];
    if(banner!=='undefined'){
      banner.remove();
    }
  }*/
  document.childNodes[1].style.overflow='';
  //Thanks to Paul Wynters for the following code:
  //////////////////////////////////////////////////
  window.addEventListener('load', (event) => {
    window.setTimeout(() => { //wait a short amount of time for twitter to set overflow:hidden
      document.childNodes[1].style.overflow=''; //html element, clear style
    }, 500);
  });
  //////////////////////////////////////////////////
  if(mutationObserver_0===undefined){
    function callback_a(mutationList) {
      mutationList.forEach(function(mutation) {
        let largeLP;
        //IF EDITING THIS LIST OF POPUPS, KEEP IT IN SYNC WITH THE POPUPS LISTED IN CUSTOM CSS
        largeLP=document.querySelector(".css-1dbjc4n.r-aqfbo4.r-1d2f490.r-12vffkv.r-1xcajam.r-zchlnj.r-ipm5af");//big login popup triggered when clicking on "like" or "comment" (mobile)
        if(largeLP===null){
          largeLP=document.querySelector(".css-1dbjc4n.r-1awozwy.r-1kihuf0.r-18u37iz.r-1pi2tsx.r-1777fci.r-1pjcn9w.r-1xcajam.r-ipm5af.r-g6jmlv");//big login popup triggered when clicking on "like" or "comment" (desktop);
        }
        if(largeLP===null){
          largeLP=document.querySelector(".css-1dbjc4n.r-1awozwy.r-14lw9ot.r-1dgieki.r-1efd50x.r-5kkj8d.r-18u37iz.r-16y2uox.r-1a1dyw.r-1swwhx3.r-1j3t67a.r-1qxgc49");//another popup (i don't remember which one)
        }
        if(largeLP===null){
          largeLP=document.querySelector("#layers>.css-1dbjc4n.r-aqfbo4.r-1p0dtai.r-1d2f490.r-12vffkv.r-1xcajam.r-zchlnj>.css-1dbjc4n.r-12vffkv>.css-1dbjc4n.r-12vffkv>.css-1dbjc4n.r-l5o3uw");//another popup (i don't remember which one)
        }
        if(largeLP!==null){
          if(mutation.target.style.overflow=="hidden"){//definitely not the right way to do it, but setting the style attributes with mutation.target.style doesn't work.
            let restyle=mutation.target.getAttribute("style");
            restyle=restyle.replace("overflow: hidden","overflow: auto scroll");
            restyle=restyle.replace("margin-right: 17px","margin-right: auto");
            restyle=restyle.replace("overscroll-behavior-y: none;","overscroll-behavior-y: auto");
            restyle=restyle.replace("position: fixed","position: unset");
           	mutation.target.setAttribute("style",restyle);
          }
        }
      });
    }
    mutationObserver_0 = new MutationObserver(callback_a);
    mutationObserver_0.observe(document.documentElement, {attributes: true});
  }
}
function redditShadowRootPatch(){
  console.log("Patching reddit's shadowDom elements and restoring hidden content");
  let blurredContainers=document.querySelectorAll("shreddit-blurred-container");
  let containers=document.querySelectorAll("xpromo-nsfw-blocking-container");
  if(containers.length>0){
    for(let c of containers){
      let stl=document.createElement("STYLE");
      stl.innerHTML=".prompt {display: none !important}";
      c.shadowRoot.appendChild(stl);
    }
  }
  if(blurredContainers.length>0){
    for(let b of blurredContainers){
      b.removeAttribute("blurred");
      let stl=document.createElement("STYLE");
      stl.innerHTML=" *{filter: none !important}";
      console.log("SHADOWROOT");
      console.log(b.shadowRoot);
      b.shadowRoot.appendChild(stl);
    }
  }
}
function redditPatch(isMobile){
  redditShadowRootPatch();
  if(document.getElementsByClassName("NavFrame__below-top-nav").length>0){
    const nav=document.getElementsByClassName("NavFrame__below-top-nav")[0];
    const config = { attributes: false, childList: true, subtree: true };
    var firstCheck=true;
    var redditPic=document.body.getElementsByTagName("IMG");
    const callback = function(mutationsList, observer) {
      for(const mutation of mutationsList) {
        redditShadowRootPatch();
        if(isMobile){
          let articles=mutation.target.getElementsByTagName("ARTICLE");
          if(articles.length>0){
            for (const article of articles){
              if(!article.classList.contains("replaced")){
                if((article.getElementsByTagName("video").length==0)&&(article.getElementsByClassName("slideImageMainDiv").length==0)){
                  var articleCopy=article.cloneNode(true);
                  articleCopy.classList.add("replaced");
                  article.replaceWith(articleCopy);
                }
              }
            }
          }
        }
        if(firstCheck==false){
          redditPic=mutation.target.getElementsByTagName("IMG");
        }
        if(redditPic.length>0){
          for(var i=0;i<redditPic.length;i++){
            if((redditPic[i].getAttribute("src").includes("blur="))&&(redditPic[i].getAttribute("src").includes("/preview."))){
              redditPic[i].setAttribute("src",redditPic[i].getAttribute("src").split("?")[0].replace("preview","i"));
            }
          }
          firstCheck=false;
        }
      }
    };
    const bodyObserver = new MutationObserver(callback);
    bodyObserver.observe(nav, config);
  }
  var previews=document.querySelectorAll("[slot=blurred] img[src^='https://preview.']");
  for(let p of previews){
    p.src=p.src.replace("preview.redd.it","i.redd.it");
  }
  var previewLinks=document.querySelectorAll("[bundlename=media_player_loader]")
  for(let pL of previewLinks){
    pL.parentNode.removeAttribute("href");
  }
  //Thanks to https://greasyfork.org/en/users/730548-spacex for the following code
  var bodyChildren = document.getElementsByTagName('body')[0].children;
  for (var i = 0; i < bodyChildren.length; i++) {
    try {
      if (bodyChildren[i].style.backdropFilter){
          bodyChildren[i].style.backdropFilter = 'none';
      }
    }
    catch (err){}
  }
  //Thanks!
}
function quoraPatchObserver(){
  var firstCheck=true;
  var searchBox=document.createElement("DIV");
  searchBox.className="q-flex qu-alignItems--center qu-justifyContent--center";
  searchBox.style="box-sizing: border-box; display: flex; max-width: 100%;";
  let a=document.createElement("INPUT");
  a.id="search";
  a.style="height: 44px; text-align: center; border-radius: 22px; background-color:white; width: 100%; margin: 0.5em";
  a.className="selector_input text";
  a.value="";
  a.setAttribute("data-lpignore","true");
  a.setAttribute("data-group","js-editable");
  a.setAttribute("placeholder","Search");
  a.setAttribute("w2cid","wS7KcEIg18");
  searchBox.appendChild(a);
  searchBox.addEventListener('keypress', function(e){searchQuora(e);});
  try{
  	document.getElementsByClassName("q-sticky qu-zIndex--header")[0].appendChild(searchBox);
  }
  catch(e){
    try{
      document.getElementsByClassName("header_main")[0].appendChild(searchBox);
      document.getElementsByClassName("query_title")[0].style.marginTop="2em";
    }catch(err){console.log("Info: if you are browsing from a PC, this error is expected.");}
  }
  quoraPatch(firstCheck);
  firstCheck=false;
  const config = { attributes: true, childList: true, subtree: false };//this rough implementation is slower
  const callback = function(mutationsList, observer) {
    for(const mutation of mutationsList) {
				quoraPatch(firstCheck);
    }
  };
  const bodyObserver = new MutationObserver(callback);
  bodyObserver.observe(document.body, config);
}
function quoraPatch(firstCheck){
  var i;
  var wall;
  if(firstCheck==true){
    wall=document.getElementsByClassName("q-box qu-overflow--hidden");
    if(wall.length>1){
      let bgColor=getComputedStyle(wall[0].parentNode.children[0]).backgroundColor;
      if(bgColor=="rgba(68, 68, 68, 0.85)"){
        wall[0].parentNode.children[0].remove();
        wall[0].classList.remove("qu-overflow--hidden");
      }
    }
  }
  if(document.body.className.includes('signup_wall_prevent_scroll')){
    document.body.classList.remove('signup_wall_prevent_scroll');
  }
  wall=document.getElementsByClassName('new_web_signup_wall_design');
  if(wall.length>0){
    wall[0].parentNode.parentNode.remove();//desktop
  }
  wall=document.getElementsByClassName('modal_bg new_signup_dialog');
  if(wall.length>0){
    wall[0].parentNode.remove();//mobile
  }
  wall=document.getElementsByClassName('q-absolute qu-full qu-bg--blue');
  if(wall.length>0){
    wall[0].parentNode.remove();//mobile
  }
}
function searchQuora(e) {
  if (e.key === 'Enter') {
    window.open("https://www.quora.com/search?q="+document.getElementById("search").value,"_self");
  }
}
function checkVKLoginPopup(mode){
  const popups=document.getElementsByClassName("UnauthActionBoxContainer");
  if(popups.length>0){
    removeVKLoginPopup(popups[0]);
  }
  const config = { attributes: false, childList: true, subtree: false };
  const callback = function(mutationsList, observer) {
    for(const mutation of mutationsList) {
      for(let newNode of mutation.addedNodes){
      	if(newNode.className.includes("UnauthActionBoxContainer")){
          removeVKLoginPopup(newNode,mode);
        }
        else if(newNode.getElementsByClassName("UnauthorizedActionModal")!==undefined){
          removeVKLoginPopup(newNode,mode);
        }
      }
    }
  };
  const observer = new MutationObserver(callback);
  if(mode=="d"){
  	observer.observe(document.getElementById("box_layer"), config);
  }
  else{
    observer.observe(document.body, config);
  }
}
function removeVKLoginPopup(popup,mode){
  popup.remove();
  console.log("Popup removed");
  if(mode=="d"){
  	document.getElementById("box_layer_bg").remove();
  	document.getElementById("box_layer_wrap").remove();
  }
  else{
    document.getElementById("popup_wrap").remove();
		document.getElementById("popup_overlay").remove();
  }
}
function allowVideoReplayStarterIG(){
  var imgBox=document.getElementsByClassName("pR7Pc");
  if(imgBox.length==0){
    imgBox=document.getElementsByClassName("kPFhm B1JlO OAXCp");
  }
  if(imgBox.length==0){
    imgBox=document.getElementsByClassName("_aatk");
  }
  if(imgBox.length>0){
    let videoPlayer=imgBox[0].getElementsByTagName("VIDEO");
    if(videoPlayer.length>0){
      let moveTagCss=document.createElement("style");
      moveTagCss.textContent=".LcKDX, ._7zNgw.GLdVF {bottom: 3em}";
      document.body.appendChild(moveTagCss);
      for (let vp of videoPlayer){
        vp.controls=true;
      }
    }
  	allowVideoReplayIG(imgBox[0]);
    clearInterval(interval_0);
  }
}
function allowVideoReplayIG(imgBox){
  console.log("add video controls");
  const config = { attributes: false, childList: true, subtree: true };
  const callback = function(mutationsList, observer) {
    for(const mutation of mutationsList) {
      for(let newNode of mutation.addedNodes){
				let videoPlayer=newNode.getElementsByTagName("VIDEO");
        if(videoPlayer.length>0){
          for (var vp of videoPlayer){
            vp.controls=true;
          }
        }
      }
    }
  };
  const observer = new MutationObserver(callback);
  observer.observe(imgBox, config);
}
function tumblrDeClassify(){
  const config = { attributes: false, childList: true, subtree: false };
  const callback = function(mutationsList, observer) {
    for(const mutation of mutationsList) {
      for(let newNode of mutation.addedNodes){
        if(newNode.querySelector("[href^='/auth/']")){
          newNode.remove();
        }
      }
    }
  };
  const observer = new MutationObserver(callback);
  observer.observe(document.getElementById("glass-container"), config);
}
function tumblrFreeScroll(){
  unsafeWindow.scrollTo=function(){};
  unsafeWindow.scrollY=function(){};
  unsafeWindow.scrollX=function(){};
}
function tiktok_waitReady(){
  let lockStyle=document.querySelector(".tux-base-dialog.z-2000");
  if(lockStyle!==null){
    tiktok_removeAnnoyances(lockStyle);
    clearInterval(interval_0);
  }
}
function tiktok_clickList(){
  let list=document.getElementsByTagName("UL");
  if(list.length>0){
    if(list[0].getElementsByTagName("LI").length>0){
      tiktok_openLinksWithoutApp(list[0]);
      clearInterval(interval_1);
    }
  }
}
function tiktok_removeAnnoyances(lockStyle){
  lockStyle.remove();
  var possibleBanners=document.body.querySelectorAll("div[data-theme=light]");
  console.log(possibleBanners);
  for(let i=possibleBanners.length-1;i>=0;i--){//reverse check so if one element is parent of another, the child will be removed
    let res=possibleBanners[i].querySelector("g[clip-path*=TikTok_Logo_Dark]");
    if(res!==null){
      possibleBanners[i].remove();
      break;
    }
  }
}
function tiktok_openLinksWithoutApp(list){
    var recommended=document.body.getElementsByClassName("recommend-item");
    for(let r of recommended){
      r.addEventListener("click", function(event){
        event.stopPropagation();
        window.stop();
        window.open(event.target.getAttribute("href"));
        window.close();

      });
      r.setAttribute("href",r.getElementsByTagName("A")[0].href);
    }
    const targetNode = list;
    const config = { attributes: false, childList: true, subtree: true };
    const callback = (mutationList, observer) => {
        for (const mutation of mutationList) {
          console.log(mutation.target);
          if(mutation.target.classList.includes("recommend-item")){
            mutation.target.addEventListener("click", function(event){
              event.stopPropagation();
              document.location.replace(event.target.getAttribute("href"));
            });
            mutation.target.setAttribute("href",r.getElementsByTagName("A")[0].href);
          }
        }
    };
    const observer = new MutationObserver(callback);
    observer.observe(targetNode, config);
}