Mirror: The highly customizable and versatile GraphQL client with which you add on features like normalized caching as you grow.

(docs) - Update landing page (#742)

- Add shared components (panel, button, link)
- Update content/copy
- Separate Components from Features
- Update react-tile colour
- Change background colour for homepage sections
- Add <MoreOSS />
- Fix link in header

+9
packages/site/src/assets/badge_runpkg.svg
···
+
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 600 595">
+
<g fill="none" fill-rule="evenodd">
+
<path fill="#80EAC7" d="M342.277 19.4l195.971 87.165C575.801 123.268 600 160.515 600 201.616v211.961c0 41.1-24.199 78.348-61.752 95.051l-195.971 87.165c-26.914 11.97-57.64 11.97-84.554 0L61.752 508.628C24.199 491.925 0 454.678 0 413.578V201.615c0-41.1 24.199-78.348 61.752-95.05L257.723 19.4c26.914-11.97 57.64-11.97 84.554 0z" transform="translate(0 -10)"></path>
+
<path stroke="#202020" stroke-opacity=".3" stroke-width="24.247" d="M327.97 52.47c-17.86-7.951-38.259-7.938-56.109.037l-196.786 87.92c-11.832 5.286-21.557 13.803-28.324 24.238-6.767 10.435-10.576 22.788-10.576 35.747v213.356c0 13.37 3.935 26.113 10.923 36.873 6.988 10.76 17.03 19.538 29.244 24.975l195.557 87.055c17.825 7.935 38.179 7.938 56.006.008l195.732-87.07c12.219-5.436 22.265-14.214 29.256-24.975 6.991-10.762 10.928-23.508 10.928-36.881v-212.32c0-13.369-3.935-26.112-10.923-36.872-6.988-10.76-17.03-19.537-29.243-24.975L327.97 52.47z" transform="translate(0 -10)"></path>
+
<path stroke="#202020" stroke-opacity=".6" stroke-width="24.007" d="M337.399 30.368c-23.808-10.59-50.99-10.59-74.798 0l-195.97 87.165c-16.611 7.387-30.268 19.319-39.772 33.948-9.504 14.63-14.856 31.956-14.856 50.135v211.961c0 18.18 5.352 35.506 14.856 50.135 9.504 14.63 23.161 26.561 39.771 33.949l195.971 87.164c23.808 10.59 50.99 10.59 74.798 0l195.97-87.164c16.611-7.388 30.268-19.32 39.772-33.949 9.504-14.629 14.856-31.956 14.856-50.135v-211.96c0-18.18-5.352-35.507-14.856-50.136-9.504-14.63-23.161-26.56-39.771-33.948L337.399 30.368z" transform="translate(0 -10)"></path>
+
<text font-family="Helvetica" font-size="230" fill="#202020" class="abbr" x="50%" y="58%" text-anchor="middle" letter-spacing="-5.5">Rp</text>
+
<text font-size="51" font-family="Helvetica-Bold, Helvetica" fill="#202020" class="desc" x="50%" y="76%" text-anchor="middle" letter-spacing="1">RUNPKG</text>
+
</g>
+
</svg>
+15
packages/site/src/assets/badge_spectacle.svg
···
+
+
<svg xmlns="http://www.w3.org/2000/svg" width="600" height="595" viewBox="0 0 600 595">
+
<g fill="none" fill-rule="evenodd">
+
<path fill="#202020" d="M342.277 9.4l195.971 87.165C575.801 113.268 600 150.515 600 191.616v211.961c0 41.1-24.199 78.348-61.752 95.051l-195.971 87.165a104.029 104.029 0 0 1-84.554 0L61.752 498.628C24.199 481.925 0 444.678 0 403.578V191.615c0-41.1 24.199-78.348 61.752-95.05L257.723 9.4a104.029 104.029 0 0 1 84.554 0z"/>
+
<path stroke="#FC6986" stroke-width="24.247" d="M553.161 442.006a68.635 68.635 0 0 1-29.745 25.342L332.27 552.082a79.877 79.877 0 0 1-64.741 0L76.38 467.348a68.635 68.635 0 0 1-29.745-25.342 68.635 68.635 0 0 1-11.113-37.463V190.841a68.635 68.635 0 0 1 11.113-37.463 68.635 68.635 0 0 1 29.745-25.342l191.148-84.733a79.877 79.877 0 0 1 64.74 0l191.148 84.733a68.635 68.635 0 0 1 29.745 25.342 68.635 68.635 0 0 1 11.114 37.463v213.702a68.635 68.635 0 0 1-11.114 37.463z" opacity=".499"/>
+
<path stroke="#FC6986" stroke-width="24.007" d="M573.14 453.712a91.94 91.94 0 0 1-39.77 33.949l-195.971 87.164a92.026 92.026 0 0 1-74.798 0l-195.97-87.164a91.94 91.94 0 0 1-39.772-33.949 91.94 91.94 0 0 1-14.856-50.135v-211.96A91.94 91.94 0 0 1 26.86 141.48a91.94 91.94 0 0 1 39.771-33.948l195.971-87.165a92.026 92.026 0 0 1 74.798 0l195.97 87.165a91.94 91.94 0 0 1 39.772 33.948 91.94 91.94 0 0 1 14.856 50.135v211.961a91.94 91.94 0 0 1-14.856 50.135z"/>
+
<text fill="#FC6986" font-family="Helvetica-Bold, Helvetica" font-size="51" font-weight="bold" letter-spacing="2" transform="translate(0 -10)">
+
<tspan x="138.5" y="464">SPECTACLE</tspan>
+
</text>
+
<g fill="#FC6986">
+
<path d="M283.282 206.925c1.17-10.61 7.611-17.46 16.421-17.46 8.81 0 15.251 6.85 16.42 17.46L319.855 237 329 192.884c.01-15.39-12.959-27.874-28.967-27.884h-.03c-16.008-.01-28.993 12.46-29.003 27.85v.006l8.75 42.333 3.532-28.264zM275 268.855l-13.252-67.153c-5.943-14.14-22.5-20.543-37.042-14.326-14.535 6.26-21.531 22.775-15.588 36.91L269.796 313 275 268.855zM270.406 325.218v-.553l.067-.548.097-.786-70.13-94.23c-10.909-9.439-28.484-7.738-39.338 3.802-10.836 11.541-10.8 28.532.09 37.988L271 331a28.45 28.45 0 0 1-.594-5.782M438.974 232.84c-10.722-11.467-28.149-13.156-38.925-3.778l-70.305 94.885v.643A28.233 28.233 0 0 1 329 331l109.884-60.414c10.776-9.35 10.836-26.279.09-37.746M375.442 187.366c-14.393-6.178-30.824.16-36.688 14.155L325 271.304 329.915 313l60.998-89.102c5.864-13.99-1.049-30.337-15.47-36.532"/>
+
<path d="M291.703 208.715L277 326.065c0 11.562 9.85 20.935 22 20.935s22-9.373 22-20.935l-14.703-117.35c-.49-4.489-2.542-9.715-7.297-9.715s-6.808 5.226-7.297 9.715"/>
+
</g>
+
</g>
+
</svg>
+18
packages/site/src/assets/badge_victory.svg
···
+
+
<svg xmlns="http://www.w3.org/2000/svg" width="600" height="595" viewBox="0 0 600 595">
+
<g fill="none" fill-rule="evenodd">
+
<path fill="#202020" d="M342.277 9.4l195.971 87.165C575.801 113.268 600 150.515 600 191.616v211.961c0 41.1-24.199 78.348-61.752 95.051l-195.971 87.165a104.029 104.029 0 0 1-84.554 0L61.752 498.628C24.199 481.925 0 444.678 0 403.578V191.615c0-41.1 24.199-78.348 61.752-95.05L257.723 9.4a104.029 104.029 0 0 1 84.554 0z"/>
+
<path stroke="#FF684F" stroke-width="24.247" d="M553.161 442.006a68.635 68.635 0 0 1-29.745 25.342L332.27 552.082a79.877 79.877 0 0 1-64.741 0L76.38 467.348a68.635 68.635 0 0 1-29.745-25.342 68.635 68.635 0 0 1-11.113-37.463V190.841a68.635 68.635 0 0 1 11.113-37.463 68.635 68.635 0 0 1 29.745-25.342l191.148-84.733a79.877 79.877 0 0 1 64.74 0l191.148 84.733a68.635 68.635 0 0 1 29.745 25.342 68.635 68.635 0 0 1 11.114 37.463v213.702a68.635 68.635 0 0 1-11.114 37.463z" opacity=".499"/>
+
<path stroke="#FF684F" stroke-width="24.007" d="M573.14 453.712a91.94 91.94 0 0 1-39.77 33.949l-195.971 87.164a92.026 92.026 0 0 1-74.798 0l-195.97-87.164a91.94 91.94 0 0 1-39.772-33.949 91.94 91.94 0 0 1-14.856-50.135v-211.96A91.94 91.94 0 0 1 26.86 141.48a91.94 91.94 0 0 1 39.771-33.948l195.971-87.165a92.026 92.026 0 0 1 74.798 0l195.97 87.165a91.94 91.94 0 0 1 39.772 33.948 91.94 91.94 0 0 1 14.856 50.135v211.961a91.94 91.94 0 0 1-14.856 50.135z"/>
+
<text fill="#FF684F" font-family="Helvetica-Bold, Helvetica" font-size="51" font-weight="bold" letter-spacing="2" transform="translate(0 -10)">
+
<tspan x="181" y="459">VICTORY</tspan>
+
</text>
+
<path fill="#FF684F" d="M223.963 151h21.98l58.319 213H282.28z"/>
+
<path fill="#E04126" d="M362.13 233.44H340L304.263 364h21.973z"/>
+
<path fill="#FF684F" d="M401.607 169.473h-21.904l-26.201 95.357h21.814z"/>
+
<path fill="#E04126" d="M201.981 151h21.982l58.317 213a28.809 28.809 0 0 1-27.786-21.201L201.981 151z"/>
+
<path fill="#FF9877" d="M245.944 151a28.809 28.809 0 0 1 27.786 21.201L326.243 364h-21.981l-58.318-213z"/>
+
<path fill="#FF684F" d="M223.963 151l5.068 18.473h-24.595A25.079 25.079 0 0 1 180.243 151h43.72z"/>
+
<path fill="#FF684F" fill-rule="nonzero" d="M409.967 203.367c8.993-6.29 11.19-18.675 4.911-27.675l-18.836-26.996-30.789 14.49c-9.982 4.698-14.266 16.599-9.568 26.581l.01.021 33.724-15.872 20.548 29.45z"/>
+
</g>
+
</svg>
-3
packages/site/src/assets/github.svg
···
-
<svg width="1024" height="1024" viewBox="0 0 1024 1024" fill="none" xmlns="http://www.w3.org/2000/svg">
-
<path fill-rule="evenodd" clip-rule="evenodd" d="M8 0C3.58 0 0 3.58 0 8C0 11.54 2.29 14.53 5.47 15.59C5.87 15.66 6.02 15.42 6.02 15.21C6.02 15.02 6.01 14.39 6.01 13.72C4 14.09 3.48 13.23 3.32 12.78C3.23 12.55 2.84 11.84 2.5 11.65C2.22 11.5 1.82 11.13 2.49 11.12C3.12 11.11 3.57 11.7 3.72 11.94C4.44 13.15 5.59 12.81 6.05 12.6C6.12 12.08 6.33 11.73 6.56 11.53C4.78 11.33 2.92 10.64 2.92 7.58C2.92 6.71 3.23 5.99 3.74 5.43C3.66 5.23 3.38 4.41 3.82 3.31C3.82 3.31 4.49 3.1 6.02 4.13C6.66 3.95 7.34 3.86 8.02 3.86C8.7 3.86 9.38 3.95 10.02 4.13C11.55 3.09 12.22 3.31 12.22 3.31C12.66 4.41 12.38 5.23 12.3 5.43C12.81 5.99 13.12 6.7 13.12 7.58C13.12 10.65 11.25 11.33 9.47 11.53C9.76 11.78 10.01 12.26 10.01 13.01C10.01 14.08 10 14.94 10 15.21C10 15.42 10.15 15.67 10.55 15.59C13.71 14.53 16 11.53 16 8C16 3.58 12.42 0 8 0Z" transform="scale(64)" fill="#1B1F23"/>
-
</svg>
-1
packages/site/src/assets/logo_development-dashboards.svg
···
-
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="150" height="149"><image width="150" height="149" xlink:href="" fill="none" y="0"/></svg>
+4 -4
packages/site/src/assets/react-tile.svg
···
-
<svg width="492px" height="437px" viewBox="0 0 492 437" version="1.1"
-
xmlns="http://www.w3.org/2000/svg"
+
<svg width="492px" height="437px" viewBox="0 0 492 437" version="1.1"
+
xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink">
<g id="Page-1" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
-
<g id="React-icon" fill="#8196FF" fill-rule="nonzero">
+
<g id="React-icon" fill="#fff" fill-rule="nonzero">
<path d="M491.3,218.5 C491.3,186 450.6,155.2 388.2,136.1 C402.6,72.5 396.2,21.9 368,5.7 C361.5,1.9 353.9,0.1 345.6,0.1 L345.6,22.4 C350.2,22.4 353.9,23.3 357,25 C370.6,32.8 376.5,62.5 371.9,100.7 C370.8,110.1 369,120 366.8,130.1 C347.2,125.3 325.8,121.6 303.3,119.2 C289.8,100.7 275.8,83.9 261.7,69.2 C294.3,38.9 324.9,22.3 345.7,22.3 L345.7,0 C345.7,0 345.7,0 345.7,0 C318.2,0 282.2,19.6 245.8,53.6 C209.4,19.8 173.4,0.4 145.9,0.4 L145.9,22.7 C166.6,22.7 197.3,39.2 229.9,69.3 C215.9,84 201.9,100.7 188.6,119.2 C166,121.6 144.6,125.3 125,130.2 C122.7,120.2 121,110.5 119.8,101.2 C115.1,63 120.9,33.3 134.4,25.4 C137.4,23.6 141.3,22.8 145.9,22.8 L145.9,0.5 C145.9,0.5 145.9,0.5 145.9,0.5 C137.5,0.5 129.9,2.3 123.3,6.1 C95.2,22.3 88.9,72.8 103.4,136.2 C41.2,155.4 0.7,186.1 0.7,218.5 C0.7,251 41.4,281.8 103.8,300.9 C89.4,364.5 95.8,415.1 124,431.3 C130.5,435.1 138.1,436.9 146.5,436.9 C174,436.9 210,417.3 246.4,383.3 C282.8,417.1 318.8,436.5 346.3,436.5 C354.7,436.5 362.3,434.7 368.9,430.9 C397,414.7 403.3,364.2 388.8,300.8 C450.8,281.7 491.3,250.9 491.3,218.5 Z M361.1,151.8 C357.4,164.7 352.8,178 347.6,191.3 C343.5,183.3 339.2,175.3 334.5,167.3 C329.9,159.3 325,151.5 320.1,143.9 C334.3,146 348,148.6 361.1,151.8 Z M315.3,258.3 C307.5,271.8 299.5,284.6 291.2,296.5 C276.3,297.8 261.2,298.5 246,298.5 C230.9,298.5 215.8,297.8 201,296.6 C192.7,284.7 184.6,272 176.8,258.6 C169.2,245.5 162.3,232.2 156,218.8 C162.2,205.4 169.2,192 176.7,178.9 C184.5,165.4 192.5,152.6 200.8,140.7 C215.7,139.4 230.8,138.7 246,138.7 C261.1,138.7 276.2,139.4 291,140.6 C299.3,152.5 307.4,165.2 315.2,178.6 C322.8,191.7 329.7,205 336,218.4 C329.7,231.8 322.8,245.2 315.3,258.3 Z M347.6,245.3 C353,258.7 357.6,272.1 361.4,285.1 C348.3,288.3 334.5,291 320.2,293.1 C325.1,285.4 330,277.5 334.6,269.4 C339.2,261.4 343.5,253.3 347.6,245.3 Z M246.2,352 C236.9,342.4 227.6,331.7 218.4,320 C227.4,320.4 236.6,320.7 245.9,320.7 C255.3,320.7 264.6,320.5 273.7,320 C264.7,331.7 255.4,342.4 246.2,352 Z M171.8,293.1 C157.6,291 143.9,288.4 130.8,285.2 C134.5,272.3 139.1,259 144.3,245.7 C148.4,253.7 152.7,261.7 157.4,269.7 C162.1,277.7 166.9,285.5 171.8,293.1 Z M245.7,85 C255,94.6 264.3,105.3 273.5,117 C264.5,116.6 255.3,116.3 246,116.3 C236.6,116.3 227.3,116.5 218.2,117 C227.2,105.3 236.5,94.6 245.7,85 Z M171.7,143.9 C166.8,151.6 161.9,159.5 157.3,167.6 C152.7,175.6 148.4,183.6 144.3,191.6 C138.9,178.2 134.3,164.8 130.5,151.8 C143.6,148.7 157.4,146 171.7,143.9 Z M81.2,269.1 C45.8,254 22.9,234.2 22.9,218.5 C22.9,202.8 45.8,182.9 81.2,167.9 C89.8,164.2 99.2,160.9 108.9,157.8 C114.6,177.4 122.1,197.8 131.4,218.7 C122.2,239.5 114.8,259.8 109.2,279.3 C99.3,276.2 89.9,272.8 81.2,269.1 Z M135,412 C121.4,404.2 115.5,374.5 120.1,336.3 C121.2,326.9 123,317 125.2,306.9 C144.8,311.7 166.2,315.4 188.7,317.8 C202.2,336.3 216.2,353.1 230.3,367.8 C197.7,398.1 167.1,414.7 146.3,414.7 C141.8,414.6 138,413.7 135,412 Z M372.2,335.8 C376.9,374 371.1,403.7 357.6,411.6 C354.6,413.4 350.7,414.2 346.1,414.2 C325.4,414.2 294.7,397.7 262.1,367.6 C276.1,352.9 290.1,336.2 303.4,317.7 C326,315.3 347.4,311.6 367,306.7 C369.3,316.8 371.1,326.5 372.2,335.8 Z M410.7,269.1 C402.1,272.8 392.7,276.1 383,279.2 C377.3,259.6 369.8,239.2 360.5,218.3 C369.7,197.5 377.1,177.2 382.7,157.7 C392.6,160.8 402,164.2 410.8,167.9 C446.2,183 469.1,202.8 469.1,218.5 C469,234.2 446.1,254.1 410.7,269.1 Z" id="Shape"></path>
<circle id="Oval" cx="245.9" cy="218.5" r="45.7"></circle>
</g>
</g>
-
</svg>
+
</svg>
+1 -1
packages/site/src/components/body-copy.js
···
export const BodyCopy = styled.p`
font-size: 1.4rem;
line-height: 2.2rem;
-
margin: 0 auto;
width: 100%;
text-align: center;
+
${p => p.noMargin && 'margin: 0'};
@media (min-width: 768px) {
font-size: 1.5rem;
line-height: 2.4rem;
+24 -14
packages/site/src/components/button.js
···
-
import styled from 'styled-components';
-
import { Link } from 'react-router-dom';
+
import React from 'react';
+
import styled, { css } from 'styled-components';
-
export const Button = styled(Link)`
-
background: ${props => (props.light ? 'white' : '#202020')};
-
color: ${props => (props.light ? '#202020' : 'white')};
-
display: block;
-
font-size: 1.5rem;
+
export const buttonLinkStyling = css`
+
background: white;
+
color: #383838;
+
font-weight: normal;
+
font-size: 1.4rem;
+
font-style: normal;
+
font-stretch: normal;
height: 4rem;
-
letter-spacing: 0.05em;
line-height: 4rem;
-
margin: ${props => (props.noMargin ? '0' : '5rem auto 3rem')};
-
max-width: 21rem;
-
min-width: 10rem;
+
padding: 0 2rem;
+
letter-spacing: 0.01rem;
text-align: center;
text-transform: uppercase;
-
transition: background 0.4s;
-
width: 100%;
+
transition: opacity 0.4s ease-out;
+
&:hover {
-
background: ${props => (props.light ? '#f6f6f6' : '#333')};
+
opacity: 0.8;
}
&:active {
opacity: 0.6;
}
`;
+
+
const ButtonNoBorder = styled.button`
+
border: none;
+
`;
+
+
export const Button = styled(props => (
+
<ButtonNoBorder {...props}>{props.children}</ButtonNoBorder>
+
))`
+
${buttonLinkStyling}
+
`;
+11 -3
packages/site/src/components/header.js
···
import React from 'react';
-
import Hero from '../screens/home/hero';
import styled from 'styled-components';
+
import PropTypes from 'prop-types';
+
import Hero from '../screens/home/hero';
import logoFormidableWhite from '../assets/logo_formidable_white.svg';
import LeftTriangles from '../assets/left-triangles.svg';
import RightTriangles from '../assets/right-triangles.svg';
···
flex-direction: column;
color: #ffffff;
text-decoration: none;
+
z-index: 2;
`;
const HeaderText = styled.p`
···
}
`;
-
export const Header = () => (
+
export const Header = ({ content }) => (
<Container>
<LeftTrianglesImg src={LeftTriangles} />
<RightTrianglesImg src={RightTriangles} />
···
<HeaderText>Another oss project by </HeaderText>
<HeaderLogo src={logoFormidableWhite} alt="Formidable Logo" />
</HeaderContainer>
-
<Hero />
+
<Hero content={content.hero} />
</Container>
);
+
+
Header.propTypes = {
+
content: PropTypes.shape({
+
hero: PropTypes.shape({ copyText: PropTypes.string }),
+
}).isRequired,
+
};
+17
packages/site/src/components/link.js
···
+
import React from 'react';
+
import styled from 'styled-components';
+
import { Link as ReactRouterLink } from 'react-router-dom';
+
+
import { buttonLinkStyling } from './button';
+
+
export const Link = styled(({ isExternal, ...rest }) =>
+
isExternal ? (
+
<a href={rest.to} {...rest}>
+
{rest.children}
+
</a>
+
) : (
+
<ReactRouterLink {...rest} />
+
)
+
)`
+
${buttonLinkStyling}
+
`;
+44
packages/site/src/components/panel.js
···
+
import React from 'react';
+
import styled, { css } from 'styled-components';
+
import constants from '../constants';
+
+
const dark = css`
+
background-color: #0d1129;
+
`;
+
+
const light = css`
+
background: ${constants.color};
+
border-bottom: 1rem solid rgba(0, 0, 0, 0.4);
+
box-shadow: inset 0 -1rem 0 rgba(0, 0, 0, 0.2);
+
`;
+
export const FullWidthContainer = styled.div`
+
color: #e3eef8;
+
display: flex;
+
justify-content: center;
+
${p => (!!p.isLight ? light : dark)};
+
${p => p.background && `background: ${p.background}`}
+
`;
+
+
export const SectionWrapper = styled.div`
+
flex-direction: column;
+
align-items: center;
+
display: flex;
+
padding: 8rem 4rem;
+
width: 100%;
+
@media (min-width: 768px) {
+
flex-direction: column;
+
margin: 0 8rem;
+
padding: 8rem 8rem;
+
}
+
`;
+
+
export const PanelSectionWrapper = ({
+
children,
+
isLight,
+
background,
+
...rest
+
}) => (
+
<FullWidthContainer isLight={!!isLight} background={background} {...rest}>
+
<SectionWrapper>{children}</SectionWrapper>
+
</FullWidthContainer>
+
);
+2 -1
packages/site/src/components/secondary-title.js
···
import styled from 'styled-components';
export const SecondaryTitle = styled.h3`
+
color: white;
font-size: 2rem;
line-height: 2.4rem;
margin: 2rem auto 1rem;
-
color: ${p => (p.pop ? 'white' : 'inherit')};
+
text-align: center;
@media (min-width: 768px) {
font-size: 2.2rem;
line-height: 2.6rem;
+4 -4
packages/site/src/components/section-title.js
···
export const SectionTitle = styled.h2`
color: #fff;
-
font-size: 4.5rem;
+
font-size: 3.5rem;
flex: auto;
line-height: 1.3;
-
margin: 2rem 0 3rem;
+
margin: 0 0 3rem;
width: 100%;
text-align: center;
-
margin: 4rem 0;
@media (min-width: 768px) {
-
margin: 2rem 0 6rem;
+
font-size: 4.5rem;
+
margin: 0 0 6rem;
}
`;
+1 -1
packages/site/src/constants.js
···
githubIssues: 'https://www.github.com/FormidableLabs/urql/issues',
github: 'https://www.github.com/FormidableLabs/urql',
readme: 'https://github.com/FormidableLabs/urql/blob/master/README.md',
-
color: '#8196FF',
+
color: '#6B78B8',
googleAnalyticsId: 'UA-43290258-1',
};
+27 -33
packages/site/src/screens/home/_content.js
···
const content = {
+
header: {
+
hero: {
+
copyText: 'npm install urql graphql',
+
},
+
},
features: [
{
-
title: 'Fully functional GraphQL client with a single import',
+
title: 'Performant and functional',
description:
-
'Urql is a lightweight, powerful, and easy to use alternative to bulky GraphQL clients.',
+
'Lightweight, powerful, and easy to use; urql is a great alternative to bulky GraphQL clients.',
icon: require('../../assets/gql-tile.svg'),
},
{
-
title: 'Logical default behavior and caching',
+
title: 'Extensible library that grows with you',
description:
-
'Urql helps you to rapidly use GraphQL in your apps without complex configuration or large API overhead.',
-
icon: require('../../assets/clock-tile.svg'),
+
'Want to change how you fetch, cache, or subscribe to data? The urql exchanges allow you to customize your data layer to suit your needs.',
+
icon: require('../../assets/eagle-tile.svg'),
},
{
-
title: 'Extensible library that grows with you',
+
title: 'Logical default behavior and caching',
description:
-
'Want to change how you fetch, cache, or subscribe to data? Urql Exchanges allow you to customize your data layer to suit your needs.',
-
icon: require('../../assets/eagle-tile.svg'),
+
'Adding urql enables you to rapidly use GraphQL in your apps without complex configuration or large API overhead.',
+
icon: require('../../assets/clock-tile.svg'),
},
],
components: {
···
media: '',
},
getStarted: {
-
description:
-
'Dive into the documentation to see how you can get your urql client up and running.',
+
description: `With it's intiuitive set of lightweight API's, getting started with urql is a breeze. Dive into the documentation to get up and running in minutes.`,
link: '/docs',
},
-
// TODO: This needs to be shared content instead of hardcoded in each site
-
/*
oss: [
{
title: 'Victory',
description:
'An ecosystem of modular data visualization components for React. Friendly and flexible.',
-
logo: require('../../assets/logo_victory.svg'),
+
logo: require('../../assets/badge_victory.svg'),
link: 'https://formidable.com/open-source/victory',
-
hasOwnLogo: true,
},
{
-
title: 'Development Dashboards',
+
title: 'urql',
description:
-
'Dashboards to organize and intuitively display your dev server and tooling output.',
-
abbreviation: 'Dd',
-
color: '#8bd48b',
-
number: '17',
-
link: 'https://formidable.com/open-source/development-dashboards/',
+
'Universal React Query Library is a blazing-fast GraphQL client, exposed as a set of ReactJS components.',
+
logo: require('../../assets/sidebar-badge.svg'),
+
link: 'https://formidable.com/open-source/urql/',
},
{
-
title: 'React Animations',
+
title: 'Spectacle',
description:
-
'A collection of animations that can be used with many inline style libraries, such as Radium or Aphrodite.',
-
abbreviation: 'Ra',
-
color: '#86b9e6',
-
number: '03',
-
link: 'https://formidable.com/open-source/react-animations',
+
'A React.js based library for creating sleek presentations using JSX syntax that gives you the ability to live demo your code.',
+
logo: require('../../assets/badge_spectacle.svg'),
+
link: 'https://formidable.com/open-source/spectacle/',
},
{
-
title: 'Enzyme Matchers',
+
title: 'Runpkg',
description:
-
'Run common assertions on your React components using Enzyme in a Jest or Jasmine environment.',
-
abbreviation: 'Em',
-
color: '#e48055',
-
number: '09',
-
link: 'https://formidable.com/open-source/jest-enzyme/',
+
'The online package explorer. Runpkg turns any npm package into an interactive and informative browsing experience',
+
logo: require('../../assets/badge_runpkg.svg'),
+
link: 'https://www.runpkg.com/',
},
],
-
*/
};
export default content;
+59
packages/site/src/screens/home/components.js
···
+
import React from 'react';
+
import PropTypes from 'prop-types';
+
import styled from 'styled-components';
+
+
import { SecondaryTitle } from '../../components/secondary-title';
+
import { BodyCopy } from '../../components/body-copy';
+
import { PanelSectionWrapper } from '../../components/panel';
+
+
const ComponentWrapper = styled.div`
+
margin: 0 0 0;
+
display: flex;
+
flex-direction: column;
+
align-items: center;
+
text-align: center;
+
> img {
+
width: 100%;
+
max-width: 10rem;
+
margin-bottom: 20px;
+
}
+
`;
+
+
const SecondaryTitleCentred = styled(SecondaryTitle)`
+
@media (min-width: 768px) {
+
text-align: center;
+
}
+
`;
+
+
const BodyCopyCentred = styled(BodyCopy)`
+
margin-top: 2rem;
+
max-width: 28rem;
+
+
@media (min-width: 768px) {
+
text-align: center;
+
}
+
+
@media (min-width: 1024px) {
+
max-width: 20vw;
+
}
+
`;
+
+
const Components = props => {
+
return (
+
<PanelSectionWrapper isLight>
+
<ComponentWrapper>
+
<img src={props.components.icon} />
+
<SecondaryTitleCentred>{props.components.title}</SecondaryTitleCentred>
+
<BodyCopyCentred>{props.components.description}</BodyCopyCentred>
+
</ComponentWrapper>
+
</PanelSectionWrapper>
+
);
+
};
+
+
Components.displayName = 'Components';
+
+
Components.propTypes = {
+
components: PropTypes.object,
+
};
+
+
export default Components;
+17 -83
packages/site/src/screens/home/features.js
···
import { BodyCopy } from '../../components/body-copy';
import { SecondaryTitle } from '../../components/secondary-title';
import { SectionTitle } from '../../components/section-title';
-
-
const FullWidthContainer = styled.div`
-
display: flex;
-
justify-content: center;
-
background-color: #0d1129;
-
`;
-
-
const FeaturesWrapper = styled.div`
-
flex-direction: column;
-
align-items: center;
-
display: flex;
-
background-color: #0d1129;
-
color: #a3abd4;
-
padding: 8rem 8rem;
-
width: 100%;
-
@media (min-width: 768px) {
-
flex-direction: column;
-
margin: 0 8rem;
-
}
-
`;
+
import { PanelSectionWrapper } from '../../components/panel';
const FeatureWrapper = styled.div`
display: flex;
···
display: flex;
flex-direction: column;
align-items: center;
-
margin: 0 0 4rem;
width: 100%;
max-width: 28rem;
text-align: center;
···
width: 100%;
max-width: 28rem;
box-shadow: -20px 20px 0 0 rgba(0, 0, 0, 0.5);
-
margin-bottom: 20px;
+
}
+
&:not(:last-child) {
+
margin: 0 0 4rem;
}
@media (min-width: 768px) {
-
margin: 0 0 6rem;
+
margin: 0;
width: calc(1 / 3 * 100% - (1 - 1 / 3) * 40px);
align-items: flex-start;
text-align: left;
···
}
`;
-
const ComponentWrapper = styled.div`
-
margin: 0 0 0;
-
display: flex;
-
flex-direction: column;
-
align-items: center;
-
max-width: 28rem;
-
text-align: center;
-
> img {
-
width: 100%;
-
max-width: 10rem;
-
margin-bottom: 20px;
-
}
-
@media (min-width: 768px) {
-
max-width: 116rem;
-
padding: 6rem 8rem 0 8rem;
-
}
-
`;
-
-
const BodyCopyCentred = styled(BodyCopy)`
-
max-width: 28rem;
-
-
@media (min-width: 768px) {
-
text-align: center;
-
}
-
-
@media (min-width: 1024px) {
-
max-width: 20vw;
-
}
-
`;
-
-
const SecondaryTitleCentred = styled(SecondaryTitle)`
-
@media (min-width: 768px) {
-
text-align: center;
-
}
-
`;
-
const SectionTitleStyled = styled(SectionTitle)`
margin-top: 0;
margin-bottom: 4rem;
···
class Features extends React.Component {
render() {
return (
-
<FullWidthContainer>
-
<FeaturesWrapper>
-
<SectionTitleStyled>Features</SectionTitleStyled>
-
<FeatureWrapper>
-
{this.props.featureArray.map(feature => (
-
<FeatureCard key={feature.title}>
-
<img src={feature.icon} />
-
<SecondaryTitleStyled pop>{feature.title}</SecondaryTitleStyled>
-
<BodyCopy>{feature.description}</BodyCopy>
-
</FeatureCard>
-
))}
-
</FeatureWrapper>
-
<ComponentWrapper>
-
<img src={this.props.components.icon} />
-
<SecondaryTitleCentred pop>
-
{this.props.components.title}
-
</SecondaryTitleCentred>
-
<BodyCopyCentred>
-
{this.props.components.description}
-
</BodyCopyCentred>
-
</ComponentWrapper>
-
</FeaturesWrapper>
-
</FullWidthContainer>
+
<PanelSectionWrapper>
+
<SectionTitleStyled>Features</SectionTitleStyled>
+
<FeatureWrapper>
+
{this.props.featureArray.map(feature => (
+
<FeatureCard key={feature.title}>
+
<img src={feature.icon} />
+
<SecondaryTitleStyled>{feature.title}</SecondaryTitleStyled>
+
<BodyCopy>{feature.description}</BodyCopy>
+
</FeatureCard>
+
))}
+
</FeatureWrapper>
+
</PanelSectionWrapper>
);
}
}
Features.propTypes = {
-
components: PropTypes.object,
featureArray: PropTypes.array,
};
+13 -51
packages/site/src/screens/home/get-started.js
···
import React from 'react';
import PropTypes from 'prop-types';
import styled from 'styled-components';
-
import { Link } from 'react-router-dom';
import { BodyCopy } from '../../components/body-copy';
+
import { Link } from '../../components/link';
import { SectionTitle } from '../../components/section-title';
-
import constants from '../../constants';
-
-
import octoCat from '../../assets/github.svg';
-
-
const OuterWrapper = styled.div`
-
background: ${constants.color};
-
border-bottom: 1rem solid rgba(0, 0, 0, 0.4);
-
box-shadow: inset 0 -1rem 0 rgba(0, 0, 0, 0.2);
-
display: flex;
-
justify-content: center;
-
`;
+
import { PanelSectionWrapper } from '../../components/panel';
const GetStartedWrapper = styled.div`
+
align-items: center;
+
display: flex;
+
flex-direction: column;
+
max-width: 55rem;
p {
text-align: center;
}
h2 {
margin-top: 0;
}
-
display: flex;
-
flex-direction: column;
-
align-items: center;
-
margin: 8rem;
`;
const GithubButton = styled.a`
···
const ButtonsWrapper = styled.div`
display: flex;
align-items: center;
-
margin-top: 4rem;
+
margin-top: 6rem;
flex-direction: column;
@media (min-width: 768px) {
flex-direction: row;
-
margin-top: 6rem;
-
}
-
`;
-
-
const DocButton = styled(Link)`
-
width: 18rem;
-
margin-left: 0rem;
-
height: 4rem;
-
font-size: 14px;
-
background-color: #ffffff;
-
line-height: 4rem;
-
text-align: center;
-
text-transform: uppercase;
-
letter-spacing: 1px;
-
color: #383838;
-
border: 0;
-
margin-top: 2rem;
-
@media (min-width: 768px) {
-
margin-left: 2rem;
-
width: 18rem;
-
margin-top: 0;
-
}
-
&:hover {
-
background-color: #f6f6f6;
}
`;
class GetStarted extends React.Component {
render() {
-
const { getStartedObj } = this.props;
+
const { content } = this.props;
return (
-
<OuterWrapper>
+
<PanelSectionWrapper>
<GetStartedWrapper>
<GetStartedTitle>Get Started</GetStartedTitle>
-
<BodyCopy>{getStartedObj.description}</BodyCopy>
+
<BodyCopy noMargin>{content.description}</BodyCopy>
<ButtonsWrapper>
-
<GithubButton href="https://github.com/FormidableLabs/urql">
-
<img src={octoCat} />
-
<p>GitHub</p>
-
</GithubButton>
-
<DocButton to="docs/">Documentation</DocButton>
+
<Link to="docs/">Quick Start Guide</Link>
</ButtonsWrapper>
</GetStartedWrapper>
-
</OuterWrapper>
+
</PanelSectionWrapper>
);
}
}
GetStarted.propTypes = {
-
getStartedObj: PropTypes.object,
+
content: PropTypes.object,
};
export default GetStarted;
+64 -90
packages/site/src/screens/home/hero.js
···
-
import React from 'react';
-
import { Link } from 'react-router-dom';
+
import React, { useState, useCallback } from 'react';
+
import { Link as ReactRouterLink } from 'react-router-dom';
+
import { Wrapper } from '../../components/wrapper';
+
import { Button } from '../../components/button';
+
import { Link } from '../../components/link';
import styled from 'styled-components';
import badge from '../../assets/sidebar-badge.svg';
···
font-size: 14px;
margin: 0;
`;
-
const HeroNPMButton = styled.button`
+
const HeroNPMButton = styled(Button)`
width: 8rem;
-
height: 4rem;
-
background-color: #ffffff;
-
font-size: 14px;
-
font-weight: normal;
-
font-style: normal;
-
font-stretch: normal;
-
line-height: normal;
-
color: #383838;
-
border: 0;
-
text-transform: uppercase;
cursor: copy;
-
&:hover {
-
background-color: #f6f6f6;
-
}
+
text-decoration: none;
`;
export const HeroDocsButton = styled(Link)`
width: 30rem;
-
margin-left: 0rem;
-
height: 4rem;
-
font-size: 14px;
-
background-color: #ffffff;
-
line-height: 4rem;
-
text-align: center;
-
text-transform: uppercase;
-
color: #383838;
-
border: 0;
-
margin-top: 5rem;
+
margin-top: 4rem;
@media (min-width: 768px) {
margin-top: 2rem;
width: 30rem;
···
margin-top: 0;
margin-left: 2rem;
width: 18rem;
-
}
-
&:hover {
-
background-color: #f6f6f6;
}
`;
···
}
`;
-
const copyFallBack = () => {
+
const copyFallBack = copyText => {
const copyTextArea = document.createElement('textArea');
-
copyTextArea.value = 'npm install urql';
+
copyTextArea.value = copyText;
document.body.appendChild(copyTextArea);
···
copyTextArea.remove();
};
-
class Hero extends React.Component {
-
constructor() {
-
super(...arguments);
+
const Hero = props => {
+
const [animating, setAnimating] = useState(false);
+
const [copied, setCopied] = useState(false);
-
this.state = {
-
animating: false,
-
copied: false,
-
};
-
}
-
-
handleCopy(e) {
-
if (!navigator.clipboard) {
-
copyFallBack();
-
e.preventDefault();
-
return;
-
}
-
navigator.clipboard.writeText('npm install urql');
-
}
+
const handleCopy = useCallback(
+
e => {
+
if (!navigator.clipboard) {
+
copyFallBack(props.content.copyText);
+
e.preventDefault();
+
return;
+
}
+
navigator.clipboard.writeText(props.content.copyText);
+
},
+
[props.content.copyText]
+
);
-
render() {
-
return (
-
<WrapperStyled noPadding>
-
<HeroContent>
-
<HeroLogoContainer>
-
<HeroLogo src={badge} />
-
</HeroLogoContainer>
-
<HeroTitle>urql</HeroTitle>
-
<HeroBody>
-
Universal React Query Library is a blazing-fast GraphQL client,
-
exposed as a set of ReactJS components.
-
</HeroBody>
-
<HeroButtonsWrapper>
-
<HeroNPMWrapper>
-
<HeroNPMCopy>npm install urql</HeroNPMCopy>
-
<HeroNPMButton onClick={this.handleCopy}>copy</HeroNPMButton>
-
</HeroNPMWrapper>
-
<HeroDocsButton to="docs/">Documentation</HeroDocsButton>
-
</HeroButtonsWrapper>
-
</HeroContent>
-
<HeroNavList>
-
<li>
-
<Link to="docs/">Docs</Link>
-
</li>
-
<li>
-
<a
-
title="Issues"
-
href="https://www.github.com/FormidableLabs/urql/issues"
-
>
-
Issues
-
</a>
-
</li>
-
<li>
-
<a title="GitHub" href="https://github.com/FormidableLabs/urql">
-
GitHub
-
</a>
-
</li>
-
</HeroNavList>
-
</WrapperStyled>
-
);
-
}
-
}
+
return (
+
<WrapperStyled noPadding>
+
<HeroContent>
+
<HeroLogoContainer>
+
<HeroLogo src={badge} />
+
</HeroLogoContainer>
+
<HeroTitle>urql</HeroTitle>
+
<HeroBody>
+
urql is a blazing-fast GraphQL client that supports React, Preact, and
+
Svelte (alpha).
+
</HeroBody>
+
<HeroButtonsWrapper>
+
<HeroNPMWrapper>
+
<HeroNPMCopy>{props.content.copyText}</HeroNPMCopy>
+
<HeroNPMButton onClick={handleCopy}>copy</HeroNPMButton>
+
</HeroNPMWrapper>
+
<HeroDocsButton to="docs/">Documentation</HeroDocsButton>
+
</HeroButtonsWrapper>
+
</HeroContent>
+
<HeroNavList>
+
<li>
+
<ReactRouterLink to="docs/">Docs</ReactRouterLink>
+
</li>
+
<li>
+
<a
+
title="Issues"
+
href="https://www.github.com/FormidableLabs/urql/issues"
+
>
+
Issues
+
</a>
+
</li>
+
<li>
+
<a title="GitHub" href="https://github.com/FormidableLabs/urql">
+
GitHub
+
</a>
+
</li>
+
</HeroNavList>
+
</WrapperStyled>
+
);
+
};
export default Hero;
+6 -7
packages/site/src/screens/home/index.js
···
import { useMarkdownTree } from 'react-static-plugin-md-pages';
import Features from './features';
+
import Components from './components';
import GetStarted from './get-started';
import MoreOSS from './more-oss';
import content from './_content';
···
return (
<Container ref={ref}>
-
<Header />
-
<Features
-
featureArray={content.features}
-
components={content.components}
-
/>
-
<GetStarted getStartedObj={content.getStarted} />
-
<MoreOSS ossArray={content.oss} />
+
<Header content={content.header} />
+
<Features featureArray={content.features} />
+
<Components components={content.components} />
+
<GetStarted content={content.getStarted} />
+
<MoreOSS oss={content.oss} />
<Footer />
</Container>
);
+87 -101
packages/site/src/screens/home/more-oss.js
···
import React from 'react';
import PropTypes from 'prop-types';
-
// import { ProjectBadge } from "formidable-oss-badges";
import styled from 'styled-components';
+
import { BodyCopy } from '../../components/body-copy';
-
import { Button } from '../../components/button';
-
import { SecondaryTitle } from '../../components/secondary-title';
+
import { Link } from '../../components/link';
+
import { PanelSectionWrapper } from '../../components/panel';
import { SectionTitle } from '../../components/section-title';
+
import { SecondaryTitle } from '../../components/secondary-title';
import { Wrapper } from '../../components/wrapper';
-
const OuterWrapper = styled.div`
-
background-color: #000000;
-
background-size: 100% 100%;
-
color: white;
-
width: 100%;
-
display: flex;
-
justify-content: center;
+
const OSSCardContainer = styled.div`
+
display: grid;
+
grid-template-columns: 1fr;
+
grid-template-rows: repeat(4, 1fr);
+
grid-gap: 4rem;
+
width: calc(100% - 4rem);
+
max-width: 75%;
+
margin: auto auto 4rem auto;
+
+
@media (min-width: 768px) {
+
grid-template-columns: repeat(2, 1fr);
+
grid-template-rows: repeat(2, 1fr);
+
max-width: 116rem;
+
}
`;
const OSSCard = styled.div`
-
margin: 0 auto 4rem;
-
max-width: 43rem;
-
position: relative;
text-align: left;
-
width: 100%;
display: flex;
flex-direction: column;
align-items: center;
+
+
> * {
+
margin-top: 0;
+
margin-bottom: 0;
+
}
+
+
> * + * {
+
margin-top: 1rem;
+
}
+
@media (min-width: 768px) {
-
width: calc(1 / 2 * 100% - (1 - 1 / 2) * 80px);
flex-direction: row;
justify-content: space-between;
-
margin-bottom: 6rem;
-
}
-
`;
-
const OSSImage = styled.img`
-
left: 0;
-
position: relative;
-
top: 2rem;
-
width: 18rem;
-
max-width: none;
-
padding: 13% 12%;
-
@media (min-width: 1024px) {
-
padding: 14px;
-
width: 14rem;
-
}
-
`;
+
> * {
+
margin-left: 0;
+
margin-right: 0;
+
}
-
const OSSLink = styled.a`
-
& h3 {
-
color: white;
-
}
-
& h3:hover {
-
opacity: 0.7;
+
> * + * {
+
margin-top: 0;
+
margin-left: 3rem;
+
}
}
`;
-
/*
-
const StyledProjectBadge = styled(ProjectBadge)`
-
left: 0;
-
position: relative;
-
top: 2rem;
-
width: 18rem;
-
@media (min-width: 1024px) {
-
width: 14rem;
-
}
+
const OSSImage = styled.img`
+
flex: 0 0 15rem;
+
height: 15rem;
`;
-
*/
const OSSCopyContainer = styled.div`
display: flex;
flex-direction: column;
-
@media (min-width: 768px) {
-
padding-left: 1rem;
+
+
> * {
+
margin-top: 0;
+
margin-bottom: 0;
}
-
@media (min-width: 1024px) {
-
padding-left: 2rem;
+
> * + * {
+
margin-top: 2rem;
}
`;
-
const SecondaryTitleStyled = styled(SecondaryTitle)`
-
text-align: center;
+
const OSSTitle = styled(SecondaryTitle)`
+
transition: opacity 0.3s ease-out;
+
margin: 0;
+
+
&:hover {
+
opacity: 0.7;
+
}
+
@media (min-width: 768px) {
text-align: left;
}
`;
-
const SectionWrapper = styled(Wrapper)`
-
padding: 8rem 8rem;
-
h2 {
-
margin-top: 0rem;
-
}
-
@media (max-width: 768px) {
-
padding: 8rem 8rem;
-
p {
-
text-align: center;
-
}
+
const OSSDescription = styled(BodyCopy)`
+
@media (min-width: 768px) {
+
text-align: left;
}
`;
-
const ButtonStyled = styled(Button)`
-
margin-bottom: 0;
-
margin-top: 0;
-
`;
-
-
class MoreOSS extends React.Component {
-
render() {
-
return null; // TODO: formidable-oss-badges incorrectly bundles styled-components due to a broken rollup config
-
-
/*
-
return (
-
<OuterWrapper>
-
<SectionWrapper>
-
<SectionTitle>More Open Source from Formidable</SectionTitle>
-
{this.props.ossArray.map(card => (
-
<OSSCard key={card.title}>
-
<OSSLink href={card.link}>
-
{card.hasOwnLogo ? (
-
<OSSImage src={card.logo} />
-
) : null}
-
</OSSLink>
-
<OSSCopyContainer>
-
<OSSLink href={card.link}>
-
<SecondaryTitleStyled>{card.title}</SecondaryTitleStyled>
-
</OSSLink>
-
<BodyCopy>{card.description}</BodyCopy>
-
</OSSCopyContainer>
-
</OSSCard>
-
))}
-
<ButtonStyled light="true" to="https://formidable.com/open-source/">
-
View All
-
</ButtonStyled>
-
</SectionWrapper>
-
</OuterWrapper>
-
);
-
*/
-
}
-
}
+
const MoreOSS = ({ oss }) => (
+
<PanelSectionWrapper background="#000000">
+
<SectionTitle>More Open Source from Formidable</SectionTitle>
+
<OSSCardContainer>
+
{oss.map(card => {
+
return (
+
<OSSCard key={card.title}>
+
<OSSImage src={card.logo} />
+
<OSSCopyContainer>
+
<a href={card.link} target="_blank" rel="noopener noreferrer">
+
<OSSTitle>{card.title}</OSSTitle>
+
</a>
+
<OSSDescription>{card.description}</OSSDescription>
+
</OSSCopyContainer>
+
</OSSCard>
+
);
+
})}
+
</OSSCardContainer>
+
<Link isExternal to="https://formidable.com/open-source/">
+
View All
+
</Link>
+
</PanelSectionWrapper>
+
);
MoreOSS.propTypes = {
-
ossArray: PropTypes.array,
+
oss: PropTypes.arrayOf(
+
PropTypes.shape({
+
title: PropTypes.string.isRequired,
+
link: PropTypes.string.isRequired,
+
description: PropTypes.string.isRequired,
+
logo: PropTypes.string.isRequired,
+
}).isRequired
+
).isRequired,
};
export default MoreOSS;