2017-03-29 01:50:16 +00:00
<!DOCTYPE html>
2015-03-23 18:48:02 +00:00
<!--
2018-09-11 22:27:47 +00:00
Copyright (c) Facebook, Inc. and its affiliates.
2018-06-08 09:57:15 +00:00
2018-02-17 02:24:55 +00:00
This source code is licensed under the MIT license found in the
LICENSE file in the root directory of this source tree.
2015-03-23 18:48:02 +00:00
-->
2015-03-19 19:10:41 +00:00
< html >
< head >
< meta charset = utf-8 >
2017-01-26 19:51:41 +00:00
< link rel = "icon" href = "data:;base64,iVBORw0KGgo=" >
2015-03-19 19:10:41 +00:00
< title > React Native Debugger< / title >
2017-09-29 12:38:12 +00:00
< script src = "/debugger-ui/DeltaPatcher.js" > < / script >
< script src = "/debugger-ui/deltaUrlToBlobUrl.js" > < / script >
2015-08-21 17:15:04 +00:00
< script >
2016-10-31 18:16:14 +00:00
/* eslint-env browser */
2017-03-29 01:50:16 +00:00
'use strict';
2015-03-19 19:10:41 +00:00
(function() {
2017-03-29 01:50:16 +00:00
const isMacLike = /(Mac|iPhone|iPod|iPad)/i.test(navigator.platform);
const refreshShortcut = isMacLike ? '⌘R' : 'Ctrl R';
window.onload = function() {
if (!isMacLike) {
document.getElementById('shortcut').innerHTML = 'Ctrl⇧J';
}
Page.render();
};
2015-03-19 19:10:41 +00:00
2017-03-29 01:50:16 +00:00
const Assets = {
blueIcon: '',
grayIcon: '',
orangeIcon: '',
};
2017-01-26 19:51:41 +00:00
2017-03-29 01:50:16 +00:00
const Page = window.Page = {
state: {
isDark: localStorage.getItem('darkTheme') === 'on',
2017-03-29 01:50:23 +00:00
isPriorityMaintained: localStorage.getItem('maintainPriority') === 'on',
2017-03-29 01:50:16 +00:00
status: {type: 'disconnected'},
visibilityState: document.visibilityState,
},
setState(partialState) {
Page.state = Object.assign({}, Page.state, partialState);
Page.render();
},
render() {
const {
isDark,
2017-03-29 01:50:23 +00:00
isPriorityMaintained,
2017-03-29 01:50:16 +00:00
status,
visibilityState,
} = Page.state;
const statusNode = document.getElementById('status');
switch (status.type) {
case 'connected':
statusNode.innerHTML = 'Debugger session #' + status.id + ' active.';
break;
case 'error':
statusNode.innerHTML = status.error.reason || 'Disconnected from proxy. Attempting reconnection. Is node server running?';
break;
case 'connecting':
case 'disconnected':
// Fall through.
default:
statusNode.innerHTML = 'Waiting, press < span class = "shortcut" > ' + refreshShortcut + '< / span > in simulator to reload and connect.';
break;
}
2017-01-26 19:51:41 +00:00
2017-03-29 01:50:16 +00:00
const linkNode = document.querySelector('link[rel=icon]');
if (status.type === 'disconnected' ||
status.type === 'error') {
linkNode.href = Assets.grayIcon;
} else {
2017-03-29 01:50:23 +00:00
if (visibilityState === 'visible' || isPriorityMaintained) {
2017-03-29 01:50:16 +00:00
linkNode.href = Assets.blueIcon;
} else {
linkNode.href = Assets.orangeIcon;
}
}
2016-09-14 09:55:14 +00:00
2017-03-29 01:50:16 +00:00
const darkCheckbox = document.getElementById('dark');
document.body.classList.toggle('dark', isDark);
darkCheckbox.checked = isDark;
localStorage.setItem('darkTheme', isDark ? 'on' : '');
2017-03-29 01:50:23 +00:00
const maintainPriorityCheckbox = document.getElementById('maintain-priority');
const silence = document.getElementById('silence');
2017-06-19 20:30:00 +00:00
silence.volume = 0.1;
2017-03-29 01:50:23 +00:00
if (isPriorityMaintained) {
silence.play();
} else {
silence.pause();
}
maintainPriorityCheckbox.checked = isPriorityMaintained;
localStorage.setItem('maintainPriority', isPriorityMaintained ? 'on' : '');
2017-03-29 01:50:16 +00:00
},
2015-09-24 03:02:22 +00:00
2017-03-29 01:50:16 +00:00
toggleDarkTheme() {
Page.setState({isDark: !Page.state.isDark});
},
2016-02-05 23:16:16 +00:00
2017-03-29 01:50:23 +00:00
togglePriorityMaintenance() {
Page.setState({isPriorityMaintained: !Page.state.isPriorityMaintained});
},
2017-03-29 01:50:16 +00:00
};
2015-03-19 19:10:41 +00:00
2017-03-29 01:50:16 +00:00
function connectToDebuggerProxy() {
const ws = new WebSocket('ws://' + window.location.host + '/debugger-proxy?role=debugger&name=Chrome');
let worker;
2018-06-08 09:57:15 +00:00
let queuedMessages = [];
let appExecuted = false;
2017-03-29 01:50:16 +00:00
function createJSRuntime() {
// This worker will run the application JavaScript code,
// making sure that it's run in an environment without a global
// document, to make it consistent with the JSC executor environment.
worker = new Worker('debuggerWorker.js');
worker.onmessage = function(message) {
ws.send(JSON.stringify(message.data));
};
window.onbeforeunload = function() {
return 'If you reload this page, it is going to break the debugging session. ' +
'You should press' + refreshShortcut + 'in simulator to reload.';
};
updateVisibility();
2016-02-05 23:16:16 +00:00
}
2015-03-19 19:10:41 +00:00
2017-03-29 01:50:16 +00:00
function shutdownJSRuntime() {
if (worker) {
worker.terminate();
worker = null;
window.onbeforeunload = null;
}
2016-10-31 18:16:17 +00:00
}
2017-01-26 19:51:41 +00:00
2017-03-29 01:50:16 +00:00
function updateVisibility() {
2017-03-29 01:50:23 +00:00
if (worker & & !Page.state.isPriorityMaintained) {
2017-03-29 01:50:16 +00:00
worker.postMessage({
method: 'setDebuggerVisibility',
visibilityState: document.visibilityState,
});
2017-01-26 19:51:41 +00:00
}
2017-03-29 01:50:16 +00:00
Page.setState({visibilityState: document.visibilityState});
2017-01-26 19:51:41 +00:00
}
2016-10-31 18:16:17 +00:00
2017-03-29 01:50:16 +00:00
ws.onopen = function() {
Page.setState({status: {type: 'connecting'}});
};
2015-03-19 19:10:41 +00:00
2017-09-29 12:38:12 +00:00
ws.onmessage = async function(message) {
2017-03-29 01:50:16 +00:00
if (!message.data) {
return;
}
const object = JSON.parse(message.data);
2016-02-05 23:16:16 +00:00
2017-03-29 01:50:16 +00:00
if (object.$event === 'client-disconnected') {
shutdownJSRuntime();
Page.setState({status: {type: 'disconnected'}});
return;
}
2016-02-05 23:16:16 +00:00
2017-03-29 01:50:16 +00:00
if (!object.method) {
return;
}
2015-08-31 18:18:23 +00:00
2017-03-29 01:50:16 +00:00
// Special message that asks for a new JS runtime
if (object.method === 'prepareJSRuntime') {
shutdownJSRuntime();
console.clear();
createJSRuntime();
ws.send(JSON.stringify({replyID: object.id}));
Page.setState({status: {type: 'connected', id: object.id}});
} else if (object.method === '$disconnected') {
shutdownJSRuntime();
Page.setState({status: {type: 'disconnected'}});
2017-09-29 12:38:12 +00:00
} else if (object.method === 'executeApplicationScript') {
2018-07-11 11:18:34 +00:00
appExecuted = false;
2017-09-29 12:38:12 +00:00
worker.postMessage({
...object,
url: await getBlobUrl(object.url),
});
2018-06-08 09:57:15 +00:00
appExecuted = true;
// Flush any messages queued up and clear them
for (const message of queuedMessages) {
worker.postMessage(message);
}
queuedMessages = [];
2017-03-29 01:50:16 +00:00
} else {
2018-06-08 09:57:15 +00:00
// Otherwise, pass through to the worker provided the
// application script has been executed. If not add
// it to a queue until it has been executed.
if (appExecuted) {
worker.postMessage(object);
} else {
queuedMessages.push(object);
}
2017-03-29 01:50:16 +00:00
}
};
2015-07-16 21:16:50 +00:00
2017-03-29 01:50:16 +00:00
ws.onclose = function(error) {
shutdownJSRuntime();
Page.setState({status: {type: 'error', error}});
if (error.reason) {
console.warn(error.reason);
}
setTimeout(connectToDebuggerProxy, 500);
};
2016-10-31 18:16:17 +00:00
2017-03-29 01:50:16 +00:00
// Let debuggerWorker.js know when we're not visible so that we can warn about
// poor performance when using remote debugging.
document.addEventListener('visibilitychange', updateVisibility, false);
}
2015-03-19 19:10:41 +00:00
2017-03-29 01:50:16 +00:00
connectToDebuggerProxy();
2015-07-16 21:16:50 +00:00
2017-09-29 12:38:12 +00:00
async function getBlobUrl(url) {
return await window.deltaUrlToBlobUrl(url.replace('.bundle', '.delta'));
}
2015-03-19 19:10:41 +00:00
})();
< / script >
< style type = "text/css" >
2017-03-23 11:28:26 +00:00
html,
2015-08-07 22:01:40 +00:00
body {
2016-10-31 18:16:14 +00:00
font-family: Helvetica, Verdana, sans-serif;
2015-08-07 22:01:40 +00:00
font-size: large;
2016-10-31 18:16:14 +00:00
font-weight: 200;
2017-03-23 11:28:26 +00:00
height: 100%;
2015-08-21 17:15:04 +00:00
margin: 0;
2015-08-07 22:01:40 +00:00
padding: 0;
}
2015-03-19 19:10:41 +00:00
.shortcut {
2016-10-31 18:16:14 +00:00
border-radius: 4px;
2015-03-19 19:10:41 +00:00
color: #eee;
background-color: #333;
2016-10-31 18:16:14 +00:00
font-family: "Monaco", monospace;
font-size: medium;
2015-03-19 19:10:41 +00:00
letter-spacing: 3px;
2016-10-31 18:16:14 +00:00
padding: 4px;
2015-03-19 19:10:41 +00:00
}
2015-08-07 22:01:40 +00:00
.content {
padding: 10px;
2015-03-19 19:10:41 +00:00
}
2017-03-23 11:28:26 +00:00
body.dark {
background-color: #242424;
2018-08-24 09:26:55 +00:00
color: #afafaf;
2017-03-23 11:28:26 +00:00
}
.dark .shortcut {
2018-08-24 09:26:55 +00:00
color: #c1c1c1;
2017-03-23 11:28:26 +00:00
}
.dark a {
color: #3b99fc;
}
2017-03-29 01:50:16 +00:00
input[type=checkbox] {
vertical-align: middle;
}
2015-03-19 19:10:41 +00:00
< / style >
< / head >
< body >
2015-08-07 22:01:40 +00:00
< div class = "content" >
2017-03-23 11:28:26 +00:00
< label for = "dark" >
2017-03-29 01:50:16 +00:00
< input type = "checkbox" id = "dark" onclick = "Page.toggleDarkTheme()" > Dark Theme
2017-03-23 11:28:26 +00:00
< / label >
2017-03-29 01:50:23 +00:00
< label for = "maintain-priority" >
< input type = "checkbox" id = "maintain-priority" onclick = "Page.togglePriorityMaintenance()" > Maintain Priority
< / label >
2015-08-07 22:01:40 +00:00
< p >
2017-03-29 01:50:16 +00:00
React Native JS code runs as a web worker inside this tab.
2015-08-07 22:01:40 +00:00
< / p >
2017-08-25 00:30:46 +00:00
< p > Press < kbd id = "shortcut" class = "shortcut" > ⌘⌥I< / kbd > to open Developer Tools. Enable < a href = "https://stackoverflow.com/a/17324511/232122" target = "_blank" > Pause On Caught Exceptions< / a > for a better debugging experience.< / p >
2017-05-05 18:37:43 +00:00
< p > You may also install < a href = "https://github.com/facebook/react-devtools/tree/master/packages/react-devtools" target = "_blank" > the standalone version of React Developer Tools< / a > to inspect the React component hierarchy, their props, and state.< / p >
2015-08-21 17:15:04 +00:00
< p > Status: < span id = "status" > Loading...< / span > < / p >
2015-08-07 22:01:40 +00:00
< / div >
2017-06-19 20:30:00 +00:00
< audio id = "silence" loop src = "data:audio/wav;base64,UklGRmxsAQBXQVZFZm10IBIAAAABAAEARKwAAIhYAQACABAAAABkYXRhiFgBAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAHAAcABwAHAAcABwAHAAcABwAHAAcAB
2015-03-19 19:10:41 +00:00
< / body >
< / html >