New Posts New Posts RSS Feed: EntityAspect.PropertyChanged, Event bubbling and Dirtying Parent objects.
  FAQ FAQ  Forum Search   Calendar   Register Register  Login Login

EntityAspect.PropertyChanged, Event bubbling and Dirtying Parent objects.

 Post Reply Post Reply
Author
skingaby View Drop Down
DevForce MVP
DevForce MVP
Avatar

Joined: 23-Apr-2008
Location: United States
Posts: 146
Post Options Post Options   Quote skingaby Quote  Post ReplyReply Direct Link To This Post Topic: EntityAspect.PropertyChanged, Event bubbling and Dirtying Parent objects.
    Posted: 16-Dec-2009 at 7:56pm
This is a little bit weird. We have added a PropertyChanged handler to our base child entity so it can bubble events up to its parent object and bubble events for readonly properties.
Please bear with me. First I will explain why I need this foolishness. Second, I will show the code. Third, I will ask my question.

I have two example requirements:
A) When any OrderDetail changes, the UI should refresh the OrderTotal; and
B) When any part of an Order or OrderDetail changes, the Order and the changed Order Details should be saved as a set.

For A), When a user changes an OrderDetail:BaseChildEntity in the Order:BaseEntity, the readonly Order.OrderTotal property needs to fire a PropertyChanged("OrderTotal") event so the UI control that is bound to it will refresh. In order to do this, I am capturing the Entity.EntityAspect.IsChanged property change in the child entity and bubbling it up from the EntityAspect to the parent.

For B), When a user changes an OrderDetail:BaseChildEntity in the Order:BaseEntity, the Order should be marked as dirty (i.e. Order.EntityAspect.IsChanged) so that when saved, the Entity Manager will save the Order and the changed Order Details.

To accomplish this, I have coded the following:

1) In BaseEntity, which Order derives from:
public abstract class BaseEntity : Entity

{
     protected BaseEntity()
     {
          base.EntityAspect.PropertyChanged += EntityAspectPropertyChanged; // This is probably NOT the right place to do this, but until we can find something better, this seems to work. Suggestions?
     }

     private void EntityAspectPropertyChanged(object sender, PropertyChangedEventArgs e)
     {
          this.EntityAspectPropertyChangedInternal(sender, e);
     }
     
     protected virtual void EntityAspectPropertyChangedInternal(object sender, PropertyChangedEventArgs e)
     {
          if (e.PropertyName == "IsChanged") //take the Entity.EntityAspect.IsChanged property change
          {
               RaisePropertyChanged("IsChanged"); //and turn it into an Entity.IsChanged property change
          }
     }
}


2) In BaseChildEntity, which OrderDetail derives from:
public abstract class BaseChildEntity : BaseEntity

{
     protected abstract BaseEntity Parent { get; set; }

     protected override void EntityAspectPropertyChangedInternal(object sender, System.ComponentModel.PropertyChangedEventArgs e)
     {     
          base.EntityPropertyChangedInternal(sender, e);
          if (e.PropertyName == "IsChanged" && this.EntityAspect.IsChanged)
          {
               //entity just got dirtied, so dirty the parent too
               this.Parent.EntityAspect.SetModified();
          }
     }
}


3) In OrderDetail:
public partial class OrderDetail

{
     protected override BaseEntity Parent
     {
          get
          {
               return Order; //Expose the RelatedEntity>Order as this OrderDetail's Parent
          }
          set
          {
               Order = value as Order;
          }
     }
}


4) In Order:
public partial class Order

{
     public double OrderTotal()
     {
          return Sum(this.OrderDetails);
     }

     private void OrderDetailPropertyChanged(object sender, PropertyChangedEventArgs e) //there is a horrible amount of code in this class that handles ensuring that this event handler is tied to each OrderDetail.PropertyChanged event.
     {
          this.RaisePropertyChanged("OrderTotal");
     }
     
}


OK. So here's what I expect to happen:

For requirement A) When an OrderDetail is edited, the OrderDetail.EntityAspect.IsChanged property gets changed, which is caught in BaseEntity.EntityAspectPropertyChanged, which then raises the OrderDetail.IsChanged property changed event, which is caught by the Order.OrderDetailPropertyChanged event handler, which then raises the Order.OrderTotal property changed event, which tells the UI bound to Order.OrderTotal to refresh.

For requirement B) When an OrderDetail is edited, the EntityAspect.IsChanged property gets changed, which is caught in BaseEntity.EntityAspectPropertyChanged, which then delegates to the BaseChildEntity.EntityAspectPropertyChangedInternal, which calls SetModified() on the parent Order. Now, when the Order is saved, the EM will save the Order and the changed OrderDetails all at once. When the Em.SaveChangesAsync is called, I expect all items to be saved and reset to EntityState = NotModified and IsChanged = false.

But, of course, it didn't work out that way.

First, is this just ridiculous? Am I doing something silly to accomplish what I need? My actual problem is messier and more complicated than this, so I would love to hear someone say this is not insane.

Second, when the EntityManager.SaveChangesAsync processes, the BaseEntity.EntityAspectPropertyChanged is called twice. Once for the EntityAspect.EntityState property, and once for the EntityAspect.IsChanged property. I assumed this was firing when these items are being reset after the save/replace process, but when the OrderDetail calls these, the test in the BaseChildEntity.EntityAspectPropertyChangedInternal > if (e.PropertyName == "IsChanged" && this.EntityAspect.IsChanged) still returns true. Why? What am I doing wrong? How can I accomplish what I need?
Here's the call stack at that moment:

Model.DomainModel!DomainModel.BaseChildEntity.EntityAspectPropertyChangedInternal(object sender = {IdeaBlade.EntityModel.EntityAspect}, System.ComponentModel.PropertyChangedEventArgs e = {System.ComponentModel.PropertyChangedEventArgs}) Line 39     C#
>     Png.GcsAg.Model.DomainModel!Png.GcsAg.Model.DomainModel.BaseEntity.EntityAspect_PropertyChanged(object sender = {IdeaBlade.EntityModel.EntityAspect}, System.ComponentModel.PropertyChangedEventArgs e = {System.ComponentModel.PropertyChangedEventArgs}) Line 42 + 0x11 bytes     C#
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityAspect.OnPropertyChanged(System.ComponentModel.PropertyChangedEventArgs args = {System.ComponentModel.PropertyChangedEventArgs}) + 0x45 bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityAspect.add_PropertyChanged.AnonymousMethod(object o = {DealPricing: 18485}, System.ComponentModel.PropertyChangedEventArgs args = {System.ComponentModel.PropertyChangedEventArgs}) + 0x26 bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityWrapper.OnEntityAspectPropertyChanged(string propertyName = "IsChanged") + 0x92 bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityWrapper.EntityState.set(IdeaBlade.EntityModel.EntityState value = Unchanged) + 0x77 bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityWrapper.ReplaceAll(IdeaBlade.EntityModel.EntityWrapper sourceEntity = {DealPricing: 18485}, bool copy = true) + 0x9f bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityWrapper.ReplaceEntity(IdeaBlade.EntityModel.EntityWrapper sourceEntity = {DealPricing: 18485}, IdeaBlade.EntityModel.MergeStrategy mergeStrategy = OverwriteChanges) + 0x71 bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityGroup.ImportEntity(IdeaBlade.EntityModel.EntityWrapper sourceWrapper = {DealPricing: 18485}, IdeaBlade.EntityModel.MergeStrategy mergeStrategy = OverwriteChanges) + 0x89 bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityManager.SaveEntitiesPostProcessing(IdeaBlade.EntityModel.SaveWorkState workstate = {IdeaBlade.EntityModel.SaveWorkState}) + 0x795 bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.EntityManager.SaveChangesAsyncCore.AnonymousMethod(IdeaBlade.EntityModel.EntitySavedEventArgs args = {IdeaBlade.EntityModel.EntitySavedEventArgs}) + 0x51 bytes     
IdeaBlade.EntityModel.SL!IdeaBlade.EntityModel.AsyncProcessor<IdeaBlade.EntityModel.EntitySavedEventArgs>.Execute.AnonymousMethod(object x = {IdeaBlade.EntityModel.EntitySavedEventArgs}) + 0x73 bytes     
[Native to Managed Transition]     
[Managed to Native Transition]     
mscorlib.dll!System.Delegate.DynamicInvokeImpl(object[] args = {object[1]}) + 0xb4 bytes     
mscorlib.dll!System.Delegate.DynamicInvoke(object[] args = {object[1]}) + 0x2a bytes     
System.Windows.dll!System.Windows.Threading.DispatcherOperation.Invoke() + 0x47 bytes     
System.Windows.dll!System.Windows.Threading.Dispatcher.Dispatch(System.Windows.Threading.DispatcherPriority priority = 13) + 0x161 bytes     
System.Windows.dll!System.Windows.Threading.Dispatcher.OnInvoke(object context = null) + 0x28 bytes     
System.Windows.dll!System.Windows.Hosting.CallbackCookie.Invoke(object[] args = {object[0]}) + 0x37 bytes     
System.Windows.dll!System.Windows.Hosting.DelegateWrapper.InternalInvoke(object[] args = {object[0]}) + 0x2a bytes     
System.Windows.Browser.dll!System.Windows.Hosting.ScriptingInterface.InvokeDelegate(System.Windows.Hosting.DelegateWrapper delegateWrapper = {System.Windows.Hosting.CallbackCookie}, System.Windows.Hosting.NativeMethods.ScriptParam[] pParams = {System.Windows.Hosting.NativeMethods.ScriptParam[0]}, ref System.Windows.Hosting.NativeMethods.ScriptParam pResult = {System.Windows.Hosting.NativeMethods.ScriptParam}) + 0x68 bytes     
System.Windows.Browser.dll!System.Windows.Hosting.ManagedHost.InvokeDelegate(System.IntPtr pHandle = 151065016, int nParamCount = 0, System.Windows.Hosting.NativeMethods.ScriptParam[] pParams = {System.Windows.Hosting.NativeMethods.ScriptParam[0]}, ref System.Windows.Hosting.NativeMethods.ScriptParam pResult = {System.Windows.Hosting.NativeMethods.ScriptParam}) + 0x124 bytes     
[Appdomain Transition]     
[Native to Managed Transition]     

Thanks, Simon.
Back to Top
GregD View Drop Down
IdeaBlade
IdeaBlade
Avatar

Joined: 09-May-2007
Posts: 374
Post Options Post Options   Quote GregD Quote  Post ReplyReply Direct Link To This Post Posted: 22-Dec-2009 at 11:50am
>>
OK. So here's what I expect to happen:

For requirement A) When an OrderDetail is edited, the OrderDetail.EntityAspect.IsChanged property gets changed, which is caught in BaseEntity.EntityAspectPropertyChanged, which then raises the OrderDetail.IsChanged property changed event, which is caught by the Order.OrderDetailPropertyChanged event handler, which then raises the Order.OrderTotal property changed event, which tells the UI bound to Order.OrderTotal to refresh.

For requirement B) When an OrderDetail is edited, the EntityAspect.IsChanged property gets changed, which is caught in BaseEntity.EntityAspectPropertyChanged, which then delegates to the BaseChildEntity.EntityAspectPropertyChangedInternal, which calls SetModified() on the parent Order. Now, when the Order is saved, the EM will save the Order and the changed OrderDetails all at once. When the Em.SaveChangesAsync is called, I expect all items to be saved and reset to EntityState = NotModified and IsChanged = false.

But, of course, it didn't work out that way.

First, is this just ridiculous? Am I doing something silly to accomplish what I need? My actual problem is messier and more complicated than this, so I would love to hear someone say this is not insane.
<<

It seems okay on paper, and you haven't said exactly what didn't work out. I might be tempted to do something a little more direct; e.g., when any property of the OrderDetail changes, just call SetModified() directly on the parent Order; and when any property on the OrderDetail that affects price changes, call OnPropertyChanged(OrderTotal), again directly, on the parent Order.

Back to Top
skingaby View Drop Down
DevForce MVP
DevForce MVP
Avatar

Joined: 23-Apr-2008
Location: United States
Posts: 146
Post Options Post Options   Quote skingaby Quote  Post ReplyReply Direct Link To This Post Posted: 23-Dec-2009 at 9:35am
Here's the problem:

Second, when the EntityManager.SaveChangesAsync processes, the BaseEntity.EntityAspectPropertyChanged is called twice. Once for the EntityAspect.EntityState property, and once for the EntityAspect.IsChanged property. I assumed this was firing when these items are being reset after the save/replace process, but when the OrderDetail calls these, the test in the BaseChildEntity.EntityAspectPropertyChangedInternal > if (e.PropertyName == "IsChanged" && this.EntityAspect.IsChanged) still returns true. Why? What am I doing wrong? How can I accomplish what I need?

Or, more succinctly: when SaveChangesAsync comes back and the entity is replaced by the EM, the IsChanges property is set to false and its event is fired, but the IsChanges getter is still returning true. Why?
Back to Top
skingaby View Drop Down
DevForce MVP
DevForce MVP
Avatar

Joined: 23-Apr-2008
Location: United States
Posts: 146
Post Options Post Options   Quote skingaby Quote  Post ReplyReply Direct Link To This Post Posted: 23-Dec-2009 at 9:36am
Also, a couple of days ago I completely rewrote these classes to use an AfterSet interceptor. The approach shown in this thread, however, should only fire once when the entity is first dirtied, by using an AfterSet interceptor it fires a gazillion times because it fires any time any property changes. What's worse, if a property has an interceptor that changes another property, then everything fires twice.

Edited by skingaby - 23-Dec-2009 at 9:37am
Back to Top
WardBell View Drop Down
IdeaBlade
IdeaBlade
Avatar

Joined: 31-Mar-2009
Location: Emeryville, CA,
Posts: 338
Post Options Post Options   Quote WardBell Quote  Post ReplyReply Direct Link To This Post Posted: 21-Jan-2010 at 1:25pm
Hi Simon - You have a typical case of (A) cross entity dependence and (B) aggregate entity.
 
I'm not sure I would address either in the ways you have proposed. The moment I saw you listening for changes to EntityAspect.EntityState deep in a base class I tuned out. "Hitting a bug with a sledge hammer" I thought. Maybe I'm just lazy. Permit me to proceed (theoretically ... I haven't actually written this out) in a different manner.
 
--- (A) ORDER TOTAL ---
 
What affects the Order Total?
- Adding a detail
- Removing a detail
- Changing a detail's quanity, price, discount, ???
 
Let's start with Add and Remove. Personally, because this is an Aggregate, I would channel all adding and removing of OrderDetails through the AggregateRoot. I would PREVENT any process from creating/deleting an OrderDetail independently. The best way to do that is in the public constructor of the OrderDetail. Make it throw an exception.
 
Alternatively, you could throw an exception when anyone tries to set the OrderDetail's parent Order or parent OrderId.
 
Order should sport AddDetail and RemoveDetail methods. Only Order has the ability to set the OrderDetail's parent OrderId (devise an internal method for the purpose).
 
Now that you've channeled adding / deleting through the Order, there is no problem ensuring that Order raises property changed for OrderTotal during detail add/delete.
 
Still want to ignore my advice? Well you can also arrange for the Order to listen to changes in its own OrderDetail collection. You will find that OrderDetailsProperty.GetValue(this) returns the RelatedEntityList<OrderDetail> which has a CollectionChanged event you can listen to. You should be able to hook this by writing your own Order default constructor.  Haven't tried but should work.
 
On to OrderDetail changes.
 
On simple approach is to provide a RaiseOrderTotalChanged method on the Order. In the OrderDetail, you can override OnPropertyChanged and have it call RaiseOrderTotalChanged on its parent Order.
 
Interesting, I would expect this to fire when adding a detail to a parent Order. You might only have to worry about how to notify the Order when you remove a detail.
 
--- (B) A CHANGE TO ORDER DETAIL IS A CHANGE TO THE PARENT ORDER ---
 
"Order", in this example, is what is known as an "Aggregate root entity". The Order Aggregate in this example consists of Order and its OrderDetails. They are an "Aggregate" because we have several entity types in a common graph that are always treated as a single thing. An aggregate always has a root (Order) in this case; the other members of the aggregate (OrderDetail in this case) cannot exist apart from the root entity.
 
This fact, by the way, explains why Product is not in the Aggregate even though Product is clearly part of Order's extended object graph. A change to Product does not propagate to changes in all Orders of that product (at least not in our example Order management system).
 
I think its important  to treat each Aggregate entity individually. 
 
I would program for the Order Aggregate by itself. The pattern applies to other aggregates certainly. But I wouldn't drive the behavior we're discussing down to a base entity shared by all entities (except as noted below ... there is always an except :-)  ).
 
You can dirty the Order the moment a Detail changes. There are reasons to do this; perhaps you want it reflected immediately in the UI. Perhaps you need to know if the Order permits changes to one of its details!
 
We already established a mechanism for the OrderDetail to notify its parent Order of changes. It might as well dirty the parent Order at the same time via the same call. As part of that call, it checks to see if the Order is in the Unchanged state; if it is, call Order.EntityAspect.SetModified() .
 
If you need to ensure that no one modifies a locked down detail, you should do this in a custom, pre-set ValidationRule, defined for all properties of the OrderDetail.
 
---
However, I would not rely exclusively on dirtying the Order when an OrderDetail changes. The user could undo the OrderDetail change in which case the Order may not be dirty. Or the user might call undo changes on the Order even though one of its details is still dirty. You want to be sure to have integrity and consistency at the moment of save. That's why I look to the EntityManager's SavingHandler.
 
You have one right? You should. That's the place to validate the entities you are about to save before attempting to save them to the database.
 
In the SavingHandler, I validate every entity scheduled for save. I do this by calling into each entity's ValidateForSave method. "But they don't have such a method" you protest. Indeed they don't. You should add it. It's a good BaseEntity method.
 
Arguments to the ValidateForSave  method include the (a) list of entities to save, (b) an empty list of entities to add to that list, and (c) an empty list of entities to remove from that list [rarely needed].
 
Within the ValidateForSave, you can add any other related entity (entities) to lists (b) or (c).
 
Your saving handler, which is iterating over list (a), looks for (b) and (c) coming back from the validated entity and takes appropriate action to update list (a) before calling ValidateForSave on the next item.
 
The point of this is that the Detail checks to see if its parent Order is in list (a). If not, it dirties the order (as described above), and adds the dirtied parent Order to list (b).
 
Consequently, the SavingHandler will include the dirtied parent Order among its list of entities to save; don't forget that the SavingHandler must validate that dirty parent Order as well!
 
I should write this as a separate post and sample. I've been meaning to for years.
 
 
 
Back to Top
skingaby View Drop Down
DevForce MVP
DevForce MVP
Avatar

Joined: 23-Apr-2008
Location: United States
Posts: 146
Post Options Post Options   Quote skingaby Quote  Post ReplyReply Direct Link To This Post Posted: 24-Jan-2010 at 2:23pm
Interesting. I hadn't thought about using the Saving handler for this. I shall have to consider how that will work in our grid row model (yeuch).
Back to Top
 Post Reply Post Reply

Forum Jump Forum Permissions View Drop Down