1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 09:58:01 +00:00
putty-source/mac/macctrls.c
Ben Harris feac9298a0 Change the way that panel-switching works so that we only hide the panel
we're switching from and show the panel we're switching to, rather than
iterating over all the panels, hiding and showing them as appropriate.
This has that consequence that all controls have to be created invisible,
since they no longer get hidden after creation.  As usual, the scroll
bars on list boxes have a special hack of their own.

[originally from svn r3183]
2003-05-10 20:23:23 +00:00

1942 lines
51 KiB
C

/* $Id: macctrls.c,v 1.41 2003/05/10 20:23:23 ben Exp $ */
/*
* Copyright (c) 2003 Ben Harris
* All rights reserved.
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use,
* copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following
* conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
* ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
* CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
#include <MacTypes.h>
#include <Appearance.h>
#include <ColorPicker.h>
#include <Controls.h>
#include <ControlDefinitions.h>
#include <Events.h>
#include <Lists.h>
#include <Menus.h>
#include <Resources.h>
#include <Script.h>
#include <Sound.h>
#include <TextEdit.h>
#include <TextUtils.h>
#include <ToolUtils.h>
#include <Windows.h>
#include <assert.h>
#include <string.h>
#include "putty.h"
#include "mac.h"
#include "macresid.h"
#include "dialog.h"
#include "tree234.h"
/* Range of menu IDs for popup menus */
#define MENU_MIN 1024
#define MENU_MAX 2048
union macctrl {
struct macctrl_generic {
enum {
MACCTRL_TEXT,
MACCTRL_EDITBOX,
MACCTRL_RADIO,
MACCTRL_CHECKBOX,
MACCTRL_BUTTON,
MACCTRL_LISTBOX,
MACCTRL_POPUP
} type;
/* Template from which this was generated */
union control *ctrl;
/* Next control in this panel */
union macctrl *next;
void *privdata;
int freeprivdata;
} generic;
struct {
struct macctrl_generic generic;
ControlRef tbctrl;
} text;
struct {
struct macctrl_generic generic;
ControlRef tbctrl;
ControlRef tblabel;
} editbox;
struct {
struct macctrl_generic generic;
ControlRef *tbctrls;
ControlRef tblabel;
} radio;
struct {
struct macctrl_generic generic;
ControlRef tbctrl;
} checkbox;
struct {
struct macctrl_generic generic;
ControlRef tbctrl;
ControlRef tbring;
} button;
struct {
struct macctrl_generic generic;
ControlRef tbctrl;
ListHandle list;
unsigned int nids;
int *ids;
} listbox;
struct {
struct macctrl_generic generic;
ControlRef tbctrl;
MenuRef menu;
int menuid;
unsigned int nids;
int *ids;
} popup;
};
struct mac_layoutstate {
Point pos;
unsigned int width;
unsigned int panelnum;
};
#define ctrlevent(mcs, mc, event) do { \
if ((mc)->generic.ctrl->generic.handler != NULL) \
(*(mc)->generic.ctrl->generic.handler)((mc)->generic.ctrl, (mcs),\
(mcs)->data, (event)); \
} while (0)
#define findbyctrl(mcs, ctrl) \
find234((mcs)->byctrl, (ctrl), macctrl_cmp_byctrl_find)
static void macctrl_layoutset(struct mac_layoutstate *, struct controlset *,
WindowPtr, struct macctrls *);
static void macctrl_hideshowpanel(struct macctrls *, unsigned int, int);
static void macctrl_switchtopanel(struct macctrls *, unsigned int);
static void macctrl_setfocus(struct macctrls *, union macctrl *);
static void macctrl_text(struct macctrls *, WindowPtr,
struct mac_layoutstate *, union control *);
static void macctrl_editbox(struct macctrls *, WindowPtr,
struct mac_layoutstate *, union control *);
static void macctrl_radio(struct macctrls *, WindowPtr,
struct mac_layoutstate *, union control *);
static void macctrl_checkbox(struct macctrls *, WindowPtr,
struct mac_layoutstate *, union control *);
static void macctrl_button(struct macctrls *, WindowPtr,
struct mac_layoutstate *, union control *);
static void macctrl_listbox(struct macctrls *, WindowPtr,
struct mac_layoutstate *, union control *);
static void macctrl_popup(struct macctrls *, WindowPtr,
struct mac_layoutstate *, union control *);
#if !TARGET_API_MAC_CARBON
static pascal SInt32 macctrl_sys7_editbox_cdef(SInt16, ControlRef,
ControlDefProcMessage, SInt32);
static pascal SInt32 macctrl_sys7_default_cdef(SInt16, ControlRef,
ControlDefProcMessage, SInt32);
static pascal SInt32 macctrl_sys7_listbox_cdef(SInt16, ControlRef,
ControlDefProcMessage, SInt32);
#endif
#if !TARGET_API_MAC_CARBON
/*
* This trick enables us to keep all the CDEF code in the main
* application, which makes life easier. For details, see
* <http://developer.apple.com/technotes/tn/tn2003.html#custom_code_base>.
*/
#pragma options align=mac68k
typedef struct {
short jmpabs; /* 4EF9 */
ControlDefUPP theUPP;
} **PatchCDEF;
#pragma options align=reset
#endif
static void macctrl_init()
{
#if !TARGET_API_MAC_CARBON
static int inited = 0;
PatchCDEF cdef;
if (inited) return;
cdef = (PatchCDEF)GetResource(kControlDefProcResourceType, CDEF_EditBox);
(*cdef)->theUPP = NewControlDefProc(macctrl_sys7_editbox_cdef);
cdef = (PatchCDEF)GetResource(kControlDefProcResourceType, CDEF_Default);
(*cdef)->theUPP = NewControlDefProc(macctrl_sys7_default_cdef);
cdef = (PatchCDEF)GetResource(kControlDefProcResourceType, CDEF_ListBox);
(*cdef)->theUPP = NewControlDefProc(macctrl_sys7_listbox_cdef);
inited = 1;
#endif
}
static int macctrl_cmp_byctrl(void *av, void *bv)
{
union macctrl *a = (union macctrl *)av;
union macctrl *b = (union macctrl *)bv;
if (a->generic.ctrl < b->generic.ctrl)
return -1;
else if (a->generic.ctrl > b->generic.ctrl)
return +1;
else
return 0;
}
static int macctrl_cmp_byctrl_find(void *av, void *bv)
{
union control *a = (union control *)av;
union macctrl *b = (union macctrl *)bv;
if (a < b->generic.ctrl)
return -1;
else if (a > b->generic.ctrl)
return +1;
else
return 0;
}
static union control panellist;
static void panellist_handler(union control *ctrl, void *dlg, void *data,
int event)
{
struct macctrls *mcs = dlg;
/* XXX what if there's no selection? */
if (event == EVENT_SELCHANGE)
macctrl_switchtopanel(mcs, dlg_listbox_index(ctrl, dlg) + 1);
}
void macctrl_layoutbox(struct controlbox *cb, WindowPtr window,
struct macctrls *mcs)
{
int i;
struct mac_layoutstate curstate;
ControlRef root;
Rect rect;
macctrl_init();
if (mac_gestalts.apprvers >= 0x100)
CreateRootControl(window, &root);
#if TARGET_API_MAC_CARBON
GetPortBounds(GetWindowPort(window), &rect);
#else
rect = window->portRect;
#endif
mcs->window = window;
mcs->byctrl = newtree234(macctrl_cmp_byctrl);
mcs->focus = NULL;
mcs->defbutton = NULL;
mcs->canbutton = NULL;
/* Count the number of panels */
mcs->npanels = 1;
for (i = 1; i < cb->nctrlsets; i++)
if (strcmp(cb->ctrlsets[i]->pathname, cb->ctrlsets[i-1]->pathname))
mcs->npanels++;
mcs->panels = snewn(mcs->npanels, union macctrl *);
memset(mcs->panels, 0, sizeof(*mcs->panels) * mcs->npanels);
curstate.panelnum = 0;
curstate.pos.h = rect.left + 13;
curstate.pos.v = rect.top + 13;
curstate.width = 160;
panellist.listbox.type = CTRL_LISTBOX;
panellist.listbox.handler = &panellist_handler;
panellist.listbox.height = 20;
panellist.listbox.percentwidth = 100;
macctrl_listbox(mcs, window, &curstate, &panellist);
/* XXX Start with panel 1 active */
curstate.pos.h = rect.left + 13 + 160 + 13;
curstate.pos.v = rect.bottom - 33;
curstate.width = rect.right - (rect.left + 13 + 160) - (13 * 2);
for (i = 0; i < cb->nctrlsets; i++) {
if (i > 0 && strcmp(cb->ctrlsets[i]->pathname,
cb->ctrlsets[i-1]->pathname)) {
curstate.pos.v = rect.top + 13;
curstate.panelnum++;
assert(curstate.panelnum < mcs->npanels);
dlg_listbox_add(&panellist, mcs, cb->ctrlsets[i]->pathname);
}
macctrl_layoutset(&curstate, cb->ctrlsets[i], window, mcs);
}
macctrl_switchtopanel(mcs, 1);
macctrl_hideshowpanel(mcs, 0, TRUE);
/* 14 = proxies, 19 = portfwd, 20 = SSH bugs */
}
#define MAXCOLS 16
static void macctrl_layoutset(struct mac_layoutstate *curstate,
struct controlset *s,
WindowPtr window, struct macctrls *mcs)
{
unsigned int i, j, ncols, colstart;
struct mac_layoutstate cols[MAXCOLS];
cols[0] = *curstate;
ncols = 1;
for (i = 0; i < s->ncontrols; i++) {
union control *ctrl = s->ctrls[i];
colstart = COLUMN_START(ctrl->generic.column);
switch (ctrl->generic.type) {
case CTRL_COLUMNS:
if (ctrl->columns.ncols != 1) {
ncols = ctrl->columns.ncols;
assert(ncols <= MAXCOLS);
for (j = 0; j < ncols; j++) {
cols[j] = cols[0];
if (j > 0)
cols[j].pos.h = cols[j-1].pos.h + cols[j-1].width + 6;
if (j == ncols - 1)
cols[j].width = curstate->width -
(cols[j].pos.h - curstate->pos.h);
else
cols[j].width = (curstate->width + 6) *
ctrl->columns.percentages[j] / 100 - 6;
}
} else {
for (j = 0; j < ncols; j++)
if (cols[j].pos.v > cols[0].pos.v)
cols[0].pos.v = cols[j].pos.v;
cols[0].width = curstate->width;
ncols = 1;
}
break;
case CTRL_TEXT:
macctrl_text(mcs, window, &cols[colstart], ctrl);
break;
case CTRL_EDITBOX:
macctrl_editbox(mcs, window, &cols[colstart], ctrl);
break;
case CTRL_RADIO:
macctrl_radio(mcs, window, &cols[colstart], ctrl);
break;
case CTRL_CHECKBOX:
macctrl_checkbox(mcs, window, &cols[colstart], ctrl);
break;
case CTRL_BUTTON:
macctrl_button(mcs, window, &cols[colstart], ctrl);
break;
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0)
macctrl_popup(mcs, window, &cols[colstart], ctrl);
else
macctrl_listbox(mcs, window, &cols[colstart], ctrl);
break;
}
}
for (j = 0; j < ncols; j++)
if (cols[j].pos.v > curstate->pos.v)
curstate->pos.v = cols[j].pos.v;
}
static void macctrl_hideshowpanel(struct macctrls *mcs, unsigned int panel,
int showit)
{
union macctrl *mc;
int j;
#define hideshow(c) do { \
if (showit) ShowControl(c); else HideControl(c); \
} while (0)
for (mc = mcs->panels[panel]; mc != NULL; mc = mc->generic.next) {
#if !TARGET_API_MAC_CARBON
if (mcs->focus == mc)
macctrl_setfocus(mcs, NULL);
#endif
switch (mc->generic.type) {
case MACCTRL_TEXT:
hideshow(mc->text.tbctrl);
break;
case MACCTRL_EDITBOX:
hideshow(mc->editbox.tbctrl);
if (mc->editbox.tblabel != NULL)
hideshow(mc->editbox.tblabel);
break;
case MACCTRL_RADIO:
for (j = 0; j < mc->generic.ctrl->radio.nbuttons; j++)
hideshow(mc->radio.tbctrls[j]);
if (mc->radio.tblabel != NULL)
hideshow(mc->radio.tblabel);
break;
case MACCTRL_CHECKBOX:
hideshow(mc->checkbox.tbctrl);
break;
case MACCTRL_BUTTON:
hideshow(mc->button.tbctrl);
if (mc->button.tbring != NULL)
hideshow(mc->button.tbring);
break;
case MACCTRL_LISTBOX:
hideshow(mc->listbox.tbctrl);
/*
* At least under Mac OS 8.1, hiding a list box
* doesn't hide its scroll bars.
*/
#if TARGET_API_MAC_CARBON
hideshow(GetListVerticalScrollBar(mc->listbox.list));
#else
hideshow((*mc->listbox.list)->vScroll);
#endif
break;
case MACCTRL_POPUP:
hideshow(mc->popup.tbctrl);
break;
}
}
}
static void macctrl_switchtopanel(struct macctrls *mcs, unsigned int which)
{
macctrl_hideshowpanel(mcs, mcs->curpanel, FALSE);
macctrl_hideshowpanel(mcs, which, TRUE);
mcs->curpanel = which;
}
#if !TARGET_API_MAC_CARBON
/*
* System 7 focus manipulation
*/
static void macctrl_defocus(union macctrl *mc)
{
assert(mac_gestalts.apprvers < 0x100);
switch (mc->generic.type) {
case MACCTRL_EDITBOX:
TEDeactivate((TEHandle)(*mc->editbox.tbctrl)->contrlData);
break;
}
}
static void macctrl_enfocus(union macctrl *mc)
{
assert(mac_gestalts.apprvers < 0x100);
switch (mc->generic.type) {
case MACCTRL_EDITBOX:
TEActivate((TEHandle)(*mc->editbox.tbctrl)->contrlData);
break;
}
}
static void macctrl_setfocus(struct macctrls *mcs, union macctrl *mc)
{
if (mcs->focus == mc)
return;
if (mcs->focus != NULL)
macctrl_defocus(mcs->focus);
mcs->focus = mc;
if (mc != NULL)
macctrl_enfocus(mc);
}
#endif
static void macctrl_text(struct macctrls *mcs, WindowPtr window,
struct mac_layoutstate *curstate,
union control *ctrl)
{
union macctrl *mc = snew(union macctrl);
Rect bounds;
SInt16 height;
assert(ctrl->text.label != NULL);
mc->generic.type = MACCTRL_TEXT;
mc->generic.ctrl = ctrl;
mc->generic.privdata = NULL;
bounds.left = curstate->pos.h;
bounds.right = bounds.left + curstate->width;
bounds.top = curstate->pos.v;
bounds.bottom = bounds.top + 16;
if (mac_gestalts.apprvers >= 0x100) {
Size olen;
mc->text.tbctrl = NewControl(window, &bounds, NULL, FALSE, 0, 0, 0,
kControlStaticTextProc, (long)mc);
SetControlData(mc->text.tbctrl, kControlEntireControl,
kControlStaticTextTextTag,
strlen(ctrl->text.label), ctrl->text.label);
GetControlData(mc->text.tbctrl, kControlEntireControl,
kControlStaticTextTextHeightTag,
sizeof(height), &height, &olen);
}
#if !TARGET_API_MAC_CARBON
else {
TEHandle te;
mc->text.tbctrl = NewControl(window, &bounds, NULL, FALSE, 0, 0, 0,
SYS7_TEXT_PROC, (long)mc);
te = (TEHandle)(*mc->text.tbctrl)->contrlData;
TESetText(ctrl->text.label, strlen(ctrl->text.label), te);
height = TEGetHeight(1, (*te)->nLines, te);
}
#endif
SizeControl(mc->text.tbctrl, curstate->width, height);
curstate->pos.v += height + 6;
add234(mcs->byctrl, mc);
mc->generic.next = mcs->panels[curstate->panelnum];
mcs->panels[curstate->panelnum] = mc;
}
static void macctrl_editbox(struct macctrls *mcs, WindowPtr window,
struct mac_layoutstate *curstate,
union control *ctrl)
{
union macctrl *mc = snew(union macctrl);
Rect lbounds, bounds;
mc->generic.type = MACCTRL_EDITBOX;
mc->generic.ctrl = ctrl;
mc->generic.privdata = NULL;
lbounds.left = curstate->pos.h;
lbounds.top = curstate->pos.v;
if (ctrl->editbox.percentwidth == 100) {
if (ctrl->editbox.label != NULL) {
lbounds.right = lbounds.left + curstate->width;
lbounds.bottom = lbounds.top + 16;
curstate->pos.v += 18;
}
bounds.left = curstate->pos.h;
bounds.right = bounds.left + curstate->width;
} else {
lbounds.right = lbounds.left +
curstate->width * (100 - ctrl->editbox.percentwidth) / 100;
lbounds.bottom = lbounds.top + 22;
bounds.left = lbounds.right;
bounds.right = lbounds.left + curstate->width;
}
bounds.top = curstate->pos.v;
bounds.bottom = bounds.top + 22;
if (mac_gestalts.apprvers >= 0x100) {
if (ctrl->editbox.label == NULL)
mc->editbox.tblabel = NULL;
else {
mc->editbox.tblabel = NewControl(window, &lbounds, NULL, FALSE,
0, 0, 0, kControlStaticTextProc,
(long)mc);
SetControlData(mc->editbox.tblabel, kControlEntireControl,
kControlStaticTextTextTag,
strlen(ctrl->editbox.label), ctrl->editbox.label);
}
InsetRect(&bounds, 3, 3);
mc->editbox.tbctrl = NewControl(window, &bounds, NULL, FALSE, 0, 0, 0,
ctrl->editbox.password ?
kControlEditTextPasswordProc :
kControlEditTextProc, (long)mc);
}
#if !TARGET_API_MAC_CARBON
else {
if (ctrl->editbox.label == NULL)
mc->editbox.tblabel = NULL;
else {
mc->editbox.tblabel = NewControl(window, &lbounds, NULL, FALSE,
0, 0, 0, SYS7_TEXT_PROC,
(long)mc);
TESetText(ctrl->editbox.label, strlen(ctrl->editbox.label),
(TEHandle)(*mc->editbox.tblabel)->contrlData);
}
mc->editbox.tbctrl = NewControl(window, &bounds, NULL, FALSE, 0, 0, 0,
SYS7_EDITBOX_PROC, (long)mc);
}
#endif
curstate->pos.v += 28;
add234(mcs->byctrl, mc);
mc->generic.next = mcs->panels[curstate->panelnum];
mcs->panels[curstate->panelnum] = mc;
ctrlevent(mcs, mc, EVENT_REFRESH);
}
#if !TARGET_API_MAC_CARBON
static pascal SInt32 macctrl_sys7_editbox_cdef(SInt16 variant,
ControlRef control,
ControlDefProcMessage msg,
SInt32 param)
{
RgnHandle rgn;
Rect rect;
TEHandle te;
long ssfs;
Point mouse;
switch (msg) {
case initCntl:
rect = (*control)->contrlRect;
if (variant == SYS7_EDITBOX_VARIANT)
InsetRect(&rect, 3, 3); /* 2 if it's 20 pixels high */
te = TENew(&rect, &rect);
ssfs = GetScriptVariable(smSystemScript, smScriptSysFondSize);
(*te)->txSize = LoWord(ssfs);
(*te)->txFont = HiWord(ssfs);
(*control)->contrlData = (Handle)te;
return noErr;
case dispCntl:
TEDispose((TEHandle)(*control)->contrlData);
return 0;
case drawCntl:
if ((*control)->contrlVis) {
rect = (*control)->contrlRect;
if (variant == SYS7_EDITBOX_VARIANT) {
PenNormal();
FrameRect(&rect);
InsetRect(&rect, 3, 3);
}
EraseRect(&rect);
(*(TEHandle)(*control)->contrlData)->viewRect = rect;
TEUpdate(&rect, (TEHandle)(*control)->contrlData);
}
return 0;
case testCntl:
if (variant == SYS7_TEXT_VARIANT)
return kControlNoPart;
mouse.h = LoWord(param);
mouse.v = HiWord(param);
rect = (*control)->contrlRect;
InsetRect(&rect, 3, 3);
return PtInRect(mouse, &rect) ? kControlEditTextPart : kControlNoPart;
case calcCRgns:
if (param & (1 << 31)) {
param &= ~(1 << 31);
goto calcthumbrgn;
}
/* FALLTHROUGH */
case calcCntlRgn:
rgn = (RgnHandle)param;
RectRgn(rgn, &(*control)->contrlRect);
return 0;
case calcThumbRgn:
calcthumbrgn:
rgn = (RgnHandle)param;
SetEmptyRgn(rgn);
return 0;
}
return 0;
}
#endif
static void macctrl_radio(struct macctrls *mcs, WindowPtr window,
struct mac_layoutstate *curstate,
union control *ctrl)
{
union macctrl *mc = snew(union macctrl);
Rect bounds;
Str255 title;
unsigned int i, colwidth;
mc->generic.type = MACCTRL_RADIO;
mc->generic.ctrl = ctrl;
mc->generic.privdata = NULL;
mc->radio.tbctrls = snewn(ctrl->radio.nbuttons, ControlRef);
colwidth = (curstate->width + 13) / ctrl->radio.ncolumns;
bounds.top = curstate->pos.v;
bounds.bottom = bounds.top + 16;
bounds.left = curstate->pos.h;
bounds.right = bounds.left + curstate->width;
if (ctrl->radio.label == NULL)
mc->radio.tblabel = NULL;
else {
if (mac_gestalts.apprvers >= 0x100) {
mc->radio.tblabel = NewControl(window, &bounds, NULL, FALSE,
0, 0, 0, kControlStaticTextProc,
(long)mc);
SetControlData(mc->radio.tblabel, kControlEntireControl,
kControlStaticTextTextTag,
strlen(ctrl->radio.label), ctrl->radio.label);
}
#if !TARGET_API_MAC_CARBON
else {
mc->radio.tblabel = NewControl(window, &bounds, NULL, FALSE,
0, 0, 0, SYS7_TEXT_PROC, (long)mc);
TESetText(ctrl->radio.label, strlen(ctrl->radio.label),
(TEHandle)(*mc->radio.tblabel)->contrlData);
}
#endif
curstate->pos.v += 18;
}
for (i = 0; i < ctrl->radio.nbuttons; i++) {
bounds.top = curstate->pos.v - 2;
bounds.bottom = bounds.top + 18;
bounds.left = curstate->pos.h + colwidth * (i % ctrl->radio.ncolumns);
if (i == ctrl->radio.nbuttons - 1 ||
i % ctrl->radio.ncolumns == ctrl->radio.ncolumns - 1) {
bounds.right = curstate->pos.h + curstate->width;
curstate->pos.v += 18;
} else
bounds.right = bounds.left + colwidth - 13;
c2pstrcpy(title, ctrl->radio.buttons[i]);
mc->radio.tbctrls[i] = NewControl(window, &bounds, title, FALSE,
0, 0, 1, radioButProc, (long)mc);
}
curstate->pos.v += 4;
add234(mcs->byctrl, mc);
mc->generic.next = mcs->panels[curstate->panelnum];
mcs->panels[curstate->panelnum] = mc;
ctrlevent(mcs, mc, EVENT_REFRESH);
}
static void macctrl_checkbox(struct macctrls *mcs, WindowPtr window,
struct mac_layoutstate *curstate,
union control *ctrl)
{
union macctrl *mc = snew(union macctrl);
Rect bounds;
Str255 title;
assert(ctrl->checkbox.label != NULL);
mc->generic.type = MACCTRL_CHECKBOX;
mc->generic.ctrl = ctrl;
mc->generic.privdata = NULL;
bounds.left = curstate->pos.h;
bounds.right = bounds.left + curstate->width;
bounds.top = curstate->pos.v;
bounds.bottom = bounds.top + 16;
c2pstrcpy(title, ctrl->checkbox.label);
mc->checkbox.tbctrl = NewControl(window, &bounds, title, FALSE, 0, 0, 1,
checkBoxProc, (long)mc);
add234(mcs->byctrl, mc);
curstate->pos.v += 22;
mc->generic.next = mcs->panels[curstate->panelnum];
mcs->panels[curstate->panelnum] = mc;
ctrlevent(mcs, mc, EVENT_REFRESH);
}
static void macctrl_button(struct macctrls *mcs, WindowPtr window,
struct mac_layoutstate *curstate,
union control *ctrl)
{
union macctrl *mc = snew(union macctrl);
Rect bounds;
Str255 title;
assert(ctrl->button.label != NULL);
mc->generic.type = MACCTRL_BUTTON;
mc->generic.ctrl = ctrl;
mc->generic.privdata = NULL;
bounds.left = curstate->pos.h;
bounds.right = bounds.left + curstate->width;
bounds.top = curstate->pos.v;
bounds.bottom = bounds.top + 20;
c2pstrcpy(title, ctrl->button.label);
mc->button.tbctrl = NewControl(window, &bounds, title, FALSE, 0, 0, 1,
pushButProc, (long)mc);
mc->button.tbring = NULL;
if (mac_gestalts.apprvers >= 0x100) {
Boolean isdefault = ctrl->button.isdefault;
SetControlData(mc->button.tbctrl, kControlEntireControl,
kControlPushButtonDefaultTag,
sizeof(isdefault), &isdefault);
} else if (ctrl->button.isdefault) {
InsetRect(&bounds, -4, -4);
mc->button.tbring = NewControl(window, &bounds, title, FALSE, 0, 0, 1,
SYS7_DEFAULT_PROC, (long)mc);
}
if (mac_gestalts.apprvers >= 0x110) {
Boolean iscancel = ctrl->button.iscancel;
SetControlData(mc->button.tbctrl, kControlEntireControl,
kControlPushButtonCancelTag,
sizeof(iscancel), &iscancel);
}
if (ctrl->button.isdefault)
mcs->defbutton = mc;
if (ctrl->button.iscancel)
mcs->canbutton = mc;
add234(mcs->byctrl, mc);
mc->generic.next = mcs->panels[curstate->panelnum];
mcs->panels[curstate->panelnum] = mc;
curstate->pos.v += 26;
}
#if !TARGET_API_MAC_CARBON
static pascal SInt32 macctrl_sys7_default_cdef(SInt16 variant,
ControlRef control,
ControlDefProcMessage msg,
SInt32 param)
{
RgnHandle rgn;
Rect rect;
int oval;
PenState savestate;
switch (msg) {
case drawCntl:
if ((*control)->contrlVis) {
rect = (*control)->contrlRect;
GetPenState(&savestate);
PenNormal();
PenSize(3, 3);
if ((*control)->contrlHilite == kControlInactivePart)
PenPat(&qd.gray);
oval = (rect.bottom - rect.top) / 2 + 2;
FrameRoundRect(&rect, oval, oval);
SetPenState(&savestate);
}
return 0;
case calcCRgns:
if (param & (1 << 31)) {
param &= ~(1 << 31);
goto calcthumbrgn;
}
/* FALLTHROUGH */
case calcCntlRgn:
rgn = (RgnHandle)param;
RectRgn(rgn, &(*control)->contrlRect);
return 0;
case calcThumbRgn:
calcthumbrgn:
rgn = (RgnHandle)param;
SetEmptyRgn(rgn);
return 0;
}
return 0;
}
#endif
static void macctrl_listbox(struct macctrls *mcs, WindowPtr window,
struct mac_layoutstate *curstate,
union control *ctrl)
{
union macctrl *mc = snew(union macctrl);
Rect bounds;
Size olen;
/* XXX Use label */
assert(ctrl->listbox.percentwidth == 100);
mc->generic.type = MACCTRL_LISTBOX;
mc->generic.ctrl = ctrl;
mc->generic.privdata = NULL;
/* The list starts off empty */
mc->listbox.nids = 0;
mc->listbox.ids = NULL;
bounds.left = curstate->pos.h;
bounds.right = bounds.left + curstate->width;
bounds.top = curstate->pos.v;
bounds.bottom = bounds.top + 16 * ctrl->listbox.height + 2;
if (mac_gestalts.apprvers >= 0x100) {
InsetRect(&bounds, 3, 3);
mc->listbox.tbctrl = NewControl(window, &bounds, NULL, FALSE,
ldes_Default, 0, 0,
kControlListBoxProc, (long)mc);
if (GetControlData(mc->listbox.tbctrl, kControlEntireControl,
kControlListBoxListHandleTag,
sizeof(mc->listbox.list), &mc->listbox.list,
&olen) != noErr) {
DisposeControl(mc->listbox.tbctrl);
sfree(mc);
return;
}
}
#if !TARGET_API_MAC_CARBON
else {
InsetRect(&bounds, -3, -3);
mc->listbox.tbctrl = NewControl(window, &bounds, NULL, FALSE,
0, 0, 0,
SYS7_LISTBOX_PROC, (long)mc);
mc->listbox.list = (ListHandle)(*mc->listbox.tbctrl)->contrlData;
(*mc->listbox.list)->refCon = (long)mc;
}
#endif
if (!ctrl->listbox.multisel) {
#if TARGET_API_MAC_CARBON
SetListSelectionFlags(mc->listbox.list, lOnlyOne);
#else
(*mc->listbox.list)->selFlags = lOnlyOne;
#endif
}
add234(mcs->byctrl, mc);
curstate->pos.v += 6 + 16 * ctrl->listbox.height + 2;
mc->generic.next = mcs->panels[curstate->panelnum];
mcs->panels[curstate->panelnum] = mc;
ctrlevent(mcs, mc, EVENT_REFRESH);
#if TARGET_API_MAC_CARBON
HideControl(GetListVerticalScrollBar(mc->listbox.list));
#else
HideControl((*mc->listbox.list)->vScroll);
#endif
}
#if !TARGET_API_MAC_CARBON
static pascal SInt32 macctrl_sys7_listbox_cdef(SInt16 variant,
ControlRef control,
ControlDefProcMessage msg,
SInt32 param)
{
RgnHandle rgn;
Rect rect;
ListHandle list;
long ssfs;
Point mouse;
ListBounds bounds;
Point csize;
short savefont;
short savesize;
GrafPtr curport;
switch (msg) {
case initCntl:
rect = (*control)->contrlRect;
InsetRect(&rect, 4, 4);
rect.right -= 15; /* scroll bar */
bounds.top = bounds.bottom = bounds.left = 0;
bounds.right = 1;
csize.h = csize.v = 0;
GetPort(&curport);
savefont = curport->txFont;
savesize = curport->txSize;
ssfs = GetScriptVariable(smSystemScript, smScriptSysFondSize);
TextFont(HiWord(ssfs));
TextSize(LoWord(ssfs));
list = LNew(&rect, &bounds, csize, 0, (*control)->contrlOwner,
TRUE, FALSE, FALSE, TRUE);
SetControlReference((*list)->vScroll, (long)list);
(*control)->contrlData = (Handle)list;
TextFont(savefont);
TextSize(savesize);
return noErr;
case dispCntl:
/*
* If the dialogue box is being destroyed, the scroll bar
* might have gone already. In our situation, this is the
* only time we destroy a control, so NULL out the scroll bar
* handle to prevent LDispose trying to free it.
*/
list = (ListHandle)(*control)->contrlData;
(*list)->vScroll = NULL;
LDispose(list);
return 0;
case drawCntl:
if ((*control)->contrlVis) {
rect = (*control)->contrlRect;
/* XXX input focus highlighting? */
InsetRect(&rect, 3, 3);
PenNormal();
FrameRect(&rect);
list = (ListHandle)(*control)->contrlData;
LActivate((*control)->contrlHilite != kControlInactivePart, list);
GetPort(&curport);
LUpdate(curport->visRgn, list);
}
return 0;
case testCntl:
mouse.h = LoWord(param);
mouse.v = HiWord(param);
rect = (*control)->contrlRect;
InsetRect(&rect, 4, 4);
/*
* We deliberately exclude the scrollbar so that LClick() can see it.
*/
rect.right -= 15;
return PtInRect(mouse, &rect) ? kControlListBoxPart : kControlNoPart;
case calcCRgns:
if (param & (1 << 31)) {
param &= ~(1 << 31);
goto calcthumbrgn;
}
/* FALLTHROUGH */
case calcCntlRgn:
rgn = (RgnHandle)param;
RectRgn(rgn, &(*control)->contrlRect);
return 0;
case calcThumbRgn:
calcthumbrgn:
rgn = (RgnHandle)param;
SetEmptyRgn(rgn);
return 0;
}
return 0;
}
#endif
static void macctrl_popup(struct macctrls *mcs, WindowPtr window,
struct mac_layoutstate *curstate,
union control *ctrl)
{
union macctrl *mc = snew(union macctrl);
Rect bounds;
Str255 title;
unsigned int labelwidth;
static int nextmenuid = MENU_MIN;
int menuid;
MenuRef menu;
/*
* <http://developer.apple.com/qa/tb/tb42.html> explains how to
* create a popup menu with dynamic content.
*/
assert(ctrl->listbox.height == 0);
assert(!ctrl->listbox.draglist);
assert(!ctrl->listbox.multisel);
mc->generic.type = MACCTRL_POPUP;
mc->generic.ctrl = ctrl;
mc->generic.privdata = NULL;
c2pstrcpy(title, ctrl->button.label == NULL ? "" : ctrl->button.label);
/* Find a spare menu ID and create the menu */
while (GetMenuHandle(nextmenuid) != NULL)
if (++nextmenuid >= MENU_MAX) nextmenuid = MENU_MIN;
menuid = nextmenuid++;
menu = NewMenu(menuid, "\pdummy");
if (menu == NULL) return;
mc->popup.menu = menu;
mc->popup.menuid = menuid;
InsertMenu(menu, kInsertHierarchicalMenu);
/* The menu starts off empty */
mc->popup.nids = 0;
mc->popup.ids = NULL;
bounds.left = curstate->pos.h;
bounds.right = bounds.left + curstate->width;
bounds.top = curstate->pos.v;
bounds.bottom = bounds.top + 20;
/* XXX handle percentwidth == 100 */
labelwidth = curstate->width * (100 - ctrl->listbox.percentwidth) / 100;
mc->popup.tbctrl = NewControl(window, &bounds, title, FALSE,
popupTitleLeftJust, menuid, labelwidth,
popupMenuProc + popupFixedWidth, (long)mc);
add234(mcs->byctrl, mc);
curstate->pos.v += 26;
mc->generic.next = mcs->panels[curstate->panelnum];
mcs->panels[curstate->panelnum] = mc;
ctrlevent(mcs, mc, EVENT_REFRESH);
}
void macctrl_activate(WindowPtr window, EventRecord *event)
{
struct macctrls *mcs = mac_winctrls(window);
Boolean active = (event->modifiers & activeFlag) != 0;
GrafPtr saveport;
int i, j;
ControlPartCode state;
union macctrl *mc;
GetPort(&saveport);
SetPort((GrafPtr)GetWindowPort(window));
if (mac_gestalts.apprvers >= 0x100)
SetThemeWindowBackground(window, active ?
kThemeBrushModelessDialogBackgroundActive :
kThemeBrushModelessDialogBackgroundInactive,
TRUE);
state = active ? kControlNoPart : kControlInactivePart;
for (i = 0; i <= mcs->curpanel; i += mcs->curpanel)
for (mc = mcs->panels[i]; mc != NULL; mc = mc->generic.next) {
switch (mc->generic.type) {
case MACCTRL_TEXT:
HiliteControl(mc->text.tbctrl, state);
break;
case MACCTRL_EDITBOX:
HiliteControl(mc->editbox.tbctrl, state);
if (mc->editbox.tblabel != NULL)
HiliteControl(mc->editbox.tblabel, state);
break;
case MACCTRL_RADIO:
for (j = 0; j < mc->generic.ctrl->radio.nbuttons; j++)
HiliteControl(mc->radio.tbctrls[j], state);
if (mc->radio.tblabel != NULL)
HiliteControl(mc->radio.tblabel, state);
break;
case MACCTRL_CHECKBOX:
HiliteControl(mc->checkbox.tbctrl, state);
break;
case MACCTRL_BUTTON:
HiliteControl(mc->button.tbctrl, state);
if (mc->button.tbring != NULL)
HiliteControl(mc->button.tbring, state);
break;
case MACCTRL_LISTBOX:
HiliteControl(mc->listbox.tbctrl, state);
break;
case MACCTRL_POPUP:
HiliteControl(mc->popup.tbctrl, state);
break;
}
#if !TARGET_API_MAC_CARBON
if (mcs->focus == mc) {
if (active)
macctrl_enfocus(mc);
else
macctrl_defocus(mc);
}
#endif
}
SetPort(saveport);
}
void macctrl_click(WindowPtr window, EventRecord *event)
{
Point mouse;
ControlHandle control, oldfocus;
int part, trackresult;
GrafPtr saveport;
union macctrl *mc;
struct macctrls *mcs = mac_winctrls(window);
int i;
UInt32 features;
GetPort(&saveport);
SetPort((GrafPtr)GetWindowPort(window));
mouse = event->where;
GlobalToLocal(&mouse);
part = FindControl(mouse, window, &control);
if (control != NULL) {
#if !TARGET_API_MAC_CARBON
/*
* Special magic for scroll bars in list boxes, whose refcon
* is the list.
*/
if (part == kControlUpButtonPart || part == kControlDownButtonPart ||
part == kControlPageUpPart || part == kControlPageDownPart ||
part == kControlIndicatorPart)
mc = (union macctrl *)
(*(ListHandle)GetControlReference(control))->refCon;
else
#endif
mc = (union macctrl *)GetControlReference(control);
if (mac_gestalts.apprvers >= 0x100) {
if (GetControlFeatures(control, &features) == noErr &&
(features & kControlSupportsFocus) &&
(features & kControlGetsFocusOnClick) &&
GetKeyboardFocus(window, &oldfocus) == noErr &&
control != oldfocus)
SetKeyboardFocus(window, control, part);
trackresult = HandleControlClick(control, mouse, event->modifiers,
(ControlActionUPP)-1);
} else {
#if !TARGET_API_MAC_CARBON
if (mc->generic.type == MACCTRL_EDITBOX &&
control == mc->editbox.tbctrl) {
TEHandle te = (TEHandle)(*control)->contrlData;
macctrl_setfocus(mcs, mc);
TEClick(mouse, !!(event->modifiers & shiftKey), te);
goto done;
}
if (mc->generic.type == MACCTRL_LISTBOX &&
(control == mc->listbox.tbctrl ||
control == (*mc->listbox.list)->vScroll)) {
macctrl_setfocus(mcs, mc);
if (LClick(mouse, event->modifiers, mc->listbox.list))
/* double-click */
ctrlevent(mcs, mc, EVENT_ACTION);
else
ctrlevent(mcs, mc, EVENT_SELCHANGE);
goto done;
}
#endif
trackresult = TrackControl(control, mouse, (ControlActionUPP)-1);
}
switch (mc->generic.type) {
case MACCTRL_RADIO:
if (trackresult != 0) {
for (i = 0; i < mc->generic.ctrl->radio.nbuttons; i++)
if (mc->radio.tbctrls[i] == control)
SetControlValue(mc->radio.tbctrls[i],
kControlRadioButtonCheckedValue);
else
SetControlValue(mc->radio.tbctrls[i],
kControlRadioButtonUncheckedValue);
ctrlevent(mcs, mc, EVENT_VALCHANGE);
}
break;
case MACCTRL_CHECKBOX:
if (trackresult != 0) {
SetControlValue(control, !GetControlValue(control));
ctrlevent(mcs, mc, EVENT_VALCHANGE);
}
break;
case MACCTRL_BUTTON:
if (trackresult != 0)
ctrlevent(mcs, mc, EVENT_ACTION);
break;
case MACCTRL_LISTBOX:
/* FIXME spot double-click */
ctrlevent(mcs, mc, EVENT_SELCHANGE);
break;
case MACCTRL_POPUP:
ctrlevent(mcs, mc, EVENT_SELCHANGE);
break;
}
}
done:
SetPort(saveport);
}
void macctrl_key(WindowPtr window, EventRecord *event)
{
ControlRef control;
struct macctrls *mcs = mac_winctrls(window);
union macctrl *mc;
unsigned long dummy;
switch (event->message & charCodeMask) {
case kEnterCharCode:
case kReturnCharCode:
if (mcs->defbutton != NULL) {
assert(mcs->defbutton->generic.type == MACCTRL_BUTTON);
HiliteControl(mcs->defbutton->button.tbctrl, kControlButtonPart);
/*
* I'd like to delay unhilighting the button until after
* the event has been processed, but by them the entire
* dialgue box might have been destroyed.
*/
Delay(6, &dummy);
HiliteControl(mcs->defbutton->button.tbctrl, kControlNoPart);
ctrlevent(mcs, mcs->defbutton, EVENT_ACTION);
}
return;
case kEscapeCharCode:
if (mcs->canbutton != NULL) {
assert(mcs->canbutton->generic.type == MACCTRL_BUTTON);
HiliteControl(mcs->canbutton->button.tbctrl, kControlButtonPart);
Delay(6, &dummy);
HiliteControl(mcs->defbutton->button.tbctrl, kControlNoPart);
ctrlevent(mcs, mcs->canbutton, EVENT_ACTION);
}
return;
}
if (mac_gestalts.apprvers >= 0x100) {
if (GetKeyboardFocus(window, &control) == noErr && control != NULL) {
HandleControlKey(control, (event->message & keyCodeMask) >> 8,
event->message & charCodeMask, event->modifiers);
mc = (union macctrl *)GetControlReference(control);
switch (mc->generic.type) {
case MACCTRL_LISTBOX:
ctrlevent(mcs, mc, EVENT_SELCHANGE);
break;
default:
ctrlevent(mcs, mc, EVENT_VALCHANGE);
break;
}
}
}
#if !TARGET_API_MAC_CARBON
else {
TEHandle te;
if (mcs->focus != NULL) {
mc = mcs->focus;
switch (mc->generic.type) {
case MACCTRL_EDITBOX:
te = (TEHandle)(*mc->editbox.tbctrl)->contrlData;
TEKey(event->message & charCodeMask, te);
ctrlevent(mcs, mc, EVENT_VALCHANGE);
break;
}
}
}
#endif
}
void macctrl_update(WindowPtr window)
{
#if TARGET_API_MAC_CARBON
RgnHandle visrgn;
#endif
Rect rect;
GrafPtr saveport;
BeginUpdate(window);
GetPort(&saveport);
SetPort((GrafPtr)GetWindowPort(window));
if (mac_gestalts.apprvers >= 0x101) {
#if TARGET_API_MAC_CARBON
GetPortBounds(GetWindowPort(window), &rect);
#else
rect = window->portRect;
#endif
InsetRect(&rect, -1, -1);
DrawThemeModelessDialogFrame(&rect, mac_frontwindow() == window ?
kThemeStateActive : kThemeStateInactive);
}
#if TARGET_API_MAC_CARBON
visrgn = NewRgn();
GetPortVisibleRegion(GetWindowPort(window), visrgn);
UpdateControls(window, visrgn);
DisposeRgn(visrgn);
#else
UpdateControls(window, window->visRgn);
#endif
SetPort(saveport);
EndUpdate(window);
}
#if TARGET_API_MAC_CARBON
#define EnableItem EnableMenuItem
#define DisableItem DisableMenuItem
#endif
void macctrl_adjustmenus(WindowPtr window)
{
MenuHandle menu;
menu = GetMenuHandle(mFile);
DisableItem(menu, iSave); /* XXX enable if modified */
EnableItem(menu, iSaveAs);
EnableItem(menu, iDuplicate);
menu = GetMenuHandle(mEdit);
DisableItem(menu, 0);
}
void macctrl_close(WindowPtr window)
{
struct macctrls *mcs = mac_winctrls(window);
union macctrl *mc;
/*
* Mostly, we don't bother disposing of the Toolbox controls,
* since that will happen automatically when the window is
* disposed of. Popup menus are an exception, because we have to
* dispose of the menu ourselves, and doing that while the control
* still holds a reference to it seems rude.
*/
while ((mc = index234(mcs->byctrl, 0)) != NULL) {
if (mc->generic.privdata != NULL && mc->generic.freeprivdata)
sfree(mc->generic.privdata);
switch (mc->generic.type) {
case MACCTRL_POPUP:
DisposeControl(mc->popup.tbctrl);
DeleteMenu(mc->popup.menuid);
DisposeMenu(mc->popup.menu);
break;
}
del234(mcs->byctrl, mc);
sfree(mc);
}
freetree234(mcs->byctrl);
mcs->byctrl = NULL;
sfree(mcs->panels);
mcs->panels = NULL;
}
void dlg_update_start(union control *ctrl, void *dlg)
{
/* No-op for now */
}
void dlg_update_done(union control *ctrl, void *dlg)
{
/* No-op for now */
}
void dlg_set_focus(union control *ctrl, void *dlg)
{
if (mac_gestalts.apprvers >= 0x100) {
/* Use SetKeyboardFocus() */
} else {
/* Do our own mucking around */
}
}
union control *dlg_last_focused(union control *ctrl, void *dlg)
{
return NULL;
}
void dlg_beep(void *dlg)
{
SysBeep(30);
}
void dlg_error_msg(void *dlg, char *msg)
{
Str255 pmsg;
c2pstrcpy(pmsg, msg);
ParamText(pmsg, NULL, NULL, NULL);
StopAlert(128, NULL);
}
void dlg_end(void *dlg, int value)
{
struct macctrls *mcs = dlg;
if (mcs->end != NULL)
(*mcs->end)(mcs->window, value);
};
void dlg_refresh(union control *ctrl, void *dlg)
{
struct macctrls *mcs = dlg;
union macctrl *mc;
if (ctrl == NULL)
return; /* FIXME */
mc = findbyctrl(mcs, ctrl);
assert(mc != NULL);
ctrlevent(mcs, mc, EVENT_REFRESH);
};
void *dlg_get_privdata(union control *ctrl, void *dlg)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
assert(mc != NULL);
return mc->generic.privdata;
}
void dlg_set_privdata(union control *ctrl, void *dlg, void *ptr)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
assert(mc != NULL);
mc->generic.privdata = ptr;
mc->generic.freeprivdata = FALSE;
}
void *dlg_alloc_privdata(union control *ctrl, void *dlg, size_t size)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
assert(mc != NULL);
mc->generic.privdata = smalloc(size);
mc->generic.freeprivdata = TRUE;
return mc->generic.privdata;
}
/*
* Radio Button control
*/
void dlg_radiobutton_set(union control *ctrl, void *dlg, int whichbutton)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
int i;
if (mc == NULL) return;
for (i = 0; i < ctrl->radio.nbuttons; i++) {
if (i == whichbutton)
SetControlValue(mc->radio.tbctrls[i],
kControlRadioButtonCheckedValue);
else
SetControlValue(mc->radio.tbctrls[i],
kControlRadioButtonUncheckedValue);
}
};
int dlg_radiobutton_get(union control *ctrl, void *dlg)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
int i;
assert(mc != NULL);
for (i = 0; i < ctrl->radio.nbuttons; i++) {
if (GetControlValue(mc->radio.tbctrls[i]) ==
kControlRadioButtonCheckedValue)
return i;
}
return -1;
};
/*
* Check Box control
*/
void dlg_checkbox_set(union control *ctrl, void *dlg, int checked)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
if (mc == NULL) return;
SetControlValue(mc->checkbox.tbctrl,
checked ? kControlCheckBoxCheckedValue :
kControlCheckBoxUncheckedValue);
}
int dlg_checkbox_get(union control *ctrl, void *dlg)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
assert(mc != NULL);
return GetControlValue(mc->checkbox.tbctrl);
}
/*
* Edit Box control
*/
void dlg_editbox_set(union control *ctrl, void *dlg, char const *text)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
GrafPtr saveport;
if (mc == NULL) return;
assert(mc->generic.type == MACCTRL_EDITBOX);
GetPort(&saveport);
SetPort((GrafPtr)(GetWindowPort(mcs->window)));
if (mac_gestalts.apprvers >= 0x100)
SetControlData(mc->editbox.tbctrl, kControlEntireControl,
ctrl->editbox.password ?
kControlEditTextPasswordTag :
kControlEditTextTextTag,
strlen(text), text);
#if !TARGET_API_MAC_CARBON
else
TESetText(text, strlen(text),
(TEHandle)(*mc->editbox.tbctrl)->contrlData);
#endif
DrawOneControl(mc->editbox.tbctrl);
SetPort(saveport);
}
void dlg_editbox_get(union control *ctrl, void *dlg, char *buffer, int length)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
Size olen;
assert(mc != NULL);
assert(mc->generic.type == MACCTRL_EDITBOX);
if (mac_gestalts.apprvers >= 0x100) {
if (GetControlData(mc->editbox.tbctrl, kControlEntireControl,
ctrl->editbox.password ?
kControlEditTextPasswordTag :
kControlEditTextTextTag,
length - 1, buffer, &olen) != noErr)
olen = 0;
if (olen > length - 1)
olen = length - 1;
}
#if !TARGET_API_MAC_CARBON
else {
TEHandle te = (TEHandle)(*mc->editbox.tbctrl)->contrlData;
olen = (*te)->teLength;
if (olen > length - 1)
olen = length - 1;
memcpy(buffer, *(*te)->hText, olen);
}
#endif
buffer[olen] = '\0';
}
/*
* List Box control
*/
static void dlg_macpopup_clear(union control *ctrl, void *dlg)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
MenuRef menu = mc->popup.menu;
unsigned int i, n;
if (mc == NULL) return;
n = CountMenuItems(menu);
for (i = 0; i < n; i++)
DeleteMenuItem(menu, n - i);
mc->popup.nids = 0;
sfree(mc->popup.ids);
mc->popup.ids = NULL;
SetControlMaximum(mc->popup.tbctrl, CountMenuItems(menu));
}
static void dlg_maclist_clear(union control *ctrl, void *dlg)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
if (mc == NULL) return;
LDelRow(0, 0, mc->listbox.list);
mc->listbox.nids = 0;
sfree(mc->listbox.ids);
mc->listbox.ids = NULL;
DrawOneControl(mc->listbox.tbctrl);
}
void dlg_listbox_clear(union control *ctrl, void *dlg)
{
switch (ctrl->generic.type) {
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0)
dlg_macpopup_clear(ctrl, dlg);
else
dlg_maclist_clear(ctrl, dlg);
break;
}
}
static void dlg_macpopup_del(union control *ctrl, void *dlg, int index)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
MenuRef menu = mc->popup.menu;
if (mc == NULL) return;
DeleteMenuItem(menu, index + 1);
if (mc->popup.ids != NULL)
memcpy(mc->popup.ids + index, mc->popup.ids + index + 1,
(mc->popup.nids - index - 1) * sizeof(*mc->popup.ids));
SetControlMaximum(mc->popup.tbctrl, CountMenuItems(menu));
}
static void dlg_maclist_del(union control *ctrl, void *dlg, int index)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
if (mc == NULL) return;
LDelRow(1, index, mc->listbox.list);
if (mc->listbox.ids != NULL)
memcpy(mc->listbox.ids + index, mc->listbox.ids + index + 1,
(mc->listbox.nids - index - 1) * sizeof(*mc->listbox.ids));
DrawOneControl(mc->listbox.tbctrl);
}
void dlg_listbox_del(union control *ctrl, void *dlg, int index)
{
switch (ctrl->generic.type) {
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0)
dlg_macpopup_del(ctrl, dlg, index);
else
dlg_maclist_del(ctrl, dlg, index);
break;
}
}
static void dlg_macpopup_add(union control *ctrl, void *dlg, char const *text)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
MenuRef menu = mc->popup.menu;
Str255 itemstring;
if (mc == NULL) return;
assert(text[0] != '\0');
c2pstrcpy(itemstring, text);
AppendMenu(menu, "\pdummy");
SetMenuItemText(menu, CountMenuItems(menu), itemstring);
SetControlMaximum(mc->popup.tbctrl, CountMenuItems(menu));
}
static void dlg_maclist_add(union control *ctrl, void *dlg, char const *text)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
ListBounds bounds;
Cell cell = { 0, 0 };
if (mc == NULL) return;
#if TARGET_API_MAC_CARBON
GetListDataBounds(mc->listbox.list, &bounds);
#else
bounds = (*mc->listbox.list)->dataBounds;
#endif
cell.v = bounds.bottom;
LAddRow(1, cell.v, mc->listbox.list);
LSetCell(text, strlen(text), cell, mc->listbox.list);
DrawOneControl(mc->listbox.tbctrl);
}
void dlg_listbox_add(union control *ctrl, void *dlg, char const *text)
{
switch (ctrl->generic.type) {
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0)
dlg_macpopup_add(ctrl, dlg, text);
else
dlg_maclist_add(ctrl, dlg, text);
break;
}
}
static void dlg_macpopup_addwithid(union control *ctrl, void *dlg,
char const *text, int id)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
MenuRef menu = mc->popup.menu;
unsigned int index;
if (mc == NULL) return;
dlg_macpopup_add(ctrl, dlg, text);
index = CountMenuItems(menu) - 1;
if (mc->popup.nids <= index) {
mc->popup.nids = index + 1;
mc->popup.ids = sresize(mc->popup.ids, mc->popup.nids, int);
}
mc->popup.ids[index] = id;
}
static void dlg_maclist_addwithid(union control *ctrl, void *dlg,
char const *text, int id)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
ListBounds bounds;
int index;
if (mc == NULL) return;
dlg_maclist_add(ctrl, dlg, text);
#if TARGET_API_MAC_CARBON
GetListDataBounds(mc->listbox.list, &bounds);
#else
bounds = (*mc->listbox.list)->dataBounds;
#endif
index = bounds.bottom;
if (mc->listbox.nids <= index) {
mc->listbox.nids = index + 1;
mc->listbox.ids = sresize(mc->listbox.ids, mc->listbox.nids, int);
}
mc->listbox.ids[index] = id;
}
void dlg_listbox_addwithid(union control *ctrl, void *dlg,
char const *text, int id)
{
switch (ctrl->generic.type) {
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0)
dlg_macpopup_addwithid(ctrl, dlg, text, id);
else
dlg_maclist_addwithid(ctrl, dlg, text, id);
break;
}
}
int dlg_listbox_getid(union control *ctrl, void *dlg, int index)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
assert(mc != NULL);
switch (ctrl->generic.type) {
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0) {
assert(mc->popup.ids != NULL && mc->popup.nids > index);
return mc->popup.ids[index];
} else {
assert(mc->listbox.ids != NULL && mc->listbox.nids > index);
return mc->listbox.ids[index];
}
}
return -1;
}
int dlg_listbox_index(union control *ctrl, void *dlg)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
Cell cell = { 0, 0 };
assert(mc != NULL);
switch (ctrl->generic.type) {
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0)
return GetControlValue(mc->popup.tbctrl) - 1;
else {
if (LGetSelect(TRUE, &cell, mc->listbox.list))
return cell.v;
else
return -1;
}
}
return -1;
}
int dlg_listbox_issel(union control *ctrl, void *dlg, int index)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
Cell cell = { 0, 0 };
assert(mc != NULL);
switch (ctrl->generic.type) {
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0)
return GetControlValue(mc->popup.tbctrl) - 1 == index;
else {
cell.v = index;
return LGetSelect(FALSE, &cell, mc->listbox.list);
}
}
return FALSE;
}
void dlg_listbox_select(union control *ctrl, void *dlg, int index)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
if (mc == NULL) return;
switch (ctrl->generic.type) {
case CTRL_LISTBOX:
if (ctrl->listbox.height == 0)
SetControlValue(mc->popup.tbctrl, index + 1);
break;
}
}
/*
* Text control
*/
void dlg_text_set(union control *ctrl, void *dlg, char const *text)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
if (mc == NULL) return;
if (mac_gestalts.apprvers >= 0x100)
SetControlData(mc->text.tbctrl, kControlEntireControl,
kControlStaticTextTextTag, strlen(text), text);
#if !TARGET_API_MAC_CARBON
else
TESetText(text, strlen(text),
(TEHandle)(*mc->text.tbctrl)->contrlData);
#endif
}
/*
* File Selector control
*/
void dlg_filesel_set(union control *ctrl, void *dlg, Filename fn)
{
}
void dlg_filesel_get(union control *ctrl, void *dlg, Filename *fn)
{
}
/*
* Font Selector control
*/
void dlg_fontsel_set(union control *ctrl, void *dlg, FontSpec fn)
{
}
void dlg_fontsel_get(union control *ctrl, void *dlg, FontSpec *fn)
{
}
/*
* Printer enumeration
*/
printer_enum *printer_start_enum(int *nprinters)
{
*nprinters = 0;
return NULL;
}
char *printer_get_name(printer_enum *pe, int thing)
{
return "<none>";
}
void printer_finish_enum(printer_enum *pe)
{
}
/*
* Colour selection stuff
*/
void dlg_coloursel_start(union control *ctrl, void *dlg,
int r, int g, int b)
{
struct macctrls *mcs = dlg;
union macctrl *mc = findbyctrl(mcs, ctrl);
Point where = {-1, -1}; /* Screen with greatest colour depth */
RGBColor incolour;
if (HAVE_COLOR_QD()) {
incolour.red = r * 0x0101;
incolour.green = g * 0x0101;
incolour.blue = b * 0x0101;
mcs->gotcolour = GetColor(where, "\pModify Colour:", &incolour,
&mcs->thecolour);
ctrlevent(mcs, mc, EVENT_CALLBACK);
} else
dlg_beep(dlg);
}
int dlg_coloursel_results(union control *ctrl, void *dlg,
int *r, int *g, int *b)
{
struct macctrls *mcs = dlg;
if (mcs->gotcolour) {
*r = mcs->thecolour.red >> 8;
*g = mcs->thecolour.green >> 8;
*b = mcs->thecolour.blue >> 8;
return 1;
} else
return 0;
}
/*
* Local Variables:
* c-file-style: "simon"
* End:
*/