Skip to content

Latest commit

 

History

History
363 lines (240 loc) · 13 KB

components-and-jsx.md

File metadata and controls

363 lines (240 loc) · 13 KB

Title: Components And JSX
Duration: 45min+
Creator: Joe Keohan


Components and JSX

Learning Objectives

  • Identify and define React Components
  • Describe why we use Components in React
  • Build a React Component
  • Review the rules and best practices of working with Components
  • Work with JSX (JavaScript and XML) in Components

Components

The basic unit you'll be working with in React is a Component. Components are pieces of our application that we define once and can reuse throughout. They also can be extended to include additional functionality, including the following:

  • all or part of the applications state (data)
  • accept data from a parent component as props
  • leverage a Components lifecycle methods

state, props and lifecycle methods are fundamental concepts in React and we will work with them quite extensively while learning React.

Components Are Used Every

If you are familiar with Bootstrap then you already understand the concept of a Component. Bootstrap even has a section in their documentation dedicated to just components.


⏰ Activity - 2min

Let's take a look at Bootstraps section for Components.

On the left hand side you will see a link to Components and clicking on that will open the door to over 2 dozen Components.


Let's take this Bootstrap Card component for instance. It's a common design that we have all seen on the web and perhaps have even used you'reself in previous projects.

It's essentially comprised of the following html. As a developer all you would need to do is copy/paste the html and update the relevant content placeholders.

It's essentially a grouping of elements that, as a Component, is reusable.

<div class="card" style="width: 18rem;">
  <img src="..." class="card-img-top" alt="...">
  <div class="card-body">
    <h5 class="card-title">Card title</h5>
    <p class="card-text">Some quick example text to build on the card title and make up the bulk of the card's content.</p>
    <a href="#" class="btn btn-primary">Go somewhere</a>
  </div>
</div>

Bootstrap essentially allows us to pick and choose from this reusable pool of Components and to implement any web page layout/design.



React Components

Although Bootstrap comes with predefined Components out of the box, React does not, by default, provide a list of Components.

React, on the other hand, allows you to decide what constitutes a Component and then provides the framework for you to add the required HTML and JS to create one.

So instead of creating a few large files, you will organize you're web app into small, reusable Components that encompass their own content and business logic.

Train Station Schedule Example

Take for instance an example of a train station schedule. There are elements that repeat themselves in the html structure and essentially perform the same function.

These Components, much like the HTML used to render them, will be structured into various nested Components.

- TubeTracker
  - Network
      - Line
  - Predictions
      - DepartureBoard
          - Trains
  • TubeTracker contains the application and will render 2 main elements: Network and Predictions
  • Network displays each line on the network and renders a Line for each line that it contained in it's dataset
  • Line displays a single station on a line
  • Predictions controls the state of the departure board and renders DepartureBoard
  • DepartureBoard displays the current station and platforms and renders a Train for each one in its dataset
  • Trains displays the trains due to arrive at a platform

When using React, building Components will be a thing you will do quite often.

Rules To Follow

From this point on we will be creating Components so before we begin let's take a moment to discuss the requirements and best practices for creating Components.

Component Specific Rules :oncoming_police_car

  • Must import React (not so anymore with the latest version of React)
  • Must be called within the JSX using an uppercase first letter
  • Must return some form of UI (user interface)
  • Must be exported from the file to be imported into another Component

JSX Specific Rules :oncoming_police_car

  • Can return only one top level element but that element can contain numerous child elements.
  • Any JS within JSX must be enclosed in curly braces {variable_name}
  • The keyword class is reserved so CSS classes must be renamed className
  • Any Component created via a loop must include a key prop

⭐ Some Best Practices

  • Each Component should be in it's own file
  • Each Component file should be in a separate folder
  • Each Component should reference it's own CSS

Class vs Functional Components

As of React 16.8 Components now come in 3 forms, all of which follow the same requirements and best practices.

  • Class Based (makes use of state and lifecycle methods)
  • Functional Using Hooks - uses Hooks to make use of state and lifecycle methods
  • Presentational - Renders only UI only and is stateless

Class Based Components

We will introduce Class Based Components so that you have some grounding as to their structure, but for the remainder of this class all Components will be created as Functional Components.

Starter Code

Let's continue working on the previous CodeSandbox Starter.

App.js

In src create a new file called App.js.

Inside App.js we will do the following:

  • import React
  • create the Class Component
  • export the Component
// IMPORT REACT
import React from 'react'
// CREATE THE CLASS BASED COMPONENT
class App extends React.Component {
  render(){
    return (
      <div>Class Based Component</div>
    )
  }
}
// EXPORT THE COMPONENT
export default App

Let's break down the things we see here:

🚔 import React from 'react'

This imports React and supporting methods from the React library. In this case were making use of React.Component()

🚔 class App extends React.Component

Since we are creating a Class Component we must include the keyword class, as well as extend, which is used to inherit all the functionality provided via the default React.Component.

ES6 classes, and class inheritance, are a much more extensive topic and something we will not cover in this class, but they are extensively used in JavaScript and were how we originally created state enabled Components in React up until v16.8.

render(){
  return (
    <div>Class Based Component</div>
  )
}

All Class Components must use the render(){} method to return() the UI. The render() method is was inherited from React.Component via the extend keyword.

🚔 export default App

This allows other files to import the App Component. In our case, we'll be importing it into index.js using the import keyword.

Importing The Component Into index.js

In index.js let's import the App Component and render it.

// IMPORT THE APP COMPONENT
import App from './App'

And now let's render it. For this we are going to replace the existing code we copied earlier with the <App /> Component.

ReactDOM.render(
  <App />, 
document.getElementById("root"));

Once the page refreshes you should see the output.

Final Note On Class Components

There is much more to Class Components then was touched on here. This was only meant to show the most minimum instantiation of a Class Component.

Just keep in mind that the additional structure for defining state, handling props, calling lifecycle methods will be slightly different than their Hook based counterparts.

Functional Components

Being that you are familiar with standard JS functions writing a Functional Component is much easier to understand.

For now let's just comment out the Class Component and write a new Functional Component.

const App = () => (
    <div>Functional Component</div>
)

As we can see a Functional Component is more streamlined and concise. If written using the fat-arrow syntax it allows it to take advantage of the implicit return.

If we only need to return JSX then this way of writing the Component will suffice. However if we needed to include additional logic then we would need to write the Component to include the return keyword.

const App = () => {
  const name = 'Functional Component'
  return (
    <div>{name}</div>
  )
}

As we can see this requires curly braces {} and an explicit return statement.

JSX

One thing to note about Bootstrap is that there is a distinct separation between HTML, CSS and JS. It takes the separation of concerns modality and requires the import of a separate css and js file.

So in order to fully work with Bootstrap not only do you need to include the Bootstrap specific HTML but also import the following files into index.js

<!-- BOOTSTRAP CSS -->
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css" integrity="sha384-JcKb8q3iqJ61gNV9KGb8thSsNjpSL0n8PARn9HuZOnIxN0hoP+VmmDGMN5t9UJ0Z" crossorigin="anonymous">
<!-- BOOTSTRAP JS AND REQUIRED DEPENDENCIES-->
<script src="https://code.jquery.com/jquery-3.5.1.slim.min.js" integrity="sha384-DfXdz2htPH0lsSSs5nCTpuj/zy4C+OGpamoFVy38MVBnE+IbbVYUew+OrCXaRkfj" crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/popper.js@1.16.1/dist/umd/popper.min.js" integrity="sha384-9/reFTGAW83EW2RDu2S0VKaIzap3H66lZH81PoYlFhbGU+6BZp6G7niu735Sk7lN" crossorigin="anonymous"></script>
<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.5.2/js/bootstrap.min.js" integrity="sha384-B4gt1jrGC7Jh4AgTPSdUtOBvfO8shuf57BaghqFfPlYxofvL8/KUEfYiJOMMV+rV" crossorigin="anonymous"></script>

JSX is a language that allows us to enumerate JavaScipt within HTML and write more cohesive code.

JSX is not limited to React and can be implemented in other frameworks.

Rules For Writing JSX

Let's review the rules once again for working with JSX.

  • Can return only one top level element but that element can contain numerous child elements.
  • Any JS within JSX must be enclosed in curly braces {variable_name}
  • The keyword class is reserved so CSS classes must be renamed className
  • Any Component created via a loop must include a key prop
Single Top Level Parent

Let's put the first rule to the test and try adding another div as a sibling.

const App = () => {
  return (
    <div>Top</div>
    <div>Functional Component</div>
  )
}

We should get the following error regarding Adjacent JSX elements...


We can fix that by wrapping all the HTML in a top level element, such as another div or by using a React fragment.

const App = () => {
  return (
    <>
      <div>Top</div>
      <div>Functional Component</div>
    </>
  )
}

Which in pre v16.8 was written as:

const App = () => {
  return (
    <React.Fragment>
      <div>Top</div>
      <div>Functional Component</div>
    </React.Fragment>
  )
}

Replacing class with className

Although this won't cause the same fatal error as adding two top level parents elements, it will produce a warning.

Let's see that warning in action by adding a class.

const App = () => (
  <>
    <div class='top'>Top</div>
    <div>Functional Component</div>
  </>
)

In Chrome DevTools we should see the following warning in the Console

Bonus - Modern Web Components

Components, React or otherwise, are now becoming the standard for sharing small pieces of functionality.

Let's take a look at https://bit.dev/ which is an environment developers can share their own custom Components or use those created by other devs.

There are even hundreds of React Components being shared on the site as well.

Much like the Train Station design we looked at earlier, Components are used organized to implement a design and can be built using several of the shared Components on bit.dev.

Resource