From 0225bdb772d1334cc1aa7ab0fc3678df0864df6b Mon Sep 17 00:00:00 2001 From: AlisaLinUwU Date: Sun, 26 Jan 2025 10:42:28 +0500 Subject: Initialize --- .../resources/static/plugins/jsgrid/demos/db.js | 884 +++++++ .../static/plugins/jsgrid/i18n/jsgrid-de.js | 46 + .../static/plugins/jsgrid/i18n/jsgrid-es.js | 46 + .../static/plugins/jsgrid/i18n/jsgrid-fr.js | 47 + .../static/plugins/jsgrid/i18n/jsgrid-he.js | 46 + .../static/plugins/jsgrid/i18n/jsgrid-ja.js | 46 + .../static/plugins/jsgrid/i18n/jsgrid-ka.js | 46 + .../static/plugins/jsgrid/i18n/jsgrid-pl.js | 62 + .../static/plugins/jsgrid/i18n/jsgrid-pt-br.js | 46 + .../static/plugins/jsgrid/i18n/jsgrid-pt.js | 46 + .../static/plugins/jsgrid/i18n/jsgrid-ru.js | 47 + .../static/plugins/jsgrid/i18n/jsgrid-tr.js | 47 + .../static/plugins/jsgrid/i18n/jsgrid-zh-cn.js | 46 + .../static/plugins/jsgrid/i18n/jsgrid-zh-tw.js | 46 + .../static/plugins/jsgrid/jsgrid-theme.css | 258 ++ .../static/plugins/jsgrid/jsgrid-theme.min.css | 7 + .../resources/static/plugins/jsgrid/jsgrid.css | 126 + src/main/resources/static/plugins/jsgrid/jsgrid.js | 2516 ++++++++++++++++++++ .../resources/static/plugins/jsgrid/jsgrid.min.css | 7 + .../resources/static/plugins/jsgrid/jsgrid.min.js | 8 + 20 files changed, 4423 insertions(+) create mode 100644 src/main/resources/static/plugins/jsgrid/demos/db.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-de.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-es.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-fr.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-he.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ja.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ka.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pl.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pt-br.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pt.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ru.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-tr.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-zh-cn.js create mode 100644 src/main/resources/static/plugins/jsgrid/i18n/jsgrid-zh-tw.js create mode 100644 src/main/resources/static/plugins/jsgrid/jsgrid-theme.css create mode 100644 src/main/resources/static/plugins/jsgrid/jsgrid-theme.min.css create mode 100644 src/main/resources/static/plugins/jsgrid/jsgrid.css create mode 100644 src/main/resources/static/plugins/jsgrid/jsgrid.js create mode 100644 src/main/resources/static/plugins/jsgrid/jsgrid.min.css create mode 100644 src/main/resources/static/plugins/jsgrid/jsgrid.min.js (limited to 'src/main/resources/static/plugins/jsgrid') diff --git a/src/main/resources/static/plugins/jsgrid/demos/db.js b/src/main/resources/static/plugins/jsgrid/demos/db.js new file mode 100644 index 0000000..07ef1ee --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/demos/db.js @@ -0,0 +1,884 @@ +(function() { + + var db = { + + loadData: function(filter) { + return $.grep(this.clients, function(client) { + return (!filter.Name || client.Name.indexOf(filter.Name) > -1) + && (filter.Age === undefined || client.Age === filter.Age) + && (!filter.Address || client.Address.indexOf(filter.Address) > -1) + && (!filter.Country || client.Country === filter.Country) + && (filter.Married === undefined || client.Married === filter.Married); + }); + }, + + insertItem: function(insertingClient) { + this.clients.push(insertingClient); + }, + + updateItem: function(updatingClient) { }, + + deleteItem: function(deletingClient) { + var clientIndex = $.inArray(deletingClient, this.clients); + this.clients.splice(clientIndex, 1); + } + + }; + + window.db = db; + + + db.countries = [ + { Name: "", Id: 0 }, + { Name: "United States", Id: 1 }, + { Name: "Canada", Id: 2 }, + { Name: "United Kingdom", Id: 3 }, + { Name: "France", Id: 4 }, + { Name: "Brazil", Id: 5 }, + { Name: "China", Id: 6 }, + { Name: "Russia", Id: 7 } + ]; + + db.clients = [ + { + "Name": "Otto Clay", + "Age": 61, + "Country": 6, + "Address": "Ap #897-1459 Quam Avenue", + "Married": false + }, + { + "Name": "Connor Johnston", + "Age": 73, + "Country": 7, + "Address": "Ap #370-4647 Dis Av.", + "Married": false + }, + { + "Name": "Lacey Hess", + "Age": 29, + "Country": 7, + "Address": "Ap #365-8835 Integer St.", + "Married": false + }, + { + "Name": "Timothy Henson", + "Age": 78, + "Country": 1, + "Address": "911-5143 Luctus Ave", + "Married": false + }, + { + "Name": "Ramona Benton", + "Age": 43, + "Country": 5, + "Address": "Ap #614-689 Vehicula Street", + "Married": true + }, + { + "Name": "Ezra Tillman", + "Age": 51, + "Country": 1, + "Address": "P.O. Box 738, 7583 Quisque St.", + "Married": true + }, + { + "Name": "Dante Carter", + "Age": 59, + "Country": 1, + "Address": "P.O. Box 976, 6316 Lorem, St.", + "Married": false + }, + { + "Name": "Christopher Mcclure", + "Age": 58, + "Country": 1, + "Address": "847-4303 Dictum Av.", + "Married": true + }, + { + "Name": "Ruby Rocha", + "Age": 62, + "Country": 2, + "Address": "5212 Sagittis Ave", + "Married": false + }, + { + "Name": "Imelda Hardin", + "Age": 39, + "Country": 5, + "Address": "719-7009 Auctor Av.", + "Married": false + }, + { + "Name": "Jonah Johns", + "Age": 28, + "Country": 5, + "Address": "P.O. Box 939, 9310 A Ave", + "Married": false + }, + { + "Name": "Herman Rosa", + "Age": 49, + "Country": 7, + "Address": "718-7162 Molestie Av.", + "Married": true + }, + { + "Name": "Arthur Gay", + "Age": 20, + "Country": 7, + "Address": "5497 Neque Street", + "Married": false + }, + { + "Name": "Xena Wilkerson", + "Age": 63, + "Country": 1, + "Address": "Ap #303-6974 Proin Street", + "Married": true + }, + { + "Name": "Lilah Atkins", + "Age": 33, + "Country": 5, + "Address": "622-8602 Gravida Ave", + "Married": true + }, + { + "Name": "Malik Shepard", + "Age": 59, + "Country": 1, + "Address": "967-5176 Tincidunt Av.", + "Married": false + }, + { + "Name": "Keely Silva", + "Age": 24, + "Country": 1, + "Address": "P.O. Box 153, 8995 Praesent Ave", + "Married": false + }, + { + "Name": "Hunter Pate", + "Age": 73, + "Country": 7, + "Address": "P.O. Box 771, 7599 Ante, Road", + "Married": false + }, + { + "Name": "Mikayla Roach", + "Age": 55, + "Country": 5, + "Address": "Ap #438-9886 Donec Rd.", + "Married": true + }, + { + "Name": "Upton Joseph", + "Age": 48, + "Country": 4, + "Address": "Ap #896-7592 Habitant St.", + "Married": true + }, + { + "Name": "Jeanette Pate", + "Age": 59, + "Country": 2, + "Address": "P.O. Box 177, 7584 Amet, St.", + "Married": false + }, + { + "Name": "Kaden Hernandez", + "Age": 79, + "Country": 3, + "Address": "366 Ut St.", + "Married": true + }, + { + "Name": "Kenyon Stevens", + "Age": 20, + "Country": 3, + "Address": "P.O. Box 704, 4580 Gravida Rd.", + "Married": false + }, + { + "Name": "Jerome Harper", + "Age": 31, + "Country": 5, + "Address": "2464 Porttitor Road", + "Married": false + }, + { + "Name": "Jelani Patel", + "Age": 36, + "Country": 2, + "Address": "P.O. Box 541, 5805 Nec Av.", + "Married": true + }, + { + "Name": "Keaton Oconnor", + "Age": 21, + "Country": 1, + "Address": "Ap #657-1093 Nec, Street", + "Married": false + }, + { + "Name": "Bree Johnston", + "Age": 31, + "Country": 2, + "Address": "372-5942 Vulputate Avenue", + "Married": false + }, + { + "Name": "Maisie Hodges", + "Age": 70, + "Country": 7, + "Address": "P.O. Box 445, 3880 Odio, Rd.", + "Married": false + }, + { + "Name": "Kuame Calhoun", + "Age": 39, + "Country": 2, + "Address": "P.O. Box 609, 4105 Rutrum St.", + "Married": true + }, + { + "Name": "Carlos Cameron", + "Age": 38, + "Country": 5, + "Address": "Ap #215-5386 A, Avenue", + "Married": false + }, + { + "Name": "Fulton Parsons", + "Age": 25, + "Country": 7, + "Address": "P.O. Box 523, 3705 Sed Rd.", + "Married": false + }, + { + "Name": "Wallace Christian", + "Age": 43, + "Country": 3, + "Address": "416-8816 Mauris Avenue", + "Married": true + }, + { + "Name": "Caryn Maldonado", + "Age": 40, + "Country": 1, + "Address": "108-282 Nonummy Ave", + "Married": false + }, + { + "Name": "Whilemina Frank", + "Age": 20, + "Country": 7, + "Address": "P.O. Box 681, 3938 Egestas. Av.", + "Married": true + }, + { + "Name": "Emery Moon", + "Age": 41, + "Country": 4, + "Address": "Ap #717-8556 Non Road", + "Married": true + }, + { + "Name": "Price Watkins", + "Age": 35, + "Country": 4, + "Address": "832-7810 Nunc Rd.", + "Married": false + }, + { + "Name": "Lydia Castillo", + "Age": 59, + "Country": 7, + "Address": "5280 Placerat, Ave", + "Married": true + }, + { + "Name": "Lawrence Conway", + "Age": 53, + "Country": 1, + "Address": "Ap #452-2808 Imperdiet St.", + "Married": false + }, + { + "Name": "Kalia Nicholson", + "Age": 67, + "Country": 5, + "Address": "P.O. Box 871, 3023 Tellus Road", + "Married": true + }, + { + "Name": "Brielle Baxter", + "Age": 45, + "Country": 3, + "Address": "Ap #822-9526 Ut, Road", + "Married": true + }, + { + "Name": "Valentine Brady", + "Age": 72, + "Country": 7, + "Address": "8014 Enim. Road", + "Married": true + }, + { + "Name": "Rebecca Gardner", + "Age": 57, + "Country": 4, + "Address": "8655 Arcu. Road", + "Married": true + }, + { + "Name": "Vladimir Tate", + "Age": 26, + "Country": 1, + "Address": "130-1291 Non, Rd.", + "Married": true + }, + { + "Name": "Vernon Hays", + "Age": 56, + "Country": 4, + "Address": "964-5552 In Rd.", + "Married": true + }, + { + "Name": "Allegra Hull", + "Age": 22, + "Country": 4, + "Address": "245-8891 Donec St.", + "Married": true + }, + { + "Name": "Hu Hendrix", + "Age": 65, + "Country": 7, + "Address": "428-5404 Tempus Ave", + "Married": true + }, + { + "Name": "Kenyon Battle", + "Age": 32, + "Country": 2, + "Address": "921-6804 Lectus St.", + "Married": false + }, + { + "Name": "Gloria Nielsen", + "Age": 24, + "Country": 4, + "Address": "Ap #275-4345 Lorem, Street", + "Married": true + }, + { + "Name": "Illiana Kidd", + "Age": 59, + "Country": 2, + "Address": "7618 Lacus. Av.", + "Married": false + }, + { + "Name": "Adria Todd", + "Age": 68, + "Country": 6, + "Address": "1889 Tincidunt Road", + "Married": false + }, + { + "Name": "Kirsten Mayo", + "Age": 71, + "Country": 1, + "Address": "100-8640 Orci, Avenue", + "Married": false + }, + { + "Name": "Willa Hobbs", + "Age": 60, + "Country": 6, + "Address": "P.O. Box 323, 158 Tristique St.", + "Married": false + }, + { + "Name": "Alexis Clements", + "Age": 69, + "Country": 5, + "Address": "P.O. Box 176, 5107 Proin Rd.", + "Married": false + }, + { + "Name": "Akeem Conrad", + "Age": 60, + "Country": 2, + "Address": "282-495 Sed Ave", + "Married": true + }, + { + "Name": "Montana Silva", + "Age": 79, + "Country": 6, + "Address": "P.O. Box 120, 9766 Consectetuer St.", + "Married": false + }, + { + "Name": "Kaseem Hensley", + "Age": 77, + "Country": 6, + "Address": "Ap #510-8903 Mauris. Av.", + "Married": true + }, + { + "Name": "Christopher Morton", + "Age": 35, + "Country": 5, + "Address": "P.O. Box 234, 3651 Sodales Avenue", + "Married": false + }, + { + "Name": "Wade Fernandez", + "Age": 49, + "Country": 6, + "Address": "740-5059 Dolor. Road", + "Married": true + }, + { + "Name": "Illiana Kirby", + "Age": 31, + "Country": 2, + "Address": "527-3553 Mi Ave", + "Married": false + }, + { + "Name": "Kimberley Hurley", + "Age": 65, + "Country": 5, + "Address": "P.O. Box 637, 9915 Dictum St.", + "Married": false + }, + { + "Name": "Arthur Olsen", + "Age": 74, + "Country": 5, + "Address": "887-5080 Eget St.", + "Married": false + }, + { + "Name": "Brody Potts", + "Age": 59, + "Country": 2, + "Address": "Ap #577-7690 Sem Road", + "Married": false + }, + { + "Name": "Dillon Ford", + "Age": 60, + "Country": 1, + "Address": "Ap #885-9289 A, Av.", + "Married": true + }, + { + "Name": "Hannah Juarez", + "Age": 61, + "Country": 2, + "Address": "4744 Sapien, Rd.", + "Married": true + }, + { + "Name": "Vincent Shaffer", + "Age": 25, + "Country": 2, + "Address": "9203 Nunc St.", + "Married": true + }, + { + "Name": "George Holt", + "Age": 27, + "Country": 6, + "Address": "4162 Cras Rd.", + "Married": false + }, + { + "Name": "Tobias Bartlett", + "Age": 74, + "Country": 4, + "Address": "792-6145 Mauris St.", + "Married": true + }, + { + "Name": "Xavier Hooper", + "Age": 35, + "Country": 1, + "Address": "879-5026 Interdum. Rd.", + "Married": false + }, + { + "Name": "Declan Dorsey", + "Age": 31, + "Country": 2, + "Address": "Ap #926-4171 Aenean Road", + "Married": true + }, + { + "Name": "Clementine Tran", + "Age": 43, + "Country": 4, + "Address": "P.O. Box 176, 9865 Eu Rd.", + "Married": true + }, + { + "Name": "Pamela Moody", + "Age": 55, + "Country": 6, + "Address": "622-6233 Luctus Rd.", + "Married": true + }, + { + "Name": "Julie Leon", + "Age": 43, + "Country": 6, + "Address": "Ap #915-6782 Sem Av.", + "Married": true + }, + { + "Name": "Shana Nolan", + "Age": 79, + "Country": 5, + "Address": "P.O. Box 603, 899 Eu St.", + "Married": false + }, + { + "Name": "Vaughan Moody", + "Age": 37, + "Country": 5, + "Address": "880 Erat Rd.", + "Married": false + }, + { + "Name": "Randall Reeves", + "Age": 44, + "Country": 3, + "Address": "1819 Non Street", + "Married": false + }, + { + "Name": "Dominic Raymond", + "Age": 68, + "Country": 1, + "Address": "Ap #689-4874 Nisi Rd.", + "Married": true + }, + { + "Name": "Lev Pugh", + "Age": 69, + "Country": 5, + "Address": "Ap #433-6844 Auctor Avenue", + "Married": true + }, + { + "Name": "Desiree Hughes", + "Age": 80, + "Country": 4, + "Address": "605-6645 Fermentum Avenue", + "Married": true + }, + { + "Name": "Idona Oneill", + "Age": 23, + "Country": 7, + "Address": "751-8148 Aliquam Avenue", + "Married": true + }, + { + "Name": "Lani Mayo", + "Age": 76, + "Country": 1, + "Address": "635-2704 Tristique St.", + "Married": true + }, + { + "Name": "Cathleen Bonner", + "Age": 40, + "Country": 1, + "Address": "916-2910 Dolor Av.", + "Married": false + }, + { + "Name": "Sydney Murray", + "Age": 44, + "Country": 5, + "Address": "835-2330 Fringilla St.", + "Married": false + }, + { + "Name": "Brenna Rodriguez", + "Age": 77, + "Country": 6, + "Address": "3687 Imperdiet Av.", + "Married": true + }, + { + "Name": "Alfreda Mcdaniel", + "Age": 38, + "Country": 7, + "Address": "745-8221 Aliquet Rd.", + "Married": true + }, + { + "Name": "Zachery Atkins", + "Age": 30, + "Country": 1, + "Address": "549-2208 Auctor. Road", + "Married": true + }, + { + "Name": "Amelia Rich", + "Age": 56, + "Country": 4, + "Address": "P.O. Box 734, 4717 Nunc Rd.", + "Married": false + }, + { + "Name": "Kiayada Witt", + "Age": 62, + "Country": 3, + "Address": "Ap #735-3421 Malesuada Avenue", + "Married": false + }, + { + "Name": "Lysandra Pierce", + "Age": 36, + "Country": 1, + "Address": "Ap #146-2835 Curabitur St.", + "Married": true + }, + { + "Name": "Cara Rios", + "Age": 58, + "Country": 4, + "Address": "Ap #562-7811 Quam. Ave", + "Married": true + }, + { + "Name": "Austin Andrews", + "Age": 55, + "Country": 7, + "Address": "P.O. Box 274, 5505 Sociis Rd.", + "Married": false + }, + { + "Name": "Lillian Peterson", + "Age": 39, + "Country": 2, + "Address": "6212 A Avenue", + "Married": false + }, + { + "Name": "Adria Beach", + "Age": 29, + "Country": 2, + "Address": "P.O. Box 183, 2717 Nunc Avenue", + "Married": true + }, + { + "Name": "Oleg Durham", + "Age": 80, + "Country": 4, + "Address": "931-3208 Nunc Rd.", + "Married": false + }, + { + "Name": "Casey Reese", + "Age": 60, + "Country": 4, + "Address": "383-3675 Ultrices, St.", + "Married": false + }, + { + "Name": "Kane Burnett", + "Age": 80, + "Country": 1, + "Address": "759-8212 Dolor. Ave", + "Married": false + }, + { + "Name": "Stewart Wilson", + "Age": 46, + "Country": 7, + "Address": "718-7845 Sagittis. Av.", + "Married": false + }, + { + "Name": "Charity Holcomb", + "Age": 31, + "Country": 6, + "Address": "641-7892 Enim. Ave", + "Married": false + }, + { + "Name": "Kyra Cummings", + "Age": 43, + "Country": 4, + "Address": "P.O. Box 702, 6621 Mus. Av.", + "Married": false + }, + { + "Name": "Stuart Wallace", + "Age": 25, + "Country": 7, + "Address": "648-4990 Sed Rd.", + "Married": true + }, + { + "Name": "Carter Clarke", + "Age": 59, + "Country": 6, + "Address": "Ap #547-2921 A Street", + "Married": false + } + ]; + + db.users = [ + { + "ID": "x", + "Account": "A758A693-0302-03D1-AE53-EEFE22855556", + "Name": "Carson Kelley", + "RegisterDate": "2002-04-20T22:55:52-07:00" + }, + { + "Account": "D89FF524-1233-0CE7-C9E1-56EFF017A321", + "Name": "Prescott Griffin", + "RegisterDate": "2011-02-22T05:59:55-08:00" + }, + { + "Account": "06FAAD9A-5114-08F6-D60C-961B2528B4F0", + "Name": "Amir Saunders", + "RegisterDate": "2014-08-13T09:17:49-07:00" + }, + { + "Account": "EED7653D-7DD9-A722-64A8-36A55ECDBE77", + "Name": "Derek Thornton", + "RegisterDate": "2012-02-27T01:31:07-08:00" + }, + { + "Account": "2A2E6D40-FEBD-C643-A751-9AB4CAF1E2F6", + "Name": "Fletcher Romero", + "RegisterDate": "2010-06-25T15:49:54-07:00" + }, + { + "Account": "3978F8FA-DFF0-DA0E-0A5D-EB9D281A3286", + "Name": "Thaddeus Stein", + "RegisterDate": "2013-11-10T07:29:41-08:00" + }, + { + "Account": "658DBF5A-176E-569A-9273-74FB5F69FA42", + "Name": "Nash Knapp", + "RegisterDate": "2005-06-24T09:11:19-07:00" + }, + { + "Account": "76D2EE4B-7A73-1212-F6F2-957EF8C1F907", + "Name": "Quamar Vega", + "RegisterDate": "2011-04-13T20:06:29-07:00" + }, + { + "Account": "00E46809-A595-CE82-C5B4-D1CAEB7E3E58", + "Name": "Philip Galloway", + "RegisterDate": "2008-08-21T18:59:38-07:00" + }, + { + "Account": "C196781C-DDCC-AF83-DDC2-CA3E851A47A0", + "Name": "Mason French", + "RegisterDate": "2000-11-15T00:38:37-08:00" + }, + { + "Account": "5911F201-818A-B393-5888-13157CE0D63F", + "Name": "Ross Cortez", + "RegisterDate": "2010-05-27T17:35:32-07:00" + }, + { + "Account": "B8BB78F9-E1A1-A956-086F-E12B6FE168B6", + "Name": "Logan King", + "RegisterDate": "2003-07-08T16:58:06-07:00" + }, + { + "Account": "06F636C3-9599-1A2D-5FD5-86B24ADDE626", + "Name": "Cedric Leblanc", + "RegisterDate": "2011-06-30T14:30:10-07:00" + }, + { + "Account": "FE880CDD-F6E7-75CB-743C-64C6DE192412", + "Name": "Simon Sullivan", + "RegisterDate": "2013-06-11T16:35:07-07:00" + }, + { + "Account": "BBEDD673-E2C1-4872-A5D3-C4EBD4BE0A12", + "Name": "Jamal West", + "RegisterDate": "2001-03-16T20:18:29-08:00" + }, + { + "Account": "19BC22FA-C52E-0CC6-9552-10365C755FAC", + "Name": "Hector Morales", + "RegisterDate": "2012-11-01T01:56:34-07:00" + }, + { + "Account": "A8292214-2C13-5989-3419-6B83DD637D6C", + "Name": "Herrod Hart", + "RegisterDate": "2008-03-13T19:21:04-07:00" + }, + { + "Account": "0285564B-F447-0E7F-EAA1-7FB8F9C453C8", + "Name": "Clark Maxwell", + "RegisterDate": "2004-08-05T08:22:24-07:00" + }, + { + "Account": "EA78F076-4F6E-4228-268C-1F51272498AE", + "Name": "Reuben Walter", + "RegisterDate": "2011-01-23T01:55:59-08:00" + }, + { + "Account": "6A88C194-EA21-426F-4FE2-F2AE33F51793", + "Name": "Ira Ingram", + "RegisterDate": "2008-08-15T05:57:46-07:00" + }, + { + "Account": "4275E873-439C-AD26-56B3-8715E336508E", + "Name": "Damian Morrow", + "RegisterDate": "2015-09-13T01:50:55-07:00" + }, + { + "Account": "A0D733C4-9070-B8D6-4387-D44F0BA515BE", + "Name": "Macon Farrell", + "RegisterDate": "2011-03-14T05:41:40-07:00" + }, + { + "Account": "B3683DE8-C2FA-7CA0-A8A6-8FA7E954F90A", + "Name": "Joel Galloway", + "RegisterDate": "2003-02-03T04:19:01-08:00" + }, + { + "Account": "01D95A8E-91BC-2050-F5D0-4437AAFFD11F", + "Name": "Rigel Horton", + "RegisterDate": "2015-06-20T11:53:11-07:00" + }, + { + "Account": "F0D12CC0-31AC-A82E-FD73-EEEFDBD21A36", + "Name": "Sylvester Gaines", + "RegisterDate": "2004-03-12T09:57:13-08:00" + }, + { + "Account": "874FCC49-9A61-71BC-2F4E-2CE88348AD7B", + "Name": "Abbot Mckay", + "RegisterDate": "2008-12-26T20:42:57-08:00" + }, + { + "Account": "B8DA1912-20A0-FB6E-0031-5F88FD63EF90", + "Name": "Solomon Green", + "RegisterDate": "2013-09-04T01:44:47-07:00" + } + ]; + +}()); \ No newline at end of file diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-de.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-de.js new file mode 100644 index 0000000..4bad06d --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-de.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales.de = { + grid: { + noDataContent: "Die Daten konnten nicht gefunden werden", + deleteConfirm: "Möchten Sie die Daten unwiederruflich löschen?", + pagerFormat: "Seiten: {first} {prev} {pages} {next} {last}    {pageIndex} von {pageCount}", + pagePrevText: "<", + pageNextText: ">", + pageFirstText: "<<", + pageLastText: ">>", + loadMessage: "Bitte warten...", + invalidMessage: "Ihre Eingabe ist nicht zulässig!" + }, + + loadIndicator: { + message: "Lädt..." + }, + + fields: { + control: { + searchModeButtonTooltip: "Suche", + insertModeButtonTooltip: "Eintrag hinzufügen", + editButtonTooltip: "Bearbeiten", + deleteButtonTooltip: "Löschen", + searchButtonTooltip: "Eintrag finden", + clearFilterButtonTooltip: "Filter zurücksetzen", + insertButtonTooltip: "Hinzufügen", + updateButtonTooltip: "Speichern", + cancelEditButtonTooltip: "Abbrechen" + } + }, + + validators: { + required: { message: "Dies ist ein Pflichtfeld" }, + rangeLength: { message: "Die Länge der Eingabe liegt außerhalb des zulässigen Bereichs" }, + minLength: { message: "Die Eingabe ist zu kurz" }, + maxLength: { message: "Die Eingabe ist zu lang" }, + pattern: { message: "Die Eingabe entspricht nicht dem gewünschten Muster" }, + range: { message: "Der eingegebene Wert liegt außerhalb des zulässigen Bereichs" }, + min: { message: "Der eingegebene Wert ist zu niedrig" }, + max: { message: "Der eingegebene Wert ist zu hoch" } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-es.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-es.js new file mode 100644 index 0000000..e8fd7fe --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-es.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales.es = { + grid: { + noDataContent: "No encontrado", + deleteConfirm: "¿Está seguro?", + pagerFormat: "Paginas: {first} {prev} {pages} {next} {last}    {pageIndex} de {pageCount}", + pagePrevText: "Anterior", + pageNextText: "Siguiente", + pageFirstText: "Primero", + pageLastText: "Ultimo", + loadMessage: "Por favor, espere...", + invalidMessage: "¡Datos no válidos!" + }, + + loadIndicator: { + message: "Cargando..." + }, + + fields: { + control: { + searchModeButtonTooltip: "Cambiar a búsqueda", + insertModeButtonTooltip: "Cambiar a inserción", + editButtonTooltip: "Editar", + deleteButtonTooltip: "Suprimir", + searchButtonTooltip: "Buscar", + clearFilterButtonTooltip: "Borrar filtro", + insertButtonTooltip: "Insertar", + updateButtonTooltip: "Actualizar", + cancelEditButtonTooltip: "Cancelar edición" + } + }, + + validators: { + required: { message: "Campo requerido" }, + rangeLength: { message: "La longitud del valor está fuera del intervalo definido" }, + minLength: { message: "La longitud del valor es demasiado corta" }, + maxLength: { message: "La longitud del valor es demasiado larga" }, + pattern: { message: "El valor no se ajusta al patrón definido" }, + range: { message: "Valor fuera del rango definido" }, + min: { message: "Valor demasiado bajo" }, + max: { message: "Valor demasiado alto" } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-fr.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-fr.js new file mode 100644 index 0000000..c0f6f71 --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-fr.js @@ -0,0 +1,47 @@ +(function(jsGrid) { + + jsGrid.locales.fr = { + grid: { + noDataContent: "Pas de données", + deleteConfirm: "Êtes-vous sûr ?", + pagerFormat: "Pages: {first} {prev} {pages} {next} {last}    {pageIndex} de {pageCount}", + pagePrevText: "<", + pageNextText: ">", + pageFirstText: "<<", + pageLastText: ">>", + loadMessage: "Chargement en cours...", + invalidMessage: "Des données incorrectes sont entrés !" + }, + + loadIndicator: { + message: "Chargement en cours..." + }, + + fields: { + control: { + searchModeButtonTooltip: "Recherche", + insertModeButtonTooltip: "Ajouter une entrée", + editButtonTooltip: "Changer", + deleteButtonTooltip: "Effacer", + searchButtonTooltip: "Trouve", + clearFilterButtonTooltip: "Effacer", + insertButtonTooltip: "Ajouter", + updateButtonTooltip: "Sauvegarder", + cancelEditButtonTooltip: "Annuler" + } + }, + + validators: { + required: { message: "Champ requis" }, + rangeLength: { message: "Longueur de la valeur du champ est hors de la plage définie" }, + minLength: { message: "La valeur du champ est trop court" }, + maxLength: { message: "La valeur du champ est trop long" }, + pattern: { message: "La valeur du champ ne correspond pas à la configuration définie" }, + range: { message: "La valeur du champ est hors de la plage définie" }, + min: { message: "La valeur du champ est trop petit" }, + max: { message: "La valeur du champ est trop grande" } + } + }; + +}(jsGrid, jQuery)); + diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-he.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-he.js new file mode 100644 index 0000000..42c294c --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-he.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales.he = { + grid: { + noDataContent: "לא נמצא", + deleteConfirm: "האם אתה בטוח?", + pagerFormat: "עמודים: {first} {prev} {pages} {next} {last}    {pageIndex} מתוך {pageCount}", + pagePrevText: "הקודם", + pageNextText: "הבא", + pageFirstText: "ראשון", + pageLastText: "אחרון", + loadMessage: "אנא המתן ...", + invalidMessage: "נתונים לא חוקיים!" + }, + + loadIndicator: { + message: "טוען..." + }, + + fields: { + control: { + searchModeButtonTooltip: "ביצוע חיפוש", + insertModeButtonTooltip: "ביצוע עריכת שורה", + editButtonTooltip: "עריכה", + deleteButtonTooltip: "מחיקה", + searchButtonTooltip: "חיפוש", + clearFilterButtonTooltip: "ניקוי מסנן", + insertButtonTooltip: "הכנסה", + updateButtonTooltip: "עדכון", + cancelEditButtonTooltip: "ביטול עריכה" + } + }, + + validators: { + required: { message: "שדה נדרש" }, + rangeLength: { message: "אורכו של הערך הוא מחוץ לטווח המוגדר" }, + minLength: { message: "אורכו של הערך קצר מדי" }, + maxLength: { message: "אורכו של הערך ארוך מדי" }, + pattern: { message: "אורכו של הערך ארוך מדי" }, + range: { message: "ערך מחוץ לטווח" }, + min: { message: "ערך נמוך מדי" }, + max: { message: "גבוה מדי" } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ja.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ja.js new file mode 100644 index 0000000..acc506f --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ja.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales.ja = { + grid: { + noDataContent: "データが見つかりません。", + deleteConfirm: "削除しますよろしですか。", + pagerFormat: "頁: {first} {prev} {pages} {next} {last}    【{pageIndex}/{pageCount}】", + pagePrevText: "前", + pageNextText: "次", + pageFirstText: "最初", + pageLastText: "最後", + loadMessage: "しばらくお待ちください…", + invalidMessage: "入力されたデータが不正です。" + }, + + loadIndicator: { + message: "処理中…" + }, + + fields: { + control: { + searchModeButtonTooltip: "検索モードへ", + insertModeButtonTooltip: "登録モードへ", + editButtonTooltip: "編集", + deleteButtonTooltip: "削除", + searchButtonTooltip: "フィルター", + clearFilterButtonTooltip: "クリア", + insertButtonTooltip: "登録", + updateButtonTooltip: "更新", + cancelEditButtonTooltip: "編集戻す" + } + }, + + validators: { + required: { message: "項目が必要です。" }, + rangeLength: { message: "項目の桁数が範囲外です。" }, + minLength: { message: "項目の桁数が超過しています。" }, + maxLength: { message: "項目の桁数が不足しています。" }, + pattern: { message: "項目の値がパターンに一致しません。" }, + range: { message: "項目の値が範囲外です。" }, + min: { message: "項目の値が超過しています。" }, + max: { message: "項目の値が不足しています。" } + } + }; + +}(jsGrid, jQuery)); \ No newline at end of file diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ka.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ka.js new file mode 100644 index 0000000..281a230 --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ka.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales.ka = { + grid: { + noDataContent: "მონაცემები ცარიელია.", + deleteConfirm: "ნამდვილად გსურთ ჩანაწერის წაშლა?", + pagerFormat: "გვერდები: {first} {prev} {pages} {next} {last}    {pageIndex} - {pageCount} დან.", + pagePrevText: "<", + pageNextText: ">", + pageFirstText: "<<", + pageLastText: ">>", + loadMessage: "გთხოვთ დაიცადოთ...", + invalidMessage: "შეყვანილია არასწორი მონაცემები!" + }, + + loadIndicator: { + message: "მიმდინარეობს ჩატვირთვა..." + }, + + fields: { + control: { + searchModeButtonTooltip: "ძებნა", + insertModeButtonTooltip: "ჩანაწერის დამატება", + editButtonTooltip: "შესწორება", + deleteButtonTooltip: "წაშლა", + searchButtonTooltip: "ძებნა", + clearFilterButtonTooltip: "ფილტრის გასუფთავება", + insertButtonTooltip: "დამატება", + updateButtonTooltip: "შენახვა", + cancelEditButtonTooltip: "გაუქმება" + } + }, + + validators: { + required: { message: "ველი აუცილებელია შესავსებად." }, + rangeLength: { message: "შეყვანილი ჩანაწერის ზომა არ ექვემდებარება დიაპაზონს." }, + minLength: { message: "შეყვანილი ჩანაწერის ზომა საკმაოდ პატარა არის." }, + maxLength: { message: "შეყვანილი ჩანაწერის ზომა საკმაოდ დიდი არის." }, + pattern: { message: "შეყვანილი მნიშვნელობა არ ემთხვევა მითითებულ შაბლონს." }, + range: { message: "შეყვანილი ინფორმაცია არ ჯდება დიაპაზონში." }, + min: { message: "შეყვანილი ინფორმაციის ზომა საკმაოდ პატარა არის." }, + max: { message: "შეყვანილი ინფორმაციის ზომა საკმაოდ დიდი არის." } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pl.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pl.js new file mode 100644 index 0000000..e5fbf41 --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pl.js @@ -0,0 +1,62 @@ +(function(jsGrid) { + + jsGrid.locales.pl = { + grid: { + noDataContent: "Nie znaleziono", + deleteConfirm: "Czy jesteś pewien?", + pagerFormat: "Strony: {first} {prev} {pages} {next} {last}    {pageIndex} z {pageCount}", + pagePrevText: "Poprzednia", + pageNextText: "Następna", + pageFirstText: "Pierwsza", + pageLastText: "Ostatnia", + loadMessage: "Proszę czekać...", + invalidMessage: "Wprowadzono nieprawidłowe dane!" + }, + + loadIndicator: { + message: "Ładowanie..." + }, + + fields: { + control: { + searchModeButtonTooltip: "Wyszukiwanie", + insertModeButtonTooltip: "Dodawanie", + editButtonTooltip: "Edytuj", + deleteButtonTooltip: "Usuń", + searchButtonTooltip: "Szukaj", + clearFilterButtonTooltip: "Wyczyść filtr", + insertButtonTooltip: "Dodaj", + updateButtonTooltip: "Aktualizuj", + cancelEditButtonTooltip: "Anuluj edytowanie" + } + }, + + validators: { + required: { + message: "Pole jest wymagane" + }, + rangeLength: { + message: "Długość wartości pola znajduje się poza zdefiniowanym zakresem" + }, + minLength: { + message: "Wartość pola jest zbyt krótka" + }, + maxLength: { + message: "Wartość pola jest zbyt długa" + }, + pattern: { + message: "Wartość pola nie zgadza się ze zdefiniowanym wzorem" + }, + range: { + message: "Wartość pola znajduje się poza zdefiniowanym zakresem" + }, + min: { + message: "Wartość pola jest zbyt mała" + }, + max: { + message: "Wartość pola jest zbyt duża" + } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pt-br.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pt-br.js new file mode 100644 index 0000000..d9df06c --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pt-br.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales["pt-br"] = { + grid: { + noDataContent: "Não encontrado", + deleteConfirm: "Você tem certeza que deseja remover este item?", + pagerFormat: "Páginas: {first} {prev} {pages} {next} {last}    {pageIndex} de {pageCount}", + pagePrevText: "Anterior", + pageNextText: "Seguinte", + pageFirstText: "Primeira", + pageLastText: "Última", + loadMessage: "Por favor, espere...", + invalidMessage: "Dados inválidos!" + }, + + loadIndicator: { + message: "Carregando..." + }, + + fields: { + control: { + searchModeButtonTooltip: "Mudar para busca", + insertModeButtonTooltip: "Mudar para inserção", + editButtonTooltip: "Editar", + deleteButtonTooltip: "Remover", + searchButtonTooltip: "Buscar", + clearFilterButtonTooltip: "Remover filtro", + insertButtonTooltip: "Adicionar", + updateButtonTooltip: "Atualizar", + cancelEditButtonTooltip: "Cancelar Edição" + } + }, + + validators: { + required: { message: "Campo obrigatório" }, + rangeLength: { message: "O valor esta fora do intervaldo definido" }, + minLength: { message: "O comprimento do valor é muito curto" }, + maxLength: { message: "O comprimento valor é muito longo" }, + pattern: { message: "O valor informado não é compatível com o padrão" }, + range: { message: "O valor informado esta fora do limite definido" }, + min: { message: "O valor é muito curto" }, + max: { message: "O valor é muito longo" } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pt.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pt.js new file mode 100644 index 0000000..25e3e1a --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-pt.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales.pt = { + grid: { + noDataContent: "Não encontrado", + deleteConfirm: "Você tem certeza que deseja remover este item?", + pagerFormat: "Páginas: {first} {prev} {pages} {next} {last}    {pageIndex} de {pageCount}", + pagePrevText: "Anterior", + pageNextText: "Seguinte", + pageFirstText: "Primeira", + pageLastText: "Última", + loadMessage: "Por favor, espere...", + invalidMessage: "Dados inválidos!" + }, + + loadIndicator: { + message: "Carregando..." + }, + + fields: { + control: { + searchModeButtonTooltip: "Mudar para busca", + insertModeButtonTooltip: "Mudar para inserção", + editButtonTooltip: "Editar", + deleteButtonTooltip: "Remover", + searchButtonTooltip: "Buscar", + clearFilterButtonTooltip: "Remover filtro", + insertButtonTooltip: "Adicionar", + updateButtonTooltip: "Atualizar", + cancelEditButtonTooltip: "Cancelar Edição" + } + }, + + validators: { + required: { message: "Campo obrigatório" }, + rangeLength: { message: "O valor esta fora do intervaldo definido" }, + minLength: { message: "O comprimento do valor é muito curto" }, + maxLength: { message: "O comprimento valor é muito longo" }, + pattern: { message: "O valor informado não é compatível com o padrão" }, + range: { message: "O valor informado esta fora do limite definido" }, + min: { message: "O valor é muito curto" }, + max: { message: "O valor é muito longo" } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ru.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ru.js new file mode 100644 index 0000000..7fa469f --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-ru.js @@ -0,0 +1,47 @@ +(function(jsGrid) { + + jsGrid.locales.ru = { + grid: { + noDataContent: "Данных не найдено", + deleteConfirm: "Вы действительно хотите удалить запись?", + pagerFormat: "Страницы: {first} {prev} {pages} {next} {last}    {pageIndex} из {pageCount}", + pagePrevText: "<", + pageNextText: ">", + pageFirstText: "<<", + pageLastText: ">>", + loadMessage: "Пожалуйста, подождите...", + invalidMessage: "Введены неверные данные!" + }, + + loadIndicator: { + message: "Загрузка..." + }, + + fields: { + control: { + searchModeButtonTooltip: "Поиск", + insertModeButtonTooltip: "Добавить запись", + editButtonTooltip: "Изменить", + deleteButtonTooltip: "Удалить", + searchButtonTooltip: "Найти", + clearFilterButtonTooltip: "Очистить фильтр", + insertButtonTooltip: "Добавить", + updateButtonTooltip: "Сохранить", + cancelEditButtonTooltip: "Отменить" + } + }, + + validators: { + required: { message: "Поле обязательно для заполения" }, + rangeLength: { message: "Длинна введенного значения вне допустимого диапазона" }, + minLength: { message: "Введенное значение слишком короткое" }, + maxLength: { message: "Введенное значение слишком длинное" }, + pattern: { message: "Введенное значение не соответствует заданному шаблону" }, + range: { message: "Введенное значение вне допустимого диапазона" }, + min: { message: "Введенное значение слишком маленькое" }, + max: { message: "Введенное значение слишком большое" } + } + }; + +}(jsGrid, jQuery)); + diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-tr.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-tr.js new file mode 100644 index 0000000..bbb7c5c --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-tr.js @@ -0,0 +1,47 @@ +(function(jsGrid) { + + jsGrid.locales.tr = { + grid: { + noDataContent: "Kayıt Bulunamadı", + deleteConfirm: "Emin misiniz ?", + pagerFormat: "Sayfalar: {first} {prev} {pages} {next} {last}    {pageIndex} / {pageCount}", + pagePrevText: "<", + pageNextText: ">", + pageFirstText: "<<", + pageLastText: ">>", + loadMessage: "Lütfen bekleyiniz...", + invalidMessage: "Geçersiz veri girişi !" + }, + + loadIndicator: { + message: "Yükleniyor..." + }, + + fields: { + control: { + searchModeButtonTooltip: "Arama moduna geç", + insertModeButtonTooltip: "Yeni kayıt moduna geç", + editButtonTooltip: "Değiştir", + deleteButtonTooltip: "Sil", + searchButtonTooltip: "Bul", + clearFilterButtonTooltip: "Filtreyi temizle", + insertButtonTooltip: "Ekle", + updateButtonTooltip: "Güncelle", + cancelEditButtonTooltip: "Güncelleme iptali" + } + }, + + validators: { + required: { message: "Gerekli alandır" }, + rangeLength: { message: "Alan değerinin uzunluğu tanımlanan aralık dışındadır" }, + minLength: { message: "Alan değeri çok kısadır" }, + maxLength: { message: "Alan değeri çok uzundur" }, + pattern: { message: "Alan değeri tanımlanan şablon ile eşleşmiyor" }, + range: { message: "Alan değeri tanımlı aralığın dışındadır" }, + min: { message: "Alan değeri çok küçüktür" }, + max: { message: "Alan değeri çok büyüktür" } + } + }; + +}(jsGrid, jQuery)); + diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-zh-cn.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-zh-cn.js new file mode 100644 index 0000000..87c0055 --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-zh-cn.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales["zh-cn"] = { + grid: { + noDataContent: "暂无数据", + deleteConfirm: "确认删除?", + pagerFormat: "页码: {first} {prev} {pages} {next} {last}    {pageIndex} / {pageCount}", + pagePrevText: "上一页", + pageNextText: "下一页", + pageFirstText: "第一页", + pageLastText: "最后页", + loadMessage: "请稍后...", + invalidMessage: "数据有误!" + }, + + loadIndicator: { + message: "载入中..." + }, + + fields: { + control: { + searchModeButtonTooltip: "切换为搜索", + insertModeButtonTooltip: "切换为新增", + editButtonTooltip: "编辑", + deleteButtonTooltip: "删除", + searchButtonTooltip: "搜索", + clearFilterButtonTooltip: "清空过滤", + insertButtonTooltip: "插入", + updateButtonTooltip: "更新", + cancelEditButtonTooltip: "取消编辑" + } + }, + + validators: { + required: { message: "字段必填" }, + rangeLength: { message: "字段值长度超过定义范围" }, + minLength: { message: "字段长度过短" }, + maxLength: { message: "字段长度过长" }, + pattern: { message: "字段值不符合定义规则" }, + range: { message: "字段值超过定义范围" }, + min: { message: "字段值太小" }, + max: { message: "字段值太大" } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-zh-tw.js b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-zh-tw.js new file mode 100644 index 0000000..2a22845 --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/i18n/jsgrid-zh-tw.js @@ -0,0 +1,46 @@ +(function(jsGrid) { + + jsGrid.locales["zh-tw"] = { + grid: { + noDataContent: "暫無資料", + deleteConfirm: "確認刪除?", + pagerFormat: "頁碼: {first} {prev} {pages} {next} {last}    {pageIndex} / {pageCount}", + pagePrevText: "上一頁", + pageNextText: "下一頁", + pageFirstText: "第一頁", + pageLastText: "最後一頁", + loadMessage: "請稍候...", + invalidMessage: "輸入資料不正確" + }, + + loadIndicator: { + message: "載入中..." + }, + + fields: { + control: { + searchModeButtonTooltip: "切換為搜尋", + insertModeButtonTooltip: "切換為新增", + editButtonTooltip: "編輯", + deleteButtonTooltip: "刪除", + searchButtonTooltip: "搜尋", + clearFilterButtonTooltip: "清除搜尋條件", + insertButtonTooltip: "新增", + updateButtonTooltip: "修改", + cancelEditButtonTooltip: "取消編輯" + } + }, + + validators: { + required: { message: "欄位必填" }, + rangeLength: { message: "欄位字串長度超出範圍" }, + minLength: { message: "欄位字串長度太短" }, + maxLength: { message: "欄位字串長度太長" }, + pattern: { message: "欄位字串不符合規則" }, + range: { message: "欄位數值超出範圍" }, + min: { message: "欄位數值太小" }, + max: { message: "欄位數值太大" } + } + }; + +}(jsGrid, jQuery)); diff --git a/src/main/resources/static/plugins/jsgrid/jsgrid-theme.css b/src/main/resources/static/plugins/jsgrid/jsgrid-theme.css new file mode 100644 index 0000000..ebc7855 --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/jsgrid-theme.css @@ -0,0 +1,258 @@ +/* + * jsGrid v1.5.3 (http://js-grid.com) + * (c) 2016 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid-grid-header, +.jsgrid-grid-body, +.jsgrid-header-row > .jsgrid-header-cell, +.jsgrid-filter-row > .jsgrid-cell, +.jsgrid-insert-row > .jsgrid-cell, +.jsgrid-edit-row > .jsgrid-cell { + border: 1px solid #e9e9e9; +} + +.jsgrid-header-row > .jsgrid-header-cell { + border-top: 0; +} + +.jsgrid-header-row > .jsgrid-header-cell, +.jsgrid-filter-row > .jsgrid-cell, +.jsgrid-insert-row > .jsgrid-cell { + border-bottom: 0; +} + +.jsgrid-header-row > .jsgrid-header-cell:first-child, +.jsgrid-filter-row > .jsgrid-cell:first-child, +.jsgrid-insert-row > .jsgrid-cell:first-child { + border-left: none; +} + +.jsgrid-header-row > .jsgrid-header-cell:last-child, +.jsgrid-filter-row > .jsgrid-cell:last-child, +.jsgrid-insert-row > .jsgrid-cell:last-child { + border-right: none; +} + +.jsgrid-header-row .jsgrid-align-right, +.jsgrid-header-row .jsgrid-align-left { + text-align: center; +} + +.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-cell { + border: #f3f3f3 1px solid; +} + +.jsgrid-grid-body .jsgrid-row:first-child .jsgrid-cell, +.jsgrid-grid-body .jsgrid-alt-row:first-child .jsgrid-cell { + border-top: none; +} + +.jsgrid-grid-body .jsgrid-cell:first-child { + border-left: none; +} + +.jsgrid-grid-body .jsgrid-cell:last-child { + border-right: none; +} + +.jsgrid-row > .jsgrid-cell { + background: #fff; +} + +.jsgrid-alt-row > .jsgrid-cell { + background: #fcfcfc; +} + +.jsgrid-header-row > .jsgrid-header-cell { + background: #f9f9f9; +} + +.jsgrid-filter-row > .jsgrid-cell { + background: #fcfcfc; +} + +.jsgrid-insert-row > .jsgrid-cell { + background: #e3ffe5; +} + +.jsgrid-edit-row > .jsgrid-cell { + background: #fdffe3; +} + +.jsgrid-selected-row > .jsgrid-cell { + background: #c4e2ff; + border-color: #c4e2ff; +} + +.jsgrid-nodata-row > .jsgrid-cell { + background: #fff; +} + +.jsgrid-invalid input, +.jsgrid-invalid select, +.jsgrid-invalid textarea { + background: #ffe3e5; + border: 1px solid #ff808a; +} + +.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-edit-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/src/main/resources/static/plugins/jsgrid/jsgrid-theme.min.css b/src/main/resources/static/plugins/jsgrid/jsgrid-theme.min.css new file mode 100644 index 0000000..f369d7e --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/jsgrid-theme.min.css @@ -0,0 +1,7 @@ +/* + * jsGrid v1.5.3 (http://js-grid.com) + * (c) 2016 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid-edit-row>.jsgrid-cell,.jsgrid-filter-row>.jsgrid-cell,.jsgrid-grid-body,.jsgrid-grid-header,.jsgrid-header-row>.jsgrid-header-cell,.jsgrid-insert-row>.jsgrid-cell{border:1px solid #e9e9e9}.jsgrid-header-row>.jsgrid-header-cell{border-top:0}.jsgrid-filter-row>.jsgrid-cell,.jsgrid-header-row>.jsgrid-header-cell,.jsgrid-insert-row>.jsgrid-cell{border-bottom:0}.jsgrid-filter-row>.jsgrid-cell:first-child,.jsgrid-header-row>.jsgrid-header-cell:first-child,.jsgrid-insert-row>.jsgrid-cell:first-child{border-left:none}.jsgrid-filter-row>.jsgrid-cell:last-child,.jsgrid-header-row>.jsgrid-header-cell:last-child,.jsgrid-insert-row>.jsgrid-cell:last-child{border-right:none}.jsgrid-header-row .jsgrid-align-left,.jsgrid-header-row .jsgrid-align-right{text-align:center}.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-cell{border:1px solid #f3f3f3}.jsgrid-grid-body .jsgrid-alt-row:first-child .jsgrid-cell,.jsgrid-grid-body .jsgrid-row:first-child .jsgrid-cell{border-top:none}.jsgrid-grid-body .jsgrid-cell:first-child{border-left:none}.jsgrid-grid-body .jsgrid-cell:last-child{border-right:none}.jsgrid-row>.jsgrid-cell{background:#fff}.jsgrid-alt-row>.jsgrid-cell{background:#fcfcfc}.jsgrid-header-row>.jsgrid-header-cell{background:#f9f9f9}.jsgrid-filter-row>.jsgrid-cell{background:#fcfcfc}.jsgrid-insert-row>.jsgrid-cell{background:#e3ffe5}.jsgrid-edit-row>.jsgrid-cell{background:#fdffe3}.jsgrid-selected-row>.jsgrid-cell{background:#c4e2ff;border-color:#c4e2ff}.jsgrid-nodata-row>.jsgrid-cell{background:#fff}.jsgrid-invalid input,.jsgrid-invalid select,.jsgrid-invalid textarea{background:#ffe3e5;border:1px solid #ff808a}.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-edit-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/src/main/resources/static/plugins/jsgrid/jsgrid.css b/src/main/resources/static/plugins/jsgrid/jsgrid.css new file mode 100644 index 0000000..a59154f --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/jsgrid.css @@ -0,0 +1,126 @@ +/* + * jsGrid v1.5.3 (http://js-grid.com) + * (c) 2016 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-cell { + padding: 0.5em 0.5em; +} + +.jsgrid-сell, +.jsgrid-header-cell { + box-sizing: border-box; +} + +.jsgrid-align-left { + text-align: left; +} + +.jsgrid-align-center, +.jsgrid-align-center input, +.jsgrid-align-center textarea, +.jsgrid-align-center select { + text-align: center; +} + +.jsgrid-align-right, +.jsgrid-align-right input, +.jsgrid-align-right textarea, +.jsgrid-align-right select { + text-align: right; +} + +.jsgrid-header-cell { + 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: 100%; + padding: .3em .5em; +} + +.jsgrid-filter-row input[type='checkbox'], +.jsgrid-edit-row input[type='checkbox'], +.jsgrid-insert-row input[type='checkbox'] { + width: auto; +} + + +.jsgrid-selected-row .jsgrid-cell { + cursor: pointer; +} + +.jsgrid-nodata-row .jsgrid-cell { + 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/src/main/resources/static/plugins/jsgrid/jsgrid.js b/src/main/resources/static/plugins/jsgrid/jsgrid.js new file mode 100644 index 0000000..922bcdf --- /dev/null +++ b/src/main/resources/static/plugins/jsgrid/jsgrid.js @@ -0,0 +1,2516 @@ +/* + * jsGrid v1.5.3 (http://js-grid.com) + * (c) 2016 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 normalizePromise = function(promise) { + var d = $.Deferred(); + + if(promise && promise.then) { + promise.then(function() { + d.resolve.apply(d, arguments); + }, function() { + d.reject.apply(d, arguments); + }); + } else { + d.resolve(promise); + } + + return d.promise(); + }; + + 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", + headerCellClass: "jsgrid-header-cell", + + 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", + cellClass: "jsgrid-cell", + + 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, + + invalidMessage: "Invalid data entered!", + + invalidNotify: function(args) { + var messages = $.map(args.errors, function(error) { + return error.message || null; + }); + + window.alert([this.invalidMessage].concat(messages).join("\n")); + }, + + onInit: $.noop, + onRefreshing: $.noop, + onRefreshed: $.noop, + onPageChanged: $.noop, + onItemDeleting: $.noop, + onItemDeleted: $.noop, + onItemInserting: $.noop, + onItemInserted: $.noop, + onItemEditing: $.noop, + onItemUpdating: $.noop, + onItemUpdated: $.noop, + onItemInvalid: $.noop, + onDataLoading: $.noop, + onDataLoaded: $.noop, + onOptionChanging: $.noop, + onOptionChanged: $.noop, + onError: $.noop, + + invalidClass: "jsgrid-invalid", + + 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(); + this._callEventHandler(this.onInit) + }, + + 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)); + }, + + renderTemplate: function(source, context, config) { + args = []; + for(var key in config) { + args.push(config[key]); + } + + args.unshift(source, context); + + source = getOrApply.apply(null, args); + return (source === undefined || source === null) ? "" : source; + }, + + loadIndicator: function(config) { + return new jsGrid.LoadIndicator(config); + }, + + validation: function(config) { + return jsGrid.Validation && new jsGrid.Validation(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); + }, + + fieldOption: function(field, key, value) { + field = this._normalizeField(field); + + if(arguments.length === 2) + return field[key]; + + field[key] = value; + this._renderGrid(); + }, + + _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 "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; + case "updateOnResize": + this._detachWindowResizeCallback(); + this._attachWindowResizeCallback(); + break; + case "invalidNotify": + case "invalidMessage": + break; + default: + this.render(); + break; + } + }, + + destroy: function() { + this._detachWindowResizeCallback(); + this._clear(); + this._container.removeData(JSGRID_DATA_KEY); + }, + + render: function() { + this._renderGrid(); + return this.autoload ? this.loadData() : $.Deferred().resolve().promise(); + }, + + _renderGrid: 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._validation = this._createValidation(); + + this.refresh(); + }, + + _createLoadIndicator: function() { + return getOrApply(this.loadIndicator, this, { + message: this.loadMessage, + shading: this.loadShading, + container: this._container + }); + }, + + _createValidation: function() { + return getOrApply(this.validation, this); + }, + + _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) { + if(field.visible) { + callBack.call(self, field, index); + } + }); + }, + + _createHeaderRow: function() { + if($.isFunction(this.headerRowRenderer)) + return $(this.renderTemplate(this.headerRowRenderer, this)); + + var $result = $("
").addClass(this.headerRowClass); + + this._eachField(function(field, index) { + var $th = this._prepareCell("").addClass(this.filterRowClass); + + this._eachField(function(field) { + this._prepareCell("").addClass(this.insertRowClass); + + this._eachField(function(field) { + this._prepareCell("").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 = this._getItemFieldValue(item, field); + + var args = { value: fieldValue, item : item }; + if($.isFunction(field.cellRenderer)) { + $result = this.renderTemplate(field.cellRenderer, field, args); + } else { + $result = $("").addClass(this.editRowClass); + + this._eachField(function(field) { + var fieldValue = this._getItemFieldValue(item, field); + + this._prepareCell("
", field, "headercss", this.headerCellClass) + .append(this.renderTemplate(field.headerTemplate, field)) + .appendTo($result); + + if(this.sorting && field.sorting) { + $th.addClass(this.sortableClass) + .on("click", $.proxy(function() { + this.sort(index); + }, this)); + } + }); + + return $result; + }, + + _prepareCell: function(cell, field, cssprop, cellClass) { + return $(cell).css("width", field.width) + .addClass(cellClass || this.cellClass) + .addClass((cssprop && field[cssprop]) || field.css) + .addClass(field.align ? ("jsgrid-align-" + field.align) : ""); + }, + + _createFilterRow: function() { + if($.isFunction(this.filterRowRenderer)) + return $(this.renderTemplate(this.filterRowRenderer, this)); + + var $result = $("
", field, "filtercss") + .append(this.renderTemplate(field.filterTemplate, field)) + .appendTo($result); + }); + + return $result; + }, + + _createInsertRow: function() { + if($.isFunction(this.insertRowRenderer)) + return $(this.renderTemplate(this.insertRowRenderer, this)); + + var $result = $("
", field, "insertcss") + .append(this.renderTemplate(field.insertTemplate, field)) + .appendTo($result); + }); + + return $result; + }, + + _callEventHandler: function(handler, eventParams) { + handler.call(this, $.extend(eventParams, { + grid: this + })); + + return eventParams; + }, + + reset: function() { + this._resetSorting(); + this._resetPager(); + return this._loadStrategy.reset(); + }, + + _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 amountOfFields = 0; + this._eachField(function() { + amountOfFields++; + }); + + return $("
").addClass(this.cellClass).attr("colspan", amountOfFields) + .append(this.renderTemplate(this.noDataContent, this))); + }, + + _createRow: function(item, itemIndex) { + var $result; + + if($.isFunction(this.rowRenderer)) { + $result = this.renderTemplate(this.rowRenderer, this, { item: item, itemIndex: itemIndex }); + } else { + $result = $("
").append(this.renderTemplate(field.itemTemplate || fieldValue, field, args)); + } + + return this._prepareCell($result, field); + }, + + _getItemFieldValue: function(item, field) { + var props = field.name.split('.'); + var result = item[props.shift()]; + + while(result && props.length) { + result = result[props.shift()]; + } + + return result; + }, + + _setItemFieldValue: function(item, field, value) { + var props = field.name.split('.'); + var current = item; + var prop = props[0]; + + while(current && props.length) { + item = current; + prop = props.shift(); + current = item[prop]; + } + + if(!current) { + while(props.length) { + item = item[prop] = {}; + prop = props.shift(); + } + } + + item[prop] = value; + }, + + 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._normalizeField(field); + order = order || ((this._sortField === field) ? this._reversedSortOrder(this._sortOrder) : SORT_ORDER_ASC); + + this._sortField = field; + this._sortOrder = order; + }, + + _normalizeField: 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 = this._visibleFieldIndex(this._sortField); + + this._headerRow.find("th").eq(fieldIndex) + .addClass(this._sortOrder === SORT_ORDER_ASC ? this.sortAscClass : this.sortDescClass); + }, + + _visibleFieldIndex: function(field) { + return $.inArray(field, $.grep(this.fields, function(f) { return f.visible; })); + }, + + _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 = []; + + 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 width = (this.width === "auto") ? this._getAutoWidth() : this.width; + + this._container.width(width); + }, + + _getAutoWidth: function() { + var $headerGrid = this._headerGrid, + $header = this._header; + + $headerGrid.width("auto"); + + var contentWidth = $headerGrid.outerWidth(); + var borderWidth = $header.outerWidth() - $header.innerWidth(); + + $headerGrid.width(""); + + return contentWidth + borderWidth; + }, + + _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; + } + + this._callEventHandler(this.onPageChanged, { + pageIndex: pageIndex + }); + }, + + _controllerCall: function(method, param, isCanceled, doneCallback) { + if(isCanceled) + return $.Deferred().reject().promise(); + + this._showLoading(); + + var controller = this._controller; + if(!controller || !controller[method]) { + throw Error("controller has no method '" + method + "'"); + } + + return normalizePromise(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() { + if(!this.loadIndication) + return; + + clearTimeout(this._loadingTimer); + + this._loadingTimer = setTimeout($.proxy(function() { + this._loadIndicator.show(); + }, this), this.loadIndicationDelay); + }, + + _hideLoading: function() { + if(!this.loadIndication) + return; + + 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) { + this._setItemFieldValue(result, field, 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._getValidatedInsertItem(); + + if(!insertingItem) + return $.Deferred().reject().promise(); + + 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 + }); + }); + }, + + _getValidatedInsertItem: function() { + var item = this._getInsertItem(); + return this._validateItem(item, this._insertRow) ? item : null; + }, + + _getInsertItem: function() { + var result = {}; + this._eachField(function(field) { + if(field.inserting) { + this._setItemFieldValue(result, field, field.insertValue()); + } + }); + return result; + }, + + _validateItem: function(item, $row) { + var validationErrors = []; + + var args = { + item: item, + itemIndex: this._rowIndex($row), + row: $row + }; + + this._eachField(function(field) { + if(!field.validate || + ($row === this._insertRow && !field.inserting) || + ($row === this._getEditRow() && !field.editing)) + return; + + var fieldValue = this._getItemFieldValue(item, field); + + var errors = this._validation.validate($.extend({ + value: fieldValue, + rules: field.validate + }, args)); + + this._setCellValidity($row.children().eq(this._visibleFieldIndex(field)), errors); + + if(!errors.length) + return; + + validationErrors.push.apply(validationErrors, + $.map(errors, function(message) { + return { field: field, message: message }; + })); + }); + + if(!validationErrors.length) + return true; + + var invalidArgs = $.extend({ + errors: validationErrors + }, args); + this._callEventHandler(this.onItemInvalid, invalidArgs); + this.invalidNotify(invalidArgs); + + return false; + }, + + _setCellValidity: function($cell, errors) { + $cell + .toggleClass(this.invalidClass, !!errors.length) + .attr("title", errors.join("\n")); + }, + + 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; + + var item = $row.data(JSGRID_ROW_DATA_KEY); + + var args = this._callEventHandler(this.onItemEditing, { + row: $row, + item: item, + itemIndex: this._itemIndex(item) + }); + + if(args.cancel) + return; + + if(this._editingRow) { + this.cancelEdit(); + } + + var $editRow = this._createEditRow(item); + + this._editingRow = $row; + $row.hide(); + $editRow.insertBefore($row); + $row.data(JSGRID_EDIT_ROW_DATA_KEY, $editRow); + }, + + _createEditRow: function(item) { + if($.isFunction(this.editRowRenderer)) { + return $(this.renderTemplate(this.editRowRenderer, this, { item: item, itemIndex: this._itemIndex(item) })); + } + + var $result = $("
", field, "editcss") + .append(this.renderTemplate(field.editTemplate || "", field, { value: fieldValue, item: item })) + .appendTo($result); + }); + + return $result; + }, + + updateItem: function(item, editedItem) { + if(arguments.length === 1) { + editedItem = item; + } + + var $row = item ? this.rowByItem(item) : this._editingRow; + editedItem = editedItem || this._getValidatedEditedItem(); + + if(!editedItem) + return; + + return this._updateRow($row, editedItem); + }, + + _getValidatedEditedItem: function() { + var item = this._getEditedItem(); + return this._validateItem(item, this._getEditRow()) ? item : null; + }, + + _updateRow: function($updatingRow, editedItem) { + var updatingItem = $updatingRow.data(JSGRID_ROW_DATA_KEY), + updatingItemIndex = this._itemIndex(updatingItem), + updatedItem = $.extend(true, {}, updatingItem, editedItem); + + var args = this._callEventHandler(this.onItemUpdating, { + row: $updatingRow, + item: updatedItem, + itemIndex: updatingItemIndex, + previousItem: updatingItem + }); + + return this._controllerCall("updateItem", updatedItem, args.cancel, function(loadedUpdatedItem) { + var previousItem = $.extend(true, {}, updatingItem); + updatedItem = loadedUpdatedItem || $.extend(true, updatingItem, editedItem); + + var $updatedRow = this._finishUpdate($updatingRow, updatedItem, updatingItemIndex); + + this._callEventHandler(this.onItemUpdated, { + row: $updatedRow, + item: updatedItem, + itemIndex: updatingItemIndex, + previousItem: previousItem + }); + }); + }, + + _rowIndex: function(row) { + return this._content.children().index($(row)); + }, + + _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) { + this._setItemFieldValue(result, field, field.editValue()); + } + }); + return result; + }, + + cancelEdit: function() { + if(!this._editingRow) + return; + + this._getEditRow().remove(); + this._editingRow.show(); + this._editingRow = null; + }, + + _getEditRow: function() { + return this._editingRow && this._editingRow.data(JSGRID_EDIT_ROW_DATA_KEY); + }, + + 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); + }; + + var locales = {}; + + var locale = function(lang) { + var localeConfig = $.isPlainObject(lang) ? lang : locales[lang]; + + if(!localeConfig) + throw Error("unknown locale " + lang); + + setLocale(jsGrid, localeConfig); + }; + + var setLocale = function(obj, localeConfig) { + $.each(localeConfig, function(field, value) { + if($.isPlainObject(value)) { + setLocale(obj[field] || obj[field[0].toUpperCase() + field.slice(1)], value); + return; + } + + if(obj.hasOwnProperty(field)) { + obj[field] = value; + } else { + obj.prototype[field] = value; + } + }); + }; + + window.jsGrid = { + Grid: Grid, + fields: fields, + setDefaults: setDefaults, + locales: locales, + locale: locale, + version: '1.5.3' + }; + +}(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(); + }, + + reset: function() { + 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") + }; + }, + + reset: function() { + return this._grid.loadData(); + }, + + 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 Validation(config) { + this._init(config); + } + + Validation.prototype = { + + _init: function(config) { + $.extend(true, this, config); + }, + + validate: function(args) { + var errors = []; + + $.each(this._normalizeRules(args.rules), function(_, rule) { + if(rule.validator(args.value, args.item, rule.param)) + return; + + var errorMessage = $.isFunction(rule.message) ? rule.message(args.value, args.item) : rule.message; + errors.push(errorMessage); + }); + + return errors; + }, + + _normalizeRules: function(rules) { + if(!$.isArray(rules)) + rules = [rules]; + + return $.map(rules, $.proxy(function(rule) { + return this._normalizeRule(rule); + }, this)); + }, + + _normalizeRule: function(rule) { + if(typeof rule === "string") + rule = { validator: rule }; + + if($.isFunction(rule)) + rule = { validator: rule }; + + if($.isPlainObject(rule)) + rule = $.extend({}, rule); + else + throw Error("wrong validation config specified"); + + if($.isFunction(rule.validator)) + return rule; + + return this._applyNamedValidator(rule, rule.validator); + }, + + _applyNamedValidator: function(rule, validatorName) { + delete rule.validator; + + var validator = validators[validatorName]; + if(!validator) + throw Error("unknown validator \"" + validatorName + "\""); + + if($.isFunction(validator)) { + validator = { validator: validator }; + } + + return $.extend({}, validator, rule); + } + }; + + jsGrid.Validation = Validation; + + + var validators = { + required: { + message: "Field is required", + validator: function(value) { + return value !== undefined && value !== null && value !== ""; + } + }, + + rangeLength: { + message: "Field value length is out of the defined range", + validator: function(value, _, param) { + return value.length >= param[0] && value.length <= param[1]; + } + }, + + minLength: { + message: "Field value is too short", + validator: function(value, _, param) { + return value.length >= param; + } + }, + + maxLength: { + message: "Field value is too long", + validator: function(value, _, param) { + return value.length <= param; + } + }, + + pattern: { + message: "Field value is not matching the defined pattern", + validator: function(value, _, param) { + if(typeof param === "string") { + param = new RegExp("^(?:" + param + ")$"); + } + return param.test(value); + } + }, + + range: { + message: "Field value is out of the defined range", + validator: function(value, _, param) { + return value >= param[0] && value <= param[1]; + } + }, + + min: { + message: "Field value is too small", + validator: function(value, _, param) { + return value >= param; + } + }, + + max: { + message: "Field value is too large", + validator: function(value, _, param) { + return value <= param; + } + } + }; + + jsGrid.validators = validators; + +}(jsGrid, jQuery)); + +(function(jsGrid, $, undefined) { + + function Field(config) { + $.extend(true, this, config); + this.sortingFunc = this._getSortingFunc(); + } + + Field.prototype = { + name: "", + title: null, + css: "", + align: "", + width: 100, + + visible: true, + filtering: true, + inserting: true, + editing: true, + sorting: true, + sorter: "string", // name of SortStrategy or function to compare elements + + headerTemplate: function() { + return (this.title === undefined || this.title === null) ? this.name : this.title; + }, + + 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, + readOnly: false, + + 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 ""; + + return this.insertControl = this._createTextBox(); + }, + + editTemplate: function(value) { + if(!this.editing) + return this.itemTemplate.apply(this, arguments); + + 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") + .prop("readonly", !!this.readOnly); + } + }); + + 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", + readOnly: false, + + filterValue: function() { + return this.filterControl.val() + ? parseInt(this.filterControl.val() || 0, 10) + : undefined; + }, + + insertValue: function() { + return this.insertControl.val() + ? parseInt(this.insertControl.val() || 0, 10) + : undefined; + }, + + editValue: function() { + return this.editControl.val() + ? parseInt(this.editControl.val() || 0, 10) + : undefined; + }, + + _createTextBox: function() { + return $("").attr("type", "number") + .prop("readonly", !!this.readOnly); + } + }); + + 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 ""; + + return this.insertControl = this._createTextArea(); + }, + + editTemplate: function(value) { + if(!this.editing) + return this.itemTemplate.apply(this, arguments); + + var $result = this.editControl = this._createTextArea(); + $result.val(value); + return $result; + }, + + _createTextArea: function() { + return $("