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.4 to 1.1.0

.eslintignore

18

flow-libs/mocha.js

@@ -1,10 +0,8 @@

// jshint ignore:start
declare function describe(name: string, fn: (done: (err: any) => void) => ?Promise): void;
declare function xdescribe(name: string, fn: (done: (err: any) => void) => ?Promise): void;
declare function before(fn: (done: (err: any) => void) => ?Promise): void;
declare function beforeEach(fn: (done: (err: any) => void) => ?Promise): void;
declare function after(fn: (done: (err: any) => void) => ?Promise): void;
declare function afterEach(fn: (done: (err: any) => void) => ?Promise): void;
declare function it(name: string, fn: (done: (err: any) => void) => ?Promise): void;
declare function xit(name: string, fn: (done: (err: any) => void) => ?Promise): void;
declare function describe(name: string, fn: (done: (err: any) => void) => ?Promise<any>): void;
declare function xdescribe(name: string, fn: (done: (err: any) => void) => ?Promise<any>): void;
declare function before(fn: (done: (err: any) => void) => ?Promise<any>): void;
declare function beforeEach(fn: (done: (err: any) => void) => ?Promise<any>): void;
declare function after(fn: (done: (err: any) => void) => ?Promise<any>): void;
declare function afterEach(fn: (done: (err: any) => void) => ?Promise<any>): void;
declare function it(name: string, fn: (done: (err: any) => void) => ?Promise<any>): void;
declare function xit(name: string, fn: (done: (err: any) => void) => ?Promise<any>): void;

@@ -27,2 +27,3 @@ 'use strict';

if (style.position !== 'fixed') {
// eslint-disable-next-line no-console
console.error('containByScreen only works on fixed position elements', element);

@@ -48,3 +49,7 @@ }

var allPossibleChoices = (0, _flatten2.default)(positions.map(function (position) {
return position === 'top' || position === 'bottom' ? hAligns.map(function (hAlign) {
return position === 'cover' ? (0, _flatten2.default)(hAligns.map(function (hAlign) {
return vAligns.map(function (vAlign) {
return { position: position, hAlign: hAlign, vAlign: vAlign };
});
})) : position === 'top' || position === 'bottom' ? hAligns.map(function (hAlign) {
return { position: position, hAlign: hAlign, vAlign: 'center' };

@@ -107,3 +112,30 @@ }) : vAligns.map(function (vAlign) {

left = 0;
if (position === 'top' || position === 'bottom') {
if (position === 'cover') {
switch (hAlign) {
case 'center':
left = Math.round((anchorRect.left + anchorRect.right - elRect.width) / 2);
break;
case 'left':
left = Math.floor(anchorRect.left);
break;
case 'right':
left = Math.ceil(anchorRect.right - elRect.width);
break;
default:
throw new Error('Should not happen');
}
switch (vAlign) {
case 'center':
top = Math.round((anchorRect.top + anchorRect.bottom - elRect.height) / 2);
break;
case 'top':
top = Math.floor(anchorRect.top);
break;
case 'bottom':
top = Math.ceil(anchorRect.bottom - elRect.height);
break;
default:
throw new Error('Should not happen');
}
} else if (position === 'top' || position === 'bottom') {
switch (position) {

@@ -117,3 +149,3 @@ case 'top':

default:
throw new Error("Should not happen");
throw new Error('Should not happen');
}

@@ -131,3 +163,3 @@ switch (hAlign) {

default:
throw new Error("Should not happen");
throw new Error('Should not happen');
}

@@ -143,3 +175,3 @@ } else {

default:
throw new Error("Should not happen");
throw new Error('Should not happen');
}

@@ -157,3 +189,3 @@ switch (vAlign) {

default:
throw new Error("Should not happen");
throw new Error('Should not happen');
}

@@ -164,2 +196,2 @@ }

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 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"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/index.js"],"names":["containByScreen","element","anchorPoint","options","process","env","NODE_ENV","window","getComputedStyle","style","position","console","error","elRect","getBoundingClientRect","anchorRect","buffers","all","buffer","top","topBuffer","bottom","bottomBuffer","left","leftBuffer","right","rightBuffer","positions","forcePosition","filter","Boolean","concat","hAligns","hAlign","forceHAlign","vAligns","vAlign","forceVAlign","allPossibleChoices","map","choiceAndCoord","i","length","choice","coord","positionAndAlign","height","innerHeight","width","innerWidth","el","rect","Math","round","floor","ceil","Error"],"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'|'cover';\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      // eslint-disable-next-line no-console\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 === 'cover') ?\n      flatten(hAligns.map(hAlign => vAligns.map(vAlign => ({position, hAlign, vAlign})))) :\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 === 'cover') {\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.floor(anchorRect.left);\n      break;\n    case 'right':\n      left = Math.ceil(anchorRect.right - elRect.width);\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.floor(anchorRect.top);\n      break;\n    case 'bottom':\n      top = Math.ceil(anchorRect.bottom - elRect.height);\n      break;\n    default: throw new Error('Should not happen');\n    }\n  } else 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.4",
"version": "1.1.0",
"description": "Position a dropdown element near a button in a way that fits on the screen.",

@@ -8,3 +8,6 @@ "main": "js/index.js",

"prepublish": "babel -s inline -d js/ src/ && flow-copy-source -v src js",
"test": "mocha"
"test": "npm run lint && npm run flow_check && mocha",
"flow_check": "flow check",
"lint": "eslint .",
"lint-fix": "eslint . --fix"
},

@@ -32,17 +35,21 @@ "repository": {

"devDependencies": {
"babel-cli": "^6.4.0",
"babel-plugin-add-module-exports": "^0.1.2",
"babel-plugin-transform-class-properties": "^6.4.0",
"babel-plugin-transform-flow-strip-types": "^6.4.0",
"babel-plugin-transform-runtime": "^6.4.3",
"babel-preset-es2015": "^6.3.13",
"babel-register": "^6.4.3",
"babel-cli": "^6.14.0",
"babel-eslint": "^6.1.2",
"babel-plugin-add-module-exports": "^0.2.1",
"babel-plugin-transform-class-properties": "^6.11.5",
"babel-plugin-transform-flow-strip-types": "^6.14.0",
"babel-plugin-transform-runtime": "^6.12.0",
"babel-preset-es2015": "^6.14.0",
"babel-register": "^6.14.0",
"eslint": "^3.4.0",
"eslint-plugin-react": "^6.2.0",
"flow-bin": "^0.32.0",
"flow-copy-source": "^1.0.1",
"mocha": "^2.3.4"
"mocha": "^3.0.2"
},
"dependencies": {
"babel-runtime": "^6.3.19",
"envify": "^3.4.0",
"babel-runtime": "^6.11.6",
"envify": "^3.4.1",
"lodash": "^4.6.1"
}
}
# contain-by-screen
[![Circle CI](https://circleci.com/gh/AgentME/contain-by-screen.svg?style=shield)](https://circleci.com/gh/AgentME/contain-by-screen)
[![npm version](https://badge.fury.io/js/contain-by-screen.svg)](https://badge.fury.io/js/contain-by-screen)
This function is for positioning an element next to another in a way that fits

@@ -22,5 +25,5 @@ on the screen. This can be used to position a dropdown menu next to a button.

`position` sets the prioritized position for the target relative to its anchor.
It may be set to null, "top", "bottom", "left", or "right". The element will
use this position unless it is not possible to do so while fitting the element
on-screen.
It may be set to null, "top", "bottom", "left", "right", or "cover". The
element will use this position unless it is not possible to do so while fitting
the element on-screen.

@@ -32,6 +35,6 @@ `forcePosition` is a boolean which controls whether the configured position

relative to its anchor. The horizontal alignment mode is used if the element is
positioned in the top or bottom positions relative to the anchor, and causes
the element to be moved horizontally in order to make a specific edge align. It
may be set to null, "center", "left", or "right". The element will use this
alignment unless it is not possible to do so while fitting the element
positioned in the top, bottom, or cover positions relative to the anchor, and
causes the element to be moved horizontally in order to make a specific edge
align. It may be set to null, "center", "left", or "right". The element will
use this alignment unless it is not possible to do so while fitting the element
on-screen.

@@ -44,6 +47,7 @@

to its anchor. The vertical alignment mode is used if the element is positioned
in the left or right positions relative to the anchor, and causes the element
to be moved vertically in order to make a specific edge align. It may be set to
null, "center", "top", or "bottom". The element will use this alignment unless
it is not possible to do so while fitting the element on-screen.
in the left, right, or cover positions relative to the anchor, and causes the
element to be moved vertically in order to make a specific edge align. It may
be set to null, "center", "top", or "bottom". The element will use this
alignment unless it is not possible to do so while fitting the element
on-screen.

@@ -56,6 +60,6 @@ `forceVAlign` is a boolean which controls whether the configured vAlign value

this much larger in all directions, requiring it to be placed with the given
amount of space between it, the anchor element, and the edges of the screen.
The buffer option is useful if the element has children which are positioned
such that they escape the boundaries of the element. Buffers do not affect
alignment with the anchor element.
amount of space between it, the anchor element when position is not "cover",
and the edges of the screen. The buffer option is useful if the element has
children which are positioned such that they escape the boundaries of the
element. Buffers do not affect alignment with the anchor element.

@@ -72,3 +76,3 @@ `topBuffer` specifies an additional buffer space only for the top edge.

Full [Flow Type](http://flowtype.org/) declarations for this module are
Full [Flow](http://flowtype.org/) type declarations for this module are
included!

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