forked from Archive/PX4-Autopilot
Big NxWidgets simplification. Remove all hierarch logic. Widgets now exist only in a two-dimensional plane
git-svn-id: https://nuttx.svn.sourceforge.net/svnroot/nuttx/trunk@4712 7fd9a85b-ad96-42d3-883c-3090e2eb8679
This commit is contained in:
parent
f08f0709b3
commit
11961873cc
|
@ -35,3 +35,9 @@
|
||||||
window size is received.
|
window size is received.
|
||||||
* CGraphicsPort and CWidgetControl: If the underlying graphics device
|
* CGraphicsPort and CWidgetControl: If the underlying graphics device
|
||||||
is write-only, then we have to render fonts a little differently.
|
is write-only, then we have to render fonts a little differently.
|
||||||
|
* CNxWidgets, CWidgetControl, and CRectCache: Big change! Remove all support
|
||||||
|
for widgets in a "vertical" hierarchy. Now widgets exist in a flat,
|
||||||
|
two-dimensional space and should not overlap. This should greatly
|
||||||
|
reduce the memory requirements and, since, NuttX already supports
|
||||||
|
a hierarchical windowing system, does not result in loss of functionality.
|
||||||
|
|
||||||
|
|
|
@ -44,9 +44,9 @@ CSRCS =
|
||||||
CXXSRCS = cbitmap.cxx cbgwindow.cxx ccallback.cxx cgraphicsport.cxx
|
CXXSRCS = cbitmap.cxx cbgwindow.cxx ccallback.cxx cgraphicsport.cxx
|
||||||
CXXSRCS += clistdata.cxx clistdataitem.cxx cnxfont.cxx
|
CXXSRCS += clistdata.cxx clistdataitem.cxx cnxfont.cxx
|
||||||
CXXSRCS += cnxserver.cxx cnxstring.cxx cnxtimer.cxx cnxwidget.cxx cnxwindow.cxx
|
CXXSRCS += cnxserver.cxx cnxstring.cxx cnxtimer.cxx cnxwidget.cxx cnxwindow.cxx
|
||||||
CXXSRCS += cnxtkwindow.cxx cnxtoolbar.cxx crect.cxx crectcache.cxx
|
CXXSRCS += cnxtkwindow.cxx cnxtoolbar.cxx crect.cxx crlepalettebitmap.cxx
|
||||||
CXXSRCS += crlepalettebitmap.cxx cstringiterator.cxx ctext.cxx cwidgetcontrol.cxx
|
CXXSRCS += cstringiterator.cxx ctext.cxx cwidgetcontrol.cxx cwidgeteventhandlerlist.cxx
|
||||||
CXXSRCS += cwidgeteventhandlerlist.cxx singletons.cxx
|
CXXSRCS += singletons.cxx
|
||||||
# Widget APIs
|
# Widget APIs
|
||||||
CXXSRCS += cbutton.cxx cbuttonarray.cxx ccheckbox.cxx ccyclebutton.cxx
|
CXXSRCS += cbutton.cxx cbuttonarray.cxx ccheckbox.cxx ccyclebutton.cxx
|
||||||
CXXSRCS += cglyphbutton.cxx cimage.cxx ckeypad.cxx clabel.cxx clatchbutton.cxx
|
CXXSRCS += cglyphbutton.cxx cimage.cxx ckeypad.cxx clabel.cxx clatchbutton.cxx
|
||||||
|
|
|
@ -111,7 +111,6 @@ namespace NXWidgets
|
||||||
class CWidgetControl;
|
class CWidgetControl;
|
||||||
class CGraphicsPort;
|
class CGraphicsPort;
|
||||||
class CNxFont;
|
class CNxFont;
|
||||||
class CRectCache;
|
|
||||||
class CWidgetEventHandlerList;
|
class CWidgetEventHandlerList;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -206,9 +205,8 @@ namespace NXWidgets
|
||||||
CNxWidget *m_focusedChild; /**< Pointer to the child widget that has focus. */
|
CNxWidget *m_focusedChild; /**< Pointer to the child widget that has focus. */
|
||||||
TNxArray<CNxWidget*> m_children; /**< List of child widgets. */
|
TNxArray<CNxWidget*> m_children; /**< List of child widgets. */
|
||||||
|
|
||||||
// Visible regions
|
// Borders
|
||||||
|
|
||||||
CRectCache *m_rectCache; /**< List of the widget's visible regions. */
|
|
||||||
WidgetBorderSize m_borderSize; /**< Size of the widget borders. */
|
WidgetBorderSize m_borderSize; /**< Size of the widget borders. */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -237,18 +235,6 @@ namespace NXWidgets
|
||||||
|
|
||||||
virtual void drawBorder(CGraphicsPort* port) { }
|
virtual void drawBorder(CGraphicsPort* port) { }
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the supplied coordinates collide with a portion of this widget
|
|
||||||
* that is not obscured by its siblings, but that may be obscured by
|
|
||||||
* its children.
|
|
||||||
*
|
|
||||||
* @param x X coordinate of the click.
|
|
||||||
* @param y Y coordinate of the click.
|
|
||||||
* @return True if a collision occurred; false if not.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool checkCollisionWithForegroundRects(nxgl_coord_t x, nxgl_coord_t y) const;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Draw all visible regions of this widget's children.
|
* Draw all visible regions of this widget's children.
|
||||||
*/
|
*/
|
||||||
|
@ -265,34 +251,6 @@ namespace NXWidgets
|
||||||
|
|
||||||
void closeChild(CNxWidget *widget);
|
void closeChild(CNxWidget *widget);
|
||||||
|
|
||||||
/**
|
|
||||||
* Redraws all regions of child widgets that fall within the invalidRects
|
|
||||||
* regions.
|
|
||||||
*
|
|
||||||
* @param invalidRects List of invalid regions that need to be redrawn.
|
|
||||||
* @param sender Pointer to the widget that initiated the redraw.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void redrawDirtyChildren(TNxArray<CRect>* invalidRects, CNxWidget *sender);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the index of the next visible widget higher up the z-order.
|
|
||||||
*
|
|
||||||
* @param startIndex The starting index.
|
|
||||||
* @return The index of the next highest visible widget.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const int getHigherVisibleWidget(const int startIndex) const;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the index of the next visible widget lower down the z-order.
|
|
||||||
*
|
|
||||||
* @param startIndex The starting index.
|
|
||||||
* @return The index of the next lowest visible widget.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const int getLowerVisibleWidget(const int startIndex) const;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Notify this widget that it is being dragged, and set its drag point.
|
* Notify this widget that it is being dragged, and set its drag point.
|
||||||
*
|
*
|
||||||
|
@ -748,24 +706,6 @@ namespace NXWidgets
|
||||||
|
|
||||||
void getRect(CRect &rect) const;
|
void getRect(CRect &rect) const;
|
||||||
|
|
||||||
/**
|
|
||||||
* Clips the supplied rect to the boundaries defined by this widget and
|
|
||||||
* this widget's parents.
|
|
||||||
*
|
|
||||||
* @param rect Reference to a rect to populate with data.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void getRectClippedToHierarchy(CRect &rect) const;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets a pointer to the vector of all of the visible regions of this widget,
|
|
||||||
* including any covered by children.
|
|
||||||
*
|
|
||||||
* @return A pointer to a vector of all visible regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
TNxArray<CRect> *getForegroundRegions(void);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets a pointer to the widget's font.
|
* Gets a pointer to the widget's font.
|
||||||
*
|
*
|
||||||
|
@ -1061,13 +1001,6 @@ namespace NXWidgets
|
||||||
|
|
||||||
void redraw(void);
|
void redraw(void);
|
||||||
|
|
||||||
/**
|
|
||||||
* Erases the visible regions of the widget by redrawing the widgets
|
|
||||||
* behind it.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void erase(void);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Enables the widget.
|
* Enables the widget.
|
||||||
*
|
*
|
||||||
|
@ -1246,42 +1179,6 @@ namespace NXWidgets
|
||||||
bool changeDimensions(nxgl_coord_t x, nxgl_coord_t y,
|
bool changeDimensions(nxgl_coord_t x, nxgl_coord_t y,
|
||||||
nxgl_coord_t width, nxgl_coord_t height);
|
nxgl_coord_t width, nxgl_coord_t height);
|
||||||
|
|
||||||
/**
|
|
||||||
* Raises the widget to the top of its parent's widget stack.
|
|
||||||
*
|
|
||||||
* @return True if the raise was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool raiseToTop(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Lowers the widget to the bottom of its parent's widget stack.
|
|
||||||
*
|
|
||||||
* @return True if the lower was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool lowerToBottom(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Raises the supplied widget to the top of this widget's child stack.
|
|
||||||
* The supplied widget pointer must be a child of this widget.
|
|
||||||
*
|
|
||||||
* @param widget A pointer to the child widget to raise.
|
|
||||||
* @return True if the raise was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool raiseWidgetToTop(CNxWidget *widget);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Lowers the supplied widget to the bottom of this widget's child stack.
|
|
||||||
* The supplied widget pointer must be a child of this widget.
|
|
||||||
*
|
|
||||||
* @param widget A pointer to the child widget to lower.
|
|
||||||
* @return True if the lower was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool lowerWidgetToBottom(CNxWidget *widget);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Moves the supplied child widget to the deletion queue.
|
* Moves the supplied child widget to the deletion queue.
|
||||||
* For framework use only.
|
* For framework use only.
|
||||||
|
@ -1333,16 +1230,6 @@ namespace NXWidgets
|
||||||
|
|
||||||
bool checkCollision(CNxWidget *widget) const;
|
bool checkCollision(CNxWidget *widget) const;
|
||||||
|
|
||||||
/**
|
|
||||||
* Invalidate the visible region cache for all widgets below the supplied
|
|
||||||
* widget in this widget's child stack. This will cause those widgets to
|
|
||||||
*
|
|
||||||
* recalculate their visible regions next time they try to draw themselves.
|
|
||||||
* @param widget A pointer to a child widget.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void invalidateLowerWidgetsVisibleRectCache(CNxWidget *widget);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds a widget to this widget's child stack. The widget is added to the
|
* Adds a widget to this widget's child stack. The widget is added to the
|
||||||
* top of the stack. Note that the widget can only be added if it is not
|
* top of the stack. Note that the widget can only be added if it is not
|
||||||
|
@ -1377,61 +1264,6 @@ namespace NXWidgets
|
||||||
m_parent = parent;
|
m_parent = parent;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Rebuild the list of this widget's visible regions
|
|
||||||
*/
|
|
||||||
|
|
||||||
void cacheVisibleRects(void) const;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Mark this widget's visible region cache as invalid, and do the same
|
|
||||||
* to its child widgets.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void invalidateVisibleRectCache(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Erase a child widget by drawing the widgets behind it.
|
|
||||||
*
|
|
||||||
* @param widget The child widget to erase.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void eraseWidget(CNxWidget *widget);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Redraw any visible regions of this widget that have become corrupted.
|
|
||||||
*
|
|
||||||
* @param invalidRects A list of corrupt regions.
|
|
||||||
* @param sender A pointer to the widget that corrupted the regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void redrawDirty(TNxArray<CRect>* invalidRects, CNxWidget *sender);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Clips a rectangular region to the dimensions of this widget and its ancestors.
|
|
||||||
*
|
|
||||||
* @param rect The region that needs to be clipped.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void clipRectToHierarchy(CRect &rect) const;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Swaps the depth of the supplied child widget.
|
|
||||||
*
|
|
||||||
* @param widget A pointer to the child widget that needs to swap depths.
|
|
||||||
* @return True if the swap was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
virtual bool swapWidgetDepth(CNxWidget *widget);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Swap the depth of this widget.
|
|
||||||
*
|
|
||||||
* @return True if the swap was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool swapDepth(void);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete this widget. This should never be called in user code; widget
|
* Delete this widget. This should never be called in user code; widget
|
||||||
* deletion is handled internally.
|
* deletion is handled internally.
|
||||||
|
@ -1480,15 +1312,6 @@ namespace NXWidgets
|
||||||
m_flags.modal = false;
|
m_flags.modal = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the index of the specified child widget.
|
|
||||||
*
|
|
||||||
* @param widget The widget to get the index of.
|
|
||||||
* @return The index of the widget. -1 if the widget is not found.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const int getWidgetIndex(const CNxWidget *widget) const;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the child widget at the specified index.
|
* Get the child widget at the specified index.
|
||||||
*
|
*
|
||||||
|
@ -1509,17 +1332,6 @@ namespace NXWidgets
|
||||||
return m_children.size();
|
return m_children.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a pointer to the cache of visible rects.
|
|
||||||
*
|
|
||||||
* @return A pointer to the cache of visible rects.
|
|
||||||
*/
|
|
||||||
|
|
||||||
inline CRectCache *getCRectCache(void) const
|
|
||||||
{
|
|
||||||
return m_rectCache;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the border size. The border cannot be drawn over in the
|
* Sets the border size. The border cannot be drawn over in the
|
||||||
* drawContents() method.
|
* drawContents() method.
|
||||||
|
|
|
@ -1,222 +0,0 @@
|
||||||
/****************************************************************************
|
|
||||||
* NxWidgets/libnxwidgets/include/crectcache.hxx
|
|
||||||
*
|
|
||||||
* Copyright (C) 2012 Gregory Nutt. All rights reserved.
|
|
||||||
* Author: Gregory Nutt <gnutt@nuttx.org>
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions
|
|
||||||
* are met:
|
|
||||||
*
|
|
||||||
* 1. Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* 2. Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in
|
|
||||||
* the documentation and/or other materials provided with the
|
|
||||||
* distribution.
|
|
||||||
* 3. Neither the name NuttX, NxWidgets, nor the names of its contributors
|
|
||||||
* me be used to endorse or promote products derived from this software
|
|
||||||
* without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
||||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
||||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
||||||
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
||||||
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
||||||
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
||||||
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
||||||
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
||||||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
||||||
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
||||||
* POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*
|
|
||||||
****************************************************************************
|
|
||||||
*
|
|
||||||
* Portions of this package derive from Woopsi (http://woopsi.org/) and
|
|
||||||
* portions are original efforts. It is difficult to determine at this
|
|
||||||
* point what parts are original efforts and which parts derive from Woopsi.
|
|
||||||
* However, in any event, the work of Antony Dzeryn will be acknowledged
|
|
||||||
* in most NxWidget files. Thanks Antony!
|
|
||||||
*
|
|
||||||
* Copyright (c) 2007-2011, Antony Dzeryn
|
|
||||||
* All rights reserved.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
*
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the names "Woopsi", "Simian Zombie" nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY Antony Dzeryn ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL Antony Dzeryn BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#ifndef __INCLUDE_CRECTCACHE_HXX
|
|
||||||
#define __INCLUDE_CRECTCACHE_HXX
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Included Files
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#include <nuttx/config.h>
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stdbool.h>
|
|
||||||
|
|
||||||
#include <nuttx/nx/nxglib.h>
|
|
||||||
|
|
||||||
#include "cnxwidget.hxx"
|
|
||||||
#include "cwidgetstyle.hxx"
|
|
||||||
#include "tnxarray.hxx"
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Pre-Processor Definitions
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Implementation Classes
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
|
|
||||||
namespace NXWidgets
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Maintains a list of foreground (ie. above children) and background (with
|
|
||||||
* child overlapped-rects removed) rectangles representing the visible portions
|
|
||||||
* of a widget.
|
|
||||||
*/
|
|
||||||
|
|
||||||
class CRectCache
|
|
||||||
{
|
|
||||||
private:
|
|
||||||
TNxArray<CRect> m_foregroundRegions; /**< List of the widget's visible regions */
|
|
||||||
TNxArray<CRect> m_backgroundRegions; /**< List of the widget's visible regions with child rects removed */
|
|
||||||
const CNxWidget *m_widget; /**< Owning widget */
|
|
||||||
bool m_foregroundInvalid; /**< True if the foreground cache needs refreshing */
|
|
||||||
bool m_backgroundInvalid; /**< True if the background cache needs refreshing */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cache the foreground regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void cacheForegroundRegions(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cache the background regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void cacheBackgroundRegions(void);
|
|
||||||
|
|
||||||
public:
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructor.
|
|
||||||
*
|
|
||||||
* @param widget Widget that contains the rect cache.
|
|
||||||
*/
|
|
||||||
|
|
||||||
CRectCache(const CNxWidget *widget);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Destructor.
|
|
||||||
*/
|
|
||||||
|
|
||||||
inline ~CRectCache() { }
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rebuild the cache if it is invalid.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void cache(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Invalidates the cache.
|
|
||||||
*/
|
|
||||||
|
|
||||||
inline void invalidate(void)
|
|
||||||
{
|
|
||||||
m_foregroundInvalid = true;
|
|
||||||
m_backgroundInvalid = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the list of background regions. These are regions that are not overlapped by
|
|
||||||
* child widgets.
|
|
||||||
*
|
|
||||||
* @return The list of background regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
inline TNxArray<CRect> *getBackgroundRegions(void)
|
|
||||||
{
|
|
||||||
return &m_backgroundRegions;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the list of foreground regions. These are regions that represent the entire
|
|
||||||
* visible surface of the widget - that is, any regions not overlapped by ancestors or
|
|
||||||
* sublings of the widget - including any regions that are actually overlapped by
|
|
||||||
* child widgets.
|
|
||||||
*
|
|
||||||
* @return The list of foreground regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
inline TNxArray<CRect> *getForegroundRegions(void)
|
|
||||||
{
|
|
||||||
return &m_foregroundRegions;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Works out which rectangles in the invalidRects list overlap this
|
|
||||||
* widget, then cuts the rectangles into smaller pieces. The overlapping
|
|
||||||
* pieces are pushed into validRects, and the non-overlapping pieces are
|
|
||||||
* pushed back into the invalidRects vector.
|
|
||||||
*
|
|
||||||
* @param invalidRects A vector of regions that need to be tested
|
|
||||||
* for collisions against this widget; they represent regions that need
|
|
||||||
* to be redrawn.
|
|
||||||
* @param validRects A vector of regions that represents areas of the
|
|
||||||
* display that do not need to be redrawn.
|
|
||||||
* @param sender Pointer to the widget that initiated the split.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void splitRectangles(TNxArray<CRect> *invalidRects,
|
|
||||||
TNxArray<CRect> *validRects,
|
|
||||||
FAR const CNxWidget *sender) const;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Move any rectangles from the visibleRects list that overlap this widget
|
|
||||||
* into the invisibleRects list. Used during visible region calculations.
|
|
||||||
*
|
|
||||||
* @param visibleRects A vector of regions that are not overlapped.
|
|
||||||
* @param invisibleRects A vector of regions that are overlapped.
|
|
||||||
* @param widget The widget that requested the lists.
|
|
||||||
* @see splitRectangles()
|
|
||||||
*/
|
|
||||||
|
|
||||||
void removeOverlappedRects(TNxArray<CRect> *visibleRects,
|
|
||||||
TNxArray<CRect> *invisibleRects,
|
|
||||||
FAR const CNxWidget* widget) const;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __cplusplus
|
|
||||||
|
|
||||||
#endif // __INCLUDE_CRECTCACHE_HXX
|
|
||||||
|
|
|
@ -391,14 +391,6 @@ namespace NXWidgets
|
||||||
|
|
||||||
bool swapWidgetDepth(CNxWidget *widget);
|
bool swapWidgetDepth(CNxWidget *widget);
|
||||||
|
|
||||||
/**
|
|
||||||
* Redraws any dirty regions within the supplied region.
|
|
||||||
*
|
|
||||||
* @param rect The region to redraw
|
|
||||||
*/
|
|
||||||
|
|
||||||
void eraseRect(CRect rect);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add another widget to be managed by this control instance
|
* Add another widget to be managed by this control instance
|
||||||
*
|
*
|
||||||
|
|
|
@ -276,22 +276,6 @@ namespace NXWidgets
|
||||||
*/
|
*/
|
||||||
|
|
||||||
virtual void handleActionEvent(const CWidgetEventArgs &e) { }
|
virtual void handleActionEvent(const CWidgetEventArgs &e) { }
|
||||||
|
|
||||||
/**
|
|
||||||
* Handle a widget move forward event.
|
|
||||||
*
|
|
||||||
* @param e The event data.
|
|
||||||
*/
|
|
||||||
|
|
||||||
virtual void handleMoveForwardEvent(const CWidgetEventArgs &e) { }
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handle a widget move backward event.
|
|
||||||
*
|
|
||||||
* @param e The event data.
|
|
||||||
*/
|
|
||||||
|
|
||||||
virtual void handleMoveBackwardEvent(const CWidgetEventArgs &e) { }
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -247,18 +247,6 @@ namespace NXWidgets
|
||||||
|
|
||||||
void raiseDropEvent(nxgl_coord_t x, nxgl_coord_t y);
|
void raiseDropEvent(nxgl_coord_t x, nxgl_coord_t y);
|
||||||
|
|
||||||
/**
|
|
||||||
* Raise a move forward event to the event handler.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void raiseMoveForwardEvent(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Raise a move backward event to the event handler.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void raiseMoveBackwardEvent(void);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Raise a key press event to the event handler.
|
* Raise a key press event to the event handler.
|
||||||
*
|
*
|
||||||
|
|
|
@ -82,7 +82,6 @@
|
||||||
#include "cnxfont.hxx"
|
#include "cnxfont.hxx"
|
||||||
#include "cwidgetstyle.hxx"
|
#include "cwidgetstyle.hxx"
|
||||||
#include "cwidgeteventargs.hxx"
|
#include "cwidgeteventargs.hxx"
|
||||||
#include "crectcache.hxx"
|
|
||||||
#include "cwidgetcontrol.hxx"
|
#include "cwidgetcontrol.hxx"
|
||||||
#include "singletons.hxx"
|
#include "singletons.hxx"
|
||||||
|
|
||||||
|
@ -195,8 +194,7 @@ CNxWidget::CNxWidget(CWidgetControl *pWidgetControl,
|
||||||
m_borderSize.right = 1;
|
m_borderSize.right = 1;
|
||||||
m_borderSize.bottom = 1;
|
m_borderSize.bottom = 1;
|
||||||
m_borderSize.left = 1;
|
m_borderSize.left = 1;
|
||||||
|
|
||||||
m_rectCache = new CRectCache(this);
|
|
||||||
m_widgetEventHandlers = new CWidgetEventHandlerList(this);
|
m_widgetEventHandlers = new CWidgetEventHandlerList(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -237,7 +235,6 @@ CNxWidget::~CNxWidget(void)
|
||||||
// Delete instances. NOTE that we do not delete the controlling
|
// Delete instances. NOTE that we do not delete the controlling
|
||||||
// widget. It persists until the window is closed.
|
// widget. It persists until the window is closed.
|
||||||
|
|
||||||
delete m_rectCache;
|
|
||||||
delete m_widgetEventHandlers;
|
delete m_widgetEventHandlers;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -466,41 +463,6 @@ void CNxWidget::getRect(CRect &rect) const
|
||||||
rect.setY(rect.getY() + getY());
|
rect.setY(rect.getY() + getY());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Clips the supplied rect to the boundaries defined by this widget and
|
|
||||||
* this widget's parents.
|
|
||||||
*
|
|
||||||
* @param rect Reference to a rect to populate with data.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::getRectClippedToHierarchy(CRect &rect) const
|
|
||||||
{
|
|
||||||
// Copy the widget's properties into the rect. NOTE that if this is
|
|
||||||
// a child widget, getX() and getY() will return the actual X and Y
|
|
||||||
// positions (and not the parent-relative X and Y positions).
|
|
||||||
|
|
||||||
rect.setX(getX());
|
|
||||||
rect.setY(getY());
|
|
||||||
rect.setWidth(getWidth());
|
|
||||||
rect.setHeight(getHeight());
|
|
||||||
|
|
||||||
// And clip it
|
|
||||||
|
|
||||||
clipRectToHierarchy(rect);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets a pointer to the vector of all of the visible regions of this widget,
|
|
||||||
* including any covered by children.
|
|
||||||
*
|
|
||||||
* @return A pointer to a vector of all visible regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
TNxArray<CRect> *CNxWidget::getForegroundRegions(void)
|
|
||||||
{
|
|
||||||
return m_rectCache->getForegroundRegions();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets this widget's border state.
|
* Sets this widget's border state.
|
||||||
*
|
*
|
||||||
|
@ -510,7 +472,6 @@ TNxArray<CRect> *CNxWidget::getForegroundRegions(void)
|
||||||
void CNxWidget::setBorderless(bool borderless)
|
void CNxWidget::setBorderless(bool borderless)
|
||||||
{
|
{
|
||||||
m_flags.borderless = borderless;
|
m_flags.borderless = borderless;
|
||||||
invalidateVisibleRectCache();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -532,53 +493,25 @@ void CNxWidget::redraw(void)
|
||||||
{
|
{
|
||||||
if (isDrawingEnabled())
|
if (isDrawingEnabled())
|
||||||
{
|
{
|
||||||
cacheVisibleRects();
|
// Get the graphics port needed to draw on this window
|
||||||
|
|
||||||
if (m_rectCache->getBackgroundRegions()->size() > 0)
|
CGraphicsPort *port = m_widgetControl->getGraphicsPort();
|
||||||
{
|
|
||||||
// Get the graphics port for drawing in this window
|
|
||||||
|
|
||||||
CGraphicsPort *port = m_widgetControl->getGraphicsPort();
|
// Draw the Widget
|
||||||
|
|
||||||
// Draw all visible rectangles
|
drawBorder(port);
|
||||||
|
drawContents(port);
|
||||||
for (int i = 0; i < m_rectCache->getBackgroundRegions()->size(); i++)
|
|
||||||
{
|
|
||||||
drawBorder(port);
|
|
||||||
drawContents(port);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remember that the widget is no longer erased
|
// Remember that the widget is no longer erased
|
||||||
|
|
||||||
m_flags.erased = false;
|
m_flags.erased = false;
|
||||||
|
|
||||||
|
// Draw the children of the widget
|
||||||
|
|
||||||
drawChildren();
|
drawChildren();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Erases the visible regions of the widget by redrawing the widgets
|
|
||||||
* behind it.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::erase(void)
|
|
||||||
{
|
|
||||||
if (!m_flags.erased)
|
|
||||||
{
|
|
||||||
cacheVisibleRects();
|
|
||||||
|
|
||||||
if (m_parent != (CNxWidget *)NULL)
|
|
||||||
{
|
|
||||||
m_parent->eraseWidget(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remember that the widget has been erased
|
|
||||||
|
|
||||||
m_flags.erased = true;
|
|
||||||
invalidateVisibleRectCache();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Enables the widget.
|
* Enables the widget.
|
||||||
*
|
*
|
||||||
|
@ -646,7 +579,6 @@ void CNxWidget::close(void)
|
||||||
// Ensure the widget isn't running modally
|
// Ensure the widget isn't running modally
|
||||||
|
|
||||||
stopModal();
|
stopModal();
|
||||||
erase();
|
|
||||||
|
|
||||||
if (m_parent != (CNxWidget *)NULL)
|
if (m_parent != (CNxWidget *)NULL)
|
||||||
{
|
{
|
||||||
|
@ -669,10 +601,6 @@ bool CNxWidget::show(void)
|
||||||
{
|
{
|
||||||
m_flags.hidden = false;
|
m_flags.hidden = false;
|
||||||
|
|
||||||
// Ensure that widgets behind this do not draw over the
|
|
||||||
// top of the newly-visible widget
|
|
||||||
|
|
||||||
m_parent->invalidateLowerWidgetsVisibleRectCache(this);
|
|
||||||
m_widgetEventHandlers->raiseShowEvent();
|
m_widgetEventHandlers->raiseShowEvent();
|
||||||
redraw();
|
redraw();
|
||||||
return true;
|
return true;
|
||||||
|
@ -699,7 +627,6 @@ bool CNxWidget::hide(void)
|
||||||
|
|
||||||
stopModal();
|
stopModal();
|
||||||
m_widgetEventHandlers->raiseHideEvent();
|
m_widgetEventHandlers->raiseHideEvent();
|
||||||
erase();
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -741,14 +668,6 @@ bool CNxWidget::click(nxgl_coord_t x, nxgl_coord_t y)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure that the click has occurred on a region of this widget
|
|
||||||
// not obscured by its siblings
|
|
||||||
|
|
||||||
if (!checkCollisionWithForegroundRects(x, y))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle clicks on this
|
// Handle clicks on this
|
||||||
|
|
||||||
m_flags.clicked = true;
|
m_flags.clicked = true;
|
||||||
|
@ -834,14 +753,6 @@ bool CNxWidget::doubleClick(nxgl_coord_t x, nxgl_coord_t y)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure that the click has occurred on a region of this widget
|
|
||||||
// not obscured by its siblings
|
|
||||||
|
|
||||||
if (!checkCollisionWithForegroundRects(x, y))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
m_flags.clicked = true;
|
m_flags.clicked = true;
|
||||||
|
|
||||||
// Record data for double-click
|
// Record data for double-click
|
||||||
|
@ -1143,8 +1054,6 @@ bool CNxWidget::moveTo(nxgl_coord_t x, nxgl_coord_t y)
|
||||||
|
|
||||||
if ((m_rect.getX() != x) || (m_rect.getY() != y))
|
if ((m_rect.getX() != x) || (m_rect.getY() != y))
|
||||||
{
|
{
|
||||||
erase();
|
|
||||||
|
|
||||||
nxgl_coord_t oldX = m_rect.getX();
|
nxgl_coord_t oldX = m_rect.getX();
|
||||||
nxgl_coord_t oldY = m_rect.getY();
|
nxgl_coord_t oldY = m_rect.getY();
|
||||||
|
|
||||||
|
@ -1205,19 +1114,11 @@ bool CNxWidget::resize(nxgl_coord_t width, nxgl_coord_t height)
|
||||||
bool wasDrawEnabled = m_flags.drawingEnabled;
|
bool wasDrawEnabled = m_flags.drawingEnabled;
|
||||||
|
|
||||||
m_flags.permeable = true;
|
m_flags.permeable = true;
|
||||||
erase();
|
|
||||||
disableDrawing();
|
disableDrawing();
|
||||||
|
|
||||||
m_rect.setWidth(width);
|
m_rect.setWidth(width);
|
||||||
m_rect.setHeight(height);
|
m_rect.setHeight(height);
|
||||||
|
|
||||||
// Handle visible region caching
|
|
||||||
|
|
||||||
if (m_parent != (CNxWidget *)NULL)
|
|
||||||
{
|
|
||||||
m_parent->invalidateLowerWidgetsVisibleRectCache(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
onResize(width, height);
|
onResize(width, height);
|
||||||
|
|
||||||
// Reset the permeable value
|
// Reset the permeable value
|
||||||
|
@ -1257,117 +1158,6 @@ bool CNxWidget::changeDimensions(nxgl_coord_t x, nxgl_coord_t y,
|
||||||
return (resize(width, height) | moved);
|
return (resize(width, height) | moved);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Raises the widget to the top of its parent's widget stack.
|
|
||||||
*
|
|
||||||
* @return True if the raise was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool CNxWidget::raiseToTop(void)
|
|
||||||
{
|
|
||||||
if (m_parent != (CNxWidget *)NULL)
|
|
||||||
{
|
|
||||||
if (m_parent->raiseWidgetToTop(this))
|
|
||||||
{
|
|
||||||
m_widgetEventHandlers->raiseMoveForwardEvent();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Lowers the widget to the bottom of its parent's widget stack.
|
|
||||||
*
|
|
||||||
* @return True if the lower was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool CNxWidget::lowerToBottom(void)
|
|
||||||
{
|
|
||||||
if (m_parent != (CNxWidget *)NULL)
|
|
||||||
{
|
|
||||||
if (m_parent->lowerWidgetToBottom(this))
|
|
||||||
{
|
|
||||||
m_widgetEventHandlers->raiseMoveBackwardEvent();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Raises the supplied widget to the top of this widget's child stack.
|
|
||||||
* The supplied widget pointer must be a child of this widget.
|
|
||||||
*
|
|
||||||
* @param widget A pointer to the child widget to raise.
|
|
||||||
* @return True if the raise was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool CNxWidget::raiseWidgetToTop(CNxWidget *widget)
|
|
||||||
{
|
|
||||||
// Locate widget in the stack
|
|
||||||
|
|
||||||
int index = getWidgetIndex(widget);
|
|
||||||
|
|
||||||
if ((index > -1) && (index < m_children.size() - 1))
|
|
||||||
{
|
|
||||||
m_children.erase(index);
|
|
||||||
m_children.push_back(widget);
|
|
||||||
|
|
||||||
widget->invalidateVisibleRectCache();
|
|
||||||
|
|
||||||
// Invalidate all widgets that collide with the depth-swapped widget
|
|
||||||
|
|
||||||
for (int i = 0; i < m_children.size(); i++)
|
|
||||||
{
|
|
||||||
if (m_children[i]->checkCollision(widget))
|
|
||||||
{
|
|
||||||
m_children[i]->invalidateVisibleRectCache();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
widget->redraw();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Lowers the supplied widget to the bottom of this widget's child stack.
|
|
||||||
* The supplied widget pointer must be a child of this widget.
|
|
||||||
*
|
|
||||||
* @param widget A pointer to the child widget to lower.
|
|
||||||
* @return True if the lower was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool CNxWidget::lowerWidgetToBottom(CNxWidget *widget)
|
|
||||||
{
|
|
||||||
// Locate widget in the stack
|
|
||||||
|
|
||||||
int index = getWidgetIndex(widget);
|
|
||||||
|
|
||||||
if (index > 0)
|
|
||||||
{
|
|
||||||
widget->erase();
|
|
||||||
|
|
||||||
// Handle visible region caching
|
|
||||||
|
|
||||||
widget->invalidateVisibleRectCache();
|
|
||||||
invalidateLowerWidgetsVisibleRectCache(widget);
|
|
||||||
|
|
||||||
m_children.erase(index);
|
|
||||||
m_children.insert(0, widget);
|
|
||||||
|
|
||||||
widget->redraw();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Moves the supplied child widget to the deletion queue.
|
* Moves the supplied child widget to the deletion queue.
|
||||||
* For framework use only.
|
* For framework use only.
|
||||||
|
@ -1446,7 +1236,7 @@ bool CNxWidget::checkCollision(nxgl_coord_t x, nxgl_coord_t y) const
|
||||||
// Get the clipped rect
|
// Get the clipped rect
|
||||||
|
|
||||||
CRect rect;
|
CRect rect;
|
||||||
getRectClippedToHierarchy(rect);
|
getRect(rect);
|
||||||
return rect.contains(x, y);
|
return rect.contains(x, y);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1471,7 +1261,7 @@ bool CNxWidget::checkCollision(nxgl_coord_t x, nxgl_coord_t y,
|
||||||
// Get the clipped rect
|
// Get the clipped rect
|
||||||
|
|
||||||
CRect rect;
|
CRect rect;
|
||||||
getRectClippedToHierarchy(rect);
|
getRect(rect);
|
||||||
return rect.intersects(CRect(x, y, width, height));
|
return rect.intersects(CRect(x, y, width, height));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1487,35 +1277,10 @@ bool CNxWidget::checkCollision(CNxWidget *widget) const
|
||||||
// Get the clipped rect
|
// Get the clipped rect
|
||||||
|
|
||||||
CRect rect;
|
CRect rect;
|
||||||
widget->getRectClippedToHierarchy(rect);
|
widget->getRect(rect);
|
||||||
return rect.intersects(m_rect);
|
return rect.intersects(m_rect);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Invalidate the visible region cache for all widgets below the supplied
|
|
||||||
* widget in this widget's child stack. This will cause those widgets to
|
|
||||||
*
|
|
||||||
* recalculate their visible regions next time they try to draw themselves.
|
|
||||||
* @param widget A pointer to a child widget.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::invalidateLowerWidgetsVisibleRectCache(CNxWidget *widget)
|
|
||||||
{
|
|
||||||
// Find the widget
|
|
||||||
|
|
||||||
int widgetIndex = getWidgetIndex(widget);
|
|
||||||
|
|
||||||
// Invalidate lower widgets
|
|
||||||
|
|
||||||
for (int i = widgetIndex - 1; i > -1; i--)
|
|
||||||
{
|
|
||||||
if (m_children[i]->checkCollision(widget))
|
|
||||||
{
|
|
||||||
m_children[i]->invalidateVisibleRectCache();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds a widget to this widget's child stack. The widget is added to the
|
* Adds a widget to this widget's child stack. The widget is added to the
|
||||||
* top of the stack. Note that the widget can only be added if it is not
|
* top of the stack. Note that the widget can only be added if it is not
|
||||||
|
@ -1540,7 +1305,6 @@ void CNxWidget::addWidget(CNxWidget *widget)
|
||||||
}
|
}
|
||||||
|
|
||||||
widget->enableDrawing();
|
widget->enableDrawing();
|
||||||
invalidateVisibleRectCache();
|
|
||||||
widget->redraw();
|
widget->redraw();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1562,251 +1326,10 @@ void CNxWidget::insertWidget(CNxWidget *widget)
|
||||||
m_children.insert(0, widget);
|
m_children.insert(0, widget);
|
||||||
|
|
||||||
widget->enableDrawing();
|
widget->enableDrawing();
|
||||||
invalidateVisibleRectCache();
|
|
||||||
widget->redraw();
|
widget->redraw();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Rebuild the list of this widget's visible regions
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::cacheVisibleRects(void) const
|
|
||||||
{
|
|
||||||
m_rectCache->cache();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Mark this widget's visible region cache as invalid, and do the same
|
|
||||||
* to its child widgets.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::invalidateVisibleRectCache(void)
|
|
||||||
{
|
|
||||||
m_rectCache->invalidate();
|
|
||||||
|
|
||||||
// Invalidate child cache
|
|
||||||
|
|
||||||
for (int i = 0; i < m_children.size(); i++)
|
|
||||||
{
|
|
||||||
m_children[i]->invalidateVisibleRectCache();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Erase a child widget by drawing the widgets behind it.
|
|
||||||
*
|
|
||||||
* @param widget The child widget to erase.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::eraseWidget(CNxWidget *widget)
|
|
||||||
{
|
|
||||||
// Locate the widget
|
|
||||||
|
|
||||||
int widgetIndex = getWidgetIndex(widget);
|
|
||||||
|
|
||||||
// Ensure rect cache is up to date
|
|
||||||
|
|
||||||
widget->cacheVisibleRects();
|
|
||||||
|
|
||||||
// Order all lower widgets to redraw themselves based on the erased widget's
|
|
||||||
// visible rect cache
|
|
||||||
|
|
||||||
for (int i = widgetIndex - 1; i > -1; i--)
|
|
||||||
{
|
|
||||||
m_children[i]->redrawDirty(widget->getForegroundRegions(), widget);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Order this widget to redraw itself based on any remaining rectangles
|
|
||||||
// in the erased widget's rect cache
|
|
||||||
|
|
||||||
redrawDirty(widget->getForegroundRegions(), widget);
|
|
||||||
invalidateVisibleRectCache();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Redraw any visible regions of this widget that have become corrupted.
|
|
||||||
*
|
|
||||||
* @param invalidRects A list of corrupt regions.
|
|
||||||
* @param sender A pointer to the widget that corrupted the regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::redrawDirty(TNxArray<CRect> *invalidRects, CNxWidget *sender)
|
|
||||||
{
|
|
||||||
if (isDrawingEnabled())
|
|
||||||
{
|
|
||||||
// Draw any children first
|
|
||||||
|
|
||||||
redrawDirtyChildren(invalidRects, sender);
|
|
||||||
|
|
||||||
// Create an array that will contain all of the rects from the
|
|
||||||
// original array that overlap this widget
|
|
||||||
|
|
||||||
TNxArray<CRect> *overlappingRects = new TNxArray<CRect>();
|
|
||||||
|
|
||||||
// Remove any non-overlapping rectangles from dirty vector and add to
|
|
||||||
// overlapping vector
|
|
||||||
|
|
||||||
m_rectCache->splitRectangles(invalidRects, overlappingRects, sender);
|
|
||||||
|
|
||||||
// Create an array that will contain all of the rects that overlap this
|
|
||||||
// widget clipped to its parent
|
|
||||||
|
|
||||||
TNxArray<CRect> *rectsToDraw = new TNxArray<CRect>();
|
|
||||||
|
|
||||||
// Split from overlappingRects into rectsToDraw, giving us an array
|
|
||||||
// of rects that overlap only the visible portions of this widget
|
|
||||||
|
|
||||||
m_rectCache->splitRectangles(overlappingRects, rectsToDraw, sender);
|
|
||||||
|
|
||||||
// Get the graphics port for drawing in this window
|
|
||||||
|
|
||||||
CGraphicsPort *port = m_widgetControl->getGraphicsPort();
|
|
||||||
|
|
||||||
// Draw the dirty rects
|
|
||||||
|
|
||||||
if (rectsToDraw->size() > 0)
|
|
||||||
{
|
|
||||||
for (int i = 0; i < rectsToDraw->size(); i++)
|
|
||||||
{
|
|
||||||
drawBorder(port);
|
|
||||||
drawContents(port);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Copy all of the overlapping rects we didn't draw back to the main
|
|
||||||
// array of rects that need to be drawn by another widget
|
|
||||||
|
|
||||||
for (int i = 0; i < overlappingRects->size(); i++)
|
|
||||||
{
|
|
||||||
invalidRects->push_back(overlappingRects->at(i));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clean up
|
|
||||||
|
|
||||||
delete overlappingRects;
|
|
||||||
delete rectsToDraw;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Clips a rectangular region to the dimensions of this widget and its ancestors.
|
|
||||||
*
|
|
||||||
* @param rect The region that needs to be clipped.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::clipRectToHierarchy(CRect &rect) const {
|
|
||||||
|
|
||||||
const CNxWidget *parent = m_parent;
|
|
||||||
const CNxWidget *widget = this;
|
|
||||||
CRect thisRect;
|
|
||||||
|
|
||||||
while (parent != NULL)
|
|
||||||
{
|
|
||||||
// Standard widgets can draw into client space
|
|
||||||
|
|
||||||
parent->getClientRect(thisRect);
|
|
||||||
|
|
||||||
// Adjust rect to screen space
|
|
||||||
|
|
||||||
thisRect.offset(parent->getX(), parent->getY());
|
|
||||||
rect.clipToIntersect(thisRect);
|
|
||||||
|
|
||||||
// Send up to parent
|
|
||||||
|
|
||||||
widget = parent;
|
|
||||||
parent = parent->getParent();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Swaps the depth of the supplied child widget.
|
|
||||||
*
|
|
||||||
* @param widget A pointer to the child widget that needs to swap depths.
|
|
||||||
* @return True if the swap was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool CNxWidget::swapWidgetDepth(CNxWidget *widget)
|
|
||||||
{
|
|
||||||
// Can we swap?
|
|
||||||
|
|
||||||
if (m_children.size() > 1)
|
|
||||||
{
|
|
||||||
int widgetSource = 0;
|
|
||||||
int widgetDest = 0;
|
|
||||||
|
|
||||||
// Locate the widget in the vector
|
|
||||||
|
|
||||||
widgetSource = getWidgetIndex(widget);
|
|
||||||
|
|
||||||
// Attempt to raise up
|
|
||||||
|
|
||||||
int i = getHigherVisibleWidget(widgetSource);
|
|
||||||
if (i > -1)
|
|
||||||
{
|
|
||||||
// Raise
|
|
||||||
|
|
||||||
widgetDest = i;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Lower to bottom of stack
|
|
||||||
|
|
||||||
widgetDest = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Erase the widget from the screen
|
|
||||||
|
|
||||||
eraseWidget(widget);
|
|
||||||
|
|
||||||
// Swap
|
|
||||||
|
|
||||||
CNxWidget *tmp = m_children[widgetSource];
|
|
||||||
m_children[widgetSource] = m_children[widgetDest];
|
|
||||||
m_children[widgetDest] = tmp;
|
|
||||||
|
|
||||||
// Invalidate the widgets below the top affected widget
|
|
||||||
|
|
||||||
if (widgetSource < widgetDest)
|
|
||||||
{
|
|
||||||
// Source lower; invalidate from dest down
|
|
||||||
|
|
||||||
m_children[widgetDest]->invalidateVisibleRectCache();
|
|
||||||
invalidateLowerWidgetsVisibleRectCache(m_children[widgetDest]);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Dest lower; invalidate from source down
|
|
||||||
|
|
||||||
m_children[widgetSource]->invalidateVisibleRectCache();
|
|
||||||
invalidateLowerWidgetsVisibleRectCache(m_children[widgetSource]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Redraw the widget
|
|
||||||
|
|
||||||
widget->redraw();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Swap the depth of this widget.
|
|
||||||
*
|
|
||||||
* @return True if the swap was successful.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool CNxWidget::swapDepth(void)
|
|
||||||
{
|
|
||||||
if (m_parent != (CNxWidget *)NULL)
|
|
||||||
{
|
|
||||||
return m_parent->swapWidgetDepth(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Remove this widget from the widget hierarchy. Returns
|
* Remove this widget from the widget hierarchy. Returns
|
||||||
* responsibility for deleting the widget back to the developer.
|
* responsibility for deleting the widget back to the developer.
|
||||||
|
@ -1908,26 +1431,6 @@ void CNxWidget::goModal(void)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the index of the specified child widget.
|
|
||||||
*
|
|
||||||
* @param widget The widget to get the index of.
|
|
||||||
* @return The index of the widget. -1 if the widget is not found.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const int CNxWidget::getWidgetIndex(const CNxWidget *widget) const
|
|
||||||
{
|
|
||||||
for (int i = 0; i < m_children.size(); i++)
|
|
||||||
{
|
|
||||||
if (m_children[i] == widget)
|
|
||||||
{
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the child widget at the specified index.
|
* Get the child widget at the specified index.
|
||||||
*
|
*
|
||||||
|
@ -1976,38 +1479,6 @@ void CNxWidget::useWidgetStyle(const CWidgetStyle *style)
|
||||||
m_style.font = style->font;
|
m_style.font = style->font;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the supplied coordinates collide with a portion of this widget
|
|
||||||
* that is not obscured by its siblings, but that may be obscured by
|
|
||||||
* its children.
|
|
||||||
*
|
|
||||||
* @param x X coordinate of the click.
|
|
||||||
* @param y Y coordinate of the click.
|
|
||||||
* @return True if a collision occurred; false if not.
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool CNxWidget::checkCollisionWithForegroundRects(nxgl_coord_t x, nxgl_coord_t y) const
|
|
||||||
{
|
|
||||||
if (isHidden())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
cacheVisibleRects();
|
|
||||||
|
|
||||||
CRect *rect;
|
|
||||||
for (int i = 0; i < m_rectCache->getForegroundRegions()->size(); ++i)
|
|
||||||
{
|
|
||||||
rect = &(m_rectCache->getForegroundRegions()->at(i));
|
|
||||||
if (rect->contains(x, y))
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Draw all visible regions of this widget's children.
|
* Draw all visible regions of this widget's children.
|
||||||
*/
|
*/
|
||||||
|
@ -2079,72 +1550,6 @@ void CNxWidget::closeChild(CNxWidget *widget)
|
||||||
moveChildToDeleteQueue(widget);
|
moveChildToDeleteQueue(widget);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Redraws all regions of child widgets that fall within the invalidRects
|
|
||||||
* regions.
|
|
||||||
*
|
|
||||||
* @param invalidRects List of invalid regions that need to be redrawn.
|
|
||||||
* @param sender Pointer to the widget that initiated the redraw.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CNxWidget::redrawDirtyChildren(TNxArray<CRect> *invalidRects, CNxWidget *sender)
|
|
||||||
{
|
|
||||||
for (int i = m_children.size() - 1; i > -1 ; i--)
|
|
||||||
{
|
|
||||||
if (invalidRects->size() > 0)
|
|
||||||
{
|
|
||||||
if (m_children.at(i) != sender)
|
|
||||||
{
|
|
||||||
m_children[i]->redrawDirty(invalidRects, sender);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the index of the next visible widget higher up the z-order.
|
|
||||||
*
|
|
||||||
* @param startIndex The starting index.
|
|
||||||
* @return The index of the next highest visible widget.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const int CNxWidget::getHigherVisibleWidget(const int startIndex) const
|
|
||||||
{
|
|
||||||
for (int i = startIndex; i < m_children.size(); i++)
|
|
||||||
{
|
|
||||||
if (!m_children[i]->isHidden())
|
|
||||||
{
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the index of the next visible widget lower down the z-order.
|
|
||||||
*
|
|
||||||
* @param startIndex The starting index.
|
|
||||||
* @return The index of the next lowest visible widget.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const int CNxWidget::getLowerVisibleWidget(const int startIndex) const
|
|
||||||
{
|
|
||||||
for (int i = startIndex; i > -1; i--)
|
|
||||||
{
|
|
||||||
if (!m_children[i]->isHidden())
|
|
||||||
{
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Notify this widget that it is being dragged, and set its drag point.
|
* Notify this widget that it is being dragged, and set its drag point.
|
||||||
*
|
*
|
||||||
|
|
|
@ -1,521 +0,0 @@
|
||||||
/****************************************************************************
|
|
||||||
* NxWidgets/libnxwidgets/src/crectcache.cxx
|
|
||||||
*
|
|
||||||
* Copyright (C) 2012 Gregory Nutt. All rights reserved.
|
|
||||||
* Author: Gregory Nutt <gnutt@nuttx.org>
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions
|
|
||||||
* are met:
|
|
||||||
*
|
|
||||||
* 1. Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* 2. Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in
|
|
||||||
* the documentation and/or other materials provided with the
|
|
||||||
* distribution.
|
|
||||||
* 3. Neither the name NuttX, NxWidgets, nor the names of its contributors
|
|
||||||
* me be used to endorse or promote products derived from this software
|
|
||||||
* without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
||||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
||||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
||||||
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
||||||
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
||||||
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
||||||
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
||||||
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
||||||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
||||||
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
||||||
* POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*
|
|
||||||
****************************************************************************
|
|
||||||
*
|
|
||||||
* Portions of this package derive from Woopsi (http://woopsi.org/) and
|
|
||||||
* portions are original efforts. It is difficult to determine at this
|
|
||||||
* point what parts are original efforts and which parts derive from Woopsi.
|
|
||||||
* However, in any event, the work of Antony Dzeryn will be acknowledged
|
|
||||||
* in most NxWidget files. Thanks Antony!
|
|
||||||
*
|
|
||||||
* Copyright (c) 2007-2011, Antony Dzeryn
|
|
||||||
* All rights reserved.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
*
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the names "Woopsi", "Simian Zombie" nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY Antony Dzeryn ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL Antony Dzeryn BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Included Files
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#include <nuttx/config.h>
|
|
||||||
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stdbool.h>
|
|
||||||
|
|
||||||
#include <nuttx/nx/nxglib.h>
|
|
||||||
|
|
||||||
#include "crectcache.hxx"
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Pre-Processor Definitions
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* CNxFont Method Implementations
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
using namespace NXWidgets;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructor.
|
|
||||||
*
|
|
||||||
* @param widget Widget that contains the rect cache.
|
|
||||||
*/
|
|
||||||
|
|
||||||
CRectCache::CRectCache(const CNxWidget* widget)
|
|
||||||
{
|
|
||||||
m_widget = widget;
|
|
||||||
m_foregroundInvalid = true;
|
|
||||||
m_backgroundInvalid = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rebuild the cache if it is invalid.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CRectCache::cache(void)
|
|
||||||
{
|
|
||||||
cacheBackgroundRegions();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Works out which rectangles in the invalidRects list overlap this
|
|
||||||
* widget, then cuts the rectangles into smaller pieces. The overlapping
|
|
||||||
* pieces are pushed into validRects, and the non-overlapping pieces are
|
|
||||||
* pushed back into the invalidRects vector.
|
|
||||||
*
|
|
||||||
* @param invalidRects A vector of regions that need to be tested
|
|
||||||
* for collisions against this widget; they represent regions that need
|
|
||||||
* to be redrawn.
|
|
||||||
* @param validRects A vector of regions that represents areas of the
|
|
||||||
* display that do not need to be redrawn.
|
|
||||||
* @param sender Pointer to the widget that initiated the split.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CRectCache::splitRectangles(TNxArray<CRect>* invalidRects,
|
|
||||||
TNxArray<CRect>* validRects,
|
|
||||||
FAR const CNxWidget *sender) const
|
|
||||||
{
|
|
||||||
// Check for collisions with any rectangles in the vector
|
|
||||||
|
|
||||||
for (int i = 0; i < invalidRects->size(); i++)
|
|
||||||
{
|
|
||||||
// Get rectangle to check
|
|
||||||
|
|
||||||
CRect checkRect = invalidRects->at(i);
|
|
||||||
nxgl_coord_t splitX[4];
|
|
||||||
nxgl_coord_t splitY[4];
|
|
||||||
unsigned int rectXCount = 0;
|
|
||||||
unsigned int rectYCount = 0;
|
|
||||||
unsigned int overlapXRect = 0;
|
|
||||||
unsigned int overlapYRect = 0;
|
|
||||||
|
|
||||||
nxgl_coord_t width = checkRect.getWidth();
|
|
||||||
nxgl_coord_t height = checkRect.getHeight();
|
|
||||||
|
|
||||||
if (m_widget->checkCollision(checkRect.getX(), checkRect.getY(), width, height))
|
|
||||||
{
|
|
||||||
// Got a collision. We need to split this rectangle
|
|
||||||
// Get clipped dimensions of widget
|
|
||||||
|
|
||||||
CRect widgetRect;
|
|
||||||
m_widget->getRectClippedToHierarchy(widgetRect);
|
|
||||||
|
|
||||||
// Vertical split
|
|
||||||
// Start at left edge of rectangle
|
|
||||||
|
|
||||||
splitX[0] = checkRect.getX();
|
|
||||||
|
|
||||||
// Check for second split
|
|
||||||
|
|
||||||
if (checkRect.getX() < widgetRect.getX())
|
|
||||||
{
|
|
||||||
// Widget is to the right of the invalid rectangle (or in the centre)
|
|
||||||
|
|
||||||
if (splitX[rectXCount] != widgetRect.getX())
|
|
||||||
{
|
|
||||||
rectXCount++;
|
|
||||||
splitX[rectXCount] = widgetRect.getX();
|
|
||||||
|
|
||||||
// The next rectangle is the overlap
|
|
||||||
|
|
||||||
overlapXRect = rectXCount;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Widget rectangle is on the left of the invalid rectangle
|
|
||||||
|
|
||||||
if (splitX[rectXCount] != widgetRect.getX2() + 1)
|
|
||||||
{
|
|
||||||
// We've found the start of the overlapping rectangle!
|
|
||||||
|
|
||||||
overlapXRect = rectXCount;
|
|
||||||
rectXCount++;
|
|
||||||
|
|
||||||
// Split is either the end of the widget or the end of the
|
|
||||||
// invalid rect, whichever comes first
|
|
||||||
|
|
||||||
if (widgetRect.getX2() <= checkRect.getX2())
|
|
||||||
{
|
|
||||||
splitX[rectXCount] = widgetRect.getX2() + 1;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
splitX[rectXCount] = checkRect.getX2() + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Found the start of the overlapping rectangle
|
|
||||||
|
|
||||||
overlapXRect = rectXCount;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check for third split
|
|
||||||
|
|
||||||
if (widgetRect.getX2() + 1 <= checkRect.getX2() + 1)
|
|
||||||
{
|
|
||||||
// Widget ends before the invalid rectangle
|
|
||||||
|
|
||||||
if (splitX[rectXCount] != widgetRect.getX2() + 1)
|
|
||||||
{
|
|
||||||
// Record end of overlap
|
|
||||||
|
|
||||||
rectXCount++;
|
|
||||||
splitX[rectXCount] = widgetRect.getX2() + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store end of invalid rectangle
|
|
||||||
|
|
||||||
if (splitX[rectXCount] <= checkRect.getX2())
|
|
||||||
{
|
|
||||||
rectXCount++;
|
|
||||||
splitX[rectXCount] = checkRect.getX2() + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Horizontal split
|
|
||||||
// Start at left edge of rectangle
|
|
||||||
|
|
||||||
splitY[0] = checkRect.getY();
|
|
||||||
|
|
||||||
// Check for second split
|
|
||||||
|
|
||||||
if (checkRect.getY() < widgetRect.getY())
|
|
||||||
{
|
|
||||||
// Widget below the invalid rectangle (or in the centre)
|
|
||||||
|
|
||||||
if (splitY[rectYCount] != widgetRect.getY())
|
|
||||||
{
|
|
||||||
rectYCount++;
|
|
||||||
splitY[rectYCount] = widgetRect.getY();
|
|
||||||
|
|
||||||
// The next rectangle is the overlap
|
|
||||||
|
|
||||||
overlapYRect = rectYCount;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Widget rectangle above the invalid rectangle
|
|
||||||
|
|
||||||
if (splitY[rectYCount] != widgetRect.getY2() + 1)
|
|
||||||
{
|
|
||||||
// We've found the start of the overlapping rectangle!
|
|
||||||
|
|
||||||
overlapYRect = rectYCount;
|
|
||||||
rectYCount++;
|
|
||||||
|
|
||||||
// Split is either the end of the widget or the end of the
|
|
||||||
// invalid rect, whichever comes first
|
|
||||||
|
|
||||||
if (widgetRect.getY2() <= checkRect.getY2())
|
|
||||||
{
|
|
||||||
splitY[rectYCount] = widgetRect.getY2() + 1;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
splitY[rectYCount] = checkRect.getY2() + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Found the start of the overlapping rectangle
|
|
||||||
|
|
||||||
overlapYRect = rectYCount;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check for third split
|
|
||||||
|
|
||||||
if (widgetRect.getY2() < checkRect.getY2())
|
|
||||||
{
|
|
||||||
// Widget ends before the invalid rectangle
|
|
||||||
|
|
||||||
if (splitY[rectYCount] != widgetRect.getY2() + 1)
|
|
||||||
{
|
|
||||||
// Record end of overlap
|
|
||||||
|
|
||||||
rectYCount++;
|
|
||||||
splitY[rectYCount] = widgetRect.getY2() + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store end of invalid rectangle
|
|
||||||
|
|
||||||
if (splitY[rectYCount] <= checkRect.getY2())
|
|
||||||
{
|
|
||||||
rectYCount++;
|
|
||||||
splitY[rectYCount] = checkRect.getY() + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove the original rectangle
|
|
||||||
|
|
||||||
invalidRects->erase(i);
|
|
||||||
|
|
||||||
// Force the loop to re-examine the new rectangle at this index
|
|
||||||
|
|
||||||
i--;
|
|
||||||
|
|
||||||
// Add the new rectangles (not the overlap; that's the one we need to draw)
|
|
||||||
|
|
||||||
for (unsigned int xRects = 0; xRects < rectXCount; xRects++)
|
|
||||||
{
|
|
||||||
for (unsigned int yRects = 0; yRects < rectYCount; yRects++)
|
|
||||||
{
|
|
||||||
// Is this the overlap?
|
|
||||||
|
|
||||||
if ((overlapXRect == xRects) && (overlapYRect == yRects))
|
|
||||||
{
|
|
||||||
// Got the overlap, so set the output values
|
|
||||||
|
|
||||||
CRect overlapRect;
|
|
||||||
overlapRect.setX(splitX[xRects]);
|
|
||||||
overlapRect.setY(splitY[yRects]);
|
|
||||||
overlapRect.setX2(splitX[xRects + 1] - 1);
|
|
||||||
overlapRect.setY2(splitY[yRects + 1] - 1);
|
|
||||||
|
|
||||||
if (overlapRect.hasDimensions())
|
|
||||||
{
|
|
||||||
validRects->push_back(overlapRect);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Not an overlap; add to vector
|
|
||||||
|
|
||||||
CRect newRect;
|
|
||||||
newRect.setX(splitX[xRects]);
|
|
||||||
newRect.setY(splitY[yRects]);
|
|
||||||
newRect.setX2(splitX[xRects + 1] - 1);
|
|
||||||
newRect.setY2(splitY[yRects + 1] - 1);
|
|
||||||
|
|
||||||
// Insert the new rectangle at the start so we don't
|
|
||||||
|
|
||||||
if (newRect.hasDimensions())
|
|
||||||
{
|
|
||||||
invalidRects->push_back(newRect);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Increase iterator to compensate for insertion
|
|
||||||
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Move any rectangles from the visibleRects list that overlap this widget
|
|
||||||
* into the invisibleRects list. Used during visible region calculations.
|
|
||||||
*
|
|
||||||
* @param visibleRects A vector of regions that are not overlapped.
|
|
||||||
* @param invisibleRects A vector of regions that are overlapped.
|
|
||||||
* @param widget The widget that requested the lists.
|
|
||||||
* @see splitRectangles()
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CRectCache::removeOverlappedRects(TNxArray<CRect> *visibleRects,
|
|
||||||
TNxArray<CRect> *invisibleRects,
|
|
||||||
FAR const CNxWidget* widget) const
|
|
||||||
{
|
|
||||||
const CNxWidget* parent = m_widget;
|
|
||||||
int widgetIndex = -1;
|
|
||||||
|
|
||||||
while ((widget != NULL) && (parent != NULL))
|
|
||||||
{
|
|
||||||
// Locate widget in the list; we add one to the index to
|
|
||||||
// ensure that we deal with the next widget up in the z-order
|
|
||||||
|
|
||||||
widgetIndex = parent->getWidgetIndex(widget) + 1;
|
|
||||||
|
|
||||||
// Widget should never be the bottom item on the screen
|
|
||||||
|
|
||||||
if (widgetIndex > 0)
|
|
||||||
{
|
|
||||||
// Remove any overlapped rectangles
|
|
||||||
|
|
||||||
for (int i = widgetIndex; i < parent->getChildCount(); i++)
|
|
||||||
{
|
|
||||||
if (visibleRects->size() > 0)
|
|
||||||
{
|
|
||||||
parent->getChild(i)->getCRectCache()->splitRectangles(visibleRects, invisibleRects, widget);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (visibleRects->size() > 0)
|
|
||||||
{
|
|
||||||
widget = parent;
|
|
||||||
|
|
||||||
if (parent != NULL)
|
|
||||||
{
|
|
||||||
parent = parent->getParent();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cache the foreground regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CRectCache::cacheForegroundRegions(void)
|
|
||||||
{
|
|
||||||
if (m_foregroundInvalid)
|
|
||||||
{
|
|
||||||
// Use internal region cache to store the non-overlapped rectangles
|
|
||||||
// We will use this to clip the widget
|
|
||||||
|
|
||||||
m_foregroundRegions.clear();
|
|
||||||
|
|
||||||
// Create pointer to a vector to store the overlapped rectangles
|
|
||||||
// We can discard this later as we don't need it
|
|
||||||
|
|
||||||
TNxArray<CRect>* invisibleRects = new TNxArray<CRect>();
|
|
||||||
|
|
||||||
// Copy the clipped widget dimensions into a rect
|
|
||||||
|
|
||||||
CRect rect;
|
|
||||||
m_widget->getRectClippedToHierarchy(rect);
|
|
||||||
|
|
||||||
// Do we have a visible region left?
|
|
||||||
|
|
||||||
if (rect.hasDimensions())
|
|
||||||
{
|
|
||||||
// Add rect to list
|
|
||||||
|
|
||||||
m_foregroundRegions.push_back(rect);
|
|
||||||
|
|
||||||
// Request refresh
|
|
||||||
|
|
||||||
if (m_widget->getParent() != NULL)
|
|
||||||
{
|
|
||||||
m_widget->getParent()->getCRectCache()->removeOverlappedRects(&m_foregroundRegions, invisibleRects, m_widget);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tidy up
|
|
||||||
|
|
||||||
delete invisibleRects;
|
|
||||||
m_foregroundInvalid = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cache the background regions.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CRectCache::cacheBackgroundRegions(void)
|
|
||||||
{
|
|
||||||
// Ensure that foreground is up to date
|
|
||||||
|
|
||||||
cacheForegroundRegions();
|
|
||||||
|
|
||||||
if (m_backgroundInvalid)
|
|
||||||
{
|
|
||||||
// Cache visible regions not overlapped by children
|
|
||||||
|
|
||||||
m_backgroundRegions.clear();
|
|
||||||
|
|
||||||
// Create pointer to a vector to store the overlapped rectangles
|
|
||||||
// We can discard this later as we don't need it
|
|
||||||
|
|
||||||
TNxArray<CRect>* invisibleRects = new TNxArray<CRect>();
|
|
||||||
|
|
||||||
// Copy all foreground regions into the new vector
|
|
||||||
|
|
||||||
for (int i = 0; i < m_foregroundRegions.size(); i++)
|
|
||||||
{
|
|
||||||
m_backgroundRegions.push_back(m_foregroundRegions[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove all child rects from the visible vector
|
|
||||||
|
|
||||||
for (int i = 0; i < m_widget->getChildCount(); i++)
|
|
||||||
{
|
|
||||||
if (m_backgroundRegions.size() > 0)
|
|
||||||
{
|
|
||||||
m_widget->getChild(i)->getCRectCache()->splitRectangles(&m_backgroundRegions, invisibleRects, m_widget);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tidy up
|
|
||||||
|
|
||||||
delete invisibleRects;
|
|
||||||
m_backgroundInvalid = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
|
@ -290,44 +290,6 @@ const int CWidgetControl::getWidgetIndex(const CNxWidget *widget) const
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Redraws any dirty regions within the supplied region.
|
|
||||||
*
|
|
||||||
* @param rect The region to redraw
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CWidgetControl::eraseRect(CRect rect)
|
|
||||||
{
|
|
||||||
// Create pointer to a vector to store the invalid rectangles
|
|
||||||
|
|
||||||
TNxArray<CRect> *invalidRectangles = new TNxArray<CRect>();
|
|
||||||
|
|
||||||
if (invalidRectangles != (TNxArray<CRect> *)NULL)
|
|
||||||
{
|
|
||||||
// Add rectangle into the vector
|
|
||||||
|
|
||||||
invalidRectangles->push_back(rect);
|
|
||||||
|
|
||||||
// Refresh children
|
|
||||||
|
|
||||||
for (int i = m_widgets.size() - 1; i > -1 ; i--)
|
|
||||||
{
|
|
||||||
if (invalidRectangles->size() > 0)
|
|
||||||
{
|
|
||||||
m_widgets[i]->redrawDirty(invalidRectangles, (CNxWidget *)NULL);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tidy up
|
|
||||||
|
|
||||||
delete invalidRectangles;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Remove a controlled widget
|
* Remove a controlled widget
|
||||||
*
|
*
|
||||||
|
@ -477,7 +439,6 @@ void CWidgetControl::redrawEvent(FAR const struct nxgl_rect_s *nxRect, bool more
|
||||||
{
|
{
|
||||||
CRect rect;
|
CRect rect;
|
||||||
rect.setNxRect(nxRect);
|
rect.setNxRect(nxRect);
|
||||||
eraseRect(rect);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -279,40 +279,6 @@ void CWidgetEventHandlerList::raiseDropEvent(nxgl_coord_t x, nxgl_coord_t y)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Raise a move forward event to the event handler.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CWidgetEventHandlerList::raiseMoveForwardEvent(void)
|
|
||||||
{
|
|
||||||
if (isEnabled())
|
|
||||||
{
|
|
||||||
CWidgetEventArgs e(m_widget, 0, 0, 0, 0, KEY_CODE_NONE);
|
|
||||||
|
|
||||||
for (int i = 0; i < m_widgetEventHandlers.size(); ++i)
|
|
||||||
{
|
|
||||||
m_widgetEventHandlers.at(i)->handleMoveForwardEvent(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Raise a move backward event to the event handler.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void CWidgetEventHandlerList::raiseMoveBackwardEvent(void)
|
|
||||||
{
|
|
||||||
if (isEnabled())
|
|
||||||
{
|
|
||||||
CWidgetEventArgs e(m_widget, 0, 0, 0, 0, KEY_CODE_NONE);
|
|
||||||
|
|
||||||
for (int i = 0; i < m_widgetEventHandlers.size(); ++i)
|
|
||||||
{
|
|
||||||
m_widgetEventHandlers.at(i)->handleMoveBackwardEvent(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Raise a key press event to the event handler.
|
* Raise a key press event to the event handler.
|
||||||
*
|
*
|
||||||
|
|
Loading…
Reference in New Issue