Completion of Form Component, Component Viewer, and Documentation
This commit is contained in:
parent
c5397aa793
commit
6d89fb8657
45
README.md
45
README.md
@ -1,17 +1,46 @@
|
||||
# react-interview-q1
|
||||
|
||||
## Instructions
|
||||
## Summary
|
||||
|
||||
This was a coding challenge. To see the original prompt, scroll down to the "Original Prompt" heading. Nothing is edited below that. Above that, under the "My Implementation" heading, I will give some explanations as to why I made the choices I did. This was a really fun challenge because I actually used it as an opportunity to work outside of React's data model and leverage RxJS. This came with a couple fun edge cases to solve for and I am happy to nerd out about the solutions I came up with - Feel free to reach out!
|
||||
|
||||
## My Implementation
|
||||
|
||||
### Styling Notes
|
||||
|
||||
1. CSS follows in the footsteps of create-react-app and are external files pulled in via imports. There are no frameworks or libraries used for styling. Modern features like Flexbox and container queries are used to make the form responsive. Components are styled with component name prefixes to reduce the chance of class name collisions.
|
||||
2. I wanted to easily adjust the form components size when styling, so I created the `ComponentViewer` component to wrap the form. This provides a slider to adjust the width of the nested component (in this case `Form`) dynamically to see how it looks at different sizes without opening dev tools.
|
||||
|
||||
### Component Composition Notes
|
||||
|
||||
1. The job description this coding challenge was for listed RxJS as a requirement. I hadn't used RxJS before, so I used this as an opportunity to learn it. Because of this, I broke up the form into smaller components and chose not to pass data through props. Instead, I used RxJS `Subjects` to push data from the child components to anyone who would listen. In this case, the parent component, `Form`, listens to the `NameInput` and `LocationInput` components found in `src/components/atoms`.
|
||||
2. The `RenderTable` component is kept pretty simple and more closely follows idiomatic React. It receives data from the `Form` component and iterates through to render the table rows. There is also a minimum of 5 rows that will always be rendered. In order to achieve this though, key's are explicitly not provided so that React doesn't accidentally persist data when clearing. This is a trade off, but I think the user experience of seeing the table is more important than the minuscule performance loss. If this were populated with a large amount of data, I would reconsider this decision. While there would be a technical solution to achieve both I didn't want to sacrifice code readability for a small performance gain.
|
||||
|
||||
### Just Cool Stuff
|
||||
|
||||
1. `LocationInput` is more like a Solid component in that once React renders it, there is no reason it needs to be rendered agin (YAY Performance!) and it relies on 'signals'. This is because the data is fetched from the API on mount and populated initially. Alongside this action, a listener is mounted so that whenever there is a change on the select element, it is dispatched straight to the `Subject` which makes the selection available to any other component outside of Reacts rendering cycle.
|
||||
2. `NameInput` is a little more complex. Similar to `LocationInput`, it pushes its state changes through a `Subject` making it available anywhere. Before we can do this though, we need to validate the input. Name validation is an async operation though and if we just validate every keystroke, we won't know if the response we get back is for the most recent keystroke. To solve this, I used an RxJS Operator to ignore all validation responses except the most recent. This eliminated the risk of the user seeing a false positive or negative caused by an older request coming back after a newer one.
|
||||
|
||||
This still has a massive problem though because the response to the most recent request could come back after the user has changed the input again. This is because the user could have typed in a new character and we are still waiting for that response to be received. There is no way to verify that the response is for the current string in the input box. The API is not available for us to make changes to as well so I solved this by wrapping every request in a new promise that includes the name we asked to validate. This way, we can save the name in a ref that is kept in sync with the input box and compare that with the validation response. If they are the same, we know the user has stopped typing and we can dispatch the valid name to the subject.
|
||||
|
||||
To make this a little better of a user experience, and because all the building blocks were there, I added additional states for validation. Every time a new key is pressed a `pending` is dispatched to the subject and a message is shown to the user. If the response comes back and the name is the same as the current input, the message is updated to reflect the validity of the name and an additional message is dispatched to the `subject` to inform other components that the name is valid or not.
|
||||
|
||||
This is actually consumed by the `Form` component to enable or disable the submit button only when the name is guaranteed to be valid. (using the Mock API of course, no way to solve the two generals problem here)
|
||||
|
||||
## Original Prompt
|
||||
|
||||
### Instructions
|
||||
|
||||
Fork this repo first into your own github account. Make sure to thoroughly read the instructions and implement the react component to meet the provided requirements. Send back a link to your cloned repo. You are expected to make implementation choices around customer experience and efficiency. Please make sure to explain your choices in comments.
|
||||
|
||||
## Requirements
|
||||
### Requirements
|
||||
|
||||
Please build the following form component
|
||||

|
||||
|
||||
* Name input should be validated using the provided mock API to check whether the chosen name is taken or not.
|
||||
* Name input should be validated as the user is typing.
|
||||
* Location dropdown options should be fetched using the provided mock API.
|
||||
* Component should have a responsive layout
|
||||
* Component should be appropriately styled
|
||||
* Unit tests are not required
|
||||
- Name input should be validated using the provided mock API to check whether the chosen name is taken or not.
|
||||
- Name input should be validated as the user is typing.
|
||||
- Location dropdown options should be fetched using the provided mock API.
|
||||
- Component should have a responsive layout
|
||||
- Component should be appropriately styled
|
||||
- Unit tests are not required
|
||||
|
16
package-lock.json
generated
16
package-lock.json
generated
@ -8,6 +8,9 @@
|
||||
"name": "react-interview-q1",
|
||||
"version": "1.0.0",
|
||||
"license": "MIT",
|
||||
"dependencies": {
|
||||
"rxjs": "^7.8.1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"create-react-app": "^5.0.1"
|
||||
}
|
||||
@ -453,6 +456,14 @@
|
||||
"rimraf": "bin.js"
|
||||
}
|
||||
},
|
||||
"node_modules/rxjs": {
|
||||
"version": "7.8.1",
|
||||
"resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.1.tgz",
|
||||
"integrity": "sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==",
|
||||
"dependencies": {
|
||||
"tslib": "^2.1.0"
|
||||
}
|
||||
},
|
||||
"node_modules/safe-buffer": {
|
||||
"version": "5.1.2",
|
||||
"resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
|
||||
@ -626,6 +637,11 @@
|
||||
"url": "https://github.com/sponsors/isaacs"
|
||||
}
|
||||
},
|
||||
"node_modules/tslib": {
|
||||
"version": "2.6.2",
|
||||
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz",
|
||||
"integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q=="
|
||||
},
|
||||
"node_modules/uid-number": {
|
||||
"version": "0.0.6",
|
||||
"resolved": "https://registry.npmjs.org/uid-number/-/uid-number-0.0.6.tgz",
|
||||
|
@ -17,5 +17,8 @@
|
||||
"homepage": "https://github.com/MasterRyd3l/react-interview-q1#readme",
|
||||
"devDependencies": {
|
||||
"create-react-app": "^5.0.1"
|
||||
},
|
||||
"dependencies": {
|
||||
"rxjs": "^7.8.1"
|
||||
}
|
||||
}
|
||||
|
@ -1,38 +1,5 @@
|
||||
.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);
|
||||
}
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
}
|
||||
|
@ -1,23 +1,13 @@
|
||||
import logo from './logo.svg';
|
||||
import './App.css';
|
||||
import ComponentViewer from "./components/ComponentViewer";
|
||||
import Form from "./components/Form";
|
||||
import "./App.css";
|
||||
|
||||
function App() {
|
||||
return (
|
||||
<div className="App">
|
||||
<header className="App-header">
|
||||
<img src={logo} className="App-logo" alt="logo" />
|
||||
<p>
|
||||
Edit <code>src/App.js</code> and save to reload.
|
||||
</p>
|
||||
<a
|
||||
className="App-link"
|
||||
href="https://reactjs.org"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
>
|
||||
Learn React
|
||||
</a>
|
||||
</header>
|
||||
<ComponentViewer name="Form Component">
|
||||
<Form />
|
||||
</ComponentViewer>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
@ -0,0 +1,5 @@
|
||||
/* Style Injection to size component
|
||||
to that of what the slider specifies*/
|
||||
.ComponentViewer div {
|
||||
box-sizing: border-box;
|
||||
}
|
25
solution/src/components/ComponentViewer/index.js
Normal file
25
solution/src/components/ComponentViewer/index.js
Normal file
@ -0,0 +1,25 @@
|
||||
import React, { useState } from "react";
|
||||
|
||||
export default function ComponentViewer({ name, children }) {
|
||||
const [width, setWidth] = useState(450);
|
||||
|
||||
return (
|
||||
<>
|
||||
<h1>{name}</h1>
|
||||
<label htmlFor="width-slider">Component Width: {width + "px"}</label>
|
||||
<input
|
||||
type="range"
|
||||
id="width-slider"
|
||||
name="width-slider"
|
||||
value={width}
|
||||
min="200"
|
||||
max="900"
|
||||
onChange={(e) => setWidth(e.target.value)}
|
||||
></input>
|
||||
<br />
|
||||
|
||||
{/* Set the container width to the slider definition */}
|
||||
<div style={{ width: `${width}px` }}>{children}</div>
|
||||
</>
|
||||
);
|
||||
}
|
18
solution/src/components/Form/Form.css
Normal file
18
solution/src/components/Form/Form.css
Normal file
@ -0,0 +1,18 @@
|
||||
.Form-container {
|
||||
border: solid black;
|
||||
width: 100%;
|
||||
container-type: inline-size;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.Form-actions {
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
gap: 20px;
|
||||
margin: 10px;
|
||||
}
|
||||
|
||||
.Form-container > form {
|
||||
max-width: 515px;
|
||||
margin: 1.5rem auto;
|
||||
}
|
50
solution/src/components/Form/index.js
Normal file
50
solution/src/components/Form/index.js
Normal file
@ -0,0 +1,50 @@
|
||||
import NameInput, { selectedName } from "../atoms/NameInput";
|
||||
import LocationInput, { selectedLocation } from "../atoms/LocationInput";
|
||||
import RenderTable from "../atoms/RenderTable";
|
||||
import "./Form.css";
|
||||
import { useEffect, useRef, useState } from "react";
|
||||
|
||||
export default function Form() {
|
||||
const [table, setTable] = useState([]);
|
||||
const [isNameValid, setIsNameValid] = useState(false);
|
||||
const locationRef = useRef();
|
||||
const nameRef = useRef();
|
||||
|
||||
useEffect(() => {
|
||||
selectedLocation.subscribe((value) => {
|
||||
locationRef.current = value;
|
||||
});
|
||||
|
||||
selectedName.subscribe((value) => {
|
||||
nameRef.current = value;
|
||||
value.isValid === true ? setIsNameValid(true) : setIsNameValid(false);
|
||||
});
|
||||
}, []);
|
||||
|
||||
function addToTable() {
|
||||
setTable([
|
||||
{ name: nameRef.current.name, location: locationRef.current },
|
||||
...table,
|
||||
]);
|
||||
}
|
||||
|
||||
function clearTable() {
|
||||
setTable([]);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="Form-container">
|
||||
<form onSubmit={(e) => e.preventDefault()}>
|
||||
<NameInput />
|
||||
<LocationInput />
|
||||
<div className="Form-actions">
|
||||
<button onClick={clearTable}>Clear</button>
|
||||
<button type="submit" disabled={!isNameValid} onClick={addToTable}>
|
||||
Add
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
<RenderTable records={table} />
|
||||
</div>
|
||||
);
|
||||
}
|
@ -0,0 +1,45 @@
|
||||
@container (width > 0px) {
|
||||
.LocationInput {
|
||||
container: inline-size;
|
||||
max-width: 515px;
|
||||
margin: 5px 5px 25px 5px;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
.LocationInput label {
|
||||
display: block;
|
||||
}
|
||||
|
||||
.LocationInput > select {
|
||||
box-sizing: border-box;
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
|
||||
@container (width > 300px) {
|
||||
.LocationInput {
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.LocationInput label {
|
||||
flex: 0 80px;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
padding-top: 5px;
|
||||
padding-bottom: 5px;
|
||||
margin-right: 15px;
|
||||
}
|
||||
|
||||
.LocationInput > select {
|
||||
flex: 1;
|
||||
justify-content: flex-start;
|
||||
font-size: 1rem;
|
||||
padding: 0px;
|
||||
}
|
||||
}
|
||||
|
||||
@container (width > 430px) {
|
||||
.LocationInput label {
|
||||
font-size: 1.2rem;
|
||||
}
|
||||
}
|
51
solution/src/components/atoms/LocationInput/index.js
Normal file
51
solution/src/components/atoms/LocationInput/index.js
Normal file
@ -0,0 +1,51 @@
|
||||
import { useState, useEffect } from "react";
|
||||
import { getLocations } from "../../../mock-api/apis.js";
|
||||
import "./LocationInput.css";
|
||||
import { Subject } from "rxjs";
|
||||
|
||||
// Fetches location options on mount
|
||||
// Shares currently selected location with any listeners (anywhere!)
|
||||
// Finite number of renders -> no pass by props re-rendering
|
||||
export default function LocationInput() {
|
||||
const [options, setOptions] = useState([]);
|
||||
|
||||
// on component mount (Only runs once!)
|
||||
useEffect(() => {
|
||||
// Fetch selectable locations
|
||||
getLocations().then((locations) => {
|
||||
// push the first location to the selectedLocation subject
|
||||
// This is how we share the selected location with listeners
|
||||
selectedLocation.next(locations[0]);
|
||||
|
||||
// Render the location options and save to state for re-render
|
||||
// Options are static -> no need to re-render after this.
|
||||
setOptions(
|
||||
locations.map((option) => {
|
||||
return (
|
||||
<option key={option} value={option}>
|
||||
{option}
|
||||
</option>
|
||||
);
|
||||
}),
|
||||
);
|
||||
});
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<div className="LocationInput">
|
||||
<label htmlFor="location">Location</label>
|
||||
<select
|
||||
name="location"
|
||||
id="location"
|
||||
onChange={(e) => {
|
||||
// Push new location to listeners
|
||||
selectedLocation.next(e.target.value);
|
||||
}}
|
||||
>
|
||||
{options}
|
||||
</select>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export const selectedLocation = new Subject();
|
89
solution/src/components/atoms/NameInput/NameInput.css
Normal file
89
solution/src/components/atoms/NameInput/NameInput.css
Normal file
@ -0,0 +1,89 @@
|
||||
@container (width > 0px) {
|
||||
.NameInput {
|
||||
container: inline-size;
|
||||
max-width: 515px;
|
||||
margin: 5px 5px 25px 5px;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
.NameInput label {
|
||||
display: block;
|
||||
}
|
||||
|
||||
.NameInput > span > input {
|
||||
box-sizing: border-box;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.NameInput > span:nth-child(2) > span {
|
||||
height: 16px;
|
||||
}
|
||||
|
||||
#Form-error {
|
||||
color: red;
|
||||
}
|
||||
|
||||
#Form-pending {
|
||||
color: gold;
|
||||
}
|
||||
|
||||
#Form-valid {
|
||||
color: green;
|
||||
}
|
||||
}
|
||||
|
||||
@container (width > 300px) {
|
||||
.NameInput {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
/* .NameInput label {
|
||||
padding-top: 5px;
|
||||
padding-bottom: 5px;
|
||||
margin-left: 5px;
|
||||
width: 80px;
|
||||
text-align: right;
|
||||
display: inline-block;
|
||||
} */
|
||||
|
||||
.NameInput span {
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.NameInput > span > label,
|
||||
.NameInput > span > span {
|
||||
flex: 0 80px;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
padding-top: 5px;
|
||||
padding-bottom: 5px;
|
||||
margin-right: 15px;
|
||||
}
|
||||
|
||||
.NameInput > span > input,
|
||||
.NameInput > span > span:nth-child(2) {
|
||||
flex: 1;
|
||||
justify-content: flex-start;
|
||||
margin-left: 5px;
|
||||
font-size: 1rem;
|
||||
padding: 0px;
|
||||
}
|
||||
|
||||
.NameInput > span:nth-child(2) > span {
|
||||
font-size: 1rem;
|
||||
height: 1rem;
|
||||
}
|
||||
}
|
||||
|
||||
@container (width > 430px) {
|
||||
.NameInput label,
|
||||
.NameInput > span > input {
|
||||
font-size: 1.2rem;
|
||||
}
|
||||
|
||||
.NameInput > span > span:nth-child(2) {
|
||||
font-size: 1.5rem;
|
||||
}
|
||||
}
|
101
solution/src/components/atoms/NameInput/index.js
Normal file
101
solution/src/components/atoms/NameInput/index.js
Normal file
@ -0,0 +1,101 @@
|
||||
import "./NameInput.css";
|
||||
import { isNameValid as getNameValidity } from "../../../mock-api/apis.js";
|
||||
import { useState, useEffect, useRef } from "react";
|
||||
import { Subject, from, switchAll } from "rxjs";
|
||||
|
||||
// The native api does not include the name that was checked
|
||||
// This can cause an incorrect determination of the validity
|
||||
// of the name if the name is changed before the api call is
|
||||
// completed. This function wraps the api call to include the
|
||||
// name that was checked to compare with the current name.
|
||||
function isNameValid(name) {
|
||||
return new Promise((resolve) => {
|
||||
getNameValidity(name).then((value) => {
|
||||
resolve({ name: name, isValid: value });
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
// Checks if the name is valid
|
||||
// Shares the name with any listeners (anywhere!)
|
||||
// Finite number of renders -> no pass by props re-rendering
|
||||
// Guarantees the following
|
||||
// A subscriber will always get the latest name
|
||||
// A subscriber will always know if the 'settled' name is valid
|
||||
// A subscriber can read if a name has not 'settled' and is still pending validation
|
||||
export default function NameInput() {
|
||||
const [name, setName] = useState("");
|
||||
const [isError, setIsError] = useState(false);
|
||||
const [isPending, setIsPending] = useState(false);
|
||||
const [isValid, setIsValid] = useState(false);
|
||||
const nameRef = useRef(name);
|
||||
|
||||
// Anytime name is updated
|
||||
useEffect(() => {
|
||||
if (name.length > 0) {
|
||||
// Send the name to the api to check if it is valid (async)
|
||||
let observable = from(isNameValid(name));
|
||||
latestObservable.next(observable);
|
||||
|
||||
// Update the nameRef to validate that the name has not changed
|
||||
// after the above api call was made
|
||||
nameRef.current = name;
|
||||
|
||||
// Broadcast the latest name and it's pending status
|
||||
selectedName.next({ name: name, isValid: "pending" });
|
||||
setIsPending(true);
|
||||
setIsError(false);
|
||||
setIsValid(false);
|
||||
} else {
|
||||
selectedName.next({ name: "", isValid: "false" });
|
||||
setIsPending(false);
|
||||
setIsError(false);
|
||||
setIsValid(false);
|
||||
}
|
||||
}, [name]);
|
||||
|
||||
// Mount the observable listener on component mount
|
||||
useEffect(() => {
|
||||
// Only read the most recent API response and ignore the rest
|
||||
latestObservable.pipe(switchAll()).subscribe((nameObject) => {
|
||||
// Check that the name has not changed since the api call was made
|
||||
if (nameObject.name === nameRef.current) {
|
||||
setIsPending(false);
|
||||
// Show error if name is invalid
|
||||
nameObject.isValid ? setIsError(false) : setIsError(true);
|
||||
nameObject.isValid ? setIsValid(true) : setIsValid(false);
|
||||
|
||||
// Broadcast the latest name and it's validity
|
||||
selectedName.next(nameObject);
|
||||
}
|
||||
});
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<div className="NameInput">
|
||||
<span>
|
||||
<label htmlFor="name">Name</label>
|
||||
<input
|
||||
type="text"
|
||||
id="name"
|
||||
name="name"
|
||||
value={name}
|
||||
onChange={(e) => {
|
||||
setName(e.target.value);
|
||||
}}
|
||||
/>
|
||||
</span>
|
||||
<span>
|
||||
<span></span> {/* Spacer */}
|
||||
{isError ? (
|
||||
<span id="Form-error">This name has already been taken</span>
|
||||
) : null}
|
||||
{isPending ? <span id="Form-pending">Checking...</span> : null}
|
||||
{isValid ? <span id="Form-valid">Available!</span> : null}
|
||||
</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
const latestObservable = new Subject();
|
||||
export const selectedName = new Subject();
|
35
solution/src/components/atoms/RenderTable/RenderTable.css
Normal file
35
solution/src/components/atoms/RenderTable/RenderTable.css
Normal file
@ -0,0 +1,35 @@
|
||||
.Form-table {
|
||||
border-top: solid black 2px;
|
||||
border-spacing: 0;
|
||||
empty-cells: show;
|
||||
width: 100%;
|
||||
max-width: 515px;
|
||||
}
|
||||
|
||||
.Form-table th:nth-child(1),
|
||||
.Form-table td:nth-child(1) {
|
||||
border-right: solid black 2px;
|
||||
}
|
||||
|
||||
.Form-table thead tr {
|
||||
background-color: lightgray;
|
||||
text-align: left;
|
||||
border: solid lightgray 2px;
|
||||
}
|
||||
|
||||
.Form-table tbody tr:nth-child(even) {
|
||||
background-color: #ffffff;
|
||||
height: 22px;
|
||||
}
|
||||
|
||||
.Form-table tbody tr:nth-child(odd) {
|
||||
background-color: #f2f2f2;
|
||||
height: 22px;
|
||||
}
|
||||
|
||||
@container (width > 515px) {
|
||||
.Form-table {
|
||||
border: solid black 2px;
|
||||
margin: 25px auto;
|
||||
}
|
||||
}
|
33
solution/src/components/atoms/RenderTable/index.js
Normal file
33
solution/src/components/atoms/RenderTable/index.js
Normal file
@ -0,0 +1,33 @@
|
||||
import "./RenderTable.css";
|
||||
|
||||
export default function RenderTable({ records }) {
|
||||
const tableRows = records.map((record) => {
|
||||
return (
|
||||
<tr>
|
||||
<td>{record.name}</td>
|
||||
<td>{record.location}</td>
|
||||
</tr>
|
||||
);
|
||||
});
|
||||
|
||||
while (tableRows.length < 5) {
|
||||
tableRows.push(
|
||||
<tr>
|
||||
<td></td>
|
||||
<td></td>
|
||||
</tr>,
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<table className="Form-table">
|
||||
<thead>
|
||||
<tr>
|
||||
<th scope="col">Name</th>
|
||||
<th scope="col">Location</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>{tableRows}</tbody>
|
||||
</table>
|
||||
);
|
||||
}
|
Loading…
Reference in New Issue
Block a user