fly_out_nav.js 4.87 KB
Newer Older
1
import bp from './breakpoints';
2

3 4 5 6 7 8 9
const HIDE_INTERVAL_TIMEOUT = 300;
const IS_OVER_CLASS = 'is-over';
const IS_ABOVE_CLASS = 'is-above';
const IS_SHOWING_FLY_OUT_CLASS = 'is-showing-fly-out';
let currentOpenMenu = null;
let menuCornerLocs;
let timeoutId;
10
let sidebar;
Phil Hughes's avatar
Phil Hughes committed
11

12
export const mousePos = [];
13

14
export const setSidebar = (el) => { sidebar = el; };
15 16 17
export const setOpenMenu = (menu = null) => { currentOpenMenu = menu; };

export const slope = (a, b) => (b.y - a.y) / (b.x - a.x);
18

19 20 21
let headerHeight = 50;

export const getHeaderHeight = () => headerHeight;
22

23 24 25
export const canShowActiveSubItems = (el) => {
  if (el.classList.contains('active') && (sidebar && !sidebar.classList.contains('sidebar-icons-only'))) {
    return false;
26 27 28 29
  }

  return true;
};
30

31
export const canShowSubItems = () => bp.getBreakpointSize() === 'sm' || bp.getBreakpointSize() === 'md' || bp.getBreakpointSize() === 'lg';
32

33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51
export const getHideSubItemsInterval = () => {
  if (!currentOpenMenu) return 0;

  const currentMousePos = mousePos[mousePos.length - 1];
  const prevMousePos = mousePos[0];
  const currentMousePosY = currentMousePos.y;
  const [menuTop, menuBottom] = menuCornerLocs;

  if (currentMousePosY < menuTop.y ||
    currentMousePosY > menuBottom.y) return 0;

  if (slope(prevMousePos, menuBottom) < slope(currentMousePos, menuBottom) &&
    slope(prevMousePos, menuTop) > slope(currentMousePos, menuTop)) {
    return HIDE_INTERVAL_TIMEOUT;
  }

  return 0;
};

52 53 54 55
export const calculateTop = (boundingRect, outerHeight) => {
  const windowHeight = window.innerHeight;
  const bottomOverflow = windowHeight - (boundingRect.top + outerHeight);

56 57
  return bottomOverflow < 0 ? (boundingRect.top - outerHeight) + boundingRect.height :
    boundingRect.top;
58 59
};

60 61
export const hideMenu = (el) => {
  if (!el) return;
62

63
  const parentEl = el.parentNode;
64

65 66 67 68 69 70 71 72
  el.style.display = ''; // eslint-disable-line no-param-reassign
  el.style.transform = ''; // eslint-disable-line no-param-reassign
  el.classList.remove(IS_ABOVE_CLASS);
  parentEl.classList.remove(IS_OVER_CLASS);
  parentEl.classList.remove(IS_SHOWING_FLY_OUT_CLASS);

  setOpenMenu();
};
73

74
export const moveSubItemsToPosition = (el, subItems) => {
75
  const boundingRect = el.getBoundingClientRect();
Phil Hughes's avatar
Phil Hughes committed
76
  const top = calculateTop(boundingRect, subItems.offsetHeight);
77 78
  const isAbove = top < boundingRect.top;

79
  subItems.classList.add('fly-out-list');
80
  subItems.style.transform = `translate3d(0, ${Math.floor(top) - headerHeight}px, 0)`; // eslint-disable-line no-param-reassign
81 82 83 84 85 86 87 88 89 90 91 92 93

  const subItemsRect = subItems.getBoundingClientRect();

  menuCornerLocs = [
    {
      x: subItemsRect.left, // left position of the sub items
      y: subItemsRect.top, // top position of the sub items
    },
    {
      x: subItemsRect.left, // left position of the sub items
      y: subItemsRect.top + subItemsRect.height, // bottom position of the sub items
    },
  ];
94 95

  if (isAbove) {
96
    subItems.classList.add(IS_ABOVE_CLASS);
97 98 99
  }
};

100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
export const showSubLevelItems = (el) => {
  const subItems = el.querySelector('.sidebar-sub-level-items');

  if (!canShowSubItems() || !canShowActiveSubItems(el)) return;

  el.classList.add(IS_OVER_CLASS);

  if (!subItems) return;

  subItems.style.display = 'block';
  el.classList.add(IS_SHOWING_FLY_OUT_CLASS);

  setOpenMenu(subItems);
  moveSubItemsToPosition(el, subItems);
};

export const mouseEnterTopItems = (el) => {
  clearTimeout(timeoutId);

  timeoutId = setTimeout(() => {
    if (currentOpenMenu) hideMenu(currentOpenMenu);

    showSubLevelItems(el);
  }, getHideSubItemsInterval());
};

export const mouseLeaveTopItem = (el) => {
127 128
  const subItems = el.querySelector('.sidebar-sub-level-items');

129 130
  if (!canShowSubItems() || !canShowActiveSubItems(el) ||
    (subItems && subItems === currentOpenMenu)) return;
131

132 133 134 135 136 137 138 139 140 141
  el.classList.remove(IS_OVER_CLASS);
};

export const documentMouseMove = (e) => {
  mousePos.push({
    x: e.clientX,
    y: e.clientY,
  });

  if (mousePos.length > 6) mousePos.shift();
142 143
};

144
export default () => {
145
  sidebar = document.querySelector('.nav-sidebar');
146

147
  if (!sidebar) return;
148

149
  const items = [...sidebar.querySelectorAll('.sidebar-top-level-items > li')];
Phil Hughes's avatar
Phil Hughes committed
150

151
  sidebar.querySelector('.sidebar-top-level-items').addEventListener('mouseleave', () => {
152 153 154 155 156 157 158
    clearTimeout(timeoutId);

    timeoutId = setTimeout(() => {
      if (currentOpenMenu) hideMenu(currentOpenMenu);
    }, getHideSubItemsInterval());
  });

159 160
  headerHeight = document.querySelector('.nav-sidebar').offsetTop;

161 162 163 164 165 166 167 168 169 170 171 172 173 174 175
  items.forEach((el) => {
    const subItems = el.querySelector('.sidebar-sub-level-items');

    if (subItems) {
      subItems.addEventListener('mouseleave', () => {
        clearTimeout(timeoutId);
        hideMenu(currentOpenMenu);
      });
    }

    el.addEventListener('mouseenter', e => mouseEnterTopItems(e.currentTarget));
    el.addEventListener('mouseleave', e => mouseLeaveTopItem(e.currentTarget));
  });

  document.addEventListener('mousemove', documentMouseMove);
176
};