228 lines
5.1 KiB
Objective-C
228 lines
5.1 KiB
Objective-C
// 15 august 2015
|
|
#import "uipriv_darwin.h"
|
|
|
|
struct uiWindow {
|
|
uiDarwinControl c;
|
|
NSWindow *window;
|
|
uiControl *child;
|
|
int margined;
|
|
int (*onClosing)(uiWindow *, void *);
|
|
void *onClosingData;
|
|
};
|
|
|
|
@interface windowDelegateClass : NSObject<NSWindowDelegate> {
|
|
NSMapTable *windows;
|
|
}
|
|
- (BOOL)windowShouldClose:(id)sender;
|
|
- (void)registerWindow:(uiWindow *)w;
|
|
- (void)unregisterWindow:(uiWindow *)w;
|
|
- (uiWindow *)lookupWindow:(NSWindow *)w;
|
|
@end
|
|
|
|
@implementation windowDelegateClass
|
|
|
|
- (id)init
|
|
{
|
|
self = [super init];
|
|
if (self)
|
|
self->windows = newMap();
|
|
return self;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
if ([self->windows count] != 0)
|
|
complain("attempt to destroy shared window delegate but windows are still registered to it");
|
|
[self->windows release];
|
|
[super dealloc];
|
|
}
|
|
|
|
- (BOOL)windowShouldClose:(id)sender
|
|
{
|
|
uiWindow *w;
|
|
|
|
w = [self lookupWindow:((NSWindow *) sender)];
|
|
// w should not be NULL; we are only the delegate of registered windows
|
|
if ((*(w->onClosing))(w, w->onClosingData))
|
|
uiControlDestroy(uiControl(w));
|
|
return NO;
|
|
}
|
|
|
|
- (void)registerWindow:(uiWindow *)w
|
|
{
|
|
mapSet(self->windows, w->window, w);
|
|
[w->window setDelegate:self];
|
|
}
|
|
|
|
- (void)unregisterWindow:(uiWindow *)w
|
|
{
|
|
[w->window setDelegate:nil];
|
|
[self->windows removeObjectForKey:w->window];
|
|
}
|
|
|
|
- (uiWindow *)lookupWindow:(NSWindow *)w
|
|
{
|
|
NSValue *v;
|
|
|
|
v = (NSValue *) [self->windows objectForKey:w];
|
|
if (v == nil) // just in case we're called with some OS X-provided window as the key window
|
|
return NULL;
|
|
return (uiWindow *) [v pointerValue];
|
|
}
|
|
|
|
@end
|
|
|
|
static windowDelegateClass *windowDelegate = nil;
|
|
|
|
static void onDestroy(uiWindow *);
|
|
|
|
uiDarwinDefineControlWithOnDestroy(
|
|
uiWindow, // type name
|
|
uiWindowType, // type function
|
|
window, // handle
|
|
onDestroy(this); // on destroy
|
|
)
|
|
|
|
static void onDestroy(uiWindow *w)
|
|
{
|
|
NSView *childView;
|
|
|
|
// hide the window
|
|
[w->window orderOut:w->window];
|
|
if (w->child != NULL) {
|
|
childView = (NSView *) uiControlHandle(w->child);
|
|
[childView removeFromSuperview];
|
|
uiControlSetParent(w->child, NULL);
|
|
uiControlDestroy(w->child);
|
|
}
|
|
[windowDelegate unregisterWindow:w];
|
|
}
|
|
|
|
// TODO forbid setting a parent
|
|
|
|
static void windowCommitShow(uiControl *c)
|
|
{
|
|
uiWindow *w = (uiWindow *) c;
|
|
|
|
[w->window makeKeyAndOrderFront:w->window];
|
|
}
|
|
|
|
static void windowCommitHide(uiControl *c)
|
|
{
|
|
uiWindow *w = (uiWindow *) c;
|
|
|
|
[w->window orderOut:w->window];
|
|
}
|
|
|
|
// TODO container update state
|
|
|
|
static void windowRelayout(uiDarwinControl *c)
|
|
{
|
|
uiWindow *w = uiWindow(c);
|
|
uiDarwinControl *cc;
|
|
NSView *childView;
|
|
|
|
if (w->child == NULL)
|
|
return;
|
|
cc = uiDarwinControl(w->child);
|
|
childView = (NSView *) uiControlHandle(w->child);
|
|
// first relayout the child
|
|
(*(cc->Relayout))(cc);
|
|
// now relayout ourselves
|
|
layoutSingleView([w->window contentView], childView, w->margined);
|
|
}
|
|
|
|
char *uiWindowTitle(uiWindow *w)
|
|
{
|
|
return uiDarwinNSStringToText([w->window title]);
|
|
}
|
|
|
|
void uiWindowSetTitle(uiWindow *w, const char *title)
|
|
{
|
|
[w->window setTitle:toNSString(title)];
|
|
}
|
|
|
|
void uiWindowOnClosing(uiWindow *w, int (*f)(uiWindow *, void *), void *data)
|
|
{
|
|
w->onClosing = f;
|
|
w->onClosingData = data;
|
|
}
|
|
|
|
void uiWindowSetChild(uiWindow *w, uiControl *child)
|
|
{
|
|
NSView *childView;
|
|
|
|
if (w->child != NULL) {
|
|
childView = (NSView *) uiControlHandle(w->child);
|
|
[childView removeFromSuperview];
|
|
uiControlSetParent(w->child, NULL);
|
|
}
|
|
w->child = child;
|
|
if (w->child != NULL) {
|
|
uiControlSetParent(w->child, uiControl(w));
|
|
childView = (NSView *) uiControlHandle(w->child);
|
|
[[w->window contentView] addSubview:childView];
|
|
uiDarwinControlTriggerRelayout(uiDarwinControl(w));
|
|
}
|
|
}
|
|
|
|
int uiWindowMargined(uiWindow *w)
|
|
{
|
|
return w->margined;
|
|
}
|
|
|
|
void uiWindowSetMargined(uiWindow *w, int margined)
|
|
{
|
|
w->margined = margined;
|
|
if (w->child != NULL)
|
|
uiDarwinControlTriggerRelayout(uiDarwinControl(w));
|
|
}
|
|
|
|
static int defaultOnClosing(uiWindow *w, void *data)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
uiWindow *uiNewWindow(const char *title, int width, int height, int hasMenubar)
|
|
{
|
|
uiWindow *w;
|
|
|
|
finalizeMenus();
|
|
|
|
w = (uiWindow *) uiNewControl(uiWindowType());
|
|
|
|
w->window = [[NSWindow alloc] initWithContentRect:NSMakeRect(0, 0, (CGFloat) width, (CGFloat) height)
|
|
styleMask:(NSTitledWindowMask | NSClosableWindowMask | NSMiniaturizableWindowMask | NSResizableWindowMask)
|
|
backing:NSBackingStoreBuffered
|
|
defer:YES];
|
|
[w->window setTitle:toNSString(title)];
|
|
|
|
// explicitly release when closed
|
|
// the only thing that closes the window is us anyway
|
|
[w->window setReleasedWhenClosed:YES];
|
|
|
|
if (windowDelegate == nil) {
|
|
windowDelegate = [windowDelegateClass new];
|
|
[delegates addObject:windowDelegate];
|
|
}
|
|
[windowDelegate registerWindow:w];
|
|
uiWindowOnClosing(w, defaultOnClosing, NULL);
|
|
|
|
uiDarwinFinishNewControl(w, uiWindow);
|
|
//TODO uiControl(w)->CommitShow = windowCommitShow;
|
|
//TODO uiControl(w)->CommitHide = windowCommitHide;
|
|
uiDarwinControl(w)->Relayout = windowRelayout;
|
|
|
|
return w;
|
|
}
|
|
|
|
// utility function for menus
|
|
uiWindow *windowFromNSWindow(NSWindow *w)
|
|
{
|
|
if (w == nil)
|
|
return NULL;
|
|
if (windowDelegate == nil) // no windows were created yet; we're called with some OS X-provided window
|
|
return NULL;
|
|
return [windowDelegate lookupWindow:w];
|
|
}
|