pub(crate) mod base_helper;
pub(crate) mod window_helper;
pub(crate) mod resources_helper;
pub(crate) mod window;
pub(crate) mod message_box;
pub(crate) mod high_dpi;
pub(crate) mod monitor;
#[cfg(feature = "menu")]
pub(crate) mod menu;
#[cfg(feature = "cursor")]
pub(crate) mod cursor;
#[cfg(feature = "clipboard")]
pub(crate) mod clipboard;
#[cfg(feature = "tabs")]
pub(crate) mod tabs;
#[cfg(feature = "extern-canvas")]
pub(crate) mod extern_canvas;
#[cfg(feature = "image-decoder")]
pub(crate) mod image_decoder;
#[cfg(feature = "rich-textbox")]
pub(crate) mod richedit;
#[cfg(feature = "plotting")]
pub(crate) mod plotters_d2d;
use std::{fs, mem, ptr};
use crate::errors::NwgError;
use winapi::um::winuser::{IsDialogMessageW, GetAncestor, TranslateMessage, DispatchMessageW, GA_ROOT};
pub fn dispatch_thread_events() {
use winapi::um::winuser::MSG;
use winapi::um::winuser::GetMessageW;
unsafe {
let mut msg: MSG = mem::zeroed();
while GetMessageW(&mut msg, ptr::null_mut(), 0, 0) != 0 {
if IsDialogMessageW(GetAncestor(msg.hwnd, GA_ROOT), &mut msg) == 0 {
TranslateMessage(&msg);
DispatchMessageW(&msg);
}
}
}
}
pub fn dispatch_thread_events_with_callback<F>(mut cb: F)
where F: FnMut() -> () + 'static
{
use winapi::um::winuser::MSG;
use winapi::um::winuser::{PeekMessageW, PM_REMOVE, WM_QUIT};
unsafe {
let mut msg: MSG = mem::zeroed();
while msg.message != WM_QUIT {
let has_message = PeekMessageW(&mut msg, ptr::null_mut(), 0, 0, PM_REMOVE) != 0;
if has_message {
if IsDialogMessageW(GetAncestor(msg.hwnd, GA_ROOT), &mut msg) == 0 {
TranslateMessage(&msg);
DispatchMessageW(&msg);
}
}
cb();
}
}
}
pub fn stop_thread_dispatch() {
use winapi::um::winuser::PostMessageW;
use winapi::um::winuser::WM_QUIT;
unsafe { PostMessageW(ptr::null_mut(), WM_QUIT, 0, 0) };
}
pub fn enable_visual_styles() {
use winapi::shared::minwindef::{ULONG, DWORD, MAX_PATH};
use winapi::shared::basetsd::ULONG_PTR;
use winapi::um::fileapi::{GetTempFileNameW, GetTempPathW};
use winapi::um::winbase::{ACTCTXW, CreateActCtxW, ActivateActCtx};
const MANIFEST_CONTENT: &str = r#"
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0">
<description>native-windows-gui comctl32 manifest</description>
<dependency>
<dependentAssembly>
<assemblyIdentity type="win32" name="Microsoft.Windows.Common-Controls" version="6.0.0.0" processorArchitecture="*" publicKeyToken="6595b64144ccf1df" language="*" />
</dependentAssembly>
</dependency>
</assembly>
"#;
const ACTCTX_FLAG_SET_PROCESS_DEFAULT: DWORD = 0x010;
let mut tmp_dir = [0u16; MAX_PATH + 1];
if unsafe { GetTempPathW(tmp_dir.len() as u32, tmp_dir.as_mut_ptr()) } == 0 {
return;
}
let mut tmp_path = [0u16; MAX_PATH]; let prefix = ['n' as u16, 'w' as u16, 'g' as u16];
if unsafe { GetTempFileNameW(tmp_dir.as_ptr(), prefix.as_ptr(), 0, tmp_path.as_mut_ptr())} == 0 {
return;
}
let manifest_path_raw = tmp_path;
let manifest_path = base_helper::from_utf16(&tmp_path);
let _ = fs::write(&manifest_path, MANIFEST_CONTENT);
let mut activation_cookie: ULONG_PTR = 0;
let mut act_ctx = ACTCTXW {
cbSize: mem::size_of::<ACTCTXW> as ULONG,
dwFlags: ACTCTX_FLAG_SET_PROCESS_DEFAULT,
lpSource: manifest_path_raw.as_ptr(),
wProcessorArchitecture: 0,
wLangId: 0,
lpAssemblyDirectory: ptr::null_mut(),
lpResourceName: ptr::null_mut(),
lpApplicationName: ptr::null_mut(),
hModule: ptr::null_mut()
};
unsafe {
let handle = CreateActCtxW(&mut act_ctx);
ActivateActCtx(handle, &mut activation_cookie);
}
let _ = fs::remove_file(&manifest_path);
}
pub fn init_common_controls() -> Result<(), NwgError> {
use winapi::um::objbase::CoInitialize;
use winapi::um::libloaderapi::LoadLibraryW;
use winapi::um::commctrl::{InitCommonControlsEx, INITCOMMONCONTROLSEX};
use winapi::um::commctrl::{ICC_BAR_CLASSES, ICC_STANDARD_CLASSES, ICC_DATE_CLASSES, ICC_PROGRESS_CLASS,
ICC_TAB_CLASSES, ICC_TREEVIEW_CLASSES, ICC_LISTVIEW_CLASSES};
use winapi::shared::winerror::{S_OK, S_FALSE};
unsafe {
let mut classes = ICC_BAR_CLASSES | ICC_STANDARD_CLASSES;
if cfg!(feature = "datetime-picker") {
classes |= ICC_DATE_CLASSES;
}
if cfg!(feature = "progress-bar") {
classes |= ICC_PROGRESS_CLASS;
}
if cfg!(feature = "tabs") {
classes |= ICC_TAB_CLASSES;
}
if cfg!(feature = "tree-view") {
classes |= ICC_TREEVIEW_CLASSES;
}
if cfg!(feature = "list-view") {
classes |= ICC_LISTVIEW_CLASSES;
}
if cfg!(feature = "rich-textbox") {
let lib = base_helper::to_utf16("Msftedit.dll");
LoadLibraryW(lib.as_ptr());
}
let data = INITCOMMONCONTROLSEX {
dwSize: mem::size_of::<INITCOMMONCONTROLSEX>() as u32,
dwICC: classes
};
InitCommonControlsEx(&data);
}
window::init_window_class()?;
tabs_init()?;
extern_canvas_init()?;
frame_init()?;
match unsafe { CoInitialize(ptr::null_mut()) } {
S_OK | S_FALSE => Ok(()),
_ => Err(NwgError::initialization("CoInitialize failed"))
}
}
#[cfg(feature = "tabs")]
fn tabs_init() -> Result<(), NwgError> { tabs::create_tab_classes() }
#[cfg(not(feature = "tabs"))]
fn tabs_init() -> Result<(), NwgError> { Ok(()) }
#[cfg(feature = "extern-canvas")]
fn extern_canvas_init() -> Result<(), NwgError> { extern_canvas::create_extern_canvas_classes() }
#[cfg(not(feature = "extern-canvas"))]
fn extern_canvas_init() -> Result<(), NwgError> { Ok(()) }
#[cfg(feature = "frame")]
fn frame_init() -> Result<(), NwgError> { window::create_frame_classes() }
#[cfg(not(feature = "frame"))]
fn frame_init() -> Result<(), NwgError> { Ok(()) }