- Documentation >
- Tutorials >
- Extending Admin UI >
- 5. Creating a UDW tab
Step 5 - Creating a UDW tab
The Universal Discovery Widget (UDW) is a separate React module.
By default it contains two tabs: Browse and Search.
In this step you will add a new tab called Images which will display all Content items of the type 'Image'
using a gallery of thumbnails.
Tip
To be able to view the results of this step, create a few Content items of the type "Image".
Add a tab
First add a src/EzSystems/ExtendingTutorialBundle/Resources/public/js/add.tab.js
which governs the creation of the new tab:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15 | (function (global, eZ) {
eZ = eZ || {};
eZ.adminUiConfig = eZ.adminUiConfig || {};
eZ.adminUiConfig.universalDiscoveryWidget = eZ.adminUiConfig.universalDiscoveryWidget || {};
eZ.adminUiConfig.universalDiscoveryWidget.extraTabs = eZ.adminUiConfig.universalDiscoveryWidget.extraTabs || [];
eZ.adminUiConfig.universalDiscoveryWidget.extraTabs.push({
id: 'images',
title: 'Images',
iconIdentifier: 'image',
panel: eZ.modules.ImagesPanel, // React component that represents content of a tab
attrs: {}
});
})(window, window.eZ);
|
The highlighted line indicates the actual panel inside the tab that will display the images.
Provide ReactJS development files
Next you need to provide a set of files that will later be compiled into the module:
images.service.js
handles fetching the data
images.panel.js
renders the panel inside the tab
images.list.js
renders the image list
image.js
renders a single image
Place all these files in src/EzSystems/ExtendingTutorialBundle/Resources/ui-dev/src
:
images.service.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69 | const handleRequestResponse = response => {
if (!response.ok) {
throw Error(response.statusText);
}
return response.json();
};
export const getImages = ({token, siteaccess, contentId}, callback) => {
const body = JSON.stringify({
ViewInput: {
identifier: 'images',
public: false,
LocationQuery: {
Criteria: {},
FacetBuilders: {},
SortClauses: {},
Filter: { ContentTypeIdCriterion: 5 }
}
}
});
const request = new Request('/api/ezp/v2/views', {
method: 'POST',
headers: {
'Accept': 'application/vnd.ez.api.View+json; version=1.1',
'Content-Type': 'application/vnd.ez.api.ViewInput+json; version=1.1',
'X-Siteaccess': siteaccess,
'X-CSRF-Token': token,
},
body,
mode: 'cors',
});
fetch(request)
.then(handleRequestResponse)
.then(callback)
.catch(error => console.log('error:load:images', error));
};
export const loadImageContent = ({token, siteaccess, contentId}, callback) => {
const body = JSON.stringify({
ViewInput: {
identifier: `image-content-${contentId}`,
public: false,
ContentQuery: {
Criteria: {},
FacetBuilders: {},
SortClauses: {},
Filter: { ContentIdCriterion: contentId }
}
}
});
const request = new Request('/api/ezp/v2/views', {
method: 'POST',
headers: {
'Accept': 'application/vnd.ez.api.View+json; version=1.1',
'Content-Type': 'application/vnd.ez.api.ViewInput+json; version=1.1',
'X-Siteaccess': siteaccess,
'X-CSRF-Token': token,
},
body,
mode: 'cors',
});
fetch(request)
.then(handleRequestResponse)
.then(callback)
.catch(error => console.log('error:load:images', error));
};
|
images.panel.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19 | import React from 'react';
import ImagesList from './images.list';
const ImagesPanel = (props) => {
const wrapperAttrs = {className: 'c-images-panel'};
if (!props.isVisible) {
wrapperAttrs.hidden = true;
}
return (
<div {...wrapperAttrs}>
<h1>Images panel</h1>
<ImagesList {...props} />
</div>
);
};
export default ImagesPanel;
|
images.list.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112 | import React, { Component } from 'react';
import Image from './image';
import { getImages } from './images.service';
import './css/images.list.css';
class ImagesList extends Component {
constructor(props) {
super(props);
this.updateImagesState = this.updateImagesState.bind(this);
this.showPrevPage = this.showPrevPage.bind(this);
this.showNextPage = this.showNextPage.bind(this);
this.state = {
images: [],
page: 0,
itemsPerPage: 5,
maxPageIndex: 0
};
}
componentDidMount() {
getImages(this.props.restInfo, this.updateImagesState);
}
updateImagesState(response) {
const images = response.View.Result.searchHits.searchHit.map(item => item.value.Location);
const perPage = this.state.itemsPerPage;
const modulo = images.length % perPage;
const maxPageIndex = modulo ? (images.length - modulo) / perPage : images.length / perPage - 1;
this.setState({ images, maxPageIndex });
}
showPrevPage() {
this.setState(state => Object.assign({}, state, {
page: state.page > 0 ? state.page - 1 : 0
}));
}
showNextPage() {
this.setState(state => Object.assign({}, state, {
page: state.maxPageIndex > state.page ? state.page + 1 : state.maxPageIndex
}));
}
renderItems() {
const attrs = {
className: 'c-images-list__items',
style: {
transform: `translate3d(-${this.state.page * this.state.itemsPerPage * 316}px, 0, 0)`
}
};
return (
<div className="c-images-list__items-wrapper">
<div {...attrs}>
{this.state.images.map(imageLocation => <Image key={imageLocation.id} location={imageLocation} restInfo={this.props.restInfo}/>)}
</div>
</div>
);
}
renderPrevBtn() {
const attrs = {
className: 'c-images-list__btn--prev',
onClick: this.showPrevPage
};
if (this.state.page <= 0) {
attrs.disabled = true;
}
return (
<div {...attrs}>
<svg className="ez-icon">
<use xlinkHref="/bundles/ezplatformadminui/img/ez-icons.svg#caret-back"></use>
</svg>
</div>
);
}
renderNextBtn() {
const attrs = {
className: 'c-images-list__btn--next',
onClick: this.showNextPage
};
if (this.state.page >= this.state.maxPageIndex) {
attrs.disabled = true;
}
return (
<div {...attrs}>
<svg className="ez-icon">
<use xlinkHref="/bundles/ezplatformadminui/img/ez-icons.svg#caret-next"></use>
</svg>
</div>
);
}
render() {
return (
<div className="c-images-list">
{this.renderPrevBtn()}
{this.renderItems()}
{this.renderNextBtn()}
</div>
);
}
}
export default ImagesList;
|
image.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44 | import React, { Component } from 'react';
import { loadImageContent } from './images.service';
import './css/image.css';
class Image extends Component {
constructor(props) {
super(props);
this.updateVersionInfoState = this.updateVersionInfoState.bind(this);
this.state = {
content: null
};
}
componentDidMount() {
loadImageContent({ ...this.props.restInfo, contentId: this.props.location.ContentInfo.Content._id}, this.updateVersionInfoState);
}
updateVersionInfoState(response) {
this.setState(state => Object.assign({}, state, {
content: response.View.Result.searchHits.searchHit[0].value.Content
}));
}
render() {
let src = '';
let alt = 'Loading meta data ...';
if (this.state.content) {
const imageField = this.state.content.CurrentVersion.Version.Fields.field.find(field => field.fieldTypeIdentifier === 'ezimage').fieldValue;
src = imageField.uri;
alt = imageField.fileName;
}
return (
<div className="c-image" data-title={alt} onClick={() => console.log(data)}>
<img className="c-image__thumb" src={src} alt={alt} />
</div>
);
}
}
export default Image;
|
Finally, ensure that the new tab is styled by adding the following two files to src/EzSystems/ExtendingTutorialBundle/Resources/ui-dev/src/css
:
image.css
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43 | .c-image {
width: 300px;
height: 200px;
background: #fff;
transition: box-shadow .3s ease-in-out;
position: relative;
cursor: pointer;
display: flex;
}
.c-image:before {
content: attr(data-title);
display: flex;
background: rgba(0,0,0,.75);
color: #fff;
width: 300px;
align-items: center;
justify-content: center;
font-weight: 700;
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
opacity: 0;
padding: 1rem;
transition: opacity .3s ease-in-out;
overflow: hidden;
}
.c-image:hover:before,
.c-image:focus:before {
opacity: 1;
}
.c-image__thumb {
display: block;
max-width: 300px;
max-height: 200px;
width: auto;
height: auto;
margin: auto;
}
|
images.list.css
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59 | .c-images-list {
display: grid;
grid-template-areas: 'prev list next';
grid-template-columns: 32px 1fr 32px;
grid-gap: 16px;
overflow: hidden;
}
.c-images-list__items-wrapper {
overflow: hidden;
max-width: 1564px;
}
[class*="c-images-list__btn--"] {
display: flex;
align-items: center;
justify-content: center;
background: #f15a10;
transition: background .2s ease-in-out, opacity .2s ease-in-out;
}
[class*="c-images-list__btn--"]:focus,
[class*="c-images-list__btn--"]:hover {
background: #ab3f0a;
}
[class*="c-images-list__btn--"][disabled],
[class*="c-images-list__btn--"][disabled]:focus,
[class*="c-images-list__btn--"][disabled]:hover {
background: #f15a10;
opacity: .5;
}
[class*="c-images-list__btn--"] .ez-icon {
fill: #fff;
}
.c-images-list__btn--prev {
grid-area: prev;
}
.c-images-list__btn--next {
grid-area: next;
}
.c-images-list__items {
grid-area: list;
display: flex;
flex-wrap: nowrap;
transition: transform .3s ease-in-out;
}
.c-images-list__items .c-image {
flex: 0 0 300px;
}
.c-images-list__items .c-image + .c-image {
margin-left: 1rem;
}
|
Compile the files
Now you need to make the React module available (in umd version) in the window
object in the browser.
Place the compiled files in src/EzSystems/ExtendingTutorialBundle/Resources/public/js/modules
.
How to compile the React module
If you are not familiar with compiling React modules, you can use the following steps:
- Copy the
webpack.*.js
, package.json
, and .babelrc
files from vendor/ezsystems/ezplatform-admin-ui-modules
to src/EzSystems/ExtendingTutorialBundle/Resources/ui-dev
.
- Replace the existing modules in the
entry
property in webpack.common.js
with ImagesPanel: './src/images.panel.js',
.
- In the terminal, in
src/EzSystems/ExtendingTutorialBundle/Resources/ui-dev
, run npm install
.
- Next, run
npm run build
.
Add configuration
Next, ensure that the React modules will be correctly initialized.
To do this, add the following instructions to src/EzSystems/ExtendingTutorialBundle/Resources/config/services.yml
.
This adds JavaScript files into specific zones inside the built-in layout.html.twig
file.
ReactJS modules (the compiled files) should be placed in the custom-admin-ui-modules
group,
and configuration in the custom-admin-ui-config
group.
1
2
3
4
5
6
7
8
9
10
11
12
13 | ezplatform.udw.image.panel.module.js:
parent: EzSystems\EzPlatformAdminUi\Component\ScriptComponent
arguments:
$src: /bundles/ezsystemsextendingtutorial/js/modules/ImagesPanel.module.js
tags:
- { name: ezplatform.admin_ui.component, group: custom-admin-ui-modules }
ezplatform.udw.add.tab.js:
parent: EzSystems\EzPlatformAdminUi\Component\ScriptComponent
arguments:
$src: /bundles/ezsystemsextendingtutorial/js/add.tab.js
tags:
- { name: ezplatform.admin_ui.component, group: custom-admin-ui-config }
|
Finally, make sure that the Symfony bundle configuration is imported. In src/EzSystems/ExtendingTutorialBundle/DependencyInjection
add the following files:
Configuration.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29 | <?php
namespace EzSystems\ExtendingTutorialBundle\DependencyInjection;
use Symfony\Component\Config\Definition\Builder\TreeBuilder;
use Symfony\Component\Config\Definition\ConfigurationInterface;
/**
* This is the class that validates and merges configuration from your app/config files.
*
* To learn more see {@link http://symfony.com/doc/current/cookbook/bundles/configuration.html}
*/
class Configuration implements ConfigurationInterface
{
/**
* {@inheritdoc}
*/
public function getConfigTreeBuilder()
{
$treeBuilder = new TreeBuilder();
$rootNode = $treeBuilder->root('ez_systems_udw_tab_tutorial');
// Here you should define the parameters that are allowed to
// configure your bundle. See the documentation linked above for
// more information on that topic.
return $treeBuilder;
}
}
|
EzSystemsExtendingTutorialExtension.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28 | <?php
namespace EzSystems\ExtendingTutorialBundle\DependencyInjection;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\Config\FileLocator;
use Symfony\Component\HttpKernel\DependencyInjection\Extension;
use Symfony\Component\DependencyInjection\Loader;
/**
* This is the class that loads and manages your bundle configuration.
*
* @link http://symfony.com/doc/current/cookbook/bundles/extension.html
*/
class EzSystemsExtendingTutorialExtension extends Extension
{
/**
* {@inheritdoc}
*/
public function load(array $configs, ContainerBuilder $container)
{
$configuration = new Configuration();
$config = $this->processConfiguration($configuration, $configs);
$loader = new Loader\YamlFileLoader($container, new FileLocator(__DIR__.'/../Resources/config'));
$loader->load('services.yml');
}
}
|
Check results
Tip
If you cannot see the results or encounter an error, clear the cache and reload the application.
At this point you can go to the Back Office and choose Browse under Content/Content structure.
In the UDW a new "Images" tab will appear, listing all Images from the Repository.