#region PDFsharp - A .NET library for processing PDF
//
// Authors:
// Stefan Lange
//
// Copyright (c) 2005-2017 empira Software GmbH, Cologne Area (Germany)
//
// http://www.pdfsharp.com
// http://sourceforge.net/projects/pdfsharp
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the "Software"),
// to deal in the Software without restriction, including without limitation
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
// and/or sell copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included
// in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.
#endregion
using System;
using PdfSharp.Internal;
#if GDI
using System.Drawing;
using System.Drawing.Drawing2D;
using GdiPen = System.Drawing.Pen;
#endif
#if WPF
using System.Windows;
using System.Windows.Media;
using WpfPen =System.Windows.Media.Pen;
using WpfBrush =System.Windows.Media.Brush;
#endif
#if UWP
#endif
namespace PdfSharp.Drawing
{
// TODO Free GDI objects (pens, brushes, ...) automatically without IDisposable.
///
/// Defines an object used to draw lines and curves.
///
public sealed class XPen
{
///
/// Initializes a new instance of the class.
///
public XPen(XColor color)
: this(color, 1, false)
{ }
///
/// Initializes a new instance of the class.
///
public XPen(XColor color, double width)
: this(color, width, false)
{ }
internal XPen(XColor color, double width, bool immutable)
{
_color = color;
_width = width;
_lineJoin = XLineJoin.Miter;
_lineCap = XLineCap.Flat;
_dashStyle = XDashStyle.Solid;
_dashOffset = 0f;
_immutable = immutable;
}
///
/// Initializes a new instance of the class.
///
public XPen(XPen pen)
{
_color = pen._color;
_width = pen._width;
_lineJoin = pen._lineJoin;
_lineCap = pen._lineCap;
_dashStyle = pen._dashStyle;
_dashOffset = pen._dashOffset;
_dashPattern = pen._dashPattern;
if (_dashPattern != null)
_dashPattern = (double[])_dashPattern.Clone();
}
///
/// Clones this instance.
///
public XPen Clone()
{
return new XPen(this);
}
///
/// Gets or sets the color.
///
public XColor Color
{
get { return _color; }
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
_dirty = _dirty || _color != value;
_color = value;
}
}
internal XColor _color;
///
/// Gets or sets the width.
///
public double Width
{
get { return _width; }
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
_dirty = _dirty || _width != value;
_width = value;
}
}
internal double _width;
///
/// Gets or sets the line join.
///
public XLineJoin LineJoin
{
get { return _lineJoin; }
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
_dirty = _dirty || _lineJoin != value;
_lineJoin = value;
}
}
internal XLineJoin _lineJoin;
///
/// Gets or sets the line cap.
///
public XLineCap LineCap
{
get { return _lineCap; }
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
_dirty = _dirty || _lineCap != value;
_lineCap = value;
}
}
internal XLineCap _lineCap;
///
/// Gets or sets the miter limit.
///
public double MiterLimit
{
get { return _miterLimit; }
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
_dirty = _dirty || _miterLimit != value;
_miterLimit = value;
}
}
internal double _miterLimit;
///
/// Gets or sets the dash style.
///
public XDashStyle DashStyle
{
get { return _dashStyle; }
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
_dirty = _dirty || _dashStyle != value;
_dashStyle = value;
}
}
internal XDashStyle _dashStyle;
///
/// Gets or sets the dash offset.
///
public double DashOffset
{
get { return _dashOffset; }
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
_dirty = _dirty || _dashOffset != value;
_dashOffset = value;
}
}
internal double _dashOffset;
///
/// Gets or sets the dash pattern.
///
public double[] DashPattern
{
get
{
if (_dashPattern == null)
_dashPattern = new double[0];
return _dashPattern;
}
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
int length = value.Length;
//if (length == 0)
// throw new ArgumentException("Dash pattern array must not be empty.");
for (int idx = 0; idx < length; idx++)
{
if (value[idx] <= 0)
throw new ArgumentException("Dash pattern value must greater than zero.");
}
_dirty = true;
_dashStyle = XDashStyle.Custom;
_dashPattern = (double[])value.Clone();
}
}
internal double[] _dashPattern;
///
/// Gets or sets a value indicating whether the pen enables overprint when used in a PDF document.
/// Experimental, takes effect only on CMYK color mode.
///
public bool Overprint
{
get { return _overprint; }
set
{
if (_immutable)
throw new ArgumentException(PSSR.CannotChangeImmutableObject("XPen"));
_overprint = value;
}
}
internal bool _overprint;
#if GDI
#if UseGdiObjects
///
/// Implicit conversion from Pen to XPen
///
public static implicit operator XPen(Pen pen)
{
XPen xpen;
try
{
Lock.EnterGdiPlus();
switch (pen.PenType)
{
case PenType.SolidColor:
xpen = new XPen(pen.Color, pen.Width);
xpen.LineJoin = (XLineJoin)pen.LineJoin;
xpen.DashStyle = (XDashStyle)pen.DashStyle;
xpen._miterLimit = pen.MiterLimit;
break;
default:
throw new NotImplementedException("Pen type not supported by PDFsharp.");
}
// Bug fixed by drice2@ageone.de
if (pen.DashStyle == System.Drawing.Drawing2D.DashStyle.Custom)
{
int length = pen.DashPattern.Length;
double[] pattern = new double[length];
for (int idx = 0; idx < length; idx++)
pattern[idx] = pen.DashPattern[idx];
xpen.DashPattern = pattern;
xpen._dashOffset = pen.DashOffset;
}
}
finally { Lock.ExitGdiPlus(); }
return xpen;
}
#endif
internal System.Drawing.Pen RealizeGdiPen()
{
if (_dirty)
{
if (_gdiPen == null)
_gdiPen = new System.Drawing.Pen(_color.ToGdiColor(), (float)_width);
else
{
_gdiPen.Color = _color.ToGdiColor();
_gdiPen.Width = (float)_width;
}
LineCap lineCap = XConvert.ToLineCap(_lineCap);
_gdiPen.StartCap = lineCap;
_gdiPen.EndCap = lineCap;
_gdiPen.LineJoin = XConvert.ToLineJoin(_lineJoin);
_gdiPen.DashOffset = (float)_dashOffset;
if (_dashStyle == XDashStyle.Custom)
{
int len = _dashPattern == null ? 0 : _dashPattern.Length;
float[] pattern = new float[len];
for (int idx = 0; idx < len; idx++)
pattern[idx] = (float)_dashPattern[idx];
_gdiPen.DashPattern = pattern;
}
else
_gdiPen.DashStyle = (System.Drawing.Drawing2D.DashStyle)_dashStyle;
}
return _gdiPen;
}
#endif
#if WPF
internal WpfPen RealizeWpfPen()
{
#if !SILVERLIGHT
if (_dirty || !_dirty) // TODOWPF: XPen is frozen by design, WPF Pen can change
{
//if (_wpfPen == null)
_wpfPen = new WpfPen(new SolidColorBrush(_color.ToWpfColor()), _width);
//else
//{
// _wpfPen.Brush = new SolidColorBrush(_color.ToWpfColor());
// _wpfPen.Thickness = _width;
//}
PenLineCap lineCap = XConvert.ToPenLineCap(_lineCap);
_wpfPen.StartLineCap = lineCap;
_wpfPen.EndLineCap = lineCap;
_wpfPen.LineJoin = XConvert.ToPenLineJoin(_lineJoin);
if (_dashStyle == XDashStyle.Custom)
{
// TODOWPF: does not work in all cases
_wpfPen.DashStyle = new System.Windows.Media.DashStyle(_dashPattern, _dashOffset);
}
else
{
switch (_dashStyle)
{
case XDashStyle.Solid:
_wpfPen.DashStyle = DashStyles.Solid;
break;
case XDashStyle.Dash:
//_wpfPen.DashStyle = DashStyles.Dash;
_wpfPen.DashStyle = new System.Windows.Media.DashStyle(new double[] { 2, 2 }, 0);
break;
case XDashStyle.Dot:
//_wpfPen.DashStyle = DashStyles.Dot;
_wpfPen.DashStyle = new System.Windows.Media.DashStyle(new double[] { 0, 2 }, 1.5);
break;
case XDashStyle.DashDot:
//_wpfPen.DashStyle = DashStyles.DashDot;
_wpfPen.DashStyle = new System.Windows.Media.DashStyle(new double[] { 2, 2, 0, 2 }, 0);
break;
case XDashStyle.DashDotDot:
//_wpfPen.DashStyle = DashStyles.DashDotDot;
_wpfPen.DashStyle = new System.Windows.Media.DashStyle(new double[] { 2, 2, 0, 2, 0, 2 }, 0);
break;
}
}
}
#else
_wpfPen = new System.Windows.Media.Pen();
_wpfPen.Brush = new SolidColorBrush(_color.ToWpfColor());
_wpfPen.Thickness = _width;
#endif
return _wpfPen;
}
#endif
bool _dirty = true;
readonly bool _immutable;
#if GDI
GdiPen _gdiPen;
#endif
#if WPF
WpfPen _wpfPen;
#endif
}
}