Date & Time

class Cake\I18n\Time

Si vous avez besoin de fonctionnalités TimeHelper en-dehors d’une View, utilisez la classe Time:

use Cake\I18n\Time;

class UsersController extends AppController
{

    public function initialize()
    {
        parent::initialize();
        $this->loadComponent('Auth');
    }

    public function afterLogin()
    {
        $time = new Time($this->Auth->user('date_of_birth'));
        if ($time->isToday()) {
            // accueillir l'utilisateur avec un message de bon anniversaire
            $this->Flash->success(__('Bon anniversaire à toi...'));
        }
    }
}

En interne, CakePHP utilise Chronos pour faire fonctionner l’utilitaire Time. Tout ce que vous pouvez faire avec Chronos et DateTime, vous pouvez le faire avec Time et Date.

Note

Avant 3.2.0, CakePHP utilisait Carbon.

Pour plus d’informations sur Chronos, rendez-vous sur la documentation de l’API.

Créer des Instances Time

Il y a plusieurs façons de créer des instances Time:

use Cake\I18n\Time;

// Crée à partir d'une chaîne datetime.
$time = Time::createFromFormat(
    'Y-m-d H:i:s',
    $datetime,
    'America/New_York'
);

// Crée à partir d'un timestamp
$time = Time::createFromTimestamp($ts);

// Récupère le temps actuel.
$time = Time::now();

// Ou utilise juste 'new'
$time = new Time('2014-01-10 11:11', 'America/New_York');

$time = new Time('2 hours ago');

Le constructeur de la classe Time peut prendre les mêmes paramètres que la classe PHP interne DateTime. Quand vous passez un nombre ou une valeur numérique, elle sera interprétée comme un timestamp UNIX.

Dans les cas de test, vous pouvez mock out now() en utilisant setTestNow():

// Fixe le temps.
$now = new Time('2014-04-12 12:22:30');
Time::setTestNow($now);

// Retourne '2014-04-12 12:22:30'
$now = Time::now();

// Retourne '2014-04-12 12:22:30'
$now = Time::parse('now');

Manipulation

Une fois créées, vous pouvez manipuler les instances Time en utilisant les méthodes setter:

$now = Time::now();
$now->year(2013)
    ->month(10)
    ->day(31);

Vous pouvez aussi utiliser les méthodes fournies nativement par la classe PHP DateTime:

$now->setDate(2013, 10, 31);

Les dates peuvent être modifiées à travers la soustraction et l’addition de leurs composantes:

$now = Time::now();
$now->subDays(5);
$now->addMonth(1);

// Utilisation des chaînes strtotime.
$now->modify('+5 days');

Vous pouvez obtenir des composantes internes d’une date en accédant à ses propriétés:

$now = Time::now();
echo $now->year; // 2014
echo $now->month; // 5
echo $now->day; // 10
echo $now->timezone; // America/New_York

Il est aussi permis d’assigner directement ces propriétés pour modifier la date:

$time->year = 2015;
$time->timezone = 'Europe/Paris';

Formatage

static Cake\I18n\Time::setJsonEncodeFormat($format)

Cette méthode définit le format par défaut utilisé lors de la conversion d’un objet en json:

Time::setJsonEncodeFormat('yyyy-MM-dd HH:mm:ss');  // For any mutable DateTime
FrozenTime::setJsonEncodeFormat('yyyy-MM-dd HH:mm:ss');  // For any immutable DateTime
Date::setJsonEncodeFormat('yyyy-MM-dd HH:mm:ss');  // For any mutable Date
FrozenDate::setJsonEncodeFormat('yyyy-MM-dd HH:mm:ss');  // For any immutable Date

Note

Cette méthode doit être appelée statiquement.

Cake\I18n\Time::i18nFormat($format = null, $timezone = null, $locale = null)

Une chose habituelle à faire avec les instances Time est d’afficher les dates formatées. CakePHP facilite cela:

$now = Time::parse('2014-10-31');

// Affiche un stamp datetime localisé.
echo $now;

// Affiche '10/31/14, 12:00 AM' pour la locale en-US
$now->i18nFormat();

// Utilise la date complète et le format time
$now->i18nFormat(\IntlDateFormatter::FULL);

// Utilise la date complète mais un format court de temps
$now->i18nFormat([\IntlDateFormatter::FULL, \IntlDateFormatter::SHORT]);

// affiche '2014-10-31 00:00:00'
$now->i18nFormat('yyyy-MM-dd HH:mm:ss');

Il est possible de spécifier le format d’affichage désiré. Vous pouvez soit passer une constante IntlDateFormatter ou une chaine complète de formatage tel que spécifié dans cette ressource: https://unicode-org.github.io/icu/userguide/format_parse/datetime/#datetime-format-syntax.

Vous pouvez aussi formater les dates avec des calendriers non-grégoriens:

// Affiche 'Friday, Aban 9, 1393 AP at 12:00:00 AM GMT'
$result = $now->i18nFormat(\IntlDateFormatter::FULL, null, 'en-IR@calendar=persian');

Les types de calendrier suivants sont supportés:

  • japanese

  • buddhist

  • chinese

  • persian

  • indian

  • islamic

  • hebrew

  • coptic

  • ethiopic

Nouveau dans la version 3.1: Le support des calendriers non-grégoriens a été ajouté dans 3.1

Note

Pour les chaînes constantes, par exemple pour IntlDateFormatter::FULL, Intl utilise la librairie ICU qui alimente ses données à partir de CLDR (http://cldr.unicode.org/) dont la version peut varier selon l’installation PHP et donner des résultats différents.

Cake\I18n\Time::nice()

Affiche un format prédéfini “nice”:

$now = Time::parse('2014-10-31');

// Affiche 'Oct 31, 2014 12:32pm' en en-US
echo $now->nice();

Vous pouvez modifier le timezone avec lequel la date est affichée sans modifier l’objet Time lui-même. C’est utile quand vous stockez des dates dans un timezone, mais que vous voulez les afficher dans un timezone propre à un utilisateur:

$now->i18nFormat(\IntlDateFormatter::FULL, 'Europe/Paris');

Laisser le premier paramètre à null va utiliser la chaine de formatage par défaut:

$now->i18nFormat(null, 'Europe/Paris');

Enfin, il est possible d’utiliser une locale différente pour l’affichage d’une date:

echo $now->i18nFormat(\IntlDateFormatter::FULL, 'Europe/Paris', 'fr-FR');

echo $now->nice('Europe/Paris', 'fr-FR');

Définir la Locale par défaut et la Chaîne Format

La locale par défaut avec laquelle les dates sont affichées quand vous utilisez nice i18nFormat est prise à partir de la directive intl.default_locale. Vous pouvez cependant modifier ceci par défaut à la volée:

Time::setDefaultLocale('es-ES'); // For any mutable DateTime
FrozenTime::setDefaultLocale('es-ES'); // For any immutable DateTime
Date::setDefaultLocale('es-ES'); // For any mutable Date
FrozenDate::setDefaultLocale('es-ES'); // For any immutable Date

A partir de maintenant, les datetimes vont s’afficher avec un format de préférence Espagnol, à moins qu’une locale différente ne soit spécifiée directement dans la méthode de formatage.

De même, il est possible de modifier la chaîne de formatage par défaut à utiliser pour le format i18nFormat:

Time::setToStringFormat(\IntlDateFormatter::SHORT); // For any mutable DateTime
FrozenTime::setToStringFormat(\IntlDateFormatter::SHORT); // For any immutable DateTime
Date::setToStringFormat(\IntlDateFormatter::SHORT); // For any mutable Date
FrozenDate::setToStringFormat(\IntlDateFormatter::SHORT); // For any immutable Date

// La même méthode existe pour les Date, FrozenDate, et FrozenTime
Time::setToStringFormat([
    \IntlDateFormatter::FULL,
    \IntlDateFormatter::SHORT
]);

// La même méthode existe pour les Date, FrozenDate, et FrozenTime
Time::setToStringFormat('yyyy-MM-dd HH:mm:ss');

Il est recommandé de toujours utiliser les constantes plutôt que de directement passer une date en format chaîne de caractère.

Formater les Temps Relatifs

Cake\I18n\Time::timeAgoInWords(array $options = [])

Souvent, il est utile d’afficher les temps liés au présent:

$now = new Time('Aug 22, 2011');
echo $now->timeAgoInWords(
    ['format' => 'MMM d, YYY', 'end' => '+1 year']
);
// On Nov 10th, 2011 this would display: 2 months, 2 weeks, 6 days ago

L’option end vous laisse définir à partir de quel point les temps relatifs doivent être formatés en utilisant l’option format. L’option accuracy nous laisse contrôler le niveau de détail qui devra être utilisé pour chaque intervalle:

// Si $timestamp est 1 month, 1 week, 5 days et 6 hours ago
echo $timestamp->timeAgoInWords([
    'accuracy' => ['month' => 'month'],
    'end' => '1 year'
]);
// Affiche '1 month ago'

En configurant accuracy en une chaîne, vous pouvez spécifier le niveau maximum de détail que vous souhaitez afficher:

$time = new Time('+23 hours');
// Affiche 'in about a day'
$result = $time->timeAgoInWords([
    'accuracy' => 'day'
]);

Conversion

Cake\I18n\Time::toQuarter()

Une fois créées, vous pouvez convertir les instances Time en timestamps ou valeurs quarter:

$time = new Time('2014-06-15');
$time->toQuarter();
$time->toUnixString();

Comparer Avec le Present

Cake\I18n\Time::isYesterday()
Cake\I18n\Time::isThisWeek()
Cake\I18n\Time::isThisMonth()
Cake\I18n\Time::isThisYear()

Vous pouvez comparer une instance Time avec le temps présent de plusieurs façons:

$time = new Time('2014-06-15');

echo $time->isYesterday();
echo $time->isThisWeek();
echo $time->isThisMonth();
echo $time->isThisYear();

Chacune des méthodes ci-dessus va retourner true/false selon si oui ou non l’instance Time correspond au temps présent.

Comparer Avec les Intervals

Cake\I18n\Time::isWithinNext($interval)

Vous pouvez regarder si une instance Time tombe dans un intervalle en utilisant wasWithinLast() et isWithinNext():

$time = new Time('2014-06-15');

// A moins de 2 jours.
echo $time->isWithinNext(2);

// A moins de 2 semaines.
echo $time->isWithinNext('2 weeks');
Cake\I18n\Time::wasWithinLast($interval)

Vous pouvez aussi comparer une instance Time dans un intervalle dans le passé:

// Dans les 2 derniers jours.
echo $time->wasWithinLast(2);

// Dans les 2 dernières semaines.
echo $time->wasWithinLast('2 weeks');

Dates

Nouveau dans la version 3.2.

La classe Date dans CakePHP implémente les mêmes API et méthodes que Cake\I18n\Time. La différence principale entre Time et Date est que Date ne suit pas les composants liés à l’heure et est toujours en UTC. Par exemple:

use Cake\I18n\Date;
$date = new Date('2015-06-15');

$date->modify('+2 hours');
// Affiche 2015-06-15 00:00:00
echo $date->format('Y-m-d H:i:s');

$date->modify('+36 hours');
// Affiche 2015-06-15 00:00:00
echo $date->format('Y-m-d H:i:s');

Les tentatives de modification de timezone sur une instance de Date seront toujours ignorées:

use Cake\I18n\Date;
$date = new Date('2015-06-15');
$date->setTimezone(new \DateTimeZone('America/New_York'));

// Affiche UTC
echo $date->format('e');

Dates et Heures Immutables

class Cake\I18n\FrozenTime
class Cake\I18n\FrozenDate

CakePHP offre des classes de date et d’heure immutables qui implémentent la même interface que leurs équivalents mutables. Les objets immutables sont utiles pour éviter les modifications accidentelles de données, ou lorsque vous voulez éviter les problèmes liés à l’ordre de dépendances. Prenez le code suivant:

use Cake\I18n\Time;
$time = new Time('2015-06-15 08:23:45');
$time->modify('+2 hours');

// Cette méthode modifie également l'instance $time
$this->someOtherFunction($time);

// La sorie ici est inconnue.
echo $time->format('Y-m-d H:i:s');

Si les appels aux méthodes sont réordonnés, ou si someOtherFunction évolue la sortie peut être inattendue. La mutabilité de vos objets crée un couplage temporal. Si nous utilisions des objets immutables, nous pourrions éviter ce type de problème:

use Cake\I18n\FrozenTime;
$time = new FrozenTime('2015-06-15 08:23:45');
$time = $time->modify('+2 hours');

// La modification de cette méthode ne change pas $time
$this->someOtherFunction($time);

// La sortie est connue.
echo $time->format('Y-m-d H:i:s');

Les Date et heures immutables sont utiles dans les entities car elles évitent les modifications accidentelles, et forcent les modifications à être explicitement exprimées. Utiliser des objets immutables aide l’ORM à mieux suivre les modifications et assurer que les colones date/datetime sont persistées correctement:

// Cette modification sera perdue lrsque l'article sera enregistré.
$article->updated->modify('+1 hour');

// En remplaçant l'objet time, la propriété sera auvegardée
$article->updated = $article->updated->modify('+1 hour');

Accepter des Données Requêtées Localisées

Quand vous créez des inputs de texte qui manipulent des dates, vous voudrez probablement accepter et parser des chaînes datetime localisées. Consultez Parser les Données Datetime Localisées.