Reactivity in JavaScript

in #javascript7 years ago (edited)

Reactivity in JS

<blockquote> <p dir="auto">context: I have been using <a href="https://vuejs.org/" target="_blank" rel="nofollow noreferrer noopener" title="This link will take you away from hive.blog" class="external_link">Vue.jsfor a number of side projects and really enjoy working with it, I also use <a href="http://knockoutjs.com/" target="_blank" rel="nofollow noreferrer noopener" title="This link will take you away from hive.blog" class="external_link">Knockout.jsa lot in work which uses a similar approach. I wanted to dive into the abstracted functionality a little deeper to help me understand the libraries I work with. <p dir="auto">How do changes in state translate to changes in the DOM? <p dir="auto">Let's start with a variable: <p dir="auto"><code>let a = 10 <p dir="auto">and another one: <p dir="auto"><code>let b = a * 10 <p dir="auto">the complete picture now looks like this: <pre><code> let a = 10 let b = a * 10 <p dir="auto">We want <code>b to always be 10 times of a. We now have a problem, when <code>a changes <code>b will not be aware of that change and not update, no matter what <code>a is,<code>b will always be the original value of <code>a times 10. <p dir="auto">This is a imperative and procedural implementation which does not solve the problem fully, what we need is an approach that is declarative and object oriented. <p dir="auto">In an ideal world we would have a function that looked like this: <pre><code> onStateChanged( () => { b = a * 10 }) <p dir="auto">This function would run every time <code>a changes to update the value of <code>b. So how do we implement this function? We can see that it takes a function that is then ran to set the new state. <blockquote> <p dir="auto">jargon buster: a program is described as stateful if it is designed to remember preceding events or user interactions; the remembered information is called the state of the system. <p dir="auto">// pseudo code <pre><code>let update; // the function that manipulates the state. const onSateChanged = _update => { update = _update; // sets update the passed in function } const setState = newState => { // receives the new state and then called update state = newState update(); } <blockquote> <p dir="auto">This is an approach that React uses and why setState is always used to well.. set the state. <p dir="auto">There is another way that involves taking the state object and converting it into a reactive object. <p dir="auto">Our <code>onStateChanged function will be renamed <code>autorun why will become clearer as we move through the examples. <p dir="auto">Let's implement a mini data observer and look at how Vue.js handles things. <p dir="auto">first we need to define accessors using the <a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty" target="_blank" rel="nofollow noreferrer noopener" title="This link will take you away from hive.blog" class="external_link">Object.defineProperty() API. This will allow us later to get and set the current state. <p dir="auto"><img src="https://images.hive.blog/768x0/https://steemitimages.com/DQmXYQesGUkgJ2EVcGWdH1tKFo7wGAtw8v9hACRbC5Sb9TM/convert.png" alt="convert.png" srcset="https://images.hive.blog/768x0/https://steemitimages.com/DQmXYQesGUkgJ2EVcGWdH1tKFo7wGAtw8v9hACRbC5Sb9TM/convert.png 1x, https://images.hive.blog/1536x0/https://steemitimages.com/DQmXYQesGUkgJ2EVcGWdH1tKFo7wGAtw8v9hACRbC5Sb9TM/convert.png 2x" /> <p dir="auto"><a href="https://jsfiddle.net/harps116/wc2e0vgv/16/" target="_blank" rel="nofollow noreferrer noopener" title="This link will take you away from hive.blog" class="external_link">View the code <p dir="auto">That seems to be working as expected, we have a few more parts to create. Later we will change the name to observe rather than convert. <p dir="auto">We are going to create a dependency class that keeps track of the things our state depends on, it will also notify those that depend on it that it has changed. <p dir="auto"><img src="https://images.hive.blog/768x0/https://steemitimages.com/DQmVjNcmz6vjhJhSbjhbisgdHeWKntN1ezUQtnQuzB4Z5xm/dep.png" alt="dep.png" srcset="https://images.hive.blog/768x0/https://steemitimages.com/DQmVjNcmz6vjhJhSbjhbisgdHeWKntN1ezUQtnQuzB4Z5xm/dep.png 1x, https://images.hive.blog/1536x0/https://steemitimages.com/DQmVjNcmz6vjhJhSbjhbisgdHeWKntN1ezUQtnQuzB4Z5xm/dep.png 2x" /> <p dir="auto"><img src="https://images.hive.blog/768x0/https://steemitimages.com/DQmWb7LoH3tqzyzYRB9reE4N946rgsYYBYRvR64RMghMCfj/dep-2.png" alt="dep-2.png" srcset="https://images.hive.blog/768x0/https://steemitimages.com/DQmWb7LoH3tqzyzYRB9reE4N946rgsYYBYRvR64RMghMCfj/dep-2.png 1x, https://images.hive.blog/1536x0/https://steemitimages.com/DQmWb7LoH3tqzyzYRB9reE4N946rgsYYBYRvR64RMghMCfj/dep-2.png 2x" /> <p dir="auto"><a href="https://jsfiddle.net/harps116/7wf1vc2j/16/" target="_blank" rel="nofollow noreferrer noopener" title="This link will take you away from hive.blog" class="external_link">View the code <p dir="auto">This of course is not the complete picture and a simplified version of the general idea. Notice the wrappedUpdate. This is a little trick to ensure that the correct function is added to the subscribers list. As it is assigned to the activeUpdate which our dependency instance checks for in its depend method, we can ensure that what we add to the subscribers list is currently active. <p dir="auto">Putting it all together: <p dir="auto"><img src="https://images.hive.blog/768x0/https://steemitimages.com/DQmU9uNzVuRC6xFwjikr3sD2XLyU4bJLPu3UjJvWwjLCJND/3.png" alt="3.png" srcset="https://images.hive.blog/768x0/https://steemitimages.com/DQmU9uNzVuRC6xFwjikr3sD2XLyU4bJLPu3UjJvWwjLCJND/3.png 1x, https://images.hive.blog/1536x0/https://steemitimages.com/DQmU9uNzVuRC6xFwjikr3sD2XLyU4bJLPu3UjJvWwjLCJND/3.png 2x" /><br /> <img src="https://images.hive.blog/768x0/https://steemitimages.com/DQmccyMkujPF8CqmGqunJY43H6hbYwH9vUuxaaAy91GkzfX/3.2.png" alt="3.2.png" srcset="https://images.hive.blog/768x0/https://steemitimages.com/DQmccyMkujPF8CqmGqunJY43H6hbYwH9vUuxaaAy91GkzfX/3.2.png 1x, https://images.hive.blog/1536x0/https://steemitimages.com/DQmccyMkujPF8CqmGqunJY43H6hbYwH9vUuxaaAy91GkzfX/3.2.png 2x" /><br /> <img src="https://images.hive.blog/768x0/https://steemitimages.com/DQmNyz6mtQmaYaFkGsHLtqCzQfHviMtX8sbFNdEAxhZw4Pp/3.3.png" alt="3.3.png" srcset="https://images.hive.blog/768x0/https://steemitimages.com/DQmNyz6mtQmaYaFkGsHLtqCzQfHviMtX8sbFNdEAxhZw4Pp/3.3.png 1x, https://images.hive.blog/1536x0/https://steemitimages.com/DQmNyz6mtQmaYaFkGsHLtqCzQfHviMtX8sbFNdEAxhZw4Pp/3.3.png 2x" /> <p dir="auto"><a href="https://jsfiddle.net/harps116/gL46sfxs/4/" target="_blank" rel="nofollow noreferrer noopener" title="This link will take you away from hive.blog" class="external_link">View the code <p dir="auto">Read through the code and play around with the state. You see it log out the changes. <p dir="auto">You can read more about reactivity in Vue <a href="https://vuejs.org/v2/guide/reactivity.html" target="_blank" rel="nofollow noreferrer noopener" title="This link will take you away from hive.blog" class="external_link">here. <p dir="auto">This idea may change a little in up coming version of Vue. You can read more about it <a href="https://blog.cloudboost.io/reactivity-in-vue-js-2-vs-vue-js-3-dcdd0728dcdf" target="_blank" rel="nofollow noreferrer noopener" title="This link will take you away from hive.blog" class="external_link">here. <p dir="auto">Happy Coding!
Sort:  

I'm a React guy, but if I would have to switch, I would switch to Vue (or HyperApp). Thanks for sharing.

Congratulations @harps116! You received a personal award!

Happy Birthday! - You are on the Steem blockchain for 2 years!

You can view your badges on your Steem Board and compare to others on the Steem Ranking

Vote for @Steemitboard as a witness to get one more award and increased upvotes!