Tests content moderation with translatable content entities.
public function testTranslatableContentEntities() {
$session = $this
->getSession();
$page = $session
->getPage();
$assert_session = $this
->assertSession();
$this
->drupalGet('node/add/paragraphed_test');
// Translation selection should be available on content creation page.
$option = $assert_session
->optionExists('langcode[0][value]', 'en');
$this
->assertTrue($option
->hasAttribute('selected'));
$assert_session
->optionExists('langcode[0][value]', 'de');
// Create a text paragraphs, a container paragraph with a text inside.
$page
->fillField('title[0][value]', 'Page 1 EN');
$page
->pressButton('Add text');
$page
->pressButton('Add container');
$page
->pressButton('field_paragraphs_1_subform_field_paragraphs_text_add_more');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'Initial paragraph text EN');
$page
->fillField('field_paragraphs[0][subform][field_untranslatable][0][value]', 'Untranslatable text');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'Initial paragraph container text EN');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_untranslatable][0][value]', 'Untranslatable container text');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->fillField('revision_log[0][value]', 'Node revision #1 EN');
$page
->pressButton('Save');
$assert_session
->pageTextContains('paragraphed_test Page 1 EN has been created.');
$assert_session
->pageTextContains('Initial paragraph text EN');
$assert_session
->pageTextContains('Untranslatable text');
$assert_session
->pageTextContains('Initial paragraph container text EN');
$assert_session
->pageTextContains('Untranslatable container text');
$host_node = $this
->getLastEntityOfType('node', TRUE);
$host_node_id = $host_node
->id();
// Create a translation.
$this
->drupalGet("/de/node/{$host_node_id}/translations/add/en/de");
$assert_session
->pageTextNotContains('Fields that apply to all languages are hidden to avoid conflicting changes');
$page
->fillField('title[0][value]', 'Page 1 DE');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'Initial paragraph text DE');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'Initial paragraph container text DE');
$assert_session
->fieldNotExists('field_paragraphs_1_subform_field_paragraphs_container_add_more');
$assert_session
->fieldNotExists('field_paragraphs[0][subform][field_untranslatable][0][value]');
$assert_session
->fieldNotExists('field_paragraphs[1][subform][field_paragraphs][0][subform][field_untranslatable][0][value]');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->pressButton('Save (this translation)');
$assert_session
->pageTextContains('paragraphed_test Page 1 DE has been updated.');
$assert_session
->pageTextContains('Initial paragraph text DE');
$assert_session
->pageTextContains('Untranslatable text');
$assert_session
->pageTextContains('Initial paragraph container text DE');
$assert_session
->pageTextContains('Untranslatable container text');
// Test the original translation.
$this
->drupalGet("/node/{$host_node_id}");
$assert_session
->pageTextContains('Page 1 EN');
$assert_session
->pageTextContains('Initial paragraph text EN');
$assert_session
->pageTextContains('Untranslatable text');
$assert_session
->pageTextContains('Initial paragraph container text EN');
$assert_session
->pageTextContains('Untranslatable container text');
// Create revision.
$this
->drupalGet("/de/node/{$host_node_id}/edit");
$page
->fillField('title[0][value]', 'Changed Page 1 DE');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'Changed paragraph text DE');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'Changed paragraph container text DE');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->pressButton('Save (this translation)');
$assert_session
->pageTextContains('paragraphed_test Changed Page 1 DE has been updated.');
$assert_session
->pageTextNotContains('Initial paragraph text DE');
$assert_session
->pageTextContains('Changed paragraph container text DE');
// Create revision draft for DE.
$this
->drupalGet("/de/node/{$host_node_id}/edit");
$page
->fillField('title[0][value]', 'Draft Page 1 DE');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'Draft paragraph text DE');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'Draft paragraph container text DE');
$page
->selectFieldOption('moderation_state[0][state]', 'draft');
$page
->pressButton('Save (this translation)');
$assert_session
->pageTextContains('paragraphed_test Draft Page 1 DE has been updated.');
$assert_session
->pageTextContains('Draft paragraph text DE');
$assert_session
->pageTextContains('Draft paragraph container text DE');
$this
->drupalGet("/de/node/{$host_node_id}");
$assert_session
->pageTextContains('Changed paragraph text DE');
$this
->drupalGet("/de/node/{$host_node_id}/latest");
$assert_session
->pageTextContains('Draft paragraph text DE');
// Create revision draft for EN.
$this
->drupalGet("node/{$host_node_id}/edit");
$page
->fillField('title[0][value]', 'Draft Page 1 EN');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'Draft paragraph text EN');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'Draft paragraph container text EN');
$page
->fillField('field_paragraphs[0][subform][field_untranslatable][0][value]', 'Untranslatable draft text');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_untranslatable][0][value]', 'Untranslatable container draft text');
$page
->selectFieldOption('moderation_state[0][state]', 'draft');
$page
->pressButton('Save (this translation)');
$assert_session
->pageTextContains('paragraphed_test Draft Page 1 EN has been updated.');
$assert_session
->pageTextContains('Draft paragraph text EN');
$assert_session
->pageTextContains('Untranslatable draft text');
$assert_session
->pageTextContains('Draft paragraph container text EN');
$assert_session
->pageTextContains('Untranslatable container draft text');
$this
->drupalGet("/node/{$host_node_id}");
$assert_session
->pageTextContains('Initial paragraph text EN');
$this
->drupalGet("/node/{$host_node_id}/latest");
$assert_session
->pageTextContains('Draft paragraph text EN');
// Assert the DE draft is still accessible.
$this
->drupalGet("/de/node/{$host_node_id}");
$assert_session
->pageTextContains('Changed paragraph text DE');
$this
->drupalGet("/de/node/{$host_node_id}/latest");
$assert_session
->pageTextContains('Draft paragraph text DE');
// Publish the EN draft.
$this
->drupalGet("/node/{$host_node_id}/latest");
$assert_session
->pageTextContains('Draft paragraph text EN');
$assert_session
->pageTextContains('Untranslatable draft text');
$page
->pressButton('Apply');
$assert_session
->pageTextContains('Draft paragraph text EN');
$assert_session
->pageTextContains('Untranslatable draft text');
// The untranslatable fields are really stored per translation revision too
// so the DE draft still has the old values for them.
$this
->drupalGet("/de/node/{$host_node_id}/latest");
$assert_session
->pageTextContains('Draft paragraph text DE');
$assert_session
->pageTextContains('Untranslatable text');
$assert_session
->pageTextContains('Draft paragraph container text DE');
$assert_session
->pageTextContains('Untranslatable container text');
// Publish the DE draft through the edit form.
$this
->drupalGet("/de/node/{$host_node_id}/edit");
$assert_session
->fieldValueEquals('field_paragraphs[0][subform][field_text][0][value]', 'Draft paragraph text DE');
$assert_session
->fieldValueEquals('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'Draft paragraph container text DE');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->pressButton('Save (this translation)');
$this
->drupalGet("/de/node/{$host_node_id}");
$assert_session
->pageTextContains('Draft paragraph text DE');
$assert_session
->pageTextContains('Untranslatable draft text');
$assert_session
->pageTextContains('Draft paragraph container text DE');
$assert_session
->pageTextContains('Untranslatable container draft text');
// Assert that the EN translation as not affected.
$this
->drupalGet("/node/{$host_node_id}");
$assert_session
->pageTextContains('Draft paragraph text EN');
$assert_session
->pageTextContains('Untranslatable draft text');
$assert_session
->pageTextContains('Draft paragraph container text EN');
$assert_session
->pageTextContains('Untranslatable container draft text');
// Revert the DE translation to the previous state, ensure that the EN
// translation does not get reverted with it.
$this
->drupalGet("/node/{$host_node_id}/revisions");
$this
->drupalGet("/de/node/{$host_node_id}/revisions");
// The revision lists 4 german revisions, the current, the draft, the
// changed and the original revision. Revert to the changed revision, which
// is the second Revert link on the page.
$this
->clickLink('Revert', 1);
$page
->pressButton('Revert');
$this
->drupalGet("/de/node/{$host_node_id}");
$assert_session
->pageTextContains('Changed paragraph text DE');
$assert_session
->pageTextContains('Untranslatable draft text');
$assert_session
->pageTextContains('Changed paragraph container text DE');
$assert_session
->pageTextContains('Untranslatable container draft text');
// Assert that the EN translation as not affected.
$this
->drupalGet("/node/{$host_node_id}");
$assert_session
->pageTextContains('Draft paragraph text EN');
$assert_session
->pageTextContains('Untranslatable draft text');
$assert_session
->pageTextContains('Draft paragraph container text EN');
$assert_session
->pageTextContains('Untranslatable container draft text');
/** @var \Drupal\Core\Entity\EntityDisplayRepositoryInterface $display_repository */
$display_repository = \Drupal::service('entity_display.repository');
$display_repository
->getFormDisplay('node', 'paragraphed_test')
->setComponent('field_paragraphs', [
'type' => 'entity_reference_paragraphs',
'settings' => [
'edit_mode' => 'open',
'add_mode' => 'dropdown',
'form_display_mode' => 'default',
'default_paragraph_type' => '_none',
],
])
->save();
$display_repository
->getFormDisplay('paragraph', 'container')
->setComponent('field_paragraphs', [
'type' => 'entity_reference_paragraphs',
'settings' => [
'edit_mode' => 'open',
'add_mode' => 'dropdown',
'form_display_mode' => 'default',
'default_paragraph_type' => '_none',
],
])
->save();
// @TODO when https://www.drupal.org/project/paragraphs/issues/3004099 gets
// committed, update the following two scenarios.
// When an EN node is published, we add a draft translation and we edit the
// original EN adding a new Paragraph and keeping it published, the new
// created Paragraph should appear in the translation draft too.
// Create a new published EN node with a paragraph.
$this
->drupalGet('node/add/paragraphed_test');
$page
->fillField('title[0][value]', 'Moderation test 1 EN');
$page
->pressButton('Add text');
$page
->pressButton('Add container');
$page
->pressButton('field_paragraphs_1_subform_field_paragraphs_text_add_more');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'EN First level text');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'EN Second level text');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->fillField('revision_log[0][value]', 'Revision 1 EN');
$page
->pressButton('Save');
$assert_session
->pageTextContains('paragraphed_test Moderation test 1 EN has been created.');
$assert_session
->pageTextContains('EN First level text');
$assert_session
->pageTextContains('EN Second level text');
$node = $this
->getLastEntityOfType('node', TRUE);
$node = $node
->id();
// Create a draft translation.
$this
->drupalGet("/de/node/{$node}/translations/add/en/de");
$page
->fillField('title[0][value]', 'Moderation test 1 DE');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'DE First level text');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'DE Second level text');
$page
->selectFieldOption('moderation_state[0][state]', 'draft');
$page
->pressButton('Save (this translation)');
$assert_session
->pageTextContains('paragraphed_test Moderation test 1 DE has been updated.');
$assert_session
->pageTextContains('DE First level text');
$assert_session
->pageTextContains('DE Second level text');
// Change the structure of Paragraphs on the new published EN node.
$this
->drupalGet("/node/{$node}/edit");
$assert_session
->pageTextContains('Moderation test 1 EN');
$page
->pressButton('field_paragraphs_0_remove');
$page
->pressButton('field_paragraphs_1_subform_field_paragraphs_text_add_more');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'EN Draft second level text 1');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][1][subform][field_text][0][value]', 'EN Draft second level text 2');
$page
->fillField('revision_log[0][value]', 'Revision 1 EN');
$page
->pressButton('Save');
$assert_session
->pageTextContains('paragraphed_test Moderation test 1 EN has been updated.');
$assert_session
->pageTextContains('EN Draft second level text 1');
$assert_session
->pageTextContains('EN Draft second level text 2');
// Assert that the draft DE translation has the new paragraph structure.
$this
->drupalGet("/de/node/{$node}/edit");
$assert_session
->fieldExists('field_paragraphs[0][subform][field_paragraphs][0][subform][field_text][0][value]');
$assert_session
->fieldExists('field_paragraphs[0][subform][field_paragraphs][1][subform][field_text][0][value]');
$page
->fillField('field_paragraphs[0][subform][field_paragraphs][0][subform][field_text][0][value]', 'DE Draft second level text 1');
$page
->fillField('field_paragraphs[0][subform][field_paragraphs][1][subform][field_text][0][value]', 'DE Draft second level text 2');
$page
->selectFieldOption('moderation_state[0][state]', 'draft');
$page
->pressButton('Save (this translation)');
$assert_session
->pageTextContains('paragraphed_test Moderation test 1 DE has been updated.');
$assert_session
->pageTextContains('DE Draft second level text 1');
$assert_session
->pageTextContains('DE Draft second level text 2');
// Create a published EN node, add a draft with a different Paragraphs
// structure. When translating, the translation should have the same
// Paragraphs structure as the last published EN node.
// Create a new published EN node with a paragraph.
$this
->drupalGet('node/add/paragraphed_test');
$page
->fillField('title[0][value]', 'EN Moderation');
$page
->pressButton('Add text');
$page
->pressButton('Add container');
$page
->pressButton('field_paragraphs_1_subform_field_paragraphs_text_add_more');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'EN First level text');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'EN Second level text');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->fillField('revision_log[0][value]', 'Revision 1 EN');
$page
->pressButton('Save');
$assert_session
->pageTextContains('paragraphed_test EN Moderation has been created.');
$assert_session
->pageTextContains('EN First level text');
$assert_session
->pageTextContains('EN Second level text');
$node = $this
->getLastEntityOfType('node', TRUE);
$node = $node
->id();
// Create an EN draft of the node.
$this
->drupalGet("node/{$node}/edit");
$page
->pressButton('field_paragraphs_0_remove');
$page
->pressButton('field_paragraphs_1_subform_field_paragraphs_text_add_more');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'EN Draft second level first text');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][1][subform][field_text][0][value]', 'EN Draft second level second text');
$page
->selectFieldOption('moderation_state[0][state]', 'draft');
$page
->fillField('revision_log[0][value]', 'Revision 2 EN');
$page
->pressButton('Save');
$assert_session
->pageTextContains('paragraphed_test EN Moderation has been updated.');
$assert_session
->pageTextContains('EN Draft second level first text');
$assert_session
->pageTextContains('EN Draft second level second text');
$node = $this
->getLastEntityOfType('node', TRUE);
$node = $node
->id();
// Create a translation and save, it should have the same structure as the
// published EN node.
$this
->drupalGet("/de/node/{$node}/translations/add/en/de");
$page
->fillField('title[0][value]', 'DE Moderation');
$page
->fillField('field_paragraphs[0][subform][field_text][0][value]', 'DE First level text');
$page
->fillField('field_paragraphs[1][subform][field_paragraphs][0][subform][field_text][0][value]', 'DE Second level text');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->pressButton('Save (this translation)');
$assert_session
->pageTextNotContains('Non-translatable fields can only be changed when updating the original language.');
$assert_session
->pageTextContains('paragraphed_test DE Moderation has been updated.');
$assert_session
->pageTextContains('DE First level text');
$assert_session
->pageTextContains('DE Second level text');
// Publish the EN draft.
$this
->drupalGet("node/{$node}/edit");
$page
->fillField('field_paragraphs[0][subform][field_paragraphs][0][subform][field_text][0][value]', 'EN Second level first text');
$page
->fillField('field_paragraphs[0][subform][field_paragraphs][1][subform][field_text][0][value]', 'EN Second level second text');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->pressButton('Save');
$assert_session
->pageTextNotContains('Non-translatable fields can only be changed when updating the original language.');
$assert_session
->pageTextContains('paragraphed_test EN Moderation has been updated.');
$assert_session
->pageTextContains('EN Second level first text');
$assert_session
->pageTextContains('EN Second level second text');
// Assert that the translation node has the same structure as the new
// published node.
$this
->drupalGet("de/node/{$node}/edit");
$page
->fillField('field_paragraphs[0][subform][field_paragraphs][0][subform][field_text][0][value]', 'DE Second level first text');
$page
->fillField('field_paragraphs[0][subform][field_paragraphs][1][subform][field_text][0][value]', 'DE Second level second text');
$page
->selectFieldOption('moderation_state[0][state]', 'published');
$page
->pressButton('Save');
$assert_session
->pageTextNotContains('Non-translatable fields can only be changed when updating the original language.');
$assert_session
->pageTextContains('paragraphed_test DE Moderation has been updated.');
$assert_session
->pageTextContains('DE Second level first text');
$assert_session
->pageTextContains('DE Second level second text');
}