source: mainline/uspace/lib/ui/src/slider.c@ 453f9645

ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 453f9645 was 0eca788, checked in by Jiri Svoboda <jiri@…>, 3 years ago

Slider does not have a caption

  • Property mode set to 100644
File size: 12.5 KB
Line 
1/*
2 * Copyright (c) 2022 Jiri Svoboda
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - 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 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup libui
30 * @{
31 */
32/**
33 * @file Slider
34 */
35
36#include <errno.h>
37#include <gfx/color.h>
38#include <gfx/context.h>
39#include <gfx/render.h>
40#include <gfx/text.h>
41#include <io/pos_event.h>
42#include <stdlib.h>
43#include <str.h>
44#include <ui/control.h>
45#include <ui/paint.h>
46#include <ui/slider.h>
47#include "../private/resource.h"
48#include "../private/slider.h"
49
50/*
51 * The kind reader will appreciate that slider button dimensions 23:15
52 * are chosen such that, after subtracting the frame width (2 times 1),
53 * we get 21:13, which is a good approximation of the golden ratio.
54 */
55enum {
56 /** Slider button width */
57 ui_slider_btn_w = 15,
58 /** Slider button height */
59 ui_slider_btn_h = 23,
60 /** Slider button width in text mode */
61 ui_slider_btn_w_text = 3,
62 /** Slider button height in text mode */
63 ui_slider_btn_h_text = 1,
64 /** Slider button frame thickness */
65 ui_slider_btn_frame_thickness = 1,
66 /** Slider button bevel width */
67 ui_slider_btn_bevel_width = 2,
68 /** Slider groove width (total) */
69 ui_slider_groove_width = 4,
70 /** Amount by which slider groove bevel extendeds on each side
71 * beyond nominal groove length.
72 */
73 ui_slider_groove_margin = 2
74};
75
76static void ui_slider_ctl_destroy(void *);
77static errno_t ui_slider_ctl_paint(void *);
78static ui_evclaim_t ui_slider_ctl_pos_event(void *, pos_event_t *);
79
80/** Slider control ops */
81ui_control_ops_t ui_slider_ops = {
82 .destroy = ui_slider_ctl_destroy,
83 .paint = ui_slider_ctl_paint,
84 .pos_event = ui_slider_ctl_pos_event
85};
86
87/** Create new slider.
88 *
89 * @param resource UI resource
90 * @param rslider Place to store pointer to new slider
91 * @return EOK on success, ENOMEM if out of memory
92 */
93errno_t ui_slider_create(ui_resource_t *resource, ui_slider_t **rslider)
94{
95 ui_slider_t *slider;
96 errno_t rc;
97
98 slider = calloc(1, sizeof(ui_slider_t));
99 if (slider == NULL)
100 return ENOMEM;
101
102 rc = ui_control_new(&ui_slider_ops, (void *) slider,
103 &slider->control);
104 if (rc != EOK) {
105 free(slider);
106 return rc;
107 }
108
109 slider->res = resource;
110 *rslider = slider;
111 return EOK;
112}
113
114/** Destroy slider.
115 *
116 * @param slider Slider or @c NULL
117 */
118void ui_slider_destroy(ui_slider_t *slider)
119{
120 if (slider == NULL)
121 return;
122
123 ui_control_delete(slider->control);
124 free(slider);
125}
126
127/** Get base control from slider.
128 *
129 * @param slider Slider
130 * @return Control
131 */
132ui_control_t *ui_slider_ctl(ui_slider_t *slider)
133{
134 return slider->control;
135}
136
137/** Set slider callbacks.
138 *
139 * @param slider Slider
140 * @param cb Slider callbacks
141 * @param arg Callback argument
142 */
143void ui_slider_set_cb(ui_slider_t *slider, ui_slider_cb_t *cb, void *arg)
144{
145 slider->cb = cb;
146 slider->arg = arg;
147}
148
149/** Set slider rectangle.
150 *
151 * @param slider Slider
152 * @param rect New slider rectangle
153 */
154void ui_slider_set_rect(ui_slider_t *slider, gfx_rect_t *rect)
155{
156 slider->rect = *rect;
157}
158
159/** Paint outer slider frame.
160 *
161 * @param slider Slider
162 * @return EOK on success or an error code
163 */
164static errno_t ui_slider_paint_frame(ui_resource_t *res, gfx_rect_t *rect,
165 gfx_coord_t thickness, gfx_rect_t *inside)
166{
167 gfx_rect_t drect;
168 errno_t rc;
169
170 rc = gfx_set_color(res->gc, res->btn_frame_color);
171 if (rc != EOK)
172 goto error;
173
174 drect.p0.x = rect->p0.x + 1;
175 drect.p0.y = rect->p0.y;
176 drect.p1.x = rect->p1.x - 1;
177 drect.p1.y = rect->p0.y + thickness;
178 rc = gfx_fill_rect(res->gc, &drect);
179 if (rc != EOK)
180 goto error;
181
182 drect.p0.x = rect->p0.x + 1;
183 drect.p0.y = rect->p1.y - thickness;
184 drect.p1.x = rect->p1.x - 1;
185 drect.p1.y = rect->p1.y;
186 rc = gfx_fill_rect(res->gc, &drect);
187 if (rc != EOK)
188 goto error;
189
190 drect.p0.x = rect->p0.x;
191 drect.p0.y = rect->p0.y + 1;
192 drect.p1.x = rect->p0.x + thickness;
193 drect.p1.y = rect->p1.y - 1;
194 rc = gfx_fill_rect(res->gc, &drect);
195 if (rc != EOK)
196 goto error;
197
198 drect.p0.x = rect->p1.x - thickness;
199 drect.p0.y = rect->p0.y + 1;
200 drect.p1.x = rect->p1.x;
201 drect.p1.y = rect->p1.y - 1;
202 rc = gfx_fill_rect(res->gc, &drect);
203 if (rc != EOK)
204 goto error;
205
206 if (inside != NULL) {
207 inside->p0.x = rect->p0.x + thickness;
208 inside->p0.y = rect->p0.y + thickness;
209 inside->p1.x = rect->p1.x - thickness;
210 inside->p1.y = rect->p1.y - thickness;
211 }
212
213 return EOK;
214error:
215 return rc;
216}
217
218/** Paint outset slider bevel.
219 *
220 * @param slider Slider
221 * @return EOK on success or an error code
222 */
223static errno_t ui_slider_paint_outset(ui_slider_t *slider,
224 gfx_rect_t *rect, gfx_rect_t *inside)
225{
226 return ui_paint_bevel(slider->res->gc, rect,
227 slider->res->btn_highlight_color,
228 slider->res->btn_shadow_color, ui_slider_btn_bevel_width, inside);
229}
230
231/** Determine slider button rectagle.
232 *
233 * @param slider Slider
234 */
235static void ui_slider_btn_rect(ui_slider_t *slider, gfx_rect_t *rect)
236{
237 gfx_coord2_t pos;
238
239 pos.x = slider->rect.p0.x + slider->pos;
240 pos.y = slider->rect.p0.y;
241
242 rect->p0.x = pos.x;
243 rect->p0.y = pos.y;
244
245 if (slider->res->textmode) {
246 rect->p1.x = pos.x + ui_slider_btn_w_text;
247 rect->p1.y = pos.y + ui_slider_btn_h_text;
248 } else {
249 rect->p1.x = pos.x + ui_slider_btn_w;
250 rect->p1.y = pos.y + ui_slider_btn_h;
251 }
252}
253
254/** Determine slider length.
255 *
256 * @param slider Slider
257 * @return Slider length in pixels
258 */
259gfx_coord_t ui_slider_length(ui_slider_t *slider)
260{
261 gfx_coord2_t dims;
262 gfx_coord_t w;
263
264 gfx_rect_dims(&slider->rect, &dims);
265 w = slider->res->textmode ? ui_slider_btn_w_text :
266 ui_slider_btn_w;
267 return dims.x - w;
268}
269
270/** Paint slider in graphics mode.
271 *
272 * @param slider Slider
273 * @return EOK on success or an error code
274 */
275errno_t ui_slider_paint_gfx(ui_slider_t *slider)
276{
277 gfx_coord2_t pos;
278 gfx_coord_t length;
279 gfx_rect_t rect;
280 gfx_rect_t brect;
281 gfx_rect_t irect;
282 errno_t rc;
283
284 /* Paint slider groove */
285
286 pos = slider->rect.p0;
287 length = ui_slider_length(slider);
288
289 rect.p0.x = pos.x + ui_slider_btn_w / 2 - ui_slider_groove_margin;
290 rect.p0.y = pos.y + ui_slider_btn_h / 2 - ui_slider_groove_width / 2;
291 rect.p1.x = pos.x + ui_slider_btn_w / 2 + length +
292 ui_slider_groove_margin;
293 rect.p1.y = pos.y + ui_slider_btn_h / 2 + ui_slider_groove_width / 2;
294
295 rc = ui_paint_inset_frame(slider->res, &rect, NULL);
296 if (rc != EOK)
297 goto error;
298
299 /* Paint slider button */
300
301 ui_slider_btn_rect(slider, &rect);
302
303 rc = ui_slider_paint_frame(slider->res, &rect,
304 ui_slider_btn_frame_thickness, &brect);
305 if (rc != EOK)
306 goto error;
307
308 rc = ui_slider_paint_outset(slider, &brect, &irect);
309 if (rc != EOK)
310 goto error;
311
312 rc = gfx_set_color(slider->res->gc, slider->res->btn_face_color);
313 if (rc != EOK)
314 goto error;
315
316 rc = gfx_fill_rect(slider->res->gc, &irect);
317 if (rc != EOK)
318 goto error;
319
320 rc = gfx_update(slider->res->gc);
321 if (rc != EOK)
322 goto error;
323
324 return EOK;
325error:
326 return rc;
327}
328
329/** Paint slider in text mode.
330 *
331 * @param slider Slider
332 * @return EOK on success or an error code
333 */
334errno_t ui_slider_paint_text(ui_slider_t *slider)
335{
336 gfx_coord2_t pos;
337 gfx_text_fmt_t fmt;
338 gfx_coord_t w, i;
339 char *buf;
340 const char *gchar;
341 size_t gcharsz;
342 errno_t rc;
343
344 /* Paint slider groove */
345
346 pos = slider->rect.p0;
347
348 gfx_text_fmt_init(&fmt);
349 fmt.font = slider->res->font;
350 fmt.color = slider->res->wnd_text_color;
351 fmt.halign = gfx_halign_left;
352 fmt.valign = gfx_valign_top;
353
354 w = slider->rect.p1.x - slider->rect.p0.x;
355 gchar = "\u2550";
356 gcharsz = str_size(gchar);
357
358 buf = malloc(w * gcharsz + 1);
359 if (buf == NULL)
360 return ENOMEM;
361
362 for (i = 0; i < w; i++)
363 str_cpy(buf + i * gcharsz, (w - i) * gcharsz + 1, gchar);
364 buf[w * gcharsz] = '\0';
365
366 rc = gfx_puttext(&pos, &fmt, buf);
367 free(buf);
368 if (rc != EOK)
369 goto error;
370
371 /* Paint slider button */
372
373 pos.x += slider->pos;
374
375 rc = gfx_puttext(&pos, &fmt, "[O]");
376 if (rc != EOK)
377 goto error;
378
379 rc = gfx_update(slider->res->gc);
380 if (rc != EOK)
381 goto error;
382
383 return EOK;
384error:
385 return rc;
386}
387
388/** Paint slider.
389 *
390 * @param slider Slider
391 * @return EOK on success or an error code
392 */
393errno_t ui_slider_paint(ui_slider_t *slider)
394{
395 if (slider->res->textmode)
396 return ui_slider_paint_text(slider);
397 else
398 return ui_slider_paint_gfx(slider);
399}
400
401/** Clear slider button.
402 *
403 * @param slider Slider
404 * @return EOK on success or an error code
405 */
406errno_t ui_slider_btn_clear(ui_slider_t *slider)
407{
408 gfx_rect_t rect;
409 errno_t rc;
410
411 /* No need to clear button in text mode */
412 if (slider->res->textmode)
413 return EOK;
414
415 ui_slider_btn_rect(slider, &rect);
416
417 rc = gfx_set_color(slider->res->gc, slider->res->wnd_face_color);
418 if (rc != EOK)
419 goto error;
420
421 rc = gfx_fill_rect(slider->res->gc, &rect);
422 if (rc != EOK)
423 goto error;
424
425 return EOK;
426error:
427 return rc;
428}
429
430/** Press down slider.
431 *
432 * @param slider Slider
433 * @param pos Pointer position
434 */
435void ui_slider_press(ui_slider_t *slider, gfx_coord2_t *pos)
436{
437 if (slider->held)
438 return;
439
440 slider->held = true;
441 slider->press_pos = *pos;
442 slider->last_pos = slider->pos;
443
444 (void) ui_slider_paint(slider);
445}
446
447/** Release slider.
448 *
449 * @param slider Slider
450 * @param pos Pointer position
451 */
452void ui_slider_release(ui_slider_t *slider, gfx_coord2_t *pos)
453{
454 if (!slider->held)
455 return;
456
457 ui_slider_update(slider, pos);
458 slider->held = false;
459}
460
461/** Pointer moved.
462 *
463 * @param slider Slider
464 * @param pos New pointer position
465 */
466void ui_slider_update(ui_slider_t *slider, gfx_coord2_t *pos)
467{
468 gfx_coord_t spos;
469 gfx_coord_t length;
470
471 if (slider->held) {
472 spos = slider->last_pos + pos->x - slider->press_pos.x;
473 length = ui_slider_length(slider);
474 if (spos < 0)
475 spos = 0;
476 if (spos > length)
477 spos = length;
478
479 if (spos != slider->pos) {
480 (void) ui_slider_btn_clear(slider);
481 slider->pos = spos;
482 (void) ui_slider_paint(slider);
483 ui_slider_moved(slider, spos);
484 }
485 }
486}
487
488/** Slider was moved.
489 *
490 * @param slider Slider
491 */
492void ui_slider_moved(ui_slider_t *slider, gfx_coord_t pos)
493{
494 if (slider->cb != NULL && slider->cb->moved != NULL)
495 slider->cb->moved(slider, slider->arg, pos);
496}
497
498/** Handle slider position event.
499 *
500 * @param slider Slider
501 * @param pos_event Position event
502 * @return @c ui_claimed iff the event is claimed
503 */
504ui_evclaim_t ui_slider_pos_event(ui_slider_t *slider, pos_event_t *event)
505{
506 gfx_coord2_t pos;
507 gfx_rect_t rect;
508
509 pos.x = event->hpos;
510 pos.y = event->vpos;
511
512 switch (event->type) {
513 case POS_PRESS:
514 ui_slider_btn_rect(slider, &rect);
515 if (gfx_pix_inside_rect(&pos, &rect)) {
516 ui_slider_press(slider, &pos);
517 slider->press_pos = pos;
518 return ui_claimed;
519 }
520 break;
521 case POS_RELEASE:
522 if (slider->held) {
523 ui_slider_release(slider, &pos);
524 return ui_claimed;
525 }
526 break;
527 case POS_UPDATE:
528 ui_slider_update(slider, &pos);
529 break;
530 case POS_DCLICK:
531 break;
532 }
533
534 return ui_unclaimed;
535}
536
537/** Destroy slider control.
538 *
539 * @param arg Argument (ui_slider_t *)
540 */
541void ui_slider_ctl_destroy(void *arg)
542{
543 ui_slider_t *slider = (ui_slider_t *) arg;
544
545 ui_slider_destroy(slider);
546}
547
548/** Paint slider control.
549 *
550 * @param arg Argument (ui_slider_t *)
551 * @return EOK on success or an error code
552 */
553errno_t ui_slider_ctl_paint(void *arg)
554{
555 ui_slider_t *slider = (ui_slider_t *) arg;
556
557 return ui_slider_paint(slider);
558}
559
560/** Handle slider control position event.
561 *
562 * @param arg Argument (ui_slider_t *)
563 * @param pos_event Position event
564 * @return @c ui_claimed iff the event is claimed
565 */
566ui_evclaim_t ui_slider_ctl_pos_event(void *arg, pos_event_t *event)
567{
568 ui_slider_t *slider = (ui_slider_t *) arg;
569
570 return ui_slider_pos_event(slider, event);
571}
572
573/** @}
574 */
Note: See TracBrowser for help on using the repository browser.