@nextop-os/browser-node 0.0.28 → 0.0.29
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/assets/workspace-dock-website.png +0 -0
- package/dist/{chunk-JRJTAIK5.js → chunk-G3H2RFWQ.js} +3 -1
- package/dist/chunk-G3H2RFWQ.js.map +1 -0
- package/dist/{chunk-F2AFX4OE.js → chunk-HBFCSUQO.js} +31 -2
- package/dist/chunk-HBFCSUQO.js.map +1 -0
- package/dist/electron-main/index.d.ts +11 -2
- package/dist/electron-main/index.js +100 -12
- package/dist/electron-main/index.js.map +1 -1
- package/dist/electron-preload/index.d.ts +23 -1
- package/dist/electron-preload/index.js +112 -1
- package/dist/electron-preload/index.js.map +1 -1
- package/dist/i18n/index.d.ts +1 -1
- package/dist/i18n/index.js +1 -1
- package/dist/index.d.ts +9 -1
- package/dist/index.js +1 -1
- package/dist/react/index.d.ts +2 -1
- package/dist/react/index.js +1 -1
- package/dist/workbench/index.js +5 -1
- package/dist/workbench/index.js.map +1 -1
- package/package.json +5 -5
- package/dist/chunk-F2AFX4OE.js.map +0 -1
- package/dist/chunk-JRJTAIK5.js.map +0 -1
|
Binary file
|
|
@@ -18,6 +18,7 @@ var browserNodeEn = {
|
|
|
18
18
|
actions: {
|
|
19
19
|
back: "Back",
|
|
20
20
|
forward: "Forward",
|
|
21
|
+
openExternal: "Open in browser",
|
|
21
22
|
reload: "Reload"
|
|
22
23
|
},
|
|
23
24
|
addressLabel: "Address",
|
|
@@ -37,6 +38,7 @@ var browserNodeZhCN = {
|
|
|
37
38
|
actions: {
|
|
38
39
|
back: "\u540E\u9000",
|
|
39
40
|
forward: "\u524D\u8FDB",
|
|
41
|
+
openExternal: "\u5728\u6D4F\u89C8\u5668\u4E2D\u6253\u5F00",
|
|
40
42
|
reload: "\u91CD\u65B0\u52A0\u8F7D"
|
|
41
43
|
},
|
|
42
44
|
addressLabel: "\u5730\u5740",
|
|
@@ -80,4 +82,4 @@ export {
|
|
|
80
82
|
browserNodeI18nResources,
|
|
81
83
|
createBrowserNodeI18nRuntime
|
|
82
84
|
};
|
|
83
|
-
//# sourceMappingURL=chunk-
|
|
85
|
+
//# sourceMappingURL=chunk-G3H2RFWQ.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/i18n/browserNodeI18n.ts"],"sourcesContent":["import {\n createI18nRuntime,\n createScopedI18nRuntime,\n createScopedLocaleObjectsI18nModuleManifest,\n type I18nDictionary,\n type I18nRuntime\n} from \"@nextop-os/ui-i18n-runtime\";\n\ntype BrowserNodeI18nLocale = \"en\" | \"zh-CN\";\nexport const browserNodeI18nNamespace = \"browserNode\";\nexport const browserNodeI18nModule =\n createScopedLocaleObjectsI18nModuleManifest({\n localeObjectByLocale: {\n en: \"browserNodeEn\",\n \"zh-CN\": \"browserNodeZhCN\"\n },\n name: \"browser-node\",\n namespace: \"browserNode\",\n sourceRoot: \"packages/browser/workbench-node/src\"\n });\n\nconst browserNodeEn = {\n actions: {\n back: \"Back\",\n forward: \"Forward\",\n openExternal: \"Open in browser\",\n reload: \"Reload\"\n },\n addressLabel: \"Address\",\n addressPlaceholder: \"Search or enter address\",\n coldStatus: \"Sleeping\",\n dockLabel: \"Browser\",\n errors: {\n invalidUrl: \"Enter a valid web address.\",\n navigationFailed: \"The page could not be loaded.\",\n unsupportedProtocol: \"This address type is not supported.\",\n unsupportedUrl: \"This page cannot be opened.\"\n },\n loadFailed: \"Page load failed\",\n title: \"Browser\"\n} as const satisfies I18nDictionary;\n\nconst browserNodeZhCN = {\n actions: {\n back: \"后退\",\n forward: \"前进\",\n openExternal: \"在浏览器中打开\",\n reload: \"重新加载\"\n },\n addressLabel: \"地址\",\n addressPlaceholder: \"搜索或输入地址\",\n coldStatus: \"休眠中\",\n dockLabel: \"浏览器\",\n errors: {\n invalidUrl: \"请输入有效的网址。\",\n navigationFailed: \"页面无法加载。\",\n unsupportedProtocol: \"不支持此地址类型。\",\n unsupportedUrl: \"无法打开此页面。\"\n },\n loadFailed: \"页面加载失败\",\n title: \"浏览器\"\n} as const satisfies I18nDictionary;\n\nexport type BrowserNodeI18nKey =\n | \"actions.back\"\n | \"actions.forward\"\n | \"actions.openExternal\"\n | \"actions.reload\"\n | \"addressLabel\"\n | \"addressPlaceholder\"\n | \"coldStatus\"\n | \"dockLabel\"\n | \"errors.invalidUrl\"\n | \"errors.navigationFailed\"\n | \"errors.unsupportedProtocol\"\n | \"errors.unsupportedUrl\"\n | \"loadFailed\"\n | \"title\";\n\nexport type BrowserNodeI18nRuntime = I18nRuntime<BrowserNodeI18nKey>;\n\nconst browserNodeDefaults: Record<BrowserNodeI18nLocale, I18nDictionary> = {\n en: browserNodeEn,\n \"zh-CN\": browserNodeZhCN\n};\n\nexport const browserNodeI18nResources = {\n en: {\n [browserNodeI18nNamespace]: browserNodeDefaults.en\n },\n \"zh-CN\": {\n [browserNodeI18nNamespace]: browserNodeDefaults[\"zh-CN\"]\n }\n} as const satisfies Record<BrowserNodeI18nLocale, I18nDictionary>;\n\nconst defaultBrowserNodeI18n = createI18nRuntime({\n dictionaries: [browserNodeI18nResources.en]\n});\n\nexport function createBrowserNodeI18nRuntime(\n runtime: I18nRuntime<string> | undefined\n): BrowserNodeI18nRuntime {\n return createScopedI18nRuntime<BrowserNodeI18nKey>(\n runtime ?? defaultBrowserNodeI18n,\n browserNodeI18nNamespace\n );\n}\n"],"mappings":";AAAA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,OAGK;AAGA,IAAM,2BAA2B;AACjC,IAAM,wBACX,4CAA4C;AAAA,EAC1C,sBAAsB;AAAA,IACpB,IAAI;AAAA,IACJ,SAAS;AAAA,EACX;AAAA,EACA,MAAM;AAAA,EACN,WAAW;AAAA,EACX,YAAY;AACd,CAAC;AAEH,IAAM,gBAAgB;AAAA,EACpB,SAAS;AAAA,IACP,MAAM;AAAA,IACN,SAAS;AAAA,IACT,cAAc;AAAA,IACd,QAAQ;AAAA,EACV;AAAA,EACA,cAAc;AAAA,EACd,oBAAoB;AAAA,EACpB,YAAY;AAAA,EACZ,WAAW;AAAA,EACX,QAAQ;AAAA,IACN,YAAY;AAAA,IACZ,kBAAkB;AAAA,IAClB,qBAAqB;AAAA,IACrB,gBAAgB;AAAA,EAClB;AAAA,EACA,YAAY;AAAA,EACZ,OAAO;AACT;AAEA,IAAM,kBAAkB;AAAA,EACtB,SAAS;AAAA,IACP,MAAM;AAAA,IACN,SAAS;AAAA,IACT,cAAc;AAAA,IACd,QAAQ;AAAA,EACV;AAAA,EACA,cAAc;AAAA,EACd,oBAAoB;AAAA,EACpB,YAAY;AAAA,EACZ,WAAW;AAAA,EACX,QAAQ;AAAA,IACN,YAAY;AAAA,IACZ,kBAAkB;AAAA,IAClB,qBAAqB;AAAA,IACrB,gBAAgB;AAAA,EAClB;AAAA,EACA,YAAY;AAAA,EACZ,OAAO;AACT;AAoBA,IAAM,sBAAqE;AAAA,EACzE,IAAI;AAAA,EACJ,SAAS;AACX;AAEO,IAAM,2BAA2B;AAAA,EACtC,IAAI;AAAA,IACF,CAAC,wBAAwB,GAAG,oBAAoB;AAAA,EAClD;AAAA,EACA,SAAS;AAAA,IACP,CAAC,wBAAwB,GAAG,oBAAoB,OAAO;AAAA,EACzD;AACF;AAEA,IAAM,yBAAyB,kBAAkB;AAAA,EAC/C,cAAc,CAAC,yBAAyB,EAAE;AAC5C,CAAC;AAEM,SAAS,6BACd,SACwB;AACxB,SAAO;AAAA,IACL,WAAW;AAAA,IACX;AAAA,EACF;AACF;","names":[]}
|
|
@@ -63,7 +63,10 @@ function createBrowserNodeControllerEntry(context) {
|
|
|
63
63
|
connectedRelease: null,
|
|
64
64
|
controller: null,
|
|
65
65
|
context,
|
|
66
|
-
lastColdActivationUrl:
|
|
66
|
+
lastColdActivationUrl: resolveInitialLastColdActivationUrl(
|
|
67
|
+
runtime,
|
|
68
|
+
context.defaultUrl
|
|
69
|
+
),
|
|
67
70
|
listeners: /* @__PURE__ */ new Set(),
|
|
68
71
|
pendingColdActivationUrl: null,
|
|
69
72
|
refCount: 0,
|
|
@@ -171,6 +174,15 @@ function createBrowserNodeControllerEntry(context) {
|
|
|
171
174
|
};
|
|
172
175
|
return entry;
|
|
173
176
|
}
|
|
177
|
+
function resolveInitialLastColdActivationUrl(runtime, defaultUrl) {
|
|
178
|
+
const trimmedUrl = defaultUrl.trim();
|
|
179
|
+
if (runtime.lifecycle === "cold" || runtime.error !== null || trimmedUrl.length === 0 || trimmedUrl === "about:blank") {
|
|
180
|
+
return null;
|
|
181
|
+
}
|
|
182
|
+
const comparableDefaultUrl = normalizeBrowserComparableUrl(trimmedUrl);
|
|
183
|
+
const comparableRuntimeUrl = runtime.url ? normalizeBrowserComparableUrl(runtime.url) : null;
|
|
184
|
+
return comparableDefaultUrl !== null && comparableDefaultUrl === comparableRuntimeUrl ? trimmedUrl : null;
|
|
185
|
+
}
|
|
174
186
|
function notifyBrowserNodeControllerListeners(entry) {
|
|
175
187
|
for (const listener of entry.listeners) {
|
|
176
188
|
listener();
|
|
@@ -642,6 +654,7 @@ function BrowserNode({
|
|
|
642
654
|
});
|
|
643
655
|
const runtime = state.runtime;
|
|
644
656
|
const errorMessage = runtime.error ? formatBrowserNodeErrorMessage(feature, runtime.error) : null;
|
|
657
|
+
const openExternalUrl = feature.hostApi.openExternal ? feature.resolveAddressInput(state.displayUrl).url : null;
|
|
645
658
|
const {
|
|
646
659
|
shouldRenderWebview,
|
|
647
660
|
setWebviewRef,
|
|
@@ -674,6 +687,9 @@ function BrowserNode({
|
|
|
674
687
|
onSubmitUrl: () => {
|
|
675
688
|
void controller.submitDraftUrl().catch(() => void 0);
|
|
676
689
|
},
|
|
690
|
+
onOpenExternal: openExternalUrl ? () => {
|
|
691
|
+
void feature.hostApi.openExternal?.({ url: openExternalUrl }).catch(() => void 0);
|
|
692
|
+
} : void 0,
|
|
677
693
|
onGoBack: () => {
|
|
678
694
|
void controller.goBack().catch(() => void 0);
|
|
679
695
|
},
|
|
@@ -728,6 +744,7 @@ function BrowserNodeWorkbenchHeader({
|
|
|
728
744
|
nodeId
|
|
729
745
|
});
|
|
730
746
|
const runtime = state.runtime;
|
|
747
|
+
const openExternalUrl = feature.hostApi.openExternal ? feature.resolveAddressInput(state.displayUrl).url : null;
|
|
731
748
|
return /* @__PURE__ */ jsx(
|
|
732
749
|
BrowserNodeHeader,
|
|
733
750
|
{
|
|
@@ -746,6 +763,9 @@ function BrowserNodeWorkbenchHeader({
|
|
|
746
763
|
onSubmitUrl: () => {
|
|
747
764
|
void controller.submitDraftUrl().catch(() => void 0);
|
|
748
765
|
},
|
|
766
|
+
onOpenExternal: openExternalUrl ? () => {
|
|
767
|
+
void feature.hostApi.openExternal?.({ url: openExternalUrl }).catch(() => void 0);
|
|
768
|
+
} : void 0,
|
|
749
769
|
onGoBack: () => {
|
|
750
770
|
void controller.goBack().catch(() => void 0);
|
|
751
771
|
},
|
|
@@ -774,6 +794,7 @@ function BrowserNodeHeader({
|
|
|
774
794
|
onFocusRequest,
|
|
775
795
|
onGoBack,
|
|
776
796
|
onGoForward,
|
|
797
|
+
onOpenExternal,
|
|
777
798
|
onReload,
|
|
778
799
|
onSubmitUrl,
|
|
779
800
|
withBorder = true
|
|
@@ -874,6 +895,14 @@ function BrowserNodeHeader({
|
|
|
874
895
|
]
|
|
875
896
|
}
|
|
876
897
|
),
|
|
898
|
+
onOpenExternal ? /* @__PURE__ */ jsx(
|
|
899
|
+
BrowserNodeHeaderButton,
|
|
900
|
+
{
|
|
901
|
+
label: feature.i18n.t("actions.openExternal"),
|
|
902
|
+
onClick: onOpenExternal,
|
|
903
|
+
children: /* @__PURE__ */ jsx(LaunchIcon, { className: "size-[15px]" })
|
|
904
|
+
}
|
|
905
|
+
) : null,
|
|
877
906
|
defaultActions ? /* @__PURE__ */ jsxs("div", { className: "nodrag flex shrink-0 items-center gap-1.5", children: [
|
|
878
907
|
isCold ? /* @__PURE__ */ jsx(
|
|
879
908
|
Badge,
|
|
@@ -940,4 +969,4 @@ export {
|
|
|
940
969
|
BrowserNodeWorkbenchHeader,
|
|
941
970
|
BrowserNodeHeader
|
|
942
971
|
};
|
|
943
|
-
//# sourceMappingURL=chunk-
|
|
972
|
+
//# sourceMappingURL=chunk-HBFCSUQO.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/react/BrowserNode.tsx","../src/react/useBrowserNodeController.ts","../src/core/nodeController.ts","../src/react/useBrowserNodeWebview.ts","../src/core/webviewController.ts"],"sourcesContent":["import {\n ArrowLeftIcon,\n ArrowRightIcon,\n Badge,\n Button,\n Input,\n LaunchIcon,\n LoadingIcon,\n RefreshIcon,\n cn\n} from \"@nextop-os/ui-system\";\nimport { useState } from \"react\";\nimport type { HTMLAttributes, JSX, ReactNode } from \"react\";\nimport type { BrowserNodeFeature } from \"../core/feature.ts\";\nimport type {\n BrowserNodeNavigationPolicy,\n BrowserNodeRuntimeError,\n BrowserNodeSessionMode\n} from \"../core/types.ts\";\nimport { useBrowserNodeController } from \"./useBrowserNodeController.ts\";\nimport { useBrowserNodeWebview } from \"./useBrowserNodeWebview.ts\";\n\nexport interface BrowserNodeProps {\n defaultUrl: string;\n feature: BrowserNodeFeature;\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n onFocusRequest?: () => void;\n profileId?: string | null;\n sessionMode?: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n showHeader?: boolean;\n syncDefaultUrl?: boolean;\n}\n\nexport function BrowserNode({\n defaultUrl,\n feature,\n navigationPolicy = null,\n nodeId,\n onFocusRequest,\n profileId = null,\n sessionMode = \"shared\",\n sessionPartition = null,\n showHeader = true,\n syncDefaultUrl = false\n}: BrowserNodeProps): JSX.Element {\n const { controller, state } = useBrowserNodeController({\n defaultUrl,\n feature,\n navigationPolicy,\n nodeId,\n profileId,\n sessionMode,\n sessionPartition,\n syncDefaultUrl\n });\n const runtime = state.runtime;\n const errorMessage = runtime.error\n ? formatBrowserNodeErrorMessage(feature, runtime.error)\n : null;\n const openExternalUrl = feature.hostApi.openExternal\n ? feature.resolveAddressInput(state.displayUrl).url\n : null;\n const {\n shouldRenderWebview,\n setWebviewRef,\n webviewKey,\n webviewPartition,\n webviewSrc\n } = useBrowserNodeWebview({\n feature,\n initialUrl: state.displayUrl,\n lifecycle: runtime.lifecycle,\n navigationPolicy,\n nodeId,\n onGuestInteraction: onFocusRequest,\n profileId,\n sessionMode,\n sessionPartition\n });\n\n return (\n <div className=\"flex h-full min-h-0 flex-col overflow-hidden bg-[var(--background-panel)]\">\n {showHeader ? (\n <BrowserNodeHeader\n canGoBack={runtime.canGoBack}\n canGoForward={runtime.canGoForward}\n draftUrl={state.draftUrl}\n feature={feature}\n isCold={runtime.lifecycle === \"cold\"}\n isLoading={runtime.isLoading}\n onDraftUrlChange={(nextUrl) => controller.setDraftUrl(nextUrl)}\n onFocusRequest={onFocusRequest}\n onSubmitUrl={() => {\n void controller.submitDraftUrl().catch(() => undefined);\n }}\n onOpenExternal={\n openExternalUrl\n ? () => {\n void feature.hostApi\n .openExternal?.({ url: openExternalUrl })\n .catch(() => undefined);\n }\n : undefined\n }\n onGoBack={() => {\n void controller.goBack().catch(() => undefined);\n }}\n onGoForward={() => {\n void controller.goForward().catch(() => undefined);\n }}\n onReload={() => {\n void controller.reload().catch(() => undefined);\n }}\n />\n ) : null}\n <div className=\"relative min-h-0 flex-1 overflow-hidden bg-[var(--background-panel)]\">\n {shouldRenderWebview ? (\n <webview\n key={webviewKey}\n ref={setWebviewRef}\n className=\"absolute inset-0 h-full w-full border-0 bg-[var(--background-panel)]\"\n data-browser-node-webview=\"true\"\n partition={webviewPartition}\n src={webviewSrc}\n />\n ) : null}\n {errorMessage ? (\n <div className=\"pointer-events-none absolute inset-0 z-10 flex items-center justify-end p-3 text-center\">\n <div\n className=\"max-w-[min(320px,100%)] rounded-md border border-border bg-card/95 px-3 py-2 text-sm text-card-foreground shadow-panel\"\n role=\"status\"\n aria-live=\"polite\"\n >\n <div className=\"font-medium\">{feature.i18n.t(\"loadFailed\")}</div>\n <div className=\"mt-1 text-xs text-muted-foreground\">\n {errorMessage}\n </div>\n </div>\n </div>\n ) : null}\n </div>\n </div>\n );\n}\n\nexport interface BrowserNodeWorkbenchHeaderProps {\n className?: string;\n defaultActions?: ReactNode;\n defaultUrl: string;\n dragHandleProps?: HTMLAttributes<HTMLElement>;\n feature: BrowserNodeFeature;\n nodeId: string;\n onCloseRequest?: () => void;\n onFocusRequest?: () => void;\n}\n\nexport function BrowserNodeWorkbenchHeader({\n className,\n defaultActions,\n defaultUrl,\n dragHandleProps,\n feature,\n nodeId,\n onCloseRequest,\n onFocusRequest\n}: BrowserNodeWorkbenchHeaderProps): JSX.Element {\n const { controller, state } = useBrowserNodeController({\n defaultUrl,\n feature,\n nodeId\n });\n const runtime = state.runtime;\n const openExternalUrl = feature.hostApi.openExternal\n ? feature.resolveAddressInput(state.displayUrl).url\n : null;\n\n return (\n <BrowserNodeHeader\n canGoBack={runtime.canGoBack}\n canGoForward={runtime.canGoForward}\n className={className}\n defaultActions={defaultActions}\n draftUrl={state.draftUrl}\n dragHandleProps={dragHandleProps}\n feature={feature}\n isCold={runtime.lifecycle === \"cold\"}\n isLoading={runtime.isLoading}\n onCloseRequest={onCloseRequest}\n onDraftUrlChange={(nextUrl) => controller.setDraftUrl(nextUrl)}\n onFocusRequest={onFocusRequest}\n onSubmitUrl={() => {\n void controller.submitDraftUrl().catch(() => undefined);\n }}\n onOpenExternal={\n openExternalUrl\n ? () => {\n void feature.hostApi\n .openExternal?.({ url: openExternalUrl })\n .catch(() => undefined);\n }\n : undefined\n }\n onGoBack={() => {\n void controller.goBack().catch(() => undefined);\n }}\n onGoForward={() => {\n void controller.goForward().catch(() => undefined);\n }}\n onReload={() => {\n void controller.reload().catch(() => undefined);\n }}\n withBorder={false}\n />\n );\n}\n\nexport function BrowserNodeHeader({\n canGoBack,\n canGoForward,\n className,\n defaultActions,\n draftUrl,\n dragHandleProps,\n feature,\n isCold = false,\n isLoading,\n onCloseRequest,\n onDraftUrlChange,\n onFocusRequest,\n onGoBack,\n onGoForward,\n onOpenExternal,\n onReload,\n onSubmitUrl,\n withBorder = true\n}: {\n canGoBack: boolean;\n canGoForward: boolean;\n className?: string;\n defaultActions?: ReactNode;\n draftUrl: string;\n dragHandleProps?: HTMLAttributes<HTMLElement>;\n feature: BrowserNodeFeature;\n isCold?: boolean;\n isLoading: boolean;\n onCloseRequest?: () => void;\n onDraftUrlChange: (nextUrl: string) => void;\n onFocusRequest?: () => void;\n onGoBack: () => void;\n onGoForward: () => void;\n onOpenExternal?: () => void;\n onReload: () => void;\n onSubmitUrl: () => void;\n withBorder?: boolean;\n}): JSX.Element {\n const [reloadAnimationKey, setReloadAnimationKey] = useState(0);\n\n const handleReload = (): void => {\n setReloadAnimationKey((currentKey) => currentKey + 1);\n onReload();\n };\n\n return (\n <div\n className={cn(\n \"flex h-[var(--workbench-header-height,38px)] min-h-[var(--workbench-header-height,38px)] items-center gap-2 bg-[var(--background-panel)] px-2 pl-3\",\n withBorder ? \"border-b border-border\" : null,\n className\n )}\n data-browser-node-header=\"true\"\n onDoubleClick={(event) => {\n if (\n event.target instanceof Element &&\n event.target.closest(\".nodrag\")\n ) {\n return;\n }\n event.stopPropagation();\n dragHandleProps?.onDoubleClick?.(event);\n }}\n >\n <div className=\"inline-flex items-center gap-1\">\n <BrowserNodeHeaderButton\n disabled={!canGoBack}\n label={feature.i18n.t(\"actions.back\")}\n onClick={onGoBack}\n >\n <ArrowLeftIcon className=\"size-[15px]\" />\n </BrowserNodeHeaderButton>\n <BrowserNodeHeaderButton\n disabled={!canGoForward}\n label={feature.i18n.t(\"actions.forward\")}\n onClick={onGoForward}\n >\n <ArrowRightIcon className=\"size-[15px]\" />\n </BrowserNodeHeaderButton>\n <BrowserNodeHeaderButton\n label={feature.i18n.t(\"actions.reload\")}\n onClick={handleReload}\n >\n <RefreshIcon\n key={reloadAnimationKey}\n className={cn(\n \"size-[15px]\",\n reloadAnimationKey > 0 &&\n \"motion-safe:animate-[spin_520ms_cubic-bezier(0.4,0,0.2,1)_1_reverse]\"\n )}\n />\n </BrowserNodeHeaderButton>\n </div>\n <div\n {...dragHandleProps}\n className=\"h-full w-8 shrink-0 cursor-grab active:cursor-grabbing\"\n data-browser-node-drag-gutter=\"true\"\n data-node-drag-handle=\"true\"\n aria-hidden=\"true\"\n />\n <form\n className=\"nodrag relative min-w-0 flex-1\"\n onSubmit={(event) => {\n event.preventDefault();\n event.stopPropagation();\n onSubmitUrl();\n }}\n >\n <LaunchIcon className=\"pointer-events-none absolute left-2 top-1/2 z-[1] size-4 -translate-y-1/2 text-[var(--text-tertiary)]\" />\n <Input\n aria-label={feature.i18n.t(\"addressLabel\")}\n className=\"pl-8 pr-8 focus-visible:border-input focus-visible:ring-0 focus-visible:ring-offset-0\"\n placeholder={feature.i18n.t(\"addressPlaceholder\")}\n size=\"sm\"\n value={draftUrl}\n onChange={(event) => onDraftUrlChange(event.target.value)}\n onFocus={onFocusRequest}\n />\n {isLoading ? (\n <LoadingIcon className=\"pointer-events-none absolute right-2 top-1/2 z-[1] size-4 -translate-y-1/2 animate-spin text-[var(--text-tertiary)]\" />\n ) : null}\n </form>\n {onOpenExternal ? (\n <BrowserNodeHeaderButton\n label={feature.i18n.t(\"actions.openExternal\")}\n onClick={onOpenExternal}\n >\n <LaunchIcon className=\"size-[15px]\" />\n </BrowserNodeHeaderButton>\n ) : null}\n {defaultActions ? (\n <div className=\"nodrag flex shrink-0 items-center gap-1.5\">\n {isCold ? (\n <Badge\n className=\"h-[26px] min-w-7 rounded-md text-[10px] font-semibold lowercase tracking-[0.08em]\"\n aria-label={feature.i18n.t(\"coldStatus\")}\n >\n {feature.i18n.t(\"coldStatus\")}\n </Badge>\n ) : null}\n <span\n className=\"contents\"\n onClickCapture={(event) => {\n if (\n !onCloseRequest ||\n !(event.target instanceof Element) ||\n !event.target.closest('[data-workbench-action=\"close\"]')\n ) {\n return;\n }\n onCloseRequest();\n }}\n >\n {defaultActions}\n </span>\n </div>\n ) : null}\n </div>\n );\n}\n\nfunction formatBrowserNodeErrorMessage(\n feature: BrowserNodeFeature,\n error: BrowserNodeRuntimeError\n): string {\n switch (error.code) {\n case \"invalid-url\":\n return feature.i18n.t(\"errors.invalidUrl\", error.params);\n case \"navigation-failed\":\n return feature.i18n.t(\"errors.navigationFailed\", error.params);\n case \"unsupported-protocol\":\n return feature.i18n.t(\"errors.unsupportedProtocol\", error.params);\n case \"unsupported-url\":\n return feature.i18n.t(\"errors.unsupportedUrl\", error.params);\n }\n}\n\nfunction BrowserNodeHeaderButton({\n children,\n disabled,\n label,\n onClick\n}: {\n children: ReactNode;\n disabled?: boolean;\n label: string;\n onClick: () => void;\n}) {\n return (\n <Button\n aria-label={label}\n className=\"rounded-md\"\n disabled={disabled}\n size=\"icon-sm\"\n title={label}\n type=\"button\"\n variant=\"chrome\"\n onClick={onClick}\n >\n {children}\n </Button>\n );\n}\n","import { useEffect, useMemo } from \"react\";\nimport { useExternalStoreSnapshot } from \"@nextop-os/ui-react-hooks\";\nimport type { BrowserNodeFeature } from \"../core/feature.ts\";\nimport {\n acquireBrowserNodeController,\n type BrowserNodeControllerState\n} from \"../core/nodeController.ts\";\nimport type {\n BrowserNodeNavigationPolicy,\n BrowserNodeSessionMode\n} from \"../core/types.ts\";\n\nexport function useBrowserNodeController(input: {\n defaultUrl: string;\n feature: BrowserNodeFeature;\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n profileId?: string | null;\n sessionMode?: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n syncDefaultUrl?: boolean;\n}) {\n const controller = useMemo(\n () =>\n acquireBrowserNodeController({\n defaultUrl: input.defaultUrl,\n feature: input.feature,\n navigationPolicy: input.navigationPolicy,\n nodeId: input.nodeId,\n profileId: input.profileId ?? null,\n sessionMode: input.sessionMode ?? \"shared\",\n sessionPartition: input.sessionPartition,\n syncDefaultUrl: input.syncDefaultUrl ?? false\n }),\n [\n input.defaultUrl,\n input.feature,\n input.navigationPolicy,\n input.nodeId,\n input.profileId,\n input.sessionMode,\n input.sessionPartition,\n input.syncDefaultUrl\n ]\n );\n\n useEffect(() => {\n controller.retain();\n return () => {\n controller.release();\n };\n }, [controller]);\n\n useEffect(() => {\n controller.sync();\n }, [\n controller,\n input.defaultUrl,\n input.feature,\n input.navigationPolicy,\n input.nodeId,\n input.profileId,\n input.sessionMode,\n input.sessionPartition,\n input.syncDefaultUrl\n ]);\n const state = useExternalStoreSnapshot<BrowserNodeControllerState>({\n getSnapshot() {\n return controller.getState();\n },\n subscribe(listener) {\n return controller.subscribe(listener);\n }\n });\n\n return {\n controller,\n state\n };\n}\n","import type { BrowserNodeFeature } from \"./feature.ts\";\nimport { normalizeBrowserComparableUrl } from \"./url.ts\";\nimport type {\n BrowserNodeNavigationPolicy,\n BrowserNodeRuntimeState,\n BrowserNodeSessionMode\n} from \"./types.ts\";\n\nexport interface BrowserNodeControllerState {\n displayUrl: string;\n draftUrl: string;\n runtime: BrowserNodeRuntimeState;\n}\n\nexport interface BrowserNodeController {\n getState(): BrowserNodeControllerState;\n goBack(): Promise<void>;\n goForward(): Promise<void>;\n reload(): Promise<void>;\n release(): void;\n retain(): void;\n setDraftUrl(nextUrl: string): void;\n sync(): void;\n subscribe(listener: () => void): () => void;\n submitDraftUrl(): Promise<void>;\n}\n\ninterface BrowserNodeControllerContext {\n defaultUrl: string;\n feature: BrowserNodeFeature;\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n syncDefaultUrl: boolean;\n}\n\ninterface BrowserNodeControllerEntry {\n connectedRelease: (() => void) | null;\n controller: BrowserNodeController;\n context: BrowserNodeControllerContext;\n lastColdActivationUrl: string | null;\n listeners: Set<() => void>;\n pendingColdActivationUrl: string | null;\n refCount: number;\n runtimeUnsubscribe: (() => void) | null;\n state: BrowserNodeControllerState;\n}\n\nconst controllerRegistry = new Map<string, BrowserNodeControllerEntry>();\n\nexport function acquireBrowserNodeController(input: {\n defaultUrl: string;\n feature: BrowserNodeFeature;\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n profileId?: string | null;\n sessionMode?: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n syncDefaultUrl?: boolean;\n}): BrowserNodeController {\n const existing = controllerRegistry.get(input.nodeId);\n const entry =\n existing ??\n createBrowserNodeControllerEntry({\n defaultUrl: input.defaultUrl,\n feature: input.feature,\n navigationPolicy: input.navigationPolicy,\n nodeId: input.nodeId,\n profileId: input.profileId ?? null,\n sessionMode: input.sessionMode ?? \"shared\",\n sessionPartition: input.sessionPartition,\n syncDefaultUrl: input.syncDefaultUrl ?? false\n });\n\n entry.context = {\n defaultUrl: input.defaultUrl,\n feature: input.feature,\n navigationPolicy: input.navigationPolicy,\n nodeId: input.nodeId,\n profileId: input.profileId ?? null,\n sessionMode: input.sessionMode ?? \"shared\",\n sessionPartition: input.sessionPartition,\n syncDefaultUrl: input.syncDefaultUrl ?? false\n };\n\n if (!existing) {\n controllerRegistry.set(input.nodeId, entry);\n }\n\n reconcileBrowserNodeControllerState(entry, {\n allowAutoActivate: false,\n notifyListeners: false\n });\n\n return entry.controller;\n}\n\nfunction createBrowserNodeControllerEntry(\n context: BrowserNodeControllerContext\n): BrowserNodeControllerEntry {\n const runtime = context.feature.runtimeStore.getNodeState(context.nodeId);\n const displayUrl = resolveBrowserNodeDisplayUrl(runtime, context.defaultUrl);\n const entry = {\n connectedRelease: null,\n controller: null as unknown as BrowserNodeController,\n context,\n lastColdActivationUrl: resolveInitialLastColdActivationUrl(\n runtime,\n context.defaultUrl\n ),\n listeners: new Set(),\n pendingColdActivationUrl: null,\n refCount: 0,\n runtimeUnsubscribe: null,\n state: {\n displayUrl,\n draftUrl: displayUrl,\n runtime\n }\n } as BrowserNodeControllerEntry;\n\n entry.controller = {\n getState() {\n return entry.state;\n },\n goBack() {\n return entry.context.feature.hostApi.goBack({\n nodeId: entry.context.nodeId\n });\n },\n goForward() {\n return entry.context.feature.hostApi.goForward({\n nodeId: entry.context.nodeId\n });\n },\n reload() {\n return entry.context.feature.hostApi.reload({\n nodeId: entry.context.nodeId\n });\n },\n release() {\n entry.refCount = Math.max(0, entry.refCount - 1);\n if (entry.refCount > 0) {\n return;\n }\n\n entry.connectedRelease?.();\n entry.connectedRelease = null;\n entry.runtimeUnsubscribe?.();\n entry.runtimeUnsubscribe = null;\n controllerRegistry.delete(entry.context.nodeId);\n },\n retain() {\n entry.refCount += 1;\n if (entry.refCount > 1) {\n return;\n }\n\n if (!controllerRegistry.has(entry.context.nodeId)) {\n controllerRegistry.set(entry.context.nodeId, entry);\n }\n entry.connectedRelease = entry.context.feature.connect();\n entry.runtimeUnsubscribe = entry.context.feature.runtimeStore.subscribe(\n () => {\n reconcileBrowserNodeControllerState(entry, {\n allowAutoActivate: true,\n notifyListeners: true\n });\n }\n );\n reconcileBrowserNodeControllerState(entry, {\n allowAutoActivate: true,\n notifyListeners: true\n });\n },\n setDraftUrl(nextUrl) {\n if (entry.state.draftUrl === nextUrl) {\n return;\n }\n\n entry.state = {\n ...entry.state,\n draftUrl: nextUrl\n };\n notifyBrowserNodeControllerListeners(entry);\n },\n sync() {\n reconcileBrowserNodeControllerState(entry, {\n allowAutoActivate: true,\n notifyListeners: true\n });\n },\n subscribe(listener) {\n entry.listeners.add(listener);\n return () => {\n entry.listeners.delete(listener);\n };\n },\n async submitDraftUrl() {\n const resolved = entry.context.feature.resolveAddressInput(\n entry.state.draftUrl\n );\n if (!resolved.url) {\n return;\n }\n\n if (entry.state.draftUrl !== resolved.url) {\n entry.state = {\n ...entry.state,\n draftUrl: resolved.url\n };\n notifyBrowserNodeControllerListeners(entry);\n }\n\n await entry.context.feature.hostApi.navigate({\n navigationPolicy: entry.context.navigationPolicy,\n nodeId: entry.context.nodeId,\n url: resolved.url\n });\n }\n };\n\n return entry;\n}\n\nfunction resolveInitialLastColdActivationUrl(\n runtime: BrowserNodeRuntimeState,\n defaultUrl: string\n): string | null {\n const trimmedUrl = defaultUrl.trim();\n if (\n runtime.lifecycle === \"cold\" ||\n runtime.error !== null ||\n trimmedUrl.length === 0 ||\n trimmedUrl === \"about:blank\"\n ) {\n return null;\n }\n\n const comparableDefaultUrl = normalizeBrowserComparableUrl(trimmedUrl);\n const comparableRuntimeUrl = runtime.url\n ? normalizeBrowserComparableUrl(runtime.url)\n : null;\n return comparableDefaultUrl !== null &&\n comparableDefaultUrl === comparableRuntimeUrl\n ? trimmedUrl\n : null;\n}\n\nfunction notifyBrowserNodeControllerListeners(\n entry: BrowserNodeControllerEntry\n): void {\n for (const listener of entry.listeners) {\n listener();\n }\n}\n\nfunction resolveBrowserNodeDisplayUrl(\n runtime: BrowserNodeRuntimeState,\n defaultUrl: string\n): string {\n const resolvedRuntimeUrl = runtime.url?.trim() ?? \"\";\n return resolvedRuntimeUrl.length > 0 ? resolvedRuntimeUrl : defaultUrl;\n}\n\nfunction reconcileBrowserNodeControllerState(\n entry: BrowserNodeControllerEntry,\n options: {\n allowAutoActivate: boolean;\n notifyListeners: boolean;\n }\n): void {\n const runtime = entry.context.feature.runtimeStore.getNodeState(\n entry.context.nodeId\n );\n const displayUrl = resolveBrowserNodeDisplayUrl(\n runtime,\n entry.context.defaultUrl\n );\n const nextDraftUrl =\n displayUrl !== entry.state.displayUrl ? displayUrl : entry.state.draftUrl;\n\n const changed =\n entry.state.runtime !== runtime ||\n entry.state.displayUrl !== displayUrl ||\n entry.state.draftUrl !== nextDraftUrl;\n\n if (changed) {\n entry.state = {\n displayUrl,\n draftUrl: nextDraftUrl,\n runtime\n };\n if (options.notifyListeners) {\n notifyBrowserNodeControllerListeners(entry);\n }\n }\n\n if (options.allowAutoActivate) {\n void maybeActivateBrowserNodeDefaultUrl(entry).catch(() => undefined);\n }\n}\n\nasync function maybeActivateBrowserNodeDefaultUrl(\n entry: BrowserNodeControllerEntry\n): Promise<void> {\n const {\n defaultUrl,\n feature,\n navigationPolicy,\n nodeId,\n profileId,\n sessionMode,\n sessionPartition,\n syncDefaultUrl\n } = entry.context;\n const trimmedUrl = defaultUrl.trim();\n const comparableDefaultUrl = normalizeBrowserComparableUrl(trimmedUrl);\n const comparableRuntimeUrl = entry.state.runtime.url\n ? normalizeBrowserComparableUrl(entry.state.runtime.url)\n : null;\n const shouldActivateColdNode = entry.state.runtime.lifecycle === \"cold\";\n const shouldSyncDefaultUrl =\n syncDefaultUrl &&\n entry.state.runtime.lifecycle !== \"cold\" &&\n comparableDefaultUrl !== null &&\n (comparableRuntimeUrl !== comparableDefaultUrl ||\n entry.state.runtime.error !== null) &&\n entry.lastColdActivationUrl !== trimmedUrl;\n if (\n trimmedUrl.length === 0 ||\n trimmedUrl === \"about:blank\" ||\n entry.state.runtime.isLoading ||\n entry.pendingColdActivationUrl === trimmedUrl ||\n (!shouldActivateColdNode && !shouldSyncDefaultUrl) ||\n (shouldActivateColdNode &&\n entry.state.runtime.error !== null &&\n entry.lastColdActivationUrl === trimmedUrl)\n ) {\n return;\n }\n\n entry.pendingColdActivationUrl = trimmedUrl;\n try {\n await feature.hostApi.activate({\n navigationPolicy,\n nodeId,\n profileId,\n sessionMode,\n sessionPartition,\n url: trimmedUrl\n });\n entry.lastColdActivationUrl = trimmedUrl;\n } finally {\n if (entry.pendingColdActivationUrl === trimmedUrl) {\n entry.pendingColdActivationUrl = null;\n }\n }\n}\n","import { useCallback, useEffect, useMemo } from \"react\";\nimport { useExternalStoreSnapshot } from \"@nextop-os/ui-react-hooks\";\nimport {\n acquireBrowserNodeWebviewController,\n type BrowserNodeWebviewControllerState\n} from \"../core/webviewController.ts\";\nimport type { BrowserNodeFeature } from \"../core/feature.ts\";\nimport type {\n BrowserNodeLifecycle,\n BrowserNodeNavigationPolicy,\n BrowserNodeSessionMode\n} from \"../core/types.ts\";\nimport type { BrowserNodeWebviewTag } from \"./webviewTag.ts\";\n\nexport function useBrowserNodeWebview({\n feature,\n initialUrl,\n lifecycle,\n navigationPolicy,\n nodeId,\n onGuestInteraction,\n profileId,\n sessionMode,\n sessionPartition\n}: {\n feature: BrowserNodeFeature;\n initialUrl: string;\n lifecycle: BrowserNodeLifecycle;\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n onGuestInteraction?: () => void;\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n}): {\n shouldRenderWebview: boolean;\n setWebviewRef: (element: BrowserNodeWebviewTag | null) => void;\n webviewKey: string;\n webviewPartition: string;\n webviewSrc: string;\n} {\n const controller = useMemo(\n () =>\n acquireBrowserNodeWebviewController({\n feature,\n initialUrl,\n lifecycle,\n navigationPolicy,\n nodeId,\n onGuestInteraction,\n profileId,\n sessionMode,\n sessionPartition\n }),\n [\n feature,\n initialUrl,\n lifecycle,\n navigationPolicy,\n nodeId,\n onGuestInteraction,\n profileId,\n sessionMode,\n sessionPartition\n ]\n );\n\n useEffect(() => {\n controller.retain();\n return () => {\n controller.release();\n };\n }, [controller]);\n\n useEffect(() => {\n controller.sync();\n }, [\n controller,\n initialUrl,\n lifecycle,\n navigationPolicy,\n nodeId,\n onGuestInteraction,\n profileId,\n sessionMode,\n sessionPartition\n ]);\n const state = useExternalStoreSnapshot<BrowserNodeWebviewControllerState>({\n getSnapshot() {\n return controller.getState();\n },\n subscribe(listener) {\n return controller.subscribe(listener);\n }\n });\n\n const setWebviewRef = useCallback(\n (element: BrowserNodeWebviewTag | null) => {\n controller.setWebview(element);\n },\n [controller]\n );\n\n return {\n shouldRenderWebview: state.shouldRenderWebview,\n setWebviewRef,\n webviewKey: state.webviewKey,\n webviewPartition: state.webviewPartition,\n webviewSrc: state.webviewSrc\n };\n}\n","import type { BrowserNodeFeature } from \"./feature.ts\";\nimport { resolveBrowserSessionPartition } from \"./session.ts\";\nimport type {\n BrowserNodeLifecycle,\n BrowserNodeNavigationPolicy,\n BrowserNodeSessionMode\n} from \"./types.ts\";\nimport type { BrowserNodeWebviewTag } from \"../react/webviewTag.ts\";\n\nconst browserGuestUnregisterGraceMs = 250;\nconst browserNodeInitialWebviewSrc = \"about:blank\";\n\nexport interface BrowserNodeWebviewControllerState {\n shouldRenderWebview: boolean;\n webviewKey: string;\n webviewPartition: string;\n webviewSrc: string;\n}\n\nexport interface BrowserNodeWebviewController {\n getState(): BrowserNodeWebviewControllerState;\n release(): void;\n retain(): void;\n setWebview(element: BrowserNodeWebviewTag | null): void;\n sync(): void;\n subscribe(listener: () => void): () => void;\n}\n\ninterface BrowserNodeWebviewControllerContext {\n feature: BrowserNodeFeature;\n initialUrl: string;\n lifecycle: BrowserNodeLifecycle;\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n onGuestInteraction?: () => void;\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n}\n\ninterface BrowserNodeWebviewControllerEntry {\n attachedListeners: Array<{\n event: string;\n listener: EventListener;\n }>;\n context: BrowserNodeWebviewControllerContext;\n controller: BrowserNodeWebviewController;\n listeners: Set<() => void>;\n refCount: number;\n registeredGuestId: number | null;\n registeringGuestId: number | null;\n state: BrowserNodeWebviewControllerState;\n webview: BrowserNodeWebviewTag | null;\n}\n\nconst webviewControllerRegistry = new Map<\n string,\n BrowserNodeWebviewControllerEntry\n>();\nconst pendingGuestIdsByNodeId = new Map<string, number>();\nconst pendingUnregisterTimersByNodeId = new Map<\n string,\n ReturnType<typeof globalThis.setTimeout>\n>();\n\nexport function acquireBrowserNodeWebviewController(input: {\n feature: BrowserNodeFeature;\n initialUrl: string;\n lifecycle: BrowserNodeLifecycle;\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n onGuestInteraction?: () => void;\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n}): BrowserNodeWebviewController {\n const existing = webviewControllerRegistry.get(input.nodeId);\n const entry =\n existing ??\n createBrowserNodeWebviewControllerEntry({\n feature: input.feature,\n initialUrl: input.initialUrl,\n lifecycle: input.lifecycle,\n navigationPolicy: input.navigationPolicy,\n nodeId: input.nodeId,\n onGuestInteraction: input.onGuestInteraction,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition\n });\n\n entry.context = {\n feature: input.feature,\n initialUrl: input.initialUrl,\n lifecycle: input.lifecycle,\n navigationPolicy: input.navigationPolicy,\n nodeId: input.nodeId,\n onGuestInteraction: input.onGuestInteraction,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition\n };\n if (!existing) {\n webviewControllerRegistry.set(input.nodeId, entry);\n }\n return entry.controller;\n}\n\nfunction createBrowserNodeWebviewControllerEntry(\n context: BrowserNodeWebviewControllerContext\n): BrowserNodeWebviewControllerEntry {\n const state = resolveBrowserNodeWebviewControllerState(context);\n const entry = {\n attachedListeners: [],\n context,\n controller: null as unknown as BrowserNodeWebviewController,\n listeners: new Set(),\n refCount: 0,\n registeredGuestId: null,\n registeringGuestId: null,\n state,\n webview: null\n } as BrowserNodeWebviewControllerEntry;\n\n entry.controller = {\n getState() {\n return entry.state;\n },\n release() {\n entry.refCount = Math.max(0, entry.refCount - 1);\n if (entry.refCount > 0) {\n return;\n }\n scheduleBrowserNodeGuestUnregister(entry);\n detachBrowserNodeWebview(entry);\n webviewControllerRegistry.delete(entry.context.nodeId);\n },\n retain() {\n entry.refCount += 1;\n if (entry.refCount > 1) {\n return;\n }\n reconcileBrowserNodeWebviewControllerState(entry, {\n allowHostEffects: true,\n notifyListeners: true,\n rebindWebview: true\n });\n },\n setWebview(element) {\n if (entry.webview === element) {\n return;\n }\n detachBrowserNodeWebview(entry);\n entry.webview = element;\n attachBrowserNodeWebview(entry);\n },\n sync() {\n reconcileBrowserNodeWebviewControllerState(entry, {\n allowHostEffects: true,\n notifyListeners: true,\n rebindWebview: true\n });\n },\n subscribe(listener) {\n entry.listeners.add(listener);\n return () => {\n entry.listeners.delete(listener);\n };\n }\n };\n\n return entry;\n}\n\nfunction resolveBrowserNodeWebviewControllerState(\n context: BrowserNodeWebviewControllerContext\n): BrowserNodeWebviewControllerState {\n const webviewPartition = resolveBrowserSessionPartition({\n profileId: context.profileId,\n sessionMode: context.sessionMode,\n sessionPartition: context.sessionPartition\n });\n return {\n shouldRenderWebview: context.lifecycle !== \"cold\",\n webviewKey: `${context.nodeId}:${webviewPartition}`,\n webviewPartition,\n webviewSrc: browserNodeInitialWebviewSrc\n };\n}\n\nfunction reconcileBrowserNodeWebviewControllerState(\n entry: BrowserNodeWebviewControllerEntry,\n options: {\n allowHostEffects: boolean;\n notifyListeners: boolean;\n rebindWebview: boolean;\n }\n): void {\n const nextState = resolveBrowserNodeWebviewControllerState(entry.context);\n const changed =\n entry.state.shouldRenderWebview !== nextState.shouldRenderWebview ||\n entry.state.webviewKey !== nextState.webviewKey ||\n entry.state.webviewPartition !== nextState.webviewPartition ||\n entry.state.webviewSrc !== nextState.webviewSrc;\n\n if (options.allowHostEffects) {\n if (entry.context.lifecycle === \"cold\") {\n scheduleBrowserNodeGuestUnregister(entry);\n } else {\n clearPendingBrowserNodeGuestUnregister(entry.context.nodeId);\n void entry.context.feature.hostApi\n .prepareSession({\n navigationPolicy: entry.context.navigationPolicy,\n nodeId: entry.context.nodeId,\n profileId: entry.context.profileId,\n sessionMode: entry.context.sessionMode,\n sessionPartition: entry.context.sessionPartition\n })\n .catch(() => undefined);\n }\n }\n\n if (!changed) {\n if (\n options.rebindWebview &&\n entry.webview &&\n entry.attachedListeners.length === 0\n ) {\n attachBrowserNodeWebview(entry);\n }\n return;\n }\n\n entry.state = nextState;\n detachBrowserNodeWebview(entry);\n attachBrowserNodeWebview(entry);\n if (options.notifyListeners) {\n notifyBrowserNodeWebviewControllerListeners(entry);\n }\n}\n\nfunction notifyBrowserNodeWebviewControllerListeners(\n entry: BrowserNodeWebviewControllerEntry\n): void {\n for (const listener of entry.listeners) {\n listener();\n }\n}\n\nfunction clearPendingBrowserNodeGuestUnregister(nodeId: string): void {\n const timerId = pendingUnregisterTimersByNodeId.get(nodeId);\n if (timerId !== undefined) {\n globalThis.clearTimeout(timerId);\n pendingUnregisterTimersByNodeId.delete(nodeId);\n }\n pendingGuestIdsByNodeId.delete(nodeId);\n}\n\nfunction scheduleBrowserNodeGuestUnregister(\n entry: BrowserNodeWebviewControllerEntry\n): void {\n const guestId = entry.registeredGuestId;\n const nodeId = entry.context.nodeId;\n entry.registeringGuestId = null;\n if (guestId === null) {\n clearPendingBrowserNodeGuestUnregister(nodeId);\n return;\n }\n\n entry.registeredGuestId = null;\n clearPendingBrowserNodeGuestUnregister(nodeId);\n pendingGuestIdsByNodeId.set(nodeId, guestId);\n const timerId = globalThis.setTimeout(() => {\n pendingUnregisterTimersByNodeId.delete(nodeId);\n const pendingGuestId = pendingGuestIdsByNodeId.get(nodeId);\n pendingGuestIdsByNodeId.delete(nodeId);\n if (\n typeof pendingGuestId !== \"number\" ||\n !Number.isFinite(pendingGuestId)\n ) {\n return;\n }\n\n void entry.context.feature.hostApi\n .unregisterGuest({\n nodeId: entry.context.nodeId,\n webContentsId: pendingGuestId\n })\n .catch(() => undefined);\n }, browserGuestUnregisterGraceMs);\n pendingUnregisterTimersByNodeId.set(nodeId, timerId);\n}\n\nfunction detachBrowserNodeWebview(\n entry: BrowserNodeWebviewControllerEntry\n): void {\n if (!entry.webview) {\n entry.attachedListeners = [];\n return;\n }\n for (const record of entry.attachedListeners) {\n entry.webview.removeEventListener(record.event, record.listener);\n }\n entry.attachedListeners = [];\n}\n\nfunction attachBrowserNodeWebview(\n entry: BrowserNodeWebviewControllerEntry\n): void {\n const webview = entry.webview;\n if (!webview || !entry.state.shouldRenderWebview) {\n return;\n }\n\n const registerGuest = async (): Promise<void> => {\n const guestId = webview.getWebContentsId?.();\n if (\n typeof guestId !== \"number\" ||\n !Number.isFinite(guestId) ||\n guestId <= 0 ||\n entry.registeredGuestId === guestId ||\n entry.registeringGuestId === guestId\n ) {\n return;\n }\n\n clearPendingBrowserNodeGuestUnregister(entry.context.nodeId);\n entry.registeringGuestId = guestId;\n try {\n await entry.context.feature.hostApi.registerGuest({\n navigationPolicy: entry.context.navigationPolicy,\n nodeId: entry.context.nodeId,\n profileId: entry.context.profileId,\n sessionMode: entry.context.sessionMode,\n sessionPartition: entry.context.sessionPartition,\n webContentsId: guestId\n });\n entry.registeredGuestId = guestId;\n } finally {\n if (entry.registeringGuestId === guestId) {\n entry.registeringGuestId = null;\n }\n }\n };\n\n const handleDidAttach: EventListener = () => {\n void registerGuest().catch(() => undefined);\n };\n const handleDomReady: EventListener = () => {\n void registerGuest().catch(() => undefined);\n };\n const handleGuestInteraction: EventListener = () => {\n entry.context.onGuestInteraction?.();\n };\n\n const records = [\n { event: \"did-attach\", listener: handleDidAttach },\n { event: \"dom-ready\", listener: handleDomReady },\n { event: \"focus\", listener: handleGuestInteraction },\n { event: \"ipc-message\", listener: handleGuestInteraction }\n ];\n for (const record of records) {\n webview.addEventListener(record.event, record.listener);\n }\n entry.attachedListeners = records;\n}\n"],"mappings":"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","names":["useEffect","useMemo","useExternalStoreSnapshot","useMemo","useEffect","useExternalStoreSnapshot"]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { BrowserWindow,
|
|
1
|
+
import { BrowserWindow, WebContents, WebPreferences } from 'electron';
|
|
2
2
|
|
|
3
3
|
interface BrowserNodeLoopbackPreviewTarget {
|
|
4
4
|
readonly targetUrl: string;
|
|
@@ -19,6 +19,7 @@ interface BrowserNodeLoopbackPreviewRoutingOptions {
|
|
|
19
19
|
type BrowserPreferredColorScheme = "dark" | "light";
|
|
20
20
|
interface BrowserNodeElectronLogger {
|
|
21
21
|
debug?(message: string, metadata?: Record<string, unknown>): void;
|
|
22
|
+
info?(message: string, metadata?: Record<string, unknown>): void;
|
|
22
23
|
warn?(message: string, metadata?: Record<string, unknown>): void;
|
|
23
24
|
}
|
|
24
25
|
interface BrowserGuestWebContents {
|
|
@@ -34,6 +35,7 @@ interface BrowserGuestWebContents {
|
|
|
34
35
|
capturePage?(): Promise<BrowserGuestNativeImage>;
|
|
35
36
|
getTitle(): string;
|
|
36
37
|
getURL(): string;
|
|
38
|
+
getUserAgent?(): string;
|
|
37
39
|
goBack(): void;
|
|
38
40
|
goForward(): void;
|
|
39
41
|
isDestroyed(): boolean;
|
|
@@ -42,6 +44,7 @@ interface BrowserGuestWebContents {
|
|
|
42
44
|
off(event: string, listener: (...args: unknown[]) => void): this;
|
|
43
45
|
on(event: string, listener: (...args: unknown[]) => void): this;
|
|
44
46
|
reload(): void;
|
|
47
|
+
setUserAgent?(userAgent: string): void;
|
|
45
48
|
setWindowOpenHandler?(handler: (details: {
|
|
46
49
|
url: string;
|
|
47
50
|
}) => {
|
|
@@ -70,7 +73,9 @@ interface BrowserNodeElectronMainChannels {
|
|
|
70
73
|
readonly event: string;
|
|
71
74
|
readonly goBack: string;
|
|
72
75
|
readonly goForward: string;
|
|
76
|
+
readonly guestOpenUrl?: string;
|
|
73
77
|
readonly navigate: string;
|
|
78
|
+
readonly openExternal?: string;
|
|
74
79
|
readonly prepareSession: string;
|
|
75
80
|
readonly registerGuest: string;
|
|
76
81
|
readonly reload: string;
|
|
@@ -84,6 +89,7 @@ interface RegisterBrowserNodeElectronMainInput {
|
|
|
84
89
|
readonly loopbackPreviewRouting?: BrowserNodeLoopbackPreviewRoutingOptions;
|
|
85
90
|
readonly openExternal: (url: string) => Promise<void> | void;
|
|
86
91
|
readonly registerHandler: <TPayload, TResult>(channel: string, handler: (event: unknown, payload: TPayload) => Promise<TResult> | TResult) => void;
|
|
92
|
+
readonly registerListener?: <TPayload>(channel: string, handler: (event: unknown, payload: TPayload) => void) => void;
|
|
87
93
|
readonly resolveWebContents: (input: {
|
|
88
94
|
event: unknown;
|
|
89
95
|
ownerWindow: BrowserWindow;
|
|
@@ -94,6 +100,9 @@ interface RegisterBrowserNodeElectronMainInput {
|
|
|
94
100
|
}
|
|
95
101
|
declare function registerBrowserNodeElectronMain(input: RegisterBrowserNodeElectronMainInput): void;
|
|
96
102
|
|
|
103
|
+
declare function sanitizeBrowserGuestUserAgent(userAgent: string): string;
|
|
104
|
+
declare function applyBrowserGuestUserAgent(contents: WebContents, logger?: BrowserNodeElectronLogger): void;
|
|
105
|
+
|
|
97
106
|
interface BrowserSessionPartitionAllowedOptions {
|
|
98
107
|
additionalAllowedPrefixes?: readonly string[];
|
|
99
108
|
}
|
|
@@ -126,4 +135,4 @@ interface InstallBrowserWebviewSecurityInput {
|
|
|
126
135
|
}
|
|
127
136
|
declare function installBrowserWebviewSecurity({ allowedSessionPartitions, contents, logger, onGuestAttached, openExternal, resolvePreload, shouldHandleWebview }: InstallBrowserWebviewSecurityInput): () => void;
|
|
128
137
|
|
|
129
|
-
export { type BrowserNodeElectronLogger, type BrowserNodeElectronMainChannels, type BrowserNodeLoopbackPreviewResolver, type BrowserNodeLoopbackPreviewRoutingOptions, type BrowserNodeLoopbackPreviewTarget, type BrowserNodeWebviewMatcher, type BrowserWebviewPreloadResolver, type BrowserWebviewPreloadResolverInput, type BrowserWebviewSecurityInput, type BrowserWebviewSecurityResult, type InstallBrowserWebviewSecurityInput, type RegisterBrowserNodeElectronMainInput, enforceBrowserWebviewSecurity, installBrowserWebviewSecurity, isBrowserNodeWebviewAttach, registerBrowserNodeElectronMain };
|
|
138
|
+
export { type BrowserNodeElectronLogger, type BrowserNodeElectronMainChannels, type BrowserNodeLoopbackPreviewResolver, type BrowserNodeLoopbackPreviewRoutingOptions, type BrowserNodeLoopbackPreviewTarget, type BrowserNodeWebviewMatcher, type BrowserWebviewPreloadResolver, type BrowserWebviewPreloadResolverInput, type BrowserWebviewSecurityInput, type BrowserWebviewSecurityResult, type InstallBrowserWebviewSecurityInput, type RegisterBrowserNodeElectronMainInput, applyBrowserGuestUserAgent, enforceBrowserWebviewSecurity, installBrowserWebviewSecurity, isBrowserNodeWebviewAttach, registerBrowserNodeElectronMain, sanitizeBrowserGuestUserAgent };
|
|
@@ -695,7 +695,7 @@ function createBrowserGuestManager({
|
|
|
695
695
|
return;
|
|
696
696
|
}
|
|
697
697
|
event?.preventDefault?.();
|
|
698
|
-
|
|
698
|
+
emitOpenUrlFromGuest(session, url);
|
|
699
699
|
publishState(session);
|
|
700
700
|
};
|
|
701
701
|
const records = [
|
|
@@ -733,7 +733,7 @@ function createBrowserGuestManager({
|
|
|
733
733
|
policy: session.navigationPolicy,
|
|
734
734
|
url: resolved.url
|
|
735
735
|
})) {
|
|
736
|
-
|
|
736
|
+
emitOpenUrlFromGuest(session, resolved.url);
|
|
737
737
|
publishState(session);
|
|
738
738
|
return;
|
|
739
739
|
}
|
|
@@ -759,17 +759,27 @@ function createBrowserGuestManager({
|
|
|
759
759
|
publishState(session);
|
|
760
760
|
}
|
|
761
761
|
};
|
|
762
|
-
const
|
|
762
|
+
const emitOpenUrlFromGuest = (session, url) => {
|
|
763
763
|
const resolved = resolveBrowserNavigationUrl(url);
|
|
764
764
|
if (resolved.url) {
|
|
765
|
-
|
|
766
|
-
|
|
767
|
-
|
|
768
|
-
|
|
769
|
-
|
|
770
|
-
|
|
771
|
-
|
|
765
|
+
logger?.info?.("Browser Node guest emitted open-url", {
|
|
766
|
+
nodeId: session.nodeId,
|
|
767
|
+
url: resolved.url,
|
|
768
|
+
webContentsId: session.webContentsId
|
|
769
|
+
});
|
|
770
|
+
emit({
|
|
771
|
+
reuseIfOpen: false,
|
|
772
|
+
sourceNodeId: session.nodeId,
|
|
773
|
+
type: "open-url",
|
|
774
|
+
url: resolved.url
|
|
775
|
+
});
|
|
776
|
+
return { action: "deny" };
|
|
772
777
|
}
|
|
778
|
+
void Promise.resolve(openExternal(url)).catch((error) => {
|
|
779
|
+
logger?.warn?.("Browser Node openExternal failed", {
|
|
780
|
+
error: error instanceof Error ? error.message : String(error)
|
|
781
|
+
});
|
|
782
|
+
});
|
|
773
783
|
return { action: "deny" };
|
|
774
784
|
};
|
|
775
785
|
return {
|
|
@@ -828,6 +838,7 @@ function createBrowserGuestManager({
|
|
|
828
838
|
sessionMode: session.sessionMode,
|
|
829
839
|
sessionPartition: session.sessionPartition,
|
|
830
840
|
title: contents ? contents.getTitle() : null,
|
|
841
|
+
userAgent: contents?.getUserAgent?.() ?? null,
|
|
831
842
|
webContentsDestroyed: session.contents ? session.contents.isDestroyed() : null,
|
|
832
843
|
webContentsId: session.webContentsId
|
|
833
844
|
};
|
|
@@ -846,6 +857,23 @@ function createBrowserGuestManager({
|
|
|
846
857
|
}
|
|
847
858
|
return Promise.resolve();
|
|
848
859
|
},
|
|
860
|
+
handleGuestOpenUrl(webContentsId, input) {
|
|
861
|
+
const nodeId = nodeIdByWebContentsId.get(webContentsId);
|
|
862
|
+
const session = nodeId ? sessions.get(nodeId) : null;
|
|
863
|
+
if (!session) {
|
|
864
|
+
logger?.warn?.("Browser Node ignored guest open-url request", {
|
|
865
|
+
url: input.url,
|
|
866
|
+
webContentsId
|
|
867
|
+
});
|
|
868
|
+
return;
|
|
869
|
+
}
|
|
870
|
+
logger?.info?.("Browser Node handling guest open-url request", {
|
|
871
|
+
nodeId: session.nodeId,
|
|
872
|
+
url: input.url,
|
|
873
|
+
webContentsId
|
|
874
|
+
});
|
|
875
|
+
emitOpenUrlFromGuest(session, input.url);
|
|
876
|
+
},
|
|
849
877
|
async navigate(input) {
|
|
850
878
|
const resolved = resolveBrowserNavigationUrl(input.url);
|
|
851
879
|
if (!resolved.url) {
|
|
@@ -858,6 +886,13 @@ function createBrowserGuestManager({
|
|
|
858
886
|
session.lifecycle = "active";
|
|
859
887
|
await loadDesiredUrl(session);
|
|
860
888
|
},
|
|
889
|
+
async openExternal(input) {
|
|
890
|
+
const resolved = resolveBrowserNavigationUrl(input.url);
|
|
891
|
+
if (!resolved.url) {
|
|
892
|
+
throw new Error("Browser Node rejected external URL");
|
|
893
|
+
}
|
|
894
|
+
await Promise.resolve(openExternal(resolved.url));
|
|
895
|
+
},
|
|
861
896
|
async prepareSession(input) {
|
|
862
897
|
await prepareSession?.(input);
|
|
863
898
|
getSession(input.nodeId, {
|
|
@@ -904,7 +939,14 @@ function createBrowserGuestManager({
|
|
|
904
939
|
session.webContentsId = input.webContentsId;
|
|
905
940
|
nodeIdByWebContentsId.set(input.webContentsId, input.nodeId);
|
|
906
941
|
session.lifecycle = "active";
|
|
907
|
-
|
|
942
|
+
logger?.info?.("Browser Node registered guest owner", {
|
|
943
|
+
nodeId: input.nodeId,
|
|
944
|
+
sessionPartition: session.sessionPartition,
|
|
945
|
+
webContentsId: input.webContentsId
|
|
946
|
+
});
|
|
947
|
+
contents.setWindowOpenHandler?.(
|
|
948
|
+
({ url }) => emitOpenUrlFromGuest(session, url)
|
|
949
|
+
);
|
|
908
950
|
attachGuestListeners(session);
|
|
909
951
|
await applyPreferredColorSchemeToGuest(
|
|
910
952
|
session,
|
|
@@ -1023,6 +1065,14 @@ function registerBrowserNodeElectronMain(input) {
|
|
|
1023
1065
|
input.channels.navigate,
|
|
1024
1066
|
(event, payload) => resolveManager(event).navigate(payload)
|
|
1025
1067
|
);
|
|
1068
|
+
if (input.channels.openExternal) {
|
|
1069
|
+
input.registerHandler(
|
|
1070
|
+
input.channels.openExternal,
|
|
1071
|
+
(event, payload) => resolveManager(event).openExternal(
|
|
1072
|
+
payload
|
|
1073
|
+
)
|
|
1074
|
+
);
|
|
1075
|
+
}
|
|
1026
1076
|
input.registerHandler(
|
|
1027
1077
|
input.channels.goBack,
|
|
1028
1078
|
(event, payload) => resolveManager(event).goBack(payload)
|
|
@@ -1045,6 +1095,41 @@ function registerBrowserNodeElectronMain(input) {
|
|
|
1045
1095
|
(event, payload) => resolveManager(event).debugDump(payload)
|
|
1046
1096
|
);
|
|
1047
1097
|
}
|
|
1098
|
+
if (input.channels.guestOpenUrl && input.registerListener) {
|
|
1099
|
+
input.registerListener(input.channels.guestOpenUrl, (event, payload) => {
|
|
1100
|
+
const senderId = readBrowserNodeIpcSenderId(event);
|
|
1101
|
+
const openUrlPayload = payload;
|
|
1102
|
+
if (typeof senderId !== "number" || !Number.isFinite(senderId) || !openUrlPayload || typeof openUrlPayload.url !== "string") {
|
|
1103
|
+
return;
|
|
1104
|
+
}
|
|
1105
|
+
resolveManager(event).handleGuestOpenUrl(senderId, openUrlPayload);
|
|
1106
|
+
});
|
|
1107
|
+
}
|
|
1108
|
+
}
|
|
1109
|
+
function readBrowserNodeIpcSenderId(event) {
|
|
1110
|
+
const sender = event?.sender;
|
|
1111
|
+
return typeof sender?.id === "number" ? sender.id : null;
|
|
1112
|
+
}
|
|
1113
|
+
|
|
1114
|
+
// src/electron-main/userAgent.ts
|
|
1115
|
+
var electronUserAgentTokenPattern = /\sElectron\/[^\s]+/g;
|
|
1116
|
+
function sanitizeBrowserGuestUserAgent(userAgent) {
|
|
1117
|
+
return userAgent.trim().replace(electronUserAgentTokenPattern, "").replace(/\s{2,}/g, " ");
|
|
1118
|
+
}
|
|
1119
|
+
function applyBrowserGuestUserAgent(contents, logger) {
|
|
1120
|
+
const guestContents = contents;
|
|
1121
|
+
if (typeof guestContents.getUserAgent !== "function" || typeof guestContents.setUserAgent !== "function") {
|
|
1122
|
+
return;
|
|
1123
|
+
}
|
|
1124
|
+
const currentUserAgent = guestContents.getUserAgent().trim();
|
|
1125
|
+
const nextUserAgent = sanitizeBrowserGuestUserAgent(currentUserAgent);
|
|
1126
|
+
if (!nextUserAgent || nextUserAgent === currentUserAgent) {
|
|
1127
|
+
return;
|
|
1128
|
+
}
|
|
1129
|
+
guestContents.setUserAgent(nextUserAgent);
|
|
1130
|
+
logger?.debug?.("Browser Node sanitized guest user agent", {
|
|
1131
|
+
webContentsId: contents.id ?? null
|
|
1132
|
+
});
|
|
1048
1133
|
}
|
|
1049
1134
|
|
|
1050
1135
|
// src/electron-main/webviewSecurity.ts
|
|
@@ -1141,6 +1226,7 @@ function installBrowserWebviewSecurity({
|
|
|
1141
1226
|
return;
|
|
1142
1227
|
}
|
|
1143
1228
|
pendingBrowserAttachCount -= 1;
|
|
1229
|
+
applyBrowserGuestUserAgent(guestContents, logger);
|
|
1144
1230
|
onGuestAttached?.(guestContents);
|
|
1145
1231
|
logger?.debug?.("Browser Node webview guest attached", {
|
|
1146
1232
|
guestWebContentsId: guestContents.id ?? null,
|
|
@@ -1162,9 +1248,11 @@ function installBrowserWebviewSecurity({
|
|
|
1162
1248
|
};
|
|
1163
1249
|
}
|
|
1164
1250
|
export {
|
|
1251
|
+
applyBrowserGuestUserAgent,
|
|
1165
1252
|
enforceBrowserWebviewSecurity,
|
|
1166
1253
|
installBrowserWebviewSecurity,
|
|
1167
1254
|
isBrowserNodeWebviewAttach,
|
|
1168
|
-
registerBrowserNodeElectronMain
|
|
1255
|
+
registerBrowserNodeElectronMain,
|
|
1256
|
+
sanitizeBrowserGuestUserAgent
|
|
1169
1257
|
};
|
|
1170
1258
|
//# sourceMappingURL=index.js.map
|