diff --git a/modules/duktape-debugger.js b/modules/duktape-debugger.js index 3c1b481..98568a1 100644 --- a/modules/duktape-debugger.js +++ b/modules/duktape-debugger.js @@ -130,7 +130,7 @@ var events = require('events'); var Promise = require('promise'); -var dbgHTML = '﻿<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html>
<head>
<meta content="text/html; charset=utf-8" http-equiv="Content-Type" />
<meta http-equiv="X-UA-Compatible" content="IE=EDGE" />
<meta name="format-detection" content="telephone=no" />
<title>WebRTC Debug</title>
<style type="text/css">
body 
{
    margin: 0;
	padding: 0;
	border: 0;
	color: black;
	font-size: 13px;
	font-family: "Trebuchet MS", Arial, Helvetica, sans-serif;
	background-color: #d3d9d6;
}
#container {
	background-color: #fff;
	width: 960px;
	margin: 0 auto;
	border-top: 0;
	border-right: 1px solid #b7b7b7;
	border-bottom: 0;
	border-left: 1px solid #b7b7b7;
	padding: 0;
}
#masthead {
	width: auto;
	margin: 0;
	padding: 0;
	overflow: auto;
	text-align: right;
	background-color: #036;
	width: 960px;

    background: rgb(45,86,137); /* Old browsers */
    background: -moz-linear-gradient(left,  rgba(45,86,137,1) 0%, rgba(0,51,102,1) 29%); /* FF3.6+ */
    background: -webkit-gradient(linear, left top, right top, color-stop(0%,rgba(45,86,137,1)), color-stop(29%,rgba(0,51,102,1))); /* Chrome,Safari4+ */
    background: -webkit-linear-gradient(left,  rgba(45,86,137,1) 0%,rgba(0,51,102,1) 29%); /* Chrome10+,Safari5.1+ */
    background: -o-linear-gradient(left,  rgba(45,86,137,1) 0%,rgba(0,51,102,1) 29%); /* Opera 11.10+ */
    background: -ms-linear-gradient(left,  rgba(45,86,137,1) 0%,rgba(0,51,102,1) 29%); /* IE10+ */
    background: linear-gradient(to right,  rgba(45,86,137,1) 0%,rgba(0,51,102,1) 29%); /* W3C */
    filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#2d5689', endColorstr='#003366',GradientType=1 ); /* IE6-9 */
}
#column_l {
	position: relative;
	float: left;
	width: 930px;
	margin: 0;
	padding: 0 15px;
	background-color: #fff;
}
#footer {
	clear: both;
	overflow: auto;
	width: 960px;
	text-align: center;
	background-color: #113962;
	padding-top: 5px;
	padding-bottom: 5px;
}
#footer a {
	color: #fff;
	text-decoration: underline;
}
#footer a:hover {
	color: #fff;
	text-decoration: none;
}
a {
	color: #036;
	text-decoration: underline;
}
.style3 {
	text-align: center;
	color: white;
	background-color: #808080;
	font-weight: bold;
}
.style6 {
    padding-top: 2px;
    padding-bottom: 2px;
	background-color: #C0C0C0;
}
.xsection {
    background-color: #E0E0E0;
    padding: 5px 20px 25px 20px;
    border-radius: 10px;
}
.fixedfont {
    font-family:courier, "courier new", monospace;
}
.infoentry {
    padding-top:1px;
}
.GENERIC_MODULE{
    color:black;
}
.VARQUERY{
    color:black;
}
.VARRESULT{
    color:darkgreen;
}
.BREAKPOINT{
    color:darkviolet
}
.BREAKPOINT:hover 
{
	background-color: #EEE;
}
.STATECHANGE{
    color:darkviolet
}
.SOURCECODEPARENT 
{
    width:10000px
}
.SOURCECODEPARENT:hover 
{
	background-color: #EEE;
}
.SOURCECODE{
    color:darkgreen;
    padding-left:10px;
    display:inline-block;
}
.SOURCECODELINE
{
    width:30px;
    text-align:right;
    color:gray;
    display:inline-block;
    -webkit-user-select:none;
    -khtml-user-select:none;
    -moz-user-select:none;
    -ms-user-select:none;
    -o-user-select:none;
    -user-select:none;
}
.CALLSTACK
{
    color:saddlebrown;
    font-weight:normal;
    -webkit-user-select:none;
    -khtml-user-select:none;
    -moz-user-select:none;
    -ms-user-select:none;
    -o-user-select:none;
    -user-select:none;
}
.CALLSTACK:hover
{
	background-color: #EEE;
}
.ERROR{
    color:red;
}
.UNKNOWN_MODULE{
    color:red;
}
</style>
</head>
<body onload="if (typeof(startup) !== 'undefined') startup();">
	<div id="container">
		<div id="masthead" style="height: 66px; width: 100%; overflow:hidden">
	        <div style="float:left; height: 66px; color:#c8c8c8; padding-left:20px; padding-top:8px"><strong><font style="font-size:46px; font-family: Arial, Helvetica, sans-serif;">Built-in JS Debugger</font></strong></div>
	        <div style="float:left; height: 66px; color:#c8c8c8; padding-left:5px; padding-top:14px"><strong><font style="font-size:14px; font-family: Arial, Helvetica, sans-serif;"><span id="titlehost"></span></font></strong></div>
		</div>
		<div id="topbar">
			<table style="width: 100%; height: 22px;" cellpadding="0" cellspacing="0" class="style1">
				<tr>
					<td id="AttachButton" style="width: 100px; height: 24px; cursor:default;" onclick="attachDebugger()" class="style3">ATTACH</td>
					<td id="PauseResumeButton" style="width: 100px; height: 24px; cursor:default;" onclick="pauseResume()" class="style3">PAUSE</td>
                    <td id="StepOverButton" style="width: 100px; height: 24px; cursor:default;" onclick="stepOver()" class="style3">Step-Over</td>
                    <td id="StepIntoButton" style="width: 100px; height: 24px; cursor:default;" onclick="stepInto()" class="style3">Step-Into</td>
                    <td id="StepOutButton" style="width: 100px; height: 24px; cursor:default;" onclick="stepOut()" class="style3">Step-Out</td>
                    <!--<td id="HeapButton" style="width: 100px; height: 24px; cursor:default;" onclick="onDumpHeap()" class="style3">View Heap</td> -->
				</tr>
			</table>
            <div id="column_l">
                <div id="statustext" align="center" style="height:20px; font-size:15px">
                </div>
            </div>
		</div>
		<div id="page_content">
            <div style="width: 75%; float:left">
                <div id="CallstackWindow" style="width: 100%;height:100px;overflow-y:scroll">
                </div>	
                <div id="LogWindow" style="width: 100%;height:500px;overflow-y:scroll">
                </div>	
            </div>
            <div style="width: 25%; float:right">
                <div id="LogWindow2" style="width: 100%;height:600px;overflow-y:scroll">
                    <textarea id="loadSourceText" disabled style="width:95%;height:30px"></textarea>
                    <input id="loadSourceButton" type="button" value="Load Source" disabled onclick="manuallyLoadSource()" />
                    <p/>
                    <div class='GENERIC_MODULE'>Breakpoints</div>
                    <div id="CurrentBreakpoints" style="width: 95%;height:200px;overflow-y:scroll">
                    </div>
                    <input id="delBreakpointButton" type="button" value="Del Breakpoint" disabled onclick="delBreakpoint()" />
                    <p />
                    <textarea id="queryText" disabled style="width:95%;height:15px"></textarea>
                    <input id="queryButton" type="button" value="Query" disabled onclick="queryVal()" />
                    <input id="evalButton" type="button" value="Eval" disabled onclick="evalString()" />
                    <input id="clearButton" type="button" value="Clear" disabled onclick="clearQuery()" />
                    <input id="localsButton" type="button" value="Locals" disabled onclick="localsQuery()" />
                    <p />
                    <div class='GENERIC_MODULE'>Results</div>
                    <div id="queryResults" style="width: 95%;height:200px;overflow-y:scroll">
                    </div>	

                </div>	
            </div>

            <div id="footer">
                <table cellpadding="0" cellspacing="10" style="width: 100%">
                    <tr>
                        <td id='xfooter' style="text-align:left; display:none"></td>
                        <td style="text-align:center;color:lightgray">Total Memory Allocation: <b id="memusage" style="color:yellow"></b></td>
                        <td><input id="gcButton" type="button" value="GC" onclick="forceGC()" /></td>
                     </tr>
                </table>
            </div>
       </div>

    <script type="text/javascript">        var breakpointID = 0;        var selectedBreakpoint = "";        var wsocket;        var connected = false;        var currentLine = 0;        var currentModule = '';        var currentModuleTokens = null;        var sourceHighlight = 'yellow';        var exceptionHighlight = 'deeppink';        var bpColor = 'blue';        var exceptionMessage = null;        var highlight = sourceHighlight;        var callstack_scope = 'royalblue';        var callstack_default = 'saddlebrown';        var currentCallstack = [];        var isNative = false;        function Q(x) { return document.getElementById(x); } // "Q"
        function QC(x, y) { QS(x)["cursor"] = y;}
        function QS(x) { return Q(x).style; }                // "Q" style
        function QE(x, y) { Q(x).disabled = !y; }            // "Q" enable
        function QV(x, y) { QS(x).display = (y ? '' : 'none'); } // "Q" visible
        function QA(x, y) { Q(x).innerHTML += y; }           // "Q" append
        function QH(x, y) { Q(x).innerHTML = y; }            // "Q" html
        function QHVAL(x) { return (Q(x).innerHTML); }
        function QVAL(x, y) { if (y == undefined) { return (Q(x).value); } else { Q(x).value = y; } }
        function QBC(x, y) { QS(x)["background-color"] = y; }
        function QCHILDREN(x) { return (Q(x).getElementsByTagName('div')); }        function Int32ToStr(v) { return String.fromCharCode((v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF); }
        function Int16ToStr(v) { return String.fromCharCode((v >> 8) & 0xFF, v & 0xFF); }
        function ReadShort(data, ptr) { return (data.charCodeAt(ptr) << 8) + data.charCodeAt(ptr + 1); }
        function ReadInt(data, ptr) { return (data.charCodeAt(ptr) * 16777216) + (data.charCodeAt(ptr + 1) * 65536) + (data.charCodeAt(ptr + 2) * 256) + data.charCodeAt(ptr + 3); }
        function ReadLong(data, ptr) { return ((data.charCodeAt(ptr) * 72057594037927936) + (data.charCodeAt(ptr + 1) * 281474976710656) + (data.charCodeAt(ptr + 2) * 1099511627776) + (data.charCodeAt(ptr + 3) * 4294967296) + data.charCodeAt(ptr + 4) * 16777216) + (data.charCodeAt(ptr + 5) * 65536) + (data.charCodeAt(ptr + 6) * 256) + data.charCodeAt(ptr + 7); }
        function SendCommand(cmd, data)
        {
            wsocket.send(Int16ToStr(cmd) + data);
        }        function AddLog(code, msg)
        {
            var dclass = "";
            if (msg == undefined) {
                QA("LogWindow", "<div class='GENERIC_MODULE' > " + code + "</br></div>");
                return;
            }

            dclass = code;

            QA("LogWindow", "<div class='" + dclass + "' > " + msg + "</br></div>");
        }        function startup()
        {
            displayStatus("Establishing connection...");          
            wsocket = new WebSocket("ws://" + window.location.hostname + ":" + window.location.port + "/");
            wsocket.binaryType = "arraybuffer";
            wsocket.onopen = function (evt)
            {
                connected = true; displayStatus("Debug Client Connected...");
                toggleColor(QS("AttachButton"), 1);
                attachDebugger();
            }
            wsocket.onclose = function (evt)
            {
                connected = false; displayStatus("Debug Client Connection lost... Retrying...");
                toggleColor(QS("AttachButton"), 0);
                toggleColor(QS("StepOverButton"), 0);
                toggleColor(QS("StepIntoButton"), 0);
                toggleColor(QS("StepOutButton"), 0);
                toggleColor(QS("PauseResumeButton"), 0);
                QE("loadSourceButton", 0);
                QE("loadSourceText", 0);
                QE('queryButton', 0);
                QE('clearButton', 0);
                QE('localsButton', 0);
                QE('evalButton', 0);
                QV('page_content', false);
                setTimeout(serverPoll, 1000);
                currentModule = '';
            }
            wsocket.onmessage = function (evt)
            {
                var msg = JSON.parse(evt.data);

                switch(msg.cmd)
                {
                    case 'MEMORY':
                        QH('memusage', Math.round(msg.total/1024) + ' kb');
                        break;
                    case 'ATTACH':
                        displayStatus('Attached...');
                        QH("AttachButton", "DETACH");
                        toggleColor(QS("PauseResumeButton"), 1);
                        QE("loadSourceButton", 1);
                        QE("loadSourceText", 1);
                        QV('page_content', true);
                        break;
                    case 'PAUSE':
                        isNative = msg.native;

                        if (highlight == exceptionHighlight)
                        {
                            QH('statustext', '<div class="STATECHANGE">' + exceptionMessage + ' in ' + msg.file + ':' + msg.line + '</div>');
                        }
                        else
                        {
                            if (msg.native)
                            {
                                QH('statustext', '<div class="STATECHANGE">PAUSED in [NATIVE CODE]</div>');
                                QH('LogWindow', '');
                            }
                            else
                            {
                                QH('statustext', '<div class="STATECHANGE">PAUSED in ' + msg.file + ':' + msg.line + '</div>');
                            }
                        }

                        toggleColor(QS("StepOverButton"), 1);
                        toggleColor(QS("StepIntoButton"), 1);
                        toggleColor(QS("StepOutButton"), 1);
                        QH("PauseResumeButton", "RESUME");
                        QE('queryText', 1);
                        QE('queryButton', 1);
                        QE('clearButton', 1);
                        QE('localsButton', 1);
                        QE('evalButton', 1);

                        if (msg.file == currentModule)
                        {
                            QBC(sourceLineDiv(msg.line), highlight);
                            currentLine = msg.line;
                            Q(sourceLineDiv(currentLine)).scrollIntoView({block: 'center'});
                        }
                        else
                        {
                            currentLine = msg.line;
                            var jj = { cmd: 'SOURCE', name: msg.file };
                            wsocket.send(JSON.stringify(jj));
                        }
                        break;
                    case 'RUNNING':
                        QH('statustext', '<div class="STATECHANGE">RUNNING ' + msg.file + '</div>');
                        QH("PauseResumeButton", "PAUSE");
                        toggleColor(QS("StepOverButton"), 0);
                        toggleColor(QS("StepIntoButton"), 0);
                        toggleColor(QS("StepOutButton"), 0);
                        QE('queryText', 0);
                        QE('queryButton', 0);
                        QE('clearButton', 0);
                        QE('localsButton', 0);
                        QE('evalButton', 0);


                        break;
                    case 'THROW':
                        highlight = exceptionHighlight;
                        exceptionMessage = '** ' + msg.msg + ' **';
                        break;
                    case 'DETACH':
                        displayStatus("Debugger Detached...");
                        toggleColor(QS("PauseResumeButton"), 0);
                        QH("AttachButton", "ATTACH");

                        toggleColor(QS("StepOverButton"), 0);
                        toggleColor(QS("StepIntoButton"), 0);
                        toggleColor(QS("StepOutButton"), 0);
                        QE("loadSourceButton", 0);
                        QE("loadSourceText", 0);
                        QE('queryText', 0);
                        QE('queryButton', 0);
                        currentModule = '';
                        break;
                    case 'QUERY':
                        QA("queryResults", '<div class="VARQUERY">' + msg.var + '=</div>');
                        QA("queryResults", '<div class="VARRESULT">   ' + msg.val + '</div>');
                        break;
                    case 'EVAL':
                        QA("queryResults", '<div class="VARQUERY">' + msg.eval + '=</div>');
                        QA("queryResults", '<div class="VARRESULT">   ' + msg.val + '</div>');
                        break;
                    case 'SOURCE':
                        currentModule = msg.name;
                        currentModuleTokens = msg.source.split('\r').join('').split(' ').join('&nbsp;').split('\t').join('&nbsp;&nbsp;&nbsp;').split('\n');
                        QH('LogWindow', '');
                        var lines = [];
                        for (var i = 0; i < currentModuleTokens.length;++i)
                        {
                            lines.push('<div class="SOURCECODEPARENT" ondblclick="onToggleBreakpoint(' + (i+1) + ')">');
                            lines.push('<div class="SOURCECODELINE" id="L' + sourceLineDiv(i + 1) + '">' + (i + 1) + '</div>');
                            lines.push('<div class="SOURCECODE" id="' + sourceLineDiv(i + 1) + '"><code style="white-space:normal">' + currentModuleTokens[i] + '</code></div>');
                            lines.push('</div>');
                        }

                        var cline = getSourceLine();

                        QA('LogWindow', lines.join(''));

                        if (cline >= 0)
                        {
                            QBC(sourceLineDiv(cline), highlight);
                            Q(sourceLineDiv(cline)).scrollIntoView({ block: 'center' });
                            Q('L' + sourceLineDiv(cline)).scrollIntoView({ inline: 'start' });
                        }

                        var divs = QCHILDREN("CurrentBreakpoints");
                        for (var i in divs)
                        {
                            if(currentModule.toLowerCase().endsWith('.js') || currentModule == '<<NATIVE>>')
                            {
                                if(divs[i].id.startsWith('_bp__'))
                                {
                                    QBC('L' + sourceLineDiv(divs[i].id.slice(5)), bpColor);
                                }
                            }
                            else
                            {
                                if (divs[i].id.startsWith('_bp_' + currentModule + '_'))
                                {
                                    QBC('L' + sourceLineDiv(divs[i].id.slice(5 + currentModule.length)), bpColor);
                                }
                            }
                        }
                        break;
                    case 'BREAKPOINT':
                        var items = [];
                        for (var i in msg.list)
                        {
                            var fn = msg.list[i].fileName;
                            var id;
                            if(fn.toLowerCase().endsWith('.js'))
                            {
                                id = '_bp__' + msg.list[i].lineNumber;
                                items.push('<div id="' + id + '" class="BREAKPOINT" onclick="onBreakpointClick(\'' + id + '\')">:' + msg.list[i].lineNumber + '</div>');
                            }
                            else
                            {
                                id = '_bp_' + msg.list[i].fileName + '_' + msg.list[i].lineNumber;
                                items.push('<div id="' + id + '" class="BREAKPOINT" onclick="onBreakpointClick(\'' + id + '\')">' + msg.list[i].fileName + ':' + msg.list[i].lineNumber + '</div>');
                            }
                        }
                        QH("CurrentBreakpoints", items.join(''));
                        break;
                    case 'CALLSTACK':
                        currentCallstack = msg.callstack;
                        displayCallstack();
                        break
                    case 'LOCALS':
                        for (var v in msg.val)
                        {
                            if (msg.val[v].type == 'undefined')
                            {
                                QA("queryResults", '<div class="VARQUERY">' + msg.val[v].key + '=</div>');
                            }
                            else
                            {
                                wsocket.send('{"cmd": "QUERY", "var": "' + msg.val[v].key + '", "level": ' + getCallstackLevel() + '}');
                            }
                        }
                        break;
                }
            }
        }        function getSourceLine()        {            for(var i = 0; i < currentCallstack.length; ++i)
            {
                if (QS('_stack_' + i).color == callstack_scope && currentCallstack[i].fileName == currentModule)
                {
                    return (currentCallstack[i].lineNumber);
                }
            }
            return (-1);
        }        function toggleColor(istyle, state)
        {
            istyle["background-color"] = state ? 'blue' : 'grey';
            istyle["cursor"] = state ? 'pointer' : 'default';
        }        function attachDebugger()
        {
            switch(Q("AttachButton").innerHTML)
            {
                case 'ATTACH':
                    displayStatus("Attaching debugger...");
                    wsocket.send('{"cmd": "ATTACH"}');
                    break;
                case 'DETACH':
                    currentCallstack = [];
                    displayCallstack();

                    displayStatus("Detaching debugger...");
                    currentModule = '';
                    wsocket.send('{"cmd": "DETACH"}');
                    QV('page_content', false);
                    break;
            }
        }        function pauseResume()
        {
            switch(Q("PauseResumeButton").innerHTML)
            {
                case 'PAUSE':
                    wsocket.send('{"cmd": "PAUSE"}');
                    break;
                case 'RESUME':
                    highlight = sourceHighlight;
                    wsocket.send('{"cmd": "RESUME"}');
                    QBC(sourceLineDiv(currentLine), 'white');

                    currentCallstack = [];
                    displayCallstack();
                    break;
            }
        }        function stepOver()
        {
            highlight = sourceHighlight;
            QBC(sourceLineDiv(currentLine), 'white');
            wsocket.send('{"cmd": "STEPOVER"}');
        }        function stepInto()        {
            highlight = sourceHighlight;
            QBC(sourceLineDiv(currentLine), 'white');
            wsocket.send('{"cmd": "STEPINTO"}');
        }        function stepOut()
        {
            highlight = sourceHighlight;
            QBC(sourceLineDiv(currentLine), 'white');
            wsocket.send('{"cmd": "STEPOUT"}');
        }                function delBreakpoint()        {
            QV(selectedBreakpoint, 0);
            selectedBreakpoint = "";
            QE("delBreakpointButton", 0);
        }        function onBreakpointClick(id)        {
            if (selectedBreakpoint == id)
            {
                QS(id)["background-color"] = 'white';
                selectedBreakpoint = "";
                QE("delBreakpointButton", 0);
            }
            else
            {
                if (selectedBreakpoint != "")
                {
                    QS(selectedBreakpoint)["background-color"] = 'white';
                }
                QS(id)["background-color"] = 'yellow';
                selectedBreakpoint = id;
                QE("delBreakpointButton", 1);
            }

            for (var i = 0; i < currentCallstack.length; ++i)
            {
                QS('_stack_' + i).color = callstack_default;
            }
            var bp = QHVAL(id).split(':')[0];
            var jj = { cmd: 'SOURCE', name: bp == '' ? '<<NATIVE>>' : bp };
            wsocket.send(JSON.stringify(jj));        
        }        function getCallstackLevel()        {
            // Determine the Callstack Level
            var level = -1;
            for (var i = 0; i < currentCallstack.length; ++i) {
                if (QS('_stack_' + i).color == callstack_scope) {
                    level = 0 - (i + 1);
                    break;
                }
            }
            return (level);
        }        function queryVal()        {
            //QA("queryResults", '<div class="VARQUERY">' + QVAL('queryText') + '=</div>');
            wsocket.send('{"cmd": "QUERY", "var": "' + QVAL('queryText') + '", "level": ' + getCallstackLevel() + '}');
            QVAL('queryText', "");
        }        function evalString()        {
            wsocket.send('{"cmd": "EVAL", "eval": "' + QVAL('queryText') + '", "level": ' + getCallstackLevel() + '}');
            QVAL('queryText', "");
        }        function localsQuery()        {
            QA("queryResults", '<div class="VARQUERY">Locals:&nbsp;</div>');
            wsocket.send('{"cmd": "LOCALS", "level": ' + getCallstackLevel() + '}');
        }        function clearQuery()        {
            QH('queryResults', '');
        }        function displayStatus(msg, msgClass)        {
            QH('statustext', '<div class="' + (msgClass?msgClass:'GENERIC_MODULE') + '">' + msg + '</div>');
        }        function sourceLineDiv(line)        {
            return ('__src' + line);
        }        function onToggleBreakpoint(line)        {
            QBC('L' + sourceLineDiv(line), bpColor);
            var fn = '';

            for (var i = 0; i < currentCallstack.length; ++i)
            {
                if (QS('_stack_' + i).color == callstack_scope)
                {
                    fn = currentCallstack[i].fileName.split('\\').join('\\\\');
                }
            }

            wsocket.send('{"cmd": "BREAKPOINT", "file":"' + fn + '", "line":' + line + ', "mode":"add"}');
        }        function onDumpHeap()        {
            wsocket.send('{"cmd": "HEAP"}');
        }        function forceGC()        {
            wsocket.send('{"cmd": "GC"}');
        }        function displayCallstack()        {
            var display = [];
            var indent = '-->&nbsp;';
            for (var i = 0; i < currentCallstack.length; ++i)
            {
                if (currentCallstack[i].funcName == '') { currentCallstack[i].funcName = '(anonymous)'; }
                var fname = currentCallstack[i].fileName.split('\\').join('\\\\');

                display.push('<div depth="' + (i+1) + '" id="_stack_' + i + '" class="CALLSTACK" onclick="onCallstackClick(\'' + currentCallstack.length + '\', \'_stack_' + i + '\', \'' + fname + '\')">' + indent + currentCallstack[i].funcName + ':' + currentCallstack[i].lineNumber + '</div>');
                indent = '&nbsp;&nbsp;&nbsp;' + indent;
            }
            QH('CallstackWindow', display.join(''));

            if (currentCallstack.length > 0)
            {
                QS('_stack_0').color = callstack_scope;
            }
        }        function onCallstackClick(len, id, filename)        {
            for (var i = 0; i < len; ++i)
            {
                QS('_stack_' + i).color = callstack_default;
            }
            QS(id).color = callstack_scope;
            var jj = { cmd: 'SOURCE', name: filename };
            wsocket.send(JSON.stringify(jj));
        }        function manuallyLoadSource()        {
            for (var i = 0; i < currentCallstack.length; ++i)
            {
                QS('_stack_' + i).color = callstack_default;
            }
            QH('LogWindow', '');

            var jj = { cmd: 'SOURCE', name: QVAL('loadSourceText') == ''?'<<NATIVE>>':QVAL('loadSourceText') };
            wsocket.send(JSON.stringify(jj));
        }        // Poll the server, if it responds, refresh the page.
        function serverPoll()
        {
            xdr = null;
            try { xdr = new XDomainRequest(); } catch (e) { }
            if (!xdr) xdr = new XMLHttpRequest();
            xdr.open("HEAD", window.location.href);
            xdr.timeout = 15000;
            xdr.onload = function () { reload(); };
            xdr.onerror = xdr.ontimeout = function () { setTimeout(serverPoll, 2000); };
            xdr.send();
        }

        function reload() { window.location.href = window.location.href; }
    </script>
</body>
</html>
'; +var dbgHTML = '﻿<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html>
<head>
<meta content="text/html; charset=utf-8" http-equiv="Content-Type" />
<meta http-equiv="X-UA-Compatible" content="IE=EDGE" />
<meta name="format-detection" content="telephone=no" />
<title>WebRTC Debug</title>
<style type="text/css">
body 
{
    margin: 0;
	padding: 0;
	border: 0;
	color: black;
	font-size: 13px;
	font-family: "Trebuchet MS", Arial, Helvetica, sans-serif;
	background-color: #d3d9d6;
}
#container {
	background-color: #fff;
	width: 960px;
	margin: 0 auto;
	border-top: 0;
	border-right: 1px solid #b7b7b7;
	border-bottom: 0;
	border-left: 1px solid #b7b7b7;
	padding: 0;
}
#masthead {
	width: auto;
	margin: 0;
	padding: 0;
	overflow: auto;
	text-align: right;
	background-color: #036;
	width: 960px;

    background: rgb(45,86,137); /* Old browsers */
    background: -moz-linear-gradient(left,  rgba(45,86,137,1) 0%, rgba(0,51,102,1) 29%); /* FF3.6+ */
    background: -webkit-gradient(linear, left top, right top, color-stop(0%,rgba(45,86,137,1)), color-stop(29%,rgba(0,51,102,1))); /* Chrome,Safari4+ */
    background: -webkit-linear-gradient(left,  rgba(45,86,137,1) 0%,rgba(0,51,102,1) 29%); /* Chrome10+,Safari5.1+ */
    background: -o-linear-gradient(left,  rgba(45,86,137,1) 0%,rgba(0,51,102,1) 29%); /* Opera 11.10+ */
    background: -ms-linear-gradient(left,  rgba(45,86,137,1) 0%,rgba(0,51,102,1) 29%); /* IE10+ */
    background: linear-gradient(to right,  rgba(45,86,137,1) 0%,rgba(0,51,102,1) 29%); /* W3C */
    filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#2d5689', endColorstr='#003366',GradientType=1 ); /* IE6-9 */
}
#column_l {
	position: relative;
	float: left;
	width: 930px;
	margin: 0;
	padding: 0 15px;
	background-color: #fff;
}
#footer {
	clear: both;
	overflow: auto;
	width: 960px;
	text-align: center;
	background-color: #113962;
	padding-top: 5px;
	padding-bottom: 5px;
}
#footer a {
	color: #fff;
	text-decoration: underline;
}
#footer a:hover {
	color: #fff;
	text-decoration: none;
}
a {
	color: #036;
	text-decoration: underline;
}
.style3 {
	text-align: center;
	color: white;
	background-color: #808080;
	font-weight: bold;
}
.style6 {
    padding-top: 2px;
    padding-bottom: 2px;
	background-color: #C0C0C0;
}
.xsection {
    background-color: #E0E0E0;
    padding: 5px 20px 25px 20px;
    border-radius: 10px;
}
.fixedfont {
    font-family:courier, "courier new", monospace;
}
.infoentry {
    padding-top:1px;
}
.GENERIC_MODULE{
    color:black;
}
.VARQUERY{
    color:black;
}
.VARRESULT{
    color:darkgreen;
}
.BREAKPOINT{
    color:darkviolet
}
.BREAKPOINT:hover 
{
	background-color: #EEE;
}
.STATECHANGE{
    color:darkviolet
}
.SOURCECODEPARENT 
{
    width:10000px
}
.SOURCECODEPARENT:hover 
{
	background-color: #EEE;
}
.SOURCECODE{
    color:darkgreen;
    padding-left:10px;
    display:inline-block;
}
.SOURCECODELINE
{
    width:30px;
    text-align:right;
    color:gray;
    display:inline-block;
    -webkit-user-select:none;
    -khtml-user-select:none;
    -moz-user-select:none;
    -ms-user-select:none;
    -o-user-select:none;
    -user-select:none;
}
.CALLSTACK
{
    color:saddlebrown;
    font-weight:normal;
    -webkit-user-select:none;
    -khtml-user-select:none;
    -moz-user-select:none;
    -ms-user-select:none;
    -o-user-select:none;
    -user-select:none;
}
.CALLSTACK:hover
{
	background-color: #EEE;
}
.ERROR{
    color:red;
}
.UNKNOWN_MODULE{
    color:red;
}
</style>
</head>
<body onload="if (typeof(startup) !== 'undefined') startup();">
	<div id="container">
		<div id="masthead" style="height: 66px; width: 100%; overflow:hidden">
	        <div style="float:left; height: 66px; color:#c8c8c8; padding-left:20px; padding-top:8px"><strong><font style="font-size:46px; font-family: Arial, Helvetica, sans-serif;">Built-in JS Debugger</font></strong></div>
	        <div style="float:left; height: 66px; color:#c8c8c8; padding-left:5px; padding-top:14px"><strong><font style="font-size:14px; font-family: Arial, Helvetica, sans-serif;"><span id="titlehost"></span></font></strong></div>
		</div>
		<div id="topbar">
			<table style="width: 100%; height: 22px;" cellpadding="0" cellspacing="0" class="style1">
				<tr>
					<td id="AttachButton" style="width: 100px; height: 24px; cursor:default;" onclick="attachDebugger()" class="style3">ATTACH</td>
					<td id="PauseResumeButton" style="width: 100px; height: 24px; cursor:default;" onclick="pauseResume()" class="style3">PAUSE</td>
                    <td id="StepOverButton" style="width: 100px; height: 24px; cursor:default;" onclick="stepOver()" class="style3">Step-Over</td>
                    <td id="StepIntoButton" style="width: 100px; height: 24px; cursor:default;" onclick="stepInto()" class="style3">Step-Into</td>
                    <td id="StepOutButton" style="width: 100px; height: 24px; cursor:default;" onclick="stepOut()" class="style3">Step-Out</td>
                    <!--<td id="HeapButton" style="width: 100px; height: 24px; cursor:default;" onclick="onDumpHeap()" class="style3">View Heap</td> -->
				</tr>
			</table>
            <div id="column_l">
                <div id="statustext" align="center" style="height:20px; font-size:15px">
                </div>
            </div>
		</div>
		<div id="page_content">
            <div style="width: 75%; float:left">
                <div id="CallstackWindow" style="width: 100%;height:100px;overflow-y:scroll">
                </div>	
                <div id="LogWindow" style="width: 100%;height:500px;overflow-y:scroll">
                </div>	
            </div>
            <div style="width: 25%; float:right">
                <div id="LogWindow2" style="width: 100%;height:600px;overflow-y:scroll">
                    <textarea id="loadSourceText" disabled style="width:95%;height:30px"></textarea>
                    <input id="loadSourceButton" type="button" value="Load Source" disabled onclick="manuallyLoadSource()" />
                    <p/>
                    <div class='GENERIC_MODULE'>Breakpoints</div>
                    <div id="CurrentBreakpoints" style="width: 95%;height:200px;overflow-y:scroll">
                    </div>
                    <input id="delBreakpointButton" type="button" value="Del Breakpoint" disabled onclick="delBreakpoint()" />
                    <p />
                    <textarea id="queryText" disabled style="width:95%;height:15px"></textarea>
                    <input id="queryButton" type="button" value="Query" disabled onclick="queryVal()" />
                    <input id="evalButton" type="button" value="Eval" disabled onclick="evalString()" />
                    <input id="clearButton" type="button" value="Clear" disabled onclick="clearQuery()" />
                    <input id="localsButton" type="button" value="Locals" disabled onclick="localsQuery()" />
                    <p />
                    <div class='GENERIC_MODULE'>Results</div>
                    <div id="queryResults" style="width: 95%;height:200px;overflow-y:scroll">
                    </div>	

                </div>	
            </div>

            <div id="footer">
                <table cellpadding="0" cellspacing="10" style="width: 100%">
                    <tr>
                        <td id='xfooter' style="text-align:left; display:none"></td>
                        <td style="text-align:center;color:lightgray">Total Memory Allocation: <b id="memusage" style="color:yellow"></b></td>
                        <td><input id="gcButton" type="button" value="GC" onclick="forceGC()" /></td>
                     </tr>
                </table>
            </div>
       </div>

    <script type="text/javascript">        var breakpointID = 0;        var selectedBreakpoint = "";        var wsocket;        var connected = false;        var currentLine = 0;        var currentModule = '';        var currentModuleTokens = null;        var sourceHighlight = 'yellow';        var exceptionHighlight = 'deeppink';        var bpColor = 'blue';        var exceptionMessage = null;        var highlight = sourceHighlight;        var callstack_scope = 'royalblue';        var callstack_default = 'saddlebrown';        var currentCallstack = [];        var isNative = false;        function Q(x) { return document.getElementById(x); } // "Q"
        function QC(x, y) { QS(x)["cursor"] = y;}
        function QS(x) { return Q(x).style; }                // "Q" style
        function QE(x, y) { Q(x).disabled = !y; }            // "Q" enable
        function QV(x, y) { QS(x).display = (y ? '' : 'none'); } // "Q" visible
        function QA(x, y) { Q(x).innerHTML += y; }           // "Q" append
        function QH(x, y) { Q(x).innerHTML = y; }            // "Q" html
        function QHVAL(x) { return (Q(x).innerHTML); }
        function QVAL(x, y) { if (y == undefined) { return (Q(x).value); } else { Q(x).value = y; } }
        function QBC(x, y) { QS(x)["background-color"] = y; }
        function QCHILDREN(x) { return (Q(x).getElementsByTagName('div')); }        function Int32ToStr(v) { return String.fromCharCode((v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF); }
        function Int16ToStr(v) { return String.fromCharCode((v >> 8) & 0xFF, v & 0xFF); }
        function ReadShort(data, ptr) { return (data.charCodeAt(ptr) << 8) + data.charCodeAt(ptr + 1); }
        function ReadInt(data, ptr) { return (data.charCodeAt(ptr) * 16777216) + (data.charCodeAt(ptr + 1) * 65536) + (data.charCodeAt(ptr + 2) * 256) + data.charCodeAt(ptr + 3); }
        function ReadLong(data, ptr) { return ((data.charCodeAt(ptr) * 72057594037927936) + (data.charCodeAt(ptr + 1) * 281474976710656) + (data.charCodeAt(ptr + 2) * 1099511627776) + (data.charCodeAt(ptr + 3) * 4294967296) + data.charCodeAt(ptr + 4) * 16777216) + (data.charCodeAt(ptr + 5) * 65536) + (data.charCodeAt(ptr + 6) * 256) + data.charCodeAt(ptr + 7); }
        function SendCommand(cmd, data)
        {
            wsocket.send(Int16ToStr(cmd) + data);
        }        function AddLog(code, msg)
        {
            var dclass = "";
            if (msg == undefined) {
                QA("LogWindow", "<div class='GENERIC_MODULE' > " + code + "</br></div>");
                return;
            }

            dclass = code;

            QA("LogWindow", "<div class='" + dclass + "' > " + msg + "</br></div>");
        }        function startup()
        {
            displayStatus("Establishing connection...");          
            wsocket = new WebSocket("ws://" + window.location.hostname + ":" + window.location.port + "/");
            wsocket.binaryType = "arraybuffer";
            wsocket.onopen = function (evt)
            {
                connected = true; displayStatus("Debug Client Connected...");
                toggleColor(QS("AttachButton"), 1);
                attachDebugger();
            }
            wsocket.onclose = function (evt)
            {
                connected = false; displayStatus("Debug Client Connection lost... Retrying...");
                toggleColor(QS("AttachButton"), 0);
                toggleColor(QS("StepOverButton"), 0);
                toggleColor(QS("StepIntoButton"), 0);
                toggleColor(QS("StepOutButton"), 0);
                toggleColor(QS("PauseResumeButton"), 0);
                QE("loadSourceButton", 0);
                QE("loadSourceText", 0);
                QE('queryButton', 0);
                QE('clearButton', 0);
                QE('localsButton', 0);
                QE('evalButton', 0);
                QV('page_content', false);
                setTimeout(serverPoll, 1000);
                currentModule = '';
            }
            wsocket.onmessage = function (evt)
            {
                var msg = JSON.parse(evt.data);

                switch(msg.cmd)
                {
                    case 'MEMORY':
                        QH('memusage', Math.round(msg.total/1024) + ' kb');
                        break;
                    case 'ATTACH':
                        displayStatus('Attached...');
                        QH("AttachButton", "DETACH");
                        toggleColor(QS("PauseResumeButton"), 1);
                        QE("loadSourceButton", 1);
                        QE("loadSourceText", 1);
                        QV('page_content', true);
                        break;
                    case 'PAUSE':
                        isNative = msg.native;

                        if (highlight == exceptionHighlight)
                        {
                            QH('statustext', '<div class="STATECHANGE">' + exceptionMessage + ' in ' + msg.file + ':' + msg.line + '</div>');
                        }
                        else
                        {
                            if (msg.native)
                            {
                                QH('statustext', '<div class="STATECHANGE">PAUSED in [NATIVE CODE]</div>');
                                QH('LogWindow', '');
                            }
                            else
                            {
                                QH('statustext', '<div class="STATECHANGE">PAUSED in ' + msg.file + ':' + msg.line + '</div>');
                            }
                        }

                        toggleColor(QS("StepOverButton"), 1);
                        toggleColor(QS("StepIntoButton"), 1);
                        toggleColor(QS("StepOutButton"), 1);
                        QH("PauseResumeButton", "RESUME");
                        QE('queryText', 1);
                        QE('queryButton', 1);
                        QE('clearButton', 1);
                        QE('localsButton', 1);
                        QE('evalButton', 1);

                        if (msg.file == currentModule)
                        {
                            QBC(sourceLineDiv(msg.line), highlight);
                            currentLine = msg.line;
                            Q(sourceLineDiv(currentLine)).scrollIntoView({block: 'center'});
                        }
                        else
                        {
                            currentLine = msg.line;
                            var jj = { cmd: 'SOURCE', name: msg.file };
                            wsocket.send(JSON.stringify(jj));
                        }
                        break;
                    case 'RUNNING':
                        QH('statustext', '<div class="STATECHANGE">RUNNING ' + msg.file + '</div>');
                        QH("PauseResumeButton", "PAUSE");
                        toggleColor(QS("StepOverButton"), 0);
                        toggleColor(QS("StepIntoButton"), 0);
                        toggleColor(QS("StepOutButton"), 0);
                        QE('queryText', 0);
                        QE('queryButton', 0);
                        QE('clearButton', 0);
                        QE('localsButton', 0);
                        QE('evalButton', 0);


                        break;
                    case 'THROW':
                        highlight = exceptionHighlight;
                        exceptionMessage = '** ' + msg.msg + ' **';
                        break;
                    case 'DETACH':
                        displayStatus("Debugger Detached...");
                        toggleColor(QS("PauseResumeButton"), 0);
                        QH("AttachButton", "ATTACH");

                        toggleColor(QS("StepOverButton"), 0);
                        toggleColor(QS("StepIntoButton"), 0);
                        toggleColor(QS("StepOutButton"), 0);
                        QE("loadSourceButton", 0);
                        QE("loadSourceText", 0);
                        QE('queryText', 0);
                        QE('queryButton', 0);
                        currentModule = '';
                        break;
                    case 'QUERY':
                        QA("queryResults", '<div class="VARQUERY">' + msg.var + '=</div>');
                        QA("queryResults", '<div class="VARRESULT">   ' + msg.val + '</div>');
                        break;
                    case 'EVAL':
                        QA("queryResults", '<div class="VARQUERY">' + msg.eval + '=</div>');
                        QA("queryResults", '<div class="VARRESULT">   ' + msg.val + '</div>');
                        break;
                    case 'SOURCE':
                        currentModule = msg.name;
                        currentModuleTokens = msg.source.split('\r').join('').split(' ').join('&nbsp;').split('\t').join('&nbsp;&nbsp;&nbsp;').split('\n');
                        QH('LogWindow', '');
                        var lines = [];
                        for (var i = 0; i < currentModuleTokens.length;++i)
                        {
                            lines.push('<div class="SOURCECODEPARENT" ondblclick="onToggleBreakpoint(' + (i+1) + ')">');
                            lines.push('<div class="SOURCECODELINE" id="L' + sourceLineDiv(i + 1) + '">' + (i + 1) + '</div>');
                            lines.push('<div class="SOURCECODE" id="' + sourceLineDiv(i + 1) + '"><code style="white-space:normal">' + currentModuleTokens[i] + '</code></div>');
                            lines.push('</div>');
                        }

                        var cline = getSourceLine();

                        QA('LogWindow', lines.join(''));

                        if (cline >= 0)
                        {
                            QBC(sourceLineDiv(cline), highlight);
                            Q(sourceLineDiv(cline)).scrollIntoView({ block: 'center' });
                            Q('L' + sourceLineDiv(cline)).scrollIntoView({ inline: 'start' });
                        }

                        var divs = QCHILDREN("CurrentBreakpoints");
                        for (var i in divs)
                        {
                            if(currentModule.toLowerCase().endsWith('.js') || currentModule == '<<NATIVE>>')
                            {
                                if(divs[i].id.startsWith('_bp__'))
                                {
                                    QBC('L' + sourceLineDiv(divs[i].id.slice(5)), bpColor);
                                }
                            }
                            else
                            {
                                if (divs[i].id.startsWith('_bp_' + currentModule + '_'))
                                {
                                    QBC('L' + sourceLineDiv(divs[i].id.slice(5 + currentModule.length)), bpColor);
                                }
                            }
                        }
                        break;
                    case 'BREAKPOINT':
                        var items = [];
                        for (var i in msg.list)
                        {
                            var fn = msg.list[i].fileName;
                            var id;
                            if(fn.toLowerCase().endsWith('.js'))
                            {
                                id = '_bp__' + msg.list[i].lineNumber;
                                items.push('<div id="' + id + '" class="BREAKPOINT" onclick="onBreakpointClick(\'' + id + '\')">:' + msg.list[i].lineNumber + '</div>');
                            }
                            else
                            {
                                id = '_bp_' + msg.list[i].fileName + '_' + msg.list[i].lineNumber;
                                items.push('<div id="' + id + '" class="BREAKPOINT" onclick="onBreakpointClick(\'' + id + '\')">' + msg.list[i].fileName + ':' + msg.list[i].lineNumber + '</div>');
                            }
                        }
                        QH("CurrentBreakpoints", items.join(''));
                        break;
                    case 'CALLSTACK':
                        currentCallstack = msg.callstack;
                        displayCallstack();
                        break
                    case 'LOCALS':
                        for (var v in msg.val)
                        {
                            if (msg.val[v].type == 'undefined')
                            {
                                QA("queryResults", '<div class="VARQUERY">' + msg.val[v].key + '=</div>');
                            }
                            else
                            {
                                wsocket.send('{"cmd": "QUERY", "var": "' + msg.val[v].key + '", "level": ' + getCallstackLevel() + '}');
                            }
                        }
                        break;
                }
            }
        }        function getSourceLine()        {            for(var i = 0; i < currentCallstack.length; ++i)
            {
                if (QS('_stack_' + i).color == callstack_scope && currentCallstack[i].fileName == currentModule)
                {
                    return (currentCallstack[i].lineNumber);
                }
            }
            return (-1);
        }        function toggleColor(istyle, state)
        {
            istyle["background-color"] = state ? 'blue' : 'grey';
            istyle["cursor"] = state ? 'pointer' : 'default';
        }        function attachDebugger()
        {
            switch(Q("AttachButton").innerHTML)
            {
                case 'ATTACH':
                    displayStatus("Attaching debugger...");
                    wsocket.send('{"cmd": "ATTACH"}');
                    break;
                case 'DETACH':
                    currentCallstack = [];
                    displayCallstack();

                    displayStatus("Detaching debugger...");
                    currentModule = '';
                    wsocket.send('{"cmd": "DETACH"}');
                    QV('page_content', false);
                    break;
            }
        }        function pauseResume()
        {
            switch(Q("PauseResumeButton").innerHTML)
            {
                case 'PAUSE':
                    wsocket.send('{"cmd": "PAUSE"}');
                    break;
                case 'RESUME':
                    highlight = sourceHighlight;
                    wsocket.send('{"cmd": "RESUME"}');
                    QBC(sourceLineDiv(currentLine), 'white');

                    currentCallstack = [];
                    displayCallstack();
                    break;
            }
        }        function stepOver()
        {
            highlight = sourceHighlight;
            QBC(sourceLineDiv(currentLine), 'white');
            wsocket.send('{"cmd": "STEPOVER"}');
        }        function stepInto()        {
            highlight = sourceHighlight;
            QBC(sourceLineDiv(currentLine), 'white');
            wsocket.send('{"cmd": "STEPINTO"}');
        }        function stepOut()
        {
            highlight = sourceHighlight;
            QBC(sourceLineDiv(currentLine), 'white');
            wsocket.send('{"cmd": "STEPOUT"}');
        }                function delBreakpoint()        {
            QV(selectedBreakpoint, 0);
            selectedBreakpoint = "";
            QE("delBreakpointButton", 0);
        }        function onBreakpointClick(id)        {
            if (selectedBreakpoint == id)
            {
                QS(id)["background-color"] = 'white';
                selectedBreakpoint = "";
                QE("delBreakpointButton", 0);
            }
            else
            {
                if (selectedBreakpoint != "")
                {
                    QS(selectedBreakpoint)["background-color"] = 'white';
                }
                QS(id)["background-color"] = 'yellow';
                selectedBreakpoint = id;
                QE("delBreakpointButton", 1);
            }

            for (var i = 0; i < currentCallstack.length; ++i)
            {
                QS('_stack_' + i).color = callstack_default;
            }
            var bp = QHVAL(id).split(':')[0];
            var jj = { cmd: 'SOURCE', name: bp == '' ? '<<NATIVE>>' : bp };
            wsocket.send(JSON.stringify(jj));        
        }        function getCallstackLevel()        {
            // Determine the Callstack Level
            var level = -1;
            for (var i = 0; i < currentCallstack.length; ++i) {
                if (QS('_stack_' + i).color == callstack_scope) {
                    level = 0 - (i + 1);
                    break;
                }
            }
            return (level);
        }        function queryVal()        {
            //QA("queryResults", '<div class="VARQUERY">' + QVAL('queryText') + '=</div>');
            wsocket.send('{"cmd": "QUERY", "var": "' + QVAL('queryText') + '", "level": ' + getCallstackLevel() + '}');
            QVAL('queryText', "");
        }        function evalString()        {
            wsocket.send('{"cmd": "EVAL", "eval": "' + QVAL('queryText') + '", "level": ' + getCallstackLevel() + '}');
            QVAL('queryText', "");
        }        function localsQuery()        {
            QA("queryResults", '<div class="VARQUERY">Locals:&nbsp;</div>');
            wsocket.send('{"cmd": "LOCALS", "level": ' + getCallstackLevel() + '}');
        }        function clearQuery()        {
            QH('queryResults', '');
        }        function displayStatus(msg, msgClass)        {
            QH('statustext', '<div class="' + (msgClass?msgClass:'GENERIC_MODULE') + '">' + msg + '</div>');
        }        function sourceLineDiv(line)        {
            return ('__src' + line);
        }        function onToggleBreakpoint(line)        {
            QBC('L' + sourceLineDiv(line), bpColor);
            var fn = currentModule;

            for (var i = 0; i < currentCallstack.length; ++i)
            {
                if (QS('_stack_' + i).color == callstack_scope)
                {
                    fn = currentCallstack[i].fileName.split('\\').join('\\\\');
                }
            }

            wsocket.send('{"cmd": "BREAKPOINT", "file":"' + fn + '", "line":' + line + ', "mode":"add"}');
        }        function onDumpHeap()        {
            wsocket.send('{"cmd": "HEAP"}');
        }        function forceGC()        {
            wsocket.send('{"cmd": "GC"}');
        }        function displayCallstack()        {
            var display = [];
            var indent = '-->&nbsp;';
            for (var i = 0; i < currentCallstack.length; ++i)
            {
                if (currentCallstack[i].funcName == '') { currentCallstack[i].funcName = '(anonymous)'; }
                var fname = currentCallstack[i].fileName.split('\\').join('\\\\');

                display.push('<div depth="' + (i+1) + '" id="_stack_' + i + '" class="CALLSTACK" onclick="onCallstackClick(\'' + currentCallstack.length + '\', \'_stack_' + i + '\', \'' + fname + '\')">' + indent + currentCallstack[i].funcName + ':' + currentCallstack[i].lineNumber + '</div>');
                indent = '&nbsp;&nbsp;&nbsp;' + indent;
            }
            QH('CallstackWindow', display.join(''));

            if (currentCallstack.length > 0)
            {
                QS('_stack_0').color = callstack_scope;
            }
        }        function onCallstackClick(len, id, filename)        {
            for (var i = 0; i < len; ++i)
            {
                QS('_stack_' + i).color = callstack_default;
            }
            QS(id).color = callstack_scope;
            var jj = { cmd: 'SOURCE', name: filename };
            wsocket.send(JSON.stringify(jj));
        }        function manuallyLoadSource()        {
            for (var i = 0; i < currentCallstack.length; ++i)
            {
                QS('_stack_' + i).color = callstack_default;
            }
            QH('LogWindow', '');

            var jj = { cmd: 'SOURCE', name: QVAL('loadSourceText') == ''?'<<NATIVE>>':QVAL('loadSourceText') };
            wsocket.send(JSON.stringify(jj));
        }        // Poll the server, if it responds, refresh the page.
        function serverPoll()
        {
            xdr = null;
            try { xdr = new XDomainRequest(); } catch (e) { }
            if (!xdr) xdr = new XMLHttpRequest();
            xdr.open("HEAD", window.location.href);
            xdr.timeout = 15000;
            xdr.onload = function () { reload(); };
            xdr.onerror = xdr.ontimeout = function () { setTimeout(serverPoll, 2000); };
            xdr.send();
        }

        function reload() { window.location.href = window.location.href; }
    </script>
</body>
</html>
'; var optTargetHost = '127.0.0.1'; var optTargetPort = 9091;