2021-03-15 17:09:50 -07:00
|
|
|
//! A progress indicator widget.
|
|
|
|
//!
|
2021-08-08 18:42:07 -07:00
|
|
|
//! This control wraps `NSProgressIndicator` in AppKit, and
|
|
|
|
//! `UIProgressView+UIActivityIndicatorView` in iOS/tvOS. It operates in two modes: determinate
|
2021-03-15 17:09:50 -07:00
|
|
|
//! (where you have a fixed start and end) and indeterminate (infinite; it will go and go until you
|
|
|
|
//! tell it to stop).
|
|
|
|
//!
|
|
|
|
//! ```rust,no_run
|
|
|
|
//! let indicator = ProgressIndicator::new();
|
|
|
|
//! indicator.set_indeterminate(true);
|
|
|
|
//! my_view.add_subview(&indicator);
|
|
|
|
//! ```
|
|
|
|
|
|
|
|
use core_graphics::base::CGFloat;
|
|
|
|
|
2021-01-19 00:11:52 -08:00
|
|
|
use objc_id::ShareId;
|
|
|
|
use objc::runtime::{Class, Object};
|
2021-03-15 17:09:50 -07:00
|
|
|
use objc::{class, msg_send, sel, sel_impl};
|
2021-01-19 00:11:52 -08:00
|
|
|
|
|
|
|
use crate::foundation::{id, nil, YES, NO, NSUInteger};
|
|
|
|
use crate::color::Color;
|
2021-08-08 18:42:07 -07:00
|
|
|
use crate::layout::Layout;
|
2021-03-26 13:29:39 -07:00
|
|
|
use crate::utils::properties::ObjcProperty;
|
2021-01-19 00:11:52 -08:00
|
|
|
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
|
|
|
use crate::layout::{LayoutAnchorX, LayoutAnchorY, LayoutAnchorDimension};
|
|
|
|
|
2021-01-19 00:11:52 -08:00
|
|
|
mod enums;
|
|
|
|
pub use enums::ProgressIndicatorStyle;
|
|
|
|
|
2021-03-15 17:09:50 -07:00
|
|
|
/// A control used for reporting progress to a user visually.
|
2021-01-19 00:11:52 -08:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct ProgressIndicator {
|
2021-03-15 17:09:50 -07:00
|
|
|
/// A pointer to the Objective-C Object.
|
2021-03-26 13:29:39 -07:00
|
|
|
pub objc: ObjcProperty,
|
2021-03-16 18:21:31 -07:00
|
|
|
|
|
|
|
/// A pointer to the Objective-C runtime top layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
pub top: LayoutAnchorY,
|
|
|
|
|
2021-03-16 18:21:31 -07:00
|
|
|
/// A pointer to the Objective-C runtime leading layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
pub leading: LayoutAnchorX,
|
|
|
|
|
2021-03-16 18:21:31 -07:00
|
|
|
/// A pointer to the Objective-C runtime left layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
pub left: LayoutAnchorX,
|
|
|
|
|
|
|
|
/// A pointer to the Objective-C runtime trailing layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
pub trailing: LayoutAnchorX,
|
|
|
|
|
2021-03-16 18:21:31 -07:00
|
|
|
/// A pointer to the Objective-C runtime right layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
pub right: LayoutAnchorX,
|
|
|
|
|
|
|
|
/// A pointer to the Objective-C runtime bottom layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
pub bottom: LayoutAnchorY,
|
|
|
|
|
2021-03-16 18:21:31 -07:00
|
|
|
/// A pointer to the Objective-C runtime width layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
pub width: LayoutAnchorDimension,
|
|
|
|
|
2021-03-16 18:21:31 -07:00
|
|
|
/// A pointer to the Objective-C runtime height layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
pub height: LayoutAnchorDimension,
|
|
|
|
|
2021-03-16 18:21:31 -07:00
|
|
|
/// A pointer to the Objective-C runtime center X layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
pub center_x: LayoutAnchorX,
|
|
|
|
|
2021-03-16 18:21:31 -07:00
|
|
|
/// A pointer to the Objective-C runtime center Y layout constraint.
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
pub center_y: LayoutAnchorY
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for ProgressIndicator {
|
|
|
|
fn default() -> Self {
|
|
|
|
ProgressIndicator::new()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ProgressIndicator {
|
2021-03-15 17:09:50 -07:00
|
|
|
/// Returns a default `ProgressIndicator`. You should retain this yourself for as long as you
|
|
|
|
/// need it to stay around.
|
2021-01-19 00:11:52 -08:00
|
|
|
pub fn new() -> Self {
|
|
|
|
let view = unsafe {
|
2021-08-07 22:31:48 -07:00
|
|
|
#[cfg(feature = "appkit")]
|
2021-03-15 17:09:50 -07:00
|
|
|
let view: id = msg_send![class!(NSProgressIndicator), new];
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-01-19 00:11:52 -08:00
|
|
|
let _: () = msg_send![view, setTranslatesAutoresizingMaskIntoConstraints:NO];
|
|
|
|
|
2021-08-07 22:31:48 -07:00
|
|
|
#[cfg(feature = "appkit")]
|
2021-01-19 00:11:52 -08:00
|
|
|
let _: () = msg_send![view, setWantsLayer:YES];
|
|
|
|
|
|
|
|
view
|
|
|
|
};
|
|
|
|
|
|
|
|
ProgressIndicator {
|
2021-08-08 18:42:07 -07:00
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
top: LayoutAnchorY::top(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
left: LayoutAnchorX::left(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
leading: LayoutAnchorX::leading(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
right: LayoutAnchorX::right(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
trailing: LayoutAnchorX::trailing(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
bottom: LayoutAnchorY::bottom(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
width: LayoutAnchorDimension::width(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
height: LayoutAnchorDimension::height(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
center_x: LayoutAnchorX::center(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
|
|
|
#[cfg(feature = "autolayout")]
|
2021-03-16 18:21:31 -07:00
|
|
|
center_y: LayoutAnchorY::center(view),
|
2021-08-08 18:42:07 -07:00
|
|
|
|
2021-03-26 13:29:39 -07:00
|
|
|
objc: ObjcProperty::retain(view),
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ProgressIndicator {
|
|
|
|
/// Starts the animation for an indeterminate indicator.
|
|
|
|
pub fn start_animation(&self) {
|
2021-03-26 13:29:39 -07:00
|
|
|
self.objc.with_mut(|obj| unsafe {
|
|
|
|
let _: () = msg_send![obj, startAnimation:nil];
|
|
|
|
});
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
|
|
|
|
2021-03-15 17:09:50 -07:00
|
|
|
/// Stops any animations that are currently happening on this indicator (e.g, if it's an
|
|
|
|
/// indeterminate looping animation).
|
2021-01-19 00:11:52 -08:00
|
|
|
pub fn stop_animation(&self) {
|
2021-03-26 13:29:39 -07:00
|
|
|
self.objc.with_mut(|obj| unsafe {
|
|
|
|
let _: () = msg_send![obj, stopAnimation:nil];
|
|
|
|
});
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
|
|
|
|
2021-03-15 17:09:50 -07:00
|
|
|
/// Increment the progress indicator by the amount specified.
|
|
|
|
pub fn increment(&self, amount: f64) {
|
2021-03-26 13:29:39 -07:00
|
|
|
self.objc.with_mut(|obj| unsafe {
|
|
|
|
let _: () = msg_send![obj, incrementBy:amount];
|
|
|
|
});
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
|
|
|
|
2021-03-15 17:09:50 -07:00
|
|
|
/// Set the style for the progress indicator.
|
2021-01-19 00:11:52 -08:00
|
|
|
pub fn set_style(&self, style: ProgressIndicatorStyle) {
|
2021-03-26 13:29:39 -07:00
|
|
|
let style = style as NSUInteger;
|
|
|
|
|
|
|
|
self.objc.with_mut(move |obj| unsafe {
|
|
|
|
let _: () = msg_send![obj, setStyle:style];
|
|
|
|
});
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
|
|
|
|
2021-03-15 17:09:50 -07:00
|
|
|
/// Set whether this is an indeterminate indicator or not. Indeterminate indicators are
|
|
|
|
/// "infinite" and their appearance is that of a circular spinner.
|
|
|
|
///
|
|
|
|
/// Invert this to go back to a bar appearance.
|
2021-01-19 00:11:52 -08:00
|
|
|
pub fn set_indeterminate(&self, is_indeterminate: bool) {
|
2021-03-26 13:29:39 -07:00
|
|
|
self.objc.with_mut(|obj| unsafe {
|
|
|
|
let _: () = msg_send![obj, setIndeterminate:match is_indeterminate {
|
2021-01-19 00:11:52 -08:00
|
|
|
true => YES,
|
|
|
|
false => NO
|
|
|
|
}];
|
2021-03-26 13:29:39 -07:00
|
|
|
});
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
2021-03-15 17:09:50 -07:00
|
|
|
|
|
|
|
/// Sets the value of this progress indicator.
|
|
|
|
///
|
|
|
|
/// If this progress indicator is indeterminate, this will have no effect.
|
|
|
|
pub fn set_value(&self, value: f64) {
|
|
|
|
let value = value as CGFloat;
|
|
|
|
|
2021-03-26 13:29:39 -07:00
|
|
|
self.objc.with_mut(|obj| unsafe {
|
|
|
|
let _: () = msg_send![obj, setDoubleValue:value];
|
|
|
|
});
|
2021-03-15 17:09:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Set whether this control is hidden or not.
|
|
|
|
pub fn set_hidden(&self, hidden: bool) {
|
2021-03-26 13:29:39 -07:00
|
|
|
self.objc.with_mut(|obj| unsafe {
|
|
|
|
let _: () = msg_send![obj, setHidden:match hidden {
|
2021-03-15 17:09:50 -07:00
|
|
|
true => YES,
|
|
|
|
false => NO
|
|
|
|
}];
|
2021-03-26 13:29:39 -07:00
|
|
|
});
|
2021-03-15 17:09:50 -07:00
|
|
|
}
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Layout for ProgressIndicator {
|
2021-03-26 13:29:39 -07:00
|
|
|
fn with_backing_node<F: Fn(id)>(&self, handler: F) {
|
|
|
|
self.objc.with_mut(handler);
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
2021-03-26 17:51:37 -07:00
|
|
|
|
|
|
|
fn get_from_backing_node<F: Fn(&Object) -> R, R>(&self, handler: F) -> R {
|
|
|
|
self.objc.get(handler)
|
|
|
|
}
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for ProgressIndicator {
|
2021-03-15 17:09:50 -07:00
|
|
|
/// A bit of extra cleanup for delegate callback pointers.
|
|
|
|
/// If the originating `ProgressIndicator` is being
|
2021-01-19 00:11:52 -08:00
|
|
|
/// dropped, we do some logic to clean it all up (e.g, we go ahead and check to see if
|
2021-03-15 17:09:50 -07:00
|
|
|
/// this has a superview (i.e, it's in the heirarchy) on the Objective-C side. If it does, we go
|
2021-01-19 00:11:52 -08:00
|
|
|
/// ahead and remove it - this is intended to match the semantics of how Rust handles things).
|
|
|
|
///
|
|
|
|
/// There are, thankfully, no delegates we need to break here.
|
|
|
|
fn drop(&mut self) {
|
2021-03-26 13:29:39 -07:00
|
|
|
/*unsafe {
|
2021-03-15 17:09:50 -07:00
|
|
|
let superview: id = msg_send![&*self.objc, superview];
|
|
|
|
if superview != nil {
|
|
|
|
let _: () = msg_send![&*self.objc, removeFromSuperview];
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
2021-03-26 13:29:39 -07:00
|
|
|
}*/
|
2021-01-19 00:11:52 -08:00
|
|
|
}
|
|
|
|
}
|