|
|
|
/*==LICENSE==*
|
|
|
|
|
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools
|
|
|
|
Copyright (C) 2011 Cyan Worlds, Inc.
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
Additional permissions under GNU GPL version 3 section 7
|
|
|
|
|
|
|
|
If you modify this Program, or any covered work, by linking or
|
|
|
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
|
|
|
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
|
|
|
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
|
|
|
|
(or a modified version of those libraries),
|
|
|
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
|
|
|
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
|
|
|
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
|
|
|
|
licensors of this Program grant you additional
|
|
|
|
permission to convey the resulting work. Corresponding Source for a
|
|
|
|
non-source form of such a combination shall include the source code for
|
|
|
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
|
|
|
|
work.
|
|
|
|
|
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
|
|
|
or by snail mail at:
|
|
|
|
Cyan Worlds, Inc.
|
|
|
|
14617 N Newport Hwy
|
|
|
|
Mead, WA 99021
|
|
|
|
|
|
|
|
*==LICENSE==*/
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
// //
|
|
|
|
// pfGUIListBoxMod Definition //
|
|
|
|
// //
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#include "hsTypes.h"
|
|
|
|
#include "pfGUIListBoxMod.h"
|
|
|
|
#include "pfGUIListElement.h"
|
|
|
|
#include "pfGameGUIMgr.h"
|
|
|
|
#include "pfGUIUpDownPairMod.h"
|
|
|
|
#include "pfGUIControlHandlers.h"
|
|
|
|
#include "pfGUIDialogMod.h"
|
|
|
|
|
|
|
|
#include "pnMessage/plRefMsg.h"
|
|
|
|
#include "pfMessage/pfGameGUIMsg.h"
|
|
|
|
#include "plMessage/plAnimCmdMsg.h"
|
|
|
|
#include "plAvatar/plAGModifier.h"
|
|
|
|
#include "plGImage/plDynamicTextMap.h"
|
|
|
|
#include "plInputCore/plInputInterface.h"
|
|
|
|
#include "plgDispatch.h"
|
|
|
|
#include "hsResMgr.h"
|
|
|
|
|
|
|
|
|
|
|
|
#define kIndentAmount 16
|
|
|
|
|
|
|
|
|
|
|
|
//// plSmallRect Stuff ///////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::plSmallRect::Set( int16_t l, int16_t t, int16_t r, int16_t b )
|
|
|
|
{
|
|
|
|
fLeft = l;
|
|
|
|
fTop = t;
|
|
|
|
fRight = r;
|
|
|
|
fBottom = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool pfGUIListBoxMod::plSmallRect::Contains( int16_t x, int16_t y )
|
|
|
|
{
|
|
|
|
if( x < fLeft || x > fRight || y < fTop || y > fBottom )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// Wee Little Control Proc for scrolling ///////////////////////////////////
|
|
|
|
|
|
|
|
class pfScrollProc : public pfGUICtrlProcObject
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
|
|
|
|
pfGUIListBoxMod *fParent;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
pfScrollProc( pfGUIListBoxMod *parent ) : fParent( parent ) {}
|
|
|
|
|
|
|
|
virtual void DoSomething( pfGUIControlMod *ctrl )
|
|
|
|
{
|
|
|
|
// Do a check here to make sure we actually changed scroll
|
|
|
|
// positions--if not, we don't want to update, since that'll be
|
|
|
|
// slow as hell
|
|
|
|
int newScrollPos = (int)fParent->fScrollControl->GetMax() - (int)fParent->fScrollControl->GetCurrValue();
|
|
|
|
if( newScrollPos != fParent->fScrollPos )
|
|
|
|
{
|
|
|
|
fParent->IUpdate();
|
|
|
|
fParent->HandleExtendedEvent( pfGUIListBoxMod::kScrollPosChanged );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
//// Constructor/Destructor //////////////////////////////////////////////////
|
|
|
|
|
|
|
|
pfGUIListBoxMod::pfGUIListBoxMod()
|
|
|
|
{
|
|
|
|
SetFlag( kWantsInterest );
|
|
|
|
fCurrClick = -1;
|
|
|
|
fCurrHover = -1;
|
|
|
|
fModsAtDragTime = 0;
|
|
|
|
fScrollControl = nil;
|
|
|
|
fCheckScroll = false;
|
|
|
|
fSingleSelElement = -1;
|
|
|
|
fScrollRangeUpdateDeferred = false;
|
|
|
|
fScrollPos = 0;
|
|
|
|
fLocked = false;
|
|
|
|
fReadyToRoll = false;
|
|
|
|
fClicking = false;
|
|
|
|
|
|
|
|
fScrollProc = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
pfGUIListBoxMod::~pfGUIListBoxMod()
|
|
|
|
{
|
|
|
|
ClearAllElements();
|
|
|
|
|
|
|
|
if( fScrollProc && fScrollProc->DecRef() )
|
|
|
|
delete fScrollProc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IEval ///////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
hsBool pfGUIListBoxMod::IEval( double secs, float del, uint32_t dirty )
|
|
|
|
{
|
|
|
|
return pfGUIControlMod::IEval( secs, del, dirty );
|
|
|
|
}
|
|
|
|
|
|
|
|
//// MsgReceive //////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
hsBool pfGUIListBoxMod::MsgReceive( plMessage *msg )
|
|
|
|
{
|
|
|
|
plGenRefMsg *refMsg = plGenRefMsg::ConvertNoRef( msg );
|
|
|
|
if( refMsg != nil )
|
|
|
|
{
|
|
|
|
if( refMsg->fType == kRefScrollCtrl )
|
|
|
|
{
|
|
|
|
if( refMsg->GetContext() & ( plRefMsg::kOnCreate | plRefMsg::kOnRequest | plRefMsg::kOnReplace ) )
|
|
|
|
{
|
|
|
|
fScrollControl = pfGUIValueCtrl::ConvertNoRef( refMsg->GetRef() );
|
|
|
|
fScrollControl->SetHandler( fScrollProc );
|
|
|
|
fScrollControl->SetStep( 1.f );
|
|
|
|
ICalcScrollRange();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
fScrollControl = nil;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if( refMsg->fType == kRefDynTextMap )
|
|
|
|
{
|
|
|
|
// Capture this so we can reset our flag, but do NOT just return from it!
|
|
|
|
if( !( refMsg->GetContext() & ( plRefMsg::kOnCreate | plRefMsg::kOnRequest | plRefMsg::kOnReplace ) ) )
|
|
|
|
fReadyToRoll = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pfGUIControlMod::MsgReceive( msg );
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IPostSetUpDynTextMap ////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::IPostSetUpDynTextMap( void )
|
|
|
|
{
|
|
|
|
ICalcWrapStarts();
|
|
|
|
ICalcScrollRange();
|
|
|
|
fReadyToRoll = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IUpdate /////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::IUpdate( void )
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
uint16_t x, y, width, height, maxHeight;
|
|
|
|
|
|
|
|
|
|
|
|
if( !fReadyToRoll )
|
|
|
|
return;
|
|
|
|
if( fScrollRangeUpdateDeferred )
|
|
|
|
{
|
|
|
|
fScrollRangeUpdateDeferred = false;
|
|
|
|
ICalcScrollRange();
|
|
|
|
}
|
|
|
|
|
|
|
|
fDynTextMap->ClearToColor( GetColorScheme()->fBackColor );
|
|
|
|
if( fElements.GetCount() == 0 )
|
|
|
|
{
|
|
|
|
fDynTextMap->FlushToHost();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( fLocked )
|
|
|
|
{
|
|
|
|
hsStatusMessage( "WARNING: Forcing unlock on GUI list box due to call to IUpdate()\n" );
|
|
|
|
UnlockList();
|
|
|
|
}
|
|
|
|
|
|
|
|
if( fScrollControl != nil )
|
|
|
|
{
|
|
|
|
// We reverse the value, since we want the "up" button to scroll "up", which actually
|
|
|
|
// *decreases* the scrollPos
|
|
|
|
fScrollPos = (int)fScrollControl->GetMax() - (int)fScrollControl->GetCurrValue();
|
|
|
|
|
|
|
|
if( fCheckScroll )
|
|
|
|
{
|
|
|
|
// If this is set, we just did something to change the selection to a single item.
|
|
|
|
// Since it makes sense to try to ensure this item is on the screen, we thus do so here.
|
|
|
|
// We don't want to do this every time, because we could be simply clicking on the
|
|
|
|
// scroll bars, in which case we DON'T want to do this check, since it would be
|
|
|
|
// fighting against the user, which is a big UI design no-no.
|
|
|
|
// To do this check, we search on either side of our scroll range (we can only
|
|
|
|
// search past after we draw, unfortunately). If there's a selected item out
|
|
|
|
// of our range, then we best move the scrollPos to move it into view
|
|
|
|
for( j = 0; j < fScrollPos; j++ )
|
|
|
|
{
|
|
|
|
int end = ( j < fWrapStartIdxs.GetCount() - 1 ) ? fWrapStartIdxs[ j + 1 ] : fElements.GetCount();
|
|
|
|
|
|
|
|
hsBool anySelected = false;
|
|
|
|
for( i = fWrapStartIdxs[ j ]; i < end; i++ )
|
|
|
|
anySelected |= fElements[ i ]->IsSelected();
|
|
|
|
|
|
|
|
if( anySelected )
|
|
|
|
{
|
|
|
|
// Shit. Move the scrollPos up to this item at the very least
|
|
|
|
fScrollPos = j;
|
|
|
|
fScrollControl->SetCurrValue( (float)( (int)fScrollControl->GetMax() - fScrollPos ) );
|
|
|
|
fCheckScroll = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
fScrollPos = 0;
|
|
|
|
|
|
|
|
fElementBounds.SetCountAndZero( fElements.GetCount() );
|
|
|
|
|
|
|
|
if( !HasFlag( kScrollLeftToRight ) )
|
|
|
|
{
|
|
|
|
for( j = fScrollPos, y = 4; j < fWrapStartIdxs.GetCount() && y < fDynTextMap->GetVisibleHeight() - 8; j++ )
|
|
|
|
{
|
|
|
|
int end = ( j < fWrapStartIdxs.GetCount() - 1 ) ? fWrapStartIdxs[ j + 1 ] : fElements.GetCount();
|
|
|
|
|
|
|
|
for( maxHeight = 0, i = fWrapStartIdxs[ j ], x = 0; i < end; i++ )
|
|
|
|
{
|
|
|
|
if( HasFlag( kGrowLeavesAndProcessOxygen ) && fElements[ i ]->IsCollapsed() )
|
|
|
|
continue; // Skip collapsed items
|
|
|
|
|
|
|
|
fElements[ i ]->GetSize( fDynTextMap, &width, &height );
|
|
|
|
|
|
|
|
if( !HasFlag( kAllowMultipleElementsPerRow ) )
|
|
|
|
width = fDynTextMap->GetVisibleWidth();
|
|
|
|
|
|
|
|
if( y + height >= fDynTextMap->GetVisibleHeight() - 8 )
|
|
|
|
height = fDynTextMap->GetVisibleHeight() - 8 - y;
|
|
|
|
|
|
|
|
if( HasFlag( kGrowLeavesAndProcessOxygen ) )
|
|
|
|
{
|
|
|
|
x += fElements[ i ]->GetIndentLevel() * kIndentAmount;
|
|
|
|
if( x + width > fDynTextMap->GetVisibleWidth() )
|
|
|
|
width = fDynTextMap->GetVisibleWidth() - x;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( !fElements[ i ]->Draw( fDynTextMap, x, y, width, height ) )
|
|
|
|
{
|
|
|
|
// Couldn't draw, so force it to be at the end of the scroll range
|
|
|
|
y = fDynTextMap->GetVisibleHeight() - 8;
|
|
|
|
j--;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
maxHeight = ( height > maxHeight ) ? height : maxHeight;
|
|
|
|
fElementBounds[ i ].Set( x, y, x + width - 1, y + height - 1 );
|
|
|
|
x += width;
|
|
|
|
}
|
|
|
|
|
|
|
|
y += maxHeight;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for( j = fScrollPos, x = 4; j < fWrapStartIdxs.GetCount() && x < fDynTextMap->GetVisibleWidth() - 8; j++ )
|
|
|
|
{
|
|
|
|
int end = ( j < fWrapStartIdxs.GetCount() - 1 ) ? fWrapStartIdxs[ j + 1 ] : fElements.GetCount();
|
|
|
|
|
|
|
|
for( maxHeight = 0, i = fWrapStartIdxs[ j ], y = 0; i < end; i++ )
|
|
|
|
{
|
|
|
|
fElements[ i ]->GetSize( fDynTextMap, &width, &height );
|
|
|
|
|
|
|
|
if( !HasFlag( kAllowMultipleElementsPerRow ) )
|
|
|
|
height = fDynTextMap->GetVisibleHeight();
|
|
|
|
|
|
|
|
if( x + width >= fDynTextMap->GetVisibleWidth() - 8 )
|
|
|
|
width = fDynTextMap->GetVisibleWidth() - 8 - x;
|
|
|
|
|
|
|
|
if( !fElements[ i ]->Draw( fDynTextMap, x, y, width, height ) )
|
|
|
|
{
|
|
|
|
// Couldn't draw, so force it to be at the end of the scroll range
|
|
|
|
x = fDynTextMap->GetVisibleWidth() - 8;
|
|
|
|
j--;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
maxHeight = ( width > maxHeight ) ? width : maxHeight;
|
|
|
|
fElementBounds[ i ].Set( x, y, x + width - 1, y + height - 1 );
|
|
|
|
y += height;
|
|
|
|
}
|
|
|
|
|
|
|
|
x += maxHeight;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Second part of our scroll check--if we got here, then there was nothing selected
|
|
|
|
// before the viewing area. So check the rest
|
|
|
|
if( fCheckScroll && fScrollControl != nil )
|
|
|
|
{
|
|
|
|
fCheckScroll = false;
|
|
|
|
for( ; j < fWrapStartIdxs.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
int end = ( j < fWrapStartIdxs.GetCount() - 1 ) ? fWrapStartIdxs[ j + 1 ] : fElements.GetCount();
|
|
|
|
|
|
|
|
hsBool anySelected = false;
|
|
|
|
for( i = fWrapStartIdxs[ j ]; i < end; i++ )
|
|
|
|
anySelected |= fElements[ i ]->IsSelected();
|
|
|
|
|
|
|
|
if( anySelected )
|
|
|
|
{
|
|
|
|
fScrollPos = j;
|
|
|
|
fScrollControl->SetCurrValue( (float)( (int)fScrollControl->GetMax() - fScrollPos ) );
|
|
|
|
IUpdate(); // Gotta update again, since we just changed the scrollPos after the fact
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fDynTextMap->FlushToHost();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ICalcWrapStarts /////////////////////////////////////////////////////////
|
|
|
|
// Calculates the starting indexes for each row of items. Call whenever you
|
|
|
|
// update the element list.
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::ICalcWrapStarts( void )
|
|
|
|
{
|
|
|
|
uint16_t i, x, y, maxHeight, width, height;
|
|
|
|
|
|
|
|
|
|
|
|
fWrapStartIdxs.Reset();
|
|
|
|
|
|
|
|
if( HasFlag( kAllowMultipleElementsPerRow ) )
|
|
|
|
{
|
|
|
|
if( !HasFlag( kScrollLeftToRight ) )
|
|
|
|
{
|
|
|
|
for( i = 0, x = (uint16_t)-1, y = 4, maxHeight = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
fElements[ i ]->GetSize( fDynTextMap, &width, &height );
|
|
|
|
|
|
|
|
if( x + width >= fDynTextMap->GetVisibleWidth() )
|
|
|
|
{
|
|
|
|
x = 0;
|
|
|
|
y += maxHeight;
|
|
|
|
maxHeight = 0;
|
|
|
|
fWrapStartIdxs.Append( i );
|
|
|
|
}
|
|
|
|
x += width;
|
|
|
|
maxHeight = ( height > maxHeight ) ? height : maxHeight;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for( i = 0, y = (uint16_t)-1, x = 4, maxHeight = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
fElements[ i ]->GetSize( fDynTextMap, &width, &height );
|
|
|
|
if( y + height >= fDynTextMap->GetVisibleHeight() )
|
|
|
|
{
|
|
|
|
y = 0;
|
|
|
|
x += maxHeight;
|
|
|
|
maxHeight = 0;
|
|
|
|
fWrapStartIdxs.Append( i );
|
|
|
|
}
|
|
|
|
y += height;
|
|
|
|
maxHeight = ( width > maxHeight ) ? width : maxHeight;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for( i = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( HasFlag( kGrowLeavesAndProcessOxygen ) && fElements[ i ]->IsCollapsed() )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
fWrapStartIdxs.Append( i );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ICalcScrollRange ////////////////////////////////////////////////////////
|
|
|
|
// Call whenever you update the element list
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::ICalcScrollRange( void )
|
|
|
|
{
|
|
|
|
uint16_t ctrlHt, ctrlWd, height, width, maxHeight;
|
|
|
|
int i, j, end;
|
|
|
|
|
|
|
|
|
|
|
|
if( !fReadyToRoll )
|
|
|
|
{
|
|
|
|
fScrollRangeUpdateDeferred = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Basically, the scroll range is 0 to (numElements-1), but since if we scroll
|
|
|
|
// that far, we won't see anything left except the last element, we have to calculate
|
|
|
|
// how many elements away from the end we can get before we see blank space. Which means
|
|
|
|
// counting up from the end until we have passed the height of our control.
|
|
|
|
|
|
|
|
end = fElements.GetCount();
|
|
|
|
if( !HasFlag( kScrollLeftToRight ) )
|
|
|
|
{
|
|
|
|
if( HasFlag( kGrowLeavesAndProcessOxygen ) )
|
|
|
|
{
|
|
|
|
// OK, that really isn't the end, the end will the count of non-collapsed elements. Haha.
|
|
|
|
for( i = 0, end = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( !fElements[ i ]->IsCollapsed() )
|
|
|
|
end++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( i = fWrapStartIdxs.GetCount() - 1, height = 0; i >= 0; i-- )
|
|
|
|
{
|
|
|
|
// Get the max height of this row
|
|
|
|
maxHeight = 0;
|
|
|
|
for( j = fWrapStartIdxs[ i ]; j < end; j++ )
|
|
|
|
{
|
|
|
|
fElements[ j ]->GetSize( fDynTextMap, &width, &ctrlHt );
|
|
|
|
maxHeight = ( ctrlHt > maxHeight ) ? ctrlHt : maxHeight;
|
|
|
|
}
|
|
|
|
end = fWrapStartIdxs[ i ];
|
|
|
|
height += maxHeight;
|
|
|
|
|
|
|
|
if( height > fDynTextMap->GetVisibleHeight() - 8 )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for( i = fWrapStartIdxs.GetCount() - 1, width = 0; i >= 0; i-- )
|
|
|
|
{
|
|
|
|
// Get the max width of this column
|
|
|
|
maxHeight = 0;
|
|
|
|
for( j = fWrapStartIdxs[ i ]; j < end; j++ )
|
|
|
|
{
|
|
|
|
fElements[ j ]->GetSize( fDynTextMap, &ctrlWd, &ctrlHt );
|
|
|
|
maxHeight = ( ctrlWd > maxHeight ) ? ctrlWd : maxHeight;
|
|
|
|
}
|
|
|
|
end = fWrapStartIdxs[ i ];
|
|
|
|
width += maxHeight;
|
|
|
|
|
|
|
|
if( width > fDynTextMap->GetVisibleWidth() - 8 )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note: i went one past, so our scroll range is really 0 to i+1
|
|
|
|
if( fScrollControl )
|
|
|
|
{
|
|
|
|
// Because we reverse the position on draw, we have to do some special trick here
|
|
|
|
// to make sure the reversed position stays the same
|
|
|
|
fScrollPos = (int)fScrollControl->GetMax() - (int)fScrollControl->GetCurrValue();
|
|
|
|
if( fScrollPos >= fWrapStartIdxs.GetCount() )
|
|
|
|
fScrollPos = fWrapStartIdxs.GetCount() - 1;
|
|
|
|
|
|
|
|
if( i < 0 )
|
|
|
|
// Smaller than the viewing area, so we don't scroll at all
|
|
|
|
fScrollControl->SetRange( 0.f, 0.f );
|
|
|
|
else
|
|
|
|
fScrollControl->SetRange( 0.f, (float)( i + 1 ) );
|
|
|
|
|
|
|
|
fScrollControl->SetCurrValue( (float)( (int)fScrollControl->GetMax() - fScrollPos ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::PurgeDynaTextMapImage()
|
|
|
|
{
|
|
|
|
if ( fDynTextMap != nil )
|
|
|
|
fDynTextMap->PurgeImage();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// Read/Write //////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::Read( hsStream *s, hsResMgr *mgr )
|
|
|
|
{
|
|
|
|
pfGUIControlMod::Read(s, mgr);
|
|
|
|
|
|
|
|
fScrollControl = nil;
|
|
|
|
if( s->ReadBool() )
|
|
|
|
{
|
|
|
|
fScrollProc = TRACKED_NEW pfScrollProc( this );
|
|
|
|
fScrollProc->IncRef();
|
|
|
|
mgr->ReadKeyNotifyMe( s, TRACKED_NEW plGenRefMsg( GetKey(), plRefMsg::kOnCreate, -1, kRefScrollCtrl ), plRefFlags::kActiveRef );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( HasFlag( kDisableSelection ) )
|
|
|
|
ClearFlag( kWantsInterest );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::Write( hsStream *s, hsResMgr *mgr )
|
|
|
|
{
|
|
|
|
pfGUIControlMod::Write( s, mgr );
|
|
|
|
|
|
|
|
if( fScrollControl != nil )
|
|
|
|
{
|
|
|
|
s->WriteBool( true );
|
|
|
|
mgr->WriteKey( s, fScrollControl->GetKey() );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
s->WriteBool( false );
|
|
|
|
}
|
|
|
|
|
|
|
|
//// FilterMousePosition /////////////////////////////////////////////////////
|
|
|
|
// Tests the mouse point and decides whether we still want to accept input
|
|
|
|
// based on the position. This allows us to act etheral (i.e. pass mouse
|
|
|
|
// messages through) when the mouse is over an empty portion of the list.
|
|
|
|
|
|
|
|
hsBool pfGUIListBoxMod::FilterMousePosition( hsPoint3 &mousePt )
|
|
|
|
{
|
|
|
|
if( !HasFlag( kAllowMousePassThrough ) )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
int32_t hover = fCurrClick = IGetItemFromPoint( mousePt );
|
|
|
|
if( hover == -1 )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// HandleMouseDown /////////////////////////////////////////////////////////
|
|
|
|
// What we do: normal click deselects all and selects the item clicked on
|
|
|
|
// (if any). Shift-click and ctrl-click avoids the deselect and toggles
|
|
|
|
// the item clicked on.
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::HandleMouseDown( hsPoint3 &mousePt, uint8_t modifiers )
|
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
|
|
|
|
int lastSelection = -1;
|
|
|
|
if (HasFlag(kForbidNoSelection))
|
|
|
|
{
|
|
|
|
// grab the last item we had selected just in case they clicked outside the list of selectable objects
|
|
|
|
for (i=0; i<fElements.GetCount(); i++)
|
|
|
|
if (fElements[i]->IsSelected())
|
|
|
|
{
|
|
|
|
lastSelection = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( HasFlag(kSingleSelect) || ( !( modifiers & ( pfGameGUIMgr::kShiftDown | pfGameGUIMgr::kCtrlDown ) ) && !HasFlag( kHandsOffMultiSelect ) ) )
|
|
|
|
{
|
|
|
|
// Deselect everyone!
|
|
|
|
for( i = 0; i < fElements.GetCount(); i++ )
|
|
|
|
fElements[ i ]->SetSelected( false );
|
|
|
|
fSingleSelElement = -1;
|
|
|
|
}
|
|
|
|
else if( modifiers & pfGameGUIMgr::kShiftDown )
|
|
|
|
{
|
|
|
|
IFindSelectionRange( &fMinSel, &fMaxSel );
|
|
|
|
}
|
|
|
|
|
|
|
|
fCurrHover = fCurrClick = IGetItemFromPoint( mousePt );
|
|
|
|
if( fCurrClick != -1 )
|
|
|
|
{
|
|
|
|
if( ( ( modifiers & pfGameGUIMgr::kShiftDown ) && !HasFlag( kSingleSelect ) ) )
|
|
|
|
{
|
|
|
|
// Select our new range, deselect everything outside
|
|
|
|
if( fCurrClick <= fMaxSel )
|
|
|
|
{
|
|
|
|
ISelectRange( 0, (int8_t)(fCurrClick - 1), false );
|
|
|
|
ISelectRange( (int8_t)fCurrClick, (int8_t)fMaxSel, true );
|
|
|
|
ISelectRange( (int8_t)(fMaxSel + 1), -1, false );
|
|
|
|
}
|
|
|
|
else if( fCurrClick >= fMinSel )
|
|
|
|
{
|
|
|
|
ISelectRange( 0, (int8_t)(fMinSel - 1), false );
|
|
|
|
ISelectRange( (int8_t)fMinSel, (int8_t)fCurrClick, true );
|
|
|
|
ISelectRange( (int8_t)(fCurrClick + 1), -1, false );
|
|
|
|
}
|
|
|
|
fElements[ fCurrClick ]->SetSelected( true );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fElements[ fCurrClick ]->SetSelected( true );
|
|
|
|
fSingleSelElement = fCurrClick;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (HasFlag(kForbidNoSelection) && (lastSelection != -1))
|
|
|
|
{
|
|
|
|
fElements[ lastSelection ]->SetSelected(true);
|
|
|
|
fSingleSelElement = lastSelection;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We want drag/up events to be processed with the modifiers we had on
|
|
|
|
// mouse down, not the current ones. So store 'em for reference
|
|
|
|
fModsAtDragTime = modifiers;
|
|
|
|
fClicking = true;
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// HandleMouseUp ///////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::HandleMouseUp( hsPoint3 &mousePt, uint8_t modifiers )
|
|
|
|
{
|
|
|
|
fClicking = false;
|
|
|
|
if( !( fModsAtDragTime & ( pfGameGUIMgr::kShiftDown | pfGameGUIMgr::kCtrlDown ) ) && !HasFlag( kHandsOffMultiSelect ) )
|
|
|
|
{
|
|
|
|
// No modifiers--simply select whatever item we're on
|
|
|
|
int32_t item = IGetItemFromPoint( mousePt );
|
|
|
|
if( item != fCurrClick )
|
|
|
|
{
|
|
|
|
if( fCurrClick >= 0 && fCurrClick < fElements.GetCount() )
|
|
|
|
fElements[ fCurrClick ]->SetSelected( false );
|
|
|
|
fCurrHover = fCurrClick = item;
|
|
|
|
if( fCurrClick >= 0 && fCurrClick < fElements.GetCount() )
|
|
|
|
fElements[ fCurrClick ]->SetSelected( true );
|
|
|
|
else if (HasFlag(kForbidNoSelection) && fCurrClick == -1)
|
|
|
|
{
|
|
|
|
fElements[fSingleSelElement]->SetSelected(true);
|
|
|
|
fCurrClick = fSingleSelElement;
|
|
|
|
}
|
|
|
|
fCheckScroll = true;
|
|
|
|
fSingleSelElement = fCurrClick;
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We didn't change selection, so go ahead and pass the click on to
|
|
|
|
// the item, in case it wants to do something
|
|
|
|
if( fCurrClick >= 0 && fCurrClick < fElements.GetCount() )
|
|
|
|
{
|
|
|
|
hsPoint3 localPt = mousePt;
|
|
|
|
IScreenToLocalPt( localPt );
|
|
|
|
|
|
|
|
uint16_t lX = (uint16_t)(( localPt.fX * ( fDynTextMap->GetVisibleWidth() - 1 ) ) - fElementBounds[ fCurrClick ].fLeft);
|
|
|
|
uint16_t lY = (uint16_t)(( localPt.fY * ( fDynTextMap->GetVisibleHeight() - 1 ) )- fElementBounds[ fCurrClick ].fTop);
|
|
|
|
|
|
|
|
if( fElements[ fCurrClick ]->MouseClicked( lX, lY ) )
|
|
|
|
{
|
|
|
|
ICalcWrapStarts();
|
|
|
|
ICalcScrollRange();
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DoSomething(); // Change in selection triggers our DoSomething() call
|
|
|
|
}
|
|
|
|
|
|
|
|
//// HandleMouseHover ////////////////////////////////////////////////////////
|
|
|
|
// Just updates our currHover item so we can update the mouse appropriately
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::HandleMouseHover( hsPoint3 &mousePt, uint8_t modifiers )
|
|
|
|
{
|
|
|
|
fCurrHover = IGetItemFromPoint( mousePt );
|
|
|
|
}
|
|
|
|
|
|
|
|
//// HandleMouseDrag /////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::HandleMouseDrag( hsPoint3 &mousePt, uint8_t modifiers )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
if( ( fModsAtDragTime & pfGameGUIMgr::kShiftDown && !HasFlag( kSingleSelect ) ) )
|
|
|
|
{
|
|
|
|
// Select our new range, deselect everything outside
|
|
|
|
if( fCurrClick <= fMaxSel )
|
|
|
|
{
|
|
|
|
ISelectRange( 0, (int8_t)(fCurrClick - 1), false );
|
|
|
|
ISelectRange( (int8_t)fCurrClick, (int8_t)fMaxSel, true );
|
|
|
|
ISelectRange( (int8_t)(fMaxSel + 1), -1, false );
|
|
|
|
}
|
|
|
|
else if( fCurrClick >= fMinSel )
|
|
|
|
{
|
|
|
|
ISelectRange( 0, (int8_t)(fMinSel - 1), false );
|
|
|
|
ISelectRange( (int8_t)fMinSel, (int8_t)fCurrClick, true );
|
|
|
|
ISelectRange( (int8_t)(fCurrClick + 1), -1, false );
|
|
|
|
}
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
else if( !( fModsAtDragTime & ( pfGameGUIMgr::kShiftDown | pfGameGUIMgr::kCtrlDown ) ) )
|
|
|
|
{
|
|
|
|
// No modifiers--simply select whatever item we're on
|
|
|
|
if( HasFlag( kDragAndDropCapable ) )
|
|
|
|
{
|
|
|
|
// If we're drag and drop capable, then a mouse drag function results in
|
|
|
|
// the dialog entering Drag Mode(tm). Basically, we tell our parent dialog
|
|
|
|
// which items to drag and it'll run with the rest
|
|
|
|
fDialog->ClearDragList();
|
|
|
|
for( i = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fElements[ i ]->IsSelected() )
|
|
|
|
fDialog->AddToDragList( fElements[ i ] );
|
|
|
|
}
|
|
|
|
fDialog->EnterDragMode( this );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int32_t item = IGetItemFromPoint( mousePt );
|
|
|
|
if( item != fCurrClick )
|
|
|
|
{
|
|
|
|
if( fCurrClick >= 0 && fCurrClick < fElements.GetCount() )
|
|
|
|
fElements[ fCurrClick ]->SetSelected( false );
|
|
|
|
fCurrHover = fCurrClick = item;
|
|
|
|
if( fCurrClick >= 0 && fCurrClick < fElements.GetCount() )
|
|
|
|
fElements[ fCurrClick ]->SetSelected( true );
|
|
|
|
fCheckScroll = true;
|
|
|
|
fSingleSelElement = fCurrClick;
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// HandleMouseDblClick /////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::HandleMouseDblClick( hsPoint3 &mousePt, uint8_t modifiers )
|
|
|
|
{
|
|
|
|
if( !HasFlag( kGrowLeavesAndProcessOxygen ) )
|
|
|
|
return; // We only care about double clicks if we make oxygen
|
|
|
|
|
|
|
|
int32_t item = IGetItemFromPoint( mousePt );
|
|
|
|
if( item != -1 )
|
|
|
|
{
|
|
|
|
if( fElements[ item ]->GetType() == pfGUIListElement::kTreeRoot )
|
|
|
|
{
|
|
|
|
pfGUIListTreeRoot *root = (pfGUIListTreeRoot *)fElements[ item ];
|
|
|
|
|
|
|
|
root->ShowChildren( !root->IsShowingChildren() );
|
|
|
|
if( !fLocked )
|
|
|
|
{
|
|
|
|
ICalcWrapStarts();
|
|
|
|
ICalcScrollRange();
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IGetItemFromPoint ///////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
int32_t pfGUIListBoxMod::IGetItemFromPoint( hsPoint3 &mousePt )
|
|
|
|
{
|
|
|
|
if( !fBounds.IsInside( &mousePt ) )
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
hsPoint3 localPt = mousePt; // despite getting a ref to the point (why?) we do NOT want to modify it
|
|
|
|
IScreenToLocalPt( localPt );
|
|
|
|
uint32_t i;
|
|
|
|
int16_t clickItem = -1 , clickY = (int16_t)( localPt.fY * ( fDynTextMap->GetVisibleHeight() - 1 ) );
|
|
|
|
int16_t clickX = (int16_t)( localPt.fX * ( fDynTextMap->GetVisibleWidth() - 1 ) );
|
|
|
|
|
|
|
|
// We have a nice array that has the starting (top) Y's of each visible element. So we just
|
|
|
|
// check against that.
|
|
|
|
uint32_t startAt = 0;
|
|
|
|
// make sure that we have a valid fScrollPos
|
|
|
|
if ( fScrollPos != -1 )
|
|
|
|
{
|
|
|
|
// make sure that there is an Idx at fScrollPos
|
|
|
|
if ( fWrapStartIdxs.GetCount() > fScrollPos )
|
|
|
|
{
|
|
|
|
startAt = fWrapStartIdxs[ fScrollPos ];
|
|
|
|
clickItem = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for( i = startAt; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( i<fElementBounds.GetCount() && fElementBounds[ i ].Contains( clickX, clickY ) )
|
|
|
|
{
|
|
|
|
clickItem = (int16_t)i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( clickItem > fElements.GetCount() - 1 )
|
|
|
|
clickItem = -1;
|
|
|
|
|
|
|
|
return clickItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IFindSelectionRange /////////////////////////////////////////////////////
|
|
|
|
// Find the min and max item that's selected. Returns -1 for both if nobody
|
|
|
|
// is selected
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::IFindSelectionRange( int32_t *min, int32_t *max )
|
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
|
|
|
|
|
|
|
|
*min = *max = -1;
|
|
|
|
|
|
|
|
for( i = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fElements[ i ]->IsSelected() )
|
|
|
|
{
|
|
|
|
*min = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( i = fElements.GetCount() - 1; i >= 0; i-- )
|
|
|
|
{
|
|
|
|
if( fElements[ i ]->IsSelected() )
|
|
|
|
{
|
|
|
|
*max = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ISelectRange ////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::ISelectRange( int8_t min, int8_t max, hsBool select )
|
|
|
|
{
|
|
|
|
int16_t i;
|
|
|
|
|
|
|
|
|
|
|
|
if( max == -1 )
|
|
|
|
max = fElements.GetCount() - 1;
|
|
|
|
|
|
|
|
for( i = min; i <= max; i++ )
|
|
|
|
fElements[ i ]->SetSelected( select );
|
|
|
|
}
|
|
|
|
|
|
|
|
//// SetSelection ////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::SetSelection( int32_t item )
|
|
|
|
{
|
|
|
|
if( HasFlag( kSingleSelect ) )
|
|
|
|
{
|
|
|
|
// Easy, only one item selected
|
|
|
|
if( fSingleSelElement != -1 )
|
|
|
|
fElements[ fSingleSelElement ]->SetSelected( false );
|
|
|
|
|
|
|
|
fSingleSelElement = item;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ISelectRange( 0, -1, false );
|
|
|
|
|
|
|
|
if( item != -1 && item < fElements.GetCount() )
|
|
|
|
fElements[ item ]->SetSelected( true );
|
|
|
|
|
|
|
|
fCheckScroll = true;
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::RemoveSelection( int32_t item )
|
|
|
|
{
|
|
|
|
// make sure the item is valid
|
|
|
|
if ( item != -1 && item < fElements.GetCount() )
|
|
|
|
{
|
|
|
|
// if single select and its what is selected
|
|
|
|
if ( HasFlag( kSingleSelect) && fSingleSelElement == item )
|
|
|
|
{
|
|
|
|
// then remove the selection and make nothing selected
|
|
|
|
fElements[ fSingleSelElement ]->SetSelected( false );
|
|
|
|
fSingleSelElement = item;
|
|
|
|
}
|
|
|
|
// else if multiple selection
|
|
|
|
else
|
|
|
|
// just remove the selection
|
|
|
|
fElements[ item ]->SetSelected( false );
|
|
|
|
}
|
|
|
|
fCheckScroll = true;
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::AddSelection( int32_t item )
|
|
|
|
{
|
|
|
|
// make sure the item is valid (can't add a non-selection!)
|
|
|
|
if ( item != -1 && item < fElements.GetCount() )
|
|
|
|
{
|
|
|
|
// if single select then just like SetSelection
|
|
|
|
if ( HasFlag( kSingleSelect) )
|
|
|
|
{
|
|
|
|
SetSelection(item);
|
|
|
|
}
|
|
|
|
// else if multiple selection
|
|
|
|
else
|
|
|
|
// just set its selection
|
|
|
|
fElements[ item ]->SetSelected( true );
|
|
|
|
}
|
|
|
|
fCheckScroll = true;
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// HandleKeyPress //////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
hsBool pfGUIListBoxMod::HandleKeyPress( wchar_t key, uint8_t modifiers )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool pfGUIListBoxMod::HandleKeyEvent( pfGameGUIMgr::EventType event, plKeyDef key, uint8_t modifiers )
|
|
|
|
{
|
|
|
|
if( key == KEY_CAPSLOCK )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if( HasFlag( kDisableKeyActions ) )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if( event == pfGameGUIMgr::kKeyDown || event == pfGameGUIMgr::kKeyRepeat )
|
|
|
|
{
|
|
|
|
// Use arrow keys to do our dirty work
|
|
|
|
if( key == KEY_UP )
|
|
|
|
{
|
|
|
|
if( fCurrClick == -1 && fElements.GetCount() > 0 )
|
|
|
|
fCurrClick = 0;
|
|
|
|
else while( fCurrClick > 0 )
|
|
|
|
{
|
|
|
|
fCurrClick--;
|
|
|
|
if( !HasFlag( kGrowLeavesAndProcessOxygen ) || !fElements[ fCurrClick ]->IsCollapsed() )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( key == KEY_DOWN )
|
|
|
|
{
|
|
|
|
if( fCurrClick == -1 && fElements.GetCount() > 0 )
|
|
|
|
fCurrClick = fElements.GetCount() - 1;
|
|
|
|
else while( fCurrClick < fElements.GetCount() - 1 )
|
|
|
|
{
|
|
|
|
fCurrClick++;
|
|
|
|
if( !HasFlag( kGrowLeavesAndProcessOxygen ) || !fElements[ fCurrClick ]->IsCollapsed() )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( key == KEY_HOME )
|
|
|
|
{
|
|
|
|
if( fElements.GetCount() > 0 )
|
|
|
|
fCurrClick = 0;
|
|
|
|
}
|
|
|
|
else if( key == KEY_END )
|
|
|
|
{
|
|
|
|
if( fElements.GetCount() > 0 )
|
|
|
|
fCurrClick = fElements.GetCount() - 1;
|
|
|
|
}
|
|
|
|
else if( key == KEY_ENTER && HasFlag( kGrowLeavesAndProcessOxygen ) )
|
|
|
|
{
|
|
|
|
if( fCurrClick != -1 && fElements[ fCurrClick ]->GetType() == pfGUIListElement::kTreeRoot )
|
|
|
|
{
|
|
|
|
pfGUIListTreeRoot *root = (pfGUIListTreeRoot *)fElements[ fCurrClick ];
|
|
|
|
root->ShowChildren( !root->IsShowingChildren() );
|
|
|
|
|
|
|
|
if( !fLocked )
|
|
|
|
{
|
|
|
|
ICalcWrapStarts();
|
|
|
|
ICalcScrollRange();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ISelectRange( 0, -1, false );
|
|
|
|
if( fCurrClick != -1 )
|
|
|
|
fElements[ fCurrClick ]->SetSelected( true );
|
|
|
|
fSingleSelElement = fCurrClick;
|
|
|
|
|
|
|
|
DoSomething(); // Change in selection triggers our DoSomething() call
|
|
|
|
|
|
|
|
fCheckScroll = true;
|
|
|
|
IUpdate();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ScrollToEnd /////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::ScrollToEnd( void )
|
|
|
|
{
|
|
|
|
if( fScrollControl != nil )
|
|
|
|
{
|
|
|
|
fScrollPos = (int)fScrollControl->GetMin();
|
|
|
|
fScrollControl->SetCurrValue( fScrollControl->GetMax() );
|
|
|
|
}
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ScrollToBegin ///////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::ScrollToBegin( void )
|
|
|
|
{
|
|
|
|
if( fScrollControl != nil )
|
|
|
|
{
|
|
|
|
fScrollPos = (int)fScrollControl->GetMax();
|
|
|
|
fScrollControl->SetCurrValue( fScrollControl->GetMin() );
|
|
|
|
}
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::SetColorScheme( pfGUIColorScheme *newScheme )
|
|
|
|
{
|
|
|
|
uint16_t i;
|
|
|
|
|
|
|
|
pfGUIControlMod::SetColorScheme( newScheme );
|
|
|
|
|
|
|
|
for( i = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
fElements[ i ]->SetColorScheme( newScheme );
|
|
|
|
fElements[ i ]->SetSkin( fSkin );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::SetScrollPos( int32_t pos )
|
|
|
|
{
|
|
|
|
if ( pos >= (int)fScrollControl->GetMin() && pos <= (int)fScrollControl->GetMax() )
|
|
|
|
{
|
|
|
|
fScrollControl->SetCurrValue( (float)pos );
|
|
|
|
fScrollPos = (int)fScrollControl->GetMax() - pos;
|
|
|
|
}
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t pfGUIListBoxMod::GetScrollPos( void )
|
|
|
|
{
|
|
|
|
return (int)fScrollControl->GetCurrValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t pfGUIListBoxMod::GetScrollRange( void )
|
|
|
|
{
|
|
|
|
return (int)fScrollControl->GetMax() - (int)fScrollControl->GetMin();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
//// Element Manipulation ////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
uint16_t pfGUIListBoxMod::AddElement( pfGUIListElement *el )
|
|
|
|
{
|
|
|
|
uint16_t idx = fElements.GetCount();
|
|
|
|
fElements.Append( el );
|
|
|
|
el->SetColorScheme( GetColorScheme() );
|
|
|
|
el->SetSkin( fSkin );
|
|
|
|
if( !fLocked )
|
|
|
|
{
|
|
|
|
ICalcWrapStarts();
|
|
|
|
ICalcScrollRange();
|
|
|
|
IUpdate();
|
|
|
|
HandleExtendedEvent( pfGUIListBoxMod::kItemAdded );
|
|
|
|
}
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::RemoveElement( uint16_t index )
|
|
|
|
{
|
|
|
|
// Make sure no other elements care about this one
|
|
|
|
uint16_t i, j;
|
|
|
|
for( i = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fElements[ i ]->GetType() == pfGUIListElement::kTreeRoot )
|
|
|
|
{
|
|
|
|
pfGUIListTreeRoot *root = (pfGUIListTreeRoot *)fElements[ i ];
|
|
|
|
for( j = 0; j < root->GetNumChildren(); )
|
|
|
|
{
|
|
|
|
if( root->GetChild( j ) == fElements[ index ] )
|
|
|
|
root->RemoveChild( j );
|
|
|
|
else
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delete fElements[ index ];
|
|
|
|
fElements.Remove( index );
|
|
|
|
|
|
|
|
if( index == fSingleSelElement )
|
|
|
|
fSingleSelElement = -1;
|
|
|
|
else if( index < fSingleSelElement )
|
|
|
|
fSingleSelElement--;
|
|
|
|
fCurrHover = fCurrClick = -1;
|
|
|
|
|
|
|
|
if( !fLocked )
|
|
|
|
{
|
|
|
|
ICalcWrapStarts();
|
|
|
|
ICalcScrollRange();
|
|
|
|
IUpdate();
|
|
|
|
HandleExtendedEvent( pfGUIListBoxMod::kItemRemoved );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t pfGUIListBoxMod::FindElement( pfGUIListElement *toCompareTo )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
for( i = 0; i < fElements.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fElements[ i ]->CompareTo( toCompareTo ) == 0 )
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (int16_t)-1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::ClearAllElements( void )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
for( i = 0; i < fElements.GetCount(); i++ )
|
|
|
|
delete fElements[ i ];
|
|
|
|
fElements.Reset();
|
|
|
|
fSingleSelElement = -1;
|
|
|
|
|
|
|
|
if( !fLocked )
|
|
|
|
{
|
|
|
|
ICalcWrapStarts();
|
|
|
|
ICalcScrollRange();
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
HandleExtendedEvent( pfGUIListBoxMod::kListCleared );
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t pfGUIListBoxMod::AddString( const char *string )
|
|
|
|
{
|
|
|
|
return AddElement( TRACKED_NEW pfGUIListText( string ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t pfGUIListBoxMod::AddString( const wchar_t *string )
|
|
|
|
{
|
|
|
|
return AddElement( TRACKED_NEW pfGUIListText( string ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t pfGUIListBoxMod::FindString( const char *toCompareTo )
|
|
|
|
{
|
|
|
|
pfGUIListText text( toCompareTo );
|
|
|
|
return FindElement( &text );
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t pfGUIListBoxMod::FindString( const wchar_t *toCompareTo )
|
|
|
|
{
|
|
|
|
pfGUIListText text( toCompareTo );
|
|
|
|
return FindElement( &text );
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t pfGUIListBoxMod::GetNumElements( void )
|
|
|
|
{
|
|
|
|
return fElements.GetCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
pfGUIListElement *pfGUIListBoxMod::GetElement( uint16_t idx )
|
|
|
|
{
|
|
|
|
return fElements[ idx ];
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::LockList( void )
|
|
|
|
{
|
|
|
|
fLocked = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pfGUIListBoxMod::UnlockList( void )
|
|
|
|
{
|
|
|
|
fLocked = false;
|
|
|
|
ICalcWrapStarts();
|
|
|
|
ICalcScrollRange();
|
|
|
|
IUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IGetDesiredCursor ///////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
uint32_t pfGUIListBoxMod::IGetDesiredCursor( void ) const
|
|
|
|
{
|
|
|
|
if( fCurrHover == -1 )
|
|
|
|
return plInputInterface::kNullCursor;
|
|
|
|
|
|
|
|
if( fClicking )
|
|
|
|
return plInputInterface::kCursorClicked;
|
|
|
|
|
|
|
|
return plInputInterface::kCursorPoised;
|
|
|
|
}
|
|
|
|
|