重新托管的WF4删除活动方案

时间:2015-05-14 07:46:04

标签: wpf workflow-foundation-4 workflow-foundation workflow-activity

为了达到Assign活动的上下文,我将其托管到以下自定义活动中:

<sap:ActivityDesigner x:Class="ARIASquibLibrary.Design.CustomAsignDesigner"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:sap="clr-namespace:System.Activities.Presentation;assembly=System.Activities.Presentation"
xmlns:sapv="clr-namespace:System.Activities.Presentation.View;assembly=System.Activities.Presentation"
xmlns:statements="http://schemas.microsoft.com/netfx/2009/xaml/activities" Collapsible="False">
<sap:ActivityDesigner.Template>
    <ControlTemplate TargetType="sap:ActivityDesigner">
        <Grid>
            <ContentPresenter HorizontalAlignment="Center"
                                      VerticalAlignment="Center"/>
        </Grid>
    </ControlTemplate>
</sap:ActivityDesigner.Template>
<DockPanel LastChildFill="True">
    <sap:WorkflowItemPresenter Item="{Binding Path=ModelItem.Body, Mode=TwoWay}"/>
</DockPanel>

来自图书馆的代码:

public sealed class CustomAssign : NativeActivity, IActivityTemplateFactory
{
    [Browsable(false)]
    public Activity Body { get; set; }


    protected override void Execute(NativeActivityContext context)
    {
        ActivityInstance res = context.ScheduleActivity(Body, new CompletionCallback(OnExecuteComplete));
    }

    /// <summary>
    /// Called from Execute when Condition evaluates to true.
    /// </summary>
    /// <param name="context">The context.</param>
    /// <param name="instance">The instance.</param>
    public void OnExecuteComplete(NativeActivityContext context, ActivityInstance instance)
    {
        //to be added
    }

    Activity IActivityTemplateFactory.Create(System.Windows.DependencyObject target)
    {
        return new CustomAssign
        {
            Body = new Assign()
        };
    }
}

在设计器上拖动此活动时,它看起来与默认的Assign完全相同。但尝试删除它时会出现此问题。被托管时,其父母将留在设计师身上,因为事实上,即使用户没有看到这个,我也会删除身体。

有没有办法传播到父级,以便从设计器中删除整个活动?

1 个答案:

答案 0 :(得分:1)

好的,您需要覆盖工作流设计器上的OnModelChanged事件。

        ModelService modelSvc = (ModelService)designer.Context.Services.GetService<System.Activities.Presentation.Services.ModelService>();

        if (modelSvc != null)
        {
            modelSvc.ModelChanged += OnModelChanged;
        }

完成此操作后,您可以访问已从设计器中删除的活动。

            void modelSvc_ModelChanged(object sender, ModelChangedEventArgs e)
            {
                //we have deleted something from the design surfaace
                if (e.ModelChangeInfo.ModelChangeType == ModelChangeType.CollectionItemRemoved)
                {
                   ModelItem modelItem = (ModelItem)e.ModelChangeInfo.Value;
                }
            }

根据已删除活动中的模型项,您可以遍历工作流树以查找要删除的相关活动。

这个概念很好地解释了here

然后,您可以使用ModelEditingScope删除活动,并使用以下代码。

 using (ModelEditingScope editingScope = modelSvc.Root.BeginEdit("Activities"))
 {
    //to add an activity
    modelSvc.Root.Properties["Activities"].Collection.Add(new Sequence());
    //to remove an activity
    modelSvc.Root.Properties["Activities"].Collection.Remove(*A model Item to remove);

    editingScope.Complete();

 }

这不是一项微不足道的任务,而且这样做有一些好玩的东西。就个人而言,我没有在我自己的应用程序中实现这个确切的要求,但我已经完成了与删除的活动相关的变量的类似删除,并且有各种各样的问题,但上面的代码应该给你一个起点。