Skip to content

Instantly share code, notes, and snippets.

@zaydek
Created March 12, 2020 09:07
Show Gist options
  • Save zaydek/f80406342d8bb5951ab83c3435106d7b to your computer and use it in GitHub Desktop.
Save zaydek/f80406342d8bb5951ab83c3435106d7b to your computer and use it in GitHub Desktop.
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