lib.rs 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. // Copyright 2019-2024 Tauri Programme within The Commons Conservancy
  2. // SPDX-License-Identifier: Apache-2.0
  3. // SPDX-License-Identifier: MIT
  4. mod cmd;
  5. #[cfg(desktop)]
  6. mod menu_plugin;
  7. #[cfg(desktop)]
  8. mod tray;
  9. use serde::Serialize;
  10. use tauri::{
  11. ipc::Channel,
  12. webview::{PageLoadEvent, WebviewWindowBuilder},
  13. App, AppHandle, Manager, RunEvent, Runtime, WebviewUrl,
  14. };
  15. use tauri_plugin_sample::{PingRequest, SampleExt};
  16. pub type SetupHook = Box<dyn FnOnce(&mut App) -> Result<(), Box<dyn std::error::Error>> + Send>;
  17. pub type OnEvent = Box<dyn FnMut(&AppHandle, RunEvent)>;
  18. #[derive(Clone, Serialize)]
  19. struct Reply {
  20. data: String,
  21. }
  22. #[cfg(target_os = "macos")]
  23. pub struct AppMenu<R: Runtime>(pub std::sync::Mutex<Option<tauri::menu::Menu<R>>>);
  24. #[cfg(desktop)]
  25. pub struct PopupMenu<R: Runtime>(tauri::menu::Menu<R>);
  26. #[cfg_attr(mobile, tauri::mobile_entry_point)]
  27. pub fn run() {
  28. run_app(tauri::Builder::default(), |_app| {})
  29. }
  30. pub fn run_app<R: Runtime, F: FnOnce(&App<R>) + Send + 'static>(
  31. builder: tauri::Builder<R>,
  32. setup: F,
  33. ) {
  34. #[allow(unused_mut)]
  35. let mut builder = builder
  36. .plugin(tauri_plugin_sample::init())
  37. .setup(move |app| {
  38. #[cfg(all(desktop, not(test)))]
  39. {
  40. let handle = app.handle();
  41. tray::create_tray(handle)?;
  42. handle.plugin(menu_plugin::init())?;
  43. }
  44. #[cfg(target_os = "macos")]
  45. app.manage(AppMenu::<R>(Default::default()));
  46. #[cfg(all(desktop, not(test)))]
  47. app.manage(PopupMenu(
  48. tauri::menu::MenuBuilder::new(app)
  49. .check("check", "Tauri is awesome!")
  50. .text("text", "Do something")
  51. .copy()
  52. .build()?,
  53. ));
  54. let mut window_builder = WebviewWindowBuilder::new(app, "main", WebviewUrl::default());
  55. #[cfg(all(desktop, not(test)))]
  56. {
  57. window_builder = window_builder
  58. .title("Tauri API Validation")
  59. .inner_size(1000., 800.)
  60. .min_inner_size(600., 400.)
  61. .content_protected(true)
  62. .menu(tauri::menu::Menu::default(app.handle())?);
  63. }
  64. let webview = window_builder.build()?;
  65. #[cfg(debug_assertions)]
  66. webview.open_devtools();
  67. let value = Some("test".to_string());
  68. let response = app.sample().ping(PingRequest {
  69. value: value.clone(),
  70. on_event: Channel::new(|event| {
  71. println!("got channel event: {:?}", event);
  72. Ok(())
  73. }),
  74. });
  75. log::info!("got response: {:?}", response);
  76. if let Ok(res) = response {
  77. assert_eq!(res.value, value);
  78. }
  79. #[cfg(desktop)]
  80. std::thread::spawn(|| {
  81. let server = match tiny_http::Server::http("localhost:3003") {
  82. Ok(s) => s,
  83. Err(e) => {
  84. eprintln!("{}", e);
  85. std::process::exit(1);
  86. }
  87. };
  88. loop {
  89. if let Ok(mut request) = server.recv() {
  90. let mut body = Vec::new();
  91. let _ = request.as_reader().read_to_end(&mut body);
  92. let response = tiny_http::Response::new(
  93. tiny_http::StatusCode(200),
  94. request.headers().to_vec(),
  95. std::io::Cursor::new(body),
  96. request.body_length(),
  97. None,
  98. );
  99. let _ = request.respond(response);
  100. }
  101. }
  102. });
  103. setup(app);
  104. Ok(())
  105. })
  106. .on_page_load(|webview, payload| {
  107. if payload.event() == PageLoadEvent::Finished {
  108. let webview_ = webview.clone();
  109. webview.listen("js-event", move |event| {
  110. println!("got js-event with message '{:?}'", event.payload());
  111. let reply = Reply {
  112. data: "something else".to_string(),
  113. };
  114. webview_
  115. .emit("rust-event", Some(reply))
  116. .expect("failed to emit");
  117. });
  118. }
  119. });
  120. #[allow(unused_mut)]
  121. let mut app = builder
  122. .invoke_handler(tauri::generate_handler![
  123. cmd::log_operation,
  124. cmd::perform_request,
  125. ])
  126. .build(tauri::tauri_build_context!())
  127. .expect("error while building tauri application");
  128. #[cfg(target_os = "macos")]
  129. app.set_activation_policy(tauri::ActivationPolicy::Regular);
  130. app.run(move |_app_handle, _event| {
  131. #[cfg(all(desktop, not(test)))]
  132. match &_event {
  133. RunEvent::ExitRequested { api, code, .. } => {
  134. // Keep the event loop running even if all windows are closed
  135. // This allow us to catch tray icon events when there is no window
  136. // if we manually requested an exit (code is Some(_)) we will let it go through
  137. if code.is_none() {
  138. api.prevent_exit();
  139. }
  140. }
  141. RunEvent::WindowEvent {
  142. event: tauri::WindowEvent::CloseRequested { api, .. },
  143. label,
  144. ..
  145. } => {
  146. println!("closing window...");
  147. // run the window destroy manually just for fun :)
  148. // usually you'd show a dialog here to ask for confirmation or whatever
  149. api.prevent_close();
  150. _app_handle
  151. .get_webview_window(label)
  152. .unwrap()
  153. .destroy()
  154. .unwrap();
  155. }
  156. _ => (),
  157. }
  158. })
  159. }
  160. #[cfg(test)]
  161. mod tests {
  162. use tauri::Manager;
  163. #[test]
  164. fn run_app() {
  165. super::run_app(tauri::test::mock_builder(), |app| {
  166. let window = app.get_webview_window("main").unwrap();
  167. std::thread::spawn(move || {
  168. std::thread::sleep(std::time::Duration::from_secs(1));
  169. window.close().unwrap();
  170. });
  171. })
  172. }
  173. }