ShareX/ShareX.ScreenCaptureLib/RegionHelpers/AreaManager.cs
2016-01-03 17:16:01 +02:00

595 lines
18 KiB
C#

#region License Information (GPL v3)
/*
ShareX - A program that allows you to take screenshots and share any file type
Copyright (c) 2007-2016 ShareX Team
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;
using System.Collections.Generic;
using System.Drawing;
using System.Linq;
using System.Windows.Forms;
namespace ShareX.ScreenCaptureLib
{
public class AreaManager
{
public List<RegionInfo> Areas { get; private set; }
public RegionInfo[] ValidAreas
{
get
{
return Areas.Where(x => IsAreaValid(x.Area)).ToArray();
}
}
public int SelectedAreaIndex { get; private set; }
public RegionInfo CurrentRegionInfo
{
get
{
if (SelectedAreaIndex > -1)
{
return Areas[SelectedAreaIndex];
}
return null;
}
}
public Rectangle CurrentArea
{
get
{
if (SelectedAreaIndex > -1)
{
return Areas[SelectedAreaIndex].Area;
}
return Rectangle.Empty;
}
set
{
if (SelectedAreaIndex > -1)
{
Areas[SelectedAreaIndex].Area = value;
}
}
}
public bool IsCurrentAreaValid
{
get
{
return IsAreaValid(CurrentArea);
}
}
public Rectangle CurrentHoverArea { get; private set; }
public bool IsCurrentHoverAreaValid
{
get
{
return !CurrentHoverArea.IsEmpty;
}
}
public float RoundedRectangleRadius { get; set; }
public int RoundedRectangleRadiusIncrement { get; set; }
public TriangleAngle TriangleAngle { get; set; }
public Point CurrentPosition { get; private set; }
public Point PositionOnClick { get; private set; }
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; }
public bool WindowCaptureMode { get; set; }
public bool IncludeControls { get; set; }
public int MinimumSize { get; set; }
private RectangleRegion surface;
public AreaManager(RectangleRegion surface)
{
this.surface = surface;
ResizeManager = new ResizeManager(surface, this);
Areas = new List<RegionInfo>();
SelectedAreaIndex = -1;
RoundedRectangleRadius = 25;
RoundedRectangleRadiusIncrement = 3;
TriangleAngle = TriangleAngle.Top;
MinimumSize = 10;
surface.MouseDown += surface_MouseDown;
surface.MouseUp += surface_MouseUp;
surface.KeyDown += surface_KeyDown;
surface.KeyUp += surface_KeyUp;
}
private void surface_KeyDown(object sender, KeyEventArgs e)
{
switch (e.KeyCode)
{
case Keys.Insert:
if (IsCreating)
{
EndRegionSelection();
}
else
{
int areaIndex = SelectedAreaIndex;
if (ResizeManager.Visible)
{
DeselectArea();
}
if (0 > areaIndex || areaIndex != AreaIntersect())
{
RegionSelection(InputManager.MousePosition);
}
}
break;
case Keys.ShiftKey:
IsProportionalResizing = true;
break;
case Keys.Menu:
IsSnapResizing = true;
break;
case Keys.NumPad1:
ChangeCurrentShape(RegionShape.Rectangle);
break;
case Keys.NumPad2:
ChangeCurrentShape(RegionShape.RoundedRectangle);
break;
case Keys.NumPad3:
ChangeCurrentShape(RegionShape.Ellipse);
break;
case Keys.NumPad4:
ChangeCurrentShape(RegionShape.Triangle);
break;
case Keys.NumPad5:
ChangeCurrentShape(RegionShape.Diamond);
break;
case Keys.Add:
switch (surface.Config.CurrentRegionShape)
{
case RegionShape.RoundedRectangle:
RoundedRectangleRadius += RoundedRectangleRadiusIncrement;
break;
case RegionShape.Triangle:
if (TriangleAngle == TriangleAngle.Left)
{
TriangleAngle = TriangleAngle.Top;
}
else
{
TriangleAngle++;
}
break;
}
UpdateCurrentRegionInfo();
break;
case Keys.Subtract:
switch (surface.Config.CurrentRegionShape)
{
case RegionShape.RoundedRectangle:
RoundedRectangleRadius = Math.Max(0, RoundedRectangleRadius - RoundedRectangleRadiusIncrement);
break;
case RegionShape.Triangle:
if (TriangleAngle == TriangleAngle.Top)
{
TriangleAngle = TriangleAngle.Left;
}
else
{
TriangleAngle--;
}
break;
}
UpdateCurrentRegionInfo();
break;
}
}
private void surface_KeyUp(object sender, KeyEventArgs e)
{
switch (e.KeyCode)
{
case Keys.ShiftKey:
IsProportionalResizing = false;
break;
case Keys.Menu:
IsSnapResizing = false;
break;
case Keys.Delete:
RemoveCurrentArea();
if (IsCreating)
{
EndRegionSelection();
}
break;
}
}
public void Update()
{
if (IsMoving)
{
Rectangle rect = CurrentArea;
rect.X += InputManager.MouseVelocity.X;
rect.Y += InputManager.MouseVelocity.Y;
CurrentArea = rect;
}
if (IsCreating && !CurrentArea.IsEmpty)
{
CurrentPosition = InputManager.MousePosition0Based;
Point newPosition = CurrentPosition;
if (IsProportionalResizing)
{
newPosition = CaptureHelpers.ProportionalPosition(PositionOnClick, CurrentPosition);
}
if (IsSnapResizing)
{
newPosition = SnapPosition(PositionOnClick, newPosition);
}
CurrentArea = CaptureHelpers.CreateRectangle(PositionOnClick, newPosition);
}
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;
}
private void CheckHover()
{
CurrentHoverArea = Rectangle.Empty;
if (!ResizeManager.IsCursorOnNode() && !IsCreating && !IsMoving && !IsResizing)
{
Rectangle hoverArea = GetAreaIntersectWithMouse();
if (!hoverArea.IsEmpty)
{
CurrentHoverArea = hoverArea;
}
else
{
SimpleWindowInfo window = FindSelectedWindow();
if (window != null && !window.Rectangle.IsEmpty)
{
hoverArea = CaptureHelpers.ScreenToClient(window.Rectangle);
CurrentHoverArea = Rectangle.Intersect(surface.ScreenRectangle0Based, hoverArea);
}
}
}
}
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(InputManager.MousePosition));
if (windowInfo != null)
{
return windowInfo.WindowInfo;
}
}
return null;
}
private void surface_MouseDown(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
if (!IsCreating)
{
RegionSelection(e.Location);
}
}
}
private void surface_MouseUp(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
if (IsMoving || IsCreating)
{
EndRegionSelection();
}
}
else if (e.Button == MouseButtons.Right)
{
CancelRegionSelection();
if (IsCreating)
{
EndRegionSelection();
}
}
}
private void RegionSelection(Point location)
{
if (ResizeManager.IsCursorOnNode())
{
return;
}
int areaIndex = AreaIntersect(InputManager.MousePosition0Based);
PositionOnClick = InputManager.MousePosition0Based;
if (areaIndex > -1) // Select area
{
IsMoving = true;
SelectedAreaIndex = areaIndex;
SelectArea();
}
else if (!IsCreating) // Create new area
{
DeselectArea();
Rectangle rect;
if (surface.Config.IsFixedSize)
{
IsMoving = true;
rect = new Rectangle(new Point(location.X - surface.Config.FixedSize.Width / 2, location.Y - surface.Config.FixedSize.Height / 2), surface.Config.FixedSize);
}
else
{
IsCreating = true;
rect = new Rectangle(location, new Size(1, 1));
}
AddRegionInfo(rect);
}
}
private void EndRegionSelection()
{
IsCreating = false;
IsMoving = false;
if (!CurrentArea.IsEmpty)
{
if (!IsCurrentAreaValid)
{
RemoveCurrentArea();
CheckHover();
}
else if (surface.Config.QuickCrop)
{
surface.UpdateRegionPath();
surface.Close(SurfaceResult.Region);
}
else
{
SelectArea();
}
}
if (!CurrentHoverArea.IsEmpty)
{
AddRegionInfo(CurrentHoverArea);
if (surface.Config.QuickCrop)
{
surface.UpdateRegionPath();
surface.Close(SurfaceResult.Region);
}
else
{
SelectArea();
}
}
}
private void CancelRegionSelection()
{
int areaIndex = AreaIntersect();
if (areaIndex > -1)
{
Areas.RemoveAt(areaIndex);
DeselectArea();
}
else
{
surface.Close(SurfaceResult.Close);
}
}
private void ChangeCurrentShape(RegionShape shape)
{
surface.Config.CurrentRegionShape = shape;
UpdateCurrentRegionInfo();
}
private void AddRegionInfo(Rectangle rect)
{
Areas.Add(GetRegionInfo(rect));
SelectedAreaIndex = Areas.Count - 1;
}
public RegionInfo GetRegionInfo(Rectangle rect)
{
RegionInfo regionInfo = new RegionInfo(rect, surface.Config.CurrentRegionShape);
regionInfo.RoundedRectangleRadius = RoundedRectangleRadius;
regionInfo.TriangleAngle = TriangleAngle;
return regionInfo;
}
private void UpdateCurrentRegionInfo()
{
RegionInfo regionInfo = CurrentRegionInfo;
if (regionInfo != null)
{
regionInfo.Shape = surface.Config.CurrentRegionShape;
regionInfo.RoundedRectangleRadius = RoundedRectangleRadius;
regionInfo.TriangleAngle = TriangleAngle;
}
}
private void SelectArea()
{
if (!CurrentArea.IsEmpty && !surface.Config.IsFixedSize)
{
ResizeManager.Show();
}
}
private void DeselectArea()
{
SelectedAreaIndex = -1;
ResizeManager.Hide();
}
private void RemoveCurrentArea()
{
if (SelectedAreaIndex > -1)
{
Areas.RemoveAt(SelectedAreaIndex);
DeselectArea();
}
}
private bool IsAreaValid(Rectangle rect)
{
return !rect.IsEmpty && rect.Width >= MinimumSize && rect.Height >= MinimumSize;
}
public int AreaIntersect(Point mousePosition)
{
for (int i = Areas.Count - 1; i >= 0; i--)
{
if (Areas[i].Area.Contains(mousePosition))
{
return i;
}
}
return -1;
}
public int AreaIntersect()
{
return AreaIntersect(InputManager.MousePosition0Based);
}
public Rectangle GetAreaIntersectWithMouse()
{
int areaIndex = AreaIntersect();
if (areaIndex > -1)
{
return Areas[areaIndex].Area;
}
return Rectangle.Empty;
}
public bool IsAreaIntersect()
{
return AreaIntersect() > -1;
}
public Rectangle CombineAreas()
{
RegionInfo[] areas = ValidAreas;
if (areas.Length > 0)
{
Rectangle rect = areas[0].Area;
for (int i = 1; i < areas.Length; i++)
{
rect = Rectangle.Union(rect, areas[i].Area);
}
return rect;
}
return Rectangle.Empty;
}
}
}