Joins

A cornerstone of CRUD applications is the ability to combine information from multiple SQL tables, representing the combined data as a common data set that the end user can easily understand and manipulate. Relational databases are, after all, designed for exactly this sort of data referencing. Often in CRUD applications, working with such joined data can significantly increase the complexity of the application and increase development time, Editor makes working with such join tables extremely easy through its leftJoin() method. Complex joins with full CRUD support can be added in just minutes.

Left Join

Editor provides a leftJoin() method, as an SQL Left (outer) Join is the most common type of join performed when working in CRUD applications. The end result is that the focus is on a single table that is being edited, with additional (potentially optional) data added to it. If you are already comfortable with an SQL left join, skip over this section, but if not, it is important to understand the data manipulation being performed.

An SQL left join returns all rows from the left table (in the case of Editor, the table that the Editor class is initialised with), even if there are no matches in the right table (the table being joined on). If there is no data in the right table, null values will be used for the columns being read from that table.

Consider for example the following two tables:

Table: staff                     Table: sites
+----+---------+-------+------+  +----+-----------+
| id | name    | title | site |  | id | name      |
+----+---------+-------+------+  +----+-----------+
| 1  | Allan   | CEO   | 1    |  | 1  | London    |
| 2  | Charlie | CTO   | 1    |  | 2  | Edinburgh |
| 3  | Fred    | CFO   | null |  +----+-----------+
+----+---------+-------+------+

If we perform the following join query:

SELECT staff.name, sites.name
FROM staff
LEFT JOIN sites ON staff.site = sites.id`

The result set will be:

+------------+------------+
| staff.name | sites.name |
+------------+------------+
| Allan      | London     |
| Charlie    | London     |
| Fred       | null       |
+------------+------------+

For a more detailed explanation of left joins, and the other join options that SQL has, please review Jeff Atwood's excellent A Visual Explanation of SQL Joins.

leftJoin method

The Editor leftJoin() method is as similar as possible to the standard SQL JOIN ON syntax. Specifically it typically takes four parameters:

  1. The table to join onto (optionally with an alias)
  2. The first join column name
  3. The join operator (=, >=, etc.)
  4. The second join column name.

Consider for example a table users which has a column site which points to an id column in a sites table and we want to include information from the site table. In SQL the Join syntax would be:

LEFT JOIN sites ON sites.id = users.site

In Editor, the leftJoin() method is:

->leftJoin( 'sites', 'sites.id', '=', 'users.site' )

With the join in place, to read information form the joined table is as trivial as adding the field to the Editor instance' field list. For example, to read the name column from the site table use Field::inst( 'sites.name' ).

Complex left joins

Editor 2.0 added support for complex join expressions to the leftJoin(). In this case it uses just two parameters:

  1. The table to join onto (optionally with an alias)
  2. The join expression. This is raw SQL that will not be parsed by the libraries, but rather just passed to the SQL database. It may include multiple join conditions with logical expressions and / or sub-selects.

For example the above join could be written as:

->leftJoin( 'sites', 'sites.id = users.site' )

More complex expressions can be used, e.g. do a standard join, but only show details about joined tables that match the sub-select:

->leftJoin(
    'sites',
    'sites.id = users.site AND sites.id IN (SELECT id FROM sites WHERE name LIKE "L%")'
);

It is important to note that because Editor does not perform any parsing on the complex join expression, if you have any user input in the condition it must be fully validated before being used, otherwise you leave yourself open to an SQL injection attack.

Table aliases

It can sometimes be useful to perform multiple left joins to the same table so you can read different, but like information from the joined table. For example, in our staff tables above we could have a Main site and a Backup site (called main_site and backup_site in the users table, respectively), rather than just a single one. The site information would still need to come from the sites table, but the value would be different for each of the two fields.

In SQL this can be done with an alias - effectively renaming the joined table (i.e. aliasing it to a different name to ensure that it can be uniquely identified). We can do this in Editor as well using the as key word in the first parameter given to the leftJoin method.

For example:

Editor::inst( $db, 'users' )
    ->field( 
        Field::inst( 'users.main_site' ),
        Field::inst( 'users.backup_site' )
        Field::inst( 'mainSite.name' ),
        Field::inst( 'backupSite.name' )
    )
    ->leftJoin( 'sites as mainSite',   'mainSite.id',   '=', 'users.main_site' )
    ->leftJoin( 'sites as backupSite', 'backupSite.id', '=', 'users.backup_site' );

Note that the alias name (mainSite and backupSite is used in the join condition and the field name. The client-side code would also refer to the alias name.

Options

Inevitably when you are working with an editable joined table, you will wish to present the end user with a list of options that they can select as the value for the field. This list of options will be defined by the data in the joined table - continuing the above example, this is the list of sites that the staff member might be assigned to.

Editor's PHP libraries provide an Options class which is used to define the list of options for use in a field such as select, datatable, tags and others. The most simple use of the options class is to specify the table and two columns (label and value) from where the options should be read:

Field::inst( 'users.site' )
    ->options( Options::inst()
        ->table( 'sites' )
        ->value( 'id' )
        ->label( 'name' )
    );

Please see the Options documentation for full details on how to use the Options class to get and display the options to show the end user for a field.

Validation

Editor's libraries provide a number of useful validation methods that can easily be used to ensure that the data submitted from the client-side is valid. When a join is being used that validation should ensure that referential integrity is retained by checking that the value to write to the database exists in the joined table before using it. This can be done using the Validate::dbValues validation method (note this requires Editor 1.5.4 or newer).

By default Validate::dbValues will attempt to use the table and value column defined by the Field->options() method (described above). If this is not possible (either the options haven't been defined or a closure was used) the options can be passed in using the [validator's configuration options](validation#Database](validation#Database). As a result, in most cases, validating the joined data is as simple as using:

Field::inst( 'users.site' )
    ->options( Options::inst()
        ->table( 'sites' )
        ->value( 'id' )
        ->label( 'name' )
    )
    ->validator( 'Validate::dbValues' );

Example

The Editor examples contain an example of a join table and here we will consider the code from that example in detail. The example uses the two SQL tables defined above to present a list of staff with a location that we wish to be editable.

Server-side

In the PHP we use the leftJoin() method for the Editor instance:

Editor::inst( $db, 'users' )
    ->field( 
        Field::inst( 'users.first_name' ),
        Field::inst( 'users.last_name' ),
        Field::inst( 'users.phone' ),
        Field::inst( 'users.site' )
            ->options( Options::inst()
                ->table( 'sites' )
                ->value( 'id' )
                ->label( 'name' )
            )
            ->validator( 'Validate::dbValues' ),
        Field::inst( 'sites.name' )
    )
    ->leftJoin( 'sites', 'sites.id', '=', 'users.site' )
    ->process($_POST)
    ->json();

You'll also likely notice that the dot separator also very conveniently is used in Javascript as the object parameter accessor and the same basic format can be used to access the data on the client-side.

The above PHP will generate JSON data in the format for each row:

{
    "users": {
        "first_name": "Quynn",
        "last_name": "Contreras",
        "phone": "1-971-977-4681",
        "site": "1"
    },
    "sites": {
        "name": "Edinburgh"
    }
}

Client-side

On the client-side we would use the following script - notice in particular the use of the Javascript dotted object notation in the fields.name and columns.data options:

var editor = new DataTable.Editor({
    ajax: '../php/join.php',
    table: '#example',
    fields: [
        {
            label: 'First name:',
            name: 'users.first_name'
        },
        {
            label: 'Last name:',
            name: 'users.last_name'
        },
        {
            label: 'Phone #:',
            name: 'users.phone'
        },
        {
            label: 'Site:',
            name: 'users.site',
            type: 'select'
        }
    ]
});

$('#example').dataTable({
    ajax: {
        url: '../php/join.php',
        type: 'POST'
    },
    columns: [
        {data: 'users.first_name'},
        {data: 'users.last_name'},
        {data: 'users.phone'},
        {data: 'sites.name'}
    ],
    layout: {
        topStart: {
            buttons: [
                {extend: 'create', editor: editor},
                {extend: 'edit', editor: editor},
                {extend: 'remove', editor: editor}
            ]
        }
    },
    select: true
});

This particular example can be seen running here.

PHP API documentation

The PHP API developer documentation for the Editor PHP classes is available for detailed and technical discussion about the methods and classes discussed above.