mirror of
https://github.com/italicsjenga/rust_minifb.git
synced 2025-01-11 11:31:32 +11:00
WIP on new menu stuff
This commit is contained in:
parent
4a3eabe8e6
commit
42ce3b9286
|
@ -89,9 +89,11 @@ fn main() {
|
||||||
//window.add_menu("Test", &menu).expect("Unable to add menu");
|
//window.add_menu("Test", &menu).expect("Unable to add menu");
|
||||||
|
|
||||||
let mut menu = Menu::new("TestMenu").unwrap();
|
let mut menu = Menu::new("TestMenu").unwrap();
|
||||||
let mut item = MenuItem::new("Item", 1).enabled(true);
|
|
||||||
|
|
||||||
menu.add_item(&mut item);
|
menu.add_item(&MenuItem::new("Item 1", 1));
|
||||||
|
menu.add_item(&MenuItem::new("Item 2", 2));
|
||||||
|
menu.add_item(&MenuItem::new("Item 3", 3));
|
||||||
|
|
||||||
let _ = window.add_menu(&menu);
|
let _ = window.add_menu(&menu);
|
||||||
|
|
||||||
let color_mul = 1;
|
let color_mul = 1;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/// Key is used by the get key functions to check if some keys on the keyboard has been pressed
|
/// Key is used by the get key functions to check if some keys on the keyboard has been pressed
|
||||||
#[derive(PartialEq, Clone, Copy)]
|
#[derive(Debug, PartialEq, Clone, Copy)]
|
||||||
pub enum Key {
|
pub enum Key {
|
||||||
Key0 = 0,
|
Key0 = 0,
|
||||||
Key1 = 1,
|
Key1 = 1,
|
||||||
|
|
10
src/lib.rs
10
src/lib.rs
|
@ -495,6 +495,7 @@ impl Window {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// Command key on Mac OS
|
/// Command key on Mac OS
|
||||||
pub const MENU_KEY_COMMAND: usize = 1;
|
pub const MENU_KEY_COMMAND: usize = 1;
|
||||||
/// Windows key on Windows
|
/// Windows key on Windows
|
||||||
|
@ -510,6 +511,9 @@ const MENU_ID_SEPARATOR:usize = 0xffffffff;
|
||||||
|
|
||||||
pub struct Menu(imp::Menu);
|
pub struct Menu(imp::Menu);
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct MenuItemHandle(pub u64);
|
||||||
|
|
||||||
impl Menu {
|
impl Menu {
|
||||||
pub fn new(name: &str) -> Result<Menu> {
|
pub fn new(name: &str) -> Result<Menu> {
|
||||||
imp::Menu::new(name).map(Menu)
|
imp::Menu::new(name).map(Menu)
|
||||||
|
@ -526,13 +530,13 @@ impl Menu {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn add_item(&mut self, item: &mut MenuItem) {
|
pub fn add_item(&mut self, item: &MenuItem) -> MenuItemHandle {
|
||||||
self.0.add_item(item)
|
self.0.add_item(item)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn remove_item(&mut self, _item: &mut MenuItem) {
|
pub fn remove_item(&mut self, item: &MenuItemHandle) {
|
||||||
//self.0.remove_item(item)
|
self.0.remove_item(item)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -462,6 +462,8 @@ void* mfb_create_menu(const char* name) {
|
||||||
//NSMenuItem* menu_item = [[NSMenuItem alloc] initWithTitle:name action:NULL keyEquivalent:@""];
|
//NSMenuItem* menu_item = [[NSMenuItem alloc] initWithTitle:name action:NULL keyEquivalent:@""];
|
||||||
NSMenu* menu = [[NSMenu alloc] initWithTitle:ns_name];
|
NSMenu* menu = [[NSMenu alloc] initWithTitle:ns_name];
|
||||||
//[menu_item setSubmenu:menu];
|
//[menu_item setSubmenu:menu];
|
||||||
|
|
||||||
|
//printf("created menu %p\n");
|
||||||
|
|
||||||
return (void*)menu;
|
return (void*)menu;
|
||||||
}
|
}
|
||||||
|
@ -475,5 +477,16 @@ void mfb_destroy_menu(void* menu_item, const char* name)
|
||||||
[main_menu removeItem:item];
|
[main_menu removeItem:item];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void mfb_remove_menu_item(void* parent, uint64_t menu_item) {
|
||||||
|
NSMenu* menu = (NSMenu*)parent;
|
||||||
|
NSMenuItem* item = (NSMenuItem*)(uintptr_t)menu_item;
|
||||||
|
|
||||||
|
printf("remove item menu %p item %p\n", menu, item);
|
||||||
|
[menu removeItem:item];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -265,7 +265,7 @@ const uint32_t MENU_KEY_CTRL = 8;
|
||||||
const uint32_t MENU_KEY_ALT = 16;
|
const uint32_t MENU_KEY_ALT = 16;
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
/*
|
||||||
void build_submenu(NSMenu* menu, MenuDesc* desc)
|
void build_submenu(NSMenu* menu, MenuDesc* desc)
|
||||||
{
|
{
|
||||||
[menu removeAllItems];
|
[menu removeAllItems];
|
||||||
|
@ -332,10 +332,11 @@ void build_submenu(NSMenu* menu, MenuDesc* desc)
|
||||||
desc++;
|
desc++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void mfb_add_menu_item(
|
uint64_t mfb_add_menu_item(
|
||||||
void* in_menu,
|
void* in_menu,
|
||||||
int32_t menu_id,
|
int32_t menu_id,
|
||||||
const char* item_name,
|
const char* item_name,
|
||||||
|
@ -345,7 +346,9 @@ void mfb_add_menu_item(
|
||||||
{
|
{
|
||||||
NSMenu* menu = (NSMenu*)in_menu;
|
NSMenu* menu = (NSMenu*)in_menu;
|
||||||
|
|
||||||
NSString* name = [NSString stringWithUTF8String: item_name];
|
const char* t = strdup(item_name);
|
||||||
|
|
||||||
|
NSString* name = [NSString stringWithUTF8String: t];
|
||||||
|
|
||||||
if (menu_id == -1)
|
if (menu_id == -1)
|
||||||
{
|
{
|
||||||
|
@ -386,7 +389,11 @@ void mfb_add_menu_item(
|
||||||
[newItem setOnStateImage: newItem.offStateImage];
|
[newItem setOnStateImage: newItem.offStateImage];
|
||||||
[menu addItem:newItem];
|
[menu addItem:newItem];
|
||||||
[newItem release];
|
[newItem release];
|
||||||
|
|
||||||
|
return (uint64_t)newItem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@end
|
@end
|
||||||
|
|
|
@ -8,7 +8,7 @@ use Result;
|
||||||
use InputCallback;
|
use InputCallback;
|
||||||
use mouse_handler;
|
use mouse_handler;
|
||||||
use window_flags;
|
use window_flags;
|
||||||
use MenuItem;
|
use {MenuItem, MenuItemHandle};
|
||||||
//use menu::Menu;
|
//use menu::Menu;
|
||||||
|
|
||||||
use libc::{c_void, c_char, c_uchar};
|
use libc::{c_void, c_char, c_uchar};
|
||||||
|
@ -186,13 +186,14 @@ extern {
|
||||||
//fn mfb_active_menu(window: *mut c_void) -> i32;
|
//fn mfb_active_menu(window: *mut c_void) -> i32;
|
||||||
|
|
||||||
fn mfb_create_menu(name: *const c_char) -> *mut c_void;
|
fn mfb_create_menu(name: *const c_char) -> *mut c_void;
|
||||||
fn mfb_destroy_menu(menu_item: *mut c_void);
|
//fn mfb_destroy_menu(menu_item: *mut c_void);
|
||||||
|
|
||||||
fn mfb_add_menu_item(menu_item: *mut c_void,
|
fn mfb_add_menu_item(menu_item: *mut c_void,
|
||||||
menu_id: i32,
|
menu_id: i32,
|
||||||
name: *const c_char,
|
name: *const c_char,
|
||||||
key: u32,
|
key: u32,
|
||||||
modifier: u32) -> *mut c_void;
|
modifier: u32) -> u64;
|
||||||
|
fn mfb_remove_menu_item(menu: *mut c_void, item_handle: u64);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
|
@ -391,45 +392,6 @@ impl Window {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
|
|
||||||
pub fn add_menu(&mut self, name: &str, menu: &Vec<Menu>) -> Result<()> {
|
|
||||||
let mut build_menu = Vec::<Vec<CMenu>>::new();
|
|
||||||
|
|
||||||
unsafe {
|
|
||||||
Self::recursive_convert(&mut build_menu, &Some(menu));
|
|
||||||
let menu_len = build_menu.len();
|
|
||||||
mfb_add_menu(self.window_handle,
|
|
||||||
CString::new(name).unwrap().as_ptr(),
|
|
||||||
build_menu[menu_len - 1].as_mut_ptr() as *mut c_void);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn update_menu(&mut self, name: &str, menu: &Vec<Menu>) -> Result<()> {
|
|
||||||
let mut build_menu = Vec::<Vec<CMenu>>::new();
|
|
||||||
|
|
||||||
unsafe {
|
|
||||||
Self::recursive_convert(&mut build_menu, &Some(menu));
|
|
||||||
let menu_len = build_menu.len();
|
|
||||||
mfb_update_menu(self.window_handle,
|
|
||||||
CString::new(name).unwrap().as_ptr(),
|
|
||||||
build_menu[menu_len - 1].as_mut_ptr() as *mut c_void);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remove_menu(&mut self, name: &str) -> Result<()> {
|
|
||||||
unsafe {
|
|
||||||
mfb_remove_menu(self.window_handle, CString::new(name).unwrap().as_ptr());
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn is_open(&self) -> bool {
|
pub fn is_open(&self) -> bool {
|
||||||
unsafe { mfb_should_close(self.window_handle) == 0 }
|
unsafe { mfb_should_close(self.window_handle) == 0 }
|
||||||
|
@ -474,7 +436,75 @@ impl Window {
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
unsafe fn map_key_to_menu_key(key: Key) -> i32 {
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
unsafe fn recursive_convert(menu_build_vec: &mut Vec<Vec<CMenu>>, in_menu: &Option<&Vec<Menu>>) -> *mut raw::c_void {
|
||||||
|
if in_menu.is_none() {
|
||||||
|
return ptr::null_mut();
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut menu_build = Vec::<CMenu>::new();
|
||||||
|
let menu_vec = in_menu.as_ref().unwrap();
|
||||||
|
|
||||||
|
for m in menu_vec.iter() {
|
||||||
|
let key_map = Self::map_key_to_menu_key(m.key);
|
||||||
|
|
||||||
|
let mut menu = CMenu {
|
||||||
|
name: mem::uninitialized(),
|
||||||
|
id: m.id as raw::c_int,
|
||||||
|
key: key_map as raw::c_int,
|
||||||
|
special_key: 0,
|
||||||
|
modifier: m.modifier as raw::c_int,
|
||||||
|
mac_mod: m.mac_mod as raw::c_int,
|
||||||
|
enabled: m.enabled as raw::c_int,
|
||||||
|
sub_menu : Self::recursive_convert(menu_build_vec, &m.sub_menu),
|
||||||
|
};
|
||||||
|
|
||||||
|
let name = CString::new(m.name).unwrap();
|
||||||
|
let name_len = m.name.len();
|
||||||
|
|
||||||
|
ptr::copy_nonoverlapping(name.as_ptr(),
|
||||||
|
menu.name.as_mut_ptr() as *mut i8,
|
||||||
|
name_len);
|
||||||
|
menu.name[name_len] = 0;
|
||||||
|
|
||||||
|
menu_build.push(menu);
|
||||||
|
}
|
||||||
|
|
||||||
|
// end marker
|
||||||
|
|
||||||
|
menu_build.push(CMenu {
|
||||||
|
name: [0; STRING_SIZE],
|
||||||
|
id: -2,
|
||||||
|
key: 0,
|
||||||
|
special_key: 0,
|
||||||
|
modifier: 0,
|
||||||
|
mac_mod: 0,
|
||||||
|
enabled: 0,
|
||||||
|
sub_menu : ptr::null_mut(),
|
||||||
|
});
|
||||||
|
|
||||||
|
let ptr = menu_build.as_mut_ptr() as *mut raw::c_void ;
|
||||||
|
menu_build_vec.push(menu_build);
|
||||||
|
ptr
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Menu {
|
||||||
|
menu_handle: *mut c_void,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Menu {
|
||||||
|
pub fn new(name: &str) -> Result<Menu> {
|
||||||
|
unsafe {
|
||||||
|
let menu_name = CString::new(name).unwrap();
|
||||||
|
Ok(Menu { menu_handle: mfb_create_menu(menu_name.as_ptr()) })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe fn map_key_to_menu_key(key: Key) -> u32 {
|
||||||
match key {
|
match key {
|
||||||
Key::A => 0x00,
|
Key::A => 0x00,
|
||||||
Key::S => 0x01,
|
Key::S => 0x01,
|
||||||
|
@ -584,140 +614,25 @@ impl Window {
|
||||||
_ => 0x7f,
|
_ => 0x7f,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
pub fn add_item(&mut self, item: &MenuItem) -> MenuItemHandle {
|
||||||
unsafe fn recursive_convert(menu_build_vec: &mut Vec<Vec<CMenu>>, in_menu: &Option<&Vec<Menu>>) -> *mut raw::c_void {
|
|
||||||
if in_menu.is_none() {
|
|
||||||
return ptr::null_mut();
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut menu_build = Vec::<CMenu>::new();
|
|
||||||
let menu_vec = in_menu.as_ref().unwrap();
|
|
||||||
|
|
||||||
for m in menu_vec.iter() {
|
|
||||||
let key_map = Self::map_key_to_menu_key(m.key);
|
|
||||||
|
|
||||||
let mut menu = CMenu {
|
|
||||||
name: mem::uninitialized(),
|
|
||||||
id: m.id as raw::c_int,
|
|
||||||
key: key_map as raw::c_int,
|
|
||||||
special_key: 0,
|
|
||||||
modifier: m.modifier as raw::c_int,
|
|
||||||
mac_mod: m.mac_mod as raw::c_int,
|
|
||||||
enabled: m.enabled as raw::c_int,
|
|
||||||
sub_menu : Self::recursive_convert(menu_build_vec, &m.sub_menu),
|
|
||||||
};
|
|
||||||
|
|
||||||
let name = CString::new(m.name).unwrap();
|
|
||||||
let name_len = m.name.len();
|
|
||||||
|
|
||||||
ptr::copy_nonoverlapping(name.as_ptr(),
|
|
||||||
menu.name.as_mut_ptr() as *mut i8,
|
|
||||||
name_len);
|
|
||||||
menu.name[name_len] = 0;
|
|
||||||
|
|
||||||
menu_build.push(menu);
|
|
||||||
}
|
|
||||||
|
|
||||||
// end marker
|
|
||||||
|
|
||||||
menu_build.push(CMenu {
|
|
||||||
name: [0; STRING_SIZE],
|
|
||||||
id: -2,
|
|
||||||
key: 0,
|
|
||||||
special_key: 0,
|
|
||||||
modifier: 0,
|
|
||||||
mac_mod: 0,
|
|
||||||
enabled: 0,
|
|
||||||
sub_menu : ptr::null_mut(),
|
|
||||||
});
|
|
||||||
|
|
||||||
let ptr = menu_build.as_mut_ptr() as *mut raw::c_void ;
|
|
||||||
menu_build_vec.push(menu_build);
|
|
||||||
ptr
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Menu {
|
|
||||||
menu_handle: *mut c_void,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Menu {
|
|
||||||
pub fn new(name: &str) -> Result<Menu> {
|
|
||||||
unsafe {
|
unsafe {
|
||||||
let menu_name = CString::new(name).unwrap().as_ptr();
|
let item_name = CString::new(item.label.as_str()).unwrap();
|
||||||
Ok(Menu { menu_handle: mfb_create_menu(menu_name) })
|
let conv_key = Self::map_key_to_menu_key(item.key);
|
||||||
|
|
||||||
|
println!("key {:?} conv {}", item.key, conv_key);
|
||||||
|
|
||||||
|
MenuItemHandle(mfb_add_menu_item(self.menu_handle, item.id as i32, item_name.as_ptr(),
|
||||||
|
Self::map_key_to_menu_key(item.key), item.modifier))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn add_item(&mut self, item: &mut MenuItem) {
|
pub fn remove_item(&mut self, handle: &MenuItemHandle) {
|
||||||
unsafe {
|
unsafe {
|
||||||
let item_name = CString::new(item.label.as_str()).unwrap().as_ptr();
|
mfb_remove_menu_item(self.menu_handle, handle.0);
|
||||||
|
|
||||||
mfb_add_menu_item(self.menu_handle, item.id as i32, item_name,
|
|
||||||
item.key as u32, item.modifier);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
#[inline]
|
|
||||||
pub fn remove_item(&mut self, item: &mut MenuItem) {
|
|
||||||
self.0.remove_item(item)
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
pub struct MenuItem {
|
|
||||||
id: usize,
|
|
||||||
label: String,
|
|
||||||
separator: bool,
|
|
||||||
key: Key,
|
|
||||||
modifier: u32,
|
|
||||||
impl_handle: u64,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl MenuItem {
|
|
||||||
pub fn new(name: &str, id: usize) -> MenuItem {
|
|
||||||
MenuItem {
|
|
||||||
id: id,
|
|
||||||
label: "".to_owned(),
|
|
||||||
enabled: true,
|
|
||||||
separator: false,
|
|
||||||
key: Key::Unknown,
|
|
||||||
modifier: 0,
|
|
||||||
impl_handle: 0,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#[inline]
|
|
||||||
pub fn shortcut(self, key: Key, modifier: u32) -> Self {
|
|
||||||
MenuItem {
|
|
||||||
shortcut: key,
|
|
||||||
modifier: modifier,
|
|
||||||
.. self
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#[inline]
|
|
||||||
pub fn separator(self, enabled: bool) -> Self {
|
|
||||||
MenuItem {
|
|
||||||
id: 0xffffffff,
|
|
||||||
.. self
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#[inline]
|
|
||||||
pub fn enabled(self, enabled: bool) -> Self {
|
|
||||||
MenuItem {
|
|
||||||
enabled: enabled,
|
|
||||||
.. self
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
impl Drop for Window {
|
impl Drop for Window {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
|
@ -727,6 +642,7 @@ impl Drop for Window {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
impl Drop for Menu {
|
impl Drop for Menu {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe {
|
unsafe {
|
||||||
|
@ -734,6 +650,7 @@ impl Drop for Menu {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue