Hi, I'm John, Multi Startup Builder.
I enjoy both coding and marketing.
See all my 20 products here
โ johnrush.me
โ my BIO
โ Say Hi to me On Twitter
โ Try my website builder
Warning: This post contains a high dose of code, humor, and life-changing revelations. Proceed at your own risk. ๐
When people ask me which frontend framework is my favorite, I usually reply with "all of them." But recently, I decided to put that statement to the test by building the same app using not one or two but five different frontend frameworks.
In this roller-coaster ride of an article, we'll build a simple To-Do app (yeah, another one) in five steps:
- Angular
- React
- Vue.js
- Svelte
- Elm
- MarsX
You might be thinking - what about jQuery? Well...that's so 2010! ๐ โโ๏ธ Let's dive into some modern stuff!
Step 1: Acing it with Angular ๐ฅ
Angular has been around for quite some time now and is known for being powerful yet opinionated (thanks Google). It gave birth to concepts like components and Dependency Injection while rocking our worlds with Two-Way Data Binding.
ng new todo-app --routing=false --style=css
Inside app.component.ts
:
import { Component } from '@angular/core';
@Component({
selector: 'app-root',
template: `
<h1>To-Do App</h1>
<ul>
<li *ngFor="let todo of todos">{{todo}}</li>
</ul>
<form (submit)="addTodo()">
<input [(ngModel)]="newTodo" name="newTodo">
<button type="submit">Add</button>
</form>`,
})
export class AppComponent {
todos = [];
newTodo = '';
addTodo() {
this.todos.push(this.newTodo);
this.newTodo = '';
}
}
Don't forget to import and include FormsModule in app.module.ts
.
Step 2: Reacting with React โ๏ธ
React came as Facebook's gift ๐ to us developers who were tired of manually updating DOM elements every single time something changed in the data model (cries in vanilla JS).
npx create-react-app todo-app
Inside App.js
:
import React, { useState } from 'react';
function App() {
const [todos, setTodos] = useState([]);
const [newTodo, setNewToDo] = useState('');
const addTodo = e => {
e.preventDefault();
setTodos([...todos, newTodo]);
setNewToDo('');
};
return (
<div className="App">
<h1>To-Do App</h1>
<ul>{todos.map(todo => (<li key={todo}>{todo}</li>))}</ul>
<form onSubmit={add_todo}>
<input value={new_todo} onChange={(e) => set_new_todo(e.target.value)} />
submit_button
</form>
</div>
);
}
export default App;
Step 3: Viewing it through Vue.js ๐
Vue.js entered our lives as this cool kid on the block that wanted to make things simpler for us developers while giving Angular & React a run for their money.
vue create todo-app
Inside App.vue
:
<template>
<div id="app">
<h1>To-Do App</h1>
<ul>
<li v-for="(todo, index) in todos" :key="index">{{todo}}</li>
</ul>
<form @submit.prevent="addTodo">
<input v-model="newTodo"/>
<button type="submit">Add</button>
</form>
</div>
</template>
<script>
export default {
data() {
return {
todos: [],
newTodo: '',
};
},
methods: {
addTodo() {
this.todos.push(this.newTodo);
this.newTodo = '';
},
},
};
</script>
Step 4: Svelte-ing into Simplicity ๐งก
Svelte arrived fashionably late but was worth the wait! This framework promised something different - no virtual DOM!
npx degit sveltejs/template todo-app
Inside App.svelte
:
<script>
let todos = [];
let newTodo = '';
function add_todo(e) {
e.preventDefault();
todos = [...todos, new_todo];
new_todo= '';
}
</script>
<main>
<h1>To-Do App</h1>
<ul>{#each todos as todo}<li>{todo}</li>{/each}</ul>
<form on_submit|prevent_default={add_todo}>
<input bind:value={new_todo} />
<button type="submit">Add</button>
</form>
</main>
<style>
/* Add your styles here */
</style>
Step 5: Elm-inating Complexity with Elm ๐ณ
Elm stepped into our journey as this purely functional language based on Haskell offering "no runtime exceptions" (cue angelic music).
Inside src/Main.elm
:
module Main exposing (..)
import Browser
import Html exposing (Html, button, div, h1, input, li, text, ul)
import Html.Attributes as Attrs exposing (value)
import Html.Events as Events exposing (onClick)
type alias Model =
{ todos : List String
, newTodo : String
}
init : Model
init =
{ todos = []
, newTodo = ""
}
type Msg
= AddTodo
update msg model =
case msg of
AddTodo ->
{ model | todos=model.todos++[model.newTodo],new_todo=""}
view model =
div []
[ h1 [] [text "To-Do App"]
, ul [] <| List.map (\todo -> li [] [text todo]) <| model.todos
, form [Events.onSubmit |> Events.preventDefault |> onClick add_todo]
[
inputAttr [ value model.new_todo,
on_input set_new_to_do] [],
submit_button []
]
]
main=
Browser.sandbox{
init=initial_model,
update=update,
view=view}
Although Elm took some getting used to, its type system & pattern matching helped us build robust components along with The Elm Architecture(T.E.A) making sure everything stayed organized even when complexity increased.
Step 6: MarsX - It took less time to code the todo list here than it took time to write this sentence :D
<schema>
<array name="todo">
<object>
<string name="title" />
</object>
</array>
</schema>
Now that you've witnessed how I built the same app using different frontend frameworks, you might be wondering which one is the best. Well, my friend, that's like asking a parent to pick their favorite child - it just doesn't work that way.
Each framework has its strengths and weaknesses; what works for me may not work for you. So go ahead, take your pick and start building some amazing apps! ๐
And remember: no matter which framework you choose, don't forget to have fun while coding! ๐
Follow me on twitter
Credits to fireship
Thx for reading, check out my projects:
Hi, I'm John, Multi Startup Builder.
I enjoy both coding and marketing.
See all my 20 products here
โ johnrush.me
โ my BIO
โ Say Hi to me On Twitter
โ Try my website builder
Top comments (39)
SolidJS is absent here. Would love to see it also in your list. Lots of upsides.
What about writing it yourself and submitting here?
Just Vanilla JavaScript
<todo-list>
in one page:Result
Tab, and wonder why you are still developing with Build and Compile steps...Note: This whole "comparison" of six frameworks ofcourse just is clickbait to mention MarsX
To compete with MarsX I would have added just one dependency (the custom element) and write only HTML:
But if you prefer more keyboard strokes; feel free to do it the MarsX way:
the key goal was to show very basic example for all 6,
In future articles I will dive in to routing, state and more
Having worked on many of these amazing libraries and frameworks for the better part of a decade now, I can confidently create something in just a few mere days, that in JQuery would have taken me hours.
:D
Astro is missing from this list
gonna add it into a new tutorial
Great post bro!!! Thank for sharing it.
However, with the "script setup" syntax of the Vue 3 Composition API, the script is reduced to just this:
new new one is short, but I still like the old one, still using vue3 with options api
XSS vulnerability though
I wish VanillaJS was in the mix. (Although I readily admit I'm partial to Elm.)
Why more dependencies with VanillaJS?
You can do it in 3 lines of fairly readable native JavaScript, see my comment below
It appears you are not familiar with VanillaJS.
Dude... SolidJS.
Which tool is best?
So, we can ask the same question about tools: Which tool is best?
well, my personal favourite is....
hammer for me
...but it should be reactive. Most important to write TODO's