Passed
Push — master ( 12098f...ce5e60 )
by John
14:20 queued 11s
created

BirthdayService   F

Complexity

Total Complexity 65

Size/Duplication

Total Lines 434
Duplicated Lines 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
eloc 211
c 1
b 0
f 0
dl 0
loc 434
rs 3.2
wmc 65

13 Methods

Rating   Name   Duplication   Size   Complexity  
A onCardChanged() 0 24 5
A onCardDeleted() 0 18 5
A ensureCalendarExists() 0 12 2
A __construct() 0 12 1
D buildDateFromContact() 0 114 18
A isUserEnabled() 0 9 2
A getAllAffectedPrincipals() 0 14 4
A syncUser() 0 8 3
A isGloballyEnabled() 0 2 1
A birthdayEvenChanged() 0 11 3
C formatTitle() 0 53 14
A updateCalendar() 0 18 5
A resetForUser() 0 7 2

How to fix   Complexity   

Complex Class

Complex classes like BirthdayService often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes.

Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.

While breaking up the class, it is a good idea to analyze how other classes use BirthdayService, and based on these observations, apply Extract Interface, too.

1
<?php
2
3
declare(strict_types=1);
4
5
/**
6
 * @copyright Copyright (c) 2016, ownCloud, Inc.
7
 * @copyright Copyright (c) 2019, Georg Ehrke
8
 *
9
 * @author Achim Königs <[email protected]>
10
 * @author Christoph Wurst <[email protected]>
11
 * @author Georg Ehrke <[email protected]>
12
 * @author Robin Appelman <[email protected]>
13
 * @author Thomas Müller <[email protected]>
14
 *
15
 * @license AGPL-3.0
16
 *
17
 * This code is free software: you can redistribute it and/or modify
18
 * it under the terms of the GNU Affero General Public License, version 3,
19
 * as published by the Free Software Foundation.
20
 *
21
 * This program is distributed in the hope that it will be useful,
22
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
23
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
24
 * GNU Affero General Public License for more details.
25
 *
26
 * You should have received a copy of the GNU Affero General Public License, version 3,
27
 * along with this program. If not, see <http://www.gnu.org/licenses/>
28
 *
29
 */
30
31
namespace OCA\DAV\CalDAV;
32
33
use Exception;
34
use OCA\DAV\CardDAV\CardDavBackend;
35
use OCA\DAV\DAV\GroupPrincipalBackend;
36
use OCP\IConfig;
37
use OCP\IDBConnection;
38
use OCP\IL10N;
39
use Sabre\VObject\Component\VCalendar;
40
use Sabre\VObject\Component\VCard;
41
use Sabre\VObject\DateTimeParser;
42
use Sabre\VObject\Document;
43
use Sabre\VObject\InvalidDataException;
44
use Sabre\VObject\Property\VCard\DateAndOrTime;
45
use Sabre\VObject\Reader;
46
47
/**
48
 * Class BirthdayService
49
 *
50
 * @package OCA\DAV\CalDAV
51
 */
52
class BirthdayService {
53
	public const BIRTHDAY_CALENDAR_URI = 'contact_birthdays';
54
55
	/** @var GroupPrincipalBackend */
56
	private $principalBackend;
57
58
	/** @var CalDavBackend  */
59
	private $calDavBackEnd;
60
61
	/** @var CardDavBackend  */
62
	private $cardDavBackEnd;
63
64
	/** @var IConfig */
65
	private $config;
66
67
	/** @var IDBConnection */
68
	private $dbConnection;
69
70
	/** @var IL10N */
71
	private $l10n;
72
73
	/**
74
	 * BirthdayService constructor.
75
	 *
76
	 * @param CalDavBackend $calDavBackEnd
77
	 * @param CardDavBackend $cardDavBackEnd
78
	 * @param GroupPrincipalBackend $principalBackend
79
	 * @param IConfig $config
80
	 * @param IDBConnection $dbConnection
81
	 * @param IL10N $l10n
82
	 */
83
	public function __construct(CalDavBackend $calDavBackEnd,
84
								CardDavBackend $cardDavBackEnd,
85
								GroupPrincipalBackend $principalBackend,
86
								IConfig $config,
87
								IDBConnection $dbConnection,
88
								IL10N $l10n) {
89
		$this->calDavBackEnd = $calDavBackEnd;
90
		$this->cardDavBackEnd = $cardDavBackEnd;
91
		$this->principalBackend = $principalBackend;
92
		$this->config = $config;
93
		$this->dbConnection = $dbConnection;
94
		$this->l10n = $l10n;
95
	}
96
97
	/**
98
	 * @param int $addressBookId
99
	 * @param string $cardUri
100
	 * @param string $cardData
101
	 */
102
	public function onCardChanged(int $addressBookId,
103
								  string $cardUri,
104
								  string $cardData) {
105
		if (!$this->isGloballyEnabled()) {
106
			return;
107
		}
108
109
		$targetPrincipals = $this->getAllAffectedPrincipals($addressBookId);
110
		$book = $this->cardDavBackEnd->getAddressBookById($addressBookId);
0 ignored issues
show
Bug introduced by
Are you sure the assignment to $book is correct as $this->cardDavBackEnd->g...ookById($addressBookId) targeting OCA\DAV\CardDAV\CardDavB...d::getAddressBookById() seems to always return null.

This check looks for function or method calls that always return null and whose return value is assigned to a variable.

class A
{
    function getObject()
    {
        return null;
    }

}

$a = new A();
$object = $a->getObject();

The method getObject() can return nothing but null, so it makes no sense to assign that value to a variable.

The reason is most likely that a function or method is imcomplete or has been reduced for debug purposes.

Loading history...
111
		$targetPrincipals[] = $book['principaluri'];
112
		$datesToSync = [
113
			['postfix' => '', 'field' => 'BDAY'],
114
			['postfix' => '-death', 'field' => 'DEATHDATE'],
115
			['postfix' => '-anniversary', 'field' => 'ANNIVERSARY'],
116
		];
117
118
		foreach ($targetPrincipals as $principalUri) {
119
			if (!$this->isUserEnabled($principalUri)) {
120
				continue;
121
			}
122
123
			$calendar = $this->ensureCalendarExists($principalUri);
0 ignored issues
show
Bug introduced by
Are you sure the assignment to $calendar is correct as $this->ensureCalendarExists($principalUri) targeting OCA\DAV\CalDAV\BirthdayS...:ensureCalendarExists() seems to always return null.

This check looks for function or method calls that always return null and whose return value is assigned to a variable.

class A
{
    function getObject()
    {
        return null;
    }

}

$a = new A();
$object = $a->getObject();

The method getObject() can return nothing but null, so it makes no sense to assign that value to a variable.

The reason is most likely that a function or method is imcomplete or has been reduced for debug purposes.

Loading history...
124
			foreach ($datesToSync as $type) {
125
				$this->updateCalendar($cardUri, $cardData, $book, (int) $calendar['id'], $type);
0 ignored issues
show
Bug introduced by
$book of type null is incompatible with the type array expected by parameter $book of OCA\DAV\CalDAV\BirthdayService::updateCalendar(). ( Ignorable by Annotation )

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

125
				$this->updateCalendar($cardUri, $cardData, /** @scrutinizer ignore-type */ $book, (int) $calendar['id'], $type);
Loading history...
126
			}
127
		}
128
	}
129
130
	/**
131
	 * @param int $addressBookId
132
	 * @param string $cardUri
133
	 */
134
	public function onCardDeleted(int $addressBookId,
135
								  string $cardUri) {
136
		if (!$this->isGloballyEnabled()) {
137
			return;
138
		}
139
140
		$targetPrincipals = $this->getAllAffectedPrincipals($addressBookId);
141
		$book = $this->cardDavBackEnd->getAddressBookById($addressBookId);
0 ignored issues
show
Bug introduced by
Are you sure the assignment to $book is correct as $this->cardDavBackEnd->g...ookById($addressBookId) targeting OCA\DAV\CardDAV\CardDavB...d::getAddressBookById() seems to always return null.

This check looks for function or method calls that always return null and whose return value is assigned to a variable.

class A
{
    function getObject()
    {
        return null;
    }

}

$a = new A();
$object = $a->getObject();

The method getObject() can return nothing but null, so it makes no sense to assign that value to a variable.

The reason is most likely that a function or method is imcomplete or has been reduced for debug purposes.

Loading history...
142
		$targetPrincipals[] = $book['principaluri'];
143
		foreach ($targetPrincipals as $principalUri) {
144
			if (!$this->isUserEnabled($principalUri)) {
145
				continue;
146
			}
147
148
			$calendar = $this->ensureCalendarExists($principalUri);
0 ignored issues
show
Bug introduced by
Are you sure the assignment to $calendar is correct as $this->ensureCalendarExists($principalUri) targeting OCA\DAV\CalDAV\BirthdayS...:ensureCalendarExists() seems to always return null.

This check looks for function or method calls that always return null and whose return value is assigned to a variable.

class A
{
    function getObject()
    {
        return null;
    }

}

$a = new A();
$object = $a->getObject();

The method getObject() can return nothing but null, so it makes no sense to assign that value to a variable.

The reason is most likely that a function or method is imcomplete or has been reduced for debug purposes.

Loading history...
149
			foreach (['', '-death', '-anniversary'] as $tag) {
150
				$objectUri = $book['uri'] . '-' . $cardUri . $tag .'.ics';
151
				$this->calDavBackEnd->deleteCalendarObject($calendar['id'], $objectUri);
152
			}
153
		}
154
	}
155
156
	/**
157
	 * @param string $principal
158
	 * @return array|null
159
	 * @throws \Sabre\DAV\Exception\BadRequest
160
	 */
161
	public function ensureCalendarExists(string $principal):?array {
162
		$calendar = $this->calDavBackEnd->getCalendarByUri($principal, self::BIRTHDAY_CALENDAR_URI);
0 ignored issues
show
Bug introduced by
Are you sure the assignment to $calendar is correct as $this->calDavBackEnd->ge...:BIRTHDAY_CALENDAR_URI) targeting OCA\DAV\CalDAV\CalDavBackend::getCalendarByUri() seems to always return null.

This check looks for function or method calls that always return null and whose return value is assigned to a variable.

class A
{
    function getObject()
    {
        return null;
    }

}

$a = new A();
$object = $a->getObject();

The method getObject() can return nothing but null, so it makes no sense to assign that value to a variable.

The reason is most likely that a function or method is imcomplete or has been reduced for debug purposes.

Loading history...
163
		if (!is_null($calendar)) {
0 ignored issues
show
introduced by
The condition is_null($calendar) is always true.
Loading history...
164
			return $calendar;
165
		}
166
		$this->calDavBackEnd->createCalendar($principal, self::BIRTHDAY_CALENDAR_URI, [
167
			'{DAV:}displayname' => 'Contact birthdays',
168
			'{http://apple.com/ns/ical/}calendar-color' => '#E9D859',
169
			'components' => 'VEVENT',
170
		]);
171
172
		return $this->calDavBackEnd->getCalendarByUri($principal, self::BIRTHDAY_CALENDAR_URI);
0 ignored issues
show
Bug introduced by
Are you sure the usage of $this->calDavBackEnd->ge...:BIRTHDAY_CALENDAR_URI) targeting OCA\DAV\CalDAV\CalDavBackend::getCalendarByUri() seems to always return null.

This check looks for function or method calls that always return null and whose return value is used.

class A
{
    function getObject()
    {
        return null;
    }

}

$a = new A();
if ($a->getObject()) {

The method getObject() can return nothing but null, so it makes no sense to use the return value.

The reason is most likely that a function or method is imcomplete or has been reduced for debug purposes.

Loading history...
173
	}
174
175
	/**
176
	 * @param $cardData
177
	 * @param $dateField
178
	 * @param $postfix
179
	 * @return VCalendar|null
180
	 * @throws InvalidDataException
181
	 */
182
	public function buildDateFromContact(string $cardData,
183
										 string $dateField,
184
										 string $postfix):?VCalendar {
185
		if (empty($cardData)) {
186
			return null;
187
		}
188
		try {
189
			$doc = Reader::read($cardData);
190
			// We're always converting to vCard 4.0 so we can rely on the
191
			// VCardConverter handling the X-APPLE-OMIT-YEAR property for us.
192
			if (!$doc instanceof VCard) {
193
				return null;
194
			}
195
			$doc = $doc->convert(Document::VCARD40);
196
		} catch (Exception $e) {
197
			return null;
198
		}
199
200
		if (!isset($doc->{$dateField})) {
201
			return null;
202
		}
203
		if (!isset($doc->FN)) {
204
			return null;
205
		}
206
		$birthday = $doc->{$dateField};
207
		if (!(string)$birthday) {
208
			return null;
209
		}
210
		// Skip if the BDAY property is not of the right type.
211
		if (!$birthday instanceof DateAndOrTime) {
212
			return null;
213
		}
214
215
		// Skip if we can't parse the BDAY value.
216
		try {
217
			$dateParts = DateTimeParser::parseVCardDateTime($birthday->getValue());
218
		} catch (InvalidDataException $e) {
219
			return null;
220
		}
221
222
		$unknownYear = false;
223
		$originalYear = null;
224
		if (!$dateParts['year']) {
225
			$birthday = '1970-' . $dateParts['month'] . '-' . $dateParts['date'];
226
227
			$unknownYear = true;
228
		} else {
229
			$parameters = $birthday->parameters();
230
			if (isset($parameters['X-APPLE-OMIT-YEAR'])) {
231
				$omitYear = $parameters['X-APPLE-OMIT-YEAR'];
232
				if ($dateParts['year'] === $omitYear) {
233
					$birthday = '1970-' . $dateParts['month'] . '-' . $dateParts['date'];
234
					$unknownYear = true;
235
				}
236
			} else {
237
				$originalYear = (int)$dateParts['year'];
238
				// 'X-APPLE-OMIT-YEAR' is not always present, at least iOS 12.4 uses the hard coded date of 1604 (the start of the gregorian calendar) when the year is unknown
239
				if ($originalYear == 1604) {
240
					$originalYear = null;
241
					$unknownYear = true;
242
					$birthday = '1970-' . $dateParts['month'] . '-' . $dateParts['date'];
243
				}
244
				if ($originalYear < 1970) {
245
					$birthday = '1970-' . $dateParts['month'] . '-' . $dateParts['date'];
246
				}
247
			}
248
		}
249
250
		try {
251
			if ($birthday instanceof DateAndOrTime) {
252
				$date = $birthday->getDateTime();
253
			} else {
254
				$date = new \DateTimeImmutable($birthday);
255
			}
256
		} catch (Exception $e) {
257
			return null;
258
		}
259
260
		$summary = $this->formatTitle($dateField, $doc->FN->getValue(), $originalYear, $this->dbConnection->supports4ByteText());
261
262
		$vCal = new VCalendar();
263
		$vCal->VERSION = '2.0';
264
		$vCal->PRODID = '-//IDN nextcloud.com//Birthday calendar//EN';
265
		$vEvent = $vCal->createComponent('VEVENT');
266
		$vEvent->add('DTSTART');
267
		$vEvent->DTSTART->setDateTime(
0 ignored issues
show
Bug introduced by
The method setDateTime() does not exist on Sabre\VObject\Property. It seems like you code against a sub-type of Sabre\VObject\Property such as Sabre\VObject\Property\ICalendar\DateTime or Sabre\VObject\Property\VCard\DateAndOrTime. ( Ignorable by Annotation )

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

267
		$vEvent->DTSTART->/** @scrutinizer ignore-call */ 
268
                    setDateTime(
Loading history...
268
			$date
269
		);
270
		$vEvent->DTSTART['VALUE'] = 'DATE';
271
		$vEvent->add('DTEND');
272
273
		$dtEndDate = (new \DateTime())->setTimestamp($date->getTimeStamp());
274
		$dtEndDate->add(new \DateInterval('P1D'));
275
		$vEvent->DTEND->setDateTime(
276
			$dtEndDate
277
		);
278
279
		$vEvent->DTEND['VALUE'] = 'DATE';
280
		$vEvent->{'UID'} = $doc->UID . $postfix;
281
		$vEvent->{'RRULE'} = 'FREQ=YEARLY';
282
		$vEvent->{'SUMMARY'} = $summary;
283
		$vEvent->{'TRANSP'} = 'TRANSPARENT';
284
		$vEvent->{'X-NEXTCLOUD-BC-FIELD-TYPE'} = $dateField;
285
		$vEvent->{'X-NEXTCLOUD-BC-UNKNOWN-YEAR'} = $unknownYear ? '1' : '0';
286
		if ($originalYear !== null) {
287
			$vEvent->{'X-NEXTCLOUD-BC-YEAR'} = (string) $originalYear;
288
		}
289
		$alarm = $vCal->createComponent('VALARM');
290
		$alarm->add($vCal->createProperty('TRIGGER', '-PT0M', ['VALUE' => 'DURATION']));
291
		$alarm->add($vCal->createProperty('ACTION', 'DISPLAY'));
292
		$alarm->add($vCal->createProperty('DESCRIPTION', $vEvent->{'SUMMARY'}));
293
		$vEvent->add($alarm);
294
		$vCal->add($vEvent);
295
		return $vCal;
296
	}
297
298
	/**
299
	 * @param string $user
300
	 */
301
	public function resetForUser(string $user):void {
302
		$principal = 'principals/users/'.$user;
303
		$calendar = $this->calDavBackEnd->getCalendarByUri($principal, self::BIRTHDAY_CALENDAR_URI);
0 ignored issues
show
Bug introduced by
Are you sure the assignment to $calendar is correct as $this->calDavBackEnd->ge...:BIRTHDAY_CALENDAR_URI) targeting OCA\DAV\CalDAV\CalDavBackend::getCalendarByUri() seems to always return null.

This check looks for function or method calls that always return null and whose return value is assigned to a variable.

class A
{
    function getObject()
    {
        return null;
    }

}

$a = new A();
$object = $a->getObject();

The method getObject() can return nothing but null, so it makes no sense to assign that value to a variable.

The reason is most likely that a function or method is imcomplete or has been reduced for debug purposes.

Loading history...
304
		$calendarObjects = $this->calDavBackEnd->getCalendarObjects($calendar['id'], CalDavBackend::CALENDAR_TYPE_CALENDAR);
305
306
		foreach ($calendarObjects as $calendarObject) {
307
			$this->calDavBackEnd->deleteCalendarObject($calendar['id'], $calendarObject['uri'], CalDavBackend::CALENDAR_TYPE_CALENDAR);
308
		}
309
	}
310
311
	/**
312
	 * @param string $user
313
	 * @throws \Sabre\DAV\Exception\BadRequest
314
	 */
315
	public function syncUser(string $user):void {
316
		$principal = 'principals/users/'.$user;
317
		$this->ensureCalendarExists($principal);
318
		$books = $this->cardDavBackEnd->getAddressBooksForUser($principal);
319
		foreach ($books as $book) {
320
			$cards = $this->cardDavBackEnd->getCards($book['id']);
321
			foreach ($cards as $card) {
322
				$this->onCardChanged((int) $book['id'], $card['uri'], $card['carddata']);
323
			}
324
		}
325
	}
326
327
	/**
328
	 * @param string $existingCalendarData
329
	 * @param VCalendar $newCalendarData
330
	 * @return bool
331
	 */
332
	public function birthdayEvenChanged(string $existingCalendarData,
333
										VCalendar $newCalendarData):bool {
334
		try {
335
			$existingBirthday = Reader::read($existingCalendarData);
336
		} catch (Exception $ex) {
337
			return true;
338
		}
339
340
		return (
341
			$newCalendarData->VEVENT->DTSTART->getValue() !== $existingBirthday->VEVENT->DTSTART->getValue() ||
0 ignored issues
show
Bug introduced by
The property DTSTART does not seem to exist on Sabre\VObject\Property.
Loading history...
342
			$newCalendarData->VEVENT->SUMMARY->getValue() !== $existingBirthday->VEVENT->SUMMARY->getValue()
0 ignored issues
show
Bug introduced by
The property SUMMARY does not seem to exist on Sabre\VObject\Property.
Loading history...
343
		);
344
	}
345
346
	/**
347
	 * @param integer $addressBookId
348
	 * @return mixed
349
	 */
350
	protected function getAllAffectedPrincipals(int $addressBookId) {
351
		$targetPrincipals = [];
352
		$shares = $this->cardDavBackEnd->getShares($addressBookId);
353
		foreach ($shares as $share) {
354
			if ($share['{http://owncloud.org/ns}group-share']) {
355
				$users = $this->principalBackend->getGroupMemberSet($share['{http://owncloud.org/ns}principal']);
356
				foreach ($users as $user) {
357
					$targetPrincipals[] = $user['uri'];
358
				}
359
			} else {
360
				$targetPrincipals[] = $share['{http://owncloud.org/ns}principal'];
361
			}
362
		}
363
		return array_values(array_unique($targetPrincipals, SORT_STRING));
364
	}
365
366
	/**
367
	 * @param string $cardUri
368
	 * @param string $cardData
369
	 * @param array $book
370
	 * @param int $calendarId
371
	 * @param array $type
372
	 * @throws InvalidDataException
373
	 * @throws \Sabre\DAV\Exception\BadRequest
374
	 */
375
	private function updateCalendar(string $cardUri,
376
									string $cardData,
377
									array $book,
378
									int $calendarId,
379
									array $type):void {
380
		$objectUri = $book['uri'] . '-' . $cardUri . $type['postfix'] . '.ics';
381
		$calendarData = $this->buildDateFromContact($cardData, $type['field'], $type['postfix']);
382
		$existing = $this->calDavBackEnd->getCalendarObject($calendarId, $objectUri);
383
		if (is_null($calendarData)) {
384
			if (!is_null($existing)) {
385
				$this->calDavBackEnd->deleteCalendarObject($calendarId, $objectUri);
386
			}
387
		} else {
388
			if (is_null($existing)) {
389
				$this->calDavBackEnd->createCalendarObject($calendarId, $objectUri, $calendarData->serialize());
390
			} else {
391
				if ($this->birthdayEvenChanged($existing['calendardata'], $calendarData)) {
392
					$this->calDavBackEnd->updateCalendarObject($calendarId, $objectUri, $calendarData->serialize());
393
				}
394
			}
395
		}
396
	}
397
398
	/**
399
	 * checks if the admin opted-out of birthday calendars
400
	 *
401
	 * @return bool
402
	 */
403
	private function isGloballyEnabled():bool {
404
		return $this->config->getAppValue('dav', 'generateBirthdayCalendar', 'yes') === 'yes';
405
	}
406
407
	/**
408
	 * Checks if the user opted-out of birthday calendars
409
	 *
410
	 * @param string $userPrincipal The user principal to check for
411
	 * @return bool
412
	 */
413
	private function isUserEnabled(string $userPrincipal):bool {
414
		if (strpos($userPrincipal, 'principals/users/') === 0) {
415
			$userId = substr($userPrincipal, 17);
416
			$isEnabled = $this->config->getUserValue($userId, 'dav', 'generateBirthdayCalendar', 'yes');
417
			return $isEnabled === 'yes';
418
		}
419
420
		// not sure how we got here, just be on the safe side and return true
421
		return true;
422
	}
423
424
	/**
425
	 * Formats title of Birthday event
426
	 *
427
	 * @param string $field Field name like BDAY, ANNIVERSARY, ...
428
	 * @param string $name Name of contact
429
	 * @param int|null $year Year of birth, anniversary, ...
430
	 * @param bool $supports4Byte Whether or not the database supports 4 byte chars
431
	 * @return string The formatted title
432
	 */
433
	private function formatTitle(string $field,
434
								 string $name,
435
								 int $year = null,
436
								 bool $supports4Byte = true):string {
437
		if ($supports4Byte) {
438
			switch ($field) {
439
				case 'BDAY':
440
					return implode('', [
441
						'🎂 ',
442
						$name,
443
						$year ? (' (' . $year . ')') : '',
444
					]);
445
446
				case 'DEATHDATE':
447
					return implode('', [
448
						$this->l10n->t('Death of %s', [$name]),
449
						$year ? (' (' . $year . ')') : '',
450
					]);
451
452
				case 'ANNIVERSARY':
453
					return implode('', [
454
						'💍 ',
455
						$name,
456
						$year ? (' (' . $year . ')') : '',
457
					]);
458
459
				default:
460
					return '';
461
			}
462
		} else {
463
			switch ($field) {
464
				case 'BDAY':
465
					return implode('', [
466
						$name,
467
						' ',
468
						$year ? ('(*' . $year . ')') : '*',
469
					]);
470
471
				case 'DEATHDATE':
472
					return implode('', [
473
						$this->l10n->t('Death of %s', [$name]),
474
						$year ? (' (' . $year . ')') : '',
475
					]);
476
477
				case 'ANNIVERSARY':
478
					return implode('', [
479
						$name,
480
						' ',
481
						$year ? ('(⚭' . $year . ')') : '⚭',
482
					]);
483
484
				default:
485
					return '';
486
			}
487
		}
488
	}
489
}
490