Installation
Install with Composer:, (*1)
composer require felipelavinz/queulat:dev-master
, (*2)
Composer will install on wp-content/mu-plugins/queulat
, (*3)
If you need to install on a different folder, you should add something like this to your project's composer.json:, (*4)
{
"extra" : {
"installer-paths" : {
"htdocs/wp-content/mu-plugins/{$name}" : ["type:wordpress-muplugin"]
}
}
}
Where htdocs/wp-content/mu-plugins/{$name}
it's the path to your mu-plugins directory. Queulat will be installed as a sub-folder on the specified folder., (*5)
Loading Queulat as mu-plugin
Queulat uses the Composer autoloader to lazy-load most of its code, so you need to make sure that the autoloader is included before initializing Queulat., (*6)
Also, since mu-plugins installed on a sub-folder are not automatically loaded by WordPress you must manually require the main file., (*7)
You can solve this with a single file on the mu-plugins folder, such as:, (*8)
<?php
/**
* Plugin Name: Queulat Loader
* Description: Load Queulat mu-plugin
*/
// Load Composer autoloader (ABSPATH it's the path to wp-load.php).
require_once ABSPATH .'/../vendor/autoload.php';
// Load Queulat main file.
require_once __DIR__ .'/queulat/queulat.php';
Plugin headers are optional, but recommended., (*9)
You could also use something like Bedrock's autoloader, which will load all mu-plugins installed on sub-folders (you can just copy that file on your mu-plugin folder and it will automagically load Queulat)., (*10)
Leveraging WordPress architecture with Queulat
Queulat it's aimed at improving the way we create things for WordPress, so instead of fundamentally transforming it, it tries to use familiar concepts to build better-structured things for it, using custom post types, custom post queries and custom post objects., (*11)
You can generate these using its own custom post type plugin generator, which is available to admin users on the "Tools" menu., (*12)
Types, queries and objects
Each custom post type plugin it's composed of:, (*13)
- An entry file, which initializes the plugin
- A post type definition, for instance:
Song_Post_Type
. This class defines the labels and other arguments for registering the post type on WordPress. By default, the plugin activation will add the needed permisions for administrators and flush rewrites. You could extend this class if you need to define custom hooks for extra functionality for the post type.
- A query definition, like
Song_Post_Query
. You can use this class to create new database queries, using any default params that you might want to define for this type of content, and iterate over the results using a simple foreach
instead of the classic WordPress loop.
- An object definition:
Song_Post_Object
, which will be returned on the foreach
loop when using the custom query. This way, you could add any custom methods to this class, which will be available on the foreach
loop.
Using Queulat, you could do something like:, (*14)
$tracklist = new Song_Post_Query( array(
'tax_query' => array(
array(
'taxonomy' => 'albums',
'term' => 'dark-side-of-the-moon',
'field' => 'slug'
)
)
) );
foreach ( $tracklist as $track ) {
echo $track->title();
echo $track->duration();
echo $track->lyrics();
}
Post Types
@todo
, (*15)
Post Queries
@todo
, (*16)
Post Objects
@todo
, (*17)
Presently, the primary way to define meta data on post objects it's creating meta boxes; extending the included Queulat\Metabox
abstract class., (*18)
The extending class must implement the abstract methods: get_fields( ) : array
and sanitize_data( array $data ) : array
; for instance:, (*19)
<?php
use Queulat\Metabox;
use Queulat\Forms\Node_Factory;
use Queulat\Forms\Element\Input_Text;
class Track_Meta_Box extends Metabox {
/**
* Must return the list of form fields to be included on this meta box
*
* @return Queulat\Forms\Node_Interface[] Array of form fields.
**/
public function get_fields() : array {
return array(
Node_Factory::make(
Input_Text::class,
array(
'name' => 'length',
'label' => 'Track length',
'attributes.class' => 'regular-text',
'properties.description' => _x( 'Track duration, such as: 01:23:45 (1 hour, 23 minutes, 45 seconds)', 'length field description', 'track_cpt' ),
)
),
Node_Factory::class(
Input_Checkbox::class,
array(
'name' => 'colour',
'label' => 'Colour',
'options' => array(
'red' => 'Red',
'green' => 'Green',
'blue' => 'Blue',
'any' => 'Any colour'
)
)
)
);
}
/**
* Sanitize data from the metabox form.
*
* @param array $data Form data from the meta box (not the full $_POST).
* @return array Sanitized data.
**/
public function sanitize_data( array $data ) : array {
return queulat_sanitizer( $data, array(
'length' => array( 'sanitize_text' ),
'colour.*' => array( 'sanitize_key' )
) );
}
}
Queulat takes care of updating the submitted data as post meta fields, and loading the saved values on the meta box form., (*20)
For sanitizing data, you could use whatever method you prefer. The referenced queulat_sanitizer()
function it's a pretty simple way to apply callbacks to the matching values from the $data
input. You can use *
to match all properties (using dot notation)., (*21)
Check the section on (using Queulat forms)[#using-queulat-forms] for more info on available form fields or how to create your own., (*22)
Creating admin pages
@todo: add general description
, (*23)
@todo: add description for each form field
, (*24)
- Button
- Div
- Fieldset
- Form
- Google_Map
- Input
- Input_Text
- Input_Hidden
- Input_Email
- Input_Checkbox
- Input_Number
- Input_Radio
- Input_Submit
- Input_Url
- Recaptcha
- Select
- Textarea
- UI_Select2
- WP_Editor
- WP_Gallery
- WP_Image
- WP_Media
- WP_Nonce
- Yes_No
@todo
, (*25)
@todo
, (*26)
Node_Factory
it's a simple factory class that's able to create any kind of form element., (*27)
It exposes a single make
method that can instantiate and configure an element. This method takes two parameters:, (*28)
- An "element name" as string, which should be a fully qualified name for a form element or component.
- An associative array of "arguments" that are used to configure the object.
By default, Queulat is configured to handle the following attributes:, (*29)
- attributes: HTML element attributes, such as class, id, type, etc; as associative array.
- children: nested elements, which can also be created with the
Node_Factory
as an array.
- label: the text labeling a form element.
- name: the field "name" that's used on form submission.
- options: an associative array of element options (for fields such as input radios, checkboxes, selects, etc.)
- properties: an array of arbitrary node properties which can be used by the form view or whatever, as an associative array.
- text_content: the textual content of the node.
- value: the form field value.
Arguments that are not supported by the element are skipped., (*30)
You can extend the supported arguments using Node_Factory::register_argument()
., (*31)
Usage
<?php
use Queulat\Forms\Element\Div;
use Queulat\Forms\Node_Factory;
use Queulat\Forms\Element\Button;
$submit = Node_Factory::make(
Div::class, array(
'attributes' => array(
'class' => 'col-md-4 col-md-offset-8',
'id' => 'form-buttons'
),
'text_content' => '* required field',
'children' => array(
Node_Factory::make(
Button::class,
'attributes' => array(
'class' => 'btn-lg',
'type' => 'submit'
),
'text_content' => 'Submit'
)
)
);
);
echo $submit;
Node_Factory
Registering new argument handlers
You can register new arguments used by the Node_Factory using the register_argument
method., (*32)
This method takes a Node_Factory_Argument_Handler
, which needs:, (*33)
- An
$argument
(string) which is the name of the argument key that you'll handle.
- A
$method
(string) which is the name of the method that will receive the parameters used on the factory method.
- An optional
$call_type
(string) which determines how the $method will treat the received configuration values.
The $call_type
can be one of:, (*34)
Node_Factory::CALL_TYPE_VALUE
: pass all arguments as a single array to the handler. This is the default setting. Example: $obj->$method( $args );
, (*35)
Node_Factory::CALL_ARRAY
: pass arguments as individual parameters to the handler. Example: call_user_func_array( [ $obj, $method ], $args );
, (*36)
Node_Factory::CALL_KEY_VALUE
: for each item in the argument, pass its key and value as parameters to the handler. Example:, (*37)
foreach ( $args as $key => $val ) {
$obj->$method( $key, $val );
}
Node_Factory::CALL_TYPE_VALUE_ITEMS
: for each item in the argument, use the value as parameter for the handler. Example: array_walk( $args, [ $obj, $method ] );
, (*38)
Interfaces
Node_Interface
Related Interfaces |
Related Traits |
Component_Interface |
Node_Trait |
Element_Interface |
Childless_Node_Trait |
Nodes are the lowest level of objects that should be used with forms, (*39)
Use Node_Trait
to help implement this interface or Childless_Node_Trait
. In general terms,
elements should use the former and components the latter., (*40)
Component_Interface
Extends the Node_Interface and Attributes_Interface., (*41)
Element_Interface
Extends the Node_Interface and Attributes_Interface. Also, adds the get_tag_name
method., (*42)
HTML_Element_Interface
Extends the Element_Interface and Properties_Interface., (*43)
Extends HTML_Element_Interface and Form_Node_Interface., (*44)
Attributes_Interface
"Attributes" are special properties used by objects implementing this interface. They're rendered as HTML attributes key="val"
, (*45)
Use Attributes_Trait
to help implement this interface., (*46)
Objects implementing this interface (elements or components) are used as form objects. They have a "name" which is used to send data to the server, a "value" and a "label"., (*47)
The Form_Control_Trait
helps implementing the "label" and "name" getters and setters from this interface.
The "value" getter and setter should be defined by your own custom component., (*48)
Node_List_Interface
Extends ArrayAccess, SeekableIterator, Countable, Serializable., (*49)
Most commonly used to get an array-like set of children from a Node., (*50)
Option_Node_Interface
Used by controls such as checkboxes, radios, selects and every component where the user is presented with several alternatives they can choose from., (*51)
Use Options_Trait
to help implement this interface., (*52)
Properties_Interface
Node properties (not to be confused with regular object properties) can store arbitrary data, such as view settings, error data or validation state., (*53)
Use Properties_Trait
to help implement this interface., (*54)
View_Interface
Base interface to be used by form views., (*55)