Skip to content

Latest commit

 

History

History
366 lines (282 loc) · 21.2 KB

README.md

File metadata and controls

366 lines (282 loc) · 21.2 KB

Wunderbyte Table

Wunderbyte Table can be used instead of the table_sql class and then supports all the actions via Ajax.

This local plugin was developed to allow the use of the table_sql class within modals and tabs. There is no special configuration required, BUT with special configuration, your table will be much more powerful. After the installation of the local_wunderbyte_table, you can create an extended class to wunderbyte_table (instead of table_sql).

class booking_table extends wunderbyte_table {}

If your extended table class contains another class (eg myplugin_class), make sure you can instantiate it with only the cmid. Errors will be thrown if this is not possible.

From 1.1.1 on, wunderbyte_table is loaded only once the corresponding div and none of it's parent is hidden (display:none) and it will add a visbility listener on the next hidden parent element. Unhiding will trigger loading of the table.

The included demo.php is only meant to demonstrate the working of the table.

That's all it takes. Switching pages, sorting, hiding columns and downloading will now run via ajax.

Security

Wunderbyte Table uses webservices to support features like ajax reload, infinite scroll, action buttons etc. As it is a generic project it can't know about the security requirements for the projects it actually is used in. We use Wunderbyte Table in combination with shortcodes, where we can't know where in Moodle the table actually will show up and sometimes, this will be on a location without any login requirements. Therefore, the key requirelogin, which is set normally to "true" can also be set to false:

$table->requirelogin = false;

We also provide the capability 'local/wunderbyte_table:canaccess' which normally is set to 'allow' for the archetype 'user'. This capability check is only required when requirelogin is set to true. Changing this will result in a change for all wunderbyte table on the moodle instance.

If only a single table should be protected, you can use the requirecapability key on a specific table to require special capabilities. When this key is not set, it falls back to 'local/wunderbyte_table:canaccess'.

$table->requirecapability = 'local/myplugin:canaccess';

Methods that should be called via the actionbutton functionality (webservice execute_action) must include the action_ prefix ie "action_deleterow". While the transmitted methodname is "deleterow", the actual method which will be called via the webservice is action_deleterow. This prevents any other method to be called than a method defined for this purpose.

Multiple custom templates

Wunderbyte Table comes with a responsive table template, but it can and should be overriden in your project. To do so, set in your instance of your extended class (eg $yourtable) the template to your proper template, like

$yourtable->tabletemplate = 'mod_yourtemplate/yourtable'

where yourtable corresponds to yourtable.mustache in your own project.

With Version 1.2.1 Wunderbyte Table supports multiple templates in one project. The structure has to be like this:

  • A yourfirsttable_container.mustache includes possible search, filter and sort components
  • This container also contains yourfirsttable.mustache
  • This table mustache contains yourfirsttable_row.mustache. Only by obeing this structure in your own project, the reloading triggered by filter, search etc. will work.

Caching

One new feature is caching. Wunderbyte_table will automatically pass every sql call to the MODE_APPLICATION cache with the key being the hashed request.

A request for page one will be cached with a different key than a request for page 2 etc.

Invalidation of the cache is being done by

cache_helper::purge_by_event('changesinwunderbytetable');

If you don't run this every time your table changes, you won't see the changes in your DB reflected in the output of wunderbyte table, unless you otherwise purge the cache.

If you use more than one table in your plugin or if there is a possibility that more than one Plugin uses local_wunderbyte_table on your system, you should provide your own cache definitons in your plugin. Use the define_cache('mod_myplugin', 'mycachename') function to set your own caches.

JavaScript

This description is only relevant in one case: If you override a value from your table via the col_mycolumn function in the wunderbyte_table class and you use a mustache template and renderer to echo html (eg. to render a button or a modal etc.) AND if this mustache template includes javascript, then you will encounter the problem, that the JS won't be automatically included. The reason is that the js would be added to the page footer via the renderer, but it is simply skipped in this particular usecase. Therefore, you need to add the JS instead of your column template to the table template. Any JS which is on this labe (corresponding to table.mustache in the wunderbyte_table project), will be executed after the table is correctly rendered. You have to make sure to write your js in a way that your can find the necessary variables (eg. the ids of your rows) without being able to pass them directly via the mustache template.

Sticky Header, Infinite Scroll & Pagination

If you want to enable scrolling within your table, you can set the stickyheader property to true. With infinite scroll, your table will automatically reload additional rows once you scrolled to its bottom. To enable infinite scroll, you can define the number of rows that will be loaded - first on init and later with each reload - in the infinitescroll variable. Sticky Header and Infinite Scroll can be combined. If you don't use infinite scroll, pagination options will be displayed at the bottom of your table.

Action buttons

You can add a number of action buttons to your table. If you combine them with "$yourtable->addcheckbox = true", you will be able to select single lines and execute your function with it. The methods will need to be implemented in your child class of wunderbyte table and they will be called via ajax. Example:

$mytable->addcheckbox = true;
$mytable->actionbuttons[] = [
    'label' => get_string('deleterow', 'mod_myproject'), // Name of your action button.
    'methodname' => 'deleterow', // The method needs to be added to your child of wunderbyte_table class including "action_" prefix ie action_deleterow.
    'data' => [ // Will be added eg as data-id = $values->id, so values can be transmitted to the method above.
        'id' => 'id'
    ]
];

You can choose if your Action Button needs element(s) to be selected in order to trigger an action. If this is set to false, you can trigger actions with or without actions selected.

$mytable->actionbuttons[] = [
    'selectionmandatory' => false,
]

You can choose if your Action Button triggers a modal with the "nomodal" param:

$mytable->actionbuttons[] = [
    'nomodal' => false,
]

If your Action Button works with or without selected elements, you can choose a different bodystring for the modal it may trigger.

$mytable->actionbuttons[] = [
'data' => [
                'id' => 'id',
                'titlestring' => 'deletedatatitle', // Will be shown in modal title
                'bodystring' => 'deletedatabody', // Will be shown in modal body in case elements are selected
                'labelcolumn' => 'firstname', // The Labelcolumn is important because it will be picked for human verification in the modal.
                'noselectionbodystring' => 'specialbody', // Will be displayed in modal bode in case no data is selected
                'submitbuttonstring' => 'deletedatasubmit', // Modal Button String
                'component' => 'local_wunderbyte_table', // Localization of strings
            ]
]

You can choose if your Action Button transmits a single or multiple call(s) in case multiple elements are selected.

$mytable->actionbuttons[] = [
    'id' => -1, // This forces single call execution.
]

If you want your Action Button to trigger a form instead of a simple modal, hand over namespace and title.

$mytable->actionbuttons[] = [
        'formname' => 'local_myplugin\\form\\edit_mytableentry', // To include a dynamic form to open and edit entry in modal.
        'data' => [
            'title' => get_string('title'), // Localized title to be displayed as title in dynamic form (formname).
        ]
]

You can also use actionbuttons in a column to treat the corresponding record.

public function col_action($values) { // Button will be added to action column.

    $data[] = [
        'label' => get_string('delete', 'core'), // Name of your action button.
        'class' => 'btn btn-danger', // Use bootstrap 4.
        'href' => '#', // You can either use the link, or JS, or both.
        'iclass' => 'fa fa-cog', // Add an icon before the label.
        'arialabel' => 'cogwheel', // Add an aria-label string to your icon.
        'title' => 'Edit', // Will be displayed when hovered over icon.
        'id' => $values->id.'-'.$this->uniqueid, // Access the data of the record.
        'name' => $this->uniqueid.'-'.$values->id,
        'methodname' => 'deleteitem', // The method needs to be added to your child of wunderbyte_table class including "action_" prefix ie action_deleterow.
        'nomodal' => true,
        'data' => [ // Will be added eg as data-id = $values->id, so values can be transmitted to the method above.
            'id' => $values->id,
        ]
    ];

Filter, Sort and Search

WB Table provides direct filter, search and sort functionality.

$table->define_filtercolumns(['id', 'username', 'firstname', 'lastname', 'email']);
$table->define_fulltextsearchcolumns(['username', 'firstname', 'lastname']);
$table->define_sortablecolumns(['id', 'username', 'firstname', 'lastname']);

In order for them to work, you must obey some rules: Filter basically just add another 'where' to the sql. The column name must therefor be like the filtercolumn. If the "WHERE columnname like '%myname%'" doesn't work, because you would need to write "WHERE s1.columnname like '%myname%'", then the filter will not work. You would need to wrap your SQL so to eliminate the need for the columnname prefix.

Filter, search and sort params selection triggers URL update. These params can also be applied to table on load via URL. These functions are not available if more than one table is displayed per page (multitable display).

Filter

As for the filter, you have these further functionalities:

  • localize labels and results
  • sort possible results in the filter panel (eg to have weekdays in order)

Here is an example how to set this up: By the way: 'id' will aways be obmitted, as it is not a useful filter in any case.

$table->define_filtercolumns([
    'id', 'sport' => [
        'localizedname' => get_string('sport', 'mod_myplugin')
    ], 'dayofweek' => [
        'localizedname' => get_string('dayofweek', 'mod_myplugin'),
        'monday' => get_string('monday', 'mod_myplugin'),
        'tuesday' => get_string('tuesday', 'mod_myplugin'),
        'wednesday' => get_string('wednesday', 'mod_myplugin'),
        'thursday' => get_string('thursday', 'mod_myplugin'),
        'friday' => get_string('friday', 'mod_myplugin'),
        'saturday' => get_string('saturday', 'mod_myplugin'),
        'sunday' => get_string('sunday', 'mod_myplugin')
    ],  'location' => [
        'localizedname' => get_string('location', 'mod_myplugin')
    ],
]);

For columns that contain date and time values (as Unix timestamp) you can enable a datepicker to allow users to filter the values:

$table->define_filtercolumns([
            'startdate' => [ // Columns containing Unix timestamps can be filtered.
                'localizedname' => get_string('startdate'),
                'datepicker' => [
                    'from time' => [ // Can be localized and like "Courses starting after:".
                        'operator' => '>', // Must be defined, can be any SQL comparison operator.
                        'defaultvalue' => 'now', // Can also be Unix timestamp or string "now".
                    ],
                    'end time' => [ // Can be localized and like "Courses starting after:".
                        'operator' => '<',
                        'defaultvalue' => '1670999000', // Can also be Unix timestamp or string "now".
                    ]
                ]
            ],
]);

A special type of datepicker filter is the timespan filter which will take the input of two date- and timepickers and apply to two columnvalues of a record. This enables comparison of two timespans. Possible operations are 'within', 'overlapboth', 'overlapstart', 'overlapend', 'before', 'after' and 'flexoverlap'. "Overlapstart" filter will only display records with starttime before and ending within the timespan of the filter, "within" filter will display records starting after and ending before the values of the filter timespan. "Flexoverlap" will include all kinds of overlaping: overlapping the beginning, the end, both sides or within. The possibleoperations array is containing a whitelist, if none specified, all are applied. 'datepicker' => [ 'In between' => [ // Timespan filter with two datepicker-filtercontainer applying to two columns (i.e. startdate, enddate). 'possibleoperations' => ['within', 'overlapboth', 'overlapstart', 'overlapend', 'before', 'after', 'flexoverlap'], // Will be displayed in select to choose from. 'columntimestart' => 'startdate', // Columnname as is query with lower value. Column has to be defined in $table->define_columns(). 'columntimeend' => 'enddate', // Columnname as is query with higher value. Column has to be defined in $table->define_columns(). 'labelstartvalue' => get_string('startvalue', 'local_wunderbyte_table'), // Can also be Unix timestamp or string "now". 'defaultvaluestart' => '1670999000', // Can also be Unix timestamp or string "now". 'labelendvalue' => get_string('endvalue', 'local_wunderbyte_table'), // Can also be Unix timestamp or string "now". 'defaultvalueend' => 'now', // Can also be Unix timestamp or string "now". 'checkboxlabel' => get_string('apply_filter', 'local_wunderbyte_table'), // Can be localized and will be displayed next to the checkbox. ] ]

By default filters are displayed next to the table and can be hidden with the filterbutton. If you want them to be hidden on load, set filteronloadinactive = true in the instance of your table.

The Hourslist filter can extract full hours from a unix timestamp and thus only show results which e.g. happened at a certain time. One example would be in mod_booking to show only courses which start at a given time (like 10:00 - 11:00).

This only works with mariadb, mysql and postgres db.

Example of a filter json for this:

$filtercolumns = [
    'id' => [
        'localizedname' => get_string('id', 'local_wunderbyte_table')
    ],
    'timemodified' => [ // Columns containing Unix timestamps can be filtered.
        'localizedname' => get_string('hourlastmodified', 'local_wunderbyte_table'),
        'hourlist' => true,
        0  => '00:00 - 01:00',
        1  => '01:00 - 02:00',
        2  => '02:00 - 03:00',
        3  => '03:00 - 04:00',
        4  => '04:00 - 05:00',
        5  => '05:00 - 06:00',
        6  => '06:00 - 07:00',
        7  => '07:00 - 08:00',
        8  => '08:00 - 09:00',
        9  => '09:00 - 10:00',
        10 => '10:00 - 11:00',
        11 => '11:00 - 12:00',
        12 => '12:00 - 13:00',
        13 => '13:00 - 14:00',
        14 => '14:00 - 15:00',
        15 => '15:00 - 16:00',
        16 => '16:00 - 17:00',
        17 => '17:00 - 18:00',
        18 => '18:00 - 19:00',
        19 => '19:00 - 20:00',
        20 => '20:00 - 21:00',
        21 => '21:00 - 22:00',
        22 => '22:00 - 23:00',
        23 => '23:00 - 00:00',
    ],
];

Sorting

If the output template you want to use doesn't support clickable headers to sort (eg because you use cards), you might want to use the sort select. Just add

$table->cardsort = true;

to your wunderbyte_table classs (look in the _container templates to understand how this works).

Each column defined as sortable will display carets in the table header. They have the same functionality as sortcolumn select & changesortorder element displayed in top of table.

If you want to define default sorting on table load, set:

$table->sort_default_column = 'columnname';
$table->sort_default_order = SORT_ASC; // Or SORT_DESC.

Search

The fulltext search is triggerd when more than 3 characters are typed into the searchfield. Fulltext search is checking values from all columns defined in $table->define_fulltextsearchcolumns(["column1", "column2"]). If you want to look for values in a specific column, use columnname:searchterm. This can be applied for numerous queries and combined with regular fulltext search. Combinations are seperated via whitespace and/or comma. If you want to use values containing whitespaces, use double (or single) quotes ie. "Localized Column":"value including whitespace". Searchterms will be used like wildcards, while quoted values and numbers trigger exact search (no wildcard). If the "Enter" key is pressed, search will be toggled immediately, no matter how many characters are set.

Display

If you want to display multiple tables on one page, tabs can be enabled in templates.

For the display of localized names in tableheaders, use the define_headers function.

$columns = [
            'id' => get_string('id', 'local_wunderbyte_table'),
            'username' => get_string('username'),
            'firstname' => get_string('firstname'),
            'lastname' => get_string('lastname'),
            'email' => get_string('email'),
            'action' => get_string('action'),
        ];
$table->define_headers(array_values($columns));
$table->define_columns(array_keys($columns));

Exploding strings for columns storing multiple values

The define_filtercolumns function also supports columns with multiple values stored as string with a separator.

You can define the separator like this:

$table->define_filtercolumns([
    'mycolname' => [
        'localizedname' => get_string('mystring', 'mod_myplugin'),
        'explode' => ',', // In this example, a comma is the separator, you might need another one.
    ],
    'mytimestampcolumn' => [
        'localizedname' => get_string('mystring', 'mod_myplugin'),
        'datepicker' => [
            'label' => [ // Can be localized and like "Courses starting after:".
                'operator' => '<',
                'defaultvalue' => '1680130800', // Can also be string "now".
            ]
        ]
    ],
]);

Handle JSON objects

The define_filtercolumns function also supports columns storing one or multiple JSON objects. You can define the attribute of the JSON object which should be used for the filter:

$table->define_filtercolumns([
    'mycolname' => [
        'localizedname' => get_string('mystring', 'mod_myplugin'),
        'jsonattribute' => 'name', // Replace 'name' with the actual attribute name.
    ],
]);

Lazy loading vs. direct out

To lazy load wunderbyte table (eg. for loading in tabs or modals) you need to call $table->lazyout() instead of $table->out. While out will return the html to echo, lazyout echos right away. If you want the html of lazyout, use $table->lazyouthtml();

Installing via uploaded ZIP file

  1. Log in to your Moodle site as an admin and go to Site administration > Plugins > Install plugins.
  2. Upload the ZIP file with the plugin code. You should only be prompted to add extra details if your plugin type is not automatically detected.
  3. Check the plugin validation report and finish the installation.

Installing manually

The plugin can be also installed by putting the contents of this directory to

{your/moodle/dirroot}/local/wunderbyte_table}

Afterwards, log in to your Moodle site as an admin and go to Site administration > Notifications to complete the installation.

Alternatively, you can run

$ php admin/cli/upgrade.php

to complete the installation from the command line.

License

2023 Wunderbyte GmbH info@wunderbyte.at

This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.

This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.

You should have received a copy of the GNU General Public License along with this program. If not, see https://www.gnu.org/licenses/.