From 874e98b408cbeefa3481c9f22debf175a7dfbbc5 Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Mon, 9 Dec 2013 18:40:20 -0800 Subject: [PATCH 1/9] Support for persistent state for collections --- src/UI/Missing/Collection.js | 10 ++-- src/UI/Mixins/AsPersistedStateCollection.js | 53 +++++++++++++++++++++ 2 files changed, 60 insertions(+), 3 deletions(-) create mode 100644 src/UI/Mixins/AsPersistedStateCollection.js diff --git a/src/UI/Missing/Collection.js b/src/UI/Missing/Collection.js index 4b7a4fa96..d58b6d133 100644 --- a/src/UI/Missing/Collection.js +++ b/src/UI/Missing/Collection.js @@ -2,11 +2,13 @@ define( [ 'Series/EpisodeModel', - 'backbone.pageable' - ], function (EpisodeModel, PagableCollection) { - return PagableCollection.extend({ + 'backbone.pageable', + 'Mixins/AsPersistedStateCollection' + ], function (EpisodeModel, PagableCollection, AsPersistedStateCollection) { + var collection = PagableCollection.extend({ url : window.NzbDrone.ApiRoot + '/missing', model: EpisodeModel, + tableName: 'missing', state: { pageSize: 15, @@ -38,4 +40,6 @@ define( return resp; } }); + + return AsPersistedStateCollection.call(collection); }); diff --git a/src/UI/Mixins/AsPersistedStateCollection.js b/src/UI/Mixins/AsPersistedStateCollection.js new file mode 100644 index 000000000..c74e3fcd5 --- /dev/null +++ b/src/UI/Mixins/AsPersistedStateCollection.js @@ -0,0 +1,53 @@ +'use strict'; + +define( + ['Config'], + function (Config) { + + return function () { + + var originalInit = this.prototype.initialize; + + this.prototype.initialize = function () { + + if (!this.tableName) { + throw 'tableName is required'; + } + + _setState.call(this); + + this.on('backgrid:sort', _storeState, this); + + if (originalInit) { + originalInit.call(this); + } + }; + + var _setState = function () { + var key = Config.getValue('{0}.sortKey'.format(this.tableName), this.state.sortKey); + var direction = Config.getValue('{0}.sortDirection'.format(this.tableName), this.state.order); + var order = parseInt(direction, 10); + + this.state.sortKey = key; + this.state.order = order; + }; + + var _storeState = function (sortKey, sortDirection) { + var order = _convertDirectionToInt(sortDirection); + + Config.setValue('{0}.sortKey'.format(this.tableName), sortKey); + Config.setValue('{0}.sortDirection'.format(this.tableName), order); + }; + + var _convertDirectionToInt = function (dir) { + if (dir === 'ascending') { + return '-1'; + } + + return '1'; + }; + + return this; + }; + } +); From 7e20e48023085569e8ba43d97eb229941df23366 Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Tue, 10 Dec 2013 00:00:06 -0800 Subject: [PATCH 2/9] Massive backgrid update, only one header cell left --- src/UI/Cells/Header/QualityHeaderCell.js | 69 -- src/UI/Episode/Search/ManualLayout.js | 10 +- src/UI/JsLibraries/backbone.backgrid.js | 795 +++++++++++------- .../backbone.backgrid.paginator.js | 386 ++++++--- src/UI/JsLibraries/backbone.pageable.js | 82 +- .../{Collection.js => MissingCollection.js} | 0 src/UI/Missing/MissingLayout.js | 3 +- src/UI/Mixins/AsPersistedStateCollection.js | 7 +- src/UI/Series/Index/SeriesIndexLayout.js | 18 +- src/UI/Series/SeriesCollection.js | 2 +- src/UI/Shared/Grid/DateHeaderCell.js | 66 -- src/UI/Shared/Grid/HeaderCell.js | 123 +-- src/UI/app.js | 3 +- 13 files changed, 935 insertions(+), 629 deletions(-) delete mode 100644 src/UI/Cells/Header/QualityHeaderCell.js rename src/UI/Missing/{Collection.js => MissingCollection.js} (100%) delete mode 100644 src/UI/Shared/Grid/DateHeaderCell.js diff --git a/src/UI/Cells/Header/QualityHeaderCell.js b/src/UI/Cells/Header/QualityHeaderCell.js deleted file mode 100644 index 533386cb2..000000000 --- a/src/UI/Cells/Header/QualityHeaderCell.js +++ /dev/null @@ -1,69 +0,0 @@ -'use strict'; - -define( - [ - 'backgrid', - 'Shared/Grid/HeaderCell' - ], function (Backgrid, NzbDroneHeaderCell) { - - Backgrid.QualityHeaderCell = NzbDroneHeaderCell.extend({ - events: { - 'click': 'onClick' - }, - - onClick: function (e) { - e.preventDefault(); - - var self = this; - var columnName = this.column.get('name'); - - if (this.column.get('sortable')) { - if (this.direction() === 'ascending') { - this.sort(columnName, 'descending', function (left, right) { - var leftVal = left.get(columnName); - var rightVal = right.get(columnName); - - return self._comparator(leftVal, rightVal); - }); - } - else { - this.sort(columnName, 'ascending', function (left, right) { - var leftVal = left.get(columnName); - var rightVal = right.get(columnName); - - return self._comparator(rightVal, leftVal); - }); - } - } - }, - - _comparator: function (leftVal, rightVal) { - var leftWeight = leftVal.quality.weight; - var rightWeight = rightVal.quality.weight; - - if (!leftWeight && !rightWeight) { - return 0; - } - - if (!leftWeight) { - return -1; - } - - if (!rightWeight) { - return 1; - } - - if (leftWeight === rightWeight) { - return 0; - } - - if (leftWeight > rightWeight) { - return -1; - } - - return 1; - } - }); - - return Backgrid.QualityHeaderCell; - }); diff --git a/src/UI/Episode/Search/ManualLayout.js b/src/UI/Episode/Search/ManualLayout.js index c782179df..46dcff938 100644 --- a/src/UI/Episode/Search/ManualLayout.js +++ b/src/UI/Episode/Search/ManualLayout.js @@ -6,10 +6,8 @@ define( 'Cells/FileSizeCell', 'Cells/QualityCell', 'Cells/ApprovalStatusCell', - 'Release/DownloadReportCell', - 'Cells/Header/QualityHeaderCell' - - ], function (Marionette, Backgrid, FileSizeCell, QualityCell, ApprovalStatusCell, DownloadReportCell, QualityHeaderCell) { + 'Release/DownloadReportCell' + ], function (Marionette, Backgrid, FileSizeCell, QualityCell, ApprovalStatusCell, DownloadReportCell) { return Marionette.Layout.extend({ template: 'Episode/Search/ManualLayoutTemplate', @@ -49,7 +47,9 @@ define( label : 'Quality', sortable : true, cell : QualityCell, - headerCell: QualityHeaderCell + sortValue : function (model) { + return model.get('quality').quality.weight; + } }, { diff --git a/src/UI/JsLibraries/backbone.backgrid.js b/src/UI/JsLibraries/backbone.backgrid.js index 94932e472..6a0af616c 100644 --- a/src/UI/JsLibraries/backbone.backgrid.js +++ b/src/UI/JsLibraries/backbone.backgrid.js @@ -1,11 +1,25 @@ -/* +/*! backgrid http://github.com/wyuenho/backgrid - Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors + Licensed under the MIT license. */ -(function (root, $, _, Backbone) { + +(function (factory) { + + // CommonJS + if (typeof exports == "object") { + module.exports = factory(module.exports, + require("underscore"), + require("backbone")); + } + // Browser + else if (typeof _ !== "undefined" && + typeof Backbone !== "undefined") { + factory(window, _, Backbone); + } +}(function (root, _, Backbone) { "use strict"; /* @@ -13,11 +27,9 @@ http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ -var window = root; - // Copyright 2009, 2010 Kristopher Michael Kowal // https://github.com/kriskowal/es5-shim // ES5 15.5.4.20 @@ -41,12 +53,6 @@ if (!String.prototype.trim || ws.trim()) { }; } -function capitalize(s) { - return s.charAt(0).toUpperCase() + s.slice(1); - -// return String.fromCharCode(s.charCodeAt(0) - 32) + s.slice(1); -} - function lpad(str, length, padstr) { var paddingLen = length - (str + '').length; paddingLen = paddingLen < 0 ? 0 : paddingLen; @@ -57,24 +63,19 @@ function lpad(str, length, padstr) { return padding + str; } +var $ = Backbone.$; + var Backgrid = root.Backgrid = { - VERSION: "0.2.6", + VERSION: "0.3.0", Extension: {}, - requireOptions: function (options, requireOptionKeys) { - for (var i = 0; i < requireOptionKeys.length; i++) { - var key = requireOptionKeys[i]; - if (_.isUndefined(options[key])) { - throw new TypeError("'" + key + "' is required"); - } - } - }, - resolveNameToClass: function (name, suffix) { if (_.isString(name)) { - var key = _.map(name.split('-'), function (e) { return capitalize(e); }).join('') + suffix; + var key = _.map(name.split('-'), function (e) { + return e.slice(0, 1).toUpperCase() + e.slice(1); + }).join('') + suffix; var klass = Backgrid[key] || Backgrid.Extension[key]; if (_.isUndefined(klass)) { throw new ReferenceError("Class '" + key + "' not found"); @@ -83,7 +84,17 @@ var Backgrid = root.Backgrid = { } return name; + }, + + callByNeed: function () { + var value = arguments[0]; + if (!_.isFunction(value)) return value; + + var context = arguments[1]; + var args = [].slice.call(arguments, 2); + return value.apply(context, !!(args + '') ? args : void 0); } + }; _.extend(Backgrid, Backbone.Events); @@ -101,7 +112,7 @@ _.extend(Backgrid, Backbone.Events); var Command = Backgrid.Command = function (evt) { _.extend(this, { altKey: !!evt.altKey, - char: evt.char, + "char": evt["char"], charCode: evt.charCode, ctrlKey: !!evt.ctrlKey, key: evt.key, @@ -162,12 +173,13 @@ _.extend(Command.prototype, { } }); + /* backgrid http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ /** @@ -191,9 +203,10 @@ _.extend(CellFormatter.prototype, { @member Backgrid.CellFormatter @param {*} rawData + @param {Backbone.Model} model Used for more complicated formatting @return {*} */ - fromRaw: function (rawData) { + fromRaw: function (rawData, model) { return rawData; }, @@ -206,16 +219,18 @@ _.extend(CellFormatter.prototype, { @member Backgrid.CellFormatter @param {string} formattedData + @param {Backbone.Model} model Used for more complicated formatting @return {*|undefined} */ - toRaw: function (formattedData) { + toRaw: function (formattedData, model) { return formattedData; } }); /** - A floating point number formatter. Doesn't understand notation at the moment. + A floating point number formatter. Doesn't understand scientific notation at + the moment. @class Backgrid.NumberFormatter @extends Backgrid.CellFormatter @@ -223,8 +238,7 @@ _.extend(CellFormatter.prototype, { @throws {RangeError} If decimals < 0 or > 20. */ var NumberFormatter = Backgrid.NumberFormatter = function (options) { - options = options ? _.clone(options) : {}; - _.extend(this, this.defaults, options); + _.extend(this, this.defaults, options || {}); if (this.decimals < 0 || this.decimals > 20) { throw new RangeError("decimals must be between 0 and 20"); @@ -261,9 +275,10 @@ _.extend(NumberFormatter.prototype, { @member Backgrid.NumberFormatter @param {number} number + @param {Backbone.Model} model Used for more complicated formatting @return {string} */ - fromRaw: function (number) { + fromRaw: function (number, model) { if (_.isNull(number) || _.isUndefined(number)) return ''; number = number.toFixed(~~this.decimals); @@ -281,13 +296,18 @@ _.extend(NumberFormatter.prototype, { @member Backgrid.NumberFormatter @param {string} formattedData + @param {Backbone.Model} model Used for more complicated formatting @return {number|undefined} Undefined if the string cannot be converted to a number. */ - toRaw: function (formattedData) { + toRaw: function (formattedData, model) { + formattedData = formattedData.trim(); + + if (formattedData === '') return null; + var rawData = ''; - var thousands = formattedData.trim().split(this.orderSeparator); + var thousands = formattedData.split(this.orderSeparator); for (var i = 0; i < thousands.length; i++) { rawData += thousands[i]; } @@ -322,8 +342,7 @@ _.extend(NumberFormatter.prototype, { @throws {Error} If both `includeDate` and `includeTime` are false. */ var DatetimeFormatter = Backgrid.DatetimeFormatter = function (options) { - options = options ? _.clone(options) : {}; - _.extend(this, this.defaults, options); + _.extend(this, this.defaults, options || {}); if (!this.includeDate && !this.includeTime) { throw new Error("Either includeDate or includeTime must be true"); @@ -357,6 +376,8 @@ _.extend(DatetimeFormatter.prototype, { ISO_SPLITTER_RE: /T|Z| +/, _convert: function (data, validate) { + if ((data + '').trim() === '') return null; + var date, time = null; if (_.isNumber(data)) { var jsDate = new Date(data); @@ -415,10 +436,11 @@ _.extend(DatetimeFormatter.prototype, { @member Backgrid.DatetimeFormatter @param {string} rawData + @param {Backbone.Model} model Used for more complicated formatting @return {string|null|undefined} ISO-8601 string in UTC. Null and undefined values are returned as is. */ - fromRaw: function (rawData) { + fromRaw: function (rawData, model) { if (_.isNull(rawData) || _.isUndefined(rawData)) return ''; return this._convert(rawData); }, @@ -432,12 +454,13 @@ _.extend(DatetimeFormatter.prototype, { @member Backgrid.DatetimeFormatter @param {string} formattedData + @param {Backbone.Model} model Used for more complicated formatting @return {string|undefined} ISO-8601 string in UTC. Undefined if a date is found when `includeDate` is false, or a time is found when `includeTime` is false, or if `includeDate` is true and a date is not found, or if `includeTime` is true and a time is not found. */ - toRaw: function (formattedData) { + toRaw: function (formattedData, model) { return this._convert(formattedData, true); } @@ -460,9 +483,10 @@ _.extend(StringFormatter.prototype, { @member Backgrid.StringFormatter @param {*} rawValue + @param {Backbone.Model} model Used for more complicated formatting @return {string} */ - fromRaw: function (rawValue) { + fromRaw: function (rawValue, model) { if (_.isUndefined(rawValue) || _.isNull(rawValue)) return ''; return rawValue + ''; } @@ -485,9 +509,10 @@ _.extend(EmailFormatter.prototype, { @member Backgrid.EmailFormatter @param {*} formattedData + @param {Backbone.Model} model Used for more complicated formatting @return {string|undefined} */ - toRaw: function (formattedData) { + toRaw: function (formattedData, model) { var parts = formattedData.trim().split("@"); if (parts.length === 2 && _.all(parts)) { return formattedData; @@ -511,19 +536,21 @@ _.extend(SelectFormatter.prototype, { @member Backgrid.SelectFormatter @param {*} rawValue + @param {Backbone.Model} model Used for more complicated formatting @return {Array.<*>} */ - fromRaw: function (rawValue) { + fromRaw: function (rawValue, model) { return _.isArray(rawValue) ? rawValue : rawValue != null ? [rawValue] : []; } }); + /* backgrid http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ /** @@ -548,7 +575,6 @@ var CellEditor = Backgrid.CellEditor = Backbone.View.extend({ `model` or `column` are undefined. */ initialize: function (options) { - Backgrid.requireOptions(options, ["formatter", "column", "model"]); this.formatter = options.formatter; this.column = options.column; if (!(this.column instanceof Column)) { @@ -607,7 +633,7 @@ var InputCellEditor = Backgrid.InputCellEditor = CellEditor.extend({ @param {string} [options.placeholder] */ initialize: function (options) { - CellEditor.prototype.initialize.apply(this, arguments); + InputCellEditor.__super__.initialize.apply(this, arguments); if (options.placeholder) { this.$el.attr("placeholder", options.placeholder); @@ -619,7 +645,8 @@ var InputCellEditor = Backgrid.InputCellEditor = CellEditor.extend({ exists. */ render: function () { - this.$el.val(this.formatter.fromRaw(this.model.get(this.column.get("name")))); + var model = this.model + this.$el.val(this.formatter.fromRaw(model.get(this.column.get("name")), model)); return this; }, @@ -656,7 +683,7 @@ var InputCellEditor = Backgrid.InputCellEditor = CellEditor.extend({ e.stopPropagation(); var val = this.$el.val(); - var newValue = formatter.toRaw(val); + var newValue = formatter.toRaw(val, model); if (_.isUndefined(newValue)) { model.trigger("backgrid:error", model, column, val); } @@ -705,9 +732,9 @@ var Cell = Backgrid.Cell = Backbone.View.extend({ tagName: "td", /** - @property {Backgrid.CellFormatter|Object|string} [formatter=new CellFormatter()] + @property {Backgrid.CellFormatter|Object|string} [formatter=CellFormatter] */ - formatter: new CellFormatter(), + formatter: CellFormatter, /** @property {Backgrid.CellEditor} [editor=Backgrid.InputCellEditor] The @@ -734,17 +761,43 @@ var Cell = Backgrid.Cell = Backbone.View.extend({ said name cannot be found in the Backgrid module. */ initialize: function (options) { - Backgrid.requireOptions(options, ["model", "column"]); this.column = options.column; if (!(this.column instanceof Column)) { this.column = new Column(this.column); } - this.formatter = Backgrid.resolveNameToClass(this.column.get("formatter") || this.formatter, "Formatter"); + + var column = this.column, model = this.model, $el = this.$el; + + var formatter = Backgrid.resolveNameToClass(column.get("formatter") || + this.formatter, "Formatter"); + + if (!_.isFunction(formatter.fromRaw) && !_.isFunction(formatter.toRaw)) { + formatter = new formatter(); + } + + this.formatter = formatter; + this.editor = Backgrid.resolveNameToClass(this.editor, "CellEditor"); - this.listenTo(this.model, "change:" + this.column.get("name"), function () { - if (!this.$el.hasClass("editor")) this.render(); + + this.listenTo(model, "change:" + column.get("name"), function () { + if (!$el.hasClass("editor")) this.render(); }); - this.listenTo(this.model, "backgrid:error", this.renderError); + + this.listenTo(model, "backgrid:error", this.renderError); + + this.listenTo(column, "change:editable change:sortable change:renderable", + function (column) { + var changed = column.changedAttributes(); + for (var key in changed) { + if (changed.hasOwnProperty(key)) { + $el.toggleClass(key, changed[key]); + } + } + }); + + if (column.get("editable")) $el.addClass("editable"); + if (column.get("sortable")) $el.addClass("sortable"); + if (column.get("renderable")) $el.addClass("renderable"); }, /** @@ -753,7 +806,8 @@ var Cell = Backgrid.Cell = Backbone.View.extend({ */ render: function () { this.$el.empty(); - this.$el.text(this.formatter.fromRaw(this.model.get(this.column.get("name")))); + var model = this.model; + this.$el.text(this.formatter.fromRaw(model.get(this.column.get("name")), model)); this.delegateEvents(); return this; }, @@ -781,7 +835,8 @@ var Cell = Backgrid.Cell = Backbone.View.extend({ var model = this.model; var column = this.column; - if (column.get("editable")) { + var editable = Backgrid.callByNeed(column.editable(), column, model); + if (editable) { this.currentEditor = new this.editor({ column: this.column, @@ -830,10 +885,10 @@ var Cell = Backgrid.Cell = Backbone.View.extend({ */ remove: function () { if (this.currentEditor) { - this.currentEditor.remove.apply(this, arguments); + this.currentEditor.remove.apply(this.currentEditor, arguments); delete this.currentEditor; } - return Backbone.View.prototype.remove.apply(this, arguments); + return Cell.__super__.remove.apply(this, arguments); } }); @@ -849,7 +904,7 @@ var StringCell = Backgrid.StringCell = Cell.extend({ /** @property */ className: "string-cell", - formatter: new StringFormatter() + formatter: StringFormatter }); @@ -869,14 +924,33 @@ var UriCell = Backgrid.UriCell = Cell.extend({ /** @property */ className: "uri-cell", + /** + @property {string} [title] The title attribute of the generated anchor. It + uses the display value formatted by the `formatter.fromRaw` by default. + */ + title: null, + + /** + @property {string} [target="_blank"] The target attribute of the generated + anchor. + */ + target: "_blank", + + initialize: function (options) { + UriCell.__super__.initialize.apply(this, arguments); + this.title = options.title || this.title; + this.target = options.target || this.target; + }, + render: function () { this.$el.empty(); - var formattedValue = this.formatter.fromRaw(this.model.get(this.column.get("name"))); + var rawValue = this.model.get(this.column.get("name")); + var formattedValue = this.formatter.fromRaw(rawValue, this.model); this.$el.append($("", { tabIndex: -1, - href: formattedValue, - title: formattedValue, - target: "_blank" + href: rawValue, + title: this.title || formattedValue, + target: this.target, }).text(formattedValue)); this.delegateEvents(); return this; @@ -897,11 +971,12 @@ var EmailCell = Backgrid.EmailCell = StringCell.extend({ /** @property */ className: "email-cell", - formatter: new EmailFormatter(), + formatter: EmailFormatter, render: function () { this.$el.empty(); - var formattedValue = this.formatter.fromRaw(this.model.get(this.column.get("name"))); + var model = this.model; + var formattedValue = this.formatter.fromRaw(model.get(this.column.get("name")), model); this.$el.append($("", { tabIndex: -1, href: "mailto:" + formattedValue, @@ -947,12 +1022,11 @@ var NumberCell = Backgrid.NumberCell = Cell.extend({ @param {Backgrid.Column} options.column */ initialize: function (options) { - Cell.prototype.initialize.apply(this, arguments); - this.formatter = new this.formatter({ - decimals: this.decimals, - decimalSeparator: this.decimalSeparator, - orderSeparator: this.orderSeparator - }); + NumberCell.__super__.initialize.apply(this, arguments); + var formatter = this.formatter; + formatter.decimals = this.decimals; + formatter.decimalSeparator = this.decimalSeparator; + formatter.orderSeparator = this.orderSeparator; } }); @@ -1021,12 +1095,11 @@ var DatetimeCell = Backgrid.DatetimeCell = Cell.extend({ @param {Backgrid.Column} options.column */ initialize: function (options) { - Cell.prototype.initialize.apply(this, arguments); - this.formatter = new this.formatter({ - includeDate: this.includeDate, - includeTime: this.includeTime, - includeMilli: this.includeMilli - }); + DatetimeCell.__super__.initialize.apply(this, arguments); + var formatter = this.formatter; + formatter.includeDate = this.includeDate; + formatter.includeTime = this.includeTime; + formatter.includeMilli = this.includeMilli; var placeholder = this.includeDate ? "YYYY-MM-DD" : ""; placeholder += (this.includeDate && this.includeTime) ? "T" : ""; @@ -1109,7 +1182,8 @@ var BooleanCellEditor = Backgrid.BooleanCellEditor = CellEditor.extend({ uncheck otherwise. */ render: function () { - var val = this.formatter.fromRaw(this.model.get(this.column.get("name"))); + var model = this.model; + var val = this.formatter.fromRaw(model.get(this.column.get("name")), model); this.$el.prop("checked", val); return this; }, @@ -1147,12 +1221,12 @@ var BooleanCellEditor = Backgrid.BooleanCellEditor = CellEditor.extend({ command.moveDown()) { e.preventDefault(); e.stopPropagation(); - var val = formatter.toRaw($el.prop("checked")); + var val = formatter.toRaw($el.prop("checked"), model); model.set(column.get("name"), val); model.trigger("backgrid:edited", model, column, command); } else if (e.type == "change") { - var val = formatter.toRaw($el.prop("checked")); + var val = formatter.toRaw($el.prop("checked"), model); model.set(column.get("name"), val); $el.focus(); } @@ -1186,10 +1260,13 @@ var BooleanCell = Backgrid.BooleanCell = Cell.extend({ */ render: function () { this.$el.empty(); + var model = this.model, column = this.column; + var editable = Backgrid.callByNeed(column.editable(), column, model); this.$el.append($("", { tabIndex: -1, type: "checkbox", - checked: this.formatter.fromRaw(this.model.get(this.column.get("name"))) + checked: this.formatter.fromRaw(model.get(column.get("name")), model), + disabled: !editable })); this.delegateEvents(); return this; @@ -1216,10 +1293,11 @@ var SelectCellEditor = Backgrid.SelectCellEditor = CellEditor.extend({ }, /** @property {function(Object, ?Object=): string} template */ - template: _.template(''), + template: _.template('', null, {variable: null}), setOptionValues: function (optionValues) { this.optionValues = optionValues; + this.optionValues = _.result(this, "optionValues"); }, setMultiple: function (multiple) { @@ -1250,9 +1328,10 @@ var SelectCellEditor = Backgrid.SelectCellEditor = CellEditor.extend({ this.$el.empty(); var optionValues = _.result(this, "optionValues"); - var selectedValues = this.formatter.fromRaw(this.model.get(this.column.get("name"))); + var model = this.model; + var selectedValues = this.formatter.fromRaw(model.get(this.column.get("name")), model); - if (!_.isArray(optionValues)) throw TypeError("optionValues must be an array"); + if (!_.isArray(optionValues)) throw new TypeError("optionValues must be an array"); var optionValue = null; var optionText = null; @@ -1280,7 +1359,7 @@ var SelectCellEditor = Backgrid.SelectCellEditor = CellEditor.extend({ this.$el.append(optgroup); } else { - throw TypeError("optionValues elements must be a name-value pair or an object hash of { name: 'optgroup label', value: [option name-value pairs] }"); + throw new TypeError("optionValues elements must be a name-value pair or an object hash of { name: 'optgroup label', value: [option name-value pairs] }"); } } @@ -1296,7 +1375,7 @@ var SelectCellEditor = Backgrid.SelectCellEditor = CellEditor.extend({ save: function (e) { var model = this.model; var column = this.column; - model.set(column.get("name"), this.formatter.toRaw(this.$el.val())); + model.set(column.get("name"), this.formatter.toRaw(this.$el.val(), model)); model.trigger("backgrid:edited", model, column, new Command(e)); }, @@ -1317,7 +1396,7 @@ var SelectCellEditor = Backgrid.SelectCellEditor = CellEditor.extend({ e.preventDefault(); e.stopPropagation(); if (e.type == "blur" && this.$el.find("option").length === 1) { - model.set(column.get("name"), this.formatter.toRaw(this.$el.val())); + model.set(column.get("name"), this.formatter.toRaw(this.$el.val(), model)); } model.trigger("backgrid:edited", model, column, new Command(e)); } @@ -1369,7 +1448,7 @@ var SelectCell = Backgrid.SelectCell = Cell.extend({ multiple: false, /** @property */ - formatter: new SelectFormatter(), + formatter: SelectFormatter, /** @property {Array.|Array.<{name: string, values: Array.}>} optionValues @@ -1389,8 +1468,7 @@ var SelectCell = Backgrid.SelectCell = Cell.extend({ @throws {TypeError} If `optionsValues` is undefined. */ initialize: function (options) { - Cell.prototype.initialize.apply(this, arguments); - Backgrid.requireOptions(this, ["optionValues"]); + SelectCell.__super__.initialize.apply(this, arguments); this.listenTo(this.model, "backgrid:edit", function (model, column, cell, editor) { if (column.get("name") == this.column.get("name")) { editor.setOptionValues(this.optionValues); @@ -1407,8 +1485,9 @@ var SelectCell = Backgrid.SelectCell = Cell.extend({ render: function () { this.$el.empty(); - var optionValues = this.optionValues; - var rawData = this.formatter.fromRaw(this.model.get(this.column.get("name"))); + var optionValues = _.result(this, "optionValues"); + var model = this.model; + var rawData = this.formatter.fromRaw(model.get(this.column.get("name")), model); var selectedText = []; @@ -1447,7 +1526,7 @@ var SelectCell = Backgrid.SelectCell = Cell.extend({ } catch (ex) { if (ex instanceof TypeError) { - throw TypeError("'optionValues' must be of type {Array.|Array.<{name: string, values: Array.}>}"); + throw new TypeError("'optionValues' must be of type {Array.|Array.<{name: string, values: Array.}>}"); } throw ex; } @@ -1458,12 +1537,13 @@ var SelectCell = Backgrid.SelectCell = Cell.extend({ } }); + /* backgrid http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ /** @@ -1475,9 +1555,77 @@ var SelectCell = Backgrid.SelectCell = Cell.extend({ @class Backgrid.Column @extends Backbone.Model - */ +*/ var Column = Backgrid.Column = Backbone.Model.extend({ + /** + @cfg {Object} defaults Column defaults. To override any of these default + values, you can either change the prototype directly to override + Column.defaults globally or extend Column and supply the custom class to + Backgrid.Grid: + + // Override Column defaults globally + Column.prototype.defaults.sortable = false; + + // Override Column defaults locally + var MyColumn = Column.extend({ + defaults: _.defaults({ + editable: false + }, Column.prototype.defaults) + }); + + var grid = new Backgrid.Grid(columns: new Columns([{...}, {...}], { + model: MyColumn + })); + + @cfg {string} [defaults.name] The default name of the model attribute. + + @cfg {string} [defaults.label] The default label to show in the header. + + @cfg {string|Backgrid.Cell} [defaults.cell] The default cell type. If this + is a string, the capitalized form will be used to look up a cell class in + Backbone, i.e.: string => StringCell. If a Cell subclass is supplied, it is + initialized with a hash of parameters. If a Cell instance is supplied, it + is used directly. + + @cfg {string|Backgrid.HeaderCell} [defaults.headerCell] The default header + cell type. + + @cfg {boolean|string} [defaults.sortable=true] Whether this column is + sortable. If the value is a string, a method will the same name will be + looked up from the column instance to determine whether the column should + be sortable. The method's signature must be `function (Backgrid.Column, + Backbone.Model): boolean`. + + @cfg {boolean|string} [defaults.editable=true] Whether this column is + editable. If the value is a string, a method will the same name will be + looked up from the column instance to determine whether the column should + be editable. The method's signature must be `function (Backgrid.Column, + Backbone.Model): boolean`. + + @cfg {boolean|string} [defaults.renderable=true] Whether this column is + renderable. If the value is a string, a method will the same name will be + looked up from the column instance to determine whether the column should + be renderable. The method's signature must be `function (Backrid.Column, + Backbone.Model): boolean`. + + @cfg {Backgrid.CellFormatter | Object | string} [defaults.formatter] The + formatter to use to convert between raw model values and user input. + + @cfg {"toggle"|"cycle"} [defaults.sortType="cycle"] Whether sorting will + toggle between ascending and descending order, or cycle between insertion + order, ascending and descending order. + + @cfg {(function(Backbone.Model, string): *) | string} [defaults.sortValue] + The function to use to extract a value from the model for comparison during + sorting. If this value is a string, a method with the same name will be + looked up from the column instance. + + @cfg {"ascending"|"descending"|null} [defaults.direction=null] The initial + sorting direction for this column. The default is ordered by + Backbone.Model.cid, which usually means the collection is ordered by + insertion order. + */ defaults: { name: undefined, label: undefined, @@ -1485,6 +1633,9 @@ var Column = Backgrid.Column = Backbone.Model.extend({ editable: true, renderable: true, formatter: undefined, + sortType: "cycle", + sortValue: undefined, + direction: null, cell: undefined, headerCell: undefined }, @@ -1492,42 +1643,103 @@ var Column = Backgrid.Column = Backbone.Model.extend({ /** Initializes this Column instance. - @param {Object} attrs Column attributes. - @param {string} attrs.name The name of the model attribute. - @param {string|Backgrid.Cell} attrs.cell The cell type. - If this is a string, the capitalized form will be used to look up a - cell class in Backbone, i.e.: string => StringCell. If a Cell subclass - is supplied, it is initialized with a hash of parameters. If a Cell - instance is supplied, it is used directly. - @param {string|Backgrid.HeaderCell} [attrs.headerCell] The header cell type. - @param {string} [attrs.label] The label to show in the header. - @param {boolean} [attrs.sortable=true] - @param {boolean} [attrs.editable=true] - @param {boolean} [attrs.renderable=true] - @param {Backgrid.CellFormatter|Object|string} [attrs.formatter] The - formatter to use to convert between raw model values and user input. + @param {Object} attrs + + @param {string} attrs.name The model attribute this column is responsible + for. + + @param {string|Backgrid.Cell} attrs.cell The cell type to use to render + this column. + + @param {string} [attrs.label] + + @param {string|Backgrid.HeaderCell} [attrs.headerCell] + + @param {boolean|string} [attrs.sortable=true] + + @param {boolean|string} [attrs.editable=true] + + @param {boolean|string} [attrs.renderable=true] + + @param {Backgrid.CellFormatter | Object | string} [attrs.formatter] + + @param {"toggle"|"cycle"} [attrs.sortType="cycle"] + + @param {(function(Backbone.Model, string): *) | string} [attrs.sortValue] @throws {TypeError} If attrs.cell or attrs.options are not supplied. - @throws {ReferenceError} If attrs.cell is a string but a cell class of + + @throws {ReferenceError} If formatter is a string but a formatter class of said name cannot be found in the Backgrid module. See: + - Backgrid.Column.defaults - Backgrid.Cell - Backgrid.CellFormatter */ initialize: function (attrs) { - Backgrid.requireOptions(attrs, ["cell", "name"]); - if (!this.has("label")) { this.set({ label: this.get("name") }, { silent: true }); } var headerCell = Backgrid.resolveNameToClass(this.get("headerCell"), "HeaderCell"); + var cell = Backgrid.resolveNameToClass(this.get("cell"), "Cell"); - this.set({ cell: cell, headerCell: headerCell }, { silent: true }); + + this.set({cell: cell, headerCell: headerCell}, { silent: true }); + }, + + /** + Returns an appropriate value extraction function from a model for sorting. + + If the column model contains an attribute `sortValue`, if it is a string, a + method from the column instance identifified by the `sortValue` string is + returned. If it is a function, it it returned as is. If `sortValue` isn't + found from the column model's attributes, a default value extraction + function is returned which will compare according to the natural order of + the value's type. + + @return {function(Backbone.Model, string): *} + */ + sortValue: function () { + var sortValue = this.get("sortValue"); + if (_.isString(sortValue)) return this[sortValue]; + else if (_.isFunction(sortValue)) return sortValue; + + return function (model, colName) { + return model.get(colName); + }; } + /** + @member Backgrid.Column + @protected + @method sortable + @return {function(Backgrid.Column, Backbone.Model): boolean | boolean} + */ + + /** + @member Backgrid.Column + @protected + @method editable + @return {function(Backgrid.Column, Backbone.Model): boolean | boolean} + */ + + /** + @member Backgrid.Column + @protected + @method renderable + @return {function(Backgrid.Column, Backbone.Model): boolean | boolean} + */ +}); + +_.each(["sortable", "renderable", "editable"], function (key) { + Column.prototype[key] = function () { + var value = this.get(key); + if (_.isString(value)) return this[value]; + return !!value; + }; }); /** @@ -1543,12 +1755,13 @@ var Columns = Backgrid.Columns = Backbone.Collection.extend({ */ model: Column }); + /* backgrid http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ /** @@ -1564,8 +1777,6 @@ var Row = Backgrid.Row = Backbone.View.extend({ /** @property */ tagName: "tr", - requiredOptions: ["columns", "model"], - /** Initializes a row view instance. @@ -1577,8 +1788,6 @@ var Row = Backgrid.Row = Backbone.View.extend({ */ initialize: function (options) { - Backgrid.requireOptions(options, this.requiredOptions); - var columns = this.columns = options.columns; if (!(columns instanceof Backbone.Collection)) { columns = this.columns = new Columns(columns); @@ -1589,22 +1798,11 @@ var Row = Backgrid.Row = Backbone.View.extend({ cells.push(this.makeCell(columns.at(i), options)); } - this.listenTo(columns, "change:renderable", function (column, renderable) { - for (var i = 0; i < cells.length; i++) { - var cell = cells[i]; - if (cell.column.get("name") == column.get("name")) { - if (renderable) cell.$el.show(); else cell.$el.hide(); - } - } - }); - this.listenTo(columns, "add", function (column, columns) { var i = columns.indexOf(column); var cell = this.makeCell(column, options); cells.splice(i, 0, cell); - if (!cell.column.get("renderable")) cell.$el.hide(); - var $el = this.$el; if (i === 0) { $el.prepend(cell.render().$el); @@ -1648,11 +1846,8 @@ var Row = Backgrid.Row = Backbone.View.extend({ this.$el.empty(); var fragment = document.createDocumentFragment(); - for (var i = 0; i < this.cells.length; i++) { - var cell = this.cells[i]; - fragment.appendChild(cell.render().el); - if (!cell.column.get("renderable")) cell.$el.hide(); + fragment.appendChild(this.cells[i].render().el); } this.el.appendChild(fragment); @@ -1700,8 +1895,6 @@ var EmptyRow = Backgrid.EmptyRow = Backbone.View.extend({ @param {Backbone.Collection.|Array.|Array.} options.columns Column metadata. */ initialize: function (options) { - Backgrid.requireOptions(options, ["emptyText", "columns"]); - this.emptyText = options.emptyText; this.columns = options.columns; }, @@ -1722,12 +1915,13 @@ var EmptyRow = Backgrid.EmptyRow = Backbone.View.extend({ return this; } }); + /* backgrid http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ /** @@ -1748,12 +1942,6 @@ var HeaderCell = Backgrid.HeaderCell = Backbone.View.extend({ "click a": "onClick" }, - /** - @property {null|"ascending"|"descending"} _direction The current sorting - direction of this column. - */ - _direction: null, - /** Initializer. @@ -1763,12 +1951,30 @@ var HeaderCell = Backgrid.HeaderCell = Backbone.View.extend({ @throws {TypeError} If options.column or options.collection is undefined. */ initialize: function (options) { - Backgrid.requireOptions(options, ["column", "collection"]); this.column = options.column; if (!(this.column instanceof Column)) { this.column = new Column(this.column); } + this.listenTo(this.collection, "backgrid:sort", this._resetCellDirection); + + var column = this.column, $el = this.$el; + + this.listenTo(column, "change:editable change:sortable change:renderable", + function (column) { + var changed = column.changedAttributes(); + for (var key in changed) { + if (changed.hasOwnProperty(key)) { + $el.toggleClass(key, changed[key]); + } + } + }); + + this.listenTo(column, "change:name change:label", this.render); + + if (column.get("editable")) $el.addClass("editable"); + if (column.get("sortable")) $el.addClass("sortable"); + if (column.get("renderable")) $el.addClass("renderable"); }, /** @@ -1781,12 +1987,13 @@ var HeaderCell = Backgrid.HeaderCell = Backbone.View.extend({ */ direction: function (dir) { if (arguments.length) { - if (this._direction) this.$el.removeClass(this._direction); + var direction = this.column.get('direction'); + if (direction) this.$el.removeClass(direction); if (dir) this.$el.addClass(dir); - this._direction = dir; + this.column.set('direction', dir) } - return this._direction; + return this.column.get('direction'); }, /** @@ -1795,11 +2002,9 @@ var HeaderCell = Backgrid.HeaderCell = Backbone.View.extend({ @private */ - _resetCellDirection: function (sortByColName, direction, comparator, collection) { - if (collection == this.collection) { - if (sortByColName !== this.column.get("name")) this.direction(null); - else this.direction(direction); - } + _resetCellDirection: function (columnToSort, direction) { + if (columnToSort !== this.column) this.direction(null); + else this.direction(direction); }, /** @@ -1810,118 +2015,44 @@ var HeaderCell = Backgrid.HeaderCell = Backbone.View.extend({ onClick: function (e) { e.preventDefault(); - var columnName = this.column.get("name"); + var collection = this.collection, event = "backgrid:sort"; - if (this.column.get("sortable")) { - if (this.direction() === "ascending") { - this.sort(columnName, "descending", function (left, right) { - var leftVal = left.get(columnName); - var rightVal = right.get(columnName); - if (leftVal === rightVal) { - return 0; - } - else if (leftVal > rightVal) { return -1; } - return 1; - }); - } - else if (this.direction() === "descending") { - this.sort(columnName, null); - } - else { - this.sort(columnName, "ascending", function (left, right) { - var leftVal = left.get(columnName); - var rightVal = right.get(columnName); - if (leftVal === rightVal) { - return 0; - } - else if (leftVal < rightVal) { return -1; } - return 1; - }); - } + function cycleSort(header, col) { + if (header.direction() === "ascending") collection.trigger(event, col, "descending"); + else if (header.direction() === "descending") collection.trigger(event, col, null); + else collection.trigger(event, col, "ascending"); } - }, - - /** - If the underlying collection is a Backbone.PageableCollection in - server-mode or infinite-mode, a page of models is fetched after sorting is - done on the server. - If the underlying collection is a Backbone.PageableCollection in - client-mode, or any - [Backbone.Collection](http://backbonejs.org/#Collection) instance, sorting - is done on the client side. If the collection is an instance of a - Backbone.PageableCollection, sorting will be done globally on all the pages - and the current page will then be returned. - - Triggers a Backbone `backgrid:sort` event from the collection when done - with the column name, direction, comparator and a reference to the - collection. - - @param {string} columnName - @param {null|"ascending"|"descending"} direction - @param {function(*, *): number} [comparator] - - See [Backbone.Collection#comparator](http://backbonejs.org/#Collection-comparator) - */ - sort: function (columnName, direction, comparator) { - - comparator = comparator || this._cidComparator; - - var collection = this.collection; - - if (Backbone.PageableCollection && collection instanceof Backbone.PageableCollection) { - var order; - if (direction === "ascending") order = -1; - else if (direction === "descending") order = 1; - else order = null; - - collection.setSorting(order ? columnName : null, order); - - if (collection.mode == "client") { - if (!collection.fullCollection.comparator) { - collection.fullCollection.comparator = comparator; - } - collection.fullCollection.sort(); - } - else collection.fetch({reset: true}); - } - else { - collection.comparator = comparator; - collection.sort(); + function toggleSort(header, col) { + if (header.direction() === "ascending") collection.trigger(event, col, "descending"); + else collection.trigger(event, col, "ascending"); } - this.collection.trigger("backgrid:sort", columnName, direction, comparator, this.collection); - }, - - /** - Default comparator for Backbone.Collections. Sorts cids in ascending - order. The cids of the models are assumed to be in insertion order. - - @private - @param {*} left - @param {*} right - */ - _cidComparator: function (left, right) { - var lcid = left.cid, rcid = right.cid; - if (!_.isUndefined(lcid) && !_.isUndefined(rcid)) { - lcid = lcid.slice(1) * 1, rcid = rcid.slice(1) * 1; - if (lcid < rcid) return -1; - else if (lcid > rcid) return 1; + var column = this.column; + var sortable = Backgrid.callByNeed(column.sortable(), column, this.collection); + if (sortable) { + var sortType = column.get("sortType"); + if (sortType === "toggle") toggleSort(this, column); + else cycleSort(this, column); } - - return 0; }, /** - Renders a header cell with a sorter and a label. + Renders a header cell with a sorter, a label, and a class name for this + column. */ render: function () { this.$el.empty(); - var $label = $("").text(this.column.get("label")).append(""); + var column = this.column; + var $label = $("").text(column.get("label")); + var sortable = Backgrid.callByNeed(column.sortable(), column, this.collection); + if (sortable) $label.append(""); this.$el.append($label); + this.$el.addClass(column.get("name")); this.delegateEvents(); + this.direction(column.get("direction")); return this; - } +} }); @@ -1985,8 +2116,6 @@ var Header = Backgrid.Header = Backbone.View.extend({ @throws {TypeError} If options.columns or options.model is undefined. */ initialize: function (options) { - Backgrid.requireOptions(options, ["columns", "collection"]); - this.columns = options.columns; if (!(this.columns instanceof Backbone.Collection)) { this.columns = new Columns(this.columns); @@ -2018,12 +2147,13 @@ var Header = Backgrid.Header = Backbone.View.extend({ } }); + /* backgrid http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ /** @@ -2053,7 +2183,6 @@ var Body = Backgrid.Body = Backbone.View.extend({ See Backgrid.Row. */ initialize: function (options) { - Backgrid.requireOptions(options, ["columns", "collection"]); this.columns = options.columns; if (!(this.columns instanceof Backbone.Collection)) { @@ -2078,6 +2207,7 @@ var Body = Backgrid.Body = Backbone.View.extend({ this.listenTo(collection, "remove", this.removeRow); this.listenTo(collection, "sort", this.refresh); this.listenTo(collection, "reset", this.refresh); + this.listenTo(collection, "backgrid:sort", this.sort); this.listenTo(collection, "backgrid:edited", this.moveToNextCell); }, @@ -2145,6 +2275,8 @@ var Body = Backgrid.Body = Backbone.View.extend({ $children.eq(index).before($rowEl); } } + + return this; }, /** @@ -2186,6 +2318,8 @@ var Body = Backgrid.Body = Backbone.View.extend({ this.rows.splice(options.index, 1); this._unshiftEmptyRowMayBe(); + + return this; }, /** @@ -2249,6 +2383,82 @@ var Body = Backgrid.Body = Backbone.View.extend({ return Backbone.View.prototype.remove.apply(this, arguments); }, + /** + If the underlying collection is a Backbone.PageableCollection in + server-mode or infinite-mode, a page of models is fetched after sorting is + done on the server. + + If the underlying collection is a Backbone.PageableCollection in + client-mode, or any + [Backbone.Collection](http://backbonejs.org/#Collection) instance, sorting + is done on the client side. If the collection is an instance of a + Backbone.PageableCollection, sorting will be done globally on all the pages + and the current page will then be returned. + + Triggers a Backbone `backgrid:sort` event from the collection when done + with the column, direction, comparator and a reference to the collection. + + @param {Backgrid.Column} column + @param {null|"ascending"|"descending"} direction + + See [Backbone.Collection#comparator](http://backbonejs.org/#Collection-comparator) + */ + sort: function (column, direction) { + + if (_.isString(column)) column = this.columns.findWhere({name: column}); + + var collection = this.collection; + + var order; + if (direction === "ascending") order = -1; + else if (direction === "descending") order = 1; + else order = null; + + var comparator = this.makeComparator(column.get("name"), order, + order ? + column.sortValue() : + function (model) { + return model.cid; + }); + + if (Backbone.PageableCollection && + collection instanceof Backbone.PageableCollection) { + + collection.setSorting(order && column.get("name"), order, + {sortValue: column.sortValue()}); + + if (collection.mode == "client") { + if (collection.fullCollection.comparator == null) { + collection.fullCollection.comparator = comparator; + } + collection.fullCollection.sort(); + } + else collection.fetch({reset: true}); + } + else { + collection.comparator = comparator; + collection.sort(); + } + + return this; + }, + + makeComparator: function (attr, order, func) { + + return function (left, right) { + // extract the values from the models + var l = func(left, attr), r = func(right, attr), t; + + // if descending order, swap left and right + if (order === 1) t = l, l = r, r = t; + + // compare as usual + if (l === r) return 0; + else if (l < r) return -1; + return 1; + }; + }, + /** Moves focus to the next renderable and editable cell and return the currently editing cell to display mode. @@ -2261,6 +2471,9 @@ var Body = Backgrid.Body = Backbone.View.extend({ moveToNextCell: function (model, column, command) { var i = this.collection.indexOf(model); var j = this.columns.indexOf(column); + var cell, renderable, editable; + + this.rows[i].cells[j].exitEditMode(); if (command.moveUp() || command.moveDown() || command.moveLeft() || command.moveRight() || command.save()) { @@ -2269,7 +2482,12 @@ var Body = Backgrid.Body = Backbone.View.extend({ if (command.moveUp() || command.moveDown()) { var row = this.rows[i + (command.moveUp() ? -1 : 1)]; - if (row) row.cells[j].enterEditMode(); + if (row) { + cell = row.cells[j]; + if (Backgrid.callByNeed(cell.column.editable(), cell.column, model)) { + cell.enterEditMode(); + } + } } else if (command.moveLeft() || command.moveRight()) { var right = command.moveRight(); @@ -2278,8 +2496,10 @@ var Body = Backgrid.Body = Backbone.View.extend({ right ? offset++ : offset--) { var m = ~~(offset / l); var n = offset - m * l; - var cell = this.rows[m].cells[n]; - if (cell.column.get("renderable") && cell.column.get("editable")) { + cell = this.rows[m].cells[n]; + renderable = Backgrid.callByNeed(cell.column.renderable(), cell.column, cell.model); + editable = Backgrid.callByNeed(cell.column.editable(), cell.column, model); + if (renderable && editable) { cell.enterEditMode(); break; } @@ -2287,15 +2507,16 @@ var Body = Backgrid.Body = Backbone.View.extend({ } } - this.rows[i].cells[j].exitEditMode(); + return this; } }); + /* backgrid http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ /** @@ -2315,7 +2536,6 @@ var Footer = Backgrid.Footer = Backbone.View.extend({ Initializer. @param {Object} options - @param {*} options.parent The parent view class of this footer. @param {Backbone.Collection.|Array.|Array.} options.columns Column metadata. @param {Backbone.Collection} options.collection @@ -2323,7 +2543,6 @@ var Footer = Backgrid.Footer = Backbone.View.extend({ @throws {TypeError} If options.columns or options.collection is undefined. */ initialize: function (options) { - Backgrid.requireOptions(options, ["columns", "collection"]); this.columns = options.columns; if (!(this.columns instanceof Backbone.Collection)) { this.columns = new Backgrid.Columns(this.columns); @@ -2331,12 +2550,13 @@ var Footer = Backgrid.Footer = Backbone.View.extend({ } }); + /* backgrid http://github.com/wyuenho/backgrid Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors - Licensed under the MIT @license. + Licensed under the MIT license. */ /** @@ -2407,7 +2627,7 @@ var Grid = Backgrid.Grid = Backbone.View.extend({ Initializes a Grid instance. @param {Object} options - @param {Backbone.Collection.|Array.|Array.} options.columns Column metadata. + @param {Backbone.Collection.|Array.|Array.} options.columns Column metadata. @param {Backbone.Collection} options.collection The collection of tabular model data to display. @param {Backgrid.Header} [options.header=Backgrid.Header] An optional Header class to override the default. @param {Backgrid.Body} [options.body=Backgrid.Body] An optional Body class to override the default. @@ -2415,8 +2635,6 @@ var Grid = Backgrid.Grid = Backbone.View.extend({ @param {Backgrid.Footer} [options.footer=Backgrid.Footer] An optional Footer class. */ initialize: function (options) { - Backgrid.requireOptions(options, ["columns", "collection"]); - // Convert the list of column objects here first so the subviews don't have // to. if (!(options.columns instanceof Backbone.Collection)) { @@ -2424,25 +2642,30 @@ var Grid = Backgrid.Grid = Backbone.View.extend({ } this.columns = options.columns; - var passedThruOptions = _.omit(options, ["el", "id", "attributes", - "className", "tagName", "events"]); - - this.header = options.header || this.header; - this.header = new this.header(passedThruOptions); + var filteredOptions = _.omit(options, ["el", "id", "attributes", + "className", "tagName", "events"]); + // must construct body first so it listens to backgrid:sort first this.body = options.body || this.body; - this.body = new this.body(passedThruOptions); + this.body = new this.body(filteredOptions); + + this.header = options.header || this.header; + if (this.header) { + this.header = new this.header(filteredOptions); + } this.footer = options.footer || this.footer; if (this.footer) { - this.footer = new this.footer(passedThruOptions); + this.footer = new this.footer(filteredOptions); } this.listenTo(this.columns, "reset", function () { - this.header = new (this.header.remove().constructor)(passedThruOptions); - this.body = new (this.body.remove().constructor)(passedThruOptions); + if (this.header) { + this.header = new (this.header.remove().constructor)(filteredOptions); + } + this.body = new (this.body.remove().constructor)(filteredOptions); if (this.footer) { - this.footer = new (this.footer.remove().constructor)(passedThruOptions); + this.footer = new (this.footer.remove().constructor)(filteredOptions); } this.render(); }); @@ -2452,14 +2675,16 @@ var Grid = Backgrid.Grid = Backbone.View.extend({ Delegates to Backgrid.Body#insertRow. */ insertRow: function (model, collection, options) { - return this.body.insertRow(model, collection, options); + this.body.insertRow(model, collection, options); + return this; }, /** Delegates to Backgrid.Body#removeRow. */ removeRow: function (model, collection, options) { - return this.body.removeRow(model, collection, options); + this.body.removeRow(model, collection, options); + return this; }, /** @@ -2470,8 +2695,6 @@ var Grid = Backgrid.Grid = Backbone.View.extend({ @param {Object} [options] Options for `Backgrid.Columns#add`. @param {boolean} [options.render=true] Whether to render the column immediately after insertion. - - @chainable */ insertColumn: function (column, options) { options = options || {render: true}; @@ -2485,14 +2708,20 @@ var Grid = Backgrid.Grid = Backbone.View.extend({ needs to happen. @param {Object} [options] Options for `Backgrid.Columns#remove`. - - @chainable */ removeColumn: function (column, options) { this.columns.remove(column, options); return this; }, + /** + Delegates to Backgrid.Body#sort. + */ + sort: function () { + this.body.sort(arguments); + return this; + }, + /** Renders the grid's header, then footer, then finally the body. Triggers a Backbone `backgrid:rendered` event along with a reference to the grid when @@ -2501,7 +2730,9 @@ var Grid = Backgrid.Grid = Backbone.View.extend({ render: function () { this.$el.empty(); - this.$el.append(this.header.render().$el); + if (this.header) { + this.$el.append(this.header.render().$el); + } if (this.footer) { this.$el.append(this.footer.render().$el); @@ -2522,12 +2753,12 @@ var Grid = Backgrid.Grid = Backbone.View.extend({ @chainable */ remove: function () { - this.header.remove.apply(this.header, arguments); + this.header && this.header.remove.apply(this.header, arguments); this.body.remove.apply(this.body, arguments); this.footer && this.footer.remove.apply(this.footer, arguments); return Backbone.View.prototype.remove.apply(this, arguments); } }); - -}(this, jQuery, _, Backbone)); \ No newline at end of file +return Backgrid; +})); \ No newline at end of file diff --git a/src/UI/JsLibraries/backbone.backgrid.paginator.js b/src/UI/JsLibraries/backbone.backgrid.paginator.js index cceabca00..03255f84d 100644 --- a/src/UI/JsLibraries/backbone.backgrid.paginator.js +++ b/src/UI/JsLibraries/backbone.backgrid.paginator.js @@ -5,122 +5,271 @@ Copyright (c) 2013 Jimmy Yuen Ho Wong and contributors Licensed under the MIT @license. */ - -(function ($, _, Backbone, Backgrid) { +(function (factory) { + + // CommonJS + if (typeof exports == "object") { + module.exports = factory(require("underscore"), + require("backbone"), + require("backgrid"), + require("backbone-pageable")); + } + // Browser + else if (typeof _ !== "undefined" && + typeof Backbone !== "undefined" && + typeof Backgrid !== "undefined") { + factory(_, Backbone, Backgrid); + } + +}(function (_, Backbone, Backgrid) { "use strict"; /** - Paginator is a Backgrid extension that renders a series of configurable - pagination handles. This extension is best used for splitting a large data - set across multiple pages. If the number of pages is larger then a - threshold, which is set to 10 by default, the page handles are rendered - within a sliding window, plus the fast forward, fast backward, previous and - next page handles. The fast forward, fast backward, previous and next page - handles can be turned off. - - @class Backgrid.Extension.Paginator + PageHandle is a class that renders the actual page handles and reacts to + click events for pagination. + + This class acts in two modes - control or discrete page handle modes. If + one of the `is*` flags is `true`, an instance of this class is under + control page handle mode. Setting a `pageIndex` to an instance of this + class under control mode has no effect and the correct page index will + always be inferred from the `is*` flag. Only one of the `is*` flags should + be set to `true` at a time. For example, an instance of this class cannot + simultaneously be a rewind control and a fast forward control. A `label` + and a `title` template or a string are required to be passed to the + constuctor under this mode. If a `title` template is provided, it __MUST__ + accept a parameter `label`. When the `label` is provided to the `title` + template function, its result will be used to render the generated anchor's + title attribute. + + If all of the `is*` flags is set to `false`, which is the default, an + instance of this class will be in discrete page handle mode. An instance + under this mode requires the `pageIndex` to be passed from the constructor + as an option and it __MUST__ be a 0-based index of the list of page numbers + to render. The constuctor will normalize the base to the same base the + underlying PageableCollection collection instance uses. A `label` is not + required under this mode, which will default to the equivalent 1-based page + index calculated from `pageIndex` and the underlying PageableCollection + instance. A provided `label` will still be honored however. The `title` + parameter is also not required under this mode, in which case the default + `title` template will be used. You are encouraged to provide your own + `title` template however if you wish to localize the title strings. + + If this page handle represents the current page, an `active` class will be + placed on the root list element. + + if this page handle is at the border of the list of pages, a `disabled` + class will be placed on the root list element. + + Only page handles that are neither `active` nor `disabled` will respond to + click events and triggers pagination. + + @class Backgrid.Extension.PageHandle */ - Backgrid.Extension.Paginator = Backbone.View.extend({ + var PageHandle = Backgrid.Extension.PageHandle = Backbone.View.extend({ /** @property */ - className: "backgrid-paginator", + tagName: "li", /** @property */ - windowSize: 10, + events: { + "click a": "changePage" + }, /** - @property {Object} fastForwardHandleLabels You can disable specific - handles by setting its value to `null`. + @property {string|function(Object.): string} title + The title to use for the `title` attribute of the generated page handle + anchor elements. It can be a string or an Underscore template function + that takes a mandatory `label` parameter. */ - fastForwardHandleLabels: { - first: "《", - prev: "〈", - next: "〉", - last: "》" - }, + title: _.template('Page <%- label %>', null, {variable: null}), - /** @property */ - template: _.template(''), + /** + @property {boolean} isRewind Whether this handle represents a rewind + control + */ + isRewind: false, - /** @property */ - events: { - "click a": "changePage" - }, + /** + @property {boolean} isBack Whether this handle represents a back + control + */ + isBack: false, + + /** + @property {boolean} isForward Whether this handle represents a forward + control + */ + isForward: false, + + /** + @property {boolean} isFastForward Whether this handle represents a fast + forward control + */ + isFastForward: false, /** Initializer. @param {Object} options @param {Backbone.Collection} options.collection - @param {boolean} [options.fastForwardHandleLabels] Whether to render fast forward buttons. + @param {number} pageIndex 0-based index of the page number this handle + handles. This parameter will be normalized to the base the underlying + PageableCollection uses. + @param {string} [options.label] If provided it is used to render the + anchor text, otherwise the normalized pageIndex will be used + instead. Required if any of the `is*` flags is set to `true`. + @param {string} [options.title] + @param {boolean} [options.isRewind=false] + @param {boolean} [options.isBack=false] + @param {boolean} [options.isForward=false] + @param {boolean} [options.isFastForward=false] */ initialize: function (options) { - Backgrid.requireOptions(options, ["collection"]); + Backbone.View.prototype.initialize.apply(this, arguments); var collection = this.collection; - var fullCollection = collection.fullCollection; - if (fullCollection) { - this.listenTo(fullCollection, "add", this.render); - this.listenTo(fullCollection, "remove", this.render); - this.listenTo(fullCollection, "reset", this.render); - } + var state = collection.state; + var currentPage = state.currentPage; + var firstPage = state.firstPage; + var lastPage = state.lastPage; + + _.extend(this, _.pick(options, + ["isRewind", "isBack", "isForward", "isFastForward"])); + + var pageIndex; + if (this.isRewind) pageIndex = firstPage; + else if (this.isBack) pageIndex = Math.max(firstPage, currentPage - 1); + else if (this.isForward) pageIndex = Math.min(lastPage, currentPage + 1); + else if (this.isFastForward) pageIndex = lastPage; else { - this.listenTo(collection, "add", this.render); - this.listenTo(collection, "remove", this.render); - this.listenTo(collection, "reset", this.render); + pageIndex = +options.pageIndex; + pageIndex = (firstPage ? pageIndex + 1 : pageIndex); } + this.pageIndex = pageIndex; + + if (((this.isRewind || this.isBack) && currentPage == firstPage) || + ((this.isForward || this.isFastForward) && currentPage == lastPage)) { + this.$el.addClass("disabled"); + } + else if (!(this.isRewind || + this.isBack || + this.isForward || + this.isFastForward) && + currentPage == pageIndex) { + this.$el.addClass("active"); + } + + this.label = (options.label || (firstPage ? pageIndex : pageIndex + 1)) + ''; + var title = options.title || this.title; + this.title = _.isFunction(title) ? title({label: this.label}) : title; }, /** - jQuery event handler for the page handlers. Goes to the right page upon - clicking. + Renders a clickable anchor element under a list item. + */ + render: function () { + this.$el.empty(); + var anchor = document.createElement("a"); + anchor.href = '#'; + if (this.title) anchor.title = this.title; + anchor.innerHTML = this.label; + this.el.appendChild(anchor); + this.delegateEvents(); + return this; + }, - @param {Event} e - */ + /** + jQuery click event handler. Goes to the page this PageHandle instance + represents. No-op if this page handle is currently active or disabled. + */ changePage: function (e) { e.preventDefault(); + var $el = this.$el; + if (!$el.hasClass("active") && !$el.hasClass("disabled")) { + this.collection.getPage(this.pageIndex); + } + return this; + } - var $li = $(e.target).parent(); - if (!$li.hasClass("active") && !$li.hasClass("disabled")) { - - var label = $(e.target).text(); - var ffLabels = this.fastForwardHandleLabels; - - var collection = this.collection; - - if (ffLabels) { - switch (label) { - case ffLabels.first: - collection.getFirstPage(); - return; - case ffLabels.prev: - collection.getPreviousPage(); - return; - case ffLabels.next: - collection.getNextPage(); - return; - case ffLabels.last: - collection.getLastPage(); - return; - } - } + }); + + /** + Paginator is a Backgrid extension that renders a series of configurable + pagination handles. This extension is best used for splitting a large data + set across multiple pages. If the number of pages is larger then a + threshold, which is set to 10 by default, the page handles are rendered + within a sliding window, plus the rewind, back, forward and fast forward + control handles. The individual control handles can be turned off. + + @class Backgrid.Extension.Paginator + */ + Backgrid.Extension.Paginator = Backbone.View.extend({ + + /** @property */ + className: "backgrid-paginator", + + /** @property */ + windowSize: 10, - var state = collection.state; - var pageIndex = +label; - collection.getPage(state.firstPage === 0 ? pageIndex - 1 : pageIndex); + /** + @property {Object.>} controls You can + disable specific control handles by omitting certain keys. + */ + controls: { + rewind: { + label: "《", + title: "First" + }, + back: { + label: "〈", + title: "Previous" + }, + forward: { + label: "〉", + title: "Next" + }, + fastForward: { + label: "》", + title: "Last" } }, /** - Internal method to create a list of page handle objects for the template - to render them. + @property {Backgrid.Extension.PageHandle} pageHandle. The PageHandle + class to use for rendering individual handles + */ + pageHandle: PageHandle, - @return {Array.} an array of page handle objects hashes - */ - makeHandles: function () { + /** @property */ + goBackFirstOnSort: true, + + /** + Initializer. + + @param {Object} options + @param {Backbone.Collection} options.collection + @param {boolean} [options.controls] + @param {boolean} [options.pageHandle=Backgrid.Extension.PageHandle] + @param {boolean} [options.goBackFirstOnSort=true] + */ + initialize: function (options) { + this.controls = options.controls || this.controls; + this.pageHandle = options.pageHandle || this.pageHandle; - var handles = []; + var collection = this.collection; + this.listenTo(collection, "add", this.render); + this.listenTo(collection, "remove", this.render); + this.listenTo(collection, "reset", this.render); + if ((options.goBackFirstOnSort || this.goBackFirstOnSort) && + collection.fullCollection) { + this.listenTo(collection.fullCollection, "sort", function () { + collection.getFirstPage(); + }); + } + }, + + _calculateWindow: function () { var collection = this.collection; var state = collection.state; @@ -132,48 +281,44 @@ currentPage = firstPage ? currentPage - 1 : currentPage; var windowStart = Math.floor(currentPage / this.windowSize) * this.windowSize; var windowEnd = Math.min(lastPage + 1, windowStart + this.windowSize); + return [windowStart, windowEnd]; + }, - if (collection.mode !== "infinite") { - for (var i = windowStart; i < windowEnd; i++) { - handles.push({ - label: i + 1, - title: "No. " + (i + 1), - className: currentPage === i ? "active" : undefined - }); - } - } + /** + Creates a list of page handle objects for rendering. - var ffLabels = this.fastForwardHandleLabels; - if (ffLabels) { + @return {Array.} an array of page handle objects hashes + */ + makeHandles: function () { - if (ffLabels.prev) { - handles.unshift({ - label: ffLabels.prev, - className: collection.hasPrevious() ? void 0 : "disabled" - }); - } + var handles = []; + var collection = this.collection; - if (ffLabels.first) { - handles.unshift({ - label: ffLabels.first, - className: collection.hasPrevious() ? void 0 : "disabled" - }); - } + var window = this._calculateWindow(); + var winStart = window[0], winEnd = window[1]; - if (ffLabels.next) { - handles.push({ - label: ffLabels.next, - className: collection.hasNext() ? void 0 : "disabled" - }); - } + for (var i = winStart; i < winEnd; i++) { + handles.push(new this.pageHandle({ + collection: collection, + pageIndex: i + })); + } - if (ffLabels.last) { - handles.push({ - label: ffLabels.last, - className: collection.hasNext() ? void 0 : "disabled" - }); + var controls = this.controls; + _.each(["back", "rewind", "forward", "fastForward"], function (key) { + var value = controls[key]; + if (value) { + var handleCtorOpts = { + collection: collection, + title: value.title, + label: value.label + }; + handleCtorOpts["is" + key.slice(0, 1).toUpperCase() + key.slice(1)] = true; + var handle = new this.pageHandle(handleCtorOpts); + if (key == "rewind" || key == "back") handles.unshift(handle); + else handles.push(handle); } - } + }, this); return handles; }, @@ -184,15 +329,24 @@ render: function () { this.$el.empty(); - this.$el.append(this.template({ - handles: this.makeHandles() - })); + if (this.handles) { + for (var i = 0, l = this.handles.length; i < l; i++) { + this.handles[i].remove(); + } + } - this.delegateEvents(); + var handles = this.handles = this.makeHandles(); + + var ul = document.createElement("ul"); + for (var i = 0; i < handles.length; i++) { + ul.appendChild(handles[i].render().el); + } + + this.el.appendChild(ul); return this; } }); -}(jQuery, _, Backbone, Backgrid)); +})); diff --git a/src/UI/JsLibraries/backbone.pageable.js b/src/UI/JsLibraries/backbone.pageable.js index 83feb3f7e..f6cdbcacd 100644 --- a/src/UI/JsLibraries/backbone.pageable.js +++ b/src/UI/JsLibraries/backbone.pageable.js @@ -1,5 +1,5 @@ /* - backbone-pageable 1.3.2 + backbone-pageable 1.4.1 http://github.com/wyuenho/backbone-pageable Copyright (c) 2013 Jimmy Yuen Ho Wong @@ -83,7 +83,7 @@ for (var i = 0, l = kvps.length; i < l; i++) { var param = kvps[i]; kvp = param.split('='), k = kvp[0], v = kvp[1] || true; - k = decode(k), ls = params[k]; + k = decode(k), v = decode(v), ls = params[k]; if (_isArray(ls)) ls.push(v); else if (ls) params[k] = [ls, v]; else params[k] = v; @@ -91,6 +91,29 @@ return params; } + // hack to make sure the whatever event handlers for this event is run + // before func is, and the event handlers that func will trigger. + function runOnceAtLastHandler (col, event, func) { + var eventHandlers = col._events[event]; + if (eventHandlers && eventHandlers.length) { + var lastHandler = eventHandlers[eventHandlers.length - 1]; + var oldCallback = lastHandler.callback; + lastHandler.callback = function () { + try { + oldCallback.apply(this, arguments); + func(); + } + catch (e) { + throw e; + } + finally { + lastHandler.callback = oldCallback; + } + }; + } + else func(); + } + var PARAM_TRIM_RE = /[\s'"]/g; var URL_TRIM_RE = /[<>\s'"]/g; @@ -256,7 +279,7 @@ */ constructor: function (models, options) { - Backbone.Collection.apply(this, arguments); + BBColProto.constructor.apply(this, arguments); options = options || {}; @@ -299,7 +322,7 @@ var fullCollection = this.fullCollection; if (comparator && options.full) { - delete this.comparator; + this.comparator = null; fullCollection.comparator = comparator; } @@ -308,6 +331,7 @@ // make sure the models in the current page and full collection have the // same references if (models && !_isEmpty(models)) { + this.reset([].slice.call(models), _extend({silent: true}, options)); this.getPage(state.currentPage); models.splice.apply(models, [0, models.length].concat(this.models)); } @@ -412,22 +436,10 @@ pageCol.at(pageSize) : null; if (modelToRemove) { - var addHandlers = collection._events.add || [], - popOptions = {onAdd: true}; - if (addHandlers.length) { - var lastAddHandler = addHandlers[addHandlers.length - 1]; - var oldCallback = lastAddHandler.callback; - lastAddHandler.callback = function () { - try { - oldCallback.apply(this, arguments); - pageCol.remove(modelToRemove, popOptions); - } - finally { - lastAddHandler.callback = oldCallback; - } - }; - } - else pageCol.remove(modelToRemove, popOptions); + var popOptions = {onAdd: true}; + runOnceAtLastHandler(collection, event, function () { + pageCol.remove(modelToRemove, popOptions); + }); } } } @@ -442,20 +454,25 @@ } else { var totalPages = state.totalPages = ceil(state.totalRecords / pageSize); - state.lastPage = firstPage === 0 ? totalPages - 1 : totalPages; + state.lastPage = firstPage === 0 ? totalPages - 1 : totalPages || firstPage; if (state.currentPage > totalPages) state.currentPage = state.lastPage; } pageCol.state = pageCol._checkState(state); var nextModel, removedIndex = options.index; if (collection == pageCol) { - if (nextModel = fullCol.at(pageEnd)) pageCol.push(nextModel); + if (nextModel = fullCol.at(pageEnd)) { + runOnceAtLastHandler(pageCol, event, function () { + pageCol.push(nextModel); + }); + } fullCol.remove(model); } else if (removedIndex >= pageStart && removedIndex < pageEnd) { pageCol.remove(model); - nextModel = fullCol.at(currentPage * (pageSize + removedIndex)); - if (nextModel) pageCol.push(nextModel); + var at = removedIndex + 1 + nextModel = fullCol.at(at) || fullCol.last(); + if (nextModel) pageCol.add(nextModel, {at: at}); } } else delete options.onAdd; @@ -466,13 +483,13 @@ collection = model; // Reset that's not a result of getPage - if (collection === pageCol && options.from == null && + if (collection == pageCol && options.from == null && options.to == null) { var head = fullCol.models.slice(0, pageStart); var tail = fullCol.models.slice(pageStart + pageCol.models.length); fullCol.reset(head.concat(pageCol.models).concat(tail), options); } - else if (collection === fullCol) { + else if (collection == fullCol) { if (!(state.totalRecords = fullCol.models.length)) { state.totalRecords = null; state.totalPages = null; @@ -551,7 +568,7 @@ throw new RangeError("`firstPage must be 0 or 1`"); } - state.lastPage = firstPage === 0 ? max(0, totalPages - 1) : totalPages; + state.lastPage = firstPage === 0 ? max(0, totalPages - 1) : totalPages || firstPage; if (mode == "infinite") { if (!links[currentPage + '']) { @@ -561,6 +578,8 @@ else if (currentPage < firstPage || (totalPages > 0 && (firstPage ? currentPage > totalPages : currentPage >= totalPages))) { + var op = firstPage ? ">=" : ">"; + throw new RangeError("`currentPage` must be firstPage <= currentPage " + (firstPage ? ">" : ">=") + " totalPages if " + firstPage + "-based. Got " + @@ -681,7 +700,7 @@ var fullCollection = this.fullCollection; var handlers = this._handlers = this._handlers || {}, handler; if (mode != "server" && !fullCollection) { - fullCollection = this._makeFullCollection(options.models || []); + fullCollection = this._makeFullCollection(options.models || [], options); fullCollection.pageableCollection = this; this.fullCollection = fullCollection; var allHandler = this._makeCollectionEventHandler(this, fullCollection); @@ -856,7 +875,8 @@ []; if ((mode == "client" || (mode == "infinite" && !_isEmpty(pageModels))) && !options.fetch) { - return this.reset(pageModels, _omit(options, "fetch")); + this.reset(pageModels, _omit(options, "fetch")); + return this; } if (mode == "infinite") options.url = this.links[pageNum]; @@ -1310,8 +1330,8 @@ this.comparator = comparator; } - if (delComp) delete this.comparator; - if (delFullComp && fullCollection) delete fullCollection.comparator; + if (delComp) this.comparator = null; + if (delFullComp && fullCollection) fullCollection.comparator = null; return this; } diff --git a/src/UI/Missing/Collection.js b/src/UI/Missing/MissingCollection.js similarity index 100% rename from src/UI/Missing/Collection.js rename to src/UI/Missing/MissingCollection.js diff --git a/src/UI/Missing/MissingLayout.js b/src/UI/Missing/MissingLayout.js index a6ecef90a..7240277f5 100644 --- a/src/UI/Missing/MissingLayout.js +++ b/src/UI/Missing/MissingLayout.js @@ -4,7 +4,7 @@ define( 'underscore', 'marionette', 'backgrid', - 'Missing/Collection', + 'Missing/MissingCollection', 'Cells/SeriesTitleCell', 'Cells/EpisodeNumberCell', 'Cells/EpisodeTitleCell', @@ -121,7 +121,6 @@ define( ] }; - this.toolbar.show(new ToolbarLayout({ left : [ diff --git a/src/UI/Mixins/AsPersistedStateCollection.js b/src/UI/Mixins/AsPersistedStateCollection.js index c74e3fcd5..de663b794 100644 --- a/src/UI/Mixins/AsPersistedStateCollection.js +++ b/src/UI/Mixins/AsPersistedStateCollection.js @@ -14,7 +14,7 @@ define( throw 'tableName is required'; } - _setState.call(this); + _setInitialState.call(this); this.on('backgrid:sort', _storeState, this); @@ -23,7 +23,7 @@ define( } }; - var _setState = function () { + var _setInitialState = function () { var key = Config.getValue('{0}.sortKey'.format(this.tableName), this.state.sortKey); var direction = Config.getValue('{0}.sortDirection'.format(this.tableName), this.state.order); var order = parseInt(direction, 10); @@ -32,8 +32,9 @@ define( this.state.order = order; }; - var _storeState = function (sortKey, sortDirection) { + var _storeState = function (column, sortDirection) { var order = _convertDirectionToInt(sortDirection); + var sortKey = column.get('name'); Config.setValue('{0}.sortKey'.format(this.tableName), sortKey); Config.setValue('{0}.sortDirection'.format(this.tableName), order); diff --git a/src/UI/Series/Index/SeriesIndexLayout.js b/src/UI/Series/Index/SeriesIndexLayout.js index 4d794cf07..9a6390cb7 100644 --- a/src/UI/Series/Index/SeriesIndexLayout.js +++ b/src/UI/Series/Index/SeriesIndexLayout.js @@ -13,12 +13,12 @@ define( 'Cells/QualityProfileCell', 'Cells/EpisodeProgressCell', 'Cells/SeriesActionsCell', - 'Shared/Grid/DateHeaderCell', 'Cells/SeriesStatusCell', 'Series/Index/FooterView', 'Series/Index/FooterModel', 'Shared/Toolbar/ToolbarLayout', - 'underscore' + 'underscore', + 'moment' ], function (Marionette, Backgrid, PosterCollectionView, @@ -31,12 +31,12 @@ define( QualityProfileCell, EpisodeProgressCell, SeriesActionsCell, - DateHeaderCell, SeriesStatusCell, FooterView, FooterModel, ToolbarLayout, - _) { + _, + Moment) { return Marionette.Layout.extend({ template: 'Series/Index/SeriesIndexLayoutTemplate', @@ -78,7 +78,15 @@ define( name : 'nextAiring', label : 'Next Airing', cell : RelativeDateCell, - headerCell: DateHeaderCell + sortValue : function (model) { + var nextAiring = model.get('nextAiring'); + + if (!nextAiring) { + return Number.MAX_VALUE; + } + + return Moment(nextAiring).unix(); + } }, { name : 'percentOfEpisodes', diff --git a/src/UI/Series/SeriesCollection.js b/src/UI/Series/SeriesCollection.js index 204a8759b..437a83768 100644 --- a/src/UI/Series/SeriesCollection.js +++ b/src/UI/Series/SeriesCollection.js @@ -16,7 +16,7 @@ define( state: { sortKey: 'title', - order : -1 + order : 'ascending' }, save: function () { diff --git a/src/UI/Shared/Grid/DateHeaderCell.js b/src/UI/Shared/Grid/DateHeaderCell.js deleted file mode 100644 index fcc3e2ba4..000000000 --- a/src/UI/Shared/Grid/DateHeaderCell.js +++ /dev/null @@ -1,66 +0,0 @@ -'use strict'; - -define( - [ - 'backgrid', - 'Shared/Grid/HeaderCell' - ], function (Backgrid, NzbDroneHeaderCell) { - - Backgrid.DateHeaderCell = NzbDroneHeaderCell.extend({ - events: { - 'click': 'onClick' - }, - - onClick: function (e) { - e.preventDefault(); - - var self = this; - var columnName = this.column.get('name'); - - if (this.column.get('sortable')) { - if (this.direction() === 'ascending') { - this.sort(columnName, 'descending', function (left, right) { - var leftVal = left.get(columnName); - var rightVal = right.get(columnName); - - return self._comparator(leftVal, rightVal); - }); - } - else { - this.sort(columnName, 'ascending', function (left, right) { - var leftVal = left.get(columnName); - var rightVal = right.get(columnName); - - return self._comparator(rightVal, leftVal); - }); - } - } - }, - - _comparator: function (leftVal, rightVal) { - if (!leftVal && !rightVal) { - return 0; - } - - if (!leftVal) { - return -1; - } - - if (!rightVal) { - return 1; - } - - if (leftVal === rightVal) { - return 0; - } - - if (leftVal > rightVal) { - return -1; - } - - return 1; - } - }); - - return Backgrid.DateHeaderCell; - }); diff --git a/src/UI/Shared/Grid/HeaderCell.js b/src/UI/Shared/Grid/HeaderCell.js index 4d1cbdeb5..2f1f3c429 100644 --- a/src/UI/Shared/Grid/HeaderCell.js +++ b/src/UI/Shared/Grid/HeaderCell.js @@ -6,6 +6,7 @@ define( ], function (Backgrid) { Backgrid.NzbDroneHeaderCell = Backgrid.HeaderCell.extend({ + events: { 'click': 'onClick' }, @@ -14,87 +15,113 @@ define( this.$el.empty(); this.$el.append(this.column.get('label')); - if (this.column.get('sortable')) { + var column = this.column; + var sortable = Backgrid.callByNeed(column.sortable(), column, this.collection); + + if (sortable) + { this.$el.addClass('sortable'); this.$el.append(' '); + } + + //Do we need this? + this.$el.addClass(column.get('name')); + + this.delegateEvents(); + this.direction(column.get('direction')); - if (this.collection.state) { - var sortKey = this.collection.state.sortKey; - var sortDir = this._convertIntToDirection(this.collection.state.order); + if (this.collection.state) { + var key = this.collection.state.sortKey; + var order = this.collection.state.order; - if (sortKey === this.column.get('name')) { - this.$el.children('i').addClass(this._convertDirectionToIcon(sortDir)); - this._direction = sortDir; - } + if (key === this.column.get('name')) { + this._setSortIcon(order); } } - this.delegateEvents(); + return this; }, direction: function (dir) { + this.$el.children('i').removeClass('icon-sort-up icon-sort-down'); + if (arguments.length) { - if (this._direction) { - this.$el.children('i').removeClass(this._convertDirectionToIcon(this._direction)); + if (dir) + { + this._setSortIcon(dir); } - if (dir) { - this.$el.children('i').addClass(this._convertDirectionToIcon(dir)); + + this.column.set('direction', dir); + } + + var columnDirection = this.column.get('direction'); + + if (!columnDirection && this.collection.state) { + var key = this.collection.state.sortKey; + var order = this.collection.state.order; + + if (key === this.column.get('name')) { + columnDirection = order; } - this._direction = dir; } - return this._direction; + return columnDirection; }, onClick: function (e) { e.preventDefault(); - var columnName = this.column.get('name'); - - if (this.column.get('sortable')) { - if (this.direction() === 'ascending') { - this.sort(columnName, 'descending', function (left, right) { - var leftVal = left.get(columnName); - var rightVal = right.get(columnName); - if (leftVal === rightVal) { - return 0; - } - else if (leftVal > rightVal) { - return -1; - } - return 1; - }); + var collection = this.collection; + var event = 'backgrid:sort'; + + function toggleSort(header, col) { + collection.state.sortKey = col.get('name'); + var direction = header.direction(); + if (direction === 'ascending' || direction === -1) + { + collection.state.order = 'descending'; + collection.trigger(event, col, 'descending'); } - else { - this.sort(columnName, 'ascending', function (left, right) { - var leftVal = left.get(columnName); - var rightVal = right.get(columnName); - if (leftVal === rightVal) { - return 0; - } - else if (leftVal < rightVal) { - return -1; - } - return 1; - }); + else + { + collection.state.order = 'ascending'; + collection.trigger(event, col, 'ascending'); } } + + var column = this.column; + var sortable = Backgrid.callByNeed(column.sortable(), column, this.collection); + if (sortable) { + toggleSort(this, column); + } + }, + + _resetCellDirection: function (columnToSort, direction) { + if (columnToSort !== this.column) + { + this.direction(null); + } + else + { + this.direction(direction); + } }, _convertDirectionToIcon: function (dir) { - if (dir === 'ascending') { + if (dir === 'ascending' || dir === -1) { return 'icon-sort-up'; } return 'icon-sort-down'; }, - _convertIntToDirection: function (dir) { - if (dir === '-1') { - return 'ascending'; - } + _setSortIcon: function (dir) { + this._removeSortIcon(); + this.$el.children('i').addClass(this._convertDirectionToIcon(dir)); + }, - return 'descending'; + _removeSortIcon: function () { + this.$el.children('i').removeClass('icon-sort-up icon-sort-down'); } }); diff --git a/src/UI/app.js b/src/UI/app.js index 4edde0dc1..e585603b3 100644 --- a/src/UI/app.js +++ b/src/UI/app.js @@ -165,7 +165,8 @@ require.config({ renderable: true, formatter : undefined, cell : undefined, - headerCell: 'NzbDrone' + headerCell: 'NzbDrone', + sortType : 'toggle' }; }); From 7dc641b3a9f9b3e34455d91a094ff510ccbb6f6a Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Thu, 12 Dec 2013 22:37:37 -0800 Subject: [PATCH 3/9] History state is persisted across page reloads now --- src/NzbDrone.Api/History/HistoryModule.cs | 5 +++++ src/UI/Episode/Activity/EpisodeActivityLayout.js | 2 +- src/UI/History/HistoryCollection.js | 9 ++++++--- src/UI/History/Table/HistoryTableLayout.js | 2 +- src/UI/Mixins/AsPersistedStateCollection.js | 12 +++++++++--- 5 files changed, 22 insertions(+), 8 deletions(-) diff --git a/src/NzbDrone.Api/History/HistoryModule.cs b/src/NzbDrone.Api/History/HistoryModule.cs index 7b0aa3c7e..bae183df6 100644 --- a/src/NzbDrone.Api/History/HistoryModule.cs +++ b/src/NzbDrone.Api/History/HistoryModule.cs @@ -39,6 +39,11 @@ namespace NzbDrone.Api.History pagingSpec.FilterExpression = h => h.EpisodeId == i; } + if (pagingSpec.SortKey.Equals("series", StringComparison.InvariantCultureIgnoreCase)) + { + pagingSpec.SortKey = "series.title"; + } + return ApplyToPage(_historyService.Paged, pagingSpec); } diff --git a/src/UI/Episode/Activity/EpisodeActivityLayout.js b/src/UI/Episode/Activity/EpisodeActivityLayout.js index f1ed7cbbb..53de87fdc 100644 --- a/src/UI/Episode/Activity/EpisodeActivityLayout.js +++ b/src/UI/Episode/Activity/EpisodeActivityLayout.js @@ -47,7 +47,7 @@ define( this.model = options.model; this.series = options.series; - this.collection = new HistoryCollection({ episodeId: this.model.id }); + this.collection = new HistoryCollection({ episodeId: this.model.id, tableName: 'episodeActivity' }); this.collection.fetch(); this.listenTo(this.collection, 'sync', this._showTable); }, diff --git a/src/UI/History/HistoryCollection.js b/src/UI/History/HistoryCollection.js index 5341c6507..89abc33a8 100644 --- a/src/UI/History/HistoryCollection.js +++ b/src/UI/History/HistoryCollection.js @@ -2,9 +2,10 @@ define( [ 'History/HistoryModel', - 'backbone.pageable' - ], function (HistoryModel, PageableCollection) { - return PageableCollection.extend({ + 'backbone.pageable', + 'Mixins/AsPersistedStateCollection' + ], function (HistoryModel, PageableCollection, AsPersistedStateCollection) { + var collection = PageableCollection.extend({ url : window.NzbDrone.ApiRoot + '/history', model: HistoryModel, @@ -48,4 +49,6 @@ define( return resp; } }); + + return AsPersistedStateCollection.apply(collection); }); diff --git a/src/UI/History/Table/HistoryTableLayout.js b/src/UI/History/Table/HistoryTableLayout.js index 9e3263c71..25dff7526 100644 --- a/src/UI/History/Table/HistoryTableLayout.js +++ b/src/UI/History/Table/HistoryTableLayout.js @@ -80,7 +80,7 @@ define( initialize: function () { - this.collection = new HistoryCollection(); + this.collection = new HistoryCollection({ tableName: 'history' }); this.listenTo(this.collection, 'sync', this._showTable); }, diff --git a/src/UI/Mixins/AsPersistedStateCollection.js b/src/UI/Mixins/AsPersistedStateCollection.js index de663b794..518209fe7 100644 --- a/src/UI/Mixins/AsPersistedStateCollection.js +++ b/src/UI/Mixins/AsPersistedStateCollection.js @@ -8,9 +8,15 @@ define( var originalInit = this.prototype.initialize; - this.prototype.initialize = function () { + this.prototype.initialize = function (options) { - if (!this.tableName) { + options = options || {}; + + if (options.tableName) { + this.tableName = options.tableName; + } + + if (!this.tableName && !options.tableName) { throw 'tableName is required'; } @@ -19,7 +25,7 @@ define( this.on('backgrid:sort', _storeState, this); if (originalInit) { - originalInit.call(this); + originalInit.call(this, options); } }; From 96868d088fb3437933ce83792b447df1db63fc93 Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Fri, 13 Dec 2013 00:06:56 -0800 Subject: [PATCH 4/9] Fixed series editor --- src/UI/Cells/SeasonFolderCell.js | 5 ++++- src/UI/Series/Editor/SeriesEditorFooterView.js | 3 +++ src/UI/Series/Editor/SeriesEditorLayout.js | 2 +- 3 files changed, 8 insertions(+), 2 deletions(-) diff --git a/src/UI/Cells/SeasonFolderCell.js b/src/UI/Cells/SeasonFolderCell.js index 5c9cc6d1b..e3b437b39 100644 --- a/src/UI/Cells/SeasonFolderCell.js +++ b/src/UI/Cells/SeasonFolderCell.js @@ -8,8 +8,11 @@ define( className : 'season-folder-cell', render: function () { - var seasonFolder = this.model.get('seasonFolder'); + this.$el.empty(); + + var seasonFolder = this.model.get(this.column.get('name')); this.$el.html(seasonFolder.toString()); + return this; } }); diff --git a/src/UI/Series/Editor/SeriesEditorFooterView.js b/src/UI/Series/Editor/SeriesEditorFooterView.js index 42b2b85f4..28a27bca4 100644 --- a/src/UI/Series/Editor/SeriesEditorFooterView.js +++ b/src/UI/Series/Editor/SeriesEditorFooterView.js @@ -94,6 +94,8 @@ define( model.set('rootFolderPath', rootFolderPath.get('path')); } + + model.edited = true; }); SeriesCollection.save(); @@ -150,6 +152,7 @@ define( SeriesCollection.each(function (model) { model.trigger('backgrid:select', model, false); + model.edited = false; }); } }); diff --git a/src/UI/Series/Editor/SeriesEditorLayout.js b/src/UI/Series/Editor/SeriesEditorLayout.js index e82149fa9..84ee8bca7 100644 --- a/src/UI/Series/Editor/SeriesEditorLayout.js +++ b/src/UI/Series/Editor/SeriesEditorLayout.js @@ -68,7 +68,7 @@ define( cell : QualityProfileCell }, { - name : 'monitored', + name : 'seasonFolder', label: 'Season Folder', cell : SeasonFolderCell }, From df055d191fc286639168ff9298100cec604b0e20 Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Fri, 13 Dec 2013 00:07:13 -0800 Subject: [PATCH 5/9] Logs and series are now persisted --- src/UI/Series/SeriesCollection.js | 21 ++++++++++++--------- src/UI/System/Logs/LogsCollection.js | 14 +++++++++++--- 2 files changed, 23 insertions(+), 12 deletions(-) diff --git a/src/UI/Series/SeriesCollection.js b/src/UI/Series/SeriesCollection.js index 437a83768..462e88fdc 100644 --- a/src/UI/Series/SeriesCollection.js +++ b/src/UI/Series/SeriesCollection.js @@ -3,22 +3,24 @@ define( [ 'underscore', 'backbone', + 'backbone.pageable', 'Series/SeriesModel', - 'api!series' - ], function (_, Backbone, SeriesModel, SeriesData) { + 'api!series', + 'Mixins/AsPersistedStateCollection' + ], function (_, Backbone, PageableCollection, SeriesModel, SeriesData, AsPersistedStateCollection) { var Collection = Backbone.Collection.extend({ url : window.NzbDrone.ApiRoot + '/series', model: SeriesModel, - - comparator: function (model) { - return model.get('title'); - }, + tableName: 'series', state: { sortKey: 'title', - order : 'ascending' + order : -1, + pageSize: 1000 }, + mode: 'client', + save: function () { var self = this; @@ -31,7 +33,7 @@ define( toJSON: function() { return self.filter(function (model) { - return model.hasChanged(); + return model.edited; }); } }); @@ -45,6 +47,7 @@ define( } }); - var collection = new Collection(SeriesData); + var mixedIn = AsPersistedStateCollection.call(Collection); + var collection = new mixedIn(SeriesData); return collection; }); diff --git a/src/UI/System/Logs/LogsCollection.js b/src/UI/System/Logs/LogsCollection.js index 14c80374f..350ed1522 100644 --- a/src/UI/System/Logs/LogsCollection.js +++ b/src/UI/System/Logs/LogsCollection.js @@ -1,10 +1,16 @@ 'use strict'; -define(['backbone.pageable', 'System/Logs/LogsModel'], - function (PagableCollection, LogsModel) { - return PagableCollection.extend({ +define( + [ + 'backbone.pageable', + 'System/Logs/LogsModel', + 'Mixins/AsPersistedStateCollection' + ], + function (PagableCollection, LogsModel, AsPersistedStateCollection) { + var collection = PagableCollection.extend({ url : window.NzbDrone.ApiRoot + '/log', model: LogsModel, + tableName: 'logs', state: { pageSize: 50, @@ -36,4 +42,6 @@ define(['backbone.pageable', 'System/Logs/LogsModel'], return resp; } }); + + return AsPersistedStateCollection.call(collection); }); From 4d6d477947057da8a0fbfa2b3a06dfde91b5a6de Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Fri, 13 Dec 2013 08:20:48 -0800 Subject: [PATCH 6/9] Using SortValue instead of API hack for history Fixed jshint for series collection --- src/NzbDrone.Api/History/HistoryModule.cs | 5 ----- src/UI/History/Table/HistoryTableLayout.js | 3 ++- src/UI/Mixins/AsPersistedStateCollection.js | 2 +- src/UI/Series/SeriesCollection.js | 4 ++-- 4 files changed, 5 insertions(+), 9 deletions(-) diff --git a/src/NzbDrone.Api/History/HistoryModule.cs b/src/NzbDrone.Api/History/HistoryModule.cs index bae183df6..7b0aa3c7e 100644 --- a/src/NzbDrone.Api/History/HistoryModule.cs +++ b/src/NzbDrone.Api/History/HistoryModule.cs @@ -39,11 +39,6 @@ namespace NzbDrone.Api.History pagingSpec.FilterExpression = h => h.EpisodeId == i; } - if (pagingSpec.SortKey.Equals("series", StringComparison.InvariantCultureIgnoreCase)) - { - pagingSpec.SortKey = "series.title"; - } - return ApplyToPage(_historyService.Paged, pagingSpec); } diff --git a/src/UI/History/Table/HistoryTableLayout.js b/src/UI/History/Table/HistoryTableLayout.js index 25dff7526..3571c69d1 100644 --- a/src/UI/History/Table/HistoryTableLayout.js +++ b/src/UI/History/Table/HistoryTableLayout.js @@ -45,7 +45,8 @@ define( { name : 'series', label: 'Series', - cell : SeriesTitleCell + cell : SeriesTitleCell, + sortValue: 'series.title' }, { name : 'episode', diff --git a/src/UI/Mixins/AsPersistedStateCollection.js b/src/UI/Mixins/AsPersistedStateCollection.js index 518209fe7..721c4c68a 100644 --- a/src/UI/Mixins/AsPersistedStateCollection.js +++ b/src/UI/Mixins/AsPersistedStateCollection.js @@ -40,7 +40,7 @@ define( var _storeState = function (column, sortDirection) { var order = _convertDirectionToInt(sortDirection); - var sortKey = column.get('name'); + var sortKey = column.has('sortValue') ? column.get('sortValue') : column.get('name'); Config.setValue('{0}.sortKey'.format(this.tableName), sortKey); Config.setValue('{0}.sortDirection'.format(this.tableName), order); diff --git a/src/UI/Series/SeriesCollection.js b/src/UI/Series/SeriesCollection.js index 462e88fdc..74c16f27d 100644 --- a/src/UI/Series/SeriesCollection.js +++ b/src/UI/Series/SeriesCollection.js @@ -47,7 +47,7 @@ define( } }); - var mixedIn = AsPersistedStateCollection.call(Collection); - var collection = new mixedIn(SeriesData); + var MixedIn = AsPersistedStateCollection.call(Collection); + var collection = new MixedIn(SeriesData); return collection; }); From 6ba17782aa2134de479f288b252e4c6ae119372a Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Fri, 27 Dec 2013 00:31:34 -0800 Subject: [PATCH 7/9] Sorting on all series views is now working New: Sorting is now persisted on a per page and browser basis New: Series lists now support sorting on all views --- src/UI/Content/theme.less | 15 ++ src/UI/Mixins/AsPersistedStateCollection.js | 19 +- src/UI/Series/Index/SeriesIndexLayout.js | 248 +++++++++++------- src/UI/Series/SeriesCollection.js | 2 +- src/UI/Shared/Grid/HeaderCell.js | 12 + src/UI/Shared/Toolbar/ButtonModel.js | 20 +- .../Shared/Toolbar/Radio/RadioButtonView.js | 2 - .../Sorting/SortingButtonCollectionView.js | 87 ++++++ .../SortingButtonCollectionViewTemplate.html | 8 + .../Toolbar/Sorting/SortingButtonView.js | 84 ++++++ .../Sorting/SortingButtonViewTemplate.html | 4 + src/UI/Shared/Toolbar/ToolbarLayout.js | 12 +- .../Shared/Toolbar/ToolbarLayoutTemplate.html | 8 +- 13 files changed, 406 insertions(+), 115 deletions(-) create mode 100644 src/UI/Shared/Toolbar/Sorting/SortingButtonCollectionView.js create mode 100644 src/UI/Shared/Toolbar/Sorting/SortingButtonCollectionViewTemplate.html create mode 100644 src/UI/Shared/Toolbar/Sorting/SortingButtonView.js create mode 100644 src/UI/Shared/Toolbar/Sorting/SortingButtonViewTemplate.html diff --git a/src/UI/Content/theme.less b/src/UI/Content/theme.less index 8bf71358e..bf90d7856 100644 --- a/src/UI/Content/theme.less +++ b/src/UI/Content/theme.less @@ -46,6 +46,21 @@ .page-toolbar { margin-top : 10px; margin-bottom : 30px; + + .toolbar-group { + display: inline-block; + } + + .sorting-buttons { + li { + a { + span { + display: inline-block; + width: 110px; + } + } + } + } } .page-container { diff --git a/src/UI/Mixins/AsPersistedStateCollection.js b/src/UI/Mixins/AsPersistedStateCollection.js index 721c4c68a..2a957cc84 100644 --- a/src/UI/Mixins/AsPersistedStateCollection.js +++ b/src/UI/Mixins/AsPersistedStateCollection.js @@ -1,8 +1,8 @@ 'use strict'; define( - ['Config'], - function (Config) { + ['underscore', 'Config'], + function (_, Config) { return function () { @@ -22,7 +22,8 @@ define( _setInitialState.call(this); - this.on('backgrid:sort', _storeState, this); + this.on('backgrid:sort', _storeStateFromBackgrid, this); + this.on('drone:sort', _storeState, this); if (originalInit) { originalInit.call(this, options); @@ -38,9 +39,17 @@ define( this.state.order = order; }; - var _storeState = function (column, sortDirection) { + var _storeStateFromBackgrid = function (column, sortDirection) { var order = _convertDirectionToInt(sortDirection); - var sortKey = column.has('sortValue') ? column.get('sortValue') : column.get('name'); + var sortKey = column.has('sortValue') && _.isString(column.get('sortValue')) ? column.get('sortValue') : column.get('name'); + + Config.setValue('{0}.sortKey'.format(this.tableName), sortKey); + Config.setValue('{0}.sortDirection'.format(this.tableName), order); + }; + + var _storeState = function (sortModel, sortDirection) { + var order = _convertDirectionToInt(sortDirection); + var sortKey = sortModel.get('name'); Config.setValue('{0}.sortKey'.format(this.tableName), sortKey); Config.setValue('{0}.sortDirection'.format(this.tableName), order); diff --git a/src/UI/Series/Index/SeriesIndexLayout.js b/src/UI/Series/Index/SeriesIndexLayout.js index 9a6390cb7..adbd0a84c 100644 --- a/src/UI/Series/Index/SeriesIndexLayout.js +++ b/src/UI/Series/Index/SeriesIndexLayout.js @@ -41,66 +41,65 @@ define( template: 'Series/Index/SeriesIndexLayoutTemplate', regions: { - seriesRegion: '#x-series', - toolbar : '#x-toolbar', - footer : '#x-series-footer' + seriesRegion : '#x-series', + toolbar : '#x-toolbar', + footer : '#x-series-footer' }, - columns: - [ - { - name : 'statusWeight', - label : '', - cell : SeriesStatusCell - }, - { - name : 'title', - label : 'Title', - cell : SeriesTitleCell, - cellValue: 'this' - }, - { - name : 'seasonCount', - label: 'Seasons', - cell : 'integer' - }, - { - name : 'qualityProfileId', - label: 'Quality', - cell : QualityProfileCell - }, - { - name : 'network', - label: 'Network', - cell : 'string' - }, - { - name : 'nextAiring', - label : 'Next Airing', - cell : RelativeDateCell, - sortValue : function (model) { - var nextAiring = model.get('nextAiring'); - - if (!nextAiring) { - return Number.MAX_VALUE; - } + columns: [ + { + name : 'statusWeight', + label : '', + cell : SeriesStatusCell + }, + { + name : 'title', + label : 'Title', + cell : SeriesTitleCell, + cellValue: 'this' + }, + { + name : 'seasonCount', + label: 'Seasons', + cell : 'integer' + }, + { + name : 'qualityProfileId', + label: 'Quality', + cell : QualityProfileCell + }, + { + name : 'network', + label: 'Network', + cell : 'string' + }, + { + name : 'nextAiring', + label : 'Next Airing', + cell : RelativeDateCell, + sortValue : function (model) { + var nextAiring = model.get('nextAiring'); - return Moment(nextAiring).unix(); + if (!nextAiring) { + return Number.MAX_VALUE; } - }, - { - name : 'percentOfEpisodes', - label : 'Episodes', - cell : EpisodeProgressCell, - className: 'episode-progress-cell' - }, - { - name : 'this', - label : '', - sortable: false, - cell : SeriesActionsCell + + return Moment(nextAiring).unix(); } - ], + }, + { + name : 'percentOfEpisodes', + label : 'Episodes', + cell : EpisodeProgressCell, + className: 'episode-progress-cell' + }, + { + name : 'this', + label : '', + sortable: false, + cell : SeriesActionsCell + } + ], leftSideButtons: { type : 'default', @@ -138,6 +137,86 @@ define( ] }, + sortingOptions: { + type : 'sorting', + storeState : false, + viewCollection: SeriesCollection, + items : + [ + { + title: 'Title', + name : 'title' + }, + { + title: 'Seasons', + name : 'seasonCount' + }, + { + title: 'Quality', + name : 'qualityProfileId' + }, + { + title: 'Network', + name : 'network' + }, + { + title : 'Next Airing', + name : 'nextAiring', + sortValue : function (model) { + var nextAiring = model.get('nextAiring'); + + if (!nextAiring) { + return Number.MAX_VALUE; + } + + return Moment(nextAiring).unix(); + } + }, + { + title: 'Episodes', + name : 'percentOfEpisodes' + } + ] + }, + + initialize: function () { + this.seriesCollection = SeriesCollection; + + this.listenTo(SeriesCollection, 'sync', this._renderView); + this.listenTo(SeriesCollection, 'remove', this._renderView); + + this.viewButtons = { + type : 'radio', + storeState : true, + menuKey : 'seriesViewMode', + defaultAction: 'listView', + items : + [ + { + key : 'posterView', + title : '', + tooltip : 'Posters', + icon : 'icon-th-large', + callback: this._showPosters + }, + { + key : 'listView', + title : '', + tooltip : 'Overview List', + icon : 'icon-th-list', + callback: this._showList + }, + { + key : 'tableView', + title : '', + tooltip : 'Table', + icon : 'icon-table', + callback: this._showTable + } + ] + }; + }, + _showTable: function () { this.currentView = new Backgrid.Grid({ collection: SeriesCollection, @@ -145,25 +224,19 @@ define( className : 'table table-hover' }); - this._renderView(); this._fetchCollection(); }, _showList: function () { - this.currentView = new ListCollectionView(); - this._fetchCollection(); - }, + this.currentView = new ListCollectionView({ collection: SeriesCollection }); - _showPosters: function () { - this.currentView = new PosterCollectionView(); this._fetchCollection(); }, - initialize: function () { - this.seriesCollection = SeriesCollection; + _showPosters: function () { + this.currentView = new PosterCollectionView({ collection: SeriesCollection }); - this.listenTo(SeriesCollection, 'sync', this._renderView); - this.listenTo(SeriesCollection, 'remove', this._renderView); + this._fetchCollection(); }, _renderView: function () { @@ -173,7 +246,6 @@ define( this.toolbar.close(); } else { - this.currentView.collection = SeriesCollection; this.seriesRegion.show(this.currentView); this._showToolbar(); @@ -196,42 +268,18 @@ define( return; } - var viewButtons = { - type : 'radio', - storeState : true, - menuKey : 'seriesViewMode', - defaultAction: 'listView', - items : - [ - { - key : 'posterView', - title : '', - tooltip : 'Posters', - icon : 'icon-th-large', - callback: this._showPosters - }, - { - key : 'listView', - title : '', - tooltip : 'Overview List', - icon : 'icon-th-list', - callback: this._showList - }, - { - key : 'tableView', - title : '', - tooltip : 'Table', - icon : 'icon-table', - callback: this._showTable - } - ] - }; + var rightButtons = [ + this.viewButtons + ]; + + if (this.showSortingButton) { + rightButtons.splice(0, 0, this.sortingOptions); + } + + rightButtons.splice(0, 0, this.sortingOptions); this.toolbar.show(new ToolbarLayout({ - right : - [ - viewButtons - ], + right : rightButtons, left : [ this.leftSideButtons diff --git a/src/UI/Series/SeriesCollection.js b/src/UI/Series/SeriesCollection.js index 74c16f27d..2c7b049f8 100644 --- a/src/UI/Series/SeriesCollection.js +++ b/src/UI/Series/SeriesCollection.js @@ -8,7 +8,7 @@ define( 'api!series', 'Mixins/AsPersistedStateCollection' ], function (_, Backbone, PageableCollection, SeriesModel, SeriesData, AsPersistedStateCollection) { - var Collection = Backbone.Collection.extend({ + var Collection = PageableCollection.extend({ url : window.NzbDrone.ApiRoot + '/series', model: SeriesModel, tableName: 'series', diff --git a/src/UI/Shared/Grid/HeaderCell.js b/src/UI/Shared/Grid/HeaderCell.js index 2f1f3c429..45b402589 100644 --- a/src/UI/Shared/Grid/HeaderCell.js +++ b/src/UI/Shared/Grid/HeaderCell.js @@ -11,6 +11,14 @@ define( 'click': 'onClick' }, + _originalInit: Backgrid.HeaderCell.prototype.initialize, + + initialize: function (options) { + this._originalInit.call(this, options); + + this.listenTo(this.collection, 'drone:sort', this.render); + }, + render: function () { this.$el.empty(); this.$el.append(this.column.get('label')); @@ -37,6 +45,10 @@ define( if (key === this.column.get('name')) { this._setSortIcon(order); } + + else { + this._removeSortIcon(); + } } return this; diff --git a/src/UI/Shared/Toolbar/ButtonModel.js b/src/UI/Shared/Toolbar/ButtonModel.js index f6c7c8a64..b7bd6d4dc 100644 --- a/src/UI/Shared/Toolbar/ButtonModel.js +++ b/src/UI/Shared/Toolbar/ButtonModel.js @@ -1,13 +1,29 @@ 'use strict'; define( [ + 'underscore', 'backbone' - ], function (Backbone) { + ], function (_, Backbone) { return Backbone.Model.extend({ defaults: { 'target' : '/nzbdrone/route', 'title' : '', 'active' : false, - 'tooltip': undefined } + 'tooltip': undefined + }, + + sortValue: function () { + var sortValue = this.get('sortValue'); + if (_.isString(sortValue)) { + return this[sortValue]; + } + else if (_.isFunction(sortValue)) { + return sortValue; + } + + return function (model, colName) { + return model.get(colName); + }; + } }); }); diff --git a/src/UI/Shared/Toolbar/Radio/RadioButtonView.js b/src/UI/Shared/Toolbar/Radio/RadioButtonView.js index 3f5be2a6f..1fb788250 100644 --- a/src/UI/Shared/Toolbar/Radio/RadioButtonView.js +++ b/src/UI/Shared/Toolbar/Radio/RadioButtonView.js @@ -13,7 +13,6 @@ define( 'click': 'onClick' }, - initialize: function () { this.storageKey = this.model.get('menuKey') + ':' + this.model.get('key'); @@ -53,7 +52,6 @@ define( callback.call(this.model.ownerContext); } } - }); }); diff --git a/src/UI/Shared/Toolbar/Sorting/SortingButtonCollectionView.js b/src/UI/Shared/Toolbar/Sorting/SortingButtonCollectionView.js new file mode 100644 index 000000000..a79abc2f0 --- /dev/null +++ b/src/UI/Shared/Toolbar/Sorting/SortingButtonCollectionView.js @@ -0,0 +1,87 @@ +'use strict'; +define( + [ + 'backbone.pageable', + 'marionette', + 'Shared/Toolbar/Sorting/SortingButtonView' + ], function (PageableCollection, Marionette, ButtonView) { + return Marionette.CompositeView.extend({ + itemView : ButtonView, + template : 'Shared/Toolbar/Sorting/SortingButtonCollectionViewTemplate', + itemViewContainer: '.dropdown-menu', + + initialize: function (options) { + this.viewCollection = options.viewCollection; + this.listenTo(this.viewCollection, 'drone:sort', this.sort); + }, + + itemViewOptions: function () { + return { + viewCollection: this.viewCollection + }; + }, + + sort: function (sortModel, sortDirection) { + var collection = this.viewCollection; + + var order; + if (sortDirection === 'ascending') { + order = -1; + } + else if (sortDirection === 'descending') { + order = 1; + } + else { + order = null; + } + + var comparator = this.makeComparator(sortModel.get('name'), order, + order ? + sortModel.sortValue() : + function (model) { + return model.cid; + }); + + if (PageableCollection && + collection instanceof PageableCollection) { + + collection.setSorting(order && sortModel.get('name'), order, + {sortValue: sortModel.sortValue()}); + + if (collection.mode === 'client') { + if (collection.fullCollection.comparator === null) { + collection.fullCollection.comparator = comparator; + } + collection.fullCollection.sort(); + } + else { + collection.fetch({reset: true}); + } + } + else { + collection.comparator = comparator; + collection.sort(); + } + + return this; + }, + + makeComparator: function (attr, order, func) { + + return function (left, right) { + // extract the values from the models + var l = func(left, attr), r = func(right, attr), t; + + // if descending order, swap left and right + if (order === 1) t = l, l = r, r = t; + + // compare as usual + if (l === r) return 0; + else if (l < r) return -1; + return 1; + }; + } + }); + }); + + diff --git a/src/UI/Shared/Toolbar/Sorting/SortingButtonCollectionViewTemplate.html b/src/UI/Shared/Toolbar/Sorting/SortingButtonCollectionViewTemplate.html new file mode 100644 index 000000000..62f6da91e --- /dev/null +++ b/src/UI/Shared/Toolbar/Sorting/SortingButtonCollectionViewTemplate.html @@ -0,0 +1,8 @@ +
+ + Sort + + +
diff --git a/src/UI/Shared/Toolbar/Sorting/SortingButtonView.js b/src/UI/Shared/Toolbar/Sorting/SortingButtonView.js new file mode 100644 index 000000000..7421e628f --- /dev/null +++ b/src/UI/Shared/Toolbar/Sorting/SortingButtonView.js @@ -0,0 +1,84 @@ +'use strict'; +define( + [ + 'backbone', + 'marionette', + 'underscore' + ], function (Backbone, Marionette, _) { + + return Marionette.ItemView.extend({ + template : 'Shared/Toolbar/Sorting/SortingButtonViewTemplate', + tagName : 'li', + + ui: { + icon: 'i' + }, + + events: { + 'click': 'onClick' + }, + + initialize: function (options) { + this.viewCollection = options.viewCollection; + this.listenTo(this.viewCollection, 'drone:sort', this.render); + this.listenTo(this.viewCollection, 'backgrid:sort', this.render); + }, + + onRender: function () { + if (this.viewCollection.state) { + var key = this.viewCollection.state.sortKey; + var order = this.viewCollection.state.order; + + if (key === this.model.get('name')) { + this._setSortIcon(order); + } + + else { + this._removeSortIcon(); + } + } + }, + + onClick: function (e) { + e.preventDefault(); + + var collection = this.viewCollection; + var event = 'drone:sort'; + + collection.state.sortKey = this.model.get('name'); + var direction = collection.state.order; + + if (direction === 'ascending' || direction === -1) + { + collection.state.order = 'descending'; + collection.trigger(event, this.model, 'descending'); + } + else + { + collection.state.order = 'ascending'; + collection.trigger(event, this.model, 'ascending'); + } + }, + + _convertDirectionToIcon: function (dir) { + if (dir === 'ascending' || dir === -1) { + return 'icon-sort-up'; + } + + return 'icon-sort-down'; + }, + + _setSortIcon: function (dir) { + this._removeSortIcon(); + this.ui.icon.addClass(this._convertDirectionToIcon(dir)); + }, + + _removeSortIcon: function () { + this.ui.icon.removeClass('icon-sort-up icon-sort-down'); + } + }); + }); + + + + diff --git a/src/UI/Shared/Toolbar/Sorting/SortingButtonViewTemplate.html b/src/UI/Shared/Toolbar/Sorting/SortingButtonViewTemplate.html new file mode 100644 index 000000000..a969d5dc2 --- /dev/null +++ b/src/UI/Shared/Toolbar/Sorting/SortingButtonViewTemplate.html @@ -0,0 +1,4 @@ + + {{title}} + + \ No newline at end of file diff --git a/src/UI/Shared/Toolbar/ToolbarLayout.js b/src/UI/Shared/Toolbar/ToolbarLayout.js index 7f4b64bf9..8a907dad6 100644 --- a/src/UI/Shared/Toolbar/ToolbarLayout.js +++ b/src/UI/Shared/Toolbar/ToolbarLayout.js @@ -6,8 +6,9 @@ define( 'Shared/Toolbar/ButtonModel', 'Shared/Toolbar/Radio/RadioButtonCollectionView', 'Shared/Toolbar/Button/ButtonCollectionView', + 'Shared/Toolbar/Sorting/SortingButtonCollectionView', 'underscore' - ], function (Marionette, ButtonCollection, ButtonModel, RadioButtonCollectionView, ButtonCollectionView,_) { + ], function (Marionette, ButtonCollection, ButtonModel, RadioButtonCollectionView, ButtonCollectionView, SortingButtonCollectionView, _) { return Marionette.Layout.extend({ template: 'Shared/Toolbar/ToolbarLayoutTemplate', @@ -78,6 +79,15 @@ define( }); break; } + case 'sorting': + { + buttonGroupView = new SortingButtonCollectionView({ + collection : groupCollection, + menu : buttonGroup, + viewCollection: buttonGroup.viewCollection + }); + break; + } default : { buttonGroupView = new ButtonCollectionView({ diff --git a/src/UI/Shared/Toolbar/ToolbarLayoutTemplate.html b/src/UI/Shared/Toolbar/ToolbarLayoutTemplate.html index c2f497b3f..b4cd4dcda 100644 --- a/src/UI/Shared/Toolbar/ToolbarLayoutTemplate.html +++ b/src/UI/Shared/Toolbar/ToolbarLayoutTemplate.html @@ -1,8 +1,8 @@ 
-
-
+
+
-
-
+
+
From 8dcfbe7b9d59a92e5effdec7164cb273e934bd66 Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Fri, 27 Dec 2013 01:47:51 -0800 Subject: [PATCH 8/9] Little hack to deal with backgrid's setting of sortKey --- src/NzbDrone.Api/History/HistoryModule.cs | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/src/NzbDrone.Api/History/HistoryModule.cs b/src/NzbDrone.Api/History/HistoryModule.cs index 7b0aa3c7e..080e74a46 100644 --- a/src/NzbDrone.Api/History/HistoryModule.cs +++ b/src/NzbDrone.Api/History/HistoryModule.cs @@ -33,6 +33,13 @@ namespace NzbDrone.Api.History SortDirection = pagingResource.SortDirection }; + //This is a hack to deal with backgrid setting the sortKey to the column name instead of sortValue + if (pagingSpec.SortKey.Equals("series", StringComparison.InvariantCultureIgnoreCase)) + { + pagingSpec.SortKey = "series.title"; + } + + if (episodeId.HasValue) { int i = (int)episodeId; From 5c1f2b2dd7a1fd951cabdebfaaa806e77feb7dbd Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Wed, 1 Jan 2014 23:11:29 -0800 Subject: [PATCH 9/9] Better css for sorting button text --- src/UI/Content/theme.less | 10 +++------- .../Toolbar/Sorting/SortingButtonViewTemplate.html | 2 +- 2 files changed, 4 insertions(+), 8 deletions(-) diff --git a/src/UI/Content/theme.less b/src/UI/Content/theme.less index bf90d7856..4e72e020b 100644 --- a/src/UI/Content/theme.less +++ b/src/UI/Content/theme.less @@ -52,13 +52,9 @@ } .sorting-buttons { - li { - a { - span { - display: inline-block; - width: 110px; - } - } + .sorting-title { + display: inline-block; + width: 110px; } } } diff --git a/src/UI/Shared/Toolbar/Sorting/SortingButtonViewTemplate.html b/src/UI/Shared/Toolbar/Sorting/SortingButtonViewTemplate.html index a969d5dc2..b2663c548 100644 --- a/src/UI/Shared/Toolbar/Sorting/SortingButtonViewTemplate.html +++ b/src/UI/Shared/Toolbar/Sorting/SortingButtonViewTemplate.html @@ -1,4 +1,4 @@  - {{title}} + {{title}} \ No newline at end of file