Source: widgets/slider.js

Source: widgets/slider.js

  1. /*
  2. * This file is part of Toolkit.
  3. *
  4. * Toolkit is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU General Public
  6. * License as published by the Free Software Foundation; either
  7. * version 3 of the License, or (at your option) any later version.
  8. *
  9. * Toolkit is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. * Lesser General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General
  15. * Public License along with this program; if not, write to the
  16. * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
  17. * Boston, MA 02110-1301 USA
  18. */
  19. /**
  20. * The <code>useraction</code> event is emitted when a widget gets modified by user interaction.
  21. * The event is emitted for the option <code>value</code>.
  22. *
  23. * @event TK.Slider#useraction
  24. *
  25. * @param {string} name - The name of the option which was changed due to the users action
  26. * @param {mixed} value - The new value of the option
  27. */
  28. "use strict";
  29. (function(w, TK){
  30. function dblclick() {
  31. this.userset("value", this.options.reset);
  32. /**
  33. * Is fired when the slider receives a double click in order to reset to initial value.
  34. *
  35. * @event TK.Slider#doubleclick
  36. *
  37. * @param {number} value - The value of the widget.
  38. */
  39. this.fire_event("doubleclick", this.options.value);
  40. }
  41. function set_background(horiz, vert, size) {
  42. var E = this.element;
  43. E.style["background-position"] = "-"+horiz+"px -"+vert+"px";
  44. E.style["-webkit-background-size"] = size;
  45. E.style["-moz-background-size"] = size;
  46. E.style["-ms-background-size"] = size;
  47. E.style["-o-background-size"] = size;
  48. E.style["background-size"] = size;
  49. }
  50. /**
  51. * TK.Slider is a {@link TK.Widget} moving its background image
  52. * according to its value. It can be used to show strips of
  53. * e.g. 3D-rendered faders or knobs. It's important to set the
  54. * width and height of the widget in CSS according to the frames in
  55. * the background file. If alignment is `horizontal` the background image
  56. * is as height as the widget, the width keeps the ratio intact. Overall
  57. * width of the image should be frames * width. If alignment is `vertical`
  58. * the background image is as wide as the widget and the height of the
  59. * image keeps the ratio intact. The height should be height of widget
  60. * times the amount of frames.
  61. * TK.Slider uses {@link TK.DragValue} and {@link TK.ScrollValue}
  62. * for setting its value.
  63. * It inherits all options of {@link TK.DragValue}, {@link TK.Ranged} and {@link TK.Warning}.
  64. *
  65. * @class TK.Slider
  66. *
  67. * @extends TK.Widget
  68. *
  69. * @param {Object} [options={ }] - An object containing initial options.
  70. *
  71. * @property {Number} [options.value=0] - The current value.
  72. * @property {Integer} [options.frames=1] - The amount of frames contained
  73. * in the background image.
  74. * @property {String} [options.alignment="horizontal"] - The direction
  75. * of the frames in the image, next to (`horizontal`) or among each other (`vertical`).
  76. * @property {String|Booelan} [options.image=false] - The image containing all frames for the slider.
  77. * Set to `false` to set the background image via external CSS.
  78. *
  79. */
  80. TK.Slider = TK.class({
  81. _class: "Slider",
  82. Extends: TK.Widget,
  83. Implements: [TK.Ranged, TK.Warning],
  84. _options: Object.assign(Object.create(TK.Widget.prototype._options),
  85. TK.Ranged.prototype._options,
  86. TK.DragValue.prototype._options, {
  87. value: "number",
  88. frames: "int",
  89. alignment: "string",
  90. image: "string|boolean",
  91. _width: "number",
  92. _height: "number",
  93. }),
  94. options: {
  95. value: 0,
  96. frames: 1,
  97. alignment: "horizontal",
  98. image: false,
  99. direction: "polar",
  100. rotation: 45,
  101. blind_angle: 20,
  102. basis: 300,
  103. },
  104. static_events: {
  105. dblclick: dblclick,
  106. },
  107. initialize: function (options) {
  108. TK.Widget.prototype.initialize.call(this, options);
  109. var E;
  110. /**
  111. * @member {HTMLDivElement} TK.Slider#element - The main DIV container.
  112. * Has class <code>toolkit-slider</code>.
  113. */
  114. if (!(E = this.element)) this.element = E = TK.element("div");
  115. TK.add_class(E, "toolkit-slider");
  116. this.widgetize(E, true, true, true);
  117. /**
  118. * @member {TK.DragValue} TK.Knob#drag - Instance of {@link TK.DragValue} used for
  119. * interaction.
  120. */
  121. this.drag = new TK.DragValue(this, {
  122. node: E,
  123. classes: E,
  124. direction: this.options.direction,
  125. rotation: this.options.rotation,
  126. blind_angle: this.options.blind_angle,
  127. });
  128. /**
  129. * @member {TK.ScrollValue} TK.Knob#scroll - Instance of {@link TK.ScrollValue} used for
  130. * interaction.
  131. */
  132. this.scroll = new TK.ScrollValue(this, {
  133. node: E,
  134. classes: E,
  135. });
  136. if (options.reset === void(0))
  137. options.reset = options.value;
  138. },
  139. destroy: function () {
  140. this.drag.destroy();
  141. this.scroll.destroy();
  142. TK.Widget.prototype.destroy.call(this);
  143. },
  144. redraw: function() {
  145. var I = this.invalid;
  146. var O = this.options;
  147. var E = this.element;
  148. if (I.image) {
  149. I.image = false;
  150. if (O.image)
  151. this.element.style["background-image"] = "url('" + O.image + "')";
  152. else
  153. this.element.style["background-image"] = void 0;
  154. I.value = true;
  155. }
  156. if (I.value || I.alignment || O.frames) {
  157. I.value = false;
  158. I.alignment = false;
  159. I.frames = false;
  160. var coef = this.val2coef(O.value);
  161. var frame = Math.round(Math.max(0, O.frames - 1) * coef);
  162. switch (O.alignment) {
  163. default:
  164. TK.warn("Unknown alignment, only 'vertical' and 'horizontal' are allowed");
  165. break;
  166. case "vertical":
  167. set_background.call(this, 0, frame * O._width, "100% auto");
  168. break;
  169. case "horizontal":
  170. set_background.call(this, frame * O._height, 0, "auto 100%");
  171. break;
  172. }
  173. }
  174. TK.Widget.prototype.redraw.call(this);
  175. },
  176. resize: function () {
  177. this.set("_width", TK.outer_width(this.element));
  178. this.set("_height", TK.outer_height(this.element));
  179. },
  180. set: function(key, value) {
  181. switch (key) {
  182. case "value":
  183. if (value > this.options.max || value < this.options.min)
  184. this.warning(this.element);
  185. value = this.snap(value);
  186. break;
  187. }
  188. if (TK.DragValue.prototype._options[key])
  189. this.drag.set(key, value);
  190. return TK.Widget.prototype.set.call(this, key, value);
  191. },
  192. });
  193. })(this, this.TK);