Skip to content

Commit bfc8f9c

Browse files
committed
Docs for version v8.12.2
1 parent eb59482 commit bfc8f9c

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

44 files changed

+2760
-1
lines changed

v8.12.2/guides.json

+142
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,142 @@
1+
[
2+
{
3+
"text": "Guides",
4+
"items": [
5+
{
6+
"text": "Getting Started",
7+
"as": "/guides/getting_started",
8+
"href": "/guides/[...guide]"
9+
},
10+
{
11+
"text": "Overview",
12+
"as": "/guides/overview",
13+
"href": "/guides/[...guide]"
14+
},
15+
{
16+
"text": "Next.js",
17+
"as": "/guides/nextjs",
18+
"href": "/guides/[...guide]"
19+
},
20+
{
21+
"text": "Managing State",
22+
"as": "/guides/managing_state",
23+
"href": "/guides/[...guide]"
24+
},
25+
{
26+
"text": "Connectors",
27+
"as": "/guides/connectors",
28+
"href": "/guides/[...guide]"
29+
},
30+
{
31+
"text": "API",
32+
"as": "/guides/api",
33+
"href": "/guides/[...guide]"
34+
},
35+
{
36+
"text": "Navigation",
37+
"as": "/guides/navigation",
38+
"href": "/guides/[...guide]"
39+
},
40+
{
41+
"text": "Prefetching",
42+
"as": "/guides/prefetching",
43+
"href": "/guides/[...guide]"
44+
},
45+
{
46+
"text": "Styling",
47+
"as": "/guides/styling",
48+
"href": "/guides/[...guide]"
49+
},
50+
{
51+
"text": "Sessions",
52+
"as": "/guides/sessions",
53+
"href": "/guides/[...guide]"
54+
},
55+
{
56+
"text": "CMS Integration",
57+
"as": "/guides/cms",
58+
"href": "/guides/[...guide]"
59+
},
60+
{
61+
"text": "Lazy Hydration",
62+
"as": "/guides/lazy_hydration",
63+
"href": "/guides/[...guide]"
64+
},
65+
{
66+
"text": "Performance",
67+
"as": "/guides/performance",
68+
"href": "/guides/[...guide]"
69+
}
70+
]
71+
},
72+
{
73+
"text": "Pages and Features",
74+
"items": [
75+
{
76+
"text": "Main Menu",
77+
"as": "/guides/main_menu",
78+
"href": "/guides/[...guide]"
79+
},
80+
{
81+
"text": "Home",
82+
"as": "/guides/home",
83+
"href": "/guides/[...guide]"
84+
},
85+
{
86+
"text": "Subcategory",
87+
"as": "/guides/subcategory",
88+
"href": "/guides/[...guide]"
89+
},
90+
{
91+
"text": "Product",
92+
"as": "/guides/product",
93+
"href": "/guides/[...guide]"
94+
},
95+
{
96+
"text": "Search",
97+
"as": "/guides/search",
98+
"href": "/guides/[...guide]"
99+
}
100+
]
101+
},
102+
{
103+
"text": "Premium Features",
104+
"items": [
105+
{
106+
"text": "AMP",
107+
"as": "/guides/amp",
108+
"href": "/guides/[...guide]"
109+
},
110+
{
111+
"text": "Analytics",
112+
"as": "/guides/analytics",
113+
"href": "/guides/[...guide]"
114+
}
115+
]
116+
},
117+
{
118+
"text": "Moovweb XDN",
119+
"items": [
120+
{
121+
"text": "Deploying",
122+
"as": "/guides/xdn_deploying",
123+
"href": "/guides/[...guide]"
124+
},
125+
{
126+
"text": "Routing",
127+
"as": "/guides/xdn_routing",
128+
"href": "/guides/[...guide]"
129+
},
130+
{
131+
"text": "Caching",
132+
"as": "/guides/xdn_caching",
133+
"href": "/guides/[...guide]"
134+
},
135+
{
136+
"text": "Image Optimization",
137+
"as": "/guides/xdn_image_optimization",
138+
"href": "/guides/[...guide]"
139+
}
140+
]
141+
}
142+
]

v8.12.2/guides/amp.md

+143
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,143 @@
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

Comments
 (0)