Budgtr
Build an express app that let's you create, and read
Learning Objectives
- Practice building an express app
- Practice making an index route
- Practice making a show route
- Practice making new/create routes
- Practice adding static assets (CSS)
Prerequisites
- Express Basics (Create/Read/Static Assets)
Objectives
You'll be creating an app that can let you
- see a list of your income and expenditures
- show you one income/expenditure item
- create a new income/expenditure item
Technical Requirements
- Must be able to run without syntax errors
- Must have index, show, create, new routes, using REST
- Must have basic MVC structure (more details below)
Getting Started - The Data
Add the data below to a file in your project called budget.js
.
You'll use this data to populate your index and show routes
module.exports = [
{
date: "April 1",
name: "Income",
from: "Old Glory Insurance",
amount: 1000,
tags: ["income", "yay"]
},
{
date: "April 1",
name: "Taxes",
from: "Government",
amount: -300,
tags: ["taxes"]
},
{
date: "April 1",
name: "Retirement",
from: "Country Bank",
amount: -200,
tags: ["retirement", "investing in the future"]
},
{
date: "April 1",
name: "Savings",
from: "Country Bank",
amount: -100,
tags: ["savings", "rainy day fund"]
},
{
date: "April 1",
name: "Credit Card Payment",
from: "NPM Express",
amount: -100,
tags: ["credit card"]
},
{
date: "April 5",
name: "Monthy Birthday Money from Aunt Tilda",
from: "Aunt Tilda",
amount: 20,
tags: ["Aunt Tilda is the best"]
},
{
date: "April 5",
name: "Coffee",
from: "Moon Coin",
amount: -4,
tags: ["coffee"]
},
{
date: "April 5",
name: "Internet",
from: "Horizon",
amount: -100,
tags: ["utilities"]
},
{
date: "April 3",
name: "Groceries",
from: "Merchant Jack's",
amount: -76,
tags: ["groceries"]
},
{
date: "April 3",
name: "Pet Food",
from: "Pet Precious Inc",
amount: -7,
tags: ["pets"]
},
];
Routes
-
Index
- GET
/budgets
- GET
-
Show
- GET
/budgets/:index
- GET
-
New
- GET
/budgets/new
- GET
-
Create
- POST
/budgets
- POST
MVC
Models, Views, Controller
We only have one model, so it may seem all these folders are overkill. As we start building with more complexity these folders will gain utility
- Your app should follow the MVC format
-
Models
budget.js
- the data we provided
-
Controllers
- since we just have one set of routes, we'll include them in our
server.js
stay tuned for lessons that'll teach us how to organize our code when we have more sets of routes
- since we just have one set of routes, we'll include them in our
-
Views
- your EJS files go in here
-
Public
- your css file(s) go(es) here
- recommended to go to try a new css framework:
http://getskeleton.com/ - download the
normalize.css
andskeleton.css
files and add them to your public directory (or use the CDNs:
https://cdnjs.cloudflare.com/ajax/libs/normalize/8.0.0/normalize.css
andhttps://cdnjs.cloudflare.com/ajax/libs/skeleton/2.0.4/skeleton.css
- thanks for the Tip Sam Whitleton!) -
link them in your ejs files:
<link rel="stylesheet" href="/normalize.css"> <link rel="stylesheet" href="/skeleton.css">
In order to immediately modernize your HTML
At the root of your project:
server.js
package.json
created withnpm init
.gitignore
- create it and addnode_modules
, you don't need it yet, but it is good practice to use this
File Structure
Commits
The order doesn't matter, but this will help you check your progress of completing this lan:
commit at least each time you get a route and/or view working
Getting Started
- server is working and displays a plain
res.send
index page ` with some text like 'hello world' - Added and configured npm packages
express
,ejs
- configured
express.static
-
created index.ejs
- html boiler plate
- link
normalize.css
andskeleton.css
- div with the class
container
h1
with the text Budgtr ALL (or similar text)
Expected appearance with normalize.css
and skeleton.css
are properly linked:
Index
- Set your
models/budget.js
data to a variable namedBudget
- Display your data on your index page as a
<table>
- each income/expenditure should be a
<tr>
- each piece of data should be its own table cell
<td>
- the
name
should be wrapped in an<a>
(anchor) tag, that goes nowhere for the moment, but will eventually link to the show page for that item
Expected Apperance:
Show
- A show route and
show.ejs
-
a link from
index.ejs
item name to its show page- html boilerplate
- remember to add your css links (copy from your
index.ejs
) - a
<div>
with the class ofcontainer
- an
<h1>
with the name of your item - a
<button>
that takes you back to theindex.ejs
page - the rest of the details of the item
Sample Apperance:
Hint: the button should already be styled if your normalize.css
and skeleton.css
are linked properly
Feel free to organize the rest of this page any way you like. Hold off on styling this more until the HFM section - just use the base styles provided by our already linked css files
New
- configured
body-parser
- Add a button in your
index.ejs
that links to a new route that displaysnew.ejs
- The
new.ejs
should contain - html boilerplate
- links to your css files
- a
div
with the class container - an
h1
with a descriptive title - a form, with the appropriate action and method
-
an input field for
- date
- name
- amount
- from
- submit
- tags (bonus)
- Styling forms are a pain, keep it simple for now
Hint: use the attribute placeholder
in the input field to see a placeholder value, rather than putting a label of the input field outside. Here is an example:
GOTCHA - make sure this get route is above your URL parameters route
New
- A new route that is a post route
- first just console.log the
req.body
- once the
req.body
is what you'd an expect (an object with keys that match our data in ourmodels/budget.js
and values that were entered in your form).push()
thereq.body
to yourBudget
- then redirect to the index
- when you redirect to your index page, your new item should appear
See new item at the bottom:
Show page should also render properly
Hints
Server.js
Innovation Time!
Remember, this is just JavaScript, so you can write as much JS logic as you want to make the app more useful beyond just showing the data as is. Be sure you solve this on your own.
-
Add a variable 'bankAccount'
- display it at the top of the index.ejs
- have this value update based on each item
- if the value is less than 0, change the background to red
- if the value is greater than 1000 change the background to blue or green
-
Tips
- pseudocode on your own, figure out YOUR way of solving it
- have a code graveyard
- talk it through with the TA
- link to helpful articles on stack overflow/elsewhere
- collaborate - have a friend help you solve it your way or help a friend solve it their way
Hungry for More
Click me for additional challenges
Feel free to choose what you want and if you want to implement something differently than the suggestion, go for it! It is hungry for more time!
- Go back to the afternoon lab and use cURL
- render the tags as list items in an undordered list
- input the tags, and add them to the tags array, figure out how to properly add multiple tags
- override the amount input so that it defaults to a negative number
- add logic to check whether the item is an expenditure or income (use two different fields? a check box? separate buttons? The choice is yours) and then input the amount as negative or positive based on the user's input
- style your app, add a
main.css
that adds your personal style on top of normalize and skeleton
SUPER BONUS
- try to store data in localStorage
- If you implemented tags, sort/filter your list by tags (no hints! There are many ways to solve this!)