Bloomreach Vue.js SDK provides simplified headless integration with Bloomreach Experience Manager for Vue-based applications. This library interacts with the Page Model API and Bloomreach SPA SDK and exposes a simplified declarative Vue.js interface over the Page Model.
- Bloomreach Page Vue component;
- Bloomreach Components Vue component;
- Manage Content Button;
- Manage Menu Button;
- Slots support;
- Vue Server-Side Rendering support;
- Nuxt.js support;
- Vue Router support;
- Jest support.
Important notice: Vue SDK does not work in combination with nomarkup
xtype container components. (The xtype of a container component is described here and here)
The problem is that Vue 2 does not support multi-root templates and the fragement libraries such as Vue Fragment that enable this for Vue 2 all add 'magic' parentNodes which is incompatible with some of the Bloomreach Experience Manager code constructs. We are looking into updating to Vue 3 when possible which does support multi-root templates without extra magic.
To get the SDK into your project with NPM:
npm install @bloomreach/vue-sdk
And with Yarn:
yarn add @bloomreach/vue-sdk
The following code snippets render a simple page with a Banner component.
In the app's entry file, it needs to import and
install the BrSdk
plugin to make the SDK components available globally.
import Vue from 'vue';
import { BrSdk } from '@bloomreach/vue-sdk';
Vue.use(BrSdk);
// ...
In the App
component, it needs to pass the configuration and brXM components
mapping into the br-page
component inputs.
<template>
<br-page :configuration="configuration" :mapping="mapping">
<template v-slot:default="props">
<header>
<a :href="props.page.getUrl('/')">Home</a>
<br-component component="menu" />
</header>
<section>
<br-component component="main" />
</section>
<footer>
<br-component component="footer" />
</footer>
</template>
</br-page>
</template>
<script>
import Banner from './components/Banner.vue';
export default {
data() {
return {
configuration: {
/* ... */
},
mapping: { Banner },
};
},
};
</script>
Finally, in the Banner
component, it can consume the component data via the
component
prop.
<template>
<div>Banner: {{ component.getName() }}</div>
</template>
<script>
export default { props: ['component'] };
</script>
Non-blocking rendering mode can be used to decrease the time for your application to load fully on the client side. By
default the NBRMode configuration is false
to avoid breaking existing setups. Setting it to true
will enable
non-blocking render mode. When the mode is active the children of the BrPage component will start mounting while the
Page Model is being fetched. These children might contain logic themselves that queries some external API and using
non-blocking render mode would allow this to be executed in parallel to requesting the Page Model.
<template>
Hello
</template>
<script>
export default {
name: 'MyComponent',
async mounted() {
// This will run in parallel to fetching the PageModel from the Delivery API
await fetch('https://yourapi.com');
}
};
</script>
<template>
<br-page :configuration="configuration" :mapping="mapping">
<my-component></my-component>
<template v-slot:default="props">
...
</br-page>
</template>
<script>
export default {
data() {
return {
configuration: {
/* ... */
NBRMode: true,
},
mapping: { ... },
};
},
};
</script>
The br-page
component supports several options you may use to customize page
initialization. These options will be passed to the initialize
function from
@bloomreach/spa-sdk
. See
here for the
full configuration documentation.
The br-page
component provides a way to link Vue components with the brXM
ones. It requires to pass the mapping
property that maps the component type
with its representation.
The Container Items can be mapped by their labels.
<template>
<br-page :configuration="configuration" :mapping="mapping" />
</template>
<script>
import NewsList from './components/NewsList.vue';
export default {
data() {
return {
configuration: {
/* ... */
},
mapping: { 'News List': NewsList },
};
},
};
</script>
The Containers
can be only mapped by their type,
so you need to use constants from
@bloomreach/spa-sdk
. By default,
the Vue.js SDK provides an implementation for all the container types as it is
defined in the documentation.
<template>
<br-page :configuration="configuration" :mapping="mapping" />
</template>
<script>
import { TYPE_CONTAINER_INLINE } from '@bloomreach/spa-sdk';
import InlineContainer from './components/InlineContainer.vue';
export default {
data() {
return {
configuration: {
/* ... */
},
mapping: { [TYPE_CONTAINER_INLINE]: InlineContainer },
};
},
};
</script>
From within the Container component, the Container Items can be accessed via the
getChildren
method. This can be used to reorder or wrap child elements.
<template>
<div>
<span v-for="(child, key) in component.getChildren()" :key="key">
<br-component :component="child" />
</span>
</div>
</template>
<script>
export default { props: ['component'] };
</script>
That is also possible to render children via the default slot.
export default {
render(createElement) {
return createElement(
'div',
this.$slots.default.map((node) => createElement('span', [node])),
);
},
};
The Components can be mapped by their names. It is useful for a menu component mapping.
<template>
<br-page :configuration="configuration" :mapping="mapping" />
</template>
<script>
import Menu from './components/Menu.vue';
export default {
data() {
return {
configuration: {
/* ... */
},
mapping: { menu: Menu },
};
},
};
</script>
By default, container items that are not mapped will be rendered as a warning text. There is an option to override the fallback.
<template>
<br-page :configuration="configuration" :mapping="mapping" />
</template>
<script>
import { TYPE_CONTAINER_ITEM_UNDEFINED } from '@bloomreach/spa-sdk';
import Fallback from './components/Fallback.vue';
export default {
data() {
return {
configuration: {
/* ... */
},
mapping: { [TYPE_CONTAINER_ITEM_UNDEFINED]: Fallback },
};
},
};
</script>
There is also another way to render a component. In case you need to show a static component or a component from the abstract page, you can use inline component mapping.
<template>
<br-page :configuration="configuration" :mapping="mapping">
<template>
<br-component component="menu">
<template v-slot:default="{ component, page }">
<menu :component="component" :page="page" />
</template>
</br-component>
</template>
</br-page>
</template>
<script>
import Menu from './components/Menu.vue';
export default {
components: { Menu },
data() {
return {
configuration: {
/* ... */
},
mapping: {},
};
},
};
</script>
In a component it is also possible to point where the component's children are going to be placed.
<template>
<div>
@copy; Bloomreach
<br-component />
</div>
</template>
<script>
export default { props: ['component'] };
</script>
The component data in case of inline mapping can be accessed via the template context.
<template>
<br-page :configuration="configuration" :mapping="mapping">
<template>
<br-component component="menu">
<template v-slot:default="{ component, page }">
<ul>
<li><a :href="page.getUrl('/')">Home</a></li>
<li v-for="(item, key) in component.getModels()" :key="key">...</li>
</ul>
</template>
</br-component>
</template>
</br-page>
</template>
<script>
export default {
data() {
return {
configuration: {
/* ... */
},
mapping: {},
};
},
};
</script>
It is recommended to add the css style position: relative
to the Buttons
so they will position correctly within their parent container component.
Manage menu button can be placed inside a menu component using
br-manage-menu-button
component.
<template>
<ul v-if="menu" :class="{ 'has-edit-button': page.isPreview() }">
<!-- ... -->
<br-manage-menu-button :menu="menu" />
</ul>
</template>
<script>
import { Menu, Reference } from '@bloomreach/spa-sdk';
interface MenuModels {
menu: Reference;
}
export default {
computed: {
menu() {
const { menu: menuRef } = this.component.getModels<MenuModels>();
return menuRef && this.page.getContent<Menu>(menuRef);
},
},
props: ['component', 'page'],
};
</script>
Manage content button can be placed inside a component using
br-manage-content-button
component with non-empty content
property.
<template>
<div v-if="document" :class="{ 'has-edit-button': page.isPreview() }">
<!-- ... -->
<br-manage-content-button
:content="document"
document-template-query="new-banner-document"
folder-template-query="new-banner-folder"
parameter="document"
root="banners"
:relative="true"
/>
</div>
</template>
<script>
import { Document, Reference } from '@bloomreach/spa-sdk';
interface BannerModels {
document: Reference;
}
export default {
computed: {
document() {
const { document: documentRef } = this.component.getModels<BannerModels>();
return documentRef && this.page.getContent<Document>(documentRef);
},
},
props: ['component', 'page'],
};
</script>
Add new content button can be placed inside a component using
br-manage-content-button
directive but without passing a content entity.
<template>
<div :class="{ 'has-edit-button': page.isPreview() }">
<!-- ... -->
<br-manage-content-button
document-template-query="new-news-document"
folder-template-query="new-news-folder"
root="news"
/>
</div>
</template>
<script>
export default {
props: ['component', 'page'],
};
</script>
If you are navigating between pages that have and those that don't have a SDK instance you will need to persist the preview related data. See detailed guide how to setup it in framework specific examples, nuxtjs, vue.
Published under Apache 2.0 license.
The Vue.js SDK is using Bloomreach SPA SDK to interact with the brXM.
This is the entry point to the page model. This component requests and initializes the page model, and then renders the page root component recursively.
Property | Required | Description |
---|---|---|
configuration |
yes | The configuration of the SPA SDK. |
mapping |
yes | The brXM and Vue.js components mapping. |
page |
no | Preinitialized page instance or prefetched page model. Mostly that should be used to transfer state from the server-side to the client-side. |
This component also supports a default slot transclusion. <template>
from the
component contents will be rendered in the root component context.
Variable | Description |
---|---|
component |
The root component. |
page |
The current page instance. |
This component points to where children or some component should be placed.
br-component
can be used inside br-page
or mapped components only. If it
contains <template>
in the contents, then the template will be rendered in the
context of every matched component. Otherwise, it will try to render all
children components recursively.
Property | Required | Description |
---|---|---|
component |
no | The component instance or a path to a component. The path is defined as a slash-separated components name chain relative to the current component (e.g. main/container ). If it is omitted, all the children will be rendered. |
The template context holds references to the current component and the current page instance.
Variable | Description |
---|---|
component |
The current component. |
page |
The current page instance. |
This component places a button on the page that opens the linked content in the document editor or opens a document editor to create a new one. The button will only be shown in preview mode.
Property | Required | Description |
---|---|---|
content |
no | The content entity to open for editing. |
document-template-query |
no | Template query to use for creating new documents. For Content SaaS, the key is like new-banner-document for banner document type. In case the document type is in a namespace other than brxsaas , the key also includes namespace prefix, such as new-vuestorefront-accordion-document for accordion document type in vuestorefront namespace. |
folder-template-query |
no | Template query to use in case folders specified by path do not yet exist and must be created. For Content SaaS, the key is like new-banner-folder for banner document type. In case the document type is in a namespace other than brxsaas , the key also includes namespace prefix, such as new-vuestorefront-accordion-folder for accordion document type in vuestorefront namespace. |
path |
no | Initial location of a new document, relative to the root . |
parameter |
no | Name of the component parameter in which the document path is stored. |
pickerConfiguration |
no | The root path of the CMS configuration to use for the picker, relative to /hippo:configuration/hippo:frontend/cms . |
pickerEnableUpload |
no | When this picker is used for images, this flag determines if uploads are enabled. |
pickerInitialPath |
no | The initial path to use in the picker if nothing has been selected yet, relative to the pickerRootPath. |
pickerRemembersLastVisited |
no | Whether the picker remembers the last visited path. |
pickerRootPath |
no | The absolute root path to use in the picker, or an empty string if the channel content path is used. |
pickerSelectableNodeTypes |
no | Types of nodes to be able to select in the picker, separated by a comma. |
relative |
no | Flag indicating that the picked value should be stored as a relative path. |
root |
no | Path to the root folder of selectable document locations. |
This directive places a button on the page that opens the linked menu in the menu editor. The button will only be shown in preview mode.
Property | Required | Description |
---|---|---|
menu |
yes | The related menu model. |