use winapi::um::winuser::{WS_OVERLAPPEDWINDOW, WS_VISIBLE, WS_DISABLED, WS_MAXIMIZE, WS_MINIMIZE, WS_CAPTION,
WS_MINIMIZEBOX, WS_MAXIMIZEBOX, WS_SYSMENU, WS_THICKFRAME, WS_CLIPCHILDREN, WS_CLIPSIBLINGS };
use crate::win32::base_helper::check_hwnd;
use crate::win32::window_helper as wh;
use crate::{NwgError, Icon};
use super::{ControlBase, ControlHandle};
const NOT_BOUND: &'static str = "ExternCanvas is not yet bound to a winapi object";
const BAD_HANDLE: &'static str = "INTERNAL ERROR: ExternCanvas handle is not HWND!";
bitflags! {
pub struct ExternCanvasFlags: u32 {
const NONE = 0;
const MAIN_WINDOW = WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX | WS_THICKFRAME | WS_MAXIMIZEBOX;
const WINDOW = WS_CAPTION | WS_SYSMENU;
const MINIMIZE_BOX = WS_MINIMIZEBOX;
const MAXIMIZE_BOX = WS_MAXIMIZEBOX;
const SYS_MENU = WS_SYSMENU;
const VISIBLE = WS_VISIBLE;
const DISABLED = WS_DISABLED;
const MAXIMIZED = WS_MAXIMIZE;
const MINIMIZED = WS_MINIMIZE;
const RESIZABLE = WS_THICKFRAME | WS_MAXIMIZEBOX;
}
}
#[derive(Default)]
pub struct ExternCanvas {
pub handle: ControlHandle
}
impl ExternCanvas {
pub fn builder<'a>() -> ExternCanvasBuilder<'a> {
ExternCanvasBuilder {
title: "New Canvas",
size: (500, 500),
position: (300, 300),
flags: None,
icon: None,
parent: None
}
}
pub fn invalidate(&self) {
use winapi::um::winuser::InvalidateRect;
use std::ptr;
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { InvalidateRect(handle, ptr::null(), 1); }
}
pub fn icon(&self) -> Option<Icon> {
use winapi::um::winuser::WM_GETICON;
use winapi::um::winnt::HANDLE;
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
let icon_handle = wh::send_message(handle, WM_GETICON, 0, 0);
if icon_handle == 0 {
None
} else {
Some(Icon { handle: icon_handle as HANDLE, owned: false })
}
}
pub fn set_icon(&self, icon: Option<&Icon>) {
use winapi::um::winuser::WM_SETICON;
use std::{mem, ptr};
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
let image_handle = icon.map(|i| i.handle).unwrap_or(ptr::null_mut());
unsafe {
wh::send_message(handle, WM_SETICON, 0, mem::transmute(image_handle));
}
}
pub fn focus(&self) -> bool {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::get_focus(handle) }
}
pub fn set_focus(&self) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::set_focus(handle); }
}
pub fn enabled(&self) -> bool {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::get_window_enabled(handle) }
}
pub fn set_enabled(&self, v: bool) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::set_window_enabled(handle, v) }
}
pub fn visible(&self) -> bool {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::get_window_visibility(handle) }
}
pub fn set_visible(&self, v: bool) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::set_window_visibility(handle, v) }
}
pub fn size(&self) -> (u32, u32) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::get_window_size(handle) }
}
pub fn physical_size(&self) -> (u32, u32) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::get_window_physical_size(handle) }
}
pub fn set_size(&self, x: u32, y: u32) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::set_window_size(handle, x, y, true) }
}
pub fn position(&self) -> (i32, i32) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::get_window_position(handle) }
}
pub fn set_position(&self, x: i32, y: i32) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::set_window_position(handle, x, y) }
}
pub fn text(&self) -> String {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::get_window_text(handle) }
}
pub fn set_text<'a>(&self, v: &'a str) {
let handle = check_hwnd(&self.handle, NOT_BOUND, BAD_HANDLE);
unsafe { wh::set_window_text(handle, v) }
}
pub fn class_name(&self) -> &'static str {
"NWG_EXTERN_CANVAS"
}
pub fn flags(&self) -> u32 {
WS_OVERLAPPEDWINDOW | WS_VISIBLE
}
pub fn forced_flags(&self) -> u32 {
WS_CLIPCHILDREN | WS_CLIPSIBLINGS
}
}
impl Drop for ExternCanvas {
fn drop(&mut self) {
self.handle.destroy();
}
}
pub struct ExternCanvasBuilder<'a> {
title: &'a str,
size: (i32, i32),
position: (i32, i32),
flags: Option<ExternCanvasFlags>,
icon: Option<&'a Icon>,
parent: Option<ControlHandle>
}
impl<'a> ExternCanvasBuilder<'a> {
pub fn flags(mut self, flags: ExternCanvasFlags) -> ExternCanvasBuilder<'a> {
self.flags = Some(flags);
self
}
pub fn title(mut self, text: &'a str) -> ExternCanvasBuilder<'a> {
self.title = text;
self
}
pub fn size(mut self, size: (i32, i32)) -> ExternCanvasBuilder<'a> {
self.size = size;
self
}
pub fn position(mut self, pos: (i32, i32)) -> ExternCanvasBuilder<'a> {
self.position = pos;
self
}
pub fn icon(mut self, ico: Option<&'a Icon>) -> ExternCanvasBuilder<'a> {
self.icon = ico;
self
}
pub fn parent<C: Into<ControlHandle>>(mut self, p: Option<C>) -> ExternCanvasBuilder<'a> {
self.parent = p.map(|p2| p2.into());
self
}
pub fn build(self, out: &mut ExternCanvas) -> Result<(), NwgError> {
use winapi::um::winuser::{WS_CHILD};
let mut flags = self.flags.map(|f| f.bits()).unwrap_or(out.flags());
if self.parent.is_some() {
flags &= !(WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX | WS_THICKFRAME | WS_MAXIMIZEBOX);
flags |= WS_CHILD;
}
out.handle = ControlBase::build_hwnd()
.class_name(out.class_name())
.forced_flags(out.forced_flags())
.flags(flags)
.size(self.size)
.position(self.position)
.text(self.title)
.parent(self.parent)
.build()?;
if self.icon.is_some() {
out.set_icon(self.icon);
}
Ok(())
}
}