xref: /freebsd/contrib/bsddialog/lib/lib_util.h (revision 3f0efe05432b1633991114ca4ca330102a561959)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2021-2024 Alfonso Sabato Siciliano
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25  * SUCH DAMAGE.
26  */
27 
28 #ifndef _LIBBSDDIALOG_UTIL_H_
29 #define _LIBBSDDIALOG_UTIL_H_
30 
31 #define BORDER          1
32 #define BORDERS         (BORDER + BORDER)
33 #define TEXTHMARGIN     1
34 #define TEXTHMARGINS    (TEXTHMARGIN + TEXTHMARGIN)
35 #define HBUTTONS        2
36 #define OK_LABEL        "OK"
37 #define CANCEL_LABEL    "Cancel"
38 
39 /* theme util */
40 extern struct bsddialog_theme t;
41 extern bool hastermcolors;
42 
43 #define	MIN(a,b) (((a)<(b))?(a):(b))
44 #define	MAX(a,b) (((a)>(b))?(a):(b))
45 /* debug */
46 #define BSDDIALOG_DEBUG(y,x,fmt, ...) do {                                     \
47 	mvprintw(y, x, fmt, __VA_ARGS__);                                      \
48 	refresh();                                                             \
49 } while (0)
50 /* error and diagnostic */
51 #define RETURN_ERROR(str) do {                                                 \
52 	set_error_string(str);                                                 \
53 	return (BSDDIALOG_ERROR);                                              \
54 } while (0)
55 #define RETURN_FMTERROR(fmt, ...) do {                                         \
56 	set_fmt_error_string(fmt, __VA_ARGS__);                                \
57 	return (BSDDIALOG_ERROR);                                              \
58 } while (0)
59 /* check ptr */
60 #define CHECK_PTR(p) do {                                                      \
61 	if (p == NULL)                                                         \
62 		RETURN_ERROR("*" #p " is NULL");                               \
63 } while (0)
64 #define CHECK_ARRAY(nitem, a) do {                                             \
65 	if (nitem > 0 && a == NULL)                                             \
66 		RETURN_FMTERROR(#nitem " is %d but *" #a " is NULL", nitem);   \
67 } while (0)
68 /* widget utils */
69 #define KEY_CTRL(c) (c & 037)
70 #define TEXTPAD(d, downnotext) rtextpad(d, 0, 0, 0, downnotext)
71 #define SCREENLINES (getmaxy(stdscr))
72 #define SCREENCOLS  (getmaxx(stdscr))
73 #define CHECK_STR(s) (s == NULL ? "" : s)
74 #define UARROW(c) (c->ascii_lines ? '^' : ACS_UARROW)
75 #define DARROW(c) (c->ascii_lines ? 'v' : ACS_DARROW)
76 #define LARROW(c) (c->ascii_lines ? '<' : ACS_LARROW)
77 #define RARROW(c) (c->ascii_lines ? '>' : ACS_RARROW)
78 #define DRAW_BUTTONS(d) do {                                                   \
79 	draw_buttons(&d);                                                      \
80 	wnoutrefresh(d.widget);                                                \
81 } while (0)
82 
83 /* internal types */
84 enum elevation { RAISED, LOWERED };
85 
86 struct buttons {
87 	unsigned int nbuttons;
88 #define MAXBUTTONS 10 /* 3left + ok + extra + cancel + help + 3 right */
89 	const char *label[MAXBUTTONS];
90 	bool shortcut;
91 	wchar_t first[MAXBUTTONS];
92 	int value[MAXBUTTONS];
93 	int curr;
94 #define BUTTONVALUE(bs) bs.value[bs.curr]
95 	unsigned int sizebutton; /* including left and right delimiters */
96 };
97 
98 struct dialog {
99 	bool built;         /* true after the first draw_dialog() */
100 	struct bsddialog_conf *conf;  /* Checked API conf */
101 	WINDOW *widget;     /* Size and position refer to widget */
102 	int y, x;           /* Current position, API conf.[y|x]: -1, >=0 */
103 	int rows, cols;     /* API rows and cols: -1, 0, >0 */
104 	int h, w;           /* Current height and width */
105 	const char *text;   /* Checked API text, at least "" */
106 	WINDOW *textpad;    /* Fake for textbox */
107 	struct buttons bs;  /* bs.nbuttons = 0 for no buttons */
108 	WINDOW *shadow;
109 };
110 
111 /* error and diagnostic */
112 const char *get_error_string(void);
113 void set_error_string(const char *string);
114 void set_fmt_error_string(const char *fmt, ...);
115 
116 /* multicolumn character string */
117 unsigned int strcols(const char *mbstring);
118 int str_props(const char *mbstring, unsigned int *cols, bool *has_multi_col);
119 void mvwaddwch(WINDOW *w, int y, int x, wchar_t wch);
120 wchar_t* alloc_mbstows(const char *mbstring);
121 
122 /* buttons */
123 void
124 set_buttons(struct dialog *d, bool shortcut, const char *oklabel,
125     const char *canclabel);
126 void draw_buttons(struct dialog *d);
127 bool shortcut_buttons(wint_t key, struct buttons *bs);
128 
129 /* widget utils */
130 int hide_dialog(struct dialog *d);
131 int f1help_dialog(struct bsddialog_conf *conf);
132 
133 void
134 draw_borders(struct bsddialog_conf *conf, WINDOW *win, enum elevation elev);
135 
136 void
137 update_box(struct bsddialog_conf *conf, WINDOW *win, int y, int x, int h, int w,
138     enum elevation elev);
139 
140 void
141 rtextpad(struct dialog *d, int ytext, int xtext, int upnotext, int downnotext);
142 
143 /* (auto) sizing and (auto) position */
144 int
145 set_widget_size(struct bsddialog_conf *conf, int rows, int cols, int *h,
146     int *w);
147 
148 int
149 set_widget_autosize(struct bsddialog_conf *conf, int rows, int cols, int *h,
150     int *w, const char *text, int *rowstext, struct buttons *bs, int hnotext,
151     int minw);
152 
153 int widget_checksize(int h, int w, struct buttons *bs, int hnotext, int minw);
154 
155 int
156 set_widget_position(struct bsddialog_conf *conf, int *y, int *x, int h, int w);
157 
158 int dialog_size_position(struct dialog *d, int hnotext, int minw, int *htext);
159 
160 /* dialog */
161 void end_dialog(struct dialog *d);
162 int draw_dialog(struct dialog *d);
163 
164 int
165 prepare_dialog(struct bsddialog_conf *conf, const char *text, int rows,
166     int cols, struct dialog *d);
167 
168 #endif
169