Skip to content

Commit 5c57315

Browse files
committed
Initial commit
0 parents  commit 5c57315

27 files changed

+7519
-0
lines changed

.babelrc

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
{
2+
"presets": ["next/babel"],
3+
"plugins": ["babel-plugin-inline-import-graphql-ast"]
4+
}

.dockerignore

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
.env
2+
.env.*
3+
.next/
4+
.eslint*
5+
README.md
6+
node_modules/

.eslintignore

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
node_modules/
2+
dist/
3+
build/
4+
.next/
5+
.git/

.eslintrc.js

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module.exports = {
2+
extends: '@codeday',
3+
};

.gitignore

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
node_modules/
2+
.next/
3+
.env
4+
.env.*

Dockerfile

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
FROM node:13-alpine
2+
3+
ENV NODE_ENV=production
4+
RUN mkdir /app
5+
COPY package.json /app
6+
COPY yarn.lock /app
7+
WORKDIR /app
8+
9+
RUN yarn install
10+
RUN yarn add --dev strip-ansi
11+
12+
COPY . /app
13+
COPY ./docker-entrypoint.sh /docker-entrypoint.sh
14+
CMD /docker-entrypoint.sh

docker-entrypoint.sh

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
npm run build
2+
npm run start

package.json

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
{
2+
"name": "www-corp",
3+
"version": "0.0.1",
4+
"private": true,
5+
"scripts": {
6+
"dev": "next dev",
7+
"build": "next build",
8+
"start": "next start"
9+
},
10+
"dependencies": {
11+
"@codeday/topo": "^3.5.0",
12+
"@codeday/topocons": "^1.3.0",
13+
"@emotion/core": "^10.0.34",
14+
"@emotion/styled": "^10.0.27",
15+
"@types/react": "^16.9.46",
16+
"babel-plugin-inline-import-graphql-ast": "^2.4.1",
17+
"emotion-theming": "^10.0.27",
18+
"graphql": "^15.3.0",
19+
"graphql-tag": "^2.11.0",
20+
"hls.js": "^0.14.8",
21+
"knuth-shuffle-seeded": "^1.0.6",
22+
"next": "^9.5.2",
23+
"next-seo": "^4.7.3",
24+
"prop-types": "^15.7.2",
25+
"random-seed": "^0.3.0",
26+
"react": "^16.13.1",
27+
"react-countup": "^4.3.3",
28+
"react-dom": "^16.13.1",
29+
"react-responsive-modal": "^5.0.2"
30+
},
31+
"devDependencies": {
32+
"@codeday/eslint-config": "^2.1.2",
33+
"eslint": "^7.6.0"
34+
}
35+
}

public/favicon.ico

1.12 KB
Binary file not shown.

src/components/FlexScatter.js

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
import React from 'react';
2+
import PropTypes from 'prop-types';
3+
import { create } from 'random-seed';
4+
import Box, { Flex } from '@codeday/topo/Atom/Box';
5+
6+
export default function FlexScatter({
7+
children, seed, gapMin, gapMax, yOffsetMin, yOffsetMax, ...props
8+
}) {
9+
const rand = create(seed);
10+
const offsetChildren = children.map((c) => {
11+
const offset = rand.intBetween(yOffsetMin, yOffsetMax);
12+
return (
13+
<Box
14+
position="relative"
15+
marginTop={`${offset}px`}
16+
key="s-c.key"
17+
>
18+
{c}
19+
</Box>
20+
);
21+
});
22+
23+
const gaps = (new Array(children.length + 1)).fill(undefined).map((_, i) => (
24+
<Box
25+
flexBasis={`${rand.intBetween(gapMin, gapMax)}px`}
26+
flexGrow={rand.intBetween(1, 8)}
27+
flexShrink={rand.intBetween(1, 8)}
28+
// eslint-disable-next-line react/no-array-index-key
29+
key={`flex-gap-${i}`}
30+
/>
31+
));
32+
const zipper = (new Array(children.length * 2)).fill(undefined).map((_, i) => (
33+
i % 2 === 0 ? gaps : offsetChildren)[Math.floor(i / 2)]);
34+
zipper.push(gaps[gaps.length - 1]);
35+
36+
return (
37+
<Flex justify="space-around" alignContent="space-around" wrap="wrap" {...props}>
38+
{zipper}
39+
</Flex>
40+
);
41+
}
42+
FlexScatter.propTypes = {
43+
children: PropTypes.arrayOf(PropTypes.element).isRequired,
44+
seed: PropTypes.oneOfType([PropTypes.string, PropTypes.number]).isRequired,
45+
gapMin: PropTypes.number.isRequired,
46+
gapMax: PropTypes.number.isRequired,
47+
yOffsetMin: PropTypes.number.isRequired,
48+
yOffsetMax: PropTypes.number.isRequired,
49+
};

src/components/Index/Hero.gql

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
fragment IndexHeroComponent on Query {
2+
cms {
3+
mission: strings (where: { key: "common.mission" }) {
4+
items {
5+
value
6+
}
7+
}
8+
explainer: asset(id:"6BoY9Ec9J8YRrHcj2RqsCg"){
9+
url
10+
}
11+
indexHeroPhotos: pressPhotos(limit: 50) {
12+
items {
13+
region {
14+
name
15+
}
16+
event {
17+
program {
18+
name
19+
}
20+
startsAt
21+
}
22+
photo {
23+
jpg: url(transform: { width: 200, height: 150, format: JPG_PROGRESSIVE, resizeStrategy: FILL })
24+
webp: url(transform: { width: 200, height: 150, format: JPG_PROGRESSIVE, resizeStrategy: FILL })
25+
}
26+
}
27+
}
28+
}
29+
}

src/components/Index/Hero.js

+95
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
1+
import React from 'react';
2+
import { create } from 'random-seed';
3+
import shuffle from 'knuth-shuffle-seeded';
4+
import Box, { Grid } from '@codeday/topo/Atom/Box';
5+
import Text, { Heading } from '@codeday/topo/Atom/Text';
6+
import Button from '@codeday/topo/Atom/Button';
7+
import Play from '@codeday/topocons/Icon/MediaPlay';
8+
import Image from '@codeday/topo/Atom/Image';
9+
import Slides from '@codeday/topo/Molecule/Slides';
10+
import FlexScatter from '../FlexScatter';
11+
import VideoLink from '../VideoLink';
12+
import { useQuery } from '../../query';
13+
import splitGroups from '../../utils/splitGroups';
14+
15+
function photoSlides(rand, photoGroups) {
16+
return photoGroups.map((pg, i) => (
17+
<Slides key={i} w="200px" h="150px" duration={rand.floatBetween(8, 15)}>
18+
{pg.map((p) => (
19+
<Box key={p.photo.jpg} position="relative">
20+
<Image src={p.photo.jpg} />
21+
<Text
22+
position="absolute"
23+
bottom={0}
24+
left={0}
25+
right={0}
26+
mb={0}
27+
bg={`blackAlpha.500`}
28+
color="white"
29+
p={1}
30+
fontSize="sm"
31+
>
32+
{[p.event?.program?.name, p.region?.name, p.event?.startsAt?.substring(0,4)].join(' ')}
33+
</Text>
34+
</Box>
35+
))}
36+
</Slides>
37+
));
38+
}
39+
40+
export default function Hero({ seed, ...props }) {
41+
const { cms: { indexHeroPhotos, mission, explainer } } = useQuery();
42+
43+
const rand = create(seed);
44+
const photos = shuffle(indexHeroPhotos?.items || [], seed);
45+
46+
return (
47+
<Box {...props}>
48+
<Grid
49+
templateColumns={{ base: '1fr', md: '8fr 5fr', lg: '8fr 10fr', xl: '4fr 3fr' }}
50+
gap={4}
51+
pl={4}
52+
overflow="hidden"
53+
>
54+
<Box m={{ base: 0, lg: 16 }} mt={{ base: 0, lg: 0 }} textAlign={{ base: 'center', md: 'left'}}>
55+
<Heading as="h2" fontSize="6xl" fontWeight="bold" lineHeight="1.1" mt={8}>
56+
There's a place in tech for everyone.
57+
</Heading>
58+
<Text fontSize="xl" mt={8} mb={8}>{mission?.items[0]?.value}</Text>
59+
{explainer && (
60+
<VideoLink url={explainer.url} autoPlay>
61+
<Button variantColor="red">Learn More&nbsp;<Play /></Button>
62+
</VideoLink>
63+
)}
64+
</Box>
65+
66+
67+
{/* Small display: 4x images */}
68+
<Box d={{ base: 'none', md: 'block', lg: 'none'}} marginRight='-100px'>
69+
<FlexScatter
70+
seed={seed}
71+
gapMin={5}
72+
gapMax={15}
73+
yOffsetMin={-25}
74+
yOffsetMax={75}
75+
>
76+
{photoSlides(rand, splitGroups(photos, 4))}
77+
</FlexScatter>
78+
</Box>
79+
80+
{/* Large display: 6x images */}
81+
<Box d={{ base: 'none', lg: 'block'}} marginRight={{ base: '-125px', xl: '0'}}>
82+
<FlexScatter
83+
seed={seed}
84+
gapMin={25}
85+
gapMax={50}
86+
yOffsetMin={-25}
87+
yOffsetMax={75}
88+
>
89+
{photoSlides(rand, splitGroups(photos, 6))}
90+
</FlexScatter>
91+
</Box>
92+
</Grid>
93+
</Box>
94+
);
95+
}

src/components/Index/Programs.gql

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
fragment IndexProgramsComponent on Query {
2+
cms {
3+
codeDayProgram: programs(where: { name:"CodeDay" }) {
4+
items {
5+
shortDescription
6+
}
7+
}
8+
9+
mainPrograms: programs(where: { name_not:"CodeDay", type:"primary" }) {
10+
items {
11+
name
12+
url
13+
shortDescription
14+
logo {
15+
url
16+
}
17+
}
18+
}
19+
20+
otherPrograms: programs(where: { name_not:"CodeDay", type_not:"primary" }) {
21+
items {
22+
name
23+
url
24+
shortDescription
25+
logo {
26+
url
27+
}
28+
}
29+
}
30+
31+
regions (limit: 500) {
32+
items {
33+
name
34+
webname
35+
location {
36+
lat
37+
lon
38+
}
39+
}
40+
}
41+
}
42+
}

src/components/Index/Programs.js

+92
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
import React from 'react';
2+
import Box, { Grid } from '@codeday/topo/Atom/Box';
3+
import Image from '@codeday/topo/Atom/Image';
4+
import Text, { Heading } from '@codeday/topo/Atom/Text';
5+
import Content from '@codeday/topo/Molecule/Content';
6+
import { CodeDay } from '@codeday/topo/Atom/Logo';
7+
import { useQuery } from '../../query';
8+
9+
export default function Programs() {
10+
const { cms: { regions, mainPrograms, otherPrograms, codeDayProgram }} = useQuery();
11+
return (
12+
<Content>
13+
14+
<Heading as="h2" fontSize="5xl" textAlign="center" mb={16} mt={16} bold>Attend an Event</Heading>
15+
16+
{/* CodeDay */}
17+
<Grid templateColumns={{ base: '1fr', md: '1fr 1fr' }} gap={8}>
18+
<Box borderWidth={1} borderColor="current.border" borderRadius={2} p={4} boxShadow="md">
19+
<CodeDay fontSize="4xl" withText />
20+
<Text fontSize="md" mt={4} mb={4}>{codeDayProgram?.items[0]?.shortDescription}</Text>
21+
<Text fontSize="md" mb={4} bold>Choose a location:</Text>
22+
<Box borderWidth={1} border="current.border" maxHeight={{ base: "sm", md: "md" }} overflowY="auto">
23+
{regions?.items?.map((region) => (
24+
<Box
25+
p={2}
26+
as="a"
27+
d="block"
28+
href={`https://www.codeday.org/${region.webname}`}
29+
target="_blank"
30+
fontSize="xl"
31+
borderColor="current.border"
32+
borderBottomWidth="1px"
33+
>
34+
{region.name}
35+
</Box>
36+
))}
37+
</Box>
38+
</Box>
39+
40+
{/* More Programs */}
41+
<Box>
42+
{mainPrograms?.items?.map((prog) => (
43+
<Box
44+
borderWidth={1}
45+
borderColor="current.border"
46+
borderRadius={2}
47+
p={4}
48+
mb={4}
49+
boxShadow="sm"
50+
d="block"
51+
as="a"
52+
href={prog.url}
53+
target="_blank"
54+
rel="noopener"
55+
>
56+
<Box>
57+
<Box float="left" width={10} pr={4}>
58+
<Image src={prog.logo.url} height={6} alt="" />
59+
</Box>
60+
<Text fontSize="lg" bold>{prog.name}</Text>
61+
</Box>
62+
<Text clear="both" mb={0}>{prog.shortDescription}</Text>
63+
</Box>
64+
))}
65+
66+
{/* Even more programs! */}
67+
<Grid templateColumns="repeat(3, 1fr)" textAlign="center" gap={4}>
68+
{otherPrograms?.items?.map((prog) => (
69+
<Box
70+
borderWidth={1}
71+
borderColor="current.border"
72+
borderRadius={2}
73+
p={4}
74+
mb={4}
75+
boxShadow="sm"
76+
d="block"
77+
as="a"
78+
href={prog.url}
79+
target="_blank"
80+
rel="noopener"
81+
>
82+
<Image d="inline-block" src={prog.logo.url} height={12} mb={2} />
83+
<br />
84+
<Text fontSize="lg" mb={0} bold>{prog.name}</Text>
85+
</Box>
86+
))}
87+
</Grid>
88+
</Box>
89+
</Grid>
90+
</Content>
91+
)
92+
}

0 commit comments

Comments
 (0)