shithub: libnate

ref: aab184987c79fc7a8e5467200f9ac448700b797f
dir: /n_hbox.c/

View raw version
#include <u.h>
#include <libc.h>
#include <draw.h>
#include <event.h>
#include "nate.h"
#include "nate_construct.h"
#include "n_hbox.h"

#define N_TYPE NHBox_Type
char* NHBox_Type = "NHBox";

typedef struct csizep csizep;
struct csizep {
	Rectangle crect;
	Image *screen;
	int autowidth;
	int fillheight;
};

static void
childsize(Nelem* nelem, int, void *aux)
{
	Point pt;
	Rectangle r;
	csizep *p = (csizep*)aux;
	
	if (!nelem->slot.fill&FILLX || !nelem->slot.fill&FILLY)
		pt = ncalldesiredsize(nelem, p->screen);
	
	if (nelem->slot.fill&FILLX)
		pt.x = p->autowidth;
	if (nelem->slot.fill&FILLY)
		pt.y = p->fillheight;
	
	r = p->crect;
	r.max = addpt(r.min, pt);
	ncallcalcrect(nelem, p->screen, r);
	p->crect.min.x = r.max.x;
}

typedef struct dprepassp dprepassp;
struct dprepassp {
	Image *screen;
	int numfill;
	int fixedwidth;
};

static void
sizeprepass(Nelem *nelem, int, void *aux)
{
	Point pt;
	dprepassp *p = (dprepassp*)aux;
	if (nelem->slot.fill&FILLX) {
		p->numfill++;
		return;
	}
	pt = ncalldesiredsize(nelem, p->screen);
	p->fixedwidth += pt.x;
}

static Rectangle
hbox_calcrect(Nelem* nelem, Image* screen, Rectangle r)
{
	csizep cp;
	dprepassp pp;
	NHBox* b = (NHBox*)nelem;
	GUARD(b);
	
	nelem->slot.r = r;
	
	pp.screen = screen;
	pp.numfill = 0;
	pp.fixedwidth = 0;
	
	lforeach(&b->children, sizeprepass, &pp);
	
	cp.crect = r;
	cp.screen = screen;
	cp.fillheight = Dy(r);
	cp.autowidth = Dx(r) - pp.fixedwidth;
	if (pp.numfill)
		cp.autowidth /= pp.numfill;
	
	lforeach(&b->children, childsize, &cp);

	return nelem->slot.r;
}

typedef struct cdsizep cdsizep;
struct cdsizep {
	Image *screen;
	Point size;
};

static void
dsizechild(Nelem *el, int, void *aux)
{
	Point pt;
	cdsizep *p = (cdsizep*)aux;
	pt = ncalldesiredsize(el, p->screen);
	if (pt.y > p->size.y)
		p->size.y = pt.y;
	p->size.x += pt.x;
}

static Point
hbox_desiredsize(Nelem *nelem, Image *screen)
{
	cdsizep p;
	NHBox *b = (NHBox*)nelem;
	GUARD(b);
	
	p.screen = screen;
	p.size = ZP;
	lforeach(&b->children, dsizechild, &p);
	return p.size;
}

static void
hbox_childdraw(Nelem* elem, int, void *aux)
{
	ncalldraw(elem, (Image*)aux);
}

static void
hbox_draw(Nelem* nelem, Image* img)
{
	NHBox* b = (NHBox*)nelem;
	GUARD(b);
	
	lforeach(&b->children, hbox_childdraw, img);
}

static Nelemfunctions Nhboxfunctions = {
	.calcrect = hbox_calcrect,
	.desiredsize = hbox_desiredsize,
	.draw = hbox_draw,
};

#define NTYPE NHBox
#define NACCS NHBoxAccessors

DEF_SLOTFUNC(hbox_slot);

static NACCS*
hbox_autoheight(int h)
{
	NHBox *b = (NHBox*)nc_get();
	GUARD(b);
	if (h)
		b->slot.fill |= FILLX;
	else
		if (b->slot.fill & FILLX)
			b->slot.fill -= FILLX;
	return (NACCS*)b->accs;
}

static NACCS accs = {
	.Slot = hbox_slot,
	.AutoHeight = hbox_autoheight,
};

NACCS*
New_HBox(char *name)
{
	NHBox *b = MakeNelem(NHBox, NHBox_Type, &Nhboxfunctions, &accs, name, -1);
	
	linit(&b->children);
	b->autoheight = 0;
	nc_push(b);
	return (NACCS*)b->accs;
}