ShareX/ShareX.ScreenCaptureLib/Shapes/BaseShape.cs

538 lines
18 KiB
C#
Raw Normal View History

2015-07-17 03:21:02 +12:00
#region License Information (GPL v3)
/*
ShareX - A program that allows you to take screenshots and share any file type
2024-01-03 12:57:14 +13:00
Copyright (c) 2007-2024 ShareX Team
2015-07-17 03:21:02 +12:00
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, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
Optionally you can also view the license at <http://www.gnu.org/licenses/>.
*/
#endregion License Information (GPL v3)
using ShareX.HelpersLib;
using System;
2015-07-17 03:21:02 +12:00
using System.Drawing;
using System.Drawing.Drawing2D;
2016-08-06 01:35:44 +12:00
using System.Windows.Forms;
2015-07-17 03:21:02 +12:00
namespace ShareX.ScreenCaptureLib
{
public abstract class BaseShape : IDisposable
2015-07-17 03:21:02 +12:00
{
2016-09-02 19:09:24 +12:00
public abstract ShapeCategory ShapeCategory { get; }
public abstract ShapeType ShapeType { get; }
private RectangleF rectangle;
public RectangleF Rectangle
{
get
{
return rectangle;
}
set
{
rectangle = value;
startPosition = rectangle.Location;
endPosition = new PointF(rectangle.X + rectangle.Width - 1, rectangle.Y + rectangle.Height - 1);
}
}
public RectangleF RectangleInsideCanvas => RectangleF.Intersect(Rectangle, Manager.Form.CanvasRectangle);
2017-11-15 01:05:12 +13:00
public bool IsInsideCanvas => !RectangleInsideCanvas.IsEmpty;
2017-12-13 12:02:00 +13:00
public virtual bool LimitRectangleToInsideCanvas { get; }
private PointF startPosition;
public PointF StartPosition
{
get
{
return startPosition;
}
2017-11-16 22:40:56 +13:00
private set
{
startPosition = value;
rectangle = CaptureHelpers.CreateRectangle(startPosition, endPosition);
}
}
private PointF endPosition;
public PointF EndPosition
{
get
{
return endPosition;
}
2017-11-16 22:40:56 +13:00
private set
{
endPosition = value;
rectangle = CaptureHelpers.CreateRectangle(startPosition, endPosition);
}
}
2015-07-17 03:21:02 +12:00
public SizeF InitialSize { get; set; }
2019-07-22 12:17:18 +12:00
public virtual bool IsValidShape => !Rectangle.IsEmpty && Rectangle.Width >= Options.MinimumSize && Rectangle.Height >= Options.MinimumSize;
public virtual bool IsSelectable => Manager.CurrentTool == ShapeType || Manager.CurrentTool == ShapeType.ToolSelect;
public bool ForceProportionalResizing { get; protected set; }
internal ShapeManager Manager { get; set; }
2017-10-24 10:40:37 +13:00
protected InputManager InputManager => Manager.InputManager;
2017-11-07 05:01:02 +13:00
protected RegionCaptureOptions Options => Manager.Options;
protected AnnotationOptions AnnotationOptions => Manager.Options.AnnotationOptions;
2016-08-24 01:03:48 +12:00
private PointF tempNodePos, tempStartPos, tempEndPos;
private RectangleF tempRectangle;
2016-08-06 01:35:44 +12:00
public bool IsHandledBySelectTool
{
get
{
switch (ShapeCategory)
{
case ShapeCategory.Drawing:
case ShapeCategory.Effect:
return true;
default:
return false;
}
}
}
public virtual bool Intersects(PointF position)
2016-05-29 22:10:48 +12:00
{
return Rectangle.Contains(position);
}
2018-08-23 22:20:55 +12:00
internal void ChangeNodeShape(NodeShape nodeShape)
{
foreach (ResizeNode node in Manager.ResizeNodes)
{
node.Shape = nodeShape;
}
}
protected virtual void UseLightResizeNodes()
2018-08-23 22:20:55 +12:00
{
ChangeNodeShape(NodeShape.Square);
}
protected void UpdateNodeShape()
{
if (Options.UseLightResizeNodes)
2018-08-23 22:20:55 +12:00
{
UseLightResizeNodes();
2018-08-23 22:20:55 +12:00
}
else
{
ChangeNodeShape(NodeShape.CustomNode);
}
}
public virtual void ShowNodes()
{
2018-08-23 22:20:55 +12:00
UpdateNodeShape();
Manager.NodesVisible = true;
}
public virtual void Remove()
{
Manager.DeleteShape(this);
}
public void AddShapePath(GraphicsPath gp, int sizeOffset = 0)
2016-05-11 10:17:51 +12:00
{
RectangleF rect = Rectangle;
if (sizeOffset != 0)
{
rect = rect.SizeOffset(sizeOffset);
}
OnShapePathRequested(gp, rect);
}
public virtual void Move(float x, float y)
{
2017-11-16 23:42:13 +13:00
StartPosition = StartPosition.Add(x, y);
EndPosition = EndPosition.Add(x, y);
}
public void Move(PointF offset)
{
2017-10-18 03:44:46 +13:00
Move(offset.X, offset.Y);
}
public void MoveAbsolute(float x, float y, bool center = false)
2020-09-16 07:12:57 +12:00
{
if (center)
{
x -= Rectangle.Size.Width / 2;
y -= Rectangle.Size.Height / 2;
}
Move(x - Rectangle.X, y - Rectangle.Y);
}
public void MoveAbsolute(PointF point, bool center = false)
2020-09-16 07:12:57 +12:00
{
MoveAbsolute(point.X, point.Y, center);
}
public virtual void Resize(int x, int y, bool fromBottomRight)
{
if (fromBottomRight)
{
2017-11-16 19:36:21 +13:00
Rectangle = Rectangle.SizeOffset(x, y);
}
else
{
2017-11-16 19:36:21 +13:00
Rectangle = Rectangle.LocationOffset(x, y).SizeOffset(-x, -y);
}
}
public virtual BaseShape Duplicate()
{
ShapeManager manager = Manager;
Manager = null;
BaseShape shape = this.Copy();
Manager = manager;
shape.Manager = manager;
return shape;
}
2016-08-24 01:03:48 +12:00
public virtual void OnCreating()
{
PointF pos = Manager.Form.ScaledClientMousePosition;
2016-08-24 01:03:48 +12:00
2016-09-02 19:09:24 +12:00
if (Options.IsFixedSize && ShapeCategory == ShapeCategory.Region)
2016-08-24 01:03:48 +12:00
{
Manager.IsMoving = true;
Rectangle = new RectangleF(new PointF(pos.X - (Options.FixedSize.Width / 2), pos.Y - (Options.FixedSize.Height / 2)), Options.FixedSize);
OnCreated();
2016-08-24 01:03:48 +12:00
}
else
{
Manager.IsCreating = true;
Rectangle = new RectangleF(pos.X, pos.Y, 1, 1);
2016-08-24 01:03:48 +12:00
}
}
public virtual void OnCreated()
{
InitialSize = Rectangle.Size;
if (ShapeCategory == ShapeCategory.Drawing || ShapeCategory == ShapeCategory.Effect)
{
Manager.OnImageModified();
}
}
2018-03-01 11:52:11 +13:00
public virtual void OnMoving()
{
}
2018-03-01 11:52:11 +13:00
public virtual void OnMoved()
{
}
2018-03-01 11:52:11 +13:00
public virtual void OnResizing()
{
}
2018-03-01 11:52:11 +13:00
public virtual void OnResized()
{
}
public virtual void OnUpdate()
{
if (Manager.IsCreating)
{
PointF pos = Manager.Form.ScaledClientMousePosition;
if (Manager.IsCornerMoving && !Manager.IsPanning)
{
StartPosition = StartPosition.Add(Manager.Form.ScaledClientMouseVelocity);
}
if (Manager.IsProportionalResizing || ForceProportionalResizing)
{
float degree, startDegree;
if (ShapeType == ShapeType.DrawingLine || ShapeType == ShapeType.DrawingArrow)
{
degree = 45;
startDegree = 0;
}
else
{
degree = 90;
startDegree = 45;
}
pos = CaptureHelpers.SnapPositionToDegree(StartPosition, pos, degree, startDegree).Round();
}
else if (Manager.IsSnapResizing)
{
pos = Manager.SnapPosition(StartPosition, pos);
}
EndPosition = pos;
}
else if (Manager.IsMoving && !Manager.IsPanning)
{
Move(Manager.Form.ScaledClientMouseVelocity);
}
2017-12-13 12:02:00 +13:00
if (LimitRectangleToInsideCanvas)
{
2017-12-13 12:22:24 +13:00
StartPosition = StartPosition.Restrict(Manager.Form.CanvasRectangle);
EndPosition = EndPosition.Restrict(Manager.Form.CanvasRectangle);
2017-12-13 12:02:00 +13:00
}
}
public virtual void OnShapePathRequested(GraphicsPath gp, RectangleF rect)
{
gp.AddRectangle(rect);
}
public virtual void OnConfigLoad()
{
}
public virtual void OnConfigSave()
{
}
public virtual void OnDoubleClicked()
{
}
2016-08-06 01:35:44 +12:00
public virtual void OnNodeVisible()
{
2016-08-25 01:29:50 +12:00
for (int i = 0; i < 8; i++)
2016-08-06 01:35:44 +12:00
{
2016-08-25 01:29:50 +12:00
ResizeNode node = Manager.ResizeNodes[i];
2016-08-06 01:35:44 +12:00
node.Visible = true;
}
}
public virtual void OnNodeUpdate()
{
for (int i = 0; i < 8; i++)
{
2016-08-17 20:35:06 +12:00
ResizeNode node = Manager.ResizeNodes[i];
if (node.IsDragging)
2016-08-06 01:35:44 +12:00
{
Manager.IsResizing = true;
if (!InputManager.IsBeforeMouseDown(MouseButtons.Left))
{
tempNodePos = node.Position;
2017-11-21 08:31:16 +13:00
tempStartPos = Rectangle.Location;
tempEndPos = new PointF(Rectangle.X + Rectangle.Width - 1, Rectangle.Y + Rectangle.Height - 1);
tempRectangle = Rectangle;
2017-11-15 00:21:36 +13:00
OnResizing();
2016-08-06 01:35:44 +12:00
}
if (Manager.IsCornerMoving || Manager.IsPanning)
{
tempStartPos.Offset(Manager.Form.ScaledClientMouseVelocity);
tempEndPos.Offset(Manager.Form.ScaledClientMouseVelocity);
tempNodePos.Offset(Manager.Form.ScaledClientMouseVelocity);
tempRectangle.LocationOffset(Manager.Form.ScaledClientMouseVelocity);
}
PointF pos = Manager.Form.ScaledClientMousePosition;
PointF startPos = tempStartPos;
PointF endPos = tempEndPos;
2016-08-06 01:35:44 +12:00
NodePosition nodePosition = (NodePosition)i;
float x = pos.X - tempNodePos.X;
2016-08-06 01:35:44 +12:00
switch (nodePosition)
{
case NodePosition.TopLeft:
case NodePosition.Left:
case NodePosition.BottomLeft:
startPos.X += x;
2016-08-06 01:35:44 +12:00
break;
case NodePosition.TopRight:
case NodePosition.Right:
case NodePosition.BottomRight:
endPos.X += x;
2016-08-06 01:35:44 +12:00
break;
}
float y = pos.Y - tempNodePos.Y;
2016-08-06 01:35:44 +12:00
switch (nodePosition)
{
case NodePosition.TopLeft:
case NodePosition.Top:
case NodePosition.TopRight:
startPos.Y += y;
2016-08-06 01:35:44 +12:00
break;
case NodePosition.BottomLeft:
case NodePosition.Bottom:
case NodePosition.BottomRight:
endPos.Y += y;
2016-08-06 01:35:44 +12:00
break;
}
StartPosition = startPos;
EndPosition = endPos;
if (Manager.IsProportionalResizing && !InitialSize.IsEmpty)
{
switch (nodePosition)
{
case NodePosition.Top:
case NodePosition.Right:
case NodePosition.Bottom:
case NodePosition.Left:
return;
}
double ratio = Math.Min(Rectangle.Width / (double)InitialSize.Width, Rectangle.Height / (double)InitialSize.Height);
int newWidth = (int)Math.Round(InitialSize.Width * ratio);
int newHeight = (int)Math.Round(InitialSize.Height * ratio);
PointF anchor = new PointF();
switch (nodePosition)
{
case NodePosition.TopLeft:
case NodePosition.Left:
case NodePosition.BottomLeft:
anchor.X = tempRectangle.Right - 1;
break;
case NodePosition.TopRight:
case NodePosition.Right:
case NodePosition.BottomRight:
anchor.X = tempRectangle.X;
break;
}
switch (nodePosition)
{
case NodePosition.TopLeft:
case NodePosition.Top:
case NodePosition.TopRight:
anchor.Y = tempRectangle.Bottom - 1;
break;
case NodePosition.BottomLeft:
case NodePosition.Bottom:
case NodePosition.BottomRight:
anchor.Y = tempRectangle.Y;
break;
}
RectangleF newRect = Rectangle;
if (pos.X < anchor.X)
{
newRect.X = newRect.Right - newWidth;
}
newRect.Width = newWidth;
if (pos.Y < anchor.Y)
{
newRect.Y = newRect.Bottom - newHeight;
}
newRect.Height = newHeight;
Rectangle = newRect;
}
2017-12-13 12:02:00 +13:00
2017-12-13 12:22:24 +13:00
if (LimitRectangleToInsideCanvas)
{
Rectangle = RectangleInsideCanvas;
}
2016-08-06 01:35:44 +12:00
}
}
}
public virtual void OnNodePositionUpdate()
{
float xStart = Rectangle.X;
float xMid = Rectangle.X + (Rectangle.Width / 2);
float xEnd = Rectangle.X + Rectangle.Width - 1;
float yStart = Rectangle.Y;
float yMid = Rectangle.Y + (Rectangle.Height / 2);
float yEnd = Rectangle.Y + Rectangle.Height - 1;
Manager.ResizeNodes[(int)NodePosition.TopLeft].Position = new PointF(xStart, yStart);
Manager.ResizeNodes[(int)NodePosition.Top].Position = new PointF(xMid, yStart);
Manager.ResizeNodes[(int)NodePosition.TopRight].Position = new PointF(xEnd, yStart);
Manager.ResizeNodes[(int)NodePosition.Right].Position = new PointF(xEnd, yMid);
Manager.ResizeNodes[(int)NodePosition.BottomRight].Position = new PointF(xEnd, yEnd);
Manager.ResizeNodes[(int)NodePosition.Bottom].Position = new PointF(xMid, yEnd);
Manager.ResizeNodes[(int)NodePosition.BottomLeft].Position = new PointF(xStart, yEnd);
Manager.ResizeNodes[(int)NodePosition.Left].Position = new PointF(xStart, yMid);
for (int i = 0; i < 8; i++)
{
Manager.ResizeNodes[i].Visible = true;
}
if (Manager.ResizeNodes[(int)NodePosition.Right].Rectangle.IntersectsWith(Manager.ResizeNodes[(int)NodePosition.BottomRight].Rectangle))
{
Manager.ResizeNodes[(int)NodePosition.Left].Visible =
Manager.ResizeNodes[(int)NodePosition.Right].Visible = false;
}
if (Manager.ResizeNodes[(int)NodePosition.Bottom].Rectangle.IntersectsWith(Manager.ResizeNodes[(int)NodePosition.BottomRight].Rectangle))
{
Manager.ResizeNodes[(int)NodePosition.Top].Visible =
Manager.ResizeNodes[(int)NodePosition.Bottom].Visible = false;
}
if (Manager.ResizeNodes[(int)NodePosition.TopRight].Rectangle.IntersectsWith(Manager.ResizeNodes[(int)NodePosition.BottomRight].Rectangle))
{
Manager.ResizeNodes[(int)NodePosition.TopLeft].Visible =
Manager.ResizeNodes[(int)NodePosition.Top].Visible =
Manager.ResizeNodes[(int)NodePosition.TopRight].Visible = false;
}
if (Manager.ResizeNodes[(int)NodePosition.BottomLeft].Rectangle.IntersectsWith(Manager.ResizeNodes[(int)NodePosition.BottomRight].Rectangle))
{
Manager.ResizeNodes[(int)NodePosition.TopLeft].Visible =
Manager.ResizeNodes[(int)NodePosition.Left].Visible =
Manager.ResizeNodes[(int)NodePosition.BottomLeft].Visible = false;
}
2016-08-06 01:35:44 +12:00
}
public virtual void Dispose()
{
}
2015-07-17 03:21:02 +12:00
}
}