A React application can contain dozens, or even hundreds, of components.
Each component might be small and relatively unremarkable on its own. When combined, however, they can form enormous, fantastically complex ecosystems of information.
In other words, React apps are made out of components, but what makes React special isn't components themselves. What makes React special is the ways in which components interact.
This unit is an introduction to components interacting.
Here is a .render() method that returns an HTML-like JSX element:
class Example extends React.Component {
render() {
return <h1>Hello world</h1>;
}
}
You've seen render methods return div, p, and h1, just like in the above example.
Render methods can also return another kind of JSX: component instances.
class OMG extends React.Component {
render() {
return <h1>Whooaa!</h1>;
}
}
class Crazy extends React.Component {
render() {
return <OMG />;
}
}
In the above example, Crazy's render method returns an instance of the OMG component class. You could say that Crazy renders an OMG.
You've never seen a component rendered by another component before.
You have seen a component rendered before, though, but not by another component. Instead, you've seen a component rendered by ReactDOM.render().
When a component renders another component, what happens is very similar to what happens when ReactDOM.render() renders a component.
When you use React.js, every JavaScript file in your application is invisible to every other JavaScript file by default.
This is a problem!
//ProfilePage.js
import React from 'react';
import ReactDOM from 'react-dom';
class ProfilePage extends React.Component {
render() {
return (
<div>
<NavBar />
<h1>All About Me!</h1>
<p>I like movies and blah blah blah blah blah</p>
<img src="https://s3.amazonaws.com/codecademy-content/courses/React/react_photo-monkeyselfie.jpg" />
</div>
);
}
}
On line 9, you just added an instance of the NavBar component class. But since you're in ProfilePage.js, JavaScript has no idea what NavBar means.
If you want to use a variable that's declared in a different file, such as NavBar, then you have to import the variable that you want. To import a variable, you can use an import statement:
import { NavBar } from './NavBar.js';
We've used import before, but not like this! Notice the differences between the above line of code and this familiar line:
import React from 'react';
The first important difference is the curly braces around NavBar. We'll get to those soon!
The second important difference involves the contents of the string at the end of the string at the end of the statement: 'react' vs './NavBar.js'.
If you use an import statement, and the string at the end begins with either a dot or a slash, then import will treat that string as a filepath. import will follow that filepath, and import the file that it finds.
If your filepath doesn't have a file extension, then ".js" is assumed. So the above example could be shortened:
import { NavBar } from './NavBar';
One final, important note:
None of this behavior is specific to React! Module systems of independent, importable files are a very popular way to organize code. React's specific module system comes from ES6. More on all of that later.
Alright! You've learned how to use import to grab a variable from a file other than the file that is currently executing.
When you import a variable from a file that is not the current file, then an import statement isn't quite enough. You also need an export statement, written in the other file, exporting the variable that you hope to grab.
export comes from ES6's module system, just like import does. export and import are meant to be used together, and you rarely see one without the other.
There are a few different ways to use export. In this course, we will be using a style called "named exports". Here's how named exports works:
In one file, place the keyword export immediately before something that you want to export. That something can be any top-level var, let, const, function or class:
// Manifestos.js:
export const faveManifestos = {
futurist: 'http://www.artype.de/Sammlung/pdf/russolo_noise.pdf',
agile: 'https://agilemanifesto.org/iso/en/manifesto.html',
cyborg: 'http://faculty.georgetown.edu/irvinem/theory/Haraway-CyborgManifesto-1.pdf'
};
You can export multiple things from the same file:
// Manifestos.js:
export const faveManifestos = {
futurist: 'http://www.artype.de/Sammlung/pdf/russolo_noise.pdf',
agile: 'https://agilemanifesto.org/iso/en/manifesto.html',
cyborg: 'http://faculty.georgetown.edu/irvinem/theory/Haraway-CyborgManifesto-1.pdf'
};
export const alsoRan = 'TimeCube';
In a different file, import the name of the var, let, const, function, or class from the first file:
// App.js:
// Import faveManifestos and alsoRan from ./Manifestos.js:
import { faveManifestos, alsoRan } from './Manifestos';
// Use faveManifestos:
console.log(`A Cyborg Manifesto: ${faveManifestos.cyborg}`);
This style of importing and exporting in JavaScript is known was "named exports". When you use named exports, you always need to wrap your imported names in curly braces, such as:
import { faveManifestos, alsoRan } from './Manifestos';`
JavaScript's ES6 module system goes beyond named exports and has several advanced syntax features.