Switched to typescript CRA, added router, global user context and apollo client
This commit is contained in:
parent
cc63aede25
commit
16cdf92bfe
2
.gitignore
vendored
2
.gitignore
vendored
@ -21,3 +21,5 @@
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
|
||||
*.local*
|
@ -5,4 +5,6 @@ A frontend for QuestionForm application.
|
||||
# Built with:
|
||||
|
||||
- React
|
||||
- Graphql
|
||||
- Graphql
|
||||
- Apollo Client
|
||||
- React Router
|
||||
|
22
package.json
22
package.json
@ -1,25 +1,23 @@
|
||||
{
|
||||
"name": "front",
|
||||
"name": "frontend",
|
||||
"version": "0.1.0",
|
||||
"private": true,
|
||||
"dependencies": {
|
||||
"@testing-library/jest-dom": "^4.2.4",
|
||||
"@testing-library/react": "^9.3.2",
|
||||
"@testing-library/user-event": "^7.1.2",
|
||||
"@types/jest": "^26.0.14",
|
||||
"@types/node": "^14.11.2",
|
||||
"@types/react": "^16.9.50",
|
||||
"@types/react-dom": "^16.9.8",
|
||||
"@apollo/client": "^3.2.2",
|
||||
"@types/node": "^12.0.0",
|
||||
"@types/react": "^16.9.0",
|
||||
"@types/react-dom": "^16.9.0",
|
||||
"@types/react-router-dom": "^5.1.6",
|
||||
"graphql": "^15.3.0",
|
||||
"react": "^16.13.1",
|
||||
"react-dom": "^16.13.1",
|
||||
"react-router-dom": "^5.2.0",
|
||||
"react-scripts": "3.4.3",
|
||||
"typescript": "^4.0.3"
|
||||
"typescript": "~3.7.2"
|
||||
},
|
||||
"scripts": {
|
||||
"start": "react-scripts start",
|
||||
"build": "react-scripts build",
|
||||
"test": "react-scripts test",
|
||||
"eject": "react-scripts eject"
|
||||
"build": "react-scripts build"
|
||||
},
|
||||
"eslintConfig": {
|
||||
"extends": "react-app"
|
||||
|
@ -19,7 +19,6 @@
|
||||
Notice the use of %PUBLIC_URL% in the tags above.
|
||||
It will be replaced with the URL of the `public` folder during the build.
|
||||
Only files inside the `public` folder can be referenced from the HTML.
|
||||
|
||||
Unlike "/favicon.ico" or "favicon.ico", "%PUBLIC_URL%/favicon.ico" will
|
||||
work correctly both with client-side routing and a non-root public URL.
|
||||
Learn how to configure a non-root public URL by running `npm run build`.
|
||||
@ -32,10 +31,8 @@
|
||||
<!--
|
||||
This HTML file is a template.
|
||||
If you open it directly in the browser, you will see an empty page.
|
||||
|
||||
You can add webfonts, meta tags, or analytics to this file.
|
||||
The build step will place the bundled scripts into the <body> tag.
|
||||
|
||||
To begin the development, run `npm start` or `yarn start`.
|
||||
To create a production bundle, use `npm run build` or `yarn build`.
|
||||
-->
|
||||
|
@ -1,7 +0,0 @@
|
||||
import React from "react"
|
||||
|
||||
const App: React.FC = () => {
|
||||
return <div className="App"></div>
|
||||
}
|
||||
|
||||
export default App
|
31
src/apollo/index.ts
Normal file
31
src/apollo/index.ts
Normal file
@ -0,0 +1,31 @@
|
||||
import { ApolloClient, createHttpLink, InMemoryCache } from '@apollo/client'
|
||||
|
||||
import { setContext } from '@apollo/client/link/context'
|
||||
|
||||
import { LOGIN } from './queries'
|
||||
|
||||
const httpLink = createHttpLink({
|
||||
uri: process.env.GRAPHQL_URL || process.env.REACT_APP_GRAPHQL_URL || undefined
|
||||
// fetchOptions: {
|
||||
// mode: 'no-cors'
|
||||
// }
|
||||
})
|
||||
|
||||
const authLink = setContext((_, { headers }) => {
|
||||
const token = localStorage.getItem('token')
|
||||
|
||||
return {
|
||||
headers: {
|
||||
...headers,
|
||||
authorization: token ? `Bearer ${token}` : ''
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
const client = new ApolloClient({
|
||||
link: authLink.concat(httpLink),
|
||||
cache: new InMemoryCache()
|
||||
})
|
||||
|
||||
export default client
|
||||
export { LOGIN }
|
20
src/apollo/queries.ts
Normal file
20
src/apollo/queries.ts
Normal file
@ -0,0 +1,20 @@
|
||||
import { gql } from '@apollo/client'
|
||||
|
||||
const LOGIN = gql`
|
||||
mutation Login($email: String!) {
|
||||
login(email: $email) {
|
||||
success
|
||||
}
|
||||
}
|
||||
`
|
||||
|
||||
const FORM = gql`
|
||||
query Form($id: Int!) {
|
||||
form(id: $id) {
|
||||
id
|
||||
title
|
||||
}
|
||||
}
|
||||
`
|
||||
|
||||
export { LOGIN, FORM }
|
55
src/components/App.tsx
Normal file
55
src/components/App.tsx
Normal file
@ -0,0 +1,55 @@
|
||||
import { ApolloProvider, useMutation, useQuery } from '@apollo/client'
|
||||
import React, { useContext } from 'react'
|
||||
import { BrowserRouter as Router, Route, Switch } from 'react-router-dom'
|
||||
|
||||
import client from '../apollo'
|
||||
import { FORM, LOGIN } from '../apollo/queries'
|
||||
import Context from '../context'
|
||||
import { useUser } from '../hooks'
|
||||
|
||||
const TestComponent: React.FC = () => {
|
||||
const { loading, error, data } = useQuery(FORM, {
|
||||
variables: {
|
||||
id: 1
|
||||
}
|
||||
})
|
||||
|
||||
const { user } = useContext(Context)
|
||||
|
||||
const [doLogin] = useMutation(LOGIN)
|
||||
|
||||
if (loading) return <p>Loading...</p>
|
||||
if (error) return <p>Error :(</p>
|
||||
|
||||
return (
|
||||
<div>
|
||||
<button
|
||||
onClick={() => doLogin({ variables: { email: 'test@test.test' } })}
|
||||
>
|
||||
Click!
|
||||
</button>
|
||||
{user.id}
|
||||
{data.form.id}
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
const App: React.FC = () => {
|
||||
const userContext = useUser()
|
||||
|
||||
return (
|
||||
<div className="App">
|
||||
<ApolloProvider client={client}>
|
||||
<Context.Provider value={userContext}>
|
||||
<Router>
|
||||
<Switch>
|
||||
<Route path="/" component={TestComponent} />
|
||||
</Switch>
|
||||
</Router>
|
||||
</Context.Provider>
|
||||
</ApolloProvider>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
export default App
|
27
src/context.ts
Normal file
27
src/context.ts
Normal file
@ -0,0 +1,27 @@
|
||||
import { createContext } from 'react'
|
||||
|
||||
export type UserType = {
|
||||
id: number
|
||||
email: string
|
||||
name: string
|
||||
}
|
||||
|
||||
export type ContextType = {
|
||||
user: UserType
|
||||
setUser: (user: UserType) => void
|
||||
}
|
||||
|
||||
export const initialUser: UserType = {
|
||||
email: '',
|
||||
id: 0,
|
||||
name: ''
|
||||
}
|
||||
|
||||
const initialState: ContextType = {
|
||||
user: initialUser,
|
||||
setUser: () => {}
|
||||
}
|
||||
|
||||
const Context = createContext<ContextType>(initialState)
|
||||
|
||||
export default Context
|
12
src/hooks.ts
Normal file
12
src/hooks.ts
Normal file
@ -0,0 +1,12 @@
|
||||
import { useCallback, useState } from 'react'
|
||||
import { ContextType, initialUser, UserType } from './context'
|
||||
|
||||
export const useUser = (): ContextType => {
|
||||
const [user, internalSerUser] = useState<UserType>(initialUser)
|
||||
|
||||
const setUser = useCallback((user: UserType): void => {
|
||||
internalSerUser(user)
|
||||
}, [])
|
||||
|
||||
return { user, setUser }
|
||||
}
|
@ -1,13 +1,13 @@
|
||||
body {
|
||||
margin: 0;
|
||||
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", "Roboto", "Oxygen",
|
||||
"Ubuntu", "Cantarell", "Fira Sans", "Droid Sans", "Helvetica Neue",
|
||||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
|
||||
'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
|
||||
sans-serif;
|
||||
-webkit-font-smoothing: antialiased;
|
||||
-moz-osx-font-smoothing: grayscale;
|
||||
}
|
||||
|
||||
code {
|
||||
font-family: source-code-pro, Menlo, Monaco, Consolas, "Courier New",
|
||||
font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
|
||||
monospace;
|
||||
}
|
||||
|
@ -1,17 +1,12 @@
|
||||
import React from "react"
|
||||
import ReactDOM from "react-dom"
|
||||
import "./index.css"
|
||||
import App from "./App"
|
||||
import * as serviceWorker from "./serviceWorker"
|
||||
import React from 'react'
|
||||
import { render } from 'react-dom'
|
||||
|
||||
ReactDOM.render(
|
||||
import './index.css'
|
||||
import App from './components/App'
|
||||
|
||||
render(
|
||||
<React.StrictMode>
|
||||
<App />
|
||||
</React.StrictMode>,
|
||||
document.getElementById("root")
|
||||
document.getElementById('root')
|
||||
)
|
||||
|
||||
// If you want your app to work offline and load faster, you can change
|
||||
// unregister() to register() below. Note this comes with some pitfalls.
|
||||
// Learn more about service workers: https://bit.ly/CRA-PWA
|
||||
serviceWorker.unregister()
|
||||
|
@ -12,18 +12,26 @@
|
||||
|
||||
const isLocalhost = Boolean(
|
||||
window.location.hostname === 'localhost' ||
|
||||
// [::1] is the IPv6 localhost address.
|
||||
window.location.hostname === '[::1]' ||
|
||||
// 127.0.0.0/8 are considered localhost for IPv4.
|
||||
window.location.hostname.match(
|
||||
/^127(?:\.(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/
|
||||
)
|
||||
// [::1] is the IPv6 localhost address.
|
||||
window.location.hostname === '[::1]' ||
|
||||
// 127.0.0.0/8 are considered localhost for IPv4.
|
||||
window.location.hostname.match(
|
||||
/^127(?:\.(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/
|
||||
)
|
||||
);
|
||||
|
||||
export function register(config) {
|
||||
type Config = {
|
||||
onSuccess?: (registration: ServiceWorkerRegistration) => void;
|
||||
onUpdate?: (registration: ServiceWorkerRegistration) => void;
|
||||
};
|
||||
|
||||
export function register(config?: Config) {
|
||||
if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) {
|
||||
// The URL constructor is available in all browsers that support SW.
|
||||
const publicUrl = new URL(process.env.PUBLIC_URL, window.location.href);
|
||||
const publicUrl = new URL(
|
||||
process.env.PUBLIC_URL,
|
||||
window.location.href
|
||||
);
|
||||
if (publicUrl.origin !== window.location.origin) {
|
||||
// Our service worker won't work if PUBLIC_URL is on a different origin
|
||||
// from what our page is served on. This might happen if a CDN is used to
|
||||
@ -43,7 +51,7 @@ export function register(config) {
|
||||
navigator.serviceWorker.ready.then(() => {
|
||||
console.log(
|
||||
'This web app is being served cache-first by a service ' +
|
||||
'worker. To learn more, visit https://bit.ly/CRA-PWA'
|
||||
'worker. To learn more, visit https://bit.ly/CRA-PWA'
|
||||
);
|
||||
});
|
||||
} else {
|
||||
@ -54,7 +62,7 @@ export function register(config) {
|
||||
}
|
||||
}
|
||||
|
||||
function registerValidSW(swUrl, config) {
|
||||
function registerValidSW(swUrl: string, config?: Config) {
|
||||
navigator.serviceWorker
|
||||
.register(swUrl)
|
||||
.then(registration => {
|
||||
@ -71,7 +79,7 @@ function registerValidSW(swUrl, config) {
|
||||
// content until all client tabs are closed.
|
||||
console.log(
|
||||
'New content is available and will be used when all ' +
|
||||
'tabs for this page are closed. See https://bit.ly/CRA-PWA.'
|
||||
'tabs for this page are closed. See https://bit.ly/CRA-PWA.'
|
||||
);
|
||||
|
||||
// Execute callback
|
||||
@ -98,10 +106,10 @@ function registerValidSW(swUrl, config) {
|
||||
});
|
||||
}
|
||||
|
||||
function checkValidServiceWorker(swUrl, config) {
|
||||
function checkValidServiceWorker(swUrl: string, config?: Config) {
|
||||
// Check if the service worker can be found. If it can't reload the page.
|
||||
fetch(swUrl, {
|
||||
headers: { 'Service-Worker': 'script' },
|
||||
headers: { 'Service-Worker': 'script' }
|
||||
})
|
||||
.then(response => {
|
||||
// Ensure service worker exists, and that we really are getting a JS file.
|
@ -1,11 +1,7 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"target": "es5",
|
||||
"lib": [
|
||||
"dom",
|
||||
"dom.iterable",
|
||||
"esnext"
|
||||
],
|
||||
"lib": ["dom", "dom.iterable", "esnext"],
|
||||
"allowJs": true,
|
||||
"skipLibCheck": true,
|
||||
"esModuleInterop": true,
|
||||
@ -19,7 +15,5 @@
|
||||
"noEmit": true,
|
||||
"jsx": "react"
|
||||
},
|
||||
"include": [
|
||||
"src"
|
||||
]
|
||||
"include": ["src"]
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user