diff --git a/bower.json b/bower.json index 10938d70..c1e89b5b 100644 --- a/bower.json +++ b/bower.json @@ -1,6 +1,6 @@ { "name": "jsgrid", - "version": "1.1.0", + "version": "1.2.0", "main": [ "dist/jsgrid.js", "dist/jsgrid.css", diff --git a/dist/jsgrid-theme.css b/dist/jsgrid-theme.css new file mode 100644 index 00000000..14ecb864 --- /dev/null +++ b/dist/jsgrid-theme.css @@ -0,0 +1,239 @@ +/* + * jsGrid v1.2.0 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid-grid-header, +.jsgrid-grid-body, +.jsgrid-header-row > th, +.jsgrid-filter-row > td, +.jsgrid-insert-row > td, +.jsgrid-edit-row > td { + border: 1px solid #e9e9e9; +} + +.jsgrid-header-row > th { + border-top: 0; +} + +.jsgrid-header-row > th, .jsgrid-filter-row > td, .jsgrid-insert-row > td { + border-bottom: 0; +} + +.jsgrid-header-row > th:first-child, .jsgrid-filter-row > td:first-child, .jsgrid-insert-row > td:first-child { + border-left: none; +} + +.jsgrid-header-row > th:last-child, .jsgrid-filter-row > td:last-child, .jsgrid-insert-row > td:last-child { + border-right: none; +} + +.jsgrid-grid-header { + background: #f9f9f9; +} + +.jsgrid-header-scrollbar { + scrollbar-arrow-color: #f1f1f1; + scrollbar-base-color: #f1f1f1; + scrollbar-3dlight-color: #f1f1f1; + scrollbar-highlight-color: #f1f1f1; + scrollbar-track-color: #f1f1f1; + scrollbar-shadow-color: #f1f1f1; + scrollbar-dark-shadow-color: #f1f1f1; +} + +.jsgrid-header-scrollbar::-webkit-scrollbar { + visibility: hidden; +} + +.jsgrid-header-scrollbar::-webkit-scrollbar-track { + background: #f1f1f1; +} + +.jsgrid-header-sortable:hover { + cursor: pointer; + background: #fcfcfc; +} + +.jsgrid-header-row .jsgrid-header-sort { + background: #c4e2ff; +} + +.jsgrid-header-sort:before { + content: " "; + display: block; + float: left; + width: 0; + height: 0; + border-style: solid; +} + +.jsgrid-header-sort-asc:before { + border-width: 0 5px 5px 5px; + border-color: transparent transparent #009a67 transparent; +} + +.jsgrid-header-sort-desc:before { + border-width: 5px 5px 0 5px; + border-color: #009a67 transparent transparent transparent; +} + +.jsgrid-grid-body { + border-top: none; +} + +.jsgrid-grid-body td { + border: #f3f3f3 1px solid; +} + +.jsgrid-grid-body tr:first-child td { + border-top: none; +} + +.jsgrid-grid-body tr td:first-child { + border-left: none; +} + +.jsgrid-grid-body tr td:last-child { + border-right: none; +} + +.jsgrid-row > td { + background: #fff; +} + +.jsgrid-alt-row > td { + background: #fcfcfc; +} + +.jsgrid-header-row > th { + background: #f9f9f9; +} + +.jsgrid-filter-row > td { + background: #fcfcfc; +} + +.jsgrid-insert-row > td { + background: #e3ffe5; +} + +.jsgrid-edit-row > td { + background: #fdffe3; +} + +.jsgrid-selected-row > td { + background: #c4e2ff; + border-color: #c4e2ff; +} + +.jsgrid-nodata-row td { + background: #fff; +} + +.jsgrid-pager-current-page { + font-weight: bold; +} + +.jsgrid-pager-nav-inactive-button a { + color: #d3d3d3; +} + +.jsgrid-button + .jsgrid-button { + margin-left: 5px; +} + +.jsgrid-button:hover { + opacity: .5; + transition: opacity 200ms linear; +} + +.jsgrid .jsgrid-button { + width: 16px; + height: 16px; + border: none; + cursor: pointer; + background-image: url(); + background-repeat: no-repeat; + background-color: transparent; +} + +@media only screen and (-webkit-min-device-pixel-ratio: 2), only screen and (min-device-pixel-ratio: 2) { + .jsgrid .jsgrid-button { + background-image: url(); + background-size: 24px 352px; + } +} + +.jsgrid .jsgrid-mode-button { + width: 24px; + height: 24px; +} + +.jsgrid-mode-on-button { + opacity: .5; +} + +.jsgrid-cancel-button { background-position: 0 0; width: 16px; height: 16px; } +.jsgrid-clear-filter-button { background-position: 0 -40px; width: 16px; height: 16px; } +.jsgrid-delete-button { background-position: 0 -80px; width: 16px; height: 16px; } +.jsgrid-edit-button { background-position: 0 -120px; width: 16px; height: 16px; } +.jsgrid-insert-mode-button { background-position: 0 -160px; width: 24px; height: 24px; } +.jsgrid-insert-button { background-position: 0 -208px; width: 16px; height: 16px; } +.jsgrid-search-mode-button { background-position: 0 -248px; width: 24px; height: 24px; } +.jsgrid-search-button { background-position: 0 -296px; width: 16px; height: 16px; } +.jsgrid-update-button { background-position: 0 -336px; width: 16px; height: 16px; } + + +.jsgrid-load-shader { + background: #ddd; + opacity: .5; + filter: alpha(opacity=50); +} + +.jsgrid-load-panel { + width: 15em; + height: 5em; + background: #fff; + border: 1px solid #e9e9e9; + padding-top: 3em; + text-align: center; +} + +.jsgrid-load-panel:before { + content: ' '; + position: absolute; + top: .5em; + left: 50%; + margin-left: -1em; + width: 2em; + height: 2em; + border: 2px solid #009a67; + border-right-color: transparent; + border-radius: 50%; + -webkit-animation: indicator 1s linear infinite; + animation: indicator 1s linear infinite; +} + +@-webkit-keyframes indicator +{ + from { -webkit-transform: rotate(0deg); } + 50% { -webkit-transform: rotate(180deg); } + to { -webkit-transform: rotate(360deg); } +} + +@keyframes indicator +{ + from { transform: rotate(0deg); } + 50% { transform: rotate(180deg); } + to { transform: rotate(360deg); } +} + +/* old IE */ +.jsgrid-load-panel { + padding-top: 1.5em\9; +} +.jsgrid-load-panel:before { + display: none\9; +} diff --git a/dist/jsgrid-theme.min.css b/dist/jsgrid-theme.min.css new file mode 100644 index 00000000..9afc8407 --- /dev/null +++ b/dist/jsgrid-theme.min.css @@ -0,0 +1,7 @@ +/* + * jsGrid v1.2.0 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid-edit-row>td,.jsgrid-filter-row>td,.jsgrid-grid-body,.jsgrid-grid-header,.jsgrid-header-row>th,.jsgrid-insert-row>td{border:1px solid #e9e9e9}.jsgrid-header-row>th{border-top:0}.jsgrid-filter-row>td,.jsgrid-header-row>th,.jsgrid-insert-row>td{border-bottom:0}.jsgrid-filter-row>td:first-child,.jsgrid-header-row>th:first-child,.jsgrid-insert-row>td:first-child{border-left:none}.jsgrid-filter-row>td:last-child,.jsgrid-header-row>th:last-child,.jsgrid-insert-row>td:last-child{border-right:none}.jsgrid-grid-header{background:#f9f9f9}.jsgrid-header-scrollbar{scrollbar-arrow-color:#f1f1f1;scrollbar-base-color:#f1f1f1;scrollbar-3dlight-color:#f1f1f1;scrollbar-highlight-color:#f1f1f1;scrollbar-track-color:#f1f1f1;scrollbar-shadow-color:#f1f1f1;scrollbar-dark-shadow-color:#f1f1f1}.jsgrid-header-scrollbar::-webkit-scrollbar{visibility:hidden}.jsgrid-header-scrollbar::-webkit-scrollbar-track{background:#f1f1f1}.jsgrid-header-sortable:hover{cursor:pointer;background:#fcfcfc}.jsgrid-header-row .jsgrid-header-sort{background:#c4e2ff}.jsgrid-header-sort:before{content:" ";display:block;float:left;width:0;height:0;border-style:solid}.jsgrid-header-sort-asc:before{border-width:0 5px 5px;border-color:transparent transparent #009a67}.jsgrid-header-sort-desc:before{border-width:5px 5px 0;border-color:#009a67 transparent transparent}.jsgrid-grid-body{border-top:none}.jsgrid-grid-body td{border:1px solid #f3f3f3}.jsgrid-grid-body tr:first-child td{border-top:none}.jsgrid-grid-body tr td:first-child{border-left:none}.jsgrid-grid-body tr td:last-child{border-right:none}.jsgrid-row>td{background:#fff}.jsgrid-alt-row>td{background:#fcfcfc}.jsgrid-header-row>th{background:#f9f9f9}.jsgrid-filter-row>td{background:#fcfcfc}.jsgrid-insert-row>td{background:#e3ffe5}.jsgrid-edit-row>td{background:#fdffe3}.jsgrid-selected-row>td{background:#c4e2ff;border-color:#c4e2ff}.jsgrid-nodata-row td{background:#fff}.jsgrid-pager-current-page{font-weight:700}.jsgrid-pager-nav-inactive-button a{color:#d3d3d3}.jsgrid-button+.jsgrid-button{margin-left:5px}.jsgrid-button:hover{opacity:.5;transition:opacity 200ms linear}.jsgrid .jsgrid-button{width:16px;height:16px;border:none;cursor:pointer;background-image:url();background-repeat:no-repeat;background-color:transparent}@media only screen and (-webkit-min-device-pixel-ratio:2),only screen and (min-device-pixel-ratio:2){.jsgrid .jsgrid-button{background-image:url();background-size:24px 352px}}.jsgrid .jsgrid-mode-button{width:24px;height:24px}.jsgrid-mode-on-button{opacity:.5}.jsgrid-cancel-button{background-position:0 0;width:16px;height:16px}.jsgrid-clear-filter-button{background-position:0 -40px;width:16px;height:16px}.jsgrid-delete-button{background-position:0 -80px;width:16px;height:16px}.jsgrid-edit-button{background-position:0 -120px;width:16px;height:16px}.jsgrid-insert-mode-button{background-position:0 -160px;width:24px;height:24px}.jsgrid-insert-button{background-position:0 -208px;width:16px;height:16px}.jsgrid-search-mode-button{background-position:0 -248px;width:24px;height:24px}.jsgrid-search-button{background-position:0 -296px;width:16px;height:16px}.jsgrid-update-button{background-position:0 -336px;width:16px;height:16px}.jsgrid-load-shader{background:#ddd;opacity:.5;filter:alpha(opacity=50)}.jsgrid-load-panel{width:15em;height:5em;background:#fff;border:1px solid #e9e9e9;padding-top:3em;text-align:center}.jsgrid-load-panel:before{content:' ';position:absolute;top:.5em;left:50%;margin-left:-1em;width:2em;height:2em;border:2px solid #009a67;border-right-color:transparent;border-radius:50%;-webkit-animation:indicator 1s linear infinite;animation:indicator 1s linear infinite}@-webkit-keyframes indicator{from{-webkit-transform:rotate(0deg)}50%{-webkit-transform:rotate(180deg)}to{-webkit-transform:rotate(360deg)}}@keyframes indicator{from{transform:rotate(0deg)}50%{transform:rotate(180deg)}to{transform:rotate(360deg)}} \ No newline at end of file diff --git a/dist/jsgrid.css b/dist/jsgrid.css new file mode 100644 index 00000000..7db96615 --- /dev/null +++ b/dist/jsgrid.css @@ -0,0 +1,126 @@ +/* + * jsGrid v1.2.0 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid { + position: relative; + overflow: hidden; + font-size: 1em; +} + +.jsgrid, .jsgrid *, .jsgrid *:before, .jsgrid *:after { + box-sizing: border-box; +} + +.jsgrid input, +.jsgrid textarea, +.jsgrid select { + font-size: 1em; +} + +.jsgrid-grid-header { + overflow-x: hidden; + overflow-y: scroll; + -webkit-user-select: none; + -khtml-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + -o-user-select: none; + user-select: none; +} + +.jsgrid-grid-body { + overflow-x: auto; + overflow-y: scroll; + -webkit-overflow-scrolling: touch; +} + +.jsgrid-table { + width: 100%; + table-layout: fixed; + border-collapse: collapse; + border-spacing: 0; +} + +.jsgrid-table td { + padding: 0.5em 0.5em; +} + +.jsgrid-table td, +.jsgrid-table th { + box-sizing: border-box; +} + +.jsgrid-align-left { + text-align: left; +} + +.jsgrid-align-center { + text-align: center; +} + +.jsgrid-align-right { + text-align: right; +} + +.jsgrid-header-row > th { + padding: .5em .5em; +} + +.jsgrid-filter-row input, +.jsgrid-filter-row textarea, +.jsgrid-filter-row select, +.jsgrid-edit-row input, +.jsgrid-edit-row textarea, +.jsgrid-edit-row select, +.jsgrid-insert-row input, +.jsgrid-insert-row textarea, +.jsgrid-insert-row select { + width: 90%; + padding: .3em .5em; +} + +.jsgrid-filter-row input[type='checkbox'], +.jsgrid-edit-row input[type='checkbox'], +.jsgrid-insert-row input[type='checkbox'] { + width: auto; +} + +.jsgrid-header-row > th, +.jsgrid-filter-row > td, +.jsgrid-insert-row > td, +.jsgrid-edit-row > td { + text-align: center; +} + +.jsgrid-selected-row td { + cursor: pointer; +} + +.jsgrid-nodata-row td { + padding: .5em 0; + text-align: center; +} + +.jsgrid-header-sort { + cursor: pointer; +} + +.jsgrid-pager { + padding: .5em 0; +} + +.jsgrid-pager-nav-button { + padding: .2em .6em; +} + +.jsgrid-pager-nav-inactive-button { + display: none; + pointer-events: none; +} + +.jsgrid-pager-page { + padding: .2em .6em; +} diff --git a/dist/jsgrid.js b/dist/jsgrid.js new file mode 100644 index 00000000..e03b080c --- /dev/null +++ b/dist/jsgrid.js @@ -0,0 +1,2126 @@ +/* + * jsGrid v1.2.0 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +(function(window, $, undefined) { + + var JSGRID = "JSGrid", + JSGRID_DATA_KEY = JSGRID, + JSGRID_ROW_DATA_KEY = "JSGridItem", + JSGRID_EDIT_ROW_DATA_KEY = "JSGridEditRow", + + SORT_ORDER_ASC = "asc", + SORT_ORDER_DESC = "desc", + + FIRST_PAGE_PLACEHOLDER = "{first}", + PAGES_PLACEHOLDER = "{pages}", + PREV_PAGE_PLACEHOLDER = "{prev}", + NEXT_PAGE_PLACEHOLDER = "{next}", + LAST_PAGE_PLACEHOLDER = "{last}", + PAGE_INDEX_PLACEHOLDER = "{pageIndex}", + PAGE_COUNT_PLACEHOLDER = "{pageCount}", + ITEM_COUNT_PLACEHOLDER = "{itemCount}", + + EMPTY_HREF = "javascript:void(0);"; + + var getOrApply = function(value, context) { + if($.isFunction(value)) { + return value.apply(context, $.makeArray(arguments).slice(2)); + } + return value; + }; + + var defaultController = { + loadData: $.noop, + insertItem: $.noop, + updateItem: $.noop, + deleteItem: $.noop + }; + + + function Grid(element, config) { + var $element = $(element); + + $element.data(JSGRID_DATA_KEY, this); + + this._container = $element; + + this.data = []; + this.fields = []; + + this._editingRow = null; + this._sortField = null; + this._sortOrder = SORT_ORDER_ASC; + this._firstDisplayingPage = 1; + + this._init(config); + this.render(); + } + + Grid.prototype = { + width: "auto", + height: "auto", + updateOnResize: true, + + rowClass: $.noop, + rowRenderer: null, + + rowClick: function(args) { + if(this.editing) { + this.editItem($(args.event.target).closest("tr")); + } + }, + rowDoubleClick: $.noop, + + noDataContent: "Not found", + noDataRowClass: "jsgrid-nodata-row", + + heading: true, + headerRowRenderer: null, + headerRowClass: "jsgrid-header-row", + + filtering: false, + filterRowRenderer: null, + filterRowClass: "jsgrid-filter-row", + + inserting: false, + insertRowRenderer: null, + insertRowClass: "jsgrid-insert-row", + + editing: false, + editRowRenderer: null, + editRowClass: "jsgrid-edit-row", + + confirmDeleting: true, + deleteConfirm: "Are you sure?", + + selecting: true, + selectedRowClass: "jsgrid-selected-row", + oddRowClass: "jsgrid-row", + evenRowClass: "jsgrid-alt-row", + + sorting: false, + sortableClass: "jsgrid-header-sortable", + sortAscClass: "jsgrid-header-sort jsgrid-header-sort-asc", + sortDescClass: "jsgrid-header-sort jsgrid-header-sort-desc", + + paging: false, + pagerContainer: null, + pageIndex: 1, + pageSize: 20, + pageButtonCount: 15, + pagerFormat: "Pages: {first} {prev} {pages} {next} {last}    {pageIndex} of {pageCount}", + pagePrevText: "Prev", + pageNextText: "Next", + pageFirstText: "First", + pageLastText: "Last", + pageNavigatorNextText: "...", + pageNavigatorPrevText: "...", + pagerContainerClass: "jsgrid-pager-container", + pagerClass: "jsgrid-pager", + pagerNavButtonClass: "jsgrid-pager-nav-button", + pagerNavButtonInactiveClass: "jsgrid-pager-nav-inactive-button", + pageClass: "jsgrid-pager-page", + currentPageClass: "jsgrid-pager-current-page", + + customLoading: false, + pageLoading: false, + + autoload: false, + controller: defaultController, + + loadIndication: true, + loadIndicationDelay: 500, + loadMessage: "Please, wait...", + loadShading: true, + + onRefreshing: $.noop, + onRefreshed: $.noop, + onItemDeleting: $.noop, + onItemDeleted: $.noop, + onItemInserting: $.noop, + onItemInserted: $.noop, + onItemUpdating: $.noop, + onItemUpdated: $.noop, + onDataLoading: $.noop, + onDataLoaded: $.noop, + onOptionChanging: $.noop, + onOptionChanged: $.noop, + onError: $.noop, + + containerClass: "jsgrid", + tableClass: "jsgrid-table", + gridHeaderClass: "jsgrid-grid-header", + gridBodyClass: "jsgrid-grid-body", + + _init: function(config) { + $.extend(this, config); + this._initLoadStrategy(); + this._initController(); + this._initFields(); + this._attachWindowLoadResize(); + this._attachWindowResizeCallback(); + }, + + loadStrategy: function() { + return this.pageLoading + ? new jsGrid.loadStrategies.PageLoadingStrategy(this) + : new jsGrid.loadStrategies.DirectLoadingStrategy(this); + }, + + _initLoadStrategy: function() { + this._loadStrategy = getOrApply(this.loadStrategy, this); + }, + + _initController: function() { + this._controller = $.extend({}, defaultController, getOrApply(this.controller, this)); + }, + + loadIndicator: function(config) { + return new jsGrid.LoadIndicator(config); + }, + + _initFields: function() { + var self = this; + self.fields = $.map(self.fields, function(field) { + if($.isPlainObject(field)) { + var fieldConstructor = (field.type && jsGrid.fields[field.type]) || jsGrid.Field; + field = new fieldConstructor(field); + } + field._grid = self; + return field; + }); + }, + + _attachWindowLoadResize: function() { + $(window).on("load", $.proxy(this._refreshSize, this)); + }, + + _attachWindowResizeCallback: function() { + if(this.updateOnResize) { + $(window).on("resize", $.proxy(this._refreshSize, this)); + } + }, + + _detachWindowResizeCallback: function() { + $(window).off("resize", this._refreshSize); + }, + + option: function(key, value) { + var optionChangingEventArgs, + optionChangedEventArgs; + + if(arguments.length === 1) { + return this[key]; + } + + optionChangingEventArgs = { + option: key, + oldValue: this[key], + newValue: value + }; + this._callEventHandler(this.onOptionChanging, optionChangingEventArgs); + + this._handleOptionChange(optionChangingEventArgs.option, optionChangingEventArgs.newValue); + + optionChangedEventArgs = { + option: optionChangingEventArgs.option, + value: optionChangingEventArgs.newValue + }; + this._callEventHandler(this.onOptionChanged, optionChangedEventArgs); + }, + + _handleOptionChange: function(name, value) { + this[name] = value; + + switch(name) { + case "width": + case "height": + this._refreshSize(); + break; + case "rowClass": + case "rowRenderer": + case "rowClick": + case "rowDoubleClick": + case "noDataText": + case "noDataRowClass": + case "noDataContent": + case "selecting": + case "selectedRowClass": + case "oddRowClass": + case "evenRowClass": + this._refreshContent(); + break; + case "pageButtonCount": + case "pagerFormat": + case "pagePrevText": + case "pageNextText": + case "pageFirstText": + case "pageLastText": + case "pageNavigatorNextText": + case "pageNavigatorPrevText": + case "pagerClass": + case "pagerNavButtonClass": + case "pageClass": + case "currentPageClass": + case "pagerRenderer": + this._refreshPager(); + break; + case "fields": + this._initFields(); + this.render(); + break; + case "data": + case "editing": + case "heading": + case "filtering": + case "inserting": + case "paging": + this.refresh(); + break; + case "loadStrategy": + case "pageLoading": + this._initLoadStrategy(); + this.search(); + break; + case "pageIndex": + this.openPage(value); + break; + case "pageSize": + this.refresh(); + this.search(); + break; + case "editRowRenderer": + case "editRowClass": + this.cancelEdit(); + break; + default: + this.render(); + break; + } + }, + + destroy: function() { + this._detachWindowResizeCallback(); + this._clear(); + this._container.removeData(JSGRID_DATA_KEY); + }, + + render: function() { + this._clear(); + + this._container.addClass(this.containerClass) + .css("position", "relative") + .append(this._createHeader()) + .append(this._createBody()); + + this._pagerContainer = this._createPagerContainer(); + this._loadIndicator = this._createLoadIndicator(); + + this.refresh(); + + return this.autoload ? this.loadData() : $.Deferred().resolve().promise(); + }, + + _createLoadIndicator: function() { + return getOrApply(this.loadIndicator, this, { + message: this.loadMessage, + shading: this.loadShading, + container: this._container + }); + }, + + _clear: function() { + this.cancelEdit(); + + clearTimeout(this._loadingTimer); + + this._pagerContainer && this._pagerContainer.empty(); + + this._container.empty() + .css({ position: "", width: "", height: "" }); + }, + + _createHeader: function() { + var $headerRow = this._headerRow = this._createHeaderRow(), + $filterRow = this._filterRow = this._createFilterRow(), + $insertRow = this._insertRow = this._createInsertRow(); + + var $headerGrid = this._headerGrid = $("").addClass(this.tableClass) + .append($headerRow) + .append($filterRow) + .append($insertRow); + + var $header = this._header = $("
").addClass(this.gridHeaderClass) + .addClass(this._scrollBarWidth() ? "jsgrid-header-scrollbar" : "") + .append($headerGrid); + + return $header; + }, + + _createBody: function() { + var $content = this._content = $("
"); + + var $bodyGrid = this._bodyGrid = $("
").addClass(this.tableClass) + .append($content); + + var $body = this._body = $("
").addClass(this.gridBodyClass) + .append($bodyGrid) + .on("scroll", $.proxy(function(e) { + this._header.scrollLeft(e.target.scrollLeft); + }, this)); + + return $body; + }, + + _createPagerContainer: function() { + var pagerContainer = this.pagerContainer || $("
").appendTo(this._container); + return $(pagerContainer).addClass(this.pagerContainerClass); + }, + + _eachField: function(callBack) { + var self = this; + $.each(this.fields, function(index, field) { + return callBack.call(self, field, index); + }); + }, + + _createHeaderRow: function() { + if($.isFunction(this.headerRowRenderer)) { + return $(this.headerRowRenderer()); + } + + var $result = $("
").addClass(this.headerRowClass); + + this._eachField(function(field, index) { + var $th = $("").addClass(this.filterRowClass); + + this._eachField(function(field) { + $("").addClass(this.insertRowClass); + + this._eachField(function(field) { + $("").addClass(this.noDataRowClass) + .append($(""); + this._renderCells($result, item); + } + + $result.addClass(this._getRowClasses(item, itemIndex)) + .data(JSGRID_ROW_DATA_KEY, item) + .on("click", $.proxy(function(e) { + this.rowClick({ + item: item, + itemIndex: itemIndex, + event: e + }); + }, this)) + .on("dblclick", $.proxy(function(e) { + this.rowDoubleClick({ + item: item, + itemIndex: itemIndex, + event: e + }); + }, this)); + + if(this.selecting) { + this._attachRowHover($result); + } + + return $result; + }, + + _getRowClasses: function(item, itemIndex) { + var classes = []; + classes.push(((itemIndex + 1) % 2) ? this.oddRowClass : this.evenRowClass); + classes.push(getOrApply(this.rowClass, this, item, itemIndex)); + return classes.join(" "); + }, + + _attachRowHover: function($row) { + var selectedRowClass = this.selectedRowClass; + $row.hover(function() { + $(this).addClass(selectedRowClass); + }, + function() { + $(this).removeClass(selectedRowClass); + } + ); + }, + + _renderCells: function($row, item) { + this._eachField(function(field) { + $row.append(this._createCell(item, field)); + }); + return this; + }, + + _createCell: function(item, field) { + var $result; + var fieldValue = item[field.name]; + + if($.isFunction(field.cellRenderer)) { + $result = $(field.cellRenderer(fieldValue, item)); + } else { + $result = $("").addClass(this.editRowClass); + + this._eachField(function(field) { + $("
").addClass(field.headercss || field.css) + .appendTo($result) + .append(field.headerTemplate ? field.headerTemplate() : "") + .css("width", field.width); + + if(this.sorting && field.sorting) { + $th.addClass(this.sortableClass) + .on("click", $.proxy(function() { + this.sort(index); + }, this)); + } + }); + + return $result; + }, + + _createFilterRow: function() { + if($.isFunction(this.filterRowRenderer)) { + return $(this.filterRowRenderer()); + } + + var $result = $("
").addClass(field.filtercss || field.css) + .appendTo($result) + .append(field.filterTemplate ? field.filterTemplate() : "") + .width(field.width); + }); + + return $result; + }, + + _createInsertRow: function() { + if($.isFunction(this.insertRowRenderer)) { + return $(this.insertRowRenderer()); + } + + var $result = $("
").addClass(field.insertcss || field.css) + .appendTo($result) + .append(field.insertTemplate ? field.insertTemplate() : "") + .width(field.width); + }); + + return $result; + }, + + _callEventHandler: function(handler, eventParams) { + handler.call(this, $.extend(eventParams, { + grid: this + })); + + return eventParams; + }, + + reset: function() { + this._resetSorting(); + this._resetPager(); + this.refresh(); + }, + + _resetPager: function() { + this._firstDisplayingPage = 1; + this._setPage(1); + }, + + _resetSorting: function() { + this._sortField = null; + this._sortOrder = SORT_ORDER_ASC; + this._clearSortingCss(); + }, + + refresh: function() { + this._callEventHandler(this.onRefreshing); + + this.cancelEdit(); + + this._refreshHeading(); + this._refreshFiltering(); + this._refreshInserting(); + this._refreshContent(); + this._refreshPager(); + this._refreshSize(); + + this._callEventHandler(this.onRefreshed); + }, + + _refreshHeading: function() { + this._headerRow.toggle(this.heading); + }, + + _refreshFiltering: function() { + this._filterRow.toggle(this.filtering); + }, + + _refreshInserting: function() { + this._insertRow.toggle(this.inserting); + }, + + _refreshContent: function() { + var $content = this._content; + $content.empty(); + + if(!this.data.length) { + $content.append(this._createNoDataRow()); + return this; + } + + var indexFrom = this._loadStrategy.firstDisplayIndex(); + var indexTo = this._loadStrategy.lastDisplayIndex(); + + for(var itemIndex = indexFrom; itemIndex < indexTo; itemIndex++) { + var item = this.data[itemIndex]; + $content.append(this._createRow(item, itemIndex)); + } + }, + + _createNoDataRow: function() { + var noDataContent = getOrApply(this.noDataContent, this); + return $("
").attr("colspan", this.fields.length).append(noDataContent)); + }, + + _createNoDataContent: function() { + return $.isFunction(this.noDataRenderer) + ? this.noDataRenderer() + : this.noDataText; + }, + + _createRow: function(item, itemIndex) { + var $result; + + if($.isFunction(this.rowRenderer)) { + $result = $(this.rowRenderer(item, itemIndex)); + } else { + $result = $("
").append(field.itemTemplate ? field.itemTemplate(fieldValue, item) : fieldValue); + } + + $result.addClass(field.css) + .width(field.width); + + field.align && $result.addClass("jsgrid-align-" + field.align); + + return $result; + }, + + sort: function(field, order) { + if($.isPlainObject(field)) { + order = field.order; + field = field.field; + } + + this._clearSortingCss(); + this._setSortingParams(field, order); + this._setSortingCss(); + return this._loadStrategy.sort(); + }, + + _clearSortingCss: function() { + this._headerRow.find("th") + .removeClass(this.sortAscClass) + .removeClass(this.sortDescClass); + }, + + _setSortingParams: function(field, order) { + field = this._normalizeSortingField(field); + order = order || ((this._sortField === field) ? this._reversedSortOrder(this._sortOrder) : SORT_ORDER_ASC); + + this._sortField = field; + this._sortOrder = order; + }, + + _normalizeSortingField: function(field) { + if($.isNumeric(field)) { + return this.fields[field]; + } + + if(typeof field === "string") { + return $.grep(this.fields, function(f) { + return f.name === field; + })[0]; + } + + return field; + }, + + _reversedSortOrder: function(order) { + return (order === SORT_ORDER_ASC ? SORT_ORDER_DESC : SORT_ORDER_ASC); + }, + + _setSortingCss: function() { + var fieldIndex = $.inArray(this._sortField, this.fields); + + this._headerRow.find("th").eq(fieldIndex) + .addClass(this._sortOrder === SORT_ORDER_ASC ? this.sortAscClass : this.sortDescClass); + }, + + _sortData: function() { + var sortFactor = this._sortFactor(), + sortField = this._sortField; + + if(sortField) { + this.data.sort(function(item1, item2) { + return sortFactor * sortField.sortingFunc(item1[sortField.name], item2[sortField.name]); + }); + } + }, + + _sortFactor: function() { + return this._sortOrder === SORT_ORDER_ASC ? 1 : -1; + }, + + _itemsCount: function() { + return this._loadStrategy.itemsCount(); + }, + + _pagesCount: function() { + var itemsCount = this._itemsCount(), + pageSize = this.pageSize; + return Math.floor(itemsCount / pageSize) + (itemsCount % pageSize ? 1 : 0); + }, + + _refreshPager: function() { + var $pagerContainer = this._pagerContainer; + $pagerContainer.empty(); + + if(this.paging) { + $pagerContainer.append(this._createPager()); + } + + var showPager = this.paging && this._pagesCount() > 1; + $pagerContainer.toggle(showPager); + }, + + _createPager: function() { + var $result; + + if($.isFunction(this.pagerRenderer)) { + $result = $(this.pagerRenderer({ + pageIndex: this.pageIndex, + pageCount: this._pagesCount() + })); + } else { + $result = $("
").append(this._createPagerByFormat()); + } + + $result.addClass(this.pagerClass); + + return $result; + }, + + _createPagerByFormat: function() { + var pageIndex = this.pageIndex, + pageCount = this._pagesCount(), + itemCount = this._itemsCount(), + pagerParts = this.pagerFormat.split(" "); + + return $.map(pagerParts, $.proxy(function(pagerPart) { + var result = pagerPart; + + if(pagerPart === PAGES_PLACEHOLDER) { + result = this._createPages(); + } else if(pagerPart === FIRST_PAGE_PLACEHOLDER) { + result = this._createPagerNavButton(this.pageFirstText, 1, pageIndex > 1); + } else if(pagerPart === PREV_PAGE_PLACEHOLDER) { + result = this._createPagerNavButton(this.pagePrevText, pageIndex - 1, pageIndex > 1); + } else if(pagerPart === NEXT_PAGE_PLACEHOLDER) { + result = this._createPagerNavButton(this.pageNextText, pageIndex + 1, pageIndex < pageCount); + } else if(pagerPart === LAST_PAGE_PLACEHOLDER) { + result = this._createPagerNavButton(this.pageLastText, pageCount, pageIndex < pageCount); + } else if(pagerPart === PAGE_INDEX_PLACEHOLDER) { + result = pageIndex; + } else if(pagerPart === PAGE_COUNT_PLACEHOLDER) { + result = pageCount; + } else if(pagerPart === ITEM_COUNT_PLACEHOLDER) { + result = itemCount; + } + + return $.isArray(result) ? result.concat([" "]) : [result, " "]; + }, this)); + }, + + _createPages: function() { + var pageCount = this._pagesCount(), + pageButtonCount = this.pageButtonCount, + firstDisplayingPage = this._firstDisplayingPage, + pages = [], + pageNumber; + + if(firstDisplayingPage > 1) { + pages.push(this._createPagerPageNavButton(this.pageNavigatorPrevText, this.showPrevPages)); + } + + for(var i = 0, pageNumber = firstDisplayingPage; i < pageButtonCount && pageNumber <= pageCount; i++, pageNumber++) { + pages.push(pageNumber === this.pageIndex + ? this._createPagerCurrentPage() + : this._createPagerPage(pageNumber)); + } + + if((firstDisplayingPage + pageButtonCount - 1) < pageCount) { + pages.push(this._createPagerPageNavButton(this.pageNavigatorNextText, this.showNextPages)); + } + + return pages; + }, + + _createPagerNavButton: function(text, pageIndex, isActive) { + return this._createPagerButton(text, this.pagerNavButtonClass + (isActive ? "" : " " + this.pagerNavButtonInactiveClass), + isActive ? function() { this.openPage(pageIndex); } : $.noop); + }, + + _createPagerPageNavButton: function(text, handler) { + return this._createPagerButton(text, this.pagerNavButtonClass, handler); + }, + + _createPagerPage: function(pageIndex) { + return this._createPagerButton(pageIndex, this.pageClass, function() { + this.openPage(pageIndex); + }); + }, + + _createPagerButton: function(text, css, handler) { + var $link = $("").attr("href", EMPTY_HREF) + .html(text) + .on("click", $.proxy(handler, this)); + + return $("").addClass(css).append($link); + }, + + _createPagerCurrentPage: function() { + return $("") + .addClass(this.pageClass) + .addClass(this.currentPageClass) + .text(this.pageIndex); + }, + + _refreshSize: function() { + this._refreshHeight(); + this._refreshWidth(); + }, + + _refreshWidth: function() { + var $headerGrid = this._headerGrid, + $bodyGrid = this._bodyGrid, + width = this.width; + + if(width === "auto") { + $headerGrid.width("auto"); + width = $headerGrid.outerWidth(); + } + + $headerGrid.width(""); + $bodyGrid.width(""); + this._container.width(width); + width = $headerGrid.outerWidth(); + $bodyGrid.width(width); + }, + + _scrollBarWidth: (function() { + var result; + + return function() { + if(result === undefined) { + var $ghostContainer = $("
"); + var $ghostContent = $("
"); + $ghostContainer.append($ghostContent).appendTo("body"); + var width = $ghostContent.innerWidth(); + $ghostContainer.css("overflow-y", "auto"); + var widthExcludingScrollBar = $ghostContent.innerWidth(); + $ghostContainer.remove(); + result = width - widthExcludingScrollBar; + } + return result; + }; + })(), + + _refreshHeight: function() { + var container = this._container, + pagerContainer = this._pagerContainer, + height = this.height, + nonBodyHeight; + + container.height(height); + + if(height !== "auto") { + height = container.height(); + + nonBodyHeight = this._header.outerHeight(true); + if(pagerContainer.parents(container).length) { + nonBodyHeight += pagerContainer.outerHeight(true); + } + + this._body.outerHeight(height - nonBodyHeight); + } + }, + + showPrevPages: function() { + var firstDisplayingPage = this._firstDisplayingPage, + pageButtonCount = this.pageButtonCount; + + this._firstDisplayingPage = (firstDisplayingPage > pageButtonCount) ? firstDisplayingPage - pageButtonCount : 1; + + this._refreshPager(); + }, + + showNextPages: function() { + var firstDisplayingPage = this._firstDisplayingPage, + pageButtonCount = this.pageButtonCount, + pageCount = this._pagesCount(); + + this._firstDisplayingPage = (firstDisplayingPage + 2 * pageButtonCount > pageCount) + ? pageCount - pageButtonCount + 1 + : firstDisplayingPage + pageButtonCount; + + this._refreshPager(); + }, + + openPage: function(pageIndex) { + if(pageIndex < 1 || pageIndex > this._pagesCount()) + return; + + this._setPage(pageIndex); + this._loadStrategy.openPage(pageIndex); + }, + + _setPage: function(pageIndex) { + var firstDisplayingPage = this._firstDisplayingPage, + pageButtonCount = this.pageButtonCount; + + this.pageIndex = pageIndex; + + if(pageIndex < firstDisplayingPage) { + this._firstDisplayingPage = pageIndex; + } + + if(pageIndex > firstDisplayingPage + pageButtonCount - 1) { + this._firstDisplayingPage = pageIndex - pageButtonCount + 1; + } + }, + + _controllerCall: function(method, param, isCanceled, doneCallback) { + if(isCanceled) + return $.Deferred().reject().promise(); + + this._showLoading(); + + var controller = this._controller; + if(!controller || !controller[method]) { + throw new Error("controller has no method '" + method + "'"); + } + + return $.when(controller[method](param)) + .done($.proxy(doneCallback, this)) + .fail($.proxy(this._errorHandler, this)) + .always($.proxy(this._hideLoading, this)); + }, + + _errorHandler: function() { + this._callEventHandler(this.onError, { + args: $.makeArray(arguments) + }); + }, + + _showLoading: function() { + clearTimeout(this._loadingTimer); + + this._loadingTimer = setTimeout($.proxy(function() { + this._loadIndicator.show(); + }, this), this.loadIndicationDelay); + }, + + _hideLoading: function() { + clearTimeout(this._loadingTimer); + this._loadIndicator.hide(); + }, + + search: function(filter) { + this._resetSorting(); + this._resetPager(); + return this.loadData(filter); + }, + + loadData: function(filter) { + filter = filter || (this.filtering ? this.getFilter() : {}); + + $.extend(filter, this._loadStrategy.loadParams(), this._sortingParams()); + + var args = this._callEventHandler(this.onDataLoading, { + filter: filter + }); + + return this._controllerCall("loadData", filter, args.cancel, function(loadedData) { + if(!loadedData) + return; + + this._loadStrategy.finishLoad(loadedData); + + this._callEventHandler(this.onDataLoaded, { + data: loadedData + }); + }); + }, + + getFilter: function() { + var result = {}; + this._eachField(function(field) { + if(field.filtering) { + result[field.name] = field.filterValue(); + } + }); + return result; + }, + + _sortingParams: function() { + if(this.sorting && this._sortField) { + return { + sortField: this._sortField.name, + sortOrder: this._sortOrder + }; + } + return {}; + }, + + getSorting: function() { + var sortingParams = this._sortingParams(); + return { + field: sortingParams.sortField, + order: sortingParams.sortOrder + }; + }, + + clearFilter: function() { + var $filterRow = this._createFilterRow(); + this._filterRow.replaceWith($filterRow); + this._filterRow = $filterRow; + return this.search(); + }, + + insertItem: function(item) { + var insertingItem = item || this._getInsertItem(); + + var args = this._callEventHandler(this.onItemInserting, { + item: insertingItem + }); + + return this._controllerCall("insertItem", insertingItem, args.cancel, function(insertedItem) { + insertedItem = insertedItem || insertingItem; + this._loadStrategy.finishInsert(insertedItem); + + this._callEventHandler(this.onItemInserted, { + item: insertedItem + }); + }); + }, + + _getInsertItem: function() { + var result = {}; + this._eachField(function(field) { + if(field.inserting) { + result[field.name] = field.insertValue(); + } + }); + return result; + }, + + clearInsert: function() { + var insertRow = this._createInsertRow(); + this._insertRow.replaceWith(insertRow); + this._insertRow = insertRow; + this.refresh(); + }, + + editItem: function(item) { + var $row = this._rowByItem(item); + if($row.length) { + this._editRow($row); + } + }, + + _rowByItem: function(item) { + if(item.jquery || item.nodeType) + return $(item); + + return this._content.find("tr").filter(function() { + return $.data(this, JSGRID_ROW_DATA_KEY) === item; + }); + }, + + _editRow: function($row) { + if(!this.editing) + return; + + if(this._editingRow) { + this.cancelEdit(); + } + + var item = $row.data(JSGRID_ROW_DATA_KEY), + $editRow = this._createEditRow(item); + + this._editingRow = $row; + $row.hide(); + $editRow.insertAfter($row); + $row.data(JSGRID_EDIT_ROW_DATA_KEY, $editRow); + }, + + _createEditRow: function(item) { + if($.isFunction(this.editRowRenderer)) { + return $(this.editRowRenderer(item, this._itemIndex(item))); + } + + var $result = $("
").addClass(field.editcss || field.css) + .appendTo($result) + .append(field.editTemplate ? field.editTemplate(item[field.name], item) : "") + .width(field.width || "auto"); + }); + + return $result; + }, + + updateItem: function(item, editedItem) { + if(arguments.length === 1) { + editedItem = item; + } + + var $row = item ? this._rowByItem(item) : this._editingRow; + editedItem = editedItem || this._getEditedItem(); + + return this._updateRow($row, editedItem); + }, + + _updateRow: function($updatingRow, editedItem) { + var updatingItem = $updatingRow.data(JSGRID_ROW_DATA_KEY), + updatingItemIndex = this._itemIndex(updatingItem), + previousItem = $.extend({}, updatingItem); + + $.extend(updatingItem, editedItem); + + var args = this._callEventHandler(this.onItemUpdating, { + row: $updatingRow, + item: updatingItem, + itemIndex: updatingItemIndex, + previousItem: previousItem + }); + + return this._controllerCall("updateItem", updatingItem, args.cancel, function(updatedItem) { + updatedItem = updatedItem || updatingItem; + var $updatedRow = this._finishUpdate($updatingRow, updatedItem, updatingItemIndex); + + this._callEventHandler(this.onItemUpdated, { + row: $updatedRow, + item: updatedItem, + itemIndex: updatingItemIndex, + previousItem: previousItem + }); + }); + }, + + _itemIndex: function(item) { + return $.inArray(item, this.data); + }, + + _finishUpdate: function($updatingRow, updatedItem, updatedItemIndex) { + this.cancelEdit(); + this.data[updatedItemIndex] = updatedItem; + + var $updatedRow = this._createRow(updatedItem, updatedItemIndex); + $updatingRow.replaceWith($updatedRow); + return $updatedRow; + }, + + _getEditedItem: function() { + var result = {}; + this._eachField(function(field) { + if(field.editing) { + result[field.name] = field.editValue(); + } + }); + return result; + }, + + cancelEdit: function() { + if(!this._editingRow) { + return; + } + + var $row = this._editingRow, + $editRow = $row.data(JSGRID_EDIT_ROW_DATA_KEY); + + $editRow.remove(); + $row.show(); + this._editingRow = null; + }, + + deleteItem: function(item) { + var $row = this._rowByItem(item); + + if(!$row.length) + return; + + if(this.confirmDeleting && !window.confirm(getOrApply(this.deleteConfirm, this, $row.data(JSGRID_ROW_DATA_KEY)))) + return; + + return this._deleteRow($row); + }, + + _deleteRow: function($row) { + var deletingItem = $row.data(JSGRID_ROW_DATA_KEY), + deletingItemIndex = this._itemIndex(deletingItem); + + var args = this._callEventHandler(this.onItemDeleting, { + row: $row, + item: deletingItem, + itemIndex: deletingItemIndex + }); + + return this._controllerCall("deleteItem", deletingItem, args.cancel, function() { + this._loadStrategy.finishDelete(deletingItem, deletingItemIndex); + + this._callEventHandler(this.onItemDeleted, { + row: $row, + item: deletingItem, + itemIndex: deletingItemIndex + }); + }); + } + }; + + $.fn.jsGrid = function(config) { + var args = $.makeArray(arguments), + methodArgs = args.slice(1), + result = this; + + this.each(function() { + var $element = $(this), + instance = $element.data(JSGRID_DATA_KEY), + methodResult; + + if(instance) { + if(typeof config === "string") { + methodResult = instance[config].apply(instance, methodArgs); + if(methodResult !== undefined && methodResult !== instance) { + result = methodResult; + return false; + } + } else { + instance._detachWindowResizeCallback(); + instance._init(config); + instance.render(); + } + } else { + new Grid($element, config); + } + }); + + return result; + }; + + var fields = {}; + + var setDefaults = function(config) { + var componentPrototype; + + if($.isPlainObject(config)) { + componentPrototype = Grid.prototype; + } else { + componentPrototype = fields[config].prototype; + config = arguments[1] || {}; + } + + $.extend(componentPrototype, config); + }; + + window.jsGrid = { + Grid: Grid, + fields: fields, + setDefaults: setDefaults + }; + +}(window, jQuery)); + +(function(jsGrid, $, undefined) { + + function LoadIndicator(config) { + this._init(config); + } + + LoadIndicator.prototype = { + + container: "body", + message: "Loading...", + shading: true, + + zIndex: 1000, + shaderClass: "jsgrid-load-shader", + loadPanelClass: "jsgrid-load-panel", + + _init: function(config) { + $.extend(true, this, config); + + this._initContainer(); + this._initShader(); + this._initLoadPanel(); + }, + + _initContainer: function() { + this._container = $(this.container); + }, + + _initShader: function() { + if(!this.shading) + return; + + this._shader = $("
").addClass(this.shaderClass) + .hide() + .css({ + position: "absolute", + top: 0, + right: 0, + bottom: 0, + left: 0, + zIndex: this.zIndex + }) + .appendTo(this._container); + }, + + _initLoadPanel: function() { + this._loadPanel = $("
").addClass(this.loadPanelClass) + .text(this.message) + .hide() + .css({ + position: "absolute", + top: "50%", + left: "50%", + zIndex: this.zIndex + }) + .appendTo(this._container); + }, + + show: function() { + var $loadPanel = this._loadPanel.show(); + + var actualWidth = $loadPanel.outerWidth(); + var actualHeight = $loadPanel.outerHeight(); + + $loadPanel.css({ + marginTop: -actualHeight / 2, + marginLeft: -actualWidth / 2 + }); + + this._shader.show(); + }, + + hide: function() { + this._loadPanel.hide(); + this._shader.hide(); + } + + }; + + jsGrid.LoadIndicator = LoadIndicator; + +}(jsGrid, jQuery)); + +(function(jsGrid, $, undefined) { + + function DirectLoadingStrategy(grid) { + this._grid = grid; + } + + DirectLoadingStrategy.prototype = { + + firstDisplayIndex: function() { + var grid = this._grid; + return grid.option("paging") ? (grid.option("pageIndex") - 1) * grid.option("pageSize") : 0; + }, + + lastDisplayIndex: function() { + var grid = this._grid; + var itemsCount = grid.option("data").length; + + return grid.option("paging") + ? Math.min(grid.option("pageIndex") * grid.option("pageSize"), itemsCount) + : itemsCount; + }, + + itemsCount: function() { + return this._grid.option("data").length; + }, + + openPage: function(index) { + this._grid.refresh(); + }, + + loadParams: function() { + return {}; + }, + + sort: function() { + this._grid._sortData(); + this._grid.refresh(); + return $.Deferred().resolve().promise(); + }, + + finishLoad: function(loadedData) { + this._grid.option("data", loadedData); + }, + + finishInsert: function(insertedItem) { + var grid = this._grid; + grid.option("data").push(insertedItem); + grid.refresh(); + }, + + finishDelete: function(deletedItem, deletedItemIndex) { + var grid = this._grid; + grid.option("data").splice(deletedItemIndex, 1); + grid.reset(); + } + }; + + + function PageLoadingStrategy(grid) { + this._grid = grid; + this._itemsCount = 0; + } + + PageLoadingStrategy.prototype = { + firstDisplayIndex: function() { + return 0; + }, + + lastDisplayIndex: function() { + return this._grid.option("data").length; + }, + + itemsCount: function() { + return this._itemsCount; + }, + + openPage: function(index) { + this._grid.loadData(); + }, + + loadParams: function() { + var grid = this._grid; + return { + pageIndex: grid.option("pageIndex"), + pageSize: grid.option("pageSize") + }; + }, + + sort: function() { + return this._grid.loadData(); + }, + + finishLoad: function(loadedData) { + this._itemsCount = loadedData.itemsCount; + this._grid.option("data", loadedData.data); + }, + + finishInsert: function(insertedItem) { + this._grid.search(); + }, + + finishDelete: function(deletedItem, deletedItemIndex) { + this._grid.search(); + } + }; + + jsGrid.loadStrategies = { + DirectLoadingStrategy: DirectLoadingStrategy, + PageLoadingStrategy: PageLoadingStrategy + }; + +}(jsGrid, jQuery)); + +(function(jsGrid, $, undefined) { + + var isDefined = function(val) { + return typeof(val) !== "undefined" && val !== null; + }; + + var sortStrategies = { + string: function(str1, str2) { + if(!isDefined(str1) && !isDefined(str2)) + return 0; + + if(!isDefined(str1)) + return -1; + + if(!isDefined(str2)) + return 1; + + return ("" + str1).localeCompare("" + str2); + }, + + number: function(n1, n2) { + return n1 - n2; + }, + + date: function(dt1, dt2) { + return dt1 - dt2; + }, + + numberAsString: function(n1, n2) { + return parseFloat(n1) - parseFloat(n2); + } + }; + + jsGrid.sortStrategies = sortStrategies; + +}(jsGrid, jQuery)); + +(function(jsGrid, $, undefined) { + + function Field(config) { + $.extend(true, this, config); + this.sortingFunc = this._getSortingFunc(); + } + + Field.prototype = { + name: "", + title: "", + css: "", + align: "", + width: 100, + + filtering: true, + inserting: true, + editing: true, + sorting: true, + sorter: "string", // name of SortStrategy or function to compare elements + + headerTemplate: function() { + return this.title || this.name; + }, + + itemTemplate: function(value, item) { + return value; + }, + + filterTemplate: function() { + return ""; + }, + + insertTemplate: function() { + return ""; + }, + + editTemplate: function(value, item) { + this._value = value; + return this.itemTemplate(value, item); + }, + + filterValue: function() { + return ""; + }, + + insertValue: function() { + return ""; + }, + + editValue: function() { + return this._value; + }, + + _getSortingFunc: function() { + var sorter = this.sorter; + + if($.isFunction(sorter)) { + return sorter; + } + + if(typeof sorter === "string") { + return jsGrid.sortStrategies[sorter]; + } + + throw Error("Wrong sorter for the field \"" + this.name + "\"!"); + } + }; + + jsGrid.Field = Field; + +}(jsGrid, jQuery)); + +(function(jsGrid, $, undefined) { + + var Field = jsGrid.Field; + + function TextField(config) { + Field.call(this, config); + } + + TextField.prototype = new Field({ + + autosearch: true, + + filterTemplate: function() { + if(!this.filtering) + return ""; + + var grid = this._grid, + $result = this.filterControl = this._createTextBox(); + + if(this.autosearch) { + $result.on("keypress", function(e) { + if(e.which === 13) { + grid.search(); + e.preventDefault(); + } + }); + } + + return $result; + }, + + insertTemplate: function() { + if(!this.inserting) + return ""; + + var $result = this.insertControl = this._createTextBox(); + return $result; + }, + + editTemplate: function(value) { + if(!this.editing) + return this.itemTemplate(value); + + var $result = this.editControl = this._createTextBox(); + $result.val(value); + return $result; + }, + + filterValue: function() { + return this.filterControl.val(); + }, + + insertValue: function() { + return this.insertControl.val(); + }, + + editValue: function() { + return this.editControl.val(); + }, + + _createTextBox: function() { + return $("").attr("type", "text"); + } + }); + + jsGrid.fields.text = jsGrid.TextField = TextField; + +}(jsGrid, jQuery)); + +(function(jsGrid, $, undefined) { + + var TextField = jsGrid.TextField; + + function NumberField(config) { + TextField.call(this, config); + } + + NumberField.prototype = new TextField({ + + sorter: "number", + align: "right", + + filterValue: function() { + return parseInt(this.filterControl.val() || 0, 10); + }, + + insertValue: function() { + return parseInt(this.insertControl.val() || 0, 10); + }, + + editValue: function() { + return parseInt(this.editControl.val() || 0, 10); + }, + + _createTextBox: function() { + return $("").attr("type", "number"); + } + }); + + jsGrid.fields.number = jsGrid.NumberField = NumberField; + +}(jsGrid, jQuery)); + +(function(jsGrid, $, undefined) { + + var TextField = jsGrid.TextField; + + function TextAreaField(config) { + TextField.call(this, config); + } + + TextAreaField.prototype = new TextField({ + + insertTemplate: function() { + if(!this.inserting) + return ""; + + var $result = this.insertControl = this._createTextArea(); + return $result; + }, + + editTemplate: function(value) { + if(!this.editing) + return this.itemTemplate(value); + + var $result = this.editControl = this._createTextArea(); + $result.val(value); + return $result; + }, + + _createTextArea: function() { + return $("