mirror of
https://github.com/wavetermdev/waveterm.git
synced 2024-12-21 16:38:23 +01:00
webview zoom (#1531)
This commit is contained in:
parent
2285ac99db
commit
38deb28da5
@ -2,7 +2,7 @@
|
|||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
import { BlockNodeModel } from "@/app/block/blocktypes";
|
import { BlockNodeModel } from "@/app/block/blocktypes";
|
||||||
import { getApi, getSettingsKeyAtom, openLink } from "@/app/store/global";
|
import { getApi, getBlockMetaKeyAtom, getSettingsKeyAtom, openLink } from "@/app/store/global";
|
||||||
import { getSimpleControlShiftAtom } from "@/app/store/keymodel";
|
import { getSimpleControlShiftAtom } from "@/app/store/keymodel";
|
||||||
import { ObjectService } from "@/app/store/services";
|
import { ObjectService } from "@/app/store/services";
|
||||||
import { RpcApi } from "@/app/store/wshclientapi";
|
import { RpcApi } from "@/app/store/wshclientapi";
|
||||||
@ -49,13 +49,13 @@ export class WebViewModel implements ViewModel {
|
|||||||
mediaPlaying: PrimitiveAtom<boolean>;
|
mediaPlaying: PrimitiveAtom<boolean>;
|
||||||
mediaMuted: PrimitiveAtom<boolean>;
|
mediaMuted: PrimitiveAtom<boolean>;
|
||||||
modifyExternalUrl?: (url: string) => string;
|
modifyExternalUrl?: (url: string) => string;
|
||||||
|
domReady: PrimitiveAtom<boolean>;
|
||||||
|
|
||||||
constructor(blockId: string, nodeModel: BlockNodeModel) {
|
constructor(blockId: string, nodeModel: BlockNodeModel) {
|
||||||
this.nodeModel = nodeModel;
|
this.nodeModel = nodeModel;
|
||||||
this.viewType = "web";
|
this.viewType = "web";
|
||||||
this.blockId = blockId;
|
this.blockId = blockId;
|
||||||
this.blockAtom = WOS.getWaveObjectAtom<Block>(`block:${blockId}`);
|
this.blockAtom = WOS.getWaveObjectAtom<Block>(`block:${blockId}`);
|
||||||
|
|
||||||
this.url = atom();
|
this.url = atom();
|
||||||
const defaultUrlAtom = getSettingsKeyAtom("web:defaulturl");
|
const defaultUrlAtom = getSettingsKeyAtom("web:defaulturl");
|
||||||
this.homepageUrl = atom((get) => {
|
this.homepageUrl = atom((get) => {
|
||||||
@ -71,6 +71,7 @@ export class WebViewModel implements ViewModel {
|
|||||||
this.viewName = atom("Web");
|
this.viewName = atom("Web");
|
||||||
this.urlInputRef = createRef<HTMLInputElement>();
|
this.urlInputRef = createRef<HTMLInputElement>();
|
||||||
this.webviewRef = createRef<WebviewTag>();
|
this.webviewRef = createRef<WebviewTag>();
|
||||||
|
this.domReady = atom(false);
|
||||||
|
|
||||||
this.mediaPlaying = atom(false);
|
this.mediaPlaying = atom(false);
|
||||||
this.mediaMuted = atom(false);
|
this.mediaMuted = atom(false);
|
||||||
@ -339,7 +340,7 @@ export class WebViewModel implements ViewModel {
|
|||||||
const defaultSearchAtom = getSettingsKeyAtom("web:defaultsearch");
|
const defaultSearchAtom = getSettingsKeyAtom("web:defaultsearch");
|
||||||
const searchTemplate = globalStore.get(defaultSearchAtom);
|
const searchTemplate = globalStore.get(defaultSearchAtom);
|
||||||
const nextUrl = this.ensureUrlScheme(newUrl, searchTemplate);
|
const nextUrl = this.ensureUrlScheme(newUrl, searchTemplate);
|
||||||
console.log("webview loadUrl", reason, nextUrl, "cur=", this.webviewRef?.current.getURL());
|
console.log("webview loadUrl", reason, nextUrl, "cur=", this.webviewRef.current.getURL());
|
||||||
if (!this.webviewRef.current) {
|
if (!this.webviewRef.current) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -414,7 +415,7 @@ export class WebViewModel implements ViewModel {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (checkKeyPressed(e, "Cmd:r")) {
|
if (checkKeyPressed(e, "Cmd:r")) {
|
||||||
this.webviewRef?.current?.reload();
|
this.webviewRef.current?.reload();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (checkKeyPressed(e, "Cmd:ArrowLeft")) {
|
if (checkKeyPressed(e, "Cmd:ArrowLeft")) {
|
||||||
@ -428,7 +429,61 @@ export class WebViewModel implements ViewModel {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setZoomFactor(factor: number | null) {
|
||||||
|
// null is ok (will reset to default)
|
||||||
|
if (factor != null && factor < 0.1) {
|
||||||
|
factor = 0.1;
|
||||||
|
}
|
||||||
|
if (factor != null && factor > 5) {
|
||||||
|
factor = 5;
|
||||||
|
}
|
||||||
|
const domReady = globalStore.get(this.domReady);
|
||||||
|
if (!domReady) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.webviewRef.current?.setZoomFactor(factor || 1);
|
||||||
|
RpcApi.SetMetaCommand(TabRpcClient, {
|
||||||
|
oref: WOS.makeORef("block", this.blockId),
|
||||||
|
meta: { "web:zoom": factor }, // allow null so we can remove the zoom factor here
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
getSettingsMenuItems(): ContextMenuItem[] {
|
getSettingsMenuItems(): ContextMenuItem[] {
|
||||||
|
const zoomSubMenu: ContextMenuItem[] = [];
|
||||||
|
let curZoom = 1;
|
||||||
|
if (globalStore.get(this.domReady)) {
|
||||||
|
curZoom = this.webviewRef.current?.getZoomFactor() || 1;
|
||||||
|
}
|
||||||
|
const model = this; // for the closure to work (this is getting unset)
|
||||||
|
function makeZoomFactorMenuItem(label: string, factor: number): ContextMenuItem {
|
||||||
|
return {
|
||||||
|
label: label,
|
||||||
|
type: "checkbox",
|
||||||
|
click: () => {
|
||||||
|
model.setZoomFactor(factor);
|
||||||
|
},
|
||||||
|
checked: curZoom == factor,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
zoomSubMenu.push({
|
||||||
|
label: "Reset",
|
||||||
|
click: () => {
|
||||||
|
model.setZoomFactor(null);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("25%", 0.25));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("50%", 0.5));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("70%", 0.7));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("80%", 0.8));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("90%", 0.9));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("100%", 1));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("110%", 1.1));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("120%", 1.2));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("130%", 1.3));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("150%", 1.5));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("175%", 1.75));
|
||||||
|
zoomSubMenu.push(makeZoomFactorMenuItem("200%", 2));
|
||||||
|
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
label: "Set Block Homepage",
|
label: "Set Block Homepage",
|
||||||
@ -441,6 +496,10 @@ export class WebViewModel implements ViewModel {
|
|||||||
{
|
{
|
||||||
type: "separator",
|
type: "separator",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: "Set Zoom Factor",
|
||||||
|
submenu: zoomSubMenu,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: this.webviewRef.current?.isDevToolsOpened() ? "Close DevTools" : "Open DevTools",
|
label: this.webviewRef.current?.isDevToolsOpened() ? "Close DevTools" : "Open DevTools",
|
||||||
click: () => {
|
click: () => {
|
||||||
@ -476,12 +535,13 @@ const WebView = memo(({ model, onFailLoad }: WebViewProps) => {
|
|||||||
let metaUrl = blockData?.meta?.url || defaultUrl;
|
let metaUrl = blockData?.meta?.url || defaultUrl;
|
||||||
metaUrl = model.ensureUrlScheme(metaUrl, defaultSearch);
|
metaUrl = model.ensureUrlScheme(metaUrl, defaultSearch);
|
||||||
const metaUrlRef = useRef(metaUrl);
|
const metaUrlRef = useRef(metaUrl);
|
||||||
|
const zoomFactor = useAtomValue(getBlockMetaKeyAtom(model.blockId, "web:zoom")) || 1;
|
||||||
|
|
||||||
// The initial value of the block metadata URL when the component first renders. Used to set the starting src value for the webview.
|
// The initial value of the block metadata URL when the component first renders. Used to set the starting src value for the webview.
|
||||||
const [metaUrlInitial] = useState(metaUrl);
|
const [metaUrlInitial] = useState(metaUrl);
|
||||||
|
|
||||||
const [webContentsId, setWebContentsId] = useState(null);
|
const [webContentsId, setWebContentsId] = useState(null);
|
||||||
const [domReady, setDomReady] = useState(false);
|
const domReady = useAtomValue(model.domReady);
|
||||||
|
|
||||||
const [errorText, setErrorText] = useState("");
|
const [errorText, setErrorText] = useState("");
|
||||||
|
|
||||||
@ -510,13 +570,27 @@ const WebView = memo(({ model, onFailLoad }: WebViewProps) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (model.webviewRef.current && domReady) {
|
return () => {
|
||||||
|
globalStore.set(model.domReady, false);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (model.webviewRef.current == null || !domReady) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
const wcId = model.webviewRef.current.getWebContentsId?.();
|
const wcId = model.webviewRef.current.getWebContentsId?.();
|
||||||
if (wcId) {
|
if (wcId) {
|
||||||
setWebContentsId(wcId);
|
setWebContentsId(wcId);
|
||||||
|
if (model.webviewRef.current.getZoomFactor() != zoomFactor) {
|
||||||
|
model.webviewRef.current.setZoomFactor(zoomFactor);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, [model.webviewRef.current, domReady]);
|
} catch (e) {
|
||||||
|
console.error("Failed to get webcontentsid / setzoomlevel (webview)", e);
|
||||||
|
}
|
||||||
|
}, [model.webviewRef.current, domReady, zoomFactor]);
|
||||||
|
|
||||||
// Load a new URL if the block metadata is updated.
|
// Load a new URL if the block metadata is updated.
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@ -560,7 +634,7 @@ const WebView = memo(({ model, onFailLoad }: WebViewProps) => {
|
|||||||
console.error(errorMessage);
|
console.error(errorMessage);
|
||||||
setErrorText(errorMessage);
|
setErrorText(errorMessage);
|
||||||
if (onFailLoad) {
|
if (onFailLoad) {
|
||||||
const curUrl = model.webviewRef?.current.getURL();
|
const curUrl = model.webviewRef.current.getURL();
|
||||||
onFailLoad(curUrl);
|
onFailLoad(curUrl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -573,7 +647,7 @@ const WebView = memo(({ model, onFailLoad }: WebViewProps) => {
|
|||||||
getApi().setWebviewFocus(null);
|
getApi().setWebviewFocus(null);
|
||||||
};
|
};
|
||||||
const handleDomReady = () => {
|
const handleDomReady = () => {
|
||||||
setDomReady(true);
|
globalStore.set(model.domReady, true);
|
||||||
setBgColor();
|
setBgColor();
|
||||||
};
|
};
|
||||||
const handleMediaPlaying = () => {
|
const handleMediaPlaying = () => {
|
||||||
|
1
frontend/types/gotypes.d.ts
vendored
1
frontend/types/gotypes.d.ts
vendored
@ -489,6 +489,7 @@ declare global {
|
|||||||
"term:scrollback"?: number;
|
"term:scrollback"?: number;
|
||||||
"term:vdomblockid"?: string;
|
"term:vdomblockid"?: string;
|
||||||
"term:vdomtoolbarblockid"?: string;
|
"term:vdomtoolbarblockid"?: string;
|
||||||
|
"web:zoom"?: number;
|
||||||
"vdom:*"?: boolean;
|
"vdom:*"?: boolean;
|
||||||
"vdom:initialized"?: boolean;
|
"vdom:initialized"?: boolean;
|
||||||
"vdom:correlationid"?: string;
|
"vdom:correlationid"?: string;
|
||||||
|
@ -94,6 +94,8 @@ const (
|
|||||||
MetaKey_TermVDomSubBlockId = "term:vdomblockid"
|
MetaKey_TermVDomSubBlockId = "term:vdomblockid"
|
||||||
MetaKey_TermVDomToolbarBlockId = "term:vdomtoolbarblockid"
|
MetaKey_TermVDomToolbarBlockId = "term:vdomtoolbarblockid"
|
||||||
|
|
||||||
|
MetaKey_WebZoom = "web:zoom"
|
||||||
|
|
||||||
MetaKey_VDomClear = "vdom:*"
|
MetaKey_VDomClear = "vdom:*"
|
||||||
MetaKey_VDomInitialized = "vdom:initialized"
|
MetaKey_VDomInitialized = "vdom:initialized"
|
||||||
MetaKey_VDomCorrelationId = "vdom:correlationid"
|
MetaKey_VDomCorrelationId = "vdom:correlationid"
|
||||||
|
@ -95,6 +95,8 @@ type MetaTSType struct {
|
|||||||
TermVDomSubBlockId string `json:"term:vdomblockid,omitempty"`
|
TermVDomSubBlockId string `json:"term:vdomblockid,omitempty"`
|
||||||
TermVDomToolbarBlockId string `json:"term:vdomtoolbarblockid,omitempty"`
|
TermVDomToolbarBlockId string `json:"term:vdomtoolbarblockid,omitempty"`
|
||||||
|
|
||||||
|
WebZoom float64 `json:"web:zoom,omitempty"`
|
||||||
|
|
||||||
VDomClear bool `json:"vdom:*,omitempty"`
|
VDomClear bool `json:"vdom:*,omitempty"`
|
||||||
VDomInitialized bool `json:"vdom:initialized,omitempty"`
|
VDomInitialized bool `json:"vdom:initialized,omitempty"`
|
||||||
VDomCorrelationId string `json:"vdom:correlationid,omitempty"`
|
VDomCorrelationId string `json:"vdom:correlationid,omitempty"`
|
||||||
|
Loading…
Reference in New Issue
Block a user