mirror of
https://github.com/lvgl/lvgl.git
synced 2026-05-11 05:37:37 +08:00
d76a346376
Co-authored-by: Gabor Kiss-Vamosi <kisvegabor@gmail.com>
90 lines
3.2 KiB
C
90 lines
3.2 KiB
C
#include "../lv_examples.h"
|
|
#if LV_BUILD_EXAMPLES && LV_USE_FLEX
|
|
|
|
static void scroll_event_cb(lv_event_t * e)
|
|
{
|
|
lv_obj_t * cont = lv_event_get_target_obj(e);
|
|
|
|
lv_area_t cont_a;
|
|
lv_obj_get_coords(cont, &cont_a);
|
|
int32_t cont_y_center = cont_a.y1 + lv_area_get_height(&cont_a) / 2;
|
|
|
|
int32_t r = lv_obj_get_height(cont) * 7 / 10;
|
|
int32_t i;
|
|
int32_t child_cnt = (int32_t)lv_obj_get_child_count(cont);
|
|
for(i = 0; i < child_cnt; i++) {
|
|
lv_obj_t * child = lv_obj_get_child(cont, i);
|
|
lv_area_t child_a;
|
|
lv_obj_get_coords(child, &child_a);
|
|
|
|
int32_t child_y_center = child_a.y1 + lv_area_get_height(&child_a) / 2;
|
|
|
|
int32_t diff_y = child_y_center - cont_y_center;
|
|
diff_y = LV_ABS(diff_y);
|
|
|
|
/*Get the x of diff_y on a circle.*/
|
|
int32_t x;
|
|
/*If diff_y is out of the circle use the last point of the circle (the radius)*/
|
|
if(diff_y >= r) {
|
|
x = r;
|
|
}
|
|
else {
|
|
/*Use Pythagoras theorem to get x from radius and y*/
|
|
uint32_t x_sqr = r * r - diff_y * diff_y;
|
|
lv_sqrt_res_t res;
|
|
lv_sqrt(x_sqr, &res, 0x8000); /*Use lvgl's built in sqrt root function*/
|
|
x = r - res.i;
|
|
}
|
|
|
|
/*Translate the item by the calculated X coordinate*/
|
|
lv_obj_set_style_translate_x(child, x, 0);
|
|
|
|
/*Use some opacity with larger translations*/
|
|
lv_opa_t opa = (lv_opa_t)lv_map(x, 0, r, LV_OPA_TRANSP, LV_OPA_COVER);
|
|
lv_obj_set_style_opa(child, LV_OPA_COVER - opa, 0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @title Circular scroll translate effect
|
|
* @brief Bow a column of buttons along a circle while they scroll past center.
|
|
*
|
|
* A 200x200 container is given `LV_RADIUS_CIRCLE`, `clip_corner`,
|
|
* `LV_FLEX_FLOW_COLUMN`, `LV_SCROLL_SNAP_CENTER` on Y, and
|
|
* `LV_SCROLLBAR_MODE_OFF`. Twenty full-width buttons are scrolled vertically.
|
|
* An `LV_EVENT_SCROLL` callback computes each child's vertical offset from
|
|
* the container center, maps it onto a circle with radius 7/10 of the height
|
|
* via `lv_sqrt`, writes the result to `translate_x`, and fades children with
|
|
* larger offsets toward `LV_OPA_TRANSP`.
|
|
*/
|
|
void lv_example_scroll_6(void)
|
|
{
|
|
lv_obj_t * cont = lv_obj_create(lv_screen_active());
|
|
lv_obj_set_size(cont, 200, 200);
|
|
lv_obj_center(cont);
|
|
lv_obj_set_flex_flow(cont, LV_FLEX_FLOW_COLUMN);
|
|
lv_obj_add_event_cb(cont, scroll_event_cb, LV_EVENT_SCROLL, NULL);
|
|
lv_obj_set_style_radius(cont, LV_RADIUS_CIRCLE, 0);
|
|
lv_obj_set_style_clip_corner(cont, true, 0);
|
|
lv_obj_set_scroll_dir(cont, LV_DIR_VER);
|
|
lv_obj_set_scroll_snap_y(cont, LV_SCROLL_SNAP_CENTER);
|
|
lv_obj_set_scrollbar_mode(cont, LV_SCROLLBAR_MODE_OFF);
|
|
|
|
uint32_t i;
|
|
for(i = 0; i < 20; i++) {
|
|
lv_obj_t * btn = lv_button_create(cont);
|
|
lv_obj_set_width(btn, lv_pct(100));
|
|
|
|
lv_obj_t * label = lv_label_create(btn);
|
|
lv_label_set_text_fmt(label, "Button %" LV_PRIu32, i);
|
|
}
|
|
|
|
/*Update the buttons position manually for first*/
|
|
lv_obj_send_event(cont, LV_EVENT_SCROLL, NULL);
|
|
|
|
/*Be sure the fist button is in the middle*/
|
|
lv_obj_scroll_to_view(lv_obj_get_child(cont, 0), LV_ANIM_OFF);
|
|
}
|
|
|
|
#endif
|