Created
March 12, 2020 09:07
-
-
Save zaydek/f80406342d8bb5951ab83c3435106d7b to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import React, { Component } from "react"; | |
import { CatServiceClient } from "../../proto/api/v1/ApiServiceClientPb"; | |
import { | |
GetFeaturedCatsRequest, | |
Cat, | |
GetFeaturedCatsResponse | |
} from "../../proto/api/v1/api_pb"; | |
import { | |
Icon, | |
Text, | |
Box, | |
Image, | |
Heading, | |
Grid, | |
CircularProgress, | |
Alert, | |
AlertIcon, | |
AlertDescription, | |
AlertTitle | |
} from "@chakra-ui/core"; | |
import { Error, Status } from "grpc-web"; | |
// interface FeaturedCatsProps { | |
// client: CatServiceClient; | |
// } | |
// | |
// interface FeaturedCatsState { | |
// error: Error | undefined; | |
// isLoaded: boolean; | |
// cats: Cat[]; | |
// } | |
// You can destructure client to make things a little easier | |
// to understand. You’re using TypeScript and interfaces to | |
// achieve the same effect, but this also works. | |
// | |
const FeaturedCats = ({ client, ...props }) => { | |
const [state, setState] = React.useState({ | |
isLoaded: false, | |
error: null, | |
cats: [], | |
}) | |
// React.useEffect is a simple hook that ‘reacts’ to | |
// dependencies. Right now, this hook has one dependency; | |
// client. | |
// | |
// Everytime client changes, this hook would rerun. But | |
// for all intents and purposes, client is probably not | |
// going to change, so you can expect this hook to run | |
// once, when your component mounts. | |
// | |
React.useEffect(() => { | |
let req = new GetFeaturedCatsRequest(); | |
const call = client.getFeaturedCats(req, null, (err: Error, response: GetFeaturedCatsResponse) => { | |
console.log(err, response); | |
// NOTE: You can probably collapse these three | |
// setState calls to once. But sometimes you want to | |
// separate them to keep things cleaner, easier to | |
// understand. | |
setState(current => { | |
...current, | |
isLoaded: true, | |
}); | |
if (err) { | |
setState(current => { | |
...current, | |
error: err, | |
}); | |
return; | |
} | |
setState(current => { | |
...current, | |
cats: response.getCatsList(), | |
}); | |
}); | |
call.on("status", (status: Status) => { | |
if (status.metadata) { | |
console.log("Received metadata"); | |
console.log(status.metadata); | |
} | |
}); | |
// NOTE: If you need to tear down any resources when | |
// your component unmounts, you can optionally return a | |
// function/closure. This is very similar to how defer | |
// works in Go. | |
// | |
// For example: | |
// | |
// return () => { | |
// client.defer() | |
// } | |
// | |
// Or: | |
// | |
// return client.defer | |
// | |
}, [client]) | |
// You can collocate your loading, error states but you | |
// could also extract them as separate components like | |
// this: | |
// | |
// const LoadingState = ({ state, ...props }) => ( | |
// <Box> | |
// {/* ... */} | |
// </Box> | |
// ) | |
// | |
// And then simply return: | |
// | |
// return <LoadingState state={state} /> | |
if (!state.isLoaded) { | |
return ( | |
<Box | |
display="flex" | |
flexDirection="column" | |
alignItems="center" | |
justifyContent="flex-start" | |
border={5} | |
borderRadius={5} | |
p={5} | |
textAlign="left" | |
> | |
<Box> | |
<Box> | |
<Heading | |
size="xl" | |
as="h2" | |
lineHeight="shorter" | |
fontWeight="bold" | |
fontFamily="heading" | |
> | |
Featured cats | |
</Heading> | |
</Box> | |
</Box> | |
<CircularProgress isIndeterminate color="gray.300"></CircularProgress> | |
</Box> | |
); | |
} | |
if (state.error) { | |
return ( | |
<Box | |
display="flex" | |
flexDirection="column" | |
alignItems="center" | |
justifyContent="flex-start" | |
border={5} | |
borderRadius={5} | |
p={5} | |
textAlign="left" | |
> | |
<Box> | |
<Box> | |
<Heading | |
size="xl" | |
as="h2" | |
lineHeight="shorter" | |
fontWeight="bold" | |
fontFamily="heading" | |
> | |
Featured cats | |
</Heading> | |
</Box> | |
</Box> | |
<Alert status="error"> | |
<AlertIcon /> | |
<AlertTitle mr={1}>Error</AlertTitle> | |
<AlertDescription>{state.error.message}</AlertDescription> | |
</Alert> | |
</Box> | |
); | |
} | |
return ( | |
<> | |
<Box | |
display="flex" | |
flexDirection="column" | |
alignItems="center" | |
justifyContent="flex-start" | |
border={5} | |
borderRadius={5} | |
p={5} | |
textAlign="left" | |
> | |
<Box> | |
<Box> | |
<Heading | |
size="xl" | |
as="h2" | |
lineHeight="shorter" | |
fontWeight="bold" | |
fontFamily="heading" | |
> | |
Featured cats | |
</Heading> | |
</Box> | |
</Box> | |
</Box> | |
<Grid | |
justifyContent="center" | |
p={2} | |
gap={2} | |
templateColumns={"repeat(auto-fit, 224px)"} | |
> | |
{state.cats.map((value: Cat, index: Number, array: Cat[]) => { | |
return ( | |
<Box | |
display="block" | |
borderRadius="lg" | |
backgroundColor="white" | |
pl={3} | |
pr={3} | |
pt={5} | |
pb={5} | |
shadow="sm" | |
width="max-content" | |
> | |
<Box shadow={0} border={0} borderRadius={0}> | |
<Image | |
height={200} | |
width={200} | |
fallbackSrc="" | |
/> | |
<Box | |
display="flex" | |
flexDirection="row" | |
alignItems="center" | |
justifyContent="flex-start" | |
pt={3} | |
pl={1} | |
pr={1} | |
> | |
<Icon name="chat" mr={1} /> | |
<Text fontSize="lg" textAlign="center" fontStyle="italic"> | |
{value.getName()} | |
</Text> | |
</Box> | |
</Box> | |
</Box> | |
); | |
})} | |
</Grid> | |
</> | |
); | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment