Enhance the experience of viewing local files on Edge.
// ==UserScript== // @name Edge Local File Enhancer // @namespace http://tampermonkey.net/ // @version 0.1.5 // @description Enhance the experience of viewing local files on Edge. // @author PRO // @match file:///*/ // @icon  // @grant none // @run-at document-end // @license gpl-3.0 // ==/UserScript== (function() { 'use strict'; // Helper functions const path = location.href; if (!path.startsWith("file:///") || !path.endsWith("/")) return; const debug = false; const log = debug ? console.log.bind(console, "[ELFE]") : () => {}; const $ = document.querySelector.bind(document); const $$ = document.querySelectorAll.bind(document); const getModifier = (e) => (e.ctrlKey << 2 | e.shiftKey << 1 | e.altKey); const hasModifier = (e, test = 0b111) => Boolean(getModifier(e) & test); const modifierDict = (e) => { return { "button": e.button || 0, "ctrlKey": e.ctrlKey, "shiftKey": e.shiftKey, "altKey": e.altKey, "bubble": false, } }; // CSS const header = $("h1#header"); const css = document.createElement("style"); css.id = "elfe-css"; css.textContent = ` html { scroll-behavior: smooth; } h1#header > a { color: initial; text-decoration: none; transition: color 0.2s ease-in-out; } h1#header > a:hover { color: -webkit-link; } table { margin: 0.5rem 0; width: auto; } table td, table th { padding: 0.3rem 0.5rem; vertical-align: middle; } div#parentDirLinkBox { display: none !important; } #parentDir { padding: 0 0.5em 0; } thead th { border-left: 1px solid gray; border-right: 1px solid gray; } thead th, tbody tr { transition: background-color 0.2s ease-in-out; cursor: pointer; } thead th:hover, tbody tr:hover { background-color: #333333; } tbody tr.selected { background-color: #4d4d4d; } `; $("head").appendChild(css); // Navigation const delimeter = header.textContent.includes("\\") ? "\\" : "/"; const split = header.textContent.split(delimeter); log(split); const parts = split.slice(0, -1); header.innerHTML = '<a href="../" id="parentDir">↑</a>' + parts.map((part, i) => `<a href="${parts.slice(0, i + 1).join(delimeter)}${delimeter}">${part}</a>` ).join(delimeter) + delimeter + split.slice(-1)[0]; // Table const rows = $$("tbody tr"); rows.forEach(row => { row.addEventListener("click", e => { if (e.button != 0) return true; log(e); e.preventDefault(); row.querySelector("a").dispatchEvent(new MouseEvent("click", modifierDict(e))); return true; }); }); $$("td").forEach(td => { td.title = td.getAttribute("data-value") || td.textContent; }); // Shortcuts const parentDir = $("#parentDir"); const count = rows.length; var selected = 0; function select(i) { rows[selected].classList.remove("selected"); selected = i; rows[selected].classList.add("selected"); rows[selected].scrollIntoView({ block: "center" }); } function delta(d) { select((selected + d + count) % count); } document.addEventListener("keydown", e => { switch (e.key) { case "ArrowUp": e.preventDefault(); if (e.altKey && parentDir) { // Go to parent directory parentDir.click(); break; } if (e.ctrlKey) { // Scroll up window.scrollBy({ top: -window.innerHeight / 2}); break; } if (count == 0) break; if (e.shiftKey) { // Select top select(0); break; } // Select previous delta(-1); break; case "ArrowDown": e.preventDefault(); if (e.ctrlKey) { // Scroll down window.scrollBy({ top: window.innerHeight / 2}); break; } if (count == 0) break; if (e.shiftKey) { // Select bottom select(count - 1); break; } // Select next delta(1); break; case "ArrowLeft": if (hasModifier(e)) break; // Try to be none-intrusive history.back(); break; case "ArrowRight": if (hasModifier(e)) break; history.forward(); break; case "Enter": { if (count == 0) break; e.preventDefault(); const link = rows[selected].querySelector("a"); link.dispatchEvent(new MouseEvent("click", modifierDict(e))); break; } default: break; } }); })();