Passed
Push — master ( 863c7f...5d3493 )
by Aimeos
03:44
created

Standard.php$0 ➔ type()   A

Complexity

Conditions 1

Size

Total Lines 3

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 1
dl 0
loc 3
rs 10
c 0
b 0
f 0
1
<?php
2
3
/**
4
 * @license LGPLv3, http://opensource.org/licenses/LGPL-3.0
5
 * @copyright Aimeos (aimeos.org), 2018-2024
6
 * @package Controller
7
 * @subpackage Order
8
 */
9
10
11
namespace Aimeos\Controller\Jobs\Order\Email\Voucher;
12
13
14
/**
15
 * Order voucher e-mail job controller.
16
 *
17
 * @package Controller
18
 * @subpackage Order
19
 */
20
class Standard
21
	extends \Aimeos\Controller\Jobs\Base
22
	implements \Aimeos\Controller\Jobs\Iface
23
{
24
	/** controller/jobs/order/email/voucher/name
25
	 * Class name of the used order email voucher scheduler controller implementation
26
	 *
27
	 * Each default job controller 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 controller factory knows which class it
30
	 * has to instantiate.
31
	 *
32
	 * For example, if the name of the default class is
33
	 *
34
	 *  \Aimeos\Controller\Jobs\Order\Email\Voucher\Standard
35
	 *
36
	 * and you want to replace it with your own version named
37
	 *
38
	 *  \Aimeos\Controller\Jobs\Order\Email\Voucher\Myvoucher
39
	 *
40
	 * then you have to set the this configuration option:
41
	 *
42
	 *  controller/jobs/order/email/voucher/name = Myvoucher
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 "MyVoucher"!
52
	 *
53
	 * @param string Last part of the class name
54
	 * @since 2014.03
55
	 */
56
57
	/** controller/jobs/order/email/voucher/decorators/excludes
58
	 * Excludes decorators added by the "common" option from the order email voucher controllers
59
	 *
60
	 * Decorators extend the functionality of a class by adding new aspects
61
	 * (e.g. log what is currently done), executing the methods of the underlying
62
	 * class only in certain conditions (e.g. only for logged in users) or
63
	 * modify what is returned to the caller.
64
	 *
65
	 * This option allows you to remove a decorator added via
66
	 * "controller/jobs/common/decorators/default" before they are wrapped
67
	 * around the job controller.
68
	 *
69
	 *  controller/jobs/order/email/voucher/decorators/excludes = array( 'decorator1' )
70
	 *
71
	 * This would remove the decorator named "decorator1" from the list of
72
	 * common decorators ("\Aimeos\Controller\Jobs\Common\Decorator\*") added via
73
	 * "controller/jobs/common/decorators/default" to this job controller.
74
	 *
75
	 * @param array List of decorator names
76
	 * @since 2015.09
77
	 * @see controller/jobs/common/decorators/default
78
	 * @see controller/jobs/order/email/voucher/decorators/global
79
	 * @see controller/jobs/order/email/voucher/decorators/local
80
	 */
81
82
	/** controller/jobs/order/email/voucher/decorators/global
83
	 * Adds a list of globally available decorators only to the order email voucher controllers
84
	 *
85
	 * Decorators extend the functionality of a class by adding new aspects
86
	 * (e.g. log what is currently done), executing the methods of the underlying
87
	 * class only in certain conditions (e.g. only for logged in users) or
88
	 * modify what is returned to the caller.
89
	 *
90
	 * This option allows you to wrap global decorators
91
	 * ("\Aimeos\Controller\Jobs\Common\Decorator\*") around the job controller.
92
	 *
93
	 *  controller/jobs/order/email/voucher/decorators/global = array( 'decorator1' )
94
	 *
95
	 * This would add the decorator named "decorator1" defined by
96
	 * "\Aimeos\Controller\Jobs\Common\Decorator\Decorator1" only to this job controller.
97
	 *
98
	 * @param array List of decorator names
99
	 * @since 2015.09
100
	 * @see controller/jobs/common/decorators/default
101
	 * @see controller/jobs/order/email/voucher/decorators/excludes
102
	 * @see controller/jobs/order/email/voucher/decorators/local
103
	 */
104
105
	/** controller/jobs/order/email/voucher/decorators/local
106
	 * Adds a list of local decorators only to the order email voucher controllers
107
	 *
108
	 * Decorators extend the functionality of a class by adding new aspects
109
	 * (e.g. log what is currently done), executing the methods of the underlying
110
	 * class only in certain conditions (e.g. only for logged in users) or
111
	 * modify what is returned to the caller.
112
	 *
113
	 * This option allows you to wrap local decorators
114
	 * ("\Aimeos\Controller\Jobs\Order\Email\Voucher\Decorator\*") around this job controller.
115
	 *
116
	 *  controller/jobs/order/email/voucher/decorators/local = array( 'decorator2' )
117
	 *
118
	 * This would add the decorator named "decorator2" defined by
119
	 * "\Aimeos\Controller\Jobs\Order\Email\Voucher\Decorator\Decorator2" only to this job
120
	 * controller.
121
	 *
122
	 * @param array List of decorator names
123
	 * @since 2015.09
124
	 * @see controller/jobs/common/decorators/default
125
	 * @see controller/jobs/order/email/voucher/decorators/excludes
126
	 * @see controller/jobs/order/email/voucher/decorators/global
127
	 */
128
129
130
	use \Aimeos\Controller\Jobs\Mail;
131
132
133
	private ?string $couponId = null;
134
135
136
	/**
137
	 * Returns the localized name of the job.
138
	 *
139
	 * @return string Name of the job
140
	 */
141
	public function getName() : string
142
	{
143
		return $this->context()->translate( 'controller/jobs', 'Voucher related e-mails' );
144
	}
145
146
147
	/**
148
	 * Returns the localized description of the job.
149
	 *
150
	 * @return string Description of the job
151
	 */
152
	public function getDescription() : string
153
	{
154
		return $this->context()->translate( 'controller/jobs', 'Sends the e-mail with the voucher to the customer' );
155
	}
156
157
158
	/**
159
	 * Executes the job.
160
	 *
161
	 * @throws \Aimeos\Controller\Jobs\Exception If an error occurs
162
	 */
163
	public function run()
164
	{
165
		$context = $this->context();
166
		$manager = \Aimeos\MShop::create( $context, 'order' );
167
168
		$filter = $this->filter( $manager->filter() );
169
		$cursor = $manager->cursor( $filter );
170
171
		while( $items = $manager->iterate( $cursor, ['order/address', 'order/product'] ) ) {
172
			$this->notify( $items );
173
		}
174
	}
175
176
177
	/**
178
	 * Returns the delivery address item of the order
179
	 *
180
	 * @param \Aimeos\MShop\Order\Item\Iface $orderBaseItem Order including address items
181
	 * @return \Aimeos\MShop\Order\Item\Address\Iface Delivery or voucher address item
182
	 * @throws \Aimeos\Controller\Jobs\Exception If no address item is available
183
	 */
184
	protected function address( \Aimeos\MShop\Order\Item\Iface $orderBaseItem ) : \Aimeos\MShop\Order\Item\Address\Iface
185
	{
186
		$type = \Aimeos\MShop\Order\Item\Address\Base::TYPE_DELIVERY;
187
		if( ( $addr = current( $orderBaseItem->getAddress( $type ) ) ) !== false && $addr->getEmail() !== '' ) {
188
			return $addr;
189
		}
190
191
		$type = \Aimeos\MShop\Order\Item\Address\Base::TYPE_PAYMENT;
192
		if( ( $addr = current( $orderBaseItem->getAddress( $type ) ) ) !== false && $addr->getEmail() !== '' ) {
193
			return $addr;
194
		}
195
196
		$msg = sprintf( 'No address with e-mail found in order base with ID "%1$s"', $orderBaseItem->getId() );
197
		throw new \Aimeos\Controller\Jobs\Exception( $msg );
198
	}
199
200
201
	/**
202
	 * Creates coupon codes for the bought vouchers
203
	 *
204
	 * @param \Aimeos\Map $orderProdItems Complete order including addresses, products, services
205
	 */
206
	protected function createCoupons( \Aimeos\Map $orderProdItems )
207
	{
208
		$map = [];
209
		$manager = \Aimeos\MShop::create( $this->context(), 'order/product/attribute' );
210
211
		foreach( $orderProdItems as $orderProductItem )
212
		{
213
			if( $orderProductItem->getAttribute( 'coupon-code', 'coupon' ) ) {
214
				continue;
215
			}
216
217
			$codes = [];
218
219
			for( $i = 0; $i < $orderProductItem->getQuantity(); $i++ )
220
			{
221
				$str = $i . getmypid() . microtime( true ) . $orderProductItem->getId();
222
				$code = substr( strtoupper( sha1( $str ) ), -8 );
223
				$map[$code] = $orderProductItem->getId();
224
				$codes[] = $code;
225
			}
226
227
			$item = $manager->create()->setCode( 'coupon-code' )->setType( 'coupon' )->setValue( $codes );
228
			$orderProductItem->setAttributeItem( $item );
229
		}
230
231
		$this->saveCoupons( $map );
232
		return $orderProdItems;
233
	}
234
235
236
	/**
237
	 * Returns the coupon ID for the voucher coupon
238
	 *
239
	 * @return string Unique ID of the coupon item
240
	 */
241
	protected function couponId() : string
242
	{
243
		if( !isset( $this->couponId ) )
244
		{
245
			$manager = \Aimeos\MShop::create( $this->context(), 'coupon' );
246
			$filter = $manager->filter()->add( 'coupon.provider', '=~', 'Voucher' )->slice( 0, 1 );
247
248
			if( ( $item = $manager->search( $filter )->first() ) === null ) {
249
				throw new \Aimeos\Controller\Jobs\Exception( 'No coupon provider "Voucher" available' );
250
			}
251
252
			$this->couponId = $item->getId();
253
		}
254
255
		return $this->couponId;
256
	}
257
258
259
	/**
260
	 * Returns the PDF file name
261
	 *
262
	 * @param string $code Voucher code
263
	 * @return string PDF file name
264
	 */
265
	protected function filename( string $code ) : string
266
	{
267
		return $this->context()->translate( 'controller/jobs', 'Voucher' ) . '-' . $code . '.pdf';
268
	}
269
270
271
	/**
272
	 * Returns the filter for searching the appropriate orders
273
	 *
274
	 * @param \Aimeos\Base\Criteria\Iface $filter Order filter object
275
	 * @return \Aimeos\Base\Criteria\Iface Filter object with conditions set
276
	 */
277
	protected function filter( \Aimeos\Base\Criteria\Iface $filter ) : \Aimeos\Base\Criteria\Iface
278
	{
279
		$limitDate = date( 'Y-m-d H:i:s', time() - $this->limit() * 86400 );
280
281
		$filter->add( $filter->and( [
282
			$filter->compare( '>=', 'order.mtime', $limitDate ),
283
			$filter->compare( '==', 'order.statuspayment', $this->status() ),
284
			$filter->compare( '==', 'order.product.type', 'voucher' ),
285
			$filter->compare( '==', $filter->make( 'order:status', [$this->type(), '1'] ), 0 ),
286
		] ) );
287
288
289
		return $filter;
290
	}
291
292
293
	/**
294
	 * Returns the number of days after no e-mail will be sent anymore
295
	 *
296
	 * @return int Number of days
297
	 */
298
	protected function limit() : int
299
	{
300
		/** controller/jobs/order/email/voucher/limit-days
301
		 * Only send voucher e-mails of orders that were created in the past within the configured number of days
302
		 *
303
		 * The voucher e-mails are normally send immediately after the voucher
304
		 * status has changed. This option prevents e-mails for old order from
305
		 * being send in case anything went wrong or an update failed to avoid
306
		 * confusion of customers.
307
		 *
308
		 * @param integer Number of days
309
		 * @since 2014.03
310
		 * @see controller/jobs/order/email/delivery/limit-days
311
		 * @see controller/jobs/service/delivery/process/limit-days
312
		 */
313
		return (int) $this->context()->config()->get( 'controller/jobs/order/email/voucher/limit-days', 30 );
314
	}
315
316
317
	/**
318
	 * Sends the voucher e-mail for the given orders
319
	 *
320
	 * @param \Aimeos\Map $items List of order items implementing \Aimeos\MShop\Order\Item\Iface with their IDs as keys
321
	 */
322
	protected function notify( \Aimeos\Map $items )
323
	{
324
		$context = $this->context();
325
		$sites = $this->sites( $items->getSiteId()->unique() );
326
327
		$couponManager = \Aimeos\MShop::create( $context, 'coupon' );
328
		$orderProdManager = \Aimeos\MShop::create( $context, 'order/product' );
329
330
		foreach( $items as $id => $item )
331
		{
332
			$couponManager->begin();
333
			$orderProdManager->begin();
334
335
			try
336
			{
337
				$products = $this->products( $item );
338
				$orderProdManager->save( $this->createCoupons( $products ) );
339
340
				$addr = $this->address( $item );
341
				$context->locale()->setLanguageId( $addr->getLanguageId() );
342
343
				$list = $sites->get( $item->getSiteId(), map() );
344
				$view = $this->view( $item, $list->getTheme()->filter()->last() );
345
346
				$this->send( $view, $products, $addr, $list->getLogo()->filter()->last() );
347
				$this->update( $id );
348
349
				$orderProdManager->commit();
350
				$couponManager->commit();
351
352
				$str = sprintf( 'Sent voucher e-mails for order ID "%1$s"', $item->getId() );
353
				$context->logger()->info( $str, 'email/order/voucher' );
354
			}
355
			catch( \Exception $e )
356
			{
357
				$orderProdManager->rollback();
358
				$couponManager->rollback();
359
360
				$str = 'Error while trying to send voucher e-mails for order ID "%1$s": %2$s';
361
				$msg = sprintf( $str, $item->getId(), $e->getMessage() . PHP_EOL . $e->getTraceAsString() );
362
				$context->logger()->info( $msg, 'email/order/voucher' );
363
			}
364
		}
365
	}
366
367
368
	/**
369
	 * Returns the generated PDF file for the order
370
	 *
371
	 * @param \Aimeos\Base\View\Iface $view View object with address and order item assigned
372
	 * @return string|null PDF content or NULL for no PDF file
373
	 */
374
	protected function pdf( \Aimeos\Base\View\Iface $view ) : ?string
375
	{
376
		$config = $this->context()->config();
377
378
		/** controller/jobs/order/email/voucher/pdf
379
		 * Enables attaching a PDF to the voucher e-mail
380
		 *
381
		 * The voucher PDF contains the same information like the HTML e-mail.
382
		 *
383
		 * @param bool TRUE to enable attaching the PDF, FALSE to skip the PDF
384
		 * @since 2022.10
385
		 */
386
		if( !$config->get( 'controller/jobs/order/email/voucher/pdf', true ) ) {
387
			return null;
388
		}
389
390
		$pdf = new class( PDF_PAGE_ORIENTATION, PDF_UNIT, PDF_PAGE_FORMAT, true, 'UTF-8', false ) extends \TCPDF {
391
			private ?\Closure $headerFcn = null;
392
			private ?\Closure $footerFcn = null;
393
394
			public function Footer() { return ( $fcn = $this->footerFcn ) ? $fcn( $this ) : null; }
395
			public function Header() { return ( $fcn = $this->headerFcn ) ? $fcn( $this ) : null; }
396
			public function setFooterFunction( \Closure $fcn ) { $this->footerFcn = $fcn; }
397
			public function setHeaderFunction( \Closure $fcn ) { $this->headerFcn = $fcn; }
398
		};
399
		$pdf->setCreator( PDF_CREATOR );
400
		$pdf->setAuthor( 'Aimeos' );
401
402
		/** controller/jobs/order/email/voucher/template-pdf
403
		 * Relative path to the template for the PDF part of the voucher emails.
404
		 *
405
		 * The template file contains the text and processing instructions
406
		 * to generate the result shown in the body of the frontend. The
407
		 * configuration string is the path to the template file relative
408
		 * to the templates directory (usually in templates/controller/jobs).
409
		 * You can overwrite the template file configuration in extensions and
410
		 * provide alternative templates.
411
		 *
412
		 * @param string Relative path to the template
413
		 * @since 2022.10
414
		 * @see controller/jobs/order/email/voucher/template-html
415
		 * @see controller/jobs/order/email/voucher/template-text
416
		 */
417
		$template = $config->get( 'controller/jobs/order/email/voucher/template-pdf', 'order/email/voucher/pdf' );
418
419
		// Generate HTML before creating first PDF page to include header added in template
420
		$content = $view->set( 'pdf', $pdf )->render( $template );
421
422
		$pdf->addPage();
423
		$pdf->writeHtml( $content );
424
		$pdf->lastPage();
425
426
		return $pdf->output( '', 'S' );
427
	}
428
429
430
	/**
431
	 * Returns the ordered voucher products from the basket.
432
	 *
433
	 * @param \Aimeos\MShop\Order\Item\Iface $orderBaseItem Basket object
434
	 * @return \Aimeos\Map List of order product items for the voucher products
435
	 */
436
	protected function products( \Aimeos\MShop\Order\Item\Iface $orderBaseItem ) : \Aimeos\Map
437
	{
438
		$list = [];
439
440
		foreach( $orderBaseItem->getProducts() as $orderProductItem )
441
		{
442
			if( $orderProductItem->getType() === 'voucher' ) {
443
				$list[] = $orderProductItem;
444
			}
445
446
			foreach( $orderProductItem->getProducts() as $subProductItem )
447
			{
448
				if( $subProductItem->getType() === 'voucher' ) {
449
					$list[] = $subProductItem;
450
				}
451
			}
452
		}
453
454
		return map( $list );
455
	}
456
457
458
	/**
459
	 * Saves the given coupon codes
460
	 *
461
	 * @param array $map Associative list of coupon codes as keys and reference Ids as values
462
	 */
463
	protected function saveCoupons( array $map )
464
	{
465
		$couponId = $this->couponId();
466
		$manager = \Aimeos\MShop::create( $this->context(), 'coupon/code' );
467
468
		foreach( $map as $code => $ref )
469
		{
470
			$item = $manager->create()->setParentId( $couponId )
471
				->setCode( $code )->setRef( $ref )->setCount( null ); // unlimited
472
473
			$manager->save( $item );
474
		}
475
	}
476
477
478
	/**
479
	 * Sends the voucher related e-mail for a single order
480
	 *
481
	 * @param \Aimeos\Base\View\Iface $view Populated view object
482
	 * @param \Aimeos\Map $orderProducts List of ordered voucher products
483
	 * @param \Aimeos\MShop\Common\Item\Address\Iface $address Address item
484
	 * @param string|null $logoPath Relative path to the logo in the fs-media file system
485
	 */
486
	protected function send( \Aimeos\Base\View\Iface $view, \Aimeos\Map $orderProducts,
487
		\Aimeos\MShop\Common\Item\Address\Iface $address, string $logoPath = null )
488
	{
489
		/** controller/jobs/order/email/voucher/template-html
490
		 * Relative path to the template for the HTML part of the voucher emails.
491
		 *
492
		 * The template file contains the HTML code and processing instructions
493
		 * to generate the result shown in the body of the frontend. The
494
		 * configuration string is the path to the template file relative
495
		 * to the templates directory (usually in templates/controller/jobs).
496
		 * You can overwrite the template file configuration in extensions and
497
		 * provide alternative templates.
498
		 *
499
		 * @param string Relative path to the template
500
		 * @since 2022.04
501
		 * @see controller/jobs/order/email/voucher/template-text
502
		 */
503
504
		/** controller/jobs/order/email/voucher/template-text
505
		 * Relative path to the template for the text part of the voucher emails.
506
		 *
507
		 * The template file contains the text and processing instructions
508
		 * to generate the result shown in the body of the frontend. The
509
		 * configuration string is the path to the template file relative
510
		 * to the templates directory (usually in templates/controller/jobs).
511
		 * You can overwrite the template file configuration in extensions and
512
		 * provide alternative templates.
513
		 *
514
		 * @param string Relative path to the template
515
		 * @since 2022.04
516
		 * @see controller/jobs/order/email/voucher/template-html
517
		 */
518
519
		$context = $this->context();
520
		$config = $context->config();
521
		$logo = $this->call( 'mailLogo', $logoPath );
522
		$view->orderAddressItem = $address;
523
524
		foreach( $orderProducts as $orderProductItem )
525
		{
526
			if( !empty( $codes = $orderProductItem->getAttribute( 'coupon-code', 'coupon' ) ) )
527
			{
528
				foreach( (array) $codes as $code )
529
				{
530
					$view->orderProductItem = $orderProductItem;
531
					$view->voucher = $code;
532
533
					$msg = $this->call( 'mailTo', $address );
534
					$view->logo = $msg->embed( $logo, basename( (string) $logoPath ) );
535
536
					$msg->subject( $context->translate( 'controller/jobs', 'Your voucher' ) )
537
						->html( $view->render( $config->get( 'controller/jobs/order/email/voucher/template-html', 'order/email/voucher/html' ) ) )
538
						->text( $view->render( $config->get( 'controller/jobs/order/email/voucher/template-text', 'order/email/voucher/text' ) ) )
539
						->attach( $this->pdf( $view ), $this->call( 'filename', $code ), 'application/pdf' )
540
						->send();
541
				}
542
			}
543
		}
544
	}
545
546
547
	/**
548
	 * Returns the site items for the given site codes
549
	 *
550
	 * @param iterable $siteIds List of site IDs
551
	 * @return \Aimeos\Map Site items with codes as keys
552
	 */
553
	protected function sites( iterable $siteIds ) : \Aimeos\Map
554
	{
555
		$map = [];
556
		$manager = \Aimeos\MShop::create( $this->context(), 'locale/site' );
557
558
		foreach( $siteIds as $siteId )
559
		{
560
			$list = explode( '.', trim( $siteId, '.' ) );
561
			$map[$siteId] = $manager->getPath( end( $list ) );
562
		}
563
564
		return map( $map );
565
	}
566
567
568
	/**
569
	 * Returns the payment status for which the e-mails should be sent
570
	 *
571
	 * @return int Payment status
572
	 */
573
	protected function status() : int
574
	{
575
		/** controller/jobs/order/email/voucher/status
576
		 * Only send e-mails containing voucher for these payment status values
577
		 *
578
		 * E-mail containing vouchers can be sent for these payment status values:
579
		 *
580
		 * * 0: deleted
581
		 * * 1: canceled
582
		 * * 2: refused
583
		 * * 3: refund
584
		 * * 4: pending
585
		 * * 5: authorized
586
		 * * 6: received
587
		 *
588
		 * @param integer Payment status constant
589
		 * @since 2018.07
590
		 * @see controller/jobs/order/email/voucher/limit-days
591
		 */
592
		return (int) $this->context()->config()->get( 'controller/jobs/order/email/voucher/status', \Aimeos\MShop\Order\Item\Base::PAY_RECEIVED );
593
	}
594
595
596
	/**
597
	 * Returns the status type for filtering the orders
598
	 *
599
	 * @return string Status type
600
	 */
601
	protected function type() : string
602
	{
603
		return \Aimeos\MShop\Order\Item\Status\Base::EMAIL_VOUCHER;
604
	}
605
606
607
	/**
608
	 * Adds the status of the delivered e-mail for the given order ID
609
	 *
610
	 * @param string $orderId Unique order ID
611
	 */
612
	protected function update( string $orderId )
613
	{
614
		$orderStatusManager = \Aimeos\MShop::create( $this->context(), 'order/status' );
615
616
		$statusItem = $orderStatusManager->create()->setParentId( $orderId )->setValue( 1 )
617
			->setType( \Aimeos\MShop\Order\Item\Status\Base::EMAIL_VOUCHER );
618
619
		$orderStatusManager->save( $statusItem );
620
	}
621
622
623
	/**
624
	 * Returns the view populated with common data
625
	 *
626
	 * @param \Aimeos\MShop\Order\Item\Iface $base Basket including addresses
627
	 * @param string|null $theme Theme name
628
	 * @return \Aimeos\Base\View\Iface View object
629
	 */
630
	protected function view( \Aimeos\MShop\Order\Item\Iface $base, string $theme = null ) : \Aimeos\Base\View\Iface
631
	{
632
		$address = $this->address( $base );
633
		$langId = $address->getLanguageId() ?: $base->locale()->getLanguageId();
634
635
		$view = $this->call( 'mailView', $langId );
636
		$view->intro = $this->call( 'mailIntro', $address );
637
		$view->css = $this->call( 'mailCss', $theme );
638
		$view->address = $address;
639
		$view->urlparams = [
640
			'currency' => $base->getPrice()->getCurrencyId(),
641
			'site' => $base->getSiteCode(),
642
			'locale' => $langId,
643
		];
644
645
		return $view;
646
	}
647
}
648