268 lines
134 KiB
HTML
268 lines
134 KiB
HTML
|
<!DOCTYPE html>
|
||
|
<!--
|
||
|
Copyright (c) 2015-present, Facebook, Inc.
|
||
|
All rights reserved.
|
||
|
|
||
|
This source code is licensed under the BSD-style license found in the
|
||
|
LICENSE file in the root directory of this source tree. An additional grant
|
||
|
of patent rights can be found in the PATENTS file in the same directory.
|
||
|
-->
|
||
|
<html>
|
||
|
<head>
|
||
|
<meta charset=utf-8>
|
||
|
<link rel="icon" href="data:;base64,iVBORw0KGgo=">
|
||
|
<title>React Native Debugger</title>
|
||
|
<script src="/debugger-ui/DeltaPatcher.js"></script>
|
||
|
<script src="/debugger-ui/deltaUrlToBlobUrl.js"></script>
|
||
|
<script>
|
||
|
/* eslint-env browser */
|
||
|
'use strict';
|
||
|
(function() {
|
||
|
|
||
|
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();
|
||
|
};
|
||
|
|
||
|
const Assets = {
|
||
|
blueIcon: '',
|
||
|
grayIcon: '',
|
||
|
orangeIcon: '',
|
||
|
};
|
||
|
|
||
|
const Page = window.Page = {
|
||
|
|
||
|
state: {
|
||
|
isDark: localStorage.getItem('darkTheme') === 'on',
|
||
|
isPriorityMaintained: localStorage.getItem('maintainPriority') === 'on',
|
||
|
status: {type: 'disconnected'},
|
||
|
visibilityState: document.visibilityState,
|
||
|
},
|
||
|
|
||
|
setState(partialState) {
|
||
|
Page.state = Object.assign({}, Page.state, partialState);
|
||
|
Page.render();
|
||
|
},
|
||
|
|
||
|
render() {
|
||
|
const {
|
||
|
isDark,
|
||
|
isPriorityMaintained,
|
||
|
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;
|
||
|
}
|
||
|
|
||
|
const linkNode = document.querySelector('link[rel=icon]');
|
||
|
if (status.type === 'disconnected' ||
|
||
|
status.type === 'error') {
|
||
|
linkNode.href = Assets.grayIcon;
|
||
|
} else {
|
||
|
if (visibilityState === 'visible' || isPriorityMaintained) {
|
||
|
linkNode.href = Assets.blueIcon;
|
||
|
} else {
|
||
|
linkNode.href = Assets.orangeIcon;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const darkCheckbox = document.getElementById('dark');
|
||
|
document.body.classList.toggle('dark', isDark);
|
||
|
darkCheckbox.checked = isDark;
|
||
|
localStorage.setItem('darkTheme', isDark ? 'on' : '');
|
||
|
|
||
|
const maintainPriorityCheckbox = document.getElementById('maintain-priority');
|
||
|
const silence = document.getElementById('silence');
|
||
|
silence.volume = 0.1;
|
||
|
if (isPriorityMaintained) {
|
||
|
silence.play();
|
||
|
} else {
|
||
|
silence.pause();
|
||
|
}
|
||
|
maintainPriorityCheckbox.checked = isPriorityMaintained;
|
||
|
localStorage.setItem('maintainPriority', isPriorityMaintained ? 'on' : '');
|
||
|
},
|
||
|
|
||
|
toggleDarkTheme() {
|
||
|
Page.setState({isDark: !Page.state.isDark});
|
||
|
},
|
||
|
|
||
|
togglePriorityMaintenance() {
|
||
|
Page.setState({isPriorityMaintained: !Page.state.isPriorityMaintained});
|
||
|
},
|
||
|
|
||
|
};
|
||
|
|
||
|
function connectToDebuggerProxy() {
|
||
|
const ws = new WebSocket('ws://' + window.location.host + '/debugger-proxy?role=debugger&name=Chrome');
|
||
|
let worker;
|
||
|
|
||
|
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();
|
||
|
}
|
||
|
|
||
|
function shutdownJSRuntime() {
|
||
|
if (worker) {
|
||
|
worker.terminate();
|
||
|
worker = null;
|
||
|
window.onbeforeunload = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function updateVisibility() {
|
||
|
if (worker && !Page.state.isPriorityMaintained) {
|
||
|
worker.postMessage({
|
||
|
method: 'setDebuggerVisibility',
|
||
|
visibilityState: document.visibilityState,
|
||
|
});
|
||
|
}
|
||
|
Page.setState({visibilityState: document.visibilityState});
|
||
|
}
|
||
|
|
||
|
ws.onopen = function() {
|
||
|
Page.setState({status: {type: 'connecting'}});
|
||
|
};
|
||
|
|
||
|
ws.onmessage = async function(message) {
|
||
|
if (!message.data) {
|
||
|
return;
|
||
|
}
|
||
|
const object = JSON.parse(message.data);
|
||
|
|
||
|
if (object.$event === 'client-disconnected') {
|
||
|
shutdownJSRuntime();
|
||
|
Page.setState({status: {type: 'disconnected'}});
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (!object.method) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// 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'}});
|
||
|
} else if (object.method === 'executeApplicationScript') {
|
||
|
worker.postMessage({
|
||
|
...object,
|
||
|
url: await getBlobUrl(object.url),
|
||
|
});
|
||
|
} else {
|
||
|
// Otherwise, pass through to the worker.
|
||
|
worker.postMessage(object);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
ws.onclose = function(error) {
|
||
|
shutdownJSRuntime();
|
||
|
Page.setState({status: {type: 'error', error}});
|
||
|
if (error.reason) {
|
||
|
console.warn(error.reason);
|
||
|
}
|
||
|
setTimeout(connectToDebuggerProxy, 500);
|
||
|
};
|
||
|
|
||
|
// 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);
|
||
|
}
|
||
|
|
||
|
connectToDebuggerProxy();
|
||
|
|
||
|
async function getBlobUrl(url) {
|
||
|
return await window.deltaUrlToBlobUrl(url.replace('.bundle', '.delta'));
|
||
|
}
|
||
|
})();
|
||
|
</script>
|
||
|
<style type="text/css">
|
||
|
html,
|
||
|
body {
|
||
|
font-family: Helvetica, Verdana, sans-serif;
|
||
|
font-size: large;
|
||
|
font-weight: 200;
|
||
|
height: 100%;
|
||
|
margin: 0;
|
||
|
padding: 0;
|
||
|
}
|
||
|
.shortcut {
|
||
|
border-radius: 4px;
|
||
|
color: #eee;
|
||
|
background-color: #333;
|
||
|
font-family: "Monaco", monospace;
|
||
|
font-size: medium;
|
||
|
letter-spacing: 3px;
|
||
|
padding: 4px;
|
||
|
}
|
||
|
.content {
|
||
|
padding: 10px;
|
||
|
}
|
||
|
body.dark {
|
||
|
background-color: #242424;
|
||
|
color: #666;
|
||
|
}
|
||
|
.dark .shortcut {
|
||
|
color: #777;
|
||
|
}
|
||
|
.dark a {
|
||
|
color: #3b99fc;
|
||
|
}
|
||
|
input[type=checkbox] {
|
||
|
vertical-align: middle;
|
||
|
}
|
||
|
</style>
|
||
|
</head>
|
||
|
<body>
|
||
|
<div class="content">
|
||
|
<label for="dark">
|
||
|
<input type="checkbox" id="dark" onclick="Page.toggleDarkTheme()"> Dark Theme
|
||
|
</label>
|
||
|
<label for="maintain-priority">
|
||
|
<input type="checkbox" id="maintain-priority" onclick="Page.togglePriorityMaintenance()"> Maintain Priority
|
||
|
</label>
|
||
|
<p>
|
||
|
React Native JS code runs as a web worker inside this tab.
|
||
|
</p>
|
||
|
<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>
|
||
|
<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>
|
||
|
<p>Status: <span id="status">Loading...</span></p>
|
||
|
</div>
|
||
|
<audio id="silence" loop src="data:audio/wav;base64,UklGRmxsAQBXQVZFZm10IBIAAAABAAEARKwAAIhYAQACABAAAABkYXRhiFgBAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwADAAMAAwAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABAAEAAQABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAUABQAFAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABgAGAAYABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAHAAcABwAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAIAAgACAAHAAcABwAHAAcABwAHAAcABwAHAAcAB
|
||
|
</body>
|
||
|
</html>
|