180 lines
4.3 KiB
C
180 lines
4.3 KiB
C
|
/*
|
||
|
Copyright (C) 2001-2006, William Joseph.
|
||
|
All Rights Reserved.
|
||
|
|
||
|
This file is part of GtkRadiant.
|
||
|
|
||
|
GtkRadiant is free software; you can redistribute it and/or modify
|
||
|
it under the terms of the GNU General Public License as published by
|
||
|
the Free Software Foundation; either version 2 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
GtkRadiant 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 GtkRadiant; if not, write to the Free Software
|
||
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||
|
*/
|
||
|
|
||
|
#if !defined ( INCLUDED_SELECTIONLIB_H )
|
||
|
#define INCLUDED_SELECTIONLIB_H
|
||
|
|
||
|
#include "iselection.h"
|
||
|
#include "generic/callback.h"
|
||
|
#include "scenelib.h"
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
class SelectableBool : public Selectable
|
||
|
{
|
||
|
bool m_selected;
|
||
|
public:
|
||
|
SelectableBool()
|
||
|
: m_selected( false )
|
||
|
{}
|
||
|
|
||
|
void setSelected( bool select = true ){
|
||
|
m_selected = select;
|
||
|
}
|
||
|
bool isSelected() const {
|
||
|
return m_selected;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class ObservedSelectable : public Selectable
|
||
|
{
|
||
|
SelectionChangeCallback m_onchanged;
|
||
|
bool m_selected;
|
||
|
public:
|
||
|
ObservedSelectable( const SelectionChangeCallback& onchanged ) : m_onchanged( onchanged ), m_selected( false ){
|
||
|
}
|
||
|
ObservedSelectable( const ObservedSelectable& other ) : Selectable( other ), m_onchanged( other.m_onchanged ), m_selected( false ){
|
||
|
setSelected( other.isSelected() );
|
||
|
}
|
||
|
ObservedSelectable& operator=( const ObservedSelectable& other ){
|
||
|
setSelected( other.isSelected() );
|
||
|
return *this;
|
||
|
}
|
||
|
~ObservedSelectable(){
|
||
|
setSelected( false );
|
||
|
}
|
||
|
|
||
|
void setSelected( bool select ){
|
||
|
if ( select ^ m_selected ) {
|
||
|
m_selected = select;
|
||
|
|
||
|
m_onchanged( *this );
|
||
|
}
|
||
|
}
|
||
|
bool isSelected() const {
|
||
|
return m_selected;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class SelectableInstance : public scene::Instance
|
||
|
{
|
||
|
class TypeCasts
|
||
|
{
|
||
|
InstanceTypeCastTable m_casts;
|
||
|
public:
|
||
|
TypeCasts(){
|
||
|
InstanceContainedCast<SelectableInstance, Selectable>::install( m_casts );
|
||
|
}
|
||
|
InstanceTypeCastTable& get(){
|
||
|
return m_casts;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
ObservedSelectable m_selectable;
|
||
|
public:
|
||
|
|
||
|
typedef LazyStatic<TypeCasts> StaticTypeCasts;
|
||
|
|
||
|
SelectableInstance( const scene::Path& path, scene::Instance* parent, void* instance = 0, InstanceTypeCastTable& casts = StaticTypeCasts::instance().get() ) :
|
||
|
Instance( path, parent, instance != 0 ? instance : this, casts ),
|
||
|
m_selectable( SelectedChangedCaller( *this ) ){
|
||
|
}
|
||
|
|
||
|
Selectable& get( NullType<Selectable>){
|
||
|
return m_selectable;
|
||
|
}
|
||
|
|
||
|
Selectable& getSelectable(){
|
||
|
return m_selectable;
|
||
|
}
|
||
|
const Selectable& getSelectable() const {
|
||
|
return m_selectable;
|
||
|
}
|
||
|
|
||
|
void selectedChanged( const Selectable& selectable ){
|
||
|
GlobalSelectionSystem().getObserver ( SelectionSystem::ePrimitive )( selectable );
|
||
|
GlobalSelectionSystem().onSelectedChanged( *this, selectable );
|
||
|
|
||
|
Instance::selectedChanged();
|
||
|
}
|
||
|
typedef MemberCaller<SelectableInstance, void(const Selectable&), &SelectableInstance::selectedChanged> SelectedChangedCaller;
|
||
|
};
|
||
|
|
||
|
|
||
|
template<typename Iterator>
|
||
|
inline bool range_check( Iterator start, Iterator finish, Iterator iter ){
|
||
|
for (; start != finish; ++start )
|
||
|
{
|
||
|
if ( start == iter ) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
#include <list>
|
||
|
|
||
|
template<typename Selected>
|
||
|
class SelectionList
|
||
|
{
|
||
|
typedef std::list<Selected*> List;
|
||
|
List m_selection;
|
||
|
public:
|
||
|
typedef typename List::iterator iterator;
|
||
|
typedef typename List::const_iterator const_iterator;
|
||
|
|
||
|
iterator begin(){
|
||
|
return m_selection.begin();
|
||
|
}
|
||
|
const_iterator begin() const {
|
||
|
return m_selection.begin();
|
||
|
}
|
||
|
iterator end(){
|
||
|
return m_selection.end();
|
||
|
}
|
||
|
const_iterator end() const {
|
||
|
return m_selection.end();
|
||
|
}
|
||
|
bool empty() const {
|
||
|
return m_selection.empty();
|
||
|
}
|
||
|
std::size_t size() const {
|
||
|
return m_selection.size();
|
||
|
}
|
||
|
Selected& back(){
|
||
|
return *m_selection.back();
|
||
|
}
|
||
|
Selected& back() const {
|
||
|
return *m_selection.back();
|
||
|
}
|
||
|
void append( Selected& selected ){
|
||
|
m_selection.push_back( &selected );
|
||
|
}
|
||
|
void erase( Selected& selected ){
|
||
|
typename List::reverse_iterator i = std::find( m_selection.rbegin(), m_selection.rend(), &selected );
|
||
|
ASSERT_MESSAGE( i != m_selection.rend(), "selection-tracking error" );
|
||
|
ASSERT_MESSAGE( range_check( m_selection.begin(), m_selection.end(), --i.base() ), "selection-tracking error" );
|
||
|
m_selection.erase( --i.base() );
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
#endif
|