Passed
Push — master ( 5bd948...6fd2d1 )
by Aimeos
15:17
created

Standard::sortProperties()   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 12
Code Lines 6

Duplication

Lines 0
Ratio 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 2
eloc 6
c 1
b 0
f 0
nc 2
nop 1
dl 0
loc 12
rs 10
1
<?php
2
3
/**
4
 * @license LGPLv3, http://opensource.org/licenses/LGPL-3.0
5
 * @copyright Aimeos (aimeos.org), 2015-2023
6
 * @package Client
7
 * @subpackage Html
8
 */
9
10
11
namespace Aimeos\Client\Html\Catalog\Detail;
12
13
14
/**
15
 * Default implementation of catalog detail section HTML clients.
16
 *
17
 * @package Client
18
 * @subpackage Html
19
 */
20
class Standard
21
	extends \Aimeos\Client\Html\Catalog\Base
22
	implements \Aimeos\Client\Html\Common\Client\Factory\Iface
23
{
24
	/** client/html/catalog/detail/name
25
	 * Class name of the used catalog detail client implementation
26
	 *
27
	 * Each default HTML client can be replace by an alternative imlementation.
28
	 * To use this implementation, you have to set the last part of the class
29
	 * name as configuration value so the client factory knows which class it
30
	 * has to instantiate.
31
	 *
32
	 * For example, if the name of the default class is
33
	 *
34
	 *  \Aimeos\Client\Html\Catalog\Detail\Standard
35
	 *
36
	 * and you want to replace it with your own version named
37
	 *
38
	 *  \Aimeos\Client\Html\Catalog\Detail\Mydetail
39
	 *
40
	 * then you have to set the this configuration option:
41
	 *
42
	 *  client/html/catalog/detail/name = Mydetail
43
	 *
44
	 * The value is the last part of your own class name and it's case sensitive,
45
	 * so take care that the configuration value is exactly named like the last
46
	 * part of the class name.
47
	 *
48
	 * The allowed characters of the class name are A-Z, a-z and 0-9. No other
49
	 * characters are possible! You should always start the last part of the class
50
	 * name with an upper case character and continue only with lower case characters
51
	 * or numbers. Avoid chamel case names like "MyDetail"!
52
	 *
53
	 * @param string Last part of the class name
54
	 * @since 2014.03
55
	 */
56
57
58
	private array $tags = [];
59
	private ?string $expire = null;
60
	private ?\Aimeos\Base\View\Iface $view = null;
61
62
63
	/**
64
	 * Returns the HTML code for insertion into the body.
65
	 *
66
	 * @param string $uid Unique identifier for the output if the content is placed more than once on the same page
67
	 * @return string HTML code
68
	 */
69
	public function body( string $uid = '' ) : string
70
	{
71
		$view = $this->view();
72
		$config = $this->context()->config();
73
		$prefixes = ['d_prodid', 'd_name'];
74
75
		$code = $config->get( 'client/html/catalog/detail/prodcode-default' );
76
		$id = $config->get( 'client/html/catalog/detail/prodid-default', $code );
77
78
		if( !$view->param( 'd_name', $view->param( 'd_prodid', $id ) ) ) {
79
			return '';
80
		}
81
82
		/** client/html/catalog/detail/cache
83
		 * Enables or disables caching only for the catalog detail component
84
		 *
85
		 * Disable caching for components can be useful if you would have too much
86
		 * entries to cache or if the component contains non-cacheable parts that
87
		 * can't be replaced using the modify() method.
88
		 *
89
		 * @param boolean True to enable caching, false to disable
90
		 * @see client/html/catalog/filter/cache
91
		 * @see client/html/catalog/lists/cache
92
		 * @see client/html/catalog/stage/cache
93
		 */
94
95
		/** client/html/catalog/detail
96
		 * All parameters defined for the catalog detail component and its subparts
97
		 *
98
		 * This returns all settings related to the detail component.
99
		 * Please refer to the single settings for details.
100
		 *
101
		 * @param array Associative list of name/value settings
102
		 * @see client/html/catalog#detail
103
		 */
104
		$confkey = 'client/html/catalog/detail';
105
106
		if( $html = $this->cached( 'body', $uid, $prefixes, $confkey ) ) {
107
			return $this->modify( $html, $uid );
108
		}
109
110
		/** client/html/catalog/detail/template-body
111
		 * Relative path to the HTML body template of the catalog detail client.
112
		 *
113
		 * The template file contains the HTML code and processing instructions
114
		 * to generate the result shown in the body of the frontend. The
115
		 * configuration string is the path to the template file relative
116
		 * to the templates directory (usually in templates/client/html).
117
		 *
118
		 * You can overwrite the template file configuration in extensions and
119
		 * provide alternative templates. These alternative templates should be
120
		 * named like the default one but suffixed by
121
		 * an unique name. You may use the name of your project for this. If
122
		 * you've implemented an alternative client class as well, it
123
		 * should be suffixed by the name of the new class.
124
		 *
125
		 * @param string Relative path to the template creating code for the HTML page body
126
		 * @since 2014.03
127
		 * @see client/html/catalog/detail/template-header
128
		 * @see client/html/catalog/detail/404
129
		 */
130
		$template = $config->get( 'client/html/catalog/detail/template-body', 'catalog/detail/body' );
131
132
		$view = $this->view = $this->view ?? $this->object()->data( $view, $this->tags, $this->expire );
133
		$html = $this->modify( $view->render( $template ), $uid );
134
135
		return $this->cache( 'body', $uid, $prefixes, $confkey, $html, $this->tags, $this->expire );
136
	}
137
138
139
	/**
140
	 * Returns the HTML string for insertion into the header.
141
	 *
142
	 * @param string $uid Unique identifier for the output if the content is placed more than once on the same page
143
	 * @return string|null String including HTML tags for the header on error
144
	 */
145
	public function header( string $uid = '' ) : ?string
146
	{
147
		$view = $this->view();
148
		$config = $this->context()->config();
149
		$prefixes = ['d_prodid', 'd_name'];
150
		$confkey = 'client/html/catalog/detail';
151
152
		$code = $config->get( 'client/html/catalog/detail/prodcode-default' );
153
		$id = $config->get( 'client/html/catalog/detail/prodid-default', $code );
154
155
		if( !$view->param( 'd_name', $view->param( 'd_prodid', $id ) ) ) {
156
			return '';
157
		}
158
159
		if( $html = $this->cached( 'header', $uid, $prefixes, $confkey ) ) {
160
			return $this->modify( $html, $uid );
161
		}
162
163
		/** client/html/catalog/detail/template-header
164
		 * Relative path to the HTML header template of the catalog detail client.
165
		 *
166
		 * The template file contains the HTML code and processing instructions
167
		 * to generate the HTML code that is inserted into the HTML page header
168
		 * of the rendered page in the frontend. The configuration string is the
169
		 * path to the template file relative to the templates directory (usually
170
		 * in templates/client/html).
171
		 *
172
		 * You can overwrite the template file configuration in extensions and
173
		 * provide alternative templates. These alternative templates should be
174
		 * named like the default one but suffixed by
175
		 * an unique name. You may use the name of your project for this. If
176
		 * you've implemented an alternative client class as well, it
177
		 * should be suffixed by the name of the new class.
178
		 *
179
		 * @param string Relative path to the template creating code for the HTML page head
180
		 * @since 2014.03
181
		 * @see client/html/catalog/detail/template-body
182
		 * @see client/html/catalog/detail/404
183
		 */
184
		$template = $config->get( 'client/html/catalog/detail/template-header', 'catalog/detail/header' );
185
186
		$view = $this->view = $this->view ?? $this->object()->data( $this->view(), $this->tags, $this->expire );
187
		$html = $view->render( $template );
188
189
		return $this->cache( 'header', $uid, $prefixes, $confkey, $html, $this->tags, $this->expire );
190
	}
191
192
193
	/**
194
	 * Modifies the cached content to replace content based on sessions or cookies.
195
	 *
196
	 * @param string $content Cached content
197
	 * @param string $uid Unique identifier for the output if the content is placed more than once on the same page
198
	 * @return string Modified content
199
	 */
200
	public function modify( string $content, string $uid ) : string
201
	{
202
		$content = $this->replaceSection( $content, $this->navigator(), 'catalog.detail.navigator' );
203
		return $this->replaceSection( $content, $this->view()->csrf()->formfield(), 'catalog.detail.csrf' );
204
	}
205
206
207
	/**
208
	 * Processes the input, e.g. store given values.
209
	 *
210
	 * A view must be available and this method doesn't generate any output
211
	 * besides setting view variables if necessary.
212
	 */
213
	public function init()
214
	{
215
		$view = $this->view();
216
		$context = $this->context();
217
		$session = $context->session();
218
219
		$params = $view->param();
220
		$site = $context->locale()->getSiteItem()->getCode();
221
222
		$session->set( 'aimeos/catalog/last/' . $site, $view->link( 'client/html/catalog/detail/url', $params ) );
223
	}
224
225
226
	/**
227
	 * Sets the necessary parameter values in the view.
228
	 *
229
	 * @param \Aimeos\Base\View\Iface $view The view object which generates the HTML output
230
	 * @param array &$tags Result array for the list of tags that are associated to the output
231
	 * @param string|null &$expire Result variable for the expiration date of the output (null for no expiry)
232
	 * @return \Aimeos\Base\View\Iface Modified view object
233
	 */
234
	public function data( \Aimeos\Base\View\Iface $view, array &$tags = [], string &$expire = null ) : \Aimeos\Base\View\Iface
235
	{
236
		$productItem = $this->product( $view );
237
238
		$propItems = $productItem->getPropertyItems();
239
		$attrItems = $productItem->getRefItems( 'attribute', null, 'default' );
240
		$mediaItems = $productItem->getRefItems( 'media', 'default', 'default' );
241
242
		$this->addMetaItems( $productItem, $expire, $tags, ['product'] );
243
244
		if( in_array( $productItem->getType(), ['bundle', 'select'] ) )
245
		{
246
			\Aimeos\Map::method( 'attrparent', function( $subProdId ) {
247
				foreach( $this->list as $item ) {
0 ignored issues
show
Bug Best Practice introduced by
The property list does not exist on Aimeos\Client\Html\Catalog\Detail\Standard. Did you maybe forget to declare it?
Loading history...
248
					$item->parent = array_merge( $item->parent ?? [], [$subProdId] );
249
				}
250
				return $this;
251
			} );
252
253
			foreach( $productItem->getRefItems( 'product', null, 'default' ) as $subProdId => $subProduct )
254
			{
255
				$propItems->merge( $subProduct->getPropertyItems()->assign( ['parent' => $subProdId] ) );
256
				$mediaItems->merge( $subProduct->getRefItems( 'media', 'default', 'default' ) );
257
				$attrItems->replace(
258
					$subProduct->getRefItems( 'attribute', null, ['default', 'variant'] )->attrparent( $subProdId )
259
				);
260
			}
261
		}
262
263
		$view->detailMediaItems = $mediaItems;
264
		$view->detailProductItem = $productItem;
265
		$view->detailAttributeMap = $this->sortAttributes( $attrItems->groupBy( 'attribute.type' ) );
266
		$view->detailPropertyMap = $this->sortProperties( $propItems->groupBy( 'product.property.type' ) );
267
		$view->detailStockTypes = $productItem->getStockItems()->getType();
268
		$view->detailStockUrl = $this->stockUrl( $productItem );
269
270
		$this->call( 'seen', $productItem );
271
272
		return parent::data( $view, $tags, $expire );
273
	}
274
275
276
	/**
277
	 * Returns the data domains fetched along with the products
278
	 *
279
	 * @return array List of domain names
280
	 */
281
	protected function domains() : array
282
	{
283
		$context = $this->context();
284
		$config = $context->config();
285
286
		$domains = [
287
			'attribute', 'attribute/property', 'catalog', 'media', 'media/property', 'price',
288
			'product', 'product/property', 'supplier', 'supplier/address', 'text', 'stock'
289
		];
290
291
		/** client/html/catalog/domains
292
		 * A list of domain names whose items should be available in the catalog view templates
293
		 *
294
		 * @see client/html/catalog/detail/domains
295
		 */
296
		$domains = $config->get( 'client/html/catalog/domains', $domains );
297
298
		/** client/html/catalog/detail/domains
299
		 * A list of domain names whose items should be available in the product detail view template
300
		 *
301
		 * The templates rendering product details usually add the images,
302
		 * prices, texts, attributes, products, etc. associated to the product
303
		 * item. If you want to display additional or less content, you can
304
		 * configure your own list of domains (attribute, media, price, product,
305
		 * text, etc. are domains) whose items are fetched from the storage.
306
		 * Please keep in mind that the more domains you add to the configuration,
307
		 * the more time is required for fetching the content!
308
		 *
309
		 * Since version 2014.05 this configuration option overwrites the
310
		 * "client/html/catalog/domains" option that allows to configure the
311
		 * domain names of the items fetched for all catalog related data.
312
		 *
313
		 * @param array List of domain names
314
		 * @since 2014.03
315
		 * @see client/html/catalog/domains
316
		 * @see client/html/catalog/lists/domains
317
		 */
318
		$domains = $config->get( 'client/html/catalog/detail/domains', $domains );
319
320
		return $domains;
321
	}
322
323
324
	/**
325
	 * Sets the necessary parameter values in the view.
326
	 *
327
	 * @param \Aimeos\Base\View\Iface $view The view object which generates the HTML output
328
	 * @param array &$tags Result array for the list of tags that are associated to the output
329
	 * @param string|null &$expire Result variable for the expiration date of the output (null for no expiry)
330
	 * @return \Aimeos\Base\View\Iface Modified view object
331
	 */
332
	public function navigator() : string
333
	{
334
		$view = $this->view();
335
		$context = $this->context();
336
337
		if( is_numeric( $pos = $view->param( 'd_pos' ) ) )
338
		{
339
			if( $pos < 1 ) {
340
				$pos = $start = 0; $size = 2;
341
			} else {
342
				$start = $pos - 1; $size = 3;
343
			}
344
345
			$site = $context->locale()->getSiteItem()->getCode();
346
			$params = $context->session()->get( 'aimeos/catalog/lists/params/last/' . $site, [] );
347
			$level = $view->config( 'client/html/catalog/lists/levels', \Aimeos\MW\Tree\Manager\Base::LEVEL_ONE );
348
349
			$catids = $view->value( $params, 'f_catid', $view->config( 'client/html/catalog/lists/catid-default' ) );
350
			$sort = $view->value( $params, 'f_sort', $view->config( 'client/html/catalog/lists/sort', 'relevance' ) );
351
352
			$products = \Aimeos\Controller\Frontend::create( $context, 'product' )
353
				->sort( $sort ) // prioritize user sorting over the sorting through relevance and category
354
				->allOf( $view->value( $params, 'f_attrid', [] ) )
355
				->oneOf( $view->value( $params, 'f_optid', [] ) )
356
				->oneOf( $view->value( $params, 'f_oneid', [] ) )
357
				->text( $view->value( $params, 'f_search' ) )
358
				->category( $catids, 'default', $level )
359
				->slice( $start, $size )
360
				->uses( ['text'] )
361
				->search();
362
363
			if( ( $count = count( $products ) ) > 1 )
364
			{
365
				if( $pos > 0 && ( $product = $products->first() ) !== null )
366
				{
367
					$param = ['d_name' => $product->getName( 'url' ), 'd_prodid' => $product->getId(), 'd_pos' => $pos - 1];
368
					$view->navigationPrev = $view->link( 'client/html/catalog/detail/url', $param );
369
				}
370
371
				if( ( $pos === 0 || $count === 3 ) && ( $product = $products->last() ) !== null )
372
				{
373
					$param = ['d_name' => $product->getName( 'url' ), 'd_prodid' => $product->getId(), 'd_pos' => $pos + 1];
374
					$view->navigationNext = $view->link( 'client/html/catalog/detail/url', $param );
375
				}
376
			}
377
378
			/** client/html/catalog/detail/template-navigator
379
			 * Relative path to the HTML template of the catalog detail navigator partial.
380
			 *
381
			 * The template file contains the HTML code and processing instructions
382
			 * to generate the result shown in the body of the frontend. The
383
			 * configuration string is the path to the template file relative
384
			 * to the templates directory (usually in templates/client/html).
385
			 *
386
			 * You can overwrite the template file configuration in extensions and
387
			 * provide alternative templates. These alternative templates should be
388
			 * named like the default one but suffixed by
389
			 * an unique name. You may use the name of your project for this. If
390
			 * you've implemented an alternative client class as well, it
391
			 * should be suffixed by the name of the new class.
392
			 *
393
			 * @param string Relative path to the template creating the HTML fragment
394
			 * @since 2022.10
395
			 */
396
			$template = $context->config()->get( 'client/html/catalog/detail/template-navigator', 'catalog/detail/navigator' );
397
398
			return $view->render( $template );
399
		}
400
401
		return '';
402
	}
403
404
405
	/**
406
	 * Returns the product item
407
	 *
408
	 * @param \Aimeos\Base\View\Iface $view The view object which generates the HTML output
409
	 * @return \Aimeos\MShop\Product\Item\Iface Product item
410
	 */
411
	protected function product( \Aimeos\Base\View\Iface $view ) : \Aimeos\MShop\Product\Item\Iface
412
	{
413
		$context = $this->context();
414
		$config = $context->config();
415
416
		/** client/html/catalog/detail/prodid-default
417
		 * The default product ID used if none is given as parameter
418
		 *
419
		 * To display a product detail view or a part of it for a specific
420
		 * product, you can configure its ID using this setting. This is
421
		 * most useful in a CMS where the product ID can be configured
422
		 * separately for each content node.
423
		 *
424
		 * @param string Product ID
425
		 * @since 2016.01
426
		 * @see client/html/catalog/detail/prodid-default
427
		 * @see client/html/catalog/lists/catid-default
428
		 */
429
		$id = $view->param( 'd_prodid', $config->get( 'client/html/catalog/detail/prodid-default' ) );
430
431
		/** client/html/catalog/detail/prodcode-default
432
		 * The default product code used if none is given as parameter
433
		 *
434
		 * To display a product detail view or a part of it for a specific
435
		 * product, you can configure its code using this setting. This is
436
		 * most useful in a CMS where the product code can be configured
437
		 * separately for each content node.
438
		 *
439
		 * @param string Product code
440
		 * @since 2019.10
441
		 * @see client/html/catalog/detail/prodid-default
442
		 * @see client/html/catalog/lists/catid-default
443
		 */
444
		$code = $config->get( 'client/html/catalog/detail/prodcode-default' );
445
446
		$cntl = \Aimeos\Controller\Frontend::create( $context, 'product' )->uses( $this->domains() );
447
		return ( $id ? $cntl->get( $id ) : ( $code ? $cntl->find( $code ) : $cntl->resolve( $view->param( 'd_name' ) ) ) );
448
	}
449
450
451
	/**
452
	 * Adds the product to the list of last seen products.
453
	 *
454
	 * @param \Aimeos\MShop\Product\Item\Iface $product Product item
455
	 */
456
	protected function seen( \Aimeos\MShop\Product\Item\Iface $product )
457
	{
458
		$id = $product->getId();
459
		$context = $this->context();
460
		$session = $context->session();
461
		$lastSeen = map( $session->get( 'aimeos/catalog/session/seen/list', [] ) );
462
463
		if( !$lastSeen->has( $id ) )
464
		{
465
			$config = $context->config();
466
467
			/** client/html/catalog/detail/partials/seen
468
			 * Relative path to the HTML template of the catalog detail seen partial.
469
			 *
470
			 * The template file contains the HTML code and processing instructions
471
			 * to generate the result shown in the body of the frontend. The
472
			 * configuration string is the path to the template file relative
473
			 * to the templates directory (usually in templates/client/html).
474
			 *
475
			 * You can overwrite the template file configuration in extensions and
476
			 * provide alternative templates. These alternative templates should be
477
			 * named like the default one but suffixed by
478
			 * an unique name. You may use the name of your project for this. If
479
			 * you've implemented an alternative client class as well, it
480
			 * should be suffixed by the name of the new class.
481
			 *
482
			 * @param string Relative path to the template creating the HTML fragment
483
			 * @since 2014.03
484
			 */
485
			$template = $config->get( 'client/html/catalog/detail/partials/seen', 'catalog/detail/seen' );
486
487
			/** client/html/catalog/session/seen/maxitems
488
			 * Maximum number of products displayed in the "last seen" section
489
			 *
490
			 * This option limits the number of products that are shown in the
491
			 * "last seen" section after the user visited their detail pages. It
492
			 * must be a positive integer value greater than 0.
493
			 *
494
			 * @param integer Number of products
495
			 * @since 2014.03
496
			 */
497
			$max = $config->get( 'client/html/catalog/session/seen/maxitems', 6 );
498
499
			$html = $this->view()->set( 'product', $product )->render( $template );
500
			$lastSeen = $lastSeen->put( $id, $html )->slice( -$max );
501
		}
502
503
		$session->set( 'aimeos/catalog/session/seen/list', $lastSeen->put( $id, $lastSeen->pull( $id ) )->all() );
504
	}
505
506
507
	/**
508
	 * Sorts the attribute map by type
509
	 *
510
	 * @param \Aimeos\Map $map Attribute item map with types as keys
511
	 * @return \Aimeos\Map Sorted attribute map
512
	 */
513
	protected function sortAttributes( \Aimeos\Map $map ) : \Aimeos\Map
514
	{
515
		$sorted = [];
516
517
		$manager = \Aimeos\MShop::create( $this->context(), 'attribute/type' );
518
		$filter = $manager->filter( true )
519
			->add( 'attribute.type.domain', '==', 'product' )
520
			->add( 'attribute.type.code', '==', $map->keys() )
521
			->order( 'attribute.type.position' );
522
523
		foreach( $manager->search( $filter->slice( 0, 10000 ) ) as $typeItem )
524
		{
525
			$list = $map[$typeItem->getCode()];
526
527
			uasort( $list, function( $a, $b ) {
0 ignored issues
show
Bug introduced by
It seems like $list can also be of type null; however, parameter $array of uasort() does only seem to accept array, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

527
			uasort( /** @scrutinizer ignore-type */ $list, function( $a, $b ) {
Loading history...
528
				return $a->getPosition() <=> $b->getPosition();
529
			} );
530
531
			$sorted[$typeItem->getCode()] = $list;
532
		}
533
534
		return map( $sorted );
535
	}
536
537
538
	/**
539
	 * Sorts the property map by type
540
	 *
541
	 * @param \Aimeos\Map $map Property item map with types as keys
542
	 * @return \Aimeos\Map Sorted property map
543
	 */
544
	protected function sortProperties( \Aimeos\Map $map ) : \Aimeos\Map
545
	{
546
		$sorted = [];
547
548
		$manager = \Aimeos\MShop::create( $this->context(), 'product/property/type' );
549
		$filter = $manager->filter( true )->add( 'product.property.type.code', '==', $map->keys() );
550
551
		foreach( $manager->search( $filter->slice( 0, 10000 ) ) as $typeItem ) {
552
			$sorted[$typeItem->getCode()] = $map[$typeItem->getCode()];
553
		}
554
555
		return map( $sorted );
556
	}
557
558
559
	/**
560
	 * Returns the URL for fetching stock levels
561
	 *
562
	 * @param \Aimeos\MShop\Product\Item\Iface $product Product item
563
	 * @return \Aimeos\Map List of stock URLs
564
	 */
565
	protected function stockUrl( \Aimeos\MShop\Product\Item\Iface $productItem ) : \Aimeos\Map
566
	{
567
		/** client/html/catalog/detail/stock/enable
568
		 * Enables or disables displaying product stock levels in product detail view
569
		 *
570
		 * This configuration option allows shop owners to display product
571
		 * stock levels for each product in the detail views or to disable
572
		 * fetching product stock information.
573
		 *
574
		 * The stock information is fetched via AJAX and inserted via Javascript.
575
		 * This allows to cache product items by leaving out such highly
576
		 * dynamic content like stock levels which changes with each order.
577
		 *
578
		 * @param boolean Value of "1" to display stock levels, "0" to disable displaying them
579
		 * @since 2014.03
580
		 * @see client/html/catalog/lists/stock/enable
581
		 * @see client/html/catalog/stock/url/target
582
		 * @see client/html/catalog/stock/url/controller
583
		 * @see client/html/catalog/stock/url/action
584
		 * @see client/html/catalog/stock/url/config
585
		 */
586
		if( !$this->context()->config()->get( 'client/html/catalog/detail/stock/enable', true ) ) {
587
			return map();
588
		}
589
590
		$products = $productItem->getRefItems( 'product', null, 'default' )->push( $productItem );
591
592
		return $this->getStockUrl( $this->view(), $products );
593
	}
594
595
596
	/** client/html/catalog/detail/decorators/excludes
597
	 * Excludes decorators added by the "common" option from the catalog detail html client
598
	 *
599
	 * Decorators extend the functionality of a class by adding new aspects
600
	 * (e.g. log what is currently done), executing the methods of the underlying
601
	 * class only in certain conditions (e.g. only for logged in users) or
602
	 * modify what is returned to the caller.
603
	 *
604
	 * This option allows you to remove a decorator added via
605
	 * "client/html/common/decorators/default" before they are wrapped
606
	 * around the html client.
607
	 *
608
	 *  client/html/catalog/detail/decorators/excludes = array( 'decorator1' )
609
	 *
610
	 * This would remove the decorator named "decorator1" from the list of
611
	 * common decorators ("\Aimeos\Client\Html\Common\Decorator\*") added via
612
	 * "client/html/common/decorators/default" to the html client.
613
	 *
614
	 * @param array List of decorator names
615
	 * @see client/html/common/decorators/default
616
	 * @see client/html/catalog/detail/decorators/global
617
	 * @see client/html/catalog/detail/decorators/local
618
	 */
619
620
	/** client/html/catalog/detail/decorators/global
621
	 * Adds a list of globally available decorators only to the catalog detail html client
622
	 *
623
	 * Decorators extend the functionality of a class by adding new aspects
624
	 * (e.g. log what is currently done), executing the methods of the underlying
625
	 * class only in certain conditions (e.g. only for logged in users) or
626
	 * modify what is returned to the caller.
627
	 *
628
	 * This option allows you to wrap global decorators
629
	 * ("\Aimeos\Client\Html\Common\Decorator\*") around the html client.
630
	 *
631
	 *  client/html/catalog/detail/decorators/global = array( 'decorator1' )
632
	 *
633
	 * This would add the decorator named "decorator1" defined by
634
	 * "\Aimeos\Client\Html\Common\Decorator\Decorator1" only to the html client.
635
	 *
636
	 * @param array List of decorator names
637
	 * @see client/html/common/decorators/default
638
	 * @see client/html/catalog/detail/decorators/excludes
639
	 * @see client/html/catalog/detail/decorators/local
640
	 */
641
642
	/** client/html/catalog/detail/decorators/local
643
	 * Adds a list of local decorators only to the catalog detail html client
644
	 *
645
	 * Decorators extend the functionality of a class by adding new aspects
646
	 * (e.g. log what is currently done), executing the methods of the underlying
647
	 * class only in certain conditions (e.g. only for logged in users) or
648
	 * modify what is returned to the caller.
649
	 *
650
	 * This option allows you to wrap local decorators
651
	 * ("\Aimeos\Client\Html\Catalog\Decorator\*") around the html client.
652
	 *
653
	 *  client/html/catalog/detail/decorators/local = array( 'decorator2' )
654
	 *
655
	 * This would add the decorator named "decorator2" defined by
656
	 * "\Aimeos\Client\Html\Catalog\Decorator\Decorator2" only to the html client.
657
	 *
658
	 * @param array List of decorator names
659
	 * @see client/html/common/decorators/default
660
	 * @see client/html/catalog/detail/decorators/excludes
661
	 * @see client/html/catalog/detail/decorators/global
662
	 */
663
}
664