STRATO-apps/wordpress_03/app/wp-content/plugins/tutor-pro/tutorai/ImageController.php

SHA-256: 5d364a874665edcc0b0ea7ff04abbc45fc46a34692d4ab0b9e7fcc0e15a8e585
<?php
/**
 * Handle AI Generations
 *
 * @package TutorPro\TutorAI
 * @author Themeum <support@themeum.com>
 * @link https://themeum.com
 * @since 3.0.0
 */

namespace TutorPro\TutorAI;

use Exception;
use RuntimeException;
use Tutor\Helpers\HttpHelper;
use TUTOR\Input;
use Tutor\Traits\JsonResponse;
use TutorPro\OpenAI\Constants\Models;
use TutorPro\OpenAI\Constants\Sizes;

if ( ! defined( 'ABSPATH' ) ) {
	exit;
}

/**
 * Image controller class.
 * This class is responsible for generating image using openai.
 *
 * @since 3.0.0
 */
class ImageController {

	/**
	 * Use the JsonResponse trait for sending HTTP Response.
	 *
	 * @since 3.0.0
	 */
	use JsonResponse;

	/**
	 * Constructor method for generating AI Images.
	 *
	 * @since 3.0.0
	 */
	public function __construct() {
		/**
		 * Handle AJAX request for generating AI images
		 *
		 * @since 3.0.0
		 */
		add_action( 'wp_ajax_tutor_pro_generate_image', array( $this, 'generate_image' ) );

		/**
		 * Handle AJAX request for editing AI image
		 *
		 * @since 3.0.0
		 */
		add_action( 'wp_ajax_tutor_pro_magic_fill_image', array( $this, 'magic_fill_image' ) );

		/**
		 * Handle AJAX request for using the AI generated image to the WP system.
		 *
		 * @since 3.0.0
		 */
		add_action( 'wp_ajax_tutor_pro_use_magic_image', array( $this, 'use_magic_image' ) );
	}

	/**
	 * Generate the prompt for the specific styles
	 *
	 * @since 3.0.0
	 *
	 * @param string $prompt The user prompt for generating image.
	 * @param string $style The style of the output image.
	 *
	 * @return string
	 */
	private static function generate_prompt( $prompt, $style ) {
		$style_prompts = array(
			'filmic'          => 'Create an image of {user_prompt} with a cinematic quality, incorporating deep contrasts, rich colors, and dramatic lighting. The scene should evoke the feeling of a classic film.',
			'photo'           => 'Generate a high-resolution photograph {user_prompt} of with realistic lighting, shadows, and textures. The image should have a natural, lifelike quality as if captured by a professional camera.',
			'neon'            => 'Create an image of {user_prompt} with vibrant neon colors and glowing elements. The design should feature bright, fluorescent lights and a modern, urban aesthetic reminiscent of neon signs and cityscapes.',
			'dreamy'          => 'Design an image of {user_prompt} with a dreamy, ethereal quality, using soft focus, pastel colors, and gentle lighting. The scene should evoke a sense of whimsy and surreal beauty, like something out of a fantasy.',
			'black_and_white' => 'Generate a black and white image of {user_prompt} with high contrast and a wide range of grays. The absence of color should emphasize the shapes, textures, and lighting to create a dramatic and timeless look.',
			'retrowave'       => 'Design an image of {user_prompt} with a retro 80s aesthetic, featuring neon colors, grid patterns, and futuristic elements that evoke the style of synthwave music and retro video games.',
			'3d'              => 'Create an image of {user_prompt} 3D low poly, featuring game style, clean edges, and vibrant colors. The render should emphasize the crafted nature of 3D, focusing on expressive forms and controlled lighting.',
			'concept_art'     => 'Produce a piece of concept art of {user_prompt} that showcases a creative and imaginative design. Use detailed textures, dynamic compositions, and a strong visual narrative to convey the concept effectively.',
			'sketch'          => 'Create a sketch-style image of {user_prompt} with clean, hand-drawn lines and minimal shading. The design should look like a detailed pencil or ink drawing, capturing the essence of the subject with simplicity and elegance.',
			'illustration'    => "Create an illustration of {user_prompt} with vibrant colors, clear outlines, and stylized elements. The design should have a playful and imaginative quality, with detailed characters and scenes that capture the viewer's attention and convey a strong visual story.",
			'painting'        => 'Design an image of {user_prompt} with the texture and style of a traditional painting. Use brushstroke effects, rich colors, and painterly techniques to create a piece that looks like it was painted by hand on canvas.',
		);

		if ( empty( $style ) || 'none' === $style ) {
			return $prompt;
		}

		if ( empty( $style_prompts[ $style ] ) ) {
			return $prompt;
		}

		$style_prompt = 'You are an intelligent assistant tasked with generating banner images for an e-learning application. ' . $style_prompts[ $style ];

		return str_replace( '{user_prompt}', $prompt, $style_prompt );
	}

	/**
	 * Generate image using the user prompt and the styles
	 *
	 * @since 3.0.0
	 *
	 * @return void
	 */
	public function generate_image() {
		tutor_utils()->check_nonce();

		$prompt = Input::post( 'prompt' );
		$style  = Input::post( 'style' );

		if ( empty( $prompt ) ) {
			$this->json_response(
				__( 'Prompt is required to generating images', 'tutor-pro' ),
				null,
				HttpHelper::STATUS_BAD_REQUEST
			);
		}

		$prompt = self::generate_prompt( $prompt, $style );

		$input = array(
			'model'           => Models::DALL_E_3,
			'prompt'          => $prompt,
			'n'               => 1,
			'size'            => Sizes::LANDSCAPE,
			'response_format' => 'b64_json',
		);

		try {
			$client   = Helper::get_openai_client();
			$response = $client->images()->create( $input );
			$response = Helper::check_openai_response( $response );
			$this->json_response( __( 'Image created', 'tutor-pro' ), $response );
		} catch ( Exception $error ) {
			$this->json_response( $error->getMessage(), null, HttpHelper::STATUS_INTERNAL_SERVER_ERROR );
		}
	}

	/**
	 * Edit image by selecting an area.
	 *
	 * @since 3.0.0
	 *
	 * @return void
	 */
	public function magic_fill_image() {
		tutor_utils()->check_nonce();

		$prompt         = Input::post( 'prompt' );
		$image          = Input::post( 'image' );
		$revised_prompt = 'Fill the image and replace the selected area by {prompt}';

		$input = array(
			'model'           => Models::DALL_E_2,
			'image'           => $image,
			'prompt'          => str_replace( '{prompt}', $prompt, $revised_prompt ),
			'n'               => 1,
			'size'            => Sizes::REGULAR,
			'response_format' => 'b64_json',
		);

		try {
			$client   = Helper::get_openai_client();
			$response = $client->edits()->create( $input );
			$response = Helper::check_openai_response( $response );
			$this->json_response( __( 'Mask applied successfully.', 'tutor-pro' ), $response );
		} catch ( Exception $error ) {
			$this->json_response( $error->getMessage(), null, HttpHelper::STATUS_INTERNAL_SERVER_ERROR );
		}
	}

	/**
	 * Use the image generated by the AI, upload this image to the media.
	 *
	 * @since 3.0.0
	 *
	 * @return void
	 *
	 * @throws RuntimeException Throws an exception if any error happens while uploading the bits.
	 */
	public function use_magic_image() {
		tutor_utils()->check_nonce();

		$image = Input::post( 'image' );

		if ( empty( $image ) ) {
			$this->json_response( __( 'Image is missing to use', 'tutor-pro' ), null, HttpHelper::STATUS_BAD_REQUEST );
		}

		try {
			$response = tutor_utils()->upload_base64_image( $image );
			$this->json_response( __( 'Image stored', 'tutor-pro' ), $response );
		} catch ( Exception $error ) {
			$this->json_response( $error->getMessage(), null, HttpHelper::STATUS_INTERNAL_SERVER_ERROR );
		}
	}
}