contain-by-screen
Advanced tools
Comparing version 1.0.4 to 1.1.0
@@ -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
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
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
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
37052
12
208
74
13
1
Updatedbabel-runtime@^6.11.6
Updatedenvify@^3.4.1