mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-04 18:59:08 +01:00
webview preload
This commit is contained in:
parent
a73ca23229
commit
a513072ff6
@ -38,6 +38,7 @@ export default defineConfig({
|
|||||||
rollupOptions: {
|
rollupOptions: {
|
||||||
input: {
|
input: {
|
||||||
index: "emain/preload.ts",
|
index: "emain/preload.ts",
|
||||||
|
"preload-webview": "emain/preload-webview.ts",
|
||||||
},
|
},
|
||||||
output: {
|
output: {
|
||||||
format: "cjs",
|
format: "cjs",
|
||||||
|
@ -582,6 +582,41 @@ electron.ipcMain.on("open-external", (event, url) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function getUrlInSession(session: Electron.Session, url: string): Readable {
|
||||||
|
const request = electron.net.request({
|
||||||
|
url,
|
||||||
|
method: "GET",
|
||||||
|
session,
|
||||||
|
});
|
||||||
|
const readable = new Readable({
|
||||||
|
read() {}, // No-op, we'll push data manually
|
||||||
|
});
|
||||||
|
request.on("response", (response) => {
|
||||||
|
response.on("data", (chunk) => {
|
||||||
|
readable.push(chunk);
|
||||||
|
});
|
||||||
|
|
||||||
|
response.on("end", () => {
|
||||||
|
readable.push(null);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
request.on("error", (err) => {
|
||||||
|
console.error("Request error:", err);
|
||||||
|
readable.destroy(err);
|
||||||
|
});
|
||||||
|
request.end();
|
||||||
|
return readable;
|
||||||
|
}
|
||||||
|
|
||||||
|
electron.ipcMain.on("save-image", (event: electron.IpcMainEvent, payload: { src: string }) => {
|
||||||
|
console.log("save-image", payload.src);
|
||||||
|
const wc = event.sender;
|
||||||
|
if (wc == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
getUrlInSession(wc.session, payload.src);
|
||||||
|
});
|
||||||
|
|
||||||
electron.ipcMain.on("download", (event, payload) => {
|
electron.ipcMain.on("download", (event, payload) => {
|
||||||
const window = electron.BrowserWindow.fromWebContents(event.sender);
|
const window = electron.BrowserWindow.fromWebContents(event.sender);
|
||||||
const streamingUrl = getWebServerEndpoint() + "/wave/stream-file?path=" + encodeURIComponent(payload.filePath);
|
const streamingUrl = getWebServerEndpoint() + "/wave/stream-file?path=" + encodeURIComponent(payload.filePath);
|
||||||
|
@ -36,6 +36,9 @@ ipcMain.on("get-user-name", (event) => {
|
|||||||
ipcMain.on("get-host-name", (event) => {
|
ipcMain.on("get-host-name", (event) => {
|
||||||
event.returnValue = os.hostname();
|
event.returnValue = os.hostname();
|
||||||
});
|
});
|
||||||
|
ipcMain.on("get-webview-preload", (event) => {
|
||||||
|
event.returnValue = path.join(getElectronAppBasePath(), "preload", "preload-webview.cjs");
|
||||||
|
});
|
||||||
|
|
||||||
// must match golang
|
// must match golang
|
||||||
function getWaveHomeDir() {
|
function getWaveHomeDir() {
|
||||||
|
20
emain/preload-webview.ts
Normal file
20
emain/preload-webview.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
// Copyright 2024, Command Line Inc.
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
const { ipcRenderer } = require("electron");
|
||||||
|
|
||||||
|
document.addEventListener("contextmenu", (event) => {
|
||||||
|
console.log("contextmenu event", event);
|
||||||
|
if (event.target == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const targetElement = event.target as HTMLElement;
|
||||||
|
// Check if the right-click is on an image
|
||||||
|
if (targetElement.tagName === "IMG") {
|
||||||
|
const imgElem = targetElement as HTMLImageElement;
|
||||||
|
const imageUrl = imgElem.src;
|
||||||
|
ipcRenderer.send("save-image", { src: imageUrl });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log("loaded wave preload-webview.ts");
|
@ -12,6 +12,7 @@ contextBridge.exposeInMainWorld("api", {
|
|||||||
getHostName: () => ipcRenderer.sendSync("get-host-name"),
|
getHostName: () => ipcRenderer.sendSync("get-host-name"),
|
||||||
getAboutModalDetails: () => ipcRenderer.sendSync("get-about-modal-details"),
|
getAboutModalDetails: () => ipcRenderer.sendSync("get-about-modal-details"),
|
||||||
getDocsiteUrl: () => ipcRenderer.sendSync("get-docsite-url"),
|
getDocsiteUrl: () => ipcRenderer.sendSync("get-docsite-url"),
|
||||||
|
getWebviewPreload: () => ipcRenderer.sendSync("get-webview-preload"),
|
||||||
openNewWindow: () => ipcRenderer.send("open-new-window"),
|
openNewWindow: () => ipcRenderer.send("open-new-window"),
|
||||||
showContextMenu: (menu, position) => ipcRenderer.send("contextmenu-show", menu, position),
|
showContextMenu: (menu, position) => ipcRenderer.send("contextmenu-show", menu, position),
|
||||||
onContextMenuClick: (callback) => ipcRenderer.on("contextmenu-click", (_event, id) => callback(id)),
|
onContextMenuClick: (callback) => ipcRenderer.on("contextmenu-click", (_event, id) => callback(id)),
|
||||||
|
@ -16,6 +16,19 @@ import * as jotai from "jotai";
|
|||||||
import React, { memo, useEffect, useState } from "react";
|
import React, { memo, useEffect, useState } from "react";
|
||||||
import "./webview.less";
|
import "./webview.less";
|
||||||
|
|
||||||
|
let webviewPreloadUrl = null;
|
||||||
|
|
||||||
|
function getWebviewPreloadUrl() {
|
||||||
|
if (webviewPreloadUrl == null) {
|
||||||
|
webviewPreloadUrl = getApi().getWebviewPreload();
|
||||||
|
console.log("webviewPreloadUrl", webviewPreloadUrl);
|
||||||
|
}
|
||||||
|
if (webviewPreloadUrl == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return "file://" + webviewPreloadUrl;
|
||||||
|
}
|
||||||
|
|
||||||
export class WebViewModel implements ViewModel {
|
export class WebViewModel implements ViewModel {
|
||||||
viewType: string;
|
viewType: string;
|
||||||
blockId: string;
|
blockId: string;
|
||||||
@ -501,6 +514,7 @@ const WebView = memo(({ model }: WebViewProps) => {
|
|||||||
src={metaUrlInitial}
|
src={metaUrlInitial}
|
||||||
data-blockid={model.blockId}
|
data-blockid={model.blockId}
|
||||||
data-webcontentsid={webContentsId} // needed for emain
|
data-webcontentsid={webContentsId} // needed for emain
|
||||||
|
preload={getWebviewPreloadUrl()}
|
||||||
// @ts-ignore This is a discrepancy between the React typing and the Chromium impl for webviewTag. Chrome webviewTag expects a string, while React expects a boolean.
|
// @ts-ignore This is a discrepancy between the React typing and the Chromium impl for webviewTag. Chrome webviewTag expects a string, while React expects a boolean.
|
||||||
allowpopups="true"
|
allowpopups="true"
|
||||||
></webview>
|
></webview>
|
||||||
|
1
frontend/types/custom.d.ts
vendored
1
frontend/types/custom.d.ts
vendored
@ -57,6 +57,7 @@ declare global {
|
|||||||
getEnv: (varName: string) => string;
|
getEnv: (varName: string) => string;
|
||||||
getUserName: () => string;
|
getUserName: () => string;
|
||||||
getHostName: () => string;
|
getHostName: () => string;
|
||||||
|
getWebviewPreload: () => string;
|
||||||
getAboutModalDetails: () => AboutModalDetails;
|
getAboutModalDetails: () => AboutModalDetails;
|
||||||
getDocsiteUrl: () => string;
|
getDocsiteUrl: () => string;
|
||||||
showContextMenu: (menu?: ElectronContextMenuItem[]) => void;
|
showContextMenu: (menu?: ElectronContextMenuItem[]) => void;
|
||||||
|
Loading…
Reference in New Issue
Block a user