diff --git a/.github/labeler.yml b/.github/labeler.yml
index f1f3fe17cc7..b0958c5aea8 100644
--- a/.github/labeler.yml
+++ b/.github/labeler.yml
@@ -31,5 +31,9 @@
'Content: Foundations':
- 'foundations/**/*'
+'Content: Markdownlint':
+ - 'markdownlint/**/*'
+ - '**/*.markdownlint-cli2.jsonc'
+
'Type: Chore':
- 'archive/**/*'
diff --git a/.github/workflows/testing.yml b/.github/workflows/codespell.yml
similarity index 94%
rename from .github/workflows/testing.yml
rename to .github/workflows/codespell.yml
index b3fc0b46c42..9289a3e13c3 100644
--- a/.github/workflows/testing.yml
+++ b/.github/workflows/codespell.yml
@@ -14,5 +14,4 @@ jobs:
check_filenames: true
check_hidden: true
skip: ./.git,*.png,*.csv,./archive,./legacy_submissions
- only_warn: 1
ignore_words_file: './.codespellignore'
diff --git a/.github/workflows/markdownlint-lessons.yml b/.github/workflows/markdownlint-lessons.yml
new file mode 100644
index 00000000000..604e456d619
--- /dev/null
+++ b/.github/workflows/markdownlint-lessons.yml
@@ -0,0 +1,34 @@
+name: MarkdownLint
+on:
+ pull_request:
+ paths:
+ - '**.md'
+ - '!./*.md'
+ - '!**/project*.md'
+ - '!./archive/**.md'
+ - '!./templates/**.md'
+ - '!./markdownlint/docs/**.md'
+
+jobs:
+ lesson_lint:
+ name: Lint lesson files
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v4
+ - uses: tj-actions/changed-files@v41
+ id: changed-files
+ with:
+ files: |
+ **.md
+ !./*.md
+ !README.md
+ !**/project*.md
+ !./archive/**.md
+ !./templates/**.md
+ !./markdownlint/docs/**.md
+ separator: ','
+ - uses: DavidAnson/markdownlint-cli2-action@v14
+ with:
+ config: './lesson.markdownlint-cli2.jsonc'
+ globs: ${{ steps.changed-files.outputs.all_changed_files }}
+ separator: ','
diff --git a/.github/workflows/markdownlint-projects.yml b/.github/workflows/markdownlint-projects.yml
new file mode 100644
index 00000000000..6093ca8e69d
--- /dev/null
+++ b/.github/workflows/markdownlint-projects.yml
@@ -0,0 +1,32 @@
+name: MarkDownLint
+on:
+ pull_request:
+ paths:
+ - '**/project*.md'
+ - '!./*.md'
+ - '!./archive/**.md'
+ - '!./templates/**.md'
+ - '!./markdownlint/docs/**.md'
+
+jobs:
+ project_lint:
+ name: Lint project files
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v4
+ - uses: tj-actions/changed-files@v41
+ id: changed-files
+ with:
+ files: |
+ **/project*.md
+ !./*.md
+ !README.md
+ !./archive/**.md
+ !./templates/**.md
+ !./markdownlint/docs/**.md
+ separator: ','
+ - uses: DavidAnson/markdownlint-cli2-action@v14
+ with:
+ config: './project.markdownlint-cli2.jsonc'
+ globs: ${{ steps.changed-files.outputs.all_changed_files }}
+ separator: ','
diff --git a/.github/workflows/markdownlint.yml b/.github/workflows/markdownlint.yml
deleted file mode 100644
index e95a2e2c860..00000000000
--- a/.github/workflows/markdownlint.yml
+++ /dev/null
@@ -1,49 +0,0 @@
-name: MarkdownLint
-on: pull_request
-
-jobs:
- project_lint:
- name: Lint project files
- runs-on: ubuntu-latest
- steps:
- - uses: actions/checkout@v3
- - uses: tj-actions/changed-files@v41
- id: changed-files
- with:
- files: |
- **/project*.md
- !./*.md
- !README.md
- !./archive/**.md
- !./templates/**.md
- !./markdownlint/docs/**.md
- separator: ','
- - uses: DavidAnson/markdownlint-cli2-action@v14
- if: steps.changed-files.outputs.any_changed == 'true'
- with:
- config: './project.markdownlint-cli2.jsonc'
- globs: ${{ steps.changed-files.outputs.all_changed_files }}
- separator: ','
- lesson_lint:
- name: Lint lesson files
- runs-on: ubuntu-latest
- steps:
- - uses: actions/checkout@v3
- - uses: tj-actions/changed-files@v41
- id: changed-files
- with:
- files: |
- **.md
- !./*.md
- !README.md
- !**/project*.md
- !./archive/**.md
- !./templates/**.md
- !./markdownlint/docs/**.md
- separator: ','
- - uses: DavidAnson/markdownlint-cli2-action@v14
- if: steps.changed-files.outputs.any_changed == 'true'
- with:
- config: './lesson.markdownlint-cli2.jsonc'
- globs: ${{ steps.changed-files.outputs.all_changed_files }}
- separator: ','
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index b662bf93126..85eee72bc65 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -31,4 +31,11 @@ To help enforce the layout specified in our layout style guide, we use [markdown
- Lint projects: `npm run lint:project -- "./path/to/project"`
- Autofix projects: `npm run fix:project -- "./path/to/project"`
-With either of these two methods, keep in mind that not all issues that get flagged will have an autofix available. Some rules require fixes that are more dependent on context and cannot - and should not - be automatically fixed, such as our custom rule `TOP001` for descriptive link text.
+> [!IMPORTANT]
+> You *must* run these npm scripts in the root of the curriculum repo (the same location as this file and `package.json`). If you navigate to a different directory and run a script on files there, it will not pick up the right configuration files and rules.
+
+> [!TIP]
+> In some cases, you may need to run a fix script more than once to catch and fix all fixable errors. This typically occurs when a line has multiple errors affecting the same parts and fix actions collide, so Markdownlint only applies some of the fixes.
+
+> [!NOTE]
+>With either of these two methods, keep in mind that not all issues that get flagged will have an autofix available. Some rules require fixes that are more dependent on context and cannot - and should not - be automatically fixed, such as our custom rule `TOP001` for descriptive link text.
diff --git a/README.md b/README.md
index 9c61741316f..bb58f57138b 100644
--- a/README.md
+++ b/README.md
@@ -13,15 +13,17 @@ Our community can be found on the [TOP Discord server](https://discord.gg/fbFCkY
The Odin Project depends on open-source contributions to improve, grow, and thrive. We welcome contributors of all experience levels and backgrounds to help maintain this awesome curriculum and community. If you would like to contribute to our curriculum, be sure to thoroughly read our [contributing guide](https://github.com/TheOdinProject/.github/blob/main/CONTRIBUTING.md).
Some of the things you can do to contribute to our curriculum include:
-* Correct typos and other grammar errors.
-* Rewrite parts of existing lessons to make them clearer and easier to understand.
-* Fix broken links.
-* Add new resource links you think would make a lesson better.
-* Work on entirely new lessons after getting approval.
+
+- Correct typos and other grammar errors.
+- Rewrite parts of existing lessons to make them clearer and easier to understand.
+- Fix broken links.
+- Add new resource links you think would make a lesson better.
+- Work on entirely new lessons after getting approval.
**Happy Coding!**
-\* See [license.md](https://github.com/TheOdinProject/curriculum/blob/main/license.md) for usage details.
+See [license.md](https://github.com/TheOdinProject/curriculum/blob/main/license.md) for usage details.
___
-Created by [Erik Trautman](http://www.github.com/eriktrautman)
\ No newline at end of file
+
+Created by [Erik Trautman](http://www.github.com/eriktrautman).
diff --git a/advanced_html_css/accessibility/the_web_content_accessibility_guidelines_wcag.md b/advanced_html_css/accessibility/the_web_content_accessibility_guidelines_wcag.md
index f253d7ca1a8..edf80a0e754 100644
--- a/advanced_html_css/accessibility/the_web_content_accessibility_guidelines_wcag.md
+++ b/advanced_html_css/accessibility/the_web_content_accessibility_guidelines_wcag.md
@@ -48,8 +48,10 @@ The second thing to keep in mind is that just taking those first few steps towar
### Assignment
+
1. Read through the [WCAG Overview](https://www.w3.org/WAI/standards-guidelines/wcag/) page. Don't worry about all of the other links for now. The goal of reading this page is to understand more of the included overview and to get familiar with the site itself for when you *do* need to visit other pages on it.
-2. Skim through [WebAIM's WCAG 2 Checklist](https://webaim.org/standards/wcag/checklist), keeping in mind to read the important disclaimer. For now the goal is just to get an idea of common accessibility issues, some of which you'll be ready to fix by the time you finish this set of lessons, rather than read through every issue listed on the page. Keep this resource bookmarked, though, as using it as your checklist when you start actively implementing accessibility will be really handy.
+1. Skim through [WebAIM's WCAG 2 Checklist](https://webaim.org/standards/wcag/checklist), keeping in mind to read the important disclaimer. For now the goal is just to get an idea of common accessibility issues, some of which you'll be ready to fix by the time you finish this set of lessons, rather than read through every issue listed on the page. Keep this resource bookmarked, though, as using it as your checklist when you start actively implementing accessibility will be really handy.
+
### Knowledge check
@@ -63,4 +65,4 @@ The following questions are an opportunity to reflect on key topics in this less
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- It looks like this lesson doesn't have any additional resources yet. Help us expand this section by contributing to our curriculum.
+- [The A11Y Project](https://www.a11yproject.com/): An open-source resource that provides information on how to make your web content more accessible. The A11Y Project includes checklists, guidelines, and tools that can help you implement WCAG standards effectively.
diff --git a/advanced_html_css/animation/transforms.md b/advanced_html_css/animation/transforms.md
index 39ff1a5f34c..76e79c8d29f 100644
--- a/advanced_html_css/animation/transforms.md
+++ b/advanced_html_css/animation/transforms.md
@@ -157,7 +157,9 @@ on CodePen.
-If you guessed correctly, congratulations! But this is a tricky concept. There is a bit of debate on how to read a chain of transform functions. According to [MDN's transform docs](https://developer.mozilla.org/en-US/docs/Web/CSS/transform#values): "The transform functions are multiplied in order from left to right, meaning that composite transforms are effectively applied in order from right to left."
+If you guessed correctly, congratulations! But this is a tricky concept. MDN's transform docs state that "[composite transforms are effectively applied in order from right to left](https://developer.mozilla.org/en-US/docs/Web/CSS/transform#values)".
+
+The blue box rotates 45 degrees on the spot, then translates on the X axis by 200%, moving it directly to the right. The red box translates by 200% first, so moves to the right, but the transform origin is still where it used to be. Therefore, it rotates 45 degrees around that original point, making the red box "swing down" to end up diagonally from where it started.
While you can generally chain multiple transforms in any order for various results, there is one exception: `perspective`. This brings us nicely to the next section where `perspective` is involved.
@@ -269,7 +271,7 @@ Another benefit of `transform` is that it can be hardware-accelerated via a devi
-1. Take a look at this [MDN demonstration of `rotate3d`](https://developer.mozilla.org/en-US/docs/Web/CSS/transform-function/rotate3d()) then read more about the property in this [Quackit article on `rotate3d`](https://www.quackit.com/css/functions/css_rotate3d_function.cfm).
+1. Take a look at this [MDN demonstration of `rotate3d`](https://developer.mozilla.org/en-US/docs/Web/CSS/transform-function/rotate3d()) then read more about the property in this [Quackit article on `rotate3d`](https://www.qhmit.com/css/functions/css_rotate3d_function.cfm).
1. Learn more about [the `perspective` property on CSS Tricks](https://css-tricks.com/how-css-perspective-works/).
1. MDN has another great [demonstration using `translate3d`](https://developer.mozilla.org/en-US/docs/Web/CSS/transform-function/translate3d()).
1. Go through [The World of CSS Transforms](https://www.joshwcomeau.com/css/transforms/) by Josh Comeau.
diff --git a/advanced_html_css/animation/transitions.md b/advanced_html_css/animation/transitions.md
index 2da8649998a..7c303fe37ca 100644
--- a/advanced_html_css/animation/transitions.md
+++ b/advanced_html_css/animation/transitions.md
@@ -24,7 +24,7 @@ CSS transitions let you animate a change from an element's initial state to an e
-When your mouse cursor is away from the button, the button is in the initial state. When you hover over it you introduce the end state, the hover state, causing the transition of the color smoothly fading from white to black to occur.
+When your mouse cursor is away from the button, the button is in the initial state. When you hover over it, you introduce the end state - the hover state - causing the background color to transition smoothly from white to black.
This was achieved using the `transition` property, which is actually a shorthand property for `transition-property`, `transition-duration`, `transition-timing-function` and `transition-delay`.
diff --git a/nodeJS/authentication/security_configuration.md b/archive/nodeJS/authentication/security_configuration.md
similarity index 100%
rename from nodeJS/authentication/security_configuration.md
rename to archive/nodeJS/authentication/security_configuration.md
diff --git a/nodeJS/express_and_mongoose/deployment.md b/archive/nodeJS/express_and_mongoose/deployment.md
similarity index 100%
rename from nodeJS/express_and_mongoose/deployment.md
rename to archive/nodeJS/express_and_mongoose/deployment.md
diff --git a/nodeJS/express_and_mongoose/express_101.md b/archive/nodeJS/express_and_mongoose/express_101.md
similarity index 100%
rename from nodeJS/express_and_mongoose/express_101.md
rename to archive/nodeJS/express_and_mongoose/express_101.md
diff --git a/nodeJS/express_and_mongoose/express_102_crud_and_mvc.md b/archive/nodeJS/express_and_mongoose/express_102_crud_and_mvc.md
similarity index 100%
rename from nodeJS/express_and_mongoose/express_102_crud_and_mvc.md
rename to archive/nodeJS/express_and_mongoose/express_102_crud_and_mvc.md
diff --git a/nodeJS/express_and_mongoose/express_103_routes_and_controllers.md b/archive/nodeJS/express_and_mongoose/express_103_routes_and_controllers.md
similarity index 100%
rename from nodeJS/express_and_mongoose/express_103_routes_and_controllers.md
rename to archive/nodeJS/express_and_mongoose/express_103_routes_and_controllers.md
diff --git a/nodeJS/express_and_mongoose/express_104_view_templates.md b/archive/nodeJS/express_and_mongoose/express_104_view_templates.md
similarity index 100%
rename from nodeJS/express_and_mongoose/express_104_view_templates.md
rename to archive/nodeJS/express_and_mongoose/express_104_view_templates.md
diff --git a/nodeJS/express_and_mongoose/introduction_to_express.md b/archive/nodeJS/express_and_mongoose/introduction_to_express.md
similarity index 100%
rename from nodeJS/express_and_mongoose/introduction_to_express.md
rename to archive/nodeJS/express_and_mongoose/introduction_to_express.md
diff --git a/nodeJS/express_and_mongoose/project_express_105_forms_and_deployment.md b/archive/nodeJS/express_and_mongoose/project_express_105_forms_and_deployment.md
similarity index 100%
rename from nodeJS/express_and_mongoose/project_express_105_forms_and_deployment.md
rename to archive/nodeJS/express_and_mongoose/project_express_105_forms_and_deployment.md
diff --git a/nodeJS/express_and_mongoose/project_inventory_application.md b/archive/nodeJS/express_and_mongoose/project_inventory_application.md
similarity index 100%
rename from nodeJS/express_and_mongoose/project_inventory_application.md
rename to archive/nodeJS/express_and_mongoose/project_inventory_application.md
diff --git a/nodeJS/express_and_mongoose/project_mini_message_board.md b/archive/nodeJS/express_and_mongoose/project_mini_message_board.md
similarity index 100%
rename from nodeJS/express_and_mongoose/project_mini_message_board.md
rename to archive/nodeJS/express_and_mongoose/project_mini_message_board.md
diff --git a/nodeJS/mongoDB/intro_to_mongoDB.md b/archive/nodeJS/mongoDB/intro_to_mongoDB.md
similarity index 100%
rename from nodeJS/mongoDB/intro_to_mongoDB.md
rename to archive/nodeJS/mongoDB/intro_to_mongoDB.md
diff --git a/databases/databases/databases.md b/databases/databases/databases.md
index 94da2159c73..838462511df 100644
--- a/databases/databases/databases.md
+++ b/databases/databases/databases.md
@@ -1,6 +1,6 @@
### Introduction
-We've talked about the client-side and the server-side but how do we keep ahold of all our user's data? Who remembers that your login password is `CatLover1985` so you can sign into the website? The bottom layer of any web application is the database and it handles all the remembering for you (we'll cover caching much later). It can be relatively simple, like an Excel spreadsheet, or incredibly complex and split into many giant pieces, like Facebook's.
+You might have wondered how you keep track of all your users' data. Who remembers that your login password is `CatLover1985` so you can sign into the website? The bottom layer of any web application is the database and it handles all the remembering for you (we'll cover caching much later). It can be relatively simple, like an Excel spreadsheet, or incredibly complex and split into many giant pieces, like Facebook's.
Databases are kind of hidden in the back of the web application, so people approach them with a sense of suspicion and awe. However, do not fret or feel intimidated. As you dive deeper into the subject, your databases and you are going to become very good friends (or at least frenemies). Rest assured, by the end of this curriculum, you're going to understand what's going on with your databases and be able to interact with them like a pro (and probably better than some people you'll work with). This lesson is a teaser for that.
@@ -10,41 +10,42 @@ Compared to a normal programming language like you've already learned, SQL (Stru
This section contains a general overview of topics that you will learn in this lesson.
-- What a database is.
-- What relational databases are.
-- In what way relational databases are different from XML.
-- What SQL is.
-- What SQL is used for.
-- How to get all the records from a table in SQL.
-- How to insert a record in SQL.
+- What a database is.
+- What relational databases are.
+- In what way relational databases are different from XML.
+- What SQL is.
+- What SQL is used for.
+- How to get all the records from a table in SQL.
+- How to insert a record in SQL.
### Assignment
- 1. Check out this [introduction](https://launchschool.com/books/sql/read/introduction) of how SQL can be used to organise and manage an overwhelming amount of data.
- - You do not need to go any further than the first page on introductions.
+ 1. Check out this introduction of [how SQL can be used to organise and manage an overwhelming amount of data](https://launchschool.com/books/sql/read/introduction).
+ - You do not need to go any further than the first page on introductions.
- 2. Watch this [short video introduction to relational databases](http://www.youtube.com/watch?v=z2kbsG8zsLM) to get a feel for why this stuff is useful and some more exposure to the terminology we'll use.
- 3. Go through this [Khan Academy tutorial](https://www.khanacademy.org/computing/hour-of-code/hour-of-sql/v/welcome-to-sql), to get a feel for actually creating and manipulating databases.
+ 1. Watch this [short video introduction to relational databases](http://www.youtube.com/watch?v=z2kbsG8zsLM) to get a feel for why this stuff is useful and some more exposure to the terminology we'll use.
+ 1. Go through this [Khan Academy SQL tutorial](https://www.khanacademy.org/computing/hour-of-code/hour-of-sql/v/welcome-to-sql) to get a feel for actually creating and manipulating databases.
### Knowledge check
-This section contains questions for you to check your understanding of this lesson on your own. If you’re having trouble answering a question, click it and review the material it links to.
-- [What is a database?](https://launchschool.com/books/sql/read/introduction#structureddata)
-- [What are relational databases?](https://launchschool.com/books/sql/read/introduction#rdbms)
-- [What is a Primary Key?](https://youtu.be/z2kbsG8zsLM?t=200)
-- [What is SQL?](https://launchschool.com/books/sql/read/introduction#sql)
-- [How do you get all the records from a table in SQL?](https://www.khanacademy.org/computing/hour-of-code/hour-of-code-lessons/hour-of-sql/pt/querying-the-table)
-- [How do you insert a record in SQL?](https://www.khanacademy.org/computing/hour-of-code/hour-of-code-lessons/hour-of-sql/pt/creating-a-table-and-inserting-data)
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
+
+- [What is a database?](https://launchschool.com/books/sql/read/introduction#structureddata)
+- [What are relational databases?](https://launchschool.com/books/sql/read/introduction#rdbms)
+- [What is a Primary Key?](https://youtu.be/z2kbsG8zsLM?t=200)
+- [What is SQL?](https://launchschool.com/books/sql/read/introduction#sql)
+- [How do you get all the records from a table in SQL?](https://www.khanacademy.org/computing/hour-of-code/hour-of-code-lessons/hour-of-sql/pt/querying-the-table)
+- [How do you insert a record in SQL?](https://www.khanacademy.org/computing/hour-of-code/hour-of-code-lessons/hour-of-sql/pt/creating-a-table-and-inserting-data)
### Additional resources
-This section contains helpful links to related content. It isn’t required, so consider it supplemental.
-- [What is a Relational Database?](http://computer.howstuffworks.com/question599.htm) from HowStuffWorks.com
-- A brief [Simple Wiki article describing relational databases](http://simple.wikipedia.org/wiki/Relational_database)
-- Hunter Ducharme created [an e-book](https://hunter-ducharme.gitbook.io/sql-basics/) which is a great documentation on how to do all the basics in SQL.
-- David J. Malan's excellent SQL lecture and resources in [Harvard's CS50x](https://cs50.harvard.edu/x/2023/weeks/7/).
-- Relational databases aren't the only way to store data. Non-relational (aka NoSQL) databases have emerged over the last couple decades. Check out [this article](https://circleci.com/blog/SQL-vs-NoSQL-databases/) to learn the difference between SQL and NoSQL.
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
+
+- [What is a Relational Database?](http://computer.howstuffworks.com/question599.htm) from HowStuffWorks.com
+- A brief [Simple Wiki article describing relational databases](http://simple.wikipedia.org/wiki/Relational_database)
+- [David J. Malan’s excellent SQL lecture](https://cs50.harvard.edu/x/2024/weeks/7/) and resources in Harvard’s CS50x.
+- Relational databases aren't the only way to store data. Non-relational (aka NoSQL) databases have emerged over the last couple decades. Check out this article to learn the [difference between SQL and NoSQL](https://circleci.com/blog/SQL-vs-NoSQL-databases/).
diff --git a/databases/databases/databases_and_sql.md b/databases/databases/databases_and_sql.md
index 861d15f8c50..38875e67cc4 100644
--- a/databases/databases/databases_and_sql.md
+++ b/databases/databases/databases_and_sql.md
@@ -1,6 +1,6 @@
### Introduction
-Data is the core of any good web app and a good working knowledge of SQL will take you a long way. That knowledge lets you not just understand what's going on behind the scenes with your ORM tool (e.g. Active Record) but also to feel comfortable asking more complicated questions of your data. And that's really what SQL is all about -- asking questions of your database and occasionally also adding or changing things in it. Querying can be incredibly useful for you.
+Data is the core of any good web app and a good working knowledge of SQL will take you a long way. That knowledge lets you not just understand what's going on behind the scenes with your ORM tool (e.g. Active Record in Rails, or Prisma in NodeJS) but also to feel comfortable asking more complicated questions of your data. And that's really what SQL is all about -- asking questions of your database and occasionally also adding or changing things in it. Querying can be incredibly useful for you.
In more straightforward cases, you might want to display all users who signed up in December via the promotion code "FREESTUFF". You might want to display all comments created by the current user and sorted by topic and creation date. In more complex cases, you may want to show a list of all the orders shipped to states with more than 1,000 users by quantity and total order value. Or, for internal reasons, you might ask marketing analysis questions like which promotion channels produce users who meet your specified engagement criteria of reading 5 articles per work week.
@@ -10,8 +10,7 @@ You will start with the questions like the ones above and then have to figure ou
We'll move beyond just the `SELECT "users".* FROM "users" LIMIT 1` queries and into more dynamic topics like joining tables together, performing calculations on the results, and grouping results together in new ways.
-All this stuff is being used by Rails behind the scenes so understanding it will make you much better at writing queries in Rails. This is why we're going over databases before learning Rails.
-
+
#### A note on resources
SQL is one of those topics that's been stored away in dusty old technical manuals and 90's style websites. Even the best books out there can make it seem oddly complicated because they tend to write for the database engineer who actually does need to know all the nitty gritty details.
@@ -53,7 +52,7 @@ Once your database is set up and you've got empty tables to work with, you use S
Every CRUDdy command in SQL contains a few parts -- the action ("statement"), the table it should run on, and the conditions ("clauses"). If you just do an action on a table without specifying conditions, it will apply to the whole database and you'll probably break something.
-For "Destroy" queries, the classic mistake is typing `DELETE FROM users` without a `WHERE` clause, which removes all your users from the table. You probably needed to delete just one user, who you would specify based on some (hopefully unique) attribute like "name" or "id" as part of your condition clause, e.g. `DELETE FROM users WHERE users.id = 1`. You can do all kinds of common sense things like using `>`, `<`, `<=` etc. comparison operators to specify groups of rows to run commands on or logical operators like `AND`, `OR`, `NOT` etc to chain multiple clauses together, e.g. `DELETE FROM users WHERE id > 12 AND name = 'foo'`.
+For "Destroy" queries, the classic mistake is typing `DELETE FROM users` without a `WHERE` clause, which removes all your users from the table. You probably needed to delete just one user, who you would specify based on some (hopefully unique) attribute like "name" or "id" as part of your condition clause, e.g. `DELETE FROM users WHERE users.id = 1`. You can do all kinds of common sense things, such as using comparison operators (`>`, `<`, `<=` etc.) to specify groups of rows to run commands on, or logical operators (`AND`, `OR`, `NOT` etc.) to chain multiple clauses together, e.g. `DELETE FROM users WHERE id > 12 AND name = 'foo'`.
"Create" queries use `INSERT INTO` and you'll need to specify which columns to insert stuff into and then which values to put in those columns, which looks something like `INSERT INTO users (name, email) VALUES ('foobar','foo@bar.com');`. This is one of the few queries that you don't need to be careful about which rows you've selected since you're actually just adding new ones into the table.
@@ -65,7 +64,7 @@ For "Destroy" queries, the classic mistake is typing `DELETE FROM users` without
WHERE email='foo@bar.com';
```
-"Read" queries, which use `SELECT`, are the most common, e.g. `SELECT * FROM users WHERE created_at < '2013-12-11 15:35:59 -0800'`. The `*` you see just says "all the columns". Specify a column using both the table name and the column name. You can get away with just the column name for queries of one table, but as soon as there are more than one table involved, SQL will yell at you so just always specify the table name: `SELECT users.id, users.name FROM users`.
+"Read" queries, which use `SELECT`, are the most common, e.g. `SELECT * FROM users WHERE created_at < '2013-12-11 15:35:59 -0800'`. The `*` you see just says "all the columns". Specify a column using both the table name and the column name. You can get away with just the column name for queries of one table, but as soon as there is more than one table involved, SQL will yell at you so just always specify the table name: `SELECT users.id, users.name FROM users`.
A close cousin of `SELECT`, for if you only want unique values of a column, is `SELECT DISTINCT`. Say you want a list of all the different names of your users without any duplicates... try `SELECT DISTINCT users.name FROM users`.
@@ -80,9 +79,9 @@ The "left" table is the original table (the one that the `FROM` clause was `ON`)
1. `INNER JOIN`, aka `JOIN` -- Your best friend and 95% of what you'll use. Keeps only the rows from both tables where they match up. If you asked for all the posts for all users (`SELECT * FROM users JOIN posts ON users.id = posts.user_id`), it would return only the users who have actually written posts and only posts which have specified their author in the `user_id` column. If an author has written multiple posts, there will be multiple rows returned (but the columns containing the user data will just be repeated).
-2. `LEFT OUTER JOIN` -- keep all the rows from the left table and add on any rows from the right table which match up to the left table's. Set any empty cells this produces to `NULL`. E.g. return all the users whether they have written posts or not. If they do have posts, list those posts as above. If not, set the columns we asked for from the "posts" table to `NULL`.
-3. `RIGHT OUTER JOIN` -- the opposite... keep all rows in the right table.
-4. `FULL OUTER JOIN` -- Keep all rows from all tables, even if there are mismatches between them. Set any mismatched cells to `NULL`.
+1. `LEFT OUTER JOIN` -- keep all the rows from the left table and add on any rows from the right table which match up to the left table's. Set any empty cells this produces to `NULL`. E.g. return all the users whether they have written posts or not. If they do have posts, list those posts as above. If not, set the columns we asked for from the "posts" table to `NULL`.
+1. `RIGHT OUTER JOIN` -- the opposite... keep all rows in the right table.
+1. `FULL OUTER JOIN` -- Keep all rows from all tables, even if there are mismatches between them. Set any mismatched cells to `NULL`.
Joins naturally let you specify conditions too, like if you only want the posts from a specific user: `SELECT * FROM users JOIN posts ON users.id = posts.user_id WHERE users.id = 42`.
@@ -105,7 +104,7 @@ Now we're getting into the fun stuff. Aggregate functions like `COUNT` which re
GROUP BY users.id;
```
-See [W3 Schools' article](http://www.w3schools.com/sql/trysql.asp?filename=trysql_select_groupby) and play around with the SQL in the window (try deleting the `GROUP BY` line) for an interactive visual.
+See [W3Schools' browser-based SQL playground](http://www.w3schools.com/sql/trysql.asp?filename=trysql_select_groupby) for an interactive visual.
The last nifty trick is if you want to only display a subset of your data. In a normal situation, you'd use a `WHERE` clause to narrow it down. But if you've used an aggregate function like `COUNT` (say to get the count of posts written for each user in the example above), `WHERE` won't work anymore. So to conditionally retrieve records based on aggregate functions, you use the `HAVING` function, which is essentially the `WHERE` for aggregates. So say you only want to display users who have written more than 10 posts:
@@ -117,35 +116,37 @@ The last nifty trick is if you want to only display a subset of your data. In a
HAVING posts_written >= 10;
```
-Try going back to [the W3 Schools' example](http://www.w3schools.com/sql/trysql.asp?filename=trysql_select_groupby) and joining the `Customers` and the `Orders` tables to get the number of orders in each country and adding the line `HAVING COUNT(*) > 10;` after `GROUP BY` (and delete the extra semicolon in the previous line).
+Try going back to [W3Schools' browser-based SQL playground](http://www.w3schools.com/sql/trysql.asp?filename=trysql_select_groupby) and joining the `Customers` and the `Orders` tables to get the number of orders in each country and adding the line `HAVING COUNT(*) > 10;` after `GROUP BY` (and delete the extra semicolon in the previous line).
You probably got lost somewhere in the above explanation and that's just fine... it's covering way more stuff than anyone can pick up in 10 minutes. The assigned reading will do a better job of explaining things but, more importantly, you'll get plenty of opportunities to solidify your understanding by applying it in the project. If you've still got blind spots, check out the Additional Resources section below. Fear not and stick with it!
-### SQL is faster than Ruby!
-
-Learning this stuff is particularly relevant because it's MUCH faster for you to build queries that use SQL intelligently than to just grab a whole bunch of data out of your database and then use Ruby to process it. For instance, if you want all the unique names of your users, you COULD just grab the whole list from your database using SQL like `SELECT users.name FROM users` (which Active Record will do for you with `User.select(:name)`) then remove duplicates using Ruby's `#uniq` method, e.g. `User.select(:name).uniq`... but that requires you to pull all that data out of your database and then put it into memory and then iterate through it using Ruby. Use `SELECT DISTINCT users.name FROM users` instead to have SQL do it all in one step.
+### SQL is faster than your code!
-SQL is built to be fast. It has a special query optimizer which takes a look at the whole query you're about to run and it figures out exactly which tables it needs to join together and how it can most quickly execute the query. The difference between using `SELECT` and `SELECT DISTINCT` is negligible compared to the time cost of doing it in Ruby. Learning your SQL will help you write Active Record queries that can do more which will make your app much faster.
+Learning this stuff is particularly relevant because it's MUCH faster for you to build queries that use SQL intelligently than to just grab a whole bunch of data out of your database and then use a programming language (like Ruby or JavaScript) to process it. For instance, if you want all the unique names of your users, you COULD just grab the whole list from your database using SQL like `SELECT users.name FROM users` then use a JavaScript/Ruby method to remove duplicates... but that requires you to pull all that data out of your database and then put it into memory and then iterate through it in your code. Use `SELECT DISTINCT users.name FROM users` instead to have SQL do it all in one step.
+SQL is built to be fast. It has a special query optimizer which takes a look at the whole query you're about to run and it figures out exactly which tables it needs to join together and how it can most quickly execute the query. The difference between using `SELECT` and `SELECT DISTINCT` is negligible compared to the time cost of doing it yourself. Learning your SQL will help you write better queries that can do more which will make your app much faster.
### Assignment
- 1. Go through this interactive SQL tutorial from [SQL Teaching](https://www.sqlteaching.com/)
- 2. Go through this more in-depth interactive SQL tutorial from [SQL Bolt](http://sqlbolt.com)
- 3. Go through the basics at [Part 1](https://www.sqlcourse.com/beginner-course/) and the advanced at [Part 2](https://www.sqlcourse.com/advanced-course/) of SQL Course
+
+ 1. Go through this [interactive SQL tutorial from SQL Teaching](https://www.sqlteaching.com/).
+ 1. Go through this more in-depth [interactive SQL tutorial from SQL Bolt](http://sqlbolt.com/).
+ 1. Go through [SQLCourse's beginner course](https://www.sqlcourse.com/beginner-course/) then [SQLCourse's advanced course](https://www.sqlcourse.com/advanced-course/).
+
### Conclusion
-SQL can be a tricky set of concepts to wrap your head around, particularly when it comes to conditionally displaying and grouping the results of multiple joins. We've emphasized that this stuff is useful for understanding what's going on behind the scenes with Rails and you'll get a chance to apply it in the project. Everything up to vanilla joins and vanilla aggregate functions is core knowledge that you should really make an effort to assimilate.
+SQL can be a tricky set of concepts to wrap your head around, particularly when it comes to conditionally displaying and grouping the results of multiple joins. Everything up to vanilla joins and vanilla aggregate functions is core knowledge that you should really make an effort to assimilate.
If you never quite get to the point where you're comfortable with the really advanced concepts, you'll luckily not need to use them except in a small fraction of situations in your future. It's good to learn it all up front but you'll probably find yourself Googling for how to perform certain advanced queries when the time comes anyway.
-The next step, once you've had a chance to practice this all in the project, is to apply it to Rails with Active Record. You'll quickly find that Active Record makes your life much, much, much better. Just don't forget about ol' SQL when you've moved onto those better and brighter things, okay?
+The next step, once you've had a chance to practice this all in the project, is to apply it to your codebase in the upcoming courses. You'll quickly find that using ORM tools makes your life much, much, much better. Just don't forget about ol' SQL when you've moved onto those better and brighter things, okay?
### Knowledge check
-This section contains questions for you to check your understanding of this lesson on your own. If you’re having trouble answering a question, click it and review the material it links to.
+
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
- [What is the difference between a foreign key and a primary key?](#foreign-key)
- [Where is the setup information for your database stored?](#schema)
@@ -154,13 +155,14 @@ This section contains questions for you to check your understanding of this less
- [Which `JOIN` statement keeps only the rows from both tables where they match up?](#inner-join)
- [How do you use an aggregate function?](#aggregate-function)
- [In which situation would you use the `HAVING` function?](#having-function)
-- [Why can't I just use Ruby to process my database data?](#sql-is-faster-than-ruby)
+- [Why can't I just use code to process my database data?](#sql-is-faster-than-your-code)
### Additional resources
-This section contains helpful links to related content. It isn’t required, so consider it supplemental.
-- Odinite Hunter D made his excellent notes into a [Git Book on SQL](https://hunter-ducharme.gitbook.io/sql-basics) which you should totally check out if you want a decent resource.
-- [SQL "tutorial" from tutorialspoint](http://www.tutorialspoint.com/sql/index.htm)... doesn't really give much guidance, but can be a useful reference for the language.
-- [A Beginners Guide to SQL](http://www.sohamkamani.com/blog/2016/07/07/a-beginners-guide-to-sql/) by Soham Kamani.
-- [SQL Flashcards](https://flashcards.github.io/sql/introduction.html) by flashcards.github.io.
-- If you feel like doing more SQL exercises, make sure to check out [SQL Exercises](http://www.sql-ex.com/).
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
+
+- Odinite Hunter D made his excellent notes into a [Git Book on SQL](https://hunter-ducharme.gitbook.io/sql-basics) which you should totally check out if you want a decent resource.
+- [SQL "tutorial" from tutorialspoint](http://www.tutorialspoint.com/sql/index.htm)... doesn't really give much guidance, but can be a useful reference for the language.
+- [A Beginners Guide to SQL](http://www.sohamkamani.com/blog/2016/07/07/a-beginners-guide-to-sql/) by Soham Kamani.
+- [SQL Flashcards](https://flashcards.github.io/sql/introduction.html) by flashcards.github.io.
+- If you feel like doing more SQL exercises, make sure to check out [SQL Exercises](http://www.sql-ex.com/).
diff --git a/databases/databases/project_sql_zoo.md b/databases/databases/project_sql_zoo.md
index 84ac7035120..984dc6095ba 100644
--- a/databases/databases/project_sql_zoo.md
+++ b/databases/databases/project_sql_zoo.md
@@ -5,16 +5,18 @@ SQL Zoo is one of the few resources online that actually lets you build and run
### Assignment
- 1. Go to [SQL Zoo](https://sqlzoo.net/) and do Tutorials 0-9 listed under the "Tutorial Section" and the quizzes listed at the end of each. The first is the ["Select" tutorial](https://sqlzoo.net/wiki/SELECT_basics).
- - Make sure the dropdown on the upper right of the main page for "Engine" says "MySQL" (the default). Large results will be cut off and not all rows or columns shown, so the "answers" may not look 100% correct.
- 2. Before you move on, we would like your feedback - [please fill this form](https://docs.google.com/forms/d/e/1FAIpQLSenvMG6WFbOOEap_biQOwqfbH-j-xsf5Eyv4ir2Rx5FsYSecQ/viewform?usp=sf_link). Getting user (you) feedback is important so we can continue to improve the curriculum and get an idea of your experience.
+
+ 1. Go to [SQL Zoo](https://sqlzoo.net/) and do Tutorials 0-9 listed under the "Tutorial Section" and the quizzes listed at the end of each. The first tutorial is called "SELECT basics".
+ - Make sure the dropdown on the upper right of the main page for "Engine" says "MySQL" (the default). Large results will be cut off and not all rows or columns shown, so the "answers" may not look 100% correct.
+ 1. Before you move on, we would like your feedback - please fill out this [feedback form for the SQL course](https://docs.google.com/forms/d/e/1FAIpQLSenvMG6WFbOOEap_biQOwqfbH-j-xsf5Eyv4ir2Rx5FsYSecQ/viewform?usp=sf_link). Getting user (you) feedback is important so we can continue to improve the curriculum and get an idea of your experience.
+
### Additional resources
-This section contains helpful links to related content. It isn’t required, so consider it supplemental.
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- [Explanation of SELF JOIN by Caleb Curry](https://www.youtube.com/watch?v=W0p8KP0o8g4). He covers many other topics on his [Database Design](https://www.youtube.com/watch?v=e7Pr1VgPK4w&list=PL_c9BZzLwBRK0Pc28IdvPQizD2mJlgoID) series.
-- [Khan Academy's SQL Course](https://www.khanacademy.org/computing/computer-programming/sql)
-- [OverAPI's SQL Cheat Sheet](http://overapi.com/mysql)
-- [GalaXQL](http://sol.gfxile.net/galaxql.html), an interactive space-themed SQL tutorial.
+- [Explanation of SELF JOIN by Caleb Curry](https://www.youtube.com/watch?v=W0p8KP0o8g4). He covers many other topics on his [Database Design](https://www.youtube.com/watch?v=e7Pr1VgPK4w&list=PL_c9BZzLwBRK0Pc28IdvPQizD2mJlgoID) series.
+- [Khan Academy's SQL Course](https://www.khanacademy.org/computing/computer-programming/sql)
+- [OverAPI's SQL Cheat Sheet](http://overapi.com/mysql)
+- [GalaXQL](http://sol.gfxile.net/galaxql.html), an interactive space-themed SQL tutorial.
diff --git a/foundations/html_css/css-foundations/block-and-inline.md b/foundations/html_css/css-foundations/block-and-inline.md
index 06cb116c509..16e698b2be3 100644
--- a/foundations/html_css/css-foundations/block-and-inline.md
+++ b/foundations/html_css/css-foundations/block-and-inline.md
@@ -83,6 +83,6 @@ The following questions are an opportunity to reflect on key topics in this less
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- [Learn CSS Layout](https://learnlayout.com/no-layout.html) is tutorial that is a little dated at this point, but its examples are clear. The first 6 slides cover the material we've seen so far.
+- [Learn CSS Layout](https://learnlayout.com/no-layout.html) is a tutorial that is a little dated at this point, but its examples are clear. The first 6 slides cover the material we've seen so far.
- Watch this short video on [what the term “Normal Flow” means](https://www.youtube.com/watch?v=nfXRw06FgK8) in CSS.
- For a more interactive explanation and example, try this [Scrim on block and inline display](https://scrimba.com/scrim/co5024997a7e46c232d9abe55).
diff --git a/foundations/html_css/css-foundations/inspecting-html-and-css.md b/foundations/html_css/css-foundations/inspecting-html-and-css.md
index 17e4d25be2b..3ca2ee3fb9b 100644
--- a/foundations/html_css/css-foundations/inspecting-html-and-css.md
+++ b/foundations/html_css/css-foundations/inspecting-html-and-css.md
@@ -1,6 +1,6 @@
### Introduction
-Being able to inspect and debug your HTML and CSS is critical to frontend development. This lesson will take us through the Chrome Dev Tools, which allow you to see detailed information about your elements and CSS rules, as well as assist you in finding and fixing problems in your code.
+Being able to inspect and debug your HTML and CSS is critical for frontend development. This lesson will take us through the Chrome Dev Tools, which allow you to see detailed information about your elements and CSS rules, as well as assist you in finding and fixing problems in your code.
### Lesson overview
diff --git a/foundations/html_css/css-foundations/intro-to-css.md b/foundations/html_css/css-foundations/intro-to-css.md
index fee21686b2d..46ba2b6377f 100644
--- a/foundations/html_css/css-foundations/intro-to-css.md
+++ b/foundations/html_css/css-foundations/intro-to-css.md
@@ -283,7 +283,7 @@ Now that we've learned some basic syntax, you might be wondering *how* to add al
#### External CSS
-External CSS is the most common method you will come across, and it involves creating a separate file for the CSS and linking it inside of an HTML's opening and closing `` tags with a self-closing `` element:
+External CSS is the most common method you will come across, and it involves creating a separate file for the CSS and linking it inside of an HTML's opening and closing `` tags with a void `` element:
```html
@@ -306,7 +306,7 @@ p {
}
```
-First, we add a self-closing `` element inside of the opening and closing `` tags of the HTML file. The `href` attribute is the location of the CSS file, either an absolute URL or, what you'll be utilizing, a URL relative to the location of the HTML file. In our example above, we are assuming both files are located in the same directory. The `rel` attribute is required, and it specifies the relationship between the HTML file and the linked file.
+First, we add a void `` element inside of the opening and closing `` tags of the HTML file. The `href` attribute is the location of the CSS file, either an absolute URL or, what you'll be utilizing, a URL relative to the location of the HTML file. In our example above, we are assuming both files are located in the same directory. The `rel` attribute is required, and it specifies the relationship between the HTML file and the linked file.
Then inside of the newly created `styles.css` file, we have the selector (the `div` and `p`), followed by a pair of opening and closing curly braces, which create a "declaration block". Finally, we place any declarations inside of the declaration block. `color: white;` is one declaration, with `color` being the property and `white` being the value, and `background-color: black;` is another declaration.
diff --git a/foundations/html_css/flexbox/flexbox-intro.md b/foundations/html_css/flexbox/flexbox-intro.md
index 2e2f7a483c1..e4add2b7ac1 100644
--- a/foundations/html_css/flexbox/flexbox-intro.md
+++ b/foundations/html_css/flexbox/flexbox-intro.md
@@ -86,5 +86,5 @@ The following questions are an opportunity to reflect on key topics in this less
This section contains helpful links to related content. It isn't required, so consider it supplemental.
- Interneting Is Hard has a tutorial on [modern CSS layouts with flexbox](https://internetingishard.netlify.app/html-and-css/flexbox/index.html).
-
+- Slaying the dragon tutorial on [Flexbox in 8 minutes](https://youtu.be/phWxA89Dy94?si=UOXlsTa0BMfQYG3q).
- For a more interactive explanation and example, try this [Scrim on Flexbox](https://scrimba.com/learn/flexbox/your-first-flexbox-layout-flexbox-tutorial-canLGCw). Note that this Scrim requires logging into Scrimba in order to view.
diff --git a/foundations/html_css/flexbox/project-landing-page.md b/foundations/html_css/flexbox/project-landing-page.md
index 7435dd04e79..28e709058e5 100644
--- a/foundations/html_css/flexbox/project-landing-page.md
+++ b/foundations/html_css/flexbox/project-landing-page.md
@@ -4,11 +4,11 @@ For this project you'll be creating an entire web page from a design we'll provi
The design we're providing you comes in the form of 2 images: one is an image of the complete website, and one has some details about some of the fonts and colors we've used.
-Do _not_ be afraid to use google or go back to previous lessons to look something up. **In real life, professional developers use google _constantly_ for things that they have been doing for years.** At this point it is not expected that you will have everything memorized, so don't worry about it. Additionally, there are a few small details that you may not have encountered in our lessons yet. _This is by design._ These details are minor, and easily searched (e.g. google `css rounded corners`).
+Do *not* be afraid to use google or go back to previous lessons to look something up. **In real life, professional developers use google *constantly* for things that they have been doing for years.** At this point it is not expected that you will have everything memorized, so don't worry about it. Additionally, there are a few small details that you may not have encountered in our lessons yet. *This is by design.* These details are minor, and easily searched (e.g. google `css rounded corners`).
Get your project as close as you can to the design, but do not worry about getting it pixel-perfect. Don't get out your ruler or count pixels to find the exact margins between the various sections. The point of this assignment is to create something from scratch and get the various elements in more or less the right position relative to the rest. It doesn't matter if you use `margin: 24px` when the design actually has `margin: 48px`.
-_Finally_, feel free to substitute your own content into this design. The images have some meaningless dummy content, but if you want to make up a business and personalize this page, please feel free to do so! Insert actual images in the placeholders, and feel free to play with the colors and fonts a bit too.
+*Finally*, feel free to substitute your own content into this design. The images have some meaningless dummy content, but if you want to make up a business and personalize this page, please feel free to do so! Insert actual images in the placeholders, and feel free to play with the colors and fonts a bit too.
@@ -37,21 +37,23 @@ We know it may be tempting to look at the solutions when you get stuck or don't
Once you have completed your project, feel free to look at the other solutions to see alternative ways the project was completed.
-**DO NOT PEEK. Come ask for help in our [Discord server!](https://discord.com/channels/505093832157691914/516751477306294273) _(You need to [join the Discord server](https://discord.gg/fbFCkYabZB) first in order to see the channel)._**
+**DO NOT PEEK. Come ask for help in our [Discord server!](https://discord.com/channels/505093832157691914/516751477306294273) *(You need to [join the Discord server](https://discord.gg/fbFCkYabZB) first in order to see the channel).***
### Assignment
-Don't forget to commit early & often! You can [reference the Commit Message lesson here](https://www.theodinproject.com/paths/foundations/courses/foundations/lessons/commit-messages)!
+
+Don't forget to commit early & often! You can [reference the Commit Message lesson](https://www.theodinproject.com/paths/foundations/courses/foundations/lessons/commit-messages)!
1. Download the design images and take a look at what you're going to be creating here. [Image One (Full Design)](https://cdn.statically.io/gh/TheOdinProject/curriculum/81a5d553f4073e593d23a6ab00d50eef8620796d/foundations/html_css/project/imgs/01.png), [Image Two (Color and Fonts)](https://cdn.statically.io/gh/TheOdinProject/curriculum/81a5d553f4073e593d23a6ab00d50eef8620796d/foundations/html_css/project/imgs/02.png)
- 1. The font that's being used in the images is roboto.
- 2. Hero text is the statement that appears at the top of a web page.
-2. There are many ways to tackle a project like this, and it can be overwhelming to look at a blank HTML document and not know where to start. Our suggestion: take it one section at a time. The website you're creating has 4 main sections (and a footer), so pick one and get it into pretty good shape before moving on. Starting at the top is always a solid plan.
-3. For the section you're working on, begin by getting all the content onto the page before beginning to style it. In other words, do the HTML and _then_ do the CSS. You'll probably have to go back to the HTML once you start styling, but bouncing back and forth from the beginning will take more time and may cause more frustration. (Note: you don't need to use more than one stylesheet. Using only one CSS file is adequate for this project).
-4. Many of the elements on this page are very similar to things you saw in our flexbox exercises... feel free to go back to those if you need a refresher.
-5. Do not worry about making your project look nice on a mobile device. We'll learn that later.
-6. When you finish, don't forget to push it up to GitHub!
+ 1. The font that's being used in the images is `Roboto`.
+ 1. Hero text is the statement that appears at the top of a web page.
+1. There are many ways to tackle a project like this, and it can be overwhelming to look at a blank HTML document and not know where to start. Our suggestion: take it one section at a time. The website you're creating has 4 main sections (and a footer), so pick one and get it into pretty good shape before moving on. Starting at the top is always a solid plan.
+1. For the section you're working on, begin by getting all the content onto the page before beginning to style it. In other words, do the HTML and *then* do the CSS. You'll probably have to go back to the HTML once you start styling, but bouncing back and forth from the beginning will take more time and may cause more frustration. (Note: you don't need to use more than one stylesheet. Using only one CSS file is adequate for this project).
+1. Many of the elements on this page are very similar to things you saw in our flexbox exercises... feel free to go back to those if you need a refresher.
+1. Do not worry about making your project look nice on a mobile device. We'll learn that later.
+1. When you finish, don't forget to push it up to GitHub!
+
### Viewing your project on the web
@@ -66,5 +68,5 @@ There are a couple of ways to go about publishing your project, but the simplest
- Go to your GitHub repo on the web and click the **Settings** button as shown in the screenshot below.
![Screenshot pointing to the Settings located in an example repository](https://cdn.statically.io/gh/TheOdinProject/curriculum/81a5d553f4073e593d23a6ab00d50eef8620796d/foundations/html_css/project/imgs/03.png)
- Click on **Pages** on the left side bar.
-- Change the **Branch** from _none_ to _main branch_ and click Save.
+- Change the **Branch** from `None` to `main` and click Save.
- It may take a few minutes (the GitHub website says up to 10), but your project should be accessible over the web from `your-github-username.github.io/your-github-repo-name` (obviously substituting your own details in the link).
diff --git a/foundations/html_css/html-foundations/elements-and-tags.md b/foundations/html_css/html-foundations/elements-and-tags.md
index f1f58f60028..6a09a7d53ad 100644
--- a/foundations/html_css/html-foundations/elements-and-tags.md
+++ b/foundations/html_css/html-foundations/elements-and-tags.md
@@ -6,7 +6,7 @@ HTML (HyperText Markup Language) defines the structure and content of webpages.
This section contains a general overview of topics that you will learn in this lesson.
-- Explain what HTML Tags are.
+- Explain what HTML tags are.
- Explain what HTML elements are.
### Elements and tags
@@ -60,4 +60,4 @@ The following questions are an opportunity to reflect on key topics in this less
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- [Don't Fear the Internet's video about HTML](http://www.dontfeartheinternet.com/02-html)
+- [Don't Fear the Internet's video about HTML](https://player.vimeo.com/video/24549728)
diff --git a/foundations/html_css/html-foundations/html-boilerplate.md b/foundations/html_css/html-foundations/html-boilerplate.md
index 197a0a65c16..551848cf532 100644
--- a/foundations/html_css/html-foundations/html-boilerplate.md
+++ b/foundations/html_css/html-foundations/html-boilerplate.md
@@ -114,8 +114,13 @@ To complete the boilerplate, add a `` element to the `index.html` file. Th
The HTML boilerplate in the `index.html` file is complete at this point, but how do you view it in the browser? There are a couple of different options:
-> A note:
-> In order to avoid branching our lesson's instructions to accommodate for all of the differences between browsers, we are going to be using Google Chrome as our primary browser for the remainder of this course. All references to the browser will pertain specifically to Google Chrome. We **strongly** suggest that you use Google Chrome for all of your testing going forward.
+
+
+#### Use Google Chrome
+
+In order to avoid branching our lesson's instructions to accommodate for all of the differences between browsers, we are going to be using Google Chrome as our primary browser for the remainder of this course. All references to the browser will pertain specifically to Google Chrome. We **strongly** suggest that you use Google Chrome for all of your testing going forward.
+
+
1. You can drag and drop an HTML file from your text editor into the address bar of your browser.
@@ -168,7 +173,7 @@ It's still good to know how to write the boilerplate yourself in case you find y
1. Build some muscle memory by deleting the contents of the `index.html` file and trying to write out all the boilerplate again from memory. Don't worry if you have to peek at the lesson content the first few times if you get stuck. Just keep going until you can do it a couple of times from memory.
-1. Run your boilerplate through the W3 [HTML validator](https://validator.w3.org/). Validators ensure your markup is correct and are an excellent learning tool, as they provide feedback on syntax errors you may be making often and aren't aware of, such as missing closing tags and extra spaces in your HTML.
+1. Run your boilerplate through the W3 [HTML validator](https://validator.w3.org/#validate_by_input). Validators ensure your markup is correct and are an excellent learning tool, as they provide feedback on syntax errors you may be making often and aren't aware of, such as missing closing tags and extra spaces in your HTML.
diff --git a/foundations/html_css/html-foundations/links-and-images.md b/foundations/html_css/html-foundations/links-and-images.md
index 452406a7e1d..1d0cabc60a0 100644
--- a/foundations/html_css/html-foundations/links-and-images.md
+++ b/foundations/html_css/html-foundations/links-and-images.md
@@ -62,7 +62,9 @@ While `href` specifies the destination link, `target` specifies where the linked
You may have noticed that we snuck in the `rel` attribute above. This attribute is used to describe the relation between the current page and the linked document.
-The `noopener` value prevents the opened link from gaining access to the webpage from which it was opened. The `noreferrer` value prevents the opened link from knowing which webpage or resource has a link (or 'reference') to it. It also includes the `noopener` behaviour and thus can be used by itself as well.
+The `noopener` value prevents the opened link from gaining access to the webpage from which it was opened.
+
+The `noreferrer` value prevents the opened link from knowing which webpage or resource has a link (or 'reference') to it. The `noreferrer` value also includes the `noopener` behaviour and thus can be used by itself as well.
Why do we need this added behaviour for opening links in new tabs? Security reasons. The prevention of access that is caused by `noopener` prevents [phishing attacks](https://www.ibm.com/topics/phishing) where the opened link may change the original webpage to a different one to trick users. This is referred to as [tabnabbing](https://owasp.org/www-community/attacks/Reverse_Tabnabbing). Adding the `noreferrer` value can be done if you wish to not let the opened link know that your webpage links to it.
@@ -156,7 +158,7 @@ Think of your domain name (`town.com`) as a town, the directory in which your we
Websites would be fairly boring if they could only display text. Luckily HTML provides a wide variety of elements for displaying all sorts of different media. The most widely used of these is the image element.
-To display an image in HTML we use the `` element. Unlike the other elements we have encountered, the `` element is self-closing. Empty, self-closing HTML elements do not need a closing tag.
+To display an image in HTML we use the `` element. Unlike the other elements we have encountered, the `` element is a void element. As we have seen earlier in the course, void elements do not need a closing tag because they are naturally empty and do not contain any content.
Instead of wrapping content with an opening and closing tag, it embeds an image into the page using a src attribute which tells the browser where the image file is located. The src attribute works much like the href attribute for anchor tags. It can embed an image using both absolute and relative paths.
@@ -172,7 +174,7 @@ For example, using an absolute path we can display an image located on The Odin
-To use images that we have on our own websites, we can use a relative path.
+To display images on your website that are hosted on your own web server, you can use a relative path.
@@ -322,3 +324,4 @@ This section contains helpful links to related content. It isn't required, so co
- [Interneting is hard's treatment on HTML links and images](https://internetingishard.netlify.app/html-and-css/links-and-images)
- [What happened the day Google decided links including (`/`) were malware](https://www.itpro.co.uk/609724/google-apologises-after-blacklisting-entire-internet)
- [Chris Coyier's When to use target="_blank" on CSS-Tricks](https://css-tricks.com/use-target_blank/)
+- If you're looking to deepen your understanding of the various image formats used on the web, [the following article which is titled: Which is the Best Image Format for Your Website?](https://imagekit.io/blog/best-image-format-for-web/) from imagekit.io is a great resource. It offers a detailed comparison of JPEG, PNG, GIF, and WebP formats, helping you choose the right one for your needs. Note that the article doesn't cover SVG, but it's still an excellent guide for the other common formats.
diff --git a/foundations/html_css/html-foundations/lists.md b/foundations/html_css/html-foundations/lists.md
index bae11bc8fdb..d5a218384de 100644
--- a/foundations/html_css/html-foundations/lists.md
+++ b/foundations/html_css/html-foundations/lists.md
@@ -20,10 +20,13 @@ Unordered lists are created using the `
` element, and e
Each list item in an unordered list begins with a bullet point:
+
### Ordered lists
@@ -33,15 +36,19 @@ If you instead want to create a list of items where the order *does* matter, lik
Ordered lists are created using the `` element. Each individual item in them is again created using the list item element `
`. However, each list item in an ordered list begins with a number instead:
+
To get some practice using lists, create a new HTML document and create the following lists:
1. An unordered shopping list of your favorite foods
@@ -53,7 +60,7 @@ To get some practice using lists, create a new HTML document and create the foll
### Knowledge check
-This section contains questions for you to check your understanding of this lesson on your own. If you’re having trouble answering a question, click it and review the material it links to.
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
- [What HTML element is used to create an unordered list?](#unordered-lists)
- [What HTML element is used to create an ordered list?](#ordered-lists)
@@ -61,7 +68,7 @@ This section contains questions for you to check your understanding of this less
### Additional resources
-This section contains helpful links to related content. It isn’t required, so consider it supplemental.
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
- [MDN documentation on the unordered list element](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/ul)
- [MDN documentation on the ordered list element](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/ol)
diff --git a/foundations/html_css/html-foundations/working-with-text.md b/foundations/html_css/html-foundations/working-with-text.md
index 97e11b1833a..78a6c5b4411 100644
--- a/foundations/html_css/html-foundations/working-with-text.md
+++ b/foundations/html_css/html-foundations/working-with-text.md
@@ -167,7 +167,7 @@ For example, the two paragraphs in the following code are siblings, since they a
-We use indentation to make the level of nesting clear and readable for ourselves and other developers who will work with our HTML in the future. It is recommended to indent any child elements by two spaces.
+We use indentation to make the level of nesting clear and readable for ourselves and other developers who will work with our HTML in the future. In our examples, we have indented any child elements by two spaces per nesting level.
The parent, child, and sibling relationships between elements will become much more important later when we start styling our HTML with CSS and adding behavior with JavaScript. For now, however, it is just important to know the distinction between how elements are related and the terminology used to describe their relationships.
diff --git a/foundations/installations/command_line_basics.md b/foundations/installations/command_line_basics.md
index a82642adb6b..b9a812ccab0 100644
--- a/foundations/installations/command_line_basics.md
+++ b/foundations/installations/command_line_basics.md
@@ -29,7 +29,7 @@ The window that opens will be mostly blank, with the exception of some text that
It returns your username. Cool!
-Often, guides and instructions for using the terminal will indicate commands by putting the symbol first, like `$ whoami`. This tells you to type the command in your terminal, but don't enter the `$`. And remember, if you are using a newer Mac, the `%` is the same as `$`
+Often, guides and instructions for using the terminal will indicate commands by putting the symbol first, like `$ whoami`. This tells you to type the command in your terminal, but don't enter the `$`. And remember, if you are using a newer Mac, the `%` is the same as `$`.
#### Why learn this now?
@@ -108,7 +108,7 @@ Many of these resources assume you're using a Mac or Linux environment. If you d
1. Create a new directory in your home directory with the name `test`.
1. Navigate to the `test` directory.
- 1. Create a new file called `test.txt`. *Hint: use the `touch` or `echo` command.*
+ 1. Create a new file called `test.txt`. *Hint: use the `touch` command.*
1. Open your newly created file in VSCode, make some changes, save the file, and close it.
1. Navigate back out of the `test` directory.
1. Delete the `test` directory.
diff --git a/foundations/installations/installation_overview.md b/foundations/installations/installation_overview.md
index 5ecd27e87ea..c7b9b0c028f 100644
--- a/foundations/installations/installation_overview.md
+++ b/foundations/installations/installation_overview.md
@@ -37,7 +37,7 @@ If you're using a Mac, you're in great shape. The Odin Project instructions assu
#### Windows
-Windows, by itself, **is not natively supported** by The Odin Project, or on our Discord server. Because many of the tools you'll be using were written with a Linux environment in mind, you'll need to have one even if you plan to use Windows as your development OS. If you are currently using Windows you can use one of the following options to create your development environment:
+Windows, by itself, **is not natively supported** by The Odin Project, or on our Discord server. Because many of the tools you'll be using were written with a Linux environment in mind, you'll need to have one even if you plan to use Windows as your development OS. If you are currently using Windows, you can use one of the following options to create your development environment (we provide full step-by-step instructions for these options in the next lesson):
- A VirtualBox Virtual Machine
- Dual-boot Ubuntu installation
@@ -71,7 +71,7 @@ Many learners come to our Discord channel to ask if the directions on this page
Before we can continue, we must first stress one detail of importance:
-**We can only support what is provided within the scope of our curriculum. We do not support native Windows as a development environment.** Using Windows has been discussed many times and it is not feasible to do so at this time. Please do not ask us to support Windows, and **do not bring it up in the Discord**. We are constantly evaluating our curriculum to keep content as fresh and accessible as possible, and Windows has not proven to be a path of low resistance.
+**We can only support what is provided within the scope of our curriculum. We do not support native Windows as a development environment.** Using Windows has been discussed many times and it is not feasible to do so at this time. Please do not ask us to support Windows, and **do not bring it up in the Discord**. We are constantly evaluating our curriculum to keep content as fresh and accessible as possible, and Windows has not proven to be a path of low resistance. For more information on The Odin Project and Windows, we have a [list of reasons why Windows is not a supported OS in The Odin Project](https://github.com/TheOdinProject/blog/wiki/Why-We-Do-Not-Support-Windows).
With that out of the way, we need to set up an appropriate development environment!
diff --git a/foundations/installations/installations.md b/foundations/installations/installations.md
index b9aac380b75..b8e51be5ec6 100644
--- a/foundations/installations/installations.md
+++ b/foundations/installations/installations.md
@@ -1,12 +1,16 @@
-
+
+
### Introduction
If you are already using **MacOS**, **Ubuntu**, or [an official flavor of Ubuntu](https://wiki.ubuntu.com/UbuntuFlavors) as your operating system and have **Google Chrome** as an installed browser, you can skip this lesson. Otherwise, click on the small arrow to the left of the method you would like to use below to expand that section, and then follow the installation instructions.
+
#### Be mindful of the OS you are using
+
We can only support the operating systems indicated above. Our instructions have been tested with MacOS, Ubuntu, and official flavors of Ubuntu. We do not recommend installing an OS that is only based on Ubuntu (like Mint, Pop!_OS, ElementaryOS, etc).
+
### Lesson overview
@@ -36,7 +40,9 @@ This curriculum only supports using a laptop, desktop or supported Chromebook. W
Pick your method of installation below:
+
Virtual Machine (Recommended)
+
Installing a Virtual Machine (VM) is the easiest and most reliable way to get started creating an environment for web development. A VM is an entire computer emulation that runs inside your current Operating System (OS), like Windows. The main drawback of a VM is that it can be slow because you’re essentially running two computers at the same time. We’ll do a few things to improve its performance.
@@ -61,7 +67,7 @@ There are thousands of distributions of Linux out there, but Xubuntu is undoubte
#### Step 2.1: Install VirtualBox
-Installing VirtualBox is very straightforward. It doesn’t require much technical knowledge and is the same process as installing any other program on your Windows computer. Double clicking the downloaded VirtualBox file will start the installation process. If you receive an error about needing Microsoft Visual C++ 2019 Redistributable Package, you can find it on [official Microsoft Learn page](https://learn.microsoft.com/en-us/cpp/windows/latest-supported-vc-redist?view=msvc-170#visual-studio-2015-2017-2019-and-2022). You most likely want the version with `X64` Architecture (that means 64-bit) - download and install it then try installing VirtualBox again.
+Installing VirtualBox is very straightforward. It doesn’t require much technical knowledge and is the same process as installing any other program on your Windows computer. Double clicking the downloaded VirtualBox file will start the installation process. If you receive an error about needing Microsoft Visual C++ 2019 Redistributable Package, you can find it on the [official Microsoft Learn page](https://learn.microsoft.com/en-us/cpp/windows/latest-supported-vc-redist?view=msvc-170#visual-studio-2015-2017-2019-and-2022). You most likely want the version with `X64` Architecture (that means 64-bit) - download and install it then try installing VirtualBox again.
During the installation, you’ll be presented with various options. We suggest dropping the Python Support as you don't need it by clicking on the drive icon with an arrow and choosing **Entire feature will be unavailable**:
@@ -177,6 +183,7 @@ When using a command in the terminal that requires you to enter your password fo
This is a security feature to protect confidential information, like how password fields on websites use asterisks or dots. By not displaying the characters you write, the terminal keeps your password secure.
You can still enter your password as normal and press Enter to submit it.
+
```bash
@@ -190,10 +197,16 @@ After `sudo apt upgrade` runs for a while you will be asked whether you want to
### Step 4: Understand your new VM
+#### Step 4.1: Close TOP in the Windows browser and open the website in your VM
+
+From now on, stick with The Odin Project (TOP) website within your VM and follow the Linux instructions provided. The curriculum will often require you to copy and paste code between the lesson and your coding space, as well as the terminal. This won't work smoothly if you switch between the VM and Windows because they are entirely separate environments.
+
+**Remember:** For the remainder of the TOP curriculum, refer to the Linux instructions only.
+
Here are some tips to help you get started in a virtual environment:
- Enable the toolbar in your VM settings - there are useful options there that you might want to play around with, especially the ones concerning full screen or multiple displays. To do so, click on **Settings** and then navigate to **User Interface** and finally tick **Show at Top of Screen**.
- ![It's a good idea to look around the settings overall to get a feel of what's possible.](https://cdn.statically.io/gh/TheOdinProject/curriculum/96d534641514fe4d62aabe2919fac3c52cb286e7/foundations/installations/installations/imgs/16_toolbar.png)
+ ![It's a good idea to look around the settings overall to get a feel of what's possible.](https://cdn.statically.io/gh/TheOdinProject/curriculum/96d534641514fe4d62aabe2919fac3c52cb286e7/foundations/installations/installations/imgs/16_toolbar.png)
- All your work should happen in the VM. You will install everything you need for coding, including your text editor, language environments and various tools inside the VM. The Xubuntu installation inside of your VM also comes with a web browser pre-installed but we'll be installing Chrome shortly.
- To install software on your VM, you will follow the Linux (Ubuntu) installation instructions from inside the Xubuntu VM.
- You might need to take screenshots when asking for help on our Discord, here's how depending on where you use it:
@@ -242,7 +255,9 @@ To be safe, click the **Send the shutdown signal** radio and hit OK. This will s
+
Ubuntu/Windows Dual-Boot
+
### Read this entire section before starting
@@ -286,7 +301,9 @@ If you encounter an error requesting you to disable **Intel RST** while attempti
+
ChromeOS/ChromeOS Flex
+
With the recent addition of being able to run a Linux terminal, the ChromeOS platform has been opened up to the ability to install native Linux applications. If you wish to use your Chromebook to complete The Odin Project, you will need to ensure you meet a couple requirements:
@@ -301,6 +318,7 @@ Once you have successfully met both of these requirements, you should be able to
+
WSL2 (Advanced)
Using WSL2 is an quick and easy way to get started with using Linux, allowing you to run a Linux distribution from within Windows. WSL2 is available on Windows 10 version 2004 and higher (Build 19041 and higher) and Windows 11.
@@ -308,8 +326,11 @@ Using WSL2 is an quick and easy way to get started with using Linux, allowing yo
To make it clear: you are going to be using a different OS, this is not a way to avoid using Linux. Due to how WSL2 is integrated with Windows it often causes significant confusion to new learners. Use the Virtual Machine if you want a clear separation between your Windows and Linux so the curriculum is easier to follow.
+
#### WSL2 and Linux instructions
+
Because WSL2 is a full-fledged Linux distribution, almost everything that the curriculum teaches about Linux is also applicable to WSL2. In future lessons, whenever there are instructions that differ by OS, you should follow the Linux instructions, unless the lesson includes WSL2-specific instructions.
+
### Step 1: Installations
@@ -352,12 +373,17 @@ On Windows there are three primary ways to open WSL2.
- If you search for Ubuntu in the application search bar you should see a application titled Ubuntu; open it to start a new terminal session.
+
You might notice when you open WSL2 via Windows Terminal, you'll see a window with a different color scheme and a different icon compared to opening a terminal through Ubuntu in your applications. This is because Windows Terminal comes with a default color scheme for Ubuntu meant to emulate how a real Ubuntu terminal looks. This difference is purely cosmetic, and there is no practical difference between the two.
+
+
When opening your WSL2 terminal ensure that you do not see `/mnt/c` at the start of the line. `/mnt/c` is where your Windows installation lives when working within WSL2, and messing around there can have unintended consequences.
+
+
### Google Chrome installation
@@ -370,6 +396,7 @@ Look at this [usage share of web browsers](https://en.wikipedia.org/wiki/Usage_s
Choose your Operating System:
+
Linux
#### Step 1: Download Google Chrome
@@ -382,17 +409,20 @@ wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb
```
-### Copy and paste keyboard shortcuts
-You have probably noticed that the common keyboard shortcut: Ctrl + V to paste something doesn't work in the terminal. In order to paste your text input into your terminal you can use: Ctrl + Shift + V keyboard shortcut combination, instead. It is also very handy to know that the: Ctrl + Shift + C keyboard combination will copy any highlighted text from your terminal, which can then be pasted later.
+
+#### Copy and paste keyboard shortcuts
+
+You have probably noticed that the common keyboard shortcut: Ctrl + V to paste something doesn't work in the terminal. In order to paste your text input into your terminal you can use: Ctrl + Shift + V keyboard shortcut combination, instead. It is also very handy to know that the: Ctrl + Shift + C keyboard combination will copy any highlighted text from your terminal, which can then be pasted later.
+
#### Step 2: Install Google Chrome
- Enter the following command in your terminal to install **Google Chrome** `.deb` package
- ```bash
- sudo apt install ./google-chrome-stable_current_amd64.deb
- ```
+ ```bash
+ sudo apt install ./google-chrome-stable_current_amd64.deb
+ ```
- Enter your password, if needed
@@ -400,11 +430,12 @@ You have probably noticed that the common keyboard shortcut: Ctrl + <
#### A note on typing passwords in the terminal
- When using a command in the terminal that requires you to enter your password for authentication (such as sudo), you will notice that the characters aren't visible as you type them. While it might seem like the terminal isn’t responding, don’t worry!
+When using a command in the terminal that requires you to enter your password for authentication (such as sudo), you will notice that the characters aren't visible as you type them. While it might seem like the terminal isn’t responding, don’t worry!
+
+This is a security feature to protect confidential information, like how password fields on websites use asterisks or dots. By not displaying the characters you write, the terminal keeps your password secure.
- This is a security feature to protect confidential information, like how password fields on websites use asterisks or dots. By not displaying the characters you write, the terminal keeps your password secure.
+You can still enter your password as normal and press Enter to submit it.
- You can still enter your password as normal and press Enter to submit it.
@@ -439,6 +470,7 @@ Chrome is going to use this terminal to output various messages and won't let yo
+
MacOS
#### Step 1: Download Google Chrome
@@ -467,6 +499,7 @@ Chrome is going to use this terminal to output various messages and won't let yo
+
WSL2
#### Step 1: Download Google Chrome
@@ -502,4 +535,4 @@ The following questions are an opportunity to reflect on key topics in this less
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- It looks like this lesson doesn't have any additional resources yet. Help us expand this section by contributing to our curriculum.
+- Google Chrome is one of the most important tools that you will be using throughout the curriculum. This reference has many [common shortcuts that you may find useful when using Chrome](https://support.google.com/chrome/answer/157179?hl=en&co=GENIE.Platform%3DDesktop#zippy=%2Ctab-window-shortcuts).
diff --git a/foundations/introduction/how_this_course_will_work.md b/foundations/introduction/how_this_course_will_work.md
index 795859ab9a6..5342b8ac95c 100644
--- a/foundations/introduction/how_this_course_will_work.md
+++ b/foundations/introduction/how_this_course_will_work.md
@@ -12,13 +12,20 @@ By the end of this unit, you should not only understand how the web works but al
This section intentionally covers a very broad range of topics. It's silly to go diving straight into server-side programming without having a context for what it is and why it's useful (and why you should learn it!).
+### Lesson overview
+
+This section contains a general overview of topics that you will learn in this lesson.
+
+- Understand how The Odin Project's curriculum works.
+- Understand how the Foundations course works.
+
### How it works
This curriculum works by aggregating the best content from across the internet to teach a specific topic. In each lesson, we'll introduce the topic and try to provide some useful context before pointing you to external resources made by others.
Most lessons will contain questions that you should be able to answer before moving on. Some lessons will include exercises for you to complete. In addition, we provide several projects throughout the curriculum to help you grow your understanding by actually building things.
-Try not to think of The Odin Project, or programming, as a class in school. It's not material you learn all at once to take a test, and then pass or fail. You can think of it as a snowball. You, yourself, are a snowball. You're rolling down a hill full of snow; the further you roll, the more snow will stick to you. Sure, snow will also fall off you, and you'll forget things often, but that's just part of the process. Don't worry if you reach a project and feel like you haven't retained or memorized anything. That's natural and happens to everyone. The information will come back to you as you start solving your problems one at a time, relying on Google and the Odin Community for help.
+Try not to think of The Odin Project, or programming, as a class in school. It's not material you learn all at once to take a test, and then pass or fail. You can think of it as a snowball. You, yourself, are a snowball. You're rolling down a hill full of snow; the further you roll, the more snow will stick to you. Sure, snow will also fall off you, and you'll forget things often, but that's just part of the process. Don't worry if you reach a project and feel like you haven't retained or memorized anything. That's natural and happens to everyone. **The projects are not tests of what you have memorized so far.** They are practice points where the information will come back to you as you start solving your problems one at a time, relying on Google and the Odin Community for help.
### A note about tools
@@ -29,7 +36,7 @@ The truth is that once you complete this curriculum, you will **not need** a cou
One of the most difficult parts about this course is adopting the right mindset. An even bigger challenge is learning the core ideas of programming. It isn't just about coding, it is about problem-solving, asking the right questions, and doing enough research. Getting into a position where you are comfortable with the skills will mean that you will also be comfortable at expanding your skill-set. Once you have a solid *base*, it will be simpler to expand upon it. This is exactly what The Odin Project is about: developing these *skills*.
After completing the course you will be in a position where you can simply jump straight into the documentation for anything you are trying to learn.
-Reading the documentation is the best way to learn any tooling. But it takes skill to master. Many people feel intimidated by the technical jargon, which is understandable. People who write documentations usually assume a background in programming. This is exactly why we give you plenty of material to read on your own.
+Reading the documentation is the best way to learn any tooling. But it takes skill to master. Many people feel intimidated by the technical jargon, which is understandable. People who write documentation usually assume a background in programming. This is exactly why we give you plenty of material to read on your own.
The skills you will gain from completing The Odin Project will be the foundation that you will be building upon for years and decades to come. If you come out of the course thinking that you need another course like this one to learn something like Python, then you either don't believe in yourself or you haven't taken away the important ideas that are covered in this course.
@@ -68,6 +75,13 @@ Additional resources are the only thing that is considered optional unless expli
+### Knowledge check
+
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
+
+- [What is The Odin Project?](#introduction)
+- [Should you skip anything in The Odin Project or complete it non-linearly?](#what-comes-next)
+
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
diff --git a/foundations/introduction/introduction_to_web_development.md b/foundations/introduction/introduction_to_web_development.md
index f0b262525a8..364415eac6e 100644
--- a/foundations/introduction/introduction_to_web_development.md
+++ b/foundations/introduction/introduction_to_web_development.md
@@ -55,7 +55,7 @@ The web development industry has a long history of successful developers with va
### Why Odin?
-I want you to know that this will not be easy.
+We want you to know that this will not be easy.
There are plenty of other online curriculums for beginners, but they are often taught in an extremely isolated and controlled environment and cover only a specific topic.
diff --git a/foundations/introduction/join_the_odin_community.md b/foundations/introduction/join_the_odin_community.md
index c4c1b6aeccc..578930b0e42 100644
--- a/foundations/introduction/join_the_odin_community.md
+++ b/foundations/introduction/join_the_odin_community.md
@@ -1,6 +1,14 @@
### Introduction
-Working and collaborating with other people is an important part of working as a web developer. Therefore, we at The Odin Project encourage you to participate in our online chat community, which we'll talk more about below. By joining the community, you can grow alongside other Odinites and help each other learn web development. While you're at it, you can check out our [Facebook page](https://www.facebook.com/theodinproject/), [follow us on X](https://x.com/TheOdinProject) and catch up on [Instagram](https://www.instagram.com/theodinproject/). Use #TheOdinProject to share your Odin Project progress, updates, thoughts and to see what other Odin students are up to!
+Working and collaborating with other people is an important part of working as a web developer. Therefore, we at The Odin Project encourage you to participate in our online chat community, which we'll talk more about below. By joining the community, you can grow alongside other Odinites and help each other learn web development. While you're at it, you can check out [our Facebook page](https://www.facebook.com/theodinproject/), [follow us on X](https://x.com/TheOdinProject) and catch up on [our Instagram page](https://www.instagram.com/theodinproject/). Use #TheOdinProject to share your Odin Project progress, updates, thoughts and to see what other Odin students are up to!
+
+### Lesson overview
+
+This section contains a general overview of topics that you will learn in this lesson.
+
+- Learn about The Odin's community and how to join it.
+- Explain how to ask good and detailed questions.
+- Explain good practices for helping others with their questions.
### Why a community is awesome for you
@@ -56,7 +64,7 @@ In the chat rooms, code can be displayed differently from normal sentences by us
\`Your Code\`
-**For multiple lines of code:** use three backticks _on a separate line_ above and below your code.
+**For multiple lines of code:** use three backticks *on a separate line* above and below your code.
\`\`\`
@@ -64,9 +72,9 @@ Your Multiple Lines of Code
\`\`\`
-You can also use _code highlighting_ to add color to your multi-line code by specifying the language:
+You can also use *code highlighting* to add color to your multi-line code by specifying the language:
-\`\`\`**js** or **javascript**
+\`\`\`**js** or **JavaScript**
Your Multiple Lines of Colorful Code
@@ -83,45 +91,45 @@ Your Multiple Lines of Colorful Code
Not only is it important to know how to ask an effective question, but it is also important to know how to help others effectively. Please take a moment to review these guidelines so that you will have proper expectations of the help you will receive in our Discord community. In addition, come back and review these guidelines when you are ready to start helping others.
-#### 1. Instead of answering the question, guide them to the answer.
+#### 1. Instead of answering the question, guide them to the answer
Unless the problem is a simple typo or syntax error, it is more beneficial to guide them to find their own answer. This approach will teach good debugging skills and will increase their ability to solve future problems.
Start by asking probing questions, such as "What have you already tried?", “What do you expect this function to do?”, or “What do you think that error means?”.
-#### 2. Help only when you are certain of the answer.
+#### 2. Help only when you are certain of the answer
If you are not 100% certain of the answer, you may end up doing more harm than good, so please let someone else answer it.
Do not worry about how long someone has to wait for an answer. The right answer is worth the wait.
-#### 3. Help only when no one else is currently helping.
+#### 3. Help only when no one else is currently helping
If somebody is already getting help, do not jump in the middle of the conversation. We know you mean well, but it is overwhelming for the person receiving help to follow multiple conversations.
-#### 4. Help only when you have plenty of time.
+#### 4. Help only when you have plenty of time
If you do not have much time to help, please let someone else answer the question.
-#### 5. Adjust your expectations to their level.
+#### 5. Adjust your expectations to their level
If the question does not reveal where they are in the curriculum, ask them so that you can adjust your expectations to their knowledge level.
-#### 6. Ask for clarifications.
+#### 6. Ask for clarifications
If the question seems confusing or ambiguous, ask for more clarity, or politely link them to our bot command `/question`, which links to the [How to be great at asking coding questions](https://medium.com/@gordon_zhu/how-to-be-great-at-asking-questions-e37be04d0603) article.
-#### 7. Ask for live code.
+#### 7. Ask for live code
If the question needs to have live code to fully understand or debug, ask them to use [replit](https://replit.com) to provide it. If the problem is difficult to isolate, they should recreate the problem with isolated code.
-#### 8. Do not answer googleable questions.
+#### 8. Do not answer googleable questions
Learning how to research these questions is a very important skill for developers, so we need to empower them to find their own answer. When we answer these questions, it hinders their personal growth and makes them codependent on our community.
-Instead of answering these questions, politely ask them to google their question or use our bot command `/google` with the search terms.
+Instead of answering these questions, politely ask them to google their question or use our bot command `/search google` with the search terms.
-#### 9. Do not answer questions covered in our curriculum.
+#### 9. Do not answer questions covered in our curriculum
If you know that the answer is provided in our curriculum, ask them where they are at in the curriculum.
@@ -129,48 +137,47 @@ If they have not reached that portion of the curriculum, let them know they will
If they have already been through that portion of the curriculum, politely direct them to review that lesson.
-#### 10. Answer the question before pointing out other problems.
+#### 10. Answer the question before pointing out other problems
When helping someone it can be easy to spot other problems in their code. Resolve the original question, before pointing out any other problems that need attention.
-#### 11. Encourage students to use a debugger.
+#### 11. Encourage students to use a debugger
It is common for students to not understand the importance of using a debugger to look at the values of their variables at different points in their program. When students are getting unexpected values, politely encourage them to use a debugger with our bot command `/debug`.
-#### 12. Watch for students that need to take a step back.
+#### 12. Watch for students that need to take a step back
It is common for students to focus too hard on a problem and not be able to clearly see everything. When this situation arises, politely encourage them to step back from the problem and take a break. Oftentimes, stepping away from a problem will help them see the bigger picture and how to solve it.
-#### 13. Watch for students that are in over their head.
+#### 13. Watch for students that are in over their head
It is common for students to skip a lesson/project or think they know more than they actually do. When this situation arises, politely encourage them to go back and reread a section of the curriculum for more understanding.
-#### 14. Admit when the problem goes beyond your current knowledge.
+#### 14. Admit when the problem goes beyond your current knowledge
It is common for the actual issue to go beyond the initial question. If it goes beyond your current knowledge, it is important to admit that you are unsure of the correct answer and let someone else help.
After digging deeper into the problem, they might be able to continue troubleshooting on their own or they can wait for someone more experienced to help.
-#### 15. Be patient.
+#### 15. Be patient
Helping others solve a problem is not always easy. Remember to be patient as they struggle through the problem.
-#### 16. Duck out of the conversation if you get frustrated.
+#### 16. Duck out of the conversation if you get frustrated
Sometimes there are misunderstandings and interactions go poorly. You are a volunteer and are not obligated to help when things get out of hand. Politely duck out of the conversation and let someone else step up.
-
### Assignment
-1. First, create a free [GitHub account](https://github.com/join). As you will discover, GitHub is an integral part of the development workflow.
+1. First, [create a free GitHub account](https://github.com/join). As you will discover, GitHub is an integral part of the development workflow.
-2. Now, sign in to our [Discord server](https://discord.gg/theodinproject). Pop in and say hello! We've created an introductions room which is a great place to introduce yourself and we're always happy to welcome new community members. We have chat rooms for every development topic covered in our curriculum. Log into the chat and start exploring!
+1. Now, sign in to [our Discord server](https://discord.gg/fbFCkYabZB). Pop in and say hello! We've created an introductions room which is a great place to introduce yourself and we're always happy to welcome new community members. We have chat rooms for every development topic covered in our curriculum. Log into the chat and start exploring!
- **Link your GitHub to your Discord profile:** Go to `Discord Settings > Connections`, then click the GitHub icon. In the new tab that opens click "Allow Access", then back in Discord make sure "Display on profile" is toggled on. This will allow others to see what you're working on and vice versa!
-3. Here are some guidelines before you dive in:
+1. Here are some guidelines before you dive in:
- **Read the rules and FAQ:** Please take the time to read and understand our rules and FAQ. On the left sidebar in Discord, navigate to `TOP META`-> `rules` and `faq`.
- **Remember the human:** Behind every username there is a person with feelings! Be kind! If you don't have anything nice to say, don't say anything at all.
@@ -182,8 +189,16 @@ Sometimes there are misunderstandings and interactions go poorly. You are a volu
+### Knowledge check
+
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
+
+- [How do you join The Odin Project's Discord server?](https://discord.gg/fbFCkYabZB)
+- [What makes a good question that's easier for others to help with?](#asking-for-help)
+- [How can you more effectively help others with their coding problems?](#how-to-help-others-solve-coding-problems)
+
### Additional resources
-This section contains helpful links to related content. It isn’t required, so consider it supplemental.
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
- For more information on formatting, check out [Discord Markdown 101](https://support.discord.com/hc/en-us/articles/210298617-Markdown-Text-101-Chat-Formatting-Bold-Italic-Underline-).
diff --git a/foundations/introduction/motivation_and_mindset.md b/foundations/introduction/motivation_and_mindset.md
index 9e1e2269550..221b3816c45 100644
--- a/foundations/introduction/motivation_and_mindset.md
+++ b/foundations/introduction/motivation_and_mindset.md
@@ -2,7 +2,7 @@
Learning to code is incredibly rewarding but can also be difficult and frustrating. Like any skill worth knowing, it takes time to acquire, and it can't be learned in a weekend or even a month. With that said, we believe anyone can learn how to program as long as they are willing to put in the time and effort.
-So before we get into the meat of the curriculum, we're going to go over the following to help you get the most out of The Odin Project: the things that will help you succeed in your goal of learning to code and the pitfalls that you should try to avoid.
+So before we get into the meat of the curriculum, we'll cover the following aspects to help you get the most out of The Odin Project: factors that will help you succeed in learning to code and the pitfalls that you should try to avoid.
### Motivation
@@ -54,7 +54,7 @@ You will inevitably get stuck at some point in the curriculum, perhaps due to a
- Take a break: Allow your diffuse learning state to work on the problem.
- Ask for help in the [TOP Discord server](https://discord.gg/theodinproject); come prepared with your research. People will be more willing to help you when they can see you have already put effort into trying to figure out the solution on your own.
-Additionally, feel free to follow the diagram below that should help you going through problems that you may come across:
+Additionally, feel free to follow the diagram below to help you navigate problems that you may come across:
![diagram to help learners go through problems when they are stuck](https://cdn.statically.io/gh/TheOdinProject/curriculum/978ba0e91a6a47452de819246f4c495e6b092f54/foundations/introduction/motivation_and_mindset/imgs/00.png)
@@ -64,13 +64,13 @@ As technology advances, we have seen some incredible tools emerge that can help
While these tools are amazing, learners should be aware of the impacts that using such tools can have on core competency. David Humphrey, a computer science professor, [wrote about ChatGPT and its potentially negative impacts on core learning.](https://blog.humphd.org/cheatgpt/) It is a good read about the pitfalls of using generative AI in an educational context.
-For learners that are new to programming, tools like ChatGPT or Github Copilot can result in some blindspots and gaps in your knowledge.
+For learners who are new to programming, tools like ChatGPT or GitHub Copilot can result in blindspots and gaps in your knowledge.
1. By using a generative AI, learners may miss the opportunity to discover how something works and why things are done that way.
-1. Asking good questions is an important skill to have, and using generative AI to skip asking people (like our helpful Discord community) means that your development of this skill is delayed.
+1. Asking good questions is an important skill, and relying on generative AI instead of asking people (like our helpful Discord community) can delay the development of this skill.
1. Learning to talk about the code that one writes is another important skill. In a professional environment, it is highly likely that you will be doing code reviews where you have to explain the how and why behind what you are presenting. Using the Odin community to ask good questions about your code when you require help can help develop this skill as well.
1. As David Humphrey mentions, the output of generative AI must be closely scrutinized, and learners who are new to programming generally don't have a good overall understanding to be able to determine if the output is good or bad.
-1. While learning how to provide good prompts to an AI tool is a skill, it is one that is supplementary to developing foundational programming skills. The Odin curriculum strives to teach that foundational skill.
+1. While learning how to provide good prompts to an AI tool is a skill, it is supplementary to developing foundational programming skills. The Odin curriculum strives to teach that foundational skill.
1. AI tools are designed to answer questions and are not designed to help learners develop research and problem solving skills. If you ask an AI for information, it will provide information to you. If you ask a human, they may invite you to share your understanding of the problem and offer some guidance on how to discover a solution.
1. Without practice in research, problem solving and critical thinking, interviews could be a struggle since it is very likely that applicants would not be allowed to use AI tools.
@@ -80,11 +80,11 @@ We do not recommend using AI tools for your learning.
You will have more success with Odin by putting **consistent** time into it rather than working on it once a week. Building a habit of studying every day at a specific time and with a specific goal will ensure that you make consistent progress.
-It may take you longer than others to grasp concepts, or it may take you less time. This doesn't mean you're smarter or dumber than others, it means you've had differing life experiences that may or may not have prepared you for learning this stuff. Someone who grew up around an engineer may have some advantages over someone who didn't, but it doesn't mean you can't learn those skills.
+It may take you longer than others to grasp concepts, or it may take you less time. This doesn't mean you're smarter or dumber than others, it means you've had differing life experiences that may or may not have prepared you for learning those concepts. Someone who grew up around an engineer may have some advantages over someone who didn't, but it doesn't mean you can't learn those skills.
-The Odin Project isn't like college or university, it is self paced and allows you to get a really solid grasp of concepts before moving on. In school, you're forced to keep up or you will fail. The difference here is that coming into The Odin Project, you're not expected to have much knowledge; there are no prerequisites. We've had people be successful coming through here who only knew how to check their email with a computer. We've also seen success from computer science degree holders. Treating The Odin Project like a static timeline is understandable, but is a sign of misplaced expectations. You don't know what you don't know yet, and that's OK! There are no due-dates on things in The Odin Project so you can spend the time to do it right and discuss the topics.
+The Odin Project isn't like college or university, it is self-paced and allows you to get a solid grasp of concepts before moving on. In school, you're forced to keep up, or you will fail. The difference with The Odin Project is that you’re not expected to have much prior knowledge; there are no prerequisites. We've had people be successful coming through here who only knew how to check their email with a computer. We've also seen success from computer science degree holders. Treating The Odin Project like a static timeline is understandable, but is a sign of misplaced expectations. You don't know what you don't know yet, and that's OK! There are no due-dates on lessons in The Odin Project, so you can spend enough time to do it right and discuss the topics.
-Deadlines cause un-needed stress. Since The Odin Project is a free and open platform, you are not beholden to a deadline. Creating your own deadlines is a good way to rush concepts that should not be rushed. This course is very research based, meaning you will have to do research to complete tasks and projects. There's no telling if you can find the article or post that helps you in the right way, quickly to meet your deadlines, but we bet you learned a TON along the way that you can use in the future. People that do this kind of research and strive to write better solutions tend to become better developers in the future. There's no knowing how long it could take you to learn how to query stuff to find your answers. There are no solid guidelines on that. If you're doing The Odin Project because you need a high paying job *right-now*, you're not going to become a solid developer within the timeframe you have set. Stress and anxiety absolutely do not help you learn either. Relax and just enjoy the ride.
+Deadlines cause unneeded stress. Since The Odin Project is a free and open platform, you are not beholden to a deadline. Creating your own deadlines can lead to rushing through concepts that should not be rushed. This course is very research based, meaning you will have to do research to complete tasks and projects. There’s no guarantee you will find the right article or post quickly enough to meet your deadlines, but you will likely learn a TON along the way that you can use in the future. People that do this kind of research and strive to write better solutions tend to become better developers in the future. There’s no way to know how long it will take you to learn how to query concepts to find your answers. There are no solid guidelines on that. If you're doing The Odin Project because you need a high-paying job *right now*, you’re unlikely to become a solid developer within the timeframe you have set. Stress and anxiety absolutely do not help you learn either. Relax and just enjoy the ride.
Long story short: Don't worry, just go learn!
@@ -119,7 +119,7 @@ What to do during your break:
- Play a quick game.
- Go for a short walk outside.
-Checkout this article for [more information on breaks & productivity](https://simpleprogrammer.com/taking-breaks-will-boost-productivity/).
+Check out this article for [more information on breaks & productivity](https://simpleprogrammer.com/taking-breaks-will-boost-productivity/).
#### Digital distractions
@@ -131,13 +131,13 @@ Digital distractions are email and Facebook notifications and time-wasting websi
Physical distractions are distractions from your environment, like a TV in the background or other people talking. These distractions can be just as damaging to your focus as digital distractions.
-**Solution:** Find a quiet place to study where you can go to focus in your home. If that's not an option, you can use noise cancelling headphones to block out noisy distractions in your environment. There are also complimentary public and university libraries that are serene and comfortable. Some libraries even operate 24/7, uninterrupted. Beyond just providing a pleasant study space, the presence of others studying around you instills a sense of productivity.
+**Solution:** Find a quiet place in your home where you can focus on your studies. If that's not an option, you can use noise cancelling headphones to block out noisy distractions in your environment. There are also complimentary public and university libraries that are serene and comfortable. Some libraries even operate 24/7, uninterrupted. Beyond just providing a pleasant study space, the presence of others studying around you instills a sense of productivity.
#### Rabbit holes
Because we cover so much material on The Odin Project and link to so many high quality courses and tools, it is easy for students to get pulled into rabbit holes by spending time trying to learn all there is to know about a subject that they aren't ready for or won't benefit them much. **We have put a lot of effort into structuring the curriculum** so that all of the important things that you need to know about web development are covered exactly when you need to know them.
-**Solution:** Stick to the path laid out as much as possible. Try to limit time spent going down rabbit holes as these sidetracks can really ruin your momentum.
+**Solution:** Stick to the path laid out as much as possible. Try to limit time spent going down rabbit holes, as these sidetracks can really ruin your momentum.
#### Comparing yourself to others
@@ -145,7 +145,7 @@ Students often compare themselves to others who are farther along in their codin
**Solution:** Only compare yourself to your past self. Have your abilities and knowledge improved from where you were last week, last month, or last year? Be proud of the progress that you've made!
-#### Counter productive note-taking
+#### Counterproductive note-taking
The Odin Project does not recommend taking a lot of notes throughout your web development educational journey because it can be time-consuming and often leads to wasted effort.
@@ -153,7 +153,7 @@ The Odin Project does not recommend taking a lot of notes throughout your web de
### Conclusion
-Learning any new skill is a journey full of speed bumps and obstacles to be overcome. We hope that the principles laid out here will put you in a much better position to succeed and get the most out of The Odin Project.
+Learning any new skill is a journey full of speed bumps and obstacles to overcome. We hope that the principles laid out here will put you in a much better position to succeed and get the most out of The Odin Project.
### Assignment
diff --git a/foundations/javascript_basics/clean_code.md b/foundations/javascript_basics/clean_code.md
index a91bf9bb8b4..5576306d089 100644
--- a/foundations/javascript_basics/clean_code.md
+++ b/foundations/javascript_basics/clean_code.md
@@ -1,4 +1,3 @@
-
### Introduction
You might think that the majority of a developer's work involves writing code. However, in reality, a significant amount of time is spent on *reading* code. This includes code written by other team members, code written by people who are no longer part of your team, and even code that you wrote two weeks ago but may not remember much about.
@@ -24,11 +23,11 @@ Consider the following examples:
Example A:
```javascript
+const x= function (z){
+ const w = "Hello ";
+return w + z
-const x = function (z) {
- const w = "Hello ";
-return w + z
-};
+ }
x("John");
```
@@ -44,13 +43,13 @@ const generateUserGreeting = function (name) {
generateUserGreeting("John");
```
-Which of these examples do you find easier to read? It's immediately clear that the latter one is more meaningful. Surprisingly, both of these functions perform the exact same task \(in the exact same way!\), and both are valid code. But the second one is much more readable. Why?
+Which of these examples do you find easier to read? It's immediately clear that the latter one is more meaningful. Surprisingly, both of these functions perform the exact same task (in the exact same way!), and both are valid code. But the second one is much more readable. Why?
-In the first example, single-letter variables are used and the indentation is inconsistent. The result is a piece of code that is confusing and messy.
+In the first example, single-letter variables are used and the indentation and spacing are inconsistent. The result is a piece of code that is confusing and messy.
Imagine you're collaborating on a project with someone who has written the first function. How long will it take you to decipher what's going on so you can continue with your work? Or perhaps you've written it yourself some time ago and completely forgotten that it even existed. In both situations, you will eventually understand what is happening, but it's not going to be fun.
-Example B represents clean code. While you may not know what each part does, it's much easier to guess what's happening because the functions and variables are named clearly. The indentation follows a consistent and logical pattern.
+Example B represents cleaner code. While you may not know what each part does, it's much easier to guess what's happening because the functions and variables are named clearly. The indentation and spacing follow a consistent and logical pattern.
Single characters can be used as variable names in the context of a loop or a callback function, but avoid them elsewhere.
@@ -58,7 +57,17 @@ Single characters can be used as variable names in the context of a loop or a ca
camelCase is a naming convention that allows writing multiple words together without spaces or punctuation. In camelCase, when a variable name consists of multiple words like our `setTimeout` example, the first word is written completely in lowercase, while the first letter of the second word (and any subsequent words) are capitalized.
-Throughout this lesson, most of our variables and functions (at least in the good examples!) will be named using camelCase. It's a good example to follow.
+Throughout this lesson, most of our variables and functions will be named using camelCase. While not every language uses this convention, it's very common in JavaScript so it'll be a good example to follow.
+
+
+
+#### Conventions are only conventions
+
+While this lesson shares some examples on ways to clean up code, in reality, every organization will have different specific approaches, some of which may differ slightly from our examples in this lesson. Nothing is absolute.
+
+What matters most is that these approaches all serve the same overall purpose - improve code readability and maintainability. Until a time comes where you need to follow a specific set of conventions, it is sensible to follow some convention and be consistent with them.
+
+
### Naming functions and variables
@@ -70,44 +79,46 @@ In our good example, we have a variable `greeting`, to which the parameter `name
Now, try picturing a conversation with someone about the bad example. The function is named `x` with variables like `z`, and `w`. Oof, not nice.
-#### Use a consistent vocabulary
+#### Use consistent vocabulary
-Variables of the same type should have consistent naming. Consider the following examples from a game:
+Variables of the same type ideally follow a consistent naming system. Consider the following examples from a game:
```javascript
- // Good
+// Consistent naming
function getPlayerScore();
function getPlayerName();
function getPlayerTag();
+```
-// Bad
+They all follow the same naming system of "get a thing". Now consider the following:
+
+```javascript
+// Inconsistent naming
function getUserScore();
function fetchPlayerName();
function retrievePlayer1Tag();
```
-In the bad example, three different names are used to refer to the player and the actions taken. Additionally, three different verbs are used to describe these actions. The good example maintains consistency in both variable naming and the verbs used.
-
-Variables should always begin with a noun or an adjective (that is, a noun phrase) and functions with a verb.
+In the inconsistent example, three different verbs are used for the functions. While they all mean a similar thing, at a glance you might assume different verbs were used for a specific reason (e.g. "getting" might not be *quite* the same thing as "fetching" in some contexts). Additionally, what's the difference between `User`, `Player` and `Player1`? If there is no difference then ideally, you'd use the same name e.g. `Player`. Consistency allows for predictability.
-Another set of examples can illustrate why this matters:
+Variables should preferably begin with a noun or an adjective (that is, a noun phrase), as they typically represent "things", whether that thing is a string, a number etc. Functions represent actions so ideally begin with a verb.
```javascript
-// Good
+// Preferable
const numberOfThings = 10;
const myName = "Thor";
const selected = true;
-// Bad (these start with verbs, could be confused for functions)
+// Not preferable (these start with verbs, could be confused for functions)
const getCount = 10;
-const isSelected = true;
+const showNorseGods = ["Odin", "Thor", "Loki"];
-// Good
+// Preferable
function getCount() {
return numberOfThings;
}
-// Bad (it's a noun)
+// Not preferable (myName doesn't represent some kind of action)
function myName() {
return "Thor";
}
@@ -115,35 +126,35 @@ function myName() {
### Use searchable and immediately understandable names
-Sometimes, it can be tempting to use an undeclared variable. Let's take another look at an example:
+Sometimes, it can be tempting to use "magic values" i.e. explicit values, such as bare numbers or strings. Let's take another look at an example:
```javascript
setTimeout(stopTimer, 3600000);
```
-The problem is obvious. What does the undeclared variable `3600000` mean, and how long is this timeout going to count down before executing `stopTimer`? Even if you know that JavaScript understands time in milliseconds, a calculator is needed.
+The problem is obvious. What does the magic number `3600000` mean, and how long is this timeout going to count down before executing `stopTimer`? Even if you know that JavaScript understands time in milliseconds, you'd probably need a calculator or Google to figure out how many seconds or minutes it represents.
Now, let's make this code more meaningful by introducing a descriptive variable:
```javascript
-const MILLISECONDS_PER_HOUR = 60 * 60 * 1000; // 3,600,000;
+const ONE_HOUR = 3600000; // Can even write as 60 * 60 * 1000;
-setTimeout(stopTimer, MILLISECONDS_PER_HOUR);
+setTimeout(stopTimer, ONE_HOUR);
```
Much better, isn't it? The variable is declared with a descriptive name, and you don't need to perform any calculations when reading this code.
-You might wonder why this variable is declared with all caps when we recommended camelCase earlier. This is a convention to be used when the programmer is absolutely sure that the variable is *truly* a constant. We know that the milliseconds in an hour will never change, so it's appropriate here.
+You might wonder why this variable is declared with all caps when we recommended camelCase earlier. This is a convention to be used when the programmer is absolutely sure that the variable is *truly* a constant, especially if it represents some kind of concept like a specific duration of time. We know that the milliseconds in an hour will never change, so it's appropriate here. Remember, this is only a convention. Not everyone will necessarily do things the same way.
### Indentation and line length
Now it's time to head to more controversial topics. The war between coders that use tabs and coders that use spaces to indent their code is essentially a joke by now, as demonstrated in the [tabs versus spaces scene from the show Silicon Valley](https://www.youtube.com/watch?v=SsoOG6ZeyUI).
-What actually matters is *consistency*. Choose a way to indent and stick to it. Various JS style-guides recommend different options, and one is not really superior to the other. A few popular ones are linked in the additional resources.
+What actually matters is *consistency*. Choose a way to indent and stick to it. Various JavaScript style guides recommend different options, and one is not really superior to the other. We will look at style guides and related tools in more detail later in the curriculum.
#### Line length
-Again, different style guides will recommend different options for this one, but just about ALL of them suggest limiting the length of each line of code.
+Again, different style guides will recommend different options for this one, but just about *all* of them suggest limiting the length of each line of code.
Generally, your code will be easier to read if you manually break lines that are longer than about 80 characters. Many code editors have a line in the display to show when you have crossed this threshold. When manually breaking lines, you should try to break immediately *after* an operator or comma.
@@ -162,21 +173,21 @@ let reallyReallyLongLine =
oneMoreReallyLongThing;
// Or maybe like this
- let anotherReallyReallyLongLine = something + somethingElse + anotherThing +
- howManyTacos + oneMoreReallyLongThing;
+let anotherReallyReallyLongLine = something + somethingElse + anotherThing +
+ howManyTacos + oneMoreReallyLongThing;
```
+Different formats aren't necessarily right or wrong, and different people may prefer different things. Do things in a way that makes sense to you, and stay consistent with it.
+
### Semicolons
-Semicolons are *mostly* optional in JavaScript because the JS compiler will automatically insert them if they are omitted. This functionality CAN break in certain situations, leading to bugs in your code, so it is better to get used to adding semi-colons.
+Semicolons are *mostly* optional in JavaScript because the JavaScript interpreter will automatically insert them if they are omitted. This functionality *can* break in certain situations, leading to bugs in your code, so we'd recommend getting used to adding semicolons.
-Again: consistency is the main thing.
+Whether you do or not, again, consistency is the main thing.
### About comments
-Comments are a great tool. But like any good tool, it can be misused. Especially for someone early in their coding journey, it might be tempting to have comments that explain *everything* the code is doing. This is not a good practice.
-
-Next, we'll look into some common pitfalls in commenting and *why* they are pitfalls.
+Comments are a great tool but like any good tool, they can be misused. Especially for someone early in their coding journey, it might be tempting to have comments that explain *everything* the code is doing. This is generally not a good practice. Let's look at some common pitfalls when commenting and *why* they are pitfalls.
#### Don't comment when you should be using git
@@ -206,49 +217,48 @@ theFunctionInUse();
#### Tell why, not how
-The purpose of comments is not to provide pseudo code that duplicates your code. Good comments explain the *reasons* behind a piece of code.
+Ideally, comments do not provide pseudocode that duplicates your code. Good comments explain the *reasons* behind a piece of code. Sometimes you won't even need a comment at all!
-Let's look at an example to see this in practice:
+Say we had a string where part of the text was inside square brackets and we wanted to extract the text within those brackets.
```javascript
-// Bad Example - comment doesn't tell why, only what and how
-
-// This function increments the value of i by 1
-function incrementI(i) {
- i = i + 1; // Add one to i
- return i;
+// Function to extract text
+function extractText(s) {
+ // Return the string starting after the "[" and ending at "]"
+ return s.substring(s.indexOf("[") + 1, s.indexOf("]"));
}
+```
-// Better Example - comment tells a why
+The comments just describe what we can tell from the code itself. Slightly more useful comments could explain the reasons behind the code.
-// This function increments the value of index to move to the next element
-function incrementI(i) {
- i = i + 1;
- return i;
+```javascript
+// Extracts text inside square brackets (excluding the brackets)
+function extractText(s) {
+ return s.substring(s.indexOf("[") + 1, s.indexOf("]"));
}
+```
-// Good Example - the code tells all that is needed
+But often, we can make the code speak for itself without comments.
-function moveToNextElement(index) {
- index = index + 1;
- return index;
+```javascript
+function extractTextWithinBrackets(text) {
+ const bracketTextStart = text.indexOf("[") + 1;
+ const bracketTextEnd = text.indexOf("]");
+ return text.substring(bracketTextStart, bracketTextEnd);
}
```
-In the bad example, the comments explain twice what the code does. But for this, you could've just read the code, so the comments are redundant.
-
-In the better example, the comment clarifies the purpose of the function: moving to the next element. That's good, but we can do *even* better.
+In the first example, the comments repeat twice what the code does. But for this, you could've just read the code, so the comments are redundant.
-In the good example, no comments are needed at all. The use of descriptive functions and variable names eliminates the need for additional explanations. Pretty neat, huh?
+In the second example, the comment clarifies the purpose of the function: extracting the text between square brackets from a string and not just "extracting text". That's handy, but we can do *even* better.
-*This doesn't mean good code should lack comments*. In many situations, well-placed comments are priceless. The article linked in the assignment section goes into more depth on this. We don't want you to avoid comments; just be mindful of how they are best used.
+In the last example, no comments are needed at all. The use of descriptive functions and variable names eliminates the need for additional explanations. Pretty neat, huh?
-Let's look at one final example where a comment serves a good purpose:
+**This doesn't mean good code should lack comments.** Let's look at an example where a comment serves a helpful purpose:
```javascript
-
function calculateBMI(height, weight) {
- // The formula for BMI is weight in kilograms divided by height in meters squared
+ // The formula for BMI is weight in kilograms divided by height in meters squared
const heightInMeters = height / 100;
const bmi = weight / (heightInMeters * heightInMeters);
return bmi;
@@ -257,18 +267,22 @@ function calculateBMI(height, weight) {
This comment helps to refresh the reader on how BMI is calculated in plain English, helping the reader to see why the height needs to be converted and what the following calculation is doing. We are almost there with the naming, but the comment still adds further clarity.
+In many situations, well-placed comments are priceless. They might explain why an unintuitive bit of code is necessary, or perhaps the bigger picture of why a certain function is *particularly* important to be called here and not there. The article linked in the assignment section goes into more depth on this.
+
### In conclusion
-Now that we've covered these ideas, it's good to return to the reminder we shared at the start. Don't try to write perfectly clean code; this will only lead to frustration. Writing "spaghetti" is inevitable; everyone does it sometimes. Just keep these ideas in mind, and with time and patience, your code will start to get cleaner.
+Now that we've covered these ideas, it's good to return to the reminder we shared at the start. Don't try to write perfectly clean code, this will only lead to frustration. Writing "spaghetti" is inevitable; everyone does it sometimes. Just keep these ideas in mind, and with time and patience, your code will start to get cleaner.
Learning to write clean code is a process of constant improvement. One that will extend beyond you *completing* The Odin Project. This lesson is meant to serve as a primer and a starting point for that journey.
+> Great code comes from experience. Experience comes from not-so-great code.
+
### Assignment
1. Read [10 Principles for Keeping Your Programming Code Clean](https://onextrapixel.com/10-principles-for-keeping-your-programming-code-clean/) to get some great tips for clean code.
-1. To help better understand good comment practices, read about [comments telling us how code works](https://blog.codinghorror.com/code-tells-you-how-comments-tell-you-why/) as well as how to [code without comments](https://blog.codinghorror.com/coding-without-comments/).
+1. To help better understand good comment practices, read about [comments telling us why code works](https://blog.codinghorror.com/code-tells-you-how-comments-tell-you-why/) as well as how to [code without comments](https://blog.codinghorror.com/coding-without-comments/).
@@ -277,14 +291,14 @@ Learning to write clean code is a process of constant improvement. One that will
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
- [Why is it important to write clean code?](#introduction)
-- [Name 5 clean code principles previously mentioned](https://onextrapixel.com/10-principles-for-keeping-your-programming-code-clean/)
+- [What are some good principles for keeping code clean?](https://onextrapixel.com/10-principles-for-keeping-your-programming-code-clean/)
- [What is the difference between good comments and bad comments?](https://onextrapixel.com/10-principles-for-keeping-your-programming-code-clean/)
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- [A nice op-ed](https://www.martinfowler.com/bliki/CodeAsDocumentation.html)
+- [A nice op-ed on code as documentation](https://www.martinfowler.com/bliki/CodeAsDocumentation.html)
- [Airbnb style guide](https://github.com/airbnb/javascript)
- [Chaining methods to write sentences](https://web.archive.org/web/20190211152543/https://javascriptissexy.com/beautiful-javascript-easily-create-chainable-cascading-methods-for-expressiveness/)
- [Clean code in JavaScript](https://github.com/ryanmcdermott/clean-code-javascript)
diff --git a/foundations/javascript_basics/data_types_and_conditionals.md b/foundations/javascript_basics/data_types_and_conditionals.md
index 319aef14d20..9cee7c580cc 100644
--- a/foundations/javascript_basics/data_types_and_conditionals.md
+++ b/foundations/javascript_basics/data_types_and_conditionals.md
@@ -48,6 +48,14 @@ Note: Feel free to browse the files on the left column to gain familiarity with
+#### Replit fork limit
+
+Replit now limits free accounts to having 3 repls at a time. If you reach the limit of 3 repls, you can delete one or more of the previous forks to create a room for the new one. To do so, go to [your repls](https://replit.com/repls) and delete what you no longer need.
+
+
+
+
+
#### Replit and AI
Replit recently introduced an AI assistant, which is on by default. Before trying any of the exercises, you should first disable it, in order to prevent it from spoiling the exercise. You can do so by clicking on the button labeled AI in the bottom left corner of the code view and then unchecking the "Enable" checkbox.
diff --git a/foundations/javascript_basics/function_basics.md b/foundations/javascript_basics/function_basics.md
index 9124da841b6..74b8c17faef 100644
--- a/foundations/javascript_basics/function_basics.md
+++ b/foundations/javascript_basics/function_basics.md
@@ -38,7 +38,7 @@ Here is a diagram to help you visualize how parameters are passed to a function,
Make note of the fact that by calling `favoriteAnimal()` inside of `console.log()` with the argument `'Goat'` we get the return value of the function, string of `"Goat is my favorite animal!"`, printed to the console. We're passing in a function call `favoriteAnimal('Goat')` as an argument in a different function call - `log()`.
-Keep this possibility in mind because you'll be passing in function calls as arguments somewhat often. If we just called the function without `console.log`ging what it returns, nothing would appear in the console **but** nonetheless the function would return that string.
+Keep this possibility in mind because you'll be passing in function calls as arguments somewhat often. If we just called the function without using `console.log` to print it's return value, nothing would appear in the console **but** nonetheless the function would return that string.
Feel free to experiment with the code on your own and replace `'Goat'` with your favorite animal. Notice how we can change the argument to anything we like? Try changing `animal` in the function declaration and in the function body, too. What happens when you do?
diff --git a/foundations/javascript_basics/javascript_developer_tools.md b/foundations/javascript_basics/javascript_developer_tools.md
index f015a07da50..aa9c11c8661 100644
--- a/foundations/javascript_basics/javascript_developer_tools.md
+++ b/foundations/javascript_basics/javascript_developer_tools.md
@@ -41,7 +41,7 @@ Google has updated some of the required sections in the below tutorials and some
- Mobile Simulation
1. [Simulate mobile devices with Device Mode](https://developer.chrome.com/docs/devtools/device-mode/)
- JavaScript
- 1. [Debug JavaScript](https://developer.chrome.com/docs/devtools/javascript/) - Warning: In point 4 of step 3 of the tutorial, devtools will pause on the second line (`if (inputsAreEmpty()) {`) rather than at the declaration of the function. Don’t worry this is expected.
+ 1. [Debug JavaScript](https://developer.chrome.com/docs/devtools/javascript/) - Warning: In point 4 of step 3 of the tutorial, devtools will pause on the second line (`if (inputsAreEmpty()) {`) rather than at the declaration of the function. Don’t worry, this is expected.
1. [Pause your code with breakpoints](https://developer.chrome.com/docs/devtools/javascript/breakpoints/)
1. Go through the [Chrome DevTools console overview](https://developer.chrome.com/docs/devtools/console/) to familiarize yourself with the console and its usage.
diff --git a/foundations/javascript_basics/object_basics.md b/foundations/javascript_basics/object_basics.md
index 76acbe3671e..5037725aa14 100644
--- a/foundations/javascript_basics/object_basics.md
+++ b/foundations/javascript_basics/object_basics.md
@@ -11,6 +11,7 @@ This section contains a general overview of topics that you will learn in this l
- Creating objects.
- Accessing object properties.
- Using multiple object operators.
+- Understanding the differences between object and primitive data types.
- Using the `map`, `filter` and `reduce` array methods.
### Objects
@@ -20,6 +21,89 @@ Objects are a *very* important part of the JavaScript language, and while for th
1. This JavaScript.info [article on objects](https://javascript.info/object) is the best place to get started.
1. The [MDN tutorial on objects](https://developer.mozilla.org/en-US/docs/Learn/JavaScript/Objects/Basics) isn't bad either, so check it out if you need another take on the subject.
+### Differences between objects and primitives
+
+Earlier in the curriculum you've learned about [primitive data types](https://www.theodinproject.com/lessons/foundations-data-types-and-conditionals). Now that you've seen the object data type, which includes but is not limited to, objects ({key: value}), arrays, and functions. The main difference between the two is that primitives can contain only a single thing (string, number, etc). Objects data types are used to store a collection of data and more complex entities.
+
+Besides the formal differences, there are also some technical differences which affect how we use each data type. When you define a primitive variable, it will contain a copy of the information provided to it:
+
+```javascript
+let data = 42;
+// dataCopy will store a copy of what data contains, so a copy of 42
+let dataCopy = data;
+
+// which means that making changes to dataCopy won't affect data
+dataCopy = 43;
+
+console.log(data); // 42
+console.log(dataCopy); // 43
+```
+
+On the other hand, when you define an object variable, it will contain a *reference* to the object provided to it:
+
+```javascript
+// obj contains a reference to the object we defined on the right side
+const obj = { data: 42 };
+// objCopy will contain a reference to the object referenced by obj
+const objCopy = obj;
+
+// making changes to objCopy will make changes to the object that it refers to
+objCopy.data = 43;
+
+console.log(obj); // { data: 43 }
+console.log(objCopy); // { data: 43 }
+```
+
+This behavior isn't new to you. In your last project, you made changes to the cells in the Etch-A-Sketch grid by using references. Let's take this code snippet as an example:
+
+```javascript
+const element = document.querySelector("#container");
+element.style.backgroundColor = "red";
+```
+
+We're mutating the variable we declared (`element`), yet the changes affect the corresponding node in the DOM. Why does it happen? That's because the node we have in our code is a **reference** to the same node that our DOM uses. If that wasn't a reference, but a copy like primitive data types behave, our changes would have **no** effect! Because the changes would be made to the local copy we have.
+
+This behavior is also something to consider when we pass arguments to a function. Let's take the following functions for example:
+
+```javascript
+function increaseCounterObject(objectCounter) {
+ objectCounter.counter += 1;
+}
+
+function increaseCounterPrimitive(primitiveCounter) {
+ primitiveCounter += 1;
+}
+
+const object = { counter: 0 };
+let primitive = 0;
+
+increaseCounterObject(object);
+increaseCounterPrimitive(primitive);
+```
+
+Take a moment and guess what will happen to `object` and what will happen to `primitive` after we make the function calls.
+
+If you answered that the object counter would increase by 1, and the primitive counter wouldn't change, you're correct. Remember that the parameter `objectCounter` contains a *reference* to the same object as the `object` variable we gave it, while `primitiveCounter` contains only a copy of the primitive value only.
+
+
+
+#### Reassigning object data type variables
+
+While mutating the object we have a reference to will affect all other variables that reference it, reassigning a variable won’t affect what the other variables refer to. For example:
+
+```javascript
+let animal = { species: "dog" };
+let dog = animal;
+
+// reassigning animal variable with a completely new object
+animal = { species: "cat" };
+
+console.log(animal); // { species: "cat" }
+console.log(dog); // { species: "dog" }
+```
+
+
+
### Intermediate/advanced array magic
Besides being a quick and handy way to store data, arrays also have a set of functions for manipulating that data in very powerful ways. Once you begin to master these functions you will start to see ways to use them all over the place! There are really only a handful of these functions, but as you'll soon see, the possibilities of what you can do with them are near endless.
@@ -177,9 +261,9 @@ function sumOfTripledEvens(array) {
-1. Start out by watching [JavaScript Array Cardio Practice - Day 1](https://www.youtube.com/watch?v=HB1ZC7czKRs) from Wes Bos. To follow along, fork and clone the [JavaScript30 repository](https://github.com/wesbos/JavaScript30).
+1. Read through the [array method guide](https://javascript.info/array-methods) for a comprehensive overview of array methods in JavaScript. Complete the exercises at the end, except for "Create an extendable calculator," as it involves more advanced concepts that we have not yet covered.
+1. Follow up by watching [JavaScript Array Cardio Practice - Day 1](https://www.youtube.com/watch?v=HB1ZC7czKRs) by Wes Bos. To follow along, fork and clone the [JavaScript30 repository](https://github.com/wesbos/JavaScript30).
1. Watch and code along with [Array Cardio Day 2](https://www.youtube.com/watch?v=QNmRfyNg1lw).
-1. Read through the [array method guide](https://javascript.info/array-methods) for a more comprehensive and in-depth guide to array methods in JavaScript. Complete the exercises at the end except for "Create an extendable calculator", as that involves more advanced concepts we have not yet covered.
1. At this point you just need a little more practice! Go back to the [JavaScript exercises repository](https://github.com/TheOdinProject/javascript-exercises) that we introduced in the [Fundamentals Part 4](https://www.theodinproject.com/lessons/foundations-fundamentals-part-4) assignment. Review each README file prior to completing the following exercises in order:
- `08_calculator`
- `09_palindromes`
@@ -199,6 +283,7 @@ The following questions are an opportunity to reflect on key topics in this less
- [What is the difference between objects and arrays?](https://javascript.info/object#summary)
- [How do you access object properties?](https://developer.mozilla.org/en-US/docs/Learn/JavaScript/Objects/Basics#bracket_notation)
+- [How do primitives and object types differ when you assign them to other variables, or pass them into functions?](https://www.theodinproject.com/lessons/foundations-object-basics#differences-between-objects-and-primitives)
- [What is `Array.prototype.map()` useful for?](https://www.youtube.com/watch?v=HB1ZC7czKRs&t=233s)
- [What is `Array.prototype.filter()` useful for?](https://www.youtube.com/watch?v=HB1ZC7czKRs&t=84s)
- [What is `Array.prototype.reduce()` useful for?](https://youtu.be/HB1ZC7czKRs?t=467)
diff --git a/getting_hired/applying_and_interviewing/project_resume.md b/getting_hired/applying_and_interviewing/project_resume.md
index 1e972b40b9e..26001c6c14b 100644
--- a/getting_hired/applying_and_interviewing/project_resume.md
+++ b/getting_hired/applying_and_interviewing/project_resume.md
@@ -14,7 +14,7 @@ Remember: ONE PAGE.
- This Career Tool Belt article lists [6 free resume builder websites](https://web.archive.org/web/20230930114027/https://www.careertoolbelt.com/5-best-free-resume-builder-websites/)
- [Novorésumé also has a free tier with templates available, as an alternate resource](https://novoresume.com/)
-- Formatting could use some work, but here’s an [example resume from CareerCup.com](http://www.careercup.com/resume)
+- Formatting could use some work, but here’s an [example resume from CareerCup.com](https://www.gayle.com/resume)
- [Another great (totally free) resume builder](https://flowcv.io/)
### Optional reading
diff --git a/git/foundations_git/commit_messages.md b/git/foundations_git/commit_messages.md
index 659731d72c5..67c759c953b 100644
--- a/git/foundations_git/commit_messages.md
+++ b/git/foundations_git/commit_messages.md
@@ -68,6 +68,12 @@ Ahh, that's better! :) Now, developers can gain a better understanding of this c
- Contains a body that provides a concise yet clear description of why the commit needed to be made (e.g., "Screen readers won't read the images to users with disabilities without this information").
- Separates the subject from the body with a new/blank line. This is a best practice we highly recommend following. It makes commit messages easier for other developers to read.
+### How to commit with a subject and body in the message
+
+Up until now, you've been told to commit with `git commit -m `. To make a commit with subject and body in message, the simplest way is to type `git commit` without the `-m` flag and message argument.
+
+Doing so will open a new Visual Studio Code tab if you had [set Visual Studio Code as the Git editor](https://www.theodinproject.com/lessons/foundations-git-basics#changing-the-git-commit-message-editor). You can remove any comments and enter your multi-line messages. When you save and close the tab, your commit will be created.
+
### When to commit
A good way to view a commit is like a “snapshot” of your code at the moment that it was made. That version of your code up to that point will be saved for you to revert back to or look back at.
diff --git a/git/foundations_git/introduction_to_git.md b/git/foundations_git/introduction_to_git.md
index de8332bb4e0..a8046eb2e95 100644
--- a/git/foundations_git/introduction_to_git.md
+++ b/git/foundations_git/introduction_to_git.md
@@ -57,4 +57,4 @@ This section contains helpful links to related content. It isn't required, so co
- [What is Git and GitHub?](https://content.red-badger.com/resources/what-is-git-and-github)
- [What is version control?](https://www.atlassian.com/git/tutorials/what-is-version-control)
-- [What is Git](https://www.atlassian.com/git/tutorials/what-is-git)
+- [What is Git?](https://www.atlassian.com/git/tutorials/what-is-git)
diff --git a/git/intermediate_git/a_deeper_look_at_git.md b/git/intermediate_git/a_deeper_look_at_git.md
index e7ebe5482a1..7dad430b6e4 100644
--- a/git/intermediate_git/a_deeper_look_at_git.md
+++ b/git/intermediate_git/a_deeper_look_at_git.md
@@ -80,10 +80,13 @@ pick 92ad0af Create third file and create fourth file
This would allow us to edit the typo in the `Create send file` commit to be `Create second file`. Perform similar changes in your interactive rebase tool, but don't copy and paste the above code since it won't work. Save and exit the editor, which will allow us to edit the commit with the following instructions:
```bash
-You can amend the commit now, with
- git commit --amend
-Once you're satisfied with your changes, run
- git rebase --continue
+git commit --amend
+```
+
+The command above will allow you to amend the commit. Once you're satisfied with your changes, you can complete the rebase with the following:
+
+```bash
+git rebase --continue
```
So let's edit our commit by typing `git commit --amend`, fixing the typo in the title, and then finishing the rebase by typing `git rebase --continue`. That's all there is to it! Have a look at your handiwork by typing `git log`, and seeing the changed history. It seems simple, but this is a very dangerous tool if misused, so be careful. Most importantly, remember that **if you have to rebase commits in a shared repository, make sure you're doing so for a very good reason that your coworkers are aware of.**
@@ -106,10 +109,10 @@ Rename the commit to `Create first and second file`, then finish the rebase. Tha
Before diving into Remotes, we're going to have a look at a handy Git command called `git reset`. Let's have a look at the commit `Create third file and create fourth file`. At the moment we're using blank files for convenience, but let's say these files contained functionality and the commit was describing too much at once. In that case what we could do is split it up into two smaller commits by, once again, using the interactive `rebase` tool.
-We can open up the tool just like last time, change `pick` to `edit` for the commit we're going to split. But instead, what we're going to do is run `git reset HEAD^`, which resets the commit to the one right before HEAD. This allows us to add the files individually and commit them individually. All together it would look something like this:
+We can open up the tool just like last time, change `pick` to `edit` for the commit we're going to split. But instead, what we're going to do is run `git reset HEAD~`, which resets the commit to the one right before HEAD. This allows us to add the files individually and commit them individually. All together it would look something like this:
```bash
-git reset HEAD^
+git reset HEAD~
git add test3.md && git commit -m 'Create third file'
git add test4.md && git commit -m 'Create fourth file'
```
@@ -148,9 +151,9 @@ You might be feeling overwhelmed at this point, so let's recap what we've learne
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- Explain what it means for branches to be pointers.
-- How can you amend your last commit?
-- What are some different ways to rewrite history?
+- [Explain what it means for branches to be pointers.](https://git-scm.com/book/en/v2/Git-Branching-Branches-in-a-Nutshell)
+- [How can you amend your last commit?](https://git-scm.com/book/en/v2/Git-Basics-Undoing-Things)
+- [What are some different ways to rewrite history?](https://git-scm.com/book/en/v2/Git-Tools-Rewriting-History)
### Additional resources
diff --git a/git/intermediate_git/using_git_in_the_real_world.md b/git/intermediate_git/using_git_in_the_real_world.md
index 7227d853292..bb2473669e4 100644
--- a/git/intermediate_git/using_git_in_the_real_world.md
+++ b/git/intermediate_git/using_git_in_the_real_world.md
@@ -2,7 +2,7 @@
Git basics are very straightforward, but it sometimes feels like a bottomless pit when you find yourself on the wrong side of a confusing error situation. It's doubly frustrating because you think that messing up or trying the wrong solution can lose data. It's actually very hard to "lose" data with Git but it can certainly be hiding somewhere you wouldn't think to look without an experienced dev poking around.
-The thing about Git is that, unless you've got a seriously impressive memory, you can't just learn it by reading about it up front... you need to do it. Find a problem you want to go back and fix, hit an error in your merge, etc. and Google the hell out of it, learning a new Git tactic in the process.
+The thing about Git is that, unless you've got a seriously impressive memory, you can't just master it by reading about it; you need hands-on practice! For example, find a problem you want to fix, run into a merge error, and then dive into Google to learn a new Git technique along the way to help you fix that problem.
To help you out, come back and refer to this lesson again when you're in trouble. We'll first cover a real-world example of a GitHub workflow used on this very project. The Additional Resources section below should also help you find high quality resources for when you need them later on.
@@ -17,7 +17,9 @@ This section contains a general overview of topics that you will learn in this l
Before we dive into workflows, take a minute to remind yourself about good commit messages. You can check the [Commit Messages lesson](https://www.theodinproject.com/lessons/foundations-commit-messages) for a reminder. This is a good time to draw particular attention to [Conventional Commits](https://www.conventionalcommits.org), a standard for commits that is gaining more and more popularity for collaborative projects. It helps to make sure your commit message gives a clear description of its purpose to anyone reading. You may like to implement these going forwards (if you aren't already!), or at least be aware of them for when you read other repos.
-### A Git workflow for open source contribution
+### Assignment
+
+
Let's say you want to contribute to [our curriculum repo](https://github.com/TheOdinProject/curriculum/).
@@ -59,13 +61,15 @@ Note that a `git fetch upstream` followed by a `git merge upstream/some_branch`
1. If you have been following along with the above steps to get familiar with this workflow, you should **stop at this point**. If you have completed an assigned issue, the final step is to submit a pull request to merge your feature branch into the original `upstream` repository's `main` branch. This can be done using GitHub's interface.
1. Shake your moneymaker, you're an OSS contributor!
+
+
### Knowledge check
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
- [What name is typically given for a Git remote that points to a repo that's been forked?](#initial-setup)
- [Can you directly send your changes to a repository that you don't own/have write access to?](#sending-your-pull-request)
-- [What should you do immediately before merging your feature branch into main?](#ongoing-workflow)
+- [What should you do immediately before merging main into your feature branch?](#ongoing-workflow)
### Additional resources
diff --git a/git/intermediate_git/working_with_remotes.md b/git/intermediate_git/working_with_remotes.md
index 109b3a00d6d..a02fa55defa 100644
--- a/git/intermediate_git/working_with_remotes.md
+++ b/git/intermediate_git/working_with_remotes.md
@@ -28,7 +28,7 @@ git log
```
-Huh, that's interesting. We don't see our fourth file on our local system. Let's check our GitHub repository to see if it's there.
+Huh, that's interesting. We can’t find our fourth file on our local system. Let's check our GitHub repository to see if it's there.
Oh no, we just destroyed it! In this scenario, the danger - you could potentially destroy the work of those you're collaborating with! `git push --force` is a **very dangerous command, and it should be used with caution when collaborating with others**. Instead, you can fix your outdated history error by updating your local history using `fetch`, `merge`, and then attempting to `push` again.
@@ -89,7 +89,7 @@ Let's review the dangers we've addressed so far. I know, I know, it's scary stuf
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- [What is a safe way to push history changes to a remote repository?](#force-with-lease)
+- [What is a safe way to forcefully push history changes to a remote repository?](#force-with-lease)
- [What are the dangers of history-changing operations?](#dangers)
- [What are best practices of history-changing operations?](#best-practices)
diff --git a/intermediate_html_css/forms/project_sign_up_form.md b/intermediate_html_css/forms/project_sign_up_form.md
index bd4b49bfb09..cdbc50b9d30 100644
--- a/intermediate_html_css/forms/project_sign_up_form.md
+++ b/intermediate_html_css/forms/project_sign_up_form.md
@@ -25,7 +25,7 @@ This project is intended to give you a chance to flex some of the new items you'
1. The color we've chosen for the 'Create Account' button is similar to tones found in the background image. Specifically, it is `#596D48`.
1. The inputs, by default have a very light border (`#E5E7EB`), but we’ve included 2 variations. For starters, the password inputs should have a red border if they contain an invalid password. This can be handled with the `:invalid` pseudo-class you've learned in the previous lesson.
1. The other variation is the selected input, which should have a blue border and subtle box-shadow. This can be done with the `:focus` pseudo-class you've learned about in an earlier lesson.
-1. Do not worry about making your project look nice on mobile, but DO resize your browser a little bit to make sure that it's not completely broken at different desktop resolutions.
+1. Do not worry about making your project look nice on mobile. Responsive design isn't covered until later in the curriculum.
1. Validating that the password fields match each other requires JavaScript. Using JavaScript to validate forms is covered in a future lesson. For now, just validate each field separately.
diff --git a/intermediate_html_css/grid/introduction_to_grid.md b/intermediate_html_css/grid/introduction_to_grid.md
index 7be658bca3a..c7ee97f5ba1 100644
--- a/intermediate_html_css/grid/introduction_to_grid.md
+++ b/intermediate_html_css/grid/introduction_to_grid.md
@@ -20,19 +20,25 @@ The Flex lessons covered positioning items along the [two flex axes (main and cr
You’ll remember you can line up a nice row of flex items like this:
+
For one-dimensional layouts, Flex offers a convenient tool without having to rely on floats or CSS hacks to align your items properly.
@@ -48,10 +54,13 @@ We know that was a frustrating one, but it's part of the point. While Flexbox al
But setting up a two-dimensional layout of cards would be much easier using CSS Grid:
+
### What is grid?
@@ -68,7 +77,7 @@ While some people thought CSS Grid was here to replace Flexbox, you will learn b
### Knowledge check
-This section contains questions for you to check your understanding of this lesson. If you’re having trouble answering the questions below on your own, review the material above to find the answer.
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
- [How can you use Flex to make a two-dimensional layout?](#a-look-back-at-flex)
- [Why was CSS Grid introduced?](#what-is-grid)
@@ -76,7 +85,7 @@ This section contains questions for you to check your understanding of this less
### Additional resources
-This section contains helpful links to other content. It isn’t required, so consider it supplemental.
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
- Watch [Flexbox vs. CSS Grid — Which is Better?](https://www.youtube.com/watch?v=hs3piaN4b5I) for a visual representation of the use cases for Flexbox vs CSS Grid
- Read CSS Tricks' quick take on the [differences between flex and grid](https://css-tricks.com/quick-whats-the-difference-between-flexbox-and-grid/).
diff --git a/intermediate_html_css/intermediate_css_concepts/advanced_selectors.md b/intermediate_html_css/intermediate_css_concepts/advanced_selectors.md
index a2b8dc75f77..17663665317 100644
--- a/intermediate_html_css/intermediate_css_concepts/advanced_selectors.md
+++ b/intermediate_html_css/intermediate_css_concepts/advanced_selectors.md
@@ -283,7 +283,7 @@ This section contains helpful links to related content. It isn't required, so co
- [Kevin Powell](https://www.youtube.com/kepowob/search?query=pseudo) has a variety of videos on several of these topics if you'd like a deeper dive.
- [The CSS Tricks Almanac](https://css-tricks.com/almanac/selectors/) has a great reference for all pseudo-elements and selectors. It includes examples, extra resources and browser support charts.
-- [W3 Schools](https://www.w3schools.com/cssref/css_selectors.asp) also has a solid, more concise reference list. Includes an interactive selector tool if you'd like to play around with some hands on examples.
+- [W3 Schools](https://www.w3schools.com/cssref/css_selectors.asp) also has a solid, more concise reference list. Includes an interactive selector tool if you'd like to play around with some hands-on examples.
- [The Free Code Camp Selector Cheat Sheet](https://www.freecodecamp.org/news/css-selectors-cheat-sheet/) has a solid summary of some of the most common selectors.
- A nice concise article on the [differences between pseudo-classes and pseudo-elements](https://www.growingwiththeweb.com/2012/08/pseudo-classes-vs-pseudo-elements.html). Also provides a solid summary of the different kinds of selectors.
- [Smashing Magazine on Taming Advanced CSS Selectors](http://coding.smashingmagazine.com/2009/08/17/taming-advanced-css-selectors/)
diff --git a/intermediate_html_css/intermediate_css_concepts/default_styles.md b/intermediate_html_css/intermediate_css_concepts/default_styles.md
index 01a1430a09b..d4d7480ddbb 100644
--- a/intermediate_html_css/intermediate_css_concepts/default_styles.md
+++ b/intermediate_html_css/intermediate_css_concepts/default_styles.md
@@ -11,7 +11,7 @@ This section contains a general overview of topics that you will learn in this l
### What are default styles and where do they come from?
-As you have worked on projects, you likely observed default styles applied to certain elements, such as larger and bolder headings on `h1` elements, and blue, underlined links on `a' elements. There is also a good chance you struggled with things like default margins and padding. These styles are part of the user-agent stylesheets, ensuring basic styling for webpages without CSS. Each browser has its own set of user-agent stylesheets so the default styles do vary slightly between browsers.
+As you have worked on projects, you likely observed default styles applied to certain elements, such as larger and bolder headings on `h1` elements, and blue, underlined links on `a` elements. There is also a good chance you struggled with things like default margins and padding. These styles are part of the user-agent stylesheets, ensuring basic styling for webpages without CSS. Each browser has its own set of user-agent stylesheets so the default styles do vary slightly between browsers.
### What if I don't like the defaults?
diff --git a/intermediate_html_css/intermediate_css_concepts/positioning.md b/intermediate_html_css/intermediate_css_concepts/positioning.md
index 46c56320573..208c7d760ab 100644
--- a/intermediate_html_css/intermediate_css_concepts/positioning.md
+++ b/intermediate_html_css/intermediate_css_concepts/positioning.md
@@ -43,7 +43,7 @@ Fixed elements are also removed from the normal flow of the document and are pos
### Sticky positioning
-Sticky elements will act like normal elements until you scroll past them, then they start behaving like fixed elements. They are also not taken out of the normal flow of the document. It might sound confusing, so check out this [example of sticky positioning](https://codepen.io/theanam/pen/MPLBYy) example might clear things up for you. It's useful for things like section-headings. Remember being able to still see what category you're looking at while scrolling through a shop? This is how it's done!
+Sticky elements will act like normal elements until you scroll past them, then they start behaving like fixed elements. They are also not taken out of the normal flow of the document. It might sound confusing, so check out this [sticky positioning example](https://codepen.io/theanam/pen/MPLBYy) that might clear things up for you. It's useful for things like section-headings. Remember being able to still see what category you're looking at while scrolling through a shop? This is how it's done!
### Assignment
diff --git a/javascript/asynchronous_javascript_and_apis/project_weather_app.md b/javascript/asynchronous_javascript_and_apis/project_weather_app.md
index f09943aecb1..b4bb9cb85cb 100644
--- a/javascript/asynchronous_javascript_and_apis/project_weather_app.md
+++ b/javascript/asynchronous_javascript_and_apis/project_weather_app.md
@@ -1,34 +1,26 @@
### Introduction
-Use everything we've been discussing to create a weather forecast site using the weather API from the previous lesson. You should be able to search for a specific location and toggle displaying the data in Fahrenheit or Celsius.
+Use everything we've been discussing to create a weather forecast site using the Visual Crossing API from previous lessons. You should be able to search for a specific location and toggle displaying the data in Fahrenheit or Celsius.
You should change the look of the page based on the data, maybe by changing the color of the background or by adding images that describe the weather. (You could even use the Giphy API to find appropriate weather-related gifs and display them). Feel free to use promises or async/await in your code, though you should try to become comfortable with both.
-
-
-#### WeatherAPI free tier
-
-After creating your API key from WeatherAPI, you will automatically be moved to the free tier after 14 days. This will limit the forecast data available to you to 3 days. You may want to keep this in mind while making design choices for the forecast display.
-
-
-
### API keys, secrets, and security
Not all APIs are free, and depending on how they're set up, they can cost money per use. This makes them a prime target for people looking to use the API without paying by using **your** API key. They can also be rate-limited, and if someone has access to your API key they can use up all of your uses. One way to prevent this issue is to store your API keys on the server and never send them to the frontend in the first place, this is often done using environment variables and it makes the key available only on the server the code is deployed to.
-When talking about API keys and security you'll often hear "Never trust the client" (client meaning the frontend). Often this means not to trust that data coming *from* the client is valid, but it also means that you cannot trust anything we send *to* the client. Because of this, when you leak an API key, Github will alert you that you have committed an API key publicly. After following this project, and indeed exposing the API key, you may notice that Github will send you this alert. This is totally OK for this project as this API key is publicly available and there is no consequence for exposing it. This is not to say ALL keys are this way. Later during the backend courses you will learn ways to securely deal with these topics.
+When talking about API keys and security you'll often hear "Never trust the client" (client meaning the frontend). Often this means not to trust that data coming *from* the client is valid, but it also means that you cannot trust anything we send *to* the client. Because of this, when you leak an API key, Github will alert you that you have committed an API key publicly. After following this project, and indeed exposing the API key, you may notice that Github will send you this alert. This is totally OK for this project as this API key is publicly available and there is no consequence for exposing it. This is not to say ALL keys are this way. Later during the backend courses you will learn ways to securely deal with these topics.
### Assignment
1. Set up a blank HTML document with the appropriate links to your JavaScript and CSS files.
-2. Write the functions that hit the API. You're going to want functions that can take a location and return the weather data for that location. For now, just `console.log()` the information.
-3. Write the functions that _process_ the JSON data you're getting from the API and return an object with only the data you require for your app.
-4. Set up a form that will let users input their location and will fetch the weather info (still just `console.log()` it).
-5. Display the information on your webpage!
-6. Add any styling you like!
-7. Optional: add a 'loading' component that displays from the time the form is submitted until the information comes back from the API. Use DevTools to test for low-end devices.
-8. Push that baby to github and share your solution below!
+1. Write the functions that hit the API. You're going to want functions that can take a location and return the weather data for that location. For now, just `console.log()` the information.
+1. Write the functions that *process* the JSON data you're getting from the API and return an object with only the data you require for your app.
+1. Set up a form that will let users input their location and will fetch the weather info (still just `console.log()` it).
+1. Display the information on your webpage!
+1. Add any styling you like!
+1. Optional: add a 'loading' component that displays from the time the form is submitted until the information comes back from the API. Use DevTools to test for low-end devices.
+1. Push that baby to github and share your solution below!
diff --git a/javascript/asynchronous_javascript_and_apis/working_with_apis.md b/javascript/asynchronous_javascript_and_apis/working_with_apis.md
index cd8e2c0154d..8082e736673 100644
--- a/javascript/asynchronous_javascript_and_apis/working_with_apis.md
+++ b/javascript/asynchronous_javascript_and_apis/working_with_apis.md
@@ -15,42 +15,52 @@ This section contains a general overview of topics that you will learn in this l
Servers that are created for serving data for external use (in websites or apps) are often referred to as APIs or ['Application Programming Interfaces'](https://www.youtube.com/watch?v=s7wmiS2mSXY).
-There are multiple ways of requesting data from an API, but all of them basically do the same thing. For the most part, APIs are accessed through URLs, and the specifics of how to query these URLs change based on the specific service you are using. For example, WeatherAPI has several types of data that you can request. To get the current weather in a specific location, you can pass in the name of a city (optionally, you can also pass a zip code & even an ip-address!) as a URL query string parameter, like so:
+There are multiple ways of requesting data from an API, but all of them basically do the same thing. For the most part, APIs are accessed through URLs, and the specifics of how to query these URLs change based on the specific service you are using. For example, [Visual Crossing](https://www.visualcrossing.com/weather-api) has a weather API that has several types of data you can request. To get the current weather in a specific location, you can pass in the name of a city into the path of the URL, like so:
```text
-https://api.weatherapi.com/v1/current.json?q=london
+https://weather.visualcrossing.com/VisualCrossingWebServices/rest/services/timeline/london
```
-The specifics for using any API are usually documented on the service's website. Check the [WeatherAPI documentation](https://www.weatherapi.com/docs/). If you haven't already, go ahead and paste the weather URL above, with the city of your choice, into your browser... (we'll wait).
+The specifics for using any API are usually documented on the service's website. Check out the [Visual Crossing API Documentation](https://www.visualcrossing.com/resources/documentation/weather-api/timeline-weather-api/). If you haven't already, go ahead and paste the weather URL above, with the city of your choice, into your browser... (we'll wait).
You'll probably get an error like this:
```text
-{{"error":{"code":1002,"message":"API key is invalid or not provided."}}}
+No API key or session found. Please verify that your API key parameter is correct.
```
-This brings us to another point about APIs. In most cases, you will have to create an account and request an "API key" from the API service before attempting to fetch data from their endpoints (specific URLs that you use to access a particular function or data within the API). Once obtained, an API key will usually have to be included with every data request, such as *another* URL query string parameter:
+This brings us to another point about APIs. In most cases, you will have to create an account and request an "API key" from the API service before attempting to fetch data from their endpoints (specific URLs that you use to access a particular function or data within the API). Once obtained, an API key will usually have to be included with every data request. With Visual Crossing, that's as a query string parameter:
```text
-https://api.weatherapi.com/v1/current.json?key=11111111111111111&q=london
+https://weather.visualcrossing.com/VisualCrossingWebServices/rest/services/timeline/london?key=11111111111111111
```
-As you can imagine, an API key is random and unique to you. As such, services like WeatherAPI can correlate your API key to your requests of their data, including how much and how often you are requesting it.
+As you can imagine, an API key is random and unique to you. As such, services like Visual Crossing can correlate your API key to your requests of their data, including how much and how often you are requesting it.
-On one hand, issuing API keys allows an API service to better track abuse of their systems and data. On the other hand, it can also be a way for those services to mitigate and recuperate operating costs. WeatherAPI, for example, provides not only a free tier but a variety of paid tiers that can cost up to 65 USD/month! After all, running servers costs money, and APIs are no exception. While a single request to an API might cost a fraction of a penny, imagine using that API to create an amazing weather app that gets used all over the world... you could easily have thousands of people accessing that data every minute! The cost to handle that traffic could quickly balloon up to significant sums for the API service.
+Issuing API keys allows an API service to better track abuse of their systems and data. Additionally, it can also be a way for those services to mitigate and recuperate operating costs. Visual Crossing, for example, provides not only a free tier but a variety of paid tiers that can cost up to 150 USD/month! After all, running servers costs money, and APIs are no exception. While a single request to an API might cost a fraction of a penny, imagine using that API to create an amazing weather app that gets used all over the world... you could easily have thousands of people accessing that data every minute! The cost to handle that traffic could quickly balloon up to significant sums for the API service.
-As such, you'll find that most API services, if not all, provide paid tiers that come with the ability to make more frequent requests, or provide access to more information unavailable in lower tiers. For example, WeatherAPI's free plan only allows your app to make a monthly total of 1 million requests and limits the information provided, while the "Business" tier allows up to 10,000,000 requests per month and gives you all of the available information! The free tier also comes with basic hourly and daily forecasting data, but it does not include data for a 30-day forecast ([details about the WeatherAPI tiers](https://www.weatherapi.com/pricing.aspx) if you're interested). So, if your app becomes successful and needs additional features, you'll probably need to pay for a better account.
+As such, you'll find that most API services, if not all, provide paid tiers that come with the ability to make more frequent requests, or provide access to more information unavailable in lower tiers. For example, Visual Crossing's free version allows 1000 calls/requests per day with a limited amount of information (albeit sufficient for a pet project). On the other hand, the Enterprise version provides unlimited API calls per month, has Energy data, Maritime data... all those bells and whistles ([details about Visual Crossing's pricing](https://www.visualcrossing.com/weather-data-editions) if you're curious). So, if your app becomes successful and needs additional features, you'll probably need to pay for a better account.
Because your API key is **your** key to these services and data, securing them is an important habit, especially if you are using a paid tier. There are plenty of bots that crawl GitHub repositories solely for hardcoded/unsecured API keys, allowing bad agents to then access and [utilize the services and data you've paid for](https://web.archive.org/web/20150102022540/http://www.devfactor.net/2014/12/30/2375-amazon-mistake/). In fact, the more eagle-eyed readers may have noticed a problem with the demonstration above: The API key is right there in the URL request. It would not take much for an internet traffic sniffer to pick up on the API key, least of all someone looking over your shoulder!
-At this point in the curriculum, though, this point is largely moot. After all, we're leveraging free access to APIs, and the majority of our apps are only going to be used by us and the people that view our portfolios. Just make a note of the severe limitations of using API keys as demonstrated above for now. The basics of securing and obfuscating API keys from GitHub and from your requests will be covered later in the curriculum.
+At this point in the curriculum, though, this point is largely moot. After all, we're leveraging free access to APIs, and the majority of our apps are only going to be used by us and the people that view our portfolios. Just make a note of the severe limitations of using API keys as demonstrated above for now. Securing API keys requires handling things server-side, and we are only focusing on the frontend concepts here. If you are in the Full Stack JavaScript pathway, the backend will be covered later in the curriculum.
-Back to WeatherAPI. Go ahead and [create an account](https://www.weatherapi.com/signup.aspx) to obtain an API key from their free tier. Once the key has been activated, try making a new request with the city of your choice AND the API key passed in as query string parameters, like the example above. You'll hopefully see a proper response, something like:
+Back to Visual Crossing. Go ahead and [create a free Visual Crossing account](https://www.visualcrossing.com/sign-up) to obtain an API key. You can find your API key in your account profile page. Once you have your key, try making a new request with the city of your choice, and the API key passed in as query string parameters, like the example above. You'll hopefully see a proper response, something like:
```JSON
-{"location":{"name":"London","region":"City of London, Greater London","country":"United Kingdom","lat":51.52,"lon":-0.11,"tz_id":"Europe/London","localtime_epoch":1676482062,"localtime":"2023-02-15 17:27"},"current":{"temp_c":13.0,"temp_f":55.4,"is_day":0,"condition":{"text":"Clear","icon":"//cdn.weatherapi.com/weather/64x64/night/113.png","code":1000},"wind_mph":12.5,"wind_kph":20.2,"wind_degree":210,"wind_dir":"SSW","pressure_mb":1022.0,"pressure_in":30.18,"precip_mm":0.0,"precip_in":0.0,"humidity":58,"cloud":0,"feelslike_c":11.7,"feelslike_f":53.1,"vis_km":10.0,"vis_miles":6.0,"uv":4.0,"gust_mph":12.1,"gust_kph":19.4}}
+{"queryCost":1,"latitude":51.5064,"longitude":-0.12721,"resolvedAddress":"London, England, United Kingdom","address":"london","timezone":"Europe/London","tzoffset":1.0,"description":"Similar temperatures continuing with a chance of rain tomorrow, Tuesday & Thursday.","days":[{"datetime":"2024-07-06","datetimeEpoch":1720220400,"tempmax":61.4,"tempmin":53.1,"temp":57.8,"feelslikemax":61.4,"feelslikemin":53.1,"feelslike":57.8,"dew":51.3,"humidity":79.7,"precip":0.457,"precipprob":100.0,"precipcover":75.0,"preciptype":["rain"],"snow":0.0,"snowdepth":0.0,"windgust":35.3,"windspeed":21.9,"winddir":262.6,"pressure":1001.8,"cloudcover":70.5,"visibility":8.3,"solarradiation":147.5,"solarenergy":12.9,"uvindex":6.0,"severerisk":10.0,"sunrise":"04:52:02","sunriseEpoch":1720237922,"sunset":"21:18:20","sunsetEpoch":1720297100,"moonphase":0.02,"conditions":"Rain, Partially cloudy","description":"Partly cloudy throughout the day with a chance of rain throughout the day.","icon":"rain","stations":["EGWU","EGLL","D5621","EGLC"]}]}
```
+(This preview above is just the tip of the iceberg, the actual response is a lot longer!)
+
+
+
+#### Visual Crossing's query builder
+
+Curious how to construct queries? Check out [Visual Crossing's query builder](https://www.visualcrossing.com/weather/weather-data-services)! This will help you build a query with the region as the only input. Play around with the location and query options. Neat, isn't it? The output uses the "Grid" tab, but our query is in the "API" tab, and we want to learn *how* the query was structured. Click on the "API" tab and it will show the query that led to the grid format you just saw.
+
+
+
Congratulations on making your first API request!
### Fetching data
@@ -99,7 +109,7 @@ fetch('https://url.com/some/url')
In case you've forgotten, scroll back up and look at how you would use XHR to do the same thing. While you're admiring how nice and clean that code is, notice the `.then()` and `.catch()` functions there. Do you remember what those are? (PROMISES!)
-Let's change up our API for this example. We're going to walk through an example using fetch with the [giphy](https://giphy.com/) API to display a random gif on a webpage. The API requires you to sign up and get a free API key, so go ahead and [do that](https://developers.giphy.com/docs/api#quick-start-guide).
+Let's change up our API for this example. We're going to walk through an example using fetch with the [giphy](https://giphy.com/) API to display a random gif on a webpage. The API requires you [sign up to giphy and get a free API key](https://developers.giphy.com/docs/api/#quick-start-guide).
Giphy has several methods for searching and finding GIFs which you can read about in their documentation. Today we're just going to use the 'translate' endpoint because it's the simplest one for our purposes. You can find the appropriate URL in their documentation by scrolling down [to the translate endpoint information from Giphy](https://developers.giphy.com/docs/api/endpoint#translate). What it tells us is that the correct URL is `api.giphy.com/v1/gifs/translate` and that it requires 2 parameters, your `api_key` and a `search term`. If you put it all together correctly (with YOUR API key) you should get something like this:
@@ -214,7 +224,7 @@ Running the file should now log the URL of the image. All that's left to do is s
If all goes well, you should see a new image on the page every time you refresh!
-If you've gotten lost along the way, check out this [jsbin project](http://jsbin.com/canofar/edit?html,output). Besides the glorious styling, this is what your version should look like.
+If you've gotten lost along the way, check out this [jsbin project demonstrating fetching from giphy](http://jsbin.com/canofar/edit?html,output). Besides the glorious styling, this is what your version should look like.
While we are pushing this API key to the frontend, this isn't something you should do with any key that is not free. Keys used on the client-side are considered public knowledge, so caution must be taken with sensitive and non-free keys. Handling keys without pushing them to the frontend will be taught in later sections if you haven't already learned it in the Ruby course.
diff --git a/javascript/computer_science/a_very_brief_intro_to_cs.md b/javascript/computer_science/a_very_brief_intro_to_cs.md
index c046c275c70..7c8ebf67912 100644
--- a/javascript/computer_science/a_very_brief_intro_to_cs.md
+++ b/javascript/computer_science/a_very_brief_intro_to_cs.md
@@ -25,6 +25,7 @@ This section contains a general overview of topics that you will learn in this l
1. Watch [What is an Algorithm?](https://youtu.be/e_WfC8HwVB8) on YouTube for a more structured look at solving problems using algorithms.
1. Read this Quora answer about the [importance of algorithms in web development](https://qr.ae/py3NAc) to get some context for why we're going over this stuff.
1. Watch [What is pseudocode?](https://www.youtube.com/watch?v=Rg-fO7rDsds)
+ 1. Watch the first section (from 0:00 - 7:01) of [Telusko's video on data structures and algorithms](https://www.youtube.com/watch?v=xWLxhF3b5P8) for a quick overview of DSA, and why companies may be interesting in hiring candidates familiar with DSA.
diff --git a/javascript/computer_science/hash_map_data_structure.md b/javascript/computer_science/hash_map_data_structure.md
index 2e0be6e0e09..f894a87ed2e 100644
--- a/javascript/computer_science/hash_map_data_structure.md
+++ b/javascript/computer_science/hash_map_data_structure.md
@@ -216,3 +216,4 @@ This section contains helpful links to related content. It isn't required, so co
- This discussion goes through the [usages of prime numbers in hash functions](https://stackoverflow.com/questions/299304/why-does-javas-hashcode-in-string-use-31-as-a-multiplier/299748).
- The [pigeonhole principle](https://en.wikipedia.org/wiki/Pigeonhole_principle) mathematically guarantees collisions when there are more nodes than boxes.
+- Check out [Hashing](https://samwho.dev/hashing/) if you want to get a better fundamental understanding of hash functions.
diff --git a/javascript/computer_science/project_binary_search_trees.md b/javascript/computer_science/project_binary_search_trees.md
index 90ffb1854e9..62aa3ab9565 100644
--- a/javascript/computer_science/project_binary_search_trees.md
+++ b/javascript/computer_science/project_binary_search_trees.md
@@ -43,9 +43,9 @@ You'll build a balanced BST in this assignment. Do not use duplicate values beca
1. Write a `find(value)` function that returns the node with the given value.
-1. Write a `levelOrder(callback)` function that accepts an optional callback function as its parameter. `levelOrder` should traverse the tree in breadth-first level order and provide each node as an argument to the callback. As a result, the callback will perform an operation on each node following the order in which they are traversed. `levelOrder` may be implemented using either iteration or recursion (try implementing both!). The method should return an array of values if no callback is given as an argument. **Tip:** You will want to use an array acting as a queue to keep track of all the child nodes that you have yet to traverse and to add new ones to the list ([video on level order traversal](https://www.youtube.com/watch?v=86g8jAQug04)).
+1. Write a `levelOrder(callback)` function that accepts a callback function as its parameter. `levelOrder` should traverse the tree in breadth-first level order and call the callback on each node as it traverses, passing the whole node as an argument, similarly to how `Array.prototype.forEach` might work for arrays. `levelOrder` may be implemented using either iteration or recursion (try implementing both!). If no callback function is provided, [throw an Error](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/throw) reporting that a callback is required. **Tip:** You will want to use an array acting as a queue to keep track of all the child nodes that you have yet to traverse and to add new ones to the list ([video on level order traversal](https://www.youtube.com/watch?v=86g8jAQug04)).
-1. Write `inOrder(callback)`, `preOrder(callback)`, and `postOrder(callback)` functions that also accept an optional callback as a parameter. Each of these functions should traverse the tree in their respective depth-first order and yield each node to the provided callback. The functions should return an array of values if no callback is given as an argument.
+1. Write `inOrder(callback)`, `preOrder(callback)`, and `postOrder(callback)` functions that also accept a callback as a parameter. Each of these functions should traverse the tree in their respective depth-first order and pass each node to the provided callback. The functions should throw an Error if no callback is given as an argument, like with `levelOrder`.
1. Write a `height(node)` function that returns the given node's height. Height is defined as the number of edges in the longest path from a given node to a leaf node.
diff --git a/javascript/computer_science/project_linked_lists.md b/javascript/computer_science/project_linked_lists.md
index ad19223dfcd..4944ce95cb8 100644
--- a/javascript/computer_science/project_linked_lists.md
+++ b/javascript/computer_science/project_linked_lists.md
@@ -57,10 +57,39 @@ Build the following functions in your linked list class / factory:
1. `toString` represents your LinkedList objects as strings, so you can print them out and preview them in the console.
The format should be: `( value ) -> ( value ) -> ( value ) -> null`
-### Extra credit
+#### Extra credit
1. `insertAt(value, index)` that inserts a new node with the provided `value` at the given `index`.
1. `removeAt(index)` that removes the node at the given `index`.
**Extra Credit Tip:** When you insert or remove a node, consider how it will affect the existing nodes. Some of the nodes will need their `nextNode` link updated.
+
+#### Test it out
+
+Let's test out the Linked List you made!
+
+1. Create a `main.js` file and make sure it imports your `LinkedList` class or factory. This is where we'll test the list.
+1. Create an instance of your `LinkedList` and populate it with nodes:
+
+ ```javascript
+ // example uses class syntax - adjust as necessary
+ const list = new LinkedList();
+
+ list.append("dog");
+ list.append("cat");
+ list.append("parrot");
+ list.append("hamster");
+ list.append("snake");
+ list.append("turtle");
+ ```
+
+1. Add `console.log(list.toString());` to the end of the file and run it.
+1. If everything is working, the output should be:
+
+ ```text
+ ( dog ) -> ( cat ) -> ( parrot ) -> ( hamster ) -> ( snake ) -> ( turtle ) -> null
+ ```
+
+ Feel free to use different values to test if you like.
+
diff --git a/javascript/computer_science/project_recursion.md b/javascript/computer_science/project_recursion.md
index ce849807212..0cbc02ad801 100644
--- a/javascript/computer_science/project_recursion.md
+++ b/javascript/computer_science/project_recursion.md
@@ -43,7 +43,7 @@ First up create a file and tackle the fibonacci sequence:
1. Now write another function `fibsRec` which solves the same problem recursively.
1. Test both versions of your functions by passing in various lengths as arguments.
-Hopefully you were able to solve the problem with recursion! If you need some help understanding what's going on with this function, there are some additional resources linked at the end of this page.
+Hopefully you were able to solve the problem with recursion! If you need some help understanding what's going on with this function, the "Test it out" section below will help. If you're still a bit confused, there are some additional resources linked at the end of this page.
Once you have a firm grasp on solving Fibonacci with recursion, create a new file and work on a merge sort:
@@ -54,6 +54,19 @@ Tips:
- Think about what the base case is and what behavior is happening again and again and can actually be delegated to someone else (e.g. that same function!).
- It may be helpful to check out the background videos again if you don't quite understand what should be going on.
+#### Test it out
+
+To showcase the recursive effect implemented in your Fibonacci function, do the following:
+
+1. Add the following to the start of the function:
+
+ ```javascript
+ console.log("This was printed recursively");
+ ```
+
+1. Call the function with `8` as the argument.
+1. If the function is implemented correctly, you should see that sentence printed around 8 times (keep in mind that, depending on the way you implemented the function, you may see 7 instead of 8. This isn't a bug! It simply depends on how many times the function is *actually* repeated).
+
### Additional resources
diff --git a/javascript/computer_science/recursive_methods.md b/javascript/computer_science/recursive_methods.md
index 1acffcf8169..5e4d2d1243f 100644
--- a/javascript/computer_science/recursive_methods.md
+++ b/javascript/computer_science/recursive_methods.md
@@ -32,7 +32,7 @@ This section contains a general overview of topics that you will learn in this l
-### Test yourself
+#### Test yourself
@@ -40,7 +40,7 @@ This section contains a general overview of topics that you will learn in this l
-The solution for "Question 6: Search JS object" is incorrect. See the [corrected solution](https://gist.github.com/JoshDevHub/b00125f483d4a1ecc257eaa030916973) after you solve it.
+The solution for "Question 6: Search JS object" is incomplete. See [a more accurate solution for `contains()`](https://gist.github.com/JoshDevHub/b00125f483d4a1ecc257eaa030916973) after you solve it.
diff --git a/javascript/computer_science/time_complexity.md b/javascript/computer_science/time_complexity.md
index 9a4993de6af..08d3d3fd3ab 100644
--- a/javascript/computer_science/time_complexity.md
+++ b/javascript/computer_science/time_complexity.md
@@ -325,7 +325,7 @@ The following questions are an opportunity to reflect on key topics in this less
- [What is Big O?](#what-is-big-o)
- [What are the Big O Notations?](#big-o-notation)
- [Why use Big O?](#why-big-o)
-- [What is Big Omega and why isn't it as useful?](#big--omega-notation)
+- [What is Big Omega and why isn't it as useful?](#big-omega-notation)
### Additional resources
diff --git a/javascript/javascript_in_the_real_world/dynamic_user_interface_interactions.md b/javascript/javascript_in_the_real_world/dynamic_user_interface_interactions.md
index 7db2b9f6a45..a38fa3c5e58 100644
--- a/javascript/javascript_in_the_real_world/dynamic_user_interface_interactions.md
+++ b/javascript/javascript_in_the_real_world/dynamic_user_interface_interactions.md
@@ -19,7 +19,7 @@ A dropdown is something you've most likely encountered on various other websites
Dropdowns are typically comprised of two main parts:
1. A button that toggles the dropdown content's visibility.
-1. The dropdown contents itself.
+1. The dropdown content itself.
The dropdown toggle button should typically only trigger the visibility of the dropdown content on click, while the dropdown contents should typically only contain items that will trigger an action upon clicking them. Actions can include things like "Edit", "Copy", or "Delete", or linking you to another part of the site, such as in a navbar.
@@ -46,7 +46,7 @@ Create an image carousel. It should contain arrows on each side to advance the i
Don't spend too much time worrying about getting your images to display at the correct size -- it's more important to get the carousel rotating.
-1. This one is a little more involved than the last two, so think about how you would set up the different elements within the site.
+1. This one is a little more involved than the previous task, so think about how you would set up the different elements within the site.
1. Set up a very wide `div` which will contain the individual "slides" of each image. By appropriately positioning that `div` inside a container `div` (which acts like a picture frame), you can choose which slide is visible at any given time.
1. Once you have the slider positioned properly, build functions for "next" and "previous" which will advance to the next or previous slide accordingly. The transition *doesn't* need to be smooth or animated. Only make it switch to the correct slide.
1. Set up arrow buttons which activate those functions and play with cycling through the images.
diff --git a/javascript/javascript_in_the_real_world/linting.md b/javascript/javascript_in_the_real_world/linting.md
index 360e709a948..7a25c75d65d 100644
--- a/javascript/javascript_in_the_real_world/linting.md
+++ b/javascript/javascript_in_the_real_world/linting.md
@@ -1,6 +1,6 @@
### Introduction
-Before we dive all the way into the Code, we are going to take a moment to improve your editor setup and overall productivity. Doing this now will make things much easier for you going forward. This lesson will give you some information about code style, and then give you some tools to help you maintain consistent code-style throughout your projects. In some cases it can even help adjust things like indentation for you! We will also introduce template repositories which can save you time setting up projects that share a lot of configuration with other projects.
+Before we dive further into code, we are going to take a moment to improve your editor setup and overall productivity. Doing this now will make things much easier for you going forward. This lesson will give you some information about code style, and then give you some tools to help you maintain consistent code-style throughout your projects. In some cases it can even help adjust things like indentation for you!
### Lesson overview
@@ -8,11 +8,10 @@ This section contains a general overview of topics that you will learn in this l
- Learn about style guides and why they are important.
- Set up a linter and prettier to make your code better.
-- Learn what template repositories are and how to set one up.
### Style guides
-Code style is important! Having a consistent set of style rules for things such as indentation or preferred quote style makes your code more maintainable and easier to read. There are several popular JavaScript style guides on the net that set standards for these types of things, and a little time spent reading them *will* make you a better developer.
+Code style is important! Having a consistent set of style rules for things such as indentation or preferred quote style makes your code more maintainable and easier to read. There are several popular JavaScript style guides on the net that set standards for these types of things, and a little time spent looking through them them *will* make you a better developer. Have a little look at some popular style guides for an idea of what sort of things can be done to improve consistency:
1. The [Airbnb Style Guide](https://github.com/airbnb/javascript) is one of the most popular. It is also very well formatted and easy to read.
1. There is also a [JavaScript style guide used at Google](https://google.github.io/styleguide/jsguide.html).
@@ -20,47 +19,48 @@ Code style is important! Having a consistent set of style rules for things such
### Linting
-The style guides we mentioned above are full of really helpful advice for formatting, organizing and composing your code. But there are a *lot* of rules - it can be difficult to internalize them all. **Linters** are tools that will scan your code with a set of style rules and will report any errors to you that they find. In some cases, they can even auto-fix the errors! The following articles explain in more detail the benefits of using a linter while you code.
-
-1. This article on [JavaScript linters](https://gomakethings.com/javascript-linters/) gets right to the point... start here!
-1. Read this article that goes a little further by discussing exactly [*how* linters do what they do](https://hackernoon.com/how-linting-and-eslint-improve-code-quality-fa83d2469efe).
+The style guides we mentioned above are full of really helpful advice for formatting, organizing and composing your code. But there are a *lot* of rules - it can be difficult to internalize them all. **Linters** are tools that will scan your code with a set of style rules and will report any errors to you that they find. In some cases, they can even auto-fix the errors!
There are multiple options for linting your JavaScript, but the most popular (and most common in the industry) is [ESLint](https://eslint.org/). Getting it installed and the initial set-up is straightforward.
-1. [The official 'Getting Started' page](https://eslint.org/docs/user-guide/getting-started) is a good place to start. It covers installation and basic setup. The basic way to use this tool is to run the `eslint` command in your terminal with a specific file.
+1. [ESLint's official 'Getting Started' page](https://eslint.org/docs/user-guide/getting-started) is a good place to start. It covers installation and basic setup. The basic way to use this tool is to run the `eslint` command in your terminal with a specific file.
- You may also want to look at the [docs on configuring ESLint](https://eslint.org/docs/latest/use/configure/) for a list of options that you can change.
+1. There is an [ESLint extension for Visual Studio Code](https://marketplace.visualstudio.com/items?itemName=dbaeumer.vscode-eslint) with which you can get automatic lint highlighting for your files as you write, without you needing to rerun the `eslint` command every time. If your open workspace also contains an ESLint configuration file at the top level, the extension will automatically use those rules for that project.
-1. There is an [ESLint extension for Visual Studio Code](https://marketplace.visualstudio.com/items?itemName=dbaeumer.vscode-eslint) with which you can get automatic lint highlighting for your files as you write, without you needing to rerun the `eslint` command every time. If your project also contains an ESLint configuration file, the extension will automatically use those rules for that project.
+
-
+#### ESLint v9 and flat config support
-#### A note if your ESLint config is not loading
+The above ESLint doc links take you to the docs for v9, which was released in April 2024. v9 came with a lot of big changes, including forcing all ESLint config files to use the "flat config" format (`eslint.config.(m|c)js`).
+
+Because of these changes, some community plugins like [eslint-config-airbnb-base](https://github.com/airbnb/javascript/tree/master/packages/eslint-config-airbnb-base) (which makes ESLint use Airbnb's ruleset) have not yet been able to release a version that supports v9 or flat config.
-The current version of the extension (2.4.4) will only pick up the workspace folder's ESLint config file, and not a config file for a subdirectory of that workspace. Switching to the pre-release version solves this. You will also need to enable `Eslint: Use Flat Config` in VSCode's settings.
+For the time being, if you wish to use airbnb's style guide with ESLint, you will need to use [ESLint's v8.57 version of the docs](https://eslint.org/docs/v8.x/use/getting-started) and make sure you use one of the older [eslintrc configuration file formats](https://eslint.org/docs/v8.x/use/configure/configuration-files), **not** the newer flat config format.
-
+### Formatters
-#### ESLint v9 and flat config support
+Formatters are *awesome*. They are similar to linters, but serve a slightly different function. Formatters take your JavaScript code and then automatically format it according to a set of rules. Unlike linters, they do not look for style errors, but specifically target the layout of your code, making intelligent decisions about things like spaces, indentation levels and line-breaks.
-The above ESLint doc links take you to the docs for v9, which was released in April 2024. v9 came with a lot of big changes, including forcing all ESLint config files to use the "flat config" format (`eslint.config.(m|c)js`).
+As usual, there are multiple formatters out there. [Prettier](https://prettier.io/) is a very popular choice that is highly opinionated. Besides a few options, most of its formatting decisions are not customizable. Since many of these decisions have been made for you, this reduces the time spent deciding on things like indentation size or spacing, and more time on the problems that actually matter.
-Because of these changes, some community plugins like [eslint-config-airbnb-base](https://github.com/airbnb/javascript/tree/master/packages/eslint-config-airbnb-base) (which makes ESLint use airbnb's ruleset) have not yet been able to release a version that supports v9 or flat config.
+1. Read [Prettier's installation guide](https://prettier.io/docs/en/install.html) for installing it as a dependency in your projects.
+1. Prettier also has [instructions for setting up and configuring the VSCode Prettier extension](https://github.com/prettier/prettier-vscode). This extension allows you to format with Prettier via Visual Studio Code commands or keybinds instead of commands in the terminal.
-For the time being, if you wish to use airbnb's style guide with ESLint, you will need to use [ESLint's v8.57 version of the docs](https://eslint.org/docs/v8.x/use/getting-started) and make sure you use one of the older [eslintrc configuration file formats](https://eslint.org/docs/v8.x/use/configure/configuration-files), **not** the newer flat config format.
+Using Prettier makes coding faster and easier! You don't have to worry about nailing things like indentation, or remembering every semi-colon because prettier will take care of those details for you.
-
+
-### Prettier
+#### Extensions and project dependencies
-Prettier is *awesome*. It is similar to a linter, but serves a slightly different function. Prettier will take your JS code and then automatically format it according to a set of rules. Unlike a linter, it's not looking for style errors, but specifically targeting the layout of your code and making intelligent decisions about things like spaces, indentation levels and line-breaks.
+While the Visual Studio Code extensions for ESLint and Prettier are really convenient, they are local to your machine only. It's good practice to install any linters and formatters as dev dependencies in your projects as well.
-1. Watch this [short intro to Prettier](https://www.youtube.com/watch?v=hkfBvpEfWdA) by its creator.
-1. Go to [Prettier's online playground](https://prettier.io/playground) and give it a test drive. Go ahead and copy/paste some of your old JavaScript code into that editor and see what happens.
-1. Prettier has [instructions for setting up and configuring the VSCode Prettier extension](https://github.com/prettier/prettier-vscode).
+At some point, you may need to work on code with multiple people, and others may not use all of the same tools as you. Therefore, including linters and formatters as dependencies in your project, as well as any rule configuration files, allows everyone access to the same linting and formatting tools and rules.
-Using prettier makes coding faster and easier! You don't have to worry about nailing things like indentation, or remembering every semi-colon because prettier will take care of those details for you.
+Editor extensions can then be used to make linting and formatting more convenient for you. The ESLint and Prettier extensions will recognise and use any rule files in your project. If your open workspace has ESLint installed as a dependency, then the ESLint extension can automatically detect this to apply the right setting for whether to use the flat config or legacy eslintrc format.
+
+
### Using ESLint and Prettier
@@ -70,22 +70,33 @@ For most people using the default ESLint ruleset, there will be no special setup
Some community plugins, such as [eslint-config-airbnb-base](https://github.com/airbnb/javascript/tree/master/packages/eslint-config-airbnb-base), turn on some stylistic rules that may clash with what Prettier formats. If you wish to use a plugin like `eslint-config-airbnb-base` and Prettier together, you will also need to install [eslint-config-prettier](https://github.com/prettier/eslint-config-prettier) which will turn off any of the ESLint rules that clash with Prettier. If you are using the default ESLint ruleset, you will not need this.
-### Template repositories
+
+
+#### Adding setup to template repositories
-With the last few projects, you might have felt that setting up Webpack involved a fair few files and configuration, and that you may have had to look at what you configured before to copy and paste the configuration you want to reuse. You may also have noticed that whenever you create a new repository on Github, there is an option near the top for a `Repository template`.
+Recall [template repositories](https://www.theodinproject.com/lessons/node-path-javascript-revisiting-webpack#template-repositories)? You can include linter and formatter setup in any of your templates to make things quicker and easier in the future!
-This is where template repositories can come very much in handy. Any of your existing repositories can be converted to a template in its settings (right under where you can rename the repository, there is a checkbox for whether the repository is a template or not). If you check this box, congratulations, that's all you need to do! Now when you go to create a new repository, the `Repository template` dropdown will have any templates listed for you to select. Selecting one will mean your new repository will be a copy of the chosen template, not an empty one!
+
-If you find yourself reusing a lot of setup code for multiple projects, you can make a new repository with all of the setup code you need then mark it as a template. Now you can select that template when creating a new project repository to save time getting set up, letting you dive into working on the project itself sooner!
+### Assignment
+
+
+
+1. Read this article that goes more into [the value of linters and how they work](https://hackernoon.com/how-linting-and-eslint-improve-code-quality-fa83d2469efe).
+1. Watch this [short intro to Prettier](https://www.youtube.com/watch?v=hkfBvpEfWdA) by its creator.
+1. Go to [Prettier's online playground](https://prettier.io/playground) and give it a test drive. Go ahead and copy/paste some of your old JavaScript code into that editor and see what happens.
+
+
### Knowledge check
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- [What is linting?](https://gomakethings.com/javascript-linters/)
-- [Which problems can linting prevent?](https://gomakethings.com/javascript-linters/)
-- [Why should you use Prettier?](https://www.youtube.com/watch?v=hkfBvpEfWdA)
-- [What is a template repository?](https://docs.github.com/en/repositories/creating-and-managing-repositories/creating-a-template-repository)
+- [What is linting?](#linting)
+- [Which problems can linting prevent?](https://hackernoon.com/how-linting-and-eslint-improve-code-quality-fa83d2469efe)
+- [What are some of the benefits of using a formatter?](#formatters)
+- [What is Prettier?](https://www.youtube.com/watch?v=hkfBvpEfWdA)
+- [Why should you install linters and/or formatters as dev dependencies in your project?](#extensions-and-project-dependencies)
### Additional resources
diff --git a/javascript/organizing_your_javascript_code/classes.md b/javascript/organizing_your_javascript_code/classes.md
index 82c6a8cd90f..f9838bf1520 100644
--- a/javascript/organizing_your_javascript_code/classes.md
+++ b/javascript/organizing_your_javascript_code/classes.md
@@ -15,8 +15,8 @@ This section contains a general overview of topics that you will learn in this l
- Explain the differences between an object constructor and a class.
- Explain what "getters" and "setters" are.
- Understand what computed names and class fields are.
-- Explain how to implement private class fields and methods.
- Describe function binding.
+- Explain how to implement private class fields and methods.
- Use inheritance with classes.
- Understand why composition is generally preferred to inheritance.
@@ -24,15 +24,13 @@ This section contains a general overview of topics that you will learn in this l
-1. [JavaScript.info's article on Getters and Setters](https://javascript.info/property-accessors) should get you up to speed on "Getters and Setters".
-
-1. [JavaScript.info's primer on class syntax](https://javascript.info/class) is probably just about all you need to start using `class` syntax confidently.
-
-1. [MDN's docs on classes](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes) are, as usual, a great resource for going a little deeper. Look especially at the ['extends' reference page](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/extends), including the ['Mixins' section](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/extends#mix-ins). In some frameworks like React, you can use classes to create your components and make them `extend` the core React component which gives you access to all their built-in functionality (though this is not the only way to create components. This will all be covered later in React section of the course). Classes can also have [private class properties](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/Private_class_fields) that allow you to implement privacy similarly to factory functions.
-
-1. Classes can have [static properties and methods](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/static) which are properties and methods that are accessed on the class itself and not on the instance of a class. This is similar to how some string methods are accessed on the instance of a string itself e.g. `someString.slice(0, 5)` whereas some methods are called on the String constructor directly e.g. `String.fromCharCode(79, 100, 105, 110)`.
-
-1. Read this article covering [opinions regarding the pros and cons of classes](https://medium.com/@rajaraodv/is-class-in-es6-the-new-bad-part-6c4e6fe1ee65). [FunFunFunction's video on Composition over Inheritance](https://www.youtube.com/watch?v=wfMtDGfHWpA) elaborates on the cons mentioned in the article and does a great job of going over the topic.
+1. Read this article covering [opinions regarding the pros and cons of classes](https://medium.com/@rajaraodv/is-class-in-es6-the-new-bad-part-6c4e6fe1ee65).
+1. [JavaScript.info's article on Getters and Setters](https://javascript.info/property-accessors) should get you up to speed on "Getters and Setters", and [JavaScript.info's primer on class syntax](https://javascript.info/class) is probably just about all you need to start using `class` syntax confidently.
+1. [MDN's docs on classes](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes) are, as usual, a great resource for going a little deeper.
+ - Take a look at the ['extends' documentation](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/extends), including the ['Mixins' section](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/extends#mix-ins). In some frameworks like React, you can use classes to create your components and make them `extend` the core React component which gives you access to all their built-in functionality (though this is not the only way to create components. This will all be covered later in the React section of the course).
+ - Classes can also have [private class properties](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/Private_class_fields) that allow you to implement privacy similarly to factory functions.
+ - Classes can have [static properties and methods](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/static) which are properties and methods that are accessed on the class itself and not on the instance of a class. This is similar to how some string methods are accessed on the instance of a string itself e.g. `someString.slice(0, 5)` whereas some methods are called on the String constructor directly e.g. `String.fromCharCode(79, 100, 105, 110)`.
+1. [FunFunFunction's video on Composition over Inheritance](https://www.youtube.com/watch?v=wfMtDGfHWpA) gives a great overview of the topic.
@@ -44,14 +42,14 @@ Go back to your [Library project](https://www.theodinproject.com/lessons/node-pa
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- [Describe the pros and cons of using classes in JavaScript.](https://rajaraodv.medium.com/is-class-in-es6-the-new-bad-part-6c4e6fe1ee65)
+- [What are some of the pros and cons of using classes in JavaScript?](https://rajaraodv.medium.com/is-class-in-es6-the-new-bad-part-6c4e6fe1ee65)
- [How does JavaScript's object creation differ from other object-oriented programming languages?](https://rajaraodv.medium.com/is-class-in-es6-the-new-bad-part-6c4e6fe1ee65#e6b3)
-- [Explain the differences between object constructors and classes.](https://javascript.info/class#not-just-a-syntactic-sugar)
+- [What differences are there between object constructors and classes?](https://javascript.info/class#not-just-a-syntactic-sugar)
- [What are "getters" & "setters"?](https://javascript.info/property-accessors)
-- [Describe computed names and class fields.](https://javascript.info/class)
-- [Describe function binding.](https://javascript.info/class)
-- [Describe static properties.](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/static)
-- [Describe private class features.](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/Private_class_fields)
+- [What are computed names and class fields?](https://javascript.info/class)
+- [What is function binding?](https://javascript.info/class)
+- [What are static properties?](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/static)
+- [What are some private class features?](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/Private_class_fields)
- [How is inheritance used with classes?](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes#inheritance)
- [Why is favoring Composition over Inheritance suggested?](https://www.youtube.com/watch?v=wfMtDGfHWpA)
@@ -60,5 +58,4 @@ The following questions are an opportunity to reflect on key topics in this less
This section contains helpful links to related content. It isn't required, so consider it supplemental.
- Stephen Mayeux has a [Youtube playlist on ES6 classes](https://www.youtube.com/playlist?list=PLtwj5TTsiP7uTKfTQbcmb59mWXosLP_7S) and some of their methods with easy to follow examples.
-
- Here are some more examples that try to illustrate the [benefits of composition over inheritance](https://blog.beezwax.net/composition-over-inheritance-with-javascript-examples).
diff --git a/javascript/organizing_your_javascript_code/es6_modules.md b/javascript/organizing_your_javascript_code/es6_modules.md
index cc3f3d36a8d..3049ee31db5 100644
--- a/javascript/organizing_your_javascript_code/es6_modules.md
+++ b/javascript/organizing_your_javascript_code/es6_modules.md
@@ -1,183 +1,249 @@
### Introduction
-Separate from the **module pattern** that we discussed in an earlier lesson, "modules" is a feature that arrived with ES6. ES6 modules are starting to appear in many code bases around the net and getting them up and running will give us a chance to explore some new parts of the JavaScript ecosystem, so it's going to be a worthy excursion!
+We've learned about the **module pattern** in a previous lesson and played around with using them to help organize our variables and functions. At some point in the last few projects, you may have even wondered, "How would we manage more complex projects? Files would get too long! It would be great if we could split our code up into multiple files for organization!". Using multiple files would be extremely handy for this exact reason.
-Don't be fooled! We're going to cover much more than just the new module syntax in this lesson! Before we can really *use* these modules, we're going to have to learn about **npm** and **webpack**, which are both topics that will be *very* useful to you even beyond this lesson. In the end, the modules themselves are simple to implement, so we're going to take this chance to learn about a few other things.
+While the module pattern used to play a big part in helping us manage this, the release of ES6 (sometimes referred to as ES2015) gave us actual "modules" and thus they are often referred to as "ES6 modules" or "ESM".
### Lesson overview
This section contains a general overview of topics that you will learn in this lesson.
-- Explain what npm is and where it was commonly used before being adopted on the frontend.
-- Describe what `npm init` does and what `package.json` is.
-- Know how to install packages using npm.
-- Describe what a JavaScript module bundler like webpack is.
-- Explain what the concepts "entry" and "output" mean in relation to webpack.
-- Briefly explain what a development dependency is.
-- Explain what "transpiling code" means and how it relates to front-end development.
-- Briefly describe what a task runner is and how it's used in front-end development.
-- Describe how to write an npm automation script.
-- Explain one of the main benefits of writing code in modules.
-- Explain "named" exports and "default" exports.
+- Explain what ES6 modules are and how to import and export from them.
+- Describe the difference between default and named exports.
+- Explain the main differences between CommonJS modules and ES6 modules.
-### The history of JavaScript
+### Before ES6 modules: The global scope problem
-Why do we even need or want this stuff? What do you gain from all of this added complexity? These are good questions... with good answers.
+
-- Read this little [history lesson on JavaScript, package managers, and bundling](https://peterxjang.com/blog/modern-javascript-explained-for-dinosaurs.html). It's long, but it puts what we're doing here in great perspective. This article is a bit older, and those who have coded along with the example have frequently run into issues, so we don't suggest that you code along (you'll be following along with the official Webpack documentation later). Nevertheless, this article is extremely important conceptually and really clarifies the 'WHY' of the rest of this lesson.
+Even though `let`/`const` and arrow functions were not around before ES6, we will still use them in our pre-ES6 examples. They won't change how things work regarding the global scope and the module pattern, which is the main focus of this section.
-### npm
+
-The **node package manager** is a command-line tool that gives you access to a gigantic repository of plugins, libraries and tools. If you have done our Fundamentals course, you will probably have encountered it when you [installed the Jest testing framework](https://github.com/TheOdinProject/javascript-exercises#how-to-use-these-exercises) to do our exercises.
+Let's say we have two scripts, `one.js` and `two.js`, and we link them in our HTML as separate scripts.
-Read through the npm links below but don't worry about running any of the commands on your computer. This section is about growing your awareness of npm. You will have an opportunity to use what you learn here in upcoming projects.
+```html
+
+
+```
+
+```javascript
+// one.js
+const greeting = "Hello, Odinite!";
+```
+
+```javascript
+// two.js
+console.log(greeting);
+```
+
+When we open the HTML, we see `"Hello, Odinite!"` getting logged to the console, even though `greeting` was never defined in `two.js`! That's because the two scripts were loaded one after the other into the same global scope, as if we wrote only one file with the two lines in that order. If we put the `two.js` script tag first, we would instead get an error that `greeting is not defined`, as it would try to do the console log before we define the variable.
-1. Take a couple minutes to read the [About npm](https://docs.npmjs.com/getting-started/what-is-npm) page - a great introduction to npm.
-1. This tutorial teaches you [how to install packages with npm](https://docs.npmjs.com/downloading-and-installing-packages-locally).
-1. This tutorial covers [the `package.json` file](https://docs.npmjs.com/creating-a-package-json-file), which you can use to manage your project's dependencies.
-1. Read this article on [development dependencies](https://dev.to/moimikey/demystifying-devdependencies-and-dependencies-5ege) to learn what they are and how to use them. Note that when the author says "all required dependencies are pulled together and bundled", they are referring to any dependencies needed at runtime. Dependencies like Webpack (which we will cover shortly) are only used during development to build apps and so are not required at runtime.
-1. If you run into trouble at any point you can check out [the official npm docs](https://docs.npmjs.com/) for more tutorials and documentation.
+This means that even if we use multiple JavaScript files, they will still end up sharing the same global scope. Our top-level variables are not safe!
-### Yarn?
+Before ESM, we could wrap some things in an IIFE, which would cause it to run just the same, but now any variables inside them are scoped to that function and not globally.
-At some point, you will probably run into [Yarn](https://yarnpkg.com/en/) - a replacement for the default `npm`. For the most part, it does the same things, though it *does* have a few more features. Recent versions of `npm` have incorporated some of the best features of Yarn, so using it won't offer you any real advantages at this point in your career. It *is* a fine package manager, however, and may be worth your consideration in the future.
+```javascript
+// one.js
+(() => {
+ const greeting = "Hello, Odinite!";
+})();
+```
-### Webpack and bundlers
+Now, we get an error in the console that `greeting is not defined`, because there is no global variable called `greeting` for us to log! But what if we wanted only *some* things to be exposed to other files? We can return those things from our IIFE into the global scope and keep the other things private!
-So far, your projects have had relatively basic file structures. As project complexity grows, so too will the benefits of well-organized code. A project consisting of a single, long file with lots of code can be made easier to navigate and maintain by being broken down into multiple smaller files (modules). Further benefits of writing code in modules will come below when we introduce ES6 modules.
+```javascript
+// one.js
+const greeting = (() => {
+ const greetingString = "Hello, Odinite!";
+ const farewellString = "Bye bye, Odinite!";
+ return greetingString;
+})();
+```
-But there's a problem! The browser would need to make a separate HTTP request for each file. The more files you have, the more costly this becomes, particularly on slower networks and would only increase if you also imported third-party libraries into your app.
+Now, the global variable `greeting` will contain `"Hello, Odinite!"` and so our code from `two.js` successfully logs this to the console. However, our private `farewellString` variable is not global, so that cannot be accessed anywhere in `two.js`. Through this, we are able to choose what to expose from one file to be made available to all files that follow it! This is why IIFEs were often called the "module pattern", because they allowed us to write modular code across multiple files before we were given "real modules".
-What if we had a way to write multiple files and/or import multiple third-party libraries but eventually combine them all into fewer files at the end so the browser did not have to make so many requests?
+But now, with ESM, we no longer need to use IIFEs for this specific purpose.
-Enter bundlers. Give a bundler a starting file (an entry point) and it will build a dependency graph of the modules and dependencies starting from that file, then combine them into a single output file. Provide multiple entry points and it will do the same for each separately. You can read more about [why we use bundlers and how they work](https://snipcart.com/blog/javascript-module-bundler) in this short article.
+### ES6 modules
-Webpack is one such tool for bundling modules. There is a lot of talk across the net about how difficult and complex it is to set up and use, but at the moment our needs are few and the setup is basic enough. In fact, you can see an example of getting it up and running on the [front page of the Webpack website](https://webpack.js.org/).
+With ESM, we have a little more control over things. Each file has its own private scope by default, and not only can we choose what things we export from that file, we can also choose what things we import into other files. So just because we export something, it doesn't mean it's automatically available elsewhere; it will only be available in another file if we explicitly import it there. Lots of control!
-Webpack *is* a very powerful tool, and with that power comes a decent amount of complexity. Don't let it scare you off! The basic configuration is not difficult and proficiency with webpack looks *amazing* on resumes.
+
-To get us started, we are going to refer to the official documentation.
+#### Module scope is not the global scope
-1. Code along with all of the steps of [Webpack's "Getting Started" tutorial](https://webpack.js.org/guides/getting-started/).
+When using ESM, each module has its own private scope, where we use import/export to communicate between files. A top-level variable in a module will not be accessible in the global scope.
-Let's discuss what's going on there. After installing webpack using npm, we set up a project that required an external library ([Lodash](https://lodash.com/)) using a `
+```
+
+Why is `two.js` our entry point? Well, in our above examples, `two.js` imports variables from `one.js`, meaning `two.js` depends on `one.js`, so we have the following **dependency graph**:
+
+```text
+importer depends on exporter
+two.js <-------------- one.js
+```
+
+When we load `two.js` as a module, the browser will see that it depends on `one.js` and load the code from that file as well. If we instead used `one.js` as our entry point, the browser would see that it does not depend on any other files, and so would do nothing else. Our code from `two.js` would not be used, and nothing would get logged!
+
+If we had another file, `three.js`, that exported something and `two.js` imported from it, then `two.js` would still be our entry point, now depending on both `one.js` and `three.js`.
+
+```text
+two.js <-------------- one.js
+ └------- three.js
+```
+
+Or perhaps instead of `two.js`, `one.js` imports from `three.js`. In which case, `two.js` would still be our entry point and depend on `three.js` indirectly through `one.js`.
+
+```text
+two.js <-------------- one.js <-------------- three.js
```
-Using this pattern gives you the freedom to only import the functions you need in the various files of your program. So it's perfectly fine to only import `functionOne` if that's the only one you need.
+Note that we only needed the one script tag, as the browser will handle the additional file dependencies for us. We also did not need to add the `defer` attribute, as `type="module"` will automatically defer script execution for us.
+
+If you had coded along with the IIFE example at the start of the lesson, try rewriting the JavaScript to use `import` and `export`, and link only the entry point as a module script.
+
+### CommonJS
+
+Along the way, you may have bumped into something called CommonJS (CJS), which uses syntax like `require` and `module.exports` instead of `import` and `export`. You may remember seeing this in our JavaScript exercises in the Foundations course (you've come a long way)! This is a module system that was designed for use with Node.js that works a little differently than ESM, and is not something that browsers will be able to understand.
+
+CJS is still used quite a lot in Node.js code, though in recent years, ESM in Node.js has been gaining popularity. For the time being, we are focused on writing code to run in the browser, so we will be spending time with ESM. If you are taking the Full Stack JavaScript pathway, then we will cover CJS in more detail later in the Node.js course.
+
+### Assignment
+
+
+
+1. As per usual, you can learn most about JavaScript keywords and concepts from the MDN docs, so check out the [docs on export](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/export) and [docs on import](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/import). There are little extras about them we have not covered in this lesson, such as aliases and namespace imports.
+
+
### Knowledge check
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- [Explain what npm is and where it was commonly used before being adopted on the frontend.](https://peterxjang.com/blog/modern-javascript-explained-for-dinosaurs.html)
-- [Describe what `npm init` does and what `package.json` is.](https://docs.npmjs.com/creating-a-package-json-file)
-- [Know how to install packages using npm.](https://docs.npmjs.com/downloading-and-installing-packages-locally)
-- [Describe what a JavaScript module bundler like webpack is.](https://peterxjang.com/blog/modern-javascript-explained-for-dinosaurs.html)
-- [Explain what the concepts "entry" and "output" mean in relation to webpack.](#webpack-knowledge-check)
-- [Briefly explain what a development dependency is.](https://dev.to/moimikey/demystifying-devdependencies-and-dependencies-5ege)
-- [Explain what "transpiling code" means and how it relates to frontend development.](https://peterxjang.com/blog/modern-javascript-explained-for-dinosaurs.html)
-- [Briefly describe what a task runner is and how it's used in frontend development.](https://peterxjang.com/blog/modern-javascript-explained-for-dinosaurs.html)
-- [Describe how to write an npm automation script.](https://peterxjang.com/blog/modern-javascript-explained-for-dinosaurs.html)
-- [Explain one of the main benefits of writing code in modules.](#module-knowledge-check)
-- [Explain "named exports" and "default exports".](#exports-knowledge-check)
+- [Before ES6 modules, how would you privatize a variable from being accessible in other files?](#before-es6-modules-the-global-scope-problem)
+- [Before ES6 modules, how would you expose variables to be accessible in later files?](#before-es6-modules-the-global-scope-problem)
+- [What are some benefits of writing code in modules?](#introduction)
+- [What is the difference between default and named exports?](#default-exports)
+- [What is an entry point?](#entry-points)
+- [How do you link a module script in HTML?](#entry-points)
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- Watch this video about [ES6 Modules by Web Dev Simplified](https://youtu.be/cRHQNNcYf6s) if you find video lessons easier to absorb. It covers the same topics as discussed in this lesson.
-- Here is [a brief comparison of CommonJS modules and ES6 modules](https://blog.logrocket.com/commonjs-vs-es-modules-node-js/).
+- This video on [ES6 Modules by Web Dev Simplified](https://youtu.be/cRHQNNcYf6s) summarizes much of the ESM topics discussed in this lesson. At the end, he mentions `nomodule` and support for older browsers that were unable to support ESM. Nowadays, this is not a concern as ESM is supported by basically every browser in common use.
+- Here is a [brief comparison of CommonJS modules and ES6 modules](https://blog.logrocket.com/commonjs-vs-es-modules-node-js/).
diff --git a/javascript/organizing_your_javascript_code/factory_functions_and_module_pattern.md b/javascript/organizing_your_javascript_code/factory_functions_and_module_pattern.md
index b5c05481bb8..92541e04f69 100644
--- a/javascript/organizing_your_javascript_code/factory_functions_and_module_pattern.md
+++ b/javascript/organizing_your_javascript_code/factory_functions_and_module_pattern.md
@@ -82,7 +82,7 @@ Now, while it may sound good at first glance, you may already be raising your ey
Functions in JavaScript form closures. A closure refers to the combination of a function and the **surrounding state** in which the function was declared. This surrounding state, also called its **lexical environment**, consists of any local variables that were in scope at the time the closure was made. Here, `add5` is a reference to the `resulting` function, created when the `makeAdding` function is executed, thus it has access to the lexical environment of the `resulting` function, which contains the `first` variable, making it available for use.
-This is a **crucial** behavior of functions - allowing us to associate data with functions and manipulate that data anywhere outside of the enclosing function. If you're still confused, take a small detour to examine the [second question under the Knowledge check section](#knowledge-check) - no need to read the entire thing for now, anything from "Emulating private methods with closures" onward will be discussed later in this lesson.
+This is a **crucial** behavior of functions - allowing us to associate data with functions and manipulate that data anywhere outside of the enclosing function. If you're still confused, read the [MDN documentation on Closures](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Closures#closure), but only the sections "Lexical scoping", "Closure" and "Practical closures". The other sections refer to concepts that will be discussed later in this lesson.
### So, what's wrong with constructors?
@@ -224,7 +224,7 @@ function createPlayer (name, level) {
}
```
-### The module pattern - IIFEs
+### The module pattern: IIFEs
@@ -274,14 +274,14 @@ Take the calculator example into consideration. It's very easy to imagine a scen
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- [Explain how scope works in JavaScript.](https://wesbos.com/javascript-scoping)
-- [Explain what closures are and how they help in creating private variables.](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Closures#closure)
-- [Describe the common issues that you can face when working with constructors.](#so-whats-wrong-with-constructors)
-- [Describe private variables in factory functions and how they can be useful.](#private-variables-and-functions)
-- [Describe how we can implement prototypal inheritance with factory functions.](#prototypal-inheritance-with-factories)
-- [Explain how the module pattern works.](https://dev.to/tomekbuszewski/module-pattern-in-javascript-56jm)
-- [Describe IIFEs and what they stand for.](http://adripofjavascript.com/blog/drips/an-introduction-to-iffes-immediately-invoked-function-expressions.html)
-- [Explain the concept of namespacing and how factory functions help with encapsulation.](#encapsulating-with-the-module-pattern)
+- [How does scope work in JavaScript?](#scoopfuls-of-scopes)
+- [What are closures and how do they help in creating private variables?](#closures-arent-scary)
+- [What common issues can you can face when working with constructors?](#so-whats-wrong-with-constructors)
+- [What are private variables in factory functions and how can they be useful?](#private-variables-and-functions)
+- [How can we implement prototypal inheritance with factory functions?](#prototypal-inheritance-with-factories)
+- [How does the module pattern work?](https://dev.to/tomekbuszewski/module-pattern-in-javascript-56jm)
+- [What does IIFE stand for and what are they?](#the-module-pattern-iifes)
+- [What is the concept of namespacing and how do factory functions help with encapsulation?](#encapsulating-with-the-module-pattern)
### Additional resources
diff --git a/javascript/organizing_your_javascript_code/npm.md b/javascript/organizing_your_javascript_code/npm.md
new file mode 100644
index 00000000000..f85775b8d64
--- /dev/null
+++ b/javascript/organizing_your_javascript_code/npm.md
@@ -0,0 +1,73 @@
+### Introduction
+
+In the previous lesson, we learned about ES6 modules and the syntax used for importing and exporting things between modules. As we build bigger, more complex applications, we may not want to write *everything* ourselves. We may want to import third party code to handle some things for us, which can range from helper functions other people wrote, to entire frameworks for us to build our application within. To find and import these third party packages with ease, we need the help of a package manager such as npm!
+
+### Lesson overview
+
+This section contains a general overview of topics that you will learn in this lesson.
+
+- Understand what npm is.
+- Understand the purpose of the `package.json` file.
+
+### npm
+
+**npm** (no capitals!) is a package manager - a gigantic repository of plugins, libraries, and other tools, which provides us with a command-line tool we can use to install these tools (that we call "packages") in our applications. We will then have all our installed packages' code locally, which we can import into our own files. We could even publish our own code to npm!
+
+You may recall installing npm in the Foundations course in order to install the Jest testing framework to do the JavaScript exercises. Funnily enough, [npm does not stand for "Node Package Manager"](https://www.npmjs.com/package/npm#is-npm-an-acronym-for-node-package-manager), though you will often see it referred to as such.
+
+If you are in the Full Stack Ruby on Rails pathway, you will have already been introduced to Yarn, another JavaScript package manager. For this course, we will be using npm.
+
+As our applications get more complex and more and more files are needed (whether they are our own files or files from packages we've installed and imported), managing many of these dependencies can become rather troublesome, especially when packages get updated. This can get even more troublesome when we consider that we may end up sending *many* JavaScript files to the browser to download. In the next lesson, we will introduce bundlers, tools that lets us write multiple files that are better for us to work with, then bundle them together into fewer smaller files which will ultimately be sent to the browser instead.
+
+### package.json
+
+npm revolves around a file called `package.json`. It's a JSON file containing information about our project, such as its name or any dependencies and their version numbers. npm can read this file and do things such as install all of the listed dependencies with the correct versions, and running commands that you've set as an npm script (we will cover npm scripts in a later lesson).
+
+For example, here is the `package.json` file for The Odin Project's curriculum repo that houses all of the lesson files (including this lesson you are doing right now):
+
+```json
+{
+ "name": "curriculum",
+ "version": "1.0.0",
+ "description": "[The Odin Project](https://www.theodinproject.com/) (TOP) is an open-source curriculum for learning full-stack web development. Our curriculum is divided into distinct courses, each covering the subject language in depth. Each course contains a listing of lessons interspersed with multiple projects. These projects give users the opportunity to practice what they are learning, thereby reinforcing and solidifying the theoretical knowledge learned in the lessons. Completed projects may then be included in the user's portfolio.",
+ "scripts": {
+ "lint:lesson": "markdownlint-cli2 --config lesson.markdownlint-cli2.jsonc",
+ "lint:project": "markdownlint-cli2 --config project.markdownlint-cli2.jsonc",
+ "fix:lesson": "markdownlint-cli2 --fix --config lesson.markdownlint-cli2.jsonc",
+ "fix:project": "markdownlint-cli2 --fix --config project.markdownlint-cli2.jsonc"
+ },
+ "license": "CC BY-NC-SA 4.0",
+ "devDependencies": {
+ "markdownlint-cli2": "^0.12.1"
+ }
+}
+```
+
+There's a lot of stuff here and we don't need to understand it all yet. The point is that if you were to clone the curriculum repo, if you ran `npm install`, npm would read this `package.json` file and see that it needs to install the `markdownlint-cli2` package. Once this package is installed, you'll be able to run any of the four npm scripts that use that package. The curriculum repo itself does not actually contain the code for the `markdownlint-cli2` package, as anyone cloning the repo can just run `npm install` to let npm grab the code for them.
+
+In our own projects, as we use npm to install new packages (or uninstall any!), it will automatically update our `package.json` with any new details. We will see this in action in the next lesson when we introduce module bundling using a package called Webpack.
+
+### Assignment
+
+
+
+1. Read a bit more about npm, packages, and dependencies:
+ 1. [Installing packages with npm](https://docs.npmjs.com/downloading-and-installing-packages-locally).
+ 1. Read about [the `package.json` file](https://docs.npmjs.com/creating-a-package-json-file), the file that stores much of the information for our application.
+ 1. Any packages we install are called "dependencies", but if any packages are only used during the development process and their code is not needed for the user-facing app (such as the Jest testing framework), we call them [development dependencies](https://dev.to/mshertzberg/demystifying-devdependencies-and-dependencies-5ege).
+1. Here is a great little [history lesson about JavaScript and managing packages across multiple files](https://peterxjang.com/blog/modern-javascript-explained-for-dinosaurs.html). Only read up to "Using a JavaScript module bundler (webpack)", as we will cover bundlers and webpack in the next lesson.
+
+
+
+### Knowledge check
+
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
+
+- [What is npm?](#npm)
+- [What file does npm use that contains all information about dependencies?](https://docs.npmjs.com/creating-a-package-json-file)
+
+### Additional resources
+
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
+
+- It looks like this lesson doesn't have any additional resources yet. Help us expand this section by contributing to our curriculum.
diff --git a/javascript/organizing_your_javascript_code/objects_and_object_constructors.md b/javascript/organizing_your_javascript_code/objects_and_object_constructors.md
index a9ec4962035..5ebcd9af129 100644
--- a/javascript/organizing_your_javascript_code/objects_and_object_constructors.md
+++ b/javascript/organizing_your_javascript_code/objects_and_object_constructors.md
@@ -358,7 +358,8 @@ If we had used `Object.setPrototypeOf()` in this example, then we could safely e
1. To go a bit deeper into both the chain and inheritance, spend some time with [JavaScript.Info's article on Prototypal Inheritance](http://javascript.info/prototype-inheritance). As usual, doing the exercises at the end will help cement this knowledge in your mind. Don't skip them! Important note: This article makes heavy use of `__proto__` which is not generally recommended. The concepts here are what we're looking for at the moment. We will soon learn another method or two for setting the prototype.
1. You might have noticed us using the `this` keyword in object constructors and prototype methods in the examples above.
- 1. [Dmitri Pavlutin's article on the `this` keyword](https://dmitripavlutin.com/gentle-explanation-of-this-in-javascript/) is very comprehensive and covers how `this` changes in various situations. You should have a solid understanding of the concept after reading it. Pay special attention to the pitfalls mentioned in each section.
+ 1. [JavaScript Tutorial's article on the `this` keyword](https://www.javascripttutorial.net/javascript-this/) covers how `this` changes in various situations. Pay special attention to the pitfalls mentioned in each section.
+1. Read the article [[[Prototype]] vs __proto__ vs .prototype in Javascript](https://medium.com/@eamonocallaghan/prototype-vs-proto-vs-prototype-in-javascript-6758cadcbae8)
@@ -366,22 +367,17 @@ If we had used `Object.setPrototypeOf()` in this example, then we could safely e
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- [Write an object constructor and instantiate the object.](#object-constructors)
-- [Describe what a prototype is and how it can be used.](#the-prototype)
-- [Explain prototypal inheritance.](https://javascript.info/prototype-inheritance)
-- [Understand the basic do's and don't's of prototypal inheritance.](#recommended-method-for-prototypal-inheritance)
-- [How does `this` behave in different situations?](https://dmitripavlutin.com/gentle-explanation-of-this-in-javascript/)
+- [How do you write an object constructor and instantiate the object?](#object-constructors)
+- [What is a prototype and how can it be used?](#the-prototype)
+- [What is prototypal inheritance?](https://javascript.info/prototype-inheritance)
+- [What are the basic do's and don't's of prototypal inheritance?](#recommended-method-for-prototypal-inheritance)
+- [How does `this` behave in different situations?](https://www.javascripttutorial.net/javascript-this/)
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- Lydia Hallie has a [visual article on prototypal inheritance](https://dev.to/lydiahallie/javascript-visualized-prototypal-inheritance-47co) and [prototype inheritance by Programming With Avelx](https://www.youtube.com/watch?v=sOrtAjyk4lQ) explains the Prototype concept with graphics and beginner friendly language. Try using these resources if you want another perspective to understand the concept.
-- This [`Object.create` video by FunFunFunction](https://www.youtube.com/watch?v=CDFN1VatiJA) explains the method with great details about it, he walks through what it is, why `Object.create` exists in JavaScript, and how to use `Object.create`. Also you can check this [`Object.create` method video by techsith](https://www.youtube.com/watch?v=MACDGu96wrA) for another point of view on extending objects.
-- [The Principles of Object-Oriented JavaScript](https://www.amazon.com/Principles-Object-Oriented-JavaScript-Nicholas-Zakas/dp/1593275404) book by
-Nicholas C. Zakas is really great to understand OOP in JavaScript, which explains concepts in-depth, which explores JavaScript's object-oriented nature, revealing the language's unique implementation of inheritance and other key characteristics, it's not free but it's very valuable.
+- This [`Object.create` method video by techsith](https://www.youtube.com/watch?v=MACDGu96wrA) provides another point of view on how to use `Object.create` to extend objects by setting the prototype.
- The first answer on this StackOverflow question regarding [defining methods via the prototype vs in the constructor](https://stackoverflow.com/questions/9772307/declaring-javascript-object-method-in-constructor-function-vs-in-prototype/9772864#9772864) helps explain when you might want to use one over the other.
-- [A Beginner’s Guide to JavaScript’s Prototype](https://medium.com/free-code-camp/a-beginners-guide-to-javascript-s-prototype-9c049fe7b34) and [JavaScript Inheritance and the Prototype Chain](https://medium.com/free-code-camp/javascript-inheritance-and-the-prototype-chain-d4298619bdae) from Tyler Mcginnis has great examples to help you understand Prototype and Prototype Chain better from the beginner's perspective.
-- This video from Akshay Saini is an easy way to understand the [concept of Prototype, Prototype Chain and prototypal inheritance.](https://www.youtube.com/watch?v=wstwjQ1yqWQ).
- [Interactive Scrim on objects and object constructors.](https://scrimba.com/scrim/co2624f87981575448091d5a2)
- Check out this video explanation on the [`this` keyword from DevSage](https://www.youtube.com/watch?v=cwChC4BQF0Q) that gives a different perspective on how its context changes, as well as scenarios in which `this` behaves unexpectedly.
diff --git a/javascript/organizing_your_javascript_code/oop_principles.md b/javascript/organizing_your_javascript_code/oop_principles.md
index e7912c59f0b..851cd1eb871 100644
--- a/javascript/organizing_your_javascript_code/oop_principles.md
+++ b/javascript/organizing_your_javascript_code/oop_principles.md
@@ -67,8 +67,8 @@ This one is related pretty strongly to 'Single Responsibility' but takes a diffe
-1. The following article mentions the acronym **SOLID** before going on to talk about Single Responsibility. Single Responsibility is definitely the most relevant of the 5. Feel free to dig into the rest of the SOLID principles if you like, but pay special attention to Single Responsibility.
- 1. [SOLID principle #1: Single responsibility (JavaScript)](https://duncan-mcardle.medium.com/solid-principle-1-single-responsibility-javascript-5d9ce2c6f4a5) has links to other very brief articles that cover the rest of 'SOLID'. They're optional, but recommended nonetheless. **Note**: this article riffs off what the SOLID videos in the next link goes in-depth on.
+1. The following article and videos mention the acronym **SOLID** before going on to talk about Single Responsibility. Single Responsibility is definitely the most relevant of the 5, but feel free to dig into the rest of the SOLID principles, if you like.
+ 1. Read this article about [SOLID principle #1: Single responsibility (JavaScript)](https://duncan-mcardle.medium.com/solid-principle-1-single-responsibility-javascript-5d9ce2c6f4a5). It simplifies that which is covered in more detail in the SOLID videos below.
1. Watch [The SOLID Design Principles by WDS](https://www.youtube.com/playlist?list=PLZlA0Gpn_vH9kocFX7R7BAe_CvvOCO_p9) to see code examples for each principle.
1. [How to Write Highly Scalable and Maintainable JavaScript: Coupling](https://web.archive.org/web/20200810210808/https://medium.com/@alexcastrounis/how-to-write-highly-scalable-and-maintainable-javascript-coupling-c860787dbdd4) explains loosely coupled objects pretty well.
@@ -78,16 +78,14 @@ This one is related pretty strongly to 'Single Responsibility' but takes a diffe
The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- [Explain the "Single Responsibility Principle".](#single-responsibility)
-- [Briefly explain the additional SOLID principles.](https://medium.com/@cramirez92/s-o-l-i-d-the-first-5-priciples-of-object-oriented-design-with-javascript-790f6ac9b9fa)
-- [Explain what "tightly coupled" objects are and why we want to avoid them.](https://web.archive.org/web/20200810210808/https://medium.com/@alexcastrounis/how-to-write-highly-scalable-and-maintainable-javascript-coupling-c860787dbdd4)
+- [What is the "Single Responsibility Principle"?](#single-responsibility)
+- [What are the additional SOLID principles?](https://duncan-mcardle.medium.com/solid-principle-1-single-responsibility-javascript-5d9ce2c6f4a5)
+- [What are "tightly coupled" objects and why do we want to avoid them?](https://web.archive.org/web/20200810210808/https://medium.com/@alexcastrounis/how-to-write-highly-scalable-and-maintainable-javascript-coupling-c860787dbdd4)
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
- The best book we've ever read on the subject of loose coupling is [Practical Object-Oriented Design In Ruby](http://www.poodr.com/). Unfortunately, it is not free... and not JavaScript. We feel confident in recommending it anyway. If you don't know Ruby, it is a clear enough language that you don't really need to learn it to follow the examples and the content of the book is sincerely fantastic. Alternatively, [99 Bottles of OOP](https://sandimetz.com/products) is written in both JavaScript and Ruby. It is written by the same author and may be a better option if you are brand new to OOP (it is not free either).
-
- [Building a house from the inside out](https://www.ayweb.dev/blog/building-a-house-from-the-inside-out) will walk you through the process of separating your core logic and DOM logic.
-
- This [brief video by Coderized](https://www.youtube.com/watch?v=q1qKv5TBaOA) covers the SOLID programming principles and more, within the context of embracing clean coding practices and establishing a maintainable code structure. You may find it helpful if you are still confused about why these principles exist and how they can work together to improve your code, code architecture, and your skills as a programmer!
diff --git a/javascript/organizing_your_javascript_code/project_restaurant_page.md b/javascript/organizing_your_javascript_code/project_restaurant_page.md
index beaf4ada7eb..616a204e94f 100644
--- a/javascript/organizing_your_javascript_code/project_restaurant_page.md
+++ b/javascript/organizing_your_javascript_code/project_restaurant_page.md
@@ -8,11 +8,11 @@ Let's use what we've learned and take a chance to continue practicing DOM manipu
#### .gitignore
-When working with packages that are installed with npm, you don't need to track the contents of `node_modules` with git, nor push those files to GitHub. This is because the `package.json` file contains all the information, so that anyone can clone your project and install them on their machine with `npm install`.
+When working with packages that are installed with npm, you don't need to track the contents of `node_modules` with git, nor push those files to GitHub. As we learned in the [npm lesson](https://www.theodinproject.com/lessons/node-path-javascript-npm), the `package.json` file contains all the dependency information, so that anyone can clone your project and install them on their machine with `npm install`.
-You can make a `.gitignore` file in the root of the project, and by writing file or directory names in it, you can tell git what things you don't want to track. It's customary to add `node_modules` to `.gitignore`, since it can get really big.
+You can make a `.gitignore` file in the root of the project, and by writing file or directory names in it, you can tell git what things you don't want to track. It's customary to add `node_modules` to `.gitignore`, since it can get really big. Similarly, `dist` is often ignored as it can be generated when someone runs the command to bundle/build the application.
-When creating a new repo on GitHub there is an option to specify a `.gitignore` template. There are many templates out there that include common files and directories that are not typically tracked based on the type of project or language used. When looking for a template for JavaScript projects there is a `node` template that includes `node_modules`.
+When creating a new repo on GitHub, there is an option to specify a `.gitignore` template. There are many templates out there that include common files and directories that are not typically tracked based on the type of project or language used. For JavaScript projects, there is a `node` template that includes `node_modules` and `dist`.
@@ -20,49 +20,24 @@ When creating a new repo on GitHub there is an option to specify a `.gitignore`
-1. Start the project the same way you began the webpack tutorial project.
- 1. Run `npm init` in your project directory to generate a `package.json` file.
+1. Start the project the same way you began the webpack tutorial project, by creating the `package.json` file and setting up Webpack.
+ - Remember, you only need to install and configure the things you need for your project. For example, if you do not plan to have local image files linked in your HTML template, you will not need to install and configure `html-loader`.
+1. Create a `.gitignore` file in the root of your project. It should contain the text `node_modules` and `dist` on separate lines.
- 1. Run `npm install webpack webpack-cli --save-dev` to install webpack to the node_modules directory of your project.
-
- 1. Create a `src` and `dist` directory with the following contents:
- 1. An `index.js` file in `src`.
-
- 1. An `index.html` file in `src`. This file will not need a script tag, because we're using `html-webpack-plugin`, which automatically adds that in. You will also not need to link a CSS stylesheet as you should be importing it into your JavaScript and letting your webpack configuration handle bundling.
-
- 1. Create a `webpack.config.js` file that looks just like our file from the [Webpack "Getting Started" tutorial](https://webpack.js.org/guides/getting-started/#using-a-configuration). Don't forget to add the `html-webpack-plugin` config to your `webpack.config.js` and set its `template` option with a path to `src/index.html`.
-
-1. Create a `.gitignore` file in the root of your project. It should contain `node_modules` and `dist` on separate lines.
-
-1. Set up an HTML skeleton inside of `src/index.html`. Inside the body, add a `` element that contains a `
### Knowledge check
-This section contains questions for you to check your understanding of this lesson on your own. If you’re having trouble answering a question, click it and review the material it links to.
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
- [How does lazy evaluation help make Active Record more efficient?](#relations-and-lazy-evaluation)
- [How do you check whether a database already contains a record?](#checking-for-existence)
@@ -249,7 +250,7 @@ This section contains questions for you to check your understanding of this less
### Additional resources
-This section contains helpful links to related content. It isn’t required, so consider it supplemental.
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
- [SO post on Using Scopes vs Class Methods](http://stackoverflow.com/questions/5899765/activerecord-rails-3-scope-vs-class-method)
- [Platformatec diving more into the use case of scopes vs class methods](http://blog.plataformatec.com.br/2013/02/active-record-scopes-vs-class-methods/)
diff --git a/ruby_on_rails/advanced_forms_and_activerecord/forms_advanced.md b/ruby_on_rails/advanced_forms_and_activerecord/forms_advanced.md
index 104bbcedf19..e2081162beb 100644
--- a/ruby_on_rails/advanced_forms_and_activerecord/forms_advanced.md
+++ b/ruby_on_rails/advanced_forms_and_activerecord/forms_advanced.md
@@ -14,13 +14,13 @@ This section contains a general overview of topics that you will learn in this l
- How to whitelist nested parameters.
- How to delete records via form fields.
-### Prepopulating `select` tags with collections
+### Prepopulating select tags with collections
Rails provides you with a few handy ways of making dropdown menus which already contain data when the form is loaded (otherwise they're not that useful).
Let's say you want to build a New Post form for your blog but you want to be able to select who the author is from among your list of users. You will need to make a dropdown which submits the user's ID as a part of your `params` hash. So you might populate `@users` in your posts controller:
-~~~ruby
+```ruby
# app/controllers/posts_controller.rb
...
def new
@@ -28,11 +28,11 @@ Let's say you want to build a New Post form for your blog but you want to be abl
@post = Post.new
end
...
-~~~
+```
The bare HTML way is to build a bunch of `
diff --git a/ruby_on_rails/apis/apis_and_building_your_own.md b/ruby_on_rails/apis/apis_and_building_your_own.md
index 92ddea9db68..5cd99840eff 100644
--- a/ruby_on_rails/apis/apis_and_building_your_own.md
+++ b/ruby_on_rails/apis/apis_and_building_your_own.md
@@ -2,7 +2,7 @@
Working with APIs is awesome and frustrating at the same time. On the one hand, interfacing with other applications out there can greatly improve the reach and "cool factor" of your own app. On the other, it involves lots of reading through documentation, figuring out authentication strategies, and parsing bad (or nonexistent) error messages.
-Backing up, if you're still unclear on what an API (Application Programming Interface) basically is, [read this FCC explanation](https://www.freecodecamp.org/news/what-is-an-api-in-english-please-b880a3214a82/) and then [read the first bit of this article](http://money.howstuffworks.com/business-communications/how-to-leverage-an-api-for-conferencing1.htm) to catch up.
+Backing up, if you're still unclear on what an API (Application Programming Interface) basically is, [FreeCodeCamp explains what APIs are](https://www.freecodecamp.org/news/what-is-an-api-in-english-please-b880a3214a82/). [Dave Roos' summary of APIs](http://money.howstuffworks.com/business-communications/how-to-leverage-an-api-for-conferencing1.htm) is also a good way to catch up.
"API" is an incredibly broad concept -- any time your application talks to another application, that's via some sort of API. The components within your own application, e.g. the different pieces of Rails, also talk to each other via APIs... they are more or less independent sub-applications that pass along the data they each need to complete their particular task. Everything's an API in application-land!
@@ -39,17 +39,17 @@ If you want your Rails app to return JSON instead of HTML, you need to tell your
You can see which file type Rails thinks you want by checking your server log:
-~~~bash
+```bash
Started GET "/posts/new" for 127.0.0.1 at 2013-12-02 15:21:08 -0800
Processing by PostsController#new as HTML
-~~~
+```
The first line tells you which URL was requested and the second tells you where it's going and how Rails is processing it. If you use a `.json` extension, it looks like:
-~~~bash
+```bash
Started GET "/posts.json" for 127.0.0.1 at 2013-12-04 12:02:01 -0800
Processing by PostsController#index as JSON
-~~~
+```
If you've got a sample application running, try going to different URLs. If your controller isn't ready for them, you may get an error, but you should be able to see what Rails thinks you're asking for.
@@ -57,7 +57,7 @@ If you've got a sample application running, try going to different URLs. If you
Once you've decided that you want to respond to a request for JSON or XML, you need to tell your controller to render JSON or XML instead of HTML. The way to do so is by using the `#respond_to` method:
-~~~ruby
+```ruby
class UsersController < ApplicationController
def index
@@ -71,7 +71,7 @@ Once you've decided that you want to respond to a request for JSON or XML, you n
end
end
-~~~
+```
In this case, `#respond_to` passes the block a format object, to which you can attach the appropriate rendering call. If you do nothing, HTML will render using the default Rails template as normal (in this case, `app/views/index.html.erb`).
@@ -89,7 +89,7 @@ In the old days, you'd just overwrite your own version of `#to_json` but these d
In our case, we'll do this by modifying `#as_json` in our model to return only the attributes we want:
-~~~ruby
+```ruby
# app/models/user.rb
class User < ActiveRecord::Base
@@ -104,11 +104,11 @@ In our case, we'll do this by modifying `#as_json` in our model to return only t
end
end
-~~~
+```
In our controller, we then just need to render JSON as normal (in the example below, it will just always return JSON, whether it's an HTML request or not):
-~~~ruby
+```ruby
# app/controllers/users_controller.rb
class UsersController < ApplicationController
@@ -117,7 +117,7 @@ In our controller, we then just need to render JSON as normal (in the example be
end
end
-~~~
+```
Note that you don't need to call `#to_json` yourself when using `#render`... it will do it for you.
@@ -125,10 +125,10 @@ See the [as_json documentation](https://api.rubyonrails.org/classes/ActiveModel/
#### Rendering nothing or errors
-Sometimes you just want to send out an HTTP error code without any response body. [Rails guides](https://guides.rubyonrails.org/layouts_and_rendering.html#using-head-to-build-header-only-responses) once again comes in really handy with an elegant solution for this problem.
+Sometimes you just want to send out an HTTP error code without any response body. [Rails guides](https://guides.rubyonrails.org/layouts_and_rendering.html#using-head-to-build-header-only-responses) once again comes in really handy with an elegant solution for this problem.
Here's an example (again we are just rendering the error in all cases):
-~~~ruby
+```ruby
# app/controllers/users_controller.rb
class UsersController < ApplicationController
@@ -137,14 +137,13 @@ Here's an example (again we are just rendering the error in all cases):
end
end
-~~~
-
+```
#### Creating dynamic error pages
-You can create your own error pages. See [this post](https://web-crunch.com/posts/custom-error-page-ruby-on-rails).
+You can [create your own error pages](https://web-crunch.com/posts/custom-error-page-ruby-on-rails).
-Sometimes Heroku can require additional steps to properly display your error pages. See [their error page docs here](https://devcenter.heroku.com/articles/error-pages). You might need to delete the static pages in the `app/public` directory first.
+Sometimes Heroku can require additional steps to properly display your error pages, as per the [Heroku error page docs](https://devcenter.heroku.com/articles/error-pages). You might need to delete the static pages in the `app/public` directory first.
#### External facing security
@@ -168,7 +167,7 @@ It's usually a good idea to strive to keep independent pieces of your applicatio
Using an SOA architecture for your whole application is sort of like breaking up a giant and complicated Ruby script into nice neat classes and methods, just on a broader scale.
-One of the best known cases of switching to a service-oriented architecture was Amazon.com. Sometime in 2002, Jeff Bezos basically dictated that every group would switch to SOA or be fired. An [infamous blog post](https://gist.github.com/chitchcock/1281611) from a Google Employee, accidentally released to the public instead of staying internal to the company, talked about Amazon's strength with SOA. It's a great read so check it out, but the basics of Bezos' email are, as quoted from the post:
+One of the best known cases of switching to a service-oriented architecture was Amazon.com. Sometime in 2002, Jeff Bezos basically dictated that every group would switch to SOA or be fired. An infamous blog post from a Google Employee, accidentally released to the public instead of staying internal to the company, talked about [Amazon's strength with SOA](https://gist.github.com/chitchcock/1281611). It's a great read so check it out, but the basics of Bezos' email are, as quoted from the post:
1. All teams will henceforth expose their data and functionality through service interfaces.
1. Teams must communicate with each other through these interfaces.
@@ -177,7 +176,7 @@ One of the best known cases of switching to a service-oriented architecture was
1. All service interfaces, without exception, must be designed from the ground up to be externalizable. That is to say, the team must plan and design to be able to expose the interface to developers in the outside world. No exceptions.
1. Anyone who doesn't do this will be fired.
-SOA is a big deal. There are certainly a lot of issues that crop up when you're using it -- see [this post on Amazon's "lessons learned"](http://apievangelist.com/2012/01/12/the-secret-to-amazons-success-internal-apis/) -- but it ultimately has a lot of benefit.
+SOA is a big deal. There are certainly a lot of issues that crop up when you're using it -- see this post on [Amazon's “lessons learned”](http://apievangelist.com/2012/01/12/the-secret-to-amazons-success-internal-apis/) -- but it ultimately has a lot of benefit.
You probably won't be worrying too much about SOA while building "toy" applications for yourself but it will certainly come up if you find yourself working at a tech company and it's a good principle to become familiar with.
@@ -192,30 +191,32 @@ In the next lesson, we'll cover working with other people's APIs, which can add
### Assignment
+
1. Read the [Rails Guide on Controllers](https://guides.rubyonrails.org/action_controller_overview.html#rendering-xml-and-json-data) section 7 to learn about rendering JSON and XML.
- 2. They are not required viewing (because they get a bit deeper than we're scoped for), but if you're interested, go check out the Railscasts in the Additional Resources section at the bottom of this lesson for more API goodness.
+ 1. They are not required viewing (because they get a bit deeper than we're scoped for), but if you're interested, go check out the Railscasts in the Additional Resources section at the bottom of this lesson for more API goodness.
+
### Knowledge check
-This section contains questions for you to check your understanding of this lesson on your own. If you’re having trouble answering a question, click it and review the material it links to.
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- How does Rails know which type of file you are expecting back when you make an HTTP request?
-- What is the purpose of the `#respond_to` method?
-- How do you return a User object but specify that you don't want to include certain attributes (i.e. you can't just return `User.first`)?
-- What are the two steps performed behind the scenes by the `#to_json` method?
-- How do you tell a controller action to render nothing but an error message?
-- How do you build your own custom error messages?
-- Why can't you use session-based controller authentication methods if you want people to access your API programmatically?
-- What is "Service Oriented Architecture?
+- [How does Rails know which type of file you are expecting back when you make an HTTP request?](#http-request-format)
+- [What is the purpose of the `#respond_to` method?](#rendering-json-or-xml)
+- [How do you return a User object but specify that you don't want to include certain attributes (i.e. you can't just return `User.first`)?](#specifying-attributes-to-return)
+- [What are the two steps performed behind the scenes by the `#to_json` method?](#to-json-steps)
+- [How do you tell a controller action to render nothing but an error message?](https://guides.rubyonrails.org/layouts_and_rendering.html#using-head-to-build-header-only-responses)
+- [How do you build your own custom error messages?](https://web-crunch.com/posts/custom-error-page-ruby-on-rails)
+- [Why can't you use session-based controller authentication methods if you want people to access your API programmatically?](#api-tokens)
+- [What is "Service Oriented Architecture?](#service-oriented-architecture-soa)
### Additional resources
-This section contains helpful links to related content. It isn’t required, so consider it supplemental.
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- Watch [this free Railscast on making your App into an API](http://railscasts.com/episodes/348-the-rails-api-gem)
-- Watch [this free Railscast on securing your API](http://railscasts.com/episodes/352-securing-an-api)
-- Watch [this free Railscast on versioning your API](http://railscasts.com/episodes/350-rest-api-versioning)
+- Watch this free [Railscast on making your App into an API](http://railscasts.com/episodes/348-the-rails-api-gem)
+- Watch this free [Railscast on securing your API](http://railscasts.com/episodes/352-securing-an-api)
+- Watch this free [Railscast on versioning your API](http://railscasts.com/episodes/350-rest-api-versioning)
- [GoRails #162 Our First API](https://www.gorails.com/episodes/our-first-api)
- [Building a public-facing API using view templates instead of `#to_json`](http://blog.codepath.com/2011/05/16/if-youre-using-to_json-youre-doing-it-wrong/)
- [`to_json` or `as_json` by Jonathan Julian](http://jonathanjulian.com/2010/04/rails-to_json-or-as_json/) gives specific examples of digging into the `as_json` method.
diff --git a/ruby_on_rails/apis/project_flickr_api.md b/ruby_on_rails/apis/project_flickr_api.md
index 73c4aa95569..5f20b8ee45c 100644
--- a/ruby_on_rails/apis/project_flickr_api.md
+++ b/ruby_on_rails/apis/project_flickr_api.md
@@ -5,14 +5,16 @@ Remember the warm-up in the previous project when you played with the Flickr API
### Assignment
+
1. Go back to the [Flickr API Docs](http://www.flickr.com/services/api/) and click [Create an App](http://www.flickr.com/services/apps/create/) at the top.
-2. Follow the steps for getting your API key. You'll have to sign in or sign up for Flickr (someone has to these days) and give them some basic information about your app. Select "Apply for a non-commercial key" and let them know how awesome your photo feed app will be. You'll automatically get a key generated for you, in addition to a secret key. Copy both of these somewhere you can get to them later.
-3. While logged in, copy your Flickr ID from the browser address bar by navigating to the "You" link on the top navbar. It will look like `https://www.flickr.com/photos/yourIDhere/`. An example would be `1895558555@N03`. You'll need that later for some of the API methods.
-4. Upload a few photos to your photostream!
-5. Create a new Rails app and add a gem for the [Flickr API](https://www.google.com/search?q=flickr+api+gem). There are gems for pretty much every API out there. They will all require you to include your API keys and secret keys somehow. Look for gems that are maintained (have recent commits) and well-adopted (GitHub stars is one way to get a good gauge for how valuable a gem is). Alternatively, you can browse through [RubyGems](https://rubygems.org/) to see popular gems.
-6. One note is that it's not good practice to have your secret key hard coded into your app because then it's hardly a secret, especially if you're pushing to GitHub. A better practice is to store the key in an [environment variable](http://railsapps.github.io/rails-environment-variables.html) or use a gem like [`figaro` (see docs)](https://github.com/laserlemon/figaro). You can use [`Rails credentials`(see this article)](https://web-crunch.com/posts/the-complete-guide-to-ruby-on-rails-encrypted-credentials) as well (although not mentioned in the article, the value of each key can also be accessed by chaining keys as methods as shown in the [Rails Guides](https://guides.rubyonrails.org/security.html#custom-credentials) examples). Environment variables allow you to push your key to your app directly from the command line when it fires up. Figaro operates under the same principle, though it helps you out by allowing you to store the keys in an actual file that just doesn't get committed with the rest of your code. Rails credentials encrypts the keys using the master key. Use one of these techniques unless you're a cowboy.
-7. Build a basic StaticPagesController to display a home page with a basic form. The form should just be a single text field which takes the ID for a Flickr user. Once the form is submitted, the page should refresh and display the photos from that user.
-8. Ask for your friends' flickr IDs or find random photofeeds on the web. View them in your app and react appropriately.
+1. Follow the steps for getting your API key. You'll have to sign in or sign up for Flickr (someone has to these days) and give them some basic information about your app. Select "Apply for a non-commercial key" and let them know how awesome your photo feed app will be. You'll automatically get a key generated for you, in addition to a secret key. Copy both of these somewhere you can get to them later.
+1. While logged in, copy your Flickr ID from the browser address bar by navigating to the "You" link on the top navbar. It will look like `https://www.flickr.com/photos/yourIDhere/`. An example would be `1895558555@N03`. You'll need that later for some of the API methods.
+1. Upload a few photos to your photostream!
+1. Create a new Rails app and add a gem for the [Flickr API](https://www.google.com/search?q=flickr+api+gem). There are gems for pretty much every API out there. They will all require you to include your API keys and secret keys somehow. Look for gems that are maintained (have recent commits) and well-adopted (GitHub stars is one way to get a good gauge for how valuable a gem is). Alternatively, you can browse through [RubyGems](https://rubygems.org/) to see popular gems.
+1. One note is that it's not good practice to have your secret key hard coded into your app because then it's hardly a secret, especially if you're pushing to GitHub. A better practice is to store the key in an [environment variable](http://railsapps.github.io/rails-environment-variables.html) or use something like the [figaro gem](https://github.com/laserlemon/figaro). You can use [Rails credentials](https://web-crunch.com/posts/the-complete-guide-to-ruby-on-rails-encrypted-credentials) as well (although not mentioned in the article, the value of each key can also be accessed by chaining keys as methods as shown in the [Rails Guides](https://guides.rubyonrails.org/security.html#custom-credentials) examples). Environment variables allow you to push your key to your app directly from the command line when it fires up. Figaro operates under the same principle, though it helps you out by allowing you to store the keys in an actual file that just doesn't get committed with the rest of your code. Rails credentials encrypts the keys using the master key. Use one of these techniques unless you're a cowboy.
+1. Build a basic StaticPagesController to display a home page with a basic form. The form should just be a single text field which takes the ID for a Flickr user. Once the form is submitted, the page should refresh and display the photos from that user.
+1. Ask for your friends' flickr IDs or find random photofeeds on the web. View them in your app and react appropriately.
+
### Additional resources
diff --git a/ruby_on_rails/apis/project_kittens_api.md b/ruby_on_rails/apis/project_kittens_api.md
index 28b40de21a0..942d0e8a20f 100644
--- a/ruby_on_rails/apis/project_kittens_api.md
+++ b/ruby_on_rails/apis/project_kittens_api.md
@@ -6,21 +6,21 @@ In this warmup, you'll get a chance to poke around with an existing API from Fli
-1. Head over to the [Flickr API Documentation Page](http://www.flickr.com/services/api/). You can just google `XYZ API docs` to locate these pages, which is usually much faster and easier than trying to find them by navigating the websites themselves.
-1. Look around at the different methods available. They have a couple of different request formats, but check out the RESTful API format by clicking on the link on the left under "Request Formats" called [REST](http://www.flickr.com/services/api/request.rest.html).
-1. This shows you the format that a typical API call would take -- you will make your request to the endpoint at `https://www.flickr.com/services/rest/` and include any required data in the GET query string or the POST body.
-1. Look around at the various methods. How would you upload a photo? How about getting your contacts list? Lots of these methods will require you to authenticate your application or user account first.
-1. Check out the [Search method docs](http://www.flickr.com/services/api/flickr.photos.search.html). This request doesn't require you to authenticate, just provide an API key (which you would get by registering as a developer on their platform). Browse through all the arguments that can be used.
-1. If you go to the bottom of the Search docs, you'll see a link to the [Search API Explorer](http://www.flickr.com/services/api/explore/flickr.photos.search). This tool lets you actually execute requests using Flickr's API key for illustrative purposes. You just enter the options you want and it will make the request for you. Try entering "puppies" under the "tags" argument, then scroll to the bottom. Change the response "Output" dropdown to JSON then click "Call Method".
-1. When the page refreshes, you'll see your results down at the bottom. You should see a big list of photo objects (after some meta data) that were returned by your search. They look like:
-
- ~~~json
+1. Head over to the [Flickr API Documentation Page](http://www.flickr.com/services/api/). You can just google `XYZ API docs` to locate these pages, which is usually much faster and easier than trying to find them by navigating the websites themselves.
+1. Look around at the different methods available. They have a couple of different request formats, but check out the RESTful API format by clicking on the link on the left under "Request Formats" called [REST](http://www.flickr.com/services/api/request.rest.html).
+1. This shows you the format that a typical API call would take -- you will make your request to the endpoint at `https://www.flickr.com/services/rest/` and include any required data in the GET query string or the POST body.
+1. Look around at the various methods. How would you upload a photo? How about getting your contacts list? Lots of these methods will require you to authenticate your application or user account first.
+1. Check out the [Search method docs](http://www.flickr.com/services/api/flickr.photos.search.html). This request doesn't require you to authenticate, just provide an API key (which you would get by registering as a developer on their platform). Browse through all the arguments that can be used.
+1. If you go to the bottom of the Search docs, you'll see a link to the [Search API Explorer](http://www.flickr.com/services/api/explore/flickr.photos.search). This tool lets you actually execute requests using Flickr's API key for illustrative purposes. You just enter the options you want and it will make the request for you. Try entering "puppies" under the "tags" argument, then scroll to the bottom. Change the response "Output" dropdown to JSON then click "Call Method".
+1. When the page refreshes, you'll see your results down at the bottom. You should see a big list of photo objects (after some meta data) that were returned by your search. They look like:
+
+ ```json
{ "id": "11357337313", "owner": "84645040@N00", "secret": "6dd795c9c6", "server": "3805", "farm": 4, "title": "Gavin-Feb2013-0127", "ispublic": 1, "isfriend": 0, "isfamily": 0 },
- ~~~
+ ```
More interestingly, you can see the URL they used to make the request below that. Let's break it apart here to show the parameters more clearly:
- ~~~bash
+ ```bash
https://www.flickr.com/services/rest/
?method=flickr.photos.search
&api_key=bbee7f1e3a3f9cb847b87964d50bf4bc
@@ -28,11 +28,11 @@ In this warmup, you'll get a chance to poke around with an existing API from Fli
&format=json
&nojsoncallback=1
&api_sig=d207eb20abbce7c40437a01f759e1388
- ~~~
+ ```
-1. The URL contains the REST endpoint we looked at before, along with our search query and some other options like the API key and format. If you copy and paste that into your browser, you'll see the same batch of output.
-1. If you look back on the [main API docs page](http://www.flickr.com/services/api/) in the "Read these first" box, there's a link that says [URLs](http://www.flickr.com/services/api/misc.urls.html). Follow that link.
-1. Flickr's API requires two steps to actually display a photo -- you need to get a photo's meta information (which we just received in our search results) and then you need to piece it together into a URL that Flickr can understand to actually retrieve the photo. The format they call typical is:
+1. The URL contains the REST endpoint we looked at before, along with our search query and some other options like the API key and format. If you copy and paste that into your browser, you'll see the same batch of output.
+1. Learn more about [how Flickr's photo URLs are structured](http://www.flickr.com/services/api/misc.urls.html).
+1. Flickr's API requires two steps to actually display a photo -- you need to get a photo's meta information (which we just received in our search results) and then you need to piece it together into a URL that Flickr can understand to actually retrieve the photo. The format they call typical is:
`https://live.staticflickr.com/{server-id}/{id}_{secret}_{size-suffix}.jpg`
@@ -46,7 +46,7 @@ In this warmup, you'll get a chance to poke around with an existing API from Fli
As you can see, omitting `_{size-suffix}` works and defaults to the longest edge being 500px.
-1. Tada! Every API is different and you've got to read through their documentation to understand the basic format of using it. Sometimes it can be helpful to search for a [YouTube](http://www.youtube.com) or [NetTuts](http://code.tutsplus.com) video with a quick overview as well.
+1. Tada! Every API is different and you've got to read through their documentation to understand the basic format of using it. Sometimes it can be helpful to search for a [YouTube](http://www.youtube.com) or [NetTuts](http://code.tutsplus.com) video with a quick overview as well.
@@ -62,52 +62,52 @@ This is a fast and straightforward project where you'll set up a Rails app to be
We'll start by building our Kitten application to work normally in the browser with HTML.
-1. Set up a new Rails application (`odin-kittens`) and Git repo.
-1. Update the README to describe the application and link back to this project.
-1. Build a Kitten model with attributes of `:name`, `:age`, `:cuteness`, and `:softness`.
-1. Build a KittensController and `:kittens` routes for all 7 RESTful actions.
-1. Set your default route to `kittens#index`.
-1. Fill out each of your controller actions and their corresponding views to display a very basic HTML page -- `#index` should just list all Kittens, `#show` should display a single Kitten, `#new` should render a Kitten creation form, `#edit` should use the same form (which should be a partial used by both the New and Edit views) to Edit the Kitten, `#create` and `#update` should do their jobs.
-1. Make a `delete` link on the Kitten's Show and Edit pages, as well as next to each Kitten listed in the Index page.
-1. Implement a display of the `flash` hash which congratulates you on adding or editing or deleting kittens and makes fun of you for errors in your form.
-1. Test out your Kitten creation machine to make sure all your controller actions are running properly.
+1. Set up a new Rails application (`odin-kittens`) and Git repo.
+1. Update the README to describe the application and link back to this project.
+1. Build a Kitten model with attributes of `:name`, `:age`, `:cuteness`, and `:softness`.
+1. Build a KittensController and `:kittens` routes for all 7 RESTful actions.
+1. Set your default route to `kittens#index`.
+1. Fill out each of your controller actions and their corresponding views to display a very basic HTML page -- `#index` should just list all Kittens, `#show` should display a single Kitten, `#new` should render a Kitten creation form, `#edit` should use the same form (which should be a partial used by both the New and Edit views) to Edit the Kitten, `#create` and `#update` should do their jobs.
+1. Make a `delete` link on the Kitten's Show and Edit pages, as well as next to each Kitten listed in the Index page.
+1. Implement a display of the `flash` hash which congratulates you on adding or editing or deleting kittens and makes fun of you for errors in your form.
+1. Test out your Kitten creation machine to make sure all your controller actions are running properly.
#### JSON API
Now it's time to make the Kittens resource available via API.
-1. Open a new command line tab and fire up IRB. We'll use `rest-client` gem to send requests to our app:
+1. Open a new command line tab and fire up IRB. We'll use `rest-client` gem to send requests to our app:
- ~~~irb
+ ```irb
require 'rest-client' # If you get an error here, you most likely need to install the gem.
response = RestClient.get("http://localhost:3000/kittens")
- ~~~
+ ```
-1. Let's see what we got back:
+1. Let's see what we got back:
- ~~~irb
+ ```irb
response.body #=> Should return a sloppy mess of HTML.
# alternatively, you can do this:
response.to_s
- ~~~
+ ```
If you check out your server output, it's probably processing as \*/\* (i.e. all media types), e.g. `Processing by KittensController#index as */*`
-1. Try asking specifically for a JSON response by adding the option `accept: :json`, e.g.:
+1. Try asking specifically for a JSON response by adding the option `accept: :json`, e.g.:
- ~~~irb
+ ```irb
json_response = RestClient.get("http://localhost:3000/kittens", accept: :json)
- ~~~
+ ```
You most likely will get a 406 Not Acceptable error - check your server console and you will see ActionController talking about UnknownFormat for your controller.
-1. Now modify your KittenController's `#index` method to `#respond_to` JSON and render the proper variables.
-1. Test it out by making sure your RestClient calls return the proper JSON strings, e.g.:
+1. Now modify your KittenController's `#index` method to `#respond_to` JSON and render the proper variables.
+1. Test it out by making sure your RestClient calls return the proper JSON strings, e.g.:
- ~~~irb
+ ```irb
json_response = RestClient.get("http://localhost:3000/kittens", accept: :json)
puts json_response.body
- ~~~
+ ```
-1. Do the same for your `#show` method, which will require you to provide an ID when making your request. Your CSRF protection will prevent you from creating, updating or deleting kittens via the API, so it's not necessary to implement those.
+1. Do the same for your `#show` method, which will require you to provide an ID when making your request. Your CSRF protection will prevent you from creating, updating or deleting kittens via the API, so it's not necessary to implement those.
Now you've got a website that is both a normal HTML-producing back end AND an API that can be used to pull data from it. You could use JavaScript calls from the front end to dynamically refresh your data now or even to load the whole page in the first place. Or maybe you'll be hooking up a Kittens app to your iPhone and need a back end. It doesn't matter, since now you've got a RESTful API.
diff --git a/ruby_on_rails/apis/working_with_external_apis.md b/ruby_on_rails/apis/working_with_external_apis.md
index 809540154cf..8e8b1c3e7d0 100644
--- a/ruby_on_rails/apis/working_with_external_apis.md
+++ b/ruby_on_rails/apis/working_with_external_apis.md
@@ -6,7 +6,7 @@ Most popular services offer APIs so developers can interface with them (they lov
If you go to the documentation for an API, it can sometimes be a bit overwhelming because much of the material sort of assumes that you know what you're doing, but some are definitely better than others. There are also elements that are common across almost all of them. The more you work with APIs (and get the hang of the authentication flow), the easier they get. You'll be making mashups in no time.
-This lesson will cover some general steps that are common across APIs and will do high level overviews of some of the methods for authenticating with APIs like Omniauth. Try to gain as much conceptual understanding as you can here and use the documentation each gem or API provides to help with the implementation. If you find great free learning resources that explain APIs, please let us know [(make a pull request and add an additional resource at the bottom of this page)](https://github.com/TheOdinProject/curriculum)!
+This lesson will cover some general steps that are common across APIs and will do high level overviews of some of the methods for authenticating with APIs like Omniauth. Try to gain as much conceptual understanding as you can here and use the documentation each gem or API provides to help with the implementation. If you find great free learning resources that explain APIs, you can [suggest resources via a GitHub issue in our curriculum repo](https://github.com/TheOdinProject/curriculum/issues/new/choose)!
### Lesson overview
@@ -32,9 +32,9 @@ You'll typically also get a "secret key" or similarly named code. Whereas the A
Most APIs require a different type of "security clearance" for different requests:
1. You can usually make innocuous requests like asking for posts from X's (formerly known as Twitter) API with straightforward and unauthenticated GET requests. You can make these from any command line or an application like [Postman](http://getpostman.com). These types of API requests are usually severely rate limited to incentivize you to sign up.
-2. The next layer is making requests that include your API key. These are still usually fairly innocuous things (like getting public data) and limited by the API's pricing tiers.
-3. More sensitive requests like asking for specific user data or submitting/modifying/deleting data will likely require you to use an authentication process involving your secret token. We'll cover the basics of this in the project. Rates for these requests are subject to the API pricing tiers.
-4. Oftentimes, you actually want to make requests on behalf of a user. For instance, showing a user a dashboard of all their posts and Facebook posts together would require asking X for that user's posts and Facebook for that user's posts. This can involve a LOT of requests over a large user base, but luckily you're actually able to make them on the user's behalf by asking for the user's permission. We'll cover this as well later, but basically you send the user to the API provider's site to sign in, then the API provider will give you a user-specific token to use when making requests on their behalf in the future. Rates for this are usually more advantageous because they are covered in a per-user bucket. We typically use the OAuth protocol for this, as described below.
+1. The next layer is making requests that include your API key. These are still usually fairly innocuous things (like getting public data) and limited by the API's pricing tiers.
+1. More sensitive requests like asking for specific user data or submitting/modifying/deleting data will likely require you to use an authentication process involving your secret token. We'll cover the basics of this in the project. Rates for these requests are subject to the API pricing tiers.
+1. Oftentimes, you actually want to make requests on behalf of a user. For instance, showing a user a dashboard of all their posts and Facebook posts together would require asking X for that user's posts and Facebook for that user's posts. This can involve a LOT of requests over a large user base, but luckily you're actually able to make them on the user's behalf by asking for the user's permission. We'll cover this as well later, but basically you send the user to the API provider's site to sign in, then the API provider will give you a user-specific token to use when making requests on their behalf in the future. Rates for this are usually more advantageous because they are covered in a per-user bucket. We typically use the OAuth protocol for this, as described below.
### Versions
@@ -77,57 +77,60 @@ OAuth 2.0 is actually pretty complicated, so we'll just cover the basic process.
Basically (still using Facebook as an example):
1. User tries to access a page on your app and you ask the user to login
-2. User chooses the "Login With Facebook" option
-3. User is redirected to a Facebook page asking them to review the permissions you are asking for and telling them to sign in. The URI will contain parameters that tell Facebook who your application is and possibly which URI they should submit their response to (or maybe you specified this as a part of your API registration process with them).
-4. User decides you seem like a fun application so they'll allow you to see their email address and post to their timeline. User signs in to their Facebook account. Facebook creates an authorization code and sends it back to your application's callback URI.
-5. The user waits while your application takes that authorization code and uses it to ask Facebook for the real good stuff. Facebook makes sure your application is the same one the user authorized, then POSTs back to you a unique authentication token for the user (which likely expires in 90 days) and any data you asked for up front (like email address).
-5. You store the user's unique token in your database and use it, along with your application key(s), to make any subsequent requests on the user's behalf.
+1. User chooses the "Login With Facebook" option
+1. User is redirected to a Facebook page asking them to review the permissions you are asking for and telling them to sign in. The URI will contain parameters that tell Facebook who your application is and possibly which URI they should submit their response to (or maybe you specified this as a part of your API registration process with them).
+1. User decides you seem like a fun application so they'll allow you to see their email address and post to their timeline. User signs in to their Facebook account. Facebook creates an authorization code and sends it back to your application's callback URI.
+1. The user waits while your application takes that authorization code and uses it to ask Facebook for the real good stuff. Facebook makes sure your application is the same one the user authorized, then POSTs back to you a unique authentication token for the user (which likely expires in 90 days) and any data you asked for up front (like email address).
+1. You store the user's unique token in your database and use it, along with your application key(s), to make any subsequent requests on the user's behalf.
-See [this brief overview of OAuth 2.0](http://tutorials.jenkov.com/oauth2/overview.html) for an overview. Then check out [this more substantive explanation from tutsplus.com](https://code.tutsplus.com/articles/oauth-20-the-good-the-bad-the-ugly--net-33216).
+See this brief [overview of OAuth 2.0](http://tutorials.jenkov.com/oauth2/overview.html). Then check out this [more substantive explanation of OAuth 2.0 from tutsplus.com](https://code.tutsplus.com/articles/oauth-20-the-good-the-bad-the-ugly--net-33216).
#### Implementing OAuth 2.0 in Rails -- Use OmniAuth!
This sounds horribly complicated! Someone must have made a gem for it...
-Luckily someone has. Many someones, actually. There is a generic OAuth gem called `omniauth` (docs available [on GitHub](https://github.com/intridea/omniauth)) and then a separate gem which provides a specific authentication strategy for every major API (see the list [HERE](https://github.com/intridea/omniauth/wiki/List-of-Strategies)). Once you've gone through things with one of them, you'll become comfortable with all of them. It's also worth noting that if you install and use [Devise](https://github.com/heartcombo/devise) (to handle your user model, for example), it comes with support for Omniauth built in! Refer to the [documentation](https://github.com/heartcombo/devise/wiki/OmniAuth%3A-Overview) for details.
+Luckily someone has. Many someones, actually. There is a generic OAuth gem called `omniauth` ([Omniauth documentation](https://github.com/intridea/omniauth)) and then a separate gem which provides a specific authentication strategy for every major API ([list of Omniauth strategies](https://github.com/intridea/omniauth/wiki/List-of-Strategies)). Once you've gone through things with one of them, you'll become comfortable with all of them. It's also worth noting that if you install and use [Devise](https://github.com/heartcombo/devise) (to handle your user model, for example), it comes with support for Omniauth built in! They have documentation on [how to integrate Devise with Omniauth](https://github.com/heartcombo/devise/wiki/OmniAuth%3A-Overview).
Using Omniauth is much easier to learn by doing than reading a bunch of bullet points. You'll have the opportunity to implement it in your final project where you can ask questions if needed.
### SDKs
-In addition to or instead of API access, many companies provide SDKs (software development kits). Usually these are Javascript libraries that contain all the code necessary to access their API. This can be useful because you're then able to access the API with Javascript methods instead of doing backflips on your own backend. It comes with the downside, however, of expanding your code base and forcing you to use their conventions for everything.
+In addition to or instead of API access, many companies provide SDKs (software development kits). Usually these are JavaScript libraries that contain all the code necessary to access their API. This can be useful because you're then able to access the API with JavaScript methods instead of doing backflips on your own backend. It comes with the downside, however, of expanding your code base and forcing you to use their conventions for everything.
We won't cover SDKs explicitly in this course but they should be well within reach to pick up by reading the documentation.
### Assignment
- 1. Watch [this Railscast on using Omniauth to allow X (formerly known as Twitter) Signin](http://railscasts.com/episodes/241-simple-omniauth-revised).
- 2. Read through the [Omniauth documentation](https://github.com/intridea/omniauth)
- 3. Pick an API that you really might like to interface with or a web product you use almost every day (e.g. Google, Facebook, Instagram...). Google for its docs, e.g. with "instagram api documentation", and have a look at them. Some docs are better than others, but they will be your source for understanding which methods you can call, what they will return, how to register your application to get an API key, and more useful tidbits.
+
+ 1. Watch this Railscast on [using Omniauth to allow X (formerly known as Twitter) Signin](http://railscasts.com/episodes/241-simple-omniauth-revised).
+ 1. Read through the [Omniauth documentation](https://github.com/intridea/omniauth)
+ 1. Pick an API that you really might like to interface with or a web product you use almost every day (e.g. Google, Facebook, Instagram...). Google for its docs, e.g. with "instagram api documentation", and have a look at them. Some docs are better than others, but they will be your source for understanding which methods you can call, what they will return, how to register your application to get an API key, and more useful tidbits.
+
-### Conclusion
+#### Conclusion
-APIs are fundamental to making rich web applications and they're also a lot of fun -- it makes your app feel a lot more "real" when you're able to let your users log in using Facebook or if you can display information from different sources across the web. If you're building a startup, it improves the user experience enough that you'll likely see an uptick in conversion rates.
+APIs are fundamental to making rich web applications and they're also a lot of fun -- it makes your app feel a lot more "real" when you're able to let your users log in using Facebook or if you can display information from different sources across the web. If you're building a startup, it improves the user experience enough that you'll likely see an uptick in conversion rates.
Working with external APIs can be incredibly rewarding because you're able to leverage functionality that others have spent a lot of time perfecting but also incredibly frustrating because they're all different and you have to rely on gems which are doing things behind the scenes that you're not entirely sure of. Over time, you'll start to see a lot of similarities between APIs and the procedure for working with them will feel more and more comfortable. To help get you there, the next project will have you working with an API and your final project will have you implementing signin via API as well.
### Knowledge check
-This section contains questions for you to check your understanding of this lesson. If you’re having trouble answering the questions below on your own, review the material above to find the answer.
-* What's the best way to locate an API's docs?
-* What is an API key?
-* How do you avoid including an API's secret token in your GitHub repo (e.g. hard coding it)?
-* Why is it important to know which API version you're using?
-* Why would a user prefer to sign into your site using Facebook instead of giving you a new password?
-* What are the different types of "security clearance" for the different types of API requests you can make?
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
+
+- [What's the best way to locate an API's docs?](#introduction)
+- [What is an API key?](#first-steps)
+- [How do you avoid including an API's secret token in your GitHub repo (e.g. hard coding it)?](#first-steps)
+- [Why is it important to know which API version you're using?](#versions)
+- [Why would a user prefer to sign into your site using Facebook instead of giving you a new password?](#oauth-and-login-via-api)
+- [What are the different types of "security clearance" for the different types of API requests you can make?](#api-rates-and-security-tokens)
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- See [This SO Post on interfacing with third-party APIs](http://stackoverflow.com/questions/6228870/interfacing-with-a-third-party-api-in-rails-opeing-urls-and-parsing-xml-json) for tips.
+- See this Stack Overflow post on [interfacing with third-party APIs](http://stackoverflow.com/questions/6228870/interfacing-with-a-third-party-api-in-rails-opeing-urls-and-parsing-xml-json) for tips.
- [RailsConf 2016 - From Zero to API Hero: Consuming APIs like a Pro by Cecy Correa](https://www.youtube.com/watch?v=Af5HDgvGuXk)
-- Take a look at [this Medium article](https://revs.runtime-revolution.com/integrating-a-third-party-api-with-rails-5-134f960ddbba) over integrating a third party API with Rails 5.
-- See this other [Medium Article](https://medium.com/food4fluctuations/using-an-api-in-rails-for-noobs-5e02edb0e56b) on creating a basic rails app using the OMDB API, an open source movie database.
+- Take a look at this Medium article over [integrating a third party API with Rails 5](https://revs.runtime-revolution.com/integrating-a-third-party-api-with-rails-5-134f960ddbba).
+- See this other Medium Article on [creating a basic Rails app using the OMDB API](https://medium.com/food4fluctuations/using-an-api-in-rails-for-noobs-5e02edb0e56b), an open source movie database.
diff --git a/ruby_on_rails/assets_and_navigation/asset_pipeline.md b/ruby_on_rails/assets_and_navigation/asset_pipeline.md
index a9321b6c893..5eb79b39a60 100644
--- a/ruby_on_rails/assets_and_navigation/asset_pipeline.md
+++ b/ruby_on_rails/assets_and_navigation/asset_pipeline.md
@@ -8,7 +8,6 @@ This section contains a general overview of topics that you will learn in this l
- How files are processed in the Asset Pipeline.
- Organization of stylesheets and images in your app.
-- How to display raw HTML-code in your app.
### The asset pipeline
@@ -30,7 +29,7 @@ The below section on the JavaScript manifest isn't relevant to Rails 7 applicati
Rails needs to know which files to include in that giant blob, so it uses so-called "manifest" files to determine this. Your JavaScript manifest file will be `app/assets/javascripts/application.js`. It looks commented out, but the lines starting with `//=` tell Rails which files to go find and include. The comments in the file are pretty useful -- they say:
-~~~javascript
+```javascript
// This is a manifest file that'll be compiled into application.js, which will include all the files
// listed below.
//
@@ -47,7 +46,7 @@ Rails needs to know which files to include in that giant blob, so it uses so-cal
//= require jquery_ujs
//= require turbolinks
//= require_tree .
-~~~
+```
The `require_tree` helper method just grabs everything in the current directory.
@@ -55,7 +54,7 @@ Note that packages like jQuery aren't included out of the box in newer Rails app
Your stylesheet manifest file operates on the same principle -- it's available at `app/assets/stylesheets/application.css`:
-~~~css
+```css
/*
* This is a manifest file that'll be compiled into application.css, which will include all the files
* listed below.
@@ -71,7 +70,7 @@ Your stylesheet manifest file operates on the same principle -- it's available a
*= require_tree .
*= require_self
*/
-~~~
+```
Again, you see the `require_tree` helper method which brings in all CSS files in the current directory. You should put CSS sparingly into this top level file and instead add your own CSS stylesheet files in an organized way.
@@ -97,23 +96,23 @@ Let's also assume that you really like using `.container` classes to keep your `
The basic idea is to be able to say "all this code/css/whatever inside here only belongs to XYZ". You sort of fence it off. It's best explained with an example:
-~~~html
+```html
-~~~
+```
Now this container and all the code inside of it is also within the `.user` class. So we can set up our stylesheet to specifically address the `.container` class that's inside a `.user` class:
-~~~css
+```css
/* app/assets/stylesheets/user.css */
.user .container{
// style stuff
}
-~~~
+```
This is good because we're now specifically targeting containers used by User pages.
diff --git a/ruby_on_rails/forms_and_authentication/form_basics.md b/ruby_on_rails/forms_and_authentication/form_basics.md
index cea116cb222..73056cb30de 100644
--- a/ruby_on_rails/forms_and_authentication/form_basics.md
+++ b/ruby_on_rails/forms_and_authentication/form_basics.md
@@ -1,12 +1,12 @@
### Introduction
-You should be familiar with forms, both as a normal Internet user and as an HTML coder who has done the [Intermediate HTML and CSS course](https://www.theodinproject.com/paths/full-stack-ruby-on-rails/courses/intermediate-html-and-css). But how much do you REALLY know about forms? It may sound strange, but forms are possibly the most complicated thing about learning web development. Not necessarily because the code itself is difficult, but because you usually want to build forms that accomplish so many different things at once.
+You should be familiar with forms, both as a normal Internet user and as an HTML coder who has done the [Intermediate HTML and CSS course](https://www.theodinproject.com/paths/full-stack-ruby-on-rails/courses/intermediate-html-and-css). But how much do you REALLY know about forms? It may sound strange, but forms are possibly the most complicated thing about learning web development. Not necessarily because the code itself is difficult, but because you usually want to build forms that accomplish so many different things at once.
-Up until now, we've been thinking about Models in Rails on sort of a one-off basis. The User model. The Post model. Sometimes we've had the models relate to each other via associations, like that a Post can `has_many` Comment objects. Usually, though, we tend to silo our thoughts to only deal with one at a time.
+Up until now, we've been thinking about Models in Rails on sort of a one-off basis. The User model. The Post model. Sometimes we've had the models relate to each other via associations, like that a Post can `has_many` Comment objects. Usually, though, we tend to silo our thoughts to only deal with one at a time.
-Now think about a web form to buy an airline ticket. You probably need to enter your name, address, phone number, email, the airline, the flight number, the flight date, your credit card number, your credit card security number, your card expiration date, your card's zipcode, and a bunch of checkboxes for additional things like trip insurance. That's a whole lot of different models embedded in one form! But you still submit it with a single button. Holy macaroni!
+Now think about a web form to buy an airline ticket. You probably need to enter your name, address, phone number, email, the airline, the flight number, the flight date, your credit card number, your credit card security number, your card expiration date, your card's zipcode, and a bunch of checkboxes for additional things like trip insurance. That's a whole lot of different models embedded in one form! But you still submit it with a single button. Holy macaroni!
-Most forms won't be that long or complicated for you, but it's useful to appreciate all the things you can (and one day will) do with them. It's incredibly easy to make a basic form so the first thing we'll do is make sure you've got an intimate understanding of how forms are created in HTML and then how Rails offers you some helpers to make your life easier. We'll cover the way data is structured and sent to the controller until you feel pretty comfortable with that. Then a later lesson will deal with how to take that basic understanding and make forms handle some more firepower.
+Most forms won't be that long or complicated for you, but it's useful to appreciate all the things you can (and one day will) do with them. It's incredibly easy to make a basic form so the first thing we'll do is make sure you've got an intimate understanding of how forms are created in HTML and then how Rails offers you some helpers to make your life easier. We'll cover the way data is structured and sent to the controller until you feel pretty comfortable with that. Then a later lesson will deal with how to take that basic understanding and make forms handle some more firepower.
### Lesson overview
@@ -19,171 +19,183 @@ This section contains a general overview of topics that you will learn in this l
### Forms in HTML
-Step one is to be able to create a form in HTML. Remember how that looks?
+Step one is to be able to create a form in HTML. Remember how that looks?
-~~~html
-
-~~~
+```html
+
+```
-There are plenty of `input` tags to choose from, including `button`, `checkbox`, `date`, `hidden`, `password`, `radio` and many more (see [the full list from W3 Schools](https://www.w3schools.com/html/html_form_input_types.asp)).
+There are plenty of `input` tags to choose from, including `button`, `checkbox`, `date`, `hidden`, `password`, `radio` and many more (see [the full list of HTML input types from W3Schools](https://www.w3schools.com/html/html_form_input_types.asp)).
### Viewing what your form submits
-If you want to see what your forms are submitting to your Rails app, look through the output that gets printed into your console when you run your `$ rails server`. Whenever you submit a very basic form for a user email signup, it should include lines that look something like:
+If you want to see what your forms are submitting to your Rails app, look through the output that gets printed into your console when you run your `$ rails server`. Whenever you submit a very basic form for a user email signup, it should include lines that look something like:
-~~~bash
+```bash
Started POST "/user" for 127.0.0.1 at 2013-11-21 19:10:47 -0800
Processing by UsersController#create as TURBO_STREAM
Parameters: {"utf8"=>"✓", "authenticity_token"=>"jJa87aK1OpXfjojryBk2Db6thv0K3bSZeYTuW8hF4Ns=", "email"=>"foo@bar.com", "commit"=>"Submit Form"}
-~~~
+```
-*Note: this is from a form that might be generated by a Rails helper method, as explained in a later section below*
+**Note** this is from a form that might be generated by a Rails helper method, as explained in a later section below.
-The first line tells us which HTTP method was used and which route the form went to. The second line tells us which controller and action the form will be handled by, note the TURBO_STREAM action as the default behaviour. The third line contains everything that will get stuffed into the `params` hash for the controller to use. We'll talk about the contents in the next sections.
+The first line tells us which HTTP method was used and which route the form went to. The second line tells us which controller and action the form will be handled by, note the TURBO_STREAM action as the default behaviour. The third line contains everything that will get stuffed into the `params` hash for the controller to use. We'll talk about the contents in the next sections.
-You'll find yourself looking at this server output a lot when you start building forms. It'll keep you sane because it tells you exactly what the browser sent back to your application so you can see if there's been a... misunderstanding.
+You'll find yourself looking at this server output a lot when you start building forms. It'll keep you sane because it tells you exactly what the browser sent back to your application so you can see if there's been a... misunderstanding.
### Railsifying your form
-The first thing you'll realize if you try to create a plain vanilla form in a Rails view is that it won't work. You'll either get an error or your user session will get zeroed out (depending on your Rails version). That's because Rails by default automatically protects you from [cross-site request forgery](https://en.wikipedia.org/wiki/Cross-site_request_forgery) and it requires you to verify that the form was actually submitted from a page you generated. In order to do so, it generates an ["authenticity token"](http://guides.rubyonrails.org/security.html#cross-site-request-forgery-csrf) which looks like gibberish but helps Rails match the form with your session and the application.
+The first thing you'll realize if you try to create a plain vanilla form in a Rails view is that it won't work. You'll either get an error or your user session will get zeroed out (depending on your Rails version). That's because Rails by default automatically protects you from [cross-site request forgery](https://en.wikipedia.org/wiki/Cross-site_request_forgery) and it requires you to verify that the form was actually submitted from a page you generated. In order to do so, it generates an ["authenticity token"](http://guides.rubyonrails.org/security.html#cross-site-request-forgery-csrf) which looks like gibberish but helps Rails match the form with your session and the application.
You'll notice the token in the server output from above:
-~~~bash
+```bash
...
Parameters: {"utf8"=>"✓", "authenticity_token"=>"jJa87aK1OpXfjojryBk2Db6thv0K3bSZeYTuW8hF4Ns=", "email"=>"foo@bar.com", "commit"=>"Submit Form"}
-~~~
+```
-So, if you want to create your own form that gets handled by Rails, you need to provide the token somehow as well. Luckily, Rails gives you a method called `form_authenticity_token` to do so, and we'll cover it in the project.
+So, if you want to create your own form that gets handled by Rails, you need to provide the token somehow as well. Luckily, Rails gives you a method called `form_authenticity_token` to do so, and we'll cover it in the project.
-~~~html
-
-~~~
+```erb
+
+```
### Making forms into params
What about the other form inputs, the ones we actually care about?
-Each one of these inputs is structured slightly differently, but there are some commonalities. One important thing to note is the `name` attribute that you can give to an input tag. In Rails, that's very important. The `name` attribute tells Rails what it should call the stuff you entered in that input field when it creates the `params` hash. For instance,
+Each one of these inputs is structured slightly differently, but there are some commonalities. One important thing to note is the `name` attribute that you can give to an input tag. In Rails, that's very important. The `name` attribute tells Rails what it should call the stuff you entered in that input field when it creates the `params` hash. For instance,
-~~~html
- ...
-
- ...
-~~~
+```html
+...
+
+...
+```
-Will result in your `params` hash containing a key called `description` that you can access as normal, e.g. `params[:description]`, inside your controller. That's also why some inputs like radio buttons (where `type="radio"`) use the `name` attribute to know which radio buttons should be grouped together such that clicking one of them will unclick the others. The `name` attribute is surprisingly important!
+Will result in your `params` hash containing a key called `description` that you can access as normal, e.g. `params[:description]`, inside your controller. That's also why some inputs like radio buttons (where `type="radio"`) use the `name` attribute to know which radio buttons should be grouped together such that clicking one of them will unclick the others. The `name` attribute is surprisingly important!
-Now another thing we talked about in the controller section was nesting data. You'll often want to tuck submitted data neatly into a hash instead of keeping them all at the top level. This can be useful because, as we saw with controllers, it lets you do a one-line `#create` (once you've allowed the parameters with `#require` and `#permit`). When you access `params[:user]`, it's actually a hash containing all the user's attributes, for instance `{first_name: "foo", last_name: "bar", email: "foo@bar.com"}`. How do you get your forms to submit parameters like this? It's easy!
+Now another thing we talked about in the controller section was nesting data. You'll often want to tuck submitted data neatly into a hash instead of keeping them all at the top level. This can be useful because, as we saw with controllers, it lets you do a one-line `#create` (once you've allowed the parameters with `#require` and `#permit`). When you access `params[:user]`, it's actually a hash containing all the user's attributes, for instance `{first_name: "foo", last_name: "bar", email: "foo@bar.com"}`. How do you get your forms to submit parameters like this? It's easy!
It all comes back to the `name` attribute of your form inputs. Just use hard brackets to nest data like so:
-~~~html
- ...
-
-
-
- ...
-~~~
+```html
+...
+
+
+
+...
+```
-Those inputs will now get transformed into a nested hash under the `:user` key. The server output becomes:
+Those inputs will now get transformed into a nested hash under the `:user` key. The server output becomes:
-~~~bash
+```bash
Parameters: {"utf8"=>"✓", "authenticity_token"=>"jJa87aK1OpXfjojryBk2Db6thv0K3bSZeYTuW8hF4Ns=", "user"=>{"first_name"=>"foo","last_name"=>"bar","email"=>"foo@bar.com"}, "commit"=>"Submit Form"}
-~~~
+```
Specific parameters of the `params` hash are accessed like any other nested hash `params[:user][:email]`.
-Don't forget that you have to allow the params now in your controller using `require` and `permit` because they are a hash instead of just a flat string. See the Controller section below for a refresher on the controller side of things.
+Don't forget that you have to allow the params now in your controller using `require` and `permit` because they are a hash instead of just a flat string. See the Controller section below for a refresher on the controller side of things.
This is cool stuff that you'll get a chance to play with in the project.
### Form helpers: form_with
-Rails tries to make your life as easy as it can, so naturally it provides you with helper methods that automate some of the repetitive parts of creating forms. That doesn't mean you don't need to know how to create forms the "old fashioned" way... it's actually MORE important to know your form fundamentals when using helpers because you'll need to really understand what's going on behind the scenes if something breaks.
+Rails tries to make your life as easy as it can, so naturally it provides you with helper methods that automate some of the repetitive parts of creating forms. That doesn't mean you don't need to know how to create forms the "old fashioned" way... it's actually MORE important to know your form fundamentals when using helpers because you'll need to really understand what's going on behind the scenes if something breaks.
-Start by making a form using the `form_with` helper, which takes a block representing all the inputs to the form. It takes care of the CSRF security token we talked about above by automatically creating the hidden input for it so you don't have to. You pass it arguments to tell it which path to submit to (the default is the current page) and which method to use. Then there are tag helpers that create the specified tags for you, like `text_field_tag` below. All you need to specify there is what you want to call the field when it is submitted.
+Start by making a form using the `form_with` helper, which takes a block representing all the inputs to the form. It takes care of the CSRF security token we talked about above by automatically creating the hidden input for it so you don't have to. You pass it arguments to tell it which path to submit to (the default is the current page) and which method to use. Then there are tag helpers that create the specified tags for you, like `text_field_tag` below. All you need to specify there is what you want to call the field when it is submitted.
-~~~bash
+```erb
<%= form_with(url: "/search", method: "get") do %>
<%= label_tag(:query, "Search for:") %>
<%= text_field_tag(:query) %>
<%= submit_tag("Search") %>
<% end %>
-~~~
+```
Creates the form:
-~~~html
-
-~~~
+```html
+
+```
-There are tag helpers for all the major tags and the options they accept are all a bit different. See the reading assignment for more detail.
+There are tag helpers for all the major tags and the options they accept are all a bit different. See the reading assignment for more detail.
There are a few things to take note of when using the `form_with` helper.
1. The ID of the inputs matches the name.
+1. The second line ends with `as TURBO_STREAM` when you look at your output in your console after submitting a form. By default, all forms are now submitted by Turbo Drive. This means that a full request cycle doesn't occur and the page doesn't reload when the form is submitted. In order to disable this, you need to add the necessary data attribute that we covered in the Turbo Drive lesson.
-2. The second line ends with `as TURBO_STREAM` when you look at your output in your console after submitting a form. By default, all forms are now submitted by Turbo Drive. This means that a full request cycle doesn't occur and the page doesn't reload when the form is submitted. In order to disable this, you need to add the necessary data attribute that we covered in the Turbo Drive lesson.
-
-~~~bash
+```erb
<%= form_with(url: "/search", method: "get", data: { turbo: false} ) do %>
<%= label_tag(:query, "Search for:") %>
<%= text_field_tag(:query) %>
<%= submit_tag("Search") %>
<% end %>
-~~~
+```
### Using models with the form_with helper
More often than not, you'll want your form to act on the attributes of an existing model. Like specifying a title (or whatever other fields are required for your model) of a new news Article.
-Just pass `form_with` a model object, and it will make the form submit to the URL for that object, e.g. `@article` will submit to the correct URL for creating an Article. Remember from the lesson on controllers that the `#new` action usually involves creating a new (unsaved) instance of your object and passing it to the view... now you finally get to see why by using that object in your `#form_with` forms!
+Just pass `form_with` a model object, and it will make the form submit to the URL for that object, e.g. `@article` will submit to the correct URL for creating an Article. Remember from the lesson on controllers that the `#new` action usually involves creating a new (unsaved) instance of your object and passing it to the view... now you finally get to see why by using that object in your `#form_with` forms!
From the Rails Guide:
-~~~ruby
+```ruby
# app/controllers/articles_controller.rb
def new
@article = Article.new
end
-~~~
+```
-~~~erb
+```erb
# app/views/articles/new.html.erb
<%= form_with model: @article do |form| %>
<%= form.text_field :title %>
<%= form.submit "Create" %>
<% end %>
-~~~
+```
This will produce the following HTML:
-~~~html
+```html
-~~~
+```
-The best part about `form_with` is that if you just pass it a model object like `@article` in the example above, Rails will check for you if the object has been saved yet. If it's a new object, it will send the form to your `#create` action. If the object has been saved before, so we know that we're editing an existing object, it will send the object to your `#update` action instead. This is done by automatically generating the correct URL when the form is created. Magic!
+The best part about `form_with` is that if you just pass it a model object like `@article` in the example above, Rails will check for you if the object has been saved yet. If it's a new object, it will send the form to your `#create` action. If the object has been saved before, so we know that we're editing an existing object, it will send the object to your `#update` action instead. This is done by automatically generating the correct URL when the form is created. Magic!
### Forms and validations
-What happens if your form is submitted but fails the validations you've placed on it? For instance, what if the user's password is too short? Well, first of all, you should have had some JavaScript validations to be your first line of defense and they should have caught that... but we'll get into that in another course. In any case, hopefully your controller is set up to re-render the current form.
+What happens if your form is submitted but fails the validations you've placed on it? For instance, what if the user's password is too short? Well, first of all, you should have had some JavaScript validations to be your first line of defense and they should have caught that... but we'll get into that in another course. In any case, hopefully your controller is set up to re-render the current form.
-You'll probably want to display the errors so the user knows what went wrong. Recall that when Rails tries to validate an object and fails, it attaches a new set of fields to the object called `errors`. You can see those errors by accessing `your_object_name.errors`. Those errors have a couple of handy helpers you can use to display them nicely in the browser -- `#count` and `#full_messages`. See the code below:
+You'll probably want to display the errors so the user knows what went wrong. Recall that when Rails tries to validate an object and fails, it attaches a new set of fields to the object called `errors`. You can see those errors by accessing `your_object_name.errors`. Those errors have a couple of handy helpers you can use to display them nicely in the browser -- `#count` and `#full_messages`. See the code below:
-~~~erb
+```erb
<% if @post.errors.any? %>
<%= pluralize(@post.errors.count, "error") %> prohibited this post from being saved:
@@ -195,27 +207,27 @@ You'll probably want to display the errors so the user knows what went wrong. R
<% end %>
-~~~
+```
That will give the user a message telling him/her how many errors there are and then a message for each error.
-The best part about Rails form helpers... they handle errors automatically too! If a form is rendered for a specific model object, like using `form_with model: @article` from the example above, Rails will check for errors and, if it finds any, it will automatically wrap a special `
` element around that field with the class `field_with_errors` so you can write whatever CSS you want to make it stand out. Cool!
+The best part about Rails form helpers... they handle errors automatically too! If a form is rendered for a specific model object, like using `form_with model: @article` from the example above, Rails will check for errors and, if it finds any, it will automatically wrap a special `
` element around that field with the class `field_with_errors` so you can write whatever CSS you want to make it stand out. Cool!
### Making PATCH and DELETE submissions
-Forms aren't really designed to natively delete objects because browsers only support GET and POST requests. Rails gives you a way around that by sticking a hidden field named "\_method" into your form. It tells Rails that you actually want to do either a PATCH (aka PUT) or DELETE request (whichever you specified), and might look like ``.
+Forms aren't really designed to natively delete objects because browsers only support GET and POST requests. Rails gives you a way around that by sticking a hidden field named "\_method" into your form. It tells Rails that you actually want to do either a PATCH (aka PUT) or DELETE request (whichever you specified), and might look like ``.
You get Rails to add this to your form by passing an option to `form_with` called `:method`, e.g.:
-~~~ruby
+```ruby
form_with(url: search_path, method: "patch")
-~~~
+```
### Controller-side refresher
Just as a refresher, here's a very basic controller setup for handling `#new` actions and `#create` actions.
-~~~ruby
+```ruby
# app/controllers/users_controller.rb
...
def new
@@ -235,27 +247,27 @@ Just as a refresher, here's a very basic controller setup for handling `#new` ac
params.require(:user).permit(:first_name, :last_name, :other_stuff)
end
...
-~~~
+```
-I wanted to show this again so you could remind yourself what's going on in the form's lifecycle. The user presumably went to the path for the `#new` action, likely `http://www.yourapp.com/users/new`. That ran the `#new` action, which created a new user object in memory (but not yet saved to the database) and rendered the `new.html.erb` view. The view probably used `form_with model: @user` to make things nice and easy for the developer.
+I wanted to show this again so you could remind yourself what's going on in the form's lifecycle. The user presumably went to the path for the `#new` action, likely `http://www.yourapp.com/users/new`. That ran the `#new` action, which created a new user object in memory (but not yet saved to the database) and rendered the `new.html.erb` view. The view probably used `form_with model: @user` to make things nice and easy for the developer.
-Once the form gets submitted, the `#create` action will build another new User object with the parameters we explicitly tell it are okay. Recall that our custom `#user_params` method will return the `params[:user]` hash for us, which lets `User.new` build us a complete new instance. If that instance can be saved to the database, we're all good and we go to that user's `show.html.erb` page.
+Once the form gets submitted, the `#create` action will build another new User object with the parameters we explicitly tell it are okay. Recall that our custom `#user_params` method will return the `params[:user]` hash for us, which lets `User.new` build us a complete new instance. If that instance can be saved to the database, we're all good and we go to that user's `show.html.erb` page.
-If the `@user` cannot be saved, like because the `first_name` contains numbers, we will jump straight back to rendering the `new.html.erb` view, this time using the `@user` instance that will still have errors attached to it. Our form should gracefully handle those errors by telling the user where they screwed up.
+If the `@user` cannot be saved, like because the `first_name` contains numbers, we will jump straight back to rendering the `new.html.erb` view, this time using the `@user` instance that will still have errors attached to it. Our form should gracefully handle those errors by telling the user where they screwed up.
Notice how in our else clause we specify a status
-~~~ruby
+```ruby
else
render :new, status: :unprocessable_entity
-~~~
+```
Remember our Turbo Drive lesson? Which status codes does Turbo handle? Either a redirect, a 4XX status or a 5XX status. On a successful save of our created object you can see we do a redirect
-~~~ruby
+```ruby
if @user.save
redirect_to @user
-~~~
+```
This is the 303 redirect status we mentioned in the earlier Turbo Drive lesson. If we don't redirect, and we don't want a 5XX error, as that means our application has hit an unexpected issue, that only leaves us with a 4XX status code.
@@ -263,38 +275,40 @@ What would be returned if we didn't include `status: :unprocessable_entity`?
It would be 200 OK status because that is the expected response to a failed form submission. The request has completed in the way we expected. Therefore when dealing with a form validation that fails, you need to specify the 422 HTTP status. Doing so will change a `Completed 200 OK in 206ms (Views: 125.5ms | ActiveRecord: 8.1ms | Allocations: 18793)` return status to `Completed 422 Unprocessable Entity in 52ms (Views: 44.6ms | ActiveRecord: 0.0ms | Allocations: 6391)`, and as we know, Turbo will handle this and update the HTML with the returned response.
-What is the response? The new page rendered again as html. But this time, because the `@user` object has some errors on it, the HTML will include any information you've rendered on an object having some errors.
+What is the response? The new page rendered again as HTML. But this time, because the `@user` object has some errors on it, the HTML will include any information you've rendered on an object having some errors.
If you find yourself submitting a form and nothing is happening, chances are you forgot to return the correct status. You can confirm in the server logs when you submit the form.
### Assignment
+
1. Read the [Rails Guide on Form Helpers](http://guides.rubyonrails.org/form_helpers.html), sections 1 to 3.2.
- 2. Skim 3.3 to 7 to see what kinds of things are out there. One day you'll need them, and now you know where to look.
- 3. Read sections 8.1 and 8.2 for the official explanation of how parameters are created from the `name` attribute.
- 4. Read the [Rails Guide on Validations](http://guides.rubyonrails.org/active_record_validations.html#displaying-validation-errors-in-views) section 8 for a quick look at displaying errors.
- 5. Skim through the official Rails API section on the [form_with helper](https://api.rubyonrails.org/classes/ActionView/Helpers/FormHelper.html#method-i-form_with) to see various ways to use this helper tag.
+ 1. Skim 3.3 to 7 to see what kinds of things are out there. One day you'll need them, and now you know where to look.
+ 1. Read sections 8.1 and 8.2 for the official explanation of how parameters are created from the `name` attribute.
+ 1. Read the [Rails Guide on Validations](http://guides.rubyonrails.org/active_record_validations.html#displaying-validation-errors-in-views) section 8 for a quick look at displaying errors.
+ 1. Skim through the official Rails API section on the [form_with helper](https://api.rubyonrails.org/classes/ActionView/Helpers/FormHelper.html#method-i-form_with) to see various ways to use this helper tag.
+
-### Conclusion
+#### Conclusion
-At this point, you should have a solid understanding of how forms work in general and a pretty good feel for how Rails helps you out by generating them for you. You'll get a chance to build a whole bunch of forms in the next few projects, so don't worry if it's not totally stuck for you yet. Seeing it in action will make things click.
+At this point, you should have a solid understanding of how forms work in general and a pretty good feel for how Rails helps you out by generating them for you. You'll get a chance to build a whole bunch of forms in the next few projects, so don't worry if it's not totally stuck for you yet. Seeing it in action will make things click.
### Knowledge check
-This section contains questions for you to check your understanding of this lesson on your own. If you’re having trouble answering a question, click it and review the material it links to.
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- What is a CSRF Token and why is it necessary?
-- What is the `name` attribute of a form input element and what does it do?
-- How do you nest attributes under a single hash in `params`?
-- How do you pass `form_with` a model object?
-- How do you access errors for a failed-to-save model object?
-- How do Rails forms make PATCH or DELETE requests?
-- What is one case where you may need an array of hashes within the `params` hash?
+- [What is a CSRF Token and why is it necessary?](#railsifying-your-form)
+- [What is the `name` attribute of a form input element and what does it do?](#making-forms-into-params)
+- [How do you nest attributes under a single hash in `params`?](#making-forms-into-params)
+- [How do you pass `form_with` a model object?](#using-models-with-the-form_with-helper)
+- [How do you access errors for a failed-to-save model object?](#forms-and-validations)
+- [How do Rails forms make PATCH or DELETE requests?](#making-patch-and-delete-submissions)
+- [What is one case where you may need an array of hashes within the `params` hash?](https://guides.rubyonrails.org/form_helpers.html#combining-them)
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- It looks like this lesson doesn't have any additional resources yet. Help us expand this section by contributing to our curriculum.
+- It looks like this lesson doesn't have any additional resources yet. Help us expand this section by contributing to our curriculum.
diff --git a/ruby_on_rails/forms_and_authentication/project_members_only.md b/ruby_on_rails/forms_and_authentication/project_members_only.md
index ba1412e9e39..703377282da 100644
--- a/ruby_on_rails/forms_and_authentication/project_members_only.md
+++ b/ruby_on_rails/forms_and_authentication/project_members_only.md
@@ -20,11 +20,11 @@ If you'd like to challenge yourself, don't even follow the steps below, just go
1. Think about and spec out how to set up your data models for this application. You'll need users with the usual identification attributes like name, email and password. They'll need to create posts as well.
1. Create your new `members-only` Rails app and GitHub repo. Update your README.
-1. Add devise to your Gemfile and install it in your app using set up instructions on the devise [README](https://github.com/heartcombo/devise)
+1. Add devise to your Gemfile and install it in your app using set up instructions on the [Devise README](https://github.com/heartcombo/devise).
-For getting Devise to play nicely with Turbo Drive, be sure you read [this section](https://github.com/heartcombo/devise#hotwireturbo) of the Devise README. You’ll need to install the [Responders gem](https://github.com/heartcombo/responders). Make sure that in addition to adding the gem to your Gemfile that you also run the install generator. You’ll also need to specify delete requests on your links/buttons for signing the user out. More detailed information can be found in [Devise’s Guide for Hotwire Turbo Integration](https://github.com/heartcombo/devise/wiki/How-To:-Upgrade-to-Devise-4.9.0-%5BHotwire-Turbo-integration%5D).
+You'll need to install the [Responders gem](https://github.com/heartcombo/responders) to [get Devise to play nicely with Turbo Drive](https://github.com/heartcombo/devise#hotwireturbo). Make sure that in addition to adding the gem to your Gemfile that you also run the install generator. You’ll also need to specify delete requests on your links/buttons for signing the user out. More detailed information can be found in [Devise’s Guide for Hotwire Turbo Integration](https://github.com/heartcombo/devise/wiki/How-To:-Upgrade-to-Devise-4.9.0-%5BHotwire-Turbo-integration%5D).
diff --git a/ruby_on_rails/forms_and_authentication/sessions_cookies_authentication.md b/ruby_on_rails/forms_and_authentication/sessions_cookies_authentication.md
index f5a9640c039..0b2c5eaa2f0 100644
--- a/ruby_on_rails/forms_and_authentication/sessions_cookies_authentication.md
+++ b/ruby_on_rails/forms_and_authentication/sessions_cookies_authentication.md
@@ -27,7 +27,7 @@ Cookies, Sessions and Flashes are three special objects that Rails gives you in
### Cookies
-Cookies are key-value data pairs that are stored in the user's browser until they reach their specified expiration date. They can be used for pretty much anything, most commonly to "bookmark" the user's place in a web page if she gets disconnected or to store simple site display preferences. You could also store shopping cart information or even passwords but that would be a bad idea -- you shouldn't store anything in regular browser cookies that needs to either be secure or persisted across browser sessions. It's too easy for users to clear their cache and/or steal/manipulate unsecured cookies.
+Cookies are key-value data pairs that are stored in the user's browser until they reach their specified expiration date. They can be used for pretty much anything, most commonly to "bookmark" the user's place in a web page if they get disconnected or to store simple site display preferences. You could also store shopping cart information or even passwords but that would be a bad idea -- you shouldn't store anything in regular browser cookies that needs to either be secure or persisted across browser sessions. It's too easy for users to clear their cache and/or steal/manipulate unsecured cookies.
To work with cookies, Rails gives you access to a special hash called `cookies`, where each key-value pair is stored as a separate cookie on the user's browser. If you were to save `cookies[:hair-color] = "blonde"`, you'd be able to pull up your browser's developer tools and see a cookie on the user's browser that has a key of `hair-color` and a value of `blonde`. Delete it using `cookies.delete(:hair-color)`.
@@ -37,13 +37,13 @@ With each new request to your server, the browser will send along all the cookie
Think about how websites keep track of how a user is logged in when the page reloads. HTTP requests are stateless so how can you tell that a given request actually came from that particular user who is logged in? This is why cookies are important -- they allow you to keep track of your user from one request to another until the cookie expires.
-A special case is when you want to keep track of data in the user's "session", which represents all the stuff your user does while you've chosen to "remember" her, typically until the browser window is closed. In that case, every page she visits until the browser is closed will be part of the same session.
+A special case is when you want to keep track of data in the user's "session", which represents all the stuff your user does while you've chosen to "remember" them, typically until the browser window is closed. In that case, every page they visit until the browser is closed will be part of the same session.
-To identify a user's session information, Rails stores a special secure and tamper-proof cookie on the user's browser that contains their entire session hash (look for it in your developer tools, usually under the “Application” section) and it expires when the browser is closed. Whenever the user makes a request to your application, that request will also automatically include that session cookie (along with the other cookies) and you can use it to keep track of her logged-in state. This may all seem abstract now, but you'll get a chance to see it in action shortly.
+To identify a user's session information, Rails stores a special secure and tamper-proof cookie on the user's browser that contains their entire session hash (look for it in your developer tools, usually under the “Application” section) and it expires when the browser is closed. Whenever the user makes a request to your application, that request will also automatically include that session cookie (along with the other cookies) and you can use it to keep track of their logged-in state. This may all seem abstract now, but you'll get a chance to see it in action shortly.
Rails gives you access to the `session` hash in an almost identical way to the above-mentioned `cookies` hash. Use the `session` variable in your views or controllers like so:
-~~~ruby
+```ruby
# app/controllers/users_controller.rb
...
# Set a session value
@@ -58,7 +58,7 @@ Rails gives you access to the `session` hash in an almost identical way to the a
# Reset the entire session
reset_session
...
-~~~
+```
Why would you need both cookies and sessions? They are similar but not the same. `session` is an entire hash that gets put in the secure session cookie that expires when the user closes the browser. If you look in your developer tools, the "expiration" of that cookie is "session". Each value in the `cookies` hash gets stored as an individual cookie.
@@ -81,13 +81,13 @@ The flash is there to save the day! Just store `flash[:success]` (or whatever y
You still have to write view code to display the flash messages. It's common to write a short view helper that will pin any available flash message(s) to the top of the browser. You might also add a class to the message which will allow you to write some custom CSS, for instance turning `:success` messages green and `:error` messages red.
-~~~html
+```erb
# app/views/layouts/application.html.erb
...
<% flash.each do |name, message| %>
<%= message %>
<% end %>
-~~~
+```
### Controller filters
@@ -95,7 +95,7 @@ Before we talk about authentication, we need to cover controller filters. The i
We do this through the use of a "before filter", which takes the name of the method we want to run:
-~~~ruby
+```ruby
# app/controllers/users_controller
before_action :require_login
before_action :do_something_cool
@@ -113,7 +113,7 @@ Before we talk about authentication, we need to cover controller filters. The i
def do_something_cool
# do stuff here
end
-~~~
+```
The `before_action` method takes the symbol of the method to run before anything else gets run in the controller. In the case that this callback renders or redirects, the request, as well as any callbacks that are scheduled to run after that callback, are also cancelled. So in the case above, if the user was redirected to the login page, the `before_action :do_something_cool` callback wouldn't have been executed either.
@@ -129,7 +129,7 @@ The whole point of authentication is to make sure that the user is who they say
A related concept is authorization. Yes, you may be signed in, but are you actually authorized to access what you're trying to access? The typical example is the difference between a regular user and an admin user. They both authenticate with the system but only the admin is authorized to make changes to certain things.
-Authentication and authorization go hand in hand -- you first authenticate someone so you know who they are and can check if they're authorized to view a page or perform an action. When you build your app, you'll have a system of authentication to get the user signed in and to verify the user is who he says he is. You authorize the user to do certain things (like delete stuff) based on which methods are protected by controller filters that require signin or elevated permissions (e.g. admin status).
+Authentication and authorization go hand in hand -- you first authenticate someone so you know who they are and can check if they're authorized to view a page or perform an action. When you build your app, you'll have a system of authentication to get the user signed in and to verify the user is who they says they are. You authorize the user to do certain things (like delete stuff) based on which methods are protected by controller filters that require signin or elevated permissions (e.g. admin status).
### Basic and digest authentication
@@ -153,7 +153,7 @@ Rails doesn't make you do everything yourself. It has a method called `#has_sec
To initialize a new user session (when your user signs in), you'll need to create a new controller (usually `sessions_controller.rb`) and the corresponding routes for `:new`, `:create` and `:destroy`. If the user passes the correct credentials (which we can check using the `#authenticate` method), you'll use the `session` variable to store their ID, which you can use to validate that they are who they say they are. This is a way of authenticating the user that uses Rails' existing session infrastructure, but only lasts as long as the session does.
-If your user wants to be "remembered" (you've probably seen the "remember me" checkbox plenty of times on login forms), you need a way to remember them for longer than just the length of the browser session. To do this, you'll need to create another column in your Users table for an encrypted `remember_token` (or whatever you'd like to call it). You'll use that to store a random string for that user that will be used in the future to identify him/her.
+If your user wants to be "remembered" (you've probably seen the "remember me" checkbox plenty of times on login forms), you need a way to remember them for longer than just the length of the browser session. To do this, you'll need to create another column in your Users table for an encrypted `remember_token` (or whatever you'd like to call it). You'll use that to store a random string for that user that will be used in the future to identify them.
You will drop the unencrypted token as a permanent cookie (using `cookies.permanent[:remember_token]`) into the user's browser. That cookie will be submitted with each new request, so you can check with the encrypted version in the database to see who that user is whenever they make a request. This is basically a more explicit and permanent version of what Rails is doing with sessions. It's best practice to reset the token on each new signin if the user signs out.
@@ -166,7 +166,7 @@ A generic step-by-step overview:
1. Don't forget any necessary validations for password and password confirmation length.
1. Build a sessions controller (and corresponding routes) and use the `#authenticate` method to sign in the user when the user has submitted the proper credentials using the signin form.
1. Allow the user to be remembered by creating a `remember_token` column in the Users table and saving that token as a permanent cookie in the user's browser. Reset on each new signin.
-1. On each page load that requires authentication (and using a `#before_action` in the appropriate controller(s)), first check the user's cookie `remember_token` against the database to see if he's already signed in. If not, redirect to the signin page.
+1. On each page load that requires authentication (and using a `#before_action` in the appropriate controller(s)), first check the user's cookie `remember_token` against the database to see if they're already signed in. If not, redirect to the signin page.
1. Make helper methods as necessary to let you do things like easily determine if a user is signed in or compare another user to the currently signed in user.
1. Profit.
@@ -176,22 +176,22 @@ A generic step-by-step overview:
In a short word, Devise prepackages for you a bunch of signin and signup forms and methods to help implement them. It's made up of 10 modules (and you can choose which ones you want to use). You install the `devise` gem and run the installer to drop their files into your application. You'll also need to run a database migration to add their additional fields to your Users table.
-Configuration will be dependent on your use case. Do you want to make the user confirm their signup using an email (the `Confirmable` module)? Do you want to allow the user to reset his password (the `Recoverable` module)?
+Configuration will be dependent on your use case. Do you want to make the user confirm their signup using an email (the `Confirmable` module)? Do you want to allow the user to reset their password (the `Recoverable` module)?
### Assignment
-
-1. Read [this article about how Rails sessions work](https://www.justinweiss.com/articles/how-rails-sessions-work/).
-2. Watch [this video to dive deep into sessions](https://www.youtube.com/watch?v=mqUbnZIY3OQ).
-3. Read sections 5 and 6 of the [Rails Guides on Controllers](http://guides.rubyonrails.org/action_controller_overview.html#session). Don't worry too much about the details of `session_store` configurations in 5.1 right now.
-4. Read section 8 of the [Rails Guides on Controllers](http://guides.rubyonrails.org/action_controller_overview.html#filters) to understand controller filters.
-5. Read section 11 of the [Rails guides on Controllers](http://guides.rubyonrails.org/action_controller_overview.html#http-authentications) to understand more about authentication.
-6. Glance over the [Devise Documentation](https://github.com/plataformatec/devise). Read about how to install it in your Rails App and what the different modules do. You'll be using it with upcoming projects.
+
+1. Read this article about [how Rails sessions work](https://www.justinweiss.com/articles/how-rails-sessions-work/).
+1. Watch this video to [dive deep into sessions](https://www.youtube.com/watch?v=mqUbnZIY3OQ).
+1. Read sections 5 and 6 of the [Rails Guides on Controllers](http://guides.rubyonrails.org/action_controller_overview.html#session). Don't worry too much about the details of `session_store` configurations in 5.1 right now.
+1. Read section 8 of the [Rails Guides on Controllers](http://guides.rubyonrails.org/action_controller_overview.html#filters) to understand controller filters.
+1. Read section 11 of the [Rails guides on Controllers](http://guides.rubyonrails.org/action_controller_overview.html#http-authentications) to understand more about authentication.
+1. Glance over the [Devise Documentation](https://github.com/plataformatec/devise). Read about how to install it in your Rails App and what the different modules do. You'll be using it with upcoming projects.
-### Conclusion
+#### Conclusion
Authentication can appear to be a fairly complicated topic -- there are a lot of moving parts. At it's core, though, you're just checking whether the person making a request is actually a signed in user who has the permissions to do so, all by using browser cookies in some form or another.
@@ -199,17 +199,17 @@ This lesson should have given you some appreciation for how complicated login sy
### Knowledge check
-This section contains questions for you to check your understanding of this lesson on your own. If you’re having trouble answering a question, click it and review the material it links to.
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- How would you set a cookie for hair color on a user's browser?
-- How would you require a user is logged in _before_ running some code?
-- Would you use Basic HTTP Authentication for authenticating users over alternatives such as the Devise gem?
-- How would you flash an error message on a user's browser if they put an invalid username?
-- What are some reasons you would want to use the Devise gem for user authentication over building your own authorization system?
+- [How would you set a cookie for hair color on a user's browser?](#using-cookies)
+- [How would you require a user is logged in *before* running some code?](#login-check)
+- [Would you use Basic HTTP Authentication for authenticating users over alternatives such as the Devise gem?](#http-authentication)
+- [How would you flash an error message on a user's browser if they put an invalid username?](#flash-message)
+- [What are some reasons you would want to use the Devise gem for user authentication over building your own authorization system?](#using-devise)
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- Authentication in Rails 3.1 from [Railscasts](http://railscasts.com/episodes/270-authentication-in-rails-3-1)... better than we can explain it.
+- [Authentication in Rails 3.1 from Railscasts](http://railscasts.com/episodes/270-authentication-in-rails-3-1)... better than we can explain it.
- [All About Cookies (.org)](http://www.allaboutcookies.org/)
diff --git a/ruby_on_rails/introduction/project_installing_rails.md b/ruby_on_rails/introduction/project_installing_rails.md
index 0bbd7d721b7..a5aceacdfe5 100644
--- a/ruby_on_rails/introduction/project_installing_rails.md
+++ b/ruby_on_rails/introduction/project_installing_rails.md
@@ -1,15 +1,18 @@
+
+
+
### Introduction
Ready to jump into Rails? Let's get started by installing it and creating your first Rails application!
Before continuing, let's review a few best practices to keep in mind:
-* Follow the directions closely, and don't skip over any sections.
-* **Do NOT use `sudo` unless The Odin Project specifically says to do so.** Failing to follow this advice can cause *a lot* of headaches. In some instances, you might see a message in the terminal telling you to use `sudo` and/or to install something with `apt`. **Ignore what the terminal says** and follow the instructions below.
-* Copy and paste the commands to avoid typos.
-* If you stop working on this project partway through and come back to it later, be sure to use `cd` to move back inside your project directory so that the commands will work.
+- Follow the directions closely, and don't skip over any sections.
+- **Do NOT use `sudo` unless The Odin Project specifically says to do so.** Failing to follow this advice can cause *a lot* of headaches. In some instances, you might see a message in the terminal telling you to use `sudo` and/or to install something with `apt`. **Ignore what the terminal says** and follow the instructions below.
+- Copy and paste the commands to avoid typos.
+- If you stop working on this project partway through and come back to it later, be sure to use `cd` to move back inside your project directory so that the commands will work.
-In this project, we're going to build a fully functional Rails application. The entire point of this exercise is to make sure that you have everything installed and working correctly on your computer. Do *not* worry if you don't fully understand what you're doing. You'll learn exactly what all of these commands are doing later on in the course. For now, go slowly, and be sure to follow **each and every** step closely. If you run into trouble, don't forget that you can always reach out for help on [Discord](https://discord.gg/fbFCkYabZB). You can also use the [Discord search function](https://support.discordapp.com/hc/en-us/articles/115000468588-Using-Search) to check if someone else had a similar problem and how they solved it.
+In this project, we're going to build a fully functional Rails application. The entire point of this exercise is to make sure that you have everything installed and working correctly on your computer. Do *not* worry if you don't fully understand what you're doing. You'll learn exactly what all of these commands are doing later on in the course. For now, go slowly, and be sure to follow **each and every** step closely. If you run into trouble, don't forget that you can always reach out for help on [our Discord server](https://discord.gg/fbFCkYabZB). You can also use the [Discord search function](https://support.discordapp.com/hc/en-us/articles/115000468588-Using-Search) to check if someone else had a similar problem and how they solved it.
### Your first Rails app
@@ -21,15 +24,15 @@ Don't worry if you don't totally understand what you are doing in these next ste
We've previously installed Ruby, and now it's time to install Ruby on Rails. It's just running one command!
-~~~bash
+```bash
gem install rails
-~~~
+```
Once the installation finishes, you can check if everything went well by restarting your terminal and running the following command:
-~~~bash
+```bash
rails -v
-~~~
+```
This should display the version of Rails installed on your system indicating the installation went smoothly.
@@ -45,9 +48,9 @@ Visit [The Yarn Download Page](https://classic.yarnpkg.com/en/docs/install#windo
You can verify the install by running the following command:
-~~~bash
+```bash
yarn --version
-~~~
+```
If you don't get a version number drop by the chatrooms for some assistance.
@@ -57,14 +60,14 @@ This is where it might start to be difficult to follow just what is happening. I
We're going to start by navigating to the directory that you house your projects in, then telling Rails to initialize the application for us.
-~~~bash
+```bash
cd ~/your_odin_project_directory
rails new my_first_rails_app
-~~~
+```
This will do a bunch of things, and you'll see a lot of output in the terminal. If everything worked, you should see something similar to this around the end of the terminal output:
-~~~bash
+```bash
Bundle complete! 16 Gemfile dependencies, 76 gems now installed.
Use `bundle info [gemname]` to see where a bundled gem is installed.
@@ -80,31 +83,25 @@ Import Stimulus controllers
Pin Stimulus
append config/importmap.rb
-~~~
+```
In the above process, Rails created a new directory for us. Let's `cd` into it now:
-~~~bash
+```bash
cd my_first_rails_app
-~~~
-
-Now, we're going to tell Rails to generate some templates for us. This will get us up and running in no time at all. If you are using Ruby 2.7 or higher then you may see some deprecation warnings that look like errors in the console. Ruby made some changes in version 2.7 to deprecate using hashes as the last argument of a method call. You can read more about it [here](https://www.ruby-lang.org/en/news/2019/12/12/separation-of-positional-and-keyword-arguments-in-ruby-3-0/). It will take time for gems to update their codebases to deal with this deprecation, especially if they are as large as Rails. If you do see any deprecation warnings then don't worry, they will get fixed eventually. The warnings will look something like:
-
-~~~bash
-warning: Using the last argument as keyword parameters is deprecated; maybe ** should be added to the call
-~~~
+```
Run the following in the terminal:
-~~~bash
+```bash
rails generate scaffold car make:string model:string year:integer
-~~~
+```
After generating the scaffolds, we need to migrate the database.
-~~~bash
+```bash
rails db:migrate
-~~~
+```
#### Step 1.4: Start up your app
@@ -112,14 +109,16 @@ Now that you have created a Rails application, you can start it up and see if it
In the terminal, type
-~~~bash
+```bash
rails server
-~~~
+```
-Now, open a browser and visit [http://localhost:3000/cars](http://localhost:3000/cars) to see your application!
+Now, open a browser and visit [http://localhost:3000/cars](http://localhost:3000/cars) to see your application!
+
If you're using a VM, you will need to open the browser inside of your VM in order for this to work.
+
Go ahead and create a new car, and then refresh the page to verify it is working. Add as many cars as you'd like! When you're satisfied, go back to the terminal where the Rails server is running, and type Ctrl + C to close the server.
@@ -129,24 +128,27 @@ Go ahead and create a new car, and then refresh the page to verify it is working
Like all of the projects you've done so far we need to use Git for our version control and to push our app to different remotes.
#### Step 2.1 Stage and commit local changes
+
Rails will already have initialized Git for you when it was doing its thing, so just stage and commit all of the work it's done so far.
-~~~bash
+```bash
git add .
git commit -m "Initial commit"
-~~~
+```
#### Step 2.2 Initialize on GitHub, add the remote, and push
+
Make a repo on Github and make sure you **do not** initialize the repository with a README because Rails has created one already. Add this repo as a remote and push your repo to GitHub just like you normally do.
REMINDER: Do not enter the `<` or `>` symbols below. Replace those symbols and everything between them with the URL that you copied from GitHub.
-~~~bash
+```bash
git remote add origin
git push -u origin main
-~~~
+```
#### Step 2.3 Confirm Git is working correctly
+
Confirm that the push was successful and on GitHub you see all the folders and files made locally by Rails, starting with a folder called "app".
This marks the start of your Rails journey! Later on, you'll be able to look back at this repository and marvel over how far you've come!
diff --git a/ruby_on_rails/introduction/web_refresher.md b/ruby_on_rails/introduction/web_refresher.md
index f2266ebe409..d4b320a4d8f 100644
--- a/ruby_on_rails/introduction/web_refresher.md
+++ b/ruby_on_rails/introduction/web_refresher.md
@@ -6,14 +6,14 @@ To really understand how Rails works, you need to have a solid base in the guts
This section contains a general overview of topics that you will learn in this lesson.
-- The basics of HTTP.
-- The 4 most commonly used HTTP verbs.
-- The 7 RESTful routes of Rails.
-- The different components of a URL.
-- The basics of MVC.
-- What an API is.
-- What "cookies" and "sessions" are.
-- The difference between "authentication" and "authorization".
+- The basics of HTTP.
+- The 4 most commonly used HTTP verbs.
+- The 7 RESTful routes of Rails.
+- The different components of a URL.
+- The basics of MVC.
+- What an API is.
+- What "cookies" and "sessions" are.
+- The difference between "authentication" and "authorization".
### HTTP
@@ -21,9 +21,9 @@ HTTP is just a way of structuring the request-and-response conversation between
Check out these resources:
-1. This [tutsplus post on HTTP](http://code.tutsplus.com/tutorials/http-the-protocol-every-web-developer-must-know-part-1--net-31177) describes what's going on with HTTP.
-2. This [sniffer tool](http://testuri.org/sniffer) - try retrieving a couple of websites (like http://www.theodinproject.com) on your own.
-3. This [great video](https://code.tutsplus.com/how-to-become-a-web-developer--CRS-200371c/http-and-the-web-server) on communications between http requests and the web server.
+1. This [tutsplus post on HTTP](http://code.tutsplus.com/tutorials/http-the-protocol-every-web-developer-must-know-part-1--net-31177) describes what's going on with HTTP.
+1. This [sniffer tool](http://testuri.org/sniffer) - try retrieving a couple of websites (like `https://www.theodinproject.com/`) on your own.
+1. This great video on [communications between http requests and the web server](https://code.tutsplus.com/how-to-become-a-web-developer--CRS-200371c/http-and-the-web-server).
One key component to pay attention to is the fact that the request and response both have header and (usually) body components. The header contains information about the request or response itself (meta data), including which website to send or return to and what the status of the response is. The body of the request can contain things like data submitted by a form or cookies or authentication tokens while the response will usually contain the HTML page you're trying to access.
@@ -33,13 +33,13 @@ The other key component is that each request uses one of four main "verbs" -- GE
REST (short for Representational state transfer) is a term that you'll see coming up again and again because it's a very powerful idea. It basically says that there are really only 7 different types of things that you usually want to do to an individual resource via the web and you can do them by mixing and matching the HTTP verbs we just covered. A "resource" usually means a "thing" in your database or a data model. In this case, we'll assume that resource is a blog Post model that you've set up:
-1. GET all the posts (aka **"index"** the posts)
-2. GET just one specific post (aka **"show"** that post)
-3. GET the page that lets you create a new post (aka view the **"new"** post page)
-4. POST the data you just filled out for a new post back to the server so it can create that post (aka **"create"** the post)
-5. GET the page that lets you edit an existing post (aka view the **"edit"** post page)
-6. PUT (or PATCH) the data you just filled out for editing the post back to the server so it can actually perform the update (aka **"update"** the post)
-7. DELETE one specific post by sending a delete request to the server (aka **"destroy"** the post)
+1. GET all the posts (aka **"index"** the posts)
+1. GET just one specific post (aka **"show"** that post)
+1. GET the page that lets you create a new post (aka view the **"new"** post page)
+1. POST the data you just filled out for a new post back to the server so it can create that post (aka **"create"** the post)
+1. GET the page that lets you edit an existing post (aka view the **"edit"** post page)
+1. PUT (or PATCH) the data you just filled out for editing the post back to the server so it can actually perform the update (aka **"update"** the post)
+1. DELETE one specific post by sending a delete request to the server (aka **"destroy"** the post)
The highlighted words correspond to standard Rails controller actions!
@@ -53,24 +53,25 @@ It may seem simplistic to you up front to think of things this way, but once you
You may think you know what's in a URL, but which part is the host? protocol (aka scheme)? parameters? path?
-Check out this [article by Matt Cutts](http://www.mattcutts.com/blog/seo-glossary-url-definitions/) on how Googlers pick apart URL components.
+Check out this article by Matt Cutts on [how Googlers pick apart URL components](http://www.mattcutts.com/blog/seo-glossary-url-definitions/).
-**Quick quiz:**
-The URL is: https://www.google.com/search?q=what+is+a+url
+#### Quick quiz
-1. What is the "Path"?
-2. What is the "Parameter" portion?
-3. What is the "Top Level Domain"?
-4. What is the "Protocol"?
+The URL is:
+
+1. What is the "Path"?
+1. What is the "Parameter" portion?
+1. What is the "Top Level Domain"?
+1. What is the "Protocol"?
Once you understand what these components are, you can easily use Ruby's libraries to help you build your own and send requests. You also run into specific pieces like the "path" and "parameters" again and again when using Rails.
Answers:
-1. /search
-2. q=what+is+a+url
-3. com
-4. https
+1. /search
+1. q=what+is+a+url
+1. com
+1. https
### MVC
@@ -84,12 +85,12 @@ The point of MVC is that the functions of a web application can be broken down i
Once a request from a browser comes into your application, at the most basic level:
-1. The router figures out which controller to send it to (e.g. for your blog, the Posts controller).
-2. That controller asks the model (e.g. Post model) for data and any other tough questions it has.
-3. Then that controller passes off whatever data it needs to the views (e.g. `index.html.erb`), which are basically just HTML templates that are waiting for those variables.
-4. Once the proper view has been pumped full of the data it needs (like the current user's name), it gets sent back to the client that made the original request. Presto!
+1. The router figures out which controller to send it to (e.g. for your blog, the Posts controller).
+1. That controller asks the model (e.g. Post model) for data and any other tough questions it has.
+1. Then that controller passes off whatever data it needs to the views (e.g. `index.html.erb`), which are basically just HTML templates that are waiting for those variables.
+1. Once the proper view has been pumped full of the data it needs (like the current user's name), it gets sent back to the client that made the original request. Presto!
-Check out a more detailed version of MVC in [betterexplained.com's](http://betterexplained.com/articles/intermediate-rails-understanding-models-views-and-controllers/) article.
+betterexplained.com has a [deeper explanation of MVC](http://betterexplained.com/articles/intermediate-rails-understanding-models-views-and-controllers/).
To characterize the three (badly), the model is the supersmart geek in the back room, the controller is the social middleman that talks to everyone but doesn't really do anything too intensive (it asks the model in those cases), and the view just looks pretty and waits to get its outfit from the controller.
@@ -113,7 +114,7 @@ But we get ahead of ourselves a bit here... the main point is that you'll see "A
You've heard about cookies. Cookies are basically a way for websites to remember who you are from one request to another. Remember -- every HTTP request is totally independent of each other. Meaning that when you go to the Home page of a website and then click on a link to their About page, the web server treats you as a completely new user.
-...Unless they've given you some cookies (which they almost certainly have). Cookies are little bits of data that your browser sends to the website every time you make a request to it. From the perspective of the web server, it lets the server identify you as the same person who made any of a series of previous requests. It preserves the _state_ of your session.
+...Unless they've given you some cookies (which they almost certainly have). Cookies are little bits of data that your browser sends to the website every time you make a request to it. From the perspective of the web server, it lets the server identify you as the same person who made any of a series of previous requests. It preserves the *state* of your session.
Check out [allaboutcookies.org](http://www.allaboutcookies.org/) and read the first three sections for some more info about cookies.
@@ -133,7 +134,7 @@ It's also how some ads seem to follow you from one website to another -- another
On the server side, you'll interact with cookies and session variables quite a bit. As mentioned above, one of the main uses of these is to determine who the user is, or "authentication". You'll basically retrieve the cookie that the user sends you, use it to find that user in your database, and (if the user exists) then you can display the customized web page for that user.
-It's pretty straightforward in theory, but some of the security implications get a bit hairy so luckily some nice folks created a very handy gem called ["Devise"](https://github.com/heartcombo/devise) which takes care of all this stuff for you. In this curriculum (a bit later on), you'll be creating your own authentication system before learning how to use Devise to take care of the heavy lifting.
+It's pretty straightforward in theory, but some of the security implications get a bit hairy so luckily some nice folks created a very [handy gem called Devise](https://github.com/heartcombo/devise) which takes care of all this stuff for you. In this curriculum (a bit later on), you'll be creating your own authentication system before learning how to use Devise to take care of the heavy lifting.
### Authorization
@@ -147,20 +148,20 @@ We'll dig into this stuff a bit later, but it's good to understand in the contex
### Knowledge check
-This section contains questions for you to check your understanding of this lesson on your own. If you’re having trouble answering a question, click it and review the material it links to.
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
-- What do you call an HTTP message that goes from client to server?
-- What do you call an HTTP message that goes from server to client?
-- Which HTTP message would include a status code and which would include an action verb?
-- What is the name of the additional information that is added after the path of a URL?
-- What does MVC stand for?
-- What is an "API"?
-- Why do you need "cookies" to continue your "session"?
-- What is the difference between "authentication" and "authorization"?
+- [What do you call an HTTP message that goes from client to server?](https://code.tutsplus.com/tutorials/http-the-protocol-every-web-developer-must-know-part-1--net-31177)
+- [What do you call an HTTP message that goes from server to client?](https://code.tutsplus.com/tutorials/http-the-protocol-every-web-developer-must-know-part-1--net-31177)
+- [Which HTTP message would include a status code and which would include an action verb?](https://code.tutsplus.com/tutorials/http-the-protocol-every-web-developer-must-know-part-1--net-31177)
+- [What is the name of the additional information that is added after the path of a URL?](https://www.mattcutts.com/blog/seo-glossary-url-definitions/)
+- [What does MVC stand for?](https://betterexplained.com/articles/intermediate-rails-understanding-models-views-and-controllers/)
+- [What is an "API"?](https://money.howstuffworks.com/business-communications/how-to-leverage-an-api-for-conferencing1.htm)
+- [Why do you need "cookies" to continue your "session"?](https://en.wikipedia.org/wiki/HTTP_cookie#Session_management)
+- [What is the difference between "authentication" and "authorization"?](#authorization)
### Additional resources
-This section contains helpful links to other content. It isn't required, so consider it supplemental.
+This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- [HTTP explained by Harvard's David Malan](http://www.youtube.com/watch?v=8KuO4r5CHjM)
-- [HTTP Request/Response Basics](http://justahelp.blogspot.com/2013/09/http-requestresponse-basics.html) from Pralay Roy
+- [HTTP explained by Harvard's David Malan](http://www.youtube.com/watch?v=8KuO4r5CHjM)
+- [HTTP Request/Response Basics](http://justahelp.blogspot.com/2013/09/http-requestresponse-basics.html) from Pralay Roy
diff --git a/ruby_on_rails/mailers_advanced_topics/actioncable_lesson.md b/ruby_on_rails/mailers_advanced_topics/actioncable_lesson.md
index 5e8fe1f2620..89f604d4a71 100644
--- a/ruby_on_rails/mailers_advanced_topics/actioncable_lesson.md
+++ b/ruby_on_rails/mailers_advanced_topics/actioncable_lesson.md
@@ -8,86 +8,86 @@ The app we'll build will be in two stages. The first stage will be an app for se
First let's create the app
-~~~bash
+```bash
rails new messenger -T
-~~~
+```
Add devise to the Gemfile
-~~~bash
+```bash
bundle add devise
-~~~
+```
-`bundle add` both adds the gem to our Gemfile and runs bundle install. The good thing about it is it also sets the gem version to its latest variant. The down side is that it adds the gems to the bottom of the Gemfile by default. So use it with caution. There are a few flags that can be used. You can have a read about it [here](https://bundler.io/man/bundle-add.1.html).
+`bundle add` both adds the gem to our Gemfile and runs bundle install. The good thing about it is it also sets the gem version to its latest variant. The down side is that it adds the gems to the bottom of the Gemfile by default. So use it with caution. [The `bundle add` command has a few flags you can use](https://bundler.io/man/bundle-add.1.html).
Run the devise installer
-~~~bash
+```bash
bundle exec rails generate devise:install
-~~~
+```
Open `config/environments/development.rb` and add the following line
-~~~ruby
+```ruby
config.action_mailer.default_url_options = { host: 'localhost', port: 3000 }
-~~~
+```
Generate our user model and migrate
-~~~bash
+```bash
bundle exec rails generate devise user && bundle exec rails db:migrate
-~~~
+```
To get things started we'll just create a couple of users in `db/seeds.rb`
-~~~ruby
+```ruby
User.create(email: 'humblebragger@humblebrag.com', password: 'humblebaby')
User.create(email: 'fitnessgrampacer@test.com', password: 'pacertest')
-~~~
+```
You can set the user info to whatever you want. Then run the seed file
-~~~
+```bash
bundle exec rails db:seed
-~~~
+```
Let's create a basic one page app, users will land on our homepage, get redirected to log in, and then directed back to the homepage. First let's create a controller. We'll just call it `hangouts` as our users will hangout there doing absolutely nothing.
-~~~bash
+```bash
bundle exec rails generate controller hangouts index
-~~~
+```
Set the root to our index page in `routes.rb`
-~~~ruby
+```ruby
root 'hangouts#index'
-~~~
+```
Let's also protect our app from unauthorised viewing by adding the following to `app/controllers/application_controller.rb`
-~~~ruby
+```ruby
before_action :authenticate_user!
-~~~
+```
Lastly in the initial setup, let's add a little bit of styling with the Bulma gem and a navbar just with some login and logout functionality. We'll also install dartsass-rails so that Bulma works appropriately. (Note: If you receive this error - `LoadError: cannot load such file -- sassc` run `rails tmp:clear` and try restarting the server)
-~~~bash
+```bash
bundle add bulma-rails
./bin/bundle add dartsass-rails
./bin/rails dartsass:install
-~~~
+```
Next, and just to save some time you need to go to the `app/assets/stylesheets/` directory and change `application.css` to `application.scss`. Bulma-rails uses scss and although there are other ways to handle it this is the easiest.
Now open the `application.scss` file and right at the bottom add
-~~~css
+```css
@import "bulma";
-~~~
+```
Lastly, to add the navbar functionality we can just add it straight into the main layout file. Open `app/views/layouts/application.html.erb` add the following code right below the opening `` tag and above the `<%= yield %>` line.
-~~~html
+```html
@@ -101,7 +101,7 @@ Lastly, to add the navbar functionality we can just add it straight into the mai
-~~~
+```
This mostly is just using Bulma stylings to make it look nice.
@@ -113,7 +113,7 @@ Let's now deal with setting up the server connection. As we covered in the Actio
If you open up `app/channels/application_cable/connection.rb` we can authorise a connection when the user logs in. The code is pretty identical to the [connection](https://guides.rubyonrails.org/action_cable_overview.html#server-side-components-connections) found in the Rails Guides on Action Cable, with the only difference being we can use the user details set on the warden object environment variable.
-~~~ruby
+```ruby
module ApplicationCable
class Connection < ActionCable::Connection::Base
identified_by :current_user
@@ -133,30 +133,30 @@ module ApplicationCable
end
end
end
-~~~
+```
This is used to establish a connection between client and server but it won't do anything until you create a channel. It is the channel that makes a request to the websocket server that then starts the connection authorization.
We can create a channel for messages. In the terminal we can write
-~~~bash
+```bash
bundle exec rails generate channel message
-~~~
+```
You can see from the output it creates some files for us
-~~~bash
+```bash
create app/channels/message_channel.rb
identical app/javascript/channels/index.js
identical app/javascript/channels/consumer.js
create app/javascript/channels/message_channel.js
-~~~
+```
The index and consumer js files already exist. Nice! But it has created a message_channel.rb and message_channel.js file
In `app/channels/message_channel.rb` let's establish our stream. Remember, streams are how Rails publishes broadcasts to subscribers. We won't be operating on a model so we won't need `stream_for`. We can instead use `stream_from` to just stream to our message room.
-~~~ruby
+```ruby
class MessageChannel < ApplicationCable::Channel
def subscribed
stream_from 'message'
@@ -166,13 +166,13 @@ class MessageChannel < ApplicationCable::Channel
# Any cleanup needed when channel is unsubscribed
end
end
-~~~
+```
Now that our stream is setup you should see in the server logs when you refresh that you are connected to the channel with `MessageChannel is streaming from message`. Nice.
The next thing we need is a display for our messages and a form in which to post them on our index page. Open up `app/views/hangouts/index.html.erb` and add the following code
-~~~html
+```html
@@ -189,13 +189,13 @@ The next thing we need is a display for our messages and a form in which to post
-~~~
+```
-We've written a div with an id of `message-display` where we can display the messages as they are created and then we have a form. The form doesn't have an action because we aren't posting it anywhere. We'll need to grab a reference to the form in our javascript and when it's submitted we can post a message to the websocket server. We use a basic html form because we don't have an object or url to work with so Rails form helpers aren't much use to us. We can change this later when we do create a Message object to work with. All the class names are Bulma stylings so don't worry too much about the markup.
+We've written a div with an id of `message-display` where we can display the messages as they are created and then we have a form. The form doesn't have an action because we aren't posting it anywhere. We'll need to grab a reference to the form in our JavaScript and when it's submitted we can post a message to the websocket server. We use a basic HTML form because we don't have an object or url to work with so Rails form helpers aren't much use to us. We can change this later when we do create a Message object to work with. All the class names are Bulma stylings so don't worry too much about the markup.
I'm not going to stress about the page looking amazing. But just to make the message display area a decent size open up `app/assets/stylesheets/hangouts.scss` and add the following code.
-~~~css
+```css
#message-display {
height: 80vh;
overflow: auto
@@ -204,11 +204,11 @@ I'm not going to stress about the page looking amazing. But just to make the mes
#message-input {
width: 60vw;
}
-~~~
+```
Now when the form is submitted, we need to intercept and handle it. Open up our `app/javascript/channels/message_channel.js` and take a look at the code as it stands.
-~~~js
+```javascript
import consumer from "./consumer"
const messageChannel = consumer.subscriptions.create("MessageChannel", {
@@ -221,13 +221,13 @@ const messageChannel = consumer.subscriptions.create("MessageChannel", {
received(data) {
}
});
-~~~
+```
-Yours will have some comments that aren't important, but you will need to verify that the constant `messageChannel` is declared. What we need to do here first is set an event listener on the form submit so we can grab the value entered. Once we have the value, we want to send it to the server to be broadcast to all the channel subscribers. This is called rebroadcasting and you can see how it works [In the Rails Guides](https://guides.rubyonrails.org/action_cable_overview.html#rebroadcasting-a-message). We can use `MessageChannel.send` to send the value entered in the input box to our websocket server.
+Yours will have some comments that aren't important, but you will need to verify that the constant `messageChannel` is declared. What we need to do here first is set an event listener on the form submit so we can grab the value entered. Once we have the value, we want to send it to the server to be broadcast to all the channel subscribers. This is what we call [rebroadcasting](https://guides.rubyonrails.org/action_cable_overview.html#rebroadcasting-a-message). We can use `MessageChannel.send` to send the value entered in the input box to our websocket server.
Right at the bottom of our `message_channel.js` file add the following code
-~~~js
+```javascript
document.addEventListener("turbo:load", () => {
let form = document.querySelector('#message-form')
if(form) {
@@ -242,33 +242,33 @@ document.addEventListener("turbo:load", () => {
})
}
})
-~~~
+```
-Firstly, since JS assets are loaded in the head of our html, the Javascript files will be evaluated often before the DOM has rendered, so we first want to make sure the DOM has been created. We do this using the `turbo:load` event listener. Then we want to check if we are on a page with the form. It might seem obvious since we only have one page but you can never be too careful...
+Firstly, since JS assets are loaded in the head of our HTML, the JavaScript files will be evaluated often before the DOM has rendered, so we first want to make sure the DOM has been created. We do this using the `turbo:load` event listener. Then we want to check if we are on a page with the form. It might seem obvious since we only have one page but you can never be too careful...
Next if we are on a page with the form we add an event listener to it on its submit property. When the form is submitted we first prevent the default submit from happening. This stops the page rerendering. Then we grab the value from our input field and check it isn't an empty string. If it is, then we return and do nothing. If it isn't, then we create a JS object with a key named `body` which holds the value of our messageInput. If you're wondering why we did this under its own object instead of just sending `message: messageInput` in the send method it's because often you'll submit more than one parameter, and when we do create a Message Model this is similar to how it will be returned from a controller.
The last part of our code is the important bit. We call `send` on `messageChannel` and pass it an object with a key of `message` and a value of the message object we create earlier. You can see we create `messageChannel` in this line `const messageChannel = consumer.subscriptions.create("MessageChannel", {`. So it holds a reference to the channel created for message. When we call `send` on it, Rails knows to route it to the message_channel on the server side. If you save this file and refresh your browser if you have a rails server running you should be able to submit a message without the page refreshing. Check the server logs and you should see something like
-~~~bash
+```bash
Unable to process MessageChannel#receive({"message"=>{"body"=>"hello"}})
-~~~
+```
This is because we're missing the server side handling of our message submitted to the server. The Rails Guides section linked above on rebroadcasting shows a `receive` method on the server side to handle incoming messages from the client.
Open up `app/channels/message_channel.rb` and underneath the `unsubscribed` method lets add the receive method
-~~~ruby
+```ruby
def receive(data)
ActionCable.server.broadcast('message', data)
end
-~~~
+```
Rails handles routing any incoming message from a client to the `receive` method on the class representing the channel through which the message was sent. In our `receive` method we broadcast the message right back to all subscribers of the `message` stream. Sending it right back might seem strange but it's how we can ensure a message created on one client gets sent to all other clients subscribed to the channel without them having to refresh their browser. This is one of the main benefits of WebSockets.
-When data is broadcast to subscriber it goes right back to the client side to the object handling the connection to the websocket channel. In this case it's our `messageChannel` that references the connection. Data received here goes, funnily enough, to the `received` function of our messageChannel object in `message_channel.js`. Inside `received` we need to use the data received to create a template for the message so we can display it and we then want to append it to our `#message-display` div. In order to create a template we can use Javascript string template literals. Underneath the `received` function add a new function called `template` that will receive the data and return an html string. The template does use some Bulma stylings so don't worry about the actual markup. It could be any html.
+When data is broadcast to subscriber it goes right back to the client side to the object handling the connection to the websocket channel. In this case it's our `messageChannel` that references the connection. Data received here goes, funnily enough, to the `received` function of our messageChannel object in `message_channel.js`. Inside `received` we need to use the data received to create a template for the message so we can display it and we then want to append it to our `#message-display` div. In order to create a template we can use JavaScript string template literals. Underneath the `received` function add a new function called `template` that will receive the data and return an HTML string. The template does use some Bulma stylings so don't worry about the actual markup. It could be any HTML.
-~~~js
+```javascript
const messageChannel = consumer.subscriptions.create("MessageChannel", {
connected() {
},
@@ -290,37 +290,37 @@ const messageChannel = consumer.subscriptions.create("MessageChannel", {
`
}
});
-~~~
+```
Don't forget to add a comma after the received function.
To get the message to display in our view we first find the div with the id of `message-display` that we created in our hangouts index view. We can then create the template and insert it into the document. This will happen in our `received()` function which handles incoming messages from the websocket server.
-~~~js
+```javascript
received(data) {
const messageDisplay = document.querySelector('#message-display')
messageDisplay.insertAdjacentHTML('beforeend', this.template(data))
}
-~~~
+```
If you try this now you will get an error. That is because the data received doesn't know anything about the user so in our template where we write `${data.user.email}` it won't have a reference to the user.
To fix this we need to go back to our `receive` method in `message_channel.rb`. Recall it currently looks like
-~~~ruby
+```ruby
def receive(data)
ActionCable.server.broadcast('message', data)
end
-~~~
+```
To fix this we just need to set user on data. Remember data was json that has been turned into a hash object in our Ruby code. And we have access to the user who sent the message from `current_user`.
-~~~ruby
+```ruby
def receive(data)
data['user'] = current_user
ActionCable.server.broadcast('message', data)
end
-~~~
+```
data will then be converted back to JSON as it gets sent back to the client to be handled in the js file. Not bad.
@@ -332,45 +332,45 @@ First things first. We're going to need a message model. We know from our templa
In the terminal type
-~~~bash
+```bash
bundle exec rails generate model message body:string user:references
-~~~
+```
-And then
+And then
-~~~bash
+```bash
bundle exec rails db:migrate
-~~~
+```
Next we need a controller for our messages. We only need a create action since we aren't doing anything else with them
-~~~bash
+```bash
bundle exec rails generate controller messages
-~~~
+```
Then in your `routes.rb` create the resource
-~~~ruby
+```ruby
resources :messages, only: [:create]
-~~~
+```
In our user model add the other side of the association
-~~~ruby
+```ruby
has_many :messages
-~~~
+```
We can now create a form using Rails helpers. First in our hangouts controller in the index method we can create a new message object for the form to use
-~~~ruby
+```ruby
def index
@message = Message.new
end
-~~~
+```
Then in our hangouts index view we can change the form to use `form_with`. You don't need to change anything else, just replace the form inside of the div with the `message-form` id with the following form:
-~~~html
+```html
<%= form_with model: @message, local: false do |f| %>
@@ -381,11 +381,11 @@ Then in our hangouts index view we can change the form to use `form_with`. You d
<% end %>
-~~~
+```
If you try and submit a message now it will still work. That's because in our `message_channel.js` file we still have the code hijacking the submit event and preventing the default. You should now delete the bit of the code. Open the `message_channel.js` file and delete
-~~~js
+```javascript
document.addEventListener("turbo:load", () => {
let form = document.querySelector('#message-form')
if(form) {
@@ -400,19 +400,19 @@ document.addEventListener("turbo:load", () => {
})
}
})
-~~~
+```
Submitting the form now does work in the server logs, but obviously nothing happens. Before we tackle that we firstly need to tackle displaying any current message on page load.
Go back to our hangouts controller and in the index method add the following line
-~~~ruby
+```ruby
@messages = Message.all
-~~~
+```
-Then in our index view let's iterate the messages and display them using the same html as we used in our template. In our hangouts index file in the `message-display` div add the following code.
+Then in our index view let's iterate the messages and display them using the same HTML as we used in our template. In our hangouts index file in the `message-display` div add the following code.
-~~~html
+```html
<% @messages.each do |message| %>
@@ -423,19 +423,19 @@ Then in our index view let's iterate the messages and display them using the sam
<% end %>
-~~~
+```
Kudos if you noticed the N+1 problem we just introduced. By calling `message.user` for each message. Let's fix that in the hangouts controller index method
-~~~ruby
+```ruby
@messages = Message.includes(:user)
-~~~
+```
Much better
Now let's deal with our create method in the messages controller
-~~~ruby
+```ruby
def create
@message = current_user.messages.build(message_params)
@message.save
@@ -447,7 +447,7 @@ private
def message_params
params.require(:message).permit(:body)
end
-~~~
+```
This should all be familiar with the exception of `@message.as_json(include: :user)`. To send data to the client through the websocket server it must be in a json like format. If we just supplied `@message` then Rails would call `as_json` itself for us on the object. However, since we want to include the user association we need to call it ourselves.
@@ -457,6 +457,6 @@ That's it. Try sending messages between your two users and you should see instan
### Conclusion
-The point here was to show you how Action Cable makes WebSockets pretty easy to work with. You need to have at least a basic knowledge of Javascript to handle things in the client when a message is sent but other than that you can see how powerful this can be. If you have a situation where you need to update all connected clients without waiting for them to refresh, then think WebSockets.
+The point here was to show you how Action Cable makes WebSockets pretty easy to work with. You need to have at least a basic knowledge of JavaScript to handle things in the client when a message is sent but other than that you can see how powerful this can be. If you have a situation where you need to update all connected clients without waiting for them to refresh, then think WebSockets.
This only scratches the surface of what Action Cable can do, but if you do find yourself in a situation where you think they might be useful you should be able to find what you need with a few searches.
diff --git a/ruby_on_rails/mailers_advanced_topics/advanced_topics.md b/ruby_on_rails/mailers_advanced_topics/advanced_topics.md
index 3172ad424e4..cb729c95276 100644
--- a/ruby_on_rails/mailers_advanced_topics/advanced_topics.md
+++ b/ruby_on_rails/mailers_advanced_topics/advanced_topics.md
@@ -27,45 +27,45 @@ In this case, it doesn't make a whole lot of sense to display an "index" of dash
The routes file line for a singular resource would look like:
-~~~ruby
+```ruby
# in config/routes.rb
resource :dashboard
-~~~
+```
Just note that the word "resource" is singular and so is `dashboard`. That trips up a lot of people who make the typo of writing "resource" instead of "resources" when they really want plural resources (which are more common).
The `$ rails routes` for a singular resource would only contain 6 routes (since we don't use `#index` anymore), and you would no longer see any of the `:id` portions of the routes, e.g.
-~~~bash
+```bash
edit_dashboard GET /dashboard/edit(.:format) dashboards#edit
-~~~
+```
...compared with the plural version of the same route:
-~~~bash
+```bash
edit_dashboard GET /dashboards/:id/edit(.:format) dashboards#edit
-~~~
+```
### Nested routes
Sometimes it just makes sense for one resource to be nested inside of another. For instance, a listing of lessons like this logically falls within a listing of courses -- so you'd expect a URL sort of like `http://example.com/courses/1/lessons/3`. The way to achieve this nesting is in the routes file by literally nesting one resource inside a block given to another, which might look something like:
-~~~ruby
+```ruby
# config/routes.rb
TestApp::Application.routes.draw do
resources :courses do
resources :lessons
end
end
-~~~
+```
Note that the `#resources` method now takes a block which will consist of a set of routes.
When you visit the URL, you'll have to specify the `:id` parameter for BOTH objects. The `$ rails routes` for the above would include something like:
-~~~ruby
+```ruby
course_lesson GET /courses/:course_id/lessons/:id(.:format) lessons#show
-~~~
+```
It should also be noted that you're being taken to the controller of the deepest nested resource, and that's also the `:id` parameter which will be called `:id` (any parent resource parameters, as in the above, will be specifically called something like `:course_id`).
@@ -73,14 +73,14 @@ View helpers are also automatically generated in a logical way (as you can see i
Don't nest routes too deeply! If you're more than a layer or two deep, something should be different. In fact, oftentimes you'll see only some of the controller actions nested -- only the ones that actually *need* the parent's ID to uniquely specify it. For instance, you can grab a specific Lesson by knowing only its ID. But to get all the lessons that are listed beneath a specific Course, you need the Course ID so it will have to be nested. Same is true for creating lessons, since they will need a parent specified:
-~~~ruby
+```ruby
# config/routes.rb
TestApp::Application.routes.draw do
resources :courses do
resources :lessons, :only => [:index, :create]
end
end
-~~~
+```
If this seems a bit confusing at first, you'll pick it up quickly when you actually run into it in your own coding. If you find yourself working inside your controller and needing the parent's ID, the route should have been nested. If you find that you don't need the parent's ID, it doesn't need to be nested. Easy enough.
@@ -88,7 +88,7 @@ If this seems a bit confusing at first, you'll pick it up quickly when you actua
Sometimes you want to add another non-RESTful route to a resource. If you'd like to add a route to just a single member of that resource, use the `#member` method:
-~~~ruby
+```ruby
# config/routes.rb
TestApp::Application.routes.draw do
resources :courses do
@@ -97,13 +97,13 @@ Sometimes you want to add another non-RESTful route to a resource. If you'd like
end
end
end
-~~~
+```
That route would map to the `courses#preview` action. You can add as many as you'd like.
If you'd like to add a non-RESTful route to the whole collection of your resource (so you don't need to specify the `:id` attribute, like with the `index` action), you instead use the `#collection` method:
-~~~ruby
+```ruby
# config/routes.rb
TestApp::Application.routes.draw do
resources :courses do
@@ -115,7 +115,7 @@ If you'd like to add a non-RESTful route to the whole collection of your resourc
end
end
end
-~~~
+```
The `upcoming` route will map to the `courses#upcoming` action but will not take an `:id` parameter.
@@ -125,12 +125,12 @@ If any of this seems confusing, just play around with them and run `$ rails rout
You might want to provide a URL out of convenience for your user but map it directly to another one you're already using. Use a redirect:
-~~~ruby
+```ruby
# config/routes.rb
TestApp::Application.routes.draw do
get 'courses/:course_name' => redirect('/courses/%{course_name}/lessons'), :as => "course"
end
-~~~
+```
Well, that got interesting fast. The basic principle here is to just use the `#redirect` method to send one route to another route. If your route is basic, it's a really straightforward method. But if you want to also send the original parameters, you need to do a bit of gymnastics by capturing the parameter inside `%{here}`. Note the single quotes around everything.
@@ -140,17 +140,17 @@ In the example above, we've also renamed the route for convenience by using an a
Along with the advanced routing topics covered, it can also be helpful to think about controllers in Rails that don't necessarily have their own ActiveRecord model to work with. Consider that we have a request for the application so that a `lesson` can have accompanying `images`. That seems easy enough, so we can update our model:
-~~~ruby
+```ruby
# app/models/lesson.rb
class Lesson < ApplicationRecord
# other stuff
has_many_attached :images
end
-~~~
+```
Then, we think about how we might want to manage these images from the route and controller side. We might think of something like this at first:
-~~~ruby
+```ruby
# config/routes.rb
resources :lessons do
member do
@@ -158,20 +158,20 @@ resources :lessons do
delete :remove_image
end
end
-~~~
+```
Then we have accompanying methods in the `LessonsController` to process the images. This would work well enough, but when we think about Rails controllers as standalone concepts, we might choose to implement this feature differently. Consider this second approach to the implementation:
-~~~ruby
+```ruby
# config/routes.rb
resources :lessons do
resources :images, only: [:create, :delete]
end
-~~~
+```
Along with a new controller `Lessons::ImagesController` which looks like this:
-~~~ruby
+```ruby
# app/controllers/lessons/images_controller.rb
module Lessons
class ImagesController < ApplicationController
@@ -184,7 +184,7 @@ module Lessons
end
end
end
-~~~
+```
What we've done is made the implementation more RESTful, because we no longer have any custom non-RESTful actions. Instead, we have a whole new (RESTful) controller. This controller doesn't relate to its own model to handle these actions, but works on the `Lesson` model. Not only that, by using a new controller we are able to stick to the REST actions to describe what we are doing: *creating* a new image attachment, or *destroying* an image for a lesson.
@@ -206,18 +206,18 @@ For instance, you might have a specific layout file for your static pages called
In this case, you would tell your `static_pages.html.erb` layout to call the `application.html.erb` layout but also pass it some special CSS by using the `#content_for` method, e.g.
-~~~ruby
+```erb
# app/views/layouts/static_pages.html.erb
<% content_for :stylesheets do %>
#navbar {display: none}
<% end %>
<%= render :template => "layouts/application" %>
-~~~
+```
Then your `application.html.erb` layout needs to be set up to catch that content and use it, for instance by adding this `#yield` line:
-~~~ruby
+```erb
# app/views/layouts/application.html.erb
...
@@ -227,11 +227,11 @@ Then your `application.html.erb` layout needs to be set up to catch that content
...
render :template => "static_pages.html.erb"
...
-~~~
+```
When you `#yield` to a particular content block, in this case `:stylesheets`, it will essentially drop the code from inside of that `content_for`'s block to where the `#yield` method was. So in the above example, we effectively added some CSS styling to the application layout by first rendering a special `static_pages.html.erb` layout and then passing the styles to the main `application.html.erb` layout using `#content_for`. The result would look like:
-~~~html
+```erb
# app/views/layouts/application.html.erb
...
@@ -239,7 +239,7 @@ When you `#yield` to a particular content block, in this case `:stylesheets`, it
...
-~~~
+```
This trick is useful for more than just passing stylesheet information... any time you find yourself wanting to make a section of your site look different but without totally redesigning it with a fully new layout, you might consider nesting your layouts and passing information from one to another.
@@ -253,18 +253,18 @@ The routes example almost isn't fair, though, because you wrote your `routes.rb`
Ruby provides the `#send` method to save the day. If you want to run a method on an object, just *send* that object the method and any arguments you want. A basic example you can do on your command line is `1+2`:
-~~~bash
+```bash
> 1 + 2
=> 3
> 1.send(:+, 2)
=> 3
-~~~
+```
In an ordinary situation, there's no reason to use the `#send` method but if you don't know which method you're going to need to call, it's a lifesaver. Just pass it the symbolized name of the method you want to run on that object and Ruby will go looking for it.
-But how do you define a new method on the fly anyway? In this case, you can use the `#define_method` method, which takes the symbol of what you'd like to define and a block representing the method itself. The following examples were taken from [this metaprogramming guide from ruby-metaprogramming.rubylearning.com](https://web.archive.org/web/20200801134147/http://ruby-metaprogramming.rubylearning.com/html/ruby_metaprogramming_2.html):
+But how do you define a new method on the fly anyway? In this case, you can use the `#define_method` method, which takes the symbol of what you'd like to define and a block representing the method itself. The following examples were taken from this [metaprogramming guide from ruby-metaprogramming.rubylearning.com](https://web.archive.org/web/20200801134147/http://ruby-metaprogramming.rubylearning.com/html/ruby_metaprogramming_2.html):
-~~~ruby
+```ruby
class Rubyist
define_method :hello do |my_arg|
@@ -274,13 +274,13 @@ But how do you define a new method on the fly anyway? In this case, you can use
obj = Rubyist.new
puts(obj.hello('Matz')) # => Matz
-~~~
+```
Another very powerful tool is the `#method_missing` method. You've certainly seen errors that say something to the effect of "Hey you, you tried to call a method that doesn't exist!" and the stack trace will probably run through something called `method_missing`. Most likely, you had a typo and spelled your method incorrectly.
Basically, `#method_missing` is a method of Ruby's `BasicObject` class which gets inherited by every single object in Ruby and it is called whenever you try to run a method that doesn't actually exist. It also gets passed all the arguments you tried to send and any blocks that went with it. That means that you can override `#method_missing` yourself for a given object and use whatever was previously called, for example printing out a message saying the name of the method you tried to call and its arguments:
-~~~ruby
+```ruby
class Rubyist
def method_missing(m, *args, &block)
@@ -293,9 +293,9 @@ Basically, `#method_missing` is a method of Ruby's `BasicObject` class which get
end
end
end
-~~~
+```
-~~~ruby
+```ruby
> Rubyist.new.anything
"Called anything with []"
=> nil
@@ -303,7 +303,7 @@ Basically, `#method_missing` is a method of Ruby's `BasicObject` class which get
> Rubyist.new.anything(3, 4) { "something" }
"Called anything with [3, 4] and also a block: #"
=> nil
-~~~
+```
Metaprogramming is really nifty stuff and there are tons of interesting uses for it. You don't need to master it to learn Rails, so only dive into it once you're comfortable with Rails, but it will certainly be useful to you in the real world. There are all kinds of metaprogramming tricks and patterns and tips out there but it's beyond the scope of this course to dive into them.
@@ -317,34 +317,34 @@ Design patterns have a mixed reputation among software developers. On the one h
The [Wikipedia article on SOLID](http://en.wikipedia.org/wiki/SOLID_(object-oriented_design)) provides a good overview and good links related to SOLID software design. If you're looking to write great code, you'll need to know each of the principles the letters represent (paraphrasing):
-* [**S**ingle Responsibility Principle](http://en.wikipedia.org/wiki/Single_responsibility_principle) (A class should only have a single responsibility)
-* [**O**pen/Closed Principle](http://en.wikipedia.org/wiki/Open/closed_principle) (your code entities should be open for extension but closed to modification)
-* [**L**iskov Substitution Principle](http://en.wikipedia.org/wiki/Liskov_substitution_principle) (replacing an object with one of its sub-types shouldn't break anything)
-* [**I**nterface Segregation Principle](http://en.wikipedia.org/wiki/Interface_segregation_principle) (writing many client-specific interfaces is better than one behemoth general-use interface... think APIs)
-* [**D**ependency Inversion Principle](http://en.wikipedia.org/wiki/Dependency_inversion_principle) (instead of high level constructs depending on lower level ones, make them rely on abstractions instead)
+- [Single Responsibility Principle](http://en.wikipedia.org/wiki/Single_responsibility_principle) (A class should only have a single responsibility)
+- [Open/Closed Principle](http://en.wikipedia.org/wiki/Open/closed_principle) (your code entities should be open for extension but closed to modification)
+- [Liskov Substitution Principle](http://en.wikipedia.org/wiki/Liskov_substitution_principle) (replacing an object with one of its sub-types shouldn't break anything)
+- [Interface Segregation Principle](http://en.wikipedia.org/wiki/Interface_segregation_principle) (writing many client-specific interfaces is better than one behemoth general-use interface... think APIs)
+- [Dependency Inversion Principle](http://en.wikipedia.org/wiki/Dependency_inversion_principle) (instead of high level constructs depending on lower level ones, make them rely on abstractions instead)
Luckily, Rails has done a pretty good job of following these, so you should have absorbed some good habits just through using it. But you'll want to take a minute and read up on each of them (including the odd-sounding ones) because they're fairly central to all software engineering (and a ripe interview question).
-If you're particularly interested in pursuing design patterns, check out the "Gang of Four" (GoF) Patterns laid out in [this blog post from blackwasp.co.uk](http://www.blackwasp.co.uk/GofPatterns.aspx).
+If you're particularly interested in pursuing design patterns, check out the ["Gang of Four" (GoF) Patterns](http://www.blackwasp.co.uk/GofPatterns.aspx).
There's a useful book written on anti-patterns, which can help you clean up your code by identifying bad smells, called [Rails Antipatterns](http://www.amazon.com/Rails-AntiPatterns-Refactoring-Addison-Wesley-Professional/dp/0321604814/) by Tammer Saleh and Chad Pytel.
-
### I18n: Internationalization
-[Internationalization and Localization](http://en.wikipedia.org/wiki/Internationalization_and_localization) is the process of adapting your application to fit specific geographies and/or languages. It's outside our scope to cover, but for those who are interested, check out [this Sitepoint tutorial on it](http://www.sitepoint.com/go-global-rails-i18n/), as suggested by K. Bates.
-
+[Internationalization and Localization](http://en.wikipedia.org/wiki/Internationalization_and_localization) is the process of adapting your application to fit specific geographies and/or languages. It's outside our scope to cover, but for those who are interested, check out this [Sitepoint tutorial on internationalization](http://www.sitepoint.com/go-global-rails-i18n/), as suggested by K. Bates.
### Assignment
+
1. Skim the [Rails Guide on Routing](http://guides.rubyonrails.org/routing.html#controller-namespaces-and-routing) section 2.6 about namespacing.
- 2. Read the same guide sections 2.7-3.7 to learn about nested, member and collection routes and more.
- 3. Read the same guide, sections 3.8-3.15 for a variety of different advanced routing topics including constraining the inputs to your routes and redirection.
- 4. Skim the same guide, chapter 4. Some stuff we've seen but most is just to give you a sense for what's possible. When you need it, you'll probably Google your way back there.
- 5. Read the [Rails Guide on Layouts](http://guides.rubyonrails.org/layouts_and_rendering.html#using-nested-layouts) section 3.5 to see how to pass information between your view file and your layout file, including CSS styles. Really take a minute to understand what's going on in the example there.
- 6. If you're interested in peeking at metaprogramming, read through [this resource](https://web.archive.org/web/20210514184321/http://ruby-metaprogramming.rubylearning.com/). It's not essential to building early Rails apps but you'll definitely start running into it more in "the wild".
- 7. Glance through [this Slideshare Presentation on SOLID](http://www.slideshare.net/jcfischer/solid-ruby-solid-rails) principles.
+ 1. Read the same guide sections 2.7-3.7 to learn about nested, member and collection routes and more.
+ 1. Read the same guide, sections 3.8-3.15 for a variety of different advanced routing topics including constraining the inputs to your routes and redirection.
+ 1. Skim the same guide, chapter 4. Some stuff we've seen but most is just to give you a sense for what's possible. When you need it, you'll probably Google your way back there.
+ 1. Read the [Rails Guide on Layouts](http://guides.rubyonrails.org/layouts_and_rendering.html#using-nested-layouts) section 3.5 to see how to pass information between your view file and your layout file, including CSS styles. Really take a minute to understand what's going on in the example there.
+ 1. If you're interested, take a peek at [Ruby metaprogramming](https://web.archive.org/web/20210514184321/http://ruby-metaprogramming.rubylearning.com/). It's not essential to building early Rails apps but you'll definitely start running into it more in "the wild".
+ 1. Glance through this [Slideshare Presentation on SOLID principles](http://www.slideshare.net/jcfischer/solid-ruby-solid-rails).
+
### Conclusion
@@ -354,24 +354,25 @@ In this lesson we covered some fairly random and intricate concepts but useful s
The more general principles like SOLID design and metaprogramming will be useful to you regardless of whether you stick with Ruby and Rails or move on to better and brighter things.
### Knowledge check
-This section contains questions for you to check your understanding of this lesson. If you’re having trouble answering the questions below on your own, review the material above to find the answer.
-* What would the routes file line for a singular resource look like?
-* How do you nest one resource inside another in the routes file?
-* When do you use the `#member` method?
-* When do you use a redirect?
-* What are some techniques for rendering multiple layouts for one page?
-* What does the `#send` method do?
-* What are the five design principles represented by the SOLID acronym?
+The following questions are an opportunity to reflect on key topics in this lesson. If you can't answer a question, click on it to review the material, but keep in mind you are not expected to memorize or master this knowledge.
+
+- [What would the routes file line for a singular resource look like?](#singular-resources)
+- [How do you nest one resource inside another in the routes file?](#nested-routes)
+- [When do you use the `#member` method?](#member-and-collection-routes)
+- [When do you use a redirect?](#redirects-and-wildcard-routes)
+- [What are some techniques for rendering multiple layouts for one page?](#advanced-layouts-nesting-layouts-and-passing-information)
+- [What does the `#send` method do?](#metaprogramming-rails)
+- [What are the five design principles represented by the SOLID acronym?](#design-patterns)
### Additional resources
This section contains helpful links to related content. It isn't required, so consider it supplemental.
-- [Stack Overflow question on the topic](http://stackoverflow.com/questions/6629142/having-problem-understanding-singular-resource-in-rails)
+- A Stack Overflow question on [understanding singular resource in Rails](http://stackoverflow.com/questions/6629142/having-problem-understanding-singular-resource-in-rails)
- [In Relentless Pursuit of REST by Derek Prior](https://www.youtube.com/watch?v=HctYHe-YjnE)
- [A video from Yehuda Katz on Rails Security](http://youtu.be/2Ex8EEv-WPs)
-- See the first solution to [this SO question](http://stackoverflow.com/questions/4208380/confused-on-advanced-rails-layout-nesting) for a nice way to work with multiple layouts that use classes to trigger different CSS styling.
+- The first solution in this Stack Overflow question demonstrates a nice way to [work with multiple layouts that uses classes to trigger different CSS styling](http://stackoverflow.com/questions/4208380/confused-on-advanced-rails-layout-nesting).
- [Ruby Metaprogramming](https://web.archive.org/web/20200801134147/http://ruby-metaprogramming.rubylearning.com/html/ruby_metaprogramming_2.html)
- [SO post on design patterns in Rails (2010)](http://stackoverflow.com/questions/2522065/design-patterns-in-rails)
- [A longer explanation of SOLID principles](https://www.youtube.com/watch?v=8STtzjyDTTQ)
diff --git a/ruby_on_rails/mailers_advanced_topics/conclusion.md b/ruby_on_rails/mailers_advanced_topics/conclusion.md
index 92af7a27b37..f58988cf81d 100644
--- a/ruby_on_rails/mailers_advanced_topics/conclusion.md
+++ b/ruby_on_rails/mailers_advanced_topics/conclusion.md
@@ -1,4 +1,5 @@
### Introduction
+
You've reached the end of the scope of this course and should feel pretty good about your Rails skills. That doesn't mean you should feel like you understand everything, but you should be able to build a Rails application with the kind of functionality which is actually useful in the real world. It probably won't look too pretty, but that will come in the next few courses.
This isn't the last course in the curriculum and you've got quite a way to go before hitting that "full stack" point, but it's as far as we'll get with Rails and it's pretty darn far. Despite how far you've come, it's up to you to keep building stuff, asking questions, and digging deeper into both Rails and best practices of software engineering in general.
@@ -29,26 +30,26 @@ Caching makes your application faster by reducing database calls. Check out the
Here are a few books that will take you a bit deeper as well:
-* [The Rails 5 Way](https://leanpub.com/tr5w) by Obie Fernandez is a comprehensive guide to the framework.
-* [Agile Web Development with Rails 6](https://pragprog.com/titles/rails6/agile-web-development-with-rails-6/) is sort of an in-depth guide combined with a tutorial that follows Agile practices along the way.
-* [Rails Antipatterns](http://www.amazon.com/Rails-AntiPatterns-Refactoring-Addison-Wesley-Professional/dp/0321604814) is a good best-practices guide.
+- [The Rails 5 Way](https://leanpub.com/tr5w) by Obie Fernandez is a comprehensive guide to the framework.
+- [Agile Web Development with Rails 6](https://pragprog.com/titles/rails6/agile-web-development-with-rails-6/) is sort of an in-depth guide combined with a tutorial that follows Agile practices along the way.
+- [Rails Antipatterns](http://www.amazon.com/Rails-AntiPatterns-Refactoring-Addison-Wesley-Professional/dp/0321604814) is a good best-practices guide.
### Other resources
-* [RailsCasts by Ryan Bates](http://railscasts.com/) are incredibly valuable. Great ways to actually see the implementation of various gems or practices.
-* [GoRails](https://gorails.com/) alternative to RailsCasts.
-* The [RailsApps Project](https://tutorials.railsapps.org/) provides lots of great tutorials that walk you through building real-life scenarios like integrating Devise and CanCan with your app or building a payments backend.
-* [RubyFlow community blog](http://www.rubyflow.com/) will have some interesting stuff pop up from time to time.
+- [RailsCasts by Ryan Bates](http://railscasts.com/) are incredibly valuable. Great ways to actually see the implementation of various gems or practices.
+- [GoRails](https://gorails.com/) alternative to RailsCasts.
+- The [RailsApps Project](https://tutorials.railsapps.org/) provides lots of great tutorials that walk you through building real-life scenarios like integrating Devise and CanCan with your app or building a payments backend.
+- [RubyFlow community blog](http://www.rubyflow.com/) will have some interesting stuff pop up from time to time.
### Contributing
You know more than you think. Remember when we just said that you should keep building stuff? This website is completely open source and needs your help to improve. We have a dedicated group of current and former students who help add features and proofread the curriculum. The best part is that it's completely public and free so you can watch or participate at whatever level you're comfortable.
-It's a great way to start learning about agile development methodologies and to start doing meaningful development work in a friendly and supportive environment. So check out the channel on our [Discord](https://discordapp.com/channels/505093832157691914/505093832157691916) to see what we're up to. We'd love to have you get involved!
+It's a great way to start learning about agile development methodologies and to start doing meaningful development work in a friendly and supportive environment. So check out [the general channel on our Discord server](https://discordapp.com/channels/505093832157691914/505093832157691916) to see what we're up to. We'd love to have you get involved!
### Conclusion
-I want to emphasize again that the learning never really stops but also that you've come a long way. Stick around for the next few courses and you'll take a good core and build some very interesting things. In the HTML/CSS course, you'll learn to make your Rails applications look like actual websites instead of text dumps. In the Javascript course, you'll learn how to take everyday web apps and make them much faster and more dynamic in the browser.
+I want to emphasize again that the learning never really stops but also that you've come a long way. Stick around for the next few courses and you'll take a good core and build some very interesting things. In the HTML/CSS course, you'll learn to make your Rails applications look like actual websites instead of text dumps. In the JavaScript course, you'll learn how to take everyday web apps and make them much faster and more dynamic in the browser.
So there's a lot left to go but it gets better and better. Stick with it!
diff --git a/ruby_on_rails/mailers_advanced_topics/mailers.md b/ruby_on_rails/mailers_advanced_topics/mailers.md
index 358f62cfab9..926c66a0d4e 100644
--- a/ruby_on_rails/mailers_advanced_topics/mailers.md
+++ b/ruby_on_rails/mailers_advanced_topics/mailers.md
@@ -36,7 +36,9 @@ Mailers allow you to use callbacks just like a normal controller, for instance t
In the reading you'll see how to send mail using your Gmail account, but if you're building a real application you'll obviously want something a bit more robust. There are several leading players in the space of sending your email for you. Their whole job is to handle getting your mail delivered and opened so you can focus on building your application.
-[SendGrid](https://addons.heroku.com/sendgrid#1500000) is the provider used with this website for delivering welcome emails and the like and it's pretty straightforward on Heroku. See the [documentation here](https://devcenter.heroku.com/articles/sendgrid), as well as [SendGrid's own documentation](https://docs.sendgrid.com/for-developers/sending-email/rubyonrails), which includes instructions for how you should set up your `config/environment.rb` file to get ActionMailer to interface with them. You will need to use environment variables (or the `figaro` gem) again to avoid hard coding your password and username.
+[SendGrid](https://addons.heroku.com/sendgrid#1500000) is the provider used with this website for delivering welcome emails and the like and it's pretty straightforward on Heroku.
+
+See [Heroku's documentation on Twilio SendGrid](https://devcenter.heroku.com/articles/sendgrid), as well as [SendGrid's own documentation](https://docs.sendgrid.com/for-developers/sending-email/rubyonrails), which includes instructions for how you should set up your `config/environment.rb` file to get ActionMailer to interface with them. You will need to use environment variables (or the `figaro` gem) again to avoid hard coding your password and username.
Pricing for this, as most things, is free up until a certain usage tier. While you're building toy apps, it will do just fine. Other options are out there like [MailGun](https://www.mailgun.com/), [Mailchimp](https://mailchimp.com/) and [Postmark](https://postmarkapp.com/).
@@ -46,37 +48,41 @@ You'll get a chance to play with mailers in the projects.
One key thing to note is that you don't want to fire off a bunch of emails when you're testing your app in the development environment. That's not just bad practice, it can make your users pretty unhappy and get you put on SPAM lists. No bueno. But you do want to make sure the email function is working properly. Luckily, there's a solution which is quite useful.
-The [Letter Opener gem (see docs)](https://github.com/ryanb/letter_opener), put in your `development` group of the Gemfile, will take your emails and display them in the web browser for you whenever they would otherwise be sent. You just switch a config setting in your `config/environments/development.rb` file and you're good to go. Sweet.
+The [Letter Opener gem](https://github.com/ryanb/letter_opener), put in your `development` group of the Gemfile, will take your emails and display them in the web browser for you whenever they would otherwise be sent. You just switch a config setting in your `config/environments/development.rb` file and you're good to go. Sweet.
### Email wisdom
-* Email is SLOW! It can take 1-2 seconds per email to send, so don't make your main application do it when you're serving a whole bunch of them because then anyone trying to access it will be shut out.
-* Make sure you use full URLs in any links in your mailer (so `_url` not `_path` helper methods), since the user will be opening the email and clicking the link at an external source. In your `config/environments/production.rb` file you'll want to make sure to specify your website's host name using something like `config.action_mailer.default_url_options = { :host => 'yourapp.com' }`. If it's not set, you may get an error message about your host or your links may look funny.
-* The `#deliver!` method will throw an error on delivery failure whereas `#deliver` will return false (failing silently).
-* When styling your email HTML, you won't have access to any stylesheets so you'll need to do all the styling either inline or using `