Building out browser example.

- Includes test case to reproduce textfield dealloc bug.
This commit is contained in:
Ryan McGrath 2021-04-14 19:00:34 -07:00
parent 80ec654d8d
commit 8a9c45af61
No known key found for this signature in database
GPG key ID: DA6CBD9233593DEA
4 changed files with 163 additions and 28 deletions

View file

@ -1,14 +1,29 @@
//! This example showcases setting up a basic application and window, setting up some views to //! This example showcases setting up a basic application and window, setting up some views to
//! work with autolayout, and some basic ways to handle colors. //! work with autolayout, and some basic ways to handle colors.
use cacao::notification_center::Dispatcher;
use cacao::webview::{WebView, WebViewConfig, WebViewDelegate}; use cacao::webview::{WebView, WebViewConfig, WebViewDelegate};
use cacao::input::{TextField, TextFieldDelegate};
use cacao::macos::{App, AppDelegate}; use cacao::macos::{App, AppDelegate};
use cacao::macos::toolbar::{Toolbar, ToolbarItem, ItemIdentifier, ToolbarDelegate};
use cacao::macos::menu::{Menu, MenuItem}; use cacao::macos::menu::{Menu, MenuItem};
use cacao::macos::window::{Window, WindowConfig, WindowDelegate}; use cacao::macos::toolbar::Toolbar;
use cacao::macos::window::{Window, WindowConfig, WindowDelegate, WindowToolbarStyle};
mod toolbar;
use toolbar::BrowserToolbar;
#[derive(Debug)]
pub enum Action {
Back,
Forwards,
Load(String)
}
impl Action {
pub fn dispatch(self) {
App::<BasicApp, Self>::dispatch_main(self);
}
}
struct BasicApp { struct BasicApp {
window: Window<AppWindow> window: Window<AppWindow>
@ -61,33 +76,21 @@ impl AppDelegate for BasicApp {
} }
} }
pub struct URLBar; impl Dispatcher for BasicApp {
type Message = Action;
impl TextFieldDelegate for URLBar { fn on_ui_message(&self, message: Self::Message) {
const NAME: &'static str = "URLBar"; let window = self.window.delegate.as_ref().unwrap();
} let webview = &window.content;
struct BrowserToolbar { match message {
url_bar: TextField<URLBar> Action::Back => { webview.go_back(); },
} Action::Forwards => { webview.go_forward(); },
Action::Load(url) => { window.load_url(&url); }
impl BrowserToolbar {
pub fn new() -> Self {
BrowserToolbar {
url_bar: TextField::with(URLBar)
} }
} }
} }
impl ToolbarDelegate for BrowserToolbar {
const NAME: &'static str = "BrowserToolbar";
fn allowed_item_identifiers(&self) -> Vec<ItemIdentifier> { vec![] }
fn default_item_identifiers(&self) -> Vec<ItemIdentifier> { vec![] }
fn item_for(&self, _identifier: &str) -> &ToolbarItem { std::unreachable!(); }
}
#[derive(Default)] #[derive(Default)]
pub struct WebViewInstance; pub struct WebViewInstance;
@ -105,6 +108,11 @@ impl AppWindow {
content: WebView::with(WebViewConfig::default(), WebViewInstance::default()) content: WebView::with(WebViewConfig::default(), WebViewInstance::default())
} }
} }
pub fn load_url(&self, url: &str) {
self.toolbar.delegate.as_ref().unwrap().set_url(url);
self.content.load_url(url);
}
} }
impl WindowDelegate for AppWindow { impl WindowDelegate for AppWindow {
@ -112,17 +120,25 @@ impl WindowDelegate for AppWindow {
fn did_load(&mut self, window: Window) { fn did_load(&mut self, window: Window) {
window.set_title("Browser Example"); window.set_title("Browser Example");
window.set_autosave_name("CacaoBrowserExample");
window.set_minimum_content_size(400., 400.); window.set_minimum_content_size(400., 400.);
window.set_toolbar(&self.toolbar); window.set_toolbar(&self.toolbar);
window.set_content_view(&self.content); window.set_content_view(&self.content);
self.content.load_url("https://www.duckduckgo.com/"); self.load_url("https://www.duckduckgo.com/");
} }
} }
fn main() { fn main() {
App::new("com.test.window", BasicApp { App::new("com.test.window", BasicApp {
window: Window::with(WindowConfig::default(), AppWindow::new()) window: Window::with({
let mut config = WindowConfig::default();
// This flag is necessary for Big Sur to use the correct toolbar style.
config.toolbar_style = WindowToolbarStyle::Expanded;
config
}, AppWindow::new())
}).run(); }).run();
} }

105
examples/browser/toolbar.rs Normal file
View file

@ -0,0 +1,105 @@
use cacao::objc::{msg_send, sel, sel_impl};
use cacao::button::Button;
use cacao::input::{TextField, TextFieldDelegate};
use cacao::macos::toolbar::{Toolbar, ToolbarDisplayMode, ToolbarItem, ItemIdentifier, ToolbarDelegate};
use super::Action;
const BACK_BUTTON: &'static str = "BackButton";
const FWDS_BUTTON: &'static str = "FwdsButton";
const URL_BAR: &'static str = "URLBar";
#[derive(Debug)]
pub struct URLBar;
impl TextFieldDelegate for URLBar {
const NAME: &'static str = "URLBar";
fn text_did_end_editing(&self, value: &str) {
Action::Load(value.to_string()).dispatch();
}
}
#[derive(Debug)]
pub struct BrowserToolbar {
back_item: ToolbarItem,
forwards_item: ToolbarItem,
url_bar: TextField<URLBar>,
url_bar_item: ToolbarItem
}
impl BrowserToolbar {
pub fn new() -> Self {
let back_button = Button::new("Back");
let mut back_item = ToolbarItem::new(BACK_BUTTON);
back_item.set_button(back_button);
back_item.set_action(|| Action::Back.dispatch());
let forwards_button = Button::new("Forwards");
let mut forwards_item = ToolbarItem::new(FWDS_BUTTON);
forwards_item.set_button(forwards_button);
forwards_item.set_action(|| Action::Forwards.dispatch());
let url_bar = TextField::with(URLBar);
let url_bar_item = ToolbarItem::new(URL_BAR);
// We cheat for now to link these, as there's no API for Toolbar yet
// to support arbitrary view types. The framework is designed to support this kind of
// cheating, though: it's not outlandish to need to just manage things yourself when it
// comes to Objective-C/AppKit sometimes.
//
// As long as we keep hold of things here and they all drop together, it's relatively safe.
url_bar.objc.with_mut(|obj| unsafe {
let _: () = msg_send![&*url_bar_item.objc, setView:&*obj];
});
BrowserToolbar {
back_item,
forwards_item,
url_bar,
url_bar_item
}
}
pub fn set_url(&self, url: &str) {
self.url_bar.set_text(url);
}
fn item_identifiers(&self) -> Vec<ItemIdentifier> {
vec![
ItemIdentifier::Custom(BACK_BUTTON),
ItemIdentifier::Custom(FWDS_BUTTON),
ItemIdentifier::Space,
ItemIdentifier::Custom(URL_BAR),
ItemIdentifier::Space
]
}
}
impl ToolbarDelegate for BrowserToolbar {
const NAME: &'static str = "BrowserToolbar";
fn did_load(&mut self, toolbar: Toolbar) {
toolbar.set_display_mode(ToolbarDisplayMode::IconOnly);
}
fn allowed_item_identifiers(&self) -> Vec<ItemIdentifier> {
self.item_identifiers()
}
fn default_item_identifiers(&self) -> Vec<ItemIdentifier> {
self.item_identifiers()
}
fn item_for(&self, identifier: &str) -> &ToolbarItem {
match identifier {
BACK_BUTTON => &self.back_item,
FWDS_BUTTON => &self.forwards_item,
URL_BAR => &self.url_bar_item,
_ => { std::unreachable!(); }
}
}
}

View file

@ -13,7 +13,7 @@ use crate::utils::load;
/// Called when editing this text field has ended (e.g. user pressed enter). /// Called when editing this text field has ended (e.g. user pressed enter).
extern "C" fn text_did_end_editing<T: TextFieldDelegate>(this: &mut Object, _: Sel, _info: id) { extern "C" fn text_did_end_editing<T: TextFieldDelegate>(this: &mut Object, _: Sel, _info: id) {
let view = load::<T>(this, TEXTFIELD_DELEGATE_PTR); let view = load::<T>(this, TEXTFIELD_DELEGATE_PTR);
let s = NSString::retain(unsafe { msg_send![this, stringValue] }); let s = NSString::from_retained(unsafe { msg_send![this, stringValue] });
view.text_did_end_editing(s.to_str()); view.text_did_end_editing(s.to_str());
} }

View file

@ -242,6 +242,20 @@ impl<T> WebView<T> {
let _: () = msg_send![&*obj, loadRequest:request]; let _: () = msg_send![&*obj, loadRequest:request];
}); });
} }
/// Go back in history, if possible.
pub fn go_back(&self) {
self.objc.with_mut(|obj| unsafe {
let _: () = msg_send![&*obj, goBack];
});
}
/// Go forward in history, if possible.
pub fn go_forward(&self) {
self.objc.with_mut(|obj| unsafe {
let _: () = msg_send![&*obj, goForward];
});
}
} }
impl<T> Layout for WebView<T> { impl<T> Layout for WebView<T> {