Rendez-vous sur le site https://nodejs.org/en/ pour télécharger et installer la dernière version de NodeJS (minimum version 8 pour ce workshop).
Vous pouvez également l’installer via votre gestionnaire de paquet favoris, cependant, dans ce cas, veuillez installer également le paquet npm
ou yarn
.
Sur Fedora :
$ sudo dnf install nodejs
On initialize notre projet
$ npm install -g create-react-app
Cette commande vous ajoute la commande create-react-app.
Puis dans le directory que vous voulez lancez :
$ create-react-app app
$ cd app
Nous allons utiliser plusieurs librairies :
apollo-boost
: Librairie GraphQL pour le coté clientreact-apollo
: Fait le lien entre Apollo et Reactgraphql
: Utilisé par Apollo@material-ui/core
: Des composants react material
Pour les installer :
$ npm install --save apollo-boost graphql @material-ui/core react-apollo
Dans votre terminal :
$ npm start
Vous verrez alors un hello world apparaite si vous visitez http://localhost:3000
Supprimez tout dans votre dossier src
sauf App.js
et index.js
Fichier App.js
:
import React from "react";
import { Button, Typography } from "@material-ui/core";
export default function App() {
return (
<>
<Typography>Hello World</Typography>
<Button>Click Me</Button>
</>
);
}
Fichier index.js
import React from "react";
import ReactDOM from "react-dom";
import App from "./App";
ReactDOM.render(<App />, document.getElementById("root"));
On crée un fichier LangList.js
:
import React from "react";
import { Typography, Divider } from "@material-ui/core";
const langs = [
{
name: "Javascript",
ext: "js"
},
{
name: "GraphQL",
ext: "gql"
}
];
export default function LangList() {
return (
<div>
{langs.map(({ name, ext }) => (
<>
<div
style={{
flexDirection: "row",
display: "flex",
alignItems: "center",
marginTop: 20,
justifyContent: "center"
}}
>
<Typography variant="h4">{name}</Typography>
<Typography variant="h5">(.{ext})</Typography>
</div>
<Divider style={{ marginBottom: 20 }} />
</>
))}
</div>
);
}
Et on met à jour App.js
import React from "react";
import { Typography } from "@material-ui/core";
import LangList from "./LangList";
export default function App() {
return (
<div style={{ display: "flex", flexDirection: "column" }}>
<Typography variant="h2" style={{ alignSelf: "center" }}>
The Programming Languages List
</Typography>
<LangList />
</div>
);
}
index.js
import React from "react";
import ReactDOM from "react-dom";
import App from "./App";
import ApolloClient from "apollo-boost";
import { ApolloProvider } from "react-apollo";
const client = new ApolloClient({
uri: "http://localhost:4000"
});
ReactDOM.render(
<ApolloProvider client={client}>
<App />
</ApolloProvider>,
document.getElementById("root")
);
LangList.js
import React from "react";
import { Typography, Divider } from "@material-ui/core";
import { Query } from "react-apollo";
import gql from "graphql-tag";
const allLangQuery = gql`
{
allLang {
name
ext
}
}
`;
export default function LangList() {
return (
<Query query={allLangQuery}>
{({ loading, error, data }) => {
if (loading) {
return <Typography>Loading...</Typography>
} else if (error) {
return <Typography>Error: {error}</Typography>
} else if (!data) {
return <Typography>No data found</Typography>
}
return (
<div>
{data.allLang.map(({ name, ext }) => (
<div key={`${name}-${ext}`}>
<div
style={{
flexDirection: "row",
display: "flex",
alignItems: "center",
marginTop: 20,
justifyContent: "center"
}}
>
<Typography variant="h4">{name}</Typography>
<Typography variant="h5">(.{ext})</Typography>
</div>
<Divider style={{ marginBottom: 20 }} />
</div>
))}
</div>
);
}}
</Query>
);
}
LangInput.js
import React, { useState } from "react";
import { Mutation } from "react-apollo";
import gql from "graphql-tag";
import { TextField, Button } from "@material-ui/core";
const addLangMutation = gql`
mutation AddLang($name: String, $ext: String) {
addLang(name: $name, ext: $ext) {
name
ext
}
}
`;
export default function LangInput() {
const [name, setName] = useState("");
const [ext, setExt] = useState("");
return (
<Mutation
mutation={addLangMutation}
variables={{
name,
ext
}}
>
{addLang => {
return (
<div
style={{
display: "flex",
flexDirection: "column",
alignItems: "center"
}}
>
<TextField
label="Name"
value={name}
onChange={event => setName(event.target.value)}
/>
<TextField
label="Ext"
value={ext}
onChange={event => setExt(event.target.value)}
/>
<Button
style={{ marginTop: 20 }}
color="primary"
variant="contained"
onClick={addLang}
>
Add New Lang
</Button>
</div>
);
}}
</Mutation>
);
}
App.js
import React from "react";
import { Typography } from "@material-ui/core";
import LangList from "./LangList";
import LangInput from "./LangInput";
export default function App() {
return (
<div style={{ display: "flex", flexDirection: "column" }}>
<Typography variant="h2" style={{ alignSelf: "center" }}>
The Programming Languages List
</Typography>
<LangInput />
<LangList />
</div>
);
}
LangList.js
import React from "react";
import { Typography, Divider, Fab } from "@material-ui/core";
import { Query, Mutation } from "react-apollo";
import gql from "graphql-tag";
const allLangQuery = gql`
{
allLang {
name
ext
}
}
`;
const deleteLangMutation = gql`
mutation DeleteLang($name: String) {
deleteLang(name: $name) {
name
ext
}
}
`;
export default function LangList() {
return (
<Mutation mutation={deleteLangMutation}>
{deleteLang => (
<Query query={allLangQuery}>
{({ loading, error, data }) => {
if (loading) {
return <Typography>Loading...</Typography>;
} else if (error) {
return <Typography>Error: {error}</Typography>;
} else if (!data) {
return <Typography>No data found</Typography>;
}
return (
<div>
{data.allLang.map(({ name, ext }) => (
<div key={`${name}-${ext}`}>
<div
style={{
flexDirection: "row",
display: "flex",
alignItems: "center",
marginTop: 20,
justifyContent: "space-between"
}}
>
<Typography variant="h4">
{name}.{ext}
</Typography>
<Fab
onClick={() => {
deleteLang({ variables: { name } });
}}
color="secondary"
>
x
</Fab>
</div>
<Divider style={{ marginBottom: 20 }} />
</div>
))}
</div>
);
}}
</Query>
)}
</Mutation>
);
}
mutations.js
import gql from "graphql-tag";
export const addLangMutation = gql`
mutation AddLang($name: String, $ext: String) {
addLang(name: $name, ext: $ext) {
name
ext
}
}
`;
export const deleteLangMutation = gql`
mutation DeleteLang($name: String) {
deleteLang(name: $name) {
name
ext
}
}
`;
query.js
import gql from "graphql-tag";
export const allLangQuery = gql`
{
allLang {
name
ext
}
}
`;
LangInput.js
import React, { useState } from "react";
import { Mutation } from "react-apollo";
import { TextField, Button } from "@material-ui/core";
import { addLangMutation } from "./mutations";
import { allLangQuery } from "./query";
export default function LangInput() {
const [name, setName] = useState("");
const [ext, setExt] = useState("");
return (
<Mutation
mutation={addLangMutation}
refetchQueries={() => [
{
query: allLangQuery
}
]}
variables={{
name,
ext
}}
>
{addLang => {
return (
<div
style={{
display: "flex",
flexDirection: "column",
alignItems: "center"
}}
>
<TextField
label="Name"
value={name}
onChange={event => setName(event.target.value)}
/>
<TextField
label="Ext"
value={ext}
onChange={event => setExt(event.target.value)}
/>
<Button
style={{ marginTop: 20 }}
color="primary"
variant="contained"
onClick={addLang}
>
Add New Lang
</Button>
</div>
);
}}
</Mutation>
);
}
LangList.js
import React from "react";
import { Typography, Divider, Fab } from "@material-ui/core";
import { Query, Mutation } from "react-apollo";
import { deleteLangMutation } from "./mutations";
import { allLangQuery } from "./query";
export default function LangList() {
return (
<Mutation
mutation={deleteLangMutation}
refetchQueries={() => [
{
query: allLangQuery
}
]}
>
{deleteLang => (
<Query query={allLangQuery}>
{({ loading, error, data }) => {
if (loading) {
return <Typography>Loading...</Typography>;
} else if (error) {
return <Typography>Error: {error}</Typography>;
} else if (!data) {
return <Typography>No data found</Typography>;
}
return (
<div>
{data.allLang.map(({ name, ext }) => (
<div key={`${name}-${ext}`}>
<div
style={{
flexDirection: "row",
display: "flex",
alignItems: "center",
marginTop: 20,
justifyContent: "space-between"
}}
>
<Typography variant="h4">
{name}.{ext}
</Typography>
<Fab
onClick={() => {
deleteLang({ variables: { name } });
}}
color="secondary"
>
x
</Fab>
</div>
<Divider style={{ marginBottom: 20 }} />
</div>
))}
</div>
);
}}
</Query>
)}
</Mutation>
);
}