2017-02-28 03:04:44 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2013-2017 CERN
|
|
|
|
* @author Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
|
|
|
|
* @author Maciej Suminski <maciej.suminski@cern.ch>
|
|
|
|
*
|
|
|
|
* 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 2
|
|
|
|
* 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, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
2019-06-08 21:48:22 +00:00
|
|
|
#ifndef SELECTION_H
|
|
|
|
#define SELECTION_H
|
2017-02-28 03:04:44 +00:00
|
|
|
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <core/optional.h>
|
2021-06-06 15:07:55 +00:00
|
|
|
#include <core/typeinfo.h>
|
2018-10-24 02:14:11 +00:00
|
|
|
#include <deque>
|
2019-06-08 21:48:22 +00:00
|
|
|
#include <eda_rect.h>
|
2017-02-28 03:04:44 +00:00
|
|
|
#include <view/view_group.h>
|
|
|
|
|
2021-06-06 15:07:55 +00:00
|
|
|
class EDA_ITEM;
|
|
|
|
|
2017-02-28 03:04:44 +00:00
|
|
|
class SELECTION : public KIGFX::VIEW_GROUP
|
|
|
|
{
|
|
|
|
public:
|
2020-12-25 23:37:17 +00:00
|
|
|
SELECTION() :
|
|
|
|
KIGFX::VIEW_GROUP::VIEW_GROUP()
|
2018-03-20 22:44:04 +00:00
|
|
|
{
|
|
|
|
m_isHover = false;
|
|
|
|
}
|
2017-03-03 12:42:28 +00:00
|
|
|
|
2020-12-25 23:37:17 +00:00
|
|
|
SELECTION( const SELECTION& aOther ) :
|
|
|
|
KIGFX::VIEW_GROUP::VIEW_GROUP()
|
2017-03-03 12:42:28 +00:00
|
|
|
{
|
|
|
|
m_items = aOther.m_items;
|
|
|
|
m_isHover = aOther.m_isHover;
|
|
|
|
}
|
|
|
|
|
2019-05-24 10:11:18 +00:00
|
|
|
SELECTION& operator= ( const SELECTION& aOther )
|
2017-03-03 12:42:28 +00:00
|
|
|
{
|
|
|
|
m_items = aOther.m_items;
|
|
|
|
m_isHover = aOther.m_isHover;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2018-10-24 02:14:11 +00:00
|
|
|
using ITER = std::deque<EDA_ITEM*>::iterator;
|
|
|
|
using CITER = std::deque<EDA_ITEM*>::const_iterator;
|
2017-02-28 03:04:44 +00:00
|
|
|
|
|
|
|
ITER begin() { return m_items.begin(); }
|
|
|
|
ITER end() { return m_items.end(); }
|
|
|
|
CITER begin() const { return m_items.cbegin(); }
|
|
|
|
CITER end() const { return m_items.cend(); }
|
|
|
|
|
2017-03-03 12:42:28 +00:00
|
|
|
void SetIsHover( bool aIsHover )
|
|
|
|
{
|
|
|
|
m_isHover = aIsHover;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsHover() const
|
|
|
|
{
|
|
|
|
return m_isHover;
|
|
|
|
}
|
|
|
|
|
2021-06-06 15:07:55 +00:00
|
|
|
virtual void Add( EDA_ITEM* aItem );
|
2017-02-28 03:04:44 +00:00
|
|
|
|
2021-06-06 15:07:55 +00:00
|
|
|
virtual void Remove( EDA_ITEM *aItem );
|
2017-02-28 03:04:44 +00:00
|
|
|
|
|
|
|
virtual void Clear() override
|
|
|
|
{
|
|
|
|
m_items.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned int GetSize() const override
|
|
|
|
{
|
|
|
|
return m_items.size();
|
|
|
|
}
|
|
|
|
|
2021-06-06 15:07:55 +00:00
|
|
|
virtual KIGFX::VIEW_ITEM* GetItem( unsigned int aIdx ) const override;
|
2018-10-24 02:14:11 +00:00
|
|
|
|
2021-06-06 15:07:55 +00:00
|
|
|
bool Contains( EDA_ITEM* aItem ) const;
|
2017-02-28 03:04:44 +00:00
|
|
|
|
|
|
|
/// Checks if there is anything selected
|
|
|
|
bool Empty() const
|
|
|
|
{
|
2019-06-08 21:48:22 +00:00
|
|
|
return m_items.empty();
|
2017-02-28 03:04:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the number of selected parts
|
|
|
|
int Size() const
|
|
|
|
{
|
|
|
|
return m_items.size();
|
|
|
|
}
|
|
|
|
|
2018-10-24 02:14:11 +00:00
|
|
|
const std::deque<EDA_ITEM*> GetItems() const
|
2017-02-28 03:04:44 +00:00
|
|
|
{
|
|
|
|
return m_items;
|
|
|
|
}
|
2020-01-06 19:55:39 +00:00
|
|
|
|
2017-02-28 03:04:44 +00:00
|
|
|
/// Returns the center point of the selection area bounding box.
|
2021-06-06 15:07:55 +00:00
|
|
|
virtual VECTOR2I GetCenter() const;
|
2017-02-28 03:04:44 +00:00
|
|
|
|
2019-06-08 21:48:22 +00:00
|
|
|
virtual const BOX2I ViewBBox() const override
|
|
|
|
{
|
|
|
|
BOX2I r;
|
|
|
|
r.SetMaximum();
|
|
|
|
return r;
|
|
|
|
}
|
2017-03-02 23:42:23 +00:00
|
|
|
|
2017-04-22 20:07:29 +00:00
|
|
|
/// Returns the top left point of the selection area bounding box.
|
2019-06-08 21:48:22 +00:00
|
|
|
VECTOR2I GetPosition() const
|
|
|
|
{
|
|
|
|
return static_cast<VECTOR2I>( GetBoundingBox().GetPosition() );
|
|
|
|
}
|
|
|
|
|
2021-06-06 15:07:55 +00:00
|
|
|
virtual EDA_RECT GetBoundingBox() const;
|
2020-01-06 19:55:39 +00:00
|
|
|
|
2019-06-08 21:48:22 +00:00
|
|
|
virtual EDA_ITEM* GetTopLeftItem( bool onlyModules = false ) const
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
2017-04-22 20:07:29 +00:00
|
|
|
|
2018-10-24 02:14:11 +00:00
|
|
|
EDA_ITEM* operator[]( const size_t aIdx ) const
|
2017-02-28 03:04:44 +00:00
|
|
|
{
|
2018-10-24 02:14:11 +00:00
|
|
|
if( aIdx < m_items.size() )
|
|
|
|
return m_items[ aIdx ];
|
2017-02-28 03:04:44 +00:00
|
|
|
|
2018-10-24 02:14:11 +00:00
|
|
|
return nullptr;
|
2017-02-28 03:04:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
EDA_ITEM* Front() const
|
|
|
|
{
|
2019-05-02 10:59:36 +00:00
|
|
|
return m_items.size() ? m_items.front() : nullptr;
|
2017-02-28 03:04:44 +00:00
|
|
|
}
|
|
|
|
|
2018-10-24 02:14:11 +00:00
|
|
|
std::deque<EDA_ITEM*>& Items()
|
2017-02-28 03:04:44 +00:00
|
|
|
{
|
|
|
|
return m_items;
|
|
|
|
}
|
|
|
|
|
2017-03-03 12:42:28 +00:00
|
|
|
template<class T>
|
|
|
|
T* FirstOfKind() const
|
|
|
|
{
|
|
|
|
for( auto item : m_items )
|
|
|
|
{
|
2021-06-06 16:26:12 +00:00
|
|
|
if( IsA<T, EDA_ITEM>( item ) )
|
2017-03-03 12:42:28 +00:00
|
|
|
return static_cast<T*> ( item );
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-02-12 09:17:35 +00:00
|
|
|
/**
|
|
|
|
* Checks if there is at least one item of requested kind.
|
|
|
|
*
|
|
|
|
* @param aType is the type to check for.
|
|
|
|
* @return True if there is at least one item of such kind.
|
|
|
|
*/
|
2021-06-06 15:07:55 +00:00
|
|
|
bool HasType( KICAD_T aType ) const;
|
2018-02-12 09:17:35 +00:00
|
|
|
|
2021-06-06 15:07:55 +00:00
|
|
|
virtual const VIEW_GROUP::ITEMS updateDrawList() const override;
|
2017-02-28 03:04:44 +00:00
|
|
|
|
2017-09-22 15:17:38 +00:00
|
|
|
bool HasReferencePoint() const
|
|
|
|
{
|
2017-11-01 11:14:16 +00:00
|
|
|
return m_referencePoint != NULLOPT;
|
2017-09-22 15:17:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VECTOR2I GetReferencePoint() const
|
|
|
|
{
|
|
|
|
return *m_referencePoint;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetReferencePoint( const VECTOR2I& aP )
|
|
|
|
{
|
|
|
|
m_referencePoint = aP;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClearReferencePoint()
|
|
|
|
{
|
2017-11-01 11:14:16 +00:00
|
|
|
m_referencePoint = NULLOPT;
|
2017-09-22 15:17:38 +00:00
|
|
|
}
|
|
|
|
|
2019-12-31 04:05:51 +00:00
|
|
|
/**
|
|
|
|
* Checks if all items in the selection are the same KICAD_T type
|
|
|
|
*
|
|
|
|
* @return True if all items are the same type, this includes zero or single items
|
|
|
|
*/
|
2021-06-06 15:07:55 +00:00
|
|
|
bool AreAllItemsIdentical() const;
|
2019-12-31 04:05:51 +00:00
|
|
|
|
2019-06-08 21:48:22 +00:00
|
|
|
protected:
|
|
|
|
OPT<VECTOR2I> m_referencePoint;
|
2018-10-24 02:14:11 +00:00
|
|
|
std::deque<EDA_ITEM*> m_items;
|
2019-06-08 21:48:22 +00:00
|
|
|
bool m_isHover;
|
2017-03-03 12:42:28 +00:00
|
|
|
|
2017-02-28 03:04:44 +00:00
|
|
|
// mute hidden overloaded virtual function warnings
|
|
|
|
using VIEW_GROUP::Add;
|
|
|
|
using VIEW_GROUP::Remove;
|
|
|
|
};
|
|
|
|
|
2017-03-03 12:42:28 +00:00
|
|
|
|
2017-02-28 03:04:44 +00:00
|
|
|
#endif
|