|
| 1 | +# AMP |
| 2 | + |
| 3 | +Moovweb provides a commercial package, `react-storefront-amp` that automatically makes your app compatible with [Google's AMP](https://amp.dev/). [Contact Moovweb](https://www.moovweb.com/learn/request-demo) for more information. With `react-storefront-amp` you write pages in React and they look and behave the same in AMP. All you need to do is check that each page is generating valid AMP as part of your QA process. |
| 4 | + |
| 5 | +## How automatic AMP support works |
| 6 | + |
| 7 | +Next.js provides the [building blocks for rendering AMP versions](https://nextjs.org/docs#enabling-amp-support) of pages. |
| 8 | + |
| 9 | +The `react-storefront-amp` library builds upon next.js to make AMP support automatic by: |
| 10 | + |
| 11 | +- Providing AMP-aware replacements for components in `react-storefront` that render valid AMPHTML when `?amp=1` is present on the URL. |
| 12 | +- Doing additional cleanup of the outgoing HTML before it is sent to the browser to that it passes AMP validation |
| 13 | +- Adding explicit heights and widths to images that do not already have them. AMP requires that all images are either sized by their containing element or have `height` and `width` props. |
| 14 | + |
| 15 | +## AMP-compatible starter app |
| 16 | + |
| 17 | +Developers that have purchased the `react-storefront-amp` package should create their app using the `--branch=commercial` option: |
| 18 | + |
| 19 | +``` |
| 20 | +npm create react-storefront --branch=commercial |
| 21 | +``` |
| 22 | + |
| 23 | +The main differences between the commercial and free starter apps are that the commercial app uses two-way data binding from `react-storefront-amp` to manage state: |
| 24 | + |
| 25 | +## Components |
| 26 | + |
| 27 | +The structure of the `react-storefront-amp` package mimics that of `react-storefront`. It provides AMP-compatible replacements for many of the components in `react-storefront`, including: |
| 28 | + |
| 29 | +- Accordion |
| 30 | +- Drawer |
| 31 | +- ExpandableSection |
| 32 | +- Image |
| 33 | +- QuantitySelector |
| 34 | +- TabPanel |
| 35 | +- Menu |
| 36 | +- ProductOptionSelector |
| 37 | +- FilterButton |
| 38 | +- SortButton |
| 39 | +- SearchButton |
| 40 | +- SearchDrawer |
| 41 | + |
| 42 | +Each is prefixed with `Amp*`, so for example, `react-storefront-amp/AmpAccordion` replaces `react-storefront/Accordion`. |
| 43 | + |
| 44 | +## AMPHTML cleanup |
| 45 | + |
| 46 | +The `react-storefront-amp` packages provides a `renderAmp` function that cleans up the outgoing HTML so that it passes AMP validation. This function is |
| 47 | +called in `pages/_document.js` before server-side rendered HTML is returned to the browser: |
| 48 | + |
| 49 | +```js |
| 50 | +// pages/_document.js |
| 51 | + |
| 52 | +ctx.renderPage = async () => { |
| 53 | + const document = originalRenderPage({ |
| 54 | + enhanceApp: App => props => sheets.collect(<App {...props} />) |
| 55 | + }) |
| 56 | + return isAmp ? await renderAmp(document, sheets) : document |
| 57 | +} |
| 58 | +``` |
| 59 | + |
| 60 | +## Two-way data binding |
| 61 | + |
| 62 | +AMP and React have fundamentally different patterns for managing state. React allows you to respond to events with javascript and update state using the update function returned by the `useState` hook. AMP uses a more declarative approach with [amp-bind](https://amp.dev/documentation/components/amp-bind/) that resembles to-way data-binding. In order to create a common approach to state management that works in both react and amp, `react-storefront-amp` provides two-way databinding via a `DataBindingProvider` component and the `bind` prop. |
| 63 | + |
| 64 | +Here's a simple example from `pages/p/[productId].js` that shows how the quantity selector updates the page state in both AMP and React using the same code: |
| 65 | + |
| 66 | +```js |
| 67 | +// pages/p/[productId].js |
| 68 | + |
| 69 | +import useLazyState from 'react-storefront/hooks/useLazyState' |
| 70 | +import DataBindingProvider from 'react-storefront-amp/bind/DataBindingProvider' |
| 71 | +import QuantitySelector from 'react-storefront-amp/AmpQuantitySelector' |
| 72 | + |
| 73 | +// ... |
| 74 | + |
| 75 | +const Product = React.memo(lazyProps => { |
| 76 | + // ... |
| 77 | + |
| 78 | + const [store, updateStore] = useLazyState(lazyProps, { pageData: { quantity: 1 } }) |
| 79 | + |
| 80 | + return ( |
| 81 | + <DataBindingProvider store={store} updateStore={updateStore} root="pageData"> |
| 82 | + {/* ... */} |
| 83 | + <QuantitySelector bind="quantity" /> |
| 84 | + {/* ... */} |
| 85 | + </DataBindingProvider> |
| 86 | + ) |
| 87 | +}) |
| 88 | +``` |
| 89 | + |
| 90 | +### Binding a single prop |
| 91 | + |
| 92 | +Here setting `bind="quantity"` automatically sets the `value` prop of `QuantitySelector` to the `quantity` field in the store from the object specified by the `root` prop in `DataBindingProvider`. |
| 93 | + |
| 94 | +When the user changes the quantity, `store.pageData.quantity` is automatically updated with the new value based on the presence of `bind="quantity"`. |
| 95 | + |
| 96 | +In other words, this: |
| 97 | + |
| 98 | +```js |
| 99 | +<QuantitySelector bind="quantity" /> |
| 100 | +``` |
| 101 | + |
| 102 | +... is equivalent to: |
| 103 | + |
| 104 | +```js |
| 105 | +<QuantitySelector |
| 106 | + value={store.pageData.quantity} |
| 107 | + onChange={value => { |
| 108 | + updateStore({ |
| 109 | + ...store, |
| 110 | + pageData: { |
| 111 | + ...store.pageData, |
| 112 | + quantity: value |
| 113 | + } |
| 114 | + }) |
| 115 | + } |
| 116 | +/> |
| 117 | +``` |
| 118 | +
|
| 119 | +### Binding multiple props |
| 120 | +
|
| 121 | +The `bind` prop can also be used to bind multiple props to values in a store, as in the case of `ProductOptionSelector`, which gets `options` and `value` props from the store: |
| 122 | +
|
| 123 | +```js |
| 124 | +<ProductOptionSelector bind={{ value: 'size', options: 'product.sizes' }} /> |
| 125 | +``` |
| 126 | +
|
| 127 | +Here bind takes an object whose keys are the props to bind and whose values are the paths to the values for those props in the store. |
| 128 | +
|
| 129 | +### Binding a prop to multiple values |
| 130 | +
|
| 131 | +The `bind` prop also allows you to pull the value from alternate fields in the store if the default field is `null` or `undefined` by passing an array as the value. Here is an example from the product link on the subcategory page: |
| 132 | +
|
| 133 | +```js |
| 134 | +// components/product/ProductItem.js |
| 135 | + |
| 136 | +<AmpImage |
| 137 | + bind={{ |
| 138 | + // use the thumbnail and alt for the selected color if one is selected, otherwise use the default thumbnail and alt |
| 139 | + src: ['color.media.thumbnail.src', 'thumbnail.src'], |
| 140 | + alt: ['color.media.thumbnail.alt', 'thumbnail.alt'] |
| 141 | + }} |
| 142 | +/> |
| 143 | +``` |
0 commit comments