Sidebar Menu
Introduction
The following provides an overview of how to create and manage interactive menu items in the side bar of the ExamSys user interface. It will also explain how to implement new menu items using the available templates, JavaScript functionality and data classes as well as how to do this for other pages.
The following applies to versions 7.70 and above
Main Components Overview
The sidebar system consists of three main components:
Data Layer: Data classes that mainly serve the purpose of structuring menu items (eg PaperMenuItemData.class.php)
Template Layer: The template that is used by the twig rendering engine in order to render the menu item (eg menuitem.html)
Behaviour Layer: The sidebar.js file that is the JavaScript file that handles the interactivity of the menu item.
Creating Menu Item Data Classes
Data Class Structure
/**
* Data class that handles the creation of menu items for your page context
*
* @author Your Name
* @copyright Copyright (c) 2024 The University of Nottingham
*/
class YourMenuItemData
{
/** @var array Language strings used for menu item text */
private $string;
/** @var string The root path of the installation */
private $rootPath;
/**
* Constructor for YourMenuItemData
*
* @param array $string Array of language strings for menu items
*/
public function __construct(array $string)
{
$this->rootPath = Config::get_instance()->get('cfg_root_path');
$this->string = $string;
}
/**
* Generate methods for each menu item following the standard format:
* - Clear method name describing the item's purpose
* - Return array with standard menu item structure
* - Include relevant parameters for item generation
*/
public function getYourMenuItem(array $params): array
{
return [
'classes' => 'yourclasses',
'icon' => $this->rootPath . '/artwork/your_icon.png',
'text' => $this->string['yourmenuitem'],
// ... additional properties
];
}
}
Key Principles for Data Classes
Consistent Structure
Follow the aforementioned menu item data structure
Maintain similar method naming conventions
Documentation
Document newly created public methods
Include parameter and return type assertions
Language Support
Use language strings for all text content displayed to the user
Template Structure
Basic template organisation
New templates pertaining to the sidebar are to be created inside the sidebar directory which is nested inside the templates directory:
templates/ └── sidebar/ ├── menuitem.html # Standard menu items ├── summativemenuitem.html # Summative checklist items └── yournewmenuitem.html # Custom template variations
The menuitem.html will enable you to generate menu items that navigate to a direct url, open a window on the same page and create submenus that use the makeMenu method in the sidebar_menu.inc file
The summativemenuitem.html allows for the creation of summative checklist menu items in the paper_options.php page
The styling for the menuitem.html templates is located inside the submenu.css file.
Implementing Menu Items
Create the data structure
First, we create the menu item data structure inside of its respective data class:
// In your data class (e.g., PaperMenuItemData)
public function getEditPropertiesItem(int $paperID, string $module, string $folder): array
{
return [
'classes' => 'properties',
'icon' => $this->rootPath . '/artwork/properties_icon.gif',
'text' => $this->string['editproperties'],
'href' => $this->rootPath . "/paper/properties.php?paperID=$paperID&caller=details&module=$module&folder=$folder",
'action' => 'directUrl'
];
}
Render on the page
We then render the menu item on the page that it is needed in below is how something like this may look like, taking the papertasks menu as an example:
// Initialize required objects
$menuItemData = new PaperMenuItemData($string);
$render = new render($configObject);
// Create a menu section
echo '<div class="submenuheading" id="papertasks">' . $string['papertasks'] . '</div>';
echo '<div id="menu1" role="menu">';
// Render individual items
$testPreviewItem = $menuItemData->getTestPreviewItem($properties, $properties->get_crypt_name());
$render->render($testPreviewItem, $string, 'sidebar/menuitem.html');
$editPropertiesItem = $menuItemData->getEditPropertiesItem($paperID, $module, $folder);
$render->render($editPropertiesItem, $string, 'sidebar/menuitem.html');
echo '</div>';
Menu Item Actions Guide
Menu items in the sidebar use a data-action attribute to define their behaviour. These actions are then handled by the handleMenuItemAction function in the sidebar.js file.
Core Action Types
Direct URL Navigation (directUrl)
Used for direct page navigation. The action looks for an anchor tag inside of a menu item and then navigates to its href value.
Required properties:
action: Must be set to ‘directUrl’
href: The URL to navigate to
// In your data class
public function getDirectUrlItem(): array
{
return [
'classes' => 'properties',
'icon' => $this->rootPath . '/artwork/icon.gif',
'text' => $this->string['menutext'],
'href' => $this->rootPath . '/path/to/page.php',
'action' => 'directUrl'
];
}
// How it's handled in sidebar.js
case 'directUrl':
var href = menuItem.find('a').attr('href');
if (href) {
window.location = href;
}
break;
Popup Windows (openPopup)
Opens a new window popup on the same page.
Required data_attributes:
url: The URL of the page to display in the popup
popuptype: Must be set to "window"
name: Identifier for the popup window
features: Window features string
Window feature options:
width: Width in pixels
height: Height in pixels
scrollbars: yes/no
toolbar: yes/no
location: yes/no
directories: yes/no
status: yes/no
menubar: yes/no
resizable: yes/no
left: Position from left
top: Position from top
public function getPopupItem(): array
{
return [
'classes' => 'popupclass',
'icon' => $this->rootPath . '/artwork/popup_icon.gif',
'text' => $this->string['popuptext'],
'action' => 'openPopup',
'data_attributes' => [
'url' => $this->rootPath . '/popup/content.php',
'popuptype' => 'window',
'name' => 'previewWindow',
'features' => 'width=800,height=600,scrollbars=yes',
'focus' => true
]
];
}
case 'openPopup':
var settings = menuItem.data();
if (settings.popuptype === 'window') {
var popup = window.open(settings.url, settings.name, settings.features);
if (settings.focus && window.focus) {
popup.focus();
}
}
break;
Submenus (openSubMenu)
Creates cascading menu functionality. Used when a menu item needs to display additional options.
Required properties:
action: Must be set to "openSubMenu"
type: Must be set to "button"
hasPopup: Must be set to true
popupType: Must be set to ‘menu’
Required data_attributes:
popupid: Unique identifier for the submenu
popuptype: Category of submenu (e.g., "papertasks", "banktasks")
popupname: Reference name for the submenu
public function getSubmenuItem(): array
{
return [
'classes' => 'cascade showmenu',
'icon' => $this->rootPath . '/artwork/submenu_icon.gif',
'text' => $this->string['submenutext'],
'href' => '#',
'action' => 'openSubMenu',
'type' => 'button',
'hasPopup' => true,
'popupType' => 'menu',
'data_attributes' => [
'popupid' => '1', // Unique identifier for the submenu
'popuptype' => 'papertasks',// Type of submenu
'popupname' => 'submenu1' // Name for reference
]
];
}
case 'openSubMenu':
var options = JSON.parse($("#popupmenu" + menuItem.attr('data-popupid')).attr('data-myOptions'));
var urls = JSON.parse($("#popupmenu" + menuItem.attr('data-popupid')).attr('data-myURLs'));
var id = 'popup' + menuItem.attr('data-popupid');
var type = menuItem.attr('data-popuptype');
var name = menuItem.attr('data-popupname');
this.showMenu(id, type, name, options, urls, e);
break;
Adding New Actions
You can extend the system with custom actions by-
Defining the action inside you menu item data:
public function getCustomActionItem(): array
{
return [
'classes' => 'customaction',
'icon' => $this->rootPath . '/artwork/custom_icon.gif',
'text' => $this->string['customtext'],
'action' => 'myCustomAction',
'data_attributes' => [
'custom-data' => 'value',
]
];
}
2. Adding the case handler inside the sidebar.js file.
case 'myCustomAction':
var customData = menuItem.data('custom-data');
// Implement your custom behavior
break;
Prior to ExamSys 7.7.0
The following applies to versions prior to 7.7.0
A number of sidebar menus have different states depending on what has been selected. For example, when in a paper initially most of the 'Question Tasks' options are greyed out. These become active links once the user has selected a question. To do this all the links within 'Questions Tasks' are held in two <DIV> containers: one with the greyed out options and one with the active links. When the user selects a question JavaScript hides the grey <DIV> and shows the active link <DIV>.
The sidebar menus are contained within include files, brought in from /include/ directory. The language strings that are needed for the sidebar menu should be included in the 'parent' PHP file, they do not need their own separate language include file.