libui/darwin/tab.m

228 lines
6.0 KiB
Objective-C

// 12 april 2015
#import "uipriv_darwin.h"
// TODO rewrite this whole file to take advantage of bin functions
struct tab {
uiTab t;
NSTabView *tabview;
NSMutableArray *pages;
NSMutableArray *margined;
void (*baseEnable)(uiControl *);
void (*baseDisable)(uiControl *);
void (*baseSysFunc)(uiControl *, uiControlSysFuncParams *);
};
static void destroy(void *data)
{
struct tab *t = (struct tab *) data;
// first destroy all tab pages so we can destroy all the bins
while ([t->tabview numberOfTabViewItems] != 0)
[t->tabview removeTabViewItem:[t->tabview tabViewItemAtIndex:0]];
// then destroy all the bins, destroying children in the process
// the above loop serves the purpose of binSetParent()
[t->pages enumerateObjectsUsingBlock:^(id obj, NSUInteger index, BOOL *stop) {
NSValue *v = (NSValue *) obj;
uiBin *p;
p = (uiBin *) [v pointerValue];
uiControlDestroy(uiControl(p));
}];
// and finally destroy ourselves
[t->pages release];
[t->margined release];
uiFree(t);
}
// the default new control implementation uses -sizeToFit, which we don't have with NSTabView
// fortunately, we do have -minimumSize
static void tabPreferredSize(uiControl *c, uiSizing *d, intmax_t *width, intmax_t *height)
{
struct tab *t = (struct tab *) c;
NSSize s;
s = [t->tabview minimumSize];
*width = (intmax_t) (s.width);
*height = (intmax_t) (s.height);
}
static void tabEnable(uiControl *c)
{
struct tab *t = (struct tab *) c;
(*(t->baseEnable))(uiControl(t));
[t->pages enumerateObjectsUsingBlock:^(id obj, NSUInteger index, BOOL *stop) {
NSValue *v = (NSValue *) obj;
uiBin *page;
page = (uiBin *) [v pointerValue];
uiControlEnable(uiControl(page));
}];
}
static void tabDisable(uiControl *c)
{
struct tab *t = (struct tab *) c;
(*(t->baseDisable))(uiControl(t));
[t->pages enumerateObjectsUsingBlock:^(id obj, NSUInteger index, BOOL *stop) {
NSValue *v = (NSValue *) obj;
uiBin *page;
page = (uiBin *) [v pointerValue];
uiControlDisable(uiControl(page));
}];
}
static void tabSysFunc(uiControl *c, uiControlSysFuncParams *p)
{
struct tab *t = (struct tab *) c;
(*(t->baseSysFunc))(uiControl(t), p);
[t->pages enumerateObjectsUsingBlock:^(id obj, NSUInteger index, BOOL *stop) {
NSValue *v = (NSValue *) obj;
uiBin *page;
page = (uiBin *) [v pointerValue];
uiControlSysFunc(uiControl(page), p);
}];
}
static void tabAppendPage(uiTab *tt, const char *name, uiControl *child)
{
struct tab *t = (struct tab *) tt;
uiBin *page;
NSTabViewItem *i;
page = newBin();
uiBinSetMainControl(page, child);
[t->pages addObject:[NSValue valueWithPointer:page]];
[t->margined addObject:[NSNumber numberWithInt:0]];
i = [[NSTabViewItem alloc] initWithIdentifier:nil];
[i setLabel:toNSString(name)];
[i setView:((NSView *) uiControlHandle(uiControl(page)))];
[t->tabview addTabViewItem:i];
}
static void tabInsertPageBefore(uiTab *tt, const char *name, uintmax_t n, uiControl *child)
{
struct tab *t = (struct tab *) tt;
uiBin *page;
NSTabViewItem *i;
page = newBin();
uiBinSetMainControl(page, child);
[t->pages insertObject:[NSValue valueWithPointer:page] atIndex:n];
[t->margined insertObject:[NSNumber numberWithInt:0] atIndex:n];
i = [[NSTabViewItem alloc] initWithIdentifier:nil];
[i setLabel:toNSString(name)];
[i setView:((NSView *) uiControlHandle(uiControl(page)))];
[t->tabview insertTabViewItem:i atIndex:n];
}
static void tabDeletePage(uiTab *tt, uintmax_t n)
{
struct tab *t = (struct tab *) tt;
NSValue *v;
uiBin *page;
NSTabViewItem *i;
v = (NSValue *) [t->pages objectAtIndex:n];
page = (uiBin *) [v pointerValue];
[t->pages removeObjectAtIndex:n];
[t->margined removeObjectAtIndex:n];
// make sure the children of the tab aren't destroyed
uiBinSetMainControl(page, NULL);
// remove the bin from the tab view
// this serves the purpose of uiBinRemoveOSParent()
i = [t->tabview tabViewItemAtIndex:n];
[t->tabview removeTabViewItem:i];
// then destroy the bin
uiControlDestroy(uiControl(page));
}
static uintmax_t tabNumPages(uiTab *tt)
{
struct tab *t = (struct tab *) tt;
return [t->pages count];
}
static int tabMargined(uiTab *tt, uintmax_t n)
{
struct tab *t = (struct tab *) tt;
NSNumber *v;
v = (NSNumber *) [t->margined objectAtIndex:n];
return [v intValue];
}
// These are based on measurements from Interface Builder.
// These seem to be based on Auto Layout constants, but I don't see an API that exposes these...
#define tabLeftMargin 17
#define tabTopMargin 3
#define tabRightMargin 17
#define tabBottomMargin 17
// notes:
// top margin of a tab to its parent should be 12, not 20
// our system doesn't allow this...
static void tabSetMargined(uiTab *tt, uintmax_t n, int margined)
{
struct tab *t = (struct tab *) tt;
NSNumber *v;
NSValue *pagev;
uiBin *page;
v = [NSNumber numberWithInt:margined];
[t->margined replaceObjectAtIndex:n withObject:v];
pagev = (NSValue *) [t->pages objectAtIndex:n];
page = (uiBin *) [pagev pointerValue];
if ([v intValue])
uiBinSetMargins(page, tabLeftMargin, tabTopMargin, tabRightMargin, tabBottomMargin);
else
uiBinSetMargins(page, 0, 0, 0, 0);
uiContainerUpdate(uiContainer(page));
}
uiTab *uiNewTab(void)
{
struct tab *t;
t = uiNew(struct tab);
uiDarwinMakeControl(uiControl(t), [NSTabView class], NO, NO, destroy, t);
t->tabview = (NSTabView *) uiControlHandle(uiControl(t));
// also good for NSTabView (same selector and everything)
setStandardControlFont((NSControl *) (t->tabview));
t->pages = [NSMutableArray new];
t->margined = [NSMutableArray new];
uiControl(t)->PreferredSize = tabPreferredSize;
t->baseEnable = uiControl(t)->Enable;
uiControl(t)->Enable = tabEnable;
t->baseDisable = uiControl(t)->Disable;
uiControl(t)->Disable = tabDisable;
t->baseSysFunc = uiControl(t)->SysFunc;
uiControl(t)->SysFunc = tabSysFunc;
uiTab(t)->AppendPage = tabAppendPage;
uiTab(t)->InsertPageBefore = tabInsertPageBefore;
uiTab(t)->DeletePage = tabDeletePage;
uiTab(t)->NumPages = tabNumPages;
uiTab(t)->Margined = tabMargined;
uiTab(t)->SetMargined = tabSetMargined;
return uiTab(t);
}