#pragma warning disable 1634, 1691
namespace System.Workflow.ComponentModel.Design
{
using System;
using System.IO;
using System.Drawing;
using System.CodeDom;
using System.Diagnostics;
using System.Collections;
using System.Collections.Generic;
using System.Windows.Forms;
using System.ComponentModel;
using System.Globalization;
using System.Drawing.Design;
using System.Drawing.Imaging;
using System.Drawing.Drawing2D;
using System.Windows.Forms.Design;
using System.ComponentModel.Design;
using System.Collections.Specialized;
using System.ComponentModel.Design.Serialization;
using System.Workflow.ComponentModel.Compiler;
using System.Workflow.ComponentModel.Serialization;
using System.Collections.ObjectModel;
using System.Reflection;
using System.Workflow.ComponentModel.Design;
using System.Runtime.Serialization.Formatters.Binary;
//
#region CompositeActivityDesigner Class
///
/// CompositeActivityDesigner provides a designer which allows user to visually design composite activities in the design mode.
/// CompositeActivityDesigner enables the user to customize layouting, drawing associated with the CompositeActivity, it also allows
/// managing the layouting, drawing and eventing for the contained activity designers.
///
[ActivityDesignerTheme(typeof(CompositeDesignerTheme))]
[SRCategory("CompositeActivityDesigners", "System.Workflow.ComponentModel.Design.DesignerResources")]
[DesignerSerializer(typeof(CompositeActivityDesignerLayoutSerializer), typeof(WorkflowMarkupSerializer))]
[Obsolete("The System.Workflow.* types are deprecated. Instead, please use the new types from System.Activities.*")]
public abstract class CompositeActivityDesigner : ActivityDesigner
{
#region Fields
private const string CF_DESIGNER = "CF_WINOEDESIGNERCOMPONENTS";
private const string CF_DESIGNERSTATE = "CF_WINOEDESIGNERCOMPONENTSSTATE";
private const int MaximumCharsPerLine = 8;
private const int MaximumTextLines = 1;
private Size actualTextSize = Size.Empty;
private CompositeDesignerAccessibleObject accessibilityObject;
private List containedActivityDesigners;
private bool expanded = true;
#endregion
#region Construction / Destruction
///
/// Default constructor for CompositeActivityDesigner
///
protected CompositeActivityDesigner()
{
}
#endregion
#region Properties
#region Public Properties
///
/// Gets if the activity associated with designer is structurally locked that is no activities can be added to it.
///
public bool IsEditable
{
get
{
if (!(Activity is CompositeActivity))
return false;
if (IsLocked)
return false;
if (Helpers.IsCustomActivity(Activity as CompositeActivity))
return false;
return true;
}
}
///
/// Gets if the designer can be collapsed, collapsed designer has expand/collapse button added to it
///
public virtual bool CanExpandCollapse
{
get
{
return !IsRootDesigner;
}
}
///
/// Gets or Sets if the designer currently in expanded state
///
[DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
public virtual bool Expanded
{
get
{
if (!CanExpandCollapse && !this.expanded)
Expanded = true;
return this.expanded;
}
set
{
if (this.expanded == value)
return;
//If the designer can not expand or collapse then we need to make sure that
//user does not collapse it
if (!CanExpandCollapse && !value)
return;
this.expanded = value;
PerformLayout();
}
}
///
/// Gets the array of activity designer contained within.
///
public virtual ReadOnlyCollection ContainedDesigners
{
get
{
List designers = new List();
CompositeActivity compositeActivity = Activity as CompositeActivity;
if (compositeActivity != null)
{
if (this.containedActivityDesigners == null)
{
bool foundAllDesigners = true;
//In certain cases users might try to access the activity designers
//in Initialize method of composite activity. In that case the child activities
//might not be inserted in the container. If this happens then we might not get the
//designer of the contained activity. When such a case happens we should not buffer the
//designers as it might lead to erroneous results
foreach (Activity activity in compositeActivity.Activities)
{
ActivityDesigner activityDesigner = ActivityDesigner.GetDesigner(activity);
if (activityDesigner != null)
designers.Add(activityDesigner);
else
foundAllDesigners = false;
}
if (foundAllDesigners)
this.containedActivityDesigners = designers;
}
else
{
designers = this.containedActivityDesigners;
}
}
return designers.AsReadOnly();
}
}
///
/// Gets the first selectable object in the navigation order.
///
public virtual object FirstSelectableObject
{
get
{
return null;
}
}
///
/// Gets the last selected object in the navigation order
///
public virtual object LastSelectableObject
{
get
{
return null;
}
}
public override AccessibleObject AccessibilityObject
{
get
{
if (this.accessibilityObject == null)
this.accessibilityObject = new CompositeDesignerAccessibleObject(this);
return this.accessibilityObject;
}
}
public override Point Location
{
get
{
return base.Location;
}
set
{
///If designers's location changes then we need to change location of children
if (base.Location == value)
return;
Size moveDelta = new Size(value.X - base.Location.X, value.Y - base.Location.Y);
foreach (ActivityDesigner activityDesigner in ContainedDesigners)
activityDesigner.Location = new Point(activityDesigner.Location.X + moveDelta.Width, activityDesigner.Location.Y + moveDelta.Height);
base.Location = value;
}
}
#endregion
#region Protected Properties
///
/// Gets the rectangle associated with expand/collapse button
///
protected virtual Rectangle ExpandButtonRectangle
{
get
{
if (!CanExpandCollapse)
return Rectangle.Empty;
CompositeDesignerTheme designerTheme = DesignerTheme as CompositeDesignerTheme;
if (designerTheme == null)
return Rectangle.Empty;
Size textSize = TextRectangle.Size;
Size imageSize = (Image != null) ? designerTheme.ImageSize : Size.Empty;
Rectangle bounds = Bounds;
Size anchorSize = (!textSize.IsEmpty) ? textSize : imageSize;
Rectangle expandButtonRectangle = new Rectangle(bounds.Location, designerTheme.ExpandButtonSize);
expandButtonRectangle.X += (bounds.Width - ((3 * designerTheme.ExpandButtonSize.Width / 2) + anchorSize.Width)) / 2;
expandButtonRectangle.Y += 2 * WorkflowTheme.CurrentTheme.AmbientTheme.Margin.Height;
if (anchorSize.Height > expandButtonRectangle.Height)
expandButtonRectangle.Y += (anchorSize.Height - expandButtonRectangle.Height) / 2;
return expandButtonRectangle;
}
}
///
/// Gets the height for the title area of the designer, typically this can contain the heading, icon and expand/collapse button
///
protected virtual int TitleHeight
{
get
{
Size margin = WorkflowTheme.CurrentTheme.AmbientTheme.Margin;
Rectangle expandButtonRectangle = ExpandButtonRectangle;
Rectangle textRectangle = TextRectangle;
Rectangle imageRectangle = ImageRectangle;
int titleHeight = 0;
if (!textRectangle.Size.IsEmpty)
{
titleHeight = Math.Max(expandButtonRectangle.Height, textRectangle.Height);
titleHeight += imageRectangle.Height;
}
else
{
titleHeight = Math.Max(expandButtonRectangle.Height, imageRectangle.Height);
}
if (!expandButtonRectangle.Size.IsEmpty || !textRectangle.Size.IsEmpty || !imageRectangle.Size.IsEmpty)
titleHeight += (Expanded ? 2 : 3) * margin.Height;
if (!imageRectangle.Size.IsEmpty && !textRectangle.Size.IsEmpty)
titleHeight += margin.Height;
return titleHeight;
}
}
protected override Rectangle ImageRectangle
{
get
{
if (Image == null)
return Rectangle.Empty;
CompositeDesignerTheme designerTheme = DesignerTheme as CompositeDesignerTheme;
if (designerTheme == null)
return Rectangle.Empty;
Rectangle bounds = Bounds;
Size expandButtonSize = ExpandButtonRectangle.Size;
Size imageSize = designerTheme.ImageSize;
Size textSize = TextRectangle.Size;
Size margin = WorkflowTheme.CurrentTheme.AmbientTheme.Margin;
Rectangle imageRectangle = new Rectangle(bounds.Location, imageSize);
if (textSize.Width > 0)
{
imageRectangle.X += (bounds.Width - imageSize.Width) / 2;
}
else
{
imageRectangle.X += (bounds.Width - (imageSize.Width + 3 * expandButtonSize.Width / 2)) / 2;
imageRectangle.X += 3 * expandButtonSize.Width / 2;
}
imageRectangle.Y += 2 * margin.Height;
if (textSize.Height > 0)
imageRectangle.Y += textSize.Height + margin.Height;
return imageRectangle;
}
}
protected override Rectangle TextRectangle
{
get
{
if (String.IsNullOrEmpty(Text))
return Rectangle.Empty;
CompositeDesignerTheme designerTheme = DesignerTheme as CompositeDesignerTheme;
if (designerTheme == null)
return Rectangle.Empty;
Rectangle bounds = Bounds;
Size margin = WorkflowTheme.CurrentTheme.AmbientTheme.Margin;
Size expandButtonSize = (CanExpandCollapse) ? designerTheme.ExpandButtonSize : Size.Empty;
//Calculate the text size
int maxAvailableWidth = bounds.Width - (2 * margin.Width + 3 * expandButtonSize.Width / 2);
Size requestedLineSize = this.actualTextSize;
requestedLineSize.Width /= Text.Length;
requestedLineSize.Width += ((requestedLineSize.Width % Text.Length) > 0) ? 1 : 0;
requestedLineSize.Width *= Math.Min(Text.Length, CompositeActivityDesigner.MaximumCharsPerLine - 1);
Size textSize = Size.Empty;
textSize.Width = Math.Min(maxAvailableWidth, this.actualTextSize.Width);
textSize.Width = Math.Max(1, Math.Max(textSize.Width, requestedLineSize.Width));
textSize.Height = requestedLineSize.Height;
int textLines = this.actualTextSize.Width / textSize.Width;
textLines += ((this.actualTextSize.Width % textSize.Width) > 0) ? 1 : 0;
textLines = Math.Min(textLines, CompositeActivityDesigner.MaximumTextLines);
textSize.Height *= textLines;
//Calculate the text rectangle
Rectangle textRectangle = new Rectangle(bounds.Location, textSize);
textRectangle.X += (bounds.Width - (3 * expandButtonSize.Width / 2 + textSize.Width)) / 2;
textRectangle.X += 3 * expandButtonSize.Width / 2;
textRectangle.Y += 2 * margin.Height;
if (expandButtonSize.Height > textSize.Height)
textRectangle.Y += (expandButtonSize.Height - textSize.Height) / 2;
textRectangle.Size = textSize;
return textRectangle;
}
}
protected internal override ActivityDesignerGlyphCollection Glyphs
{
get
{
ActivityDesignerGlyphCollection glyphs = new ActivityDesignerGlyphCollection();
glyphs.AddRange(base.Glyphs);
CompositeDesignerTheme compositeTheme = DesignerTheme as CompositeDesignerTheme;
if (compositeTheme != null && compositeTheme.ShowDropShadow)
glyphs.Add(ShadowGlyph.Default);
return glyphs;
}
}
#endregion
#region Private Properties
#region Properties used during serialization only
//NOTE THAT THIS WILL ONLY BE USED FOR SERIALIZATION PURPOSES
[EditorBrowsable(EditorBrowsableState.Never)]
[DesignerSerializationVisibility(DesignerSerializationVisibility.Content)]
internal List Designers
{
get
{
List childDesigners = new List();
CompositeActivity compositeActivity = Activity as CompositeActivity;
IDesignerHost host = GetService(typeof(IDesignerHost)) as IDesignerHost;
if (host != null && compositeActivity != null)
{
foreach (Activity childActivity in compositeActivity.Activities)
{
ActivityDesigner designer = host.GetDesigner(childActivity) as ActivityDesigner;
if (designer != null)
childDesigners.Add(designer);
}
}
return childDesigners;
}
}
#endregion
#endregion
#endregion
#region Methods
#region Public Static Methods
///
/// Inserts activities in specified composite activity designer by creating transaction
///
/// Designer in which to insert the activities
/// Insertion location
/// Array of activities to insert
/// Text for the designer transaction which will be created
public static void InsertActivities(CompositeActivityDesigner compositeActivityDesigner, HitTestInfo insertLocation, ReadOnlyCollection activitiesToInsert, string undoTransactionDescription)
{
if (compositeActivityDesigner == null)
throw new ArgumentNullException("compositeActivityDesigner");
if (compositeActivityDesigner.Activity == null ||
compositeActivityDesigner.Activity.Site == null ||
!(compositeActivityDesigner.Activity is CompositeActivity))
throw new ArgumentException("compositeActivityDesigner");
if (insertLocation == null)
throw new ArgumentNullException("insertLocation");
if (activitiesToInsert == null)
throw new ArgumentNullException("activitiesToInsert");
ISite site = compositeActivityDesigner.Activity.Site;
// now insert the actual activities
IDesignerHost designerHost = site.GetService(typeof(IDesignerHost)) as IDesignerHost;
DesignerTransaction trans = null;
if (designerHost != null && !string.IsNullOrEmpty(undoTransactionDescription))
trans = designerHost.CreateTransaction(undoTransactionDescription);
bool moveCase = false;
try
{
//Detect if the activities are being moved or inserted
foreach (Activity activity in activitiesToInsert)
{
if (activity == null)
throw new ArgumentException("activitiesToInsert", SR.GetString(SR.Error_CollectionHasNullEntry));
moveCase = ((IComponent)activity).Site != null;
break;
}
//We purposely create a new instance of activities list so that we do not modify the original one
if (moveCase)
compositeActivityDesigner.MoveActivities(insertLocation, activitiesToInsert);
else
compositeActivityDesigner.InsertActivities(insertLocation, activitiesToInsert);
if (trans != null)
trans.Commit();
}
catch (Exception e)
{
if (trans != null)
trans.Cancel();
throw e;
}
//If we are just moving the activities then we do not need to emit the class
//for scopes; only when we are adding the activities the class needs to be emitted for
//scope in code beside file
if (!moveCase)
{
// if everything was successful then generate classes correposnding to new scopes
// get all the activities underneath the child activities
ArrayList allActivities = new ArrayList();
foreach (Activity activity in activitiesToInsert)
{
allActivities.Add(activity);
if (activity is CompositeActivity)
allActivities.AddRange(Helpers.GetNestedActivities((CompositeActivity)activity));
}
}
}
///
/// Removes activities from designer by creating designer transaction
///
/// Service Provider associated to providing services
/// Array of activities to remove
/// Transaction text used to name the designer transaction
public static void RemoveActivities(IServiceProvider serviceProvider, ReadOnlyCollection activitiesToRemove, string transactionDescription)
{
if (serviceProvider == null)
throw new ArgumentNullException();
if (activitiesToRemove == null)
throw new ArgumentNullException("activitiesToRemove");
Activity nextSelectableActivity = null;
IDesignerHost designerHost = serviceProvider.GetService(typeof(IDesignerHost)) as IDesignerHost;
DesignerTransaction trans = null;
if (designerHost != null && !string.IsNullOrEmpty(transactionDescription))
trans = designerHost.CreateTransaction(transactionDescription);
try
{
foreach (Activity activity in activitiesToRemove)
{
ActivityDesigner designer = ActivityDesigner.GetDesigner(activity);
if (designer != null)
{
CompositeActivityDesigner parentDesigner = designer.ParentDesigner;
if (parentDesigner != null)
{
nextSelectableActivity = DesignerHelpers.GetNextSelectableActivity(activity);
parentDesigner.RemoveActivities(new List(new Activity[] { activity }).AsReadOnly());
}
}
}
if (trans != null)
trans.Commit();
}
catch
{
if (trans != null)
trans.Cancel();
throw;
}
if (nextSelectableActivity != null && nextSelectableActivity.Site != null)
{
ISelectionService selectionService = nextSelectableActivity.Site.GetService(typeof(ISelectionService)) as ISelectionService;
if (selectionService != null)
selectionService.SetSelectedComponents(new Activity[] { nextSelectableActivity }, SelectionTypes.Replace);
}
}
public static IDataObject SerializeActivitiesToDataObject(IServiceProvider serviceProvider, Activity[] activities)
{
if (serviceProvider == null)
throw new ArgumentNullException("serviceProvider");
if (activities == null)
throw new ArgumentNullException("activities");
// get component serialization service
ComponentSerializationService css = (ComponentSerializationService)serviceProvider.GetService(typeof(ComponentSerializationService));
if (css == null)
throw new InvalidOperationException(SR.GetString(SR.General_MissingService, typeof(ComponentSerializationService).Name));
// serialize all activities to the store
SerializationStore store = css.CreateStore();
using (store)
{
foreach (Activity activity in activities)
css.Serialize(store, activity);
}
// wrap it with clipboard style object
Stream stream = new MemoryStream();
BinaryFormatter formatter = new BinaryFormatter();
formatter.Serialize(stream, store);
stream.Seek(0, SeekOrigin.Begin);
DataObject dataObject = new DataObject(CF_DESIGNER, stream);
dataObject.SetData(CF_DESIGNERSTATE, Helpers.SerializeDesignersToStream(activities));
return dataObject;
}
public static Activity[] DeserializeActivitiesFromDataObject(IServiceProvider serviceProvider, IDataObject dataObj)
{
return DeserializeActivitiesFromDataObject(serviceProvider, dataObj, false);
}
internal static Activity[] DeserializeActivitiesFromDataObject(IServiceProvider serviceProvider, IDataObject dataObj, bool addAssemblyReference)
{
if (serviceProvider == null)
throw new ArgumentNullException("serviceProvider");
if (dataObj == null)
return new Activity[] { };
IDesignerHost designerHost = (IDesignerHost)serviceProvider.GetService(typeof(IDesignerHost));
if (designerHost == null)
throw new InvalidOperationException(SR.GetString(SR.General_MissingService, typeof(IDesignerHost).Name));
object data = dataObj.GetData(CF_DESIGNER);
ICollection activities = null;
if (data is Stream)
{
BinaryFormatter formatter = new BinaryFormatter();
((Stream)data).Seek(0, SeekOrigin.Begin);
object serializationData = formatter.Deserialize((Stream)data);
if (serializationData is SerializationStore)
{
// get component serialization service
ComponentSerializationService css = serviceProvider.GetService(typeof(ComponentSerializationService)) as ComponentSerializationService;
if (css == null)
throw new Exception(SR.GetString(SR.General_MissingService, typeof(ComponentSerializationService).Name));
// deserialize data
activities = css.Deserialize((SerializationStore)serializationData);
}
}
else
{
// Now check for a toolbox item.
IToolboxService ts = (IToolboxService)serviceProvider.GetService(typeof(IToolboxService));
if (ts != null && ts.IsSupported(dataObj, designerHost))
{
ToolboxItem toolBoxItem = ts.DeserializeToolboxItem(dataObj, designerHost);
if (toolBoxItem != null)
{
activities = GetActivitiesFromToolboxItem(serviceProvider, addAssemblyReference, designerHost, activities, toolBoxItem);
}
}
}
if (activities != null && Helpers.AreAllActivities(activities))
return (Activity[])new ArrayList(activities).ToArray(typeof(Activity));
else
return new Activity[] { };
}
private static ICollection GetActivitiesFromToolboxItem(IServiceProvider serviceProvider, bool addAssemblyReference, IDesignerHost designerHost, ICollection activities, ToolboxItem toolBoxItem)
{
// this will make sure that we add the assembly reference to project
if (addAssemblyReference && toolBoxItem.AssemblyName != null)
{
ITypeResolutionService trs = serviceProvider.GetService(typeof(ITypeResolutionService)) as ITypeResolutionService;
if (trs != null)
trs.ReferenceAssembly(toolBoxItem.AssemblyName);
}
ActivityToolboxItem ActivityToolboxItem = toolBoxItem as ActivityToolboxItem;
if (addAssemblyReference && ActivityToolboxItem != null)
activities = ActivityToolboxItem.CreateComponentsWithUI(designerHost);
else
activities = toolBoxItem.CreateComponents(designerHost);
return activities;
}
internal static Activity[] DeserializeActivitiesFromToolboxItem(IServiceProvider serviceProvider, ToolboxItem toolboxItem, bool addAssemblyReference)
{
if (serviceProvider == null)
throw new ArgumentNullException("serviceProvider");
IDesignerHost designerHost = (IDesignerHost)serviceProvider.GetService(typeof(IDesignerHost));
if (designerHost == null)
throw new InvalidOperationException(SR.GetString(SR.General_MissingService, typeof(IDesignerHost).Name));
ICollection activities = null;
if (toolboxItem != null)
{
activities = GetActivitiesFromToolboxItem(serviceProvider, addAssemblyReference, designerHost, activities, toolboxItem);
}
if (activities != null && Helpers.AreAllActivities(activities))
return (Activity[])new ArrayList(activities).ToArray(typeof(Activity));
else
return new Activity[] { };
}
public static ActivityDesigner[] GetIntersectingDesigners(ActivityDesigner topLevelDesigner, Rectangle rectangle)
{
if (topLevelDesigner == null)
throw new ArgumentNullException("topLevelDesigner");
List intersectingDesigners = new List();
if (!rectangle.IntersectsWith(topLevelDesigner.Bounds))
return intersectingDesigners.ToArray();
if (!topLevelDesigner.Bounds.Contains(rectangle))
intersectingDesigners.Add(topLevelDesigner);
if (topLevelDesigner is CompositeActivityDesigner)
{
Queue compositeDesigners = new Queue();
compositeDesigners.Enqueue(topLevelDesigner);
while (compositeDesigners.Count > 0)
{
CompositeActivityDesigner compositeDesigner = compositeDesigners.Dequeue() as CompositeActivityDesigner;
if (compositeDesigner != null)
{
bool bDrawingVisibleChildren = false;
foreach (ActivityDesigner activityDesigner in compositeDesigner.ContainedDesigners)
{
if (activityDesigner.IsVisible && rectangle.IntersectsWith(activityDesigner.Bounds))
{
bDrawingVisibleChildren = true;
if (!activityDesigner.Bounds.Contains(rectangle))
intersectingDesigners.Add(activityDesigner);
if (activityDesigner is CompositeActivityDesigner)
compositeDesigners.Enqueue(activityDesigner);
}
else
{
if ((!(compositeDesigner is FreeformActivityDesigner)) && bDrawingVisibleChildren)
break;
}
}
}
}
}
return intersectingDesigners.ToArray();
}
#endregion
#region Public Methods
///
/// Checks if the activities can be inserted into the composite activity associated with the designer
///
/// Location at which to insert activities
/// Array of activities to be inserted
/// True of activities can be inserted, false otherwise
public virtual bool CanInsertActivities(HitTestInfo insertLocation, ReadOnlyCollection activitiesToInsert)
{
if (insertLocation == null)
throw new ArgumentNullException("insertLocation");
if (activitiesToInsert == null)
throw new ArgumentNullException("activitiesToInsert");
CompositeActivity compositeActivity = Activity as CompositeActivity;
if (compositeActivity == null)
return false;
//If the activity state is locked then we can not insert activities.
if (!IsEditable)
return false;
IExtendedUIService2 extendedUIService = this.GetService(typeof(IExtendedUIService2)) as IExtendedUIService2;
foreach (Activity activity in activitiesToInsert)
{
if (activity == null)
throw new ArgumentException("activitiesToInsert", SR.GetString(SR.Error_CollectionHasNullEntry));
if (extendedUIService != null)
{
if (!extendedUIService.IsSupportedType(activity.GetType()))
{
return false;
}
}
if (activity is CompositeActivity && Helpers.IsAlternateFlowActivity(activity))
return false;
ActivityDesigner designerToInsert = null;
#pragma warning disable 56506//bug in presharp, activity has already been checked for null value
if (activity.Site != null)
{
//get an existing designer
IDesignerHost designerHost = activity.Site.GetService(typeof(IDesignerHost)) as IDesignerHost;
designerToInsert = (designerHost != null) ? designerHost.GetDesigner((IComponent)activity) as ActivityDesigner : null;
}
else
{
//we dont want to create a designer instance every time, so we'll cache it
//this is a fix for a perf issue - this function gets called in a loop when doing drag'n'drop operation
//from a toolbox
if (activity.UserData.Contains(typeof(ActivityDesigner)))
{
designerToInsert = activity.UserData[typeof(ActivityDesigner)] as ActivityDesigner;
}
else
{
//create a new one
designerToInsert = ActivityDesigner.CreateDesigner(Activity.Site, activity);
activity.UserData[typeof(ActivityDesigner)] = designerToInsert;
}
}
#pragma warning restore 56506//bug in presharp
if (designerToInsert == null)
return false;
if (!designerToInsert.CanBeParentedTo(this))
return false;
}
return true;
}
///
/// Inserts activity at specified location within designer
///
/// Location at which to insert activities
/// Array of activities to insert
public virtual void InsertActivities(HitTestInfo insertLocation, ReadOnlyCollection activitiesToInsert)
{
if (insertLocation == null)
throw new ArgumentNullException("insertLocation");
if (activitiesToInsert == null)
throw new ArgumentNullException("activitiesToInsert");
CompositeActivity compositeActivity = Activity as CompositeActivity;
if (compositeActivity == null)
throw new Exception(SR.GetString(SR.Error_DragDropInvalid));
int index = insertLocation.MapToIndex();
IIdentifierCreationService identifierCreationService = GetService(typeof(IIdentifierCreationService)) as IIdentifierCreationService;
if (identifierCreationService != null)
identifierCreationService.EnsureUniqueIdentifiers(compositeActivity, activitiesToInsert);
else
throw new InvalidOperationException(SR.GetString(SR.General_MissingService, typeof(IIdentifierCreationService).FullName));
foreach (Activity activity in activitiesToInsert)
{
if (activity == null)
throw new ArgumentException("activitiesToInsert", SR.GetString(SR.Error_CollectionHasNullEntry));
if (activity.Parent == null)
{
compositeActivity.Activities.Insert(index++, activity);
WorkflowDesignerLoader.AddActivityToDesigner(Activity.Site, activity);
}
}
// filter out unsupported Dependency properties
foreach (Activity activity in activitiesToInsert)
{
Walker walker = new Walker();
walker.FoundActivity += delegate(Walker w, WalkerEventArgs walkerEventArgs)
{
ExtenderHelpers.FilterDependencyProperties(this.Activity.Site, walkerEventArgs.CurrentActivity);
};
walker.Walk(activity);
}
}
///
/// Checks if the activities can be moved out from the composite activity associated with designer
///
/// Location from which to move the activities
/// Array of activities to move
///
public virtual bool CanMoveActivities(HitTestInfo moveLocation, ReadOnlyCollection activitiesToMove)
{
if (moveLocation == null)
throw new ArgumentNullException("moveLocation");
if (activitiesToMove == null)
throw new ArgumentNullException("activitiesToMove");
//If the activity has locked structure then we do not allow user to move the activity out
if (!IsEditable)
return false;
//Now go through all the movable activities and check if their position is locked
foreach (Activity activity in activitiesToMove)
{
ActivityDesigner designer = ActivityDesigner.GetDesigner(activity);
if (designer == null || designer.IsLocked)
return false;
}
return true;
}
///
/// Moves activities from one designer to other
///
/// Location at which to move the activities
/// Array of activities to move
public virtual void MoveActivities(HitTestInfo moveLocation, ReadOnlyCollection activitiesToMove)
{
if (moveLocation == null)
throw new ArgumentNullException("moveLocation");
if (activitiesToMove == null)
throw new ArgumentNullException("activitiesToMove");
//Make sure that we get the composite activity
CompositeActivity compositeActivity = Activity as CompositeActivity;
if (compositeActivity == null)
throw new Exception(SR.GetString(SR.Error_DragDropInvalid));
IIdentifierCreationService identifierCreationService = GetService(typeof(IIdentifierCreationService)) as IIdentifierCreationService;
if (identifierCreationService == null)
throw new InvalidOperationException(SR.GetString(SR.General_MissingService, typeof(IIdentifierCreationService).FullName));
IDesignerHost designerHost = GetService(typeof(IDesignerHost)) as IDesignerHost;
if (designerHost == null)
throw new InvalidOperationException(SR.GetString(SR.General_MissingService, typeof(IDesignerHost).FullName));
int index = moveLocation.MapToIndex();
foreach (Activity activity in activitiesToMove)
{
ActivityDesigner designer = ActivityDesigner.GetDesigner(activity);
if (designer != null)
{
CompositeActivityDesigner parentDesigner = designer.ParentDesigner;
if (parentDesigner == this)
{
int originalIndex = compositeActivity.Activities.IndexOf(activity);
if (index > originalIndex)
index--;
}
}
//In some cases we might get activities which are added newly, in such cases we check if the activity
//is existing activity or new one based on this decision we add it to the designer host.
Debug.Assert(activity.Parent != null);
CompositeActivity parentActivity = activity.Parent;
int positionInParent = parentActivity.Activities.IndexOf(activity);
activity.Parent.Activities.Remove(activity);
//We need to make sure that the activity is going to have unique identifier
//This might just cause problems
identifierCreationService.EnsureUniqueIdentifiers(compositeActivity, new Activity[] { activity });
//We do not need to read the activity in the designer host as this is move operation
//assign unique temporary name to avoid conflicts
DesignerHelpers.UpdateSiteName(activity, "_activityonthemove_");
CompositeActivity compositeActivityMoved = activity as CompositeActivity;
if (compositeActivityMoved != null)
{
int i = 1;
foreach (Activity nestedActivity in Helpers.GetNestedActivities(compositeActivityMoved))
{
DesignerHelpers.UpdateSiteName(nestedActivity, "_activityonthemove_" + i.ToString(CultureInfo.InvariantCulture));
i += 1;
}
}
try
{
compositeActivity.Activities.Insert(index++, activity);
}
catch (Exception ex)
{
// reconnect the activity
parentActivity.Activities.Insert(positionInParent, activity);
//
throw ex;
}
DesignerHelpers.UpdateSiteName(activity, activity.Name);
if (compositeActivityMoved != null)
{
foreach (Activity nestedActivity in Helpers.GetNestedActivities(compositeActivityMoved))
DesignerHelpers.UpdateSiteName(nestedActivity, nestedActivity.Name);
}
}
// filter out unsupported Dependency properties and refresh propertyDescriptors
foreach (Activity activity in activitiesToMove)
{
Walker walker = new Walker();
walker.FoundActivity += delegate(Walker w, WalkerEventArgs walkerEventArgs)
{
ExtenderHelpers.FilterDependencyProperties(this.Activity.Site, walkerEventArgs.CurrentActivity);
TypeDescriptor.Refresh(walkerEventArgs.CurrentActivity);
};
walker.Walk(activity);
}
}
///
/// Checks if activities can be removed from the activity associated with the designer
///
/// Array of activities to remove
/// True of the activities can be removed, False otherwise.
public virtual bool CanRemoveActivities(ReadOnlyCollection activitiesToRemove)
{
if (activitiesToRemove == null)
throw new ArgumentNullException("activitiesToRemove");
if (!IsEditable)
return false;
foreach (Activity activity in activitiesToRemove)
{
ActivityDesigner designer = ActivityDesigner.GetDesigner(activity);
if (designer == null || designer.IsLocked)
return false;
}
return true;
}
///
/// Removes activities from composite activity associated with the designer
///
/// Array of activities to remove
public virtual void RemoveActivities(ReadOnlyCollection activitiesToRemove)
{
if (activitiesToRemove == null)
throw new ArgumentNullException("activitiesToRemove");
CompositeActivity compositeActivity = Activity as CompositeActivity;
if (compositeActivity == null)
throw new Exception(SR.GetString(SR.Error_DragDropInvalid));
foreach (Activity activity in activitiesToRemove)
{
compositeActivity.Activities.Remove(activity);
//Before we destroy the activity make sure that the references it and its child activities store to its parent
//are set to null or else an undo unit will be created
//For details look at,
//\\cpvsbuild\drops\whidbey\pd6\raw\40903.19\sources\ndp\fx\src\Designer\Host\UndoEngine.cs
//OnComponentRemoving function which retains the references we hold to the parent
//This bug can be reproed by deleting a compositeactivity from design surface and then doing an undo
//VSWhidbey #312230
activity.SetParent(null);
if (activity is CompositeActivity)
{
foreach (Activity nestedActivity in Helpers.GetNestedActivities(activity as CompositeActivity))
nestedActivity.SetParent(null);
}
WorkflowDesignerLoader.RemoveActivityFromDesigner(Activity.Site, activity);
}
}
///
/// Checks if the contained designer is to be made visible
///
/// Contained designer to check for visibility
///
public virtual bool IsContainedDesignerVisible(ActivityDesigner containedDesigner)
{
if (containedDesigner == null)
throw new ArgumentNullException("containedDesigner");
return true;
}
///
/// Makes sure that the child designer will be made visible
///
/// Contained designer to make visible
public virtual void EnsureVisibleContainedDesigner(ActivityDesigner containedDesigner)
{
if (containedDesigner == null)
throw new ArgumentNullException("containedDesigner");
Expanded = true;
}
///
/// Gets the object which is next in the order of navigation
///
/// Current object in the navigation order
/// Navigation direction
///
public virtual object GetNextSelectableObject(object current, DesignerNavigationDirection direction)
{
return null;
}
public override HitTestInfo HitTest(Point point)
{
HitTestInfo hitInfo = HitTestInfo.Nowhere;
if (ExpandButtonRectangle.Contains(point))
{
hitInfo = new HitTestInfo(this, HitTestLocations.Designer | HitTestLocations.ActionArea);
}
else if (Expanded && Bounds.Contains(point))
{
//First check if any of our children are hit.
ReadOnlyCollection containedDesigners = ContainedDesigners;
for (int i = containedDesigners.Count - 1; i >= 0; i--)
{
ActivityDesigner activityDesigner = containedDesigners[i] as ActivityDesigner;
if (activityDesigner != null && activityDesigner.IsVisible)
{
hitInfo = activityDesigner.HitTest(point);
if (hitInfo.HitLocation != HitTestLocations.None)
break;
}
}
}
//If no children are hit then call base class's hittest
if (hitInfo == HitTestInfo.Nowhere)
hitInfo = base.HitTest(point);
//This is to create default hittest info in case the drawing state is invalid
if (hitInfo.AssociatedDesigner != null && hitInfo.AssociatedDesigner.DrawingState != DrawingStates.Valid)
hitInfo = new HitTestInfo(hitInfo.AssociatedDesigner, HitTestLocations.Designer | HitTestLocations.ActionArea);
return hitInfo;
}
#endregion
#region Protected Methods
///
/// Notifies that the activity associated with contained designer has changed.
///
/// ActivityChangedEventArgs containing information about the change.
protected virtual void OnContainedActivityChanged(ActivityChangedEventArgs e)
{
if (e == null)
throw new ArgumentNullException("e");
}
///
/// Notifies that number of activities contained within the designers are changing
///
/// ActivityCollectionChangeEventArgs containing information about what is about to change
protected virtual void OnContainedActivitiesChanging(ActivityCollectionChangeEventArgs listChangeArgs)
{
if (listChangeArgs == null)
throw new ArgumentNullException("listChangeArgs");
}
///
/// Notifies that number of activities contained within the designers have changed
///
/// ItemListChangeEventArgs containing information about what has changed
protected virtual void OnContainedActivitiesChanged(ActivityCollectionChangeEventArgs listChangeArgs)
{
if (listChangeArgs == null)
throw new ArgumentNullException("listChangeArgs");
// Update the status of all the designers
foreach (ActivityDesigner activityDesigner in ContainedDesigners)
{
foreach (DesignerVerb designerVerb in ((IDesigner)activityDesigner).Verbs)
{
int status = designerVerb.OleStatus;
status = 0;
}
}
RefreshDesignerVerbs();
//clear the list of activity designers (force to create the new list the next time it is requested)
this.containedActivityDesigners = null;
PerformLayout();
}
protected override void Initialize(Activity activity)
{
base.Initialize(activity);
///
CompositeActivity compositeActivity = activity as CompositeActivity;
if (compositeActivity != null)
{
compositeActivity.Activities.ListChanging += new EventHandler(OnActivityListChanging);
compositeActivity.Activities.ListChanged += new EventHandler(OnActivityListChanged);
}
if (IsRootDesigner)
{
IComponentChangeService componentChangeService = GetService(typeof(IComponentChangeService)) as IComponentChangeService;
if (componentChangeService != null)
{
componentChangeService.ComponentAdded += new ComponentEventHandler(OnComponentAdded);
componentChangeService.ComponentChanged += new ComponentChangedEventHandler(OnComponentChanged);
}
}
}
protected override void Dispose(bool disposing)
{
if (disposing)
{
CompositeActivity compositeActivity = Activity as CompositeActivity;
if (compositeActivity != null)
{
compositeActivity.Activities.ListChanging -= new EventHandler(OnActivityListChanging);
compositeActivity.Activities.ListChanged -= new EventHandler(OnActivityListChanged);
}
if (IsRootDesigner)
{
IComponentChangeService componentChangeService = GetService(typeof(IComponentChangeService)) as IComponentChangeService;
if (componentChangeService != null)
{
componentChangeService.ComponentAdded -= new ComponentEventHandler(OnComponentAdded);
componentChangeService.ComponentChanged -= new ComponentChangedEventHandler(OnComponentChanged);
}
}
}
base.Dispose(disposing);
}
protected override void OnMouseDown(MouseEventArgs e)
{
base.OnMouseDown(e);
if (ExpandButtonRectangle.Contains(new Point(e.X, e.Y)))
{
//In order to property update the menu items for expand collapse, rather than setting the property
//directly we go thru menu command service
IMenuCommandService menuCommandService = GetService(typeof(IMenuCommandService)) as IMenuCommandService;
if (menuCommandService != null)
menuCommandService.GlobalInvoke((Expanded) ? WorkflowMenuCommands.Collapse : WorkflowMenuCommands.Expand);
else
Expanded = !Expanded;
}
}
protected override void OnPaint(ActivityDesignerPaintEventArgs e)
{
base.OnPaint(e);
CompositeDesignerTheme compositeDesignerTheme = e.DesignerTheme as CompositeDesignerTheme;
if (compositeDesignerTheme == null)
return;
//Draw the expand collapse button and the connection
if (CanExpandCollapse)
{
Rectangle expandButtonRectangle = ExpandButtonRectangle;
if (!expandButtonRectangle.Size.IsEmpty)
{
ActivityDesignerPaint.DrawExpandButton(e.Graphics, expandButtonRectangle, !Expanded, compositeDesignerTheme);
}
}
if (Expanded)
PaintContainedDesigners(e);
}
protected override void OnLayoutPosition(ActivityDesignerLayoutEventArgs e)
{
if (e == null)
throw new ArgumentNullException("e");
base.OnLayoutPosition(e);
foreach (ActivityDesigner activityDesigner in ContainedDesigners)
{
try
{
((IWorkflowDesignerMessageSink)activityDesigner).OnLayoutPosition(e.Graphics);
activityDesigner.DrawingState &= (~DrawingStates.InvalidPosition);
}
catch
{
//Eat the exception thrown
activityDesigner.DrawingState |= DrawingStates.InvalidPosition;
}
}
}
protected override Size OnLayoutSize(ActivityDesignerLayoutEventArgs e)
{
Size containerSize = base.OnLayoutSize(e);
//Calculate the size of internal designers
foreach (ActivityDesigner activityDesigner in ContainedDesigners)
{
try
{
((IWorkflowDesignerMessageSink)activityDesigner).OnLayoutSize(e.Graphics);
activityDesigner.DrawingState &= (~DrawingStates.InvalidSize);
}
catch
{
//Eat the exception thrown
activityDesigner.Size = activityDesigner.DesignerTheme.Size;
activityDesigner.DrawingState |= DrawingStates.InvalidSize;
}
}
if (!String.IsNullOrEmpty(Text))
this.actualTextSize = ActivityDesignerPaint.MeasureString(e.Graphics, e.DesignerTheme.BoldFont, Text, StringAlignment.Center, Size.Empty);
else
this.actualTextSize = Size.Empty;
if (Expanded)
containerSize.Height = TitleHeight;
else
containerSize.Height = TitleHeight + WorkflowTheme.CurrentTheme.AmbientTheme.Margin.Height;
return containerSize;
}
protected override void SaveViewState(BinaryWriter writer)
{
if (writer == null)
throw new ArgumentNullException("writer");
writer.Write(Expanded);
base.SaveViewState(writer);
}
protected override void LoadViewState(BinaryReader reader)
{
if (reader == null)
throw new ArgumentNullException("reader");
Expanded = reader.ReadBoolean();
base.LoadViewState(reader);
}
protected override void OnThemeChange(ActivityDesignerTheme designerTheme)
{
base.OnThemeChange(designerTheme);
CompositeActivity compositeActivity = Activity as CompositeActivity;
if (compositeActivity != null)
{
foreach (Activity activity in compositeActivity.Activities)
{
IWorkflowDesignerMessageSink containedDesigner = ActivityDesigner.GetDesigner(activity) as IWorkflowDesignerMessageSink;
if (containedDesigner != null)
containedDesigner.OnThemeChange();
}
}
}
///
/// Call the OnPaint on the contained designers
///
/// EventArgs to be used for painting
protected void PaintContainedDesigners(ActivityDesignerPaintEventArgs e)
{
OnPaintContainedDesigners(e);
}
protected override void OnKeyDown(KeyEventArgs e)
{
if (e == null)
throw new ArgumentNullException("e");
ISelectionService selectionService = GetService(typeof(ISelectionService)) as ISelectionService;
object selectedObject = (selectionService != null) ? selectionService.PrimarySelection : null;
if (selectedObject == null)
return;
//handling of the key move events for all designers
//the freeform designer will override that to allow moving of the designers vs moving selection
object nextSelectedObject = null;
if (e.KeyCode == Keys.Down ||
(e.KeyCode == Keys.Tab && !e.Shift))
{
CompositeActivityDesigner selectedDesigner = ActivityDesigner.GetDesigner(selectedObject as Activity) as CompositeActivityDesigner;
if (selectedDesigner != null)
nextSelectedObject = selectedDesigner.FirstSelectableObject;
if (nextSelectedObject == null)
{
do
{
// get parent designer
CompositeActivityDesigner parentDesigner = ActivityDesigner.GetParentDesigner(selectedObject);
if (parentDesigner == null)
{
// IMPORTANT: This will only happen when the focus is on the last connector in ServiceDesigner
nextSelectedObject = selectedObject;
break;
}
nextSelectedObject = parentDesigner.GetNextSelectableObject(selectedObject, DesignerNavigationDirection.Down);
if (nextSelectedObject != null)
break;
selectedObject = parentDesigner.Activity;
} while (true);
}
}
else if (e.KeyCode == Keys.Up ||
(e.KeyCode == Keys.Tab && e.Shift))
{
// get parent designer
CompositeActivityDesigner parentDesigner = ActivityDesigner.GetParentDesigner(selectedObject);
if (parentDesigner == null)
{
// IMPORTANT: This will only happen when the focus is on the ServiceDesigner it self
CompositeActivityDesigner selectedDesigner = ActivityDesigner.GetDesigner(selectedObject as Activity) as CompositeActivityDesigner;
if (selectedDesigner != null)
nextSelectedObject = selectedDesigner.LastSelectableObject;
}
else
{
// ask for previous component
nextSelectedObject = parentDesigner.GetNextSelectableObject(selectedObject, DesignerNavigationDirection.Up);
if (nextSelectedObject != null)
{
CompositeActivityDesigner nextSelectedDesigner = ActivityDesigner.GetDesigner(nextSelectedObject as Activity) as CompositeActivityDesigner;
// when we go up, and then upper selection is parent designer then we look for last component
if (nextSelectedDesigner != null)
{
object lastObject = nextSelectedDesigner.LastSelectableObject;
if (lastObject != null)
nextSelectedObject = lastObject;
}
}
else
{
nextSelectedObject = parentDesigner.Activity;
}
}
}
else if (e.KeyCode == Keys.Left)
{
do
{
CompositeActivityDesigner parentDesigner = ActivityDesigner.GetParentDesigner(selectedObject);
if (parentDesigner == null)
break;
nextSelectedObject = parentDesigner.GetNextSelectableObject(selectedObject, DesignerNavigationDirection.Left);
if (nextSelectedObject != null)
break;
selectedObject = parentDesigner.Activity;
} while (true);
}
else if (e.KeyCode == Keys.Right)
{
do
{
CompositeActivityDesigner parentDesigner = ActivityDesigner.GetParentDesigner(selectedObject);
if (parentDesigner == null)
break;
nextSelectedObject = parentDesigner.GetNextSelectableObject(selectedObject, DesignerNavigationDirection.Right);
if (nextSelectedObject != null)
break;
selectedObject = parentDesigner.Activity;
} while (true);
}
// now select the component
if (nextSelectedObject != null)
{
selectionService.SetSelectedComponents(new object[] { nextSelectedObject }, SelectionTypes.Replace);
// make the selected designer visible
ParentView.EnsureVisible(nextSelectedObject);
}
}
#endregion
#region Private Methods
internal virtual void OnPaintContainedDesigners(ActivityDesignerPaintEventArgs e)
{
foreach (ActivityDesigner activityDesigner in ContainedDesigners)
{
using (PaintEventArgs paintEventArgs = new PaintEventArgs(e.Graphics, e.ViewPort))
{
((IWorkflowDesignerMessageSink)activityDesigner).OnPaint(paintEventArgs, e.ViewPort);
}
}
}
private void OnComponentAdded(object sender, ComponentEventArgs e)
{
ActivityDesigner designer = ActivityDesigner.GetDesigner(e.Component as Activity);
if (Activity != e.Component && designer != null && designer.IsLocked)
DesignerHelpers.MakePropertiesReadOnly(e.Component.Site, designer.Activity);
}
private void OnComponentChanged(object sender, ComponentChangedEventArgs e)
{
IReferenceService referenceService = GetService(typeof(IReferenceService)) as IReferenceService;
Activity changedActivity = (referenceService != null) ? referenceService.GetComponent(e.Component) as Activity : e.Component as Activity;
if (changedActivity != null)
{
ActivityDesigner designer = ActivityDesigner.GetDesigner(changedActivity);
if (designer != null)
{
CompositeActivityDesigner parentDesigner = designer.ParentDesigner;
if (parentDesigner != null)
parentDesigner.OnContainedActivityChanged(new ActivityChangedEventArgs(changedActivity, e.Member, e.OldValue, e.NewValue));
}
}
}
private void OnActivityListChanging(object sender, ActivityCollectionChangeEventArgs e)
{
OnContainedActivitiesChanging(e);
}
private void OnActivityListChanged(object sender, ActivityCollectionChangeEventArgs e)
{
OnContainedActivitiesChanged(e);
}
#endregion
#endregion
#region Properties and Methods
//
public static void MoveDesigners(ActivityDesigner activityDesigner, bool moveBack)
{
if (activityDesigner == null)
throw new ArgumentNullException("activityDesigner");
Activity activity = activityDesigner.Activity as Activity;
if (activity == null || activity.Parent == null)
return;
CompositeActivity compositeActivity = activity.Parent as CompositeActivity;
if (compositeActivity == null || !compositeActivity.Activities.Contains(activity))
return;
int index = compositeActivity.Activities.IndexOf(activity);
index += (moveBack) ? -1 : 1;
if (index < 0 || index >= compositeActivity.Activities.Count)
return;
IDesignerHost designerHost = compositeActivity.Site.GetService(typeof(IDesignerHost)) as IDesignerHost;
if (designerHost == null)
return;
DesignerTransaction trans = designerHost.CreateTransaction(SR.GetString(SR.MovingActivities));
try
{
compositeActivity.Activities.Remove(activity);
compositeActivity.Activities.Insert(index, activity);
ISelectionService selectionService = compositeActivity.Site.GetService(typeof(ISelectionService)) as ISelectionService;
if (selectionService != null)
selectionService.SetSelectedComponents(new object[] { activity });
if (trans != null)
trans.Commit();
}
catch
{
if (trans != null)
trans.Cancel();
throw;
}
CompositeActivityDesigner compositeDesigner = ActivityDesigner.GetDesigner(compositeActivity) as CompositeActivityDesigner;
if (compositeDesigner != null)
compositeDesigner.PerformLayout();
}
#endregion
}
#endregion
}