contain-by-screen
Advanced tools
Comparing version 1.0.3 to 1.0.4
'use strict'; | ||
Object.defineProperty(exports, "__esModule", { | ||
value: true | ||
}); | ||
var _assign = require('babel-runtime/core-js/object/assign'); | ||
@@ -7,11 +11,12 @@ | ||
Object.defineProperty(exports, "__esModule", { | ||
value: true | ||
}); | ||
exports.default = containByScreen; | ||
var _lodash = require('lodash'); | ||
var _flatten = require('lodash/flatten'); | ||
var _lodash2 = _interopRequireDefault(_lodash); | ||
var _flatten2 = _interopRequireDefault(_flatten); | ||
var _uniq = require('lodash/uniq'); | ||
var _uniq2 = _interopRequireDefault(_uniq); | ||
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } | ||
@@ -38,37 +43,30 @@ | ||
var positions = options.position && options.forcePosition ? [options.position] : _lodash2.default.uniq([options.position].concat(['top', 'bottom', 'left', 'right']).filter(Boolean)); | ||
var hAligns = options.hAlign && options.forceHAlign ? [options.hAlign] : _lodash2.default.uniq([options.hAlign].concat(['center', 'left', 'right']).filter(Boolean)); | ||
var vAligns = options.vAlign && options.forceVAlign ? [options.vAlign] : _lodash2.default.uniq([options.vAlign].concat(['center', 'top', 'bottom']).filter(Boolean)); | ||
var positions = options.position && options.forcePosition ? [options.position] : (0, _uniq2.default)([options.position].filter(Boolean).concat(['top', 'bottom', 'left', 'right'])); | ||
var hAligns = options.hAlign && options.forceHAlign ? [options.hAlign] : (0, _uniq2.default)([options.hAlign].filter(Boolean).concat(['center', 'left', 'right'])); | ||
var vAligns = options.vAlign && options.forceVAlign ? [options.vAlign] : (0, _uniq2.default)([options.vAlign].filter(Boolean).concat(['center', 'top', 'bottom'])); | ||
var choiceAndCoord = _lodash2.default.chain(positions).map(function (position) { | ||
var allPossibleChoices = (0, _flatten2.default)(positions.map(function (position) { | ||
return position === 'top' || position === 'bottom' ? hAligns.map(function (hAlign) { | ||
return { position: position, hAlign: hAlign }; | ||
}) : [{ position: position, hAlign: 'center' }]; | ||
}).flatten().map(function (_ref) { | ||
var position = _ref.position; | ||
var hAlign = _ref.hAlign; | ||
return position === 'top' || position === 'bottom' ? [{ position: position, hAlign: hAlign, vAlign: 'center' }] : vAligns.map(function (vAlign) { | ||
return { position: position, hAlign: hAlign, vAlign: vAlign }; | ||
return { position: position, hAlign: hAlign, vAlign: 'center' }; | ||
}) : vAligns.map(function (vAlign) { | ||
return { position: position, hAlign: 'center', vAlign: vAlign }; | ||
}); | ||
}).flatten() | ||
// We've got an array of all sensible {position, hAlign, vAlign} combinations | ||
.map(function (_ref2) { | ||
var position = _ref2.position; | ||
var hAlign = _ref2.hAlign; | ||
var vAlign = _ref2.vAlign; | ||
return { | ||
choice: { position: position, hAlign: hAlign, vAlign: vAlign }, | ||
coord: positionAndAlign(elRect, anchorRect, position, hAlign, vAlign, buffers) | ||
}; | ||
}).filter(function (_ref3) { | ||
var choice = _ref3.choice; | ||
var _ref3$coord = _ref3.coord; | ||
var top = _ref3$coord.top; | ||
var left = _ref3$coord.left; | ||
return top - buffers.all - buffers.top >= 0 && left - buffers.all - buffers.left >= 0 && top + elRect.height + buffers.all + buffers.bottom <= window.innerHeight && left + elRect.width + buffers.all + buffers.right <= window.innerWidth; | ||
}).first().value(); | ||
})); | ||
var choiceAndCoord = null; | ||
for (var i = 0; i < allPossibleChoices.length; i++) { | ||
var choice = allPossibleChoices[i]; | ||
var coord = positionAndAlign(elRect, anchorRect, choice, buffers); | ||
var _top = coord.top; | ||
var _left = coord.left; | ||
if (_top - buffers.all - buffers.top >= 0 && _left - buffers.all - buffers.left >= 0 && _top + elRect.height + buffers.all + buffers.bottom <= window.innerHeight && _left + elRect.width + buffers.all + buffers.right <= window.innerWidth) { | ||
choiceAndCoord = { choice: choice, coord: coord }; | ||
break; | ||
} | ||
} | ||
// Fallback if we failed to find a position that fit on the screen. | ||
if (!choiceAndCoord) { | ||
var choice = { | ||
var _choice = { | ||
position: options.position || 'top', | ||
@@ -79,4 +77,4 @@ hAlign: options.hAlign || 'center', | ||
choiceAndCoord = { | ||
choice: choice, | ||
coord: positionAndAlign(elRect, anchorRect, choice.position, choice.hAlign, choice.vAlign, buffers) | ||
choice: _choice, | ||
coord: positionAndAlign(elRect, anchorRect, _choice, buffers) | ||
}; | ||
@@ -103,3 +101,7 @@ } | ||
function positionAndAlign(elRect, anchorRect, position, hAlign, vAlign, buffers) { | ||
function positionAndAlign(elRect, anchorRect, _ref, buffers) { | ||
var position = _ref.position; | ||
var hAlign = _ref.hAlign; | ||
var vAlign = _ref.vAlign; | ||
var top = 0, | ||
@@ -159,2 +161,2 @@ left = 0; | ||
module.exports = exports['default']; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/index.js"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/* @flow */\n\nimport _ from 'lodash';\nimport type {Position, HAlign, VAlign, Options} from './index.js.flow';\n\ntype Rect = { // Similar to ClientRect, but not a class\n  top: number;\n  bottom: number;\n  height: number;\n  left: number;\n  right: number;\n  width: number;\n};\n\nexport default function containByScreen(element: HTMLElement, anchorPoint: HTMLElement, options: Options):\n{position: Position, hAlign: HAlign, vAlign: VAlign} {\n  if (process.env.NODE_ENV !== 'production' && window.getComputedStyle) {\n    const style = window.getComputedStyle(element);\n    if (style.position !== 'fixed') {\n      console.error('containByScreen only works on fixed position elements', element);\n    }\n  }\n\n  const elRect: Rect = getBoundingClientRect(element);\n  const anchorRect: Rect = getBoundingClientRect(anchorPoint);\n\n  const buffers = {\n    all: options.buffer || 0,\n    top: options.topBuffer || 0,\n    bottom: options.bottomBuffer || 0,\n    left: options.leftBuffer || 0,\n    right: options.rightBuffer || 0\n  };\n\n  const positions: Position[] = options.position && options.forcePosition ?\n    [options.position] :\n    _.uniq([options.position].concat(['top','bottom','left','right']).filter(Boolean));\n  const hAligns: HAlign[] = options.hAlign && options.forceHAlign ?\n    [options.hAlign] :\n    _.uniq([options.hAlign].concat(['center','left','right']).filter(Boolean));\n  const vAligns: VAlign[] = options.vAlign && options.forceVAlign ?\n    [options.vAlign] :\n    _.uniq([options.vAlign].concat(['center','top','bottom']).filter(Boolean));\n\n  let choiceAndCoord = _.chain(positions)\n    .map(position =>\n      position === 'top' || position === 'bottom' ?\n        hAligns.map(hAlign => ({position, hAlign})) :\n        [{position, hAlign: 'center'}]\n    )\n    .flatten()\n    .map(({position, hAlign}) =>\n      position === 'top' || position === 'bottom' ?\n        [{position, hAlign, vAlign: 'center'}] :\n        vAligns.map(vAlign => ({position, hAlign, vAlign}))\n    )\n    .flatten()\n    // We've got an array of all sensible {position, hAlign, vAlign} combinations\n    .map(({position, hAlign, vAlign}) => ({\n      choice: {position, hAlign, vAlign},\n      coord: positionAndAlign(elRect, anchorRect, position, hAlign, vAlign, buffers)\n    }))\n    .filter(({choice, coord: {top, left}}) =>\n      top-buffers.all-buffers.top >= 0 &&\n      left-buffers.all-buffers.left >= 0 &&\n      top+elRect.height+buffers.all+buffers.bottom <= window.innerHeight &&\n      left+elRect.width+buffers.all+buffers.right <= window.innerWidth\n    )\n    .first()\n    .value();\n\n  // Fallback if we failed to find a position that fit on the screen.\n  if (!choiceAndCoord) {\n    const choice = {\n      position: options.position||'top',\n      hAlign: options.hAlign||'center',\n      vAlign: options.vAlign||'center'\n    };\n    choiceAndCoord = {\n      choice,\n      coord: positionAndAlign(elRect, anchorRect,\n        choice.position, choice.hAlign, choice.vAlign, buffers)\n    };\n  }\n\n  element.style.top = `${choiceAndCoord.coord.top}px`;\n  element.style.left = `${choiceAndCoord.coord.left}px`;\n\n  return choiceAndCoord.choice;\n}\n\nfunction getBoundingClientRect(el: Element): Rect {\n  let rect = el.getBoundingClientRect();\n  if (!('width' in rect)) {\n    // IE <9 support\n    rect = Object.assign(({\n      width: rect.right-rect.left,\n      height: rect.bottom-rect.top\n    }: Object), rect);\n  }\n  return rect;\n}\n\nfunction positionAndAlign(elRect: Rect, anchorRect: Rect, position: Position, hAlign: HAlign, vAlign: VAlign, buffers): {top: number, left: number} {\n  let top=0, left=0;\n  if (position === 'top' || position === 'bottom') {\n    switch (position) {\n      case 'top':\n        top = Math.floor(anchorRect.top - elRect.height - buffers.all - buffers.bottom);\n        break;\n      case 'bottom':\n        top = Math.ceil(anchorRect.bottom + buffers.all + buffers.top);\n        break;\n      default: throw new Error(\"Should not happen\");\n    }\n    switch (hAlign) {\n      case 'center':\n        left = Math.round((anchorRect.left + anchorRect.right - elRect.width)/2);\n        break;\n      case 'left':\n        left = Math.round(anchorRect.left);\n        break;\n      case 'right':\n        left = Math.round(anchorRect.right - elRect.width);\n        break;\n      default: throw new Error(\"Should not happen\");\n    }\n  } else {\n    switch (position) {\n      case 'left':\n        left = Math.floor(anchorRect.left - elRect.width - buffers.all - buffers.right);\n        break;\n      case 'right':\n        left = Math.ceil(anchorRect.right + buffers.all + buffers.left);\n        break;\n      default: throw new Error(\"Should not happen\");\n    }\n    switch (vAlign) {\n      case 'center':\n        top = Math.round((anchorRect.top + anchorRect.bottom - elRect.height)/2);\n        break;\n      case 'top':\n        top = Math.round(anchorRect.top);\n        break;\n      case 'bottom':\n        top = Math.round(anchorRect.bottom - elRect.height);\n        break;\n      default: throw new Error(\"Should not happen\");\n    }\n  }\n  return {top, left};\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/index.js"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/* @flow */\n\nimport flatten from 'lodash/flatten';\nimport uniq from 'lodash/uniq';\n\nexport type Position = 'top'|'bottom'|'left'|'right';\nexport type HAlign = 'center'|'left'|'right';\nexport type VAlign = 'center'|'top'|'bottom';\nexport type Choice = {\n  position: Position;\n  hAlign: HAlign;\n  vAlign: VAlign;\n};\n\nexport type Options = {\n  position?: ?Position;\n  forcePosition?: ?boolean;\n  hAlign?: ?HAlign;\n  forceHAlign?: ?boolean;\n  vAlign?: ?VAlign;\n  forceVAlign?: ?boolean;\n  buffer?: ?number;\n  topBuffer?: ?number;\n  bottomBuffer?: ?number;\n  leftBuffer?: ?number;\n  rightBuffer?: ?number;\n};\n\ntype Rect = { // Similar to ClientRect, but not a class\n  top: number;\n  bottom: number;\n  height: number;\n  left: number;\n  right: number;\n  width: number;\n};\n\nexport default function containByScreen(element: HTMLElement, anchorPoint: HTMLElement, options: Options):\nChoice {\n  if (process.env.NODE_ENV !== 'production' && window.getComputedStyle) {\n    const style = window.getComputedStyle(element);\n    if (style.position !== 'fixed') {\n      console.error('containByScreen only works on fixed position elements', element);\n    }\n  }\n\n  const elRect: Rect = getBoundingClientRect(element);\n  const anchorRect: Rect = getBoundingClientRect(anchorPoint);\n\n  const buffers = {\n    all: options.buffer || 0,\n    top: options.topBuffer || 0,\n    bottom: options.bottomBuffer || 0,\n    left: options.leftBuffer || 0,\n    right: options.rightBuffer || 0\n  };\n\n  const positions: Position[] = options.position && options.forcePosition ?\n    [options.position] :\n    uniq([options.position].filter(Boolean).concat(['top','bottom','left','right']));\n  const hAligns: HAlign[] = options.hAlign && options.forceHAlign ?\n    [options.hAlign] :\n    uniq([options.hAlign].filter(Boolean).concat(['center','left','right']));\n  const vAligns: VAlign[] = options.vAlign && options.forceVAlign ?\n    [options.vAlign] :\n    uniq([options.vAlign].filter(Boolean).concat(['center','top','bottom']));\n\n  const allPossibleChoices = flatten(positions.map(position =>\n    (position === 'top' || position === 'bottom') ?\n      hAligns.map(hAlign => ({position, hAlign, vAlign: 'center'})) :\n      vAligns.map(vAlign => ({position, hAlign: 'center', vAlign}))\n  ));\n\n  let choiceAndCoord = null;\n  for (let i=0; i < allPossibleChoices.length; i++) {\n    const choice = allPossibleChoices[i];\n    const coord = positionAndAlign(elRect, anchorRect, choice, buffers);\n    const {top, left} = coord;\n    if (\n      top-buffers.all-buffers.top >= 0 &&\n      left-buffers.all-buffers.left >= 0 &&\n      top+elRect.height+buffers.all+buffers.bottom <= window.innerHeight &&\n      left+elRect.width+buffers.all+buffers.right <= window.innerWidth\n    ) {\n      choiceAndCoord = {choice, coord};\n      break;\n    }\n  }\n\n  // Fallback if we failed to find a position that fit on the screen.\n  if (!choiceAndCoord) {\n    const choice = {\n      position: options.position||'top',\n      hAlign: options.hAlign||'center',\n      vAlign: options.vAlign||'center'\n    };\n    choiceAndCoord = {\n      choice,\n      coord: positionAndAlign(elRect, anchorRect, choice, buffers)\n    };\n  }\n\n  element.style.top = `${choiceAndCoord.coord.top}px`;\n  element.style.left = `${choiceAndCoord.coord.left}px`;\n\n  return choiceAndCoord.choice;\n}\n\nfunction getBoundingClientRect(el: Element): Rect {\n  let rect = el.getBoundingClientRect();\n  if (!('width' in rect)) {\n    // IE <9 support\n    rect = Object.assign(({\n      width: rect.right-rect.left,\n      height: rect.bottom-rect.top\n    }: Object), rect);\n  }\n  return rect;\n}\n\nfunction positionAndAlign(elRect: Rect, anchorRect: Rect, {position, hAlign, vAlign}: Choice, buffers): {top: number, left: number} {\n  let top=0, left=0;\n  if (position === 'top' || position === 'bottom') {\n    switch (position) {\n      case 'top':\n        top = Math.floor(anchorRect.top - elRect.height - buffers.all - buffers.bottom);\n        break;\n      case 'bottom':\n        top = Math.ceil(anchorRect.bottom + buffers.all + buffers.top);\n        break;\n      default: throw new Error(\"Should not happen\");\n    }\n    switch (hAlign) {\n      case 'center':\n        left = Math.round((anchorRect.left + anchorRect.right - elRect.width)/2);\n        break;\n      case 'left':\n        left = Math.round(anchorRect.left);\n        break;\n      case 'right':\n        left = Math.round(anchorRect.right - elRect.width);\n        break;\n      default: throw new Error(\"Should not happen\");\n    }\n  } else {\n    switch (position) {\n      case 'left':\n        left = Math.floor(anchorRect.left - elRect.width - buffers.all - buffers.right);\n        break;\n      case 'right':\n        left = Math.ceil(anchorRect.right + buffers.all + buffers.left);\n        break;\n      default: throw new Error(\"Should not happen\");\n    }\n    switch (vAlign) {\n      case 'center':\n        top = Math.round((anchorRect.top + anchorRect.bottom - elRect.height)/2);\n        break;\n      case 'top':\n        top = Math.round(anchorRect.top);\n        break;\n      case 'bottom':\n        top = Math.round(anchorRect.bottom - elRect.height);\n        break;\n      default: throw new Error(\"Should not happen\");\n    }\n  }\n  return {top, left};\n}\n"]} |
{ | ||
"name": "contain-by-screen", | ||
"version": "1.0.3", | ||
"version": "1.0.4", | ||
"description": "Position a dropdown element near a button in a way that fits on the screen.", | ||
"main": "js/index.js", | ||
"scripts": { | ||
"prepublish": "babel -s inline -d js/ src/ && cp -v src/index.js.flow js/", | ||
"prepublish": "babel -s inline -d js/ src/ && flow-copy-source -v src js", | ||
"test": "mocha" | ||
@@ -21,2 +21,7 @@ }, | ||
"license": "MIT", | ||
"browserify": { | ||
"transform": [ | ||
"envify" | ||
] | ||
}, | ||
"bugs": { | ||
@@ -28,3 +33,3 @@ "url": "https://github.com/AgentME/contain-by-screen/issues" | ||
"babel-cli": "^6.4.0", | ||
"babel-plugin-add-module-exports": "^0.1.3-alpha", | ||
"babel-plugin-add-module-exports": "^0.1.2", | ||
"babel-plugin-transform-class-properties": "^6.4.0", | ||
@@ -35,2 +40,3 @@ "babel-plugin-transform-flow-strip-types": "^6.4.0", | ||
"babel-register": "^6.4.3", | ||
"flow-copy-source": "^1.0.1", | ||
"mocha": "^2.3.4" | ||
@@ -40,4 +46,5 @@ }, | ||
"babel-runtime": "^6.3.19", | ||
"lodash": "^3.10.1" | ||
"envify": "^3.4.0", | ||
"lodash": "^4.6.1" | ||
} | ||
} |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
30241
10
3
9
143
+ Addedenvify@^3.4.0
+ Addedacorn@5.7.4(transitive)
+ Addedamdefine@1.0.1(transitive)
+ Addedast-types@0.9.6(transitive)
+ Addedbalanced-match@1.0.2(transitive)
+ Addedbase62@1.2.8(transitive)
+ Addedbrace-expansion@1.1.11(transitive)
+ Addedcommander@2.20.3(transitive)
+ Addedcommoner@0.10.8(transitive)
+ Addedconcat-map@0.0.1(transitive)
+ Addeddefined@1.0.1(transitive)
+ Addeddetective@4.7.1(transitive)
+ Addedenvify@3.4.1(transitive)
+ Addedesprima@3.1.3(transitive)
+ Addedesprima-fb@15001.1.0-dev-harmony-fb(transitive)
+ Addedglob@5.0.15(transitive)
+ Addedgraceful-fs@4.2.11(transitive)
+ Addediconv-lite@0.4.24(transitive)
+ Addedinflight@1.0.6(transitive)
+ Addedinherits@2.0.4(transitive)
+ Addedjstransform@11.0.3(transitive)
+ Addedlodash@4.17.21(transitive)
+ Addedminimatch@3.1.2(transitive)
+ Addedminimist@1.2.8(transitive)
+ Addedmkdirp@0.5.6(transitive)
+ Addedobject-assign@2.1.1(transitive)
+ Addedonce@1.4.0(transitive)
+ Addedpath-is-absolute@1.0.1(transitive)
+ Addedprivate@0.1.8(transitive)
+ Addedq@1.5.1(transitive)
+ Addedrecast@0.11.23(transitive)
+ Addedsafer-buffer@2.1.2(transitive)
+ Addedsource-map@0.4.40.5.7(transitive)
+ Addedthrough@2.3.8(transitive)
+ Addedwrappy@1.0.2(transitive)
- Removedlodash@3.10.1(transitive)
Updatedlodash@^4.6.1