Difference between revisions of "Editing Client Area Menus"
|  (→Add a special offer image and link to the top of the topmost sidebar) |  (→Add a special offer image and link to the top of the topmost sidebar) | ||
| Line 432: | Line 432: | ||
| <syntaxhighlight lang="php"> | <syntaxhighlight lang="php"> | ||
| <?php | <?php | ||
| − | + | ||
| use WHMCS\View\Menu\Item as MenuItem; | use WHMCS\View\Menu\Item as MenuItem; | ||
| − | + | ||
| add_hook('ClientAreaPrimarySidebar', 1, function (MenuItem $primarySidebar) | add_hook('ClientAreaPrimarySidebar', 1, function (MenuItem $primarySidebar) | ||
| { | { | ||
| Line 444: | Line 444: | ||
| </a> | </a> | ||
| EOT; | EOT; | ||
| − | + | ||
| − | + |      // Add a link to the special to the first panel's body HTML. It will render | |
|      // above the panel's menu item list. |      // above the panel's menu item list. | ||
| − |      $primarySidebar->getFirstChild()->setBodyHtml($specialOfferHtml); | + |      $firstSidebar = $primarySidebar->getFirstChild(); | 
| + |     if ($firstSidebar) { | ||
| + |     	$firstSidebar->setBodyHtml($specialOfferHtml); | ||
| + |     } | ||
| }); | }); | ||
| </syntaxhighlight> | </syntaxhighlight> | ||
Revision as of 06:50, 1 September 2015
WHMCS version 6 introduces a programmatic way to interact with the client area navigation and sidebars through hooks and modules.
Contents
- 1 Menu structure
- 2 Menu layout
- 3 Menu items
- 4 Menu item arrangement
- 5 Interacting with menus
- 6 Examples
- 6.1 Add a social media panel to the end of the sidebar
- 6.2 Add a special offer image and link to the top of the topmost sidebar
- 6.3 Move the “Contact Us” link to the secondary navigation bar and add more contact options
- 6.4 Add support hours and a custom message to the sidebar on the submit ticket page
 
Menu structure
The client area's navigation and sidebars are defined in a tree structure of menu item objects. Each menu item has one parent item and can have many child items. The only menu item with no parent an invisible root item that is not displayed on the page.
Navigation bars consist of the invisible menu root with children representing every item displayed in the navigation bar. Each of these items may have their own child items. These child items are rendered as that navigation bar item's dropdown menu.
-  Navigation bar root Item
- navigation item
-  navigation item
- dropdown item
 
-  navigation item
- dropdown item
- dropdown item
- dropdown item
 
- navigation item
 
Navigation bars are displayed on every page in the client area, but their contents may change if a client is logged in or not. For instance, the navigation bar may show login and password recovery links if a user isn't logged in.
Sidebars
Like the navigation bar, the sidebars in WHMCS begin with an invisible menu root, but each child represents an individual panel in the side bar. Each panel item is rendered as an item within the panel in the WHMCS client area.
-  Sidebar root Item
-  panel
- panel item
- panel item
- panel item
 
-  panel
- panel item
- panel item
 
-  panel
- panel item
- panel item
- panel item
 
 
-  panel
Sidebars help provide context for the data displayed on the page. Different pages in the client area may have different sidebar items. For example, a page to view an account may contain sidebar links to view that client’s open tickets or unpaid invoices.
Menu layout
Desktop mode
There are two navigation bars in WHMCS’ client area. The primary navigation bar contains the bulk of the menu and floats to the left of the secondary navigation bar. The secondary navigation bar contains user-specific items and changes if a client is logged in to WHMCS. When a client is not logged in then the secondary navigation contains a login link, and when a client is logged in then the secondary menu contains links to the client’s account.Likewise there are two sidebars on every client area page. The primary sidebar is displayed above the secondary side bar on the left side of the page. Sidebar content varies per page, though the primary sidebar typically displays information directly relevant to the page content while the secondary sidebar usually contains more general links.
Responsive mode
Responsive mode is activated when WHMCS’ client area is viewed on a smaller screen device such as a phone or tablet. WHMCS rearranges the page layout in responsive mode for best display on mobile devices.In responsive mode the primary and secondary navigation bars are displayed above the primary sidebar, followed by the page’s content with the secondary sidebar displayed at the bottom of the page.
Menu items
Menu items are modeled in code by the \WHMCS\View\Menu\Item class. These objects contain all of the information needed to render that menu item within a template, including their parent and child menu item relationships. Menu items can have the following aspects:
- A single parent item
- Multiple optional child items
- A name used to internally refer to the menu item
- A label to display when the item is rendered to the page. If no label is defined, then WHMCS render's the menu item's name
- A URI to link to when the user clicks or taps on a menu item
- An optional icon displayed to the left of the label. WHMCS has access to both the Glyphicons and Font Awesome libraries.
- An optional badge displayed to the right of the label, usually used for contextual information, such as the number of tickets in a status next to that statuses' name
- The order that a menu item is displayed in its parent's list of children.
Menu item arrangement
This diagram represents an individual menu item in the navigation bars. All of the menu item’s children are rendered as dropdown menu items. All menu items are rendered with icons to the left of a link to the menu item’s URI followed by the menu item’s badge.
Sidebars
This diagram represents an individual menu item in the sidebars. Unlike the navigation bars this menu item renders a panel in the side bar. Items in the panel rendered by the menu item’s children. Note how body and footer HTML content are also rendered around sidebar panel items. All menu items are rendered with icons to the left of a link to the menu item’s URI followed by the menu item’s badge.Hooks
WHMCS 6.0 introduces a number of hooks to allow menu interaction before they’re sent to the template renderer. Use WHMCS’ add_hook() function to call custom code when WHMCS reaches these hook points during page generation.
-  ClientAreaPrimaryNavbar
- Called prior to rendering navigation bars.
- Passes the primary navigation bar object to the hook function.
 
-  ClientAreaSecondaryNavbar
- Called prior to rendering navigation bars.
- Passes the secondary navigation bar object to the hook function.
 
-  ClientAreaNavbars
- Called prior to rendering navigation bars.
- Passes no parameters to the hook function.
 
-  ClientAreaPrimarySidebar
- Called prior to rendering sidebars.
- Passes the primary side bar object to the hook function.
 
-  ClientAreaSecondarySidebar
- Called prior to rendering sidebars.
- Passes the secondary side bar object to the hook function.
 
-  ClientAreaSidebars
- Called prior to rendering sidebars.
- Passes no parameters to the hook function
 
Direct Access
WHMCS allows direct manipulation of menu objects outside the hooks system for modules and other custom code that don’t use the hooks system. The built-in Menu class is an alias to an object repository that can retrieve all of WHMCS’ menu objects. Please note that if the menu isn’t generated by the page yet then an empty menu structure may exist that is overwritten by normal page generation. WHMCS recommends using the hooks system to interact with menus.
The Menu class has four static methods to retrieve menus:
-  Item Menu::primaryNavbar()
- Retrieve the primary navigation bar.
 
-  Item Menu::secondaryNavbar()
- Retrieve the secondary navigation bar.
 
-  Item Menu::primarySidebar()
- Retrieve the primary sidebar.
 
-  Item Menu::secondarySidebar()
- Retrieve the secondary sidebar.
 
WHMCS employs a number of pre-built sidebars in its built-in pages. These side bars are available to hook and module developers through the Menu::PrimarySidebar() and Menu::SecondarySidebar() methods. Call either of these methods with the name of the sidebar as the first parameter to retrieve the pre-built sidebar. WHMCS will build the side bar if it isn't already defined. See the table below for the currently defined pre-built sidebars and which pages they are used on.
Context
WHMCS’ menus, especially the sidebars, render information specific to the page in the client area that’s being accessed by the user. For instance, client information is passed to the “my account” page and ticket information is passed to the “view ticket page”. This data is passed to menu item objects as context. Context can be any PHP object or data type. The Menu class has two static methods for setting and retrieving context items:
-  void Menu::addContext(string $key, mixed $value)
- Add $value to the menu context at the key $key, overriding existing values.
 
-  mixed|null Menu::context(string $key)
- Retrieve the menu context at $key or null if no context exists at the key.
 
| Sidebar name | Pages used on | Context | 
|---|---|---|
| affiliateView | 
 | 
 | 
| announcementList | 
 | 
 | 
| clientAddFunds | 
 | 
 | 
| clientRegistration | register.php | 
 | 
| clientQuoteList | 
 | 
 | 
| clientView | 
 | 
 | 
| domainList | 
 | 
 | 
| If the q request variable is defined, then the client's associated domain counts in the primary sidebar are filtered to those domains whose names contain q's value. | ||
| domainView | 
 | 
 | 
| downloadList | 
 | 
 | 
| The topFiveDownloads and 'downloadCategory contexts are only passed to download.php. | ||
| invoiceList | 
 | 
 | 
| networkIssueList | serverstatus.php | 
 | 
| serviceList | 
 | 
 | 
| If the q request variable is defined, then the client's associated service counts in the primary sidebar are filtered to those services whose domain names contain q's value. | ||
| serviceUpgrade | 
 | 
 | 
| serviceView | 
 | 
 | 
| sslCertificateOrderView | 
 | 
 | 
| support | various | 
 | 
| The support secondary sidebar is displayed when a contact does not have permission to perform an action in the client area. | ||
| supportKnowledgeBase | 
 | 
 | 
| ticketFeedback | 
 | 
 | 
| Only used on viewticket.php when a user is entering feedback for a recently closed ticket. | ||
| ticketList | 
 | 
 | 
| ticketSubmit | 
 | 
 | 
| ticketView | 
 | 
 | 
Examples
Add a social media panel to the end of the sidebar
Let's say we’re a hosting company who wants to better connect with our clients by using social media. The menu hooks introduced in WHMCS 6.0 allow us to insert links to our company’s Facebook, Twitter, and Google+ profiles directly into the interface. We’ll put them at the end of the secondary sidebar so they’re rendered last and won’t interrupt our users’ experience.This example uses the ClientAreaSecondarySidebar hook and the menu item’s addChild()and moveToBack() methods. To add panel with links to the sidebar we must:
- Create a panel at the end of the sidebar.
- Retrieve the panel we just created.
- Add social media links to the panel.
Fortunately the Font Awesome library already has icons for all of these services. Create the includes/hooks/socialMediaPanel.php file in your WHMCS installation and enter the code below. Save the file and reload your WHMCS installation’s client area. WHMCS automatically loads all hooks the includes/hooks directory. The SecondarySidebar hook is registered with add_hook() and is consequently loaded every time WHMCS renders the secondary sidebar on page load.
<?php
use WHMCS\View\Menu\Item as MenuItem;
// Add social media links to the end of all secondary sidebars.
add_hook('ClientAreaSecondarySidebar', 1, function (MenuItem $secondarySidebar)
{
    // Add a panel to the end of the secondary sidebar for social media links.
    // Declare it with the name "social-media" so we can easily retrieve it
    // later.
    $secondarySidebar->addChild('social-media', array(
        'label' => 'Social Media',
        'uri' => '#',
        'icon' => 'fa-thumbs-up',
    ));
    // Retrieve the panel we just created.
    $socialMediaPanel = $secondarySidebar->getChild('social-media');
    // Move the panel to the end of the sorting order so it's always displayed
    // as the last panel in the sidebar.
    $socialMediaPanel->moveToBack();
    // Add a Facebook link to the panel.
    $socialMediaPanel->addChild('facebook-link', array(
        'uri' => 'https://facebook.com/our-great-company',
        'label' => 'Like us on Facebook!',
        'order' => 1,
        'icon' => 'fa-facebook',
    ));
    // Add a Twitter link to the panel after the Facebook link.
    $socialMediaPanel->addChild('twitter-link', array(
        'uri' => 'https://twitter.com/ourgreatcompany',
        'label' => 'Follow us on Twitter!',
        'order' => 2,
        'icon' => 'fa-twitter',
    ));
    // Add a Google+ link to the panel after the Twitter link.
    $socialMediaPanel->addChild('google-plus-link', array(
        'uri' => 'https://plus.google.com/1234567890123456',
        'label' => 'Add us to your circles!',
        'order' => 3,
        'icon' => 'fa-google-plus',
    ));
});Add a special offer image and link to the top of the topmost sidebar
Your web host is doing amazing and to celebrate is offering a discount to all users. The marketing folks want a small image and message in the top of the first sidebar panel on every page that links to a page with the special offer.This example uses the ClientAreaPrimarySidebar hook and the menu item’s getFirstChild() and setBodyHtml() methods. To add panel with links to the sidebar we must:
- Get the first panel from the primary sidebar.
- Set the panel’s body HTML.
Create the includes/hooks/specialOfferInSidebar.php file in your WHMCS installation and enter the code below. As with the previous example the new file and hook within the file is run on page load and adds the image and link to the topmost panel in the primary sidebar, no matter which page the client is accessing.
<?php
use WHMCS\View\Menu\Item as MenuItem;
add_hook('ClientAreaPrimarySidebar', 1, function (MenuItem $primarySidebar)
{
    // The HTML for the link to the the special offer.
    $specialOfferHtml = <<<EOT
<a href="//myawesomecompany.com/special-offer/">
   <img src="/assets/img/catdeals.png" alt="Click here for amazing deals!">
   Kitten says <strong><em>thanks</em></strong> for making us the best web host!
</a>
EOT;
     // Add a link to the special to the first panel's body HTML. It will render
    // above the panel's menu item list.
    $firstSidebar = $primarySidebar->getFirstChild();
    if ($firstSidebar) {
    	$firstSidebar->setBodyHtml($specialOfferHtml);
    }
});This example requires manipulating more than one menu bar. To do that we’ll use the ClientAreaNavbars hook and the Menu class to retrieve the primary and secondary navigation bars. We’ll use the menu item’s getChild(), removeChild(), addChild(), and moveToFront() methods and the static Menu::primaryNavbar() and Menu::secondaryNavbar() methods. Here’s what we’ll do:
- Save the “Contact Us” link from the primary navigation bar.
- Remove the “Contact Us” link from the primary navigation bar.
- Add an email link child to the “Contact Us” link.
- Add a phone link child to the “Contact Us” link.
- Add a map link child to the “Contact Us” link.
- Add the “Contact Us” link to the secondary navigation bar then move it to the beginning of the menu.
Create the includes/hooks/moveContactUsLink.php file in your WHMCS installation and enter the code below. As with the other examples this hook file is picked up on page load and rearranges the navigation bars with the appropriate dropdown items.
<?php
add_hook('ClientAreaNavbars', 1, function ()
{
    // Get the current navigation bars.
    $primaryNavbar = Menu::primaryNavbar();
    $secondaryNavbar = Menu::secondaryNavbar();
    // Save the "Contact Us" link and remove it from the primary navigation bar.
    $contactUsLink = $primaryNavbar->getChild('Contact Us');
    $primaryNavbar->removeChild('Contact Us');
    // Add the email sales link to the link's drop-down menu.
    $contactUsLink->addChild('email-sales', array(
        'label' => 'Email our sales team',
        'uri' => 'sales@my-awesome-company.com',
        'order' => 1,
        'icon' => 'fa-diamond',
    ));
    // Add the call us link to the link's drop-down menu.
    $contactUsLink->addChild('call-us', array(
        'label' => 'Call us',
        'uri' => 'tel:+18005551212',
        'order' => 2,
        'icon' => 'fa-mobile',
    ));
    // Add the map to the company to the link's drop-down menu.
    $contactUsLink->addChild('map', array(
        'label' => '123 Main St. AnyTown, TX 11223, USA',
        'uri' => 'https:\/\/maps.google.com/maps/place/some-map-data',
        'order' => 3,
        'icon' => 'fa-map-marker',
    ));
    // Add the link and its drop-down children to the secondary navigation bar.
    $secondaryNavbar->addChild($contactUsLink);
    // Make sure the contact us link appears as the first item in the
    // secondary navigation bar.
    $contactUsLink->moveToFront();
});Add support hours and a custom message to the sidebar on the submit ticket page
Since the powers that be want this to appear at the top of the sidebars we'll manipulate the primary sidebar via the ClientAreaPrimarySidebar hook. We’ll use the menu item’s addChild(), moveToFront(), and setBodyHtml() methods to add the new panel. The special message to the user addresses the logged in user by first name. Every sidebar has a "client" context available which contains the record of the client that is logged in or null if no client is logged in. The Menu::context() method will retrieve the client record for us. If the client is logged in then we'll use the client object's firstName property to address the user by name. Version 6.0 uses the very helpful Carbon date library internally. Carbon is available to third party developers, so we'll use it to determine if support is currently open. Here’s what we’ll do:
- Determine if the user is visiting submitticket.php.
- Add a "Support Hours" panel to the primary sidebar and move it to the front so it displays at the top.
- Create child items in the support hours panel saying when support is open and closed.
- Determine if support is currently open.
- If there is a user logged in then determine their first name.
- Assign the support hours' panel body HTML to a special message depending on the logged in user's first name and whether support is currently open.
Create the includes/hooks/addSupportHours.php file in your WHMCS installation and enter the code below. As with the other examples this hook file is picked up on page load and adds the custom panel and message to the primary sidebar before the submit ticket page renders.
<?php
use Carbon\Carbon;
use WHMCS\View\Menu\Item as MenuItem;
// Add a helpful support hours notice to the top of the sidebar on the submit
// ticket page.
if (App::getCurrentFilename() == 'submitticket') {
    add_hook('ClientAreaPrimarySidebar', 1, function (MenuItem $primarySidebar) 
    {
        // Create the support hours panel and make sure it's the first one
        // displayed.
        /** @var MenuItem $supportHours */
        $supportHours = $primarySidebar->addChild('Support Hours');
        $supportHours->moveToFront();
        // Add hours to the panel.
        $supportHours->addChild(
            '<strong>Open</strong> 08:00-17:00 M-F',
            array(
                'icon'  => 'fa-smile-o',
                'order' => 1,
            )
        );
        $supportHours->addChild(
            '<strong>Closed</strong> Weekends',
            array(
                'icon'  => 'fa-frown-o',
                'order' => 2,
            )
        );
        // Add a custom notice to the support hours panel with the logged in
        // client's first name and a different message depending on whether
        // support is open.
        /** @var \WHMCS\User\Client $client */
        $client = Menu::context('client');
        $greeting = is_null($client)
            ? ''
            : ", <strong>{$client->firstName}</strong>";
    
        $now = Carbon::now();
        $supportIsOpen = $now->isWeekday()
            && $now->hour >= 8
            && $now->hour <= 17;
        $supportHours->setBodyHtml(
            $supportIsOpen
                ? "Hi{$greeting}! We're open and will respond to your ticket soon!"
                : "Don't worry{$greeting}! We will respond on the next business day. Sit tight!"
        );
    });
}






