Prisma PHPX

Prisma PHPX is a template system for Prisma PHP that is inspired by JSX and React.

To maintain order and consistency, create a directory in src/app/Lib/PHPX/DirectoryName/ClassName.php. This convention helps in organizing and managing your components effectively.

For example, to create a SearchIcon component, you should create a file named SearchIcon.php in the src/app/Lib/PHPX/GoogleIcons/SearchIcon.php directory. The class name should match the file name.

It is important to note that you can have one initializer file for all your components. For example, you can follow a naming convention like Accordion for the main component. The related components should follow the same naming convention, such as AccordionItem, AccordionTrigger, and AccordionContent. These classes should be in the same file and start with the main component name, in this case, Accordion, followed by the specific component name.

The file can be located in any directory you desire, but following the naming convention is important. For example, you can place it in src/app/Lib/PHPX/Components/Accordion.php.

By default, the class constructor's props will receive the children property, which represents the inner content of the component. You can also pass other props to the component, such as class and others.

The PHPX class provides three primary properties: $props, $children, and $class. These properties store the sanitized versions of the component's attributes, content, and CSS classes, respectively. When extending the PHPX class, you can utilize these properties to meet your custom requirements.

  • $props: The properties or attributes passed to the component.
  • $children: The children elements or content to be rendered within the component.
  • $class: The CSS class for custom styling.
  • $attributesArray: An array of attributes to be rendered as HTML attributes, useful when you want to spread all attributes to the component.

PHPX provides several methods that you can override to customize your components. The most important method is the constructor. When you extend the PHPX class, you should pass the props from your constructor if you plan to use any of the properties and methods provided by the PHPX class. You can override any of the methods to tailor your component to your needs. A common use case is the static init method, which you can use to register and retrieve information like dialog IDs or other data when the component is initialized. This information can then be shared with child components using the StateManager class, which is designed to facilitate information sharing within the application environment.

  • __construct(array $props = []): Constructor to initialize the component with the given properties.
  • init(array $props = []): Registers or initializes any necessary components or settings.
  • getMergeClasses(): Combines and returns the CSS classes for the component, especially useful when working with dynamic TailwindCSS classes.
  • getAttributes(): Generates and returns a string of HTML attributes from the provided props and avoid duplication of attributes.
  • render(): Renders the component as an HTML string with the appropriate classes and attributes.
  • __toString(): Converts the object to its string representation by rendering it.

In the src/Lib/PHPX directory, you will find the following files:

  • TemplateCompiler.php: The main compiler for the PHPX template system.
  • PHPX.php: The base class that all your PHPX components should extend.
  • IPHPX.php: An interface that all PHPX components must implement. Note: If you extend the PHPX class, implementing this interface is not required.
  • TwMerge.php: A helper class for merging classes, particularly useful for dynamic TailwindCSS classes.

To import your component, use use Lib\PHPX\GoogleIcons\Search; and then use the component as <Search />. If you have issues when importing components with the same name from different libraries, use an alias like use Lib\PHPX\GoogleIcons\Search as GoogleSearch; and then use the component as <GoogleSearch />. For example, if you have a component named Search in two different libraries, you can import them as use Lib\PHPX\GoogleIcons\Search as GoogleSearch; and use Lib\PHPX\LucideIcons\Search as LucideSearch; and then use the components as <GoogleSearch /> and <LucideSearch />.

The execution order of components starts from the innermost child and proceeds to the outermost parent. By default, child components are executed first, followed by the parent component. However, if you use the Component::init() method, the parent component will be executed first. For instance, if you need to assign a dynamic ID to a specific child, it's important to understand where and how to set special attributes to ensure they receive the correct values. Note that the init() method executes only once for that component because it is a static method.

Default Execution Order

<?php

use Lib\PHPX\PHPXUI\{Accordion, AccordionItem, AccordionTrigger, AccordionContent};

?>

<Accordion>
    <AccordionItem>
        <AccordionTrigger>Prisma PHP is awesome!</AccordionTrigger>
        <AccordionContent>Prisma PHPX is inspired by JSX and React.</AccordionContent>
    </AccordionItem>
</Accordion>
  • AccordionTrigger initialized
  • AccordionContent initialized
  • AccordionItem initialized
  • Accordion initialized

When Using the init() Method

<?php

use Lib\PHPX\PHPXUI\{Accordion, AccordionItem, AccordionTrigger, AccordionContent};

Accordion::init();

?>

<Accordion>
    <AccordionItem>
        <AccordionTrigger>Prisma PHP is awesome!</AccordionTrigger>
        <AccordionContent>Prisma PHPX is inspired by JSX and React.</AccordionContent>
    </AccordionItem>
</Accordion>
  • Accordion initialized
  • AccordionTrigger initialized
  • AccordionContent initialized
  • AccordionItem initialized

In Prisma PHP, all tags must follow strict XML closing rules, similar to JSX:

  • Valid: <hr />
  • Invalid: <hr>
  • Valid: <br />
  • Invalid: <br>
  • Valid: <input type="text" />
  • Invalid: <input type="text">
  • Valid: <image src="image.jpg" />
  • Invalid: <image src="image.jpg">

Attributes must be enclosed in double quotes and follow strict XML rules:

  • Valid: <input required="true" />
  • Invalid: <input required />
  • Valid: <input type="text" id="input" />
  • Invalid: <input type="text" id=input />

When displaying complex code using the <pre> or <code> tags, it is recommended to use <![CDATA[ ... ]]> to prevent any issues with code rendering during XML parsing.

<pre><code><![CDATA[
    // Your complex code here
]]></code></pre>

When using the <pre> or <code> tags, always escape the "<" character using HTML entities to prevent rendering issues during XML parsing.

When working with JSON data, you can use the json_encode() function to convert the data to a JSON string. This function is useful when you need to pass data from PHP to JavaScript or store data in a JSON file. To prevent XSS attacks, use json_encode($data, JSON_HEX_TAG | JSON_HEX_AMP | JSON_HEX_APOS | JSON_HEX_QUOT). This ensures that special characters are properly encoded, preventing the XML parser from misinterpreting them as HTML tags and avoiding rendering issues in the browser.

Explanation of JSON Encoding Options:

Constant Description
JSON_HEX_TAG Escapes < and > as \u003C and \u003E to prevent HTML injection.
JSON_HEX_AMP Escapes & as \u0026 to prevent breaking HTML attributes.
JSON_HEX_APOS Escapes ' (single quote) as \u0027 to prevent JavaScript issues.
JSON_HEX_QUOT Escapes " (double quote) as \u0022 to prevent breaking JSON inside HTML attributes.

Use MainLayout::addFooterScript($accordionScript); to add scripts to the footer dynamically.

public function __construct(array $props = [])
      {
          parent::__construct($props);

          $accordionScript = <<<HTML
          <script>
              function toggleAccordion(id) {
                  const allContents = document.querySelectorAll('[id^="content-"]');
                  const allIcons = document.querySelectorAll('[id^="icon-"]');
                  const allButtons = document.querySelectorAll('button[aria-expanded]');
  
                  // Close all other accordion items
                  allContents.forEach((content) => {
                      if (content.id !== 'content-' + id) {
                          content.classList.add('hidden');
                      }
                  });
  
                  allIcons.forEach((icon) => {
                      if (icon.id !== 'icon-' + id) {
                          icon.classList.remove('rotate-180');
                      }
                  });
  
                  allButtons.forEach((button) => {
                      if (button.getAttribute('aria-controls') !== 'content-' + id) {
                          button.setAttribute('aria-expanded', 'false');
                      }
                  });
  
                  // Toggle the selected accordion item
                  const content = document.getElementById('content-' + id);
                  const icon = document.getElementById('icon-' + id);
                  const isExpanded = content.classList.contains('hidden');
  
                  content.classList.toggle('hidden', !isExpanded);
                  icon.classList.toggle('rotate-180', isExpanded);
  
                  // Update aria-expanded for the clicked button
                  const button = document.querySelector('[aria-controls="content-' + id + '"]');
                  button.setAttribute('aria-expanded', isExpanded.toString());
              }
          </script>
          HTML;
  
          MainLayout::addFooterScript($accordionScript);
      }

To create a component by inheriting the PHPX class, simply call the parent constructor and render method. Use the inherited methods to get attributes and classes, and return the HTML code as a string.

<?php

namespace Lib\PHPX\Components;

use Lib\PHPX\PHPX;

class Button extends PHPX
{
    public function __construct(array $props = [])
    {
        parent::__construct($props);
    }

    public function render(): string
    {
        $attributes = $this->getAttributes([
            'type' => 'button',
        ]);
        $class = $this->getMergeClasses('bg-blue-500 p-2 rounded-md text-white hover:bg-blue-600');

        return <<<HTML
        <button class="$class" $attributes>
            {$this->children}
        </button>
        HTML;
    }
}