暂无描述

scrollbar.js 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524
  1. /**
  2. * @license
  3. * Visual Blocks Editor
  4. *
  5. * Copyright 2011 Google Inc.
  6. * https://developers.google.com/blockly/
  7. *
  8. * Licensed under the Apache License, Version 2.0 (the "License");
  9. * you may not use this file except in compliance with the License.
  10. * You may obtain a copy of the License at
  11. *
  12. * http://www.apache.org/licenses/LICENSE-2.0
  13. *
  14. * Unless required by applicable law or agreed to in writing, software
  15. * distributed under the License is distributed on an "AS IS" BASIS,
  16. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  17. * See the License for the specific language governing permissions and
  18. * limitations under the License.
  19. */
  20. /**
  21. * @fileoverview Library for creating scrollbars.
  22. * @author fraser@google.com (Neil Fraser)
  23. */
  24. 'use strict';
  25. goog.provide('Blockly.Scrollbar');
  26. goog.provide('Blockly.ScrollbarPair');
  27. goog.require('goog.dom');
  28. goog.require('goog.events');
  29. /**
  30. * Class for a pair of scrollbars. Horizontal and vertical.
  31. * @param {!Blockly.Workspace} workspace Workspace to bind the scrollbars to.
  32. * @constructor
  33. */
  34. Blockly.ScrollbarPair = function(workspace) {
  35. this.workspace_ = workspace;
  36. this.hScroll = new Blockly.Scrollbar(workspace, true, true);
  37. this.vScroll = new Blockly.Scrollbar(workspace, false, true);
  38. this.corner_ = Blockly.createSvgElement('rect',
  39. {'height': Blockly.Scrollbar.scrollbarThickness,
  40. 'width': Blockly.Scrollbar.scrollbarThickness,
  41. 'class': 'blocklyScrollbarBackground'}, null);
  42. Blockly.Scrollbar.insertAfter_(this.corner_, workspace.getBubbleCanvas());
  43. };
  44. /**
  45. * Previously recorded metrics from the workspace.
  46. * @type {Object}
  47. * @private
  48. */
  49. Blockly.ScrollbarPair.prototype.oldHostMetrics_ = null;
  50. /**
  51. * Dispose of this pair of scrollbars.
  52. * Unlink from all DOM elements to prevent memory leaks.
  53. */
  54. Blockly.ScrollbarPair.prototype.dispose = function() {
  55. goog.dom.removeNode(this.corner_);
  56. this.corner_ = null;
  57. this.workspace_ = null;
  58. this.oldHostMetrics_ = null;
  59. this.hScroll.dispose();
  60. this.hScroll = null;
  61. this.vScroll.dispose();
  62. this.vScroll = null;
  63. };
  64. /**
  65. * Recalculate both of the scrollbars' locations and lengths.
  66. * Also reposition the corner rectangle.
  67. */
  68. Blockly.ScrollbarPair.prototype.resize = function() {
  69. // Look up the host metrics once, and use for both scrollbars.
  70. var hostMetrics = this.workspace_.getMetrics();
  71. if (!hostMetrics) {
  72. // Host element is likely not visible.
  73. return;
  74. }
  75. // Only change the scrollbars if there has been a change in metrics.
  76. var resizeH = false;
  77. var resizeV = false;
  78. if (!this.oldHostMetrics_ ||
  79. this.oldHostMetrics_.viewWidth != hostMetrics.viewWidth ||
  80. this.oldHostMetrics_.viewHeight != hostMetrics.viewHeight ||
  81. this.oldHostMetrics_.absoluteTop != hostMetrics.absoluteTop ||
  82. this.oldHostMetrics_.absoluteLeft != hostMetrics.absoluteLeft) {
  83. // The window has been resized or repositioned.
  84. resizeH = true;
  85. resizeV = true;
  86. } else {
  87. // Has the content been resized or moved?
  88. if (!this.oldHostMetrics_ ||
  89. this.oldHostMetrics_.contentWidth != hostMetrics.contentWidth ||
  90. this.oldHostMetrics_.viewLeft != hostMetrics.viewLeft ||
  91. this.oldHostMetrics_.contentLeft != hostMetrics.contentLeft) {
  92. resizeH = true;
  93. }
  94. if (!this.oldHostMetrics_ ||
  95. this.oldHostMetrics_.contentHeight != hostMetrics.contentHeight ||
  96. this.oldHostMetrics_.viewTop != hostMetrics.viewTop ||
  97. this.oldHostMetrics_.contentTop != hostMetrics.contentTop) {
  98. resizeV = true;
  99. }
  100. }
  101. if (resizeH) {
  102. this.hScroll.resize(hostMetrics);
  103. }
  104. if (resizeV) {
  105. this.vScroll.resize(hostMetrics);
  106. }
  107. // Reposition the corner square.
  108. if (!this.oldHostMetrics_ ||
  109. this.oldHostMetrics_.viewWidth != hostMetrics.viewWidth ||
  110. this.oldHostMetrics_.absoluteLeft != hostMetrics.absoluteLeft) {
  111. this.corner_.setAttribute('x', this.vScroll.xCoordinate);
  112. }
  113. if (!this.oldHostMetrics_ ||
  114. this.oldHostMetrics_.viewHeight != hostMetrics.viewHeight ||
  115. this.oldHostMetrics_.absoluteTop != hostMetrics.absoluteTop) {
  116. this.corner_.setAttribute('y', this.hScroll.yCoordinate);
  117. }
  118. // Cache the current metrics to potentially short-cut the next resize event.
  119. this.oldHostMetrics_ = hostMetrics;
  120. };
  121. /**
  122. * Set the sliders of both scrollbars to be at a certain position.
  123. * @param {number} x Horizontal scroll value.
  124. * @param {number} y Vertical scroll value.
  125. */
  126. Blockly.ScrollbarPair.prototype.set = function(x, y) {
  127. this.hScroll.set(x);
  128. this.vScroll.set(y);
  129. };
  130. // --------------------------------------------------------------------
  131. /**
  132. * Class for a pure SVG scrollbar.
  133. * This technique offers a scrollbar that is guaranteed to work, but may not
  134. * look or behave like the system's scrollbars.
  135. * @param {!Blockly.Workspace} workspace Workspace to bind the scrollbar to.
  136. * @param {boolean} horizontal True if horizontal, false if vertical.
  137. * @param {boolean=} opt_pair True if the scrollbar is part of a horiz/vert pair.
  138. * @constructor
  139. */
  140. Blockly.Scrollbar = function(workspace, horizontal, opt_pair) {
  141. this.workspace_ = workspace;
  142. this.pair_ = opt_pair || false;
  143. this.horizontal_ = horizontal;
  144. this.createDom_();
  145. if (horizontal) {
  146. this.svgBackground_.setAttribute('height',
  147. Blockly.Scrollbar.scrollbarThickness);
  148. this.svgKnob_.setAttribute('height',
  149. Blockly.Scrollbar.scrollbarThickness - 5);
  150. this.svgKnob_.setAttribute('y', 2.5);
  151. } else {
  152. this.svgBackground_.setAttribute('width',
  153. Blockly.Scrollbar.scrollbarThickness);
  154. this.svgKnob_.setAttribute('width',
  155. Blockly.Scrollbar.scrollbarThickness - 5);
  156. this.svgKnob_.setAttribute('x', 2.5);
  157. }
  158. var scrollbar = this;
  159. this.onMouseDownBarWrapper_ = Blockly.bindEvent_(this.svgBackground_,
  160. 'mousedown', scrollbar, scrollbar.onMouseDownBar_);
  161. this.onMouseDownKnobWrapper_ = Blockly.bindEvent_(this.svgKnob_,
  162. 'mousedown', scrollbar, scrollbar.onMouseDownKnob_);
  163. };
  164. /**
  165. * Width of vertical scrollbar or height of horizontal scrollbar.
  166. * Increase the size of scrollbars on touch devices.
  167. * Don't define if there is no document object (e.g. node.js).
  168. */
  169. Blockly.Scrollbar.scrollbarThickness = 15;
  170. if (goog.events.BrowserFeature.TOUCH_ENABLED) {
  171. Blockly.Scrollbar.scrollbarThickness = 25;
  172. }
  173. /**
  174. * Dispose of this scrollbar.
  175. * Unlink from all DOM elements to prevent memory leaks.
  176. */
  177. Blockly.Scrollbar.prototype.dispose = function() {
  178. this.onMouseUpKnob_();
  179. Blockly.unbindEvent_(this.onMouseDownBarWrapper_);
  180. this.onMouseDownBarWrapper_ = null;
  181. Blockly.unbindEvent_(this.onMouseDownKnobWrapper_);
  182. this.onMouseDownKnobWrapper_ = null;
  183. goog.dom.removeNode(this.svgGroup_);
  184. this.svgGroup_ = null;
  185. this.svgBackground_ = null;
  186. this.svgKnob_ = null;
  187. this.workspace_ = null;
  188. };
  189. /**
  190. * Recalculate the scrollbar's location and its length.
  191. * @param {Object=} opt_metrics A data structure of from the describing all the
  192. * required dimensions. If not provided, it will be fetched from the host
  193. * object.
  194. */
  195. Blockly.Scrollbar.prototype.resize = function(opt_metrics) {
  196. // Determine the location, height and width of the host element.
  197. var hostMetrics = opt_metrics;
  198. if (!hostMetrics) {
  199. hostMetrics = this.workspace_.getMetrics();
  200. if (!hostMetrics) {
  201. // Host element is likely not visible.
  202. return;
  203. }
  204. }
  205. /* hostMetrics is an object with the following properties.
  206. * .viewHeight: Height of the visible rectangle,
  207. * .viewWidth: Width of the visible rectangle,
  208. * .contentHeight: Height of the contents,
  209. * .contentWidth: Width of the content,
  210. * .viewTop: Offset of top edge of visible rectangle from parent,
  211. * .viewLeft: Offset of left edge of visible rectangle from parent,
  212. * .contentTop: Offset of the top-most content from the y=0 coordinate,
  213. * .contentLeft: Offset of the left-most content from the x=0 coordinate,
  214. * .absoluteTop: Top-edge of view.
  215. * .absoluteLeft: Left-edge of view.
  216. */
  217. if (this.horizontal_) {
  218. var outerLength = hostMetrics.viewWidth - 1;
  219. if (this.pair_) {
  220. // Shorten the scrollbar to make room for the corner square.
  221. outerLength -= Blockly.Scrollbar.scrollbarThickness;
  222. } else {
  223. // Only show the scrollbar if needed.
  224. // Ideally this would also apply to scrollbar pairs, but that's a bigger
  225. // headache (due to interactions with the corner square).
  226. this.setVisible(outerLength < hostMetrics.contentHeight);
  227. }
  228. this.ratio_ = outerLength / hostMetrics.contentWidth;
  229. if (this.ratio_ === -Infinity || this.ratio_ === Infinity ||
  230. isNaN(this.ratio_)) {
  231. this.ratio_ = 0;
  232. }
  233. var innerLength = hostMetrics.viewWidth * this.ratio_;
  234. var innerOffset = (hostMetrics.viewLeft - hostMetrics.contentLeft) *
  235. this.ratio_;
  236. this.svgKnob_.setAttribute('width', Math.max(0, innerLength));
  237. this.xCoordinate = hostMetrics.absoluteLeft + 0.5;
  238. if (this.pair_ && this.workspace_.RTL) {
  239. this.xCoordinate += hostMetrics.absoluteLeft +
  240. Blockly.Scrollbar.scrollbarThickness;
  241. }
  242. this.yCoordinate = hostMetrics.absoluteTop + hostMetrics.viewHeight -
  243. Blockly.Scrollbar.scrollbarThickness - 0.5;
  244. this.svgGroup_.setAttribute('transform',
  245. 'translate(' + this.xCoordinate + ',' + this.yCoordinate + ')');
  246. this.svgBackground_.setAttribute('width', Math.max(0, outerLength));
  247. this.svgKnob_.setAttribute('x', this.constrainKnob_(innerOffset));
  248. } else {
  249. var outerLength = hostMetrics.viewHeight - 1;
  250. if (this.pair_) {
  251. // Shorten the scrollbar to make room for the corner square.
  252. outerLength -= Blockly.Scrollbar.scrollbarThickness;
  253. } else {
  254. // Only show the scrollbar if needed.
  255. this.setVisible(outerLength < hostMetrics.contentHeight);
  256. }
  257. this.ratio_ = outerLength / hostMetrics.contentHeight;
  258. if (this.ratio_ === -Infinity || this.ratio_ === Infinity ||
  259. isNaN(this.ratio_)) {
  260. this.ratio_ = 0;
  261. }
  262. var innerLength = hostMetrics.viewHeight * this.ratio_;
  263. var innerOffset = (hostMetrics.viewTop - hostMetrics.contentTop) *
  264. this.ratio_;
  265. this.svgKnob_.setAttribute('height', Math.max(0, innerLength));
  266. this.xCoordinate = hostMetrics.absoluteLeft + 0.5;
  267. if (!this.workspace_.RTL) {
  268. this.xCoordinate += hostMetrics.viewWidth -
  269. Blockly.Scrollbar.scrollbarThickness - 1;
  270. }
  271. this.yCoordinate = hostMetrics.absoluteTop + 0.5;
  272. this.svgGroup_.setAttribute('transform',
  273. 'translate(' + this.xCoordinate + ',' + this.yCoordinate + ')');
  274. this.svgBackground_.setAttribute('height', Math.max(0, outerLength));
  275. this.svgKnob_.setAttribute('y', this.constrainKnob_(innerOffset));
  276. }
  277. // Resizing may have caused some scrolling.
  278. this.onScroll_();
  279. };
  280. /**
  281. * Create all the DOM elements required for a scrollbar.
  282. * The resulting widget is not sized.
  283. * @private
  284. */
  285. Blockly.Scrollbar.prototype.createDom_ = function() {
  286. /* Create the following DOM:
  287. <g class="blocklyScrollbarHorizontal">
  288. <rect class="blocklyScrollbarBackground" />
  289. <rect class="blocklyScrollbarKnob" rx="8" ry="8" />
  290. </g>
  291. */
  292. var className = 'blocklyScrollbar' +
  293. (this.horizontal_ ? 'Horizontal' : 'Vertical');
  294. this.svgGroup_ = Blockly.createSvgElement('g', {'class': className}, null);
  295. this.svgBackground_ = Blockly.createSvgElement('rect',
  296. {'class': 'blocklyScrollbarBackground'}, this.svgGroup_);
  297. var radius = Math.floor((Blockly.Scrollbar.scrollbarThickness - 5) / 2);
  298. this.svgKnob_ = Blockly.createSvgElement('rect',
  299. {'class': 'blocklyScrollbarKnob', 'rx': radius, 'ry': radius},
  300. this.svgGroup_);
  301. Blockly.Scrollbar.insertAfter_(this.svgGroup_,
  302. this.workspace_.getBubbleCanvas());
  303. };
  304. /**
  305. * Is the scrollbar visible. Non-paired scrollbars disappear when they aren't
  306. * needed.
  307. * @return {boolean} True if visible.
  308. */
  309. Blockly.Scrollbar.prototype.isVisible = function() {
  310. return this.svgGroup_.getAttribute('display') != 'none';
  311. };
  312. /**
  313. * Set whether the scrollbar is visible.
  314. * Only applies to non-paired scrollbars.
  315. * @param {boolean} visible True if visible.
  316. */
  317. Blockly.Scrollbar.prototype.setVisible = function(visible) {
  318. if (visible == this.isVisible()) {
  319. return;
  320. }
  321. // Ideally this would also apply to scrollbar pairs, but that's a bigger
  322. // headache (due to interactions with the corner square).
  323. if (this.pair_) {
  324. throw 'Unable to toggle visibility of paired scrollbars.';
  325. }
  326. if (visible) {
  327. this.svgGroup_.setAttribute('display', 'block');
  328. } else {
  329. // Hide the scrollbar.
  330. this.workspace_.setMetrics({x: 0, y: 0});
  331. this.svgGroup_.setAttribute('display', 'none');
  332. }
  333. };
  334. /**
  335. * Scroll by one pageful.
  336. * Called when scrollbar background is clicked.
  337. * @param {!Event} e Mouse down event.
  338. * @private
  339. */
  340. Blockly.Scrollbar.prototype.onMouseDownBar_ = function(e) {
  341. this.onMouseUpKnob_();
  342. if (Blockly.isRightButton(e)) {
  343. // Right-click.
  344. // Scrollbars have no context menu.
  345. e.stopPropagation();
  346. return;
  347. }
  348. var mouseXY = Blockly.mouseToSvg(e, this.workspace_.options.svg);
  349. var mouseLocation = this.horizontal_ ? mouseXY.x : mouseXY.y;
  350. var knobXY = Blockly.getSvgXY_(this.svgKnob_, this.workspace_);
  351. var knobStart = this.horizontal_ ? knobXY.x : knobXY.y;
  352. var knobLength = parseFloat(
  353. this.svgKnob_.getAttribute(this.horizontal_ ? 'width' : 'height'));
  354. var knobValue = parseFloat(
  355. this.svgKnob_.getAttribute(this.horizontal_ ? 'x' : 'y'));
  356. var pageLength = knobLength * 0.95;
  357. if (mouseLocation <= knobStart) {
  358. // Decrease the scrollbar's value by a page.
  359. knobValue -= pageLength;
  360. } else if (mouseLocation >= knobStart + knobLength) {
  361. // Increase the scrollbar's value by a page.
  362. knobValue += pageLength;
  363. }
  364. this.svgKnob_.setAttribute(this.horizontal_ ? 'x' : 'y',
  365. this.constrainKnob_(knobValue));
  366. this.onScroll_();
  367. e.stopPropagation();
  368. };
  369. /**
  370. * Start a dragging operation.
  371. * Called when scrollbar knob is clicked.
  372. * @param {!Event} e Mouse down event.
  373. * @private
  374. */
  375. Blockly.Scrollbar.prototype.onMouseDownKnob_ = function(e) {
  376. this.onMouseUpKnob_();
  377. if (Blockly.isRightButton(e)) {
  378. // Right-click.
  379. // Scrollbars have no context menu.
  380. e.stopPropagation();
  381. return;
  382. }
  383. // Look up the current translation and record it.
  384. this.startDragKnob = parseFloat(
  385. this.svgKnob_.getAttribute(this.horizontal_ ? 'x' : 'y'));
  386. // Record the current mouse position.
  387. this.startDragMouse = this.horizontal_ ? e.clientX : e.clientY;
  388. Blockly.Scrollbar.onMouseUpWrapper_ = Blockly.bindEvent_(document,
  389. 'mouseup', this, this.onMouseUpKnob_);
  390. Blockly.Scrollbar.onMouseMoveWrapper_ = Blockly.bindEvent_(document,
  391. 'mousemove', this, this.onMouseMoveKnob_);
  392. e.stopPropagation();
  393. };
  394. /**
  395. * Drag the scrollbar's knob.
  396. * @param {!Event} e Mouse up event.
  397. * @private
  398. */
  399. Blockly.Scrollbar.prototype.onMouseMoveKnob_ = function(e) {
  400. var currentMouse = this.horizontal_ ? e.clientX : e.clientY;
  401. var mouseDelta = currentMouse - this.startDragMouse;
  402. var knobValue = this.startDragKnob + mouseDelta;
  403. // Position the bar.
  404. this.svgKnob_.setAttribute(this.horizontal_ ? 'x' : 'y',
  405. this.constrainKnob_(knobValue));
  406. this.onScroll_();
  407. };
  408. /**
  409. * Stop binding to the global mouseup and mousemove events.
  410. * @private
  411. */
  412. Blockly.Scrollbar.prototype.onMouseUpKnob_ = function() {
  413. Blockly.removeAllRanges();
  414. Blockly.hideChaff(true);
  415. if (Blockly.Scrollbar.onMouseUpWrapper_) {
  416. Blockly.unbindEvent_(Blockly.Scrollbar.onMouseUpWrapper_);
  417. Blockly.Scrollbar.onMouseUpWrapper_ = null;
  418. }
  419. if (Blockly.Scrollbar.onMouseMoveWrapper_) {
  420. Blockly.unbindEvent_(Blockly.Scrollbar.onMouseMoveWrapper_);
  421. Blockly.Scrollbar.onMouseMoveWrapper_ = null;
  422. }
  423. };
  424. /**
  425. * Constrain the knob's position within the minimum (0) and maximum
  426. * (length of scrollbar) values allowed for the scrollbar.
  427. * @param {number} value Value that is potentially out of bounds.
  428. * @return {number} Constrained value.
  429. * @private
  430. */
  431. Blockly.Scrollbar.prototype.constrainKnob_ = function(value) {
  432. if (value <= 0 || isNaN(value)) {
  433. value = 0;
  434. } else {
  435. var axis = this.horizontal_ ? 'width' : 'height';
  436. var barLength = parseFloat(this.svgBackground_.getAttribute(axis));
  437. var knobLength = parseFloat(this.svgKnob_.getAttribute(axis));
  438. value = Math.min(value, barLength - knobLength);
  439. }
  440. return value;
  441. };
  442. /**
  443. * Called when scrollbar is moved.
  444. * @private
  445. */
  446. Blockly.Scrollbar.prototype.onScroll_ = function() {
  447. var knobValue = parseFloat(
  448. this.svgKnob_.getAttribute(this.horizontal_ ? 'x' : 'y'));
  449. var barLength = parseFloat(
  450. this.svgBackground_.getAttribute(this.horizontal_ ? 'width' : 'height'));
  451. var ratio = knobValue / barLength;
  452. if (isNaN(ratio)) {
  453. ratio = 0;
  454. }
  455. var xyRatio = {};
  456. if (this.horizontal_) {
  457. xyRatio.x = ratio;
  458. } else {
  459. xyRatio.y = ratio;
  460. }
  461. this.workspace_.setMetrics(xyRatio);
  462. };
  463. /**
  464. * Set the scrollbar slider's position.
  465. * @param {number} value The distance from the top/left end of the bar.
  466. */
  467. Blockly.Scrollbar.prototype.set = function(value) {
  468. // Move the scrollbar slider.
  469. this.svgKnob_.setAttribute(this.horizontal_ ? 'x' : 'y', value * this.ratio_);
  470. this.onScroll_();
  471. };
  472. /**
  473. * Insert a node after a reference node.
  474. * Contrast with node.insertBefore function.
  475. * @param {!Element} newNode New element to insert.
  476. * @param {!Element} refNode Existing element to precede new node.
  477. * @private
  478. */
  479. Blockly.Scrollbar.insertAfter_ = function(newNode, refNode) {
  480. var siblingNode = refNode.nextSibling;
  481. var parentNode = refNode.parentNode;
  482. if (!parentNode) {
  483. throw 'Reference node has no parent.';
  484. }
  485. if (siblingNode) {
  486. parentNode.insertBefore(newNode, siblingNode);
  487. } else {
  488. parentNode.appendChild(newNode);
  489. }
  490. };