diff --git a/core/modules/datetime/src/Tests/DateTimeFieldTest.php b/core/modules/datetime/src/Tests/DateTimeFieldTest.php
index bf3ba07..ce598bf 100644
--- a/core/modules/datetime/src/Tests/DateTimeFieldTest.php
+++ b/core/modules/datetime/src/Tests/DateTimeFieldTest.php
@@ -54,17 +54,33 @@ class DateTimeFieldTest extends WebTestBase {
   protected $field;
 
   /**
+   * An array of timezone extremes to test.
+   *
+   * @var string[]
+   */
+  protected static $timezones = [
+    // UTC-12, no DST.
+    'Pacific/Kwajalein',
+    // UTC-11, no DST
+    'Pacific/Midway',
+    // UTC-7, no DST.
+    'America/Phoenix',
+    // UTC.
+    'UTC',
+    // UTC+5:30, no DST.
+    'Asia/Kolkata',
+    // UTC+12, no DST
+    'Pacific/Funafuti',
+    // UTC+13, no DST.
+    'Pacific/Tongatapu',
+  ];
+
+  /**
    * {@inheritdoc}
    */
   protected function setUp() {
     parent::setUp();
 
-    // Set an explicit site timezone, and disallow per-user timezones.
-    $this->config('system.date')
-      ->set('timezone.user.configurable', 0)
-      ->set('timezone.default', 'Asia/Tokyo')
-      ->save();
-
     $web_user = $this->drupalCreateUser(array(
       'access content',
       'view test entity',
@@ -117,128 +133,155 @@ protected function setUp() {
   function testDateField() {
     $field_name = $this->fieldStorage->getName();
 
-    // Display creation form.
-    $this->drupalGet('entity_test/add');
-    $this->assertFieldByName("{$field_name}[0][value][date]", '', 'Date element found.');
-    $this->assertFieldByXPath('//*[@id="edit-' . $field_name . '-wrapper"]/h4[contains(@class, "js-form-required")]', TRUE, 'Required markup found');
-    $this->assertNoFieldByName("{$field_name}[0][value][time]", '', 'Time element not found.');
-
-    // Build up a date in the UTC timezone.
-    $value = '2012-12-31 00:00:00';
-    $date = new DrupalDateTime($value, 'UTC');
+    // Loop through defined timezones to test that date-only fields work at the
+    // extremes.
+    foreach (static::$timezones as $timezone) {
 
-    // The expected values will use the default time.
-    datetime_date_default_time($date);
+      $this->setSiteTimezone($timezone);
 
-    // Update the timezone to the system default.
-    $date->setTimezone(timezone_open(drupal_get_user_timezone()));
-
-    // Submit a valid date and ensure it is accepted.
-    $date_format = DateFormat::load('html_date')->getPattern();
-    $time_format = DateFormat::load('html_time')->getPattern();
-
-    $edit = array(
-      "{$field_name}[0][value][date]" => $date->format($date_format),
-    );
-    $this->drupalPostForm(NULL, $edit, t('Save'));
-    preg_match('|entity_test/manage/(\d+)|', $this->url, $match);
-    $id = $match[1];
-    $this->assertText(t('entity_test @id has been created.', array('@id' => $id)));
-    $this->assertRaw($date->format($date_format));
-    $this->assertNoRaw($date->format($time_format));
-
-    // Verify that the date is output according to the formatter settings.
-    $options = array(
-      'format_type' => array('short', 'medium', 'long'),
-    );
-    foreach ($options as $setting => $values) {
-      foreach ($values as $new_value) {
-        // Update the entity display settings.
-        $this->displayOptions['settings'] = array($setting => $new_value) + $this->defaultSettings;
-        entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
-          ->setComponent($field_name, $this->displayOptions)
-          ->save();
-
-        $this->renderTestEntity($id);
-        switch ($setting) {
-          case 'format_type':
-            // Verify that a date is displayed.
-            $expected = format_date($date->getTimestamp(), $new_value);
-            $expected_iso = format_date($date->getTimestamp(), 'custom', 'Y-m-d\TH:i:s\Z', 'UTC');
-            $this->renderTestEntity($id);
-            $this->assertFieldByXPath('//time[@datetime="' . $expected_iso . '"]', $expected, SafeMarkup::format('Formatted date field using %value format displayed as %expected with %expected_iso attribute.', array('%value' => $new_value, '%expected' => $expected, '%expected_iso' => $expected_iso)));
-            break;
+      // Display creation form.
+      $this->drupalGet('entity_test/add');
+      $this->assertFieldByName("{$field_name}[0][value][date]", '', 'Date element found.');
+      $this->assertFieldByXPath('//*[@id="edit-' . $field_name . '-wrapper"]/h4[contains(@class, "js-form-required")]', TRUE, 'Required markup found');
+      $this->assertNoFieldByName("{$field_name}[0][value][time]", '', 'Time element not found.');
+
+      // Build up a date in the UTC timezone. Note that using this will also
+      // mimic the user in a different timezone simply entering '2012-12-31' via
+      // the UI.
+      $value = '2012-12-31 00:00:00';
+      $date = new DrupalDateTime($value, DATETIME_STORAGE_TIMEZONE);
+
+      // Submit a valid date and ensure it is accepted.
+      $date_format = DateFormat::load('html_date')->getPattern();
+      $time_format = DateFormat::load('html_time')->getPattern();
+
+      $edit = array(
+        "{$field_name}[0][value][date]" => $date->format($date_format),
+      );
+      $this->drupalPostForm(NULL, $edit, t('Save'));
+      preg_match('|entity_test/manage/(\d+)|', $this->url, $match);
+      $id = $match[1];
+      $this->assertText(t('entity_test @id has been created.', array('@id' => $id)));
+      $this->assertRaw($date->format($date_format));
+      $this->assertNoRaw($date->format($time_format));
+
+      // Verify the date doesn't change if using a timezone that is UTC+12 when
+      // the entity is edited through the form.
+      $entity = EntityTest::load($id);
+      $this->assertEqual('2012-12-31', $entity->{$field_name}->value);
+      $this->drupalGet('entity_test/manage/' . $id . '/edit');
+      $this->drupalPostForm(NULL, [], t('Save'));
+      $this->drupalGet('entity_test/manage/' . $id . '/edit');
+      $this->drupalPostForm(NULL, [], t('Save'));
+      $this->drupalGet('entity_test/manage/' . $id . '/edit');
+      $this->drupalPostForm(NULL, [], t('Save'));
+      $entity = EntityTest::load($id);
+      $this->assertEqual('2012-12-31', $entity->{$field_name}->value);
+
+      // Reset display options since these get changed below.
+      $this->displayOptions = array(
+        'type' => 'datetime_default',
+        'label' => 'hidden',
+        'settings' => array('format_type' => 'medium') + $this->defaultSettings,
+      );
+      // Verify that the date is output according to the formatter settings.
+      $options = array(
+        'format_type' => array('short', 'medium', 'long'),
+      );
+      // Formats that display a time component for date-only fields will display
+      // the default time, so that is applied before calculating the expected
+      // value.
+      datetime_date_default_time($date);
+      foreach ($options as $setting => $values) {
+        foreach ($values as $new_value) {
+          // Update the entity display settings.
+          $this->displayOptions['settings'] = array($setting => $new_value) + $this->defaultSettings;
+          entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
+            ->setComponent($field_name, $this->displayOptions)
+            ->save();
+
+          $this->renderTestEntity($id);
+          switch ($setting) {
+            case 'format_type':
+              // Verify that a date is displayed. Since this is a date-only
+              // field, it is expected to display the time as 00:00:00.
+              $expected = format_date($date->getTimestamp(), $new_value, '', DATETIME_STORAGE_TIMEZONE);
+              $expected_iso = format_date($date->getTimestamp(), 'custom', 'Y-m-d\TH:i:s\Z', DATETIME_STORAGE_TIMEZONE);
+              $this->renderTestEntity($id);
+              $this->assertFieldByXPath('//time[@datetime="' . $expected_iso . '"]', $expected, SafeMarkup::format('Formatted date field using %value format displayed as %expected with %expected_iso attribute.', array('%value' => $new_value, '%expected' => $expected, '%expected_iso' => $expected_iso)));
+              break;
+          }
         }
       }
-    }
 
-    // Verify that the plain formatter works.
-    $this->displayOptions['type'] = 'datetime_plain';
-    $this->displayOptions['settings'] = $this->defaultSettings;
-    entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
-      ->setComponent($field_name, $this->displayOptions)
-      ->save();
-    $expected = $date->format(DATETIME_DATE_STORAGE_FORMAT);
-    $this->renderTestEntity($id);
-    $this->assertText($expected, SafeMarkup::format('Formatted date field using plain format displayed as %expected.', array('%expected' => $expected)));
-
-    // Verify that the 'datetime_custom' formatter works.
-    $this->displayOptions['type'] = 'datetime_custom';
-    $this->displayOptions['settings'] = array('date_format' => 'm/d/Y') + $this->defaultSettings;
-    entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
-      ->setComponent($field_name, $this->displayOptions)
-      ->save();
-    $expected = $date->format($this->displayOptions['settings']['date_format']);
-    $this->renderTestEntity($id);
-    $this->assertText($expected, SafeMarkup::format('Formatted date field using datetime_custom format displayed as %expected.', array('%expected' => $expected)));
-
-    // Verify that the 'datetime_time_ago' formatter works for intervals in the
-    // past.  First update the test entity so that the date difference always
-    // has the same interval.  Since the database always stores UTC, and the
-    // interval will use this, force the test date to use UTC and not the local
-    // or user timezome.
-    $timestamp = REQUEST_TIME - 87654321;
-    $entity = EntityTest::load($id);
-    $field_name = $this->fieldStorage->getName();
-    $date = DrupalDateTime::createFromTimestamp($timestamp, 'UTC');
-    $entity->{$field_name}->value = $date->format($date_format);
-    $entity->save();
-
-    $this->displayOptions['type'] = 'datetime_time_ago';
-    $this->displayOptions['settings'] = array(
-      'future_format' => '@interval in the future',
-      'past_format' => '@interval in the past',
-      'granularity' => 3,
-    );
-    entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
-      ->setComponent($field_name, $this->displayOptions)
-      ->save();
-    $expected = SafeMarkup::format($this->displayOptions['settings']['past_format'], [
-      '@interval' => \Drupal::service('date.formatter')->formatTimeDiffSince($timestamp, ['granularity' => $this->displayOptions['settings']['granularity']])
-    ]);
-    $this->renderTestEntity($id);
-    $this->assertText($expected, SafeMarkup::format('Formatted date field using datetime_time_ago format displayed as %expected.', array('%expected' => $expected)));
-
-    // Verify that the 'datetime_time_ago' formatter works for intervals in the
-    // future.  First update the test entity so that the date difference always
-    // has the same interval.  Since the database always stores UTC, and the
-    // interval will use this, force the test date to use UTC and not the local
-    // or user timezome.
-    $timestamp = REQUEST_TIME + 87654321;
-    $entity = EntityTest::load($id);
-    $field_name = $this->fieldStorage->getName();
-    $date = DrupalDateTime::createFromTimestamp($timestamp, 'UTC');
-    $entity->{$field_name}->value = $date->format($date_format);
-    $entity->save();
-
-    entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
-      ->setComponent($field_name, $this->displayOptions)
-      ->save();
-    $expected = SafeMarkup::format($this->displayOptions['settings']['future_format'], [
-      '@interval' => \Drupal::service('date.formatter')->formatTimeDiffUntil($timestamp, ['granularity' => $this->displayOptions['settings']['granularity']])
-    ]);
-    $this->renderTestEntity($id);
-    $this->assertText($expected, SafeMarkup::format('Formatted date field using datetime_time_ago format displayed as %expected.', array('%expected' => $expected)));
+      // Verify that the plain formatter works.
+      $this->displayOptions['type'] = 'datetime_plain';
+      $this->displayOptions['settings'] = $this->defaultSettings;
+      entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
+        ->setComponent($field_name, $this->displayOptions)
+        ->save();
+      $expected = $date->format(DATETIME_DATE_STORAGE_FORMAT);
+      $this->renderTestEntity($id);
+      $this->assertText($expected, SafeMarkup::format('Formatted date field using plain format displayed as %expected.', array('%expected' => $expected)));
+
+      // Verify that the 'datetime_custom' formatter works.
+      $this->displayOptions['type'] = 'datetime_custom';
+      $this->displayOptions['settings'] = array('date_format' => 'm/d/Y') + $this->defaultSettings;
+      entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
+        ->setComponent($field_name, $this->displayOptions)
+        ->save();
+      $expected = $date->format($this->displayOptions['settings']['date_format']);
+      $this->renderTestEntity($id);
+      $this->assertText($expected, SafeMarkup::format('Formatted date field using datetime_custom format displayed as %expected.', array('%expected' => $expected)));
+
+      // Verify that the 'datetime_time_ago' formatter works for intervals in the
+      // past.  First update the test entity so that the date difference always
+      // has the same interval.  Since the database always stores UTC, and the
+      // interval will use this, force the test date to use UTC and not the local
+      // or user timezome.
+      $timestamp = REQUEST_TIME - 87654321;
+      $entity = EntityTest::load($id);
+      $field_name = $this->fieldStorage->getName();
+      $date = DrupalDateTime::createFromTimestamp($timestamp, 'UTC');
+      $entity->{$field_name}->value = $date->format($date_format);
+      $entity->save();
+
+      $this->displayOptions['type'] = 'datetime_time_ago';
+      $this->displayOptions['settings'] = array(
+        'future_format' => '@interval in the future',
+        'past_format' => '@interval in the past',
+        'granularity' => 3,
+      );
+      entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
+        ->setComponent($field_name, $this->displayOptions)
+        ->save();
+      $expected = SafeMarkup::format($this->displayOptions['settings']['past_format'], [
+        '@interval' => \Drupal::service('date.formatter')->formatTimeDiffSince($timestamp, ['granularity' => $this->displayOptions['settings']['granularity']])
+      ]);
+      $this->renderTestEntity($id);
+      $this->assertText($expected, SafeMarkup::format('Formatted date field using datetime_time_ago format displayed as %expected.', array('%expected' => $expected)));
+
+      // Verify that the 'datetime_time_ago' formatter works for intervals in the
+      // future.  First update the test entity so that the date difference always
+      // has the same interval.  Since the database always stores UTC, and the
+      // interval will use this, force the test date to use UTC and not the local
+      // or user timezome.
+      $timestamp = REQUEST_TIME + 87654321;
+      $entity = EntityTest::load($id);
+      $field_name = $this->fieldStorage->getName();
+      $date = DrupalDateTime::createFromTimestamp($timestamp, 'UTC');
+      $entity->{$field_name}->value = $date->format($date_format);
+      $entity->save();
+
+      entity_get_display($this->field->getTargetEntityTypeId(), $this->field->getTargetBundle(), 'full')
+        ->setComponent($field_name, $this->displayOptions)
+        ->save();
+      $expected = SafeMarkup::format($this->displayOptions['settings']['future_format'], [
+        '@interval' => \Drupal::service('date.formatter')->formatTimeDiffUntil($timestamp, ['granularity' => $this->displayOptions['settings']['granularity']])
+      ]);
+      $this->renderTestEntity($id);
+      $this->assertText($expected, SafeMarkup::format('Formatted date field using datetime_time_ago format displayed as %expected.', array('%expected' => $expected)));
+    }
   }
 
   /**
@@ -874,4 +917,19 @@ protected function renderTestEntity($id, $view_mode = 'full', $reset = TRUE) {
     $this->verbose($output);
   }
 
+  /**
+   * Sets the site timezone to a given timezone.
+   *
+   * @param string $timezone
+   *   The timezone identifier to set.
+   */
+  protected function setSiteTimezone($timezone) {
+    // Set an explicit site timezone, and disallow per-user timezones.
+    $this->config('system.date')
+      ->set('timezone.user.configurable', 0)
+      // A timezone with an offset greater than UTC+12 is used.
+      ->set('timezone.default', $timezone)
+      ->save();
+  }
+
 }
