ucscode / uss-element
UssElement is a PHP library for building and manipulating HTML elements programmatically.
Requires
- php: >=8.2
Requires (Dev)
- phpunit/phpunit: ^10
README
A simple, lightweight, standalone PHP library for programmatically creating and manipulating HTML elements. It simplifies the process of working with HTML structures and DOM elements, offering functionality similar to DOMDocument but with reduced boilerplate and enhanced ease of use.
With UssElement, you can effortlessly create DOM nodes, set attributes, set innerHtml, use querySelector, modify element classlist etc, and generate (or render) HTML strings with ease.
Why Uss Element?
UssElement is designed to simplify and streamline the process of working with HTML elements in PHP. If (like me) you've ever been frustrated by the complexity of PHP's DOMDocument
or found yourself writing repetitive, cumbersome code just to manipulate HTML structures, UssElement is the solution you’ve been waiting for.
This standalone library takes care of the heavy lifting, reducing boilerplate code and eliminates the need for complex XPath queries, offering a simple, intuitive API for tasks like creating elements, setting inner HTML, and selecting elements using CSS selectors.
The library is lightweight, fast, and easy to integrate into any project, making it perfect for both small and large-scale applications.
Key Features:
- Create HTML elements using PHP code.
- More inspired by Javascript DOM than PHP DOMDocument
- Set element attributes, such as class names and IDs.
- Define inner HTML content for elements.
- Generate or render HTML strings with the
NodeInterface::render()
method. - Boost your productivity by simplifying HTML generation in PHP.
- Reducing Complexity
- Providing an Intuitive API
- Encouraging Dependency-Free Development
- Efficiency in Common DOM Tasks
- Flexibility and Extensibility
- Improving Developer Experience
- Encoding element to Json format
- Decoding element from json format
Prerequisite
- PHP >= 8.2
Installation (Composer)
You can include UssElement library in your project using Composer:
composer require ucscode/uss-element
Getting Started:
- Instantiate the UssElement class with the desired HTML element type (e.g.,
NodeNameEnum::NODE_DIV
). - Use UssElement methods to set attributes and content.
- Generate HTML strings with
NodeInterface::render()
for seamless integration into your web pages.
Creating Elements
You can create elements by instantiating the type of node
use Ucscode\UssElement\Node\ElementNode; $element = new ElementNode('div');
If you prefer, you can use the NodeNameEnum
enum
use Ucscode\UssElement\Node\ElementNode; use Ucscode\UssElement\Enums\NodeNameEnum; $element = new ElementNode(NodeNameEnum::NODE_DIV);
You can also create an element and set their attributes at the point of instantiation:
$span = new ElementNode('span', [ 'id' => 'short-cut', 'class' => 'to set', 'data-what' => 'attributes' ]);
You can use many available methods to manipulate the DOM.
A summary of these methods are provided in the following sections:
$element->appendChild($span);
$element->getNextSibling();
$element->getChild(0) ->setAttribute('data-name', 'Ucscode') ->setAttribute('title', 'Uss Element') ;
Traversing Elements
Use the querySelector()
or querySelectorAll()
method to select elements based on CSS selectors:
$element->querySelector('.to.set[data-what=attributes]'); // Returns the <span> element
You can also retrieve an element by other methods such as:
getElementsByClassName
getElementsByTagName
$element->getElementsByClassName('.set'); // Returns the <span> element
Inner HTML
- You can easily set the inner HTML content of an element using the
setInnerHtml()
method: - You can also get inner HTML of an element using
getInnerHTML()
method:
$element->setInnerHtml('<p>This is a paragraph inside a div.</p>');
Loading HTML
You can convert an HTML string to NodeList
containing all elements using the HtmlLoader
class:
use Ucscode\UssElement\Parser\Translator\HtmlLoader; // An example HTML document: $html = <<< 'HERE' <html> <head> <title>TEST</title> </head> <body id='foo'> <h1>Hello World</h1> <p>This is a test of the HTML5 parser.</p> </body> </html> HERE; $htmlLoader = new HtmlLoader($html); $htmlLoader->getNodeList()->count(); // Returns the number of direct nodes (1 in this case) $htmlLoader->getNodeList()->first; // HTML ElementNode
You can also load framents
use Ucscode\UssElement\Parser\Translator\HtmlLoader; $html = <<< 'HERE' <h1>Hi there</h1> <p>Please enter your detail</p> <form name="my-form> <input name="username"/> </form> HERE; $htmlLoader = new HtmlLoader($html); $htmlLoader->getNodeList()->count(); // Returns the number of direct nodes (3 in this case) $htmlLoader->getNodeList()->get(0); // H1 ElementNode $htmlLoader->getNodeList()->get(1); // P ElementNode $htmlLoader->getNodeList()->get(2); // FORM ElementNode
Basic Example
$html = '<div class="container"><p>Hello, world!</p></div>'; $htmlLoader = new HtmlLoader($html); // Access the root div element $divElement = $htmlLoader->getNodeList()->get(0); // Set inner HTML of the root element $divElement->setInnerHtml('<i class="fa-icon"></i><h1 class="heading">New Heading</h1><br/>'); // Query the first paragraph within the container $paragraph = $divElement->querySelector('p'); // null $heading = $divElement->querySelector('h1.heading'); // H1 ElementNode // Accessing the number of direct child nodes echo $divElement->getChildNodes()->count(); // 3
Render HTML
You can get or render the ElementNode
as HTML using the render()
method.
echo $divElement->render();
Output
<div class="container"><i class="fa-icon"></i><h1 class="heading">New Heading</h1><br/></div>
If you want to indent the rendered output, pass an unsigned integer (initially zero) to the render()
method
echo $divElement->render(0);
Output
<div class="container"> <i class="fa-icon"></i> <h1 class="heading"> New Heading </h1> <br/> </div>
Element Render Visibility
To keep an element in the DOM tree but exclude it from the rendered output, set its visibility to false
.
$divElement->querySelector('.heading')->setVisible(false);
$divElement->render(0);
<div class="container"> <i class="fa-icon"></i> <br/> </div>
$divElement->getChildren()->count(); // 3
Setting Void Item
Some HTML elements, like <br>
and <img>
, do not have closing tags.
The setVoid()
method marks an element as void, ensuring that it is rendered without a closing tag. This is especially helpful when defining custom elements.
$element = new Element('x-widget', [ ':vue-binder' => 'project' ]);
$element->render(); // <x-widget :vue-binder="project"></x-widget>
$element->setVoid(true);
$element->render(); // <x-widget :vue-binder="project"/>
Encoding and Decoding Nodes
This library provides methods for encoding a node into JSON format and decoding it back to its original structure.
This is useful for transferring nodes between systems or storing them in a format that can be easily reconstructed.
Encoding a Node
To encode a node into JSON format, the toJson()
method is used.
$node->toJson(); // Node to JSON Serialization
The toJson()
method internally uses an instance of the NodeJsonEncoder
, which is the actual encoder responsible for serializing the node.
(new NodeJsonEncoder($node))->encode();
Decoding a Node
To decode a JSON string back into a node, use the NodeJsonDecoder
class.
(new NodeJsonDecoder($json))->decode(); // JSON to Node Deserilization
The decoding process restores the full structure of the original node, including its attributes, child nodes, and content.
Normalization
Both the NodeJsonEncoder
and NodeJsonDecoder
provide a normalize
method to convert the input into an array.
(new NodeJsonEncoder($node))->normalize(); // to array (new NodeJsonDecoder($json))->normalize(); // to array
NodeInterface methods
ElementInterface methods
Includes:
TextNode methods
Includes:
Collection Objects
Node Objects
Parser Objects
Providing Support For:
- Combinators: Use of combinator such as
>
,+
,~
,$
are captured but not yet supported
Contributing
Feel free to open issues or submit pull requests. We welcome contributions to improve the library.
How to Contribute
- Fork the repository.
- Create a new branch (
git checkout -b feature-xyz
). - Commit your changes (
git commit -am 'Add feature xyz'
). - Push to the branch (
git push origin feature-xyz
). - Create a new pull request.
License
This project is licensed under the MIT License - see the LICENSE file for details.