administrate-materialize-theme 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (81) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.md +49 -0
  4. data/Rakefile +1 -0
  5. data/app/assets/javascripts/administrate-materialize-theme/anime.min.js +34 -0
  6. data/app/assets/javascripts/administrate-materialize-theme/autocomplete.js +450 -0
  7. data/app/assets/javascripts/administrate-materialize-theme/bin/materialize.js +12374 -0
  8. data/app/assets/javascripts/administrate-materialize-theme/bin/materialize.min.js +6 -0
  9. data/app/assets/javascripts/administrate-materialize-theme/buttons.js +354 -0
  10. data/app/assets/javascripts/administrate-materialize-theme/cards.js +40 -0
  11. data/app/assets/javascripts/administrate-materialize-theme/carousel.js +717 -0
  12. data/app/assets/javascripts/administrate-materialize-theme/cash.js +960 -0
  13. data/app/assets/javascripts/administrate-materialize-theme/characterCounter.js +136 -0
  14. data/app/assets/javascripts/administrate-materialize-theme/chips.js +481 -0
  15. data/app/assets/javascripts/administrate-materialize-theme/collapsible.js +275 -0
  16. data/app/assets/javascripts/administrate-materialize-theme/component.js +44 -0
  17. data/app/assets/javascripts/administrate-materialize-theme/datepicker.js +975 -0
  18. data/app/assets/javascripts/administrate-materialize-theme/dropdown.js +617 -0
  19. data/app/assets/javascripts/administrate-materialize-theme/forms.js +275 -0
  20. data/app/assets/javascripts/administrate-materialize-theme/global.js +427 -0
  21. data/app/assets/javascripts/administrate-materialize-theme/materialbox.js +453 -0
  22. data/app/assets/javascripts/administrate-materialize-theme/modal.js +382 -0
  23. data/app/assets/javascripts/administrate-materialize-theme/parallax.js +138 -0
  24. data/app/assets/javascripts/administrate-materialize-theme/pushpin.js +145 -0
  25. data/app/assets/javascripts/administrate-materialize-theme/range.js +263 -0
  26. data/app/assets/javascripts/administrate-materialize-theme/scrollspy.js +295 -0
  27. data/app/assets/javascripts/administrate-materialize-theme/select.js +432 -0
  28. data/app/assets/javascripts/administrate-materialize-theme/sidenav.js +580 -0
  29. data/app/assets/javascripts/administrate-materialize-theme/slider.js +359 -0
  30. data/app/assets/javascripts/administrate-materialize-theme/tabs.js +402 -0
  31. data/app/assets/javascripts/administrate-materialize-theme/tapTarget.js +314 -0
  32. data/app/assets/javascripts/administrate-materialize-theme/theme.js +6 -0
  33. data/app/assets/javascripts/administrate-materialize-theme/timepicker.js +647 -0
  34. data/app/assets/javascripts/administrate-materialize-theme/toasts.js +310 -0
  35. data/app/assets/javascripts/administrate-materialize-theme/tooltip.js +303 -0
  36. data/app/assets/javascripts/administrate-materialize-theme/waves.js +335 -0
  37. data/app/assets/stylesheets/administrate-materialize-theme/components/_badges.scss +55 -0
  38. data/app/assets/stylesheets/administrate-materialize-theme/components/_buttons.scss +322 -0
  39. data/app/assets/stylesheets/administrate-materialize-theme/components/_cards.scss +195 -0
  40. data/app/assets/stylesheets/administrate-materialize-theme/components/_carousel.scss +90 -0
  41. data/app/assets/stylesheets/administrate-materialize-theme/components/_chips.scss +90 -0
  42. data/app/assets/stylesheets/administrate-materialize-theme/components/_collapsible.scss +91 -0
  43. data/app/assets/stylesheets/administrate-materialize-theme/components/_color-classes.scss +32 -0
  44. data/app/assets/stylesheets/administrate-materialize-theme/components/_color-variables.scss +370 -0
  45. data/app/assets/stylesheets/administrate-materialize-theme/components/_datepicker.scss +191 -0
  46. data/app/assets/stylesheets/administrate-materialize-theme/components/_dropdown.scss +85 -0
  47. data/app/assets/stylesheets/administrate-materialize-theme/components/_global.scss +769 -0
  48. data/app/assets/stylesheets/administrate-materialize-theme/components/_grid.scss +156 -0
  49. data/app/assets/stylesheets/administrate-materialize-theme/components/_icons-material-design.scss +5 -0
  50. data/app/assets/stylesheets/administrate-materialize-theme/components/_materialbox.scss +43 -0
  51. data/app/assets/stylesheets/administrate-materialize-theme/components/_modal.scss +94 -0
  52. data/app/assets/stylesheets/administrate-materialize-theme/components/_navbar.scss +208 -0
  53. data/app/assets/stylesheets/administrate-materialize-theme/components/_normalize.scss +447 -0
  54. data/app/assets/stylesheets/administrate-materialize-theme/components/_preloader.scss +334 -0
  55. data/app/assets/stylesheets/administrate-materialize-theme/components/_pulse.scss +34 -0
  56. data/app/assets/stylesheets/administrate-materialize-theme/components/_sidenav.scss +216 -0
  57. data/app/assets/stylesheets/administrate-materialize-theme/components/_slider.scss +92 -0
  58. data/app/assets/stylesheets/administrate-materialize-theme/components/_table_of_contents.scss +33 -0
  59. data/app/assets/stylesheets/administrate-materialize-theme/components/_tabs.scss +99 -0
  60. data/app/assets/stylesheets/administrate-materialize-theme/components/_tapTarget.scss +103 -0
  61. data/app/assets/stylesheets/administrate-materialize-theme/components/_timepicker.scss +183 -0
  62. data/app/assets/stylesheets/administrate-materialize-theme/components/_toast.scss +58 -0
  63. data/app/assets/stylesheets/administrate-materialize-theme/components/_tooltip.scss +32 -0
  64. data/app/assets/stylesheets/administrate-materialize-theme/components/_transitions.scss +13 -0
  65. data/app/assets/stylesheets/administrate-materialize-theme/components/_typography.scss +60 -0
  66. data/app/assets/stylesheets/administrate-materialize-theme/components/_variables.scss +349 -0
  67. data/app/assets/stylesheets/administrate-materialize-theme/components/_waves.scss +114 -0
  68. data/app/assets/stylesheets/administrate-materialize-theme/components/forms/_checkboxes.scss +200 -0
  69. data/app/assets/stylesheets/administrate-materialize-theme/components/forms/_file-input.scss +44 -0
  70. data/app/assets/stylesheets/administrate-materialize-theme/components/forms/_forms.scss +22 -0
  71. data/app/assets/stylesheets/administrate-materialize-theme/components/forms/_input-fields.scss +354 -0
  72. data/app/assets/stylesheets/administrate-materialize-theme/components/forms/_radio-buttons.scss +115 -0
  73. data/app/assets/stylesheets/administrate-materialize-theme/components/forms/_range.scss +161 -0
  74. data/app/assets/stylesheets/administrate-materialize-theme/components/forms/_select.scss +180 -0
  75. data/app/assets/stylesheets/administrate-materialize-theme/components/forms/_switches.scss +89 -0
  76. data/app/assets/stylesheets/administrate-materialize-theme/materialize.scss +41 -0
  77. data/app/assets/stylesheets/administrate-materialize-theme/theme.scss +200 -0
  78. data/lib/administrate-materialize-theme.rb +6 -0
  79. data/lib/administrate-materialize-theme/engine.rb +7 -0
  80. data/lib/administrate-materialize-theme/version.rb +5 -0
  81. metadata +150 -0
@@ -0,0 +1,314 @@
1
+ (function($) {
2
+ 'use strict';
3
+
4
+ let _defaults = {
5
+ onOpen: undefined,
6
+ onClose: undefined
7
+ };
8
+
9
+ /**
10
+ * @class
11
+ *
12
+ */
13
+ class TapTarget extends Component {
14
+ /**
15
+ * Construct TapTarget instance
16
+ * @constructor
17
+ * @param {Element} el
18
+ * @param {Object} options
19
+ */
20
+ constructor(el, options) {
21
+ super(TapTarget, el, options);
22
+
23
+ this.el.M_TapTarget = this;
24
+
25
+ /**
26
+ * Options for the select
27
+ * @member TapTarget#options
28
+ * @prop {Function} onOpen - Callback function called when feature discovery is opened
29
+ * @prop {Function} onClose - Callback function called when feature discovery is closed
30
+ */
31
+ this.options = $.extend({}, TapTarget.defaults, options);
32
+
33
+ this.isOpen = false;
34
+
35
+ // setup
36
+ this.$origin = $('#' + this.$el.attr('data-target'));
37
+ this._setup();
38
+
39
+ this._calculatePositioning();
40
+ this._setupEventHandlers();
41
+ }
42
+
43
+ static get defaults() {
44
+ return _defaults;
45
+ }
46
+
47
+ static init(els, options) {
48
+ return super.init(this, els, options);
49
+ }
50
+
51
+ /**
52
+ * Get Instance
53
+ */
54
+ static getInstance(el) {
55
+ let domElem = !!el.jquery ? el[0] : el;
56
+ return domElem.M_TapTarget;
57
+ }
58
+
59
+ /**
60
+ * Teardown component
61
+ */
62
+ destroy() {
63
+ this._removeEventHandlers();
64
+ this.el.TapTarget = undefined;
65
+ }
66
+
67
+ /**
68
+ * Setup Event Handlers
69
+ */
70
+ _setupEventHandlers() {
71
+ this._handleDocumentClickBound = this._handleDocumentClick.bind(this);
72
+ this._handleTargetClickBound = this._handleTargetClick.bind(this);
73
+ this._handleOriginClickBound = this._handleOriginClick.bind(this);
74
+
75
+ this.el.addEventListener('click', this._handleTargetClickBound);
76
+ this.originEl.addEventListener('click', this._handleOriginClickBound);
77
+
78
+ // Resize
79
+ let throttledResize = M.throttle(this._handleResize, 200);
80
+ this._handleThrottledResizeBound = throttledResize.bind(this);
81
+
82
+ window.addEventListener('resize', this._handleThrottledResizeBound);
83
+ }
84
+
85
+ /**
86
+ * Remove Event Handlers
87
+ */
88
+ _removeEventHandlers() {
89
+ this.el.removeEventListener('click', this._handleTargetClickBound);
90
+ this.originEl.removeEventListener('click', this._handleOriginClickBound);
91
+ window.removeEventListener('resize', this._handleThrottledResizeBound);
92
+ }
93
+
94
+ /**
95
+ * Handle Target Click
96
+ * @param {Event} e
97
+ */
98
+ _handleTargetClick(e) {
99
+ this.open();
100
+ }
101
+
102
+ /**
103
+ * Handle Origin Click
104
+ * @param {Event} e
105
+ */
106
+ _handleOriginClick(e) {
107
+ this.close();
108
+ }
109
+
110
+ /**
111
+ * Handle Resize
112
+ * @param {Event} e
113
+ */
114
+ _handleResize(e) {
115
+ this._calculatePositioning();
116
+ }
117
+
118
+ /**
119
+ * Handle Resize
120
+ * @param {Event} e
121
+ */
122
+ _handleDocumentClick(e) {
123
+ if (!$(e.target).closest('.tap-target-wrapper').length) {
124
+ this.close();
125
+ e.preventDefault();
126
+ e.stopPropagation();
127
+ }
128
+ }
129
+
130
+ /**
131
+ * Setup Tap Target
132
+ */
133
+ _setup() {
134
+ // Creating tap target
135
+ this.wrapper = this.$el.parent()[0];
136
+ this.waveEl = $(this.wrapper).find('.tap-target-wave')[0];
137
+ this.originEl = $(this.wrapper).find('.tap-target-origin')[0];
138
+ this.contentEl = this.$el.find('.tap-target-content')[0];
139
+
140
+ // Creating wrapper
141
+ if (!$(this.wrapper).hasClass('.tap-target-wrapper')) {
142
+ this.wrapper = document.createElement('div');
143
+ this.wrapper.classList.add('tap-target-wrapper');
144
+ this.$el.before($(this.wrapper));
145
+ this.wrapper.append(this.el);
146
+ }
147
+
148
+ // Creating content
149
+ if (!this.contentEl) {
150
+ this.contentEl = document.createElement('div');
151
+ this.contentEl.classList.add('tap-target-content');
152
+ this.$el.append(this.contentEl);
153
+ }
154
+
155
+ // Creating foreground wave
156
+ if (!this.waveEl) {
157
+ this.waveEl = document.createElement('div');
158
+ this.waveEl.classList.add('tap-target-wave');
159
+
160
+ // Creating origin
161
+ if (!this.originEl) {
162
+ this.originEl = this.$origin.clone(true, true);
163
+ this.originEl.addClass('tap-target-origin');
164
+ this.originEl.removeAttr('id');
165
+ this.originEl.removeAttr('style');
166
+ this.originEl = this.originEl[0];
167
+ this.waveEl.append(this.originEl);
168
+ }
169
+
170
+ this.wrapper.append(this.waveEl);
171
+ }
172
+ }
173
+
174
+ /**
175
+ * Calculate positioning
176
+ */
177
+ _calculatePositioning() {
178
+ // Element or parent is fixed position?
179
+ let isFixed = this.$origin.css('position') === 'fixed';
180
+ if (!isFixed) {
181
+ let parents = this.$origin.parents();
182
+ for (let i = 0; i < parents.length; i++) {
183
+ isFixed = $(parents[i]).css('position') == 'fixed';
184
+ if (isFixed) {
185
+ break;
186
+ }
187
+ }
188
+ }
189
+
190
+ // Calculating origin
191
+ let originWidth = this.$origin.outerWidth();
192
+ let originHeight = this.$origin.outerHeight();
193
+ let originTop = isFixed
194
+ ? this.$origin.offset().top - M.getDocumentScrollTop()
195
+ : this.$origin.offset().top;
196
+ let originLeft = isFixed
197
+ ? this.$origin.offset().left - M.getDocumentScrollLeft()
198
+ : this.$origin.offset().left;
199
+
200
+ // Calculating screen
201
+ let windowWidth = window.innerWidth;
202
+ let windowHeight = window.innerHeight;
203
+ let centerX = windowWidth / 2;
204
+ let centerY = windowHeight / 2;
205
+ let isLeft = originLeft <= centerX;
206
+ let isRight = originLeft > centerX;
207
+ let isTop = originTop <= centerY;
208
+ let isBottom = originTop > centerY;
209
+ let isCenterX = originLeft >= windowWidth * 0.25 && originLeft <= windowWidth * 0.75;
210
+
211
+ // Calculating tap target
212
+ let tapTargetWidth = this.$el.outerWidth();
213
+ let tapTargetHeight = this.$el.outerHeight();
214
+ let tapTargetTop = originTop + originHeight / 2 - tapTargetHeight / 2;
215
+ let tapTargetLeft = originLeft + originWidth / 2 - tapTargetWidth / 2;
216
+ let tapTargetPosition = isFixed ? 'fixed' : 'absolute';
217
+
218
+ // Calculating content
219
+ let tapTargetTextWidth = isCenterX ? tapTargetWidth : tapTargetWidth / 2 + originWidth;
220
+ let tapTargetTextHeight = tapTargetHeight / 2;
221
+ let tapTargetTextTop = isTop ? tapTargetHeight / 2 : 0;
222
+ let tapTargetTextBottom = 0;
223
+ let tapTargetTextLeft = isLeft && !isCenterX ? tapTargetWidth / 2 - originWidth : 0;
224
+ let tapTargetTextRight = 0;
225
+ let tapTargetTextPadding = originWidth;
226
+ let tapTargetTextAlign = isBottom ? 'bottom' : 'top';
227
+
228
+ // Calculating wave
229
+ let tapTargetWaveWidth = originWidth > originHeight ? originWidth * 2 : originWidth * 2;
230
+ let tapTargetWaveHeight = tapTargetWaveWidth;
231
+ let tapTargetWaveTop = tapTargetHeight / 2 - tapTargetWaveHeight / 2;
232
+ let tapTargetWaveLeft = tapTargetWidth / 2 - tapTargetWaveWidth / 2;
233
+
234
+ // Setting tap target
235
+ let tapTargetWrapperCssObj = {};
236
+ tapTargetWrapperCssObj.top = isTop ? tapTargetTop + 'px' : '';
237
+ tapTargetWrapperCssObj.right = isRight
238
+ ? windowWidth - tapTargetLeft - tapTargetWidth + 'px'
239
+ : '';
240
+ tapTargetWrapperCssObj.bottom = isBottom
241
+ ? windowHeight - tapTargetTop - tapTargetHeight + 'px'
242
+ : '';
243
+ tapTargetWrapperCssObj.left = isLeft ? tapTargetLeft + 'px' : '';
244
+ tapTargetWrapperCssObj.position = tapTargetPosition;
245
+ $(this.wrapper).css(tapTargetWrapperCssObj);
246
+
247
+ // Setting content
248
+ $(this.contentEl).css({
249
+ width: tapTargetTextWidth + 'px',
250
+ height: tapTargetTextHeight + 'px',
251
+ top: tapTargetTextTop + 'px',
252
+ right: tapTargetTextRight + 'px',
253
+ bottom: tapTargetTextBottom + 'px',
254
+ left: tapTargetTextLeft + 'px',
255
+ padding: tapTargetTextPadding + 'px',
256
+ verticalAlign: tapTargetTextAlign
257
+ });
258
+
259
+ // Setting wave
260
+ $(this.waveEl).css({
261
+ top: tapTargetWaveTop + 'px',
262
+ left: tapTargetWaveLeft + 'px',
263
+ width: tapTargetWaveWidth + 'px',
264
+ height: tapTargetWaveHeight + 'px'
265
+ });
266
+ }
267
+
268
+ /**
269
+ * Open TapTarget
270
+ */
271
+ open() {
272
+ if (this.isOpen) {
273
+ return;
274
+ }
275
+
276
+ // onOpen callback
277
+ if (typeof this.options.onOpen === 'function') {
278
+ this.options.onOpen.call(this, this.$origin[0]);
279
+ }
280
+
281
+ this.isOpen = true;
282
+ this.wrapper.classList.add('open');
283
+
284
+ document.body.addEventListener('click', this._handleDocumentClickBound, true);
285
+ document.body.addEventListener('touchend', this._handleDocumentClickBound);
286
+ }
287
+
288
+ /**
289
+ * Close Tap Target
290
+ */
291
+ close() {
292
+ if (!this.isOpen) {
293
+ return;
294
+ }
295
+
296
+ // onClose callback
297
+ if (typeof this.options.onClose === 'function') {
298
+ this.options.onClose.call(this, this.$origin[0]);
299
+ }
300
+
301
+ this.isOpen = false;
302
+ this.wrapper.classList.remove('open');
303
+
304
+ document.body.removeEventListener('click', this._handleDocumentClickBound, true);
305
+ document.body.removeEventListener('touchend', this._handleDocumentClickBound);
306
+ }
307
+ }
308
+
309
+ M.TapTarget = TapTarget;
310
+
311
+ if (M.jQueryLoaded) {
312
+ M.initializeJqueryWrapper(TapTarget, 'tapTarget', 'M_TapTarget');
313
+ }
314
+ })(cash);
@@ -0,0 +1,6 @@
1
+ //= require ./bin/materialize
2
+
3
+ // --- on ready ----------------------------------------------------------------
4
+ document.addEventListener('DOMContentLoaded', () => {
5
+ M.AutoInit();
6
+ });
@@ -0,0 +1,647 @@
1
+ (function($) {
2
+ 'use strict';
3
+
4
+ let _defaults = {
5
+ dialRadius: 135,
6
+ outerRadius: 105,
7
+ innerRadius: 70,
8
+ tickRadius: 20,
9
+ duration: 350,
10
+ container: null,
11
+ defaultTime: 'now', // default time, 'now' or '13:14' e.g.
12
+ fromNow: 0, // Millisecond offset from the defaultTime
13
+ showClearBtn: false,
14
+
15
+ // internationalization
16
+ i18n: {
17
+ cancel: 'Cancel',
18
+ clear: 'Clear',
19
+ done: 'Ok'
20
+ },
21
+
22
+ autoClose: false, // auto close when minute is selected
23
+ twelveHour: true, // change to 12 hour AM/PM clock from 24 hour
24
+ vibrate: true, // vibrate the device when dragging clock hand
25
+
26
+ // Callbacks
27
+ onOpenStart: null,
28
+ onOpenEnd: null,
29
+ onCloseStart: null,
30
+ onCloseEnd: null,
31
+ onSelect: null
32
+ };
33
+
34
+ /**
35
+ * @class
36
+ *
37
+ */
38
+ class Timepicker extends Component {
39
+ constructor(el, options) {
40
+ super(Timepicker, el, options);
41
+
42
+ this.el.M_Timepicker = this;
43
+
44
+ this.options = $.extend({}, Timepicker.defaults, options);
45
+
46
+ this.id = M.guid();
47
+ this._insertHTMLIntoDOM();
48
+ this._setupModal();
49
+ this._setupVariables();
50
+ this._setupEventHandlers();
51
+
52
+ this._clockSetup();
53
+ this._pickerSetup();
54
+ }
55
+
56
+ static get defaults() {
57
+ return _defaults;
58
+ }
59
+
60
+ static init(els, options) {
61
+ return super.init(this, els, options);
62
+ }
63
+
64
+ static _addLeadingZero(num) {
65
+ return (num < 10 ? '0' : '') + num;
66
+ }
67
+
68
+ static _createSVGEl(name) {
69
+ let svgNS = 'http://www.w3.org/2000/svg';
70
+ return document.createElementNS(svgNS, name);
71
+ }
72
+
73
+ /**
74
+ * @typedef {Object} Point
75
+ * @property {number} x The X Coordinate
76
+ * @property {number} y The Y Coordinate
77
+ */
78
+
79
+ /**
80
+ * Get x position of mouse or touch event
81
+ * @param {Event} e
82
+ * @return {Point} x and y location
83
+ */
84
+ static _Pos(e) {
85
+ if (e.targetTouches && e.targetTouches.length >= 1) {
86
+ return { x: e.targetTouches[0].clientX, y: e.targetTouches[0].clientY };
87
+ }
88
+ // mouse event
89
+ return { x: e.clientX, y: e.clientY };
90
+ }
91
+
92
+ /**
93
+ * Get Instance
94
+ */
95
+ static getInstance(el) {
96
+ let domElem = !!el.jquery ? el[0] : el;
97
+ return domElem.M_Timepicker;
98
+ }
99
+
100
+ /**
101
+ * Teardown component
102
+ */
103
+ destroy() {
104
+ this._removeEventHandlers();
105
+ this.modal.destroy();
106
+ $(this.modalEl).remove();
107
+ this.el.M_Timepicker = undefined;
108
+ }
109
+
110
+ /**
111
+ * Setup Event Handlers
112
+ */
113
+ _setupEventHandlers() {
114
+ this._handleInputKeydownBound = this._handleInputKeydown.bind(this);
115
+ this._handleInputClickBound = this._handleInputClick.bind(this);
116
+ this._handleClockClickStartBound = this._handleClockClickStart.bind(this);
117
+ this._handleDocumentClickMoveBound = this._handleDocumentClickMove.bind(this);
118
+ this._handleDocumentClickEndBound = this._handleDocumentClickEnd.bind(this);
119
+
120
+ this.el.addEventListener('click', this._handleInputClickBound);
121
+ this.el.addEventListener('keydown', this._handleInputKeydownBound);
122
+ this.plate.addEventListener('mousedown', this._handleClockClickStartBound);
123
+ this.plate.addEventListener('touchstart', this._handleClockClickStartBound);
124
+
125
+ $(this.spanHours).on('click', this.showView.bind(this, 'hours'));
126
+ $(this.spanMinutes).on('click', this.showView.bind(this, 'minutes'));
127
+ }
128
+
129
+ _removeEventHandlers() {
130
+ this.el.removeEventListener('click', this._handleInputClickBound);
131
+ this.el.removeEventListener('keydown', this._handleInputKeydownBound);
132
+ }
133
+
134
+ _handleInputClick() {
135
+ this.open();
136
+ }
137
+
138
+ _handleInputKeydown(e) {
139
+ if (e.which === M.keys.ENTER) {
140
+ e.preventDefault();
141
+ this.open();
142
+ }
143
+ }
144
+
145
+ _handleClockClickStart(e) {
146
+ e.preventDefault();
147
+ let clockPlateBR = this.plate.getBoundingClientRect();
148
+ let offset = { x: clockPlateBR.left, y: clockPlateBR.top };
149
+
150
+ this.x0 = offset.x + this.options.dialRadius;
151
+ this.y0 = offset.y + this.options.dialRadius;
152
+ this.moved = false;
153
+ let clickPos = Timepicker._Pos(e);
154
+ this.dx = clickPos.x - this.x0;
155
+ this.dy = clickPos.y - this.y0;
156
+
157
+ // Set clock hands
158
+ this.setHand(this.dx, this.dy, false);
159
+
160
+ // Mousemove on document
161
+ document.addEventListener('mousemove', this._handleDocumentClickMoveBound);
162
+ document.addEventListener('touchmove', this._handleDocumentClickMoveBound);
163
+
164
+ // Mouseup on document
165
+ document.addEventListener('mouseup', this._handleDocumentClickEndBound);
166
+ document.addEventListener('touchend', this._handleDocumentClickEndBound);
167
+ }
168
+
169
+ _handleDocumentClickMove(e) {
170
+ e.preventDefault();
171
+ let clickPos = Timepicker._Pos(e);
172
+ let x = clickPos.x - this.x0;
173
+ let y = clickPos.y - this.y0;
174
+ this.moved = true;
175
+ this.setHand(x, y, false, true);
176
+ }
177
+
178
+ _handleDocumentClickEnd(e) {
179
+ e.preventDefault();
180
+ document.removeEventListener('mouseup', this._handleDocumentClickEndBound);
181
+ document.removeEventListener('touchend', this._handleDocumentClickEndBound);
182
+ let clickPos = Timepicker._Pos(e);
183
+ let x = clickPos.x - this.x0;
184
+ let y = clickPos.y - this.y0;
185
+ if (this.moved && x === this.dx && y === this.dy) {
186
+ this.setHand(x, y);
187
+ }
188
+
189
+ if (this.currentView === 'hours') {
190
+ this.showView('minutes', this.options.duration / 2);
191
+ } else if (this.options.autoClose) {
192
+ $(this.minutesView).addClass('timepicker-dial-out');
193
+ setTimeout(() => {
194
+ this.done();
195
+ }, this.options.duration / 2);
196
+ }
197
+
198
+ if (typeof this.options.onSelect === 'function') {
199
+ this.options.onSelect.call(this, this.hours, this.minutes);
200
+ }
201
+
202
+ // Unbind mousemove event
203
+ document.removeEventListener('mousemove', this._handleDocumentClickMoveBound);
204
+ document.removeEventListener('touchmove', this._handleDocumentClickMoveBound);
205
+ }
206
+
207
+ _insertHTMLIntoDOM() {
208
+ this.$modalEl = $(Timepicker._template);
209
+ this.modalEl = this.$modalEl[0];
210
+ this.modalEl.id = 'modal-' + this.id;
211
+
212
+ // Append popover to input by default
213
+ let containerEl = document.querySelector(this.options.container);
214
+ if (this.options.container && !!containerEl) {
215
+ this.$modalEl.appendTo(containerEl);
216
+ } else {
217
+ this.$modalEl.insertBefore(this.el);
218
+ }
219
+ }
220
+
221
+ _setupModal() {
222
+ this.modal = M.Modal.init(this.modalEl, {
223
+ onOpenStart: this.options.onOpenStart,
224
+ onOpenEnd: this.options.onOpenEnd,
225
+ onCloseStart: this.options.onCloseStart,
226
+ onCloseEnd: () => {
227
+ if (typeof this.options.onCloseEnd === 'function') {
228
+ this.options.onCloseEnd.call(this);
229
+ }
230
+ this.isOpen = false;
231
+ }
232
+ });
233
+ }
234
+
235
+ _setupVariables() {
236
+ this.currentView = 'hours';
237
+ this.vibrate = navigator.vibrate
238
+ ? 'vibrate'
239
+ : navigator.webkitVibrate
240
+ ? 'webkitVibrate'
241
+ : null;
242
+
243
+ this._canvas = this.modalEl.querySelector('.timepicker-canvas');
244
+ this.plate = this.modalEl.querySelector('.timepicker-plate');
245
+
246
+ this.hoursView = this.modalEl.querySelector('.timepicker-hours');
247
+ this.minutesView = this.modalEl.querySelector('.timepicker-minutes');
248
+ this.spanHours = this.modalEl.querySelector('.timepicker-span-hours');
249
+ this.spanMinutes = this.modalEl.querySelector('.timepicker-span-minutes');
250
+ this.spanAmPm = this.modalEl.querySelector('.timepicker-span-am-pm');
251
+ this.footer = this.modalEl.querySelector('.timepicker-footer');
252
+ this.amOrPm = 'PM';
253
+ }
254
+
255
+ _pickerSetup() {
256
+ let $clearBtn = $(
257
+ `<button class="btn-flat timepicker-clear waves-effect" style="visibility: hidden;" type="button" tabindex="${
258
+ this.options.twelveHour ? '3' : '1'
259
+ }">${this.options.i18n.clear}</button>`
260
+ )
261
+ .appendTo(this.footer)
262
+ .on('click', this.clear.bind(this));
263
+ if (this.options.showClearBtn) {
264
+ $clearBtn.css({ visibility: '' });
265
+ }
266
+
267
+ let confirmationBtnsContainer = $('<div class="confirmation-btns"></div>');
268
+ $(
269
+ '<button class="btn-flat timepicker-close waves-effect" type="button" tabindex="' +
270
+ (this.options.twelveHour ? '3' : '1') +
271
+ '">' +
272
+ this.options.i18n.cancel +
273
+ '</button>'
274
+ )
275
+ .appendTo(confirmationBtnsContainer)
276
+ .on('click', this.close.bind(this));
277
+ $(
278
+ '<button class="btn-flat timepicker-close waves-effect" type="button" tabindex="' +
279
+ (this.options.twelveHour ? '3' : '1') +
280
+ '">' +
281
+ this.options.i18n.done +
282
+ '</button>'
283
+ )
284
+ .appendTo(confirmationBtnsContainer)
285
+ .on('click', this.done.bind(this));
286
+ confirmationBtnsContainer.appendTo(this.footer);
287
+ }
288
+
289
+ _clockSetup() {
290
+ if (this.options.twelveHour) {
291
+ this.$amBtn = $('<div class="am-btn">AM</div>');
292
+ this.$pmBtn = $('<div class="pm-btn">PM</div>');
293
+ this.$amBtn.on('click', this._handleAmPmClick.bind(this)).appendTo(this.spanAmPm);
294
+ this.$pmBtn.on('click', this._handleAmPmClick.bind(this)).appendTo(this.spanAmPm);
295
+ }
296
+
297
+ this._buildHoursView();
298
+ this._buildMinutesView();
299
+ this._buildSVGClock();
300
+ }
301
+
302
+ _buildSVGClock() {
303
+ // Draw clock hands and others
304
+ let dialRadius = this.options.dialRadius;
305
+ let tickRadius = this.options.tickRadius;
306
+ let diameter = dialRadius * 2;
307
+
308
+ let svg = Timepicker._createSVGEl('svg');
309
+ svg.setAttribute('class', 'timepicker-svg');
310
+ svg.setAttribute('width', diameter);
311
+ svg.setAttribute('height', diameter);
312
+ let g = Timepicker._createSVGEl('g');
313
+ g.setAttribute('transform', 'translate(' + dialRadius + ',' + dialRadius + ')');
314
+ let bearing = Timepicker._createSVGEl('circle');
315
+ bearing.setAttribute('class', 'timepicker-canvas-bearing');
316
+ bearing.setAttribute('cx', 0);
317
+ bearing.setAttribute('cy', 0);
318
+ bearing.setAttribute('r', 4);
319
+ let hand = Timepicker._createSVGEl('line');
320
+ hand.setAttribute('x1', 0);
321
+ hand.setAttribute('y1', 0);
322
+ let bg = Timepicker._createSVGEl('circle');
323
+ bg.setAttribute('class', 'timepicker-canvas-bg');
324
+ bg.setAttribute('r', tickRadius);
325
+ g.appendChild(hand);
326
+ g.appendChild(bg);
327
+ g.appendChild(bearing);
328
+ svg.appendChild(g);
329
+ this._canvas.appendChild(svg);
330
+
331
+ this.hand = hand;
332
+ this.bg = bg;
333
+ this.bearing = bearing;
334
+ this.g = g;
335
+ }
336
+
337
+ _buildHoursView() {
338
+ let $tick = $('<div class="timepicker-tick"></div>');
339
+ // Hours view
340
+ if (this.options.twelveHour) {
341
+ for (let i = 1; i < 13; i += 1) {
342
+ let tick = $tick.clone();
343
+ let radian = i / 6 * Math.PI;
344
+ let radius = this.options.outerRadius;
345
+ tick.css({
346
+ left:
347
+ this.options.dialRadius + Math.sin(radian) * radius - this.options.tickRadius + 'px',
348
+ top:
349
+ this.options.dialRadius - Math.cos(radian) * radius - this.options.tickRadius + 'px'
350
+ });
351
+ tick.html(i === 0 ? '00' : i);
352
+ this.hoursView.appendChild(tick[0]);
353
+ // tick.on(mousedownEvent, mousedown);
354
+ }
355
+ } else {
356
+ for (let i = 0; i < 24; i += 1) {
357
+ let tick = $tick.clone();
358
+ let radian = i / 6 * Math.PI;
359
+ let inner = i > 0 && i < 13;
360
+ let radius = inner ? this.options.innerRadius : this.options.outerRadius;
361
+ tick.css({
362
+ left:
363
+ this.options.dialRadius + Math.sin(radian) * radius - this.options.tickRadius + 'px',
364
+ top:
365
+ this.options.dialRadius - Math.cos(radian) * radius - this.options.tickRadius + 'px'
366
+ });
367
+ tick.html(i === 0 ? '00' : i);
368
+ this.hoursView.appendChild(tick[0]);
369
+ // tick.on(mousedownEvent, mousedown);
370
+ }
371
+ }
372
+ }
373
+
374
+ _buildMinutesView() {
375
+ let $tick = $('<div class="timepicker-tick"></div>');
376
+ // Minutes view
377
+ for (let i = 0; i < 60; i += 5) {
378
+ let tick = $tick.clone();
379
+ let radian = i / 30 * Math.PI;
380
+ tick.css({
381
+ left:
382
+ this.options.dialRadius +
383
+ Math.sin(radian) * this.options.outerRadius -
384
+ this.options.tickRadius +
385
+ 'px',
386
+ top:
387
+ this.options.dialRadius -
388
+ Math.cos(radian) * this.options.outerRadius -
389
+ this.options.tickRadius +
390
+ 'px'
391
+ });
392
+ tick.html(Timepicker._addLeadingZero(i));
393
+ this.minutesView.appendChild(tick[0]);
394
+ }
395
+ }
396
+
397
+ _handleAmPmClick(e) {
398
+ let $btnClicked = $(e.target);
399
+ this.amOrPm = $btnClicked.hasClass('am-btn') ? 'AM' : 'PM';
400
+ this._updateAmPmView();
401
+ }
402
+
403
+ _updateAmPmView() {
404
+ if (this.options.twelveHour) {
405
+ this.$amBtn.toggleClass('text-primary', this.amOrPm === 'AM');
406
+ this.$pmBtn.toggleClass('text-primary', this.amOrPm === 'PM');
407
+ }
408
+ }
409
+
410
+ _updateTimeFromInput() {
411
+ // Get the time
412
+ let value = ((this.el.value || this.options.defaultTime || '') + '').split(':');
413
+ if (this.options.twelveHour && !(typeof value[1] === 'undefined')) {
414
+ if (value[1].toUpperCase().indexOf('AM') > 0) {
415
+ this.amOrPm = 'AM';
416
+ } else {
417
+ this.amOrPm = 'PM';
418
+ }
419
+ value[1] = value[1].replace('AM', '').replace('PM', '');
420
+ }
421
+ if (value[0] === 'now') {
422
+ let now = new Date(+new Date() + this.options.fromNow);
423
+ value = [now.getHours(), now.getMinutes()];
424
+ if (this.options.twelveHour) {
425
+ this.amOrPm = value[0] >= 12 && value[0] < 24 ? 'PM' : 'AM';
426
+ }
427
+ }
428
+ this.hours = +value[0] || 0;
429
+ this.minutes = +value[1] || 0;
430
+ this.spanHours.innerHTML = this.hours;
431
+ this.spanMinutes.innerHTML = Timepicker._addLeadingZero(this.minutes);
432
+
433
+ this._updateAmPmView();
434
+ }
435
+
436
+ showView(view, delay) {
437
+ if (view === 'minutes' && $(this.hoursView).css('visibility') === 'visible') {
438
+ // raiseCallback(this.options.beforeHourSelect);
439
+ }
440
+ let isHours = view === 'hours',
441
+ nextView = isHours ? this.hoursView : this.minutesView,
442
+ hideView = isHours ? this.minutesView : this.hoursView;
443
+ this.currentView = view;
444
+
445
+ $(this.spanHours).toggleClass('text-primary', isHours);
446
+ $(this.spanMinutes).toggleClass('text-primary', !isHours);
447
+
448
+ // Transition view
449
+ hideView.classList.add('timepicker-dial-out');
450
+ $(nextView)
451
+ .css('visibility', 'visible')
452
+ .removeClass('timepicker-dial-out');
453
+
454
+ // Reset clock hand
455
+ this.resetClock(delay);
456
+
457
+ // After transitions ended
458
+ clearTimeout(this.toggleViewTimer);
459
+ this.toggleViewTimer = setTimeout(() => {
460
+ $(hideView).css('visibility', 'hidden');
461
+ }, this.options.duration);
462
+ }
463
+
464
+ resetClock(delay) {
465
+ let view = this.currentView,
466
+ value = this[view],
467
+ isHours = view === 'hours',
468
+ unit = Math.PI / (isHours ? 6 : 30),
469
+ radian = value * unit,
470
+ radius =
471
+ isHours && value > 0 && value < 13 ? this.options.innerRadius : this.options.outerRadius,
472
+ x = Math.sin(radian) * radius,
473
+ y = -Math.cos(radian) * radius,
474
+ self = this;
475
+
476
+ if (delay) {
477
+ $(this.canvas).addClass('timepicker-canvas-out');
478
+ setTimeout(() => {
479
+ $(self.canvas).removeClass('timepicker-canvas-out');
480
+ self.setHand(x, y);
481
+ }, delay);
482
+ } else {
483
+ this.setHand(x, y);
484
+ }
485
+ }
486
+
487
+ setHand(x, y, roundBy5) {
488
+ let radian = Math.atan2(x, -y),
489
+ isHours = this.currentView === 'hours',
490
+ unit = Math.PI / (isHours || roundBy5 ? 6 : 30),
491
+ z = Math.sqrt(x * x + y * y),
492
+ inner = isHours && z < (this.options.outerRadius + this.options.innerRadius) / 2,
493
+ radius = inner ? this.options.innerRadius : this.options.outerRadius;
494
+
495
+ if (this.options.twelveHour) {
496
+ radius = this.options.outerRadius;
497
+ }
498
+
499
+ // Radian should in range [0, 2PI]
500
+ if (radian < 0) {
501
+ radian = Math.PI * 2 + radian;
502
+ }
503
+
504
+ // Get the round value
505
+ let value = Math.round(radian / unit);
506
+
507
+ // Get the round radian
508
+ radian = value * unit;
509
+
510
+ // Correct the hours or minutes
511
+ if (this.options.twelveHour) {
512
+ if (isHours) {
513
+ if (value === 0) value = 12;
514
+ } else {
515
+ if (roundBy5) value *= 5;
516
+ if (value === 60) value = 0;
517
+ }
518
+ } else {
519
+ if (isHours) {
520
+ if (value === 12) {
521
+ value = 0;
522
+ }
523
+ value = inner ? (value === 0 ? 12 : value) : value === 0 ? 0 : value + 12;
524
+ } else {
525
+ if (roundBy5) {
526
+ value *= 5;
527
+ }
528
+ if (value === 60) {
529
+ value = 0;
530
+ }
531
+ }
532
+ }
533
+
534
+ // Once hours or minutes changed, vibrate the device
535
+ if (this[this.currentView] !== value) {
536
+ if (this.vibrate && this.options.vibrate) {
537
+ // Do not vibrate too frequently
538
+ if (!this.vibrateTimer) {
539
+ navigator[this.vibrate](10);
540
+ this.vibrateTimer = setTimeout(() => {
541
+ this.vibrateTimer = null;
542
+ }, 100);
543
+ }
544
+ }
545
+ }
546
+
547
+ this[this.currentView] = value;
548
+ if (isHours) {
549
+ this['spanHours'].innerHTML = value;
550
+ } else {
551
+ this['spanMinutes'].innerHTML = Timepicker._addLeadingZero(value);
552
+ }
553
+
554
+ // Set clock hand and others' position
555
+ let cx1 = Math.sin(radian) * (radius - this.options.tickRadius),
556
+ cy1 = -Math.cos(radian) * (radius - this.options.tickRadius),
557
+ cx2 = Math.sin(radian) * radius,
558
+ cy2 = -Math.cos(radian) * radius;
559
+ this.hand.setAttribute('x2', cx1);
560
+ this.hand.setAttribute('y2', cy1);
561
+ this.bg.setAttribute('cx', cx2);
562
+ this.bg.setAttribute('cy', cy2);
563
+ }
564
+
565
+ open() {
566
+ if (this.isOpen) {
567
+ return;
568
+ }
569
+
570
+ this.isOpen = true;
571
+ this._updateTimeFromInput();
572
+ this.showView('hours');
573
+
574
+ this.modal.open();
575
+ }
576
+
577
+ close() {
578
+ if (!this.isOpen) {
579
+ return;
580
+ }
581
+
582
+ this.isOpen = false;
583
+ this.modal.close();
584
+ }
585
+
586
+ /**
587
+ * Finish timepicker selection.
588
+ */
589
+ done(e, clearValue) {
590
+ // Set input value
591
+ let last = this.el.value;
592
+ let value = clearValue
593
+ ? ''
594
+ : Timepicker._addLeadingZero(this.hours) + ':' + Timepicker._addLeadingZero(this.minutes);
595
+ this.time = value;
596
+ if (!clearValue && this.options.twelveHour) {
597
+ value = `${value} ${this.amOrPm}`;
598
+ }
599
+ this.el.value = value;
600
+
601
+ // Trigger change event
602
+ if (value !== last) {
603
+ this.$el.trigger('change');
604
+ }
605
+
606
+ this.close();
607
+ this.el.focus();
608
+ }
609
+
610
+ clear() {
611
+ this.done(null, true);
612
+ }
613
+ }
614
+
615
+ Timepicker._template = [
616
+ '<div class= "modal timepicker-modal">',
617
+ '<div class="modal-content timepicker-container">',
618
+ '<div class="timepicker-digital-display">',
619
+ '<div class="timepicker-text-container">',
620
+ '<div class="timepicker-display-column">',
621
+ '<span class="timepicker-span-hours text-primary"></span>',
622
+ ':',
623
+ '<span class="timepicker-span-minutes"></span>',
624
+ '</div>',
625
+ '<div class="timepicker-display-column timepicker-display-am-pm">',
626
+ '<div class="timepicker-span-am-pm"></div>',
627
+ '</div>',
628
+ '</div>',
629
+ '</div>',
630
+ '<div class="timepicker-analog-display">',
631
+ '<div class="timepicker-plate">',
632
+ '<div class="timepicker-canvas"></div>',
633
+ '<div class="timepicker-dial timepicker-hours"></div>',
634
+ '<div class="timepicker-dial timepicker-minutes timepicker-dial-out"></div>',
635
+ '</div>',
636
+ '<div class="timepicker-footer"></div>',
637
+ '</div>',
638
+ '</div>',
639
+ '</div>'
640
+ ].join('');
641
+
642
+ M.Timepicker = Timepicker;
643
+
644
+ if (M.jQueryLoaded) {
645
+ M.initializeJqueryWrapper(Timepicker, 'timepicker', 'M_Timepicker');
646
+ }
647
+ })(cash);