LocalStorage
Table of Contents
Key Terms
localStorage
— an API for storing persistent data in a user's browser.localStorage.setItem(key, value)
— stores the given key:value pair inlocalStorage
. The value is converted to a string using the value's.toString()
methodlocalStorage.getItem(key)
— retrieves the value associated with the given key inlocalStorage
. It will always return a string.localStorage.removeItem(key)
— removes the value associated with the given key fromlocalStorage
.localStorage.clear()
— removes all key:value pairs fromlocalStorage
.toString()
— converts the source value or object to a string. How the string is converted varies depending on the source value or object.JSON.stringify(value)
— converts the given value to a string in the JSON format and returns it.JSON.parse(jsonStr)
— parses (inspects) a given JSON string, constructing and returning the JavaScript value or object described by the string. If the given string is not valid JSON, aSyntaxError
will be thrown.
Lets Make a Leaderboard!
Making video games is one of the greatest joys of programming. And for good reason! Not only are they fun to play, but they are rewarding to create. Video games also can teach us important lessons about programming.
Suppose we had a game where the user is challenged to click a button as many times as they can in 5 seconds.
To encourage the competitive spirit of gaming, a key feature that we want to implement is a leaderboard to display the user's previous scores. Take a look at the speed-clicker-game
where we've implemented this game!
localStorage
Lets Us Store Data That Persists
localStorage
Lets Us Store Data That PersistsThe localStorage
API allows you to store data in a user's browser that will persist across browser sessions. That is, the data from a previous session can be retrieved even after they close their browser.
localStorage
is a globally available Object with the following methods:
localStorage.setItem(key, value)
— stores the given key:value pair inlocalStorage
. The value is converted to a string using the value's.toString()
methodlocalStorage.getItem(key)
— retrieves the value associated with the given key inlocalStorage
. It will always return a string.
You can play around with these functions in your browser's Console to get a sense of how they work:
Notice anything weird about the 'leaderboard'
array that we stored in localStorage
? It was converted to a string!
Let's try to add a new score to the leaderboard array:
JSON.stringify() and JSON.parse()
When using setItem
, the value is turned into a string before being stored. It does this using the given value's .toString()
method (every data type has this method).
As you can see, this is particularly annoying when dealing with Arrays and Objects. Try copying and pasting these statements into your browser Console:
When we want to store Arrays and Objects in localStorage
, we often use JSON.stringify()
to convert the string into a JSON-formatted string.
When we want to get the value out of localStorage
, we use JSON.parse()
which will create a JavaScript value / object from a JSON-formatted string.
In summary, when working with arrays and objects in localStorage
, any time we need to insert a value into localStorage
we will first stringify it with JSON.stringify
and any time we want to get a value from localStorage
, we will use JSON.parse()
:
For now, lets empty out the localStorage
:
Adding the leaderboard to LocalStorage
Let's try to use localStorage
to implement our leaderboard!
Previously, we had initialized leaderboard
as an empty array "in memory" and then whenever the game ended, we would push the current score into that array.
To use localStorage
, we'll follow a common pattern:
Check to see if there is a
leaderboard
value inlocalStorage
and store it in memory.If there isn't any previously-stored
leaderboard
, make a new array in memory.Modify the in-memory array with our new data.
Store the in-memory array in
localStorage
, replacing the old value.
Return to the game and test it out! It should work:
localStorage
Helpers
localStorage
HelpersHaving to remember stringify our values before adding them to localStorage
can be a pain. In addition, using JSON.parse
can sometimes result in an error if the string we are parsing is not in valid JSON format.
For example, JSON objects need to have double-quotations ("") around the keys (single quotations won't work!):
To ensure that we are always and consistently stringifying and parsing every time we interact with localStorage
, we can write these two helper functions:
In getLocalStorageKye
, we use a try/catch
block which is new syntax to us. It works by attempting to execute the code in the try {}
block knowing that it could potentially throw an error. Normally, a thrown error would crash the program. But with the catch (err) {}
block, we can "catch" the error and handle it without crashing the program.
If
JSON.parse
works, it will return the value.If it throws an error, the error will be printed (and not break everything) and
null
will be returned.
Now, we can safely use these functions instead of the localStorage
ones and know that all values will be properly stored and retrieved.
The two helper functions that we've ensure that our interactions with localStorage
won't throw any errors.
One way to think about these helper functions is that they form an API (application programming interface) for localStorage
. In other words, these functions allow the rest of the program to access localStorage
in a controlled manner.
We can take this even further to ensure that our application works in a consistent and predictable manner.
Data Layer: Creating an API for localStorage
localStorage
Suppose we had an application with a form that lets users add names to a list. Names are displayed on the screen in a list and clicking on the name will remove it from the list. This application can be found in 3-data-layer
.
Currently, it interacts with a names
array stored in localStorage
by directly using localStorage
methods, manually stringifying and parsing. This opens the door to errors and inconsistency.
In order to restrict the program so that it can only interact with localStorage
in the way we choose, we:
Isolate the logic for dealing with
localStorage
in its own file.Create functions for interacting with the
names
key inlocalStorage
.Only export the functions that indirectly interact with
localStorage
. The exported functions will form our API.
Note that not every function is exported.
setLocalStorageKey
andgetLocalStorageKey
directly interact withlocalStorage
and take care of stringifying, parsing, and error handling.The remaining functions are all exported. They form our
localStorage
API ("application programming interface"), giving the rest of our application the ability to interact withlocalStorage
, but only in the ways that we choose.
With this localStorage
API that we've created, we can greatly simplify our code!
Q: What makes this predictable and consistent?
That file acts as a data layer. We might also decide to isolate our DOM manipulation code and create a DOM layer or create an event handling layer.
Using localStorage
, we will build a data layer that is used to inform what is rendered, know as the view layer. When users interact with the view through the form, the data layer will be updated and we re-render the view.
This cycle of data > view > handle events > data looks like this:
Removing Values
The localStorage
API provides two methods for removing values. You can either remove a single value or clear all values from localStorage
:
localStorage.removeItem(key)
— removes the value associated with the given key fromlocalStorage
.localStorage.clear()
— removes all key:value pairs fromlocalStorage
Last updated