diff --git a/internal/core/model.rs b/internal/core/model.rs index cb4f33e8031..ab655791aee 100644 --- a/internal/core/model.rs +++ b/internal/core/model.rs @@ -895,15 +895,17 @@ impl Repeater { } fn model(self: Pin<&Self>) -> ModelRc { - // Safety: Repeater does not implement drop and never allows access to model as mutable let model = self.data().project_ref().model; if model.is_dirty() { - *self.data().inner.borrow_mut() = RepeaterInner::default(); - self.data().is_dirty.set(true); + let old_model = model.get_internal(); let m = model.get(); - let peer = self.project_ref().0.model_peer(); - m.model_tracker().attach_peer(peer); + if old_model != m { + *self.data().inner.borrow_mut() = RepeaterInner::default(); + self.data().is_dirty.set(true); + let peer = self.project_ref().0.model_peer(); + m.model_tracker().attach_peer(peer); + } m } else { model.get() diff --git a/internal/core/properties.rs b/internal/core/properties.rs index 5650f5a212f..f08f88e6b9a 100644 --- a/internal/core/properties.rs +++ b/internal/core/properties.rs @@ -861,8 +861,8 @@ impl Property { self.get_internal() } - /// Get the value without registering any dependencies or executing any binding - fn get_internal(&self) -> T { + /// Get the cached value without registering any dependencies or executing any binding + pub fn get_internal(&self) -> T { self.handle.access(|_| { // Safety: PropertyHandle::access ensure that the value is locked unsafe { (*self.value.get()).clone() } diff --git a/tests/cases/models/dirty_model.slint b/tests/cases/models/dirty_model.slint new file mode 100644 index 00000000000..0fa13835532 --- /dev/null +++ b/tests/cases/models/dirty_model.slint @@ -0,0 +1,92 @@ +// Copyright © SixtyFPS GmbH +// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0 + +// Test that having the model being dirty doesn't re-create items +// Only actually changing the model does. + +export component TestCase inherits Window { + property <{xx: int, model :[string]}> model: { model: ["AA", "BB"] }; + in-out property result ; + public function mark_dirty() { + model.xx += 1; + } + public function change() { + model.model = ["CC"]; + } + + HorizontalLayout { + Rectangle {} + for m in model.model : Rectangle { + init => { + result += "Init '" + m + "' (" + model.xx + ")\n"; + } + } + } + +} + +/* +```rust +let instance = TestCase::new().unwrap(); + +slint_testing::send_mouse_click(&instance, 15., 5.); +assert_eq!(instance.get_result(), "Init 'AA' (0)\nInit 'BB' (0)\n"); +instance.set_result("".into()); +slint_testing::send_mouse_click(&instance, 15., 5.); +instance.invoke_mark_dirty(); +slint_testing::send_mouse_click(&instance, 15., 5.); +assert_eq!(instance.get_result(), ""); +slint_testing::send_mouse_click(&instance, 15., 5.); +instance.invoke_change(); +slint_testing::send_mouse_click(&instance, 15., 5.); +assert_eq!(instance.get_result(), "Init 'CC' (1)\n"); +instance.set_result("".into()); +instance.invoke_mark_dirty(); +slint_testing::send_mouse_click(&instance, 15., 5.); +assert_eq!(instance.get_result(), ""); + +``` + +```cpp +auto handle = TestCase::create(); +const TestCase &instance = *handle; + +slint_testing::send_mouse_click(&instance, 15., 5.); +assert_eq(instance.get_result(), "Init 'AA' (0)\nInit 'BB' (0)\n"); +instance.set_result(""); +slint_testing::send_mouse_click(&instance, 15., 5.); +instance.invoke_mark_dirty(); +slint_testing::send_mouse_click(&instance, 15., 5.); +assert_eq(instance.get_result(), ""); +slint_testing::send_mouse_click(&instance, 15., 5.); +instance.invoke_change(); +slint_testing::send_mouse_click(&instance, 15., 5.); +assert_eq(instance.get_result(), "Init 'CC' (1)\n"); +instance.set_result(""); +instance.invoke_mark_dirty(); +slint_testing::send_mouse_click(&instance, 15., 5.); +assert_eq(instance.get_result(), ""); + +``` + + +```js +var instance = new slint.TestCase({}); +slintlib.private_api.send_mouse_click(instance, 15., 5.); +assert.equal(instance.result, "Init 'AA' (0)\nInit 'BB' (0)\n"); +instance.result = ""; +slintlib.private_api.send_mouse_click(instance, 15., 5.); +instance.mark_dirty(); +slintlib.private_api.send_mouse_click(instance, 15., 5.); +assert.equal(instance.result, ""); +slintlib.private_api.send_mouse_click(instance, 15., 5.); +instance.change(); +slintlib.private_api.send_mouse_click(instance, 15., 5.); +assert.equal(instance.result, "Init 'CC' (1)\n"); +instance.result = ""; +instance.mark_dirty(); +slintlib.private_api.send_mouse_click(instance, 15., 5.); +assert.equal(instance.result, ""); +``` +*/ +