// ==UserScript==
// @name File Regexplorer
// @version 0.2.1
// @namespace file_regexplorer
// @description Implements local explorer - file manager and provide a means to sort the file list in custom order via regular expressions
// @description:en Implements local explorer - file manager and provide a means to sort the file list in custom order via regular expressions
// @description:it Implementa il file manager locale del browser e fornisce un sistema per ordinare l'elenco dei file in modo personalizzato tramite espressioni regolari
// @author OpenDec
// @match file:///*/
// @icon 
// @grant GM.getValue
// @grant GM.setValue
// @grant GM.deleteValue
// @run-at document-start
// @license MIT
// ==/UserScript==
/* jshint esversion: 11 */
// --------------------------------------------------
// --- START ---
// --------------------------------------------------
function start(){
'use strict';
// Exit if page is cached
const _html = document.documentElement;
if (_html.classList.contains('od_regexplorer')) return;
_html.classList.add('od_regexplorer');
const _name = GM.info.script.name;
const _vers = GM.info.script.version;
const _icon = GM.info.script.icon || GM.info.scriptMetaStr.match(/^\/\/ @icon +(.*)$/m)[1];
const _header = document.querySelector('#header, h1');
const _table = document.querySelector('table');
const _thead = _table.querySelector('#theader, thead > tr');
const _tbody = _table.querySelector('#tbody, tbody');
const _topbar = document.createElement('div');
const _form = document.createElement('div');
const _url = window.location.href;
const _urlHash = getHash(_url);
let _arrRows;
let _order;
let _orderBy;
let _userData;
// --------------------------------------------------
// --- DATA ---
// --------------------------------------------------
async function saveData(){
const data = {};
if (_inpPattern.value) data.pattern = _inpPattern.value;
if (_ckbFlagI.checked) data.flagI = 1;
if (_inpOutput.value) data.output = _inpOutput.value;
if (_ckbFilteredView.checked) data.filteredView = 1;
if (_orderBy) data.orderBy = _orderBy;
if (_order === -1) data.order = -1;
if (Object.keys(data).length === 0) GM.deleteValue(_urlHash);
else GM.setValue(_urlHash, JSON.stringify(data));
}
async function loadData(){
_html.removeAttribute('style');
const dataJSON = await GM.getValue(_urlHash);
if (!dataJSON) return;
const data = JSON.parse(dataJSON);
if (data.pattern) _inpPattern.value = data.pattern;
if (data.flagI) _ckbFlagI.checked = true;
if (data.output) _inpOutput.value = data.output;
if (data.filteredView) _ckbFilteredView.checked = true;
_orderBy = data.orderBy ? +data.orderBy : 0;
_order = data.order ? -1 : 1;
fillOutputColumn(_inpPattern.value, _inpOutput.value);
setFilteredView(_ckbFilteredView.checked, {skipTransition: true});
if (_orderBy || _order === -1){
_table.dataset.order = _order;
sortBy(_orderBy, true);
}
}
function getHash(s){
const hash = Array.from(s).reduce((hash, char) => 0 | (31 * hash + char.charCodeAt(0)), 0).toString(36);
return '_' + (hash[0] === '-' ? hash.slice(1).toUpperCase() : hash);
}
loadData();
// --------------------------------------------------
// --- TOP ---
// --------------------------------------------------
_topbar.id = 'od_topbar';
_topbar.innerHTML = `<a href="https://greasyfork.org/it/scripts/467835-file-regexplorer">${_name} v${_vers}</a>`;
document.body.appendChild(_topbar);
// --------------------------------------------------
// --- FORM ---
// --------------------------------------------------
// Pattern fields
_form.id = 'od_form';
_form.innerHTML = `
<div class="od_container_field" id="od_container_pattern">
<label class="od_lbl" id="od_lbl_pattern">Pattern (.*): <input class="od_inp" id="od_inp_pattern" value="" spellcheck="false"></label>
<label class="od_lbl od_lbl_top" id="od_lbl_pattern_flag_i"><input type="checkbox" class="od_ckb od_ckb_top" id="od_ckb_pattern_flag_i"> Case Insensitive</label>
</div>
<div class="od_container_field" id="od_container_output">
<label class="od_lbl" id="od_lbl_output">Output: <input class="od_inp" id="od_inp_output" value="" spellcheck="false"></label>
<label class="od_lbl od_lbl_top" id="od_lbl_output_filtered_view"><input type="checkbox" class="od_ckb od_ckb_top" id="od_ckb_output_filtered_view"> Filtered view</label>
</div>
`;
_header.parentNode.insertBefore(_form, _header.nextSibling);
const _inpPattern = document.getElementById('od_inp_pattern');
const _ckbFlagI = document.getElementById('od_ckb_pattern_flag_i');
const _inpOutput = document.getElementById('od_inp_output');
const _ckbFilteredView = document.getElementById('od_ckb_output_filtered_view');
// Check regex pattern validity
_inpPattern.addEventListener('input', e => {
const me = e.target;
const pat = me.value;
me.setCustomValidity('');
try {
new RegExp(pat);
} catch(err){
me.setCustomValidity('Invalid regex pattern');
}
});
// Redraw output column based on form values
function submitForm(){
fillOutputColumn(_inpPattern.value, _inpOutput.value);
if (_table.dataset.orderBy === '1') sortBy(1, true);
saveData();
}
// When press a key in the fields
function onKey(e){
const me = e.target;
if (e.keyCode === 13){
if (me ===_inpPattern) _inpOutput.focus();
else me.blur();
} else if (e.keyCode === 27){
me.value = me.dataset.currentValue;
me.blur();
}
}
// When field focus
function onFocus(e){
const me = e.target;
me.dataset.currentValue = me.value;
}
// Form fields listeners
[_inpPattern, _inpOutput].forEach(e => {
e.addEventListener('change', submitForm);
e.addEventListener('focus', onFocus);
e.addEventListener('keydown', onKey);
});
_ckbFlagI.addEventListener('change', e => {
submitForm();
});
_ckbFilteredView.addEventListener('change', e => {
const me = e.target;
saveData();
setFilteredView(me.checked);
});
// --------------------------------------------------
// --- TABLE ---
// --------------------------------------------------
(()=>{
const th = document.createElement('th');
th.innerText = '(.*)';
_thead.insertBefore(th, _thead.cells[1]);
// Clean & set column headers
setTimeout(()=>{
Array.from(_thead.cells).forEach(th => {
th.innerHTML = th.innerText;
th.tabIndex = '0';
th.setAttribute('role', 'button');
});
}, 100);
Array.from(_tbody.rows).forEach(tr => {
const td0 = tr.cells[0];
if ('value' in td0.dataset){
// On Chromium, prepend 1 or 2 for the value of folders or files respectively, to keep items separate when sorting, like Firefox does
const tdVal = td0.dataset.value;
if (tdVal){
td0.dataset.value = (tdVal.slice(-1) === '/')
? 1 + (td0.firstChild.innerHTML = tdVal.slice(0,-1))
: 2 + tdVal
;
}
} else if (td0.hasAttribute('sortable-data')){
// FF - pass the values to the data attribute for each td
Array.from(tr.cells).forEach(td => {
td.dataset.value = td.getAttribute('sortable-data');
td.removeAttribute('sortable-data');
});
// FF remove the nested table with the ellipsis from the file name cell
td0.innerHTML = td0.querySelector('a').outerHTML;
}
// Wrap cell content
Array.from(tr.cells).forEach((td, index) => {
if (td.innerHTML === '') return;
td.innerHTML = `<div>${td.innerHTML}</div>`;
});
// Add output column
let td = document.createElement('td');
tr.insertBefore(td, tr.cells[1]);
});
// Shadow thead sticky
const shadow = document.createElement('div');
shadow.id = 'od_shadow_thead_sticky';
document.body.insertBefore(shadow, _table);
// Change the sort order when you press on the headers
_thead.addEventListener('click', e => {
e.stopPropagation();
sortBy(e.target.cellIndex);
saveData();
}, true);
_thead.addEventListener('keydown', e => {
if (e.key == 'Enter' || e.key == ' ') {
e.stopPropagation();
e.preventDefault();
sortBy(e.target.cellIndex);
saveData();
}
}, true);
// Default array rows (sorted by filename ascending)
_arrRows = Array.from(_tbody.rows);
sortBy(0);
_arrRows = Array.from(_tbody.rows);
})();
function fillOutputColumn(pat, out){
let reg;
try {
reg = new RegExp(pat, _ckbFlagI.checked ? 'i' : '');
} catch(err) {
console.error('Invalid regex pattern');
return;
}
const nomatches = [];
_arrRows.forEach((tr, index) => {
const valName = tr.cells[0].innerText;
const matches = valName.match(reg);
const td = tr.cells[1];
if (matches === null){
nomatches[index] = true;
td.innerHTML = '<div></div>';
// Prepend 2 for empty values, otherwise 1, to keep items separate when sorting
td.dataset.value = 2;
} else {
const text = matches[0]?.replace(reg, out || '$&') || '';
td.innerHTML = `<div>${text}</div>`;
td.dataset.value = 1 + text;
}
});
// Apply nomatch class
applyNomatch(nomatches);
}
function applyNomatch(nomatches){
_arrRows.forEach((tr, index) => {
tr.classList.toggle('od_nomatch', nomatches[index] || false);
});
}
function sortBy(column, keepOrder = false){
const rows = Array.from(_arrRows);
_order = keepOrder ? +_table.dataset.order : (_orderBy == column && _table.dataset.order === '1') ? -1 : 1;
_table.dataset.orderBy = _orderBy = column;
_table.dataset.order = _order;
rows.sort((rowA, rowB) => {
let a = rowA.cells[_orderBy].dataset.value || '';
let b = rowB.cells[_orderBy].dataset.value || '';
return _order * a.localeCompare(b, false, {numeric: true});
});
_tbody.innerHTML = '';
for (let i = 0; i < rows.length; i++){
_tbody.appendChild(rows[i]);
}
}
function setFilteredView(b, o){
if (o?.skipTransition){
_table.classList.add('od_skip_transition');
setTimeout(()=>{_table.classList.remove('od_skip_transition');}, 0)
}
_table.classList.toggle('od_filtered_view', b);
}
// --------------------------------------------------
// --- STYLE ---
// --------------------------------------------------
addGlobalStyle(`
/* -------------------------------------------------- */
/* --- RESET --- */
/* -------------------------------------------------- */
html *,
html *::before,
html *::after {box-sizing: border-box}
:root {
padding-inline: 0;
}
/* -------------------------------------------------- */
/* --- MAIN --- */
/* -------------------------------------------------- */
body {
position: relative;
width: auto;
min-width: 500px;
margin: 4em auto;
padding: 2em 1em;
font: 12px "Segoe UI", "DejaVu Sans", "Bitstream Vera Sans", "Lucida Grande", Verdana, Tahoma, Arial, sans-serif;
border: 1px solid;
border-radius: 10px;
}
#header, h1 {
margin: 0 0 60px;
padding: 0;
white-space: normal;
word-break: break-word;
font-size: 160%;
font-weight: normal;
border-bottom: 1px solid;
}
#header, h1 {
margin: 0 0 60px;
padding: 0;
white-space: normal;
word-break: break-word;
font-size: 160%;
font-weight: normal;
border-bottom: 1px solid;
}
/* -------------------------------------------------- */
/* --- TOPBAR --- */
/* -------------------------------------------------- */
#od_topbar {
display: inline-flex;
align-items: center;
position: absolute;
top: -30px;
left: 0;
}
#od_topbar a {
display: inline-flex;
align-items: center;
text-decoration: none;
padding-left: 40px;
height: 30px;
background: 8px/24px url(${_icon}) no-repeat;
}
/* -------------------------------------------------- */
/* --- FORM --- */
/* -------------------------------------------------- */
#od_form {
position: relative;
z-index: 1;
display: grid;
grid-template-columns: 1fr;
grid-gap: 1rem;
margin: -50px 0 10px;
padding: 5px 10px;
}
input.od_inp {
width: 100%;
margin: 5px 0 0;
padding: .5em;
border: 1px solid;
}
input.od_inp:invalid {
outline: 1px solid;
}
.od_container_field {
position: relative;
}
.od_lbl {
user-select: none;
}
.od_lbl_top {
display: flex;
align-items: center;
position: absolute;
top: 0;
right: 0;
padding-left: 18px;
margin-right: 10px;
}
.od_ckb_top {
position: absolute;
left: 0;
margin: 0;
}
/* -------------------------------------------------- */
/* --- TABLE --- */
/* -------------------------------------------------- */
body > table {
min-width: 100%;
margin: 0 auto;
border-collapse: separate;
border-spacing: 0;
}
thead {
position: sticky;
top: 0;
z-index: 2;
}
#od_shadow_thead_sticky {
clear: both;
position: sticky;
top: 0;
width: 100%;
height: 2.5em;
margin-top: -2.5em;
pointer-events: none;
}
/* FF fix. Collapse margin after floated elements */
#UI_goUp, #UI_showHidden {
margin-bottom: -2.5em;
position: relative;
z-index: 1;
}
:root.od_regexplorer > body > table > * > tr > * {
padding-block: 4px;
padding-inline: 8px;
}
body > table > thead > tr > th {
position: relative;
border-width: 1px;
border-style: solid;
font-size: 15px;
font-weight: normal;
text-align: center;
white-space: nowrap;
cursor: pointer;
user-select: none;
}
body > table > tbody > tr {
outline-offset: -1px;
}
body > table > tbody > tr > td {
border: solid transparent;
border-width: 0 1px;
}
body > table > tbody > tr > td {
}
body > table > thead > tr > th:first-child {
text-align: start;
}
body > table > tbody > tr > td:nth-child(-n+2) {
text-align: start;
overflow-wrap: anywhere;
}
body > table > tbody > tr > td:nth-child(3) {
text-align: end;
}
body > table > tbody > tr > td:nth-child(4) {
text-align: center;
}
body > table > tbody > tr {
outline: 1px solid transparent;
}
body > table > tbody > tr > td:first-child a {
display: inline-block;
white-space: break-spaces;
}
/* Sorting icons - column headers */
table[data-order-by="0"] > thead > tr > th:nth-child(1)::after,
table[data-order-by="1"] > thead > tr > th:nth-child(2)::after,
table[data-order-by="2"] > thead > tr > th:nth-child(3)::after,
table[data-order-by="3"] > thead > tr > th:nth-child(4)::after {
display: inline-block;
}
table[data-order] > thead > tr > th::after {
display: none;
position: absolute;
top: -4px;
left: 0;
right: 0;
width: fit-content;
margin-inline: auto;
font-size: 9px;
text-align: center;
opacity: .4;
transform: scaleX(1.5);
}
table[data-order="1"] > thead > tr > th::after {
content: "˄";
}
table[data-order="-1"] > thead > tr > th::after {
content: "˅";
}
/* Unmatched items */
tr.od_nomatch td:nth-child(2) {
position: relative;
}
tr td:nth-child(2) > div:empty {
width: 0;
left: 50%;
transform: translateX(-50%);
}
tr.od_nomatch td:nth-child(2) > div {
position: absolute;
top: 50%;
border-bottom: 1px solid #e5683e;
width: 100%;
}
/* Filtered view - hide rows with unmatched items */
body > table > tbody > tr > td {
opacity: 1;
transition: .3s .2s;
}
body > table > tbody > tr > td > div {
overflow: hidden;
max-width: 75vw;
max-height: 10em;
transition: .3s .2s, max-width .2s;
}
body > table.od_filtered_view > tbody > tr.od_nomatch > td {
padding-block: 0;
opacity: 0;
transition: .5s;
}
body > table.od_filtered_view > tbody > tr.od_nomatch > td > div {
max-width: 0;
max-height: 0;
transition: .3s, max-width .3s .3s;
}
body > table.od_skip_transition > tbody > tr.od_nomatch > td,
body > table.od_skip_transition > tbody > tr.od_nomatch > td > div {
transition: none;
}
@media (min-width: 600px) {
#od_form {grid-template-columns: 3fr 2fr;}
body {
max-width: 800px;
min-width: fit-content;
padding: 3em;
}
}
/* -------------------------------------------------- */
/* --- COLORS --- */
/* -------------------------------------------------- */
thead > tr {box-shadow: -1px 0 0, 1px 0 0;}
:root {background-color: #efefef;}
body {background-color: #fff; border-color: #8888;}
#od_shadow_thead_sticky {box-shadow: 0 .3em .8em .3em #fff;}
#od_topbar a {color: #555;}
#od_topbar a:hover {color: #111;}
#od_form {background: #8882;}
#header, h1 {border-bottom-color: #8889;}
input.od_inp {background: #fff; border-color: #888;}
input.od_inp:invalid {background-color: #f003; outline-color: #f33;}
thead > tr {background-color: #eee; color: #eee;}
thead > tr > th {color: #333; border-color: transparent #888a #8884 transparent;}
thead > tr > th:last-child {border-right-color: transparent}
thead > tr > th:hover {background: #8883;}
body > table > tbody > tr:hover {background: #8881; outline-color: #8889;}
@media (prefers-color-scheme: dark) {
:root {background-color: #18181a;}
body {background-color: #272729;}
#od_shadow_thead_sticky {box-shadow: 0 .3em .8em .3em #272729;}
thead > tr {background-color: #333; color: #333;}
thead > tr > th {color: #eee;}
#od_topbar a {color: #bbb;}
#od_topbar a:hover {color: #eee;}
input.od_inp {background: #454546;}
}
`);
}
function addGlobalStyle(strCSS){
const h = document.querySelector('head');
if (!h) return;
const s = document.createElement('style');
s.type = 'text/css';
s.innerHTML = strCSS;
h.appendChild(s);
}
// --------------------------------------------------
// --- WHEN DOCUMENT IS READY ---
// --------------------------------------------------
if (['complete', 'interactive', 'loaded'].includes(document.readyState)){
// Document has at least been parsed
start();
} else {
// Document is not ready yet, so wait for the event
document.documentElement.style.opacity = '0';
document.addEventListener('DOMContentLoaded', start, false);
}