2015-07-23 23:56:23 +00:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* @providesModule setupDevtools
|
|
|
|
* @flow
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
2016-07-19 12:43:16 +00:00
|
|
|
var NativeModules = require('NativeModules');
|
2016-10-11 17:08:54 +00:00
|
|
|
var Platform = require('Platform');
|
2016-07-19 12:43:16 +00:00
|
|
|
|
2015-07-23 23:56:23 +00:00
|
|
|
function setupDevtools() {
|
|
|
|
var messageListeners = [];
|
|
|
|
var closeListeners = [];
|
2016-07-19 12:43:16 +00:00
|
|
|
var hostname = 'localhost';
|
|
|
|
if (Platform.OS === 'android' && NativeModules.AndroidConstants) {
|
|
|
|
hostname = NativeModules.AndroidConstants.ServerHost.split(':')[0];
|
|
|
|
}
|
2016-11-08 02:37:02 +00:00
|
|
|
var port = window.__REACT_DEVTOOLS_PORT__ || 8097;
|
|
|
|
var ws = new window.WebSocket('ws://' + hostname + ':' + port + '/devtools');
|
2015-07-23 23:56:23 +00:00
|
|
|
// this is accessed by the eval'd backend code
|
|
|
|
var FOR_BACKEND = { // eslint-disable-line no-unused-vars
|
|
|
|
resolveRNStyle: require('flattenStyle'),
|
|
|
|
wall: {
|
|
|
|
listen(fn) {
|
|
|
|
messageListeners.push(fn);
|
|
|
|
},
|
|
|
|
onClose(fn) {
|
|
|
|
closeListeners.push(fn);
|
|
|
|
},
|
|
|
|
send(data) {
|
|
|
|
ws.send(JSON.stringify(data));
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
2016-03-08 03:19:20 +00:00
|
|
|
ws.onclose = handleClose;
|
|
|
|
ws.onerror = handleClose;
|
2015-07-23 23:56:23 +00:00
|
|
|
ws.onopen = function () {
|
|
|
|
tryToConnect();
|
|
|
|
};
|
|
|
|
|
2016-03-08 03:19:20 +00:00
|
|
|
var hasClosed = false;
|
|
|
|
function handleClose() {
|
|
|
|
if (!hasClosed) {
|
|
|
|
hasClosed = true;
|
2016-04-01 14:01:51 +00:00
|
|
|
setTimeout(setupDevtools, 2000);
|
2016-03-08 03:19:20 +00:00
|
|
|
closeListeners.forEach(fn => fn());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-23 23:56:23 +00:00
|
|
|
function tryToConnect() {
|
|
|
|
ws.send('attach:agent');
|
|
|
|
var _interval = setInterval(() => ws.send('attach:agent'), 500);
|
|
|
|
ws.onmessage = evt => {
|
|
|
|
if (evt.data.indexOf('eval:') === 0) {
|
|
|
|
clearInterval(_interval);
|
|
|
|
initialize(evt.data.slice('eval:'.length));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function initialize(text) {
|
|
|
|
try {
|
|
|
|
// FOR_BACKEND is used by the eval'd code
|
|
|
|
eval(text); // eslint-disable-line no-eval
|
|
|
|
} catch (e) {
|
2015-10-29 18:15:43 +00:00
|
|
|
console.error('Failed to eval: ' + e.message);
|
2015-07-23 23:56:23 +00:00
|
|
|
return;
|
|
|
|
}
|
2016-04-21 16:27:50 +00:00
|
|
|
// This is breaking encapsulation of the React package. Move plz.
|
2016-11-04 12:40:26 +00:00
|
|
|
var ReactNativeComponentTree = require('ReactNativeComponentTree');
|
2015-07-23 23:56:23 +00:00
|
|
|
window.__REACT_DEVTOOLS_GLOBAL_HOOK__.inject({
|
2016-04-21 16:27:50 +00:00
|
|
|
ComponentTree: {
|
|
|
|
getClosestInstanceFromNode: function (node) {
|
|
|
|
return ReactNativeComponentTree.getClosestInstanceFromNode(node);
|
|
|
|
},
|
|
|
|
getNodeFromInstance: function (inst) {
|
|
|
|
// inst is an internal instance (but could be a composite)
|
|
|
|
while (inst._renderedComponent) {
|
|
|
|
inst = inst._renderedComponent;
|
|
|
|
}
|
|
|
|
if (inst) {
|
|
|
|
return ReactNativeComponentTree.getNodeFromInstance(inst);
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2016-11-04 12:40:26 +00:00
|
|
|
Mount: require('ReactNativeMount'),
|
|
|
|
Reconciler: require('ReactReconciler')
|
2015-07-23 23:56:23 +00:00
|
|
|
});
|
|
|
|
ws.onmessage = handleMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleMessage(evt) {
|
2015-09-05 00:00:21 +00:00
|
|
|
// It's hard to handle JSON in a safe manner without inspecting it at
|
|
|
|
// runtime, hence the any
|
|
|
|
var data: any;
|
2015-07-23 23:56:23 +00:00
|
|
|
try {
|
|
|
|
data = JSON.parse(evt.data);
|
|
|
|
} catch (e) {
|
|
|
|
return console.error('failed to parse json: ' + evt.data);
|
|
|
|
}
|
|
|
|
// the devtools closed
|
|
|
|
if (data.$close || data.$error) {
|
|
|
|
closeListeners.forEach(fn => fn());
|
|
|
|
window.__REACT_DEVTOOLS_GLOBAL_HOOK__.emit('shutdown');
|
|
|
|
tryToConnect();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (data.$open) {
|
|
|
|
return; // ignore
|
|
|
|
}
|
|
|
|
messageListeners.forEach(fn => {
|
|
|
|
try {
|
|
|
|
fn(data);
|
|
|
|
} catch (e) {
|
|
|
|
// jsc doesn't play so well with tracebacks that go into eval'd code,
|
|
|
|
// so the stack trace here will stop at the `eval()` call. Getting the
|
|
|
|
// message that caused the error is the best we can do for now.
|
|
|
|
console.log(data);
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = setupDevtools;
|