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

Reactivity issue with computed on array first item #12735

Open
clemator opened this issue Jan 17, 2025 · 3 comments · May be fixed by #12737
Open

Reactivity issue with computed on array first item #12735

clemator opened this issue Jan 17, 2025 · 3 comments · May be fixed by #12737
Labels
🔨 p3-minor-bug Priority 3: this fixes a bug, but is an edge case that only affects very specific usage. scope: reactivity

Comments

@clemator
Copy link

Version

3.5.13

Reproduction link

stackblitz.com

Steps to reproduce

Click on the button "Increment first array item" until the first array item object reach 5.

The computed "biggestNumber" is never recomputed, showing 4 instead of 5.

Click once on the second button "Increment second array item". The computed "biggestNumber" is now recomputed and displays 5 as expected.

Click on the first button, this time the computed is always recomputed on each clicks.

Click on the second button until it's value is superior to the first array item. Then click on the first one, it loses reactivity again.

This only happens to the first array item, redo steps with the first button and third or first and fourth.

What is expected?

The computed should react to changes on the first array item.

What is actually happening?

The first item never triggers computed reactivity until other array items changes.


Thank you for your work, love Vue <3

@skirtles-code
Copy link
Contributor

skirtles-code commented Jan 17, 2025

I think the problem is that the first argument passed to reduce isn't wrapped in a proxy. e.g.:

reactive([{ a: 1 }, { b: 2 }]).reduce((c, d) => {
  console.log(c, d)
})

d is wrapped in a proxy, but c isn't.

No initial value is being passed to reduce here, so the unwrapped first item in the array is treated as the initial value.

@danolu
Copy link

danolu commented Jan 17, 2025

Your error is caused by how the reduce function is used.

"The first time that the callback is run there is no "return value of the previous calculation". If supplied, an initial value may be used in its place. Otherwise the array element at index 0 is used as the initial value and iteration starts from the next element (index 1 instead of index 0)."

More can be found here.

If you don't want your program to skip the first execution, you must pass the optional initialValue as the second argument to your reduce function. Assuming that your values will be only positive integers, { "value": 0 } will be ideal as the initialValue argument.

@edison1105 edison1105 added scope: reactivity 🔨 p3-minor-bug Priority 3: this fixes a bug, but is an edge case that only affects very specific usage. labels Jan 20, 2025
@ics90
Copy link

ics90 commented Jan 22, 2025

const maxBy = (arr, iteratee) => {
return arr.reduce((maxItem, currentItem) => {
const cur = iteratee(currentItem);
return cur > maxItem ? cur : maxItem;
}, Number.MIN_VALUE);
};

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
🔨 p3-minor-bug Priority 3: this fixes a bug, but is an edge case that only affects very specific usage. scope: reactivity
Projects
None yet
Development

Successfully merging a pull request may close this issue.

5 participants