var HIGHLIGHT = '#cc00cc'; var REGULAR = '#666666'; var cols = { 'a':173, 'b':325, 'c':589, 'd':797, 'e':1005, 'f':1195, 'g':1402, 'gg':1515, 'h':1572, 'i':1799, 'j':1893, 'k':1988, 'l':2157, 'll':2346, 'm':2403, 'mm':2535, 'n':2554, 'o':2649, 'oo':2781, 'ooo':2801, 'p':2894, 'q':3007 }; var rows = { '1':221, '2':298, '3':373, '4':448, '5':524, '6':599, '7':675, '8':751, '9':826, '10':902, '11':977, '12':1053, '13':1129, '14':1204, '15':1280, '16':1355, '17':1431, '18':1506, '19':1583, '20':1658, '21':1734, '22':1809, '23':1885, '24':1961, '25':2036, '26':2112 }; var edges = { 'b14b13':['b14','b13'], 'b13b12':['b13','b12'], 'b13503':['b13','503'], 'b12b11':['b12','b11'], 'b12501':['b12','501'], 'b11b10':['b11','b10'], 'b11414':['b11','414'], 'b10b9':['b10','b9'], 'b10405':['b10','405'], 'b9b8':['b9','b8'], 'b9400':['b9','400'], 'b8b7':['b8','b7'], 'b8401':['b8','401'], 'b7b6':['b7','b6'], 'b7403':['b7','403'], 'b6b5':['b6','b5'], 'b6501':['b6','501a'], 'b5b4':['b5','b4'], 'b5415':['b5','415'], 'b4b3':['b4','b3'], 'b4413':['b4','b4'], 'b3c3':['b3','c3'], 'b3200':['b3','200'], 'c3c4':['c3','c4'], 'c3d4':['c3','d3','d4'], 'c4d4':['c4','d4'], 'c4406':['c4','406'], 'd4d5':['d4','d5'], 'd4e5':['d4','e4','e5'], 'd5e5':['d5','e5'], 'd5406':['d5','d7','406'], 'e5e6':['e5','e6'], 'e5f6':['e5','f5','f6'], 'e6f6':['e6','f6'], 'e6406':['e6','e7','406'], 'f6f7':['f6','f7'], 'f6g7':['f6','g6','g7'], 'f7g7':['f7','g7'], 'f7406':['f7','406'], 'g7g8':['g7','g8'], 'g7h7':['g7','h7'], 'g8g9':['g8','g9'], 'g8h10':['g8','h8','h10'], 'g9g11':['g9','g11'], 'g9h10':['g9','gg9','gg10','h10'], 'g11h10':['g11','gg11','gg10','h10'], 'g11412':['g11','g18','412a'], 'h7i7':['h7','i7'], 'h7412':['h7','412'], 'h10h11':['h10','h11'], 'h10i12':['h10','i10','i12'], 'h11h12':['h11','h12'], 'h11i12':['h11','i11','i12'], 'h12i12':['h12','i12'], 'h12412':['h12','412a'], 'i4p3':['i4','i3','p3'], 'i4301':['i4','301'], 'i7i4':['i7','i4'], 'i7k7':['i7','k7'], 'i12l13':['i12','l12','l13'], 'i12i13':['i12','i13'], 'i13k13':['i13','k13'], 'i13j18':['i13','i17','j17','j18'], 'j18412':['j18','412a'], 'j18304':['j18','304'], 'k5l5':['k5','l5'], 'k5301':['k5','301'], 'k7k5':['k7','k5'], 'k7l7':['k7','l7'], 'k13j18':['k13','k17','j17','j18'], 'k13l13':['k13','l13'], 'l5m5':['l5','m5'], 'l5307':['l5','307'], 'l7m7':['l7','m7'], 'l7404':['l7','l8','404'], 'l13l14':['l13','l14'], 'l13m16':['l13','m13','m16'], 'l14l15':['l14','l15'], 'l14m16':['l14','m14','m16'], 'l15l17':['l15','l17'], 'l15m16':['l15','ll15','ll16','m16'], 'l17m16':['l17','ll17','ll16','m16'], 'l17304':['l17','304'], 'm5n5':['m5','n5'], 'm5410':['m5','m4','410'], 'm7n11':['m7','n7','n11'], 'm7404':['m7','404'], 'm16m20':['m16','m20'], 'm16n16':['m16','n16'], 'm20o20':['m20','o20'], 'm20202':['m20','202'], 'n5n11':['n5','n11'], 'n5410':['n5','410'], 'n11p11':['n11','p11'], 'n11303':['n11','303'], 'n16n11':['n16','n11'], 'n16o16':['n16','o16'], 'o14p11':['o14','o11','p11'], 'o14409':['o14','409a'], 'o16o14':['o16','o14'], 'o16o18':['o16','o18'], 'o18200':['o18','200a'], 'o18300':['o18','oo18','300'], 'o20o18':['o20','o18'], 'o20204':['o20','204'], 'p3p11':['p3','p11'], 'p3409':['p3','409'], 'p11o20':['p11','p20','o20'], 'p11201':['p11','q11','201'] }; var ends = { '200': {col:'a', row:'3', width:190}, '200a': {col:'mm', row:'18', width:116}, '201': {col:'q', row:'12', width:154}, '202': {col:'m', row:'21', width:116}, '204': {col:'o', row:'21', width:152}, '300': {col:'oo', row:'19', width:152}, '301': {col:'k', row:'4', width:154}, '303': {col:'m', row:'11', width:116}, '304': {col:'l', row:'18', width:116}, '307': {col:'l', row:'4', width:154}, '400': {col:'a', row:'9', width:190}, '401': {col:'a', row:'8', width:190}, '403': {col:'a', row:'7', width:190}, '404': {col:'m', row:'8', width:116}, '405': {col:'a', row:'10', width:190}, '406': {col:'c', row:'7', width:152}, '409': {col:'p', row:'2', width:116}, '409a': {col:'oo', row:'14', width:116}, '410': {col:'n', row:'4', width:116}, '412': {col:'h', row:'6', width:152}, '412a': {col:'h', row:'18', width:152}, '413': {col:'a', row:'4', width:190}, '414': {col:'a', row:'11', width:190}, '415': {col:'a', row:'5', width:190}, '501a': {col:'a', row:'6', width:190}, '501': {col:'a', row:'12', width:190}, '503': {col:'a', row:'13', width:190} }; var canvas; function decorateTrace() { trace[0].x = cols[trace[0].d[0]]; trace[0].y = rows[trace[0].d.slice(1)]; trace[0].previewCalls = previewCalls(trace[0]); for (var i = 1; i < trace.length; i++) { trace[i].x = cols[trace[i].d[0]]; trace[i].y = rows[trace[i].d.slice(1)]; trace[i].previewCalls = previewCalls(trace[i]); var path = edges[trace[i-1].d+trace[i].d]; if (path) { trace[i].path = [path.length-1]; for (var p = 1; p < path.length; p++) { trace[i].path[p-1] = getSeg(path[p-1], path[p], p == path.length-1); } } else { trace[i].path = []; } } var path = edges[trace[i-1].d+response.code]; if (path) { var end = ends[path[path.length-1]]; response.x = cols[end.col]; response.y = rows[end.row]; response.width = end.width; response.type = 'normal'; response.path = [path.length-1]; for (var p = 1; p < path.length; p++) { response.path[p-1] = getSeg(path[p-1], path[p], p == path.length-1); } } else { var ld = trace[trace.length-1]; response.x = ld.x+50; response.y = ld.y-50; response.width = 38; response.type = 'other'; response.path = [ {x1: ld.x+10, y1: ld.y-10, x2: ld.x+36, y2: ld.y-36} ]; } }; function previewCalls(dec) { var prev = ''; for (var i = 0; i < dec.calls.length; i++) { if (dec.calls[i].output != "wmtrace_not_exported") prev += '<li>'+dec.calls[i].module+':'+dec.calls[i]['function']+'</li>'; } return prev; }; function drawTrace() { drawDecision(trace[0]); for (var i = 1; i < trace.length; i++) { drawPath(trace[i].path); drawDecision(trace[i]); } drawPath(response.path); drawResponse(); }; function drawResponse() { if (response.type == 'normal') { var context = canvas.getContext('2d'); context.strokeStyle=HIGHLIGHT; context.lineWidth=4; context.beginPath(); context.rect(response.x-(response.width/2), response.y-19, response.width, 38); context.stroke(); } else { var context = canvas.getContext('2d'); context.strokeStyle='#ff0000'; context.lineWidth=4; context.beginPath(); context.arc(response.x, response.y, 19, 0, 2*3.14159, false); context.stroke(); } }; function drawDecision(dec) { var context = canvas.getContext('2d'); if (dec.previewCalls == '') context.strokeStyle=REGULAR; else context.strokeStyle=HIGHLIGHT; context.lineWidth=4; context.beginPath(); context.moveTo(dec.x, dec.y-19); context.lineTo(dec.x+19, dec.y); context.lineTo(dec.x, dec.y+19); context.lineTo(dec.x-19, dec.y); context.closePath(); context.stroke(); }; function drawPath(path) { var context = canvas.getContext('2d'); context.strokeStyle=REGULAR; context.lineWidth=4; context.beginPath(); context.moveTo(path[0].x1, path[0].y1); for (var p = 0; p < path.length; p++) { context.lineTo(path[p].x2, path[p].y2); } context.stroke(); }; function getSeg(p1, p2, last) { var seg = { x1:cols[p1[0]], y1:rows[p1.slice(1)] }; if (ends[p2]) { seg.x2 = cols[ends[p2].col]; seg.y2 = rows[ends[p2].row]; } else { seg.x2 = cols[p2[0]]; seg.y2 = rows[p2.slice(1)]; } if (seg.x1 == seg.x2) { if (seg.y1 < seg.y2) { seg.y1 = seg.y1+19; if (last) seg.y2 = seg.y2-19; } else { seg.y1 = seg.y1-19; if (last) seg.y2 = seg.y2+19; } } else { //assume seg.y1 == seg.y2 if (seg.x1 < seg.x2) { seg.x1 = seg.x1+19; if (last) seg.x2 = seg.x2-(ends[p2] ? (ends[p2].width/2) : 19); } else { seg.x1 = seg.x1-19; if (last) seg.x2 = seg.x2+(ends[p2] ? (ends[p2].width/2) : 19); } } return seg; }; function traceDecision(name) { for (var i = trace.length-1; i >= 0; i--) if (trace[i].d == name) return trace[i]; }; var detailPanels = {}; function initDetailPanels() { var windowWidth = document.getElementById('sizetest').clientWidth; var infoPanel = document.getElementById('infopanel'); var panelWidth = windowWidth-infoPanel.offsetLeft; var panels = { 'request': document.getElementById('requestdetail'), 'response': document.getElementById('responsedetail'), 'decision': document.getElementById('decisiondetail') }; var tabs = { 'request': document.getElementById('requesttab'), 'response': document.getElementById('responsetab'), 'decision': document.getElementById('decisiontab') }; var decisionId = document.getElementById('decisionid'); var decisionCalls = document.getElementById('decisioncalls'); var callInput = document.getElementById('callinput'); var callOutput = document.getElementById('calloutput'); var lastUsedPanelWidth = windowWidth-infoPanel.offsetLeft; var setPanelWidth = function(width) { infoPanel.style.left = (windowWidth-width)+'px'; canvas.style.marginRight = (width+20)+'px'; panelWidth = width; }; setPanelWidth(panelWidth); var ensureVisible = function() { if (windowWidth-infoPanel.offsetLeft < 10) setPanelWidth(lastUsedPanelWidth); }; var decChoices = ''; for (var i = 0; i < trace.length; i++) { decChoices += '<option value="'+trace[i].d+'">'+trace[i].d+'</option>'; } decisionId.innerHTML = decChoices; decisionId.selectedIndex = -1; decisionId.onchange = function() { detailPanels.setDecision(traceDecision(decisionId.value)); } detailPanels.setDecision = function(dec) { decisionId.value = dec.d; var calls = []; for (var i = 0; i < dec.calls.length; i++) { calls.push('<option value="'+dec.d+'-'+i+'">'); calls.push(dec.calls[i].module+':'+dec.calls[i]['function']); calls.push('</option>'); } decisionCalls.innerHTML = calls.join(''); decisionCalls.selectedIndex = 0; decisionCalls.onchange(); }; detailPanels.show = function(name) { for (p in panels) { if (p == name) { panels[p].style.display = 'block'; tabs[p].className = 'selectedtab'; } else { panels[p].style.display = 'none'; tabs[p].className = ''; } } ensureVisible(); }; detailPanels.hide = function() { setPanelWidth(0); } decisionCalls.onchange = function() { var val = decisionCalls.value; if (val) { var dec = traceDecision(val.substring(0, val.indexOf('-'))); var call = dec.calls[parseInt(val.substring(val.indexOf('-')+1, val.length))]; if (call.output != "wmtrace_not_exported") { callInput.style.color='#000000'; callInput.innerHTML = call.input; if (call.output != null) { callOutput.style.color = '#000000'; callOutput.innerHTML = call.output; } else { callOutput.style.color = '#ff0000'; callOutput.textContent = 'Error: '+call.module+':'+call['function']+' never returned'; } } else { callInput.style.color='#999999'; callInput.textContent = call.module+':'+call['function']+' was not exported'; callOutput.textContent = ''; } } else { callInput.textContent = ''; callOutput.textContent = ''; } }; var headersList = function(headers) { var h = ''; for (n in headers) h += '<li>'+n+': '+headers[n]; return h; }; document.getElementById('requestmethod').innerHTML = request.method; document.getElementById('requestpath').innerHTML = request.path; document.getElementById('requestheaders').innerHTML = headersList(request.headers); document.getElementById('requestbody').innerHTML = request.body; document.getElementById('responsecode').innerHTML = response.code; document.getElementById('responseheaders').innerHTML = headersList(response.headers); document.getElementById('responsebody').innerHTML = response.body; var infoControls = document.getElementById('infocontrols'); var md = false; var dragged = false; var msoff = 0; infoControls.onmousedown = function(ev) { md = true; dragged = false; msoff = ev.clientX-infoPanel.offsetLeft; }; infoControls.onclick = function(ev) { if (dragged) { lastUsedPanelWidth = panelWidth; } else if (panelWidth < 10) { switch(ev.target.id) { case 'requesttab': detailPanels.show('request'); break; case 'responsetab': detailPanels.show('response'); break; case 'decisiontab': detailPanels.show('decision'); break; default: ensureVisible(); } } else { var name = 'none'; switch(ev.target.id) { case 'requesttab': name = 'request'; break; case 'responsetab': name = 'response'; break; case 'decisiontab': name = 'decision'; break; } if (panels[name] && panels[name].style.display != 'block') detailPanels.show(name); else detailPanels.hide(); } return false; }; document.onmousemove = function(ev) { if (md) { dragged = true; panelWidth = windowWidth-(ev.clientX-msoff); if (panelWidth < 0) { panelWidth = 0; infoPanel.style.left = windowWidth+"px"; } else if (panelWidth > windowWidth-21) { panelWidth = windowWidth-21; infoPanel.style.left = '21px'; } else infoPanel.style.left = (ev.clientX-msoff)+"px"; canvas.style.marginRight = panelWidth+20+"px"; return false; } }; document.onmouseup = function() { md = false; }; window.onresize = function() { windowWidth = document.getElementById('sizetest').clientWidth; infoPanel.style.left = windowWidth-panelWidth+'px'; }; }; window.onload = function() { canvas = document.getElementById('v3map'); initDetailPanels(); var scale = 0.25; var coy = canvas.offsetTop; function findDecision(ev) { var x = (ev.clientX+window.pageXOffset)/scale; var y = (ev.clientY+window.pageYOffset-coy)/scale; for (var i = trace.length-1; i >= 0; i--) { if (x >= trace[i].x-19 && x <= trace[i].x+19 && y >= trace[i].y-19 && y <= trace[i].y+19) return trace[i]; } }; var preview = document.getElementById('preview'); var previewId = document.getElementById('previewid'); var previewCalls = document.getElementById('previewcalls'); function previewDecision(dec) { preview.style.left = (dec.x*scale)+'px'; preview.style.top = (dec.y*scale+coy+15)+'px'; preview.style.display = 'block'; previewId.textContent = dec.d; previewCalls.innerHTML = dec.previewCalls; }; function overResponse(ev) { var x = (ev.clientX+window.pageXOffset)/scale; var y = (ev.clientY+window.pageYOffset-coy)/scale; return (x >= response.x-(response.width/2) && x <= response.x+(response.width/2) && y >= response.y-19 && y <= response.y+19); }; decorateTrace(); var bg = new Image(3138, 2184); function drawMap() { var ctx = canvas.getContext("2d"); ctx.save(); ctx.scale(1/scale, 1/scale); ctx.fillStyle = '#ffffff'; ctx.fillRect(0, 0, 3138, 2184); ctx.restore(); ctx.drawImage(bg, 0, 0); drawTrace(); }; bg.onload = function() { canvas.getContext("2d").scale(scale, scale); drawMap(scale); canvas.onmousemove = function(ev) { if (findDecision(ev)) { canvas.style.cursor = 'pointer'; previewDecision(findDecision(ev)); } else { preview.style.display = 'none'; if (overResponse(ev)) canvas.style.cursor = 'pointer'; else canvas.style.cursor = 'default'; } }; canvas.onclick = function(ev) { var dec = findDecision(ev); if (dec) { detailPanels.setDecision(dec); detailPanels.show('decision'); } else if (overResponse(ev)) { detailPanels.show('response'); } }; document.getElementById('zoomin').onclick = function() { scale = scale*2; canvas.getContext("2d").scale(2, 2); drawMap(); }; document.getElementById('zoomout').onclick = function() { scale = scale/2; canvas.getContext("2d").scale(0.5, 0.5); drawMap(); }; }; bg.onerror = function() { alert('Failed to load background image.'); }; bg.src = 'static/map.png'; };