Wednesday, 13 May 2020

Vue - Sorting arrays by multiple params

I'm working on this page where the user can search for videos from multiple sources.

In the data we have an array of all the videos, and a sortBy field:

data() {
return {
searchQuery: "",
videos: [],
sortBy: "",

When the user searches for something, the videos array gets populated with video data from an API, which returns an array of video objects like so:

   "source": 1,
   "id": "ImtZ5yENzgE",
   "channelName": "",
   "title": "Laravel PHP Framework Tutorial - Full Course for Beginners (2019)",
   "desc": "Learn Laravel 5.8 by creating an...",
   "imgURL": "",
   "likeCount": "11688",
   "duration": "04:25:05",
   "rawViewCount": 745908,
   "rawDate": "05/07/2019 12:39:04"

Once we have the array of videos, we can show them to the user by looping over the videos array in the template (here I'm using a video-card component that renders each video):

<section v-if="videos.length">
v-for="video in videos"

Now let's see how we can let the user sort the results by a few parameters: views, likes, and date published.

It's quite easy to implement with Vue 😀.

First let's create the dropdown for the user to choose the sorting method:

Sort by:
<option value="">Default</option>
<option value="rawViewCount">Views</option>
<option value="likeCount">Likes</option>
<option value="rawDate">Newest</option>

We bind it to the sortBy property with v-model, so when the user selects an option, the sortBy property will be the value of the selected option. Here is the trick - we give each option a value that is the property name of the video object we want to sort on.

Now in order to actually sort the videos base on the users selection, we need to create a new list of videos, that is computed based on the sorting method the user has selected. In Vue it can be done easily with a computed property:

computed: {
videosSorted() {
// if default sorting selected - return everything as is
if (!this.sortBy) return this.videos;
// sorting
return this.videos.sort((a, b) => b[this.sortBy] - a[this.sortBy]);

So this one sort function is going to sort the videos by any parameter the user selects 🎉 (then we just loop over this in the template). I think this is a very nice way of doing it...

Just in case you didn't grasp how it works: videosSorted returns the videos array sorted by the value of sortBy, which is going to be the name of the property of the video the user wants to sort.

Instead of writing 3 separate sort methods, write one. The trick is to use the exact property name as the value of the dropdown options.