mirror of
https://github.com/servo/servo.git
synced 2025-06-25 01:24:37 +01:00
410 lines
13 KiB
C++
410 lines
13 KiB
C++
#include "pch.h"
|
|
#include "Servo.h"
|
|
#include <EGL/egl.h>
|
|
#include "../DefaultUrl.h"
|
|
|
|
namespace winrt::servo {
|
|
|
|
using namespace Windows::Storage;
|
|
|
|
void on_load_started() { sServo->Delegate().OnServoLoadStarted(); }
|
|
|
|
void on_load_ended() { sServo->Delegate().OnServoLoadEnded(); }
|
|
|
|
void on_history_changed(bool back, bool forward) {
|
|
sServo->Delegate().OnServoHistoryChanged(back, forward);
|
|
}
|
|
|
|
void on_shutdown_complete() { sServo->Delegate().OnServoShutdownComplete(); }
|
|
|
|
void on_title_changed(const char *title) {
|
|
sServo->Delegate().OnServoTitleChanged(char2hstring(title));
|
|
}
|
|
|
|
void on_url_changed(const char *url) {
|
|
sServo->Delegate().OnServoURLChanged(char2hstring(url));
|
|
}
|
|
|
|
void wakeup() { sServo->Delegate().WakeUp(); }
|
|
|
|
bool on_allow_navigation(const char *url) {
|
|
return sServo->Delegate().OnServoAllowNavigation(char2hstring(url));
|
|
};
|
|
|
|
void on_animating_changed(bool aAnimating) {
|
|
sServo->Delegate().OnServoAnimatingChanged(aAnimating);
|
|
}
|
|
|
|
void on_panic(const char *backtrace) {
|
|
if (sLogHandle != INVALID_HANDLE_VALUE) {
|
|
CloseHandle(sLogHandle);
|
|
sLogHandle = INVALID_HANDLE_VALUE;
|
|
}
|
|
throw hresult_error(E_FAIL, char2hstring(backtrace));
|
|
}
|
|
|
|
void on_ime_show(const char *text, int32_t x, int32_t y, int32_t width,
|
|
int32_t height) {
|
|
hstring htext = L"";
|
|
if (text != nullptr) {
|
|
htext = char2hstring(text);
|
|
}
|
|
sServo->Delegate().OnServoIMEShow(htext, x, y, width, height);
|
|
}
|
|
|
|
void on_ime_hide() { sServo->Delegate().OnServoIMEHide(); }
|
|
|
|
void set_clipboard_contents(const char *) {
|
|
// FIXME
|
|
}
|
|
|
|
const char *get_clipboard_contents() {
|
|
// FIXME
|
|
return nullptr;
|
|
}
|
|
|
|
void on_media_session_metadata(const char *title, const char *album,
|
|
const char *artist) {
|
|
return sServo->Delegate().OnServoMediaSessionMetadata(
|
|
char2hstring(title), char2hstring(album), char2hstring(artist));
|
|
}
|
|
|
|
void on_media_session_playback_state_change(
|
|
const capi::CMediaSessionPlaybackState state) {
|
|
return sServo->Delegate().OnServoMediaSessionPlaybackStateChange(state);
|
|
}
|
|
|
|
void prompt_alert(const char *message, bool trusted) {
|
|
sServo->Delegate().OnServoPromptAlert(char2hstring(message), trusted);
|
|
}
|
|
|
|
void show_context_menu(const char *title, const char *const *items_list,
|
|
uint32_t items_size) {
|
|
std::optional<hstring> opt_title = {};
|
|
if (title != nullptr) {
|
|
opt_title = char2hstring(title);
|
|
}
|
|
std::vector<winrt::hstring> items;
|
|
for (uint32_t i = 0; i < items_size; i++) {
|
|
items.push_back(char2hstring(items_list[i]));
|
|
}
|
|
sServo->Delegate().OnServoShowContextMenu(opt_title, items);
|
|
}
|
|
|
|
void on_devtools_started(Servo::DevtoolsServerState result,
|
|
const unsigned int port, const char *token) {
|
|
auto state = result == Servo::DevtoolsServerState::Started;
|
|
sServo->Delegate().OnServoDevtoolsStarted(state, port, char2hstring(token));
|
|
}
|
|
|
|
void on_log_output(const char *buffer, uint32_t buffer_length) {
|
|
OutputDebugStringA(buffer);
|
|
|
|
if (sLogHandle == INVALID_HANDLE_VALUE) {
|
|
return;
|
|
}
|
|
|
|
DWORD bytesWritten;
|
|
auto writeResult =
|
|
WriteFile(sLogHandle, buffer, buffer_length, &bytesWritten, nullptr);
|
|
|
|
if (writeResult == FALSE || bytesWritten != buffer_length)
|
|
throw std::runtime_error(
|
|
"Failed to write log message to the log file: error code " +
|
|
std::to_string(GetLastError()));
|
|
}
|
|
|
|
Servo::PromptResult prompt_ok_cancel(const char *message, bool trusted) {
|
|
return sServo->Delegate().OnServoPromptOkCancel(char2hstring(message),
|
|
trusted);
|
|
}
|
|
|
|
Servo::PromptResult prompt_yes_no(const char *message, bool trusted) {
|
|
return sServo->Delegate().OnServoPromptYesNo(char2hstring(message), trusted);
|
|
}
|
|
|
|
const char *prompt_input(const char *message, const char *default,
|
|
bool trusted) {
|
|
auto input = sServo->Delegate().OnServoPromptInput(
|
|
char2hstring(message), char2hstring(default), trusted);
|
|
if (input.has_value()) {
|
|
return *hstring2char(*input);
|
|
} else {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
Servo::Servo(std::optional<hstring> initUrl, hstring args, GLsizei width,
|
|
GLsizei height, EGLNativeWindowType eglNativeWindow, float dpi,
|
|
ServoDelegate &aDelegate)
|
|
: mWindowHeight(height), mWindowWidth(width), mDelegate(aDelegate) {
|
|
ApplicationDataContainer localSettings =
|
|
ApplicationData::Current().LocalSettings();
|
|
if (!localSettings.Containers().HasKey(L"servoUserPrefs")) {
|
|
ApplicationDataContainer container = localSettings.CreateContainer(
|
|
L"servoUserPrefs", ApplicationDataCreateDisposition::Always);
|
|
}
|
|
|
|
auto prefs = localSettings.Containers().Lookup(L"servoUserPrefs");
|
|
|
|
std::vector<capi::CPref> cprefs;
|
|
|
|
for (auto pref : prefs.Values()) {
|
|
|
|
auto key = *hstring2char(pref.Key());
|
|
auto value = pref.Value();
|
|
|
|
auto type = value.as<Windows::Foundation::IPropertyValue>().Type();
|
|
capi::CPref cpref;
|
|
cpref.key = key;
|
|
cpref.pref_type = capi::CPrefType::Missing;
|
|
cpref.value = NULL;
|
|
if (type == Windows::Foundation::PropertyType::Boolean) {
|
|
cpref.pref_type = capi::CPrefType::Bool;
|
|
auto val = unbox_value<bool>(value);
|
|
cpref.value = &val;
|
|
} else if (type == Windows::Foundation::PropertyType::String) {
|
|
cpref.pref_type = capi::CPrefType::Str;
|
|
auto val = unbox_value<hstring>(value);
|
|
cpref.value = *hstring2char(val);
|
|
} else if (type == Windows::Foundation::PropertyType::Int64) {
|
|
cpref.pref_type = capi::CPrefType::Int;
|
|
auto val = unbox_value<int64_t>(value);
|
|
cpref.value = &val;
|
|
} else if (type == Windows::Foundation::PropertyType::Double) {
|
|
cpref.pref_type = capi::CPrefType::Float;
|
|
auto val = unbox_value<double>(value);
|
|
cpref.value = &val;
|
|
} else if (type == Windows::Foundation::PropertyType::Empty) {
|
|
cpref.pref_type = capi::CPrefType::Missing;
|
|
} else {
|
|
log(L"skipping pref %s. Unknown type", key);
|
|
continue;
|
|
}
|
|
cprefs.push_back(cpref);
|
|
}
|
|
|
|
if (initUrl.has_value()) {
|
|
setNonPersistentHomepage(*initUrl, cprefs);
|
|
} else {
|
|
#ifdef OVERRIDE_DEFAULT_URL
|
|
setNonPersistentHomepage(OVERRIDE_DEFAULT_URL, cprefs);
|
|
#endif
|
|
}
|
|
|
|
capi::CPrefList prefsList = {cprefs.size(), cprefs.data()};
|
|
|
|
capi::CInitOptions o;
|
|
o.prefs = &prefsList;
|
|
o.args = *hstring2char(args);
|
|
o.width = mWindowWidth;
|
|
o.height = mWindowHeight;
|
|
o.density = dpi;
|
|
o.enable_subpixel_text_antialiasing = false;
|
|
o.native_widget = eglNativeWindow;
|
|
|
|
// Note about logs:
|
|
// By default: all modules are enabled. Only warn level-logs are displayed.
|
|
// To change the log level, add "--vslogger-level debug" to o.args.
|
|
// To only print logs from specific modules, add their names to pfilters.
|
|
// For example:
|
|
// static char *pfilters[] = {
|
|
// "servo",
|
|
// "simpleservo",
|
|
// "script::dom::bindings::error", // Show JS errors by default.
|
|
// "canvas::webgl_thread", // Show GL errors by default.
|
|
// "compositing",
|
|
// "constellation",
|
|
// };
|
|
// o.vslogger_mod_list = pfilters;
|
|
// o.vslogger_mod_size = sizeof(pfilters) / sizeof(pfilters[0]);
|
|
|
|
o.vslogger_mod_list = NULL;
|
|
o.vslogger_mod_size = 0;
|
|
|
|
sServo = this; // FIXME;
|
|
|
|
auto current = ApplicationData::Current();
|
|
auto filePath = std::wstring(current.LocalFolder().Path()) + L"\\stdout.txt";
|
|
sLogHandle =
|
|
CreateFile2(filePath.c_str(), GENERIC_WRITE, 0, CREATE_ALWAYS, nullptr);
|
|
if (sLogHandle == INVALID_HANDLE_VALUE) {
|
|
throw std::runtime_error("Failed to open the log file: error code " +
|
|
std::to_string(GetLastError()));
|
|
}
|
|
|
|
if (SetFilePointer(sLogHandle, 0, nullptr, FILE_END) ==
|
|
INVALID_SET_FILE_POINTER) {
|
|
throw std::runtime_error(
|
|
"Failed to set file pointer to the end of file: error code " +
|
|
std::to_string(GetLastError()));
|
|
}
|
|
|
|
capi::CHostCallbacks c;
|
|
c.on_load_started = &on_load_started;
|
|
c.on_load_ended = &on_load_ended;
|
|
c.on_title_changed = &on_title_changed;
|
|
c.on_url_changed = &on_url_changed;
|
|
c.on_history_changed = &on_history_changed;
|
|
c.on_animating_changed = &on_animating_changed;
|
|
c.on_shutdown_complete = &on_shutdown_complete;
|
|
c.on_allow_navigation = &on_allow_navigation;
|
|
c.on_ime_show = &on_ime_show;
|
|
c.on_ime_hide = &on_ime_hide;
|
|
c.get_clipboard_contents = &get_clipboard_contents;
|
|
c.set_clipboard_contents = &set_clipboard_contents;
|
|
c.on_media_session_metadata = &on_media_session_metadata;
|
|
c.on_media_session_playback_state_change =
|
|
&on_media_session_playback_state_change;
|
|
c.prompt_alert = &prompt_alert;
|
|
c.prompt_ok_cancel = &prompt_ok_cancel;
|
|
c.prompt_yes_no = &prompt_yes_no;
|
|
c.prompt_input = &prompt_input;
|
|
c.on_devtools_started = &on_devtools_started;
|
|
c.show_context_menu = &show_context_menu;
|
|
c.on_log_output = &on_log_output;
|
|
|
|
capi::register_panic_handler(&on_panic);
|
|
|
|
capi::init_with_egl(o, &wakeup, c);
|
|
}
|
|
|
|
Servo::~Servo() {
|
|
sServo = nullptr;
|
|
if (sLogHandle != INVALID_HANDLE_VALUE)
|
|
CloseHandle(sLogHandle);
|
|
}
|
|
|
|
Servo::PrefTuple Servo::SetFloatPref(hstring key, double val) {
|
|
auto ckey = *hstring2char(key);
|
|
capi::set_float_pref(ckey, val);
|
|
auto updatedPref = WrapPref(capi::get_pref(ckey));
|
|
SaveUserPref(updatedPref);
|
|
return updatedPref;
|
|
}
|
|
|
|
Servo::PrefTuple Servo::SetIntPref(hstring key, int64_t val) {
|
|
auto ckey = *hstring2char(key);
|
|
capi::set_int_pref(ckey, val);
|
|
auto updatedPref = WrapPref(capi::get_pref(ckey));
|
|
SaveUserPref(updatedPref);
|
|
return updatedPref;
|
|
}
|
|
|
|
Servo::PrefTuple Servo::SetBoolPref(hstring key, bool val) {
|
|
auto ckey = *hstring2char(key);
|
|
capi::set_bool_pref(ckey, val);
|
|
auto updatedPref = WrapPref(capi::get_pref(ckey));
|
|
SaveUserPref(updatedPref);
|
|
return updatedPref;
|
|
}
|
|
|
|
Servo::PrefTuple Servo::SetStringPref(hstring key, hstring val) {
|
|
auto ckey = *hstring2char(key);
|
|
auto cval = *hstring2char(val);
|
|
capi::set_str_pref(ckey, cval);
|
|
auto updatedPref = WrapPref(capi::get_pref(ckey));
|
|
SaveUserPref(updatedPref);
|
|
return updatedPref;
|
|
}
|
|
|
|
Servo::PrefTuple Servo::ResetPref(hstring key) {
|
|
auto ckey = *hstring2char(key);
|
|
capi::reset_pref(ckey);
|
|
auto updatedPref = WrapPref(capi::get_pref(ckey));
|
|
SaveUserPref(updatedPref);
|
|
return updatedPref;
|
|
}
|
|
|
|
void Servo::GoHome() {
|
|
ApplicationDataContainer localSettings =
|
|
ApplicationData::Current().LocalSettings();
|
|
auto prefs = localSettings.Containers().Lookup(L"servoUserPrefs");
|
|
auto home = unbox_value<hstring>(prefs.Values().Lookup(L"shell.homepage"));
|
|
LoadUri(home);
|
|
}
|
|
|
|
void Servo::SaveUserPref(PrefTuple pref) {
|
|
auto localSettings = ApplicationData::Current().LocalSettings();
|
|
auto values = localSettings.Containers().Lookup(L"servoUserPrefs").Values();
|
|
auto [key, val, isDefault] = pref;
|
|
if (isDefault) {
|
|
values.Remove(key);
|
|
} else {
|
|
values.Insert(key, val);
|
|
}
|
|
}
|
|
|
|
Servo::PrefTuple Servo::WrapPref(capi::CPref pref) {
|
|
winrt::Windows::Foundation::IInspectable val;
|
|
if (pref.pref_type == capi::CPrefType::Bool) {
|
|
val = box_value(*(capi::get_pref_as_bool(pref.value)));
|
|
} else if (pref.pref_type == capi::CPrefType::Int) {
|
|
val = box_value(*(capi::get_pref_as_int(pref.value)));
|
|
} else if (pref.pref_type == capi::CPrefType::Float) {
|
|
val = box_value(*(capi::get_pref_as_float(pref.value)));
|
|
} else if (pref.pref_type == capi::CPrefType::Str) {
|
|
val = box_value(char2hstring(capi::get_pref_as_str(pref.value)));
|
|
}
|
|
auto key = char2hstring(pref.key);
|
|
auto isDefault = pref.is_default;
|
|
Servo::PrefTuple t{key, val, isDefault};
|
|
return t;
|
|
}
|
|
|
|
Servo::PrefTuple Servo::GetPref(hstring key) {
|
|
auto ckey = *hstring2char(key);
|
|
return WrapPref(capi::get_pref(ckey));
|
|
}
|
|
|
|
std::vector<Servo::PrefTuple> Servo::GetPrefs() {
|
|
if (sServo == nullptr) {
|
|
return {};
|
|
}
|
|
auto prefs = capi::get_prefs();
|
|
std::vector<
|
|
std::tuple<hstring, winrt::Windows::Foundation::IInspectable, bool>>
|
|
vec;
|
|
for (auto i = 0; i < prefs.len; i++) {
|
|
auto pref = WrapPref(prefs.list[i]);
|
|
vec.push_back(pref);
|
|
}
|
|
return vec;
|
|
}
|
|
|
|
void setNonPersistentHomepage(hstring url, std::vector<capi::CPref> &cprefs) {
|
|
for (auto cpref : cprefs) {
|
|
if (strcmp(cpref.key, "shell.homepage") == 0) {
|
|
cpref.value = *hstring2char(url);
|
|
return;
|
|
}
|
|
}
|
|
capi::CPref cpref;
|
|
cpref.key = "shell.homepage";
|
|
cpref.pref_type = capi::CPrefType::Str;
|
|
cpref.value = *hstring2char(url);
|
|
cprefs.push_back(cpref);
|
|
}
|
|
|
|
winrt::hstring char2hstring(const char *c_str) {
|
|
// FIXME: any better way of doing this?
|
|
auto str = std::string(c_str);
|
|
int size_needed =
|
|
MultiByteToWideChar(CP_UTF8, 0, &str[0], (int)str.size(), NULL, 0);
|
|
std::wstring str2(size_needed, 0);
|
|
MultiByteToWideChar(CP_UTF8, 0, &str[0], (int)str.size(), &str2[0],
|
|
size_needed);
|
|
winrt::hstring str3{str2};
|
|
return str3;
|
|
}
|
|
|
|
std::unique_ptr<char *> hstring2char(hstring hstr) {
|
|
const wchar_t *wc = hstr.c_str();
|
|
size_t size = hstr.size() + 1;
|
|
char *str = new char[size];
|
|
size_t converted = 0;
|
|
wcstombs_s(&converted, str, size, wc, hstr.size());
|
|
return std::make_unique<char *>(str);
|
|
}
|
|
|
|
} // namespace winrt::servo
|