taskwarrior-web 1.0.2 → 1.0.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (47) hide show
  1. data/lib/taskwarrior-web/app.rb +4 -14
  2. data/lib/taskwarrior-web/command_builders/base.rb +5 -4
  3. data/lib/taskwarrior-web/config.rb +2 -2
  4. data/lib/taskwarrior-web/helpers.rb +5 -5
  5. data/lib/taskwarrior-web/public/css/bootstrap-responsive.min.css +9 -0
  6. data/lib/taskwarrior-web/public/css/bootstrap.min.css +9 -0
  7. data/lib/taskwarrior-web/public/css/datepicker.css +156 -0
  8. data/lib/taskwarrior-web/public/css/styles.css +3 -307
  9. data/lib/taskwarrior-web/public/{images → img}/ajax-loader.gif +0 -0
  10. data/lib/taskwarrior-web/public/img/glyphicons-halflings-white.png +0 -0
  11. data/lib/taskwarrior-web/public/img/glyphicons-halflings.png +0 -0
  12. data/lib/taskwarrior-web/public/js/application.js +22 -55
  13. data/lib/taskwarrior-web/public/js/bootstrap-datepicker.js +401 -0
  14. data/lib/taskwarrior-web/public/js/bootstrap.min.js +6 -0
  15. data/lib/taskwarrior-web/public/js/jquery.min.js +2 -16
  16. data/lib/taskwarrior-web/views/_subnav.erb +13 -0
  17. data/lib/taskwarrior-web/views/_topbar.erb +19 -0
  18. data/lib/taskwarrior-web/views/layout.erb +8 -22
  19. data/lib/taskwarrior-web/views/listing.erb +1 -1
  20. data/lib/taskwarrior-web/views/project.erb +1 -2
  21. data/lib/taskwarrior-web/views/projects.erb +3 -4
  22. data/lib/taskwarrior-web/views/task_form.erb +26 -16
  23. data/taskwarrior-web.gemspec +1 -1
  24. metadata +27 -41
  25. data/lib/taskwarrior-web/public/css/gh-buttons.css +0 -388
  26. data/lib/taskwarrior-web/public/css/jquery-ui.css +0 -738
  27. data/lib/taskwarrior-web/public/css/tipsy.css +0 -7
  28. data/lib/taskwarrior-web/public/images/arrow_right_black.png +0 -0
  29. data/lib/taskwarrior-web/public/images/arrow_right_grey.png +0 -0
  30. data/lib/taskwarrior-web/public/images/bg_fallback.png +0 -0
  31. data/lib/taskwarrior-web/public/images/gh-icons.png +0 -0
  32. data/lib/taskwarrior-web/public/images/grid-view.png +0 -0
  33. data/lib/taskwarrior-web/public/images/icon_sprite.png +0 -0
  34. data/lib/taskwarrior-web/public/images/list-view.png +0 -0
  35. data/lib/taskwarrior-web/public/images/logo.png +0 -0
  36. data/lib/taskwarrior-web/public/images/progress_bar.gif +0 -0
  37. data/lib/taskwarrior-web/public/images/slider_handles.png +0 -0
  38. data/lib/taskwarrior-web/public/images/subnav_background.gif +0 -0
  39. data/lib/taskwarrior-web/public/images/tab_background.gif +0 -0
  40. data/lib/taskwarrior-web/public/images/tipsy.gif +0 -0
  41. data/lib/taskwarrior-web/public/images/ui-icons_222222_256x240.png +0 -0
  42. data/lib/taskwarrior-web/public/images/ui-icons_454545_256x240.png +0 -0
  43. data/lib/taskwarrior-web/public/js/jquery-ui.min.js +0 -163
  44. data/lib/taskwarrior-web/public/js/jquery.cookie.js +0 -96
  45. data/lib/taskwarrior-web/public/js/jquery.tagsinput.js +0 -218
  46. data/lib/taskwarrior-web/public/js/jquery.tipsy.js +0 -104
  47. data/lib/taskwarrior-web/views/_navigation.erb +0 -17
@@ -1,45 +1,14 @@
1
1
  $(document).ready(function() {
2
- initPolling();
3
- initTooltips();
4
2
  initDatePicker();
5
3
  initAutocomplete();
6
4
  initTaskCompletion();
7
5
 
8
6
  // Fluid-specific stuff.
9
- if (typeof window.fluid !== undefined) {
10
- refreshDockBadge();
11
- }
7
+ //if (typeof window.fluid !== undefined) {
8
+ //refreshDockBadge();
9
+ //}
12
10
  });
13
11
 
14
- var initPolling = function() {
15
- var polling = $.cookie('taskwarrior-web-polling');
16
- if (polling) {
17
- var pollingInterval = startPolling();
18
- } else {
19
- $('#polling-info a').text('Start polling');
20
- }
21
-
22
- $('#polling-info a').click(function(e) {
23
- if (polling) {
24
- window.clearInterval(pollingInterval);
25
- polling = false;
26
- $(this).text('Start polling');
27
- $.cookie('taskwarrior-web-polling', null);
28
- } else {
29
- pollingInterval = startPolling();
30
- polling = true;
31
- $(this).text('Stop polling');
32
- $.cookie('taskwarrior-web-polling', true);
33
- }
34
- e.preventDefault();
35
- });
36
- };
37
-
38
- var startPolling = function() {
39
- var pollingInterval = window.setInterval('refreshPageContents()', 3000);
40
- return pollingInterval;
41
- };
42
-
43
12
  var refreshPageContents = function() {
44
13
  $.ajax({
45
14
  url: window.location,
@@ -51,29 +20,26 @@ var refreshPageContents = function() {
51
20
  });
52
21
  };
53
22
 
54
- var initTooltips = function() {
55
- $('.tooltip').tipsy({
56
- title: 'data-tooltip',
57
- gravity: 's'
58
- });
59
- };
60
-
61
23
  var initDatePicker = function() {
62
- $('.datefield input').datepicker({
63
- dateFormat: $('.datefield input').data('format'),
64
- autoSize: true
24
+ $('.date-picker').datepicker({
25
+ autoclose: true
65
26
  });
66
27
  };
67
28
 
68
29
  var initAutocomplete = function() {
69
- $('#task-project').autocomplete({
70
- source: '/ajax/projects'
30
+ $('#task-project').typeahead({
31
+ source: function (query, process) {
32
+ return $.get('/ajax/projects/', {query: query}, function (data) {
33
+ data = $.parseJSON(data);
34
+ process(data);
35
+ });
36
+ }
71
37
  });
72
38
 
73
- $('#task-tags').tagsInput({
74
- autocomplete_url: '/ajax/tags',
75
- defaultText: ''
76
- });
39
+ //$('#task-tags').tagsInput({
40
+ //autocomplete_url: '/ajax/tags',
41
+ //defaultText: ''
42
+ //});
77
43
  };
78
44
 
79
45
  var initTaskCompletion = function() {
@@ -82,17 +48,18 @@ var initTaskCompletion = function() {
82
48
  var container = $(this).parent();
83
49
  var checkbox = container.html();
84
50
  var row = $(this).closest('tr');
85
- container.html('<img src="/images/ajax-loader.gif" />');
51
+ container.html('<img src="/img/ajax-loader.gif" />');
86
52
  $.ajax({
87
53
  url: '/ajax/task-complete/' + $(this).data('task-id'),
88
54
  type: 'POST',
89
55
  success: function(data) {
56
+ refreshPageContents();
90
57
  var message = (data === '') ? 'Task marked as completed.' : data;
91
- set_message(message, 'success');
58
+ set_message(message, 'alert-success');
92
59
  row.fadeOut();
93
60
  },
94
61
  error: function(data) {
95
- set_message('There was an error when marking the task as completed.', 'error');
62
+ set_message('There was an error when marking the task as completed.', 'alert-error');
96
63
  container.html(checkbox);
97
64
  }
98
65
  });
@@ -116,11 +83,11 @@ var refreshDockBadge = function() {
116
83
 
117
84
  var refreshSubnavCount = function() {
118
85
  getCount(function(data) {
119
- $('#subnav-bar ul li:first-child a').text('Pending ('+data+')');
86
+ $('#subnav-bar ul li:first-child a span.badge').text(data);
120
87
  });
121
88
  };
122
89
 
123
90
  function set_message(msg, severity) {
124
91
  severity = severity ? severity : 'info';
125
- $('#flash-messages').append('<div class="message ' + severity + '">' + msg + '</div>');
92
+ $('#flash-messages').append('<div class="alert ' + severity + '">' + msg + '</div>');
126
93
  }
@@ -0,0 +1,401 @@
1
+ /* =========================================================
2
+ * bootstrap-datepicker.js
3
+ * http://www.eyecon.ro/bootstrap-datepicker
4
+ * =========================================================
5
+ * Copyright 2012 Stefan Petre
6
+ *
7
+ * Licensed under the Apache License, Version 2.0 (the "License");
8
+ * you may not use this file except in compliance with the License.
9
+ * You may obtain a copy of the License at
10
+ *
11
+ * http://www.apache.org/licenses/LICENSE-2.0
12
+ *
13
+ * Unless required by applicable law or agreed to in writing, software
14
+ * distributed under the License is distributed on an "AS IS" BASIS,
15
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ * See the License for the specific language governing permissions and
17
+ * limitations under the License.
18
+ * ========================================================= */
19
+
20
+ !function( $ ) {
21
+
22
+ // Picker object
23
+
24
+ var Datepicker = function(element, options){
25
+ this.element = $(element);
26
+ this.format = DPGlobal.parseFormat(options.format||this.element.data('date-format')||'mm/dd/yyyy');
27
+ this.picker = $(DPGlobal.template)
28
+ .appendTo('body')
29
+ .on({
30
+ click: $.proxy(this.click, this),
31
+ mousedown: $.proxy(this.mousedown, this)
32
+ });
33
+ this.isInput = this.element.is('input');
34
+ this.component = this.element.is('.date') ? this.element.find('.add-on') : false;
35
+
36
+ if (this.isInput) {
37
+ this.element.on({
38
+ focus: $.proxy(this.show, this),
39
+ blur: $.proxy(this.hide, this),
40
+ keyup: $.proxy(this.update, this)
41
+ });
42
+ } else {
43
+ if (this.component){
44
+ this.component.on('click', $.proxy(this.show, this));
45
+ } else {
46
+ this.element.on('click', $.proxy(this.show, this));
47
+ }
48
+ }
49
+
50
+ this.viewMode = 0;
51
+ this.weekStart = options.weekStart||this.element.data('date-weekstart')||0;
52
+ this.weekEnd = this.weekStart == 0 ? 6 : this.weekStart - 1;
53
+ this.fillDow();
54
+ this.fillMonths();
55
+ this.update();
56
+ this.showMode();
57
+ };
58
+
59
+ Datepicker.prototype = {
60
+ constructor: Datepicker,
61
+
62
+ show: function(e) {
63
+ this.picker.show();
64
+ this.height = this.component ? this.component.outerHeight() : this.element.outerHeight();
65
+ this.place();
66
+ $(window).on('resize', $.proxy(this.place, this));
67
+ if (e ) {
68
+ e.stopPropagation();
69
+ e.preventDefault();
70
+ }
71
+ if (!this.isInput) {
72
+ $(document).on('mousedown', $.proxy(this.hide, this));
73
+ }
74
+ this.element.trigger({
75
+ type: 'show',
76
+ date: this.date
77
+ });
78
+ },
79
+
80
+ hide: function(){
81
+ this.picker.hide();
82
+ $(window).off('resize', this.place);
83
+ this.viewMode = 0;
84
+ this.showMode();
85
+ if (!this.isInput) {
86
+ $(document).off('mousedown', this.hide);
87
+ }
88
+ this.setValue();
89
+ this.element.trigger({
90
+ type: 'hide',
91
+ date: this.date
92
+ });
93
+ },
94
+
95
+ setValue: function() {
96
+ var formated = DPGlobal.formatDate(this.date, this.format);
97
+ if (!this.isInput) {
98
+ if (this.component){
99
+ this.element.find('input').prop('value', formated);
100
+ }
101
+ this.element.data('date', formated);
102
+ } else {
103
+ this.element.prop('value', formated);
104
+ }
105
+ },
106
+
107
+ place: function(){
108
+ var offset = this.component ? this.component.offset() : this.element.offset();
109
+ this.picker.css({
110
+ top: offset.top + this.height,
111
+ left: offset.left
112
+ });
113
+ },
114
+
115
+ update: function(){
116
+ this.date = DPGlobal.parseDate(
117
+ this.isInput ? this.element.prop('value') : this.element.data('date'),
118
+ this.format
119
+ );
120
+ this.viewDate = new Date(this.date);
121
+ this.fill();
122
+ },
123
+
124
+ fillDow: function(){
125
+ var dowCnt = this.weekStart;
126
+ var html = '<tr>';
127
+ while (dowCnt < this.weekStart + 7) {
128
+ html += '<th class="dow">'+DPGlobal.dates.daysMin[(dowCnt++)%7]+'</th>';
129
+ }
130
+ html += '</tr>';
131
+ this.picker.find('.datepicker-days thead').append(html);
132
+ },
133
+
134
+ fillMonths: function(){
135
+ var html = '';
136
+ var i = 0
137
+ while (i < 12) {
138
+ html += '<span class="month">'+DPGlobal.dates.monthsShort[i++]+'</span>';
139
+ }
140
+ this.picker.find('.datepicker-months td').append(html);
141
+ },
142
+
143
+ fill: function() {
144
+ var d = new Date(this.viewDate),
145
+ year = d.getFullYear(),
146
+ month = d.getMonth(),
147
+ currentDate = this.date.valueOf();
148
+ this.picker.find('.datepicker-days th:eq(1)')
149
+ .text(DPGlobal.dates.months[month]+' '+year);
150
+ var prevMonth = new Date(year, month-1, 28,0,0,0,0),
151
+ day = DPGlobal.getDaysInMonth(prevMonth.getFullYear(), prevMonth.getMonth());
152
+ prevMonth.setDate(day);
153
+ prevMonth.setDate(day - (prevMonth.getDay() - this.weekStart + 7)%7);
154
+ var nextMonth = new Date(prevMonth);
155
+ nextMonth.setDate(nextMonth.getDate() + 42);
156
+ nextMonth = nextMonth.valueOf();
157
+ html = [];
158
+ var clsName;
159
+ while(prevMonth.valueOf() < nextMonth) {
160
+ if (prevMonth.getDay() == this.weekStart) {
161
+ html.push('<tr>');
162
+ }
163
+ clsName = '';
164
+ if (prevMonth.getMonth() < month) {
165
+ clsName += ' old';
166
+ } else if (prevMonth.getMonth() > month) {
167
+ clsName += ' new';
168
+ }
169
+ if (prevMonth.valueOf() == currentDate) {
170
+ clsName += ' active';
171
+ }
172
+ html.push('<td class="day'+clsName+'">'+prevMonth.getDate() + '</td>');
173
+ if (prevMonth.getDay() == this.weekEnd) {
174
+ html.push('</tr>');
175
+ }
176
+ prevMonth.setDate(prevMonth.getDate()+1);
177
+ }
178
+ this.picker.find('.datepicker-days tbody').empty().append(html.join(''));
179
+ var currentYear = this.date.getFullYear();
180
+
181
+ var months = this.picker.find('.datepicker-months')
182
+ .find('th:eq(1)')
183
+ .text(year)
184
+ .end()
185
+ .find('span').removeClass('active');
186
+ if (currentYear == year) {
187
+ months.eq(this.date.getMonth()).addClass('active');
188
+ }
189
+
190
+ html = '';
191
+ year = parseInt(year/10, 10) * 10;
192
+ var yearCont = this.picker.find('.datepicker-years')
193
+ .find('th:eq(1)')
194
+ .text(year + '-' + (year + 9))
195
+ .end()
196
+ .find('td');
197
+ year -= 1;
198
+ for (var i = -1; i < 11; i++) {
199
+ html += '<span class="year'+(i == -1 || i == 10 ? ' old' : '')+(currentYear == year ? ' active' : '')+'">'+year+'</span>';
200
+ year += 1;
201
+ }
202
+ yearCont.html(html);
203
+ },
204
+
205
+ click: function(e) {
206
+ e.stopPropagation();
207
+ e.preventDefault();
208
+ var target = $(e.target).closest('span, td, th');
209
+ if (target.length == 1) {
210
+ switch(target[0].nodeName.toLowerCase()) {
211
+ case 'th':
212
+ switch(target[0].className) {
213
+ case 'switch':
214
+ this.showMode(1);
215
+ break;
216
+ case 'prev':
217
+ case 'next':
218
+ this.viewDate['set'+DPGlobal.modes[this.viewMode].navFnc].call(
219
+ this.viewDate,
220
+ this.viewDate['get'+DPGlobal.modes[this.viewMode].navFnc].call(this.viewDate) +
221
+ DPGlobal.modes[this.viewMode].navStep * (target[0].className == 'prev' ? -1 : 1)
222
+ );
223
+ this.fill();
224
+ break;
225
+ }
226
+ break;
227
+ case 'span':
228
+ if (target.is('.month')) {
229
+ var month = target.parent().find('span').index(target);
230
+ this.viewDate.setMonth(month);
231
+ } else {
232
+ var year = parseInt(target.text(), 10)||0;
233
+ this.viewDate.setFullYear(year);
234
+ }
235
+ this.showMode(-1);
236
+ this.fill();
237
+ break;
238
+ case 'td':
239
+ if (target.is('.day')){
240
+ var day = parseInt(target.text(), 10)||1;
241
+ var month = this.viewDate.getMonth();
242
+ if (target.is('.old')) {
243
+ month -= 1;
244
+ } else if (target.is('.new')) {
245
+ month += 1;
246
+ }
247
+ var year = this.viewDate.getFullYear();
248
+ this.date = new Date(year, month, day,0,0,0,0);
249
+ this.viewDate = new Date(year, month, day,0,0,0,0);
250
+ this.fill();
251
+ this.setValue();
252
+ this.element.trigger({
253
+ type: 'changeDate',
254
+ date: this.date
255
+ });
256
+ }
257
+ break;
258
+ }
259
+ }
260
+ },
261
+
262
+ mousedown: function(e){
263
+ e.stopPropagation();
264
+ e.preventDefault();
265
+ },
266
+
267
+ showMode: function(dir) {
268
+ if (dir) {
269
+ this.viewMode = Math.max(0, Math.min(2, this.viewMode + dir));
270
+ }
271
+ this.picker.find('>div').hide().filter('.datepicker-'+DPGlobal.modes[this.viewMode].clsName).show();
272
+ }
273
+ };
274
+
275
+ $.fn.datepicker = function ( option ) {
276
+ return this.each(function () {
277
+ var $this = $(this),
278
+ data = $this.data('datepicker'),
279
+ options = typeof option == 'object' && option;
280
+ if (!data) {
281
+ $this.data('datepicker', (data = new Datepicker(this, $.extend({}, $.fn.datepicker.defaults,options))));
282
+ }
283
+ if (typeof option == 'string') data[option]();
284
+ });
285
+ };
286
+
287
+ $.fn.datepicker.defaults = {
288
+ };
289
+ $.fn.datepicker.Constructor = Datepicker;
290
+
291
+ var DPGlobal = {
292
+ modes: [
293
+ {
294
+ clsName: 'days',
295
+ navFnc: 'Month',
296
+ navStep: 1
297
+ },
298
+ {
299
+ clsName: 'months',
300
+ navFnc: 'FullYear',
301
+ navStep: 1
302
+ },
303
+ {
304
+ clsName: 'years',
305
+ navFnc: 'FullYear',
306
+ navStep: 10
307
+ }],
308
+ dates:{
309
+ days: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"],
310
+ daysShort: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"],
311
+ daysMin: ["Su", "Mo", "Tu", "We", "Th", "Fr", "Sa", "Su"],
312
+ months: ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"],
313
+ monthsShort: ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
314
+ },
315
+ isLeapYear: function (year) {
316
+ return (((year % 4 === 0) && (year % 100 !== 0)) || (year % 400 === 0))
317
+ },
318
+ getDaysInMonth: function (year, month) {
319
+ return [31, (DPGlobal.isLeapYear(year) ? 29 : 28), 31, 30, 31, 30, 31, 31, 30, 31, 30, 31][month]
320
+ },
321
+ parseFormat: function(format){
322
+ var separator = format.match(/[.\/-].*?/),
323
+ parts = format.split(/\W+/);
324
+ if (!separator || !parts || parts.length == 0){
325
+ throw new Error("Invalid date format.");
326
+ }
327
+ return {separator: separator, parts: parts};
328
+ },
329
+ parseDate: function(date, format) {
330
+ var parts = date.split(format.separator),
331
+ date = new Date(1970, 1, 1, 0, 0, 0),
332
+ val;
333
+ if (parts.length == format.parts.length) {
334
+ for (var i=0, cnt = format.parts.length; i < cnt; i++) {
335
+ val = parseInt(parts[i], 10)||1;
336
+ switch(format.parts[i]) {
337
+ case 'dd':
338
+ case 'd':
339
+ date.setDate(val);
340
+ break;
341
+ case 'mm':
342
+ case 'm':
343
+ date.setMonth(val - 1);
344
+ break;
345
+ case 'yy':
346
+ date.setFullYear(2000 + val);
347
+ break;
348
+ case 'yyyy':
349
+ date.setFullYear(val);
350
+ break;
351
+ }
352
+ }
353
+ }
354
+ return date;
355
+ },
356
+ formatDate: function(date, format){
357
+ var val = {
358
+ d: date.getDate(),
359
+ m: date.getMonth() + 1,
360
+ yy: date.getFullYear().toString().substring(2),
361
+ yyyy: date.getFullYear()
362
+ };
363
+ val.dd = (val.d < 10 ? '0' : '') + val.d;
364
+ val.mm = (val.m < 10 ? '0' : '') + val.m;
365
+ var date = [];
366
+ for (var i=0, cnt = format.parts.length; i < cnt; i++) {
367
+ date.push(val[format.parts[i]]);
368
+ }
369
+ return date.join(format.separator);
370
+ },
371
+ headTemplate: '<thead>'+
372
+ '<tr>'+
373
+ '<th class="prev"><i class="icon-arrow-left"/></th>'+
374
+ '<th colspan="5" class="switch"></th>'+
375
+ '<th class="next"><i class="icon-arrow-right"/></th>'+
376
+ '</tr>'+
377
+ '</thead>',
378
+ contTemplate: '<tbody><tr><td colspan="7"></td></tr></tbody>'
379
+ };
380
+ DPGlobal.template = '<div class="datepicker dropdown-menu">'+
381
+ '<div class="datepicker-days">'+
382
+ '<table class=" table-condensed">'+
383
+ DPGlobal.headTemplate+
384
+ '<tbody></tbody>'+
385
+ '</table>'+
386
+ '</div>'+
387
+ '<div class="datepicker-months">'+
388
+ '<table class="table-condensed">'+
389
+ DPGlobal.headTemplate+
390
+ DPGlobal.contTemplate+
391
+ '</table>'+
392
+ '</div>'+
393
+ '<div class="datepicker-years">'+
394
+ '<table class="table-condensed">'+
395
+ DPGlobal.headTemplate+
396
+ DPGlobal.contTemplate+
397
+ '</table>'+
398
+ '</div>'+
399
+ '</div>';
400
+
401
+ }( window.jQuery )