Last active
December 23, 2024 17:29
-
-
Save derekmc/f223518b02d86e0ba04199660a203770 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<html> | |
<head> | |
<meta name="viewport" content="width=device-width, initial-scale=1"> | |
<style> | |
/* CSS */ | |
#c0{ | |
border: 2px solid blue; | |
} | |
@media (orientation:portrait){ | |
body { | |
margin: 0; | |
overflow: hidden; | |
transform-origin: right; | |
transform: translate(-50vh, 50vh) rotate(90deg); | |
writing-mode: vertical-rl; | |
} | |
.panel{ | |
transform: rotate(-90deg) translate(0vw, -50vh) ; | |
writing-mode: vertical-rl; | |
margin: 0; | |
overflow: hidden; | |
transform-origin: right; | |
writing-mode: horizontal-rl; | |
position: absolute; | |
top:0; bottom: 0; left: 0; | |
} | |
.panel button{ | |
padding: 9px; | |
} | |
} | |
body{ | |
margin: 0; | |
overflow: hidden; | |
} | |
.panel{ | |
position: absolute; | |
bottom: 0; left: 0; right: 0; | |
text-align: center; | |
z-index: 999; | |
padding: 12px; | |
background: #000; | |
color: #fff; | |
font-family: sans-serif; | |
} | |
.panel button{ | |
background: #fff; | |
color: #000; | |
font-weight: bold; | |
border: none; | |
font-size: 20px; | |
border-radius: 12px; | |
padding: 9px; | |
margin: 5px 5px; | |
} | |
/* END */ | |
</style> | |
</head> | |
<body> | |
<div class="markdown-src"> | |
<!-- HTML --> | |
<canvas id='c0'></canvas> | |
<div class='panel'> | |
Click above to focus page then click or type to see chord codes. | |
<!--<br><br> | |
<button id='record()'>Record</button> | |
<button id='play()'>Play</button> | |
<button id='load()'>Load</button> | |
<button id='tempo()'>Tempo</button>--> | |
</div> | |
<!-- END --> | |
</div> | |
<div style="display: none;"></div> | |
<script> | |
let datakey = "__HTML_NOTES__:notedataid=DPjVRX6o2y" | |
let inframe = (window!=window.top) | |
window.Note = {} | |
window.Note.data = | |
/* JSON */ | |
{}; | |
/* END */ | |
(function(){ | |
window.Note.save = saveData | |
window.Note.load = loadData | |
window.Note.autoSave = autoSave | |
function getNoteModule(){ | |
if(!window['Note']) window.Note = {} | |
return window.Note | |
} | |
function loadData(){ | |
let note = getNoteModule() | |
if(!inframe){ | |
try{ | |
let savedata = localStorage.getItem(datakey) | |
if(savedata && savedata.length){ | |
note.data = JSON.parse(savedata) | |
} | |
window.addEventListener("message", onMessage) | |
} catch(e){ | |
console.log("No local data") | |
console.log(e) | |
} | |
} else { | |
return note.data | |
} | |
} | |
let firstLogCall = true | |
let originalLogFunc = console.log | |
function log(x, ...rest){ | |
if(firstLogCall) | |
document.body.appendChild(document.createElement("hr")) | |
else | |
document.body.appendChild(document.createElement("br")) | |
document.body.appendChild( | |
document.createTextNode(x + " " + rest.join(" "))) | |
firstLogCall = false | |
originalLogFunc(x, ...rest) | |
} | |
console.log = log | |
let SaveInterval = 500 | |
window.addEventListener("load", ()=>{ | |
convertMarkdown() | |
autoSave() | |
loadData() // dont wait for async | |
}) | |
let saveIntervalRef = null | |
function autoSave(enable){ | |
if(enable === undefined) enable = true | |
clearInterval(saveIntervalRef) | |
if(enable){ | |
saveIntervalRef = window.setInterval(saveData, SaveInterval) | |
} | |
} | |
// this is preferred over a 'localstorage' polyfill for a frame, | |
// so the programmer doesn't assume this is the browsers localstorage | |
function saveData(){ | |
let note = getNoteModule() | |
if(inframe){ | |
let msg_obj = { | |
action: "saveData", | |
data: JSON.stringify(note['data']) | |
} | |
let message = JSON.stringify(msg_obj) | |
window.parent.postMessage(message) | |
} else { | |
localStorage.setItem(datakey, JSON.stringify(note['data'])) | |
} | |
} | |
let md_subs = [ | |
/(\n|^)\s*######\s([^\s#].*)\n*/g, "\n<h6 id=\"$2\">$2</h6>\n", | |
/(\n|^)\s*#####\s([^\s#].*)\n*/g, "\n<h5 id=\"$2\">$2</h5>\n", | |
/(\n|^)\s*####\s*([^\s#].*)\n*/g, "\n<h4 id=\"$2\">$2</h4>\n", | |
/(\n|^)\s*###\s*([^\s#].*)\n*/g, "\n<h3 id=\"$2\">$2</h3>\n", | |
/(\n|^)\s*##\s*([^\s#].*)\n*/g, "\n<h2 id=\"$2\">$2</h2>\n", | |
/(\n|^)\s*#\s*([^\s#].*)\n*/g, "\n<h1 id=\"$2\">$2</h1>\n", | |
/\n(\s*)[\*\-](.*)/g, '\n<ul><li>$2</li></ul>', | |
/\n+\n(?=[^#\n])/g, "\n\n<br><br>", | |
/\n+\n/g, "\n", | |
/__([^_\n]*)__/g, "<b>$1</b>", | |
/\*\*([^_\n]*)\*\*/g, "<b>$1</b>", | |
/_([^_\n]*)_/g, "<i>$1</i>", | |
/\*([^_\n]*)\*/g, "<i>$1</i>", | |
/\!\[([^\]\n]*)\]\(([^\)\n]*)\)/g, "<img src=\"$2\" alt=\"$1\"></img>", | |
/\[([^\]\n]*)\]\(([^\)\n]*)\)/g, "<a href=\"$2\" target=\"_blank\">$1</a>", | |
] | |
function onMessage(){ | |
try{ | |
let message = JSON.parse(e.data) | |
// console.log('received message', message) | |
if(message.hasOwnProperty("event")){ | |
if(message.event == "keydown" && typeof keydown != "undefined"){ | |
keydown(message) | |
} | |
if(message.event == "keyup" && typeof keyup != "undefined"){ | |
keydown(message) | |
} | |
} | |
} catch(e){ | |
console.warn("error processing message: " + e.data) | |
} | |
} | |
function convertMarkdown(){ | |
let containers = document.getElementsByClassName("markdown-src") | |
for(let j=0; j<containers.length; ++j){ | |
let container = containers[j] | |
let src = container.innerHTML | |
for(var i=0; i<md_subs.length-1; i += 2){ | |
var search = md_subs[i] | |
var replace = md_subs[i+1] | |
src = src.replace(search, replace) | |
} | |
container.innerHTML = src | |
} | |
} | |
})() | |
//window.addEventListener('load', htmlNotesMainFunc) | |
//function htmlNotesMainFunc(){ | |
/* JS */ | |
let imgdata = "" | |
let base36 = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ' | |
let resetTime = 12*1000 | |
let dt = 200 | |
let keyrows = [74, 59, 107, 127, 143, 196, 178, 263] | |
let keydx = 68.8, keysize = 60 | |
let temp = {} | |
function reset(){ | |
console.info('Resetting temp vars') | |
temp = { | |
dirty: true, | |
activeKeys: {}, | |
holdKeys: {}, | |
portrait: false, | |
shiftPress: false, | |
resetTimeout: null, | |
} | |
resize() | |
} | |
function delayReset(){ | |
clearTimeout(temp.resetTimeout) | |
temp.resetTimeout = setTimeout(reset, resetTime) | |
} | |
let keyaddresses = qw` | |
0-0 0-2 0-3 0-7 0-8 0-10 | |
1-0 1-1 1-2 1-7 1-8 1-9 | |
2-1 2-2 2-3 2-4 2-5 2-6 2-7 2-8 | |
3-2 3-3 3-5 3-6` | |
let keynames = qw` | |
1 3 4 8 9 - | |
q w e i o p | |
s d f g h j k l | |
c v n m` | |
let keyvalues = qw` | |
8 3 3 3 3 8 | |
4 2 2 2 2 4 | |
1 1 4 8 8 4 1 1 | |
G C C G` | |
let keydigits = qw` | |
1 1 0 2 3 3 | |
1 1 0 2 3 3 | |
1 0 0 0 2 2 2 3 | |
0 0 2 2` | |
let keyfingers = qw` | |
3 2 1 5 6 7 | |
3 2 1 5 6 7 | |
2 1 0 0 4 4 5 6 | |
0 0 4 4` | |
window.addEventListener('load', main) | |
function addBase36(x, y){ | |
let n = base36.indexOf(x) | |
let m = base36.indexOf(y) | |
if(n < 0 || m < 0){ | |
console.warn("invalid base36 digit", n , m) | |
} | |
let s = n + m | |
return (s > 35? '1' : '') + base36[s%36] | |
} | |
function resize(){ | |
let width = window.innerWidth | |
let height = window.innerHeight | |
temp.portrait = height > width | |
if(temp.portrait){ // show sideways in portrait | |
[width, height] = [height, width] } | |
id('c0').width = width - 4 | |
id('c0').height = height - 5 | |
} | |
function main(){ | |
reset() | |
window.addEventListener('mousedown', mousedown) | |
window.addEventListener('keydown', keydown) | |
window.addEventListener('keyup', keyup) | |
window.addEventListener('resize', resize) | |
window.addEventListener('mousedown', delayReset) | |
window.addEventListener('keydown', delayReset) | |
window.addEventListener('keyup', delayReset) | |
window.addEventListener('resize', delayReset) | |
setInterval(draw, dt) | |
setTimeout(()=>temp.dirty=true, 1500) | |
setTimeout(()=>id('c0').focus(),1000) | |
} | |
function touchstart(e){ | |
e.preventDefault() | |
mousedown({clientX: e.targetTouches[0].clientX, | |
clientY: e.targetTouches[0].clientY}) | |
} | |
function mousedown(e){ | |
let x = e.clientX, y = e.clientY | |
let w = id('c0').width, h = id('c0').height | |
if(temp.portrait){ // show sideways in portrait | |
[x, y] = [y, h-x]; | |
console.info('portrait', x, y, w, h) | |
} | |
x = Math.round(1000*x/w) | |
y = Math.round(1000*y/w) | |
let keyIndex = null | |
for(let j=0; j<4; ++j){ | |
if(keyIndex !== null) break | |
let x0 = keyrows[2*j], y0 = keyrows[2*j+1] | |
for(let i=0; i<12-j; ++i){ | |
let x1 = x0 + keydx*i | |
if(x > x1 && x < x1 + keysize && y > y0 && y < y0 + keysize){ | |
let keyaddress = j + '-' + i | |
keyIndex = keyaddresses.indexOf(keyaddress) | |
console.info('key', keyaddress, keyIndex) | |
if(keyIndex < 0) keyIndex = null | |
else break | |
} | |
} | |
} | |
if(keyIndex == null){ | |
// clear active and hold keys | |
temp.activeKeys = {} | |
temp.holdKeys = {} | |
} else { | |
// toggle the active and hold keys | |
if(temp.activeKeys[keyIndex] || temp.holdKeys[keyIndex]){ | |
delete temp.holdKeys[keyIndex] | |
delete temp.activeKeys[keyIndex] | |
} else { | |
//temp.activeKeys[keyIndex] = 1 | |
temp.holdKeys[keyIndex] = 1 | |
for(let i=0; i<keyfingers.length; ++i){ | |
if(i != keyIndex && keyfingers[i] == keyfingers[keyIndex]){ | |
delete temp.holdKeys[i] | |
delete temp.activeKeys[i] | |
} | |
} | |
} | |
} | |
//console.info(x, y) | |
} | |
function draw(){ | |
//if(!temp.dirty) return | |
temp.dirty = false | |
let {width, height} = id('c0') | |
let img = new Image | |
let ctx = id('c0').getContext('2d') | |
img.src = imgdata | |
ctx.fillStyle = 'black' | |
ctx.fillRect(0, 0, width, height) | |
let w = img.naturalWidth, h = img.naturalHeight | |
let scale = width/w | |
ctx.drawImage(img, 0, 0, scale*w, scale*h) | |
let hold = Object.keys(temp.holdKeys) | |
//console.info('active', hold, hold.map(i=>keyaddresses[i])) | |
let skip = hold.length > 0? hold.map(i=>keyaddresses[i]) : keyaddresses | |
drawKeyCovers(ctx, skip) | |
ctx.fillStyle = '#fff' | |
drawCodes(ctx, scale*w/2, scale*h + 15) | |
ctx.textAlign = 'center' | |
ctx.fillStyle = '#fff' | |
ctx.font = "18px arial" | |
ctx.textBaseline = 'top' | |
ctx.fillText('Chord-20-20 System', width/2, 10) | |
ctx.textBaseline = 'bottom' | |
ctx.font = "14px arial" | |
//ctx.fillText('Click keyboard to focus and type to see chord codes.', width/2, 0.85*height - 40) | |
} | |
function drawKeyCovers(ctx, skip){ | |
let w = id('c0').width | |
let s = w/1000 | |
ctx.fillStyle = 'rgba(0,15,30,0.82)' | |
for(let j=0; j<4; ++j){ | |
let x0 = keyrows[2*j], y = keyrows[2*j+1] | |
for(let i=0; i<12-j; ++i){ | |
if(skip.indexOf(j + "-" + i) >= 0) | |
continue | |
let x = x0 + keydx*i | |
ctx.fillRect(s*x, s*y, s*keysize, s*keysize) | |
} | |
} | |
} | |
function drawCodes(ctx, x, y){ | |
let hold = Object.keys(temp.holdKeys) | |
let code = qw`0 0 0 0` | |
for(let i=0; i<hold.length; ++i){ | |
let n = hold[i] | |
let pos = keydigits[n] | |
let val = keyvalues[n] | |
code[pos] = addBase36(code[pos], val) | |
} | |
if(code[1] == '0' && code[3] == '0'){ | |
code[1] = code[3] = ' ' | |
} | |
ctx.textAlign = 'center' | |
ctx.textBaseline = 'hanging' | |
ctx.font = "30px monospace" | |
ctx.fillText(code.join(' '), x, y) | |
//console.info('code', code.join(' ')) | |
} | |
function keydown(e){ | |
let k = e.key.toLowerCase() | |
//console.log('key', k) | |
let i = keynames.indexOf(k) | |
if(i >= 0 && !temp.activeKeys[i]){ | |
temp.dirty = true | |
temp.activeKeys[i] = temp.holdKeys[i] = 1 | |
for(let j=0; j<keyfingers.length; ++j){ | |
if(i != j && keyfingers[i] == keyfingers[j]){ | |
delete temp.holdKeys[j]} | |
} | |
} | |
} | |
function keyup(e){ | |
let k = e.key.toLowerCase() | |
let i = keynames.indexOf(k) | |
if(i >= 0){ | |
//console.info(`keyup: '${k}' #${i}`) | |
temp.dirty = true | |
delete temp.activeKeys[i] | |
//delete temp.holdKeys[i] | |
if(Object.keys(temp.activeKeys).length == 0){ | |
temp.holdKeys = {} | |
} | |
} | |
} | |
function qw(parts, ...rest){ | |
let s = '' | |
for(let i=0; i<rest.length; ++i){ | |
s += parts[i] + rest[i]} | |
s += parts[rest.length] | |
return s.trim().split(/\s+/) | |
} | |
function id(x){ | |
return document.getElementById(x) | |
} | |
/* END */ | |
//} | |
</script> | |
</body> | |
<html> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment