Add groundwork for options UI
This commit is contained in:
parent
64d309b490
commit
d7b3e01577
204
css/options.css
Normal file
204
css/options.css
Normal file
@ -0,0 +1,204 @@
|
|||||||
|
html * {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
background: black;
|
||||||
|
color: #fafafa;
|
||||||
|
margin: 0px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media only screen and (max-width: 730px) {
|
||||||
|
body {
|
||||||
|
padding: 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
h1,
|
||||||
|
h2 {
|
||||||
|
color: #fafafa;
|
||||||
|
font-family: system-ui;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
padding: 15px 0 2px 0;
|
||||||
|
text-align: center;
|
||||||
|
margin-bottom: -20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title-area {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
position: relative;
|
||||||
|
max-width: 700px;
|
||||||
|
margin: 20px auto 2px auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.setting-area {
|
||||||
|
padding-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card {
|
||||||
|
padding: 12px 18px;
|
||||||
|
background-color: #171b1f;
|
||||||
|
border: 1px solid #3e4852;
|
||||||
|
border-radius: 5px;
|
||||||
|
font-family: system-ui;
|
||||||
|
max-width: 700px;
|
||||||
|
margin: 0 auto 20px auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.setting {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 1px;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 400;
|
||||||
|
position: relative;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.setting-description {
|
||||||
|
margin: auto 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn {
|
||||||
|
background-color: rgb(42, 52, 61);
|
||||||
|
border: 1px solid #3e4852;
|
||||||
|
border-radius: 3px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
display: inline-block;
|
||||||
|
position: relative;
|
||||||
|
height: 30px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
margin-top: 1rem;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
border: 0;
|
||||||
|
border-top: 1px solid #3e4852;
|
||||||
|
box-sizing: content-box;
|
||||||
|
height: 0;
|
||||||
|
overflow: visible;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close {
|
||||||
|
margin: 6px 3px;
|
||||||
|
font-family: "Segoe UI", system-ui, sans-serif;
|
||||||
|
display: inline-block;
|
||||||
|
vertical-align: middle;
|
||||||
|
animation: pop 16ms linear;
|
||||||
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
border-radius: 1px;
|
||||||
|
background-color: rgba(0, 0, 0, 0);
|
||||||
|
opacity: 1;
|
||||||
|
border: 0;
|
||||||
|
padding: 0;
|
||||||
|
fill: #fafafa;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:hover {
|
||||||
|
fill: red;
|
||||||
|
background-color: rgba(0, 0, 0, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-text {
|
||||||
|
display: inline-block;
|
||||||
|
align-items: center;
|
||||||
|
vertical-align: middle;
|
||||||
|
margin: 6px 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hotkeys .btn-text {
|
||||||
|
margin-right: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-hov:hover {
|
||||||
|
background-color: #3e4852;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-enter {
|
||||||
|
background-color: #00b1e2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.reset-area {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
position: relative;
|
||||||
|
margin: 50px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.reset {
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 400;
|
||||||
|
background-color: rgb(202, 74, 74);
|
||||||
|
}
|
||||||
|
|
||||||
|
.reset:hover {
|
||||||
|
background-color: rgb(148, 54, 54);
|
||||||
|
}
|
||||||
|
|
||||||
|
.info {
|
||||||
|
color: #00b1e2;
|
||||||
|
position: relative;
|
||||||
|
display: inline-block;
|
||||||
|
margin-left: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info .info-content {
|
||||||
|
visibility: hidden;
|
||||||
|
width: max-content;
|
||||||
|
max-width: 450px;
|
||||||
|
background-color: rgb(44, 44, 44);
|
||||||
|
border: 1px solid rgb(1, 140, 179);
|
||||||
|
color: #fff;
|
||||||
|
padding: 5px 8px;
|
||||||
|
border-radius: 6px;
|
||||||
|
display: inline-block;
|
||||||
|
font-style: normal;
|
||||||
|
|
||||||
|
/* Positioning the tooltip */
|
||||||
|
position: absolute;
|
||||||
|
z-index: 1;
|
||||||
|
top: -5px;
|
||||||
|
left: 105%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-content img {
|
||||||
|
margin-top: 6px;
|
||||||
|
max-width: 432px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Show the tooltip when the mouse hovers the tooltip container */
|
||||||
|
.info:hover .info-content {
|
||||||
|
visibility: visible;
|
||||||
|
}
|
||||||
|
|
||||||
|
.options-container {
|
||||||
|
margin: 30px 0 15px 0;
|
||||||
|
width: 100%;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.options-container input[type="radio"] {
|
||||||
|
position: absolute;
|
||||||
|
clip: rect(0, 0, 0, 0);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.options-container img {
|
||||||
|
height: 200px;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 0 0 2px rgba(255, 255, 255, 0.1);
|
||||||
|
}
|
159
js/options.js
Normal file
159
js/options.js
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
var SETTINGS_FULL;
|
||||||
|
var HOTKEY_CODES;
|
||||||
|
|
||||||
|
function getSettings(callback) {
|
||||||
|
chrome.storage.local.get(["extension-settings"], function (result) {
|
||||||
|
SETTINGS_FULL = result["extension-settings"];
|
||||||
|
HOTKEY_CODES = SETTINGS_FULL.hotkeys.codes;
|
||||||
|
console.log(HOTKEY_CODES);
|
||||||
|
if (callback instanceof Function) {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function populateFields() {
|
||||||
|
let hotkeyDivs = document.querySelectorAll(".setting[data-type='shortcut']");
|
||||||
|
hotkeyDivs.forEach((hotkeyDiv) => {
|
||||||
|
setHotkeyBtn(hotkeyDiv);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function enterNewHotkey(event) {
|
||||||
|
let element = event.currentTarget;
|
||||||
|
element.removeEventListener("click", enterNewHotkey);
|
||||||
|
element.setAttribute("class", "btn btn-enter");
|
||||||
|
let textArea = element.children[0];
|
||||||
|
textArea.innerHTML = `
|
||||||
|
Enter the shortcut
|
||||||
|
`;
|
||||||
|
|
||||||
|
let keysDown = new Set();
|
||||||
|
let keysFinal;
|
||||||
|
window.addEventListener("keydown", keyPress);
|
||||||
|
window.addEventListener("keyup", keyRelease);
|
||||||
|
|
||||||
|
function keyPress(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
|
||||||
|
// hitting escape cancels input
|
||||||
|
if (e.key.toLowerCase() === "escape") {
|
||||||
|
window.removeEventListener("keydown", keyPress);
|
||||||
|
window.removeEventListener("keyup", keyRelease);
|
||||||
|
setHotkeyBtn(element.parentNode);
|
||||||
|
}
|
||||||
|
keysDown.add(e.key.toLowerCase(), keysDown.size);
|
||||||
|
console.log(keysDown);
|
||||||
|
textArea.innerHTML = formateHotkeys(keysDown);
|
||||||
|
keysFinal = new Set(keysDown);
|
||||||
|
}
|
||||||
|
function keyRelease(e) {
|
||||||
|
keysDown.delete(e.key.toLowerCase());
|
||||||
|
|
||||||
|
// once no more keys are pressed, the final combo is recorded
|
||||||
|
if (keysDown.size === 0) {
|
||||||
|
window.removeEventListener("keydown", keyPress);
|
||||||
|
window.removeEventListener("keyup", keyRelease);
|
||||||
|
// setHotkeyBtn(element.parentNode);
|
||||||
|
updateHotkey(element, [...keysFinal]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateHotkey(element, newVal) {
|
||||||
|
let parentSection = element.parentNode;
|
||||||
|
parentSection.removeChild(element);
|
||||||
|
let newSettings = SETTINGS_FULL;
|
||||||
|
newSettings.hotkeys.codes[parentSection.id] = newVal;
|
||||||
|
chrome.storage.local.set({ "extension-settings": newSettings }, function () {
|
||||||
|
setHotkeyBtn(parentSection);
|
||||||
|
// TODO: Send message to content scripts that settings updated
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function formateHotkeys(set1) {
|
||||||
|
let replaceTable = {
|
||||||
|
Control: "Ctrl",
|
||||||
|
Arrowup: "↑",
|
||||||
|
Arrowright: "→",
|
||||||
|
Arrowdown: "↓",
|
||||||
|
Arrowleft: "←",
|
||||||
|
" ": " Space",
|
||||||
|
Pageup: "PgUp",
|
||||||
|
Pagedown: "PgDn",
|
||||||
|
Delete: "Del",
|
||||||
|
};
|
||||||
|
|
||||||
|
let keyString = [...set1].map((c) => c.slice(0, 1).toUpperCase() + c.slice(1).toLowerCase()).join(" + ");
|
||||||
|
keyString = keyString.replace(
|
||||||
|
/Control|Arrowup|Arrowright|Arrowdown|Arrowleft|\s\s|Pageup|Pagedown|Delete/g,
|
||||||
|
function (match) {
|
||||||
|
return replaceTable[match];
|
||||||
|
}
|
||||||
|
);
|
||||||
|
keyString = keyString === " " ? "Space" : keyString;
|
||||||
|
|
||||||
|
return keyString;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setHotkeyBtn(element) {
|
||||||
|
let el = document.createElement("div");
|
||||||
|
// TODO: Check if hotkey combo in storage
|
||||||
|
|
||||||
|
// if in storage print stored combo
|
||||||
|
if (HOTKEY_CODES[element.id].length !== 0) {
|
||||||
|
console.log(element.id);
|
||||||
|
// delete previous btn
|
||||||
|
let old = document.getElementById(element.id + "-hotkey");
|
||||||
|
if (old !== null && element.parentNode) {
|
||||||
|
old.parentNode.removeChild(old);
|
||||||
|
}
|
||||||
|
el.setAttribute("id", element.id + "-hotkey");
|
||||||
|
el.setAttribute("class", "btn hotkeys");
|
||||||
|
el.innerHTML = `
|
||||||
|
<span class="btn-text">
|
||||||
|
${formateHotkeys(new Set(HOTKEY_CODES[element.id]))}
|
||||||
|
</span>
|
||||||
|
<button class="close" title="Delete shortcut">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 18 18">
|
||||||
|
<path d="M13.5 6l-1.4-1.4-3.1 3-3.1-3L4.5 6l3.1 3.1-3 2.9 1.5 1.4L9 10.5l2.9 2.9 1.5-1.4-3-2.9"></path>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
`;
|
||||||
|
el.getElementsByClassName("close")[0].addEventListener(
|
||||||
|
"click",
|
||||||
|
(deleteHotkey = function () {
|
||||||
|
updateHotkey(el, []);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
element.appendChild(el);
|
||||||
|
} else {
|
||||||
|
// delete any previous btn
|
||||||
|
let old = document.getElementById(element.id + "-btn");
|
||||||
|
if (old !== null && element.parentNode) {
|
||||||
|
old.parentNode.removeChild(old);
|
||||||
|
}
|
||||||
|
|
||||||
|
el.setAttribute("id", element.id + "-btn");
|
||||||
|
el.setAttribute("class", "btn btn-hov");
|
||||||
|
el.innerHTML = `
|
||||||
|
<span class="btn-text">
|
||||||
|
Click to type a new shortcut
|
||||||
|
</span>
|
||||||
|
`;
|
||||||
|
// Add listner for new hotkey input
|
||||||
|
el.addEventListener("click", enterNewHotkey);
|
||||||
|
element.appendChild(el);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getSettings(function () {
|
||||||
|
// once the DOM is ready file in all settings info
|
||||||
|
let stateCheck = setInterval(() => {
|
||||||
|
if (document.readyState === "complete") {
|
||||||
|
clearInterval(stateCheck);
|
||||||
|
populateFields();
|
||||||
|
}
|
||||||
|
}, 100);
|
||||||
|
});
|
@ -15,4 +15,5 @@
|
|||||||
"short_name": "spatial-navigation",
|
"short_name": "spatial-navigation",
|
||||||
"update_url": "https://clients2.google.com/service/update2/crx",
|
"update_url": "https://clients2.google.com/service/update2/crx",
|
||||||
"version": "0.4.7"
|
"version": "0.4.7"
|
||||||
|
"options_page": "options.html"
|
||||||
}
|
}
|
||||||
|
71
options.html
Normal file
71
options.html
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
|
||||||
|
<title>Spatial Navigation Options</title>
|
||||||
|
<link rel="stylesheet" href="css/options.css">
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<div class="title">
|
||||||
|
<h1>Spatial Navigation Options</h1>
|
||||||
|
<hr width="50%">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="title-area">
|
||||||
|
<h2>Keyboard Shortcuts</h2>
|
||||||
|
</div>
|
||||||
|
<div class="card">
|
||||||
|
<div class="setting-area">
|
||||||
|
<div class="setting" id="slower" data-type="shortcut">
|
||||||
|
<span class="setting-description">
|
||||||
|
Make the Video Slower
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<hr>
|
||||||
|
<div class="setting" id="normal" data-type="shortcut">
|
||||||
|
<span class="setting-description">
|
||||||
|
Make the Video Play at the Original Speed
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<hr>
|
||||||
|
<div class="setting" id="faster" data-type="shortcut">
|
||||||
|
<span class="setting-description">
|
||||||
|
Make the Video Faster
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<hr>
|
||||||
|
<div class="setting" id="pause" data-type="shortcut">
|
||||||
|
<span class="setting-description">
|
||||||
|
Toggle Play/Pause on the Video
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<hr>
|
||||||
|
<div class="setting" id="skip-back" data-type="shortcut">
|
||||||
|
<span class="setting-description">
|
||||||
|
Skip Back in the Video
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<hr>
|
||||||
|
<div class="setting" id="skip-forward" data-type="shortcut">
|
||||||
|
<span class="setting-description">
|
||||||
|
Skip Forward in the Video
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="reset-area">
|
||||||
|
<div class="btn reset">
|
||||||
|
<span class="btn-text">
|
||||||
|
Reset All Settings to Defaults
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript" src="js/options.js"></script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
Loading…
Reference in New Issue
Block a user