ShareX/ShareX.ScreenCaptureLib/RegionHelpers/AreaManager.cs

641 lines
19 KiB
C#
Raw Normal View History

2013-11-03 23:53:49 +13:00
#region License Information (GPL v3)
/*
ShareX - A program that allows you to take screenshots and share any file type
2016-01-04 04:16:01 +13:00
Copyright (c) 2007-2016 ShareX Team
2013-11-03 23:53:49 +13: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)
2014-12-11 09:25:20 +13:00
using ShareX.HelpersLib;
using System;
2013-11-03 23:53:49 +13:00
using System.Collections.Generic;
using System.Drawing;
using System.Linq;
using System.Windows.Forms;
2014-12-11 09:25:20 +13:00
namespace ShareX.ScreenCaptureLib
2013-11-03 23:53:49 +13:00
{
public class AreaManager
{
public List<BaseShape> Shapes { get; private set; } = new List<BaseShape>();
2013-11-03 23:53:49 +13:00
public BaseShape CurrentShape { get; private set; }
public ShapeType CurrentShapeType { get; set; } = ShapeType.RegionRectangle;
public Rectangle CurrentRectangle
2013-11-03 23:53:49 +13:00
{
get
{
if (CurrentShape != null)
{
return CurrentShape.Rectangle;
}
return Rectangle.Empty;
}
set
{
if (CurrentShape != null)
{
CurrentShape.Rectangle = value;
}
}
}
public BaseShape[] Regions
{
get
{
return Shapes.OfType<BaseRegionShape>().ToArray();
2013-11-03 23:53:49 +13:00
}
}
public BaseShape[] ValidRegions
{
get
{
return Regions.Where(x => IsAreaValid(x.Rectangle)).ToArray();
}
}
public bool IsCurrentRegionValid
2013-11-03 23:53:49 +13:00
{
get
{
if (CurrentShape is BaseRegionShape)
2013-11-03 23:53:49 +13:00
{
return IsAreaValid(CurrentRectangle);
2013-11-03 23:53:49 +13:00
}
return false;
2013-11-03 23:53:49 +13:00
}
}
public Rectangle CurrentHoverRectangle { get; private set; }
2013-11-03 23:53:49 +13:00
public bool IsCurrentHoverAreaValid
{
get
{
return !CurrentHoverRectangle.IsEmpty;
2013-11-03 23:53:49 +13:00
}
}
public float RoundedRectangleRadius { get; set; } = 25;
public int RoundedRectangleRadiusIncrement { get; set; } = 3;
public TriangleAngle TriangleAngle { get; set; } = TriangleAngle.Top;
public Point CurrentPosition { get; private set; }
public Point PositionOnClick { get; private set; }
2013-11-03 23:53:49 +13:00
public ResizeManager ResizeManager { get; private set; }
public bool IsCreating { get; private set; }
public bool IsMoving { get; private set; }
public bool IsResizing
{
get
{
return ResizeManager.IsResizing;
}
}
public bool IsProportionalResizing { get; private set; }
public bool IsSnapResizing { get; private set; }
public List<SimpleWindowInfo> Windows { get; set; }
2013-11-03 23:53:49 +13:00
public bool WindowCaptureMode { get; set; }
2014-07-10 09:44:08 +12:00
public bool IncludeControls { get; set; }
public int MinimumSize { get; set; } = 3;
2013-11-03 23:53:49 +13:00
private RectangleRegion surface;
public AreaManager(RectangleRegion surface)
{
this.surface = surface;
ResizeManager = new ResizeManager(surface, this);
2013-11-03 23:53:49 +13:00
surface.MouseDown += surface_MouseDown;
surface.MouseUp += surface_MouseUp;
surface.KeyDown += surface_KeyDown;
surface.KeyUp += surface_KeyUp;
}
private void surface_KeyDown(object sender, KeyEventArgs e)
{
2015-08-05 00:40:25 +12:00
switch (e.KeyCode)
2013-11-03 23:53:49 +13:00
{
case Keys.Insert:
if (IsCreating)
{
EndRegionSelection();
}
else
{
2015-08-21 04:29:10 +12:00
if (ResizeManager.Visible)
{
DeselectArea();
}
if (CurrentShape == null || CurrentShape != AreaIntersect())
2015-08-21 04:29:10 +12:00
{
RegionSelection(InputManager.MousePosition);
}
}
break;
case Keys.ShiftKey:
IsProportionalResizing = true;
break;
case Keys.Menu:
IsSnapResizing = true;
break;
case Keys.NumPad1:
CurrentShapeType = ShapeType.RegionRectangle;
break;
case Keys.NumPad2:
CurrentShapeType = ShapeType.RegionRoundedRectangle;
break;
case Keys.NumPad3:
CurrentShapeType = ShapeType.RegionEllipse;
break;
case Keys.NumPad4:
CurrentShapeType = ShapeType.RegionTriangle;
break;
case Keys.NumPad5:
CurrentShapeType = ShapeType.RegionDiamond;
break;
case Keys.NumPad0:
CurrentShapeType = ShapeType.DrawingRectangle;
break;
case Keys.Add:
switch (CurrentShapeType)
{
case ShapeType.RegionRoundedRectangle:
RoundedRectangleRadius += RoundedRectangleRadiusIncrement;
UpdateRoundedRectangle();
break;
case ShapeType.RegionTriangle:
if (TriangleAngle == TriangleAngle.Left)
{
TriangleAngle = TriangleAngle.Top;
}
else
{
TriangleAngle++;
}
UpdateTriangle();
break;
}
break;
case Keys.Subtract:
switch (CurrentShapeType)
{
case ShapeType.RegionRoundedRectangle:
RoundedRectangleRadius = Math.Max(0, RoundedRectangleRadius - RoundedRectangleRadiusIncrement);
UpdateRoundedRectangle();
break;
case ShapeType.RegionTriangle:
if (TriangleAngle == TriangleAngle.Top)
{
TriangleAngle = TriangleAngle.Left;
}
else
{
TriangleAngle--;
}
UpdateTriangle();
break;
}
break;
}
}
private void UpdateRoundedRectangle()
{
if (CurrentShape != null)
{
RoundedRectangleRegionShape roundedRectangleShape = CurrentShape as RoundedRectangleRegionShape;
if (roundedRectangleShape != null)
{
roundedRectangleShape.Radius = RoundedRectangleRadius;
}
}
}
private void UpdateTriangle()
{
if (CurrentShape != null)
{
TriangleRegionShape triangleShape = CurrentShape as TriangleRegionShape;
if (triangleShape != null)
{
triangleShape.Angle = TriangleAngle;
}
2013-11-03 23:53:49 +13:00
}
}
private void surface_KeyUp(object sender, KeyEventArgs e)
{
switch (e.KeyCode)
2013-11-03 23:53:49 +13:00
{
case Keys.ShiftKey:
IsProportionalResizing = false;
break;
case Keys.Menu:
IsSnapResizing = false;
break;
case Keys.Delete:
RemoveCurrentArea();
2015-08-21 04:29:10 +12:00
if (IsCreating)
{
EndRegionSelection();
}
break;
2013-11-03 23:53:49 +13:00
}
}
public void Update()
{
if (IsMoving)
{
Rectangle rect = CurrentRectangle;
2013-11-03 23:53:49 +13:00
rect.X += InputManager.MouseVelocity.X;
rect.Y += InputManager.MouseVelocity.Y;
CurrentRectangle = rect;
2013-11-03 23:53:49 +13:00
}
if (IsCreating && !CurrentRectangle.IsEmpty)
2013-11-03 23:53:49 +13:00
{
CurrentPosition = InputManager.MousePosition0Based;
2013-11-03 23:53:49 +13:00
Point newPosition = CurrentPosition;
2013-11-03 23:53:49 +13:00
if (IsProportionalResizing)
2013-11-03 23:53:49 +13:00
{
newPosition = CaptureHelpers.ProportionalPosition(PositionOnClick, CurrentPosition);
2013-11-03 23:53:49 +13:00
}
if (IsSnapResizing)
{
newPosition = SnapPosition(PositionOnClick, newPosition);
}
CurrentRectangle = CaptureHelpers.CreateRectangle(PositionOnClick, newPosition);
2013-11-03 23:53:49 +13:00
}
CheckHover();
ResizeManager.Update();
}
private Point SnapPosition(Point posOnClick, Point posCurrent)
{
Rectangle currentRect = CaptureHelpers.CreateRectangle(posOnClick, posCurrent);
Point newPosition = posCurrent;
foreach (SnapSize size in surface.Config.SnapSizes)
{
if (currentRect.Width.IsBetween(size.Width - surface.Config.SnapDistance, size.Width + surface.Config.SnapDistance) ||
currentRect.Height.IsBetween(size.Height - surface.Config.SnapDistance, size.Height + surface.Config.SnapDistance))
{
newPosition = CaptureHelpers.CalculateNewPosition(posOnClick, posCurrent, size);
break;
}
}
Rectangle newRect = CaptureHelpers.CreateRectangle(posOnClick, newPosition);
if (surface.ScreenRectangle0Based.Contains(newRect))
{
return newPosition;
}
return posCurrent;
}
2013-11-03 23:53:49 +13:00
private void CheckHover()
{
CurrentHoverRectangle = Rectangle.Empty;
2013-11-03 23:53:49 +13:00
if (!ResizeManager.IsCursorOnNode() && !IsCreating && !IsMoving && !IsResizing)
{
Rectangle hoverArea = GetAreaIntersectWithMouse();
if (!hoverArea.IsEmpty)
{
CurrentHoverRectangle = hoverArea;
2013-11-03 23:53:49 +13:00
}
else
2013-11-03 23:53:49 +13:00
{
SimpleWindowInfo window = FindSelectedWindow();
2013-11-03 23:53:49 +13:00
if (window != null && !window.Rectangle.IsEmpty)
2013-11-03 23:53:49 +13:00
{
hoverArea = CaptureHelpers.ScreenToClient(window.Rectangle);
CurrentHoverRectangle = Rectangle.Intersect(surface.ScreenRectangle0Based, hoverArea);
2013-11-03 23:53:49 +13:00
}
}
}
}
public SimpleWindowInfo FindSelectedWindow()
{
if (Windows != null)
{
return Windows.FirstOrDefault(x => x.Rectangle.Contains(InputManager.MousePosition));
}
return null;
}
public WindowInfo FindSelectedWindowInfo(Point mousePosition)
{
if (Windows != null)
{
SimpleWindowInfo windowInfo = Windows.FirstOrDefault(x => x.IsWindow && x.Rectangle.Contains(mousePosition));
if (windowInfo != null)
{
return windowInfo.WindowInfo;
}
}
return null;
}
2013-11-03 23:53:49 +13:00
private void surface_MouseDown(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
2015-08-21 04:29:10 +12:00
if (!IsCreating)
{
RegionSelection(e.Location);
}
}
}
private void surface_MouseUp(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
2015-08-28 02:32:30 +12:00
if (IsMoving || IsCreating)
{
EndRegionSelection();
}
}
else if (e.Button == MouseButtons.Right)
{
CancelRegionSelection();
2015-08-28 02:32:30 +12:00
if (IsCreating)
{
EndRegionSelection();
}
}
}
private void RegionSelection(Point location)
{
if (ResizeManager.IsCursorOnNode())
{
return;
}
BaseShape shape = AreaIntersect(InputManager.MousePosition0Based);
PositionOnClick = InputManager.MousePosition0Based;
2013-11-03 23:53:49 +13:00
if (shape != null) // Select area
{
IsMoving = true;
CurrentShape = shape;
SelectArea();
}
else if (!IsCreating) // Create new area
2013-11-03 23:53:49 +13:00
{
DeselectArea();
2013-11-03 23:53:49 +13:00
Rectangle rect;
if (surface.Config.IsFixedSize)
2013-11-03 23:53:49 +13:00
{
IsMoving = true;
rect = new Rectangle(new Point(location.X - surface.Config.FixedSize.Width / 2, location.Y - surface.Config.FixedSize.Height / 2), surface.Config.FixedSize);
2013-11-03 23:53:49 +13:00
}
else
2013-11-03 23:53:49 +13:00
{
IsCreating = true;
rect = new Rectangle(location, new Size(1, 1));
2013-11-03 23:53:49 +13:00
}
AddRegionShape(rect);
2013-11-03 23:53:49 +13:00
}
}
private void EndRegionSelection()
2013-11-03 23:53:49 +13:00
{
IsCreating = false;
IsMoving = false;
2013-11-03 23:53:49 +13:00
if (!CurrentRectangle.IsEmpty)
{
if (!IsCurrentRegionValid)
2013-11-03 23:53:49 +13:00
{
RemoveCurrentArea();
CheckHover();
2013-11-03 23:53:49 +13:00
}
else if (surface.Config.QuickCrop)
2013-11-03 23:53:49 +13:00
{
surface.UpdateRegionPath();
surface.Close(SurfaceResult.Region);
}
else
{
SelectArea();
2013-11-03 23:53:49 +13:00
}
}
if (!CurrentHoverRectangle.IsEmpty)
2013-11-03 23:53:49 +13:00
{
AddRegionShape(CurrentHoverRectangle);
2013-11-03 23:53:49 +13:00
if (surface.Config.QuickCrop)
2013-11-03 23:53:49 +13:00
{
surface.UpdateRegionPath();
surface.Close(SurfaceResult.Region);
2013-11-03 23:53:49 +13:00
}
else
{
SelectArea();
2013-11-03 23:53:49 +13:00
}
}
}
private void CancelRegionSelection()
{
BaseShape shape = AreaIntersect();
if (shape != null)
{
Shapes.Remove(shape);
DeselectArea();
}
else
{
surface.Close(SurfaceResult.Close);
}
}
private void AddRegionShape(Rectangle rect)
{
BaseShape shape = CreateRegionShape(rect);
Shapes.Add(shape);
CurrentShape = shape;
}
public BaseShape CreateRegionShape(Rectangle rect)
{
BaseShape shape;
2015-07-17 07:51:19 +12:00
switch (CurrentShapeType)
2015-07-17 07:51:19 +12:00
{
default:
case ShapeType.RegionRectangle:
shape = new RectangleRegionShape();
break;
case ShapeType.RegionRoundedRectangle:
shape = new RoundedRectangleRegionShape()
{
Radius = RoundedRectangleRadius
};
break;
case ShapeType.RegionEllipse:
shape = new EllipseRegionShape();
break;
case ShapeType.RegionTriangle:
shape = new TriangleRegionShape()
{
Angle = TriangleAngle
};
break;
case ShapeType.RegionDiamond:
shape = new DiamondRegionShape();
break;
case ShapeType.DrawingRectangle:
shape = new RectangleDrawingShape();
break;
case ShapeType.DrawingRoundedRectangle:
shape = new RoundedRectangleDrawingShape();
break;
2015-07-17 07:51:19 +12:00
}
shape.Rectangle = rect;
return shape;
}
2013-11-03 23:53:49 +13:00
private void SelectArea()
{
if (!CurrentRectangle.IsEmpty && !surface.Config.IsFixedSize)
2013-11-03 23:53:49 +13:00
{
ResizeManager.Show();
}
}
private void DeselectArea()
{
CurrentShape = null;
2013-11-03 23:53:49 +13:00
ResizeManager.Hide();
}
private void RemoveCurrentArea()
{
BaseShape shape = CurrentShape;
if (shape != null)
2013-11-03 23:53:49 +13:00
{
Shapes.Remove(shape);
2013-11-03 23:53:49 +13:00
DeselectArea();
}
}
private bool IsAreaValid(Rectangle rect)
{
return !rect.IsEmpty && rect.Width >= MinimumSize && rect.Height >= MinimumSize;
}
public BaseShape AreaIntersect(Point mousePosition)
2013-11-03 23:53:49 +13:00
{
for (int i = Shapes.Count - 1; i >= 0; i--)
2013-11-03 23:53:49 +13:00
{
if (Shapes[i].Rectangle.Contains(mousePosition))
2013-11-03 23:53:49 +13:00
{
return Shapes[i];
2013-11-03 23:53:49 +13:00
}
}
return null;
2013-11-03 23:53:49 +13:00
}
public BaseShape AreaIntersect()
2013-11-03 23:53:49 +13:00
{
return AreaIntersect(InputManager.MousePosition0Based);
}
public Rectangle GetAreaIntersectWithMouse()
{
BaseShape shape = AreaIntersect();
2013-11-03 23:53:49 +13:00
if (shape != null)
2013-11-03 23:53:49 +13:00
{
return shape.Rectangle;
2013-11-03 23:53:49 +13:00
}
return Rectangle.Empty;
}
public bool IsAreaIntersect()
{
return AreaIntersect() != null;
2013-11-03 23:53:49 +13:00
}
public Rectangle CombineAreas()
{
BaseShape[] areas = ValidRegions;
2013-11-03 23:53:49 +13:00
2015-07-17 03:21:02 +12:00
if (areas.Length > 0)
2013-11-03 23:53:49 +13:00
{
Rectangle rect = areas[0].Rectangle;
2013-11-03 23:53:49 +13:00
2015-07-17 03:21:02 +12:00
for (int i = 1; i < areas.Length; i++)
2013-11-03 23:53:49 +13:00
{
rect = Rectangle.Union(rect, areas[i].Rectangle);
2013-11-03 23:53:49 +13:00
}
return rect;
}
return Rectangle.Empty;
}
}
2015-09-01 22:43:03 +12:00
}