contextual_sidebar.js 2.8 KB
Newer Older
1
import $ from 'jquery';
2 3
import Cookies from 'js-cookie';
import _ from 'underscore';
4
import bp from './breakpoints';
5
import { parseBoolean } from '~/lib/utils/common_utils';
6

A
Annabel Dunstone Gray 已提交
7
export default class ContextualSidebar {
8 9
  constructor() {
    this.initDomElements();
10
    this.render();
11 12 13
  }

  initDomElements() {
14
    this.$page = $('.layout-page');
15
    this.$sidebar = $('.nav-sidebar');
16 17 18

    if (!this.$sidebar.length) return;

19
    this.$innerScroll = $('.nav-sidebar-inner-scroll', this.$sidebar);
20
    this.$overlay = $('.mobile-overlay');
21 22
    this.$openSidebar = $('.toggle-mobile-nav');
    this.$closeSidebar = $('.close-nav-button');
23
    this.$sidebarToggle = $('.js-toggle-sidebar');
24 25 26
  }

  bindEvents() {
27 28
    if (!this.$sidebar.length) return;

M
Mike Greiling 已提交
29 30 31 32 33
    document.addEventListener('click', e => {
      if (
        !e.target.closest('.nav-sidebar') &&
        (bp.getBreakpointSize() === 'sm' || bp.getBreakpointSize() === 'md')
      ) {
34
        this.toggleCollapsedSidebar(true, true);
35 36
      }
    });
37 38 39
    this.$openSidebar.on('click', () => this.toggleSidebarNav(true));
    this.$closeSidebar.on('click', () => this.toggleSidebarNav(false));
    this.$overlay.on('click', () => this.toggleSidebarNav(false));
40
    this.$sidebarToggle.on('click', () => {
41
      const value = !this.$sidebar.hasClass('sidebar-collapsed-desktop');
42
      this.toggleCollapsedSidebar(value, true);
43 44 45 46 47 48 49 50 51 52
    });

    $(window).on('resize', () => _.debounce(this.render(), 100));
  }

  static setCollapsedCookie(value) {
    if (bp.getBreakpointSize() !== 'lg') {
      return;
    }
    Cookies.set('sidebar_collapsed', value, { expires: 365 * 10 });
53 54 55
  }

  toggleSidebarNav(show) {
56
    this.$sidebar.toggleClass('sidebar-expanded-mobile', show);
57
    this.$overlay.toggleClass('mobile-nav-open', show);
58
    this.$sidebar.removeClass('sidebar-collapsed-desktop');
59 60
  }

61
  toggleCollapsedSidebar(collapsed, saveCookie) {
62 63
    const breakpoint = bp.getBreakpointSize();

64
    if (this.$sidebar.length) {
65
      this.$sidebar.toggleClass('sidebar-collapsed-desktop', collapsed);
66
      this.$page.toggleClass('page-with-icon-sidebar', breakpoint === 'sm' ? true : collapsed);
67
    }
68

69 70 71 72 73
    if (saveCookie) {
      ContextualSidebar.setCollapsedCookie(collapsed);
    }

    requestIdleCallback(this.toggleSidebarOverflow);
74 75 76 77 78 79 80 81
  }

  toggleSidebarOverflow() {
    if (this.$innerScroll.prop('scrollHeight') > this.$innerScroll.prop('offsetHeight')) {
      this.$innerScroll.css('overflow-y', 'scroll');
    } else {
      this.$innerScroll.css('overflow-y', '');
    }
82 83 84
  }

  render() {
85 86
    if (!this.$sidebar.length) return;

87 88 89
    const breakpoint = bp.getBreakpointSize();

    if (breakpoint === 'sm' || breakpoint === 'md') {
90
      this.toggleCollapsedSidebar(true, false);
91
    } else if (breakpoint === 'lg') {
92
      const collapse = parseBoolean(Cookies.get('sidebar_collapsed'));
93
      this.toggleCollapsedSidebar(collapse, false);
94
    }
95 96
  }
}