123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164 |
- using System;
- using System.IO;
- using System.Collections;
- using System.Collections.Generic;
- using System.Drawing;
- using System.ComponentModel;
- using FastReport.Utils;
- #if NETSTANDARD || NETCOREAPP
- using FastReport.Code.CodeDom.Compiler;
- #else
- using System.CodeDom.Compiler;
- #endif
- using System.Drawing.Design;
- namespace FastReport
- {
- /// <summary>
- /// Specifies a set of actions that cannot be performed on the object in the design mode.
- /// </summary>
- [Flags]
- public enum Restrictions
- {
- /// <summary>
- /// Specifies no restrictions.
- /// </summary>
- None = 0,
- /// <summary>
- /// Restricts moving the object.
- /// </summary>
- DontMove = 1,
- /// <summary>
- /// Restricts resizing the object.
- /// </summary>
- DontResize = 2,
- /// <summary>
- /// Restricts modifying the object's properties.
- /// </summary>
- DontModify = 4,
- /// <summary>
- /// Restricts editing the object.
- /// </summary>
- DontEdit = 8,
- /// <summary>
- /// Restricts deleting the object.
- /// </summary>
- DontDelete = 16,
- /// <summary>
- /// Hides all properties of the object.
- /// </summary>
- HideAllProperties = 32
- }
- /// <summary>
- /// Specifies a set of actions that can be performed on the object in the design mode.
- /// </summary>
- [Flags]
- public enum Flags
- {
- /// <summary>
- /// Specifies no actions.
- /// </summary>
- None = 0,
- /// <summary>
- /// Allows moving the object.
- /// </summary>
- CanMove = 1,
- /// <summary>
- /// Allows resizing the object.
- /// </summary>
- CanResize = 2,
- /// <summary>
- /// Allows deleting the object.
- /// </summary>
- CanDelete = 4,
- /// <summary>
- /// Allows editing the object.
- /// </summary>
- CanEdit = 8,
- /// <summary>
- /// Allows changing the Z-order of an object.
- /// </summary>
- CanChangeOrder = 16,
- /// <summary>
- /// Allows moving the object to another parent.
- /// </summary>
- CanChangeParent = 32,
- /// <summary>
- /// Allows copying the object to the clipboard.
- /// </summary>
- CanCopy = 64,
- /// <summary>
- /// Allows drawing the object.
- /// </summary>
- CanDraw = 128,
- /// <summary>
- /// Allows grouping the object.
- /// </summary>
- CanGroup = 256,
- /// <summary>
- /// Allows write children in the preview mode by itself.
- /// </summary>
- CanWriteChildren = 512,
- /// <summary>
- /// Allows write object's bounds into the report stream.
- /// </summary>
- CanWriteBounds = 1024,
- /// <summary>
- /// Allows the "smart tag" functionality.
- /// </summary>
- HasSmartTag = 2048,
- /// <summary>
- /// Specifies that the object's name is global (this is true for all report objects
- /// such as Text, Picture and so on).
- /// </summary>
- HasGlobalName = 4096,
- /// <summary>
- /// Specifies that the object can display children in the designer's Report Tree window.
- /// </summary>
- CanShowChildrenInReportTree = 8192,
- /// <summary>
- /// Specifies that the object supports mouse wheel in the preview window.
- /// </summary>
- InterceptsPreviewMouseEvents = 16384
- }
- /// <summary>
- /// Represents the root class of the FastReport object's hierarhy.
- /// </summary>
- [ToolboxItem(false)]
- public abstract partial class Base : Component, IFRSerializable
- {
- #region Fields
- private string name;
- private Restrictions restrictions;
- private Flags flags;
- private Base parent;
- private string baseName;
- private bool isAncestor;
- private bool isDesigning;
- private bool isPrinting;
- private bool isRunning;
- private Base originalComponent;
- private string alias;
- private Report report;
- private int zOrder;
- #endregion
- #region Properties
- /// <summary>
- /// Gets or sets the name of the object.
- /// </summary>
- /// <remarks>
- /// <para>Name of the report object must contain alpha, digit, underscore symbols only.
- /// Data objects such as <b>Variable</b>, <b>TableDataSource</b>
- /// etc. can have any characters in they names. Each component must have unique
- /// name.</para>
- /// </remarks>
- /// <example>The following code demonstrates how to find an object by its name:
- /// <code>
- /// TextObject text1 = report1.FindObject("Text1") as TextObject;
- /// </code>
- /// </example>
- /// <exception cref="DuplicateNameException" caption="">Another object with such name exists.</exception>
- /// <exception cref="AncestorException" caption="">Rename an object that was introduced in the ancestor report.</exception>
- [MergableProperty(false)]
- [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- [Category("Design")]
- [DisplayName("(Name)")]
- public virtual string Name
- {
- get { return name; }
- set
- {
- if (String.Compare(name, value, true) == 0)
- return;
- if (value != "" && Report != null && HasFlag(Flags.HasGlobalName))
- {
- Base c = Report.FindObject(value);
- if (c != null && c != this)
- throw new DuplicateNameException(value);
- if (IsAncestor)
- throw new AncestorException(name);
- if (IsDesigning)
- CheckValidIdent(value);
- }
- SetName(value);
- }
- }
- /// <summary>
- /// Gets or sets the flags that restrict some actions in the designer.
- /// </summary>
- /// <remarks>
- /// Use this property to restrict some user actions like move, resize, edit, delete. For example, if
- /// <b>Restriction.DontMove</b> flag is set, user cannot move the object in the designer.
- /// </remarks>
- [DefaultValue(Restrictions.None)]
- [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- [Category("Design")]
- [Editor("FastReport.TypeEditors.FlagsEditor, FastReport", typeof(UITypeEditor))]
- public Restrictions Restrictions
- {
- get { return restrictions; }
- set { restrictions = value; }
- }
- /// <summary>
- /// Gets the flags that allow some functionality in the designer.
- /// </summary>
- /// <remarks>
- /// Use this property only if you developing a new FastReport object.
- /// </remarks>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public Flags Flags
- {
- get { return flags; }
- }
- /// <summary>
- /// Gets or sets the parent of the object.
- /// </summary>
- /// <remarks>
- /// <para>Each report object must have a parent in order to appear in the report. Parent must be able to
- /// contain objects of such type.</para>
- /// <para>Another way (preferred) to set a parent is to use specific properties of the parent object.
- /// For example, the <see cref="Report"/> object has the <see cref="FastReport.Report.Pages"/> collection.
- /// To add a new page to the report, use the following code: <c>report1.Pages.Add(new ReportPage());</c>
- /// </para>
- /// </remarks>
- /// <example><code>
- /// Report report1;
- /// ReportPage page = new ReportPage();
- /// page.Parent = report1;
- /// </code></example>
- /// <exception cref="ParentException" caption="">Parent object cannot contain this object.</exception>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public Base Parent
- {
- get { return parent; }
- set
- {
- if (value != parent)
- {
- if (value != null)
- {
- if (value is IParent)
- (value as IParent).AddChild(this);
- }
- else
- {
- (parent as IParent).RemoveChild(this);
- }
- }
- }
- }
- /// <summary>
- /// The base part of the object's name.
- /// </summary>
- /// <remarks>
- /// This property is used to automatically create unique object's name. See <see cref="CreateUniqueName"/>
- /// </remarks>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public string BaseName
- {
- get { return baseName; }
- set { baseName = value; }
- }
- /// <summary>
- /// Gets the short type name.
- /// </summary>
- /// <remarks>
- /// Returns the short type name, such as "TextObject".
- /// </remarks>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public string ClassName
- {
- get { return GetType().Name; }
- }
- /// <summary>
- /// Gets reference to the parent <see cref="Report"/> object.
- /// </summary>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public Report Report
- {
- get
- {
- if (this is Report)
- return this as Report;
- if (report != null)
- return report;
- if (Parent != null)
- return Parent.Report;
- return null;
- }
- }
- /// <summary>
- /// Gets reference to the parent <see cref="PageBase"/> object.
- /// </summary>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public PageBase Page
- {
- get
- {
- if (this is PageBase)
- return (PageBase)this;
- Base c = Parent;
- while (c != null)
- {
- if (c is PageBase)
- return (PageBase)c;
- c = c.Parent;
- }
- return null;
- }
- }
- /// <summary>
- /// Gets the collection of this object's child objects.
- /// </summary>
- /// <remarks>
- /// This property returns child objects that belongs to this object. For example, <b>Report.ChildObjects</b>
- /// will return only pages that contains in the report, but not page childs such as bands. To return all
- /// child objects, use <see cref="AllObjects"/> property.
- /// </remarks>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public ObjectCollection ChildObjects
- {
- get
- {
- ObjectCollection result = new ObjectCollection();
- if (this is IParent)
- (this as IParent).GetChildObjects(result);
- return result;
- }
- }
- /// <summary>
- /// Gets the collection of all child objects.
- /// </summary>
- /// <remarks>
- /// This property returns child objects that belongs to this object and to child objects of this object.
- /// For example, <b>Report.AllObjects</b> will return all objects that contains in the report - such as
- /// pages, bands, text objects.
- /// </remarks>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public ObjectCollection AllObjects
- {
- get
- {
- ObjectCollection result = new ObjectCollection();
- EnumObjects(this, result);
- return result;
- }
- }
- /// <summary>
- /// Gets or sets the Z-order of the object.
- /// </summary>
- /// <remarks>
- /// The Z-order is also called "creation order". It is the index of an object in the parent's objects list.
- /// For example, put two text objects on a band. First object will have <b>ZOrder</b> = 0, second = 1. Setting the
- /// second object's <b>ZOrder</b> to 0 will move it to the back of the first text object.
- /// </remarks>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public int ZOrder
- {
- get
- {
- if (parent != null)
- return (parent as IParent).GetChildOrder(this);
- return zOrder;
- }
- set
- {
- if (parent != null)
- (parent as IParent).SetChildOrder(this, value);
- else
- zOrder = value;
- }
- }
- /// <summary>
- /// Gets a value indicating whether the object was introduced in the ancestor report.
- /// </summary>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public bool IsAncestor
- {
- get { return isAncestor; }
- }
- /// <summary>
- /// Gets a value indicating whether the object is in the design state.
- /// </summary>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public bool IsDesigning
- {
- get { return isDesigning; }
- }
- /// <summary>
- /// Gets a value indicating whether the object is currently printing.
- /// </summary>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public bool IsPrinting
- {
- get { return isPrinting; }
- }
- /// <summary>
- /// Gets a value indicating whether the object is currently processed by the report engine.
- /// </summary>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public bool IsRunning
- {
- get { return isRunning; }
- }
- /// <summary>
- /// Gets an original component for this object.
- /// </summary>
- /// <remarks>
- /// This property is used in the preview mode. Each object in the prepared report is bound to its
- /// original (from the report template). This technique is used to minimize the prepared report's size.
- /// </remarks>
- [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
- public Base OriginalComponent
- {
- get { return originalComponent; }
- set { originalComponent = value; }
- }
- internal string Alias
- {
- get { return alias; }
- set { alias = value; }
- }
- #endregion
- #region Private Methods
- private void CheckValidIdent(string value)
- {
- if (!CodeGenerator.IsValidLanguageIndependentIdentifier(value))
- throw new NotValidIdentifierException(value);
- }
- private void EnumObjects(Base c, ObjectCollection list)
- {
- if (c != this)
- list.Add(c);
- foreach (Base obj in c.ChildObjects)
- EnumObjects(obj, list);
- }
- #endregion
- #region Protected Methods
- /// <summary>
- /// Helper method, helps to set a reference-type value to the property.
- /// </summary>
- /// <param name="prop">Old property value.</param>
- /// <param name="value">New property value.</param>
- /// <remarks>
- /// This method is used widely to set a new value to the property that references another FastReport object.
- /// Method deals with the <see cref="Parent"/> property.
- /// </remarks>
- /// <example>This is example of the <c>DataBand.Header</c> property:<code>
- /// public DataHeaderBand Header
- /// {
- /// get { return FHeader; }
- /// set
- /// {
- /// SetProp(FHeader, value);
- /// FHeader = value;
- /// }
- /// }
- /// </code></example>
- protected void SetProp(Base prop, Base value)
- {
- if (prop != value)
- {
- if (prop != null)
- prop.SetParent(null);
- if (value != null)
- {
- value.Parent = null;
- value.SetParent(this);
- }
- }
- }
- /// <summary>
- /// Checks if two float values are different.
- /// </summary>
- /// <param name="f1">First value.</param>
- /// <param name="f2">Second value.</param>
- /// <returns><c>true</c> if values are not equal.</returns>
- /// <remarks>
- /// This method is needed to compare two float values using some precision (0.001). It is useful
- /// to compare objects' locations and sizes for equality.
- /// </remarks>
- protected bool FloatDiff(float f1, float f2)
- {
- return Math.Abs(f1 - f2) > 0.001;
- }
- /// <summary>
- /// Deserializes nested object properties.
- /// </summary>
- /// <param name="reader">Reader object.</param>
- /// <remarks>
- /// <para>Typically the object serializes all properties to the single xml item:</para>
- /// <code>
- /// <TextObject Name="Text2" Left="18.9" Top="37.8" Width="283.5" Height="28.35"/>
- /// </code>
- /// <para>Some objects like <see cref="DataBand"/> have child objects that serialized in subitems:</para>
- /// <code>
- /// <DataBand Name="Data1" Top="163" Width="718.2" Height="18.9">
- /// <TextObject Name="Text3" Left="18.9" Top="37.8" Width="283.5" Height="28.35"/>
- /// </DataBand>
- /// </code>
- /// <para>To read such subitems, the <c>DeserializeSubItems</c> method is used. Base
- /// implementation reads the child objects. You may override it to read some specific subitems.</para>
- /// </remarks>
- /// <example>The following code is used to read report's styles:
- /// <code>
- /// protected override void DeserializeSubItems(FRReader reader)
- /// {
- /// if (String.Compare(reader.ItemName, "Styles", true) == 0)
- /// reader.Read(Styles);
- /// else
- /// base.DeserializeSubItems(reader);
- /// }
- /// </code>
- /// </example>
- protected virtual void DeserializeSubItems(FRReader reader)
- {
- if (reader.ReadChildren)
- {
- Base obj = reader.Read() as Base;
- if (obj != null)
- {
- obj.Parent = this;
- if (IsAncestor && !obj.IsAncestor)
- obj.ZOrder = obj.zOrder;
- }
- }
- }
- /// <summary>
- /// Replaces the macros in the given string and returns the new string.
- /// </summary>
- /// <param name="text">The text containing macros.</param>
- /// <returns>The text with macros replaced with its values.</returns>
- protected string ExtractDefaultMacros(string text)
- {
- Dictionary<string, object> macroValues = Report.PreparedPages.MacroValues;
- text = ExtractDefaultMacrosInternal(macroValues, text);
- text = text.Replace("[TOTALPAGES#]", macroValues["TotalPages#"].ToString());
- text = text.Replace("[PAGE#]", macroValues["Page#"].ToString());
- return text;
- }
- /// <inheritdoc/>
- protected override void Dispose(bool disposing)
- {
- if (disposing)
- {
- Parent = null;
- Clear();
- }
- base.Dispose(disposing);
- }
- #endregion
- #region Public Methods
- /// <summary>
- /// Set object's flags.
- /// </summary>
- /// <param name="flags">Flag to set.</param>
- /// <param name="value"><b>true</b> to set the flag, <b>false</b> to reset.</param>
- public void SetFlags(Flags flags, bool value)
- {
- if (value)
- this.flags |= flags;
- else
- this.flags &= ~flags;
- }
- internal void SetAncestor(bool value)
- {
- isAncestor = value;
- }
- internal void SetDesigning(bool value)
- {
- isDesigning = value;
- }
- internal void SetPrinting(bool value)
- {
- isPrinting = value;
- }
- internal void SetRunning(bool value)
- {
- isRunning = value;
- }
- /// <summary>
- /// Sets the reference to a Report.
- /// </summary>
- /// <param name="value">Report to set.</param>
- public void SetReport(Report value)
- {
- report = value;
- }
- /// <summary>
- /// Sets the object's name.
- /// </summary>
- /// <remarks>
- /// This method is for internal use only. It just sets a new name without any checks
- /// (unlike the <see cref="Name"/> property setter).
- /// </remarks>
- /// <seealso cref="Name">Name Property</seealso>
- /// <param name="value">New name.</param>
- public virtual void SetName(string value)
- {
- name = value;
- }
- /// <summary>
- /// Sets the object's parent.
- /// </summary>
- /// <remarks>
- /// This method is for internal use only. You can use it if you are developing a new
- /// component for FastReport. Override it to perform some actions when the parent of an
- /// object is changing. This method checks that parent can contain a child.
- /// </remarks>
- /// <exception cref="ParentException" caption="">Parent object cannot contain this object.</exception>
- /// <param name="value">New parent.</param>
- public virtual void SetParent(Base value)
- {
- if (value != null)
- if (!(value is IParent) || !(value as IParent).CanContain(this))
- throw new ParentException(value, this);
- SetParentCore(value);
- }
- /// <summary>
- /// Sets the object's parent.
- /// </summary>
- /// <param name="value">New parent.</param>
- /// <remarks>
- /// This method is for internal use only. You can use it if you are developing a new component for FastReport.
- /// This method does not perform any checks, it just sets the new parent.
- /// </remarks>
- public void SetParentCore(Base value)
- {
- parent = value;
- }
- /// <summary>
- /// Searches for an object with given name.
- /// </summary>
- /// <param name="name">Name of the object to find.</param>
- /// <returns>Returns a null reference if object is not found</returns>
- /// <example>The following code demonstrates how to find an object by its name:
- /// <code>
- /// TextObject text1 = report1.FindObject("Text1") as TextObject;
- /// if (text1 != null)
- /// {
- /// // object found
- /// }
- /// </code>
- /// </example>
- public virtual Base FindObject(string name)
- {
- ObjectCollection l = AllObjects;
- foreach (Base c in l)
- {
- if (name == c.Name)
- return c;
- }
- return null;
- }
- /// <summary>
- /// Creates the unique object's name.
- /// </summary>
- /// <remarks>
- /// <para><b>Note:</b> you have to set object's parent before calling this method. Method uses the <see cref="BaseName"/>
- /// property to create a name.</para>
- /// <para><b>Note:</b> this method may be very slow on a report that contains lots of objects. Consider
- /// using own naming logic in this case.</para>
- /// </remarks>
- /// <example>
- /// <code>
- /// TextObject textObj = new TextObject();
- /// dataBand1.Objects.Add(textObj);
- /// textObj.CreateUniqueName();
- /// </code>
- /// </example>
- public void CreateUniqueName()
- {
- Report report = Report;
- if (report == null)
- return;
- string s;
- int i = 1;
- do
- {
- s = baseName + i.ToString();
- i++;
- }
- while (report.FindObject(s) != null);
- SetName(s);
- }
- /// <summary>
- /// Clears the object's state.
- /// </summary>
- /// <remarks>
- /// This method also disposes all object's children.
- /// </remarks>
- public virtual void Clear()
- {
- ObjectCollection list = ChildObjects;
- foreach (Base c in list)
- {
- c.Clear();
- c.Dispose();
- }
- }
- /// <summary>
- /// Serializes the object.
- /// </summary>
- /// <remarks>
- /// <para>Do not call this method directly. You should override it if you are
- /// developing a new component for FastReport.</para>
- /// <para>This method is called when the object needs to save the state. It may happen
- /// when:</para>
- /// <list type="bullet">
- /// <item>
- /// <description>saving the report to the file or stream;</description>
- /// </item>
- /// <item>
- /// <description>saving the report to the designer's undo buffer;</description>
- /// </item>
- /// <item>
- /// <description>
- /// assigning the object to another object using the
- /// <see cref="Assign"/> or <see cref="AssignAll(Base)">AssignAll</see> methods;
- /// </description>
- /// </item>
- /// <item>
- /// <description>saving the object to the designer's clipboard;</description>
- /// </item>
- /// <item>
- /// <description>saving the object to the preview (when run a
- /// report).</description>
- /// </item>
- /// </list>
- /// </remarks>
- /// <param name="writer">Writer object.</param>
- public virtual void Serialize(FRWriter writer)
- {
- Base c = writer.DiffObject as Base;
- if (writer.SerializeTo != SerializeTo.Preview)
- {
- // in the preview mode we don't need to write ItemName and Name properties. Alias is wriiten instead.
- writer.ItemName = isAncestor &&
- (writer.SerializeTo == SerializeTo.Report || writer.SerializeTo == SerializeTo.Undo) ?
- "inherited" : ClassName;
- if (Name != "")
- writer.WriteStr("Name", Name);
- if (Restrictions != c.Restrictions)
- writer.WriteValue("Restrictions", Restrictions);
- if ((writer.SerializeTo == SerializeTo.Report || writer.SerializeTo == SerializeTo.Undo) &&
- !IsAncestor && Parent != null && Parent.IsAncestor)
- writer.WriteInt("ZOrder", ZOrder);
- }
- if (writer.SaveChildren)
- {
- foreach (Base child in ChildObjects)
- {
- writer.Write(child);
- }
- }
- }
- /// <summary>
- /// Deserializes the object.
- /// </summary>
- /// <remarks>
- /// <para>Do not call this method directly. You should override it if you are
- /// developing a new component for FastReport.</para>
- /// <para>This method is called when the object needs to restore the state. It may
- /// happen when:</para>
- /// <list type="bullet">
- /// <item>
- /// <description>loading the report from a file or stream;</description>
- /// </item>
- /// <item>
- /// <description>loading the report from the designer's undo
- /// buffer;</description>
- /// </item>
- /// <item>
- /// <description>assigning another object to this object using the
- /// <see cref="Assign"/> or <see cref="AssignAll(Base)">AssignAll</see> methods;</description>
- /// </item>
- /// <item>
- /// <description>loading the object from the designer's
- /// clipboard;</description>
- /// </item>
- /// <item>loading the object from the preview pages.</item>
- /// </list>
- /// </remarks>
- /// <param name="reader">Reader object.</param>
- public virtual void Deserialize(FRReader reader)
- {
- reader.ReadProperties(this);
- while (reader.NextItem())
- {
- DeserializeSubItems(reader);
- }
- }
- /// <summary>
- /// Assigns values from another source.
- /// </summary>
- /// <remarks>
- /// <b>Note:</b> this method is relatively slow because it serializes
- /// an object to the xml and then deserializes it.
- /// </remarks>
- /// <param name="source">Source to assign from.</param>
- public void BaseAssign(Base source)
- {
- bool saveAncestor = source.IsAncestor;
- source.SetAncestor(false);
-
- try
- {
- using (XmlItem xml = new XmlItem())
- using (FRWriter writer = new FRWriter(xml))
- using (FRReader reader = new FRReader(Report, xml))
- {
- writer.SaveChildren = false;
- writer.Write(source, this);
- reader.Read(this);
- }
- Alias = source.Alias;
- OriginalComponent = source.OriginalComponent;
- }
- finally
- {
- source.SetAncestor(saveAncestor);
- }
- }
- /// <summary>Copies the contents of another, similar object.</summary>
- /// <remarks>
- /// <para>Call Assign to copy the properties from another object of the same type.
- /// The standard form of a call to Assign is</para>
- /// <para><c>destination.Assign(source);</c></para>
- /// <para>
- /// which tells the <b>destination</b> object to copy the contents of the
- /// <b>source</b> object to itself. In this method, all child objects are
- /// ignored. If you want to copy child objects, use the
- /// <see cref="AssignAll(Base)">AssignAll</see> method.
- /// </para>
- /// </remarks>
- /// <example><code>
- /// Report report1;
- /// Report report2 = new Report();
- /// // copy all report settings, do not copy report objects
- /// report2.Assign(report1);
- /// </code></example>
- /// <seealso cref="AssignAll(Base)">AssignAll Method</seealso>
- /// <param name="source">Source object to copy the contents from.</param>
- public virtual void Assign(Base source)
- {
- Restrictions = source.Restrictions;
- Alias = source.Alias;
- OriginalComponent = source.OriginalComponent;
- }
- /// <summary>Copies the contents (including children) of another, similar object.</summary>
- /// <remarks>
- /// <para>
- /// This method is similar to <see cref="Assign"/> method. It copies child
- /// objects as well.
- /// </para>
- /// </remarks>
- /// <example><code>
- /// Report report1;
- /// Report report2 = new Report();
- /// // copy all report settings and objects
- /// report2.AssignAll(report1);
- /// </code></example>
- /// <seealso cref="Assign"/>
- /// <param name="source">Source object to copy the state from.</param>
- public void AssignAll(Base source)
- {
- AssignAll(source, false);
- }
- internal void AssignAll(Base source, bool assignName)
- {
- Clear();
- Assign(source);
- if (assignName)
- SetName(source.Name);
- foreach (Base child in source.ChildObjects)
- {
- Base myChild = Activator.CreateInstance(child.GetType()) as Base;
- myChild.SetReport(Report);
- myChild.AssignAll(child, assignName);
- myChild.SetReport(null);
- myChild.Parent = this;
- }
- }
- /// <summary>
- /// Gets a value indicating whether the object has the specified parent in its parent hierarhy.
- /// </summary>
- /// <param name="obj">Parent object to check.</param>
- /// <returns>Returns <b>true</b> if the object has given parent in its parent hierarhy.</returns>
- public bool HasParent(Base obj)
- {
- Base parent = Parent;
- while (parent != null)
- {
- if (parent == obj)
- return true;
- parent = parent.Parent;
- }
- return false;
- }
- /// <summary>
- /// Gets a value indicating whether the object has a specified flag in its <see cref="Flags"/> property.
- /// </summary>
- /// <param name="flag">Flag to check.</param>
- /// <returns><b>true</b> if <b>Flags</b> property contains specified flag.</returns>
- public bool HasFlag(Flags flag)
- {
- return (Flags & flag) > 0;
- }
- /// <summary>
- /// Gets a value indicating whether the object has a specified restriction
- /// in its <see cref="Restrictions"/> property.
- /// </summary>
- /// <param name="restriction">Restriction to check.</param>
- /// <returns><b>true</b> if <b>Restrictions</b> property contains specified restriction.</returns>
- public bool HasRestriction(Restrictions restriction)
- {
- return (Restrictions & restriction) > 0;
- }
- /// <summary>
- /// Invokes script event.
- /// </summary>
- /// <param name="name">Name of the event to invoke.</param>
- /// <param name="param">Event parameters.</param>
- /// <remarks>
- /// <para>Do not call this method directly. You should use it if you are developing a new component
- /// for FastReport.</para>
- /// <para>Use this method to call an event handler that is located in the report's script.</para>
- /// </remarks>
- /// <example>Example of the OnBeforePrint method:<code>
- /// public void OnBeforePrint(EventArgs e)
- /// {
- /// if (BeforePrint != null)
- /// BeforePrint(this, e);
- /// InvokeEvent(BeforePrintEvent, e);
- /// }
- /// </code></example>
- public void InvokeEvent(string name, object param)
- {
- if (String.IsNullOrEmpty(name))
- return;
- Report report = Report;
- if (report != null)
- report.InvokeMethod(name, new object[] { this, param });
- }
- /// <summary>
- /// Called after all report objects were loaded.
- /// </summary>
- /// <remarks>
- /// Do not call this method directly. You may override it if you are developing a new component
- /// for FastReport.
- /// </remarks>
- public virtual void OnAfterLoad()
- {
- }
- /// <summary>
- /// Gets all expressions contained in the object.
- /// </summary>
- /// <returns>Array of expressions or <b>null</b> if object contains no expressions.</returns>
- /// <remarks>
- /// <para>Do not call this method directly. You may override it if you are developing a
- /// new component for FastReport.</para>
- /// <para>
- /// This method is called by FastReport each time before run a report. FastReport
- /// do this to collect all expressions and compile them. For example,
- /// <b>GetExpressions</b> method of the <see cref="TextObject"/> class
- /// parses the text and returns all expressions found in the text.
- /// </para>
- /// </remarks>
- public virtual string[] GetExpressions()
- {
- return null;
- }
- /// <summary>
- /// Returns a custom code that will be added to the report script before report is run.
- /// </summary>
- /// <returns>A custom script text, if any. Otherwise returns <b>null</b>.</returns>
- /// <remarks>
- /// <para>This method may return any valid code that may be inserted into the report script. Currently it is
- /// used in the TableObject to define the following script methods: Sum, Min, Max, Avg, Count.
- /// </para>
- /// <para>
- /// Note: you must take into account the current script language - C# or VB.Net. You may check it via
- /// <b>Report.ScriptLanguage</b> property.
- /// </para>
- /// </remarks>
- public virtual string GetCustomScript()
- {
- return null;
- }
- /// <summary>
- /// Used to extract macros such as "TotalPages#" in the preview mode.
- /// </summary>
- /// <remarks>
- /// This method is used mainly by the <b>TextObject</b> to extract macros and replace it with
- /// actual values passed in the <b>pageIndex</b> and <b>totalPages</b> parameters. This method
- /// is called automatically when the object is being previewed.
- /// </remarks>
- public virtual void ExtractMacros()
- {
- }
- /// <summary>
- /// Used to get information of the need to convertation if the function returns true, then the GetConvertedObjects function is called
- /// </summary>
- /// <param name="sender">The export or the object, that call this method</param>
- /// <returns>By default returns false</returns>
- /// <remarks>
- /// The functions IsHaveToConvert and GetConvertedObjects allow you to convert objects from one to another,
- /// for example the export will convert object before adding it to the file and convert recursive,
- /// i.e. If the new object has the ability to convert,
- /// it will be converted again but limit is 10 times.
- /// At the time of export it is called, only on objects inside the band,
- /// the child objects of converted object will be returned, and the child objects of old object will be ignored.
- /// </remarks>
- public virtual bool IsHaveToConvert(object sender)
- {
- return false;
- }
- /// <summary>
- /// Used to get an enumeration of the objects to which this object will be converted, before calling this function, the IsHaveToConvert function will be called
- /// </summary>
- /// <returns>By default returns this object</returns>
- /// <remarks>
- /// The functions IsHaveToConvert and GetConvertedObjects allow you to convert objects from one to another,
- /// for example the export will convert object before adding it to the file and convert recursive,
- /// i.e. If the new object has the ability to convert,
- /// it will be converted again but limit is 10 times.
- /// At the time of export it is called, only on objects inside the band,
- /// the child objects of converted object will be returned, and the child objects of old object will be ignored.
- /// </remarks>
- public virtual IEnumerable<Base> GetConvertedObjects()
- {
- yield return this;
- }
- /// <summary>
- /// Gets the collection of all child objects, converts objects if necessary
- /// </summary>
- /// <param name="sender">the object or export, that call this convertation</param>
- public ObjectCollection ForEachAllConvectedObjects(object sender)
- {
- ObjectCollection list = new ObjectCollection();
- EnumAllConvectedObjects(sender, this, list, 0);
- return list;
- }
- private void EnumAllConvectedObjects(object sender, Base c, ObjectCollection list, int convertValue)
- {
- if (c != this)
- {
- if (c.IsHaveToConvert(sender))
- {
- if (convertValue < 10)
- {
- foreach (Base b in c.GetConvertedObjects())
- EnumAllConvectedObjects(sender, b, list, convertValue + 1);
- return;
- }
- }
- list.Add(c);
- }
- foreach (Base obj in c.ChildObjects)
- EnumAllConvectedObjects(sender, obj, list, convertValue);
- }
- #endregion
- /// <summary>
- /// Initializes a new instance of the <b>Base</b> class with default settings.
- /// </summary>
- public Base()
- {
- name = "";
- alias = "";
- baseName = ClassName;
- restrictions = new Restrictions();
- SetFlags(Flags.CanMove | Flags.CanResize | Flags.CanDelete | Flags.CanEdit | Flags.CanChangeOrder |
- Flags.CanChangeParent | Flags.CanCopy | Flags.CanDraw | Flags.CanShowChildrenInReportTree, true);
- }
- }
- }
|