CustomNavigation.jsx 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. import React, { useEffect, useState, useRef } from 'react';
  2. import PropTypes from 'prop-types';
  3. import {
  4. Nav, NavItem, NavLink, TabContent, TabPane,
  5. } from 'reactstrap';
  6. const CustomNavigation = (props) => {
  7. const [activeTab, setActiveTab] = useState(Object.keys(props.navTabMapping)[0]);
  8. const [sliderWidth, setSliderWidth] = useState(0);
  9. const [sliderMarginLeft, setSliderMarginLeft] = useState(0);
  10. const navContainer = useRef();
  11. const navTabs = {};
  12. Object.keys(props.navTabMapping).forEach((key) => {
  13. navTabs[key] = React.createRef();
  14. });
  15. function switchActiveTab(activeTab) {
  16. setActiveTab(activeTab);
  17. }
  18. // Might make this dynamic for px, %, pt, em
  19. function getPercentage(min, max) {
  20. return min / max * 100;
  21. }
  22. function registerNavLink(key, elm) {
  23. if (elm != null) {
  24. navTabs[key] = elm;
  25. }
  26. }
  27. useEffect(() => {
  28. const navBar = navContainer;
  29. if (activeTab === '') {
  30. return;
  31. }
  32. if (navBar == null || navTabs == null) {
  33. return;
  34. }
  35. let tempML = 0;
  36. const styles = Object.entries(navTabs).map((el) => {
  37. const width = getPercentage(el[1].offsetWidth, navBar.current.offsetWidth);
  38. const marginLeft = tempML;
  39. tempML += width;
  40. return { width, marginLeft };
  41. });
  42. const { width, marginLeft } = styles[props.navTabMapping[activeTab].index];
  43. setSliderWidth(width);
  44. setSliderMarginLeft(marginLeft);
  45. }, [activeTab]);
  46. return (
  47. <React.Fragment>
  48. <div ref={navContainer}>
  49. <Nav className="nav-title grw-custom-navbar" id="grw-custom-navbar">
  50. {Object.entries(props.navTabMapping).map(([key, value]) => {
  51. return (
  52. <NavItem
  53. key={key}
  54. type="button"
  55. className={`p-0 grw-custom-navtab ${activeTab === key && 'active'}}`}
  56. >
  57. <NavLink key={key} innerRef={elm => registerNavLink(key, elm)} onClick={() => { switchActiveTab(key) }}>
  58. {value.icon}
  59. {value.i18n}
  60. </NavLink>
  61. </NavItem>
  62. );
  63. })}
  64. </Nav>
  65. <hr className="my-0 grw-nav-slide-hr border-none" style={{ width: `${sliderWidth}%`, marginLeft: `${sliderMarginLeft}%` }} />
  66. <TabContent activeTab={activeTab} className="p-4">
  67. {Object.entries(props.navTabMapping).map(([key, value]) => {
  68. return (
  69. <TabPane key={key} tabId={key}>
  70. {value.tabContent}
  71. </TabPane>
  72. );
  73. })}
  74. </TabContent>
  75. </div>
  76. </React.Fragment>
  77. );
  78. };
  79. CustomNavigation.propTypes = {
  80. navTabMapping: PropTypes.object,
  81. };
  82. export default CustomNavigation;