ReactiveCommand models calls to the Execute function as a Sequence of the value passed to each Execute. All subscriptions to it are called on the UI thread. There can be many subscriptions directly to ReactiveCommand, there can also be many async functions registered.
Each async function registered takes the value passed to Execute does work and returns a result. Each call to RegisterAsyncCommand returns a new Sequence of the results from each call to the registered function. All subscriptions to the Sequence of results are called on the UI thread.
ReactiveCommand also provides Sequences of bools, that can be subscribed, that represent its state. When the CanExecute Sequence is bound to the enabled property of the button the state of the ReactiveCommand is communicated into the ui.
By default ReactiveCommand will ignore Execute until all the subscribers (sync and async) to the previous Execute call have finished. This can be changed during construction.
Here is some (out-of-order) usage from the sample.
Subscription - async version (Scenario1.Xaml.cpp):
from(enable->RegisterAsyncFunction( | |
[](RoutedEventPattern) | |
{ | |
// background thread | |
// enable and disable commands will be disabled until this is finished | |
std::this_thread::sleep_for(std::chrono::seconds(2)); | |
return true; | |
})) // this is a subscription to the enable ReactiveCommand | |
// back on the ui thread | |
.subscribe([this](bool) // takes whatever was returned above | |
{ | |
// update the ui | |
}); |
Subscription - sync version (Scenario1.Xaml.cpp):
// enable the scenario when enable is executed | |
from(observable(enable)) | |
.where([this](RoutedEventPattern) | |
{ | |
return accelerometer != nullptr; | |
}) | |
.select_many([=](RoutedEventPattern) | |
{ | |
return from(visible) | |
.select_many([=](bool) | |
{ | |
// enable sensor input | |
return rx::from(readingChanged) | |
.take_until(invisible); | |
}) | |
.take_until(endScenario); // this is a subscription to the disable ReactiveCommand | |
}) | |
.subscribe([this](AccelerometerReading^ reading) | |
{ | |
// on the ui thread | |
this->ScenarioOutput_X->Text = reading->AccelerationX.ToString(); | |
this->ScenarioOutput_Y->Text = reading->AccelerationY.ToString(); | |
this->ScenarioOutput_Z->Text = reading->AccelerationZ.ToString(); | |
}); |
Binding (Scenario1.Xaml.cpp):
Creation (Scenario1.Xaml.cpp):
// start out disabled | |
auto enabled = std::make_shared<rx::BehaviorSubject<bool>>(false); | |
// start out not-working | |
auto working = std::make_shared<rx::BehaviorSubject<bool>>(false); | |
// use !enabled and !working to control canExecute | |
enable = std::make_shared < rxrt::ReactiveCommand < RoutedEventPattern> >(observable(from(enabled) | |
.combine_latest([](bool e, bool w) | |
{ | |
return !e && !w; | |
}, working))); | |
// use enabled and !working to control canExecute | |
disable = std::make_shared < rxrt::ReactiveCommand < RoutedEventPattern> >(observable(from(enabled) | |
.combine_latest([](bool e, bool w) | |
{ | |
return e && !w; | |
}, working))); |
Create state sequences that are used by these ReactiveCommand (Scenario1.Xaml.cpp):
// when enable or disable is executing mark as working (both commands should be disabled) | |
observable(from(enable->IsExecuting()) | |
.combine_latest([](bool ew, bool dw) | |
{ | |
return ew || dw; | |
}, disable->IsExecuting())) | |
->Subscribe(observer(working)); | |
// when enable is executed mark the scenario enabled, when disable is executed mark the scenario disabled | |
observable(from(observable(enable)) | |
.select([this](RoutedEventPattern) | |
{ | |
return accelerometer != nullptr; | |
}) | |
.merge(observable(from(observable(disable)) | |
.select([](RoutedEventPattern) | |
{ | |
return false; | |
})))) | |
->Subscribe(observer(enabled)); // this is a subscription to the enable and disable ReactiveCommands | |
typedef TypedEventHandler<Accelerometer^, AccelerometerReadingChangedEventArgs^> AccelerometerReadingChangedTypedEventHandler; | |
auto readingChanged = from(rxrt::FromEventPattern<AccelerometerReadingChangedTypedEventHandler>( | |
[this](AccelerometerReadingChangedTypedEventHandler^ h) | |
{ | |
return this->accelerometer->ReadingChanged += h; | |
}, | |
[this](Windows::Foundation::EventRegistrationToken t) | |
{ | |
this->accelerometer->ReadingChanged -= t; | |
})) | |
.select([](rxrt::EventPattern<Accelerometer^, AccelerometerReadingChangedEventArgs^> e) | |
{ | |
// on sensor thread | |
return e.EventArgs()->Reading; | |
}) | |
// push readings to ui thread | |
.observe_on_dispatcher() | |
.publish() | |
.ref_count(); | |
auto currentWindow = Window::Current; | |
auto visiblityChanged = from(rxrt::FromEventPattern<WindowVisibilityChangedEventHandler, VisibilityChangedEventArgs>( | |
[currentWindow](WindowVisibilityChangedEventHandler^ h) | |
{ | |
return currentWindow->VisibilityChanged += h; | |
}, | |
[currentWindow](Windows::Foundation::EventRegistrationToken t) | |
{ | |
currentWindow->VisibilityChanged -= t; | |
})) | |
.select([](rxrt::EventPattern<Platform::Object^, VisibilityChangedEventArgs^> e) | |
{ | |
return e.EventArgs()->Visible; | |
}) | |
.publish() | |
.ref_count(); | |
auto visible = from(visiblityChanged) | |
.where([](bool v) | |
{ | |
return !!v; | |
}).merge( | |
from(observable(navigated)) | |
.where([](bool n) | |
{ | |
return n; | |
})); | |
auto invisible = from(visiblityChanged) | |
.where([](bool v) | |
{ | |
return !v; | |
}); | |
// the scenario ends when: | |
auto endScenario = | |
// - disable is executed | |
from(observable(disable)) | |
.select([](RoutedEventPattern) | |
{ | |
return true; | |
}).merge( | |
// - the scenario is navigated from | |
from(observable(navigated)) | |
.where([](bool n) | |
{ | |
return !n; | |
})); |