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-05-28 12:44:28 +00:00
|
|
|
#include <algorithm>
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <core/optional.h>
|
2018-10-24 02:14:11 +00:00
|
|
|
#include <deque>
|
2019-06-08 21:48:22 +00:00
|
|
|
#include <eda_rect.h>
|
2020-10-14 01:06:53 +00:00
|
|
|
#include <eda_item.h>
|
2017-02-28 03:04:44 +00:00
|
|
|
#include <view/view_group.h>
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2017-02-28 03:04:44 +00:00
|
|
|
virtual void Add( EDA_ITEM* aItem )
|
|
|
|
{
|
2019-05-01 20:15:51 +00:00
|
|
|
// We're not sorting here; this is just a time-optimized way to do an
|
|
|
|
// inclusion check. std::lower_bound will return the first i >= aItem
|
|
|
|
// and the second i > aItem check rules out i == aItem.
|
2018-10-25 18:03:35 +00:00
|
|
|
ITER i = std::lower_bound( m_items.begin(), m_items.end(), aItem );
|
2018-10-24 02:14:11 +00:00
|
|
|
|
|
|
|
if( i == m_items.end() || *i > aItem )
|
|
|
|
m_items.insert( i, aItem );
|
2017-02-28 03:04:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void Remove( EDA_ITEM *aItem )
|
|
|
|
{
|
2018-10-25 18:03:35 +00:00
|
|
|
ITER i = std::lower_bound( m_items.begin(), m_items.end(), aItem );
|
2018-10-24 02:14:11 +00:00
|
|
|
|
|
|
|
if( !( i == m_items.end() || *i > aItem ) )
|
|
|
|
m_items.erase( i );
|
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();
|
|
|
|
}
|
|
|
|
|
2018-10-24 02:14:11 +00:00
|
|
|
virtual KIGFX::VIEW_ITEM* GetItem( unsigned int aIdx ) const override
|
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
|
|
|
}
|
|
|
|
|
|
|
|
bool Contains( EDA_ITEM* aItem ) const
|
|
|
|
{
|
2018-10-24 02:14:11 +00:00
|
|
|
CITER i = std::lower_bound( m_items.begin(), m_items.end(), aItem );
|
|
|
|
|
|
|
|
return !( i == m_items.end() || *i > aItem );
|
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.
|
2019-06-08 21:48:22 +00:00
|
|
|
virtual VECTOR2I GetCenter() const
|
|
|
|
{
|
2020-11-24 18:16:03 +00:00
|
|
|
KICAD_T labelTypes[] = { SCH_LABEL_T, SCH_GLOBAL_LABEL_T, SCH_HIER_LABEL_T, EOT };
|
|
|
|
bool includeLabels = true;
|
|
|
|
|
|
|
|
// If the selection contains at least one non-label then don't include labels when
|
|
|
|
// calculating the centerpoint.
|
|
|
|
|
|
|
|
for( EDA_ITEM* item : m_items )
|
|
|
|
{
|
|
|
|
if( !item->IsType( labelTypes ) )
|
|
|
|
{
|
|
|
|
includeLabels = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
EDA_RECT bbox;
|
|
|
|
|
|
|
|
for( EDA_ITEM* item : m_items )
|
|
|
|
{
|
|
|
|
if( !item->IsType( labelTypes ) || includeLabels )
|
|
|
|
bbox.Merge( item->GetBoundingBox() );
|
|
|
|
}
|
|
|
|
|
|
|
|
return static_cast<VECTOR2I>( bbox.Centre() );
|
2019-06-08 21:48:22 +00:00
|
|
|
}
|
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() );
|
|
|
|
}
|
|
|
|
|
2020-10-30 20:30:23 +00:00
|
|
|
virtual EDA_RECT GetBoundingBox() const
|
2019-06-08 21:48:22 +00:00
|
|
|
{
|
|
|
|
EDA_RECT bbox;
|
|
|
|
|
2020-10-30 20:30:23 +00:00
|
|
|
for( EDA_ITEM* item : m_items )
|
|
|
|
bbox.Merge( item->GetBoundingBox() );
|
2017-04-22 20:07:29 +00:00
|
|
|
|
2019-06-08 21:48:22 +00:00
|
|
|
return bbox;
|
|
|
|
}
|
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
|
|
|
|
{
|
|
|
|
auto refType = T( nullptr ).Type();
|
2017-03-03 19:26:18 +00:00
|
|
|
|
2017-03-03 12:42:28 +00:00
|
|
|
for( auto item : m_items )
|
|
|
|
{
|
|
|
|
if( item->Type() == refType )
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
bool HasType( KICAD_T aType ) const
|
|
|
|
{
|
|
|
|
for( auto item : m_items )
|
|
|
|
{
|
|
|
|
if( item->Type() == aType )
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-06-08 21:48:22 +00:00
|
|
|
virtual const VIEW_GROUP::ITEMS updateDrawList() const override
|
|
|
|
{
|
|
|
|
std::vector<VIEW_ITEM*> items;
|
|
|
|
|
|
|
|
for( auto item : m_items )
|
|
|
|
items.push_back( item );
|
|
|
|
|
|
|
|
return items;
|
|
|
|
}
|
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
|
|
|
|
*/
|
|
|
|
bool AreAllItemsIdentical() const
|
|
|
|
{
|
|
|
|
return ( std::all_of( m_items.begin() + 1, m_items.end(),
|
2020-01-06 19:55:39 +00:00
|
|
|
[&]( const EDA_ITEM* r )
|
2019-12-31 04:05:51 +00:00
|
|
|
{
|
|
|
|
return r->Type() == m_items.front()->Type();
|
|
|
|
} ) );
|
|
|
|
}
|
|
|
|
|
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
|