#region License Information (GPL v3) /* ShareX - A program that allows you to take screenshots and share any file type Copyright (c) 2007-2017 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 . */ #endregion License Information (GPL v3) using ShareX.HelpersLib; using System; using System.Drawing; using System.Drawing.Drawing2D; namespace ShareX.ScreenCaptureLib { public class SpeechBalloonDrawingShape : TextDrawingShape { public override ShapeType ShapeType { get; } = ShapeType.DrawingSpeechBalloon; private Point tailPosition; public Point TailPosition { get { return tailPosition; } private set { tailPosition = value; TailNode.Position = tailPosition; } } public bool TailVisible => !Rectangle.Contains(TailPosition); internal ResizeNode TailNode => Manager.ResizeNodes[(int)NodePosition.Extra]; // If rectangle average size is 100px then tail width will be 30px protected const float TailWidthMultiplier = 0.3f; public override void OnCreated() { base.OnCreated(); TailPosition = Rectangle.Location.Add(0, Rectangle.Height + 30); } public override void OnNodeVisible() { base.OnNodeVisible(); TailNode.Position = TailPosition; TailNode.Shape = NodeShape.Circle; TailNode.Visible = true; } public override void OnNodeUpdate() { base.OnNodeUpdate(); if (TailNode.IsDragging) { TailPosition = InputManager.MousePosition0Based; } } public override void Move(int x, int y) { base.Move(x, y); TailPosition = TailPosition.Add(x, y); } public override void OnDraw(Graphics g) { if (Rectangle.Width > 10 && Rectangle.Height > 10) { DrawSpeechBalloon(g); DrawText(g); } } protected void DrawSpeechBalloon(Graphics g) { if (Shadow && IsBorderVisible) { DrawSpeechBalloon(g, ShadowColor, BorderSize, Color.Transparent, Rectangle.LocationOffset(ShadowOffset), TailPosition.Add(ShadowOffset)); } DrawSpeechBalloon(g, BorderColor, BorderSize, FillColor, Rectangle, TailPosition); } protected void DrawSpeechBalloon(Graphics g, Color borderColor, int borderSize, Color fillColor, Rectangle rect, Point tailPosition) { GraphicsPath gpTail = null; if (TailVisible) { gpTail = CreateTailPath(rect, tailPosition); } if (fillColor.A > 0) { using (Brush brush = new SolidBrush(fillColor)) { g.FillRectangle(brush, rect); } } if (gpTail != null) { g.SmoothingMode = SmoothingMode.HighQuality; if (fillColor.A > 0) { g.ExcludeClip(rect); using (Brush brush = new SolidBrush(fillColor)) { g.FillPath(brush, gpTail); } g.ResetClip(); } if (borderSize > 0 && borderColor.A > 0) { g.ExcludeClip(rect.Offset(-1)); using (Pen pen = new Pen(borderColor, borderSize)) { g.DrawPath(pen, gpTail); } g.ResetClip(); } g.SmoothingMode = SmoothingMode.None; } if (borderSize > 0 && borderColor.A > 0) { if (gpTail != null) { using (Region region = new Region(gpTail)) { g.ExcludeClip(region); } } using (Pen pen = new Pen(borderColor, borderSize) { Alignment = PenAlignment.Inset }) { g.DrawRectangleProper(pen, rect.Offset(borderSize - 1)); } g.ResetClip(); } if (gpTail != null) { gpTail.Dispose(); } } protected GraphicsPath CreateTailPath(Rectangle rect, Point tailPosition) { GraphicsPath gpTail = new GraphicsPath(); Point center = rect.Center(); int rectAverageSize = (rect.Width + rect.Height) / 2; int tailWidth = (int)(TailWidthMultiplier * rectAverageSize); tailWidth = Math.Min(Math.Min(tailWidth, rect.Width), rect.Height); int tailOrigin = tailWidth / 2; int tailLength = (int)MathHelpers.Distance(center, tailPosition); gpTail.AddLine(0, -tailOrigin, 0, tailOrigin); gpTail.AddLine(0, tailOrigin, tailLength, 0); gpTail.CloseFigure(); using (Matrix matrix = new Matrix()) { matrix.Translate(center.X, center.Y); float tailDegree = MathHelpers.LookAtDegree(center, tailPosition); matrix.Rotate(tailDegree); gpTail.Transform(matrix); } return gpTail; } } }