mirror of
https://github.com/lightpanda-io/browser.git
synced 2025-10-28 22:53:28 +00:00
146 lines
4.6 KiB
Zig
146 lines
4.6 KiB
Zig
// Copyright (C) 2023-2024 Lightpanda (Selecy SAS)
|
|
//
|
|
// Francis Bouvier <francis@lightpanda.io>
|
|
// Pierre Tachoire <pierre@lightpanda.io>
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
// License, or (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
const std = @import("std");
|
|
|
|
const parser = @import("netsurf");
|
|
const jsruntime = @import("jsruntime");
|
|
const Callback = jsruntime.Callback;
|
|
const CallbackArg = jsruntime.CallbackArg;
|
|
const Loop = jsruntime.Loop;
|
|
|
|
const EventTarget = @import("../dom/event_target.zig").EventTarget;
|
|
const Navigator = @import("navigator.zig").Navigator;
|
|
const History = @import("history.zig").History;
|
|
const Location = @import("location.zig").Location;
|
|
|
|
const storage = @import("../storage/storage.zig");
|
|
|
|
// https://dom.spec.whatwg.org/#interface-window-extensions
|
|
// https://html.spec.whatwg.org/multipage/nav-history-apis.html#window
|
|
pub const Window = struct {
|
|
pub const prototype = *EventTarget;
|
|
pub const mem_guarantied = true;
|
|
pub const global_type = true;
|
|
|
|
// Extend libdom event target for pure zig struct.
|
|
base: parser.EventTargetTBase = parser.EventTargetTBase{},
|
|
|
|
document: ?*parser.DocumentHTML = null,
|
|
target: []const u8,
|
|
history: History = .{},
|
|
location: Location = .{},
|
|
|
|
storageShelf: ?*storage.Shelf = null,
|
|
|
|
// store a map between internal timeouts ids and pointers to uint.
|
|
// the maximum number of possible timeouts is fixed.
|
|
timeoutid: u32 = 0,
|
|
timeoutids: [512]u64 = undefined,
|
|
|
|
navigator: Navigator,
|
|
|
|
pub fn create(target: ?[]const u8, navigator: ?Navigator) Window {
|
|
return Window{
|
|
.target = target orelse "",
|
|
.navigator = navigator orelse .{},
|
|
};
|
|
}
|
|
|
|
pub fn replaceLocation(self: *Window, loc: Location) !void {
|
|
self.location = loc;
|
|
|
|
if (self.doc != null) {
|
|
try parser.documentHTMLSetLocation(Location, self.doc.?, &self.location);
|
|
}
|
|
}
|
|
|
|
pub fn replaceDocument(self: *Window, doc: *parser.DocumentHTML) !void {
|
|
self.document = doc;
|
|
try parser.documentHTMLSetLocation(Location, doc, &self.location);
|
|
}
|
|
|
|
pub fn setStorageShelf(self: *Window, shelf: *storage.Shelf) void {
|
|
self.storageShelf = shelf;
|
|
}
|
|
|
|
pub fn get_window(self: *Window) *Window {
|
|
return self;
|
|
}
|
|
|
|
pub fn get_navigator(self: *Window) *Navigator {
|
|
return &self.navigator;
|
|
}
|
|
|
|
pub fn get_location(self: *Window) *Location {
|
|
return &self.location;
|
|
}
|
|
|
|
pub fn get_self(self: *Window) *Window {
|
|
return self;
|
|
}
|
|
|
|
pub fn get_parent(self: *Window) *Window {
|
|
return self;
|
|
}
|
|
|
|
pub fn get_document(self: *Window) ?*parser.DocumentHTML {
|
|
return self.document;
|
|
}
|
|
|
|
pub fn get_history(self: *Window) *History {
|
|
return &self.history;
|
|
}
|
|
|
|
pub fn get_name(self: *Window) []const u8 {
|
|
return self.target;
|
|
}
|
|
|
|
pub fn get_localStorage(self: *Window) !*storage.Bottle {
|
|
if (self.storageShelf == null) return parser.DOMError.NotSupported;
|
|
return &self.storageShelf.?.bucket.local;
|
|
}
|
|
|
|
pub fn get_sessionStorage(self: *Window) !*storage.Bottle {
|
|
if (self.storageShelf == null) return parser.DOMError.NotSupported;
|
|
return &self.storageShelf.?.bucket.session;
|
|
}
|
|
|
|
// TODO handle callback arguments.
|
|
pub fn _setTimeout(self: *Window, loop: *Loop, cbk: Callback, delay: ?u32) !u32 {
|
|
if (self.timeoutid >= self.timeoutids.len) return error.TooMuchTimeout;
|
|
|
|
const ddelay: u63 = delay orelse 0;
|
|
const id = loop.timeout(ddelay * std.time.ns_per_ms, cbk);
|
|
|
|
self.timeoutids[self.timeoutid] = id;
|
|
defer self.timeoutid += 1;
|
|
|
|
return self.timeoutid;
|
|
}
|
|
|
|
pub fn _clearTimeout(self: *Window, loop: *Loop, id: u32) void {
|
|
// I do would prefer return an error in this case, but it seems some JS
|
|
// uses invalid id, in particular id 0.
|
|
// So we silently ignore invalid id for now.
|
|
if (id >= self.timeoutid) return;
|
|
|
|
loop.cancel(self.timeoutids[id], null);
|
|
}
|
|
};
|