Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

contain-by-screen

Package Overview
Dependencies
Maintainers
1
Versions
12
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

contain-by-screen - npm Package Compare versions

Comparing version 1.0.3 to 1.0.4

circle.yml

78

js/index.js
'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

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc