diff --git a/.gitignore b/.gitignore index f1527a7..441a3d6 100644 --- a/.gitignore +++ b/.gitignore @@ -2,7 +2,7 @@ node_modules config.js public/main.css -public/admin/main.css +public/admin/index.css public/admin/*.css public/emailFailures.txt sessions/ diff --git a/app/initialData.js b/app/initialData.js deleted file mode 100644 index 08a58dd..0000000 --- a/app/initialData.js +++ /dev/null @@ -1,116 +0,0 @@ -var Promise = require('bluebird'); - -module.exports = function(sequelize) { - var models = sequelize.models; - - //Pre-populate a new database with some data. - models.Category.count().then(function(count) { - if(count == 0) { - var basicJarIds = []; - - models.User.create({login: 'wcrisman', password: 'landFJ40', admin: true}); - - models.Venue.create({name: 'Boonville'}); - models.Venue.create({name: 'Clement St Farmers Market in SF'}); - models.Venue.create({name: 'Ukiah Farmers Market'}); - models.Venue.create({name: 'Mendocino Farmers Market'}); - models.Venue.create({name: 'Ft Bragg Farmers Market'}); - - Promise.each([ - models.Measure.create({name: 'Jar 4oz', postfix: '4oz'}), - models.Measure.create({name: 'Jar 8oz', postfix: '8oz'}), - models.Measure.create({name: 'Jar 12oz', postfix: '12oz'}), - models.Measure.create({name: 'Jar 16oz', postfix: '16oz'}), - models.Measure.create({name: 'Jar 32oz', postfix: '32oz'}), - models.Measure.create({name: 'Jar 64oz', postfix: '64oz'}), - models.Measure.create({name: 'Pounds', postfix: 'lbs'}), - models.Measure.create({name: 'Each', postfix: ''}), - models.Measure.create({name: 'Bags', postfix: 'bags'}) - ], function(value, index, length) { - //Collect the first 5 jar ids. - if(index < 5) basicJarIds.push(value.id); - }).then(function() { - models.Category.create({name: 'VAP'}).then(function(category) { - models.Subcategory.create({name: 'Soups'}).then(function(subcategory) { - category.addSubcategory(subcategory); - models.Item.create({name: 'Fava Bean Bisque', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Tomato Basil Soup', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Winter Squash Soup', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - }); - models.Subcategory.create({name: 'Drink Mixes & Syrups'}).then(function(subcategory) { - category.addSubcategory(subcategory); - models.Item.create({name: 'Bloody Mary Mix', defaultPrice: 12, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Grape Syrup', defaultPrice: 8, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Prickly Pear Syrup', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Quince Syrup', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Strawberry Syrup', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Wild Plum Syrup', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - }); - models.Subcategory.create({name: 'Fermented'}).then(function(subcategory) { - category.addSubcategory(subcategory); - models.Item.create({name: 'Napa Cabbage Sauerkraut', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Napa Cabbage Sauerkraut w/ Watercress & Espelette Pepper', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Red Sauerkraut', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - }); - models.Subcategory.create({name: 'Dried Goods'}).then(function(subcategory) { - category.addSubcategory(subcategory); - models.Item.create({name: 'Dried Strawberries', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Membrillo', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Sugared Jalape�os', defaultPrice: 10, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - }); - models.Subcategory.create({name: 'Spices'}).then(function(subcategory) { - category.addSubcategory(subcategory); - models.Item.create({name: 'Basque Pepper Powder', defaultPrice: 8, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Korean Pepper Powder', defaultPrice: 8, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Smoked Basque Pepper Powder', defaultPrice: 8, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - models.Item.create({name: 'Smoked Korean Pepper Powder', defaultPrice: 8, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - }); - models.Subcategory.create({name: 'Specialty'}).then(function(subcategory) { - category.addSubcategory(subcategory); - models.Item.create({name: 'Pure Lard', defaultPrice: 4, measures: basicJarIds}).then(function(item) { - subcategory.addItem(item); - }); - }); - }); - }); - } - }); -}; \ No newline at end of file diff --git a/csv.js b/csv.js new file mode 100644 index 0000000..60401cd --- /dev/null +++ b/csv.js @@ -0,0 +1,15 @@ + +var fs = require("fs"); +var csv = require('csv-parse'); + +module.exports = { + read: function(csvFilePath, callback) { + var parser = csv({delimiter: '\t'}, function(error, data) { + if(error) callback(error); + else callback(null, data); + }); + + + fs.createReadStream(csvFilePath).pipe(parser); + } +}; \ No newline at end of file diff --git a/downloaded tools/jsgrid-1.5.2.zip b/downloaded tools/jsgrid-1.5.2.zip new file mode 100644 index 0000000..a27086e Binary files /dev/null and b/downloaded tools/jsgrid-1.5.2.zip differ diff --git a/importAll.js b/importAll.js new file mode 100644 index 0000000..933a251 --- /dev/null +++ b/importAll.js @@ -0,0 +1,88 @@ +var Promise = require('bluebird'); +var models = require("./models"); + +var fm = 0; +var res = 0; +var rt = 0; +var mail = 0; + +//First get the basics, then load the items. Finally load sales and production data. +//Create the venue types. +Promise.each([ + models.VenueType.findOrCreate({where: {name: 'Farmers Market'}, defaults: {name: 'Farmers Market'}}), + models.VenueType.findOrCreate({where: {name: 'Restaurant'}, defaults: {name: 'Restaurant'}}), + models.VenueType.findOrCreate({where: {name: 'Retail'}, defaults: {name: 'Retail'}}), + models.VenueType.findOrCreate({where: {name: 'Mail'}, defaults: {name: 'Mail'}}) +], function(value, index, length) { + switch(index) { + case 0: + fm = value[0].id; + break; + case 1: + res = value[0].id; + break; + case 2: + rt = value[0].id; + break; + case 3: + mail = value[0].id; + break; + } +}).then(function() { + //Create the measures and venues. + Promise.each([ + //Initialize the users. + models.User.findOrCreate({where: {login: 'wcrisman'}, defaults: {login: 'wcrisman', password: 'landFJ40', admin: true}}), + //Initialize the measures. + models.Measure.findOrCreate({where: {name: 'Jar 4oz'}, defaults: {name: 'Jar 4oz', postfix: '4oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 8oz'}, defaults: {name: 'Jar 8oz', postfix: '8oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 12oz'}, defaults: {name: 'Jar 12oz', postfix: '12oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 16oz'}, defaults: {name: 'Jar 16oz', postfix: '16oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 32oz'}, defaults: {name: 'Jar 32oz', postfix: '32oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 64oz'}, defaults: {name: 'Jar 64oz', postfix: '64oz'}}), + models.Measure.findOrCreate({where: {name: 'Pounds'}, defaults: {name: 'Pounds', postfix: 'lbs'}}), + models.Measure.findOrCreate({where: {name: 'Each'}, defaults: {name: 'Each', postfix: ''}}), + models.Measure.findOrCreate({where: {name: 'Bags'}, defaults: {name: 'Bags', postfix: 'bags'}}), + models.Measure.findOrCreate({where: {name: 'Dozen Large'}, defaults: {name: 'Dozen Large', postfix: '12/lg'}}), + models.Measure.findOrCreate({where: {name: 'Dozen Small'}, defaults: {name: 'Dozen Small', postfix: '12/sm'}}), + models.Measure.findOrCreate({where: {name: 'Half Dozen Large'}, defaults: {name: 'Half Dozen Large', postfix: '6/lg'}}), + models.Measure.findOrCreate({where: {name: 'Half Dozen Small'}, defaults: {name: 'Half Dozen Small', postfix: '6/sm'}}), + //Initialize the venues. + models.Venue.findOrCreate({where: {name: 'Boonville'}, defaults: {name: 'Boonville', typeId: fm}}), + models.Venue.findOrCreate({where: {name: 'Clement St'}, defaults: {name: 'Clement St', typeId: fm}}), + models.Venue.findOrCreate({where: {name: 'Ukiah'}, defaults: {name: 'Ukiah', typeId: fm}}), + models.Venue.findOrCreate({where: {name: 'Mendocino'}, defaults: {name: 'Mendocino', typeId: fm}}), + models.Venue.findOrCreate({where: {name: 'Ft Bragg'}, defaults: {name: 'Ft Bragg', typeId: fm}}), + models.Venue.findOrCreate({where: {name: 'On Farm'}, defaults: {name: 'On Farm', typeId: rt}}), + models.Venue.findOrCreate({where: {name: 'Unknown Restaurant'}, defaults: {name: 'Unknown Restaurant', typeId: res}}), + models.Venue.findOrCreate({where: {name: 'Yorkville Market'}, defaults: {name: 'Yorkville Market', typeId: rt}}), + models.Venue.findOrCreate({where: {name: 'Yorkville Cellars'}, defaults: {name: 'Yorkville Cellars', typeId: rt}}), + models.Venue.findOrCreate({where: {name: 'Mail Order'}, defaults: {name: 'Mail Order', typeId: mail}}), + models.Venue.findOrCreate({where: {name: 'Website Order'}, defaults: {name: 'Website Order', typeId: mail}}) + ], function(value, index, length) { + //Ignore results for now. + }).then(function() { + // var softErrors = []; + // + // //Read all the Categories, Subcategories, and Items. + // require('./importItems.js')(models, softErrors).then(function() { + // //Read all the sales data. + // require('./importSales.js')(models, softErrors).then(function() { + // //Log all errors at the end so they are easily visible. + // if(softErrors.length) { + // for(var i = 0; i < softErrors.length; i++) { + // console.log(softErrors[i]); + // } + // } + // + // process.exit(0); + // }).catch(function(e) { + // console.log(e); + // process.exit(1); + // }); + // }).catch(function(e) { + // console.log(e); + // process.exit(1); + // }); + }); +}); \ No newline at end of file diff --git a/importItems.csv b/importItems.csv new file mode 100644 index 0000000..330c7aa --- /dev/null +++ b/importItems.csv @@ -0,0 +1,206 @@ +Item Subcategory Category 32 oz 16 oz 12 oz 8 oz 4 oz Total 32 oz 16 oz 12 oz 8 oz 4 oz Total 32 oz 16 oz 12 oz 8 oz 4 oz Total 32 oz 16 oz 12 oz 8 oz 4 oz Total 32 oz 16 oz 12 oz 8 oz 4 oz Total 32 oz 16 oz 12 oz 8 oz 4 oz Total 32 oz 16 oz 12 oz 8 oz 4 oz Total 32 oz 16 oz 12 oz 8 oz 4 oz Total +3packs Specialty VAP 75 +Adobo Sauce Condiment VAP 14 9 23 70 70 39 1 40 57% 59 59 59 59 100% 26 +Apple Butter Jams & Spreads VAP 42 42 36 36 15 16 31 45 45 145% 60 13 73 30 16 46 63% +Apple Jam Jams & Spreads VAP 48 32 80 4 3 7 9% +Apple Jam, French Jams & Spreads VAP 15 15 15 15 100% +Apple Jam, Rum Jams & Spreads VAP 47 14 +Apple Pie Jam Jams & Spreads VAP 38 20 58 16 8 24 41% 74 34 108 34 11 45 42% +Apple Raisin Chutney Chutney VAP 36 36 0 21 21 NEG 0 17 17 NEG +Apple Rhubarb Chutney Chutney VAP 27 27 2 2 7% +Apple Walnut Conserve Jams & Spreads VAP 21 +Applesauce Honey Side VAP 20 1 21 9 9 43% 13 13 12 12 92% +Applesauce, Blackberry Side VAP 20 20 9 9 45% +Applesauce, Cinnamon Side VAP 8 8 0 2 2 NEG 29 29 10 10 34% +Applesauce, Quince Side VAP 29 29 0 0% +Applesauce, Rhubarb Side VAP 18 18 5 5 28% +Applesauce, Unsweetened Side VAP 7 7 11 11 35 35 8 1 9 26% 47 1 48 33 33 69% +Applesauce. Strawberry Side VAP 36 36 10 10 28% +Artichoke Antipasto Side VAP 15 15 17 17 113% 0 +Baba Ganoush Condiment VAP 14 1 +Bangkok Peppers Side VAP 53 53 5 5 9% +Bartlett Pear jam Jams & Spreads VAP 69 69 24 10 34 58 6 64 188% 28 21 49 24 12 36 73% +Bartlett Vanilla Jam Jams & Spreads VAP 15 9 24 1 4 5 21% 35 5 40 16 2 18 45% +BBQ Sauce Condiment VAP 18 18 15 7 22 0 1 3 4 NEG 28 28 24 24 86% +Beet Marmalade Jams & Spreads VAP 23 38 61 5 16 21 34% 12 15 27 27 19 46 170% 10 18 +Blackberry Chutney Chutney VAP 26 +Blackberry Hazelnut Conserve Jams & Spreads VAP 22 +Blackberry Jalapeno Jam Jams & Spreads VAP 15 18 +Blackberry Jam Jams & Spreads VAP 32 32 24 24 67 25 92 41 23 64 70% 102 51 153 92 25 117 76% 50 +Blackberry Jelly Jams & Spreads VAP 20 20 10 10 50% 0 11 1 12 NEG +Blackberry Mint Jam Jams & Spreads VAP 21 +Blackberry Peach Jam Jams & Spreads VAP 16 2 18 10 2 12 67% 0 5 5 NEG +Blackberry Syrup Drink Mixes & Syrups VAP 1 17 18 22 22 122% +Blackberry Vanilla Jam Jams & Spreads VAP 65 20 85 38 8 46 54% 28 3 +Bloody Mary Mix Drink Mixes & Syrups VAP 9 9 72 72 53 53 74% 167 167 112 112 67% +Bosc Pear Chutney Chutney VAP 15 15 2 2 13% 0 15 15 NEG +Bosc Pear Jam Jams & Spreads VAP 18 18 2 21 23 128% 0 10 10 NEG +Brinjal Eggplant Relish Pickles & Relishes VAP 0 11 11 NEG +Candied Quince Side VAP 0 4 4 NEG +Catsup Condiment VAP 37 28 65 38 13 7 58 55 7 62 38 14 52 84% 49 49 2 40 42 86% +Chili Sauce Condiment VAP 18 18 11 11 61% 0 3 3 NEG +Chow Chow Mild Pickles & Relishes VAP 25 25 5 5 20% 0 12 12 NEG +Chow Chow Spicy Pickles & Relishes VAP 27 27 6 6 22% 0 19 19 NEG +Comice Pear Jam Jams & Spreads VAP 17 7 24 12 5 17 71% +Comice Pear Jam w/ Sherry Jams & Spreads VAP 36 36 27 27 75% +Confetti Relish Pickles & Relishes VAP 0 2 2 NEG +Cowboy Candy Side VAP 38 38 76 13 2 15 20% +Curried Onions Side VAP 11 11 0 0% +D'Anjou Pear Conserve Jams & Spreads VAP 0 2 2 NEG +D'Anjou Pear Jam Jams & Spreads VAP 38 16 54 3 3 6% 14 36 50 42 20 62 124% +Dilly Beans Pickles & Relishes VAP 17 17 34 20 20 11 134 145 15 36 51 35% 11 73 84 6 98 104 124% +Dried, Pear Dried VAP 1 +Dried, Strawberry Dried VAP 22 +Dried, Tomatillo Dried VAP 2 +Dried Tomato Dried VAP 56 18 32% +Eggplant Conserve Jams & Spreads VAP 21 21 10 14 24 22 3 25 104% 18 36 54 9 3 12 22% +Pesto, Eggplant Condiment VAP 12 2 17% +Pesto, Garlic Scape Condiment VAP 13 12 92% +Elderberry Butter Jams & Spreads VAP 23 23 23 23 100% 36 36 36 36 100% +Enchilada Sauce Condiment VAP 3 9 12 9 9 75% +Espelette Peppers Side VAP 36 36 3 3 8% 86 86 60 60 70% +Fava Bean Bisque Soup VAP 9 9 0 0% +Fava Bean Soup Soup VAP 23 7 30 20 7 27 90% 19 1 20 21 21 105% +Fennel Relish Pickles & Relishes VAP 4 4 0 4 4 NEG 22 22 10 10 45% +Fig Jam Jams & Spreads VAP 23 8 31 61 21 82 69 44 113 40 5 45 40% 97 41 138 82 33 115 83% +Ginger Peaches Side VAP 17 17 11 11 65% 0 6 6 NEG +Grape Jam Jams & Spreads VAP 20 12 32 7 3 10 26 12 38 6 5 11 29% 22 13 35 12 10 22 63% +Grape Jam, White Jams & Spreads VAP 36 36 0 0% +Grape Syrup Drink Mixes & Syrups VAP 9 4 13 2 2 15% +Green Gage Jam Jams & Spreads VAP 24 24 48 0 14 8 22 NEG 28 17 45 21 21 47% +Green Gage Jam w/ Ginger Jams & Spreads VAP 40 4 44 10 1 11 25% +Green Gage Jelly Jams & Spreads VAP 0 6 6 NEG +Green Tomato Catsup Condiment VAP 19 19 0 0% +Green Tomato Chutney Chutney VAP 24 24 19 19 42 42 17 17 40% 0 16 16 NEG +Green Tomato Jam Jams & Spreads VAP 12 6 18 58 10 68 38 3 41 60% 15 12 27 17 13 30 111% +Green Tomato Relish Pickles & Relishes VAP 72 72 1 1 1% +Green Tomato Salsa Condiment VAP 42 42 4 4 10% +Jalapeno Salsa Condiment VAP 35 35 19 19 54% 58 12 70 37 37 53% +Jalapeno Carrot Hot Sauce Condiment VAP 12 12 11 11 92% +Jalapenos Side VAP 30 30 8 8 27% 38 38 20 20 53% +Jalapeno Candy Specialty VAP 101 115 114% +Jujube & Ginger Butter Jams & Spreads VAP 4 4 4 4 100% 16 11 27 5 4 9 33% +Jujube Butter Jams & Spreads VAP 13 11 24 13 13 30 19 49 9 10 19 39% 0 22 2 24 NEG +Kadota Figs Side VAP 0 2 2 NEG +Kimchi Fermented VAP 12 23 136 136 52 52 38% 40 40 113 113 283% +Korean Peppers Side VAP 58 58 8 8 14% 0 29 29 NEG +Lard Specialty VAP 38 38 15 15 39% 20 1 21 21 21 100% +Leather, Unspecified Dried VAP 52 +Leather, Apple Dried VAP 18 4 22% +Leather, Grape Dried VAP 46 10 22% +Leather, Peach Dried VAP 18 6 33% +Leather, Persimmon Dried VAP 6 0% +Leather, Pumpkin Dried VAP 102 69 68% +Leather, Strawberry Dried VAP 60 56 93% +Leek & Potato Soup Soup VAP 21 4 25 2 2 4 16% 0 2 2 NEG +Marionberry Jam Jams & Spreads VAP 17 6 23 8 4 12 86 42 128 51 32 83 65% 110 41 151 89 28 117 77% +Membrillo Dried VAP 48 66 138% +Membrillo Truffles Specialty VAP 8 NEG +Mint Jelly Jams & Spreads VAP 23 23 3 3 13% 0 13 13 NEG +Mixed Berry Jam Jams & Spreads VAP 5 5 26 12 38 21 11 32 84% 18 33 51 18 25 43 84% +Napa Kraut watercress pepper Fermented VAP 28 28 15 15 54% +Napa Kraut, Onion Fermented VAP 26 26 19 19 73% +Onion Marmalade Jams & Spreads VAP 60 60 27 27 45% 0 27 27 NEG +Pasta sauce Condiment VAP 14 14 14 14 100% +Peach BBQ Condiment VAP 18 18 5 5 28% +Peach Butter Jams & Spreads VAP 30 30 26 26 87% 0 1 1 NEG +Peach Chutney Chutney VAP 13 7 20 58 58 9 32 41 71% 0 23 23 NEG +Peach Jam Jams & Spreads VAP 18 5 23 19 3 22 96% +Peach Jam, Babcobk w/ Ginger Jams & Spreads VAP 36 17 53 9 4 13 25% +Peach Jam, Babcock Jams & Spreads VAP 20 10 30 18 8 26 87% 89 41 130 25 12 37 28% +Peach Jam, Babcock w/ Lavender Jams & Spreads VAP 54 34 88 21 12 33 38% +Peach Jam, Indian Free Jams & Spreads VAP 55 24 79 17 4 21 27% +Peach Jam, Saturn Jams & Spreads VAP 19 19 10 10 53% +Pear Apple Jam Jams & Spreads VAP 30 20 50 0 0% 0 16 13 29 NEG +Pear Chutney Chutney VAP 18 18 6 6 33% 42 42 9 9 21% +Pear Dessert Topping Condiment VAP 14 14 2 2 14% 0 5 5 NEG +Pear Ginger Butter Jams & Spreads VAP 36 17 53 21 21 48 30 78 45 16 61 78% 29 8 37 35 20 55 149% +Pear Jam, Seckel Jams & Spreads VAP 55 55 31 21 52 36 4 40 77% 0 18 4 22 NEG +Pear Syrup Drink Mixes & Syrups VAP 9 2 11 8 2 10 91% 0 11 11 NEG +Pearsauce Side VAP 36 36 1 1 3% 0 15 15 NEG +Pepper Jelly Jams & Spreads VAP 35 32 67 16 1 17 30 12 42 247% 66 44 110 47 27 74 67% +Pesto, Mint Condiment VAP 57 47 82% +Pickle Relish Pickles & Relishes VAP 0 0 0% 0 3 3 NEG +Pickled Asparagus Pickles & Relishes VAP 50 50 47 47 94% 105 105 73 73 70% +Pickled Beets Pickles & Relishes VAP 8 8 44 44 31 31 70% 39 39 47 47 121% +Pickled Eggs Pickles & Relishes VAP 10 10 26 26 50 50 49 49 98% 53 53 37 37 70% +Pickled Eggplant Pickles & Relishes VAP 0 4 4 NEG +Pickled Fennel Pickles & Relishes VAP 22 4 26 4 27 31 3 7 10 32% 17 17 6 6 35% +Pickled Okra Pickles & Relishes VAP 7 7 9 1 7 5 22 59 4 63 6 28 3 37 59% 4 65 1 70 52 52 74% +Pickled Onions (sweet) Pickles & Relishes VAP 1 11 12 1 5 6 50% 0 2 2 NEG +Pickled Onion Dill w/Espelette Pickles & Relishes VAP 13 13 4 4 31% +Pickled Onion Dill w/Sweet Pep. Pickles & Relishes VAP 15 15 8 8 53% +Pickled Onions Fennel Pickles & Relishes VAP 42 42 23 23 55% +Pickled Onions, Dill Pickles & Relishes VAP 2 2 0 0% +Pickled Peas Pickles & Relishes VAP 0 3 3 NEG +Pickled Scapes Pickles & Relishes VAP 24 24 18 18 75% 0 5 5 NEG +Pickled Summer Squash Pickles & Relishes VAP 9 9 76 76 10 10 13% 0 59 59 NEG +Pickles, Bread Butter Pickles & Relishes VAP 22 22 20 20 18 18 23 3 26 144% 96 96 67 67 70% +Pickles, Dill Pickles & Relishes VAP 40 7 47 66 21 87 87 26 113 75 22 97 86% 165 165 75 75 45% +Pickles, Fennel Pickles & Relishes VAP 21 21 17 17 81% 109 109 47 47 43% +Pickles, Ginger Garlic Pickles & Relishes VAP 13 13 2 33 35 27 27 77% 49 49 48 48 98% +Pickles, Green Tomato Pickles & Relishes VAP 0 4 4 NEG 24 24 2 2 8% +Pippin Apple Chutney Chutney VAP 41 41 15 15 37% +Plum BBQ Condiment VAP 0 1 1 NEG +Plum Jam Jams & Spreads VAP 21 21 49 12 61 21 8 29 48% 16 7 23 38 10 48 209% +Plum Jam, Spiced Jams & Spreads VAP 25 6 31 3 4 7 23% 0 2 1 3 NEG +Plum Jam, Santa Rosa Jams & Spreads VAP 13 11 24 66 9 75 31 9 40 53% +Plum Sauce Condiment VAP 0 14 14 NEG +Plum Syrup Drink Mixes & Syrups VAP 3 3 4 4 133% +Popsicles Specialty VAP 249 205 82% +Prickly Pear Jelly Jams & Spreads VAP 46 24 70 4 4 6% 0 22 3 25 NEG +Prickly Pear Syrup Drink Mixes & Syrups VAP 20 21 8 49 11 11 22% +Pumpkin Chutney Chutney VAP 11 11 11 11 100% 0 +Quince Chutney Chutney VAP 65 65 0 35 35 NEG 0 +Quince Butter Jams & Spreads VAP 29 12 41 19 8 27 66% 0 7 7 NEG +Quince Fennel Butter Jams & Spreads VAP 69 51 120 33 21 54 45% 49 7 56 30 12 42 75% +Quince Jam Jams & Spreads VAP 51 11 62 44 25 69 50 17 67 97% 68 33 101 52 25 77 76% +Quince Jam w/ Cinnamon Jams & Spreads VAP 14 14 1 1 7% +Quince Plum Jam Jams & Spreads VAP 55 55 23 23 42% 0 17 17 NEG +Quince Syrup Drink Mixes & Syrups VAP 23 23 7 7 30% 0 21 21 NEG +Raspberry Jam Jams & Spreads VAP 25 12 37 15 6 21 57% 7 31 38 19 24 43 113% +Ratatouille Relish Pickles & Relishes VAP 43 43 5 5 12% +Rhubarb Jam Jams & Spreads VAP 38 14 52 31 14 45 87% 0 1 1 NEG +Salsa Verde Condiment VAP 6 39 11 56 94 94 38 38 40% 53 53 74 74 140% +Sauerkraut Fermented VAP 108 108 34 34 31% 99 99 +Sauerkraut, Red Fermented VAP 15 15 71 71 47 47 66% 67 67 35 35 52% +Shrubs Specialty VAP 1 2 46 49 1 25 26 53% +Sorrel Soup Soup VAP 7 3 10 4 3 7 70% 0 1 1 2 NEG +Spices, Unspecified Dried VAP 14 +Spices, Basil/Corriander Dried VAP 4 +Spices, Fennel Dried VAP 1 +Spices, Pepper Dried VAP 8 +Spices, Smoked Pepper Dried VAP 20 +Spicy Pepper Relish Pickles & Relishes VAP 73 1 74 31 1 32 43% 20 20 33 33 165% +Spicy Pepper Syrup Drink Mixes & Syrups VAP 12 12 1 1 8% 14 9 23 6 6 26% +Spicy Walla Walla w/ Bangkok Side VAP 7 7 0 0% +Spring Onion Relish Pickles & Relishes VAP 12 12 8 8 67% 0 3 3 NEG +Spring Onion Soup Soup VAP 19 5 24 5 5 21% 0 4 7 11 NEG +Strawberry Espelette Jam Jams & Spreads VAP 21 21 100 100 52 52 52% 0 25 25 NEG +Strawberry Jam Jams & Spreads VAP 20 10 30 179 51 230 105 34 139 60% 100 28 128 125 34 159 124% +Strawberry Balsamic Jam Jams & Spreads VAP 12 12 10 10 83% 8 8 0% +Strawberry Lavender Jam Jams & Spreads VAP 59 38 97 52 27 79 81% 100 49 149 92 40 132 89% +Strawberry Peach Jam Jams & Spreads VAP 55 19 74 35 8 43 58% 104 40 144 56 12 68 47% +Strawberry Pear Jam Jams & Spreads VAP 72 25 97 19 6 25 26% 0 46 16 62 NEG +Strawberry Quince Jam Jams & Spreads VAP 8 8 6 6 75% +Strawberry Rhubarb Jam Jams & Spreads VAP 48 23 71 34 7 41 58% 96 38 134 104 20 124 93% +Strawberry Syrup Drink Mixes & Syrups VAP 0 4 4 NEG 1 23 24 14 14 58% +Strawberry Vanilla Jam Jams & Spreads VAP 20 6 26 52 9 61 63 42 105 66 37 103 98% 86 27 113 80 23 103 91% +Strawberry Vanilla Syrup Drink Mixes & Syrups VAP 5 5 0 0% +Sweet Pepper Relish Pickles & Relishes VAP 26 26 12 12 46% +Sweet Peppers Condiment VAP 18 18 27 14 41 12 12 29% 0 16 4 20 NEG +Sweet Sour Red Cabbage Side VAP 43 43 16 16 37% 0 20 20 NEG +Szechuan Broccoli Pickles & Relishes VAP 0 1 2 3 NEG +Tayberry Jam Jams & Spreads VAP 12 12 11 10 21 175% 38 12 50 36 8 44 88% +Tomatillo Chutney Chutney VAP 19 19 5 5 72 72 10 3 13 18% 0 19 19 NEG +Tomatillo Jam Jams & Spreads VAP 6 27 33 4 4 12% 37 6 43 41 12 53 123% +Tomato Basil Soup Soup VAP 42 6 48 36 4 40 83% 31 1 32 26 5 31 97% +Tomato Juice Cocktail Drink Mixes & Syrups VAP 9 9 0 0% 0 9 9 NEG +Tomato Sauce, Basil Condiment VAP 29 29 24 24 4 16 20 83% 35 35 30 30 86% +Tomato Sauce, Fennel Condiment VAP 8 8 2 2 25% 0 9 9 NEG +Tomato Sauce, Oregano Condiment VAP 28 28 2 2 7% 23 23 19 19 83% +Tokyo Turnips Pickles & Relishes VAP 17 17 18 18 0 6 6 NEG +White Peach Chutney Chutney VAP 0 18 8 26 NEG 15 15 6 6 40% +Wild Plum Jam Jams & Spreads VAP 35 35 15 15 43% +Wild Plum Syrup Drink Mixes & Syrups VAP 13 1 14 8 8 57% +Winter Squash Soup Soup VAP 21 21 7 7 12 1 13 186% 22 22 8 1 9 41% diff --git a/importItems.js b/importItems.js new file mode 100644 index 0000000..1d0873b --- /dev/null +++ b/importItems.js @@ -0,0 +1,221 @@ + +var fs = require("fs"); +var csv = require('./csv.js'); +var Promise = require('bluebird'); + +var fileName = "./importItems.csv"; + +// module.exports = function(models, softErrors) { +// return new Promise(function(resolve, reject) { +var models = require("./models"); +var softErrors = []; +var p = new Promise(function(resolve, reject) { + //var sequelize = models.sequelize; + //The CSV data as an array of rows (each row is an array). + var data; + //The mapping of model attributes to CSV columns. The oz sizes are arrays of columns since there are multiple. + var map = {}; + //The id's in the db for the various jar sizes. + var measureIdMap = {}; + //The id of the 'each' measure. + var eachId = -1; + //A default price for all items. + var DEFAULT_PRICE = 11.0; + + //Collect the metadata from the first row of the CSV data - make a mapping. + function collectMetadata() { + var CATEGORY = 'category'; + var SUBCATEGORY = 'subcategory'; + var ITEM = 'item'; + var OZ32 = '32 oz'; + var OZ16 = '16 oz'; + var OZ12 = '12 oz'; + var OZ8 = '8 oz'; + var OZ4 = '4 oz'; + + //Data is an array of arrays. data[0] = array of headers. data[1] = first row of data. + //Iterate over the columns to create a mapping. + for(var i = 0; i < data[0].length; i++) { + var next = data[0][i]; + + if(next && next != '') { + switch(next.toLowerCase()) { + case CATEGORY: + map.category = i; + break; + case SUBCATEGORY: + map.subcategory = i; + break; + case ITEM: + map.item = i; + break; + case OZ32: + if(!map.oz32) map.oz32 = []; + map.oz32.push(i); + break; + case OZ16: + if(!map.oz16) map.oz16 = []; + map.oz16.push(i); + break; + case OZ12: + if(!map.oz12) map.oz12 = []; + map.oz12.push(i); + break; + case OZ8: + if(!map.oz8) map.oz8 = []; + map.oz8.push(i); + break; + case OZ4: + if(!map.oz4) map.oz4 = []; + map.oz4.push(i); + break; + } + } + } + } + + //Reads a single row of CSV data and adds it to the database. + function readRow(rowIndex) { + if(!data) { + console.log("WTF?"); + //process.exit(1); + reject(new Error("Data is null?!?")); + return; + } + else if(!data[rowIndex]) { + softErrors.push("Empty line? - Found while reading items."); + readRow(rowIndex + 1); + return; + } + + var category = data[rowIndex][map.category]; + var subcategory = data[rowIndex][map.subcategory]; + var item = data[rowIndex][map.item]; + var oz32 = 0; + var oz16 = 0; + var oz12 = 0; + var oz8 = 0; + var oz4 = 0; + + for(var o = 0; o < map.oz32.length; o++) { + oz32 += Math.max(data[rowIndex][map.oz32[o]], 0); + } + + for(var o = 0; o < map.oz16.length; o++) { + oz16 += Math.max(data[rowIndex][map.oz16[o]], 0); + } + + for(var o = 0; o < map.oz12.length; o++) { + oz12 += Math.max(data[rowIndex][map.oz12[o]], 0); + } + + for(var o = 0; o < map.oz8.length; o++) { + oz8 += Math.max(data[rowIndex][map.oz8[o]], 0); + } + + for(var o = 0; o < map.oz4.length; o++) { + oz4 += Math.max(data[rowIndex][map.oz4[o]], 0); + } + + //If the category doesn't yet exist then add it, otherwise get the ID. + models.Category.findOrCreate({where: {name: category}, defaults: {name: category}}).then(function(result) { + var categoryId = result[0].id; //Note: result[1] is a boolean indicating whether the object had to be created. + + //If the subcategory doesn't yet exist then add it, otherwise get the ID. + models.Subcategory.findOrCreate({ + where: {name: subcategory}, + defaults: {name: subcategory, categoryId: categoryId} + }).then(function(result) { + var subcategoryId = result[0].id; + var weightedMeasures = []; + var measures = []; + + if(oz32 > 0) weightedMeasures.push({name: '32 oz', count: oz32}); + if(oz16 > 0) weightedMeasures.push({name: '16 oz', count: oz16}); + if(oz12 > 0) weightedMeasures.push({name: '12 oz', count: oz12}); + if(oz8 > 0) weightedMeasures.push({name: '8 oz', count: oz8}); + if(oz4 > 0) weightedMeasures.push({name: '4 oz', count: oz4}); + + if(weightedMeasures.length == 0) { + measures.push(eachId); + } + else { + //Place the measure that had the most jars at the front of the list. + weightedMeasures.sort(function(a, b) { + return a.count >= b.count ? 1 : -1; + }); + + //Build the array of measure id's. + for(var i = 0; i < weightedMeasures.length; i++) { + measures.push(measureIdMap[weightedMeasures[i].name]); + } + } + + //Ensure the item doesn't yet exist - display an error if it does, otherwise add it. + models.Item.findOrCreate({ + where: {name: item}, + defaults: {name: item, subcategoryId: subcategoryId, measures: measures} + }).then(function(result) { + if(!result[1]) { + console.log("The item named '" + item + "' already exists in the database."); + } + + //Recursively read rows until there are no more rows to read. + if(rowIndex + 1 < data.length) readRow(rowIndex + 1); + else resolve(); + }); + }) + }); + } + + if(fs.existsSync(fileName)) { + csv.read(fileName, function(error, csvData) { + if(error) console.log(error); + else { + data = csvData; + //Collect the mapping data. + collectMetadata(); + + //Read or create the required models to make this all work: + Promise.each([ + models.Measure.findOrCreate({where: {name: 'Jar 32oz'}, defaults: {name: 'Jar 32oz', postfix: '32oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 16oz'}, defaults: {name: 'Jar 16oz', postfix: '16oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 12oz'}, defaults: {name: 'Jar 12oz', postfix: '12oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 8oz'}, defaults: {name: 'Jar 8oz', postfix: '8oz'}}), + models.Measure.findOrCreate({where: {name: 'Jar 4oz'}, defaults: {name: 'Jar 4oz', postfix: '4oz'}}), + models.Measure.findOrCreate({where: {name: 'Each'}, defaults: {name: 'Each', postfix: ''}}) + ], function(value, index, length) { + switch(index) { + case 0: + measureIdMap['32 oz'] = value[0].id; + break; + case 1: + measureIdMap['16 oz'] = value[0].id; + break; + case 2: + measureIdMap['12 oz'] = value[0].id; + break; + case 3: + measureIdMap['8 oz'] = value[0].id; + break; + case 4: + measureIdMap['4 oz'] = value[0].id; + break; + case 5: + eachId = value[0].id; + break; + } + }).then(function() { + //Read the first row. + readRow(1); + }); + } + }); + } + else { + console.log("Cannot find " + fileName); + } +}); + +p.then(function() {console.log("Import Complete!")}).error(function(error) {console.log("Ran into an error: \n" + error)}); +// }; \ No newline at end of file diff --git a/importSales.js b/importSales.js new file mode 100644 index 0000000..2297bce --- /dev/null +++ b/importSales.js @@ -0,0 +1,416 @@ + +var fs = require("fs"); +var csv = require('./csv.js'); +var Promise = require('bluebird'); + +var fileName = "./importSales.csv"; + +// module.exports = function(models, softErrors, handler) { +// return new Promise(function(resolve, reject) { +var models = require("./models"); +var softErrors = []; +var p = new Promise(function(resolve, reject) { + //The CSV data as an array of rows (each row is an array). + var data; + //The mapping of model attributes to CSV columns. The oz sizes are arrays of columns since there are multiple. + var map = {}; + + //Database maps so we don't have to constantly query for the same data repeatedly. + var vendorIdMap = {}; + var measureIdMap = {}; + var itemIdMap = {}; + + //Collect the metadata from the first row of the CSV data - make a mapping. + function collectMetadata() { + var DATE = 'date'; + var VENDOR = 'vendor'; + var ITEM = 'item'; + var OZ32 = '32 oz'; + var OZ16 = '16 oz'; + var OZ12 = '12 oz'; + var OZ8 = '8 oz'; + var OZ4 = '4 oz'; + var BAGS = 'bags'; + var EACH = 'each'; + + //Data is an array of arrays. data[0] = array of headers. data[1] = first row of data. + //Iterate over the columns to create a mapping. + for(var i = 0; i < data[0].length; i++) { + var next = data[0][i]; + + if(next && next != '') { + switch(next.toLowerCase()) { + case DATE: + map.date = i; + break; + case VENDOR: + map.vendor = i; + break; + case ITEM: + map.item = i; + break; + case OZ32: + map.oz32 = i; + break; + case OZ16: + map.oz16 = i; + break; + case OZ12: + map.oz12 = i; + break; + case OZ8: + map.oz8 = i; + break; + case OZ4: + map.oz4 = i; + break; + case BAGS: + map.bags = i; + break; + case EACH: + map.each = i; + break; + } + } + } + } + + //Reads a single row of CSV data and adds it to the database. + function readRow(rowIndex) { + if(!data) { + console.log("WTF?"); + //process.exit(1); + reject(new Error("Data is null?!?")); + return; + } + else if(!data[rowIndex]) { + console.log("Empty line?"); + readRow(rowIndex + 1); + return; + } + + var date = data[rowIndex][map.date]; + var vendor = data[rowIndex][map.vendor]; + var item = data[rowIndex][map.item]; + var oz32 = data[rowIndex][map.oz32]; + var oz16 = data[rowIndex][map.oz16]; + var oz12 = data[rowIndex][map.oz12]; + var oz8 = data[rowIndex][map.oz8]; + var oz4 = data[rowIndex][map.oz4]; + var bags = data[rowIndex][map.bags]; + var each = data[rowIndex][map.each]; + + var vendorId = vendorIdMap[vendor]; + var itemId = itemIdMap[item]; + var priceMap = {}; + + var saleQueries = []; + + //TODO: Trim and lowercase any names. + priceMap[2016] = { + oz4: 7, + oz8: 11, + oz12: 13, + oz16: 11, + oz32: 15, + "pickled onion, spicy": {oz16: 12}, + "pickled onion, sweet": {oz16: 12}, + "pickled onion, fennel": {oz16: 12}, + "pickled beets": {oz16: 15}, + "prickly pear margarita mix": {oz16: 15}, + "prickly pear syrup": {oz16: 15}, + "tomato sauce": {oz16: 15}, //TODO: Replace all 'tomato sauce, xxxx' with 'tomato sauce' for price lookup only. + "bone broth": {oz16: 20}, //TODO: Replace all 'bone broth, xxxx' with 'bone broth' for price lookup only. + "fava bean soup": {oz32: 20}, + "pickles, bread butter": {oz16: 11}, + "kimchi": {oz16: 11}, + "kraut - red & white": {oz16: 11}, + + "3 pack": {each: 21}, + "espelette, smoked": {each: 1}, + "leather": {each: 6}, + "leather, apple": {each: 6}, + "leather, grape": {each: 6}, + "leather, peach": {each: 6}, + "leather, persimmon": {each: 6}, + "leather, pumpkin": {each: 6}, + "leather, strawberry": {each: 6}, + "leather, quince": {each: 6}, + "membrillo": {each: 7}, + "membrillo candy": {each: 2.5}, + "spices": {each: 4}, + "spices, basil/coriander": {each: 4}, + "spices, fennel": {each: 4}, + "spices, other": {each: 4}, + "spices, pepper": {each: 4}, + "spices, smoked": {each: 4}, + + "baba ganoush": {bag: 8, oz4: 8}, + "dried, pear": {bag: 6}, //Price?? + "dried, tomatillo": {bag: 6}, //Price?? + "dried, strawberry": {bag: 7.5}, + "dried, tomato": {bag: 7}, + "pesto": {bags: 8}, + "pesto, cilantro jalapeno": {bags: 8}, + "pesto, eggplant": {bags: 8}, + "pesto, mint": {bags: 8}, + "pesto, jalapeno": {bags: 8}, + "jalapeno candy": {bags: 5}, + "tomato, dried": {bag: 7}, //Spelling??? Should be Dried, Tomato + + + //Fresh + "persimmons": {lbs: 5}, + "persimmon, frozen": {lbs: 5}, + "dried, persimmon": {lbs: 5}, + + "eggs": {each: 11} + }; + + priceMap[2015] = { + oz4: 6, + oz8: 10, + oz12: 13, + oz16: 10, + oz32: 15, + "blackberry jam": {oz4: 7, oz8: 12}, + "raspberry jam": {oz4: 7, oz8: 12}, + "blackberry vanilla": {oz4: 7, oz8: 12}, + "bloody mary mix": {oz16: 13}, + "pickled beets": {oz16: 15}, + "tomato juice cocktail": {oz16: 8}, + "fava bean soup": {oz32: 20}, + "kimchi": {oz16: 10}, + "kraut - red & white": {oz16: 10}, + "bread & butter pickles": {oz16: 10}, + "pickled eggs": {oz16: 15}, + + "3 pack": {each: 18}, + "leather": {each: 6}, + "leather, apple": {each: 6}, + "leather, grape": {each: 6}, + "leather, peach": {each: 6}, + "leather, persimmon": {each: 6}, + "leather, pumpkin": {each: 6}, + "leather, strawberry": {each: 6}, + "leather, quince": {each: 6}, + "membrillo": {each: 6}, + "membrillo candy": {each: 2.5}, + "popsicles": {each: 5}, + "spices": {each: 4}, + "spices, basil/coriander": {each: 4}, + "spices, fennel": {each: 4}, + "spices, other": {each: 4}, + "spices, pepper": {each: 4}, + "spices, smoked": {each: 4}, + + "baba ganoush": {bag: 8, oz4: 8}, + "dried, pear": {bag: 6}, //Price?? + "dried, tomatillo": {bag: 6}, //Price?? + "dried, strawberry": {bag: 7.5}, + "dried, tomato": {bag: 7}, + "jalapeno candy": {bag: 5}, + + "eggs": {each: 10}, + }; + + //Log errors, but keep on chugging. + if(vendorId == undefined) { + softErrors.push("ERROR: Unexpected vendor: " + vendor); + } + else if(itemId == undefined) { + softErrors.push("ERROR: Unexpected item: " + item); + } + else { + //Split it into multiple sales entries, one for each measure that has a positive value. + if(oz32 > 0) { + var measureId = measureIdMap['32 oz']; + var price = priceMap[year][item]; + + if(price == undefined) { + price = priceMap[year]['oz32']; + } + + saleQueries.push(models.Sale.create({date: date, vendorId: vendorId, itemId: itemId, measureId: measureId, amount: oz32, price: price})); + } + + if(oz16 > 0) { + var measureId = measureIdMap['16 oz']; + var price = priceMap[year][item]; + + if(price == undefined) { + price = priceMap[year]['oz16']; + } + + saleQueries.push(models.Sale.create({date: date, vendorId: vendorId, itemId: itemId, measureId: measureId, amount: oz32, price: price})); + } + + if(oz12 > 0) { + var measureId = measureIdMap['12 oz']; + var price = priceMap[year][item]; + + if(price == undefined) { + price = priceMap[year]['oz12']; + } + + saleQueries.push(models.Sale.create({date: date, vendorId: vendorId, itemId: itemId, measureId: measureId, amount: oz32, price: price})); + } + + if(oz8 > 0) { + var measureId = measureIdMap['8 oz']; + var price = priceMap[year][item]; + + if(price == undefined) { + price = priceMap[year]['oz8']; + } + + saleQueries.push(models.Sale.create({date: date, vendorId: vendorId, itemId: itemId, measureId: measureId, amount: oz32, price: price})); + } + + if(oz4 > 0) { + var measureId = measureIdMap['4 oz']; + var price = priceMap[year][item]; + + if(price == undefined) { + price = priceMap[year]['oz4']; + } + + saleQueries.push(models.Sale.create({date: date, vendorId: vendorId, itemId: itemId, measureId: measureId, amount: oz32, price: price})); + } + + if(bags > 0) { + var measureId = measureIdMap['bags']; + var price = priceMap[year][item]; + + if(price == undefined) { + price = priceMap[year]['bags']; + } + + saleQueries.push(models.Sale.create({date: date, vendorId: vendorId, itemId: itemId, measureId: measureId, amount: oz32, price: price})); + } + + if(each > 0) { + var measureId = measureIdMap['each']; + var price = priceMap[year][item]; + + if(price == undefined) { + price = priceMap[year]['each']; + } + + saleQueries.push(models.Sale.create({date: date, vendorId: vendorId, itemId: itemId, measureId: measureId, amount: oz32, price: price})); + } + + Promise.each(saleQueries, function(value, index, length) { + //Ignored. + }).then(function(result) { + //Recursively read rows until there are no more rows to read. + if(rowIndex + 1 < data.length) readRow(rowIndex + 1); + else resolve(); + }); + } + } + + if(fs.existsSync(fileName)) { + csv.read(fileName, function(error, csvData) { + if(error) console.log(error); + else { + data = csvData; + //Collect the mapping data. + collectMetadata(); + + //Read or create the required models to make this all work: + Promise.each([ + //Measures 1-6 + models.Measure.find({where: {name: 'Jar 32oz'}}), + models.Measure.find({where: {name: 'Jar 16oz'}}), + models.Measure.find({where: {name: 'Jar 12oz'}}), + models.Measure.find({where: {name: 'Jar 8oz'}}), + models.Measure.find({where: {name: 'Jar 4oz'}}), + models.Measure.find({where: {name: 'Each'}}), + models.Measure.find({where: {name: 'Bags'}}), + //Vendors 7-16 + models.Venue.find({where: {name: 'Boonville'}}), + models.Venue.find({where: {name: 'Clement St'}}), + models.Venue.find({where: {name: 'Ukiah'}}), + models.Venue.find({where: {name: 'Mendocino'}}), + models.Venue.find({where: {name: 'Ft Bragg'}}), + models.Venue.find({where: {name: 'On Farm'}}), + models.Venue.find({where: {name: 'Unknown Restaurant'}}), + models.Venue.find({where: {name: 'Yorkville Market'}}), + models.Venue.find({where: {name: 'Yorkville Cellars'}}), + models.Venue.find({where: {name: 'Mail Order'}}), + //Items + models.Item.findAll() + ], function(value, index, length) { + switch(index) { + case 0: + measureIdMap['32 oz'] = value.id; + break; + case 1: + measureIdMap['16 oz'] = value.id; + break; + case 2: + measureIdMap['12 oz'] = value.id; + break; + case 3: + measureIdMap['8 oz'] = value.id; + break; + case 4: + measureIdMap['4 oz'] = value.id; + break; + case 5: + measureIdMap['each'] = value.id; + break; + case 6: + measureIdMap['bags'] = value.id; + break; + case 7: + vendorIdMap['bv'] = value.id; + break; + case 8: + vendorIdMap['sf'] = value.id; + break; + case 9: + vendorIdMap['uk'] = value.id; + break; + case 10: + vendorIdMap['men'] = value.id; + break; + case 11: + vendorIdMap['fb'] = value.id; + break; + case 12: + vendorIdMap['of'] = value.id; + break; + case 13: + vendorIdMap['res'] = value.id; + break; + case 14: + vendorIdMap['ym'] = value.id; + break; + case 15: + vendorIdMap['yc'] = value.id; + break; + case 16: + vendorIdMap['mo'] = value.id; + break; + case 17: + //Iterate over all the items and setup the itemIdMap to reference the item id given the item name. Item names are unique. + for(var i = 0; i < value.length; i++) { + itemIdMap[value[i].name] = value[i].id; + } + } + }).then(function() { + //Read the first row. + readRow(1); + }); + } + }); + } + else { + console.log("Cannot find " + fileName); + } +}); + +p.then(function() {console.log("Import Complete!")}).error(function(error) {console.log("Ran into an error: \n" + error)}); +// }; \ No newline at end of file diff --git a/migrations/20160602030143-User.js b/migrations/20160602030143-User.js index 1018892..3ded8fa 100644 --- a/migrations/20160602030143-User.js +++ b/migrations/20160602030143-User.js @@ -12,7 +12,8 @@ module.exports = { autoIncrement: true }, login: { - type: DataTypes.STRING + type: DataTypes.STRING, + unique: true }, password: { type: DataTypes.STRING diff --git a/migrations/20160602030145-Measure.js b/migrations/20160602030145-Measure.js index 24bc0e6..c27c77a 100644 --- a/migrations/20160602030145-Measure.js +++ b/migrations/20160602030145-Measure.js @@ -13,7 +13,8 @@ module.exports = { }, name: { type: DataTypes.STRING, - allowNull: false + allowNull: false, + unique: true }, image: { type: DataTypes.STRING, diff --git a/migrations/20160602030146-Venue.js b/migrations/20160602030146-VenueType.js similarity index 83% rename from migrations/20160602030146-Venue.js rename to migrations/20160602030146-VenueType.js index efb0af2..8507b83 100644 --- a/migrations/20160602030146-Venue.js +++ b/migrations/20160602030146-VenueType.js @@ -4,7 +4,7 @@ module.exports = { up: function (query, Sequelize) { var DataTypes = Sequelize; //Allow for more cut and paste :) - return query.createTable('Venue', { + return query.createTable('VenueType', { id: { type: DataTypes.INTEGER, primaryKey: true, @@ -13,7 +13,8 @@ module.exports = { }, name: { type: DataTypes.STRING, - allowNull: false + allowNull: false, + unique: true }, createdAt: { type: DataTypes.DATE, @@ -35,6 +36,6 @@ module.exports = { }, down: function (query, Sequelize) { - return query.dropTable('Venue'); + return query.dropTable('VenueType'); } }; diff --git a/migrations/20160602030147-Venue.js b/migrations/20160602030147-Venue.js new file mode 100644 index 0000000..8cca8a2 --- /dev/null +++ b/migrations/20160602030147-Venue.js @@ -0,0 +1,49 @@ +'use strict'; + +module.exports = { + up: function (query, Sequelize) { + var DataTypes = Sequelize; //Allow for more cut and paste :) + + return query.createTable('Venue', { + id: { + type: DataTypes.INTEGER, + primaryKey: true, + allowNull: false, + autoIncrement: true + }, + name: { + type: DataTypes.STRING, + allowNull: false + }, + typeId: { + type: Sequelize.INTEGER, + references: { + model: 'VenueType', + key: 'id' + }, + onUpdate: 'cascade', + onDelete: 'cascade' + }, + createdAt: { + type: DataTypes.DATE, + allowNull: false + }, + updatedAt: { + type: DataTypes.DATE, + allowNull: false + }, + deletedAt: { + type: DataTypes.DATE, + allowNull: true + } + }, { + charset: 'utf8', + timestamps: true, + paranoid: true + }); + }, + + down: function (query, Sequelize) { + return query.dropTable('Venue'); + } +}; diff --git a/migrations/20160602030147-Category.js b/migrations/20160602030148-Category.js similarity index 94% rename from migrations/20160602030147-Category.js rename to migrations/20160602030148-Category.js index f33b1e5..918b83f 100644 --- a/migrations/20160602030147-Category.js +++ b/migrations/20160602030148-Category.js @@ -13,7 +13,8 @@ module.exports = { }, name: { type: DataTypes.STRING, - allowNull: false + allowNull: false, + unique: true }, createdAt: { type: DataTypes.DATE, diff --git a/migrations/20160602030148-Subcategory.js b/migrations/20160602030149-Subcategory.js similarity index 95% rename from migrations/20160602030148-Subcategory.js rename to migrations/20160602030149-Subcategory.js index 65a9ff2..dba0d6a 100644 --- a/migrations/20160602030148-Subcategory.js +++ b/migrations/20160602030149-Subcategory.js @@ -15,7 +15,8 @@ module.exports = { name: { type: DataTypes.STRING, field: 'name', - allowNull: false + allowNull: false, + unique: true }, categoryId: { type: Sequelize.INTEGER, diff --git a/migrations/20160602030149-Item.js b/migrations/20160602030150-Item.js similarity index 88% rename from migrations/20160602030149-Item.js rename to migrations/20160602030150-Item.js index dbcb38b..46bfc65 100644 --- a/migrations/20160602030149-Item.js +++ b/migrations/20160602030150-Item.js @@ -13,7 +13,8 @@ module.exports = { }, name: { type: DataTypes.STRING, - allowNull: false + allowNull: false, + unique: true }, measures: { type: DataTypes.JSON, @@ -23,9 +24,10 @@ module.exports = { type: DataTypes.JSON, allowNull: true }, - defaultPrice: { - type: DataTypes.DECIMAL(9,2), - allowNull: false + prices: { + type: DataTypes.JSON, + //type: DataTypes.DECIMAL(9,2), + allowNull: true }, subcategoryId: { type: Sequelize.INTEGER, diff --git a/migrations/20160602030150-Sale.js b/migrations/20160602030151-Sale.js similarity index 100% rename from migrations/20160602030150-Sale.js rename to migrations/20160602030151-Sale.js diff --git a/models/item.js b/models/item.js index 4019eb1..3230ad3 100644 --- a/models/item.js +++ b/models/item.js @@ -22,9 +22,10 @@ module.exports = function(sequelize, DataTypes) { type: DataTypes.JSON, allowNull: true }, - defaultPrice: { - type: DataTypes.DECIMAL(9,2), - allowNull: false + prices: { + type: DataTypes.JSON, + //type: DataTypes.DECIMAL(9,2), + allowNull: true }, createdAt: { type: DataTypes.DATE, diff --git a/models/venue.js b/models/venue.js index 8e76fd9..15fbd91 100644 --- a/models/venue.js +++ b/models/venue.js @@ -11,7 +11,7 @@ module.exports = function(sequelize, DataTypes) { }, name: { type: DataTypes.STRING, - allowNull: false + allowNull: false }, createdAt: { type: DataTypes.DATE, @@ -27,6 +27,11 @@ module.exports = function(sequelize, DataTypes) { } }, { freezeTableName: true, // Model tableName will be the same as the model name, - paranoid: true + paranoid: true, + classMethods: { + associate: function(models) { + models.Venue.belongsTo(models.VenueType, {as: 'type', foreignKey: {name: 'typeId', field: 'typeId'}}); + } + } }); }; diff --git a/models/venueType.js b/models/venueType.js new file mode 100644 index 0000000..4e12cab --- /dev/null +++ b/models/venueType.js @@ -0,0 +1,32 @@ +"use strict"; + +module.exports = function(sequelize, DataTypes) { + //The id field is auto added and made primary key. + return sequelize.define('VenueType', { + id: { + type: DataTypes.INTEGER, + primaryKey: true, + allowNull: false, + autoIncrement: true + }, + name: { + type: DataTypes.STRING, + allowNull: false + }, + createdAt: { + type: DataTypes.DATE, + allowNull: false + }, + updatedAt: { + type: DataTypes.DATE, + allowNull: false + }, + deletedAt: { + type: DataTypes.DATE, + allowNull: true + } + }, { + freezeTableName: true, // Model tableName will be the same as the model name, + paranoid: true + }); +}; diff --git a/package.json b/package.json index e05ad6e..fc2b726 100644 --- a/package.json +++ b/package.json @@ -11,7 +11,10 @@ "start": "node server.js", "migrateDB": "sequelize db:migrate", "dropDB": "sudo -u postgres dropdb PetitTeton", - "createDB": "sudo -u postgres createdb PetitTeton" + "createDB": "sudo -u postgres createdb PetitTeton", + "import1": "node importAll.js", + "import2": "node importItems.js", + "import3": "node %NODE_DEBUG_OPTION% importSales.js" }, "dependencies": { "bcrypt-nodejs": "^0.0.3", @@ -38,6 +41,7 @@ "serve-favicon": "~2.2.0", "session-file-store": "~0.0.24", "stylus": "~0.42.3", - "swig": "~1.4.2" + "swig": "~1.4.2", + "csv-parse": "latest" } } diff --git a/public/admin/Venues.html b/public/admin/Venues.html index e673010..9ed5f6c 100644 --- a/public/admin/Venues.html +++ b/public/admin/Venues.html @@ -15,7 +15,7 @@ -
| Name | @@ -99,7 +99,8 @@ $(function() { } }; - var dataTable = new LinkedTable($page.find('#dataTable'), { + var $table = $page.find('#dataTable'); + $table.buildTable({ url: "data/Venue/readAll", attr: "data-key-name", selection: "row", @@ -122,16 +123,19 @@ $(function() { } } }); + var dataTable = $table.getTable(); + $page.find('#dataTable').on('dblclick', 'tr', function(event) { if(dataTable.getSelectedRow()) { location.hash = "#!/venues-edit"; } }); - $page.find('#dataTable').on('keyup', 'table', function(event) { + + $page.find('#page').on('keyup', '.page', function(event) { switch(event.keyCode || event.which) { case 0x0D: //Enter if(dataTable.getSelectedRow()) { - $btnEdit.click(); + location.hash = '#!/venues-edit'; } break; case 0x08: //Backspace @@ -144,9 +148,6 @@ $(function() { } }); - //Call the refresh user table function once initially. - dataTable.build(); - //Refresh the data table if the user toggles the button to show/hide deleted elements. $page.find('#includeDeletedToggle').on('click', function(event) { dataTable.refresh(); @@ -247,6 +248,16 @@ $(function() { }); }); + //Close the view if the user uses the escape key and it isn't handled at the widget level. + $editorView.on('keyup', function(event) { + switch(event.keyCode) { + case 27: + if(!event.isDefaultPrevented()) + history.back(); + break; + } + }); + //++++++++++++++++++++++++++++++++++++++ // ++++++++++ Delete View +++++++++++ diff --git a/public/admin/bootstrap.styl b/public/admin/bootstrap.styl index b42e4a7..6c1aa6b 100644 --- a/public/admin/bootstrap.styl +++ b/public/admin/bootstrap.styl @@ -13,7 +13,8 @@ input[type="url"]:focus, input[type="search"]:focus, input[type="tel"]:focus, input[type="color"]:focus, -.uneditable-input:focus { +.uneditable-input:focus, +.list-group:focus { border-color: rgba(82, 168, 236, 0.8); outline: 0; outline: thin dotted \9; diff --git a/public/admin/categories.html b/public/admin/categories.html index 1c6cdde..12cca15 100644 --- a/public/admin/categories.html +++ b/public/admin/categories.html @@ -99,7 +99,8 @@ $(function() { } }; - var dataTable = new LinkedTable($page.find('#dataTable'), { + var $table = $page.find('#dataTable'); + $table.buildTable({ url: "data/Category/readAll", attr: "data-key-name", selection: "row", @@ -122,15 +123,14 @@ $(function() { } } }); + var dataTable = $table.getTable(); + $page.find('#dataTable').on('dblclick', 'tr', function(event) { if(dataTable.getSelectedRow()) { location.hash = "#!/categories-edit"; } }); - //Call the refresh user table function once initially. - dataTable.build(); - //Refresh the data table if the user toggles the button to show/hide deleted elements. $page.find('#includeDeletedToggle').on('click', function(event) { dataTable.refresh(); @@ -231,6 +231,16 @@ $(function() { }); }); + //Close the view if the user uses the escape key and it isn't handled at the widget level. + $editorView.on('keyup', function(event) { + switch(event.keyCode) { + case 27: + if(!event.isDefaultPrevented()) + history.back(); + break; + } + }); + //++++++++++++++++++++++++++++++++++++++ // ++++++++++ Delete View +++++++++++ diff --git a/public/admin/css/jsgrid-theme.css b/public/admin/css/jsgrid-theme.css new file mode 100644 index 0000000..0c99286 --- /dev/null +++ b/public/admin/css/jsgrid-theme.css @@ -0,0 +1,258 @@ +/* + * jsGrid v1.5.2 (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/public/admin/css/jsgrid-theme.min.css b/public/admin/css/jsgrid-theme.min.css new file mode 100644 index 0000000..f09f472 --- /dev/null +++ b/public/admin/css/jsgrid-theme.min.css @@ -0,0 +1,7 @@ +/* + * jsGrid v1.5.2 (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/public/admin/css/jsgrid.css b/public/admin/css/jsgrid.css new file mode 100644 index 0000000..56128dd --- /dev/null +++ b/public/admin/css/jsgrid.css @@ -0,0 +1,126 @@ +/* + * jsGrid v1.5.2 (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/public/admin/css/jsgrid.min.css b/public/admin/css/jsgrid.min.css new file mode 100644 index 0000000..2a218ac --- /dev/null +++ b/public/admin/css/jsgrid.min.css @@ -0,0 +1,7 @@ +/* + * jsGrid v1.5.2 (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 :after,.jsgrid :before{box-sizing:border-box}.jsgrid input,.jsgrid select,.jsgrid textarea{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:.5em}.jsgrid-header-cell,.jsgrid-сell{box-sizing:border-box}.jsgrid-align-left{text-align:left}.jsgrid-align-center,.jsgrid-align-center input,.jsgrid-align-center select,.jsgrid-align-center textarea{text-align:center}.jsgrid-align-right,.jsgrid-align-right input,.jsgrid-align-right select,.jsgrid-align-right textarea{text-align:right}.jsgrid-header-cell{padding:.5em}.jsgrid-edit-row input,.jsgrid-edit-row select,.jsgrid-edit-row textarea,.jsgrid-filter-row input,.jsgrid-filter-row select,.jsgrid-filter-row textarea,.jsgrid-insert-row input,.jsgrid-insert-row select,.jsgrid-insert-row textarea{width:100%;padding:.3em .5em}.jsgrid-edit-row input[type=checkbox],.jsgrid-filter-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} \ No newline at end of file diff --git a/public/admin/index.ejs b/public/admin/index.ejs index aba9b33..2fd3b3b 100644 --- a/public/admin/index.ejs +++ b/public/admin/index.ejs @@ -26,10 +26,12 @@ - + + + - - + + @@ -38,15 +40,16 @@ - - + + + \ No newline at end of file diff --git a/public/admin/prices.styl b/public/admin/prices.styl new file mode 100644 index 0000000..c104a77 --- /dev/null +++ b/public/admin/prices.styl @@ -0,0 +1,3 @@ +#prices { + +} \ No newline at end of file diff --git a/public/admin/sales.html b/public/admin/sales.html index b03fceb..57e61bc 100644 --- a/public/admin/sales.html +++ b/public/admin/sales.html @@ -35,8 +35,8 @@
|---|