Support to use the datetime-local element

Created on 8 February 2025, about 2 months ago

Problem/Motivation

I am trying to implement a datetime-local element but I can't get it to work

$form['datetimepicker'] = [
  '#type' => 'datetime',
  '#date_date_element' => 'datetime-local',
  '#date_date_format' => 'Y-m-d\TH:i',
  '#date_time_element' => 'none',
];

The element is correctly displayed but when I submit the form I always have an error.
I tried several '#data_date_format' without success.

All my tests led me to the catch (\Exception $e) in Datetime::valueCallback()
with the error "The date cannot be created from a format."

Can you please give me some help on this?

Steps to reproduce

Proposed resolution

Remaining tasks

User interface changes

Introduced terminology

API changes

Data model changes

Release notes snippet

πŸ’¬ Support request
Status

Active

Version

11.1 πŸ”₯

Component

datetime.module

Created by

πŸ‡«πŸ‡·France MacSim

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Merge Requests

Comments & Activities

  • Issue created by @MacSim
  • The code throwing the exception is:

    $date = \DateTime::createFromFormat($format, $time, $datetime_plus->getTimezone());
    if (!$date instanceof \DateTime) {
        throw new \InvalidArgumentException('The date cannot be created from a format.');
    }
    

    It is in DateTimePlus::createFromFormat. This code returns `false` in the PHP REPL:

    (\DateTime::createFromFormat('Y-m-d\TH:i', 'now'));
    

    So I think your time format string is invalid to the \DateTime library. I hope this helps.

  • Actually, 'now' can never work. It's just the case that the format has to match the input. What is the input in the cases where this doesn't work?

    This works: \DateTime::createFromFormat('Y-m-d H:i', '2008-06-01 12:50')

    I think \T is totally unsupported. Anyway, this is PHP's standard library, not Drupal.

  • πŸ‡«πŸ‡·France MacSim

    Well then it seems that there's a bug somewhere

    In Datetime::processDatetime() doc says:

      /**
       * [...]
       * #date_date_format: A date format string that describes the format that
       *     should be displayed to the end user for the date. When using HTML5
       *     elements the format MUST use the appropriate HTML5 format for that
       *     element, no other format will work. See the
       *     DateFormatterInterface::format() function for a list of the possible
       *     formats and HTML5 standards for the HTML5 requirements. Defaults to the
       *     right HTML5 format for the chosen element if an HTML5 element is used,
       *     otherwise defaults to DateFormat::load('html_date')->getPattern().
       * [...]
       */
    

    And in DateFormatterInterface::format() :

      /**
       * [...]
       * @param string $type
       *   (optional) The format to use, one of:
       *   - One of the built-in formats: 'short', 'medium',
       *     'long', 'html_datetime', 'html_date', 'html_time',
       *     'html_yearless_date', 'html_week', 'html_month', 'html_year'.
       *   - The name of a date type defined by a date format config entity.
       *   - The machine name of an administrator-defined date format.
       *   - 'custom', to use $format.
       *   Defaults to 'medium'.
       * @param string $format
       *   (optional) If $type is 'custom', a PHP date format string suitable for
       *   input to date(). Use a backslash to escape ordinary text, so it does not
       *   get interpreted as date format characters.
       * [...]
       */
    

    No matter what the value I give to that "#date_date_format" ("long", "short", "medium", 'html_datetime', "d/m/Y H:i", "Y-m-d H:i", etc.),

    1. the input is always asking for a "d/m/Y H:i" as you can see on the attached screenshot
    2. when I choose a date and time and submit the form, I am always reaching Datetime::valueCallback() with the following values:
      • $input =
         array:1 [β–Ό
          "date" => "2025-02-09T12:00"
        ]
        
      • $date_input = "2025-02-09T12:00"
      • $time_input = ""
      • $date_format = "Y-m-d\TH:i:sO"
      • $time_format = ""
    3. Then I am always catching the Exception with the message "The date cannot be created from a format."
  • πŸ‡³πŸ‡ΏNew Zealand quietone

    Changes are made on on 11.x (our main development branch) first, and are then back ported as needed according to the Core change policies β†’ .

  • πŸ‡«πŸ‡·France MacSim

    Sorry quietone #8 I didn't realised I changed the version

    Ok now I undestand why I've got this behavior no matter what I pass to '#date_date_format'
    It's because of
    $date_format = $element['#date_date_format'] != 'none' ? static::getHtml5DateFormat($element) : '';

    protected static function getHtml5DateFormat($element) {
      switch ($element['#date_date_element']) {
        case 'date':
          return DateFormat::load('html_date')->getPattern();
    
        case 'datetime':
        case 'datetime-local':
          return DateFormat::load('html_datetime')->getPattern();
    
        default:
          return $element['#date_date_format'];
      }
    }
    

    and as cilefen said in #4, 'html_datetime' (ie. "Y-m-d\TH:i") is not a valid format to pass to \DateTime::createFromFormat()

  • πŸ‡«πŸ‡·France MacSim

    MR provides a fix for the ValueCallback() method.

    Few thoughts:
    - We should maybe find a way to fix it in a less-dirty way (ie. not with that ugly "if" condition)
    - It only fixes the problem for the datetime-local element but datetime element also uses the 'html_datetime' format and would break the same way
    - Actual tests are failing - one error has no link with what has been commited: Behat\Mink\Exception\ExpectationException: The string "The text format ckeditor5 has been updated." was not found anywhere in the HTML response of the current page. (https://git.drupalcode.org/issue/drupal-3505318/-/jobs/4297615) ; the other one from SettingsTrayBlockFormTest.php (https://git.drupalcode.org/issue/drupal-3505318/-/jobs/4297616) might need more code changes

  • Pipeline finished with Success
    about 2 months ago
    Total: 1714s
    #419333
  • Pipeline finished with Success
    about 2 months ago
    Total: 1166s
    #419459
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Assuming NR for feedback for approach? Could the issue summary be filled out to clear up for proposed solution. Easier for reviews

    It only fixes the problem for the datetime-local element but datetime element also uses the 'html_datetime' format and would break the same way

    I'm 50/50 that this should be fixed here too or could be it's own issue. Guess it depends if the code is the same spot.

    We should introduce tests

    100% thanks for tagging for that!

  • πŸ‡«πŸ‡·France MacSim

    #13 Yes I wanted some feedback, thank you.

    I updated the issue summary ; hopefully it is clear enough and will help resolving the issue.

  • πŸ‡«πŸ‡·France MacSim

    Added kernel tests for datetime-local

  • Pipeline finished with Failed
    about 2 months ago
    Total: 160s
    #427992
  • Pipeline finished with Canceled
    about 2 months ago
    Total: 176s
    #427997
  • Pipeline finished with Failed
    about 2 months ago
    Total: 1270s
    #428001
  • Pipeline finished with Success
    about 2 months ago
    Total: 417s
    #428028
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Left comments on MR.

  • Pipeline finished with Success
    about 1 month ago
    Total: 982s
    #428979
  • Pipeline finished with Success
    about 1 month ago
    Total: 402s
    #428996
  • πŸ‡«πŸ‡·France MacSim

    Thanks smustgrave.
    I applied your suggestions except for the first one since it's how it's done on other core KernelTests.
    IMHO it will be easier to maintain the "form + tests" mix rather than to declare the form in another location even if it sounds prettier.

  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Thanks, yea personally think it's messy and think the current examples of it aren't good standards. So I'll leave in review for others to take a look.

  • πŸ‡«πŸ‡·France MacSim
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Won't hold it up longer, will see what the committers think.

Production build 0.71.5 2024