Copy Selected Links

Displays a Copy button on selected links.

// ==UserScript==
// @name        Copy Selected Links
// @namespace   leaumar
// @match       *://*/*
// @grant       GM.setClipboard
// @version     2
// @author      leaumar@mailbox.org
// @description Displays a Copy button on selected links.
// @license     MPL-2.0
// @icon        
// ==/UserScript==

function centerOf(element) {
  const rect = element.getBoundingClientRect();
  return {
    left: rect.left + rect.width / 2,
    top: rect.top + rect.height / 2,
  };
}

const sum = (a, b) => a + b;

function averageOf(offsets) {
  return {
    left: offsets.map(offset => offset.left).reduce(sum) / offsets.length,
    top: offsets.map(offset => offset.top).reduce(sum) / offsets.length,
  };
}

function offsetToCenterOf(self, refs) {
  const center = averageOf(refs.map(centerOf));
  const selfBox = self.getBoundingClientRect();

  return {
    left: `${center.left - selfBox.width / 2}px`,
    top: `${center.top - selfBox.height / 2}px`,
  };
}

// ------------

function makeButton() {
  let toCopy;

  const button = document.createElement('button');
  button.type = 'button';
  Object.assign(button.style, {
    padding: '0.5em',
    position: 'fixed',
  });

  // shadow dom stops external css from affecting the button
  const container = document.createElement('div');
  const shadow = container.attachShadow({mode: 'closed'});
  shadow.appendChild(button);

  function hide() {
    container.remove();
  }

  function attachTo(links) {
    toCopy = links.map(anchor => anchor.href).join('\n');
    button.textContent = `📋${links.length}`;
    document.body.appendChild(container);
    Object.assign(button.style, offsetToCenterOf(button, links));
  }

  button.addEventListener('mouseup', mouseUp => mouseUp.stopPropagation());
  button.addEventListener('click', click => {
    click.stopPropagation();
    GM.setClipboard(toCopy, "text/plain");
    hide();
  });

  return {
    attachTo,
    hide,
  };
}

(function main() {
  let copyButton;

  document.addEventListener('scroll', () => copyButton?.hide());

  function updateButton() {
    const selection = getSelection();
    if (selection?.type === 'Range') {
      const links = [...document.links].filter(link => selection.containsNode(link, true));
      if (links.length > 0) {
        (copyButton = copyButton ?? makeButton()).attachTo(links);
        return;
      }
    }
    copyButton?.hide();
  }

  // mouseup and click aren't triggered very reliably or consistently -_-
  document.body.addEventListener('mouseup', updateButton);
  document.body.addEventListener('click', updateButton);
})();