Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Bug]: AutoRefresh() on a collection of objects triggers listeners before their own internal OAPH listeners? #3922

Open
seleborg opened this issue Oct 11, 2024 · 4 comments

Comments

@seleborg
Copy link

Describe the bug 🐞

First off, thanks for all the work on ReactiveUI, it is a very helpful library!

Let's say I have a ViewModel object with two properties, one of which is just a different representation of the other one. I'm using OAPHs to derive the second property's from the first property's value. I set this up in the constructor.

I then create an ObservableCollection<ViewModel> of such objects and want to listen to changes to that collection based on the property. It turns out that that listener gets called before the OAPH's listener, such that I observe an inconsistent state.

I'm very confused by this, I would expect that the OAPH's listener, being set up in the constructor and running on the same thread as the property update, would always run before any external listeners, necessarily set up later. This does not seem to be the case.

A few additional notes:

  • If instead of ToObservableChangeSet().AutoRefresh() I just use vm.WhenAnyValue(x => x.Number), I get the expected result, so it seems to be something with ToObservableChangeSet().AutoRefresh().
  • I'm submitting the minimal example as a unit test, but this behaviour also occurs when running normally, i.e. not in unit tests.

Step to reproduce

Here's a minimal example, written as a unit test (apologies for the non-standard code style, that's just how my editor is set up):

public class ViewModel : ReactiveObject {
    int _number;
    public int Number {
      get => _number;
      set => this.RaiseAndSetIfChanged(ref _number, value);
    }

    ObservableAsPropertyHelper<string> _asString;
    public string AsString => _asString.Value;

    public ViewModel() {
      _number = 0;
      _asString = this.WhenAnyValue(x => x.Number)
        .Select(n => n.ToString())
        .ToProperty(this, nameof(AsString));
    }
  }
  [Fact]
  public static void TestOrdering() {
    ViewModel vm = new();
    bool initializing = true;
    ObservableCollection<ViewModel> vms = new ObservableCollection<ViewModel> { vm };
    _ = vms.ToObservableChangeSet()
      .AutoRefresh(x => x.Number)
      .Subscribe(_ => {
        Debug.WriteLine($"{initializing} vm.Number={vm.Number} vm.AsString={vm.AsString}");
      });
    initializing = false;
    vm.Number = 1;
  }

In this example, ViewModel has a Number property, which can be set, and an AsString property which is supposed to give the string representation of this number. When running this example, I get the following two log lines, the first of which can be ignored:

08:41:13:537	True vm.Number=0 vm.AsString=0
08:41:13:537	False vm.Number=1 vm.AsString=0

As you can see, Number and AsString represent different values, making my ViewModel's state look inconsistent.

Reproduction repository

No response

Expected behavior

I would expect the OAPH to run first, given I'm setting it up first and am not specifying that anything runs in the background, and would expect the following log lines (with Number and AsString showing the same thing):

08:41:13:537	True vm.Number=0 vm.AsString=0
08:41:13:537	False vm.Number=1 vm.AsString=1

Screenshots 🖼️

No response

IDE

Visual Studio 2022

Operating system

Windows

Version

No response

Device

No response

ReactiveUI Version

20.1.63

Additional information ℹ️

No response

@seleborg seleborg added the bug label Oct 11, 2024
@seleborg
Copy link
Author

seleborg commented Jan 3, 2025

This bug can be closed: I missed the part where the documentation states that the OAPH uses a scheduler to update the property:

It is important to note that OAPH will not set the property value immediately, but will rather schedule it on the provided or default scheduler. The [Reactive] property with .BindTo() should be used for business-critical code.

When using an OAPH to set up a class invariant, one possible workaround is to set ToProperty's scheduler argument to System.Reactive.Concurrency.ImmediateScheduler.Instance.

@seleborg seleborg closed this as not planned Won't fix, can't repro, duplicate, stale Jan 3, 2025
@seleborg
Copy link
Author

seleborg commented Jan 3, 2025

This bug can be closed: I missed the part where the documentation states that the OAPH uses a scheduler to update the property:

It is important to note that OAPH will not set the property value immediately, but will rather schedule it on the provided or default scheduler. The [Reactive] property with .BindTo() should be used for business-critical code.

When using an OAPH to set up a class invariant, one possible workaround is to set ToProperty's scheduler argument to System.Reactive.Concurrency.ImmediateScheduler.Instance.

@seleborg
Copy link
Author

seleborg commented Jan 3, 2025

I still wonder whether this should be make clearer through the type system, though. When using objects (as opposed to value types), the documentation recommends the property be implemented as such:

public class SomeOwnerClass
{
  // ...
  private ObservableAsPropertyHelper<SomeClass> _someProperty;
  public SomeClass SomeProperty => _someProperty.Value;
  // ...
}

This means that after the constructor is finished, having set up _someProperty with WhenAnyValue and ToProperty, calling new SomeOwnerClass().SomeProperty can return null, which is not in line with the property's declaration (otherwise it would be declared as public SomeClass? SomeProperty => _someProperty.Value;.

I guess at the very least, the documentation for OAPH should encourage to declare object properties as nullable types?

Does that make sense?

@dpvreony
Copy link
Member

dpvreony commented Jan 3, 2025

reopening so can look at the feedback

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

2 participants