frontend-generators 0.0.6 → 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (46) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +44 -0
  3. data/assets/clean_blog/app/assets/images/.keep +0 -0
  4. data/assets/clean_blog/app/assets/images/clean_blog/about-bg.jpg +0 -0
  5. data/assets/clean_blog/app/assets/images/clean_blog/contact-bg.jpg +0 -0
  6. data/assets/clean_blog/app/assets/images/clean_blog/home-bg.jpg +0 -0
  7. data/assets/clean_blog/app/assets/images/clean_blog/post-bg.jpg +0 -0
  8. data/assets/clean_blog/app/assets/images/clean_blog/post-sample-image.jpg +0 -0
  9. data/assets/clean_blog/app/assets/javascripts/clean_blog/clean-blog.js +1032 -0
  10. data/assets/clean_blog/app/assets/javascripts/clean_blog/manifest.js +6 -0
  11. data/assets/clean_blog/app/assets/stylesheets/clean_blog/clean-blog.css.scss +396 -0
  12. data/assets/clean_blog/app/assets/stylesheets/clean_blog/manifest.css +5 -0
  13. data/assets/clean_blog/app/controllers/clean_blogs_controller.rb +15 -0
  14. data/assets/clean_blog/app/helpers/clean_blogs_helper.rb +11 -0
  15. data/assets/clean_blog/app/views/clean_blogs/_footer.html.erb +35 -0
  16. data/assets/clean_blog/app/views/clean_blogs/_header.html.erb +61 -0
  17. data/assets/clean_blog/app/views/clean_blogs/_navigation.html.erb +34 -0
  18. data/assets/clean_blog/app/views/clean_blogs/_shim.html.erb +6 -0
  19. data/assets/clean_blog/app/views/clean_blogs/about.html.erb +20 -0
  20. data/assets/clean_blog/app/views/clean_blogs/contact.html.erb +58 -0
  21. data/assets/clean_blog/app/views/clean_blogs/index.html.erb +68 -0
  22. data/assets/clean_blog/app/views/clean_blogs/post.html.erb +52 -0
  23. data/assets/clean_blog/app/views/layouts/clean_blog.html.erb +16 -0
  24. data/assets/grayscale/app/assets/images/.keep +0 -0
  25. data/assets/grayscale/app/assets/images/grayscale/downloads-bg.jpg +0 -0
  26. data/assets/grayscale/app/assets/images/grayscale/intro-bg.jpg +0 -0
  27. data/assets/grayscale/app/assets/images/grayscale/map-marker.png +0 -0
  28. data/assets/grayscale/app/assets/javascripts/grayscale/grayscale.js +178 -0
  29. data/assets/grayscale/app/assets/javascripts/grayscale/jquery.easing.min.js +44 -0
  30. data/assets/grayscale/app/assets/javascripts/grayscale/manifest.js +7 -0
  31. data/assets/grayscale/app/assets/stylesheets/grayscale/grayscale.css +373 -0
  32. data/assets/grayscale/app/assets/stylesheets/grayscale/manifest.css +7 -0
  33. data/assets/grayscale/app/controllers/grayscales_controller.rb +5 -0
  34. data/assets/grayscale/app/views/grayscales/_about.html.erb +10 -0
  35. data/assets/grayscale/app/views/grayscales/_contact.html.erb +21 -0
  36. data/assets/grayscale/app/views/grayscales/_download.html.erb +11 -0
  37. data/assets/grayscale/app/views/grayscales/_footer.html.erb +5 -0
  38. data/assets/grayscale/app/views/grayscales/_header.html.erb +15 -0
  39. data/assets/grayscale/app/views/grayscales/_map.html.erb +8 -0
  40. data/assets/grayscale/app/views/grayscales/_navigation.html.erb +32 -0
  41. data/assets/grayscale/app/views/grayscales/_shim.html.erb +6 -0
  42. data/assets/grayscale/app/views/grayscales/index.html.erb +8 -0
  43. data/assets/grayscale/app/views/layouts/grayscale.html.erb +22 -0
  44. data/lib/frontend_generators/version.rb +1 -1
  45. data/lib/tasks/add_assets.rake +1 -1
  46. metadata +43 -2
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: b0bbd17d6eab1d485e134201f6d99cf732675c90
4
- data.tar.gz: 04d5f8f0fe8bd80e30ab329662a98df8e219d5ee
3
+ metadata.gz: fbbcad6a2c664b81693b4d4cf8b24848f5079855
4
+ data.tar.gz: 8de5f19248369b9bba97a686c747e156f4ae9d3e
5
5
  SHA512:
6
- metadata.gz: 61c856ea22debf30fd5f47f013921bb48fbe660782d767264592891fef686cffc3468a37809c236056e493b69950b713b5447dc798822a6c761fa4ccda9036a2
7
- data.tar.gz: c82f4f1e109e536fbfed9df9a2db4dbc8cb745845418cb0b392bb981321b0f6564e02ab63c11a361e8087124fba085de7377a64ec2e33b34829f58784892ea32
6
+ metadata.gz: a0f392f275d53065b33384d25e04fb837f6611aae0fa48fd1b295fbcf766f54449c937579ce0013ca1571ff34875a460d259e77423150bc5a510245c5564941f
7
+ data.tar.gz: fdb9f3d1c3eff961023c124f5057fad7c68a538507b3addbbd40a55740312d4b9c49cee8d5843843e7d6c6b449115665b6503e71720e6f5e1ed471dab8170b73
data/README.md CHANGED
@@ -4,6 +4,8 @@ This gem can be used to easily set up a [Start Bootstrap](http://startbootstrap.
4
4
 
5
5
  The FrontendGenerators gem also lets you easily copy the Bootstrap & Font Awesome css, js, and fonts directly into your application. Most people use [twitter-bootstrap-rails](https://github.com/seyhunak/twitter-bootstrap-rails) and [font-awesome-rails](https://github.com/bokmann/font-awesome-rails) to access these frontend libraries in their code, but I find it much easier to just have the code in my own applications. It's easier to examine the source code when it's in your application and you can avoid frusterating Rails asset pipeline bugs with this straightforward setup. Annoying bugs are what motivated me to create this gem in the first place.
6
6
 
7
+ [This video](https://www.youtube.com/watch?v=5vuF0n4Qlxk) shows how to use the gem.
8
+
7
9
  ## Installation
8
10
 
9
11
  Add this line to your application's Gemfile:
@@ -119,6 +121,48 @@ Add this line to `config/initializers/assets.rb`:
119
121
  Rails.application.config.assets.precompile += %w( freelancer/manifest.js freelancer/manifest.css )
120
122
  ```
121
123
 
124
+ ## [Start Bootstrap Clean Blog](http://startbootstrap.com/template-overviews/clean-blog/)
125
+
126
+ To move the Start Bootstrap Clean Blog theme into your Rails application, run this rake task:
127
+
128
+ ```ruby
129
+ bundle exec rake add_assets:clean_blog
130
+ ```
131
+
132
+ Add the routes to `routes.rb`:
133
+
134
+ ```ruby
135
+ root 'clean_blogs#index'
136
+ get 'about' => 'clean_blogs#about'
137
+ get 'post' => 'clean_blogs#post'
138
+ get 'contact' => 'clean_blogs#contact'
139
+ ```
140
+
141
+ Add this line to `config/initializers/assets.rb`:
142
+
143
+ ```ruby
144
+ Rails.application.config.assets.precompile += %w( clean_blog/manifest.js clean_blog/manifest.css )
145
+ ```
146
+
147
+ ## [Start Bootstrap Grayscale](http://startbootstrap.com/template-overviews/grayscale/)
148
+
149
+ To move the Start Bootstrap Grayscale theme into your Rails application, run this rake task:
150
+
151
+ ```ruby
152
+ bundle exec rake add_assets:grayscale
153
+ ```
154
+
155
+ Add the route to `routes.rb`:
156
+
157
+ ```ruby
158
+ get 'grayscales/index'
159
+ ```
160
+
161
+ Add this line to `config/initializers/assets.rb`:
162
+
163
+ ```ruby
164
+ Rails.application.config.assets.precompile += %w( grayscale/manifest.js grayscale/manifest.css )
165
+ ```
122
166
 
123
167
  ## Contributing
124
168
 
File without changes
@@ -0,0 +1,1032 @@
1
+ /*!
2
+ * Clean Blog v1.0.0 (http://startbootstrap.com)
3
+ * Copyright 2014 Start Bootstrap
4
+ * Licensed under Apache 2.0 (https://github.com/IronSummitMedia/startbootstrap/blob/gh-pages/LICENSE)
5
+ */
6
+
7
+ // Contact Form Scripts
8
+
9
+ $(function() {
10
+
11
+ $("input,textarea").jqBootstrapValidation({
12
+ preventSubmit: true,
13
+ submitError: function($form, event, errors) {
14
+ // additional error messages or events
15
+ },
16
+ submitSuccess: function($form, event) {
17
+ event.preventDefault(); // prevent default submit behaviour
18
+ // get values from FORM
19
+ var name = $("input#name").val();
20
+ var email = $("input#email").val();
21
+ var phone = $("input#phone").val();
22
+ var message = $("textarea#message").val();
23
+ var firstName = name; // For Success/Failure Message
24
+ // Check for white space in name for Success/Fail message
25
+ if (firstName.indexOf(' ') >= 0) {
26
+ firstName = name.split(' ').slice(0, -1).join(' ');
27
+ }
28
+ $.ajax({
29
+ url: "././mail/contact_me.php",
30
+ type: "POST",
31
+ data: {
32
+ name: name,
33
+ phone: phone,
34
+ email: email,
35
+ message: message
36
+ },
37
+ cache: false,
38
+ success: function() {
39
+ // Success message
40
+ $('#success').html("<div class='alert alert-success'>");
41
+ $('#success > .alert-success').html("<button type='button' class='close' data-dismiss='alert' aria-hidden='true'>&times;")
42
+ .append("</button>");
43
+ $('#success > .alert-success')
44
+ .append("<strong>Your message has been sent. </strong>");
45
+ $('#success > .alert-success')
46
+ .append('</div>');
47
+
48
+ //clear all fields
49
+ $('#contactForm').trigger("reset");
50
+ },
51
+ error: function() {
52
+ // Fail message
53
+ $('#success').html("<div class='alert alert-danger'>");
54
+ $('#success > .alert-danger').html("<button type='button' class='close' data-dismiss='alert' aria-hidden='true'>&times;")
55
+ .append("</button>");
56
+ $('#success > .alert-danger').append("<strong>Sorry " + firstName + ", it seems that my mail server is not responding. Please try again later!");
57
+ $('#success > .alert-danger').append('</div>');
58
+ //clear all fields
59
+ $('#contactForm').trigger("reset");
60
+ },
61
+ })
62
+ },
63
+ filter: function() {
64
+ return $(this).is(":visible");
65
+ },
66
+ });
67
+
68
+ $("a[data-toggle=\"tab\"]").click(function(e) {
69
+ e.preventDefault();
70
+ $(this).tab("show");
71
+ });
72
+ });
73
+
74
+
75
+ /*When clicking on Full hide fail/success boxes */
76
+ $('#name').focus(function() {
77
+ $('#success').html('');
78
+ });
79
+
80
+ // jqBootstrapValidation
81
+ // * A plugin for automating validation on Twitter Bootstrap formatted forms.
82
+ // *
83
+ // * v1.3.6
84
+ // *
85
+ // * License: MIT <http://opensource.org/licenses/mit-license.php> - see LICENSE file
86
+ // *
87
+ // * http://ReactiveRaven.github.com/jqBootstrapValidation/
88
+
89
+
90
+ (function( $ ){
91
+
92
+ var createdElements = [];
93
+
94
+ var defaults = {
95
+ options: {
96
+ prependExistingHelpBlock: false,
97
+ sniffHtml: true, // sniff for 'required', 'maxlength', etc
98
+ preventSubmit: true, // stop the form submit event from firing if validation fails
99
+ submitError: false, // function called if there is an error when trying to submit
100
+ submitSuccess: false, // function called just before a successful submit event is sent to the server
101
+ semanticallyStrict: false, // set to true to tidy up generated HTML output
102
+ autoAdd: {
103
+ helpBlocks: true
104
+ },
105
+ filter: function () {
106
+ // return $(this).is(":visible"); // only validate elements you can see
107
+ return true; // validate everything
108
+ }
109
+ },
110
+ methods: {
111
+ init : function( options ) {
112
+
113
+ var settings = $.extend(true, {}, defaults);
114
+
115
+ settings.options = $.extend(true, settings.options, options);
116
+
117
+ var $siblingElements = this;
118
+
119
+ var uniqueForms = $.unique(
120
+ $siblingElements.map( function () {
121
+ return $(this).parents("form")[0];
122
+ }).toArray()
123
+ );
124
+
125
+ $(uniqueForms).bind("submit", function (e) {
126
+ var $form = $(this);
127
+ var warningsFound = 0;
128
+ var $inputs = $form.find("input,textarea,select").not("[type=submit],[type=image]").filter(settings.options.filter);
129
+ $inputs.trigger("submit.validation").trigger("validationLostFocus.validation");
130
+
131
+ $inputs.each(function (i, el) {
132
+ var $this = $(el),
133
+ $controlGroup = $this.parents(".form-group").first();
134
+ if (
135
+ $controlGroup.hasClass("warning")
136
+ ) {
137
+ $controlGroup.removeClass("warning").addClass("error");
138
+ warningsFound++;
139
+ }
140
+ });
141
+
142
+ $inputs.trigger("validationLostFocus.validation");
143
+
144
+ if (warningsFound) {
145
+ if (settings.options.preventSubmit) {
146
+ e.preventDefault();
147
+ }
148
+ $form.addClass("error");
149
+ if ($.isFunction(settings.options.submitError)) {
150
+ settings.options.submitError($form, e, $inputs.jqBootstrapValidation("collectErrors", true));
151
+ }
152
+ } else {
153
+ $form.removeClass("error");
154
+ if ($.isFunction(settings.options.submitSuccess)) {
155
+ settings.options.submitSuccess($form, e);
156
+ }
157
+ }
158
+ });
159
+
160
+ return this.each(function(){
161
+
162
+ // Get references to everything we're interested in
163
+ var $this = $(this),
164
+ $controlGroup = $this.parents(".form-group").first(),
165
+ $helpBlock = $controlGroup.find(".help-block").first(),
166
+ $form = $this.parents("form").first(),
167
+ validatorNames = [];
168
+
169
+ // create message container if not exists
170
+ if (!$helpBlock.length && settings.options.autoAdd && settings.options.autoAdd.helpBlocks) {
171
+ $helpBlock = $('<div class="help-block" />');
172
+ $controlGroup.find('.controls').append($helpBlock);
173
+ createdElements.push($helpBlock[0]);
174
+ }
175
+
176
+ // =============================================================
177
+ // SNIFF HTML FOR VALIDATORS
178
+ // =============================================================
179
+
180
+ // *snort sniff snuffle*
181
+
182
+ if (settings.options.sniffHtml) {
183
+ var message = "";
184
+ // ---------------------------------------------------------
185
+ // PATTERN
186
+ // ---------------------------------------------------------
187
+ if ($this.attr("pattern") !== undefined) {
188
+ message = "Not in the expected format<!-- data-validation-pattern-message to override -->";
189
+ if ($this.data("validationPatternMessage")) {
190
+ message = $this.data("validationPatternMessage");
191
+ }
192
+ $this.data("validationPatternMessage", message);
193
+ $this.data("validationPatternRegex", $this.attr("pattern"));
194
+ }
195
+ // ---------------------------------------------------------
196
+ // MAX
197
+ // ---------------------------------------------------------
198
+ if ($this.attr("max") !== undefined || $this.attr("aria-valuemax") !== undefined) {
199
+ var max = ($this.attr("max") !== undefined ? $this.attr("max") : $this.attr("aria-valuemax"));
200
+ message = "Too high: Maximum of '" + max + "'<!-- data-validation-max-message to override -->";
201
+ if ($this.data("validationMaxMessage")) {
202
+ message = $this.data("validationMaxMessage");
203
+ }
204
+ $this.data("validationMaxMessage", message);
205
+ $this.data("validationMaxMax", max);
206
+ }
207
+ // ---------------------------------------------------------
208
+ // MIN
209
+ // ---------------------------------------------------------
210
+ if ($this.attr("min") !== undefined || $this.attr("aria-valuemin") !== undefined) {
211
+ var min = ($this.attr("min") !== undefined ? $this.attr("min") : $this.attr("aria-valuemin"));
212
+ message = "Too low: Minimum of '" + min + "'<!-- data-validation-min-message to override -->";
213
+ if ($this.data("validationMinMessage")) {
214
+ message = $this.data("validationMinMessage");
215
+ }
216
+ $this.data("validationMinMessage", message);
217
+ $this.data("validationMinMin", min);
218
+ }
219
+ // ---------------------------------------------------------
220
+ // MAXLENGTH
221
+ // ---------------------------------------------------------
222
+ if ($this.attr("maxlength") !== undefined) {
223
+ message = "Too long: Maximum of '" + $this.attr("maxlength") + "' characters<!-- data-validation-maxlength-message to override -->";
224
+ if ($this.data("validationMaxlengthMessage")) {
225
+ message = $this.data("validationMaxlengthMessage");
226
+ }
227
+ $this.data("validationMaxlengthMessage", message);
228
+ $this.data("validationMaxlengthMaxlength", $this.attr("maxlength"));
229
+ }
230
+ // ---------------------------------------------------------
231
+ // MINLENGTH
232
+ // ---------------------------------------------------------
233
+ if ($this.attr("minlength") !== undefined) {
234
+ message = "Too short: Minimum of '" + $this.attr("minlength") + "' characters<!-- data-validation-minlength-message to override -->";
235
+ if ($this.data("validationMinlengthMessage")) {
236
+ message = $this.data("validationMinlengthMessage");
237
+ }
238
+ $this.data("validationMinlengthMessage", message);
239
+ $this.data("validationMinlengthMinlength", $this.attr("minlength"));
240
+ }
241
+ // ---------------------------------------------------------
242
+ // REQUIRED
243
+ // ---------------------------------------------------------
244
+ if ($this.attr("required") !== undefined || $this.attr("aria-required") !== undefined) {
245
+ message = settings.builtInValidators.required.message;
246
+ if ($this.data("validationRequiredMessage")) {
247
+ message = $this.data("validationRequiredMessage");
248
+ }
249
+ $this.data("validationRequiredMessage", message);
250
+ }
251
+ // ---------------------------------------------------------
252
+ // NUMBER
253
+ // ---------------------------------------------------------
254
+ if ($this.attr("type") !== undefined && $this.attr("type").toLowerCase() === "number") {
255
+ message = settings.builtInValidators.number.message;
256
+ if ($this.data("validationNumberMessage")) {
257
+ message = $this.data("validationNumberMessage");
258
+ }
259
+ $this.data("validationNumberMessage", message);
260
+ }
261
+ // ---------------------------------------------------------
262
+ // EMAIL
263
+ // ---------------------------------------------------------
264
+ if ($this.attr("type") !== undefined && $this.attr("type").toLowerCase() === "email") {
265
+ message = "Not a valid email address<!-- data-validator-validemail-message to override -->";
266
+ if ($this.data("validationValidemailMessage")) {
267
+ message = $this.data("validationValidemailMessage");
268
+ } else if ($this.data("validationEmailMessage")) {
269
+ message = $this.data("validationEmailMessage");
270
+ }
271
+ $this.data("validationValidemailMessage", message);
272
+ }
273
+ // ---------------------------------------------------------
274
+ // MINCHECKED
275
+ // ---------------------------------------------------------
276
+ if ($this.attr("minchecked") !== undefined) {
277
+ message = "Not enough options checked; Minimum of '" + $this.attr("minchecked") + "' required<!-- data-validation-minchecked-message to override -->";
278
+ if ($this.data("validationMincheckedMessage")) {
279
+ message = $this.data("validationMincheckedMessage");
280
+ }
281
+ $this.data("validationMincheckedMessage", message);
282
+ $this.data("validationMincheckedMinchecked", $this.attr("minchecked"));
283
+ }
284
+ // ---------------------------------------------------------
285
+ // MAXCHECKED
286
+ // ---------------------------------------------------------
287
+ if ($this.attr("maxchecked") !== undefined) {
288
+ message = "Too many options checked; Maximum of '" + $this.attr("maxchecked") + "' required<!-- data-validation-maxchecked-message to override -->";
289
+ if ($this.data("validationMaxcheckedMessage")) {
290
+ message = $this.data("validationMaxcheckedMessage");
291
+ }
292
+ $this.data("validationMaxcheckedMessage", message);
293
+ $this.data("validationMaxcheckedMaxchecked", $this.attr("maxchecked"));
294
+ }
295
+ }
296
+
297
+ // =============================================================
298
+ // COLLECT VALIDATOR NAMES
299
+ // =============================================================
300
+
301
+ // Get named validators
302
+ if ($this.data("validation") !== undefined) {
303
+ validatorNames = $this.data("validation").split(",");
304
+ }
305
+
306
+ // Get extra ones defined on the element's data attributes
307
+ $.each($this.data(), function (i, el) {
308
+ var parts = i.replace(/([A-Z])/g, ",$1").split(",");
309
+ if (parts[0] === "validation" && parts[1]) {
310
+ validatorNames.push(parts[1]);
311
+ }
312
+ });
313
+
314
+ // =============================================================
315
+ // NORMALISE VALIDATOR NAMES
316
+ // =============================================================
317
+
318
+ var validatorNamesToInspect = validatorNames;
319
+ var newValidatorNamesToInspect = [];
320
+
321
+ do // repeatedly expand 'shortcut' validators into their real validators
322
+ {
323
+ // Uppercase only the first letter of each name
324
+ $.each(validatorNames, function (i, el) {
325
+ validatorNames[i] = formatValidatorName(el);
326
+ });
327
+
328
+ // Remove duplicate validator names
329
+ validatorNames = $.unique(validatorNames);
330
+
331
+ // Pull out the new validator names from each shortcut
332
+ newValidatorNamesToInspect = [];
333
+ $.each(validatorNamesToInspect, function(i, el) {
334
+ if ($this.data("validation" + el + "Shortcut") !== undefined) {
335
+ // Are these custom validators?
336
+ // Pull them out!
337
+ $.each($this.data("validation" + el + "Shortcut").split(","), function(i2, el2) {
338
+ newValidatorNamesToInspect.push(el2);
339
+ });
340
+ } else if (settings.builtInValidators[el.toLowerCase()]) {
341
+ // Is this a recognised built-in?
342
+ // Pull it out!
343
+ var validator = settings.builtInValidators[el.toLowerCase()];
344
+ if (validator.type.toLowerCase() === "shortcut") {
345
+ $.each(validator.shortcut.split(","), function (i, el) {
346
+ el = formatValidatorName(el);
347
+ newValidatorNamesToInspect.push(el);
348
+ validatorNames.push(el);
349
+ });
350
+ }
351
+ }
352
+ });
353
+
354
+ validatorNamesToInspect = newValidatorNamesToInspect;
355
+
356
+ } while (validatorNamesToInspect.length > 0)
357
+
358
+ // =============================================================
359
+ // SET UP VALIDATOR ARRAYS
360
+ // =============================================================
361
+
362
+ var validators = {};
363
+
364
+ $.each(validatorNames, function (i, el) {
365
+ // Set up the 'override' message
366
+ var message = $this.data("validation" + el + "Message");
367
+ var hasOverrideMessage = (message !== undefined);
368
+ var foundValidator = false;
369
+ message =
370
+ (
371
+ message
372
+ ? message
373
+ : "'" + el + "' validation failed <!-- Add attribute 'data-validation-" + el.toLowerCase() + "-message' to input to change this message -->"
374
+ )
375
+ ;
376
+
377
+ $.each(
378
+ settings.validatorTypes,
379
+ function (validatorType, validatorTemplate) {
380
+ if (validators[validatorType] === undefined) {
381
+ validators[validatorType] = [];
382
+ }
383
+ if (!foundValidator && $this.data("validation" + el + formatValidatorName(validatorTemplate.name)) !== undefined) {
384
+ validators[validatorType].push(
385
+ $.extend(
386
+ true,
387
+ {
388
+ name: formatValidatorName(validatorTemplate.name),
389
+ message: message
390
+ },
391
+ validatorTemplate.init($this, el)
392
+ )
393
+ );
394
+ foundValidator = true;
395
+ }
396
+ }
397
+ );
398
+
399
+ if (!foundValidator && settings.builtInValidators[el.toLowerCase()]) {
400
+
401
+ var validator = $.extend(true, {}, settings.builtInValidators[el.toLowerCase()]);
402
+ if (hasOverrideMessage) {
403
+ validator.message = message;
404
+ }
405
+ var validatorType = validator.type.toLowerCase();
406
+
407
+ if (validatorType === "shortcut") {
408
+ foundValidator = true;
409
+ } else {
410
+ $.each(
411
+ settings.validatorTypes,
412
+ function (validatorTemplateType, validatorTemplate) {
413
+ if (validators[validatorTemplateType] === undefined) {
414
+ validators[validatorTemplateType] = [];
415
+ }
416
+ if (!foundValidator && validatorType === validatorTemplateType.toLowerCase()) {
417
+ $this.data("validation" + el + formatValidatorName(validatorTemplate.name), validator[validatorTemplate.name.toLowerCase()]);
418
+ validators[validatorType].push(
419
+ $.extend(
420
+ validator,
421
+ validatorTemplate.init($this, el)
422
+ )
423
+ );
424
+ foundValidator = true;
425
+ }
426
+ }
427
+ );
428
+ }
429
+ }
430
+
431
+ if (! foundValidator) {
432
+ $.error("Cannot find validation info for '" + el + "'");
433
+ }
434
+ });
435
+
436
+ // =============================================================
437
+ // STORE FALLBACK VALUES
438
+ // =============================================================
439
+
440
+ $helpBlock.data(
441
+ "original-contents",
442
+ (
443
+ $helpBlock.data("original-contents")
444
+ ? $helpBlock.data("original-contents")
445
+ : $helpBlock.html()
446
+ )
447
+ );
448
+
449
+ $helpBlock.data(
450
+ "original-role",
451
+ (
452
+ $helpBlock.data("original-role")
453
+ ? $helpBlock.data("original-role")
454
+ : $helpBlock.attr("role")
455
+ )
456
+ );
457
+
458
+ $controlGroup.data(
459
+ "original-classes",
460
+ (
461
+ $controlGroup.data("original-clases")
462
+ ? $controlGroup.data("original-classes")
463
+ : $controlGroup.attr("class")
464
+ )
465
+ );
466
+
467
+ $this.data(
468
+ "original-aria-invalid",
469
+ (
470
+ $this.data("original-aria-invalid")
471
+ ? $this.data("original-aria-invalid")
472
+ : $this.attr("aria-invalid")
473
+ )
474
+ );
475
+
476
+ // =============================================================
477
+ // VALIDATION
478
+ // =============================================================
479
+
480
+ $this.bind(
481
+ "validation.validation",
482
+ function (event, params) {
483
+
484
+ var value = getValue($this);
485
+
486
+ // Get a list of the errors to apply
487
+ var errorsFound = [];
488
+
489
+ $.each(validators, function (validatorType, validatorTypeArray) {
490
+ if (value || value.length || (params && params.includeEmpty) || (!!settings.validatorTypes[validatorType].blockSubmit && params && !!params.submitting)) {
491
+ $.each(validatorTypeArray, function (i, validator) {
492
+ if (settings.validatorTypes[validatorType].validate($this, value, validator)) {
493
+ errorsFound.push(validator.message);
494
+ }
495
+ });
496
+ }
497
+ });
498
+
499
+ return errorsFound;
500
+ }
501
+ );
502
+
503
+ $this.bind(
504
+ "getValidators.validation",
505
+ function () {
506
+ return validators;
507
+ }
508
+ );
509
+
510
+ // =============================================================
511
+ // WATCH FOR CHANGES
512
+ // =============================================================
513
+ $this.bind(
514
+ "submit.validation",
515
+ function () {
516
+ return $this.triggerHandler("change.validation", {submitting: true});
517
+ }
518
+ );
519
+ $this.bind(
520
+ [
521
+ "keyup",
522
+ "focus",
523
+ "blur",
524
+ "click",
525
+ "keydown",
526
+ "keypress",
527
+ "change"
528
+ ].join(".validation ") + ".validation",
529
+ function (e, params) {
530
+
531
+ var value = getValue($this);
532
+
533
+ var errorsFound = [];
534
+
535
+ $controlGroup.find("input,textarea,select").each(function (i, el) {
536
+ var oldCount = errorsFound.length;
537
+ $.each($(el).triggerHandler("validation.validation", params), function (j, message) {
538
+ errorsFound.push(message);
539
+ });
540
+ if (errorsFound.length > oldCount) {
541
+ $(el).attr("aria-invalid", "true");
542
+ } else {
543
+ var original = $this.data("original-aria-invalid");
544
+ $(el).attr("aria-invalid", (original !== undefined ? original : false));
545
+ }
546
+ });
547
+
548
+ $form.find("input,select,textarea").not($this).not("[name=\"" + $this.attr("name") + "\"]").trigger("validationLostFocus.validation");
549
+
550
+ errorsFound = $.unique(errorsFound.sort());
551
+
552
+ // Were there any errors?
553
+ if (errorsFound.length) {
554
+ // Better flag it up as a warning.
555
+ $controlGroup.removeClass("success error").addClass("warning");
556
+
557
+ // How many errors did we find?
558
+ if (settings.options.semanticallyStrict && errorsFound.length === 1) {
559
+ // Only one? Being strict? Just output it.
560
+ $helpBlock.html(errorsFound[0] +
561
+ ( settings.options.prependExistingHelpBlock ? $helpBlock.data("original-contents") : "" ));
562
+ } else {
563
+ // Multiple? Being sloppy? Glue them together into an UL.
564
+ $helpBlock.html("<ul role=\"alert\"><li>" + errorsFound.join("</li><li>") + "</li></ul>" +
565
+ ( settings.options.prependExistingHelpBlock ? $helpBlock.data("original-contents") : "" ));
566
+ }
567
+ } else {
568
+ $controlGroup.removeClass("warning error success");
569
+ if (value.length > 0) {
570
+ $controlGroup.addClass("success");
571
+ }
572
+ $helpBlock.html($helpBlock.data("original-contents"));
573
+ }
574
+
575
+ if (e.type === "blur") {
576
+ $controlGroup.removeClass("success");
577
+ }
578
+ }
579
+ );
580
+ $this.bind("validationLostFocus.validation", function () {
581
+ $controlGroup.removeClass("success");
582
+ });
583
+ });
584
+ },
585
+ destroy : function( ) {
586
+
587
+ return this.each(
588
+ function() {
589
+
590
+ var
591
+ $this = $(this),
592
+ $controlGroup = $this.parents(".form-group").first(),
593
+ $helpBlock = $controlGroup.find(".help-block").first();
594
+
595
+ // remove our events
596
+ $this.unbind('.validation'); // events are namespaced.
597
+ // reset help text
598
+ $helpBlock.html($helpBlock.data("original-contents"));
599
+ // reset classes
600
+ $controlGroup.attr("class", $controlGroup.data("original-classes"));
601
+ // reset aria
602
+ $this.attr("aria-invalid", $this.data("original-aria-invalid"));
603
+ // reset role
604
+ $helpBlock.attr("role", $this.data("original-role"));
605
+ // remove all elements we created
606
+ if (createdElements.indexOf($helpBlock[0]) > -1) {
607
+ $helpBlock.remove();
608
+ }
609
+
610
+ }
611
+ );
612
+
613
+ },
614
+ collectErrors : function(includeEmpty) {
615
+
616
+ var errorMessages = {};
617
+ this.each(function (i, el) {
618
+ var $el = $(el);
619
+ var name = $el.attr("name");
620
+ var errors = $el.triggerHandler("validation.validation", {includeEmpty: true});
621
+ errorMessages[name] = $.extend(true, errors, errorMessages[name]);
622
+ });
623
+
624
+ $.each(errorMessages, function (i, el) {
625
+ if (el.length === 0) {
626
+ delete errorMessages[i];
627
+ }
628
+ });
629
+
630
+ return errorMessages;
631
+
632
+ },
633
+ hasErrors: function() {
634
+
635
+ var errorMessages = [];
636
+
637
+ this.each(function (i, el) {
638
+ errorMessages = errorMessages.concat(
639
+ $(el).triggerHandler("getValidators.validation") ? $(el).triggerHandler("validation.validation", {submitting: true}) : []
640
+ );
641
+ });
642
+
643
+ return (errorMessages.length > 0);
644
+ },
645
+ override : function (newDefaults) {
646
+ defaults = $.extend(true, defaults, newDefaults);
647
+ }
648
+ },
649
+ validatorTypes: {
650
+ callback: {
651
+ name: "callback",
652
+ init: function ($this, name) {
653
+ return {
654
+ validatorName: name,
655
+ callback: $this.data("validation" + name + "Callback"),
656
+ lastValue: $this.val(),
657
+ lastValid: true,
658
+ lastFinished: true
659
+ };
660
+ },
661
+ validate: function ($this, value, validator) {
662
+ if (validator.lastValue === value && validator.lastFinished) {
663
+ return !validator.lastValid;
664
+ }
665
+
666
+ if (validator.lastFinished === true)
667
+ {
668
+ validator.lastValue = value;
669
+ validator.lastValid = true;
670
+ validator.lastFinished = false;
671
+
672
+ var rrjqbvValidator = validator;
673
+ var rrjqbvThis = $this;
674
+ executeFunctionByName(
675
+ validator.callback,
676
+ window,
677
+ $this,
678
+ value,
679
+ function (data) {
680
+ if (rrjqbvValidator.lastValue === data.value) {
681
+ rrjqbvValidator.lastValid = data.valid;
682
+ if (data.message) {
683
+ rrjqbvValidator.message = data.message;
684
+ }
685
+ rrjqbvValidator.lastFinished = true;
686
+ rrjqbvThis.data("validation" + rrjqbvValidator.validatorName + "Message", rrjqbvValidator.message);
687
+ // Timeout is set to avoid problems with the events being considered 'already fired'
688
+ setTimeout(function () {
689
+ rrjqbvThis.trigger("change.validation");
690
+ }, 1); // doesn't need a long timeout, just long enough for the event bubble to burst
691
+ }
692
+ }
693
+ );
694
+ }
695
+
696
+ return false;
697
+
698
+ }
699
+ },
700
+ ajax: {
701
+ name: "ajax",
702
+ init: function ($this, name) {
703
+ return {
704
+ validatorName: name,
705
+ url: $this.data("validation" + name + "Ajax"),
706
+ lastValue: $this.val(),
707
+ lastValid: true,
708
+ lastFinished: true
709
+ };
710
+ },
711
+ validate: function ($this, value, validator) {
712
+ if (""+validator.lastValue === ""+value && validator.lastFinished === true) {
713
+ return validator.lastValid === false;
714
+ }
715
+
716
+ if (validator.lastFinished === true)
717
+ {
718
+ validator.lastValue = value;
719
+ validator.lastValid = true;
720
+ validator.lastFinished = false;
721
+ $.ajax({
722
+ url: validator.url,
723
+ data: "value=" + value + "&field=" + $this.attr("name"),
724
+ dataType: "json",
725
+ success: function (data) {
726
+ if (""+validator.lastValue === ""+data.value) {
727
+ validator.lastValid = !!(data.valid);
728
+ if (data.message) {
729
+ validator.message = data.message;
730
+ }
731
+ validator.lastFinished = true;
732
+ $this.data("validation" + validator.validatorName + "Message", validator.message);
733
+ // Timeout is set to avoid problems with the events being considered 'already fired'
734
+ setTimeout(function () {
735
+ $this.trigger("change.validation");
736
+ }, 1); // doesn't need a long timeout, just long enough for the event bubble to burst
737
+ }
738
+ },
739
+ failure: function () {
740
+ validator.lastValid = true;
741
+ validator.message = "ajax call failed";
742
+ validator.lastFinished = true;
743
+ $this.data("validation" + validator.validatorName + "Message", validator.message);
744
+ // Timeout is set to avoid problems with the events being considered 'already fired'
745
+ setTimeout(function () {
746
+ $this.trigger("change.validation");
747
+ }, 1); // doesn't need a long timeout, just long enough for the event bubble to burst
748
+ }
749
+ });
750
+ }
751
+
752
+ return false;
753
+
754
+ }
755
+ },
756
+ regex: {
757
+ name: "regex",
758
+ init: function ($this, name) {
759
+ return {regex: regexFromString($this.data("validation" + name + "Regex"))};
760
+ },
761
+ validate: function ($this, value, validator) {
762
+ return (!validator.regex.test(value) && ! validator.negative)
763
+ || (validator.regex.test(value) && validator.negative);
764
+ }
765
+ },
766
+ required: {
767
+ name: "required",
768
+ init: function ($this, name) {
769
+ return {};
770
+ },
771
+ validate: function ($this, value, validator) {
772
+ return !!(value.length === 0 && ! validator.negative)
773
+ || !!(value.length > 0 && validator.negative);
774
+ },
775
+ blockSubmit: true
776
+ },
777
+ match: {
778
+ name: "match",
779
+ init: function ($this, name) {
780
+ var element = $this.parents("form").first().find("[name=\"" + $this.data("validation" + name + "Match") + "\"]").first();
781
+ element.bind("validation.validation", function () {
782
+ $this.trigger("change.validation", {submitting: true});
783
+ });
784
+ return {"element": element};
785
+ },
786
+ validate: function ($this, value, validator) {
787
+ return (value !== validator.element.val() && ! validator.negative)
788
+ || (value === validator.element.val() && validator.negative);
789
+ },
790
+ blockSubmit: true
791
+ },
792
+ max: {
793
+ name: "max",
794
+ init: function ($this, name) {
795
+ return {max: $this.data("validation" + name + "Max")};
796
+ },
797
+ validate: function ($this, value, validator) {
798
+ return (parseFloat(value, 10) > parseFloat(validator.max, 10) && ! validator.negative)
799
+ || (parseFloat(value, 10) <= parseFloat(validator.max, 10) && validator.negative);
800
+ }
801
+ },
802
+ min: {
803
+ name: "min",
804
+ init: function ($this, name) {
805
+ return {min: $this.data("validation" + name + "Min")};
806
+ },
807
+ validate: function ($this, value, validator) {
808
+ return (parseFloat(value) < parseFloat(validator.min) && ! validator.negative)
809
+ || (parseFloat(value) >= parseFloat(validator.min) && validator.negative);
810
+ }
811
+ },
812
+ maxlength: {
813
+ name: "maxlength",
814
+ init: function ($this, name) {
815
+ return {maxlength: $this.data("validation" + name + "Maxlength")};
816
+ },
817
+ validate: function ($this, value, validator) {
818
+ return ((value.length > validator.maxlength) && ! validator.negative)
819
+ || ((value.length <= validator.maxlength) && validator.negative);
820
+ }
821
+ },
822
+ minlength: {
823
+ name: "minlength",
824
+ init: function ($this, name) {
825
+ return {minlength: $this.data("validation" + name + "Minlength")};
826
+ },
827
+ validate: function ($this, value, validator) {
828
+ return ((value.length < validator.minlength) && ! validator.negative)
829
+ || ((value.length >= validator.minlength) && validator.negative);
830
+ }
831
+ },
832
+ maxchecked: {
833
+ name: "maxchecked",
834
+ init: function ($this, name) {
835
+ var elements = $this.parents("form").first().find("[name=\"" + $this.attr("name") + "\"]");
836
+ elements.bind("click.validation", function () {
837
+ $this.trigger("change.validation", {includeEmpty: true});
838
+ });
839
+ return {maxchecked: $this.data("validation" + name + "Maxchecked"), elements: elements};
840
+ },
841
+ validate: function ($this, value, validator) {
842
+ return (validator.elements.filter(":checked").length > validator.maxchecked && ! validator.negative)
843
+ || (validator.elements.filter(":checked").length <= validator.maxchecked && validator.negative);
844
+ },
845
+ blockSubmit: true
846
+ },
847
+ minchecked: {
848
+ name: "minchecked",
849
+ init: function ($this, name) {
850
+ var elements = $this.parents("form").first().find("[name=\"" + $this.attr("name") + "\"]");
851
+ elements.bind("click.validation", function () {
852
+ $this.trigger("change.validation", {includeEmpty: true});
853
+ });
854
+ return {minchecked: $this.data("validation" + name + "Minchecked"), elements: elements};
855
+ },
856
+ validate: function ($this, value, validator) {
857
+ return (validator.elements.filter(":checked").length < validator.minchecked && ! validator.negative)
858
+ || (validator.elements.filter(":checked").length >= validator.minchecked && validator.negative);
859
+ },
860
+ blockSubmit: true
861
+ }
862
+ },
863
+ builtInValidators: {
864
+ email: {
865
+ name: "Email",
866
+ type: "shortcut",
867
+ shortcut: "validemail"
868
+ },
869
+ validemail: {
870
+ name: "Validemail",
871
+ type: "regex",
872
+ regex: "[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\\\.[A-Za-z]{2,4}",
873
+ message: "Not a valid email address<!-- data-validator-validemail-message to override -->"
874
+ },
875
+ passwordagain: {
876
+ name: "Passwordagain",
877
+ type: "match",
878
+ match: "password",
879
+ message: "Does not match the given password<!-- data-validator-paswordagain-message to override -->"
880
+ },
881
+ positive: {
882
+ name: "Positive",
883
+ type: "shortcut",
884
+ shortcut: "number,positivenumber"
885
+ },
886
+ negative: {
887
+ name: "Negative",
888
+ type: "shortcut",
889
+ shortcut: "number,negativenumber"
890
+ },
891
+ number: {
892
+ name: "Number",
893
+ type: "regex",
894
+ regex: "([+-]?\\\d+(\\\.\\\d*)?([eE][+-]?[0-9]+)?)?",
895
+ message: "Must be a number<!-- data-validator-number-message to override -->"
896
+ },
897
+ integer: {
898
+ name: "Integer",
899
+ type: "regex",
900
+ regex: "[+-]?\\\d+",
901
+ message: "No decimal places allowed<!-- data-validator-integer-message to override -->"
902
+ },
903
+ positivenumber: {
904
+ name: "Positivenumber",
905
+ type: "min",
906
+ min: 0,
907
+ message: "Must be a positive number<!-- data-validator-positivenumber-message to override -->"
908
+ },
909
+ negativenumber: {
910
+ name: "Negativenumber",
911
+ type: "max",
912
+ max: 0,
913
+ message: "Must be a negative number<!-- data-validator-negativenumber-message to override -->"
914
+ },
915
+ required: {
916
+ name: "Required",
917
+ type: "required",
918
+ message: "This is required<!-- data-validator-required-message to override -->"
919
+ },
920
+ checkone: {
921
+ name: "Checkone",
922
+ type: "minchecked",
923
+ minchecked: 1,
924
+ message: "Check at least one option<!-- data-validation-checkone-message to override -->"
925
+ }
926
+ }
927
+ };
928
+
929
+ var formatValidatorName = function (name) {
930
+ return name
931
+ .toLowerCase()
932
+ .replace(
933
+ /(^|\s)([a-z])/g ,
934
+ function(m,p1,p2) {
935
+ return p1+p2.toUpperCase();
936
+ }
937
+ )
938
+ ;
939
+ };
940
+
941
+ var getValue = function ($this) {
942
+ // Extract the value we're talking about
943
+ var value = $this.val();
944
+ var type = $this.attr("type");
945
+ if (type === "checkbox") {
946
+ value = ($this.is(":checked") ? value : "");
947
+ }
948
+ if (type === "radio") {
949
+ value = ($('input[name="' + $this.attr("name") + '"]:checked').length > 0 ? value : "");
950
+ }
951
+ return value;
952
+ };
953
+
954
+ function regexFromString(inputstring) {
955
+ return new RegExp("^" + inputstring + "$");
956
+ }
957
+
958
+ /**
959
+ * Thanks to Jason Bunting via StackOverflow.com
960
+ *
961
+ * http://stackoverflow.com/questions/359788/how-to-execute-a-javascript-function-when-i-have-its-name-as-a-string#answer-359910
962
+ * Short link: http://tinyurl.com/executeFunctionByName
963
+ **/
964
+ function executeFunctionByName(functionName, context /*, args*/) {
965
+ var args = Array.prototype.slice.call(arguments).splice(2);
966
+ var namespaces = functionName.split(".");
967
+ var func = namespaces.pop();
968
+ for(var i = 0; i < namespaces.length; i++) {
969
+ context = context[namespaces[i]];
970
+ }
971
+ return context[func].apply(this, args);
972
+ }
973
+
974
+ $.fn.jqBootstrapValidation = function( method ) {
975
+
976
+ if ( defaults.methods[method] ) {
977
+ return defaults.methods[method].apply( this, Array.prototype.slice.call( arguments, 1 ));
978
+ } else if ( typeof method === 'object' || ! method ) {
979
+ return defaults.methods.init.apply( this, arguments );
980
+ } else {
981
+ $.error( 'Method ' + method + ' does not exist on jQuery.jqBootstrapValidation' );
982
+ return null;
983
+ }
984
+
985
+ };
986
+
987
+ $.jqBootstrapValidation = function (options) {
988
+ $(":input").not("[type=image],[type=submit]").jqBootstrapValidation.apply(this,arguments);
989
+ };
990
+
991
+ })( jQuery );
992
+
993
+ // Floating label headings for the contact form
994
+ $(function() {
995
+ $("body").on("input propertychange", ".floating-label-form-group", function(e) {
996
+ $(this).toggleClass("floating-label-form-group-with-value", !!$(e.target).val());
997
+ }).on("focus", ".floating-label-form-group", function() {
998
+ $(this).addClass("floating-label-form-group-with-focus");
999
+ }).on("blur", ".floating-label-form-group", function() {
1000
+ $(this).removeClass("floating-label-form-group-with-focus");
1001
+ });
1002
+ });
1003
+
1004
+ // Navigation Scripts to Show Header on Scroll-Up
1005
+ jQuery(document).ready(function($) {
1006
+ var MQL = 1170;
1007
+
1008
+ //primary navigation slide-in effect
1009
+ if ($(window).width() > MQL) {
1010
+ var headerHeight = $('.navbar-custom').height();
1011
+ $(window).on('scroll', {
1012
+ previousTop: 0
1013
+ },
1014
+ function() {
1015
+ var currentTop = $(window).scrollTop();
1016
+ //check if user is scrolling up
1017
+ if (currentTop < this.previousTop) {
1018
+ //if scrolling up...
1019
+ if (currentTop > 0 && $('.navbar-custom').hasClass('is-fixed')) {
1020
+ $('.navbar-custom').addClass('is-visible');
1021
+ } else {
1022
+ $('.navbar-custom').removeClass('is-visible is-fixed');
1023
+ }
1024
+ } else {
1025
+ //if scrolling down...
1026
+ $('.navbar-custom').removeClass('is-visible');
1027
+ if (currentTop > headerHeight && !$('.navbar-custom').hasClass('is-fixed')) $('.navbar-custom').addClass('is-fixed');
1028
+ }
1029
+ this.previousTop = currentTop;
1030
+ });
1031
+ }
1032
+ });