You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
395 lines
10 KiB
395 lines
10 KiB
const ADD = 'add'; |
|
const BODY = 'body'; |
|
const CU = 'cu'; |
|
const DIV = 'DIV'; |
|
const MOVE = 'move'; |
|
const OPAC = 100; |
|
const PLAN = '#plan'; |
|
const POST = 'POST'; |
|
const PROPS = 'props'; |
|
const SQUARE = 30; |
|
const SVG = 'svg'; |
|
var selected = null; |
|
var mode = null; |
|
var messageTimer = null; |
|
var messageOpacity = 0; |
|
var pendingAssignment = null; |
|
var clickHistory = []; |
|
var messages = []; |
|
var android = false; |
|
|
|
function addBlockDirections(){ |
|
$('.BlockH').each(function (){ |
|
let block = $(this); |
|
let width = block.attr('viewBox').split(" ")[2]; |
|
block.append('<polygon points="0,0 100,50 0,100" class="direction"></polygon>'); |
|
block.append('<polygon points="'+width+',0 '+(width-100)+',50 '+width+',100" class="direction"></polygon>'); |
|
block.html(block.html()); |
|
}); |
|
$('.BlockV').each(function (){ |
|
let block = $(this); |
|
let height = block.attr('viewBox').split(" ")[3]; |
|
block.append('<polygon points="0,0 50,100 100,0" class="direction"></polygon>'); |
|
block.append('<polygon points="0,'+height+' 50,'+(height-100)+' 100,'+height+'" class="direction"></polygon>'); |
|
block.html(block.html()); |
|
}); |
|
} |
|
|
|
function addClass(data){ |
|
parts = data.split(" "); |
|
$('#'+parts[0]).addClass(parts[1]); |
|
} |
|
|
|
function addMessage(txt){ |
|
messages.unshift(txt); |
|
if (messages.length>5) messages.pop(); |
|
updateMessages(); |
|
setTimeout(fadeMessage,60000); |
|
} |
|
|
|
function addTile(x,y){ |
|
return request({realm:'plan',action:mode,tile:selected.id,x:x,y:y}); |
|
} |
|
|
|
function arrangeTabs(){ |
|
var id = 0; |
|
var tabs = $('<div/>',{'class':'tabs'}); |
|
var winId = $('.window').attr('id')+"-"; |
|
|
|
tabs.insertAfter($('.error')); |
|
var target = null; |
|
var index = null; |
|
$('.window > fieldset > legend').each(function(){ |
|
var fs = this.parentNode; |
|
if (!fs.id) fs.id = winId+id; |
|
var i = clickHistory.indexOf(fs.id); |
|
if (i>-1 && (index == null || i<index)) { |
|
index = i; |
|
target = this; |
|
} |
|
$(this).appendTo(tabs).click(fs.id,clickLegend); |
|
if (id > 0) { |
|
$(fs).hide(); |
|
} else $(this).addClass('front'); |
|
id++; |
|
}); |
|
if (index != null) clickLegend({'data':clickHistory[index],'target':target,'no-update':true}); |
|
} |
|
|
|
function assign(context){ |
|
pendingAssignment = context; |
|
closeWindows(); |
|
$(PLAN).css('cursor','help'); |
|
if (context.realm == 'train' && context.action == 'move')addBlockDirections(); |
|
return false; |
|
} |
|
|
|
function changeSpeed(inputId){ |
|
console.log(inputId); |
|
let parts = inputId.split('_'); |
|
let val = $('#'+inputId).val(); |
|
let data = { realm : parts[0], id : parts[1], action : "setSpeed", speed : val}; |
|
console.log(data); |
|
let caption = $('#'+inputId+'_caption'); |
|
caption.text(caption.text().replace(/\d+/,val)); |
|
request(data); |
|
} |
|
|
|
function clickLegend(ev){ |
|
lastTab = ev.data; |
|
$('.window > .tabs > legend').removeClass('front'); |
|
$(ev.target).addClass('front'); |
|
$('.window > fieldset').hide(); |
|
$('#'+lastTab).show(); |
|
if (!android) $('#'+lastTab+" input:not([type=hidden])").first().focus().select(); |
|
if (!('no-update' in ev)) remember(lastTab); |
|
} |
|
|
|
function clickTile(x,y,shift){ |
|
console.log("clickTile("+x+","+y+")"); |
|
var id = x+"-"+y; |
|
var tiles = $('#'+id); |
|
if (tiles.length > 0) { |
|
if (pendingAssignment != null) { |
|
var key = pendingAssignment.assign; |
|
delete pendingAssignment.assign; |
|
console.log("assigning key:",key); |
|
pendingAssignment[key] = id; |
|
console.log("pA:",pendingAssignment); |
|
request(pendingAssignment); |
|
pendingAssignment = null; |
|
$(PLAN).css('cursor',''); |
|
$('.direction').remove(); |
|
return false; |
|
} |
|
var json = {realm:'plan',action:'click',id:id}; |
|
if (shift) json.shift=1; |
|
request(json); |
|
} |
|
return false; |
|
} |
|
|
|
|
|
function closeMenu(ev){ |
|
if (selected != null) $(selected).css('border',''); |
|
$('.menu .list').css('display','') |
|
mode = null; |
|
selected = null; |
|
return false; |
|
} |
|
|
|
function closeWindows(){ |
|
$('.window').remove(); |
|
$('.preview').removeClass('preview'); |
|
$('#plan').css('height','').css('width',''); |
|
} |
|
|
|
function connectCu(){ |
|
return request({realm:CU,action:"connect"}); |
|
} |
|
|
|
function copyCv(ev){ |
|
var td = ev.parentNode.previousSibling; |
|
$('input[name=cv]').val(td.previousSibling.innerText); |
|
$('input[name=val]').val(td.innerText); |
|
} |
|
|
|
function dropClass(data){ |
|
var parts = data.split(" "); |
|
for (var i=1; i<parts.length; i++) $('#'+parts[0]).removeClass(parts[i]); |
|
} |
|
|
|
function enableAdding(ev){ |
|
// console.log('enableAdding:',ev); |
|
if (selected != null) $(selected).css('border',''); |
|
selected = ev.target; |
|
while (selected != null && selected.nodeName != SVG) selected = selected.parentNode; |
|
if (selected == null){ |
|
mode = null; |
|
} else { |
|
$(selected).css('border','2px solid red'); |
|
$('.menu .addtile .list').css('display','inherit'); |
|
mode = ADD; |
|
} |
|
return false; // otherwise body.click would also be triggered |
|
} |
|
|
|
function enableMove(ev){ |
|
// console.log('enableMove:',ev); |
|
if (selected != null) $(selected).css('border',''); |
|
selected = ev.target; |
|
while (selected != null && selected.nodeName != DIV) selected = selected.parentNode; |
|
if (selected == null){ |
|
mode = null; |
|
} else { |
|
$(selected).css('border','2px solid red'); |
|
$('.menu .move .list').css('display','inherit'); |
|
mode = MOVE; |
|
} |
|
return false; // otherwise body.click would also be triggered |
|
} |
|
|
|
function fadeMessage(){ |
|
if (messages.length > 1) { |
|
messages.pop(); |
|
updateMessages(); |
|
} else { |
|
messageOpacity -= 5; |
|
if (messageOpacity < 1) { |
|
messages.pop(); |
|
updateMessages(); |
|
return; |
|
} |
|
messageTimer = setTimeout(fadeMessage,100); |
|
var o = messageOpacity; |
|
if (o>OPAC) o=OPAC; |
|
$('#messages').css('opacity',o/OPAC); |
|
} |
|
} |
|
|
|
function getCookie(key) { |
|
var keyValue = document.cookie.match('(^|;) ?' + key + '=([^;]*)(;|$)'); |
|
return keyValue ? keyValue[2] : null; |
|
} |
|
|
|
function heartbeat(data){ |
|
$('#heartbeat').show().fadeOut(2000); |
|
return false; |
|
} |
|
|
|
function highlight(id,enable = true){ |
|
if (enable) { |
|
$('#'+id).addClass('highlight'); |
|
} else { |
|
$('#'+id).removeClass('highlight'); |
|
} |
|
} |
|
|
|
function keypress(ev){ |
|
if (ev.code === 'Escape') return request({realm:"cu",action:"emergency"}); |
|
if (ev.code === 'Enter') { |
|
elem = ev.target; |
|
while (elem != undefined && elem != null){ |
|
if (elem.nodeName == 'TEXTAREA') return false; |
|
if (elem.nodeName == 'FORM') return submitForm(elem.getAttribute('id')); |
|
elem = elem.parentNode; |
|
} |
|
} |
|
return false; |
|
} |
|
|
|
function moveTile(x,y){ |
|
var id = x+"-"+y; |
|
return request({realm:'plan',action:mode,direction:selected.id,id:id}); |
|
} |
|
|
|
function place(data){ |
|
var elem = $('#'+$(data).attr('id')); |
|
if (elem.length) { |
|
elem.replaceWith(data); |
|
} else $('#scroll').append(data); |
|
return false; |
|
} |
|
|
|
function planClick(ev){ |
|
console.log('planClick:',ev); |
|
var plan=$('#scroll').get(0); |
|
var x = Math.floor((plan.scrollLeft+ev.clientX)/SQUARE); |
|
var y = Math.floor((plan.scrollTop+ev.clientY)/SQUARE); |
|
|
|
switch (mode){ |
|
case undefined: |
|
case null: |
|
return clickTile(x,y,ev.shiftKey); |
|
case ADD: |
|
return addTile(x,y); |
|
case MOVE: |
|
return moveTile(x,y); |
|
} |
|
console.log('unknown action "'+mode+'" @ ('+ev.clientX+','+ev.clientY+')'); |
|
} |
|
|
|
function remember(lastClickedId){ |
|
console.log("lastClickedId: "+lastClickedId) |
|
var index = clickHistory.indexOf(lastClickedId); |
|
if (index > -1) clickHistory.splice(index, 1); |
|
clickHistory.unshift(lastClickedId); |
|
} |
|
|
|
function remove(id){ |
|
$('#'+id).remove(); |
|
return false; |
|
} |
|
|
|
function request(data){ |
|
console.log("request:",data); |
|
$.ajax({ |
|
url : 'plan', |
|
method : POST, |
|
data : data, |
|
success: function(resp){ |
|
var win = $('.window'); |
|
var scroll = win.scrollTop(); |
|
var winId = win.attr('id'); |
|
|
|
if (data.realm != 'car' && data.realm != 'loco') closeWindows(); |
|
if (resp.startsWith('<html')) return; |
|
if (resp.startsWith('<svg')){ |
|
place(resp); |
|
} else if (resp.startsWith('<')) { |
|
var isWindow = $(resp).attr('class') == 'window'; |
|
if (isWindow) $('.window').remove(); |
|
$(BODY).append($(resp)); |
|
if (isWindow) { |
|
tileWindow(); |
|
arrangeTabs(); |
|
var win = $('.window'); |
|
if (win.attr('id')==winId) win.scrollTop(scroll); |
|
} |
|
} else { |
|
addMessage(resp); |
|
} |
|
$("html").scrollTop(scroll); |
|
} |
|
}); |
|
return false; |
|
} |
|
|
|
function runAction(ev){ |
|
var clicked = ev.target; |
|
var realm = clicked.hasAttribute('class') ? clicked.getAttribute('class') : null; |
|
console.log("runAction: ",{action: clicked.id, realm:realm}); |
|
if (clicked.id == 'qrcode'){ |
|
window.open("https://api.qrserver.com/v1/create-qr-code/?data="+window.location.href,'_blank'); |
|
} else if (clicked.id == 'fullscreen'){ |
|
toggleFullscreen(); |
|
} else return request({action:ev.target.id,realm:realm}); // TODO: ask for name |
|
return false; |
|
} |
|
|
|
function stream(ev){ |
|
var data = ev.data; |
|
data = data.replace(/%newline%/g,"\n"); |
|
console.log("received: ",data); |
|
if (data.startsWith('<svg')) return place(data); |
|
if (data.startsWith("heartbeat")) return heartbeat(data); |
|
if (data.startsWith("place ")) return place(data.substring(6)); |
|
if (data.startsWith("remove")) return remove(data.substring(7)); |
|
if (data.startsWith("addclass")) return addClass(data.substring(9)); |
|
if (data.startsWith("dropclass")) return dropClass(data.substring(10)); |
|
|
|
if (data.startsWith("<div") && $(data).attr('class') == 'window'){ |
|
$('.window').remove(); |
|
$(BODY).append($(data)); |
|
tileWindow(); |
|
return; |
|
} |
|
|
|
addMessage(data); |
|
} |
|
|
|
function submitForm(formId){ |
|
console.log("submitForm("+formId+")"); |
|
return request($('#'+formId).serialize()); |
|
} |
|
|
|
function swapTiling(ev){ |
|
var vertical = getCookie("tiling") == 'v'; |
|
document.cookie = "tiling="+(vertical?'h':'v'); |
|
tileWindow(); |
|
} |
|
|
|
function tileWindow(){ |
|
var vertical = getCookie("tiling") == 'v'; |
|
var width = vertical ? '50%':''; |
|
var height = vertical ? '' : '50%'; |
|
$(PLAN).css('width',width).css('height',height); |
|
$('.window').css('width',width).css('height',height); |
|
$('.swapbtn').text(vertical ? '⇩' : '⇨'); |
|
} |
|
|
|
function toggleFullscreen(){ |
|
if (document.fullscreenElement == null){ |
|
document.documentElement.requestFullscreen(); |
|
} else document.exitFullscreen(); |
|
} |
|
|
|
function updateMessages() { |
|
if (messages.length>0) { |
|
messageOpacity = 1000; |
|
$('#messages').html(messages.slice().reverse().join('<br/>')).css('opacity',messageOpacity/OPAC); |
|
} else $('#messages').html(''); |
|
} |
|
|
|
window.onload = function () { |
|
android = navigator.userAgent.toLowerCase().includes('android') |
|
$('.menu > div').click(closeMenu); |
|
$('.menu .addtile .list svg').click(enableAdding); |
|
$('.menu .move .list div').click(enableMove); |
|
$('.menu .actions .list > div').click(runAction); |
|
$('.menu .trains .list > div').click(runAction); |
|
$('.menu .hardware .list > div').click(runAction); |
|
$(PLAN).click(planClick); |
|
$(document).keyup(keypress); |
|
(new EventSource("stream")).onmessage = stream; |
|
}
|
|
|