-
Notifications
You must be signed in to change notification settings - Fork 10.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
15 changed files
with
314 additions
and
15 deletions.
There are no files selected for viewing
This file contains 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
Binary file added
BIN
+555 KB
...-used-gatsby-to-be-selected-as-an-amazon-hq2-candidate-city/amazon-boston-2.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added
BIN
+305 KB
...-used-gatsby-to-be-selected-as-an-amazon-hq2-candidate-city/amazon-boston-3.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added
BIN
+391 KB
...ov-used-gatsby-to-be-selected-as-an-amazon-hq2-candidate-city/amazon-boston.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
46 changes: 46 additions & 0 deletions
46
...-boston-gov-used-gatsby-to-be-selected-as-an-amazon-hq2-candidate-city/index.md
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
--- | ||
title: How Boston.gov used Gatsby to be selected as an Amazon HQ2 candidate city | ||
date: "2018-01-18" | ||
author: "Sam Bhagwat" | ||
--- | ||
|
||
[Boston.gov](http://boston.gov) is a small team with a huge responsibility: ten people tasked with the digital presence of New England’s most iconic city. | ||
|
||
Today, the team and the city scored a huge victory when Amazon announced that Boston was [selected as one among 20 candidate cities](https://www.amazon.com/b?node=17044620011) for its North America HQ2. | ||
|
||
![Amazon Boston homepage](./amazon-boston.jpg "Boston city") | ||
|
||
When Amazon announced in September that it was looking to build a new headquarters, bringing 50,000 jobs and billions of investment to the chosen city, Boston’s city government jumped to throw their hat in the ring. | ||
|
||
As a technology hub, the city wanted to put their best digital foot forward, so they turned to the Boston.gov team to build a website as the city’s application — [amazon.boston.gov](http://amazon.boston.gov) . | ||
|
||
With a compressed application process, the Boston.gov team had a week and a half to build two websites -- a public-facing site, as well as a password-protected site with additional information. | ||
|
||
They needed tools that let them get off the ground immediately, iterate quickly, and build something beautiful -- so they turned to Gatsby. | ||
|
||
![Amazon Boston homepage](./amazon-boston-3.jpg "Mayor Martin J. Walsh") | ||
|
||
"Within 10 minutes, we had our environment set up, with the website building and deploying,” said Matthew Crist, lead engineer for the project. | ||
|
||
With Gatsby, the team was able to complete the website from basic comps in around a week, compared to about a month for similar projects in the past. “We generally haven’t been able to do anything as fast,” Crist said. | ||
|
||
As an engineer, Crist really appreciated Gatsby’s out-of-the-box hot reloading support. | ||
“The fact that you can save your code, tab back to your browser, and see the change sped up development so much,” Crist said. | ||
|
||
And the whole team appreciated Gatsby’s support for component libraries. | ||
|
||
“We were able to use our pattern library that we’ve built for Boston.gov,” said Crist. “Our city’s website is nothing but components, stacked together to create a page. [amazon.boston.gov](http://amazon.boston.gov) is the same.” | ||
|
||
“Our pattern library has already been A/B tested, and stakeholders are already bought in,” said Crystal Torman, an economic development advisor for Boston, and the project’s internal client. “We could make just one decision about website styling and then spend more time drilling into the content.” | ||
|
||
"Crystal and I were probably on the phone straight for two days, giving feedback,” said Reilly Zlab, the product manager. | ||
|
||
“Normally we don’t do so many iterations in such a short time,” laughs Torman. | ||
|
||
![Amazon Boston homepage](./amazon-boston-2.jpg "Boston city") | ||
|
||
The iterations paid off. | ||
|
||
"Both internally and externally we’ve heard overwhelmingly positive feedback,” Torman says. "People here love the website." | ||
|
||
Apparently, so did Amazon. |
This file contains 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
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,7 @@ | ||
{ | ||
"name": "gatsby", | ||
"description": "React.js Static Site Generator", | ||
"version": "1.9.158", | ||
"version": "1.9.159", | ||
"author": "Kyle Mathews <[email protected]>", | ||
"bin": { | ||
"gatsby": "./dist/bin/gatsby.js" | ||
|
This file contains 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
This file contains 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
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,231 @@ | ||
import React, { Component } from "react" | ||
import PropTypes from "prop-types" | ||
import { navigateTo } from "gatsby-link" | ||
import { rhythm } from "../utils/typography" | ||
import presets from "../utils/presets" | ||
|
||
import { css } from "glamor" | ||
|
||
// Override default search result styles | ||
css.global(`.searchWrap .algolia-docsearch-suggestion--highlight`, { | ||
backgroundColor: `${presets.lightPurple} !important`, | ||
boxShadow: `inset 0 -2px 0 0 ${presets.lightPurple} !important`, | ||
color: `black`, | ||
fontWeight: `bold`, | ||
}) | ||
css.global(`.searchWrap .algolia-docsearch-suggestion .algolia-docsearch-suggestion--subcategory-column`, { | ||
width: `100% !important`, | ||
}) | ||
css.global(`.searchWrap .algolia-docsearch-suggestion--subcategory-column-text:after`, { | ||
display: `none`, | ||
}) | ||
css.global( | ||
`.searchWrap .algolia-autocomplete .ds-dropdown-menu .ds-suggestion.ds-cursor .algolia-docsearch-suggestion:not(.suggestion-layout-simple) .algolia-docsearch-suggestion--content`, | ||
{ backgroundColor: `${presets.brandLighter} !important` } | ||
) | ||
css.global( | ||
`.searchWrap .algolia-docsearch-suggestion .algolia-docsearch-suggestion--content.algolia-docsearch-suggestion--no-results`, | ||
{ | ||
maxWidth: `100%`, | ||
paddingLeft: `0 !important`, | ||
width: `100% !important`, | ||
} | ||
) | ||
css.global( | ||
`.searchWrap .algolia-docsearch-suggestion .algolia-docsearch-suggestion--content.algolia-docsearch-suggestion--no-results:before`, | ||
{ display: `none !important` } | ||
) | ||
css.global(`.searchWrap .algolia-autocomplete .ds-dropdown-menu`, { | ||
position: `fixed !important`, | ||
top: `${rhythm(2)} !important`, | ||
left: `${rhythm(0.5)} !important`, | ||
right: `${rhythm(0.5)} !important`, | ||
minWidth: `calc(100vw - ${rhythm(1)})`, | ||
maxWidth: `calc(100vw - 2rem)`, | ||
maxHeight: `calc(100vh - 5rem)`, | ||
display: `block`, | ||
}) | ||
css.global( | ||
`.searchWrap .algolia-autocomplete.algolia-autocomplete-right .ds-dropdown-menu, .algolia-autocomplete.algolia-autocomplete-left .ds-dropdown-menu`, | ||
{ | ||
left: `${rhythm(0.5)} !important`, | ||
right: `${rhythm(0.5)} !important`, | ||
} | ||
) | ||
css.global( | ||
`.searchWrap .algolia-autocomplete.algolia-autocomplete-right .ds-dropdown-menu::before`, | ||
{ | ||
right: rhythm(5), | ||
} | ||
) | ||
css.global( | ||
`.searchWrap .algolia-autocomplete.algolia-autocomplete-left .ds-dropdown-menu::before`, | ||
{ | ||
left: rhythm(7), | ||
} | ||
) | ||
|
||
// use css.insert() for media query with global CSS | ||
css.insert(`@media ${presets.phablet}{ | ||
.searchWrap .algolia-autocomplete .algolia-docsearch-suggestion .algolia-docsearch-suggestion--subcategory-column { | ||
font-weight: 400; | ||
width: 30% !important; | ||
text-align: right; | ||
opacity: 1; | ||
padding: ${rhythm(0.5)} ${rhythm(1)} ${rhythm(0.5)} 0; | ||
} | ||
.searchWrap .algolia-autocomplete .algolia-docsearch-suggestion--subcategory-column:before { | ||
content: ""; | ||
position: absolute; | ||
display: block !important; | ||
top: 0; | ||
height: 100%; | ||
width: 1px; | ||
background: #ddd; | ||
right: 0; | ||
} | ||
.searchWrap .algolia-autocomplete .algolia-docsearch-suggestion--subcategory-column:after { | ||
display: none; | ||
} | ||
.searchWrap .algolia-autocomplete .algolia-docsearch-suggestion--content { | ||
width: 70% !important; | ||
max-width: 70%; | ||
display: block; | ||
padding: ${rhythm(0.5)} 0 ${rhythm(0.5)} ${rhythm(1)} !important; | ||
} | ||
.searchWrap .algolia-autocomplete .algolia-docsearch-suggestion--content:before { | ||
content: ""; | ||
position: absolute; | ||
display: block !important; | ||
top: 0; | ||
height: 100%; | ||
width: 1px; | ||
background: #ddd; | ||
left: -1px; | ||
} | ||
}`) | ||
|
||
css.insert(`@media ${presets.tablet}{ | ||
.searchWrap .algolia-autocomplete .ds-dropdown-menu { | ||
top: 100% !important; | ||
position: absolute !important; | ||
max-width: 600px !important; | ||
min-width: 500px !important; | ||
} | ||
.searchWrap .algolia-autocomplete.algolia-autocomplete-right .ds-dropdown-menu { | ||
right: 0 !important; | ||
left: inherit !important; | ||
} | ||
.searchWrap .algolia-autocomplete.algolia-autocomplete-right .ds-dropdown-menu::before { | ||
right: ${rhythm(2)}; | ||
} | ||
}`) | ||
|
||
class SearchForm extends Component { | ||
constructor() { | ||
super() | ||
this.state = { enabled: true } | ||
} | ||
|
||
/** | ||
* Replace the default selection event, allowing us to do client-side | ||
* navigation thus avoiding a full page refresh. | ||
* | ||
* Ref: https://github.com/algolia/autocomplete.js#events | ||
*/ | ||
autocompleteSelected(e) { | ||
e.stopPropagation() | ||
// Use an anchor tag to parse the absolute url (from autocomplete.js) into a relative url | ||
// eslint-disable-next-line no-undef | ||
const a = document.createElement(`a`) | ||
a.href = e._args[0].url | ||
navigateTo(`${a.pathname}${a.hash}`) | ||
} | ||
|
||
componentDidMount() { | ||
if ( | ||
typeof window === `undefined` || // eslint-disable-line no-undef | ||
typeof window.docsearch === `undefined` // eslint-disable-line no-undef | ||
) { | ||
console.warn(`Search has failed to load and now is being disabled`) | ||
this.setState({ enabled: false }) | ||
return | ||
} | ||
|
||
// eslint-disable-next-line no-undef | ||
window.addEventListener( | ||
`autocomplete:selected`, | ||
this.autocompleteSelected, | ||
true | ||
) | ||
|
||
// eslint-disable-next-line no-undef | ||
window.docsearch({ | ||
apiKey: `71af1f9c4bd947f0252e17051df13f9c`, | ||
indexName: `gatsbyjs`, | ||
inputSelector: `#doc-search`, | ||
debug: false, | ||
}) | ||
} | ||
|
||
render() { | ||
const { enabled } = this.state | ||
const { styles } = this.props.styles | ||
return enabled ? ( | ||
<form | ||
css={{ | ||
...styles, | ||
display: `flex`, | ||
flex: `0 0 auto`, | ||
flexDirection: `row`, | ||
alignItems: `center`, | ||
marginLeft: rhythm(1 / 2), | ||
marginBottom: 0, | ||
}} | ||
className="searchWrap" | ||
> | ||
<input | ||
id="doc-search" | ||
css={{ | ||
appearance: `none`, | ||
background: `transparent`, | ||
border: 0, | ||
color: presets.brand, | ||
paddingTop: rhythm(1 / 8), | ||
paddingRight: rhythm(1 / 4), | ||
paddingBottom: rhythm(1 / 8), | ||
paddingLeft: rhythm(1), | ||
backgroundImage: `url(/search.svg)`, | ||
backgroundSize: `16px 16px`, | ||
backgroundRepeat: `no-repeat`, | ||
backgroundPositionY: `center`, | ||
backgroundPositionX: `5px`, | ||
overflow: `hidden`, | ||
width: rhythm(1), | ||
transition: `width 0.2s ease`, | ||
|
||
":focus": { | ||
outline: 0, | ||
backgroundColor: presets.brandLighter, | ||
borderRadius: presets.radiusLg, | ||
width: rhythm(5), | ||
}, | ||
|
||
[presets.Desktop]: { | ||
width: rhythm(5), | ||
}, | ||
}} | ||
type="search" | ||
placeholder="Search docs" | ||
aria-label="Search docs" | ||
/> | ||
</form> | ||
) : null | ||
} | ||
} | ||
|
||
SearchForm.propTypes = { | ||
styles: PropTypes.object, | ||
} | ||
|
||
export default SearchForm |
Oops, something went wrong.