popover.js 2.4 KB
Newer Older
K
Kamran Ahmed 已提交
1
import Element from './element';
2
import { CLASS_POPOVER_TIP, ID_POPOVER, OVERLAY_PADDING, POPOVER_HTML } from './constants';
K
Kamran Ahmed 已提交
3

4 5 6 7 8
/**
 * Popover that is displayed on top of the highlighted element
 */
export default class Popover extends Element {
  constructor(options = {
9
    padding: OVERLAY_PADDING,
10 11 12 13 14 15 16
  }, window, document) {
    super();

    this.options = options;
    this.window = window;
    this.document = document;

17 18
    this.node = this.preparePopover();
    this.hide();
19 20
  }

21 22 23 24 25 26 27 28
  preparePopover() {
    let popover = this.document.getElementById(ID_POPOVER);
    if (popover) {
      return popover;
    }

    popover = Popover.createFromString(POPOVER_HTML);
    document.body.appendChild(popover);
29 30 31 32

    return popover;
  }

33 34 35 36 37 38 39 40 41 42 43 44 45
  /**
   * Turn a string into a node
   * @param  {String} htmlString to convert
   * @return {Node}   Converted node element
   */
  static createFromString(htmlString) {
    const div = document.createElement('div');
    div.innerHTML = htmlString.trim();

    // Change this to div.childNodes to support multiple top-level nodes
    return div.firstChild;
  }

46 47 48 49 50 51 52 53 54 55
  getHeight() {
    return Math.max(this.node.scrollHeight, this.node.offsetHeight);
  }

  hide() {
    this.node.style.display = 'none';
  }

  reset() {
    this.node.style.display = 'block';
56 57
    this.node.style.left = '0';
    this.node.style.top = '0';
58 59 60 61 62
    this.node.style.bottom = '';
    this.node.style.right = '';

    // Remove the positional classes from tip
    this.node
K
Kamran Ahmed 已提交
63 64
      .querySelector(`.${CLASS_POPOVER_TIP}`)
      .className = CLASS_POPOVER_TIP;
65 66 67 68 69 70
  }

  show(position) {
    this.reset();

    const pageHeight = this.getFullPageSize().height;
71 72

    const popoverTip = this.node.querySelector(`.${CLASS_POPOVER_TIP}`);
73
    const popoverMargin = this.options.padding + 10;
74
    const popoverHeight = this.getHeight();
75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90

    this.node.style.left = `${position.left - this.options.padding}px`;

    // Calculate different dimensions after attaching popover
    const pageHeightAfterPopOver = position.bottom + popoverHeight + popoverMargin;

    // If adding popover would go out of the window height, then show it to the top
    if (pageHeightAfterPopOver >= pageHeight) {
      this.node.style.top = `${position.top - popoverHeight - popoverMargin}px`;
      popoverTip.classList.add('bottom');
    } else {
      this.node.style.top = `${position.bottom + popoverMargin}px`;
      popoverTip.classList.add('top');
    }
  }
}