mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-06 00:10:25 +00:00
280 lines
28 KiB
HTML
280 lines
28 KiB
HTML
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
|
|
<html>
|
|
<head>
|
|
|
|
<meta http-equiv="content-type" content="text/html; charset=ISO-8859-1">
|
|
<title>Detailed Design Template</title>
|
|
</head>
|
|
<body>
|
|
|
|
<h1><font color="#cc0000">Gecko Layout Detailed Design Document Template</font></h1>
|
|
|
|
<h1>Space Manger Detailed Design</h1>
|
|
|
|
<h2>Overview</h2>
|
|
The Space Manager and related classes and structures are an important of
|
|
the Gecko Layout system, specifically Block Layout. See the High Level
|
|
Design document for an overview of the Space Manager, and as an introduction
|
|
to the classes, structures and algorithms container in this, the Detailed
|
|
Design Document.<br>
|
|
<br>
|
|
|
|
<hr width="100%" size="2">
|
|
<h2>nsSpaceManager</h2>
|
|
The Space Manger is the central class is a group of classes that manage
|
|
the occupied and available space that exists in horizontal bands across
|
|
a canvas. The primary goal of the Space Manager is to provide information
|
|
about those bands of space to support the CSS notion of floated elements.<br>
|
|
<br>
|
|
There are three important parts to the Space Manger API: the parts that
|
|
deal with the coordinate space of the Space Manager, the parts that deal with
|
|
the regions managed by the Space Manager, and the parts that manage floater
|
|
impact intervals.<br>
|
|
<br>
|
|
The class nsSpaceManager is declared in the file <a href="http://lxr.mozilla.org/seamonkey/source/layout/base/src/nsSpaceManager.h">
|
|
nsSpaceManger.h</a>
|
|
. The class is only used in the layout module and cannot be exported
|
|
outside of that module (nor does it need to be). It is not a general
|
|
purpose class, and is not intended to be subclasses<font color="#cc0000">
|
|
.</font><br>
|
|
<br>
|
|
Here is the class declaration, taken from the source file as of 01.08.02<br>
|
|
<br>
|
|
|
|
<pre>/**<br> * Class for dealing with bands of available space. The space manager<br> * defines a coordinate space with an origin at (0, 0) that grows down<br> * and to the right.<br> */<br>class nsSpaceManager {<br>public:<br> nsSpaceManager(nsIPresShell* aPresShell, nsIFrame* aFrame);<br> ~nsSpaceManager();<br><br> void* operator new(size_t aSize);<br> void operator delete(void* aPtr, size_t aSize);<br><br> static void Shutdown();<br><br> /*<br> * Get the frame that's associated with the space manager. This frame<br> * created the space manager, and the world coordinate space is<br> * relative to this frame.<br> *<br> * You can use QueryInterface() on this frame to get any additional<br> * interfaces.<br> */<br> nsIFrame* GetFrame() const { return mFrame; }<br><br> /**<br> * Translate the current origin by the specified (dx, dy). This<br> * creates a new local coordinate space relative to the current<br> * coordinate space.<br> */<br> void Translate(nscoord aDx, nscoord aDy) { mX += aDx; mY += aDy; }<br><br> /**<br> * Returns the current translation from local coordinate space to<br> * world coordinate space. This represents the accumulated calls to<br> * Translate().<br> */<br> void GetTranslation(nscoord& aX, nscoord& aY) const { aX = mX; aY = mY; }<br><br> /**<br> * Returns the y-most of the bottommost band or 0 if there are no bands.<br> *<br> * @return PR_TRUE if there are bands and PR_FALSE if there are no bands<br> */<br> PRBool YMost(nscoord& aYMost) const;<br><br> /**<br> * Returns a band starting at the specified y-offset. The band data<br> * indicates which parts of the band are available, and which parts<br> * are unavailable<br> *<br> * The band data that is returned is in the coordinate space of the<br> * local coordinate system.<br> *<br> * The local coordinate space origin, the y-offset, and the max size<br> * describe a rectangle that's used to clip the underlying band of<br> * available space, i.e.<br> * {0, aYOffset, aMaxSize.width, aMaxSize.height} in the local<br> * coordinate space<br> *<br> * @param aYOffset the y-offset of where the band begins. The coordinate is<br> * relative to the upper-left corner of the local coordinate space<br> * @param aMaxSize the size to use to constrain the band data<br> * @param aBandData [in,out] used to return the list of trapezoids that<br> * describe the available space and the unavailable space<br> * @return NS_OK if successful and NS_ERROR_FAILURE if the band data is not<br> * not large enough. The 'count' member of the band data struct<br> * indicates how large the array of trapezoids needs to be<br> */<br> nsresult GetBandData(nscoord aYOffset,<br> const nsSize& aMaxSize,<br> nsBandData& aBandData) const;<br><br> /**<br> * Add a rectangular region of unavailable space. The space is<br> * relative to the local coordinate system.<br> *<br> * The region is tagged with a frame<br> *<br> * @param aFrame the frame used to identify the region. Must not be NULL<br> * @param aUnavailableSpace the bounding rect of the unavailable space<br> * @return NS_OK if successful<br> * NS_ERROR_FAILURE if there is already a region tagged with aFrame<br> */<br> nsresult AddRectRegion(nsIFrame* aFrame,<br> const nsRect& aUnavailableSpace);<br><br> /**<br> * Resize the rectangular region associated with aFrame by the specified<br> * deltas. The height change always applies to the bottom edge or the existing<br> * rect. You specify whether the width change applies to the left or right edge<br> *<br> * Returns NS_OK if successful, NS_ERROR_INVALID_ARG if there is no region<br> * tagged with aFrame<br> */<br> enum AffectedEdge {LeftEdge, RightEdge};<br> nsresult ResizeRectRegion(nsIFrame* aFrame,<br> nscoord aDeltaWidth,<br> nscoord aDeltaHeight,<br> AffectedEdge aEdge = RightEdge);<br><br> /**<br> * Offset the region associated with aFrame by the specified amount.<br> *<br> * Returns NS_OK if successful, NS_ERROR_INVALID_ARG if there is no region<br> * tagged with aFrame<br> */<br> nsresult OffsetRegion(nsIFrame* aFrame, nscoord dx, nscoord dy);<br><br> /**<br> * Remove the region associated with aFrane.<br> *<br> * Returns NS_OK if successful and NS_ERROR_INVALID_ARG if there is no region<br> * tagged with aFrame<br> */<br> nsresult RemoveRegion(nsIFrame* aFrame);<br><br> /**<br> * Clears the list of regions representing the unavailable space.<br> */<br> void ClearRegions();<br><br> /**<br> * Methods for dealing with the propagation of float damage during<br> * reflow.<br> */<br> PRBool HasFloatDamage()<br> {<br> return !mFloatDamage.IsEmpty();<br> }<br><br> void IncludeInDamage(nscoord aIntervalBegin, nscoord aIntervalEnd)<br> {<br> mFloatDamage.IncludeInterval(aIntervalBegin + mY, aIntervalEnd + mY);<br> }<br><br> PRBool IntersectsDamage(nscoord aIntervalBegin, nscoord aIntervalEnd)<br> {<br> return mFloatDamage.Intersects(aIntervalBegin + mY, aIntervalEnd + mY);<br> }<br><br>#ifdef DEBUG<br> /**<br> * Dump the state of the spacemanager out to a file<br> */<br> nsresult List(FILE* out);<br><br> void SizeOf(nsISizeOfHandler* aHandler, PRUint32* aResult) const;<br>#endif<br><br>private:<br> // Structure that maintains information about the region associated<br> // with a particular frame<br> struct FrameInfo {<br> nsIFrame* const mFrame;<br> nsRect mRect; // rectangular region<br> FrameInfo* mNext;<br><br> FrameInfo(nsIFrame* aFrame, const nsRect& aRect);<br>#ifdef NS_BUILD_REFCNT_LOGGING<br> ~FrameInfo();<br>#endif<br> };<br><br> // Doubly linked list of band rects<br> struct BandRect : PRCListStr {<br> nscoord mLeft, mTop;<br> nscoord mRight, mBottom;<br> PRIntn mNumFrames; // number of frames occupying this rect<br> union {<br> nsIFrame* mFrame; // single frame occupying the space<br> nsVoidArray* mFrames; // list of frames occupying the space<br> };<br><br> BandRect(nscoord aLeft, nscoord aTop,<br> nscoord aRight, nscoord aBottom,<br> nsIFrame*);<br> BandRect(nscoord aLeft, nscoord aTop,<br> nscoord aRight, nscoord aBottom,<br> nsVoidArray*);<br> ~BandRect();<br><br> // List operations<br> BandRect* Next() const {return (BandRect*)PR_NEXT_LINK(this);}<br> BandRect* Prev() const {return (BandRect*)PR_PREV_LINK(this);}<br> void InsertBefore(BandRect* aBandRect) {PR_INSERT_BEFORE(aBandRect, this);}<br> void InsertAfter(BandRect* aBandRect) {PR_INSERT_AFTER(aBandRect, this);}<br> void Remove() {PR_REMOVE_LINK(this);}<br><br> // Split the band rect into two vertically, with this band rect becoming<br> // the top part, and a new band rect being allocated and returned for the<br> // bottom part<br> //<br> // Does not insert the new band rect into the linked list<br> BandRect* SplitVertically(nscoord aBottom);<br><br> // Split the band rect into two horizontally, with this band rect becoming<br> // the left part, and a new band rect being allocated and returned for the<br> // right part<br> //<br> // Does not insert the new band rect into the linked list<br> BandRect* SplitHorizontally(nscoord aRight);<br><br> // Accessor functions<br> PRBool IsOccupiedBy(const nsIFrame*) const;<br> void AddFrame(const nsIFrame*);<br> void RemoveFrame(const nsIFrame*);<br> PRBool HasSameFrameList(const BandRect* aBandRect) const;<br> PRInt32 Length() const;<br> };<br><br> // Circular linked list of band rects<br> struct BandList : BandRect {<br> BandList();<br><br> // Accessors<br> PRBool IsEmpty() const {return PR_CLIST_IS_EMPTY((PRCListStr*)this);}<br> BandRect* Head() const {return (BandRect*)PR_LIST_HEAD(this);}<br> BandRect* Tail() const {return (BandRect*)PR_LIST_TAIL(this);}<br><br> // Operations<br> void Append(BandRect* aBandRect) {PR_APPEND_LINK(aBandRect, this);}<br><br> // Remove and delete all the band rects in the list<br> void Clear();<br> };<br><br><br> FrameInfo* GetFrameInfoFor(nsIFrame* aFrame);<br> FrameInfo* CreateFrameInfo(nsIFrame* aFrame, const nsRect& aRect);<br> void DestroyFrameInfo(FrameInfo*);<br><br> void ClearFrameInfo();<br> void ClearBandRects();<br><br> BandRect* GetNextBand(const BandRect* aBandRect) const;<br> void DivideBand(BandRect* aBand, nscoord aBottom);<br> PRBool CanJoinBands(BandRect* aBand, BandRect* aPrevBand);<br> PRBool JoinBands(BandRect* aBand, BandRect* aPrevBand);<br> void AddRectToBand(BandRect* aBand, BandRect* aBandRect);<br> void InsertBandRect(BandRect* aBandRect);<br><br> nsresult GetBandAvailableSpace(const BandRect* aBand,<br> nscoord aY,<br> const nsSize& aMaxSize,<br> nsBandData& aAvailableSpace) const;<br><br> nsIFrame* const mFrame; // frame associated with the space manager<br> nscoord mX, mY; // translation from local to global coordinate space<br> BandList mBandList; // header/sentinel for circular linked list of band rects<br> FrameInfo* mFrameInfoMap;<br> nsIntervalSet mFloatDamage;<br><br> static PRInt32 sCachedSpaceManagerCount;<br> static void* sCachedSpaceManagers[NS_SPACE_MANAGER_CACHE_SIZE];<br><br> nsSpaceManager(const nsSpaceManager&); // no implementation<br> void operator=(const nsSpaceManager&); // no implementation<br>};<br><br></pre>
|
|
|
|
<h3>Public API</h3>
|
|
|
|
<h4>Life Cycle:</h4>
|
|
The Constructor requires a Presentation Shell, used for arena allocations
|
|
mostly, and a frame that this Space Manager is rooted on. The coordinate
|
|
space of this Space Manger is relative to the frame passed in to the constructor.<br>
|
|
|
|
<pre> nsSpaceManager(nsIPresShell* aPresShell, nsIFrame* aFrame);<br> ~nsSpaceManager();<br></pre>
|
|
Operators 'new' and 'delete' are overridden to support a recycler. Space
|
|
Manager instances come and go pretty frequently, and this recycler prevents
|
|
excessive heap allocations and the performance penalties associated with
|
|
it. The #define NS_SPACE_MANAGER_CACHE_SIZE is used to control the number
|
|
of Space Manager instances that can be present in the recycler, currently
|
|
4. If more than NS_SPACE_MANAGER_CACHE_SIZE are allocated at a time,
|
|
then standard allocation is used.<br>
|
|
|
|
<pre><br> void* operator new(size_t aSize);<br> void operator delete(void* aPtr, size_t aSize);<br><br></pre>
|
|
A Static method is used to shutdown the Space Manager recycling. This
|
|
method deletes all of the Space Mangers inthe recycler,and prevents further
|
|
recycling. It is meant to be called only when the layout module is being
|
|
terminated.<br>
|
|
|
|
<pre> static void Shutdown();<br><br></pre>
|
|
|
|
<h4>Origin / Coordinate Space Translation</h4>
|
|
|
|
<pre> /**<br> * Translate the current origin by the specified (dx, dy). This<br> * creates a new local coordinate space relative to the current<br> * coordinate space.<br> */<br> void Translate(nscoord aDx, nscoord aDy) { mX += aDx; mY += aDy; }<br><br> /**<br> * Returns the current translation from local coordinate space to<br> * world coordinate space. This represents the accumulated calls to<br> * Translate().<br> */<br> void GetTranslation(nscoord& aX, nscoord& aY) const { aX = mX; aY = mY; }<br><br> /**<br> * Returns the y-most of the bottommost band or 0 if there are no bands.<br> *<br> * @return PR_TRUE if there are bands and PR_FALSE if there are no bands<br> */<br> PRBool YMost(nscoord& aYMost) const;<br></pre>
|
|
|
|
<h4>Region Management</h4>
|
|
|
|
<pre> /**<br> * Returns a band starting at the specified y-offset. The band data<br> * indicates which parts of the band are available, and which parts<br> * are unavailable<br> *<br> * The band data that is returned is in the coordinate space of the<br> * local coordinate system.<br> *<br> * The local coordinate space origin, the y-offset, and the max size<br> * describe a rectangle that's used to clip the underlying band of<br> * available space, i.e.<br> * {0, aYOffset, aMaxSize.width, aMaxSize.height} in the local<br> * coordinate space<br> *<br> * @param aYOffset the y-offset of where the band begins. The coordinate is<br> * relative to the upper-left corner of the local coordinate space<br> * @param aMaxSize the size to use to constrain the band data<br> * @param aBandData [in,out] used to return the list of trapezoids that<br> * describe the available space and the unavailable space<br> * @return NS_OK if successful and NS_ERROR_FAILURE if the band data is not<br> * not large enough. The 'count' member of the band data struct<br> * indicates how large the array of trapezoids needs to be<br> */<br> nsresult GetBandData(nscoord aYOffset,<br> const nsSize& aMaxSize,<br> nsBandData& aBandData) const;<br><br> /**<br> * Add a rectangular region of unavailable space. The space is<br> * relative to the local coordinate system.<br> *<br> * The region is tagged with a frame<br> *<br> * @param aFrame the frame used to identify the region. Must not be NULL<br> * @param aUnavailableSpace the bounding rect of the unavailable space<br> * @return NS_OK if successful<br> * NS_ERROR_FAILURE if there is already a region tagged with aFrame<br> */<br> nsresult AddRectRegion(nsIFrame* aFrame,<br> const nsRect& aUnavailableSpace);<br><br> /**<br> * Resize the rectangular region associated with aFrame by the specified<br> * deltas. The height change always applies to the bottom edge or the existing<br> * rect. You specify whether the width change applies to the left or right edge<br> *<br> * Returns NS_OK if successful, NS_ERROR_INVALID_ARG if there is no region<br> * tagged with aFrame<br> */<br> enum AffectedEdge {LeftEdge, RightEdge};<br> nsresult ResizeRectRegion(nsIFrame* aFrame,<br> nscoord aDeltaWidth,<br> nscoord aDeltaHeight,<br> AffectedEdge aEdge = RightEdge);<br><br> /**<br> * Offset the region associated with aFrame by the specified amount.<br> *<br> * Returns NS_OK if successful, NS_ERROR_INVALID_ARG if there is no region<br> * tagged with aFrame<br> */<br> nsresult OffsetRegion(nsIFrame* aFrame, nscoord dx, nscoord dy);<br><br> /**<br> * Remove the region associated with aFrane.<br> *<br> * Returns NS_OK if successful and NS_ERROR_INVALID_ARG if there is no region<br> * tagged with aFrame<br> */<br> nsresult RemoveRegion(nsIFrame* aFrame);<br><br> /**<br> * Clears the list of regions representing the unavailable space.<br> */<br> void ClearRegions();<br></pre>
|
|
|
|
<h4>Floater Impact</h4>
|
|
|
|
<pre> /**<br> * Methods for dealing with the propagation of float damage during<br> * reflow.<br> */<br> PRBool HasFloatDamage()<br> {<br> return !mFloatDamage.IsEmpty();<br> }<br><br> void IncludeInDamage(nscoord aIntervalBegin, nscoord aIntervalEnd)<br> {<br> mFloatDamage.IncludeInterval(aIntervalBegin + mY, aIntervalEnd + mY);<br> }<br><br> PRBool IntersectsDamage(nscoord aIntervalBegin, nscoord aIntervalEnd)<br> {<br> return mFloatDamage.Intersects(aIntervalBegin + mY, aIntervalEnd + mY);<br> }<br></pre>
|
|
|
|
<h4>Debug Only Methods</h4>
|
|
|
|
<pre> /**<br> * Dump the state of the spacemanager out to a file<br> */<br> nsresult List(FILE* out);<br><br> void SizeOf(nsISizeOfHandler* aHandler, PRUint32* aResult) const;<br><br></pre>
|
|
|
|
<h4>Unused / Obsolete Methods</h4>
|
|
|
|
<pre> /*<br> * Get the frame that's associated with the space manager. This frame<br> * created the space manager, and the world coordinate space is<br> * relative to this frame.<br> *<br> * You can use QueryInterface() on this frame to get any additional<br> * interfaces.<br> */<br> nsIFrame* GetFrame() const { return mFrame; }<br><br></pre>
|
|
|
|
<h3>Implementation Notes</h3>
|
|
|
|
<h4></h4>
|
|
|
|
<h4>Algorithm 1: GetBandData</h4>
|
|
GetBandData is used to provide information to clients about what space if
|
|
available and unavailable in a band of space. The client provides a
|
|
vertical offset, the yOffset, that corresponds to the band that is of interest.
|
|
This will be the y offset of the frame that is being reflowed. The
|
|
caller also provides a collection of BandData objects (an array) and the
|
|
number of items that the collection can handle. If the collection is
|
|
too small, then an error is returned and the count is updated to indicate
|
|
the size required.<br>
|
|
<br>
|
|
The algorithm to provide the band data is as follows:<br>
|
|
<ul>
|
|
<li>Get a vertical offset in world coordinates (instead of frame-relative
|
|
coordinates) by adding the y-origin of the SpaceManager to the y offset passed
|
|
in</li>
|
|
<li>If the (adjusted) y value passed in is greater than the greatest band
|
|
being managed, then all space is available so a single trapezoid is returned,
|
|
marked as available and sized to the maximum size value (passed in).</li>
|
|
<li>If the (adjusted) y offset intersects a band, then gather the band
|
|
data:</li>
|
|
<ul>
|
|
<li>walk the internal bandData list from head to tail</li>
|
|
<li>for each band data entry, see if the top of the band is greater than
|
|
the (adjusted) y offset requested</li>
|
|
<li>if it is, then band is below the offset requested, so the area between
|
|
the band and the y offset is available - create a trapezoid with that region
|
|
and return it.</li>
|
|
<li>if the (adjusted) y offset is between the band top and bottom, then
|
|
get the available space for the band by calling GetBandAvailableSpace</li>
|
|
<li>otherwise, move to the next band</li>
|
|
</ul>
|
|
</ul>
|
|
<h5>GetBandAvailableSpace:</h5>
|
|
This method is called from GetBandData only. It walks all of the bands in
|
|
the space manager and determines which bands intersect with the band passed
|
|
in, and if within those bands there are regions that are available or occupied.<br>
|
|
<ul>
|
|
<li>First, walk all of the bands until a band that is to the right of the
|
|
desired offset is located</li>
|
|
<li>Starting at that band, walk the remaining bands:</li>
|
|
<ul>
|
|
<li>if the current band is to the right of the requested band, then there
|
|
is available space. </li>
|
|
<ul>
|
|
<li>if there is more room in the bandData collection, then add a trapezoid
|
|
to the bandData collection such that it is marked as available and has a
|
|
rect that represents the space between the reference band tna dht band being
|
|
examined</li>
|
|
<li>if there is no more room in the BandData collection, estimate the
|
|
number of entries requires as the current count + twice the number of bands
|
|
below the reference band, plus two. Return an error code so the caller
|
|
can reallocate the collection and try again.</li>
|
|
</ul>
|
|
<li>check the size of the collection again, if there is no room left
|
|
then estimate the number of items requires as the current count + twice the
|
|
number of bands below the band in question plus one. </li>
|
|
<li>create a new trapezoid in the band collection that has a region corresponding
|
|
to the reference band rect, marked as occupied by either a single or multiple
|
|
frames.</li>
|
|
<li>move to the next band</li>
|
|
</ul>
|
|
<li>after walking all of the band data, se if we have reached the right
|
|
edge of the band. </li>
|
|
<ul>
|
|
<li>If not, then check for space in the band collection</li>
|
|
<ul>
|
|
<li>if there is no room left, then set the count to the current count
|
|
plus 1 and return an error.</li>
|
|
<li>otherwise, create another entry in the band collection, marked
|
|
as available, and with a rect corresponding to the area remainin in the band
|
|
(eg. from the right edge of the last band rect to the right edge of the band).</li>
|
|
</ul>
|
|
</ul>
|
|
</ul>
|
|
|
|
<h4>Algorithm 2: AddRectRegion</h4>
|
|
Clients call into this method to notify the Space Manger that a new frame
|
|
is occupying some space.<br>
|
|
<ul>
|
|
<li>First, try to get frame info for the frame. If it is found, return
|
|
an error since the frame is already associated with a region in the Space
|
|
Manager.</li>
|
|
<li>Next, create a rect from the occupied space passed in by the caller,
|
|
transforming it first to world-coordinates by adding the Space Manager's
|
|
offset to the occupied space rect passed in.</li>
|
|
<li>Create a new Frame Info instance for the frame and rect, returning
|
|
an error if allocation fails.</li>
|
|
<li>Check if the occupied space rect (adjusted) is empty, if so, return
|
|
an error (<font color="#cc0000">NOTE: this could be done earlier, or
|
|
prevented by the caller</font>)</li>
|
|
<li>Allocate a new BandRect instance with the rect and frame as constructor
|
|
arguments, and insert it into the collection via InsertBandRect</li>
|
|
</ul>
|
|
<h5>InsertBandRect:</h5>
|
|
Internal method to insert a band rect into the BandList in the correct location.
|
|
There are several cases it has to handle, as specified in the source file
|
|
comments:<br>
|
|
<pre>// When comparing a rect to a band there are seven cases to consider.<br>// 'R' is the rect and 'B' is the band.<br>//<br>// Case 1 Case 2 Case 3 Case 4<br>// ------ ------ ------ ------<br>// +-----+ +-----+ +-----+ +-----+<br>// | R | | R | +-----+ +-----+ | | | |<br>// +-----+ +-----+ | | | R | | B | | B |<br>// +-----+ | B | +-----+ | | +-----+ | |<br>// | | | | +-----+ | R | +-----+<br>// | B | +-----+ +-----+<br>// | |<br>// +-----+<br>//<br>//<br>//<br>// Case 5 Case 6 Case 7<br>// ------ ------ ------<br>// +-----+ +-----+ +-----+ +-----+<br>// | | | R | | B | | | +-----+<br>// | B | +-----+ +-----+ | R | | B |<br>// | | | | +-----+<br>// +-----+ +-----+<br>// +-----+<br>// | R |<br>// +-----+<br>//<br></pre>
|
|
<ul>
|
|
<li>First, check for the easiest case, where there are no existing band
|
|
rects, or the band rect passed in is below the bottommost rect. In this case,
|
|
just append the band rect and return.</li>
|
|
<li>Starting at the head of the list of bandRects, check for intersection
|
|
with the rect passed in:</li>
|
|
<ul>
|
|
<li>case #1: the rect is totally above the current band rect, so insert
|
|
a new band rect before the current bandRect</li>
|
|
<li>cases #2 and #7: the rect is partially above the band rect, so it
|
|
is divided into two bandRects, one entirely above the band, and one containing
|
|
the remainder of the rect. Insert the part that is totally above the
|
|
bandRect before the current bandRect, as in case #1 above, and adjust the
|
|
other band rect to exclude the part already added.</li>
|
|
<li>case #5: the rect is totally below the current bandRect, so just
|
|
skip to the next band</li>
|
|
<li>case #3 and #4: rect is at least partially intersection with the
|
|
bandRect, so divide the current band into two parts, where the top part is
|
|
above the current rect. Move to the new band just created, which is
|
|
the next band.</li>
|
|
<li>case #6: the rect shares the bottom and height with the bandRect,
|
|
so just add the rect to the band.</li>
|
|
<li>case #4 and #7: create a new rect for the part that overlaps the
|
|
bandRect, and add it to the current bandRect (similar to case #6) and then
|
|
move on to the next band, removing that part from the rect passed in. If
|
|
no more bands, append the rect passed in to the end of the bandRect list.</li>
|
|
</ul>
|
|
</ul>
|
|
<i>This algorithm is pretty confusing - basically what needs to happen is
|
|
that rects and bands need to be divided up so that complicated cases like
|
|
#2, #4, and #7, are reduced to simpler cases where the rects is totally above,
|
|
below, or between a band rect. From the current implementation, it
|
|
might be worth verifying that the final result of the inserts is a correctly
|
|
ordered liest of bandRects (debug mode only).</i><br>
|
|
|
|
<h4>Algorithm 3: RemoveRegion</h4>
|
|
When a floater is removed, the Space Manager is notified by a call to RemoveRegion,
|
|
passing in the frame that is being removed.<br>
|
|
<ul>
|
|
<li>Get the FrameInfo for the frame passed in. If not found, an error is
|
|
returned.</li>
|
|
<li>If the rect for the frame is not empty, then visit each band in the
|
|
bandList:</li>
|
|
<ul>
|
|
<li>for each rect in the band:<br>
|
|
</li>
|
|
</ul>
|
|
<ul>
|
|
<ul>
|
|
<li>if the bandRect is occupied by the frame, either remove the frame
|
|
from the bandRect (if there are other frames sharing it) and remember that
|
|
it was shared</li>
|
|
<li>otherwise simply remove the bandRect (no other frames share it).</li>
|
|
<li>if the bandRect was shared, then try to coalesce adjacent bandRects</li>
|
|
<ul>
|
|
<li>if the previous bandRect is directly next to the current bandRect,
|
|
and they have the same frame list, then make the current bandRect cover the
|
|
previous bandRect's full region (adjust the left edge to be that of the previous
|
|
bandRect) and remvoe the previous bandRect.</li>
|
|
</ul>
|
|
</ul>
|
|
</ul>
|
|
<ul>
|
|
<li>if the current band or prior band had a rect occupied byu the frame,
|
|
then try to join the two bands via JoinBands</li>
|
|
</ul>
|
|
<li>Finally, destroy the frameInfo for the frame.<br>
|
|
</li>
|
|
</ul>
|
|
<br>
|
|
|
|
<hr width="100%" size="2">
|
|
<h2>Cross-Component Algorithms</h2>
|
|
<br>
|
|
<br>
|
|
|
|
<hr width="100%" size="2">
|
|
<h2>Tech Notes</h2>
|
|
<ul>
|
|
<li><br>
|
|
</li>
|
|
|
|
</ul>
|
|
<br>
|
|
<br>
|
|
|
|
</body>
|
|
</html>
|