Difference between revisions of "Editing Client Area Menus"
|  (→Add a social media panel to the end of the sidebar) | James.white (talk | contribs)   (→Context) | ||
| (22 intermediate revisions by 4 users not shown) | |||
| Line 1: | Line 1: | ||
| − | WHMCS  | + | WHMCS includes a programmatic way to interact with the client area navigation and sidebars through hooks and modules. | 
| − | <div class="docs-alert-warning"> | + | <div class="docs-alert-warning"> | 
| − | + | * We added this feature in WHMCS 6. | |
| − | + | * For copy-and-paste code examples for common customizations using this system, see [[Client Area Navigation Menus Cheatsheet]] and [[Client Area Sidebars Cheatsheet]]. | |
| </div> | </div> | ||
| − | =Menu structure= | + | ==Menu structure== | 
| The client area's navigation and sidebars are defined in a tree structure of [http://docs.whmcs.com/classes/classes/WHMCS.View.Menu.Item.html 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. | The client area's navigation and sidebars are defined in a tree structure of [http://docs.whmcs.com/classes/classes/WHMCS.View.Menu.Item.html 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== | + | ===Navigation bars=== | 
| 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 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.   | ||
| Line 26: | Line 26: | ||
| 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. | 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== | + | ===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. | 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. | ||
| Line 45: | Line 45: | ||
| 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. | 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= | + | ==Menu layout== | 
| − | ==Desktop mode== | + | ===Desktop mode=== | 
| [[Image:Desktop mode.png|thumb|The WHMCS 6.0 client area layout in 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. | [[Image:Desktop mode.png|thumb|The WHMCS 6.0 client area layout in 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. | ||
| Line 55: | Line 55: | ||
| <div style="clear: both"></div> | <div style="clear: both"></div> | ||
| − | ==Responsive mode== | + | ===Responsive mode=== | 
| [[Image:Responsive mode.png|thumb|The WHMCS 6.0 client area layout in 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.   | [[Image:Responsive mode.png|thumb|The WHMCS 6.0 client area layout in 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.   | ||
| Line 63: | Line 63: | ||
| <div style="clear: both"></div> | <div style="clear: both"></div> | ||
| − | =Menu items= | + | ==Menu items== | 
| Menu items are modeled in code by the [https://docs.whmcs.com/classes/7.6/WHMCS/View/Menu/Item.html <tt>\WHMCS\View\Menu\Item</tt>] 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: | Menu items are modeled in code by the [https://docs.whmcs.com/classes/7.6/WHMCS/View/Menu/Item.html <tt>\WHMCS\View\Menu\Item</tt>] 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: | ||
| Line 76: | Line 76: | ||
| * The order that a menu item is displayed in its parent's list of children. | * The order that a menu item is displayed in its parent's list of children. | ||
| − | =Menu item arrangement= | + | ==Menu item arrangement== | 
| − | ==Navigation bars== | + | ===Navigation bars=== | 
| [[Image:Navbar item.png|thumb|A client area navigation bar item represented as menu item properties.]] 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.   | [[Image:Navbar item.png|thumb|A client area navigation bar item represented as menu item properties.]] 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.   | ||
| Line 84: | Line 84: | ||
| <div style="clear: both"></div> | <div style="clear: both"></div> | ||
| − | ==Sidebars== | + | ===Sidebars=== | 
| [[Image:Sidebar panel.png|thumb|A client area sidebar panel represented as menu item properties.]] 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. | [[Image:Sidebar panel.png|thumb|A client area sidebar panel represented as menu item properties.]] 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. | ||
| Line 90: | Line 90: | ||
| <div style="clear: both"></div> | <div style="clear: both"></div> | ||
| − | =Interacting with menus= | + | ==Interacting with menus== | 
| − | ==Hooks== | + | ===Hooks=== | 
| WHMCS 6.0 introduces a number of hooks to allow menu interaction before they’re sent to the template renderer. Use WHMCS’ <tt>add_hook()</tt> function to call custom code when WHMCS reaches these hook points during page generation.   | WHMCS 6.0 introduces a number of hooks to allow menu interaction before they’re sent to the template renderer. Use WHMCS’ <tt>add_hook()</tt> function to call custom code when WHMCS reaches these hook points during page generation.   | ||
| Line 115: | Line 115: | ||
| ** Passes no parameters to the hook function | ** Passes no parameters to the hook function | ||
| − | ==Direct Access== | + | <div class="docs-alert-info"> | 
| + | <span class="title">Custom Actions</span><br /> | ||
| + | In WHMCS 8.5 and later, you can use the <tt>CustomActions</tt> server module function to add items to the server details sidebar menu. The item will perform a set function and then redirect the user to a specified URL. For more information, see [[Working With Client Area Home Page Panels]] and [https://developers.whmcs.com/provisioning-modules/custom-actions/ our Developer Documentation]. | ||
| + | </div> | ||
| + | |||
| + | ===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 <tt>Menu</tt> 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. | 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 <tt>Menu</tt> 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. | ||
| Line 132: | Line 137: | ||
| 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 <tt>Menu::PrimarySidebar()</tt> and <tt>Menu::SecondarySidebar()</tt> 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. | 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 <tt>Menu::PrimarySidebar()</tt> and <tt>Menu::SecondarySidebar()</tt> 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== | + | ===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  | + | 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  '''Account > My Account''' or, prior to WHMCS 8.0, '''My Account'''. 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 <tt>Menu</tt> class has two static methods for setting and retrieving context items: | 
| * ''void'' '''Menu::addContext(string $key, mixed $value)''' | * ''void'' '''Menu::addContext(string $key, mixed $value)''' | ||
| Line 161: | Line 166: | ||
| * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | ||
| ** The client currently logged in, or <tt>null</tt> if no client is logged in | ** The client currently logged in, or <tt>null</tt> if no client is logged in | ||
| − | * ''[[collection]] of [https://github.com/briannesbitt/Carbon \Carbon\Carbon]'': '''monthsWithAnnouncements''' | + | * <nowiki>''[[collection]] of [https://github.com/briannesbitt/Carbon \Carbon\Carbon]'': '''monthsWithAnnouncements'''</nowiki> | 
| ** The past ten months in which announcements have been published. | ** The past ten months in which announcements have been published. | ||
| |- | |- | ||
| Line 176: | Line 181: | ||
| * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | ||
| ** The client currently logged in, or <tt>null</tt> if no client is logged in. | ** The client currently logged in, or <tt>null</tt> if no client is logged in. | ||
| − | * ''[[collection]] of [http://docs.whmcs.com/classes/7.1/WHMCS/User/Client/SecurityQuestion.html \WHMCS\User\Client\SecurityQuestion]'': '''securityQuestions''' | + | * <nowiki>''[[collection]] of [http://docs.whmcs.com/classes/7.1/WHMCS/User/Client/SecurityQuestion.html \WHMCS\User\Client\SecurityQuestion]'': '''securityQuestions'''</nowiki> | 
| ** The configured system security questions. | ** The configured system security questions. | ||
| |- | |- | ||
| Line 192: | Line 197: | ||
| * <tt>clientarea.php?action=changepw</tt> | * <tt>clientarea.php?action=changepw</tt> | ||
| * <tt>clientarea.php?action=contacts</tt> | * <tt>clientarea.php?action=contacts</tt> | ||
| − | * <tt> | + | * <tt>index.php?rp=/account/paymentmethods</tt> | 
| * <tt>clientarea.php?action=details</tt> | * <tt>clientarea.php?action=details</tt> | ||
| * <tt>clientarea.php?action=emails</tt> | * <tt>clientarea.php?action=emails</tt> | ||
| Line 230: | Line 235: | ||
| * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | ||
| ** The client currently logged in, or <tt>null</tt> if no client is logged in | ** The client currently logged in, or <tt>null</tt> if no client is logged in | ||
| − | * ''[[collection]] of [http://docs.whmcs.com/classes/7.1/WHMCS/Download/Download.html \WHMCS\Download\Download]'': '''topFiveDownloads''' | + | * <nowiki>''[[collection]] of [http://docs.whmcs.com/classes/7.1/WHMCS/Download/Download.html \WHMCS\Download\Download]'': '''topFiveDownloads'''</nowiki> | 
| ** The five most downloaded files that are not hidden or in a hidden download category, ordered by number of downloads in descending order. | ** The five most downloaded files that are not hidden or in a hidden download category, ordered by number of downloads in descending order. | ||
| * ''[http://docs.whmcs.com/classes/7.1/WHMCS/Download/Category.html \WHMCS\Download\Category]'': '''downloadCategory''' | * ''[http://docs.whmcs.com/classes/7.1/WHMCS/Download/Category.html \WHMCS\Download\Category]'': '''downloadCategory''' | ||
| Line 258: | Line 263: | ||
| * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | ||
| ** The client currently logged in, or <tt>null</tt> if no client is logged in. | ** The client currently logged in, or <tt>null</tt> if no client is logged in. | ||
| − | * ''[[collection]] of [http://docs.whmcs.com/classes/7.1/WHMCS/Product/Group.html \WHMCS\Product\Group]'': '''productGroups''' | + | * <nowiki>''[[collection]] of [http://docs.whmcs.com/classes/7.1/WHMCS/Product/Group.html \WHMCS\Product\Group]'': '''productGroups'''</nowiki> | 
| ** A WHMCS installation's configured product groups. | ** A WHMCS installation's configured product groups. | ||
| * ''int'': '''productGroupId''' | * ''int'': '''productGroupId''' | ||
| Line 371: | Line 376: | ||
| ** The id number of the ticket the user is viewing. | ** The id number of the ticket the user is viewing. | ||
| |- | |- | ||
| + | ! Profile | ||
| + | |  | ||
| + | * <tt>clientarea.php?action=changepw</tt> | ||
| + | *<tt>clientarea.php</tt> | ||
| + | | | ||
| + | * ''[http://docs.whmcs.com/classes/7.1/WHMCS/User/Client.html \WHMCS\User\Client]|null'': '''client''' | ||
| + | ** The currently-authenticated client or <tt>null</tt> if there is not an authenticated client. | ||
| + | |- | ||
| + | |||
| |} | |} | ||
| − | =Examples | + | ==Examples== | 
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| + | ===Add a social media panel to the end of the sidebar=== | ||
| + | |||
| + | [[Image:Custom social media sidebar panel.png|thumb|The Client Area supports a secondary sidebar with a custom social media panel highlighted.]] | ||
| + | |||
| + | You may want to better connect with your clients by using social media. The menu hooks in WHMCS 6.0 and later allow you to insert links to your company’s Facebook, Twitter, and other social media profiles directly into the interface. We recommend putting them at the end of the secondary sidebar so they’re rendered last and won’t interrupt your users’ experience. | ||
| + | |||
| + | This example uses the <tt>ClientAreaSecondarySidebar</tt> hook and the menu item’s <tt>addChild()</tt>and <tt>moveToBack()</tt> methods. To add panel with links to the sidebar you will need to create a panel at the end of the sidebar, retrieve the new panel, and add social media links to the panel. The Font Awesome library already has icons for all of these services. | ||
| + | |||
| + | To do this: | ||
| + | |||
| + | # Create the <tt>includes/hooks/socialMediaPanel.php</tt> file in your WHMCS installation. | ||
| + | # Enter the code below. | ||
| + | # Save the file. | ||
| + | # Reload your WHMCS installation’s Client Area. | ||
| + | |||
| + | WHMCS automatically loads all hooks the <tt>includes/hooks</tt> directory. The <tt>SecondarySidebar</tt> hook is registered with <tt>add_hook()</tt> and is consequently loaded every time WHMCS renders the secondary sidebar on page load. | ||
| + | |||
| <div style="clear: both"></div> | <div style="clear: both"></div> | ||
| − | + | ||
| <syntaxhighlight lang="php"> | <syntaxhighlight lang="php"> | ||
| <?php | <?php | ||
| − | + | ||
| use WHMCS\View\Menu\Item as MenuItem; | use WHMCS\View\Menu\Item as MenuItem; | ||
| − | + | ||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| // Add social media links to the end of all secondary sidebars. | // Add social media links to the end of all secondary sidebars. | ||
| add_hook('ClientAreaSecondarySidebar', 1, function (MenuItem $secondarySidebar) | add_hook('ClientAreaSecondarySidebar', 1, function (MenuItem $secondarySidebar) | ||
| Line 408: | Line 422: | ||
|          'label' => 'Social Media', |          'label' => 'Social Media', | ||
|          'uri' => '#', |          'uri' => '#', | ||
| − |          'icon' => 'fa-thumbs-up', | + |          'icon' => 'fas fa-thumbs-up', | 
|      )); |      )); | ||
| − | + | ||
|      // Retrieve the panel we just created. |      // Retrieve the panel we just created. | ||
|      $socialMediaPanel = $secondarySidebar->getChild('social-media'); |      $socialMediaPanel = $secondarySidebar->getChild('social-media'); | ||
| − | + | ||
|      // Move the panel to the end of the sorting order so it's always displayed |      // Move the panel to the end of the sorting order so it's always displayed | ||
|      // as the last panel in the sidebar. |      // as the last panel in the sidebar. | ||
|      $socialMediaPanel->moveToBack(); |      $socialMediaPanel->moveToBack(); | ||
| − | + | ||
|      // Add a Facebook link to the panel. |      // Add a Facebook link to the panel. | ||
|      $socialMediaPanel->addChild('facebook-link', array( |      $socialMediaPanel->addChild('facebook-link', array( | ||
| Line 423: | Line 437: | ||
|          'label' => 'Like us on Facebook!', |          'label' => 'Like us on Facebook!', | ||
|          'order' => 1, |          'order' => 1, | ||
| − |          'icon' => 'fab fa-facebook-f', | + |          'icon' => 'fab fa-facebook-f fa-fw', | 
|      )); |      )); | ||
| − | + | ||
|      // Add a Twitter link to the panel after the Facebook link. |      // Add a Twitter link to the panel after the Facebook link. | ||
|      $socialMediaPanel->addChild('twitter-link', array( |      $socialMediaPanel->addChild('twitter-link', array( | ||
| Line 431: | Line 445: | ||
|          'label' => 'Follow us on Twitter!', |          'label' => 'Follow us on Twitter!', | ||
|          'order' => 2, |          'order' => 2, | ||
| − |          'icon' => 'fab fa-twitter | + |          'icon' => 'fab fa-twitter fa-fw', | 
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
|      )); |      )); | ||
| }); | }); | ||
| </syntaxhighlight> | </syntaxhighlight> | ||
| − | ==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=== | 
| − | [[Image:Custom panel body html highlighted.png|thumb| | + | [[Image:Custom panel body html highlighted.png|thumb|A Client Area panel displaying a custom special offer.]] 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 <tt>getFirstChild()</tt> and <tt>setBodyHtml()</tt> methods. To add panel with links to the sidebar we must: | This example uses the '''ClientAreaPrimarySidebar''' hook and the menu item’s <tt>getFirstChild()</tt> and <tt>setBodyHtml()</tt> methods. To add panel with links to the sidebar we must: | ||
| Line 459: | Line 465: | ||
| <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 471: | Line 476: | ||
| </a> | </a> | ||
| EOT; | EOT; | ||
| − | + | ||
| − | + |     if($primarySidebar->count() > 0) | |
| − | + |     { | |
| − | + | ||
| − | + |          // 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(); | ||
| + | |||
|      	$firstSidebar->setBodyHtml($specialOfferHtml); |      	$firstSidebar->setBodyHtml($specialOfferHtml); | ||
| + | |||
|      } |      } | ||
| }); | }); | ||
| + | |||
| </syntaxhighlight> | </syntaxhighlight> | ||
| − | ==Move the “Contact Us” link to the secondary navigation bar and add more contact options== | + | ===Move the “Contact Us” link to the secondary navigation bar and add more contact options=== | 
| − | [[Image:Custom navbar menu highlighted.png|thumb|The WHMCS  | + | [[Image:Custom navbar menu highlighted.png|thumb|The WHMCS Six Theme Client Area navbars with the Contact Us link moved to the secondary navigation bar with a custom menu.]] The graphic designer feels that our awesome hosting company’s WHMCS installation will more closely match the main site if the “Contact Us” link is moved to the right hand side of the installation’s navigation bars before the “Account” link. They also want a dropdown under the link with links specifically to email the sales team, call us, and provide a map to the company via Google Maps. | 
| − | This example requires manipulating more than one menu bar. To do that we’ll use the '''ClientAreaNavbars''' hook and the <tt>Menu</tt> class to retrieve the primary and secondary navigation bars.  | + | This example for the Six theme (but not Twenty-One) requires manipulating more than one menu bar. To do that we’ll use the '''ClientAreaNavbars''' hook and the <tt>Menu</tt> class to retrieve the primary and secondary navigation bars. We'll use the menu item's <tt>getChild()</tt>, <tt>removeChild()</tt>, <tt>addChild()</tt>, and <tt>moveToFront()</tt> methods and the static <tt>Menu::primaryNavbar()</tt> and <tt>Menu::secondaryNavbar()</tt> methods. Here’s what we’ll do: | 
| # Save the “Contact Us” link from the primary navigation bar. | # Save the “Contact Us” link from the primary navigation bar. | ||
| Line 518: | Line 528: | ||
|          'uri' => 'sales@my-awesome-company.com', |          'uri' => 'sales@my-awesome-company.com', | ||
|          'order' => 1, |          'order' => 1, | ||
| − |          'icon' => 'fa- | + |          'icon' => 'far fa-gem', | 
|      )); |      )); | ||
| Line 526: | Line 536: | ||
|          'uri' => 'tel:+18005551212', |          'uri' => 'tel:+18005551212', | ||
|          'order' => 2, |          'order' => 2, | ||
| − |          'icon' => 'fa-mobile', | + |          'icon' => 'fas fa-mobile-alt', | 
|      )); |      )); | ||
| Line 534: | Line 544: | ||
|          'uri' => 'https:\/\/maps.google.com/maps/place/some-map-data', |          'uri' => 'https:\/\/maps.google.com/maps/place/some-map-data', | ||
|          'order' => 3, |          'order' => 3, | ||
| − |          'icon' => 'fa-map-marker', | + |          'icon' => 'fas fa-map-marker-alt', | 
|      )); |      )); | ||
| Line 546: | Line 556: | ||
| </syntaxhighlight> | </syntaxhighlight> | ||
| − | ==Add support hours and a custom message to the sidebar on the submit ticket page== | + | ===Add support hours and a custom message to the sidebar on the submit ticket page=== | 
| [[Image:Custom support hours panel.png|thumb|The WHMCS 6 submit ticket page in responsive mode with support hours and a custom message in the primary sidebar.]] Your business has really taken off, and it's finally time to hire someone to help answer support tickets and sales inquiries. The new hire can't work 24 hours a day, 7 days a week, so it's time to implement official support hours. Your UX people feel that it's best to notify your users of these support hours on the submit ticket page, but only the submit ticket page. They'd like the hours at the top of the primary sidebar, so they're in plain view for the users. They'd also love it if we could notify the users that they can expect a reply from us soon or if they have to wait until the next business day. | [[Image:Custom support hours panel.png|thumb|The WHMCS 6 submit ticket page in responsive mode with support hours and a custom message in the primary sidebar.]] Your business has really taken off, and it's finally time to hire someone to help answer support tickets and sales inquiries. The new hire can't work 24 hours a day, 7 days a week, so it's time to implement official support hours. Your UX people feel that it's best to notify your users of these support hours on the submit ticket page, but only the submit ticket page. They'd like the hours at the top of the primary sidebar, so they're in plain view for the users. They'd also love it if we could notify the users that they can expect a reply from us soon or if they have to wait until the next business day. | ||
| Line 584: | Line 594: | ||
|              '<strong>Open</strong> 08:00-17:00 M-F', |              '<strong>Open</strong> 08:00-17:00 M-F', | ||
|              array( |              array( | ||
| − |                  'icon'  => 'fa-smile | + |                  'icon'  => 'far fa-smile', | 
|                  'order' => 1, |                  'order' => 1, | ||
|              ) |              ) | ||
| Line 591: | Line 601: | ||
|              '<strong>Closed</strong> Weekends', |              '<strong>Closed</strong> Weekends', | ||
|              array( |              array( | ||
| − |                  'icon'  => 'fa-frown | + |                  'icon'  => 'far fa-frown', | 
|                  'order' => 2, |                  'order' => 2, | ||
|              ) |              ) | ||
Latest revision as of 14:40, 7 March 2023
WHMCS includes a programmatic way to interact with the client area navigation and sidebars through hooks and modules.
- We added this feature in WHMCS 6.
- For copy-and-paste code examples for common customizations using this system, see Client Area Navigation Menus Cheatsheet and Client Area Sidebars Cheatsheet.
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
 
Custom Actions
In WHMCS 8.5 and later, you can use the CustomActions server module function to add items to the server details sidebar menu. The item will perform a set function and then redirect the user to a specified URL. For more information, see Working With Client Area Home Page Panels and our Developer Documentation.
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 Account > My Account or, prior to WHMCS 8.0, My Account. 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 | 
 | 
| orderFormView | cart.php | 
 | 
| Only the secondary sidebar is supported on cart.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 | 
 | 
 | 
| Profile | 
 | 
 | 
Examples
Add a social media panel to the end of the sidebar
You may want to better connect with your clients by using social media. The menu hooks in WHMCS 6.0 and later allow you to insert links to your company’s Facebook, Twitter, and other social media profiles directly into the interface. We recommend putting them at the end of the secondary sidebar so they’re rendered last and won’t interrupt your 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 you will need to create a panel at the end of the sidebar, retrieve the new panel, and add social media links to the panel. The Font Awesome library already has icons for all of these services.
To do this:
- Create the includes/hooks/socialMediaPanel.php file in your WHMCS installation.
- Enter the code below.
- Save the file.
- 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' => 'fas 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' => 'fab fa-facebook-f fa-fw',
    ));
 
    // 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' => 'fab fa-twitter fa-fw',
    ));
});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;
 
    if($primarySidebar->count() > 0)
    {
 
         // 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();
  
    	$firstSidebar->setBodyHtml($specialOfferHtml);
 
    }
});This example for the Six theme (but not Twenty-One) 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();
if (!is_null($primaryNavbar->getChild('Contact Us'))){
 
    // 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' => 'far fa-gem',
    ));
 
    // 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' => 'fas fa-mobile-alt',
    ));
 
    // 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' => 'fas fa-map-marker-alt',
    ));
 
    // 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'  => 'far fa-smile',
                'order' => 1,
            )
        );
        $supportHours->addChild(
            '<strong>Closed</strong> Weekends',
            array(
                'icon'  => 'far fa-frown',
                '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!"
        );
    });
}






