2014-07-09 12:23:13 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2014 CERN
|
2021-01-27 22:15:38 +00:00
|
|
|
* Copyright (C) 2020 KiCad Developers, see AUTHORS.txt for contributors.
|
|
|
|
*
|
2014-07-09 12:23:13 +00:00
|
|
|
* @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
|
|
|
|
*/
|
|
|
|
|
2017-04-18 15:32:05 +00:00
|
|
|
#ifndef ALIGN_DISTRIBUTE_TOOL_H_
|
|
|
|
#define ALIGN_DISTRIBUTE_TOOL_H_
|
2014-07-09 12:23:13 +00:00
|
|
|
|
|
|
|
#include <tool/tool_interactive.h>
|
2020-12-16 13:31:32 +00:00
|
|
|
#include <tools/pcb_selection.h>
|
2020-11-14 18:11:28 +00:00
|
|
|
#include <board_item.h>
|
2018-10-10 05:00:07 +00:00
|
|
|
#include <pcb_base_frame.h>
|
2018-01-31 21:24:32 +00:00
|
|
|
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2020-12-16 13:31:32 +00:00
|
|
|
class PCB_SELECTION_TOOL;
|
2024-04-27 19:57:24 +00:00
|
|
|
class BOARD_COMMIT;
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2017-04-18 15:32:05 +00:00
|
|
|
class ALIGN_DISTRIBUTE_TOOL : public TOOL_INTERACTIVE
|
2014-07-09 12:23:13 +00:00
|
|
|
{
|
|
|
|
public:
|
2017-04-18 15:32:05 +00:00
|
|
|
ALIGN_DISTRIBUTE_TOOL();
|
|
|
|
virtual ~ALIGN_DISTRIBUTE_TOOL();
|
2014-07-09 12:23:13 +00:00
|
|
|
|
|
|
|
/// @copydoc TOOL_INTERACTIVE::Reset()
|
2016-09-24 18:53:15 +00:00
|
|
|
void Reset( RESET_REASON aReason ) override {}
|
2014-07-09 12:23:13 +00:00
|
|
|
|
|
|
|
/// @copydoc TOOL_INTERACTIVE::Init()
|
2016-09-24 18:53:15 +00:00
|
|
|
bool Init() override;
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2015-02-14 20:49:06 +00:00
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Set Y coordinate of the selected items to the value of the top-most selected item Y
|
|
|
|
* coordinate.
|
2015-02-14 20:49:06 +00:00
|
|
|
*/
|
2015-02-14 20:28:47 +00:00
|
|
|
int AlignTop( const TOOL_EVENT& aEvent );
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2015-02-14 20:49:06 +00:00
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Sets Y coordinate of the selected items to the value of the bottom-most selected item Y
|
|
|
|
* coordinate.
|
2015-02-14 20:49:06 +00:00
|
|
|
*/
|
2015-02-14 20:28:47 +00:00
|
|
|
int AlignBottom( const TOOL_EVENT& aEvent );
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2015-02-14 20:49:06 +00:00
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Sets X coordinate of the selected items to the value of the left-most selected item X
|
|
|
|
* coordinate.
|
2015-02-14 20:49:06 +00:00
|
|
|
*/
|
2015-02-14 20:28:47 +00:00
|
|
|
int AlignLeft( const TOOL_EVENT& aEvent );
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2015-02-14 20:49:06 +00:00
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Sets X coordinate of the selected items to the value of the right-most selected item X
|
|
|
|
* coordinate.
|
2015-02-14 20:49:06 +00:00
|
|
|
*/
|
2015-02-14 20:28:47 +00:00
|
|
|
int AlignRight( const TOOL_EVENT& aEvent );
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2018-02-08 16:01:36 +00:00
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Set the x coordinate of the midpoint of each of the selected items to the value of the
|
|
|
|
* x coordinate of the center of the middle selected item.
|
2018-02-08 16:01:36 +00:00
|
|
|
*/
|
|
|
|
int AlignCenterX( const TOOL_EVENT& aEvent );
|
|
|
|
|
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Set the y coordinate of the midpoint of each of the selected items to the value of the
|
|
|
|
* y coordinate of the center of the middle selected item.
|
2018-02-08 16:01:36 +00:00
|
|
|
*/
|
|
|
|
int AlignCenterY( const TOOL_EVENT& aEvent );
|
|
|
|
|
2015-02-14 20:49:06 +00:00
|
|
|
/**
|
Rework item distribution
This splits the tool into two separate tools: by center and
by even gaps. Previously, this was automatically decided, based on
if the items could have any gaps between them. This was unintuitive
as it would appear to arrange by centre point sometimes but not others.
When items aren't all the same width, the results can then be very
different, based only on the starting positions.
The new behaviour is to have a dedicated tool for each, which echos
how graphical programs like Inkscape manage this.
The by-gaps method is then extended to work for overlapping items
(when items overlap, the overlaps are made equal). The logic is
centralised in kimath/geometry, and some QA is added. This should
make it easier to extend to eeschema, for example.
This also (attempts to) address some rounding issues which could
cause minor, but compounding, errors to build up along the list
of items.
Also, fix bugs in the collection filtering - previously items
like markers were filtered out only after the selection size
was used to compute the gaps between items.
2024-05-04 07:41:21 +00:00
|
|
|
* Distribute the selected items in some way
|
2015-02-14 20:49:06 +00:00
|
|
|
*/
|
Rework item distribution
This splits the tool into two separate tools: by center and
by even gaps. Previously, this was automatically decided, based on
if the items could have any gaps between them. This was unintuitive
as it would appear to arrange by centre point sometimes but not others.
When items aren't all the same width, the results can then be very
different, based only on the starting positions.
The new behaviour is to have a dedicated tool for each, which echos
how graphical programs like Inkscape manage this.
The by-gaps method is then extended to work for overlapping items
(when items overlap, the overlaps are made equal). The logic is
centralised in kimath/geometry, and some QA is added. This should
make it easier to extend to eeschema, for example.
This also (attempts to) address some rounding issues which could
cause minor, but compounding, errors to build up along the list
of items.
Also, fix bugs in the collection filtering - previously items
like markers were filtered out only after the selection size
was used to compute the gaps between items.
2024-05-04 07:41:21 +00:00
|
|
|
int DistributeItems( const TOOL_EVENT& aEvent );
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2021-01-27 22:15:38 +00:00
|
|
|
///< Set up handlers for various events.
|
2017-07-31 12:30:51 +00:00
|
|
|
void setTransitions() override;
|
2014-07-09 12:23:13 +00:00
|
|
|
|
2015-04-30 08:46:03 +00:00
|
|
|
private:
|
2018-12-14 12:22:00 +00:00
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Populate two vectors with the sorted selection and sorted locked items.
|
|
|
|
*
|
2020-12-27 15:00:47 +00:00
|
|
|
* Returns the size of aItemsToAlign()
|
2018-12-14 12:22:00 +00:00
|
|
|
*/
|
|
|
|
template< typename T >
|
2022-08-30 14:13:51 +00:00
|
|
|
size_t GetSelections( std::vector<std::pair<BOARD_ITEM*, BOX2I>>& aItemsToAlign,
|
|
|
|
std::vector<std::pair<BOARD_ITEM*, BOX2I>>& aLockedItems,
|
2020-12-27 15:00:47 +00:00
|
|
|
T aCompare );
|
2018-12-14 12:22:00 +00:00
|
|
|
|
|
|
|
template< typename T >
|
2022-08-30 14:13:51 +00:00
|
|
|
int selectTarget( std::vector<std::pair<BOARD_ITEM*, BOX2I>>& aItems,
|
|
|
|
std::vector<std::pair<BOARD_ITEM*, BOX2I>>& aLocked, T aGetValue );
|
2018-12-14 12:22:00 +00:00
|
|
|
|
2017-12-09 19:16:27 +00:00
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Sets X coordinate of the selected items to the value of the left-most selected item
|
|
|
|
* X coordinate.
|
2017-12-09 19:16:27 +00:00
|
|
|
*
|
2021-01-27 22:15:38 +00:00
|
|
|
* @note Uses the bounding box of items, which do not get mirrored even when
|
|
|
|
* the view is mirrored!
|
2017-12-09 19:16:27 +00:00
|
|
|
*/
|
|
|
|
int doAlignLeft();
|
|
|
|
|
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Align selected items using the right edge of their bounding boxes to the right-most item.
|
2017-12-09 19:16:27 +00:00
|
|
|
*
|
2021-01-27 22:15:38 +00:00
|
|
|
* @note Uses the bounding box of items, which do not get mirrored even when
|
|
|
|
* the view is mirrored!
|
2017-12-09 19:16:27 +00:00
|
|
|
*/
|
|
|
|
int doAlignRight();
|
|
|
|
|
2018-02-26 20:54:33 +00:00
|
|
|
/**
|
2021-01-27 22:15:38 +00:00
|
|
|
* Distribute selected items using an even spacing between the centers of their bounding
|
|
|
|
* boxes.
|
2018-02-26 20:54:33 +00:00
|
|
|
*
|
2021-01-27 22:15:38 +00:00
|
|
|
* @note Using the centers of bounding box of items can give unsatisfactory visual results
|
|
|
|
* since items of differing widths will be placed with different gaps. Is only used
|
|
|
|
* if items overlap
|
2018-02-26 20:54:33 +00:00
|
|
|
*/
|
Rework item distribution
This splits the tool into two separate tools: by center and
by even gaps. Previously, this was automatically decided, based on
if the items could have any gaps between them. This was unintuitive
as it would appear to arrange by centre point sometimes but not others.
When items aren't all the same width, the results can then be very
different, based only on the starting positions.
The new behaviour is to have a dedicated tool for each, which echos
how graphical programs like Inkscape manage this.
The by-gaps method is then extended to work for overlapping items
(when items overlap, the overlaps are made equal). The logic is
centralised in kimath/geometry, and some QA is added. This should
make it easier to extend to eeschema, for example.
This also (attempts to) address some rounding issues which could
cause minor, but compounding, errors to build up along the list
of items.
Also, fix bugs in the collection filtering - previously items
like markers were filtered out only after the selection size
was used to compute the gaps between items.
2024-05-04 07:41:21 +00:00
|
|
|
void doDistributeCenters( bool aIsXAxis, std::vector<std::pair<BOARD_ITEM*, BOX2I>>& aItems,
|
|
|
|
BOARD_COMMIT& aCommit ) const;
|
2018-02-26 20:54:33 +00:00
|
|
|
|
|
|
|
/**
|
Rework item distribution
This splits the tool into two separate tools: by center and
by even gaps. Previously, this was automatically decided, based on
if the items could have any gaps between them. This was unintuitive
as it would appear to arrange by centre point sometimes but not others.
When items aren't all the same width, the results can then be very
different, based only on the starting positions.
The new behaviour is to have a dedicated tool for each, which echos
how graphical programs like Inkscape manage this.
The by-gaps method is then extended to work for overlapping items
(when items overlap, the overlaps are made equal). The logic is
centralised in kimath/geometry, and some QA is added. This should
make it easier to extend to eeschema, for example.
This also (attempts to) address some rounding issues which could
cause minor, but compounding, errors to build up along the list
of items.
Also, fix bugs in the collection filtering - previously items
like markers were filtered out only after the selection size
was used to compute the gaps between items.
2024-05-04 07:41:21 +00:00
|
|
|
* Distributes selected items using an even spacing between their bounding boxe
|
|
|
|
* in the x or y axis.
|
2018-02-26 20:54:33 +00:00
|
|
|
*
|
Rework item distribution
This splits the tool into two separate tools: by center and
by even gaps. Previously, this was automatically decided, based on
if the items could have any gaps between them. This was unintuitive
as it would appear to arrange by centre point sometimes but not others.
When items aren't all the same width, the results can then be very
different, based only on the starting positions.
The new behaviour is to have a dedicated tool for each, which echos
how graphical programs like Inkscape manage this.
The by-gaps method is then extended to work for overlapping items
(when items overlap, the overlaps are made equal). The logic is
centralised in kimath/geometry, and some QA is added. This should
make it easier to extend to eeschema, for example.
This also (attempts to) address some rounding issues which could
cause minor, but compounding, errors to build up along the list
of items.
Also, fix bugs in the collection filtering - previously items
like markers were filtered out only after the selection size
was used to compute the gaps between items.
2024-05-04 07:41:21 +00:00
|
|
|
* @note If the total item widths exceed the available space, the overlaps will be
|
|
|
|
* distributed evenly.
|
2018-02-26 20:54:33 +00:00
|
|
|
*/
|
Rework item distribution
This splits the tool into two separate tools: by center and
by even gaps. Previously, this was automatically decided, based on
if the items could have any gaps between them. This was unintuitive
as it would appear to arrange by centre point sometimes but not others.
When items aren't all the same width, the results can then be very
different, based only on the starting positions.
The new behaviour is to have a dedicated tool for each, which echos
how graphical programs like Inkscape manage this.
The by-gaps method is then extended to work for overlapping items
(when items overlap, the overlaps are made equal). The logic is
centralised in kimath/geometry, and some QA is added. This should
make it easier to extend to eeschema, for example.
This also (attempts to) address some rounding issues which could
cause minor, but compounding, errors to build up along the list
of items.
Also, fix bugs in the collection filtering - previously items
like markers were filtered out only after the selection size
was used to compute the gaps between items.
2024-05-04 07:41:21 +00:00
|
|
|
void doDistributeGaps( bool aIsXAxis, std::vector<std::pair<BOARD_ITEM*, BOX2I>>& aItems,
|
|
|
|
BOARD_COMMIT& aCommit ) const;
|
2020-12-16 13:31:32 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
PCB_SELECTION_TOOL* m_selectionTool;
|
2024-05-04 19:28:47 +00:00
|
|
|
CONDITIONAL_MENU* m_placementMenu;
|
2020-12-16 13:31:32 +00:00
|
|
|
PCB_BASE_FRAME* m_frame;
|
2014-07-09 12:23:13 +00:00
|
|
|
};
|
|
|
|
|
2017-04-18 15:32:05 +00:00
|
|
|
#endif /* ALIGN_DISTRIBUTE_TOOL_H_ */
|