Add basic datatypes and first form implementation #1
| @ -4,13 +4,16 @@ | |||||||
|   "private": true, |   "private": true, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@material-ui/core": "^4.10.1", |     "@material-ui/core": "^4.10.1", | ||||||
|  |     "@material-ui/lab": "^4.0.0-alpha.55", | ||||||
|     "@testing-library/jest-dom": "^4.2.4", |     "@testing-library/jest-dom": "^4.2.4", | ||||||
|     "@testing-library/react": "^9.3.2", |     "@testing-library/react": "^9.3.2", | ||||||
|     "@testing-library/user-event": "^7.1.2", |     "@testing-library/user-event": "^7.1.2", | ||||||
|     "@types/jest": "^24.0.0", |     "@types/jest": "^24.0.0", | ||||||
|  |     "@types/moment": "^2.13.0", | ||||||
|     "@types/node": "^12.0.0", |     "@types/node": "^12.0.0", | ||||||
|     "@types/react": "^16.9.0", |     "@types/react": "^16.9.0", | ||||||
|     "@types/react-dom": "^16.9.0", |     "@types/react-dom": "^16.9.0", | ||||||
|  |     "moment": "^2.26.0", | ||||||
|     "react": "^16.13.1", |     "react": "^16.13.1", | ||||||
|     "react-dom": "^16.13.1", |     "react-dom": "^16.13.1", | ||||||
|     "react-scripts": "3.4.1", |     "react-scripts": "3.4.1", | ||||||
|  | |||||||
							
								
								
									
										38
									
								
								src/App.css
									
									
									
									
									
								
							
							
						
						
									
										38
									
								
								src/App.css
									
									
									
									
									
								
							| @ -1,38 +0,0 @@ | |||||||
| .App { |  | ||||||
|   text-align: center; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .App-logo { |  | ||||||
|   height: 40vmin; |  | ||||||
|   pointer-events: none; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| @media (prefers-reduced-motion: no-preference) { |  | ||||||
|   .App-logo { |  | ||||||
|     animation: App-logo-spin infinite 20s linear; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .App-header { |  | ||||||
|   background-color: #282c34; |  | ||||||
|   min-height: 100vh; |  | ||||||
|   display: flex; |  | ||||||
|   flex-direction: column; |  | ||||||
|   align-items: center; |  | ||||||
|   justify-content: center; |  | ||||||
|   font-size: calc(10px + 2vmin); |  | ||||||
|   color: white; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .App-link { |  | ||||||
|   color: #61dafb; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| @keyframes App-logo-spin { |  | ||||||
|   from { |  | ||||||
|     transform: rotate(0deg); |  | ||||||
|   } |  | ||||||
|   to { |  | ||||||
|     transform: rotate(360deg); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @ -1,9 +0,0 @@ | |||||||
| import React from 'react'; |  | ||||||
| import { render } from '@testing-library/react'; |  | ||||||
| import App from './App'; |  | ||||||
| 
 |  | ||||||
| test('renders learn react link', () => { |  | ||||||
|   const { getByText } = render(<App />); |  | ||||||
|   const linkElement = getByText(/learn react/i); |  | ||||||
|   expect(linkElement).toBeInTheDocument(); |  | ||||||
| }); |  | ||||||
							
								
								
									
										32
									
								
								src/App.tsx
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								src/App.tsx
									
									
									
									
									
								
							| @ -1,25 +1,21 @@ | |||||||
| import React from 'react'; | import React from 'react'; | ||||||
| import logo from './logo.svg'; | import { Container, ThemeProvider, Typography } from "@material-ui/core"; | ||||||
| import './App.css'; | import { CompanyForm } from "./forms/company"; | ||||||
|  | import { studentTheme } from "./ui/theme"; | ||||||
| 
 | 
 | ||||||
| function App() { | function App() { | ||||||
|   return ( |   return ( | ||||||
|     <div className="App"> |       <ThemeProvider theme={ studentTheme }> | ||||||
|       <header className="App-header"> |           <div className="app"> | ||||||
|         <img src={logo} className="App-logo" alt="logo" /> |               <Container maxWidth={"md"}> | ||||||
|         <p> |                   <Typography variant="h3">Zgłoszenie Praktyki</Typography> | ||||||
|           Edit <code>src/App.tsx</code> and save to reload. |                   <Typography variant="subtitle1" style={{ marginBottom: '100px' }}>UX Demo</Typography> | ||||||
|         </p> | 
 | ||||||
|         <a |                   <Typography variant="h5">Miejsce odbywania praktyki</Typography> | ||||||
|           className="App-link" |                   <CompanyForm /> | ||||||
|           href="https://reactjs.org" |               </Container> | ||||||
|           target="_blank" |           </div> | ||||||
|           rel="noopener noreferrer" |       </ThemeProvider> | ||||||
|         > |  | ||||||
|           Learn React - System Praktyk |  | ||||||
|         </a> |  | ||||||
|       </header> |  | ||||||
|     </div> |  | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								src/data/common.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/data/common.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,3 @@ | |||||||
|  | export interface Identifiable { | ||||||
|  |     id?: string | ||||||
|  | } | ||||||
							
								
								
									
										41
									
								
								src/data/company.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								src/data/company.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | |||||||
|  | import { Identifiable } from "./common"; | ||||||
|  | 
 | ||||||
|  | export interface Address { | ||||||
|  |     street: string; | ||||||
|  |     building: string; | ||||||
|  |     city: string; | ||||||
|  |     postalCode: string; | ||||||
|  |     country: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface Company extends Identifiable { | ||||||
|  |     name: string; | ||||||
|  |     url?: string; | ||||||
|  |     nip: string; | ||||||
|  |     offices: BranchOffice[]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface BranchOffice extends Identifiable { | ||||||
|  |     address: Address; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const emptyCompany: Company = { | ||||||
|  |     offices: [], | ||||||
|  |     url: "", | ||||||
|  |     name: "", | ||||||
|  |     nip: "", | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const emptyAddress: Address = { | ||||||
|  |     street: "", | ||||||
|  |     city: "", | ||||||
|  |     country: "", | ||||||
|  |     postalCode: "", | ||||||
|  |     building: "" | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const emptyBranchOffice: BranchOffice = { | ||||||
|  |     address: emptyAddress, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const formatAddress = ({ postalCode, building, street, city, country }: Address): string => `${street} ${building}, ${postalCode} ${city}, ${country}` | ||||||
							
								
								
									
										23
									
								
								src/data/internship.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/data/internship.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | |||||||
|  | import { Moment } from "moment"; | ||||||
|  | import { Nullable } from "../helpers"; | ||||||
|  | 
 | ||||||
|  | export enum InternshipType { } | ||||||
|  | export enum InternshipProgram { } | ||||||
|  | 
 | ||||||
|  | export interface Internship { | ||||||
|  |     type: InternshipType; | ||||||
|  |     program: InternshipProgram; | ||||||
|  |     startDate: Moment; | ||||||
|  |     endDate: Moment; | ||||||
|  |     isAccepted: boolean; | ||||||
|  |     lengthInWeeks: number; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const emptyInternship: Nullable<Internship> = { | ||||||
|  |     endDate: null, | ||||||
|  |     startDate: null, | ||||||
|  |     type: null, | ||||||
|  |     program: null, | ||||||
|  |     isAccepted: false, | ||||||
|  |     lengthInWeeks: 0, | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								src/forms/Internship.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/forms/Internship.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | |||||||
|  | import React from "react"; | ||||||
|  | import { useFormik } from "formik"; | ||||||
|  | import { emptyInternship, Internship } from "../data/internship"; | ||||||
|  | import { Nullable } from "../helpers"; | ||||||
|  | import { TextField } from "@material-ui/core"; | ||||||
|  | 
 | ||||||
|  | export type InternshipFormProps = { | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const InternshipForm: React.FunctionComponent<InternshipFormProps> = props => { | ||||||
|  |     const formik = useFormik<Nullable<Internship>>({ onSubmit: values => console.log(values), initialValues: emptyInternship }); | ||||||
|  | 
 | ||||||
|  |     const formikProps = (prop: keyof Internship) => ({ onChange: formik.handleChange, value: formik.values[prop], name: prop }) | ||||||
|  | 
 | ||||||
|  |     return ( | ||||||
|  |         <div className="internship-form"> | ||||||
|  |             <TextField {...formikProps("startDate")} /> | ||||||
|  |         </div> | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
							
								
								
									
										139
									
								
								src/forms/company.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								src/forms/company.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,139 @@ | |||||||
|  | import React, { HTMLProps, useEffect, useMemo, useState } from "react"; | ||||||
|  | import { BranchOffice, Company, emptyAddress, emptyBranchOffice, emptyCompany, formatAddress } from "../data/company"; | ||||||
|  | import { sampleCompanies } from "../provider/company"; | ||||||
|  | import { Autocomplete } from "@material-ui/lab"; | ||||||
|  | import { Grid, TextField, Typography } from "@material-ui/core"; | ||||||
|  | import { formFieldProps } from "./helpers"; | ||||||
|  | 
 | ||||||
|  | export type CompanyFormProps = {} | ||||||
|  | 
 | ||||||
|  | export type BranchOfficeProps = { | ||||||
|  |     company: Company, | ||||||
|  |     disabled?: boolean | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const CompanyItem = ({ company, ...props }: { company: Company } & HTMLProps<any>) => ( | ||||||
|  |     <div className="company-item" { ...props }> | ||||||
|  |         <div>{ company.name }</div> | ||||||
|  |         <Typography variant="caption">NIP: <strong>{ company.nip }</strong></Typography> | ||||||
|  |     </div> | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | export const OfficeItem = ({ office, ...props }: { office: BranchOffice } & HTMLProps<any>) => ( | ||||||
|  |     <div className="office-item" { ...props }> | ||||||
|  |         <div>{ office.address.city }</div> | ||||||
|  |         <Typography variant="caption">{ formatAddress(office.address) }</Typography> | ||||||
|  |     </div> | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | export const BranchForm: React.FC<BranchOfficeProps> = ({ company, disabled = false }) => { | ||||||
|  |     const [office, setOffice] = useState<BranchOffice>(emptyBranchOffice) | ||||||
|  | 
 | ||||||
|  |     const canEdit = useMemo(() => !office.id && !disabled, [office.id, disabled]); | ||||||
|  | 
 | ||||||
|  |     const fieldProps = formFieldProps(office.address, address => setOffice({ ...office, address })) | ||||||
|  | 
 | ||||||
|  |     const handleCityChange = (event: any, value: BranchOffice | string | null) => { | ||||||
|  |         if (typeof value === "string") { | ||||||
|  |             setOffice({ | ||||||
|  |                 ...emptyBranchOffice, | ||||||
|  |                 address: { | ||||||
|  |                     ...emptyAddress, | ||||||
|  |                     city: value, | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         } else if (typeof value === "object" && value !== null) { | ||||||
|  |             setOffice(value); | ||||||
|  |         } else { | ||||||
|  |             setOffice(emptyBranchOffice); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const handleCityInput = (event: any, value: string) => { | ||||||
|  |         const base = office.id ? emptyBranchOffice : office; | ||||||
|  |         setOffice({ | ||||||
|  |             ...base, | ||||||
|  |             address: { | ||||||
|  |                 ...base.address, | ||||||
|  |                 city: value, | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     useEffect(() => void (office.id && setOffice(emptyBranchOffice)), [company]) | ||||||
|  | 
 | ||||||
|  |     return ( | ||||||
|  |         <Grid container> | ||||||
|  |             <Grid item md={ 7 }> | ||||||
|  |                 <Autocomplete options={ company?.offices || [] } | ||||||
|  |                               disabled={ disabled } | ||||||
|  |                               getOptionLabel={ office => typeof office == "string" ? office : office.address.city } | ||||||
|  |                               renderOption={ office => <OfficeItem office={ office }/> } | ||||||
|  |                               renderInput={ props => <TextField { ...props } label={ "Miasto" } fullWidth/> } | ||||||
|  |                               onChange={ handleCityChange } | ||||||
|  |                               onInputChange={ handleCityInput } | ||||||
|  |                               inputValue={ office.address.city } | ||||||
|  |                               value={ office.id ? office : null } | ||||||
|  |                               freeSolo | ||||||
|  |                 /> | ||||||
|  |             </Grid> | ||||||
|  |             <Grid item md={ 2 }> | ||||||
|  |                 <TextField label={ "Kod pocztowy" } fullWidth disabled={ !canEdit } { ...fieldProps("postalCode") }/> | ||||||
|  |             </Grid> | ||||||
|  |             <Grid item md={ 3 }> | ||||||
|  |                 <TextField label={ "Kraj" } fullWidth disabled={ !canEdit } { ...fieldProps("country") }/> | ||||||
|  |             </Grid> | ||||||
|  |             <Grid item md={ 10 }> | ||||||
|  |                 <TextField label={ "Ulica" } fullWidth disabled={ !canEdit } { ...fieldProps("street") }/> | ||||||
|  |             </Grid> | ||||||
|  |             <Grid item md={ 2 }> | ||||||
|  |                 <TextField label={ "Nr Budynku" } fullWidth disabled={ !canEdit } { ...fieldProps("building") }/> | ||||||
|  |             </Grid> | ||||||
|  |         </Grid> | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const CompanyForm: React.FunctionComponent<CompanyFormProps> = props => { | ||||||
|  |     const [company, setCompany] = useState<Company>(emptyCompany); | ||||||
|  |     const canEdit = useMemo(() => !company.id, [company.id]); | ||||||
|  | 
 | ||||||
|  |     const fieldProps = formFieldProps(company, setCompany) | ||||||
|  | 
 | ||||||
|  |     const handleCompanyChange = (event: any, value: Company | string | null) => { | ||||||
|  |         if (typeof value === "string") { | ||||||
|  |             setCompany({ | ||||||
|  |                 ...emptyCompany, | ||||||
|  |                 name: value, | ||||||
|  |             }); | ||||||
|  |         } else if (typeof value === "object" && value !== null) { | ||||||
|  |             setCompany(value); | ||||||
|  |         } else { | ||||||
|  |             setCompany(emptyCompany); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return ( | ||||||
|  |         <> | ||||||
|  |             <Grid container style={ { marginBottom: 0, marginTop: 0 } }> | ||||||
|  |                 <Grid item> | ||||||
|  |                     <Autocomplete options={ sampleCompanies } | ||||||
|  |                                   getOptionLabel={ option => option.name } | ||||||
|  |                                   renderOption={ company => <CompanyItem company={ company }/> } | ||||||
|  |                                   renderInput={ props => <TextField { ...props } label={ "Nazwa firmy" } fullWidth/> } | ||||||
|  |                                   onChange={ handleCompanyChange } | ||||||
|  |                                   freeSolo | ||||||
|  |                     /> | ||||||
|  |                 </Grid> | ||||||
|  |                 <Grid item md={ 4 }> | ||||||
|  |                     <TextField label={ "NIP" } fullWidth { ...fieldProps("nip") } disabled={ !canEdit }/> | ||||||
|  |                 </Grid> | ||||||
|  |                 <Grid item md={ 8 }> | ||||||
|  |                     <TextField label={ "Url" } fullWidth { ...fieldProps("url") } disabled={ !canEdit }/> | ||||||
|  |                 </Grid> | ||||||
|  |             </Grid> | ||||||
|  |             <Typography variant="subtitle1">Oddział</Typography> | ||||||
|  |             <BranchForm company={ company }/> | ||||||
|  |         </> | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
							
								
								
									
										20
									
								
								src/forms/helpers.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								src/forms/helpers.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,20 @@ | |||||||
|  | import { DOMEvent } from "../helpers"; | ||||||
|  | 
 | ||||||
|  | type UpdatingEvent = "onBlur" | "onChange" | "onInput"; | ||||||
|  | type FormFieldHelperOptions<T> = { | ||||||
|  |     event: UpdatingEvent | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function formFieldProps<T>(subject: T, update: (value: T) => void, options: Partial<FormFieldHelperOptions<T>> = {}) { | ||||||
|  |     const { | ||||||
|  |         event = "onChange" | ||||||
|  |     } = options; | ||||||
|  | 
 | ||||||
|  |     return (field: keyof T) => ({ | ||||||
|  |         value: subject[field], | ||||||
|  |         [event]: (event: DOMEvent<HTMLInputElement>) => update({ | ||||||
|  |             ...subject, | ||||||
|  |             [field]: event.target.value, | ||||||
|  |         } as T) | ||||||
|  |     }) | ||||||
|  | } | ||||||
							
								
								
									
										5
									
								
								src/helpers.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								src/helpers.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | export type Nullable<T> = { [P in keyof T]: T[P] | null } | ||||||
|  | 
 | ||||||
|  | export interface DOMEvent<TTarget extends EventTarget> extends Event { | ||||||
|  |     target: TTarget; | ||||||
|  | } | ||||||
| @ -1,13 +0,0 @@ | |||||||
| body { |  | ||||||
|   margin: 0; |  | ||||||
|   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', |  | ||||||
|     monospace; |  | ||||||
| } |  | ||||||
| @ -1,6 +1,5 @@ | |||||||
| import React from 'react'; | import React from 'react'; | ||||||
| import ReactDOM from 'react-dom'; | import ReactDOM from 'react-dom'; | ||||||
| import './index.css'; |  | ||||||
| import App from './App'; | import App from './App'; | ||||||
| import * as serviceWorker from './serviceWorker'; | import * as serviceWorker from './serviceWorker'; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,7 +0,0 @@ | |||||||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 841.9 595.3"> |  | ||||||
|     <g fill="#61DAFB"> |  | ||||||
|         <path d="M666.3 296.5c0-32.5-40.7-63.3-103.1-82.4 14.4-63.6 8-114.2-20.2-130.4-6.5-3.8-14.1-5.6-22.4-5.6v22.3c4.6 0 8.3.9 11.4 2.6 13.6 7.8 19.5 37.5 14.9 75.7-1.1 9.4-2.9 19.3-5.1 29.4-19.6-4.8-41-8.5-63.5-10.9-13.5-18.5-27.5-35.3-41.6-50 32.6-30.3 63.2-46.9 84-46.9V78c-27.5 0-63.5 19.6-99.9 53.6-36.4-33.8-72.4-53.2-99.9-53.2v22.3c20.7 0 51.4 16.5 84 46.6-14 14.7-28 31.4-41.3 49.9-22.6 2.4-44 6.1-63.6 11-2.3-10-4-19.7-5.2-29-4.7-38.2 1.1-67.9 14.6-75.8 3-1.8 6.9-2.6 11.5-2.6V78.5c-8.4 0-16 1.8-22.6 5.6-28.1 16.2-34.4 66.7-19.9 130.1-62.2 19.2-102.7 49.9-102.7 82.3 0 32.5 40.7 63.3 103.1 82.4-14.4 63.6-8 114.2 20.2 130.4 6.5 3.8 14.1 5.6 22.5 5.6 27.5 0 63.5-19.6 99.9-53.6 36.4 33.8 72.4 53.2 99.9 53.2 8.4 0 16-1.8 22.6-5.6 28.1-16.2 34.4-66.7 19.9-130.1 62-19.1 102.5-49.9 102.5-82.3zm-130.2-66.7c-3.7 12.9-8.3 26.2-13.5 39.5-4.1-8-8.4-16-13.1-24-4.6-8-9.5-15.8-14.4-23.4 14.2 2.1 27.9 4.7 41 7.9zm-45.8 106.5c-7.8 13.5-15.8 26.3-24.1 38.2-14.9 1.3-30 2-45.2 2-15.1 0-30.2-.7-45-1.9-8.3-11.9-16.4-24.6-24.2-38-7.6-13.1-14.5-26.4-20.8-39.8 6.2-13.4 13.2-26.8 20.7-39.9 7.8-13.5 15.8-26.3 24.1-38.2 14.9-1.3 30-2 45.2-2 15.1 0 30.2.7 45 1.9 8.3 11.9 16.4 24.6 24.2 38 7.6 13.1 14.5 26.4 20.8 39.8-6.3 13.4-13.2 26.8-20.7 39.9zm32.3-13c5.4 13.4 10 26.8 13.8 39.8-13.1 3.2-26.9 5.9-41.2 8 4.9-7.7 9.8-15.6 14.4-23.7 4.6-8 8.9-16.1 13-24.1zM421.2 430c-9.3-9.6-18.6-20.3-27.8-32 9 .4 18.2.7 27.5.7 9.4 0 18.7-.2 27.8-.7-9 11.7-18.3 22.4-27.5 32zm-74.4-58.9c-14.2-2.1-27.9-4.7-41-7.9 3.7-12.9 8.3-26.2 13.5-39.5 4.1 8 8.4 16 13.1 24 4.7 8 9.5 15.8 14.4 23.4zM420.7 163c9.3 9.6 18.6 20.3 27.8 32-9-.4-18.2-.7-27.5-.7-9.4 0-18.7.2-27.8.7 9-11.7 18.3-22.4 27.5-32zm-74 58.9c-4.9 7.7-9.8 15.6-14.4 23.7-4.6 8-8.9 16-13 24-5.4-13.4-10-26.8-13.8-39.8 13.1-3.1 26.9-5.8 41.2-7.9zm-90.5 125.2c-35.4-15.1-58.3-34.9-58.3-50.6 0-15.7 22.9-35.6 58.3-50.6 8.6-3.7 18-7 27.7-10.1 5.7 19.6 13.2 40 22.5 60.9-9.2 20.8-16.6 41.1-22.2 60.6-9.9-3.1-19.3-6.5-28-10.2zM310 490c-13.6-7.8-19.5-37.5-14.9-75.7 1.1-9.4 2.9-19.3 5.1-29.4 19.6 4.8 41 8.5 63.5 10.9 13.5 18.5 27.5 35.3 41.6 50-32.6 30.3-63.2 46.9-84 46.9-4.5-.1-8.3-1-11.3-2.7zm237.2-76.2c4.7 38.2-1.1 67.9-14.6 75.8-3 1.8-6.9 2.6-11.5 2.6-20.7 0-51.4-16.5-84-46.6 14-14.7 28-31.4 41.3-49.9 22.6-2.4 44-6.1 63.6-11 2.3 10.1 4.1 19.8 5.2 29.1zm38.5-66.7c-8.6 3.7-18 7-27.7 10.1-5.7-19.6-13.2-40-22.5-60.9 9.2-20.8 16.6-41.1 22.2-60.6 9.9 3.1 19.3 6.5 28.1 10.2 35.4 15.1 58.3 34.9 58.3 50.6-.1 15.7-23 35.6-58.4 50.6zM320.8 78.4z"/> |  | ||||||
|         <circle cx="420.9" cy="296.5" r="45.7"/> |  | ||||||
|         <path d="M520.5 78.1z"/> |  | ||||||
|     </g> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 2.6 KiB | 
							
								
								
									
										74
									
								
								src/provider/company.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								src/provider/company.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,74 @@ | |||||||
|  | import { Company } from "../data/company"; | ||||||
|  | import { makeIdSequence } from "./helpers"; | ||||||
|  | 
 | ||||||
|  | const companySequence = makeIdSequence(); | ||||||
|  | const officeSequence = makeIdSequence(); | ||||||
|  | 
 | ||||||
|  | export const sampleCompanies: Company[] = [ | ||||||
|  |     { | ||||||
|  |         id: companySequence(), | ||||||
|  |         name: "Intel", | ||||||
|  |         url: "https://www.intel.com/content/www/us/en/jobs/locations/poland.html", | ||||||
|  |         nip: "9570752316", | ||||||
|  |         offices: [{ | ||||||
|  |             id: officeSequence(), | ||||||
|  |             address: { | ||||||
|  |                 city: "Gdańsk", | ||||||
|  |                 street: "ul. Słowackiego", | ||||||
|  |                 building: "173", | ||||||
|  |                 postalCode: "80-298", | ||||||
|  |                 country: "Poland" | ||||||
|  |             } | ||||||
|  |         }] | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         id: companySequence(), | ||||||
|  |         name: "IHS Markit", | ||||||
|  |         url: "http://ihsgdansk.com/", | ||||||
|  |         nip: "5842068320", | ||||||
|  |         offices: [{ | ||||||
|  |             id: officeSequence(), | ||||||
|  |             address: { | ||||||
|  |                 city: "Gdańsk", | ||||||
|  |                 street: "ul. Marynarki Polskiej", | ||||||
|  |                 building: "163", | ||||||
|  |                 postalCode: "80-868", | ||||||
|  |                 country: "Poland" | ||||||
|  |             } | ||||||
|  |         }] | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         id: companySequence(), | ||||||
|  |         name: "Asseco Poland", | ||||||
|  |         url: "http://pl.asseco.com/", | ||||||
|  |         nip: "5842068320", | ||||||
|  |         offices: [{ | ||||||
|  |             id: officeSequence(), | ||||||
|  |             address: { | ||||||
|  |                 city: "Gdynia", | ||||||
|  |                 street: "ul. Podolska", | ||||||
|  |                 building: "21", | ||||||
|  |                 postalCode: "81-321", | ||||||
|  |                 country: "Poland" | ||||||
|  |             } | ||||||
|  |         }, { | ||||||
|  |             id: officeSequence(), | ||||||
|  |             address: { | ||||||
|  |                 city: "Łódź", | ||||||
|  |                 street: "al. Marszałka Józefa Piłsudskiego", | ||||||
|  |                 building: "85", | ||||||
|  |                 postalCode: "90-332", | ||||||
|  |                 country: "Poland" | ||||||
|  |             } | ||||||
|  |         }, { | ||||||
|  |             id: officeSequence(), | ||||||
|  |             address: { | ||||||
|  |                 city: "Wrocław", | ||||||
|  |                 street: "Traugutta", | ||||||
|  |                 building: "1/7", | ||||||
|  |                 postalCode: "50-449", | ||||||
|  |                 country: "Poland" | ||||||
|  |             } | ||||||
|  |         }] | ||||||
|  |     } | ||||||
|  | ] | ||||||
							
								
								
									
										4
									
								
								src/provider/helpers.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/provider/helpers.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | export const makeIdSequence = (start: number = 1) => { | ||||||
|  |     let i = start; | ||||||
|  |     return () => i++; | ||||||
|  | } | ||||||
| @ -1,5 +0,0 @@ | |||||||
| // jest-dom adds custom jest matchers for asserting on DOM nodes.
 |  | ||||||
| // allows you to do things like:
 |  | ||||||
| // expect(element).toHaveTextContent(/react/i)
 |  | ||||||
| // learn more: https://github.com/testing-library/jest-dom
 |  | ||||||
| import '@testing-library/jest-dom/extend-expect'; |  | ||||||
							
								
								
									
										10
									
								
								src/ui/theme.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/ui/theme.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | import { createMuiTheme } from "@material-ui/core"; | ||||||
|  | 
 | ||||||
|  | export const studentTheme = createMuiTheme({ | ||||||
|  |     props: { | ||||||
|  |         MuiGrid: { | ||||||
|  |             spacing: 4, | ||||||
|  |             xs: 12, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | }) | ||||||
							
								
								
									
										23
									
								
								yarn.lock
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								yarn.lock
									
									
									
									
									
								
							| @ -1309,6 +1309,17 @@ | |||||||
|     react-is "^16.8.0" |     react-is "^16.8.0" | ||||||
|     react-transition-group "^4.4.0" |     react-transition-group "^4.4.0" | ||||||
| 
 | 
 | ||||||
|  | "@material-ui/lab@^4.0.0-alpha.55": | ||||||
|  |   version "4.0.0-alpha.55" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@material-ui/lab/-/lab-4.0.0-alpha.55.tgz#82a850dc59654e04ee3a31be1b34eb3bf64d5585" | ||||||
|  |   integrity sha512-mPHiS52Z1AT6NlWNp07xxTkoKGU3DZhoGdVtLKGy2+uMH5t4/UPtiZLRab7hR3hvuHvguxgV4tkBC9ww3xqUzA== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/runtime" "^7.4.4" | ||||||
|  |     "@material-ui/utils" "^4.9.6" | ||||||
|  |     clsx "^1.0.4" | ||||||
|  |     prop-types "^15.7.2" | ||||||
|  |     react-is "^16.8.0" | ||||||
|  | 
 | ||||||
| "@material-ui/styles@^4.10.0": | "@material-ui/styles@^4.10.0": | ||||||
|   version "4.10.0" |   version "4.10.0" | ||||||
|   resolved "https://registry.yarnpkg.com/@material-ui/styles/-/styles-4.10.0.tgz#2406dc23aa358217aa8cc772e6237bd7f0544071" |   resolved "https://registry.yarnpkg.com/@material-ui/styles/-/styles-4.10.0.tgz#2406dc23aa358217aa8cc772e6237bd7f0544071" | ||||||
| @ -1612,6 +1623,13 @@ | |||||||
|   resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-3.0.3.tgz#3dca0e3f33b200fc7d1139c0cd96c1268cadfd9d" |   resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-3.0.3.tgz#3dca0e3f33b200fc7d1139c0cd96c1268cadfd9d" | ||||||
|   integrity sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA== |   integrity sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA== | ||||||
| 
 | 
 | ||||||
|  | "@types/moment@^2.13.0": | ||||||
|  |   version "2.13.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@types/moment/-/moment-2.13.0.tgz#604ebd189bc3bc34a1548689404e61a2a4aac896" | ||||||
|  |   integrity sha1-YE69GJvDvDShVIaJQE5hoqSqyJY= | ||||||
|  |   dependencies: | ||||||
|  |     moment "*" | ||||||
|  | 
 | ||||||
| "@types/node@*": | "@types/node@*": | ||||||
|   version "13.9.2" |   version "13.9.2" | ||||||
|   resolved "https://registry.yarnpkg.com/@types/node/-/node-13.9.2.tgz#ace1880c03594cc3e80206d96847157d8e7fa349" |   resolved "https://registry.yarnpkg.com/@types/node/-/node-13.9.2.tgz#ace1880c03594cc3e80206d96847157d8e7fa349" | ||||||
| @ -7017,6 +7035,11 @@ mkdirp@^0.5.1, mkdirp@^0.5.3, mkdirp@~0.5.1: | |||||||
|   dependencies: |   dependencies: | ||||||
|     minimist "^1.2.5" |     minimist "^1.2.5" | ||||||
| 
 | 
 | ||||||
|  | moment@*, moment@^2.26.0: | ||||||
|  |   version "2.26.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/moment/-/moment-2.26.0.tgz#5e1f82c6bafca6e83e808b30c8705eed0dcbd39a" | ||||||
|  |   integrity sha512-oIixUO+OamkUkwjhAVE18rAMfRJNsNe/Stid/gwHSOfHrOtw9EhAY2AHvdKZ/k/MggcYELFCJz/Sn2pL8b8JMw== | ||||||
|  | 
 | ||||||
| move-concurrently@^1.0.1: | move-concurrently@^1.0.1: | ||||||
|   version "1.0.1" |   version "1.0.1" | ||||||
|   resolved "https://registry.yarnpkg.com/move-concurrently/-/move-concurrently-1.0.1.tgz#be2c005fda32e0b29af1f05d7c4b33214c701f92" |   resolved "https://registry.yarnpkg.com/move-concurrently/-/move-concurrently-1.0.1.tgz#be2c005fda32e0b29af1f05d7c4b33214c701f92" | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user