696 lines
21 KiB
C
696 lines
21 KiB
C
/*
|
|
* File : mouse.c
|
|
* This file is part of RT-Thread GUI Engine
|
|
* COPYRIGHT (C) 2006 - 2017, RT-Thread Development Team
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2009-10-16 Bernard first version
|
|
*/
|
|
|
|
#include "mouse.h"
|
|
#include <rtgui/region.h>
|
|
#include <rtgui/driver.h>
|
|
#include <rtgui/rtgui_system.h>
|
|
|
|
struct rtgui_cursor
|
|
{
|
|
/* screen byte per pixel */
|
|
rt_uint16_t bpp;
|
|
|
|
/* screen pitch */
|
|
rt_uint16_t screen_pitch;
|
|
|
|
/* current cursor x and y */
|
|
rt_uint16_t cx, cy;
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
/* cursor pitch */
|
|
rt_uint16_t cursor_pitch;
|
|
|
|
/* show cursor and show cursor count */
|
|
rt_bool_t show_cursor;
|
|
rt_base_t show_cursor_count;
|
|
|
|
/* cursor rect info */
|
|
rtgui_rect_t rect;
|
|
|
|
/* cursor image and saved cursor */
|
|
rtgui_image_t *cursor_image;
|
|
rt_uint8_t *cursor_saved;
|
|
#endif
|
|
|
|
#ifdef RTGUI_USING_WINMOVE
|
|
/* move window rect and border */
|
|
struct rtgui_win *win;
|
|
rtgui_rect_t win_rect;
|
|
rt_uint8_t *win_left, *win_right;
|
|
rt_uint8_t *win_top, *win_bottom;
|
|
rt_bool_t win_rect_show, win_rect_has_saved;
|
|
#endif
|
|
};
|
|
|
|
struct rtgui_cursor *_rtgui_cursor;
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
struct rt_mutex cursor_mutex;
|
|
static const rt_uint8_t *cursor_xpm[] =
|
|
{
|
|
"16 16 35 1",
|
|
" c None",
|
|
". c #A0B8D0",
|
|
"+ c #F0F0F0",
|
|
"@ c #FFFFFF",
|
|
"# c #F0F8F0",
|
|
"$ c #A0B0D0",
|
|
"% c #90A8C0",
|
|
"& c #A0B0C0",
|
|
"* c #E0E8F0",
|
|
"= c #8090B0",
|
|
"- c #D0D8E0",
|
|
"; c #7080A0",
|
|
"> c #90A0B0",
|
|
", c #FFF8FF",
|
|
"' c #F0F8FF",
|
|
") c #607090",
|
|
"! c #8098B0",
|
|
"~ c #405060",
|
|
"{ c #405070",
|
|
"] c #506070",
|
|
"^ c #607080",
|
|
"/ c #708090",
|
|
"( c #7088A0",
|
|
"_ c #D0D0E0",
|
|
": c #607890",
|
|
"< c #C0D0E0",
|
|
"[ c #C0C8D0",
|
|
"} c #506880",
|
|
"| c #5F778F",
|
|
"1 c #D0D8F0",
|
|
"2 c #506080",
|
|
"3 c #C0C8E0",
|
|
"4 c #A0A8C0",
|
|
"5 c #405870",
|
|
"6 c #5F6F8F",
|
|
" . ",
|
|
" .. ",
|
|
" .+. ",
|
|
" .@#$ ",
|
|
" $@@+% ",
|
|
" &@@@*= ",
|
|
" %@@@@-; ",
|
|
" >@@,''-) ",
|
|
" !,''+)~{] ",
|
|
" ='-^*/ ",
|
|
" (_{:<[^ ",
|
|
" ;} |:12 ",
|
|
" / )345 ",
|
|
" 6}${ ",
|
|
" 5{ ",
|
|
" "
|
|
};
|
|
|
|
static void rtgui_cursor_restore(void);
|
|
static void rtgui_cursor_save(void);
|
|
static void rtgui_cursor_show(void);
|
|
#endif
|
|
|
|
#ifdef RTGUI_USING_WINMOVE
|
|
static void rtgui_winrect_restore(void);
|
|
static void rtgui_winrect_save(void);
|
|
static void rtgui_winrect_show(void);
|
|
#endif
|
|
|
|
#define WIN_MOVE_BORDER 4
|
|
void rtgui_mouse_init(void)
|
|
{
|
|
const struct rtgui_graphic_driver *gd = rtgui_graphic_driver_get_default();
|
|
|
|
if (_rtgui_cursor != RT_NULL) rtgui_mouse_fini();
|
|
|
|
_rtgui_cursor = (struct rtgui_cursor *) rtgui_malloc(sizeof(struct rtgui_cursor));
|
|
rt_memset(_rtgui_cursor, 0, sizeof(struct rtgui_cursor));
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
rt_mutex_init(&cursor_mutex, "cursor", RT_IPC_FLAG_FIFO);
|
|
#endif
|
|
|
|
/* init cursor */
|
|
_rtgui_cursor->bpp = _UI_BITBYTES(gd->bits_per_pixel);
|
|
_rtgui_cursor->screen_pitch = _rtgui_cursor->bpp * gd->width;
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
/* init cursor image */
|
|
_rtgui_cursor->cursor_image = rtgui_image_create_from_mem("xpm",
|
|
(rt_uint8_t *)cursor_xpm,
|
|
sizeof(cursor_xpm),
|
|
RT_TRUE);
|
|
if (_rtgui_cursor->cursor_image == RT_NULL)
|
|
{
|
|
rtgui_free(_rtgui_cursor);
|
|
_rtgui_cursor = RT_NULL;
|
|
|
|
return;
|
|
}
|
|
|
|
/* init rect */
|
|
_rtgui_cursor->rect.x1 = _rtgui_cursor->rect.y1 = 0;
|
|
_rtgui_cursor->rect.x2 = _rtgui_cursor->cursor_image->w;
|
|
_rtgui_cursor->rect.y2 = _rtgui_cursor->cursor_image->h;
|
|
|
|
_rtgui_cursor->cursor_pitch = _rtgui_cursor->cursor_image->w * _rtgui_cursor->bpp;
|
|
|
|
_rtgui_cursor->show_cursor = RT_TRUE;
|
|
_rtgui_cursor->show_cursor_count = 0;
|
|
_rtgui_cursor->cursor_saved = rtgui_malloc(_rtgui_cursor->cursor_image->w *
|
|
_rtgui_cursor->cursor_image->h * _rtgui_cursor->bpp);
|
|
#endif
|
|
|
|
#ifdef RTGUI_USING_WINMOVE
|
|
/* init window move save image */
|
|
_rtgui_cursor->win_rect_has_saved = RT_FALSE;
|
|
_rtgui_cursor->win_rect_show = RT_FALSE;
|
|
|
|
_rtgui_cursor->win_left = rtgui_malloc(_rtgui_cursor->bpp * gd->height * WIN_MOVE_BORDER);
|
|
_rtgui_cursor->win_right = rtgui_malloc(_rtgui_cursor->bpp * gd->height * WIN_MOVE_BORDER);
|
|
_rtgui_cursor->win_top = rtgui_malloc(_rtgui_cursor->bpp * gd->width * WIN_MOVE_BORDER);
|
|
_rtgui_cursor->win_bottom = rtgui_malloc(_rtgui_cursor->bpp * gd->width * WIN_MOVE_BORDER);
|
|
#endif
|
|
}
|
|
|
|
void rtgui_mouse_fini(void)
|
|
{
|
|
if (_rtgui_cursor != RT_NULL)
|
|
{
|
|
#ifdef RTGUI_USING_WINMOVE
|
|
rtgui_free(_rtgui_cursor->win_left);
|
|
rtgui_free(_rtgui_cursor->win_right);
|
|
rtgui_free(_rtgui_cursor->win_top);
|
|
rtgui_free(_rtgui_cursor->win_bottom);
|
|
#endif
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
rt_mutex_detach(&cursor_mutex);
|
|
rtgui_image_destroy(_rtgui_cursor->cursor_image);
|
|
rtgui_free(_rtgui_cursor->cursor_saved);
|
|
#endif
|
|
rtgui_free(_rtgui_cursor);
|
|
|
|
_rtgui_cursor = RT_NULL;
|
|
}
|
|
}
|
|
|
|
void rtgui_mouse_moveto(int x, int y)
|
|
{
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
rt_mutex_take(&cursor_mutex, RT_WAITING_FOREVER);
|
|
#endif
|
|
|
|
if (x != _rtgui_cursor->cx ||
|
|
y != _rtgui_cursor->cy)
|
|
{
|
|
#ifdef RTGUI_USING_WINMOVE
|
|
if (_rtgui_cursor->win_rect_show)
|
|
{
|
|
if (_rtgui_cursor->win_rect_has_saved == RT_TRUE)
|
|
{
|
|
rtgui_winrect_restore();
|
|
}
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
rtgui_mouse_hide_cursor();
|
|
#endif
|
|
|
|
/* move winrect */
|
|
rtgui_rect_moveto(&(_rtgui_cursor->win_rect), x - _rtgui_cursor->cx,
|
|
y - _rtgui_cursor->cy);
|
|
rtgui_winrect_save();
|
|
|
|
/* move current cursor */
|
|
_rtgui_cursor->cx = x;
|
|
_rtgui_cursor->cy = y;
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
/* show cursor */
|
|
rtgui_mouse_show_cursor();
|
|
#endif
|
|
/* show winrect */
|
|
rtgui_winrect_show();
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
rtgui_mouse_hide_cursor();
|
|
#endif
|
|
/* move current cursor */
|
|
_rtgui_cursor->cx = x;
|
|
_rtgui_cursor->cy = y;
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
/* show cursor */
|
|
rtgui_mouse_show_cursor();
|
|
#endif
|
|
}
|
|
|
|
#ifdef RTGUI_USING_HW_CURSOR
|
|
rtgui_cursor_set_position(_rtgui_cursor->cx, _rtgui_cursor->cy);
|
|
#endif
|
|
}
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
rt_mutex_release(&cursor_mutex);
|
|
#endif
|
|
}
|
|
|
|
void rtgui_mouse_set_position(int x, int y)
|
|
{
|
|
/* move current cursor */
|
|
_rtgui_cursor->cx = x;
|
|
_rtgui_cursor->cy = y;
|
|
|
|
#ifdef RTGUI_USING_HW_CURSOR
|
|
rtgui_cursor_set_position(_rtgui_cursor->cx, _rtgui_cursor->cy);
|
|
#endif
|
|
}
|
|
|
|
#ifdef RTGUI_USING_MOUSE_CURSOR
|
|
void rtgui_mouse_set_cursor_enable(rt_bool_t enable)
|
|
{
|
|
_rtgui_cursor->show_cursor = enable;
|
|
}
|
|
|
|
/* set current cursor image */
|
|
void rtgui_mouse_set_cursor(rtgui_image_t *cursor)
|
|
{
|
|
}
|
|
|
|
void rtgui_mouse_get_cursor_rect(rtgui_rect_t *rect)
|
|
{
|
|
if (rect != RT_NULL)
|
|
{
|
|
*rect = _rtgui_cursor->rect;
|
|
}
|
|
}
|
|
|
|
void rtgui_mouse_show_cursor()
|
|
{
|
|
if (_rtgui_cursor->show_cursor == RT_FALSE)
|
|
return;
|
|
|
|
_rtgui_cursor->show_cursor_count ++;
|
|
if (_rtgui_cursor->show_cursor_count == 1)
|
|
{
|
|
/* save show mouse area */
|
|
rtgui_cursor_save();
|
|
|
|
/* show mouse cursor */
|
|
rtgui_cursor_show();
|
|
}
|
|
}
|
|
|
|
void rtgui_mouse_hide_cursor()
|
|
{
|
|
if (_rtgui_cursor->show_cursor == RT_FALSE)
|
|
return;
|
|
|
|
if (_rtgui_cursor->show_cursor_count == 1)
|
|
{
|
|
/* display the cursor coverage area */
|
|
rtgui_cursor_restore();
|
|
}
|
|
_rtgui_cursor->show_cursor_count --;
|
|
}
|
|
|
|
rt_bool_t rtgui_mouse_is_intersect(rtgui_rect_t *r)
|
|
{
|
|
return rtgui_rect_is_intersect(&(_rtgui_cursor->rect), r) == RT_EOK ? RT_TRUE : RT_FALSE;
|
|
}
|
|
|
|
/* display the saved cursor area to screen */
|
|
static void rtgui_cursor_restore()
|
|
{
|
|
rt_base_t idx, height, cursor_pitch;
|
|
rt_uint8_t *cursor_ptr, *fb_ptr;
|
|
|
|
fb_ptr = rtgui_graphic_driver_get_framebuffer(RT_NULL) + _rtgui_cursor->cy * _rtgui_cursor->screen_pitch
|
|
+ _rtgui_cursor->cx * _rtgui_cursor->bpp;
|
|
cursor_ptr = _rtgui_cursor->cursor_saved;
|
|
|
|
height = (_rtgui_cursor->cy + _rtgui_cursor->cursor_image->h <
|
|
rtgui_graphic_driver_get_default()->height) ? _rtgui_cursor->cursor_image->h :
|
|
rtgui_graphic_driver_get_default()->height - _rtgui_cursor->cy;
|
|
|
|
cursor_pitch = (_rtgui_cursor->cx + _rtgui_cursor->cursor_image->w <
|
|
rtgui_graphic_driver_get_default()->width) ? _rtgui_cursor->cursor_pitch :
|
|
(rtgui_graphic_driver_get_default()->width - _rtgui_cursor->cx) * _rtgui_cursor->bpp;
|
|
|
|
for (idx = 0; idx < height; idx ++)
|
|
{
|
|
rt_memcpy(fb_ptr, cursor_ptr, cursor_pitch);
|
|
|
|
fb_ptr += _rtgui_cursor->screen_pitch;
|
|
cursor_ptr += _rtgui_cursor->cursor_pitch;
|
|
}
|
|
}
|
|
|
|
/* save the cursor coverage area from screen */
|
|
static void rtgui_cursor_save()
|
|
{
|
|
rt_base_t idx, height, cursor_pitch;
|
|
rt_uint8_t *cursor_ptr, *fb_ptr;
|
|
|
|
fb_ptr = rtgui_graphic_driver_get_framebuffer(RT_NULL) + _rtgui_cursor->cy * _rtgui_cursor->screen_pitch +
|
|
_rtgui_cursor->cx * _rtgui_cursor->bpp;
|
|
cursor_ptr = _rtgui_cursor->cursor_saved;
|
|
|
|
height = (_rtgui_cursor->cy + _rtgui_cursor->cursor_image->h <
|
|
rtgui_graphic_driver_get_default()->height) ? _rtgui_cursor->cursor_image->h :
|
|
rtgui_graphic_driver_get_default()->height - _rtgui_cursor->cy;
|
|
|
|
cursor_pitch = (_rtgui_cursor->cx + _rtgui_cursor->cursor_image->w <
|
|
rtgui_graphic_driver_get_default()->width) ? _rtgui_cursor->cursor_pitch :
|
|
(rtgui_graphic_driver_get_default()->width - _rtgui_cursor->cx) * _rtgui_cursor->bpp;
|
|
|
|
for (idx = 0; idx < height; idx ++)
|
|
{
|
|
rt_memcpy(cursor_ptr, fb_ptr, cursor_pitch);
|
|
|
|
fb_ptr += _rtgui_cursor->screen_pitch;
|
|
cursor_ptr += _rtgui_cursor->cursor_pitch;
|
|
}
|
|
}
|
|
|
|
static void rtgui_cursor_show()
|
|
{
|
|
// FIXME: the prototype of set_pixel is using int so we have to use int
|
|
// as well. Might be uniformed with others in the future
|
|
int x, y;
|
|
rtgui_color_t *ptr;
|
|
rtgui_rect_t rect;
|
|
void (*set_pixel)(rtgui_color_t * c, int x, int y);
|
|
|
|
ptr = (rtgui_color_t *) _rtgui_cursor->cursor_image->data;
|
|
set_pixel = rtgui_graphic_driver_get_default()->ops->set_pixel;
|
|
|
|
rtgui_mouse_get_cursor_rect(&rect);
|
|
rtgui_rect_moveto(&rect, _rtgui_cursor->cx, _rtgui_cursor->cy);
|
|
|
|
/* draw each point */
|
|
for (y = rect.y1; y < rect.y2; y ++)
|
|
{
|
|
for (x = rect.x1; x < rect.x2; x++)
|
|
{
|
|
/* not alpha */
|
|
if ((*ptr >> 24) != 255)
|
|
{
|
|
set_pixel(ptr, x, y);
|
|
}
|
|
|
|
/* move to next color buffer */
|
|
ptr ++;
|
|
}
|
|
}
|
|
|
|
/* update rect */
|
|
rtgui_graphic_driver_screen_update(rtgui_graphic_driver_get_default(), &rect);
|
|
}
|
|
#endif
|
|
|
|
#ifdef RTGUI_USING_WINMOVE
|
|
void rtgui_winrect_set(struct rtgui_win *win)
|
|
{
|
|
/* set win rect show */
|
|
_rtgui_cursor->win_rect_show = RT_TRUE;
|
|
|
|
/* set win rect */
|
|
_rtgui_cursor->win_rect =
|
|
win->_title_wgt == RT_NULL ?
|
|
RTGUI_WIDGET(win)->extent :
|
|
RTGUI_WIDGET(win->_title_wgt)->extent;
|
|
|
|
_rtgui_cursor->win = win;
|
|
}
|
|
|
|
rt_bool_t rtgui_winrect_moved_done(rtgui_rect_t *winrect, struct rtgui_win **win)
|
|
{
|
|
rt_bool_t moved = RT_FALSE;
|
|
|
|
/* restore winrect */
|
|
if (_rtgui_cursor->win_rect_has_saved)
|
|
{
|
|
rtgui_winrect_restore();
|
|
|
|
moved = RT_TRUE;
|
|
}
|
|
|
|
/* clear win rect show */
|
|
_rtgui_cursor->win_rect_show = RT_FALSE;
|
|
_rtgui_cursor->win_rect_has_saved = RT_FALSE;
|
|
|
|
/* return win rect */
|
|
if (winrect)
|
|
*winrect = _rtgui_cursor->win_rect;
|
|
if (win)
|
|
*win = _rtgui_cursor->win;
|
|
|
|
return moved;
|
|
}
|
|
|
|
rt_bool_t rtgui_winrect_is_moved()
|
|
{
|
|
return _rtgui_cursor->win_rect_show;
|
|
}
|
|
|
|
/* show winrect */
|
|
static void rtgui_winrect_show()
|
|
{
|
|
rt_uint16_t x, y;
|
|
rtgui_color_t c;
|
|
rtgui_rect_t screen_rect, win_rect, win_rect_inner;
|
|
void (*set_pixel)(rtgui_color_t * c, int x, int y);
|
|
|
|
c = black;
|
|
set_pixel = rtgui_graphic_driver_get_default()->ops->set_pixel;
|
|
|
|
win_rect = _rtgui_cursor->win_rect;
|
|
win_rect_inner = win_rect;
|
|
rtgui_rect_inflate(&win_rect_inner, -WIN_MOVE_BORDER);
|
|
|
|
rtgui_graphic_driver_get_rect(rtgui_graphic_driver_get_default(),
|
|
&screen_rect);
|
|
rtgui_rect_intersect(&screen_rect, &win_rect);
|
|
rtgui_rect_intersect(&screen_rect, &win_rect_inner);
|
|
|
|
/* draw left */
|
|
for (y = win_rect.y1; y < win_rect.y2; y ++)
|
|
{
|
|
for (x = win_rect.x1; x < win_rect_inner.x1; x++)
|
|
if ((x + y) & 0x01) set_pixel(&c, x, y);
|
|
}
|
|
|
|
/* draw right */
|
|
for (y = win_rect.y1; y < win_rect.y2; y ++)
|
|
{
|
|
for (x = win_rect_inner.x2; x < win_rect.x2; x++)
|
|
if ((x + y) & 0x01) set_pixel(&c, x, y);
|
|
}
|
|
|
|
/* draw top border */
|
|
for (y = win_rect.y1; y < win_rect_inner.y1; y ++)
|
|
{
|
|
for (x = win_rect_inner.x1; x < win_rect_inner.x2; x++)
|
|
if ((x + y) & 0x01) set_pixel(&c, x, y);
|
|
}
|
|
|
|
/* draw bottom border */
|
|
for (y = win_rect_inner.y2; y < win_rect.y2; y ++)
|
|
{
|
|
for (x = win_rect_inner.x1; x < win_rect_inner.x2; x++)
|
|
if ((x + y) & 0x01) set_pixel(&c, x, y);
|
|
}
|
|
|
|
/* update rect */
|
|
rtgui_graphic_driver_screen_update(rtgui_graphic_driver_get_default(), &win_rect);
|
|
}
|
|
|
|
#define display_direct_memcpy(src, dest, src_pitch, dest_pitch, height, len) \
|
|
for (idx = 0; idx < height; idx ++) \
|
|
{ \
|
|
rt_memcpy(dest, src, len); \
|
|
src += src_pitch; \
|
|
dest += dest_pitch; \
|
|
}
|
|
|
|
static void rtgui_winrect_restore()
|
|
{
|
|
rt_uint8_t *winrect_ptr, *fb_ptr, *driver_fb;
|
|
int winrect_pitch, idx;
|
|
rtgui_rect_t screen_rect, win_rect;
|
|
|
|
driver_fb = rtgui_graphic_driver_get_framebuffer(RT_NULL);
|
|
win_rect = _rtgui_cursor->win_rect;
|
|
|
|
rtgui_graphic_driver_get_rect(rtgui_graphic_driver_get_default(),
|
|
&screen_rect);
|
|
rtgui_rect_intersect(&screen_rect, &win_rect);
|
|
|
|
/* restore winrect left */
|
|
fb_ptr = driver_fb + win_rect.y1 * _rtgui_cursor->screen_pitch +
|
|
win_rect.x1 * _rtgui_cursor->bpp;
|
|
winrect_ptr = _rtgui_cursor->win_left;
|
|
winrect_pitch = WIN_MOVE_BORDER * _rtgui_cursor->bpp;
|
|
display_direct_memcpy(winrect_ptr, fb_ptr, winrect_pitch, _rtgui_cursor->screen_pitch,
|
|
(win_rect.y2 - win_rect.y1), winrect_pitch);
|
|
|
|
/* restore winrect right */
|
|
fb_ptr = driver_fb + win_rect.y1 * _rtgui_cursor->screen_pitch +
|
|
(win_rect.x2 - WIN_MOVE_BORDER) * _rtgui_cursor->bpp;
|
|
winrect_ptr = _rtgui_cursor->win_right;
|
|
winrect_pitch = WIN_MOVE_BORDER * _rtgui_cursor->bpp;
|
|
display_direct_memcpy(winrect_ptr, fb_ptr, winrect_pitch, _rtgui_cursor->screen_pitch,
|
|
(win_rect.y2 - win_rect.y1), winrect_pitch);
|
|
|
|
/* restore winrect top */
|
|
fb_ptr = driver_fb + win_rect.y1 * _rtgui_cursor->screen_pitch +
|
|
(win_rect.x1 + WIN_MOVE_BORDER) * _rtgui_cursor->bpp;
|
|
winrect_ptr = _rtgui_cursor->win_top;
|
|
winrect_pitch = (win_rect.x2 - win_rect.x1 - 2 * WIN_MOVE_BORDER) * _rtgui_cursor->bpp;
|
|
display_direct_memcpy(winrect_ptr, fb_ptr, winrect_pitch, _rtgui_cursor->screen_pitch,
|
|
WIN_MOVE_BORDER, winrect_pitch);
|
|
|
|
/* restore winrect bottom */
|
|
fb_ptr = driver_fb + (win_rect.y2 - WIN_MOVE_BORDER) * _rtgui_cursor->screen_pitch +
|
|
(win_rect.x1 + WIN_MOVE_BORDER) * _rtgui_cursor->bpp;
|
|
winrect_ptr = _rtgui_cursor->win_bottom;
|
|
display_direct_memcpy(winrect_ptr, fb_ptr, winrect_pitch, _rtgui_cursor->screen_pitch,
|
|
WIN_MOVE_BORDER, winrect_pitch);
|
|
}
|
|
|
|
static void rtgui_winrect_save()
|
|
{
|
|
rt_uint8_t *winrect_ptr, *fb_ptr, *driver_fb;
|
|
int winrect_pitch, idx;
|
|
rtgui_rect_t screen_rect, win_rect;
|
|
|
|
driver_fb = rtgui_graphic_driver_get_framebuffer(RT_NULL);
|
|
win_rect = _rtgui_cursor->win_rect;
|
|
|
|
rtgui_graphic_driver_get_rect(rtgui_graphic_driver_get_default(),
|
|
&screen_rect);
|
|
rtgui_rect_intersect(&screen_rect, &win_rect);
|
|
|
|
/* set winrect has saved */
|
|
_rtgui_cursor->win_rect_has_saved = RT_TRUE;
|
|
|
|
/* save winrect left */
|
|
fb_ptr = driver_fb + win_rect.y1 * _rtgui_cursor->screen_pitch +
|
|
win_rect.x1 * _rtgui_cursor->bpp;
|
|
winrect_ptr = _rtgui_cursor->win_left;
|
|
winrect_pitch = WIN_MOVE_BORDER * _rtgui_cursor->bpp;
|
|
display_direct_memcpy(fb_ptr, winrect_ptr, _rtgui_cursor->screen_pitch, winrect_pitch,
|
|
(win_rect.y2 - win_rect.y1), winrect_pitch);
|
|
|
|
/* save winrect right */
|
|
fb_ptr = driver_fb + win_rect.y1 * _rtgui_cursor->screen_pitch +
|
|
(win_rect.x2 - WIN_MOVE_BORDER) * _rtgui_cursor->bpp;
|
|
winrect_ptr = _rtgui_cursor->win_right;
|
|
winrect_pitch = WIN_MOVE_BORDER * _rtgui_cursor->bpp;
|
|
display_direct_memcpy(fb_ptr, winrect_ptr, _rtgui_cursor->screen_pitch, winrect_pitch,
|
|
(win_rect.y2 - win_rect.y1), winrect_pitch);
|
|
|
|
/* save winrect top */
|
|
fb_ptr = driver_fb + win_rect.y1 * _rtgui_cursor->screen_pitch +
|
|
(win_rect.x1 + WIN_MOVE_BORDER) * _rtgui_cursor->bpp;
|
|
winrect_ptr = _rtgui_cursor->win_top;
|
|
winrect_pitch = (win_rect.x2 - win_rect.x1 - 2 * WIN_MOVE_BORDER) * _rtgui_cursor->bpp;
|
|
display_direct_memcpy(fb_ptr, winrect_ptr, _rtgui_cursor->screen_pitch, winrect_pitch,
|
|
WIN_MOVE_BORDER, winrect_pitch);
|
|
|
|
/* save winrect bottom */
|
|
fb_ptr = driver_fb + (win_rect.y2 - WIN_MOVE_BORDER) * _rtgui_cursor->screen_pitch +
|
|
(win_rect.x1 + WIN_MOVE_BORDER) * _rtgui_cursor->bpp;
|
|
winrect_ptr = _rtgui_cursor->win_bottom;
|
|
display_direct_memcpy(fb_ptr, winrect_ptr, _rtgui_cursor->screen_pitch, winrect_pitch,
|
|
WIN_MOVE_BORDER, winrect_pitch);
|
|
}
|
|
#endif
|
|
|
|
void rtgui_mouse_monitor_append(rtgui_list_t *head, rtgui_rect_t *rect)
|
|
{
|
|
struct rtgui_mouse_monitor *mmonitor;
|
|
|
|
/* check parameters */
|
|
if (head == RT_NULL || rect == RT_NULL) return;
|
|
|
|
/* create a mouse monitor node */
|
|
mmonitor = (struct rtgui_mouse_monitor *) rtgui_malloc(sizeof(struct rtgui_mouse_monitor));
|
|
if (mmonitor == RT_NULL) return; /* no memory */
|
|
|
|
/* set mouse monitor node */
|
|
mmonitor->rect = *rect;
|
|
rtgui_list_init(&(mmonitor->list));
|
|
|
|
/* append to list */
|
|
rtgui_list_append(head, &(mmonitor->list));
|
|
}
|
|
|
|
void rtgui_mouse_monitor_remove(rtgui_list_t *head, rtgui_rect_t *rect)
|
|
{
|
|
struct rtgui_list_node *node;
|
|
struct rtgui_mouse_monitor *mmonitor;
|
|
|
|
/* check parameters */
|
|
if (head == RT_NULL || rect == RT_NULL) return;
|
|
|
|
for (node = head->next; node != RT_NULL; node = node->next)
|
|
{
|
|
mmonitor = rtgui_list_entry(node, struct rtgui_mouse_monitor, list);
|
|
if (mmonitor->rect.x1 == rect->x1 &&
|
|
mmonitor->rect.x2 == rect->x2 &&
|
|
mmonitor->rect.y1 == rect->y1 &&
|
|
mmonitor->rect.y2 == rect->y2)
|
|
{
|
|
/* found node */
|
|
rtgui_list_remove(head, node);
|
|
rtgui_free(mmonitor);
|
|
|
|
return ;
|
|
}
|
|
}
|
|
}
|
|
|
|
rt_bool_t rtgui_mouse_monitor_contains_point(rtgui_list_t *head, int x, int y)
|
|
{
|
|
struct rtgui_list_node *node;
|
|
|
|
/* check parameter */
|
|
if (head == RT_NULL) return RT_FALSE;
|
|
|
|
rtgui_list_foreach(node, head)
|
|
{
|
|
struct rtgui_mouse_monitor *monitor = rtgui_list_entry(node,
|
|
struct rtgui_mouse_monitor, list);
|
|
|
|
if (rtgui_rect_contains_point(&(monitor->rect), x, y) == RT_EOK)
|
|
{
|
|
return RT_TRUE;
|
|
}
|
|
}
|
|
|
|
return RT_FALSE;
|
|
}
|