WindowsXP-SP1/shell/comctl32/v6/selrange.cpp
2020-09-30 16:53:49 +02:00

1166 lines
30 KiB
C++

//-------------------------------------------------------------------
//
// File: SelRange.cpp
//
// Contents:
// This file contians Selection Range handling code.
//
//-------------------------------------------------------------------
#include "ctlspriv.h"
#include "selrange.h"
#include "stdio.h"
#include <shguidp.h>
#define MINCOUNT 6 // number of sel ranges to start with amd maintain
#define GROWSIZE 150 // percent to grow when needed
#define COUNT_SELRANGES_NONE 2 // When count of selranges really means none
typedef struct tag_SELRANGEITEM
{
LONG iBegin;
LONG iEnd;
} SELRANGEITEM, *PSELRANGEITEM;
class CLVRange : public ILVRange
{
public:
// *** IUnknown methods ***
STDMETHODIMP QueryInterface(REFIID riid, LPVOID * ppv);
STDMETHODIMP_(ULONG) AddRef(void);
STDMETHODIMP_(ULONG) Release(void);
// *** ILVRange methods ***
STDMETHODIMP IncludeRange(LONG iBegin, LONG iEnd);
STDMETHODIMP ExcludeRange(LONG iBegin, LONG iEnd);
STDMETHODIMP InvertRange(LONG iBegin, LONG iEnd);
STDMETHODIMP InsertItem(LONG iItem);
STDMETHODIMP RemoveItem(LONG iItem);
STDMETHODIMP Clear();
STDMETHODIMP IsSelected(LONG iItem);
STDMETHODIMP IsEmpty();
STDMETHODIMP NextSelected(LONG iItem, LONG *piItem);
STDMETHODIMP NextUnSelected(LONG iItem, LONG *piItem);
STDMETHODIMP CountIncluded(LONG *pcIncluded);
protected:
// Helper Functions.
friend ILVRange *LVRange_Create();
CLVRange();
~CLVRange();
BOOL _Enlarge();
BOOL _Shrink();
BOOL _InsertRange(LONG iAfterItem, LONG iBegin, LONG iEnd);
HRESULT _RemoveRanges(LONG iStartItem, LONG iStopItem, LONG *p);
BOOL _FindValue(LONG Value, LONG* piItem);
void _InitNew();
int _cRef;
PSELRANGEITEM _VSelRanges; // Vector of sel ranges
LONG _cSize; // size of above vector in sel ranges
LONG _cSelRanges; // count of sel ranges used
LONG _cIncluded; // Count of Included items...
};
//-------------------------------------------------------------------
//
// Function: _Enlarge
//
// Summary:
// This will enlarge the number of items the Sel Range can have.
//
// Arguments:
// PSELRANGE [in] - SelRange to Enlarge
//
// Return: FALSE if failed.
//
// Notes: Though this function may fail, pselrange structure is still valid
//
// History:
// 17-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
BOOL CLVRange::_Enlarge()
{
LONG cNewSize;
PSELRANGEITEM pTempSelRange;
BOOL frt = FALSE;
cNewSize = _cSize * GROWSIZE / 100;
pTempSelRange = (PSELRANGEITEM) GlobalReAlloc( (HGLOBAL)_VSelRanges,
cNewSize * sizeof( SELRANGEITEM ),
GMEM_ZEROINIT | GMEM_MOVEABLE );
if (NULL != pTempSelRange)
{
_VSelRanges = pTempSelRange;
_cSize = cNewSize;
frt = TRUE;
}
return( frt );
}
//-------------------------------------------------------------------
//
// Function: _Shrink
//
// Summary:
// This will reduce the number of items the Sel Range can have.
//
// Arguments:
//
// Return: FALSE if failed
//
// Notes: Shrink only happens when a significant size below the next size
// is obtained and the new size is at least the minimum size.
// Though this function may fail, pselrange structure is still valid
//
// History:
// 17-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
BOOL CLVRange::_Shrink()
{
LONG cNewSize;
LONG cTriggerSize;
PSELRANGEITEM pTempSelRange;
BOOL frt = TRUE;
// check if we are below last grow area by a small percent
cTriggerSize = _cSize * 90 / GROWSIZE;
cNewSize = _cSize * 100 / GROWSIZE;
if ((_cSelRanges < cTriggerSize) && (cNewSize >= MINCOUNT))
{
pTempSelRange = (PSELRANGEITEM) GlobalReAlloc( (HGLOBAL)_VSelRanges,
cNewSize * sizeof( SELRANGEITEM ),
GMEM_ZEROINIT | GMEM_MOVEABLE );
if (NULL != pTempSelRange)
{
_VSelRanges = pTempSelRange;
_cSize = cNewSize;
}
else
{
frt = FALSE;
}
}
return( frt );
}
//-------------------------------------------------------------------
//
// Function: _InsertRange
//
// Summary:
// inserts a single range item into the range vector
//
// Arguments:
// iAfterItem [in] - Index to insert range after, -1 means insert as first item
// iBegin [in] - begin of range
// iEnd [in] - end of the range
//
// Return:
// TRUE if succesful, otherwise FALSE
//
// Notes:
//
// History:
// 17-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
BOOL CLVRange::_InsertRange(LONG iAfterItem,
LONG iBegin,
LONG iEnd )
{
LONG iItem;
BOOL frt = TRUE;
ASSERT( iAfterItem >= -1 );
ASSERT( iBegin >= SELRANGE_MINVALUE );
ASSERT( iEnd >= iBegin );
ASSERT( iEnd <= SELRANGE_MAXVALUE );
ASSERT( _cSelRanges < _cSize );
// shift all over one
for (iItem = _cSelRanges; iItem > iAfterItem + 1; iItem--)
{
_VSelRanges[iItem] = _VSelRanges[iItem-1];
}
_cSelRanges++;
// make the insertion
_VSelRanges[iAfterItem+1].iBegin = iBegin;
_VSelRanges[iAfterItem+1].iEnd = iEnd;
// make sure we have room next time
if (_cSelRanges == _cSize)
{
frt = _Enlarge();
}
return( frt );
}
//-------------------------------------------------------------------
//
// Function: _RemoveRanges
//
// Summary:
// Removes all ranged between and including the speicifed indexes
//
// Arguments:
// iStartItem [in] - Index to start removal
// iStopItem [in] - Index to stop removal
//
// Return:
// SELRANGE_ERROR on memory allocation error
// The number of items that are unselected by this removal
//
// Notes:
//
// History:
// 17-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::_RemoveRanges(LONG iStartItem, LONG iStopItem, LONG *pc )
{
LONG iItem;
LONG diff;
LONG cUnSelected = 0;
HRESULT hres = S_OK;
ASSERT( iStartItem > 0 );
ASSERT( iStopItem >= iStartItem );
ASSERT( iStartItem < _cSelRanges - 1 );
ASSERT( iStopItem < _cSelRanges - 1 );
diff = iStopItem - iStartItem + 1;
for (iItem = iStartItem; iItem <= iStopItem; iItem++)
cUnSelected += _VSelRanges[iItem].iEnd -
_VSelRanges[iItem].iBegin + 1;
// shift all over the difference
for (iItem = iStopItem+1; iItem < _cSelRanges; iItem++, iStartItem++)
_VSelRanges[iStartItem] = _VSelRanges[iItem];
_cSelRanges -= diff;
if (!_Shrink())
{
hres = E_FAIL;
}
else if (pc)
*pc = cUnSelected;
return( hres );
}
//-------------------------------------------------------------------
//
// Function: SelRange_FindValue
//
// Summary:
// This function will search the ranges for the value, returning true
// if the value was found within a range. The piItem will contain the
// the index at which it was found or the index before where it should be
// The piItem may be set to -1, meaning that there are no ranges in the list
// This functions uses a non-recursive binary search algorithm.
//
// Arguments:
// piItem [out] - Return of found range index, or one before
// Value [in] - Value to find within a range
//
// Return: True if found, False if not found
//
// Notes: The piItem will return one before if return is false.
//
// History:
// 14-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
BOOL CLVRange::_FindValue(LONG Value, LONG* piItem )
{
LONG First;
LONG Last;
LONG Item;
BOOL fFound = FALSE;
ASSERT( piItem );
ASSERT( _cSize >= COUNT_SELRANGES_NONE );
ASSERT( Value >= SELRANGE_MINVALUE );
ASSERT( Value <= SELRANGE_MAXVALUE );
First = 0;
Last = _cSelRanges - 1;
Item = Last / 2;
do
{
if (_VSelRanges[Item].iBegin > Value)
{ // Value before this Item
Last = Item;
Item = (Last - First) / 2 + First;
if (Item == Last)
{
Item = First;
break;
}
}
else if (_VSelRanges[Item].iEnd < Value)
{ // Value after this Item
First = Item;
Item = (Last - First) / 2 + First;
if (Item == First)
{
break;
}
}
else
{ // Value at this Item
fFound = TRUE;
}
} while (!fFound);
*piItem = Item;
return( fFound );
}
//-------------------------------------------------------------------
//
// Function: _InitNew
//
// Summary:
// This function will initialize a SelRange object.
//
// Arguments:
//
// Return:
//
// Notes:
//
// History:
// 18-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
void CLVRange::_InitNew()
{
_cSize = MINCOUNT;
_cSelRanges = COUNT_SELRANGES_NONE;
_VSelRanges[0].iBegin = LONG_MIN;
// -2 and +2 below are to stop consecutive joining of end markers
_VSelRanges[0].iEnd = SELRANGE_MINVALUE - 2;
_VSelRanges[1].iBegin = SELRANGE_MAXVALUE + 2;
_VSelRanges[1].iEnd = SELRANGE_MAXVALUE + 2;
_cIncluded = 0;
}
//-------------------------------------------------------------------
//
// Function: SelRange_Create
//
// Summary:
// This function will create and initialize a SelRange object.
//
// Arguments:
//
// Return: HSELRANGE that is created or NULL if it failed.
//
// Notes:
//
// History:
// 14-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
ILVRange *LVRange_Create( )
{
CLVRange *pselrange = new CLVRange;
if (NULL != pselrange)
{
pselrange->_VSelRanges = (PSELRANGEITEM) GlobalAlloc( GPTR,
sizeof( SELRANGEITEM ) * MINCOUNT );
if (NULL != pselrange->_VSelRanges)
{
pselrange->_InitNew();
}
else
{
delete pselrange;
pselrange = NULL;
}
}
return( pselrange? SAFECAST(pselrange, ILVRange*) : NULL);
}
//-------------------------------------------------------------------
//
// Function: Constructor
//
//-------------------------------------------------------------------
CLVRange::CLVRange()
{
_cRef = 1;
}
//-------------------------------------------------------------------
//
// Function: Destructor
//
//-------------------------------------------------------------------
CLVRange::~CLVRange()
{
GlobalFree( _VSelRanges );
}
//-------------------------------------------------------------------
//
// Function: QueryInterface
//
//-------------------------------------------------------------------
HRESULT CLVRange::QueryInterface(REFIID iid, void **ppv)
{
if (IsEqualIID(iid, IID_ILVRange) || IsEqualIID(iid, IID_IUnknown))
{
*ppv = SAFECAST(this, ILVRange *);
}
else
{
*ppv = NULL;
return E_NOINTERFACE;
}
_cRef++;
return NOERROR;
}
//-------------------------------------------------------------------
//
// Function: AddRef
//
//-------------------------------------------------------------------
ULONG CLVRange::AddRef()
{
return ++_cRef;
}
//-------------------------------------------------------------------
//
// Function: Release
//
//-------------------------------------------------------------------
ULONG CLVRange::Release()
{
if (--_cRef)
return _cRef;
delete this;
return 0;
}
//-------------------------------------------------------------------
//
// Function: IncludeRange
//
// Summary:
// This function will include the range defined into the current
// ranges, compacting as needed.
//
// Arguments:
// hselrange [in] - Handle to the SelRange
// iBegin [in] - Begin of new range
// iEnd [in] - End of new range
//
// Notes:
//
// History:
// 14-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::IncludeRange(LONG iBegin, LONG iEnd )
{
LONG iFirst; // index before or contains iBegin value
LONG iLast; // index before or contains iEnd value
BOOL fExtendFirst; // do we extend the iFirst or create one after it
LONG iRemoveStart; // start of ranges that need to be removed
LONG iRemoveFinish; // end of ranges that need to be removed
LONG iNewEnd; // calculate new end value as we go
BOOL fEndFound; // was the iEnd found in a range already
BOOL fBeginFound; // was the iEnd found in a range already
LONG cSelected = 0;
HRESULT hres = S_OK;
ASSERT( iEnd >= iBegin );
ASSERT( iBegin >= SELRANGE_MINVALUE );
ASSERT( iEnd <= SELRANGE_MAXVALUE );
// find approximate locations
fBeginFound = _FindValue( iBegin, &iFirst );
fEndFound = _FindValue( iEnd, &iLast );
//
// Find First values
//
// check for consecutive End-First values
if ((_VSelRanges[iFirst].iEnd == iBegin - 1) ||
(fBeginFound))
{
// extend iFirst
fExtendFirst = TRUE;
iRemoveStart = iFirst + 1;
}
else
{
// create one after the iFirst
fExtendFirst = FALSE;
iRemoveStart = iFirst + 2;
}
//
// Find Last values
//
if (fEndFound)
{
// Use [iLast].iEnd value
iRemoveFinish = iLast;
iNewEnd = _VSelRanges[iLast].iEnd;
}
else
{
// check for consecutive First-End values
if (_VSelRanges[iLast + 1].iBegin == iEnd + 1)
{
// Use [iLast + 1].iEnd value
iNewEnd = _VSelRanges[iLast+1].iEnd;
iRemoveFinish = iLast + 1;
}
else
{
// Use iEnd value
iRemoveFinish = iLast;
iNewEnd = iEnd;
}
}
//
// remove condenced items if needed
//
if (iRemoveStart <= iRemoveFinish)
{
LONG cChange;
hres = _RemoveRanges(iRemoveStart, iRemoveFinish, &cChange );
if (FAILED(hres))
return hres;
else
{
cSelected -= cChange;
}
}
//
// insert item and reset values as needed
//
if (fExtendFirst)
{
cSelected += iNewEnd - _VSelRanges[iFirst].iEnd;
_VSelRanges[iFirst].iEnd = iNewEnd;
}
else
{
if (iRemoveStart > iRemoveFinish + 1)
{
cSelected += iEnd - iBegin + 1;
// create one
if (!_InsertRange(iFirst, iBegin, iNewEnd ))
{
hres = E_FAIL;
}
}
else
{
cSelected += iNewEnd - _VSelRanges[iFirst+1].iEnd;
cSelected += _VSelRanges[iFirst+1].iBegin - iBegin;
// no need to create one since the Removal would have left us one
_VSelRanges[iFirst+1].iEnd = iNewEnd;
_VSelRanges[iFirst+1].iBegin = iBegin;
}
}
_cIncluded += cSelected;
return( hres );
}
//-------------------------------------------------------------------
//
// Function: SelRange_ExcludeRange
//
// Summary:
// This function will exclude the range defined from the current
// ranges, compacting and enlarging as needed.
//
// Arguments:
// hselrange [in] - Handle to the SelRange
// iBegin [in] - Begin of range to remove
// iEnd [in] - End of range to remove
//
// Return:
// SELRANGE_ERROR if memory allocation error
// the number actual items that changed state
//
// Notes:
//
// History:
// 17-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::ExcludeRange( LONG iBegin, LONG iEnd )
{
LONG iFirst; // index before or contains iBegin value
LONG iLast; // index before or contains iEnd value
LONG iRemoveStart; // start of ranges that need to be removed
LONG iRemoveFinish; // end of ranges that need to be removed
LONG iFirstNewEnd; // calculate new end value as we go
BOOL fBeginFound; // was the iBegin found in a range already
BOOL fEndFound; // was the iEnd found in a range already
LONG cUnSelected = 0;
HRESULT hres = S_OK;
ASSERT( iEnd >= iBegin );
ASSERT( iBegin >= SELRANGE_MINVALUE );
ASSERT( iEnd <= SELRANGE_MAXVALUE );
// find approximate locations
fBeginFound = _FindValue( iBegin, &iFirst );
fEndFound = _FindValue( iEnd, &iLast );
//
// Find First values
//
// start removal after first
iRemoveStart = iFirst + 1;
// save FirstEnd as we may need to modify it
iFirstNewEnd = _VSelRanges[iFirst].iEnd;
if (fBeginFound)
{
// check for complete removal of first
// (first is a single selection or match?)
if (_VSelRanges[iFirst].iBegin == iBegin)
{
iRemoveStart = iFirst;
}
else
{
// otherwise truncate iFirst
iFirstNewEnd = iBegin - 1;
}
}
//
// Find Last values
//
// end removal on last
iRemoveFinish = iLast;
if (fEndFound)
{
// check for complete removal of last
// (first/last is a single selection or match?)
if (_VSelRanges[iLast].iEnd != iEnd)
{
if (iFirst == iLast)
{
// split
if (!_InsertRange(iFirst, iEnd + 1, _VSelRanges[iFirst].iEnd ))
{
return( E_FAIL );
}
cUnSelected -= _VSelRanges[iFirst].iEnd - iEnd;
}
else
{
// truncate Last
iRemoveFinish = iLast - 1;
cUnSelected += (iEnd + 1) - _VSelRanges[iLast].iBegin;
_VSelRanges[iLast].iBegin = iEnd + 1;
}
}
}
// Now set the new end, since Last code may have needed the original values
cUnSelected -= iFirstNewEnd - _VSelRanges[iFirst].iEnd;
_VSelRanges[iFirst].iEnd = iFirstNewEnd;
//
// remove items if needed
//
if (iRemoveStart <= iRemoveFinish)
{
LONG cChange;
if (SUCCEEDED(hres = _RemoveRanges(iRemoveStart, iRemoveFinish, &cChange )))
cUnSelected += cChange;
}
_cIncluded -= cUnSelected;
return( hres );
}
//-------------------------------------------------------------------
//
// Function: SelRange_Clear
//
// Summary:
// This function will remove all ranges within the SelRange object.
//
// Arguments:
// hselrange [in] - the hselrange object to clear
//
// Return: FALSE if failed.
//
// Notes:
// This function may return FALSE on memory allocation problems, but
// will leave the SelRange object in the last state before this call.
//
// History:
// 14-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::Clear()
{
PSELRANGEITEM pNewItems;
HRESULT hres = S_OK;
pNewItems = (PSELRANGEITEM) GlobalAlloc( GPTR,
sizeof( SELRANGEITEM ) * MINCOUNT );
if (NULL != pNewItems)
{
GlobalFree( _VSelRanges );
_VSelRanges = pNewItems;
_InitNew();
}
else
{
hres = E_FAIL;
}
return( hres );
}
//-------------------------------------------------------------------
//
// Function: SelRange_IsSelected
//
// Summary:
// This function will return if the value iItem is within a
// selected range.
//
// Arguments:
// hselrange [in] - the hselrange object to use
// iItem [in] - value to check for
//
// Return: TRUE if selected, FALSE if not.
//
// Notes:
//
// History:
// 17-Oct-94 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::IsSelected( LONG iItem )
{
LONG iFirst;
ASSERT( iItem >= 0 );
ASSERT( iItem <= SELRANGE_MAXVALUE );
return( _FindValue( iItem, &iFirst ) ? S_OK : S_FALSE);
}
//-------------------------------------------------------------------
//
// Function: SelRange_IsEmpty
//
// Summary:
// This function will return TRUE if the range is empty
//
// Arguments:
// hselrange [in] - the hselrange object to use
//
// Return: TRUE if empty
//
// Notes:
//
// History:
//
//-------------------------------------------------------------------
HRESULT CLVRange::IsEmpty()
{
return (_cSelRanges == COUNT_SELRANGES_NONE)? S_OK : S_FALSE;
}
HRESULT CLVRange::CountIncluded(LONG *pcIncluded)
{
*pcIncluded = _cIncluded;
return S_OK;
}
//-------------------------------------------------------------------
//
// Function: SelRange_InsertItem
//
// Summary:
// This function will insert a unselected item at the location,
// which will push all selections up one index.
//
// Arguments:
// hselrange [in] - the hselrange object to use
// iItem [in] - value to check for
//
// Return:
// False on memory allocation error
// otherwise TRUE
//
// Notes:
//
// History:
// 20-Dec-94 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::InsertItem( LONG iItem )
{
LONG iFirst;
LONG i;
LONG iBegin;
LONG iEnd;
ASSERT( iItem >= 0 );
ASSERT( iItem <= SELRANGE_MAXVALUE );
if (_FindValue( iItem, &iFirst ) )
{
// split it
if ( _VSelRanges[iFirst].iBegin == iItem )
{
// but don't split if starts with value
iFirst--;
}
else
{
if (!_InsertRange(iFirst, iItem, _VSelRanges[iFirst].iEnd ))
{
return( E_FAIL );
}
_VSelRanges[iFirst].iEnd = iItem - 1;
}
}
// now walk all ranges past iFirst, incrementing all values by one
for (i = _cSelRanges-2; i > iFirst; i--)
{
iBegin = _VSelRanges[i].iBegin;
iEnd = _VSelRanges[i].iEnd;
iBegin = min( SELRANGE_MAXVALUE, iBegin + 1 );
iEnd = min( SELRANGE_MAXVALUE, iEnd + 1 );
_VSelRanges[i].iBegin = iBegin;
_VSelRanges[i].iEnd = iEnd;
}
return( S_OK );
}
//-------------------------------------------------------------------
//
// Function: SelRange_RemoveItem
//
// Summary:
// This function will remove an item at the location,
// which will pull all selections down one index.
//
// Arguments:
// hselrange [in] - the hselrange object to use
// iItem [in] - value to check for
// pfWasSelected [out] - was the removed item selected before the removal
//
// Return:
// TRUE if the item was removed
// FALSE if the an error happend
//
// Notes:
//
// History:
// 20-Dec-94 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::RemoveItem(LONG iItem )
{
LONG iFirst;
LONG i;
LONG iBegin;
LONG iEnd;
HRESULT hres = S_OK;
ASSERT( iItem >= SELRANGE_MINVALUE );
ASSERT( iItem <= SELRANGE_MAXVALUE );
if (_FindValue( iItem, &iFirst ) )
{
// item within, change the end value
iEnd = _VSelRanges[iFirst].iEnd;
iEnd = min( SELRANGE_MAXVALUE, iEnd - 1 );
_VSelRanges[iFirst].iEnd = iEnd;
_cIncluded--;
}
else
{
// check for merge situation
if ((iFirst < _cSelRanges - 1) &&
(_VSelRanges[iFirst].iEnd == iItem - 1) &&
(_VSelRanges[iFirst+1].iBegin == iItem + 1))
{
_VSelRanges[iFirst].iEnd =
_VSelRanges[iFirst + 1].iEnd - 1;
if (FAILED(hres = _RemoveRanges(iFirst + 1, iFirst + 1, NULL )))
return( hres );
}
}
// now walk all ranges past iFirst, decrementing all values by one
for (i = _cSelRanges-2; i > iFirst; i--)
{
iBegin = _VSelRanges[i].iBegin;
iEnd = _VSelRanges[i].iEnd;
iBegin = min( SELRANGE_MAXVALUE, iBegin - 1 );
iEnd = min( SELRANGE_MAXVALUE, iEnd - 1 );
_VSelRanges[i].iBegin = iBegin;
_VSelRanges[i].iEnd = iEnd;
}
return( hres );
}
//-------------------------------------------------------------------
//
// Function: NextSelected
//
// Summary:
// This function will start with given item and find the next
// item that is selected. If the given item is selected, that
// item number will be returned.
//
// Arguments:
// hselrange [in] - the hselrange object to use
// iItem [in] - value to start check at
//
// Return:
// -1 if none found, otherwise the item
//
// Notes:
//
// History:
// 04-Jan-95 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::NextSelected( LONG iItem, LONG *piItem )
{
LONG i;
ASSERT( iItem >= SELRANGE_MINVALUE );
ASSERT( iItem <= SELRANGE_MAXVALUE );
if (!_FindValue( iItem, &i ) )
{
i++;
if (i < _cSelRanges-1)
{
iItem = _VSelRanges[i].iBegin;
}
else
{
iItem = -1;
}
}
ASSERT( iItem >= -1 );
ASSERT( iItem <= SELRANGE_MAXVALUE );
*piItem = iItem;
return S_OK;
}
//-------------------------------------------------------------------
//
// Function: NextUnSelected
//
// Summary:
// This function will start with given item and find the next
// item that is not selected. If the given item is not selected, that
// item number will be returned.
//
// Arguments:
// hselrange [in] - the hselrange object to use
// iItem [in] - value to start check at
//
// Return:
// -1 if none found, otherwise the item
//
// Notes:
//
// History:
// 04-Jan-95 MikeMi Created
//
//-------------------------------------------------------------------
HRESULT CLVRange::NextUnSelected( LONG iItem, LONG *piItem )
{
LONG i;
ASSERT( iItem >= SELRANGE_MINVALUE );
ASSERT( iItem <= SELRANGE_MAXVALUE );
if (_FindValue( iItem, &i ) )
{
if (i < _cSelRanges-1)
{
iItem = _VSelRanges[i].iEnd + 1;
if (iItem > SELRANGE_MAXVALUE)
{
iItem = -1;
}
}
else
{
iItem = -1;
}
}
ASSERT( iItem >= -1 );
ASSERT( iItem <= SELRANGE_MAXVALUE );
*piItem = iItem;
return S_OK;
}
//-------------------------------------------------------------------
//
// Function: InvertRange
//
// Summary:
// This function will invert the range defined from the current
// ranges, compacting and enlarging as needed.
//
// Arguments:
// iBegin [in] - Begin of range to invert
// iEnd [in] - End of range to invert
//
// Return:
// SELRANGE_ERROR on memory error
// The difference in items selected from previous to current.
// negative values means less items are selected in that range now.
//
// Notes:
//
// History:
// 13-Dec-95 MikeMi Created
//
//-------------------------------------------------------------------
LONG CLVRange::InvertRange( LONG iBegin, LONG iEnd )
{
LONG iFirst; // index before or contains iBegin value
BOOL fSelect; // are we selecting or unselecting
LONG iTempE;
LONG iTempB;
HRESULT hres = S_OK;
ASSERT( iEnd >= iBegin );
ASSERT( iBegin >= SELRANGE_MINVALUE );
ASSERT( iEnd <= SELRANGE_MAXVALUE );
// find if first is selected or not
fSelect = !_FindValue( iBegin, &iFirst );
iTempE = iBegin - 1;
do
{
iTempB = iTempE + 1;
if (fSelect)
NextSelected( iTempB, &iTempE );
else
NextUnSelected( iTempB, &iTempE );
if (-1 == iTempE)
{
iTempE = SELRANGE_MAXVALUE;
}
else
{
iTempE--;
}
iTempE = min( iTempE, iEnd );
if (fSelect)
{
if (FAILED(hres = IncludeRange( iTempB, iTempE )))
{
return( hres );
}
}
else
{
if (FAILED(hres = ExcludeRange( iTempB, iTempE )))
{
return( hres );
}
}
fSelect = !fSelect;
} while (iTempE < iEnd );
return( hres );
}