Instagram Source Opener

Open the original source of an IG post, story or profile picture. No jQuery

目前为 2020-12-26 提交的版本。查看 最新版本

// ==UserScript==
// @name         Instagram Source Opener
// @version      1.1.13
// @description  Open the original source of an IG post, story or profile picture. No jQuery
// @author       jomifepe
// @icon         https://www.instagram.com/favicon.ico
// @match        https://www.instagram.com/*
// @grant        GM_xmlhttpRequest
// @grant        GM.xmlHttpRequest
// @grant        GM_registerMenuCommand
// @grant        GM.registerMenuCommand
// @grant        GM_getValue
// @grant        GM.getValue
// @grant        GM_setValue
// @grant        GM.setValue
// @grant        GM_openInTab
// @grant        GM.openInTab
// @connect      instagram.com
// @connect      instadp.com
// @connect      instadp.org
// @connect      izuum.com
// @website      https://github.com/jomifepe
// @homepage     https://gf.qytechs.cn/users/192987
// @supportURL   https://gf.qytechs.cn/en/scripts/372366-instagram-source-opener/feedback
// @namespace https://gf.qytechs.cn/users/192987
// ==/UserScript==

/* jshint esversion: 8 */
(function () {
  'use strict';

  const LOGGING_ENABLED = false;

  const SCRIPT_NAME = 'Instagram Source Opener';
  const SCRIPT_NAME_SHORT = 'ISO';
  const LOGGING_TAG = `[${SCRIPT_NAME_SHORT}]`;
  const HOMEPAGE_URL = 'https://gf.qytechs.cn/en/scripts/372366-instagram-source-opener';

  /* Instagram classes and selectors */
  const IG_S_STORY_CONTAINER = '.yS4wN,.vUg3G';
  const IG_S_SINGLE_POST_CONTAINER = '.JyscU';
  const IG_S_PROFILE_CONTAINER = '.v9tJq';
  const IG_S_STORY_MEDIA_CONTAINER = '.qbCDp';
  const IG_S_POST_IMG = '.FFVAD';
  const IG_S_POST_VIDEO = '.tWeCl';
  const IG_S_MULTI_POST_LIST_ITEMS = '.vi798 .Ckrof';
  const IG_S_MULTI_POST_PREV_ARROW_BTN = '.POSa_';
  const IG_S_MULTI_POST_NEXT_ARROW_BTN = '._6CZji';
  const IG_S_POST_CONTAINER = '._8Rm4L';
  const IG_S_POST_BUTTONS = '.eo2As > section';
  const IG_S_PROFILE_PIC_CONTAINER = '.RR-M-';
  const IG_S_PRIVATE_PROFILE_PIC_CONTAINER = '.M-jxE';
  const IG_S_PRIVATE_PIC_IMG_CONTAINER = '._2dbep';
  const IG_S_PRIVATE_PROFILE_PIC_IMG_CONTAINER = '.IalUJ';
  const IG_S_PROFILE_USERNAME_TITLE = '.fKFbl';
  const IG_S_POST_BLOCKER = '._9AhH0';
  const IG_S_TOP_BAR = '.Hz2lF';
  const IG_S_POST_TIME_ANCHOR = '.c-Yi7';
  const IG_S_MULTI_POST_INDICATOR = '.Yi5aA';
  const IG_C_MULTI_POST_INDICATOR_ACTIVE = 'XCodT';

  /* Custom classes and selectors */
  const C_BTN_STORY = 'iso-story-btn';
  const C_BTN_STORY_CONTAINER = 'iso-story-container';
  const C_POST_WITH_BUTTON = 'iso-post';
  const C_BTN_POST_OUTER_SPAN = 'iso-post-container';
  const C_BTN_POST = 'iso-post-btn';
  const C_BTN_POST_INNER_SPAN = 'iso-post-span';
  const C_BTN_PROFILE_PIC_CONTAINER = 'iso-profile-pic-container';
  const C_BTN_PROFILE_PIC = 'iso-profile-picture-btn';
  const C_BTN_PROFILE_PIC_SPAN = 'iso-profile-picture-span';

  const C_SETTINGS_PREFFIX = 'iso-settings';
  const C_SETTINGS_CONTAINER = `${C_SETTINGS_PREFFIX}-container`;
  const C_SETTINGS_BTN = `${C_SETTINGS_PREFFIX}-btn`;
  const C_SETTINGS_MENU = `${C_SETTINGS_PREFFIX}-menu`;
  const C_SETTINGS_MENU_TITLE_CONTAINER = `${C_SETTINGS_PREFFIX}-menu-title-container`;
  const C_SETTINGS_MENU_TITLE = `${C_SETTINGS_PREFFIX}-menu-title`;
  const C_SETTINGS_MENU_TITLE_LINK = `${C_SETTINGS_PREFFIX}-menu-title-link`;
  const C_SETTINGS_MENU_TITLE_CLOSE_BTN = `${C_SETTINGS_PREFFIX}-menu-close-btn`;
  const C_SETTINGS_MENU_OPTIONS = `${C_SETTINGS_PREFFIX}-menu-options`;
  const C_SETTINGS_MENU_OPTION = `${C_SETTINGS_PREFFIX}-menu-option`;
  const C_SETTINGS_MENU_OPTION_BTN = `${C_SETTINGS_PREFFIX}-menu-option-button`;
  const ID_SETTINGS_POST_STORY_KB_BTN = `${C_SETTINGS_PREFFIX}-post-story-kb-btn`;
  const ID_SETTINGS_PROFILE_PICTURE_KB_BTN = `${C_SETTINGS_PREFFIX}-profile-picture-kb-btn`;
  const ID_SETTINGS_BUTTON_BEHAVIOR_SELECT = `${C_SETTINGS_PREFFIX}-button-behavior-select`;

  const S_IG_POST_CONTAINER_WITHOUT_BUTTON = `${IG_S_POST_CONTAINER}:not(.${C_POST_WITH_BUTTON})`;

  /* Storage keys */
  const STORAGE_KEY_POST_STORY_KB = 'iso_post_story_kb';
  const STORAGE_KEY_PROFILE_PICTURE_KB = 'iso_profile_picture_kb';
  const STORAGE_KEY_BUTTON_BEHAVIOR = 'iso_button_behavior';

  /* Default letters for key bindings */
  const DEFAULT_KB_POST_STORY = 'O';
  const DEFAULT_KB_PROFILE_PICTURE = 'P';
  const BUTTON_BEHAVIOR_REDIR = 'bb_redirect';
  const BUTTON_BEHAVIOR_NEW_TAB_FOCUS = 'bb_tab_focus';
  const BUTTON_BEHAVIOR_NEW_TAB_BG = 'bb_tab_background';
  const DEFAULT_BUTTON_BEHAVIOR = BUTTON_BEHAVIOR_NEW_TAB_FOCUS;
  const BUTTON_BEHAVIOR_OPTIONS = [BUTTON_BEHAVIOR_REDIR, BUTTON_BEHAVIOR_NEW_TAB_FOCUS, BUTTON_BEHAVIOR_NEW_TAB_BG];

  const URL_IG_POST_INFO_API = postRelUrl => `https://www.instagram.com${postRelUrl}?__a=1`;
  const URL_INSTA_DP_COM = username => `https://www.instadp.com/fullsize/${username}`;
  const URL_INSTA_DP_ORG = 'https://instadp.org/';
  const URL_IZUUM = 'https://izuum.com/index.php';

  const trigger = {
    ARRIVE: 'arrive',
    LEAVE: 'leave',
  };

  /* eslint-disable no-undef */
  const GMFunc = {
    getValue: ['GM_getValue', 'GM.getValue'],
    setValue: ['GM_setValue', 'GM.setValue'],
    registerMenuCommand: ['GM_registerMenuCommand', 'GM.registerMenuCommand'],
    openInTab: ['GM_openInTab', 'GM.openInTab'],
    xmlHttpRequest: ['GM_xmlhttpRequest', 'GM.xmlHttpRequest'],
  };
  /* eslint-enable no-undef */

  let isStoryKeyBindingSetup, isSinglePostKeyBindingSetup, isProfileKeyBindingSetup;
  let openPostStoryKeyBinding = DEFAULT_KB_POST_STORY;
  let openProfilePictureKeyBinding = DEFAULT_KB_PROFILE_PICTURE;
  let openSourceBehavior = DEFAULT_BUTTON_BEHAVIOR;

  //#region Logging utilities

  const log = (...args) => LOGGING_ENABLED && console.log(...[LOGGING_TAG, ...args]);
  const error = (...args) => LOGGING_ENABLED && console.error(...[LOGGING_TAG, ...args]);
  const warn = (...args) => LOGGING_ENABLED && console.warn(...[LOGGING_TAG, ...args]);
  const message = (...args) => alert(`${SCRIPT_NAME}:\n\n${args.join(' ')}`);
  const errorMessage = (msg, ...errorArgs) => {
    if (LOGGING_ENABLED) error(msg, ...errorArgs);
    message(msg);
  };

  //#endregion

  /* Arrive.js library - https://github.com/uzairfarooq/arrive */
  /* eslint-disable */
  /* prettier-ignore */
  const Arrive=function(e,t,n){'use strict';function r(e,t,n){l.addMethod(t,n,e.unbindEvent),l.addMethod(t,n,e.unbindEventWithSelectorOrCallback),l.addMethod(t,n,e.unbindEventWithSelectorAndCallback);}function i(e){e.arrive=f.bindEvent,r(f,e,'unbindArrive'),e.leave=d.bindEvent,r(d,e,'unbindLeave');}if(e.MutationObserver&&'undefined'!=typeof HTMLElement){var o=0,l=function(){var t=HTMLElement.prototype.matches||HTMLElement.prototype.webkitMatchesSelector||HTMLElement.prototype.mozMatchesSelector||HTMLElement.prototype.msMatchesSelector;return{matchesSelector:function(e,n){return e instanceof HTMLElement&&t.call(e,n);},addMethod:function(e,t,r){var i=e[t];e[t]=function(){return r.length==arguments.length?r.apply(this,arguments):'function'==typeof i?i.apply(this,arguments):n;};},callCallbacks:function(e,t){t&&t.options.onceOnly&&1==t.firedElems.length&&(e=[e[0]]);for(var n,r=0;n=e[r];r++)n&&n.callback&&n.callback.call(n.elem,n.elem);t&&t.options.onceOnly&&1==t.firedElems.length&&t.me.unbindEventWithSelectorAndCallback.call(t.target,t.selector,t.callback);},checkChildNodesRecursively:function(e,t,n,r){for(var i,o=0;i=e[o];o++)n(i,t,r)&&r.push({callback:t.callback,elem:i}),i.childNodes.length>0&&l.checkChildNodesRecursively(i.childNodes,t,n,r);},mergeArrays:function(e,t){var n,r={};for(n in e)e.hasOwnProperty(n)&&(r[n]=e[n]);for(n in t)t.hasOwnProperty(n)&&(r[n]=t[n]);return r;},toElementsArray:function(t){return n===t||'number'==typeof t.length&&t!==e||(t=[t]),t;}};}(),c=function(){var e=function(){this._eventsBucket=[],this._beforeAdding=null,this._beforeRemoving=null;};return e.prototype.addEvent=function(e,t,n,r){var i={target:e,selector:t,options:n,callback:r,firedElems:[]};return this._beforeAdding&&this._beforeAdding(i),this._eventsBucket.push(i),i;},e.prototype.removeEvent=function(e){for(var t,n=this._eventsBucket.length-1;t=this._eventsBucket[n];n--)if(e(t)){this._beforeRemoving&&this._beforeRemoving(t);var r=this._eventsBucket.splice(n,1);r&&r.length&&(r[0].callback=null);}},e.prototype.beforeAdding=function(e){this._beforeAdding=e;},e.prototype.beforeRemoving=function(e){this._beforeRemoving=e;},e;}(),a=function(t,r){var i=new c,o=this,a={fireOnAttributesModification:!1};return i.beforeAdding(function(n){var i,l=n.target;(l===e.document||l===e)&&(l=document.getElementsByTagName('html')[0]),i=new MutationObserver(function(e){r.call(this,e,n);});var c=t(n.options);i.observe(l,c),n.observer=i,n.me=o;}),i.beforeRemoving(function(e){e.observer.disconnect();}),this.bindEvent=function(e,t,n){t=l.mergeArrays(a,t);for(var r=l.toElementsArray(this),o=0;o<r.length;o++)i.addEvent(r[o],e,t,n);},this.unbindEvent=function(){var e=l.toElementsArray(this);i.removeEvent(function(t){for(var r=0;r<e.length;r++)if(this===n||t.target===e[r])return!0;return!1;});},this.unbindEventWithSelectorOrCallback=function(e){var t,r=l.toElementsArray(this),o=e;t='function'==typeof e?function(e){for(var t=0;t<r.length;t++)if((this===n||e.target===r[t])&&e.callback===o)return!0;return!1;}:function(t){for(var i=0;i<r.length;i++)if((this===n||t.target===r[i])&&t.selector===e)return!0;return!1;},i.removeEvent(t);},this.unbindEventWithSelectorAndCallback=function(e,t){var r=l.toElementsArray(this);i.removeEvent(function(i){for(var o=0;o<r.length;o++)if((this===n||i.target===r[o])&&i.selector===e&&i.callback===t)return!0;return!1;});},this;},s=function(){function e(e){var t={attributes:!1,childList:!0,subtree:!0};return e.fireOnAttributesModification&&(t.attributes=!0),t;}function t(e,t){e.forEach(function(e){var n=e.addedNodes,i=e.target,o=[];null!==n&&n.length>0?l.checkChildNodesRecursively(n,t,r,o):'attributes'===e.type&&r(i,t,o)&&o.push({callback:t.callback,elem:i}),l.callCallbacks(o,t);});}function r(e,t){return l.matchesSelector(e,t.selector)&&(e._id===n&&(e._id=o++),-1==t.firedElems.indexOf(e._id))?(t.firedElems.push(e._id),!0):!1;}var i={fireOnAttributesModification:!1,onceOnly:!1,existing:!1};f=new a(e,t);var c=f.bindEvent;return f.bindEvent=function(e,t,r){n===r?(r=t,t=i):t=l.mergeArrays(i,t);var o=l.toElementsArray(this);if(t.existing){for(var a=[],s=0;s<o.length;s++)for(var u=o[s].querySelectorAll(e),f=0;f<u.length;f++)a.push({callback:r,elem:u[f]});if(t.onceOnly&&a.length)return r.call(a[0].elem,a[0].elem);setTimeout(l.callCallbacks,1,a);}c.call(this,e,t,r);},f;},u=function(){function e(){var e={childList:!0,subtree:!0};return e;}function t(e,t){e.forEach(function(e){var n=e.removedNodes,i=[];null!==n&&n.length>0&&l.checkChildNodesRecursively(n,t,r,i),l.callCallbacks(i,t);});}function r(e,t){return l.matchesSelector(e,t.selector);}var i={};d=new a(e,t);var o=d.bindEvent;return d.bindEvent=function(e,t,r){n===r?(r=t,t=i):t=l.mergeArrays(i,t),o.call(this,e,t,r);},d;},f=new s,d=new u;t&&i(t.fn),i(HTMLElement.prototype),i(NodeList.prototype),i(HTMLCollection.prototype),i(HTMLDocument.prototype),i(Window.prototype);var h={};return r(f,h,'unbindAllArrive'),r(d,h,'unbindAllLeave'),h;}}(window,'undefined'==typeof jQuery?null:jQuery,void 0);
  /* eslint-enable */

  const pages = {
    feed: {
      isVisible: () => window.location.pathname === '/',
      onLoadActions: () => {
        document.querySelectorAll(S_IG_POST_CONTAINER_WITHOUT_BUTTON).forEach(node => generatePostButton(node));
      },
    },
    story: {
      isVisible: () => window.location.pathname.startsWith('/stories/'),
      onLoadActions: () => {
        const node = document.querySelector(IG_S_STORY_CONTAINER);
        if (!node) return;
        generateStoryButton(node);
        setupStoryEventListeners();
      },
    },
    profile: {
      isVisible: () => window.location.pathname.length > 1 && document.querySelector(IG_S_PROFILE_CONTAINER),
      onLoadActions: () => {
        const node = document.querySelector(IG_S_PROFILE_CONTAINER);
        if (!node) return;
        generateProfilePictureButton(node);
        setupProfileEventListeners();
      },
    },
    post: {
      isVisible: () => window.location.pathname.startsWith('/p/'),
      onLoadActions: () => {
        const node = document.querySelector(IG_S_SINGLE_POST_CONTAINER);
        if (!node) return;
        generatePostButton(node);
        setupSinglePostEventListeners();
      },
    },
  };

  const actionTriggers = {
    [trigger.ARRIVE]: {
      /* triggered whenever a new instagram post is loaded on the feed */
      [S_IG_POST_CONTAINER_WITHOUT_BUTTON]: node => generatePostButton(node),
      /* triggered whenever a single post is opened (on a profile) */
      [IG_S_SINGLE_POST_CONTAINER]: node => {
        generatePostButton(node);
        setupSinglePostEventListeners();
      },
      /* triggered whenever a story is opened */
      [IG_S_STORY_CONTAINER]: node => {
        generateStoryButton(node);
        setupStoryEventListeners();
      },
      /* triggered whenever a profile page is loaded */
      [IG_S_PROFILE_CONTAINER]: node => {
        generateProfilePictureButton(node);
        setupProfileEventListeners();
      },
      /* triggered whener the top bar is created */
      [IG_S_TOP_BAR]: generateSettingsPageMenu,
    },
    [trigger.LEAVE]: {
      /* triggered whenever a single post is closed (on a profile) */
      [IG_S_SINGLE_POST_CONTAINER]: removeSinglePostEventListeners,
      /* triggered whenever a story is closed */
      [IG_S_STORY_CONTAINER]: removeStoryEventListeners,
      /* triggered whenever a profile page is left */
      [IG_S_PROFILE_CONTAINER]: removeProfileEventListeners,
    },
  };

  const profilePictureSources = {
    'instadp.org': username => getProfilePictureFromInstadpDotOrg(username),
    'instadp.com': username => getProfilePictureFromInstadpDotCom(username),
    'izuum.com': username => getProfilePictureFromIzuum(username),
    'current page sharedData': username => getProfilePictureFromSharedData(username),
    'user data graphql API (?__a=1)': () => getProfilePictureFromUpdatedSharedData(),
    'updated HTML profile page': () => getProfilePictureFromUpdatedHTMLPage(),
  };

  //#region Script setup and on load actions

  registerMenuCommands(); /* register GM menu commands */
  injectStyles(); /* injects the needed CSS into DOM */
  setupTriggers(); /* setup arrive and leave triggers for elements */
  performOnLoadActions(); /* first load actions */
  window.onload = performOnLoadActions; /* first load actions (backup) */

  //#endregion

  /**
   * Setup the arrive and leave triggers for relevant elements
   */
  function setupTriggers() {
    let count = 0;
    for (const [event, triggers] of Object.entries(actionTriggers)) {
      for (const [actuator, fireTrigger] of Object.entries(triggers)) {
        document[event](actuator, node => {
          if (!node) return;
          fireTrigger(node);
          log(`Triggered ${event} for selector ${actuator}`);
        });
        count++;
      }
    }
    log(`Created ${count} element triggers`);
  }

  /**
   * Performs actions that need to be performed on page load.
   */
  function performOnLoadActions() {
    for (const [name, page] of Object.entries(pages)) {
      if (page.isVisible()) {
        page.onLoadActions();
        log(`Performed onload actions for ${name} page`);
      }
    }
    loadPreferences();
    generateSettingsPageMenu();
  }

  /**
   * Loads preferences that are needed on multiple occasions from the storage to the corresponding variables
   */
  async function loadPreferences() {
    const response = await callGMFunction(GMFunc.getValue, STORAGE_KEY_BUTTON_BEHAVIOR, DEFAULT_BUTTON_BEHAVIOR);
    if (!response || !BUTTON_BEHAVIOR_OPTIONS.includes(response)) {
      error('Failed to load open button behavior option');
      return;
    }
    openSourceBehavior = response;
    log('Loaded preference: Open button behavior:', response);
  }

  //#endregion

  //#region Settings menu

  /**
   * Creates the commands to appear on the menu created by the <Any>monkey extension that's being used
   * For example, on Tampermonkey, this menu is accessible by clicking on the extension icon
   */
  function registerMenuCommands() {
    callGMFunction(GMFunc.registerMenuCommand, 'Change post & story shortcut', handleMenuPostStoryKBCommand, null);
    callGMFunction(GMFunc.registerMenuCommand, 'Change profile picture shortcut', handleMenuProfilePicKBCommand, null);
    log('Registered menu commands');
  }

  /**
   * Handle the click on the settings menu option to change the single post and story opening key binding
   */
  async function handleMenuPostStoryKBCommand() {
    const kb = await handleKBMenuCommand(STORAGE_KEY_POST_STORY_KB, DEFAULT_KB_POST_STORY, 'single post and story');
    if (!kb) return;
    openPostStoryKeyBinding = kb;
    if (pages.post.isVisible()) {
      removeSinglePostEventListeners();
      setupSinglePostEventListeners();
      return;
    }
    if (pages.story.isVisible()) {
      removeStoryEventListeners();
      setupStoryEventListeners();
      return;
    }
  }

  /**
   * Handle the click on the settings menu option to change the profile picture opening key binding
   */
  async function handleMenuProfilePicKBCommand() {
    const kb = await handleKBMenuCommand(STORAGE_KEY_PROFILE_PICTURE_KB, DEFAULT_KB_PROFILE_PICTURE, 'profile picture');
    if (!kb) return;
    openProfilePictureKeyBinding = kb;
    removeProfileEventListeners();
    setupProfileEventListeners();
  }

  /**
   * Handle the click on the settings menu option to change the profile picture opening key binding
   * @param {string} option Button behavior option to use, has to be one of BUTTON_BEHAVIOR_OPTIONS
   */
  async function handleMenuButtonBehaviorChange(option) {
    if (!BUTTON_BEHAVIOR_OPTIONS.includes(option)) {
      error('Invalid option for source button behavior');
      return;
    }
    const result = await callGMFunction(GMFunc.setValue, STORAGE_KEY_BUTTON_BEHAVIOR, option);
    if (result === null) error('Failed to save button behavior option on storage');
    openSourceBehavior = option;
    log('Changed open source button behavior to', option);
  }

  /**
   * Generic handler for the click action on the key binding changing options of the settings menu.
   * Launches a prompt that asks the user for a new key binding for a specific action, saves it locally and returns it
   * @param {string} keyBindingStorageKey Unique name used to store the key binding
   * @param {string} defaultKeyBinding Default key binding, used on the prompt message
   * @param {string} keyBindingName Key binding name to show on log messages, just for context
   * @returns {Promise<string|null>} Promise object, returns either the new key binding or null if it failed
   */
  async function handleKBMenuCommand(keyBindingStorageKey, defaultKeyBinding, keyBindingName) {
    let currentKey = await callGMFunction(GMFunc.getValue, keyBindingStorageKey, defaultKeyBinding);
    if (currentKey == null) {
      currentKey = defaultKeyBinding;
      log(`Falling back to default key binding: Alt + ${defaultKeyBinding}`);
    }

    const newKeyBinding = prompt(
      `${SCRIPT_NAME}:\n\nKey binding to open a ${keyBindingName}:\n` +
        'Choose a letter to be combined with the Alt/⌥ key\n\n' +
        `Current key binding: Alt/⌥ + ${currentKey.toUpperCase()}`
    );
    if (newKeyBinding == null) return null;
    if (!isKeyBindingValid(newKeyBinding)) {
      errorMessage(`Couldn't save new key binding to open ${keyBindingName}, invalid option`);
      return null;
    }

    const successMessage = `Saved new shortcut to open ${keyBindingName}:\nAlt + ${newKeyBinding.toUpperCase()}`;
    const result = await callGMFunction(GMFunc.setValue, keyBindingStorageKey, newKeyBinding);
    if (result === null) return null;
    message(successMessage);
    return newKeyBinding;
  }

  /**
   * Changes the visibility of the page settings menu
   * @param {boolean} visible
   */
  function setSettingsMenuVisible(visible) {
    if (visible) {
      document.querySelector(`.${C_SETTINGS_CONTAINER}`).style.display = 'flex';
      /* load values on the menu */
      const buttonBehaviorSelect = document.querySelector(`#${ID_SETTINGS_BUTTON_BEHAVIOR_SELECT}`);
      if (buttonBehaviorSelect) buttonBehaviorSelect.value = openSourceBehavior;
    } else {
      document.querySelector(`.${C_SETTINGS_CONTAINER}`).style.display = 'none';
    }
  }

  //#endregion

  //#region Element creation

  /**
   * Creates a visual settings menu on the page, as an alternative to the commands menu method,
   * since it isn't supported by all extensions
   */
  function generateSettingsPageMenu() {
    if (!document.querySelector(`.${C_SETTINGS_BTN}`)) {
      /* Create the settings button */
      const button = document.createElement('button');
      button.classList.add(C_SETTINGS_BTN);
      button.setAttribute('type', 'button');
      button.setAttribute('title', `Open ${SCRIPT_NAME_SHORT} settings`);
      button.addEventListener('click', () => setSettingsMenuVisible(true));
      document.querySelector(IG_S_TOP_BAR)?.appendChild(button);
      log('Created script settings button');
    }

    if (!document.querySelector(`.${C_SETTINGS_CONTAINER}`)) {
      /* Create the settings menu */
      const menu = document.createElement('div');
      menu.innerHTML = `<div class="${C_SETTINGS_CONTAINER}"><div class="${C_SETTINGS_MENU}"><div class="${C_SETTINGS_MENU_TITLE_CONTAINER}"><div class="${C_SETTINGS_MENU_TITLE}">${SCRIPT_NAME_SHORT} Settings<a class="${C_SETTINGS_MENU_TITLE_LINK}" href="${HOMEPAGE_URL}" target="_blank" title="What's this?">(?)</a></div><button class="${C_SETTINGS_MENU_TITLE_CLOSE_BTN}" title="Close"><div class="coreSpriteClose"></div></button></div><div class="${C_SETTINGS_MENU_OPTIONS}"><button id="${ID_SETTINGS_POST_STORY_KB_BTN}" class="${C_SETTINGS_MENU_OPTION_BTN}">Change post/story shortcut</button> <button id="${ID_SETTINGS_PROFILE_PICTURE_KB_BTN}" class="${C_SETTINGS_MENU_OPTION_BTN}">Change profile picture shortcut</button><div class="${C_SETTINGS_MENU_OPTION}"><label for="${ID_SETTINGS_BUTTON_BEHAVIOR_SELECT}">Open source click behavior:</label> <select id="${ID_SETTINGS_BUTTON_BEHAVIOR_SELECT}"><option value="${BUTTON_BEHAVIOR_REDIR}">Redirect</option><option value="${BUTTON_BEHAVIOR_NEW_TAB_FOCUS}">New tab and focus</option><option value="${BUTTON_BEHAVIOR_NEW_TAB_BG}">New tab in the background</option></select></div></div></div></div>`;

      menu.querySelector(`.${C_SETTINGS_MENU}`)?.addEventListener('click', event => event.stopPropagation());
      menu.querySelector(`.${C_SETTINGS_CONTAINER}`)?.addEventListener('click', () => setSettingsMenuVisible(false));
      menu
        .querySelector(`.${C_SETTINGS_MENU_TITLE_CLOSE_BTN}`)
        ?.addEventListener('click', () => setSettingsMenuVisible(false));
      menu.querySelector(`#${ID_SETTINGS_POST_STORY_KB_BTN}`)?.addEventListener('click', handleMenuPostStoryKBCommand);
      menu
        .querySelector(`#${ID_SETTINGS_PROFILE_PICTURE_KB_BTN}`)
        ?.addEventListener('click', handleMenuProfilePicKBCommand);
      menu
        .querySelector(`#${ID_SETTINGS_BUTTON_BEHAVIOR_SELECT}`)
        ?.addEventListener('change', e => handleMenuButtonBehaviorChange(e.target.value));

      document.body.appendChild(menu);
      log('Created script settings menu');
    }
  }

  /**
   * Appends new elements to DOM containing the story source opening button
   * @param {HTMLElement} node DOM element node
   */
  function generateStoryButton(node) {
    /* exits if the story button already exists */
    if (!node || elementExistsInNode(`.${C_BTN_STORY_CONTAINER}`, node)) return;

    try {
      const buttonStoryContainer = document.createElement('div');
      const buttonStory = document.createElement('button');

      buttonStoryContainer.classList.add(C_BTN_STORY_CONTAINER);
      buttonStory.classList.add(C_BTN_STORY);
      buttonStoryContainer.setAttribute('title', 'Open source');
      buttonStory.addEventListener('click', () => openStoryContent(node));

      buttonStoryContainer.appendChild(buttonStory);
      node.appendChild(buttonStoryContainer);
    } catch (err) {
      error('Failed to generate story button', err);
    }
  }

  /**
   * Appends new elements to DOM containing the post source opening button
   * @param {HTMLElement} node DOM element node
   */
  function generatePostButton(node) {
    /* exits if the post button already exists */
    if (!node || elementExistsInNode(`.${C_BTN_POST_OUTER_SPAN}`, node)) return;

    try {
      /* removes the div that's blocking the img element on a post */
      const blocker = node.querySelector(IG_S_POST_BLOCKER);
      if (blocker) blocker.parentNode.removeChild(blocker);

      const buttonsContainer = node.querySelector(IG_S_POST_BUTTONS);
      if (!buttonsContainer) {
        error(`Failed to generate post button, couldn't find by selector ${IG_S_POST_BUTTONS}`);
        return;
      }
      const newElementOuterSpan = document.createElement('span');
      const newElementButton = document.createElement('button');
      const newElementInnerSpan = document.createElement('span');

      newElementOuterSpan.classList.add(C_BTN_POST_OUTER_SPAN);
      newElementButton.classList.add(C_BTN_POST);
      newElementInnerSpan.classList.add(C_BTN_POST_INNER_SPAN);
      newElementOuterSpan.setAttribute('title', 'Open source');
      newElementButton.addEventListener('click', () => openPostSourceFromSrcAttribute(node));

      newElementButton.appendChild(newElementInnerSpan);
      newElementOuterSpan.appendChild(newElementButton);
      buttonsContainer.appendChild(newElementOuterSpan);
      node.classList.add(C_POST_WITH_BUTTON);

      const timeElement = node.querySelector(`${IG_S_POST_TIME_ANCHOR} time`);
      if (timeElement) {
        const fullDateStr = timeElement.getAttribute('datetime');
        if (fullDateStr) timeElement.innerHTML += ` (${fullDateStr})`;
      }
    } catch (err) {
      error('Failed to generate post button', err);
    }
  }

  /**
   * Appends new elements to DOM containing the profile picture source opening button
   * @param {HTMLElement} node DOM element node
   */
  function generateProfilePictureButton(node) {
    /* exits if the profile picture button already exists */
    if (!node || elementExistsInNode(`.${C_BTN_PROFILE_PIC_CONTAINER}`, node)) return;

    try {
      let profilePictureContainer = node.querySelector(IG_S_PROFILE_PIC_CONTAINER);
      /* if the profile is private and the user isn't following or isn't logged in */
      if (!profilePictureContainer) {
        profilePictureContainer = node.querySelector(IG_S_PRIVATE_PROFILE_PIC_CONTAINER);
      }
      if (!profilePictureContainer) {
        error(`Failed to generate profile picture button, couldn't find by selector ${IG_S_PROFILE_PIC_CONTAINER}`);
        return;
      }
      const newElementOuterSpan = document.createElement('span');
      const newElementButton = document.createElement('button');
      const newElementInnerSpan = document.createElement('span');
      newElementOuterSpan.setAttribute('title', 'Open full size picture');
      newElementButton.addEventListener('click', e => {
        e.stopPropagation();
        openProfilePicture();
      });

      newElementOuterSpan.classList.add(C_BTN_PROFILE_PIC_CONTAINER);
      newElementButton.classList.add(C_BTN_PROFILE_PIC);
      newElementInnerSpan.classList.add(C_BTN_PROFILE_PIC_SPAN);

      newElementButton.appendChild(newElementInnerSpan);
      newElementOuterSpan.appendChild(newElementButton);
      profilePictureContainer.appendChild(newElementOuterSpan);
    } catch (err) {
      error(err);
    }
  }

  //#endregion

  //#region Content parsing logic and opening

  /**
   * Finds the story source url from the src attribute on the node and opens it in a new tab
   * @param {HTMLElement} node DOM element node
   */
  function openStoryContent(node = null) {
    try {
      const container = (node || document).querySelector(IG_S_STORY_MEDIA_CONTAINER);
      const video = container.querySelector('video');
      const image = container.querySelector('img');
      if (video) {
        const source = getStoryVideoSrc(video);
        if (!source) throw 'Video source not available';
        openUrl(source);
        return;
      }
      if (image) {
        const source = getStoryImageSrc(image);
        if (!source) throw 'Video source not available';
        openUrl(source);
        return;
      }
      throw 'Story media source not available';
    } catch (exception) {
      errorMessage('Failed to open story source', exception);
    }
  }

  /**
   * Gets the source url of a post from the src attribute on the node and opens it in a new tab
   * @param {HTMLElement} node DOM element node containing the post
   */
  async function openPostSourceFromSrcAttribute(node = document.querySelector(IG_S_SINGLE_POST_CONTAINER)) {
    /* if is on single post page and the node is null, the picture container can be found, since there's only one */
    if (node == null) return;

    try {
      const postRelativeUrl = node.querySelector(IG_S_POST_TIME_ANCHOR)?.getAttribute('href');
      const sourceListItems = node.querySelectorAll(IG_S_MULTI_POST_LIST_ITEMS);
      if (sourceListItems.length == 0 /* is single post */) {
        await openPostMediaSource(node, postRelativeUrl);
        return;
      }

      const postIndex = getMultiPostIndex(node);
      if (sourceListItems.length == 2 /* is on the first or last item */) {
        if (node.querySelector(IG_S_MULTI_POST_NEXT_ARROW_BTN) /* next arrow exist */) {
          await openPostMediaSource(sourceListItems[0], postRelativeUrl, postIndex); /* opens first item */
        } else if (node.querySelector(IG_S_MULTI_POST_PREV_ARROW_BTN) /* previous arrow exists */) {
          await openPostMediaSource(sourceListItems[1], postRelativeUrl, postIndex); /* opens last item */
        } /* something is not right */ else {
          errorMessage('Failed to open post source', 'Failed to open first or last post carousel item');
        }
      } else if (sourceListItems.length == 3 /* is on any other item */) {
        await openPostMediaSource(sourceListItems[1], postRelativeUrl, postIndex);
      } /* something is not right */ else {
        errorMessage('Failed to open post source', 'Failed to open post carousel item other than first and last');
      }
    } catch (exception) {
      errorMessage('Failed to open post source', exception);
    } finally {
      document.body.style.cursor = 'default';
    }
  }

  /**
   * Gets the source url of a post from the src attribute on the node and opens it in a new tab
   * @param {HTMLElement} node DOM element node containing the post
   * @param {string} postRelativeUrl url of the post
   * @param {number} postIndex current index of the post carousel
   */
  async function openPostMediaSource(node, postRelativeUrl, postIndex) {
    let image = node.querySelector(IG_S_POST_IMG);
    let video = node.querySelector(IG_S_POST_VIDEO);
    if (image) {
      openUrl(image.getAttribute('src'));
      return;
    }
    if (video) {
      /* video url is available on the element */
      const videoSrc = video.getAttribute('src');
      if (!videoSrc?.startsWith('blob')) {
        openUrl(videoSrc);
        return;
      }
      if (!postRelativeUrl) throw 'No post relative url found';

      /* try to get the video url using the IG api */
      document.body.style.cursor = 'wait';
      const response = await httpGETRequest(URL_IG_POST_INFO_API(postRelativeUrl), true);
      let postData = response?.graphql?.shortcode_media;
      if (postIndex != undefined) postData = postData?.edge_sidecar_to_children?.edges?.[postIndex]?.node; // multi
      if (!postData) throw 'No post data found';
      if (!postData.is_video) throw 'Post is not a video';
      if (!postData.video_url) throw 'No video url found';

      openUrl(postData.video_url);
      return;
    }
    throw 'Failed to open source, no media found';
  }

  /**
   * Tries to get the source URL of the user's profile picture using multiple methods, including 3rd party websites
   * Opens the image or shows an alert if it doesn't find any URL
   */
  async function openProfilePicture() {
    try {
      const pageUsername = document.querySelector(IG_S_PROFILE_USERNAME_TITLE)?.innerText;
      if (!pageUsername) throw "Couldn't find username";

      document.body.style.cursor = 'wait';
      let pictureUrl = null;
      for (const [sourceName, getProfilePicture] of Object.entries(profilePictureSources)) {
        log(`Trying to get user's profile picture from ${sourceName}`);
        const url = await getProfilePicture(pageUsername);
        if (!url) {
          error(`Couldn't get profile picture url from ${sourceName}`);
          continue;
        }
        pictureUrl = url;
        break;
      }
      if (!pictureUrl) throw 'Nothing returned from 3rd party sources';

      log('Profile picture found, opening in a new tab...');
      openUrl(pictureUrl);
    } catch (err) {
      errorMessage("Couldn't get user's profile picture", err);
    } finally {
      document.body.style.cursor = 'default';
    }
  }

  /**
   * Get the source url of a story video
   * @param {HTMLElement} video DOM element node containing the video
   */
  function getStoryVideoSrc(video) {
    try {
      const videoElement = video.querySelector('source');
      return videoElement ? videoElement.getAttribute('src') : null;
    } catch (err) {
      error('Failed to get story video source', err);
      return null;
    }
  }

  /**
   * Get the source url of a story image
   * @param {HTMLElement} image DOM element node containing the image
   */
  function getStoryImageSrc(image) {
    const defaultSrc = image.getAttribute('src');
    try {
      const srcs = image.getAttribute('srcset').split(',');
      const images = srcs.map(src => {
        const [url, size] = src.split(' ');
        return { url, size: parseInt(size.replace(/[^0-9.,]/g, '')) };
      });
      return images ? images.reduce((pi, ci) => (pi.size > ci.size ? pi : ci)).url : defaultSrc;
    } catch (err) {
      error('Failed to get story image source', err);
      return defaultSrc || null;
    }
  }

  //#endregion

  //#region Content sources

  /**
   * Parses a whole HTML page in order to get the user's profile picture URL
   * @returns {Promise<string|null>} Profile picture URL or null if it fails
   */
  async function getProfilePictureFromUpdatedHTMLPage() {
    const response = await httpGETRequest(window.location, false);
    if (response) {
      const parser = new DOMParser();
      const doc = parser.parseFromString(response, 'text/html');
      const allScripts = doc.querySelectorAll('script');

      for (let i = 0; i < allScripts.length; i++) {
        if (/window._sharedData/.test(allScripts[i].innerText)) {
          let extractedJSON = /window._sharedData = (.+)/.exec(allScripts[i].innerText)[1];
          extractedJSON = extractedJSON.slice(0, -1);
          let sharedData = JSON.parse(extractedJSON);
          let userInfo = sharedData.entry_data.ProfilePage[0].graphql.user.profile_pic_url_hd;
          return userInfo;
        }
      }
    }
    return null;
  }

  /**
   * Extracts the profile picture URL from sharedData
   * @param {string} username Username of the user
   * @return {string|null} URL of the profile picture or null if it fails
   */
  function getProfilePictureFromSharedData(username) {
    const userSharedData = window._sharedData.entry_data.ProfilePage[0].graphql.user;
    /* return picture url if sharedData is correct */
    return userSharedData.username === username ? userSharedData.profile_pic_url_hd : null;
  }

  /**
   * Requests the user profile page data from graphql in order to get its profile picture URL
   * @returns {Promise<string|null>} URL of the profile picture or null if it fails
   */
  async function getProfilePictureFromUpdatedSharedData() {
    const response = await httpGETRequest(`${window.location}?__a=1`);
    return response ? response.graphql.user.profile_pic_url_hd : null;
  }

  /**
   * Performs a request to instadp.com and extracts the profile picture URL from the HTML response
   * @todo Adapt to be able to get pictures from a user that was never searched on their website
   * @param {string} username The user's Instagram username
   * @returns {Promise<string|null>} URL of the profile picture or null if it fails
   */
  async function getProfilePictureFromInstadpDotCom(username) {
    /*
			Instadp.com has a different process and requires a POST, probably to populate their database
			If you ever searched for a user on their website, this request succeeds, otherwise it fails
		*/
    const response = await httpGETRequest(URL_INSTA_DP_COM(username), false);
    if (!response) return null;
    const urls = extractUrlsFromString(response);
    const instagramUrls = urls.filter(u => u.includes('cdninstagram') && !u.includes('s150x150'));
    return instagramUrls.length > 0 ? instagramUrls[instagramUrls.length - 1] : null;
  }

  /**
   * Performs a request to instadp.org and extracts the profile picture URL from the HTML response
   * @param {string} username The user's Instagram username
   * @returns {Promise<string|null>} URL of the profile picture or null if it fails
   */
  async function getProfilePictureFromInstadpDotOrg(username) {
    const response = await httpPOSTRequest(
      URL_INSTA_DP_ORG,
      { 'Content-Type': 'application/x-www-form-urlencoded' },
      `username=${username}`,
      false
    );
    if (!response) return null;
    const instagramUrls = extractUrlsFromString(response).filter(u => u.includes('cdninstagram'));
    return instagramUrls.length > 0 ? instagramUrls[0] : null;
  }

  /**
   * Performs a request to izuum.com and extracts the profile picture URL from the HTML response
   * @param {string} username The user's Instagram username
   * @returns {Promise<string|null>} URL of the profile picture or null if it fails
   */
  async function getProfilePictureFromIzuum(username) {
    const response = await httpPOSTRequest(
      URL_IZUUM,
      { 'Content-Type': 'application/x-www-form-urlencoded' },
      `submit=${username}`,
      false
    );
    if (!response) return null;
    const instagramUrls = extractUrlsFromString(response).filter(u => u.includes('cdninstagram'));
    const cleanUrls = instagramUrls.map(u => u.replace(/amp;/g, ''));
    return cleanUrls.length > 0 ? cleanUrls[0] : null;
  }

  //#endregion

  //#region Key bindings and other event listeners

  /**
   * Loads the key bind to open a single post or a story from storage into a global scope variable, in order
   * to be used on the key binding handler method
   */
  async function loadPostStoryKeyBindings() {
    const kbName = 'single post and story';
    try {
      const kb = await loadKeyBindingFromStorage(STORAGE_KEY_POST_STORY_KB, DEFAULT_KB_POST_STORY, kbName);
      if (kb) openPostStoryKeyBinding = kb;
    } catch (err) {
      error(`Failed to load "${kbName}" key binding, considering default (Alt + ${DEFAULT_KB_POST_STORY})`, err);
    }
  }

  /**
   * Loads the key bind to open a profile picture from storage into a global scope variable in order
   * to be used on the key binding handler method
   */
  async function loadProfilePictureKeyBindings() {
    const kbName = 'profile picture';
    try {
      const kb = await loadKeyBindingFromStorage(STORAGE_KEY_PROFILE_PICTURE_KB, DEFAULT_KB_PROFILE_PICTURE, kbName);
      if (kb) openProfilePictureKeyBinding = kb;
    } catch (err) {
      error(`Failed to load "${kbName}" key binding, considering default (Alt + ${DEFAULT_KB_PROFILE_PICTURE})`, err);
    }
  }

  /**
   * Loads a key binding from storage, if it fails or doesn't have anything stores, returns the fallback key binding
   * @param {string} storageKey Unique name used to store the key binding
   * @param {string} defaultKeyBinding Fallback key binding
   * @param {string} keyBindingName Key binding name to show on log messages, just for context
   * @returns {Promise<string|null>} The saved letter used on the key binding or null if it fails
   */
  async function loadKeyBindingFromStorage(storageKey, defaultKeyBinding, keyBindingName) {
    let kb = await callGMFunction(GMFunc.getValue, storageKey, defaultKeyBinding);
    if (kb === null) {
      kb = defaultKeyBinding;
      log(`Falling back to default key binding: Alt + ${defaultKeyBinding}`);
    }

    try {
      if (isKeyBindingValid(kb)) {
        const newKey = kb.toUpperCase();
        log(`Discovered ${keyBindingName} key binding: Alt + ${newKey}`);
        return newKey;
      } else {
        error(
          `Couldn't load "${keyBindingName}" key binding, "${kb}" key is invalid, using default (Alt + ${defaultKeyBinding})`
        );
        return defaultKeyBinding;
      }
    } catch (err) {
      if (kb != defaultKeyBinding) {
        error(
          `Failed to load "${keyBindingName}" key binding, falling back to default: Alt + ${defaultKeyBinding}`,
          err
        );
      }
      return null;
    }
  }

  /**
   * Adds event listener(s) to the current document meant to handle key presses on a single post page
   */
  function setupSinglePostEventListeners() {
    setupKBEventListener(
      isSinglePostKeyBindingSetup,
      loadPostStoryKeyBindings,
      handleSinglePostKeyPress,
      () => {
        isSinglePostKeyBindingSetup = true;
      },
      'Defined single post opening event listener'
    );
  }

  /**
   * Adds event listener(s) to the current document meant to handle key presses on a story page
   */
  function setupStoryEventListeners() {
    setupKBEventListener(
      isStoryKeyBindingSetup,
      loadPostStoryKeyBindings,
      handleStoryKeyPress,
      () => {
        isStoryKeyBindingSetup = true;
      },
      'Defined story opening event listener'
    );
  }

  /**
   * Adds event listener(s) to the current document meant to handle key presses on a profile page
   */
  function setupProfileEventListeners() {
    setupKBEventListener(
      isProfileKeyBindingSetup,
      loadProfilePictureKeyBindings,
      handleProfileKeyPress,
      () => {
        isProfileKeyBindingSetup = true;
      },
      'Defined profile picture opening event listener'
    );
  }

  /**
   * Generic method to add an event listener for a key binding
   * @param {boolean} condition Condition that determines if the event should be added
   * @param {() => void} loadingFn Async function used to load the key binding
   * @param {() => void} keyPressHandler Handler function for the event (key binding press)
   * @param {() => void} callback Function to call after adding the event listener
   * @param {string} logMessage Message logged after adding the event listener
   */
  function setupKBEventListener(condition, loadingFn, keyPressHandler, callback, logMessage) {
    if (condition) return;
    loadingFn().then(() => {
      document.addEventListener('keydown', keyPressHandler);
      callback();
      log(logMessage);
    });
  }

  /**
   * Removes the previously added event listener(s) meant to handle key presses on a single post page
   */
  function removeSinglePostEventListeners() {
    removeKBEventListeners(
      isSinglePostKeyBindingSetup,
      handleSinglePostKeyPress,
      () => {
        isSinglePostKeyBindingSetup = false;
      },
      'Removed single post opening event listener'
    );
  }

  /**
   * Removes the previously added event listener(s) meant to handle key presses on a story page
   */
  function removeStoryEventListeners() {
    removeKBEventListeners(
      isStoryKeyBindingSetup,
      handleStoryKeyPress,
      () => {
        isStoryKeyBindingSetup = false;
      },
      'Removed story opening event listener'
    );
  }

  /**
   * Removes the previously added event listener(s) meant to handle key presses on a profile page
   */
  function removeProfileEventListeners() {
    removeKBEventListeners(
      isProfileKeyBindingSetup,
      handleProfileKeyPress,
      () => {
        isProfileKeyBindingSetup = false;
      },
      'Removed profile picture opening event listener'
    );
  }

  /**
   * Generic method to remove an event listener for a key binding
   * @param {boolean} condition Condition that determines if the event should be removed
   * @param {() => void} keyPressHandler Handler function previously assigned to the event
   * @param {() => void} callback Function to call after removing the event listener
   * @param {string} logMessage Message logged after removing the event listener
   */
  function removeKBEventListeners(condition, keyPressHandler, callback, logMessage) {
    if (!condition) return;
    document.removeEventListener('keydown', keyPressHandler);
    callback();
    log(logMessage);
  }

  /**
   * Handles key up events on a story page
   * @param {KeyboardEvent} event Keyboard event
   */
  function handleStoryKeyPress(event) {
    handleKeyPress(
      event,
      openPostStoryKeyBinding,
      () => pages.story.isVisible(),
      'Detected source opening shortcut on a story page',
      openStoryContent
    );
  }

  /**
   * Handles key up events on a single post page
   * @param {KeyboardEvent} event Keyboard even
   */
  function handleSinglePostKeyPress(event) {
    handleKeyPress(
      event,
      openPostStoryKeyBinding,
      () => pages.post.isVisible(),
      'Detected source opening shortcut on a single post page',
      openPostSourceFromSrcAttribute
    );
  }

  /**
   * Handles key up events on a profile page
   * @param {KeyboardEvent} event Keyboard even
   */
  function handleProfileKeyPress(event) {
    handleKeyPress(
      event,
      openProfilePictureKeyBinding,
      () => !(pages.story.isVisible() || pages.post.isVisible()),
      'Detected profile picture opening shortcut on a profile page',
      openProfilePicture
    );
  }

  /**
   * Handles key up with the alt key events on certain conditions and performs an action
   * @param {KeyboardEvent} event Keyboard event
   * @param {string} keyBinding Target key binding (letter)
   * @param {() => boolean} checkConditionsAreMet Function that determines if the conditions are met
   * @param {string} logMessageString Message logged when the keybinding is used and the conditions are met
   * @param {() => void} keyPressAction Function executed when the keybinding is used and the conditions are met
   */
  function handleKeyPress(event, keyBinding, checkConditionsAreMet, logMessageString, keyPressAction) {
    if (event.altKey && event.code.toLowerCase() === `key${keyBinding.toLowerCase()}` && checkConditionsAreMet()) {
      log(logMessageString);
      keyPressAction();
    }
  }

  //#endregion

  //#region Networking

  /**
   * Performs an HTTP GET request using the GM_xmlhttpRequest or GM.xmlHttpRequest function
   * @param {string} url Target url to perform the request
   * @param {boolean} [parseToJSON = true] Default true
   * @returns {Promise<string|any>} Response text or an exception error object
   */
  function httpGETRequest(url, parseToJSON = true) {
    return new Promise((resolve, reject) => {
      const options = {
        method: 'GET',
        url: url,
        timeout: 10000,
        onload: res => {
          if (res.status !== 200) {
            reject('Status Code', res?.status, res?.statusText || '');
            return;
          }
          let response = res.responseText;
          if (parseToJSON) {
            response = JSON.parse(res.responseText);
          }
          resolve(response);
        },
        onerror: error => {
          error(`Failed to perform GET request to ${url}`, error);
          reject(error);
        },
        ontimeout: () => {
          error('GET Request Timeout');
          reject('GET Request Timeout');
        },
        onabort: () => {
          error('GET Request Aborted');
          reject('GET Request Aborted');
        },
      };

      const fnResponse = callGMFunction(GMFunc.xmlHttpRequest, options);
      if (fnResponse === null) {
        error(`Failed to perform GET request to ${url}`);
        reject();
      }
    });
  }

  /**
   * Performs an HTTP POST request using the GM_xmlhttpRequest or GM.xmlHttpRequest function
   * @param {string} url Target url to perform the request
   * @param {{[key: string]: string}} [headers = null] Request header (optional)
   * @param {string} [data = null] Request body (optional)
   * @param {boolean} [parseToJSON = true] Parse the response to JSON (default true)
   * @returns {Promise<string|any>} Response text or an exception error object
   */
  function httpPOSTRequest(url, headers = null, data = null, parseToJSON = true) {
    return new Promise((resolve, reject) => {
      const options = {
        method: 'POST',
        url: url,
        ...(headers && { headers: headers }),
        ...(data && { data: data }),
        timeout: 10000,
        onload: res => {
          if (res.status !== 200) {
            reject('Status Code', res?.status, res?.statusText || '');
            return;
          }
          let response = res.responseText;
          if (parseToJSON) {
            response = JSON.parse(res.responseText);
          }
          resolve(response);
        },
        onerror: error => {
          error(`Failed to perform POST request to ${url}`, error);
          reject(error);
        },
        ontimeout: () => {
          error('POST Request Timeout');
          reject('POST Request Timeout');
        },
        onabort: () => {
          error('POST Request Aborted');
          reject('POST Request Aborted');
        },
      };

      const fnResponse = callGMFunction(GMFunc.xmlHttpRequest, options);
      if (fnResponse === null) {
        error(`Failed to perform POST request to ${url}`);
        reject();
      }
    });
  }

  //#endregion

  //#region Utils & others

  /**
   * Opens a URL depending on the behavior defined in the settings
   * @param {string} url URL to open
   */
  function openUrl(url) {
    if (openSourceBehavior === BUTTON_BEHAVIOR_NEW_TAB_BG) {
      callGMFunction(GMFunc.openInTab, url, true);
    } else if (openSourceBehavior === BUTTON_BEHAVIOR_REDIR) {
      window.location.replace(url);
    } else {
      window.open(url, '_blank');
    }
  }

  /**
   * Calls a GreaseMonkey function using the multiple formats for compatibility
   * @param {string[]} gmFunctionVariants GM functions to call (multiple variants)
   * @param {any[]} args Array of arguments passed to the GM function
   * @returns {Promise<any>} Returns the GM function result, undefined if the function doesn't succeeds but doesn't
   * return anything and null if the function(s) fail to execute
   */
  async function callGMFunction(gmFunctionVariants, ...args) {
    for (const fnName of gmFunctionVariants) {
      try {
        const fn = eval(fnName);
        if (typeof fn !== 'function') throw 'No function found';
        const res = await fn(...args);
        return res;
      } catch (error) {
        warn(`Failed to call ${fnName} function...`);
      }
    }
    error(`Failed to call all GM function variants (${gmFunctionVariants.join(', ')})`);
    return null;
  }

  /**
   * Finds the current position on a post carousel
   * @param {HTMLElement} node DOM element node containing the post
   * @return {number} current index
   */
  function getMultiPostIndex(node) {
    const indicators = node.querySelectorAll(IG_S_MULTI_POST_INDICATOR);
    for (let i = 0; i < indicators.length; i++) {
      if (indicators[i].classList.contains(IG_C_MULTI_POST_INDICATOR_ACTIVE)) {
        return i;
      }
    }
    return -1;
  }

  /**
   * Check if the key is valid to used as a key binding
   * @param {string} key Key binding key
   * @returns {boolean}
   */
  function isKeyBindingValid(key) {
    return /[a-zA-Z]/gm.test(key);
  }

  /**
   * Extracts every URL found between quotes and double quotes in a given string
   * @param {string} string String to match
   * @return {string[]} Extracted URLs
   */
  function extractUrlsFromString(string) {
    /* This regex is not bullet proof and has unnecessary rules, but it works fine */
    return string.match(
      /(?:(?:https?|ftp|file):\/\/|www\.|ftp\.)(?:\([-A-Z0-9+&@#\/%=~_|$?!:;,.]*\)|[-A-Z0-9+&@#\/%=~_|$?!:;,.])*(?:\([-A-Z0-9+&@#\/%=~_|$?!:;,.]*\)|[A-Z0-9+&@#\/%=~_|$])/gim
    );
  }

  /**
   * Matches a CSS selector against a DOM element object to check if the element exist in the node
   * @param {string} selector
   * @param {HTMLElement} node DOM element node to match
   * @returns {boolean} True if the element exists in the node, otherwise false
   */
  function elementExistsInNode(selector, node) {
    return node && node.querySelector(selector) != null;
  }

  /**
   * Appends the necessary style elements to DOM
   */
  function injectStyles() {
    const b64PostBtnIcon =
      '';
    const b64StoryBtnIcon =
      '';
    const b64SettingsBtnIcon =
      '';
    const svgSelectArrowIcon =
      "data:image/svg+xml;utf8,<svg fill='black' height='24' viewBox='0 0 24 24' width='24' xmlns='http://www.w3.org/2000/svg'><path d='M7 10l5 5 5-5z'/><path d='M0 0h24v24H0z' fill='none'/></svg>";
    const opacityTransition =
      'transition:opacity .2s ease-in-out;-webkit-transition:opacity .2s ease-in-out;-moz-transition:opacity .2s ease-in-out;-ms-transition:opacity .2s ease-in-out;-o-transition:opacity .2s ease-in-out;';
    const styles = `
      .${C_BTN_POST_OUTER_SPAN}{margin-left:8px;}
      .${C_BTN_POST}{outline:none;-webkit-box-align:center;align-items:center;background:0;border:0;cursor:pointer;display:flex;-webkit-box-flex:0;flex-grow:0;-webkit-box-pack:center;justify-content:center;padding:8px 0 8px 8px;}
      .${C_BTN_POST_INNER_SPAN},.${C_BTN_PROFILE_PIC_SPAN}{display:block;background-repeat:no-repeat;height:24px;width:24px;background-image:url(${b64PostBtnIcon});background-size:24px 24px;cursor:pointer;opacity:1;${opacityTransition};}
      .${C_BTN_POST_INNER_SPAN}:hover{opacity:.6;}
      .${C_BTN_PROFILE_PIC}{outline:none;background-color:white;border:0;cursor:pointer;min-height:40px;min-width:40px;padding:0;border-radius:50%;transition:background-color .5s ease;-webkit-transition:background-color .5s ease;}
      .${C_BTN_PROFILE_PIC}:hover{background-color:#D0D0D0;transition:background-color .5s ease;-webkit-transition:background-color .5s ease;}
      .${C_BTN_PROFILE_PIC_SPAN}{margin: auto;}
      .${C_BTN_STORY_CONTAINER}{position:fixed;top:32px;right:0;margin:16px;z-index:99}
      .${C_BTN_STORY}{width:24px;height:24px;margin:8px;border:none;cursor:pointer;background-color:transparent;background-image:url(${b64StoryBtnIcon});background-size:24px 24px;opacity:1;${opacityTransition}}
      .${C_BTN_STORY}:hover{opacity:0.8;}
      .${C_BTN_PROFILE_PIC_CONTAINER}{transition:.5s ease;opacity:0;position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);-ms-transform:translate(-50%,-50%);text-align:center}
      ${IG_S_PRIVATE_PIC_IMG_CONTAINER}>img{transition:.5s ease;backface-visibility:hidden;}
      ${IG_S_PRIVATE_PROFILE_PIC_IMG_CONTAINER}>img{transition:.5s ease;backface-visibility:hidden;}
      ${IG_S_PROFILE_PIC_CONTAINER}:hover .${C_BTN_PROFILE_PIC_CONTAINER}{opacity:1}
      ${IG_S_PRIVATE_PROFILE_PIC_CONTAINER}:hover .${C_BTN_PROFILE_PIC_CONTAINER}{opacity:1}
      .${C_SETTINGS_BTN}{width:20px;height:20px;cursor:pointer;top:16px;border:none;right:16px;position:fixed;background-color:transparent;background-image:url(${b64SettingsBtnIcon});background-size:20px 20px;opacity:.8;${opacityTransition}}
      .${C_SETTINGS_BTN}:hover{opacity:1;}
      .${C_SETTINGS_CONTAINER}{position:fixed;display:flex;justify-content:center;align-items:center;width:100vw;height:100vh;top:0;left:0;background-color:rgba(0,0,0,.7);display:none}
      .${C_SETTINGS_MENU}{width:280px;display:flex;flex-direction:column;background-color:#fff;border-radius:4px;z-index:5;box-shadow:-1px 2px 14px 3px rgba(0,0,0,.5)}
      .${C_SETTINGS_MENU_TITLE_CONTAINER}{display:flex;flex-direction:row;justify-content:space-between;font-weight:700}
      .${C_SETTINGS_MENU_TITLE}{display:flex;justify-content:center;flex-direction:row;font-size:16px;padding:16px;text-align:left}
      .${C_SETTINGS_MENU_TITLE_CLOSE_BTN}{width:24px;height:24px;border:0;padding:0;background-color:transparent;margin-top:8px;margin-right:8px;cursor:pointer}
      .${C_SETTINGS_MENU_TITLE_LINK}{margin-left:6px;color:#4287f5!important}
      .${C_SETTINGS_MENU_OPTIONS}{display:flex;flex-direction:column;border-top:1px solid rgba(219,219,219,1)}
      .${C_SETTINGS_MENU_OPTION}{padding:12px 16px;border:none;background-color:transparent;font-size:14px;padding-left:16px;text-align:left}
      .${C_SETTINGS_MENU_OPTION_BTN}{padding:12px 16px;border:none;background-color:transparent;font-size:14px;padding-left:16px;text-align:left;cursor:pointer;transition:background-color 0.2s ease;}
      .${C_SETTINGS_MENU_OPTION_BTN}:hover{background-color:rgba(214,214,214,.3)}
      .${C_SETTINGS_MENU_OPTION_BTN}:active{background-color:rgba(214,214,214,.4)}
      [for="${ID_SETTINGS_BUTTON_BEHAVIOR_SELECT}"]{font-size:12px;}
      #${ID_SETTINGS_BUTTON_BEHAVIOR_SELECT}{font-size:14px;margin-top:8px;height:32px;border-radius:4px;padding:0 6px;border:1px solid gray;-moz-appearance:none;-webkit-appearance:none;appearance:none;background-image:url("${svgSelectArrowIcon}");background-repeat:no-repeat;background-position-x:99%;background-position-y:50%;}
      `;

    const element = document.createElement('style');
    element.type = 'text/css';
    element.innerHTML = styles;
    document.head.appendChild(element);
    log('Injected CSS into DOM');
  }

  //#endregion
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址