Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
33 changes: 33 additions & 0 deletions front_end/models/react_native/BUILD.gn
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
# Copyright (c) Meta Platforms, Inc. and affiliates.
# Copyright 2024 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.

import("../../../scripts/build/ninja/devtools_entrypoint.gni")
import("../../../scripts/build/ninja/devtools_module.gni")
import("../visibility.gni")

devtools_module("react-native") {
sources = [
"ReactDevToolsBindingsModel.ts",
]

deps = [
"../../core/common:bundle",
"../../core/sdk:bundle",
]
}

devtools_entrypoint("bundle") {
entrypoint = "react_native.ts"

deps = [
":react-native",
]

visibility = [
":*",
]

visibility += devtools_models_visibility
}
121 changes: 121 additions & 0 deletions front_end/models/react_native/ReactDevToolsBindingsModel.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
// Copyright (c) Meta Platforms, Inc. and affiliates.
// Copyright 2024 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

import * as SDK from '../../core/sdk/sdk.js';
import type * as Common from '../../core/common/common.js';

type JSONValue = null | string | number | boolean | {[key: string]: JSONValue} | JSONValue[];
type DomainName = 'react-devtools';
type DomainMessageListener = (message: JSONValue) => void;
type BindingCalledEventTargetEvent = Common.EventTarget.EventTargetEvent<SDK.RuntimeModel.EventTypes[SDK.RuntimeModel.Events.BindingCalled]>;

const RUNTIME_GLOBAL = '__FUSEBOX_REACT_DEVTOOLS_DISPATCHER__';

export class ReactDevToolsBindingsModel extends SDK.SDKModel.SDKModel {
private readonly domainToListeners: Map<DomainName, Set<DomainMessageListener>> = new Map();

constructor(target: SDK.Target.Target) {
super(target);

this.initialize(target);
}

private initialize(target: SDK.Target.Target): void {
const runtimeModel = target.model(SDK.RuntimeModel.RuntimeModel);
if (!runtimeModel) {
return;
}

runtimeModel.addEventListener(SDK.RuntimeModel.Events.BindingCalled, this.bindingCalled, this);
void this.enable(target);
}

private bindingCalled(event: BindingCalledEventTargetEvent): void {
const serializedMessage = event.data.payload;

try {
const {domain, message} = JSON.parse(serializedMessage);
this.dispatchMessageToDomainEventListeners(domain, message);
} catch (err) {
throw new Error('Failed to parse bindingCalled event payload:', err);
}
}

subscribeToDomainMessages(domainName: DomainName, listener: DomainMessageListener): void {
let listeners = this.domainToListeners.get(domainName);
if (!listeners) {
listeners = new Set();
this.domainToListeners.set(domainName, listeners);
}

listeners.add(listener);
}

unsubscribeFromDomainMessages(domainName: DomainName, listener: DomainMessageListener): void {
const listeners = this.domainToListeners.get(domainName);
if (!listeners) {
return;
}

listeners.delete(listener);
}

private dispatchMessageToDomainEventListeners(domainName: DomainName, message: JSONValue): void {
const listeners = this.domainToListeners.get(domainName);
if (!listeners) {
// No subscriptions, no need to throw, just don't notify.
return;
}

const errors = [];
for (const listener of listeners) {
try {
listener(message);
} catch (e) {
errors.push(e);
}
}

if (errors.length > 0) {
throw new AggregateError(
errors,
`[ReactDevToolsBindingsModel] Error occurred when calling event listeners for domain: ${domainName}`,
);
}
}

async initializeDomain(domainName: DomainName): Promise<void> {
const runtimeModel = this.target().model(SDK.RuntimeModel.RuntimeModel);
if (!runtimeModel) {
throw new Error(`[ReactDevToolsBindingsModel] Failed to initialize domain ${domainName}: runtime model is not available`);
}

await runtimeModel.agent.invoke_evaluate({expression: `void ${RUNTIME_GLOBAL}.initializeDomain('${domainName}')`});
}

async sendMessage(domainName: DomainName, message: JSONValue): Promise<void> {
const runtimeModel = this.target().model(SDK.RuntimeModel.RuntimeModel);
if (!runtimeModel) {
throw new Error(`[ReactDevToolsBindingsModel] Failed to send message for domain ${domainName}: runtime model is not available`);
}

const serializedMessage = JSON.stringify(message);

await runtimeModel.agent.invoke_evaluate({expression: `${RUNTIME_GLOBAL}.sendMessage('${domainName}', '${serializedMessage}')`});
}

async enable(target: SDK.Target.Target): Promise<void> {
const runtimeModel = target.model(SDK.RuntimeModel.RuntimeModel);
if (!runtimeModel) {
throw new Error('[ReactDevToolsBindingsModel] Failed to enable model: runtime model is not available');
}

await runtimeModel.agent.invoke_evaluate({expression: `${RUNTIME_GLOBAL}.BINDING_NAME`})
.then(response => response.result.value)
.then(bindingName => runtimeModel.agent.invoke_addBinding({name: bindingName}));
}
}

SDK.SDKModel.SDKModel.register(ReactDevToolsBindingsModel, {capabilities: SDK.Target.Capability.JS, autostart: false});
8 changes: 8 additions & 0 deletions front_end/models/react_native/react_native.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
// Copyright (c) Meta Platforms, Inc. and affiliates.
// Copyright 2024 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

import * as ReactDevToolsBindingsModel from './ReactDevToolsBindingsModel.js';

export {ReactDevToolsBindingsModel};
5 changes: 3 additions & 2 deletions scripts/eslint_rules/lib/check_license_header.js
Original file line number Diff line number Diff line change
Expand Up @@ -67,11 +67,12 @@ const EXCLUDED_FILES = [
];

const META_CODE_PATHS = [
'core/host/RNPerfMetrics.ts',
'entrypoints/rn_inspector',
'global_typings/react_native.d.ts',
'models/react_native',
'panels/react_devtools_placeholder',
'panels/rn_welcome',
'core/host/RNPerfMetrics.ts',
'global_typings/react_native.d.ts',
];

const OTHER_LICENSE_HEADERS = [
Expand Down