250 lines
8.8 KiB
Rust
250 lines
8.8 KiB
Rust
// at the top of main.rs - that will prevent the console from showing
|
|
#![windows_subsystem = "windows"]
|
|
extern crate image;
|
|
use tauri_utils::config::{Config, WindowConfig};
|
|
use wry::{
|
|
application::{
|
|
event::{Event, StartCause, WindowEvent},
|
|
event_loop::{ControlFlow, EventLoop},
|
|
menu::MenuType,
|
|
window::{Fullscreen, Window, WindowBuilder},
|
|
},
|
|
webview::WebViewBuilder,
|
|
};
|
|
|
|
// enum UserEvent {
|
|
// NewWindow(String),
|
|
// }
|
|
|
|
#[cfg(target_os = "macos")]
|
|
use wry::application::{
|
|
accelerator::{Accelerator, SysMods},
|
|
keyboard::KeyCode,
|
|
menu::{MenuBar as Menu, MenuItem, MenuItemAttributes},
|
|
platform::macos::WindowBuilderExtMacOS,
|
|
};
|
|
|
|
#[cfg(target_os = "windows")]
|
|
use wry::application::window::Icon;
|
|
|
|
#[cfg(any(target_os = "linux", target_os = "windows"))]
|
|
use wry::webview::WebContext;
|
|
|
|
fn main() -> wry::Result<()> {
|
|
#[cfg(target_os = "macos")]
|
|
let (menu_bar_menu, close_item) = {
|
|
let mut menu_bar_menu = Menu::new();
|
|
let mut first_menu = Menu::new();
|
|
first_menu.add_native_item(MenuItem::Hide);
|
|
first_menu.add_native_item(MenuItem::EnterFullScreen);
|
|
first_menu.add_native_item(MenuItem::Minimize);
|
|
first_menu.add_native_item(MenuItem::Separator);
|
|
first_menu.add_native_item(MenuItem::Copy);
|
|
first_menu.add_native_item(MenuItem::Cut);
|
|
first_menu.add_native_item(MenuItem::Paste);
|
|
first_menu.add_native_item(MenuItem::Undo);
|
|
first_menu.add_native_item(MenuItem::Redo);
|
|
first_menu.add_native_item(MenuItem::SelectAll);
|
|
first_menu.add_native_item(MenuItem::Separator);
|
|
let close_item = first_menu.add_item(
|
|
MenuItemAttributes::new("CloseWindow")
|
|
.with_accelerators(&Accelerator::new(SysMods::Cmd, KeyCode::KeyW)),
|
|
);
|
|
first_menu.add_native_item(MenuItem::Quit);
|
|
menu_bar_menu.add_submenu("App", true, first_menu);
|
|
(menu_bar_menu, close_item)
|
|
};
|
|
|
|
#[cfg(any(target_os = "linux", target_os = "windows"))]
|
|
let (
|
|
package_name,
|
|
WindowConfig {
|
|
url,
|
|
width,
|
|
height,
|
|
resizable,
|
|
fullscreen,
|
|
..
|
|
},
|
|
) = {
|
|
let (package_name, windows_config) = get_windows_config();
|
|
(
|
|
package_name
|
|
.expect("can't get package name in config file")
|
|
.to_lowercase(),
|
|
windows_config.unwrap_or_default(),
|
|
)
|
|
};
|
|
|
|
#[cfg(target_os = "macos")]
|
|
let WindowConfig {
|
|
url,
|
|
width,
|
|
height,
|
|
resizable,
|
|
transparent,
|
|
fullscreen,
|
|
..
|
|
} = get_windows_config().1.unwrap_or_default();
|
|
|
|
// let event_loop: EventLoop<UserEvent> = EventLoop::with_user_event();
|
|
// let proxy = event_loop.create_proxy();
|
|
let event_loop = EventLoop::new();
|
|
let common_window = WindowBuilder::new()
|
|
.with_title("")
|
|
.with_resizable(resizable)
|
|
.with_fullscreen(if fullscreen {
|
|
Some(Fullscreen::Borderless(None))
|
|
} else {
|
|
None
|
|
})
|
|
.with_inner_size(wry::application::dpi::LogicalSize::new(width, height));
|
|
|
|
#[cfg(target_os = "windows")]
|
|
let window = {
|
|
let mut icon_path = format!("png/{}_32.ico", package_name);
|
|
// 假如没有设置,就使用默认的即可
|
|
if !std::path::Path::new(&icon_path).exists() {
|
|
icon_path = "png/icon_32.ico".to_string();
|
|
}
|
|
let icon = load_icon(std::path::Path::new(&icon_path));
|
|
common_window
|
|
.with_decorations(true)
|
|
.with_window_icon(Some(icon))
|
|
.build(&event_loop)
|
|
.unwrap()
|
|
};
|
|
|
|
#[cfg(target_os = "linux")]
|
|
let window = common_window.build(&event_loop).unwrap();
|
|
|
|
#[cfg(target_os = "macos")]
|
|
let window = common_window
|
|
.with_fullsize_content_view(true)
|
|
.with_titlebar_buttons_hidden(false)
|
|
.with_titlebar_transparent(transparent)
|
|
.with_title_hidden(true)
|
|
.with_menu(menu_bar_menu)
|
|
.build(&event_loop)
|
|
.unwrap();
|
|
|
|
let handler = move |window: &Window, req: String| {
|
|
if req == "drag_window" {
|
|
let _ = window.drag_window();
|
|
} else if req == "fullscreen" {
|
|
let is_maximized = window.is_maximized();
|
|
window.set_maximized(!is_maximized);
|
|
} else if req.starts_with("open_browser") {
|
|
let href = req.replace("open_browser:", "");
|
|
webbrowser::open(&href).expect("no browser");
|
|
}
|
|
};
|
|
|
|
// 用于欺骗部分页面对于浏览器的强检测
|
|
|
|
#[cfg(target_os = "macos")]
|
|
let webview = {
|
|
let user_agent_string = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.1 Safari/605.1.15";
|
|
WebViewBuilder::new(window)?
|
|
.with_user_agent(user_agent_string)
|
|
.with_url(&url.to_string())?
|
|
.with_devtools(cfg!(feature = "devtools"))
|
|
.with_initialization_script(include_str!("pake.js"))
|
|
.with_ipc_handler(handler)
|
|
// .with_new_window_req_handler(move |uri: String| {
|
|
// let submitted = proxy.send_event(UserEvent::NewWindow(uri.clone())).is_ok();
|
|
// submitted
|
|
// })
|
|
.with_back_forward_navigation_gestures(true)
|
|
.build()?
|
|
};
|
|
|
|
#[cfg(any(target_os = "linux", target_os = "windows"))]
|
|
let webview = {
|
|
let home_dir = match home::home_dir() {
|
|
Some(path1) => path1,
|
|
None => panic!("Error, can't found you home dir!!"),
|
|
};
|
|
#[cfg(target_os = "windows")]
|
|
let data_dir = home_dir.join("AppData").join("Roaming").join(package_name);
|
|
#[cfg(target_os = "linux")]
|
|
let data_dir = home_dir.join(".config").join(package_name);
|
|
if !data_dir.exists() {
|
|
std::fs::create_dir(&data_dir)
|
|
.unwrap_or_else(|_| panic!("can't create dir {}", data_dir.display()));
|
|
}
|
|
let mut web_content = WebContext::new(Some(data_dir));
|
|
#[cfg(target_os = "windows")]
|
|
let user_agent_string = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/108.0.0.0 Safari/537.36";
|
|
#[cfg(target_os = "linux")]
|
|
let user_agent_string = "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/108.0.0.0 Safari/537.36";
|
|
WebViewBuilder::new(window)?
|
|
.with_user_agent(user_agent_string)
|
|
.with_url(&url.to_string())?
|
|
.with_devtools(cfg!(feature = "devtools"))
|
|
.with_initialization_script(include_str!("pake.js"))
|
|
.with_ipc_handler(handler)
|
|
.with_web_context(&mut web_content)
|
|
// .with_new_window_req_handler(move |uri: String| {
|
|
// let submitted = proxy.send_event(UserEvent::NewWindow(uri.clone())).is_ok();
|
|
// submitted
|
|
// })
|
|
.build()?
|
|
};
|
|
#[cfg(feature = "devtools")]
|
|
{
|
|
webview.open_devtools();
|
|
}
|
|
|
|
event_loop.run(move |event, _, control_flow| {
|
|
*control_flow = ControlFlow::Wait;
|
|
|
|
match event {
|
|
Event::NewEvents(StartCause::Init) => println!("Wry has started!"),
|
|
Event::WindowEvent {
|
|
event: WindowEvent::CloseRequested,
|
|
..
|
|
} => *control_flow = ControlFlow::Exit,
|
|
Event::MenuEvent {
|
|
menu_id,
|
|
origin: MenuType::MenuBar,
|
|
..
|
|
} => {
|
|
#[cfg(target_os = "macos")]
|
|
if menu_id == close_item.clone().id() {
|
|
webview.window().set_minimized(true);
|
|
}
|
|
println!("Clicked on {:?}", menu_id);
|
|
println!("Clicked on {:?}", webview.window().is_visible());
|
|
}
|
|
// Event::UserEvent(UserEvent::NewWindow(uri)) => {
|
|
// webbrowser::open(&uri).expect("no browser");
|
|
// }
|
|
_ => (),
|
|
}
|
|
});
|
|
}
|
|
|
|
fn get_windows_config() -> (Option<String>, Option<WindowConfig>) {
|
|
let config_file = include_str!("../tauri.conf.json");
|
|
let config: Config = serde_json::from_str(config_file).expect("failed to parse windows config");
|
|
(
|
|
config.package.product_name.clone(),
|
|
config.tauri.windows.first().cloned(),
|
|
)
|
|
}
|
|
|
|
#[cfg(target_os = "windows")]
|
|
fn load_icon(path: &std::path::Path) -> Icon {
|
|
let (icon_rgba, icon_width, icon_height) = {
|
|
// alternatively, you can embed the icon in the binary through `include_bytes!` macro and use `image::load_from_memory`
|
|
let image = image::open(path)
|
|
.expect("Failed to open icon path")
|
|
.into_rgba8();
|
|
let (width, height) = image.dimensions();
|
|
let rgba = image.into_raw();
|
|
(rgba, width, height)
|
|
};
|
|
Icon::from_rgba(icon_rgba, icon_width, icon_height).expect("Failed to open icon")
|
|
}
|