2 * winctrls.c: routines to self-manage the controls in a dialog
7 * Possible TODO in new cross-platform config box stuff:
9 * - When lining up two controls alongside each other, I wonder if
10 * we could conveniently arrange to centre them vertically?
11 * Particularly ugly in the current setup is the `Add new
12 * forwarded port:' static next to the rather taller `Remove'
30 #define STATICHEIGHT 8
31 #define TITLEHEIGHT 12
32 #define CHECKBOXHEIGHT 8
36 #define LISTINCREMENT 8
37 #define COMBOHEIGHT 12
38 #define PUSHBTNHEIGHT 14
39 #define PROGBARHEIGHT 14
41 void ctlposinit(struct ctlpos
*cp
, HWND hwnd
,
42 int leftborder
, int rightborder
, int topborder
)
46 cp
->font
= SendMessage(hwnd
, WM_GETFONT
, 0, 0);
48 GetClientRect(hwnd
, &r
);
52 MapDialogRect(hwnd
, &r2
);
53 cp
->dlu4inpix
= r2
.right
;
54 cp
->width
= (r
.right
* 4) / (r2
.right
) - 2 * GAPBETWEEN
;
55 cp
->xoff
= leftborder
;
56 cp
->width
-= leftborder
+ rightborder
;
59 HWND
doctl(struct ctlpos
*cp
, RECT r
,
60 char *wclass
, int wstyle
, int exstyle
, char *wtext
, int wid
)
64 * Note nonstandard use of RECT. This is deliberate: by
65 * transforming the width and height directly we arrange to
66 * have all supposedly same-sized controls really same-sized.
70 MapDialogRect(cp
->hwnd
, &r
);
73 * We can pass in cp->hwnd == NULL, to indicate a dry run
74 * without creating any actual controls.
77 ctl
= CreateWindowEx(exstyle
, wclass
, wtext
, wstyle
,
78 r
.left
, r
.top
, r
.right
, r
.bottom
,
79 cp
->hwnd
, (HMENU
) wid
, hinst
, NULL
);
80 SendMessage(ctl
, WM_SETFONT
, cp
->font
, MAKELPARAM(TRUE
, 0));
82 if (!strcmp(wclass
, "LISTBOX")) {
84 * Bizarre Windows bug: the list box calculates its
85 * number of lines based on the font it has at creation
86 * time, but sending it WM_SETFONT doesn't cause it to
87 * recalculate. So now, _after_ we've sent it
88 * WM_SETFONT, we explicitly resize it (to the same
89 * size it was already!) to force it to reconsider.
91 SetWindowPos(ctl
, NULL
, 0, 0, r
.right
, r
.bottom
,
92 SWP_NOACTIVATE
| SWP_NOCOPYBITS
|
93 SWP_NOMOVE
| SWP_NOZORDER
);
101 * A title bar across the top of a sub-dialog.
103 void bartitle(struct ctlpos
*cp
, char *name
, int id
)
110 r
.bottom
= STATICHEIGHT
;
111 cp
->ypos
+= r
.bottom
+ GAPBETWEEN
;
112 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, name
, id
);
116 * Begin a grouping box, with or without a group title.
118 void beginbox(struct ctlpos
*cp
, char *name
, int idbox
)
120 cp
->boxystart
= cp
->ypos
;
122 cp
->boxystart
-= STATICHEIGHT
/ 2;
124 cp
->ypos
+= STATICHEIGHT
;
126 cp
->width
-= 2 * GAPXBOX
;
133 * End a grouping box.
135 void endbox(struct ctlpos
*cp
)
139 cp
->width
+= 2 * GAPXBOX
;
140 cp
->ypos
+= GAPYBOX
- GAPBETWEEN
;
143 r
.top
= cp
->boxystart
;
144 r
.bottom
= cp
->ypos
- cp
->boxystart
;
145 doctl(cp
, r
, "BUTTON", BS_GROUPBOX
| WS_CHILD
| WS_VISIBLE
, 0,
146 cp
->boxtext ? cp
->boxtext
: "", cp
->boxid
);
151 * A static line, followed by a full-width edit box.
153 void editboxfw(struct ctlpos
*cp
, int password
, char *text
,
154 int staticid
, int editid
)
163 r
.bottom
= STATICHEIGHT
;
164 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, text
, staticid
);
165 cp
->ypos
+= STATICHEIGHT
+ GAPWITHIN
;
168 r
.bottom
= EDITHEIGHT
;
170 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| ES_AUTOHSCROLL
|
171 (password ? ES_PASSWORD
: 0),
172 WS_EX_CLIENTEDGE
, "", editid
);
173 cp
->ypos
+= EDITHEIGHT
+ GAPBETWEEN
;
177 * A static line, followed by a full-width combo box.
179 void combobox(struct ctlpos
*cp
, char *text
, int staticid
, int listid
)
188 r
.bottom
= STATICHEIGHT
;
189 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, text
, staticid
);
190 cp
->ypos
+= STATICHEIGHT
+ GAPWITHIN
;
193 r
.bottom
= COMBOHEIGHT
* 10;
194 doctl(cp
, r
, "COMBOBOX",
195 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| WS_VSCROLL
|
196 CBS_DROPDOWN
| CBS_HASSTRINGS
, WS_EX_CLIENTEDGE
, "", listid
);
197 cp
->ypos
+= COMBOHEIGHT
+ GAPBETWEEN
;
200 struct radio
{ char *text
; int id
; };
202 static void radioline_common(struct ctlpos
*cp
, char *text
, int id
,
203 int nacross
, struct radio
*buttons
, int nbuttons
)
214 r
.bottom
= STATICHEIGHT
;
215 cp
->ypos
+= r
.bottom
+ GAPWITHIN
;
216 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, text
, id
);
221 for (j
= 0; j
< nbuttons
; j
++) {
222 char *btext
= buttons
[j
].text
;
223 int bid
= buttons
[j
].id
;
226 cp
->ypos
+= r
.bottom
+ (nacross
> 1 ? GAPBETWEEN
: GAPWITHIN
);
229 r
.left
= GAPBETWEEN
+ i
* (cp
->width
+ GAPBETWEEN
) / nacross
;
232 (i
+ 1) * (cp
->width
+ GAPBETWEEN
) / nacross
- r
.left
;
234 r
.right
= cp
->width
- r
.left
;
236 r
.bottom
= RADIOHEIGHT
;
237 doctl(cp
, r
, "BUTTON",
238 BS_NOTIFY
| BS_AUTORADIOBUTTON
| WS_CHILD
|
239 WS_VISIBLE
| WS_TABSTOP
| group
, 0, btext
, bid
);
243 cp
->ypos
+= r
.bottom
+ GAPBETWEEN
;
247 * A set of radio buttons on the same line, with a static above
248 * them. `nacross' dictates how many parts the line is divided into
249 * (you might want this not to equal the number of buttons if you
250 * needed to line up some 2s and some 3s to look good in the same
253 * There's a bit of a hack in here to ensure that if nacross
254 * exceeds the actual number of buttons, the rightmost button
255 * really does get all the space right to the edge of the line, so
256 * you can do things like
258 * (*) Button1 (*) Button2 (*) ButtonWithReallyLongTitle
260 void radioline(struct ctlpos
*cp
, char *text
, int id
, int nacross
, ...)
263 struct radio
*buttons
;
266 va_start(ap
, nacross
);
269 char *btext
= va_arg(ap
, char *);
273 bid
= va_arg(ap
, int);
277 buttons
= snewn(nbuttons
, struct radio
);
278 va_start(ap
, nacross
);
279 for (i
= 0; i
< nbuttons
; i
++) {
280 buttons
[i
].text
= va_arg(ap
, char *);
281 buttons
[i
].id
= va_arg(ap
, int);
284 radioline_common(cp
, text
, id
, nacross
, buttons
, nbuttons
);
289 * A set of radio buttons on the same line, without a static above
290 * them. Otherwise just like radioline.
292 void bareradioline(struct ctlpos
*cp
, int nacross
, ...)
295 struct radio
*buttons
;
298 va_start(ap
, nacross
);
301 char *btext
= va_arg(ap
, char *);
305 bid
= va_arg(ap
, int);
308 buttons
= snewn(nbuttons
, struct radio
);
309 va_start(ap
, nacross
);
310 for (i
= 0; i
< nbuttons
; i
++) {
311 buttons
[i
].text
= va_arg(ap
, char *);
312 buttons
[i
].id
= va_arg(ap
, int);
315 radioline_common(cp
, NULL
, 0, nacross
, buttons
, nbuttons
);
320 * A set of radio buttons on multiple lines, with a static above
323 void radiobig(struct ctlpos
*cp
, char *text
, int id
, ...)
326 struct radio
*buttons
;
332 char *btext
= va_arg(ap
, char *);
336 bid
= va_arg(ap
, int);
339 buttons
= snewn(nbuttons
, struct radio
);
341 for (i
= 0; i
< nbuttons
; i
++) {
342 buttons
[i
].text
= va_arg(ap
, char *);
343 buttons
[i
].id
= va_arg(ap
, int);
346 radioline_common(cp
, text
, id
, 1, buttons
, nbuttons
);
351 * A single standalone checkbox.
353 void checkbox(struct ctlpos
*cp
, char *text
, int id
)
360 r
.bottom
= CHECKBOXHEIGHT
;
361 cp
->ypos
+= r
.bottom
+ GAPBETWEEN
;
362 doctl(cp
, r
, "BUTTON",
363 BS_NOTIFY
| BS_AUTOCHECKBOX
| WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
, 0,
368 * Wrap a piece of text for a static text control. Returns the
369 * wrapped text (a malloc'ed string containing \ns), and also
370 * returns the number of lines required.
372 char *staticwrap(struct ctlpos
*cp
, HWND hwnd
, char *text
, int *lines
)
374 HDC hdc
= GetDC(hwnd
);
375 int lpx
= GetDeviceCaps(hdc
, LOGPIXELSX
);
376 int width
, nlines
, j
;
381 HFONT oldfont
, newfont
;
383 ret
= snewn(1+strlen(text
), char);
386 pwidths
= snewn(1+strlen(text
), INT
);
389 * Work out the width the text will need to fit in, by doing
390 * the same adjustment that the `statictext' function itself
393 SetMapMode(hdc
, MM_TEXT
); /* ensure logical units == pixels */
394 r
.left
= r
.top
= r
.bottom
= 0;
396 MapDialogRect(hwnd
, &r
);
402 * We must select the correct font into the HDC before calling
403 * GetTextExtent*, or silly things will happen.
405 newfont
= (HFONT
)SendMessage(hwnd
, WM_GETFONT
, 0, 0);
406 oldfont
= SelectObject(hdc
, newfont
);
409 if (!GetTextExtentExPoint(hdc
, p
, strlen(p
), width
,
410 &nfit
, pwidths
, &size
) ||
411 (size_t)nfit
>= strlen(p
)) {
413 * Either GetTextExtentExPoint returned failure, or the
414 * whole of the rest of the text fits on this line.
415 * Either way, we stop wrapping, copy the remainder of
416 * the input string unchanged to the output, and leave.
423 * Now we search backwards along the string from `nfit',
424 * looking for a space at which to break the line. If we
425 * don't find one at all, that's fine - we'll just break
426 * the line at `nfit'.
428 for (j
= nfit
; j
> 0; j
--) {
429 if (isspace((unsigned char)p
[j
])) {
440 while (*p
&& isspace((unsigned char)*p
))
446 SelectObject(hdc
, oldfont
);
447 ReleaseDC(cp
->hwnd
, hdc
);
449 if (lines
) *lines
= nlines
;
457 * A single standalone static text control.
459 void statictext(struct ctlpos
*cp
, char *text
, int lines
, int id
)
466 r
.bottom
= STATICHEIGHT
* lines
;
467 cp
->ypos
+= r
.bottom
+ GAPBETWEEN
;
468 doctl(cp
, r
, "STATIC",
469 WS_CHILD
| WS_VISIBLE
| SS_LEFTNOWORDWRAP
,
474 * An owner-drawn static text control for a panel title.
476 void paneltitle(struct ctlpos
*cp
, int id
)
483 r
.bottom
= TITLEHEIGHT
;
484 cp
->ypos
+= r
.bottom
+ GAPBETWEEN
;
485 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
| SS_OWNERDRAW
,
490 * A button on the right hand side, with a static to its left.
492 void staticbtn(struct ctlpos
*cp
, char *stext
, int sid
,
493 char *btext
, int bid
)
495 const int height
= (PUSHBTNHEIGHT
> STATICHEIGHT ?
496 PUSHBTNHEIGHT
: STATICHEIGHT
);
498 int lwid
, rwid
, rpos
;
500 rpos
= GAPBETWEEN
+ 3 * (cp
->width
+ GAPBETWEEN
) / 4;
501 lwid
= rpos
- 2 * GAPBETWEEN
;
502 rwid
= cp
->width
+ GAPBETWEEN
- rpos
;
505 r
.top
= cp
->ypos
+ (height
- STATICHEIGHT
) / 2;
507 r
.bottom
= STATICHEIGHT
;
508 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
511 r
.top
= cp
->ypos
+ (height
- PUSHBTNHEIGHT
) / 2;
513 r
.bottom
= PUSHBTNHEIGHT
;
514 doctl(cp
, r
, "BUTTON",
515 BS_NOTIFY
| WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| BS_PUSHBUTTON
,
518 cp
->ypos
+= height
+ GAPBETWEEN
;
522 * A simple push button.
524 void button(struct ctlpos
*cp
, char *btext
, int bid
, int defbtn
)
531 r
.bottom
= PUSHBTNHEIGHT
;
533 /* Q67655: the _dialog box_ must know which button is default
534 * as well as the button itself knowing */
535 if (defbtn
&& cp
->hwnd
)
536 SendMessage(cp
->hwnd
, DM_SETDEFID
, bid
, 0);
538 doctl(cp
, r
, "BUTTON",
539 BS_NOTIFY
| WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
|
540 (defbtn ? BS_DEFPUSHBUTTON
: 0) | BS_PUSHBUTTON
,
543 cp
->ypos
+= PUSHBTNHEIGHT
+ GAPBETWEEN
;
547 * Like staticbtn, but two buttons.
549 void static2btn(struct ctlpos
*cp
, char *stext
, int sid
,
550 char *btext1
, int bid1
, char *btext2
, int bid2
)
552 const int height
= (PUSHBTNHEIGHT
> STATICHEIGHT ?
553 PUSHBTNHEIGHT
: STATICHEIGHT
);
555 int lwid
, rwid1
, rwid2
, rpos1
, rpos2
;
557 rpos1
= GAPBETWEEN
+ (cp
->width
+ GAPBETWEEN
) / 2;
558 rpos2
= GAPBETWEEN
+ 3 * (cp
->width
+ GAPBETWEEN
) / 4;
559 lwid
= rpos1
- 2 * GAPBETWEEN
;
560 rwid1
= rpos2
- rpos1
- GAPBETWEEN
;
561 rwid2
= cp
->width
+ GAPBETWEEN
- rpos2
;
564 r
.top
= cp
->ypos
+ (height
- STATICHEIGHT
) / 2;
566 r
.bottom
= STATICHEIGHT
;
567 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
570 r
.top
= cp
->ypos
+ (height
- PUSHBTNHEIGHT
) / 2;
572 r
.bottom
= PUSHBTNHEIGHT
;
573 doctl(cp
, r
, "BUTTON",
574 BS_NOTIFY
| WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| BS_PUSHBUTTON
,
578 r
.top
= cp
->ypos
+ (height
- PUSHBTNHEIGHT
) / 2;
580 r
.bottom
= PUSHBTNHEIGHT
;
581 doctl(cp
, r
, "BUTTON",
582 BS_NOTIFY
| WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| BS_PUSHBUTTON
,
585 cp
->ypos
+= height
+ GAPBETWEEN
;
589 * An edit control on the right hand side, with a static to its left.
591 static void staticedit_internal(struct ctlpos
*cp
, char *stext
,
592 int sid
, int eid
, int percentedit
,
595 const int height
= (EDITHEIGHT
> STATICHEIGHT ?
596 EDITHEIGHT
: STATICHEIGHT
);
598 int lwid
, rwid
, rpos
;
601 GAPBETWEEN
+ (100 - percentedit
) * (cp
->width
+ GAPBETWEEN
) / 100;
602 lwid
= rpos
- 2 * GAPBETWEEN
;
603 rwid
= cp
->width
+ GAPBETWEEN
- rpos
;
606 r
.top
= cp
->ypos
+ (height
- STATICHEIGHT
) / 2;
608 r
.bottom
= STATICHEIGHT
;
609 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
612 r
.top
= cp
->ypos
+ (height
- EDITHEIGHT
) / 2;
614 r
.bottom
= EDITHEIGHT
;
616 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| ES_AUTOHSCROLL
| style
,
617 WS_EX_CLIENTEDGE
, "", eid
);
619 cp
->ypos
+= height
+ GAPBETWEEN
;
622 void staticedit(struct ctlpos
*cp
, char *stext
,
623 int sid
, int eid
, int percentedit
)
625 staticedit_internal(cp
, stext
, sid
, eid
, percentedit
, 0);
628 void staticpassedit(struct ctlpos
*cp
, char *stext
,
629 int sid
, int eid
, int percentedit
)
631 staticedit_internal(cp
, stext
, sid
, eid
, percentedit
, ES_PASSWORD
);
635 * A drop-down list box on the right hand side, with a static to
638 void staticddl(struct ctlpos
*cp
, char *stext
,
639 int sid
, int lid
, int percentlist
)
641 const int height
= (COMBOHEIGHT
> STATICHEIGHT ?
642 COMBOHEIGHT
: STATICHEIGHT
);
644 int lwid
, rwid
, rpos
;
647 GAPBETWEEN
+ (100 - percentlist
) * (cp
->width
+ GAPBETWEEN
) / 100;
648 lwid
= rpos
- 2 * GAPBETWEEN
;
649 rwid
= cp
->width
+ GAPBETWEEN
- rpos
;
652 r
.top
= cp
->ypos
+ (height
- STATICHEIGHT
) / 2;
654 r
.bottom
= STATICHEIGHT
;
655 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
658 r
.top
= cp
->ypos
+ (height
- EDITHEIGHT
) / 2;
660 r
.bottom
= COMBOHEIGHT
*4;
661 doctl(cp
, r
, "COMBOBOX",
662 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| WS_VSCROLL
|
663 CBS_DROPDOWNLIST
| CBS_HASSTRINGS
, WS_EX_CLIENTEDGE
, "", lid
);
665 cp
->ypos
+= height
+ GAPBETWEEN
;
669 * A combo box on the right hand side, with a static to its left.
671 void staticcombo(struct ctlpos
*cp
, char *stext
,
672 int sid
, int lid
, int percentlist
)
674 const int height
= (COMBOHEIGHT
> STATICHEIGHT ?
675 COMBOHEIGHT
: STATICHEIGHT
);
677 int lwid
, rwid
, rpos
;
680 GAPBETWEEN
+ (100 - percentlist
) * (cp
->width
+ GAPBETWEEN
) / 100;
681 lwid
= rpos
- 2 * GAPBETWEEN
;
682 rwid
= cp
->width
+ GAPBETWEEN
- rpos
;
685 r
.top
= cp
->ypos
+ (height
- STATICHEIGHT
) / 2;
687 r
.bottom
= STATICHEIGHT
;
688 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
691 r
.top
= cp
->ypos
+ (height
- EDITHEIGHT
) / 2;
693 r
.bottom
= COMBOHEIGHT
*10;
694 doctl(cp
, r
, "COMBOBOX",
695 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| WS_VSCROLL
|
696 CBS_DROPDOWN
| CBS_HASSTRINGS
, WS_EX_CLIENTEDGE
, "", lid
);
698 cp
->ypos
+= height
+ GAPBETWEEN
;
702 * A static, with a full-width drop-down list box below it.
704 void staticddlbig(struct ctlpos
*cp
, char *stext
,
713 r
.bottom
= STATICHEIGHT
;
714 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
715 cp
->ypos
+= STATICHEIGHT
;
721 r
.bottom
= COMBOHEIGHT
*4;
722 doctl(cp
, r
, "COMBOBOX",
723 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| WS_VSCROLL
|
724 CBS_DROPDOWNLIST
| CBS_HASSTRINGS
, WS_EX_CLIENTEDGE
, "", lid
);
725 cp
->ypos
+= COMBOHEIGHT
+ GAPBETWEEN
;
729 * A big multiline edit control with a static labelling it.
731 void bigeditctrl(struct ctlpos
*cp
, char *stext
,
732 int sid
, int eid
, int lines
)
740 r
.bottom
= STATICHEIGHT
;
741 cp
->ypos
+= r
.bottom
+ GAPWITHIN
;
742 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
748 r
.bottom
= EDITHEIGHT
+ (lines
- 1) * STATICHEIGHT
;
749 cp
->ypos
+= r
.bottom
+ GAPBETWEEN
;
751 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| WS_VSCROLL
| ES_MULTILINE
,
752 WS_EX_CLIENTEDGE
, "", eid
);
756 * A list box with a static labelling it.
758 void listbox(struct ctlpos
*cp
, char *stext
,
759 int sid
, int lid
, int lines
, int multi
)
767 r
.bottom
= STATICHEIGHT
;
768 cp
->ypos
+= r
.bottom
+ GAPWITHIN
;
769 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
775 r
.bottom
= LISTHEIGHT
+ (lines
- 1) * LISTINCREMENT
;
776 cp
->ypos
+= r
.bottom
+ GAPBETWEEN
;
777 doctl(cp
, r
, "LISTBOX",
778 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| WS_VSCROLL
|
779 LBS_NOTIFY
| LBS_HASSTRINGS
| LBS_USETABSTOPS
|
780 (multi ? LBS_MULTIPLESEL
: 0),
781 WS_EX_CLIENTEDGE
, "", lid
);
785 * A tab-control substitute when a real tab control is unavailable.
787 void ersatztab(struct ctlpos
*cp
, char *stext
, int sid
, int lid
, int s2id
)
789 const int height
= (COMBOHEIGHT
> STATICHEIGHT ?
790 COMBOHEIGHT
: STATICHEIGHT
);
792 int bigwid
, lwid
, rwid
, rpos
;
793 static const int BIGGAP
= 15;
794 static const int MEDGAP
= 3;
796 bigwid
= cp
->width
+ 2 * GAPBETWEEN
- 2 * BIGGAP
;
798 rpos
= BIGGAP
+ (bigwid
+ BIGGAP
) / 2;
799 lwid
= rpos
- 2 * BIGGAP
;
800 rwid
= bigwid
+ BIGGAP
- rpos
;
803 r
.top
= cp
->ypos
+ (height
- STATICHEIGHT
) / 2;
805 r
.bottom
= STATICHEIGHT
;
806 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
809 r
.top
= cp
->ypos
+ (height
- COMBOHEIGHT
) / 2;
811 r
.bottom
= COMBOHEIGHT
* 10;
812 doctl(cp
, r
, "COMBOBOX",
813 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
|
814 CBS_DROPDOWNLIST
| CBS_HASSTRINGS
, WS_EX_CLIENTEDGE
, "", lid
);
816 cp
->ypos
+= height
+ MEDGAP
+ GAPBETWEEN
;
822 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
| SS_ETCHEDHORZ
,
827 * A static line, followed by an edit control on the left hand side
828 * and a button on the right.
830 void editbutton(struct ctlpos
*cp
, char *stext
, int sid
,
831 int eid
, char *btext
, int bid
)
833 const int height
= (EDITHEIGHT
> PUSHBTNHEIGHT ?
834 EDITHEIGHT
: PUSHBTNHEIGHT
);
836 int lwid
, rwid
, rpos
;
841 r
.bottom
= STATICHEIGHT
;
842 cp
->ypos
+= r
.bottom
+ GAPWITHIN
;
843 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
845 rpos
= GAPBETWEEN
+ 3 * (cp
->width
+ GAPBETWEEN
) / 4;
846 lwid
= rpos
- 2 * GAPBETWEEN
;
847 rwid
= cp
->width
+ GAPBETWEEN
- rpos
;
850 r
.top
= cp
->ypos
+ (height
- EDITHEIGHT
) / 2;
852 r
.bottom
= EDITHEIGHT
;
854 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| ES_AUTOHSCROLL
,
855 WS_EX_CLIENTEDGE
, "", eid
);
858 r
.top
= cp
->ypos
+ (height
- PUSHBTNHEIGHT
) / 2;
860 r
.bottom
= PUSHBTNHEIGHT
;
861 doctl(cp
, r
, "BUTTON",
862 BS_NOTIFY
| WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
| BS_PUSHBUTTON
,
865 cp
->ypos
+= height
+ GAPBETWEEN
;
869 * A special control for manipulating an ordered preference list
870 * (eg. for cipher selection).
871 * XXX: this is a rough hack and could be improved.
873 void prefslist(struct prefslist
*hdl
, struct ctlpos
*cp
, int lines
,
874 char *stext
, int sid
, int listid
, int upbid
, int dnbid
)
876 const static int percents
[] = { 5, 75, 20 };
878 int xpos
, percent
= 0, i
;
879 int listheight
= LISTHEIGHT
+ (lines
- 1) * LISTINCREMENT
;
880 const int BTNSHEIGHT
= 2*PUSHBTNHEIGHT
+ GAPBETWEEN
;
881 int totalheight
, buttonpos
;
883 /* Squirrel away IDs. */
884 hdl
->listid
= listid
;
888 /* The static label. */
893 r
.bottom
= STATICHEIGHT
;
894 cp
->ypos
+= r
.bottom
+ GAPWITHIN
;
895 doctl(cp
, r
, "STATIC", WS_CHILD
| WS_VISIBLE
, 0, stext
, sid
);
898 if (listheight
> BTNSHEIGHT
) {
899 totalheight
= listheight
;
900 buttonpos
= (listheight
- BTNSHEIGHT
) / 2;
902 totalheight
= BTNSHEIGHT
;
906 for (i
=0; i
<3; i
++) {
908 xpos
= (cp
->width
+ GAPBETWEEN
) * percent
/ 100;
909 left
= xpos
+ GAPBETWEEN
;
910 percent
+= percents
[i
];
911 xpos
= (cp
->width
+ GAPBETWEEN
) * percent
/ 100;
916 /* The drag list box. */
917 r
.left
= left
; r
.right
= wid
;
918 r
.top
= cp
->ypos
; r
.bottom
= listheight
;
921 ctl
= doctl(cp
, r
, "LISTBOX",
922 WS_CHILD
| WS_VISIBLE
| WS_TABSTOP
|
923 WS_VSCROLL
| LBS_HASSTRINGS
| LBS_USETABSTOPS
,
931 /* The "Up" and "Down" buttons. */
932 /* XXX worry about accelerators if we have more than one
933 * prefslist on a panel */
934 r
.left
= left
; r
.right
= wid
;
935 r
.top
= cp
->ypos
+ buttonpos
; r
.bottom
= PUSHBTNHEIGHT
;
936 doctl(cp
, r
, "BUTTON",
937 BS_NOTIFY
| WS_CHILD
| WS_VISIBLE
|
938 WS_TABSTOP
| BS_PUSHBUTTON
,
941 r
.left
= left
; r
.right
= wid
;
942 r
.top
= cp
->ypos
+ buttonpos
+ PUSHBTNHEIGHT
+ GAPBETWEEN
;
943 r
.bottom
= PUSHBTNHEIGHT
;
944 doctl(cp
, r
, "BUTTON",
945 BS_NOTIFY
| WS_CHILD
| WS_VISIBLE
|
946 WS_TABSTOP
| BS_PUSHBUTTON
,
954 cp
->ypos
+= totalheight
+ GAPBETWEEN
;
959 * Helper function for prefslist: move item in list box.
961 static void pl_moveitem(HWND hwnd
, int listid
, int src
, int dst
)
965 /* Get the item's data. */
966 tlen
= SendDlgItemMessage (hwnd
, listid
, LB_GETTEXTLEN
, src
, 0);
967 txt
= snewn(tlen
+1, char);
968 SendDlgItemMessage (hwnd
, listid
, LB_GETTEXT
, src
, (LPARAM
) txt
);
969 val
= SendDlgItemMessage (hwnd
, listid
, LB_GETITEMDATA
, src
, 0);
970 /* Deselect old location. */
971 SendDlgItemMessage (hwnd
, listid
, LB_SETSEL
, FALSE
, src
);
972 /* Delete it at the old location. */
973 SendDlgItemMessage (hwnd
, listid
, LB_DELETESTRING
, src
, 0);
974 /* Insert it at new location. */
975 SendDlgItemMessage (hwnd
, listid
, LB_INSERTSTRING
, dst
,
977 SendDlgItemMessage (hwnd
, listid
, LB_SETITEMDATA
, dst
,
980 SendDlgItemMessage (hwnd
, listid
, LB_SETCURSEL
, dst
, 0);
984 int pl_itemfrompt(HWND hwnd
, POINT cursor
, BOOL scroll
)
987 POINT uppoint
, downpoint
;
988 int updist
, downdist
, upitem
, downitem
, i
;
991 * Ghastly hackery to try to figure out not which
992 * _item_, but which _gap between items_, the user
993 * is pointing at. We do this by first working out
994 * which list item is under the cursor, and then
995 * working out how far the cursor would have to
996 * move up or down before the answer was different.
997 * Then we put the insertion point _above_ the
998 * current item if the upper edge is closer than
999 * the lower edge, or _below_ it if vice versa.
1001 ret
= LBItemFromPt(hwnd
, cursor
, scroll
);
1004 ret
= LBItemFromPt(hwnd
, cursor
, FALSE
);
1005 updist
= downdist
= 0;
1006 for (i
= 1; i
< 4096 && (!updist
|| !downdist
); i
++) {
1007 uppoint
= downpoint
= cursor
;
1010 upitem
= LBItemFromPt(hwnd
, uppoint
, FALSE
);
1011 downitem
= LBItemFromPt(hwnd
, downpoint
, FALSE
);
1012 if (!updist
&& upitem
!= ret
)
1014 if (!downdist
&& downitem
!= ret
)
1017 if (downdist
< updist
)
1023 * Handler for prefslist above.
1025 * Return value has bit 0 set if the dialog box procedure needs to
1026 * return TRUE from handling this message; it has bit 1 set if a
1027 * change may have been made in the contents of the list.
1029 int handle_prefslist(struct prefslist
*hdl
,
1030 int *array
, int maxmemb
,
1031 int is_dlmsg
, HWND hwnd
,
1032 WPARAM wParam
, LPARAM lParam
)
1039 if ((int)wParam
== hdl
->listid
) {
1040 DRAGLISTINFO
*dlm
= (DRAGLISTINFO
*)lParam
;
1041 int dest
= 0; /* initialise to placate gcc */
1042 switch (dlm
->uNotification
) {
1044 /* Add a dummy item to make pl_itemfrompt() work
1046 * FIXME: this causes scrollbar glitches if the count of
1047 * listbox contains >= its height. */
1049 SendDlgItemMessage(hwnd
, hdl
->listid
,
1050 LB_ADDSTRING
, 0, (LPARAM
) "");
1052 hdl
->srcitem
= LBItemFromPt(dlm
->hWnd
, dlm
->ptCursor
, TRUE
);
1054 /* XXX hack Q183115 */
1055 SetWindowLongPtr(hwnd
, DWLP_MSGRESULT
, TRUE
);
1058 DrawInsert(hwnd
, dlm
->hWnd
, -1); /* Clear arrow */
1059 SendDlgItemMessage(hwnd
, hdl
->listid
,
1060 LB_DELETESTRING
, hdl
->dummyitem
, 0);
1065 dest
= pl_itemfrompt(dlm
->hWnd
, dlm
->ptCursor
, TRUE
);
1066 if (dest
> hdl
->dummyitem
) dest
= hdl
->dummyitem
;
1067 DrawInsert (hwnd
, dlm
->hWnd
, dest
);
1069 SetWindowLongPtr(hwnd
, DWLP_MSGRESULT
, DL_MOVECURSOR
);
1071 SetWindowLongPtr(hwnd
, DWLP_MSGRESULT
, DL_STOPCURSOR
);
1074 if (hdl
->dragging
) {
1075 dest
= pl_itemfrompt(dlm
->hWnd
, dlm
->ptCursor
, TRUE
);
1076 if (dest
> hdl
->dummyitem
) dest
= hdl
->dummyitem
;
1077 DrawInsert (hwnd
, dlm
->hWnd
, -1);
1079 SendDlgItemMessage(hwnd
, hdl
->listid
,
1080 LB_DELETESTRING
, hdl
->dummyitem
, 0);
1081 if (hdl
->dragging
) {
1084 /* Correct for "missing" item. */
1085 if (dest
> hdl
->srcitem
) dest
--;
1086 pl_moveitem(hwnd
, hdl
->listid
, hdl
->srcitem
, dest
);
1096 if (((LOWORD(wParam
) == hdl
->upbid
) ||
1097 (LOWORD(wParam
) == hdl
->dnbid
)) &&
1098 ((HIWORD(wParam
) == BN_CLICKED
) ||
1099 (HIWORD(wParam
) == BN_DOUBLECLICKED
))) {
1100 /* Move an item up or down the list. */
1101 /* Get the current selection, if any. */
1102 int selection
= SendDlgItemMessage (hwnd
, hdl
->listid
, LB_GETCURSEL
, 0, 0);
1103 if (selection
== LB_ERR
) {
1107 /* Get the total number of items. */
1108 nitems
= SendDlgItemMessage (hwnd
, hdl
->listid
, LB_GETCOUNT
, 0, 0);
1109 /* Should we do anything? */
1110 if (LOWORD(wParam
) == hdl
->upbid
&& (selection
> 0))
1111 pl_moveitem(hwnd
, hdl
->listid
, selection
, selection
- 1);
1112 else if (LOWORD(wParam
) == hdl
->dnbid
&& (selection
< nitems
- 1))
1113 pl_moveitem(hwnd
, hdl
->listid
, selection
, selection
+ 1);
1122 /* Update array to match the list box. */
1123 for (i
=0; i
< maxmemb
; i
++)
1124 array
[i
] = SendDlgItemMessage (hwnd
, hdl
->listid
, LB_GETITEMDATA
,
1132 * A progress bar (from Common Controls). We like our progress bars
1133 * to be smooth and unbroken, without those ugly divisions; some
1134 * older compilers may not support that, but that's life.
1136 void progressbar(struct ctlpos
*cp
, int id
)
1140 r
.left
= GAPBETWEEN
;
1142 r
.right
= cp
->width
;
1143 r
.bottom
= PROGBARHEIGHT
;
1144 cp
->ypos
+= r
.bottom
+ GAPBETWEEN
;
1146 doctl(cp
, r
, PROGRESS_CLASS
, WS_CHILD
| WS_VISIBLE
1150 , WS_EX_CLIENTEDGE
, "", id
);
1153 /* ----------------------------------------------------------------------
1154 * Platform-specific side of portable dialog-box mechanism.
1158 * This function takes a string, escapes all the ampersands, and
1159 * places a single (unescaped) ampersand in front of the first
1160 * occurrence of the given shortcut character (which may be
1163 * Return value is a malloc'ed copy of the processed version of the
1166 static char *shortcut_escape(const char *text
, char shortcut
)
1173 return NULL
; /* sfree won't choke on this */
1175 ret
= snewn(2*strlen(text
)+1, char); /* size potentially doubles! */
1176 shortcut
= tolower((unsigned char)shortcut
);
1181 if (shortcut
!= NO_SHORTCUT
&&
1182 tolower((unsigned char)*p
) == shortcut
) {
1184 shortcut
= NO_SHORTCUT
; /* stop it happening twice */
1185 } else if (*p
== '&') {
1194 void winctrl_add_shortcuts(struct dlgparam
*dp
, struct winctrl
*c
)
1197 for (i
= 0; i
< lenof(c
->shortcuts
); i
++)
1198 if (c
->shortcuts
[i
] != NO_SHORTCUT
) {
1199 unsigned char s
= tolower((unsigned char)c
->shortcuts
[i
]);
1200 assert(!dp
->shortcuts
[s
]);
1201 dp
->shortcuts
[s
] = TRUE
;
1205 void winctrl_rem_shortcuts(struct dlgparam
*dp
, struct winctrl
*c
)
1208 for (i
= 0; i
< lenof(c
->shortcuts
); i
++)
1209 if (c
->shortcuts
[i
] != NO_SHORTCUT
) {
1210 unsigned char s
= tolower((unsigned char)c
->shortcuts
[i
]);
1211 assert(dp
->shortcuts
[s
]);
1212 dp
->shortcuts
[s
] = FALSE
;
1216 static int winctrl_cmp_byctrl(void *av
, void *bv
)
1218 struct winctrl
*a
= (struct winctrl
*)av
;
1219 struct winctrl
*b
= (struct winctrl
*)bv
;
1220 if (a
->ctrl
< b
->ctrl
)
1222 else if (a
->ctrl
> b
->ctrl
)
1227 static int winctrl_cmp_byid(void *av
, void *bv
)
1229 struct winctrl
*a
= (struct winctrl
*)av
;
1230 struct winctrl
*b
= (struct winctrl
*)bv
;
1231 if (a
->base_id
< b
->base_id
)
1233 else if (a
->base_id
> b
->base_id
)
1238 static int winctrl_cmp_byctrl_find(void *av
, void *bv
)
1240 union control
*a
= (union control
*)av
;
1241 struct winctrl
*b
= (struct winctrl
*)bv
;
1244 else if (a
> b
->ctrl
)
1249 static int winctrl_cmp_byid_find(void *av
, void *bv
)
1252 struct winctrl
*b
= (struct winctrl
*)bv
;
1253 if (*a
< b
->base_id
)
1255 else if (*a
>= b
->base_id
+ b
->num_ids
)
1261 void winctrl_init(struct winctrls
*wc
)
1263 wc
->byctrl
= newtree234(winctrl_cmp_byctrl
);
1264 wc
->byid
= newtree234(winctrl_cmp_byid
);
1266 void winctrl_cleanup(struct winctrls
*wc
)
1270 while ((c
= index234(wc
->byid
, 0)) != NULL
) {
1271 winctrl_remove(wc
, c
);
1276 freetree234(wc
->byctrl
);
1277 freetree234(wc
->byid
);
1278 wc
->byctrl
= wc
->byid
= NULL
;
1281 void winctrl_add(struct winctrls
*wc
, struct winctrl
*c
)
1283 struct winctrl
*ret
;
1285 ret
= add234(wc
->byctrl
, c
);
1288 ret
= add234(wc
->byid
, c
);
1292 void winctrl_remove(struct winctrls
*wc
, struct winctrl
*c
)
1294 struct winctrl
*ret
;
1295 ret
= del234(wc
->byctrl
, c
);
1297 ret
= del234(wc
->byid
, c
);
1301 struct winctrl
*winctrl_findbyctrl(struct winctrls
*wc
, union control
*ctrl
)
1303 return find234(wc
->byctrl
, ctrl
, winctrl_cmp_byctrl_find
);
1306 struct winctrl
*winctrl_findbyid(struct winctrls
*wc
, int id
)
1308 return find234(wc
->byid
, &id
, winctrl_cmp_byid_find
);
1311 struct winctrl
*winctrl_findbyindex(struct winctrls
*wc
, int index
)
1313 return index234(wc
->byid
, index
);
1316 void winctrl_layout(struct dlgparam
*dp
, struct winctrls
*wc
,
1317 struct ctlpos
*cp
, struct controlset
*s
, int *id
)
1319 struct ctlpos columns
[16];
1320 int ncols
, colstart
, colspan
;
1322 struct ctlpos tabdelays
[16];
1323 union control
*tabdelayed
[16];
1328 char shortcuts
[MAX_SHORTCUTS_PER_CTRL
];
1331 int i
, actual_base_id
, base_id
, num_ids
;
1336 /* Start a containing box, if we have a boxname. */
1337 if (s
->boxname
&& *s
->boxname
) {
1338 struct winctrl
*c
= snew(struct winctrl
);
1340 c
->base_id
= base_id
;
1343 memset(c
->shortcuts
, NO_SHORTCUT
, lenof(c
->shortcuts
));
1345 beginbox(cp
, s
->boxtitle
, base_id
);
1349 /* Draw a title, if we have one. */
1350 if (!s
->boxname
&& s
->boxtitle
) {
1351 struct winctrl
*c
= snew(struct winctrl
);
1353 c
->base_id
= base_id
;
1355 c
->data
= dupstr(s
->boxtitle
);
1356 memset(c
->shortcuts
, NO_SHORTCUT
, lenof(c
->shortcuts
));
1358 paneltitle(cp
, base_id
);
1362 /* Initially we have just one column. */
1364 columns
[0] = *cp
; /* structure copy */
1366 /* And initially, there are no pending tab-delayed controls. */
1369 /* Loop over each control in the controlset. */
1370 for (i
= 0; i
< s
->ncontrols
; i
++) {
1371 union control
*ctrl
= s
->ctrls
[i
];
1374 * Generic processing that pertains to all control types.
1375 * At the end of this if statement, we'll have produced
1376 * `ctrl' (a pointer to the control we have to create, or
1377 * think about creating, in this iteration of the loop),
1378 * `pos' (a suitable ctlpos with which to position it), and
1379 * `c' (a winctrl structure to receive details of the
1380 * dialog IDs). Or we'll have done a `continue', if it was
1381 * CTRL_COLUMNS and doesn't require any control creation at
1384 if (ctrl
->generic
.type
== CTRL_COLUMNS
) {
1385 assert((ctrl
->columns
.ncols
== 1) ^ (ncols
== 1));
1389 * We're splitting into multiple columns.
1391 int lpercent
, rpercent
, lx
, rx
, i
;
1393 ncols
= ctrl
->columns
.ncols
;
1394 assert(ncols
<= lenof(columns
));
1395 for (i
= 1; i
< ncols
; i
++)
1396 columns
[i
] = columns
[0]; /* structure copy */
1399 for (i
= 0; i
< ncols
; i
++) {
1400 rpercent
= lpercent
+ ctrl
->columns
.percentages
[i
];
1401 lx
= columns
[i
].xoff
+ lpercent
*
1402 (columns
[i
].width
+ GAPBETWEEN
) / 100;
1403 rx
= columns
[i
].xoff
+ rpercent
*
1404 (columns
[i
].width
+ GAPBETWEEN
) / 100;
1405 columns
[i
].xoff
= lx
;
1406 columns
[i
].width
= rx
- lx
- GAPBETWEEN
;
1407 lpercent
= rpercent
;
1411 * We're recombining the various columns into one.
1413 int maxy
= columns
[0].ypos
;
1415 for (i
= 1; i
< ncols
; i
++)
1416 if (maxy
< columns
[i
].ypos
)
1417 maxy
= columns
[i
].ypos
;
1419 columns
[0] = *cp
; /* structure copy */
1420 columns
[0].ypos
= maxy
;
1424 } else if (ctrl
->generic
.type
== CTRL_TABDELAY
) {
1427 assert(!ctrl
->generic
.tabdelay
);
1428 ctrl
= ctrl
->tabdelay
.ctrl
;
1430 for (i
= 0; i
< ntabdelays
; i
++)
1431 if (tabdelayed
[i
] == ctrl
)
1433 assert(i
< ntabdelays
); /* we have to have found it */
1435 pos
= tabdelays
[i
]; /* structure copy */
1437 colstart
= colspan
= -1; /* indicate this was tab-delayed */
1441 * If it wasn't one of those, it's a genuine control;
1442 * so we'll have to compute a position for it now, by
1443 * checking its column span.
1447 colstart
= COLUMN_START(ctrl
->generic
.column
);
1448 colspan
= COLUMN_SPAN(ctrl
->generic
.column
);
1450 pos
= columns
[colstart
]; /* structure copy */
1451 pos
.width
= columns
[colstart
+colspan
-1].width
+
1452 (columns
[colstart
+colspan
-1].xoff
- columns
[colstart
].xoff
);
1454 for (col
= colstart
; col
< colstart
+colspan
; col
++)
1455 if (pos
.ypos
< columns
[col
].ypos
)
1456 pos
.ypos
= columns
[col
].ypos
;
1459 * If this control is to be tabdelayed, add it to the
1460 * tabdelay list, and unset pos.hwnd to inhibit actual
1463 if (ctrl
->generic
.tabdelay
) {
1464 assert(ntabdelays
< lenof(tabdelays
));
1465 tabdelays
[ntabdelays
] = pos
; /* structure copy */
1466 tabdelayed
[ntabdelays
] = ctrl
;
1472 /* Most controls don't need anything in c->data. */
1475 /* And they all start off with no shortcuts registered. */
1476 memset(shortcuts
, NO_SHORTCUT
, lenof(shortcuts
));
1479 /* Almost all controls start at base_id. */
1480 actual_base_id
= base_id
;
1483 * Now we're ready to actually create the control, by
1484 * switching on its type.
1486 switch (ctrl
->generic
.type
) {
1489 char *wrapped
, *escaped
;
1492 wrapped
= staticwrap(&pos
, cp
->hwnd
,
1493 ctrl
->generic
.label
, &lines
);
1494 escaped
= shortcut_escape(wrapped
, NO_SHORTCUT
);
1495 statictext(&pos
, escaped
, lines
, base_id
);
1501 num_ids
= 2; /* static, edit */
1502 escaped
= shortcut_escape(ctrl
->editbox
.label
,
1503 ctrl
->editbox
.shortcut
);
1504 shortcuts
[nshortcuts
++] = ctrl
->editbox
.shortcut
;
1505 if (ctrl
->editbox
.percentwidth
== 100) {
1506 if (ctrl
->editbox
.has_list
)
1507 combobox(&pos
, escaped
,
1508 base_id
, base_id
+1);
1510 editboxfw(&pos
, ctrl
->editbox
.password
, escaped
,
1511 base_id
, base_id
+1);
1513 if (ctrl
->editbox
.has_list
) {
1514 staticcombo(&pos
, escaped
, base_id
, base_id
+1,
1515 ctrl
->editbox
.percentwidth
);
1517 (ctrl
->editbox
.password ? staticpassedit
: staticedit
)
1518 (&pos
, escaped
, base_id
, base_id
+1,
1519 ctrl
->editbox
.percentwidth
);
1525 num_ids
= ctrl
->radio
.nbuttons
+ 1; /* label as well */
1527 struct radio
*buttons
;
1530 escaped
= shortcut_escape(ctrl
->radio
.label
,
1531 ctrl
->radio
.shortcut
);
1532 shortcuts
[nshortcuts
++] = ctrl
->radio
.shortcut
;
1534 buttons
= snewn(ctrl
->radio
.nbuttons
, struct radio
);
1536 for (i
= 0; i
< ctrl
->radio
.nbuttons
; i
++) {
1538 shortcut_escape(ctrl
->radio
.buttons
[i
],
1539 (char)(ctrl
->radio
.shortcuts ?
1540 ctrl
->radio
.shortcuts
[i
] :
1542 buttons
[i
].id
= base_id
+ 1 + i
;
1543 if (ctrl
->radio
.shortcuts
) {
1544 assert(nshortcuts
< MAX_SHORTCUTS_PER_CTRL
);
1545 shortcuts
[nshortcuts
++] = ctrl
->radio
.shortcuts
[i
];
1549 radioline_common(&pos
, escaped
, base_id
,
1550 ctrl
->radio
.ncolumns
,
1551 buttons
, ctrl
->radio
.nbuttons
);
1553 for (i
= 0; i
< ctrl
->radio
.nbuttons
; i
++) {
1554 sfree(buttons
[i
].text
);
1562 escaped
= shortcut_escape(ctrl
->checkbox
.label
,
1563 ctrl
->checkbox
.shortcut
);
1564 shortcuts
[nshortcuts
++] = ctrl
->checkbox
.shortcut
;
1565 checkbox(&pos
, escaped
, base_id
);
1569 escaped
= shortcut_escape(ctrl
->button
.label
,
1570 ctrl
->button
.shortcut
);
1571 shortcuts
[nshortcuts
++] = ctrl
->button
.shortcut
;
1572 if (ctrl
->button
.iscancel
)
1573 actual_base_id
= IDCANCEL
;
1575 button(&pos
, escaped
, actual_base_id
, ctrl
->button
.isdefault
);
1580 escaped
= shortcut_escape(ctrl
->listbox
.label
,
1581 ctrl
->listbox
.shortcut
);
1582 shortcuts
[nshortcuts
++] = ctrl
->listbox
.shortcut
;
1583 if (ctrl
->listbox
.draglist
) {
1584 data
= snew(struct prefslist
);
1586 prefslist(data
, &pos
, ctrl
->listbox
.height
, escaped
,
1587 base_id
, base_id
+1, base_id
+2, base_id
+3);
1588 shortcuts
[nshortcuts
++] = 'u'; /* Up */
1589 shortcuts
[nshortcuts
++] = 'd'; /* Down */
1590 } else if (ctrl
->listbox
.height
== 0) {
1591 /* Drop-down list. */
1592 if (ctrl
->listbox
.percentwidth
== 100) {
1593 staticddlbig(&pos
, escaped
,
1594 base_id
, base_id
+1);
1596 staticddl(&pos
, escaped
, base_id
,
1597 base_id
+1, ctrl
->listbox
.percentwidth
);
1600 /* Ordinary list. */
1601 listbox(&pos
, escaped
, base_id
, base_id
+1,
1602 ctrl
->listbox
.height
, ctrl
->listbox
.multisel
);
1604 if (ctrl
->listbox
.ncols
) {
1606 * This method of getting the box width is a bit of
1607 * a hack; we'd do better to try to retrieve the
1608 * actual width in dialog units from doctl() just
1609 * before MapDialogRect. But that's going to be no
1610 * fun, and this should be good enough accuracy.
1612 int width
= cp
->width
* ctrl
->listbox
.percentwidth
;
1616 tabarray
= snewn(ctrl
->listbox
.ncols
-1, int);
1618 for (i
= 0; i
< ctrl
->listbox
.ncols
-1; i
++) {
1619 percent
+= ctrl
->listbox
.percentages
[i
];
1620 tabarray
[i
] = width
* percent
/ 10000;
1622 SendDlgItemMessage(cp
->hwnd
, base_id
+1, LB_SETTABSTOPS
,
1623 ctrl
->listbox
.ncols
-1, (LPARAM
)tabarray
);
1628 case CTRL_FILESELECT
:
1630 escaped
= shortcut_escape(ctrl
->fileselect
.label
,
1631 ctrl
->fileselect
.shortcut
);
1632 shortcuts
[nshortcuts
++] = ctrl
->fileselect
.shortcut
;
1633 editbutton(&pos
, escaped
, base_id
, base_id
+1,
1634 "Bro&wse...", base_id
+2);
1635 shortcuts
[nshortcuts
++] = 'w';
1638 case CTRL_FONTSELECT
:
1640 escaped
= shortcut_escape(ctrl
->fontselect
.label
,
1641 ctrl
->fontselect
.shortcut
);
1642 shortcuts
[nshortcuts
++] = ctrl
->fontselect
.shortcut
;
1643 statictext(&pos
, escaped
, 1, base_id
);
1644 staticbtn(&pos
, "", base_id
+1, "Change...", base_id
+2);
1645 data
= fontspec_new("", 0, 0, 0);
1649 assert(!"Can't happen");
1650 num_ids
= 0; /* placate gcc */
1655 * Create a `struct winctrl' for this control, and advance
1656 * the dialog ID counter, if it's actually been created
1657 * (and isn't tabdelayed).
1660 struct winctrl
*c
= snew(struct winctrl
);
1663 c
->base_id
= actual_base_id
;
1664 c
->num_ids
= num_ids
;
1666 memcpy(c
->shortcuts
, shortcuts
, sizeof(shortcuts
));
1668 winctrl_add_shortcuts(dp
, c
);
1669 if (actual_base_id
== base_id
)
1675 if (colstart
>= 0) {
1677 * Update the ypos in all columns crossed by this
1681 for (i
= colstart
; i
< colstart
+colspan
; i
++)
1682 columns
[i
].ypos
= pos
.ypos
;
1687 * We've now finished laying out the controls; so now update
1688 * the ctlpos and control ID that were passed in, terminate
1689 * any containing box, and return.
1691 for (i
= 0; i
< ncols
; i
++)
1692 if (cp
->ypos
< columns
[i
].ypos
)
1693 cp
->ypos
= columns
[i
].ypos
;
1696 if (s
->boxname
&& *s
->boxname
)
1700 static void winctrl_set_focus(union control
*ctrl
, struct dlgparam
*dp
,
1705 dp
->lastfocused
= dp
->focused
;
1707 } else if (!has_focus
&& dp
->focused
== ctrl
) {
1708 dp
->lastfocused
= dp
->focused
;
1713 union control
*dlg_last_focused(union control
*ctrl
, void *dlg
)
1715 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
1716 return dp
->focused
== ctrl ? dp
->lastfocused
: dp
->focused
;
1720 * The dialog-box procedure calls this function to handle Windows
1721 * messages on a control we manage.
1723 int winctrl_handle_command(struct dlgparam
*dp
, UINT msg
,
1724 WPARAM wParam
, LPARAM lParam
)
1727 union control
*ctrl
;
1729 static UINT draglistmsg
= WM_NULL
;
1732 * Filter out pointless window messages. Our interest is in
1733 * WM_COMMAND and the drag list message, and nothing else.
1735 if (draglistmsg
== WM_NULL
)
1736 draglistmsg
= RegisterWindowMessage (DRAGLISTMSGSTRING
);
1738 if (msg
!= draglistmsg
&& msg
!= WM_COMMAND
&& msg
!= WM_DRAWITEM
)
1742 * Look up the control ID in our data.
1745 for (i
= 0; i
< dp
->nctrltrees
; i
++) {
1746 c
= winctrl_findbyid(dp
->controltrees
[i
], LOWORD(wParam
));
1751 return 0; /* we have nothing to do */
1753 if (msg
== WM_DRAWITEM
) {
1755 * Owner-draw request for a panel title.
1757 LPDRAWITEMSTRUCT di
= (LPDRAWITEMSTRUCT
) lParam
;
1759 RECT r
= di
->rcItem
;
1762 SetMapMode(hdc
, MM_TEXT
); /* ensure logical units == pixels */
1764 GetTextExtentPoint32(hdc
, (char *)c
->data
,
1765 strlen((char *)c
->data
), &s
);
1766 DrawEdge(hdc
, &r
, EDGE_ETCHED
, BF_ADJUST
| BF_RECT
);
1768 r
.left
+ (r
.right
-r
.left
-s
.cx
)/2,
1769 r
.top
+ (r
.bottom
-r
.top
-s
.cy
)/2,
1770 (char *)c
->data
, strlen((char *)c
->data
));
1776 id
= LOWORD(wParam
) - c
->base_id
;
1778 if (!ctrl
|| !ctrl
->generic
.handler
)
1779 return 0; /* nothing we can do here */
1782 * From here on we do not issue `return' statements until the
1783 * very end of the dialog box: any event handler is entitled to
1784 * ask for a colour selector, so we _must_ always allow control
1785 * to reach the end of this switch statement so that the
1786 * subsequent code can test dp->coloursel_wanted().
1789 dp
->coloursel_wanted
= FALSE
;
1792 * Now switch on the control type and the message.
1794 switch (ctrl
->generic
.type
) {
1796 if (msg
== WM_COMMAND
&& !ctrl
->editbox
.has_list
&&
1797 (HIWORD(wParam
) == EN_SETFOCUS
|| HIWORD(wParam
) == EN_KILLFOCUS
))
1798 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == EN_SETFOCUS
);
1799 if (msg
== WM_COMMAND
&& ctrl
->editbox
.has_list
&&
1800 (HIWORD(wParam
)==CBN_SETFOCUS
|| HIWORD(wParam
)==CBN_KILLFOCUS
))
1801 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == CBN_SETFOCUS
);
1803 if (msg
== WM_COMMAND
&& !ctrl
->editbox
.has_list
&&
1804 HIWORD(wParam
) == EN_CHANGE
)
1805 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1806 if (msg
== WM_COMMAND
&&
1807 ctrl
->editbox
.has_list
) {
1808 if (HIWORD(wParam
) == CBN_SELCHANGE
) {
1812 index
= SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1,
1813 CB_GETCURSEL
, 0, 0);
1814 len
= SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1,
1815 CB_GETLBTEXTLEN
, index
, 0);
1816 text
= snewn(len
+1, char);
1817 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, CB_GETLBTEXT
,
1818 index
, (LPARAM
)text
);
1819 SetDlgItemText(dp
->hwnd
, c
->base_id
+1, text
);
1821 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1822 } else if (HIWORD(wParam
) == CBN_EDITCHANGE
) {
1823 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1824 } else if (HIWORD(wParam
) == CBN_KILLFOCUS
) {
1825 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_REFRESH
);
1831 if (msg
== WM_COMMAND
&&
1832 (HIWORD(wParam
) == BN_SETFOCUS
|| HIWORD(wParam
) == BN_KILLFOCUS
))
1833 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == BN_SETFOCUS
);
1835 * We sometimes get spurious BN_CLICKED messages for the
1836 * radio button that is just about to _lose_ selection, if
1837 * we're switching using the arrow keys. Therefore we
1838 * double-check that the button in wParam is actually
1839 * checked before generating an event.
1841 if (msg
== WM_COMMAND
&&
1842 (HIWORD(wParam
) == BN_CLICKED
||
1843 HIWORD(wParam
) == BN_DOUBLECLICKED
) &&
1844 IsDlgButtonChecked(dp
->hwnd
, LOWORD(wParam
))) {
1845 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1849 if (msg
== WM_COMMAND
&&
1850 (HIWORD(wParam
) == BN_SETFOCUS
|| HIWORD(wParam
) == BN_KILLFOCUS
))
1851 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == BN_SETFOCUS
);
1852 if (msg
== WM_COMMAND
&&
1853 (HIWORD(wParam
) == BN_CLICKED
||
1854 HIWORD(wParam
) == BN_DOUBLECLICKED
)) {
1855 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1859 if (msg
== WM_COMMAND
&&
1860 (HIWORD(wParam
) == BN_SETFOCUS
|| HIWORD(wParam
) == BN_KILLFOCUS
))
1861 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == BN_SETFOCUS
);
1862 if (msg
== WM_COMMAND
&&
1863 (HIWORD(wParam
) == BN_CLICKED
||
1864 HIWORD(wParam
) == BN_DOUBLECLICKED
)) {
1865 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_ACTION
);
1869 if (msg
== WM_COMMAND
&& ctrl
->listbox
.height
!= 0 &&
1870 (HIWORD(wParam
)==LBN_SETFOCUS
|| HIWORD(wParam
)==LBN_KILLFOCUS
))
1871 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == LBN_SETFOCUS
);
1872 if (msg
== WM_COMMAND
&& ctrl
->listbox
.height
== 0 &&
1873 (HIWORD(wParam
)==CBN_SETFOCUS
|| HIWORD(wParam
)==CBN_KILLFOCUS
))
1874 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == CBN_SETFOCUS
);
1875 if (msg
== WM_COMMAND
&& id
>= 2 &&
1876 (HIWORD(wParam
) == BN_SETFOCUS
|| HIWORD(wParam
) == BN_KILLFOCUS
))
1877 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == BN_SETFOCUS
);
1878 if (ctrl
->listbox
.draglist
) {
1880 pret
= handle_prefslist(c
->data
, NULL
, 0, (msg
!= WM_COMMAND
),
1881 dp
->hwnd
, wParam
, lParam
);
1883 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1886 if (msg
== WM_COMMAND
&& HIWORD(wParam
) == LBN_DBLCLK
) {
1887 SetCapture(dp
->hwnd
);
1888 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_ACTION
);
1889 } else if (msg
== WM_COMMAND
&& HIWORD(wParam
) == LBN_SELCHANGE
) {
1890 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_SELCHANGE
);
1894 case CTRL_FILESELECT
:
1895 if (msg
== WM_COMMAND
&& id
== 1 &&
1896 (HIWORD(wParam
) == EN_SETFOCUS
|| HIWORD(wParam
) == EN_KILLFOCUS
))
1897 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == EN_SETFOCUS
);
1898 if (msg
== WM_COMMAND
&& id
== 2 &&
1899 (HIWORD(wParam
) == BN_SETFOCUS
|| HIWORD(wParam
) == BN_KILLFOCUS
))
1900 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == BN_SETFOCUS
);
1901 if (msg
== WM_COMMAND
&& id
== 1 && HIWORD(wParam
) == EN_CHANGE
)
1902 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1904 (msg
== WM_COMMAND
&&
1905 (HIWORD(wParam
) == BN_CLICKED
||
1906 HIWORD(wParam
) == BN_DOUBLECLICKED
))) {
1908 char filename
[FILENAME_MAX
];
1910 memset(&of
, 0, sizeof(of
));
1911 of
.hwndOwner
= dp
->hwnd
;
1912 if (ctrl
->fileselect
.filter
)
1913 of
.lpstrFilter
= ctrl
->fileselect
.filter
;
1915 of
.lpstrFilter
= "All Files (*.*)\0*\0\0\0";
1916 of
.lpstrCustomFilter
= NULL
;
1917 of
.nFilterIndex
= 1;
1918 of
.lpstrFile
= filename
;
1919 GetDlgItemText(dp
->hwnd
, c
->base_id
+1, filename
, lenof(filename
));
1920 filename
[lenof(filename
)-1] = '\0';
1921 of
.nMaxFile
= lenof(filename
);
1922 of
.lpstrFileTitle
= NULL
;
1923 of
.lpstrTitle
= ctrl
->fileselect
.title
;
1925 if (request_file(NULL
, &of
, FALSE
, ctrl
->fileselect
.for_writing
)) {
1926 SetDlgItemText(dp
->hwnd
, c
->base_id
+ 1, filename
);
1927 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1931 case CTRL_FONTSELECT
:
1932 if (msg
== WM_COMMAND
&& id
== 2 &&
1933 (HIWORD(wParam
) == BN_SETFOCUS
|| HIWORD(wParam
) == BN_KILLFOCUS
))
1934 winctrl_set_focus(ctrl
, dp
, HIWORD(wParam
) == BN_SETFOCUS
);
1936 (msg
== WM_COMMAND
&&
1937 (HIWORD(wParam
) == BN_CLICKED
||
1938 HIWORD(wParam
) == BN_DOUBLECLICKED
))) {
1942 FontSpec
*fs
= (FontSpec
*)c
->data
;
1945 lf
.lfHeight
= -MulDiv(fs
->height
,
1946 GetDeviceCaps(hdc
, LOGPIXELSY
), 72);
1948 lf
.lfWidth
= lf
.lfEscapement
= lf
.lfOrientation
= 0;
1949 lf
.lfItalic
= lf
.lfUnderline
= lf
.lfStrikeOut
= 0;
1950 lf
.lfWeight
= (fs
->isbold ? FW_BOLD
: 0);
1951 lf
.lfCharSet
= fs
->charset
;
1952 lf
.lfOutPrecision
= OUT_DEFAULT_PRECIS
;
1953 lf
.lfClipPrecision
= CLIP_DEFAULT_PRECIS
;
1954 lf
.lfQuality
= DEFAULT_QUALITY
;
1955 lf
.lfPitchAndFamily
= FIXED_PITCH
| FF_DONTCARE
;
1956 strncpy(lf
.lfFaceName
, fs
->name
,
1957 sizeof(lf
.lfFaceName
) - 1);
1958 lf
.lfFaceName
[sizeof(lf
.lfFaceName
) - 1] = '\0';
1960 cf
.lStructSize
= sizeof(cf
);
1961 cf
.hwndOwner
= dp
->hwnd
;
1963 cf
.Flags
= (dp
->fixed_pitch_fonts ? CF_FIXEDPITCHONLY
: 0) |
1964 CF_FORCEFONTEXIST
| CF_INITTOLOGFONTSTRUCT
| CF_SCREENFONTS
;
1966 if (ChooseFont(&cf
)) {
1967 fs
= fontspec_new(lf
.lfFaceName
, (lf
.lfWeight
== FW_BOLD
),
1968 cf
.iPointSize
/ 10, lf
.lfCharSet
);
1969 dlg_fontsel_set(ctrl
, dp
, fs
);
1972 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_VALCHANGE
);
1979 * If the above event handler has asked for a colour selector,
1980 * now is the time to generate one.
1982 if (dp
->coloursel_wanted
) {
1983 static CHOOSECOLOR cc
;
1984 static DWORD custom
[16] = { 0 }; /* zero initialisers */
1985 cc
.lStructSize
= sizeof(cc
);
1986 cc
.hwndOwner
= dp
->hwnd
;
1987 cc
.hInstance
= (HWND
) hinst
;
1988 cc
.lpCustColors
= custom
;
1989 cc
.rgbResult
= RGB(dp
->coloursel_result
.r
,
1990 dp
->coloursel_result
.g
,
1991 dp
->coloursel_result
.b
);
1992 cc
.Flags
= CC_FULLOPEN
| CC_RGBINIT
;
1993 if (ChooseColor(&cc
)) {
1994 dp
->coloursel_result
.r
=
1995 (unsigned char) (cc
.rgbResult
& 0xFF);
1996 dp
->coloursel_result
.g
=
1997 (unsigned char) (cc
.rgbResult
>> 8) & 0xFF;
1998 dp
->coloursel_result
.b
=
1999 (unsigned char) (cc
.rgbResult
>> 16) & 0xFF;
2000 dp
->coloursel_result
.ok
= TRUE
;
2002 dp
->coloursel_result
.ok
= FALSE
;
2003 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_CALLBACK
);
2010 * This function can be called to produce context help on a
2011 * control. Returns TRUE if it has actually launched some help.
2013 int winctrl_context_help(struct dlgparam
*dp
, HWND hwnd
, int id
)
2019 * Look up the control ID in our data.
2022 for (i
= 0; i
< dp
->nctrltrees
; i
++) {
2023 c
= winctrl_findbyid(dp
->controltrees
[i
], id
);
2028 return 0; /* we have nothing to do */
2031 * This is the Windows front end, so we're allowed to assume
2032 * `helpctx.p' is a context string.
2034 if (!c
->ctrl
|| !c
->ctrl
->generic
.helpctx
.p
)
2035 return 0; /* no help available for this ctrl */
2037 launch_help(hwnd
, c
->ctrl
->generic
.helpctx
.p
);
2042 * Now the various functions that the platform-independent
2043 * mechanism can call to access the dialog box entries.
2046 static struct winctrl
*dlg_findbyctrl(struct dlgparam
*dp
, union control
*ctrl
)
2050 for (i
= 0; i
< dp
->nctrltrees
; i
++) {
2051 struct winctrl
*c
= winctrl_findbyctrl(dp
->controltrees
[i
], ctrl
);
2058 void dlg_radiobutton_set(union control
*ctrl
, void *dlg
, int whichbutton
)
2060 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2061 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2062 assert(c
&& c
->ctrl
->generic
.type
== CTRL_RADIO
);
2063 CheckRadioButton(dp
->hwnd
,
2065 c
->base_id
+ c
->ctrl
->radio
.nbuttons
,
2066 c
->base_id
+ 1 + whichbutton
);
2069 int dlg_radiobutton_get(union control
*ctrl
, void *dlg
)
2071 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2072 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2074 assert(c
&& c
->ctrl
->generic
.type
== CTRL_RADIO
);
2075 for (i
= 0; i
< c
->ctrl
->radio
.nbuttons
; i
++)
2076 if (IsDlgButtonChecked(dp
->hwnd
, c
->base_id
+ 1 + i
))
2078 assert(!"No radio button was checked?!");
2082 void dlg_checkbox_set(union control
*ctrl
, void *dlg
, int checked
)
2084 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2085 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2086 assert(c
&& c
->ctrl
->generic
.type
== CTRL_CHECKBOX
);
2087 CheckDlgButton(dp
->hwnd
, c
->base_id
, (checked
!= 0));
2090 int dlg_checkbox_get(union control
*ctrl
, void *dlg
)
2092 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2093 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2094 assert(c
&& c
->ctrl
->generic
.type
== CTRL_CHECKBOX
);
2095 return 0 != IsDlgButtonChecked(dp
->hwnd
, c
->base_id
);
2098 void dlg_editbox_set(union control
*ctrl
, void *dlg
, char const *text
)
2100 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2101 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2102 assert(c
&& c
->ctrl
->generic
.type
== CTRL_EDITBOX
);
2103 SetDlgItemText(dp
->hwnd
, c
->base_id
+1, text
);
2106 char *dlg_editbox_get(union control
*ctrl
, void *dlg
)
2108 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2109 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2110 assert(c
&& c
->ctrl
->generic
.type
== CTRL_EDITBOX
);
2111 return GetDlgItemText_alloc(dp
->hwnd
, c
->base_id
+1);
2114 /* The `listbox' functions can also apply to combo boxes. */
2115 void dlg_listbox_clear(union control
*ctrl
, void *dlg
)
2117 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2118 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2121 (c
->ctrl
->generic
.type
== CTRL_LISTBOX
||
2122 (c
->ctrl
->generic
.type
== CTRL_EDITBOX
&&
2123 c
->ctrl
->editbox
.has_list
)));
2124 msg
= (c
->ctrl
->generic
.type
==CTRL_LISTBOX
&& c
->ctrl
->listbox
.height
!=0 ?
2125 LB_RESETCONTENT
: CB_RESETCONTENT
);
2126 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, msg
, 0, 0);
2129 void dlg_listbox_del(union control
*ctrl
, void *dlg
, int index
)
2131 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2132 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2135 (c
->ctrl
->generic
.type
== CTRL_LISTBOX
||
2136 (c
->ctrl
->generic
.type
== CTRL_EDITBOX
&&
2137 c
->ctrl
->editbox
.has_list
)));
2138 msg
= (c
->ctrl
->generic
.type
==CTRL_LISTBOX
&& c
->ctrl
->listbox
.height
!=0 ?
2139 LB_DELETESTRING
: CB_DELETESTRING
);
2140 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, msg
, index
, 0);
2143 void dlg_listbox_add(union control
*ctrl
, void *dlg
, char const *text
)
2145 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2146 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2149 (c
->ctrl
->generic
.type
== CTRL_LISTBOX
||
2150 (c
->ctrl
->generic
.type
== CTRL_EDITBOX
&&
2151 c
->ctrl
->editbox
.has_list
)));
2152 msg
= (c
->ctrl
->generic
.type
==CTRL_LISTBOX
&& c
->ctrl
->listbox
.height
!=0 ?
2153 LB_ADDSTRING
: CB_ADDSTRING
);
2154 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, msg
, 0, (LPARAM
)text
);
2158 * Each listbox entry may have a numeric id associated with it.
2159 * Note that some front ends only permit a string to be stored at
2160 * each position, which means that _if_ you put two identical
2161 * strings in any listbox then you MUST not assign them different
2162 * IDs and expect to get meaningful results back.
2164 void dlg_listbox_addwithid(union control
*ctrl
, void *dlg
,
2165 char const *text
, int id
)
2167 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2168 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2169 int msg
, msg2
, index
;
2171 (c
->ctrl
->generic
.type
== CTRL_LISTBOX
||
2172 (c
->ctrl
->generic
.type
== CTRL_EDITBOX
&&
2173 c
->ctrl
->editbox
.has_list
)));
2174 msg
= (c
->ctrl
->generic
.type
==CTRL_LISTBOX
&& c
->ctrl
->listbox
.height
!=0 ?
2175 LB_ADDSTRING
: CB_ADDSTRING
);
2176 msg2
= (c
->ctrl
->generic
.type
==CTRL_LISTBOX
&& c
->ctrl
->listbox
.height
!=0 ?
2177 LB_SETITEMDATA
: CB_SETITEMDATA
);
2178 index
= SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, msg
, 0, (LPARAM
)text
);
2179 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, msg2
, index
, (LPARAM
)id
);
2182 int dlg_listbox_getid(union control
*ctrl
, void *dlg
, int index
)
2184 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2185 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2187 assert(c
&& c
->ctrl
->generic
.type
== CTRL_LISTBOX
);
2188 msg
= (c
->ctrl
->listbox
.height
!= 0 ? LB_GETITEMDATA
: CB_GETITEMDATA
);
2190 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, msg
, index
, 0);
2193 /* dlg_listbox_index returns <0 if no single element is selected. */
2194 int dlg_listbox_index(union control
*ctrl
, void *dlg
)
2196 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2197 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2199 assert(c
&& c
->ctrl
->generic
.type
== CTRL_LISTBOX
);
2200 if (c
->ctrl
->listbox
.multisel
) {
2201 assert(c
->ctrl
->listbox
.height
!= 0); /* not combo box */
2202 ret
= SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, LB_GETSELCOUNT
, 0, 0);
2203 if (ret
== LB_ERR
|| ret
> 1)
2206 msg
= (c
->ctrl
->listbox
.height
!= 0 ? LB_GETCURSEL
: CB_GETCURSEL
);
2207 ret
= SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, msg
, 0, 0);
2214 int dlg_listbox_issel(union control
*ctrl
, void *dlg
, int index
)
2216 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2217 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2218 assert(c
&& c
->ctrl
->generic
.type
== CTRL_LISTBOX
&&
2219 c
->ctrl
->listbox
.multisel
&&
2220 c
->ctrl
->listbox
.height
!= 0);
2222 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, LB_GETSEL
, index
, 0);
2225 void dlg_listbox_select(union control
*ctrl
, void *dlg
, int index
)
2227 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2228 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2230 assert(c
&& c
->ctrl
->generic
.type
== CTRL_LISTBOX
&&
2231 !c
->ctrl
->listbox
.multisel
);
2232 msg
= (c
->ctrl
->listbox
.height
!= 0 ? LB_SETCURSEL
: CB_SETCURSEL
);
2233 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, msg
, index
, 0);
2236 void dlg_text_set(union control
*ctrl
, void *dlg
, char const *text
)
2238 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2239 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2240 assert(c
&& c
->ctrl
->generic
.type
== CTRL_TEXT
);
2241 SetDlgItemText(dp
->hwnd
, c
->base_id
, text
);
2244 void dlg_label_change(union control
*ctrl
, void *dlg
, char const *text
)
2246 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2247 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2248 char *escaped
= NULL
;
2252 switch (c
->ctrl
->generic
.type
) {
2254 escaped
= shortcut_escape(text
, c
->ctrl
->editbox
.shortcut
);
2258 escaped
= shortcut_escape(text
, c
->ctrl
->radio
.shortcut
);
2262 escaped
= shortcut_escape(text
, ctrl
->checkbox
.shortcut
);
2266 escaped
= shortcut_escape(text
, ctrl
->button
.shortcut
);
2270 escaped
= shortcut_escape(text
, ctrl
->listbox
.shortcut
);
2273 case CTRL_FILESELECT
:
2274 escaped
= shortcut_escape(text
, ctrl
->fileselect
.shortcut
);
2277 case CTRL_FONTSELECT
:
2278 escaped
= shortcut_escape(text
, ctrl
->fontselect
.shortcut
);
2282 assert(!"Can't happen");
2286 SetDlgItemText(dp
->hwnd
, id
, escaped
);
2291 void dlg_filesel_set(union control
*ctrl
, void *dlg
, Filename
*fn
)
2293 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2294 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2295 assert(c
&& c
->ctrl
->generic
.type
== CTRL_FILESELECT
);
2296 SetDlgItemText(dp
->hwnd
, c
->base_id
+1, fn
->path
);
2299 Filename
*dlg_filesel_get(union control
*ctrl
, void *dlg
)
2301 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2302 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2305 assert(c
&& c
->ctrl
->generic
.type
== CTRL_FILESELECT
);
2306 tmp
= GetDlgItemText_alloc(dp
->hwnd
, c
->base_id
+1);
2307 ret
= filename_from_str(tmp
);
2312 void dlg_fontsel_set(union control
*ctrl
, void *dlg
, FontSpec
*fs
)
2314 char *buf
, *boldstr
;
2315 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2316 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2317 assert(c
&& c
->ctrl
->generic
.type
== CTRL_FONTSELECT
);
2319 fontspec_free((FontSpec
*)c
->data
);
2320 c
->data
= fontspec_copy(fs
);
2322 boldstr
= (fs
->isbold ?
"bold, " : "");
2323 if (fs
->height
== 0)
2324 buf
= dupprintf("Font: %s, %sdefault height", fs
->name
, boldstr
);
2326 buf
= dupprintf("Font: %s, %s%d-%s", fs
->name
, boldstr
,
2327 (fs
->height
< 0 ?
-fs
->height
: fs
->height
),
2328 (fs
->height
< 0 ?
"pixel" : "point"));
2329 SetDlgItemText(dp
->hwnd
, c
->base_id
+1, buf
);
2332 dlg_auto_set_fixed_pitch_flag(dp
);
2335 FontSpec
*dlg_fontsel_get(union control
*ctrl
, void *dlg
)
2337 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2338 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2339 assert(c
&& c
->ctrl
->generic
.type
== CTRL_FONTSELECT
);
2340 return fontspec_copy((FontSpec
*)c
->data
);
2344 * Bracketing a large set of updates in these two functions will
2345 * cause the front end (if possible) to delay updating the screen
2346 * until it's all complete, thus avoiding flicker.
2348 void dlg_update_start(union control
*ctrl
, void *dlg
)
2350 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2351 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2352 if (c
&& c
->ctrl
->generic
.type
== CTRL_LISTBOX
) {
2353 SendDlgItemMessage(dp
->hwnd
, c
->base_id
+1, WM_SETREDRAW
, FALSE
, 0);
2357 void dlg_update_done(union control
*ctrl
, void *dlg
)
2359 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2360 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2361 if (c
&& c
->ctrl
->generic
.type
== CTRL_LISTBOX
) {
2362 HWND hw
= GetDlgItem(dp
->hwnd
, c
->base_id
+1);
2363 SendMessage(hw
, WM_SETREDRAW
, TRUE
, 0);
2364 InvalidateRect(hw
, NULL
, TRUE
);
2368 void dlg_set_focus(union control
*ctrl
, void *dlg
)
2370 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2371 struct winctrl
*c
= dlg_findbyctrl(dp
, ctrl
);
2376 switch (ctrl
->generic
.type
) {
2377 case CTRL_EDITBOX
: id
= c
->base_id
+ 1; break;
2379 for (id
= c
->base_id
+ ctrl
->radio
.nbuttons
; id
> 1; id
--)
2380 if (IsDlgButtonChecked(dp
->hwnd
, id
))
2383 * In the theoretically-unlikely case that no button was
2384 * selected, id should come out of this as 1, which is a
2385 * reasonable enough choice.
2388 case CTRL_CHECKBOX
: id
= c
->base_id
; break;
2389 case CTRL_BUTTON
: id
= c
->base_id
; break;
2390 case CTRL_LISTBOX
: id
= c
->base_id
+ 1; break;
2391 case CTRL_FILESELECT
: id
= c
->base_id
+ 1; break;
2392 case CTRL_FONTSELECT
: id
= c
->base_id
+ 2; break;
2393 default: id
= c
->base_id
; break;
2395 ctl
= GetDlgItem(dp
->hwnd
, id
);
2400 * During event processing, you might well want to give an error
2401 * indication to the user. dlg_beep() is a quick and easy generic
2402 * error; dlg_error() puts up a message-box or equivalent.
2404 void dlg_beep(void *dlg
)
2406 /* struct dlgparam *dp = (struct dlgparam *)dlg; */
2410 void dlg_error_msg(void *dlg
, char *msg
)
2412 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2413 MessageBox(dp
->hwnd
, msg
,
2414 dp
->errtitle ? dp
->errtitle
: NULL
,
2415 MB_OK
| MB_ICONERROR
);
2419 * This function signals to the front end that the dialog's
2420 * processing is completed, and passes an integer value (typically
2421 * a success status).
2423 void dlg_end(void *dlg
, int value
)
2425 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2427 dp
->endresult
= value
;
2430 void dlg_refresh(union control
*ctrl
, void *dlg
)
2432 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2438 * Send EVENT_REFRESH to absolutely everything.
2440 for (j
= 0; j
< dp
->nctrltrees
; j
++) {
2442 (c
= winctrl_findbyindex(dp
->controltrees
[j
], i
)) != NULL
;
2444 if (c
->ctrl
&& c
->ctrl
->generic
.handler
!= NULL
)
2445 c
->ctrl
->generic
.handler(c
->ctrl
, dp
,
2446 dp
->data
, EVENT_REFRESH
);
2451 * Send EVENT_REFRESH to a specific control.
2453 if (ctrl
->generic
.handler
!= NULL
)
2454 ctrl
->generic
.handler(ctrl
, dp
, dp
->data
, EVENT_REFRESH
);
2458 void dlg_coloursel_start(union control
*ctrl
, void *dlg
, int r
, int g
, int b
)
2460 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2461 dp
->coloursel_wanted
= TRUE
;
2462 dp
->coloursel_result
.r
= r
;
2463 dp
->coloursel_result
.g
= g
;
2464 dp
->coloursel_result
.b
= b
;
2467 int dlg_coloursel_results(union control
*ctrl
, void *dlg
,
2468 int *r
, int *g
, int *b
)
2470 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2471 if (dp
->coloursel_result
.ok
) {
2472 *r
= dp
->coloursel_result
.r
;
2473 *g
= dp
->coloursel_result
.g
;
2474 *b
= dp
->coloursel_result
.b
;
2480 void dlg_auto_set_fixed_pitch_flag(void *dlg
)
2482 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2483 Conf
*conf
= (Conf
*)dp
->data
;
2492 * Attempt to load the current font, and see if it's
2493 * variable-pitch. If so, start off the fixed-pitch flag for the
2494 * dialog box as false.
2496 * We assume here that any client of the dlg_* mechanism which is
2497 * using font selectors at all is also using a normal 'Conf *'
2501 quality
= conf_get_int(conf
, CONF_font_quality
);
2502 fs
= conf_get_fontspec(conf
, CONF_font
);
2504 hfont
= CreateFont(0, 0, 0, 0, FW_DONTCARE
, FALSE
, FALSE
, FALSE
,
2505 DEFAULT_CHARSET
, OUT_DEFAULT_PRECIS
,
2506 CLIP_DEFAULT_PRECIS
, FONT_QUALITY(quality
),
2507 FIXED_PITCH
| FF_DONTCARE
, fs
->name
);
2509 if (hdc
&& SelectObject(hdc
, hfont
) && GetTextMetrics(hdc
, &tm
)) {
2510 /* Note that the TMPF_FIXED_PITCH bit is defined upside down :-( */
2511 is_var
= (tm
.tmPitchAndFamily
& TMPF_FIXED_PITCH
);
2513 is_var
= FALSE
; /* assume it's basically normal */
2516 ReleaseDC(NULL
, hdc
);
2518 DeleteObject(hfont
);
2521 dp
->fixed_pitch_fonts
= FALSE
;
2524 int dlg_get_fixed_pitch_flag(void *dlg
)
2526 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2527 return dp
->fixed_pitch_fonts
;
2530 void dlg_set_fixed_pitch_flag(void *dlg
, int flag
)
2532 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2533 dp
->fixed_pitch_fonts
= flag
;
2536 struct perctrl_privdata
{
2537 union control
*ctrl
;
2542 static int perctrl_privdata_cmp(void *av
, void *bv
)
2544 struct perctrl_privdata
*a
= (struct perctrl_privdata
*)av
;
2545 struct perctrl_privdata
*b
= (struct perctrl_privdata
*)bv
;
2546 if (a
->ctrl
< b
->ctrl
)
2548 else if (a
->ctrl
> b
->ctrl
)
2553 void dp_init(struct dlgparam
*dp
)
2558 dp
->focused
= dp
->lastfocused
= NULL
;
2559 memset(dp
->shortcuts
, 0, sizeof(dp
->shortcuts
));
2561 dp
->wintitle
= dp
->errtitle
= NULL
;
2562 dp
->privdata
= newtree234(perctrl_privdata_cmp
);
2563 dp
->fixed_pitch_fonts
= TRUE
;
2566 void dp_add_tree(struct dlgparam
*dp
, struct winctrls
*wc
)
2568 assert(dp
->nctrltrees
< lenof(dp
->controltrees
));
2569 dp
->controltrees
[dp
->nctrltrees
++] = wc
;
2572 void dp_cleanup(struct dlgparam
*dp
)
2574 struct perctrl_privdata
*p
;
2577 while ( (p
= index234(dp
->privdata
, 0)) != NULL
) {
2578 del234(dp
->privdata
, p
);
2583 freetree234(dp
->privdata
);
2584 dp
->privdata
= NULL
;
2586 sfree(dp
->wintitle
);
2587 sfree(dp
->errtitle
);
2590 void *dlg_get_privdata(union control
*ctrl
, void *dlg
)
2592 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2593 struct perctrl_privdata tmp
, *p
;
2595 p
= find234(dp
->privdata
, &tmp
, NULL
);
2602 void dlg_set_privdata(union control
*ctrl
, void *dlg
, void *ptr
)
2604 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2605 struct perctrl_privdata tmp
, *p
;
2607 p
= find234(dp
->privdata
, &tmp
, NULL
);
2609 p
= snew(struct perctrl_privdata
);
2611 p
->needs_free
= FALSE
;
2612 add234(dp
->privdata
, p
);
2617 void *dlg_alloc_privdata(union control
*ctrl
, void *dlg
, size_t size
)
2619 struct dlgparam
*dp
= (struct dlgparam
*)dlg
;
2620 struct perctrl_privdata tmp
, *p
;
2622 p
= find234(dp
->privdata
, &tmp
, NULL
);
2624 p
= snew(struct perctrl_privdata
);
2626 p
->needs_free
= FALSE
;
2627 add234(dp
->privdata
, p
);
2629 assert(!p
->needs_free
);
2630 p
->needs_free
= TRUE
;
2632 * This is an internal allocation routine, so it's allowed to
2633 * use smalloc directly.
2635 p
->data
= smalloc(size
);