Version 3.18.1
Show:

File: widget-position-constrain/js/Widget-PositionConstrain.js

  1. /**
  2. * Provides constrained xy positioning support for Widgets, through an extension.
  3. *
  4. * It builds on top of the widget-position module, to provide constrained positioning support.
  5. *
  6. * @module widget-position-constrain
  7. */
  8. var CONSTRAIN = "constrain",
  9. CONSTRAIN_XYCHANGE = "constrain|xyChange",
  10. CONSTRAIN_CHANGE = "constrainChange",
  11. PREVENT_OVERLAP = "preventOverlap",
  12. ALIGN = "align",
  13. EMPTY_STR = "",
  14. BINDUI = "bindUI",
  15. XY = "xy",
  16. X_COORD = "x",
  17. Y_COORD = "y",
  18. Node = Y.Node,
  19. VIEWPORT_REGION = "viewportRegion",
  20. REGION = "region",
  21. PREVENT_OVERLAP_MAP;
  22. /**
  23. * A widget extension, which can be used to add constrained xy positioning support to the base Widget class,
  24. * through the <a href="Base.html#method_build">Base.build</a> method. This extension requires that
  25. * the WidgetPosition extension be added to the Widget (before WidgetPositionConstrain, if part of the same
  26. * extension list passed to Base.build).
  27. *
  28. * @class WidgetPositionConstrain
  29. * @param {Object} User configuration object
  30. */
  31. function PositionConstrain(config) {}
  32. /**
  33. * Static property used to define the default attribute
  34. * configuration introduced by WidgetPositionConstrain.
  35. *
  36. * @property ATTRS
  37. * @type Object
  38. * @static
  39. */
  40. PositionConstrain.ATTRS = {
  41. /**
  42. * @attribute constrain
  43. * @type boolean | Node
  44. * @default null
  45. * @description The node to constrain the widget's bounding box to, when setting xy. Can also be
  46. * set to true, to constrain to the viewport.
  47. */
  48. constrain : {
  49. value: null,
  50. setter: "_setConstrain"
  51. },
  52. /**
  53. * @attribute preventOverlap
  54. * @type boolean
  55. * @description If set to true, and WidgetPositionAlign is also added to the Widget,
  56. * constrained positioning will attempt to prevent the widget's bounding box from overlapping
  57. * the element to which it has been aligned, by flipping the orientation of the alignment
  58. * for corner based alignments
  59. */
  60. preventOverlap : {
  61. value:false
  62. }
  63. };
  64. /**
  65. * @property _PREVENT_OVERLAP
  66. * @static
  67. * @protected
  68. * @type Object
  69. * @description The set of positions for which to prevent
  70. * overlap.
  71. */
  72. PREVENT_OVERLAP_MAP = PositionConstrain._PREVENT_OVERLAP = {
  73. x: {
  74. "tltr": 1,
  75. "blbr": 1,
  76. "brbl": 1,
  77. "trtl": 1
  78. },
  79. y : {
  80. "trbr": 1,
  81. "tlbl": 1,
  82. "bltl": 1,
  83. "brtr": 1
  84. }
  85. };
  86. PositionConstrain.prototype = {
  87. initializer : function() {
  88. if (!this._posNode) {
  89. Y.error("WidgetPosition needs to be added to the Widget, before WidgetPositionConstrain is added");
  90. }
  91. Y.after(this._bindUIPosConstrained, this, BINDUI);
  92. },
  93. /**
  94. * Calculates the constrained positions for the XY positions provided, using
  95. * the provided node argument is passed in. If no node value is passed in, the value of
  96. * the "constrain" attribute is used.
  97. *
  98. * @method getConstrainedXY
  99. * @param {Array} xy The xy values to constrain
  100. * @param {Node | boolean} node Optional. The node to constrain to, or true for the viewport
  101. * @return {Array} The constrained xy values
  102. */
  103. getConstrainedXY : function(xy, node) {
  104. node = node || this.get(CONSTRAIN);
  105. var constrainingRegion = this._getRegion((node === true) ? null : node),
  106. nodeRegion = this._posNode.get(REGION);
  107. return [
  108. this._constrain(xy[0], X_COORD, nodeRegion, constrainingRegion),
  109. this._constrain(xy[1], Y_COORD, nodeRegion, constrainingRegion)
  110. ];
  111. },
  112. /**
  113. * Constrains the widget's bounding box to a node (or the viewport). If xy or node are not
  114. * passed in, the current position and the value of "constrain" will be used respectively.
  115. *
  116. * The widget's position will be changed to the constrained position.
  117. *
  118. * @method constrain
  119. * @param {Array} xy Optional. The xy values to constrain
  120. * @param {Node | boolean} node Optional. The node to constrain to, or true for the viewport
  121. */
  122. constrain : function(xy, node) {
  123. var currentXY,
  124. constrainedXY,
  125. constraint = node || this.get(CONSTRAIN);
  126. if (constraint) {
  127. currentXY = xy || this.get(XY);
  128. constrainedXY = this.getConstrainedXY(currentXY, constraint);
  129. if (constrainedXY[0] !== currentXY[0] || constrainedXY[1] !== currentXY[1]) {
  130. this.set(XY, constrainedXY, { constrained:true });
  131. }
  132. }
  133. },
  134. /**
  135. * The setter implementation for the "constrain" attribute.
  136. *
  137. * @method _setConstrain
  138. * @protected
  139. * @param {Node | boolean} val The attribute value
  140. */
  141. _setConstrain : function(val) {
  142. return (val === true) ? val : Node.one(val);
  143. },
  144. /**
  145. * The method which performs the actual constrain calculations for a given axis ("x" or "y") based
  146. * on the regions provided.
  147. *
  148. * @method _constrain
  149. * @protected
  150. *
  151. * @param {Number} val The value to constrain
  152. * @param {String} axis The axis to use for constrainment
  153. * @param {Region} nodeRegion The region of the node to constrain
  154. * @param {Region} constrainingRegion The region of the node (or viewport) to constrain to
  155. *
  156. * @return {Number} The constrained value
  157. */
  158. _constrain: function(val, axis, nodeRegion, constrainingRegion) {
  159. if (constrainingRegion) {
  160. if (this.get(PREVENT_OVERLAP)) {
  161. val = this._preventOverlap(val, axis, nodeRegion, constrainingRegion);
  162. }
  163. var x = (axis == X_COORD),
  164. regionSize = (x) ? constrainingRegion.width : constrainingRegion.height,
  165. nodeSize = (x) ? nodeRegion.width : nodeRegion.height,
  166. minConstraint = (x) ? constrainingRegion.left : constrainingRegion.top,
  167. maxConstraint = (x) ? constrainingRegion.right - nodeSize : constrainingRegion.bottom - nodeSize;
  168. if (val < minConstraint || val > maxConstraint) {
  169. if (nodeSize < regionSize) {
  170. if (val < minConstraint) {
  171. val = minConstraint;
  172. } else if (val > maxConstraint) {
  173. val = maxConstraint;
  174. }
  175. } else {
  176. val = minConstraint;
  177. }
  178. }
  179. }
  180. return val;
  181. },
  182. /**
  183. * The method which performs the preventOverlap calculations for a given axis ("x" or "y") based
  184. * on the value and regions provided.
  185. *
  186. * @method _preventOverlap
  187. * @protected
  188. *
  189. * @param {Number} val The value being constrain
  190. * @param {String} axis The axis to being constrained
  191. * @param {Region} nodeRegion The region of the node being constrained
  192. * @param {Region} constrainingRegion The region of the node (or viewport) we need to constrain to
  193. *
  194. * @return {Number} The constrained value
  195. */
  196. _preventOverlap : function(val, axis, nodeRegion, constrainingRegion) {
  197. var align = this.get(ALIGN),
  198. x = (axis === X_COORD),
  199. nodeSize,
  200. alignRegion,
  201. nearEdge,
  202. farEdge,
  203. spaceOnNearSide,
  204. spaceOnFarSide;
  205. if (align && align.points && PREVENT_OVERLAP_MAP[axis][align.points.join(EMPTY_STR)]) {
  206. alignRegion = this._getRegion(align.node);
  207. if (alignRegion) {
  208. nodeSize = (x) ? nodeRegion.width : nodeRegion.height;
  209. nearEdge = (x) ? alignRegion.left : alignRegion.top;
  210. farEdge = (x) ? alignRegion.right : alignRegion.bottom;
  211. spaceOnNearSide = (x) ? alignRegion.left - constrainingRegion.left : alignRegion.top - constrainingRegion.top;
  212. spaceOnFarSide = (x) ? constrainingRegion.right - alignRegion.right : constrainingRegion.bottom - alignRegion.bottom;
  213. }
  214. if (val > nearEdge) {
  215. if (spaceOnFarSide < nodeSize && spaceOnNearSide > nodeSize) {
  216. val = nearEdge - nodeSize;
  217. }
  218. } else {
  219. if (spaceOnNearSide < nodeSize && spaceOnFarSide > nodeSize) {
  220. val = farEdge;
  221. }
  222. }
  223. }
  224. return val;
  225. },
  226. /**
  227. * Binds event listeners responsible for updating the UI state in response to
  228. * Widget constrained positioning related state changes.
  229. * <p>
  230. * This method is invoked after bindUI is invoked for the Widget class
  231. * using YUI's aop infrastructure.
  232. * </p>
  233. *
  234. * @method _bindUIPosConstrained
  235. * @protected
  236. */
  237. _bindUIPosConstrained : function() {
  238. this.after(CONSTRAIN_CHANGE, this._afterConstrainChange);
  239. this._enableConstraints(this.get(CONSTRAIN));
  240. },
  241. /**
  242. * After change listener for the "constrain" attribute, responsible
  243. * for updating the UI, in response to attribute changes.
  244. *
  245. * @method _afterConstrainChange
  246. * @protected
  247. * @param {EventFacade} e The event facade
  248. */
  249. _afterConstrainChange : function(e) {
  250. this._enableConstraints(e.newVal);
  251. },
  252. /**
  253. * Updates the UI if enabling constraints, and sets up the xyChange event listeners
  254. * to constrain whenever the widget is moved. Disabling constraints removes the listeners.
  255. *
  256. * @method _enableConstraints
  257. * @private
  258. * @param {boolean} enable Enable or disable constraints
  259. */
  260. _enableConstraints : function(enable) {
  261. if (enable) {
  262. this.constrain();
  263. this._cxyHandle = this._cxyHandle || this.on(CONSTRAIN_XYCHANGE, this._constrainOnXYChange);
  264. } else if (this._cxyHandle) {
  265. this._cxyHandle.detach();
  266. this._cxyHandle = null;
  267. }
  268. },
  269. /**
  270. * The on change listener for the "xy" attribute. Modifies the event facade's
  271. * newVal property with the constrained XY value.
  272. *
  273. * @method _constrainOnXYChange
  274. * @protected
  275. * @param {EventFacade} e The event facade for the attribute change
  276. */
  277. _constrainOnXYChange : function(e) {
  278. if (!e.constrained) {
  279. e.newVal = this.getConstrainedXY(e.newVal);
  280. }
  281. },
  282. /**
  283. * Utility method to normalize region retrieval from a node instance,
  284. * or the viewport, if no node is provided.
  285. *
  286. * @method _getRegion
  287. * @private
  288. * @param {Node} node Optional.
  289. */
  290. _getRegion : function(node) {
  291. var region;
  292. if (!node) {
  293. region = this._posNode.get(VIEWPORT_REGION);
  294. } else {
  295. node = Node.one(node);
  296. if (node) {
  297. region = node.get(REGION);
  298. }
  299. }
  300. return region;
  301. }
  302. };
  303. Y.WidgetPositionConstrain = PositionConstrain;