1. import React, { Component } from 'react'
  2. import PropTypes from 'prop-types'
  3. import Draggable from 'react-draggable'
  4. import { DragHandle } from 'DragHandle'
  5. import { ArrowBox, Container } from './styled'
  6. /**
  7. * @component
  8. * @category Popup
  9. * @since 0.2.0
  10. */
  11. class PopupBase extends Component {
  12. state = {
  13. dragged: false,
  14. pinnedPixel: this.props.pixel,
  15. transparent: false
  16. }
  17. onStart = () => {
  18. const { arrow, onPopupDragStart, pixel, transparentDuringDrag } = this.props
  19. // this hard sets the arrow direction at the time of initial drag
  20. const lastArrow = !this.state.dragged ? arrow : this.state.lastArrow
  21. const pinnedPixel = !this.state.dragged ? pixel : this.state.pinnedPixel
  22. this.setState({
  23. dragged: true,
  24. lastArrow,
  25. pinnedPixel,
  26. transparent: transparentDuringDrag
  27. })
  28. onPopupDragStart()
  29. }
  30. onStop = e => {
  31. const { pinnedPixel } = this.state
  32. this.setState({
  33. transparent: false
  34. })
  35. this.props.onPopupDragEnd({ ...e, pinnedPixel })
  36. }
  37. handleDrag = e => {
  38. this.props.onPopupDrag(e)
  39. }
  40. render () {
  41. const { arrow: arrowProp, children, draggable, height, inline, pixel: pixelProp, show, width } = this.props
  42. const { dragged, lastArrow, pinnedPixel, transparent } = this.state
  43. const arrowTranslator = {
  44. top: 'bottom',
  45. bottom: 'top',
  46. left: 'right',
  47. right: 'left',
  48. none: 'none'
  49. }
  50. const pixel = dragged ? pinnedPixel : pixelProp
  51. const arrow = dragged ? lastArrow : arrowProp
  52. return (
  53. <Draggable
  54. axis='both'
  55. bounds='parent'
  56. handle={'.popupHandleTag'}
  57. onDrag={this.handleDrag}
  58. onStart={this.onStart}
  59. onStop={this.onStop}>
  60. <Container
  61. arrow={arrow}
  62. height={height}
  63. inline={inline}
  64. pixel={pixel}
  65. show={show}
  66. transparent={transparent}
  67. width={width}>
  68. {draggable ? <DragHandle className='popupHandleTag' /> : null}
  69. {!dragged && <ArrowBox position={arrowTranslator[arrow]} />}
  70. {children}
  71. </Container>
  72. </Draggable>
  73. )
  74. }
  75. }
  76. PopupBase.defaultProps = {
  77. arrow: 'none',
  78. draggable: true,
  79. height: 'auto',
  80. inline: false,
  81. onPopupDrag: () => {},
  82. onPopupDragEnd: () => {},
  83. onPopupDragStart: () => {},
  84. pixel: [0, 0],
  85. show: false,
  86. transparentDuringDrag: true,
  87. width: 280
  88. }
  89. PopupBase.propTypes = {
  90. /** The position of the popup's arrow (`top`, `right`, `bottom`, `left` or `none`) */
  91. arrow: PropTypes.string,
  92. /** The content of the popup */
  93. children: PropTypes.oneOfType([
  94. PropTypes.arrayOf(PropTypes.node),
  95. PropTypes.node
  96. ]),
  97. draggable: PropTypes.bool,
  98. /** The height of the popup */
  99. height: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
  100. /** Render the component inline (without absolute positioning) */
  101. inline: PropTypes.bool,
  102. /** Callback fired during drags when draggable is true */
  103. onPopupDrag: PropTypes.func,
  104. /** Callback fired after a drag when draggable is true */
  105. onPopupDragEnd: PropTypes.func,
  106. /** Callback fired at the beginning of a drag when draggable is true */
  107. onPopupDragStart: PropTypes.func,
  108. /** The pixel coordinate where the popup should render */
  109. pixel: PropTypes.array,
  110. /** Show/hide the popup */
  111. show: PropTypes.bool,
  112. /** Set PopupBase to see-through during a drag when draggable is true */
  113. transparentDuringDrag: PropTypes.bool,
  114. /** The width of the popup */
  115. width: PropTypes.number
  116. }
  117. export default PopupBase