File "components-repository.php"
Full Path: /home/adniftyx/public_html/wp-content/plugins/elementor/modules/components/components-repository.php
File size: 5.76 KB
MIME-type: text/x-php
Charset: utf-8
<?php
namespace Elementor\Modules\Components;
use Elementor\Core\Utils\Collection;
use Elementor\Modules\Components\Documents\Component as Component_Document;
use Elementor\Plugin;
use Elementor\Core\Base\Document;
if ( ! defined( 'ABSPATH' ) ) {
exit; // Exit if accessed directly.
}
class Components_Repository {
public static function make(): Components_Repository {
return new self();
}
public function all(): Collection {
// Components count is limited to 100, if we increase this number, we need to iterate the posts in batches.
$posts = get_posts( [
'post_type' => Component_Document::TYPE,
'post_status' => 'any',
'posts_per_page' => Components_REST_API::MAX_COMPONENTS,
] );
$components = [];
foreach ( $posts as $post ) {
$component = $this->get( $post->ID );
if ( ! $component ) {
continue;
}
$components[] = [
'id' => $component->get_main_id(),
'title' => $component->get_post()->post_title,
'uid' => $component->get_component_uid(),
'is_archived' => $component->get_is_archived(),
'styles' => $this->extract_styles( $component->get_elements_data() ),
];
}
return Collection::make( $components );
}
public function get( $id, bool $include_autosave = true ) {
$doc = $include_autosave
? Plugin::$instance->documents->get_doc_or_auto_save( $id, get_current_user_id() )
: Plugin::$instance->documents->get( $id );
if ( ! $doc instanceof Component_Document ) {
return null;
}
return $doc;
}
public function create( string $title, array $content, string $status, string $uid, array $settings = [] ) {
$document = Plugin::$instance->documents->create(
Component_Document::get_type(),
[
'post_title' => $title,
'post_status' => $status,
],
[
Component_Document::COMPONENT_UID_META_KEY => $uid,
]
);
try {
$saved = $document->save( [
'elements' => $content,
'settings' => $settings,
] );
} catch ( \Exception $e ) {
$document->force_delete();
throw $e;
}
if ( ! $saved ) {
$document->force_delete();
throw new \Exception( 'Failed to create component' );
}
return $document->get_main_id();
}
private function extract_styles( array $elements, array $styles = [] ) {
foreach ( $elements as $element ) {
if ( isset( $element['styles'] ) ) {
$styles = array_merge( $styles, $element['styles'] );
}
if ( isset( $element['elements'] ) ) {
$styles = $this->extract_styles( $element['elements'], $styles );
}
}
return $styles;
}
public function archive( array $ids, string $status ) {
$failed_ids = [];
$success_ids = [];
foreach ( $ids as $id ) {
try {
$component = $this->get_component_for_edit( $id, $status );
if ( ! $component ) {
$failed_ids[] = $id;
continue;
}
$component->archive();
$success_ids[] = $id;
} catch ( \Exception $e ) {
$failed_ids[] = $id;
}
}
return [
'failedIds' => $failed_ids,
'successIds' => $success_ids,
];
}
public function update_title( int $component_id, string $title, string $status ): bool {
$component = $this->get_component_for_edit( $component_id, $status );
if ( ! $component ) {
return false;
}
return $component->update_title( $title );
}
/**
* Get the component for edit.
*
* @param int $component_id The component ID.
* @param string $target_status The target status, means the status the component should be saved as.
* @return ?Component_Document The component document for edit.
*
* If target status is an autosave / draft:
* - If the component main document is autosave / draft, it will return the main document.
* - If the component main document is published, it will create a new autosave document and return it.
* If target status is publish:
* - Will return the main document. If it's an autosave, it will be published later by the publish_component method.
*/
private function get_component_for_edit( int $component_id, string $target_status ): ?Component_Document {
$component = $this->get( $component_id );
if ( ! $component ) {
return null;
}
$autosave_statuses = [ Document::STATUS_AUTOSAVE, Document::STATUS_DRAFT ];
$autosave_exists = $component->is_autosave();
$should_create_autosave = in_array( $target_status, $autosave_statuses, true ) && ! $autosave_exists;
if ( ! $should_create_autosave ) {
return $component;
}
// Create a new autosave document, based on the published version.
return $component->get_autosave( 0, true );
}
public function publish_component( Component_Document $component ): bool {
try {
$main_id = $component->get_main_id();
$main_component = $this->get( $main_id, false );
$autosave = $main_component->get_newer_autosave();
if ( $autosave ) {
$success = $this->copy_autosave_data_to_main_component_document_and_publish( $autosave, $main_component, $main_id );
} else {
$success = $main_component->update_status( Document::STATUS_PUBLISH );
}
if ( ! $success ) {
throw new \Exception( 'Failed to publish component' );
}
} catch ( \Exception $e ) {
return false;
}
return true;
}
private function copy_autosave_data_to_main_component_document_and_publish( Component_Document $autosave, Component_Document $main_component_document, int $main_id ): bool {
$autosave_id = $autosave->get_post()->ID;
// Copy component custom meta keys from the autosave to the main component.
Plugin::$instance->db->copy_elementor_meta( $autosave_id, $main_id, Component_Document::COMPONENT_CUSTOM_META_KEYS );
$autosave_elements = $autosave->get_elements_data();
$autosave_title = $autosave->get_post()->post_title;
return $main_component_document->save( [
'elements' => $autosave_elements,
'settings' => [
'post_status' => Document::STATUS_PUBLISH,
'post_title' => $autosave_title,
],
] );
}
}