My jQuery Form Plugin












1














Long story short, I wrote this jQuery form plugin as an exercise to better understand how other form plugins work. If you have any suggestions about how to improve this (or my coding style in general) let me know.



var Form = function(options) {

options = options || {};

this.url = options.url;
this.$fields = $(options.fields);
this.$status = $(options.status);
this.$submit = $(options.submit);
this.$form = options.form || 'form';
this.rFields = {
required: /[^.*]/,
nodigit: /^[^0-9]+$/,
email: /^[a-z0-9._%-]+@[a-z0-9.-]+.[a-z]{2,4}$/i,
};
this.required = [
this.rFields.nodigit,
this.rFields.email,
this.rFields.required,
this.rFields.required
];
this.errors = ['Please enter a valid name.', 'Please enter a valid email address.', 'You must enter a subject.', 'No message? But I wanted one :('];

}

Form.prototype.init = function init() {

this.$submit.click($.proxy(function(e) {
this.process(e);
}, this));

};

Form.prototype.process = function process(e) {

var filled = true;

e.stopImmediatePropagation();
e.preventDefault();

this.$fields.each($.proxy(function(i, element) {

var $element = $(element);

if (!$element.val().match(this.required[i])) {

$element.focus();
this.update(this.errors[i], true);
filled = false;
return false;

}
}, this));

filled && this.submit($(this.$form).serialize());

};

Form.prototype.update = function update(message, isError) {

if (isError && this.$status.hasClass('error')) {

this.$status.addClass('error')
.text(message);

setTimeout($.proxy(function() {
this.$status.removeClass('shake');
}, this), 500);

}
else {

this.$status.fadeOut();

setTimeout($.proxy(function() {
this.$status.text(message)
.fadeIn();

if (isError) {
this.$status.removeClass('shake')
.addClass('error');
}
else {
this.$status.removeClass('error info')
.addClass('success');
}
}, this), 500);

}

return this;

};

Form.prototype.submit = function submit(data) {

$.ajax({
type: 'POST',
url: this.url,
data: data,
success: $.proxy(function(msg) {
this.$fields.fadeOut();
this.$status
.empty()
.append('<h4 class="">Nice one! The form has been sent to the server...</h4>')

$(this.$submit)
.animate({
left: '-=500px',
opacity: 0,
}, 300);

}, this)
});
};


Here's how to create a new form:



var exampleForm = new Form({
url: mailUrl,
fields: 'form input, form textarea',
status: '#status',
submit: '#submit'
})

exampleForm.init()


Pretty simple, just pass in..




  • The url to send the form data too

  • The form fields all encapsulated in one string seperated by commas in the correct order of validation rules.

  • Selector string where you want the status message to go

  • The submit selector. This element will actually submit the form










share|improve this question














bumped to the homepage by Community 17 mins ago


This question has answers that may be good or bad; the system has marked it active so that they can be reviewed.




















    1














    Long story short, I wrote this jQuery form plugin as an exercise to better understand how other form plugins work. If you have any suggestions about how to improve this (or my coding style in general) let me know.



    var Form = function(options) {

    options = options || {};

    this.url = options.url;
    this.$fields = $(options.fields);
    this.$status = $(options.status);
    this.$submit = $(options.submit);
    this.$form = options.form || 'form';
    this.rFields = {
    required: /[^.*]/,
    nodigit: /^[^0-9]+$/,
    email: /^[a-z0-9._%-]+@[a-z0-9.-]+.[a-z]{2,4}$/i,
    };
    this.required = [
    this.rFields.nodigit,
    this.rFields.email,
    this.rFields.required,
    this.rFields.required
    ];
    this.errors = ['Please enter a valid name.', 'Please enter a valid email address.', 'You must enter a subject.', 'No message? But I wanted one :('];

    }

    Form.prototype.init = function init() {

    this.$submit.click($.proxy(function(e) {
    this.process(e);
    }, this));

    };

    Form.prototype.process = function process(e) {

    var filled = true;

    e.stopImmediatePropagation();
    e.preventDefault();

    this.$fields.each($.proxy(function(i, element) {

    var $element = $(element);

    if (!$element.val().match(this.required[i])) {

    $element.focus();
    this.update(this.errors[i], true);
    filled = false;
    return false;

    }
    }, this));

    filled && this.submit($(this.$form).serialize());

    };

    Form.prototype.update = function update(message, isError) {

    if (isError && this.$status.hasClass('error')) {

    this.$status.addClass('error')
    .text(message);

    setTimeout($.proxy(function() {
    this.$status.removeClass('shake');
    }, this), 500);

    }
    else {

    this.$status.fadeOut();

    setTimeout($.proxy(function() {
    this.$status.text(message)
    .fadeIn();

    if (isError) {
    this.$status.removeClass('shake')
    .addClass('error');
    }
    else {
    this.$status.removeClass('error info')
    .addClass('success');
    }
    }, this), 500);

    }

    return this;

    };

    Form.prototype.submit = function submit(data) {

    $.ajax({
    type: 'POST',
    url: this.url,
    data: data,
    success: $.proxy(function(msg) {
    this.$fields.fadeOut();
    this.$status
    .empty()
    .append('<h4 class="">Nice one! The form has been sent to the server...</h4>')

    $(this.$submit)
    .animate({
    left: '-=500px',
    opacity: 0,
    }, 300);

    }, this)
    });
    };


    Here's how to create a new form:



    var exampleForm = new Form({
    url: mailUrl,
    fields: 'form input, form textarea',
    status: '#status',
    submit: '#submit'
    })

    exampleForm.init()


    Pretty simple, just pass in..




    • The url to send the form data too

    • The form fields all encapsulated in one string seperated by commas in the correct order of validation rules.

    • Selector string where you want the status message to go

    • The submit selector. This element will actually submit the form










    share|improve this question














    bumped to the homepage by Community 17 mins ago


    This question has answers that may be good or bad; the system has marked it active so that they can be reviewed.


















      1












      1








      1


      1





      Long story short, I wrote this jQuery form plugin as an exercise to better understand how other form plugins work. If you have any suggestions about how to improve this (or my coding style in general) let me know.



      var Form = function(options) {

      options = options || {};

      this.url = options.url;
      this.$fields = $(options.fields);
      this.$status = $(options.status);
      this.$submit = $(options.submit);
      this.$form = options.form || 'form';
      this.rFields = {
      required: /[^.*]/,
      nodigit: /^[^0-9]+$/,
      email: /^[a-z0-9._%-]+@[a-z0-9.-]+.[a-z]{2,4}$/i,
      };
      this.required = [
      this.rFields.nodigit,
      this.rFields.email,
      this.rFields.required,
      this.rFields.required
      ];
      this.errors = ['Please enter a valid name.', 'Please enter a valid email address.', 'You must enter a subject.', 'No message? But I wanted one :('];

      }

      Form.prototype.init = function init() {

      this.$submit.click($.proxy(function(e) {
      this.process(e);
      }, this));

      };

      Form.prototype.process = function process(e) {

      var filled = true;

      e.stopImmediatePropagation();
      e.preventDefault();

      this.$fields.each($.proxy(function(i, element) {

      var $element = $(element);

      if (!$element.val().match(this.required[i])) {

      $element.focus();
      this.update(this.errors[i], true);
      filled = false;
      return false;

      }
      }, this));

      filled && this.submit($(this.$form).serialize());

      };

      Form.prototype.update = function update(message, isError) {

      if (isError && this.$status.hasClass('error')) {

      this.$status.addClass('error')
      .text(message);

      setTimeout($.proxy(function() {
      this.$status.removeClass('shake');
      }, this), 500);

      }
      else {

      this.$status.fadeOut();

      setTimeout($.proxy(function() {
      this.$status.text(message)
      .fadeIn();

      if (isError) {
      this.$status.removeClass('shake')
      .addClass('error');
      }
      else {
      this.$status.removeClass('error info')
      .addClass('success');
      }
      }, this), 500);

      }

      return this;

      };

      Form.prototype.submit = function submit(data) {

      $.ajax({
      type: 'POST',
      url: this.url,
      data: data,
      success: $.proxy(function(msg) {
      this.$fields.fadeOut();
      this.$status
      .empty()
      .append('<h4 class="">Nice one! The form has been sent to the server...</h4>')

      $(this.$submit)
      .animate({
      left: '-=500px',
      opacity: 0,
      }, 300);

      }, this)
      });
      };


      Here's how to create a new form:



      var exampleForm = new Form({
      url: mailUrl,
      fields: 'form input, form textarea',
      status: '#status',
      submit: '#submit'
      })

      exampleForm.init()


      Pretty simple, just pass in..




      • The url to send the form data too

      • The form fields all encapsulated in one string seperated by commas in the correct order of validation rules.

      • Selector string where you want the status message to go

      • The submit selector. This element will actually submit the form










      share|improve this question













      Long story short, I wrote this jQuery form plugin as an exercise to better understand how other form plugins work. If you have any suggestions about how to improve this (or my coding style in general) let me know.



      var Form = function(options) {

      options = options || {};

      this.url = options.url;
      this.$fields = $(options.fields);
      this.$status = $(options.status);
      this.$submit = $(options.submit);
      this.$form = options.form || 'form';
      this.rFields = {
      required: /[^.*]/,
      nodigit: /^[^0-9]+$/,
      email: /^[a-z0-9._%-]+@[a-z0-9.-]+.[a-z]{2,4}$/i,
      };
      this.required = [
      this.rFields.nodigit,
      this.rFields.email,
      this.rFields.required,
      this.rFields.required
      ];
      this.errors = ['Please enter a valid name.', 'Please enter a valid email address.', 'You must enter a subject.', 'No message? But I wanted one :('];

      }

      Form.prototype.init = function init() {

      this.$submit.click($.proxy(function(e) {
      this.process(e);
      }, this));

      };

      Form.prototype.process = function process(e) {

      var filled = true;

      e.stopImmediatePropagation();
      e.preventDefault();

      this.$fields.each($.proxy(function(i, element) {

      var $element = $(element);

      if (!$element.val().match(this.required[i])) {

      $element.focus();
      this.update(this.errors[i], true);
      filled = false;
      return false;

      }
      }, this));

      filled && this.submit($(this.$form).serialize());

      };

      Form.prototype.update = function update(message, isError) {

      if (isError && this.$status.hasClass('error')) {

      this.$status.addClass('error')
      .text(message);

      setTimeout($.proxy(function() {
      this.$status.removeClass('shake');
      }, this), 500);

      }
      else {

      this.$status.fadeOut();

      setTimeout($.proxy(function() {
      this.$status.text(message)
      .fadeIn();

      if (isError) {
      this.$status.removeClass('shake')
      .addClass('error');
      }
      else {
      this.$status.removeClass('error info')
      .addClass('success');
      }
      }, this), 500);

      }

      return this;

      };

      Form.prototype.submit = function submit(data) {

      $.ajax({
      type: 'POST',
      url: this.url,
      data: data,
      success: $.proxy(function(msg) {
      this.$fields.fadeOut();
      this.$status
      .empty()
      .append('<h4 class="">Nice one! The form has been sent to the server...</h4>')

      $(this.$submit)
      .animate({
      left: '-=500px',
      opacity: 0,
      }, 300);

      }, this)
      });
      };


      Here's how to create a new form:



      var exampleForm = new Form({
      url: mailUrl,
      fields: 'form input, form textarea',
      status: '#status',
      submit: '#submit'
      })

      exampleForm.init()


      Pretty simple, just pass in..




      • The url to send the form data too

      • The form fields all encapsulated in one string seperated by commas in the correct order of validation rules.

      • Selector string where you want the status message to go

      • The submit selector. This element will actually submit the form







      javascript jquery






      share|improve this question













      share|improve this question











      share|improve this question




      share|improve this question










      asked Nov 4 '14 at 3:48







      user44869












      bumped to the homepage by Community 17 mins ago


      This question has answers that may be good or bad; the system has marked it active so that they can be reviewed.







      bumped to the homepage by Community 17 mins ago


      This question has answers that may be good or bad; the system has marked it active so that they can be reviewed.
























          1 Answer
          1






          active

          oldest

          votes


















          0














          From a once over;





          • I would not split statements like



            this.$status.addClass('error')
            .text(message);


            over 2 lines, I would reserve that treatment for the longer statements.



          • You seem to cheat with the simpleness of your call, Form knows already the error messages with this.errors and the validation types with this.required. These should have come in as options.

          • I tend to comment all my regular expressions with my intent and an example, that might be overkill for your regular expressions

          • In Form.prototype.process it would have been simpler to have var self = this and use self instead of using $.proxy(. While using closures is not a panacea, it should also not be avoided at all costs.


          • You seem to like shortcuts, consider



            return filled = false;


            instead of



            filled = false;
            return false;



          • filled && this.submit($(this.$form).serialize()); <-- Too hacky

          • Why is this code removing the shake css class, it never assigns it..

          • On the whole it seems Form.prototype.update could be written in a smarter way, but I would have to see an actual snippet working to get started on that






          share|improve this answer





















            Your Answer





            StackExchange.ifUsing("editor", function () {
            return StackExchange.using("mathjaxEditing", function () {
            StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix) {
            StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
            });
            });
            }, "mathjax-editing");

            StackExchange.ifUsing("editor", function () {
            StackExchange.using("externalEditor", function () {
            StackExchange.using("snippets", function () {
            StackExchange.snippets.init();
            });
            });
            }, "code-snippets");

            StackExchange.ready(function() {
            var channelOptions = {
            tags: "".split(" "),
            id: "196"
            };
            initTagRenderer("".split(" "), "".split(" "), channelOptions);

            StackExchange.using("externalEditor", function() {
            // Have to fire editor after snippets, if snippets enabled
            if (StackExchange.settings.snippets.snippetsEnabled) {
            StackExchange.using("snippets", function() {
            createEditor();
            });
            }
            else {
            createEditor();
            }
            });

            function createEditor() {
            StackExchange.prepareEditor({
            heartbeatType: 'answer',
            autoActivateHeartbeat: false,
            convertImagesToLinks: false,
            noModals: true,
            showLowRepImageUploadWarning: true,
            reputationToPostImages: null,
            bindNavPrevention: true,
            postfix: "",
            imageUploader: {
            brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
            contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
            allowUrls: true
            },
            onDemand: true,
            discardSelector: ".discard-answer"
            ,immediatelyShowMarkdownHelp:true
            });


            }
            });














            draft saved

            draft discarded


















            StackExchange.ready(
            function () {
            StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f68817%2fmy-jquery-form-plugin%23new-answer', 'question_page');
            }
            );

            Post as a guest















            Required, but never shown
























            1 Answer
            1






            active

            oldest

            votes








            1 Answer
            1






            active

            oldest

            votes









            active

            oldest

            votes






            active

            oldest

            votes









            0














            From a once over;





            • I would not split statements like



              this.$status.addClass('error')
              .text(message);


              over 2 lines, I would reserve that treatment for the longer statements.



            • You seem to cheat with the simpleness of your call, Form knows already the error messages with this.errors and the validation types with this.required. These should have come in as options.

            • I tend to comment all my regular expressions with my intent and an example, that might be overkill for your regular expressions

            • In Form.prototype.process it would have been simpler to have var self = this and use self instead of using $.proxy(. While using closures is not a panacea, it should also not be avoided at all costs.


            • You seem to like shortcuts, consider



              return filled = false;


              instead of



              filled = false;
              return false;



            • filled && this.submit($(this.$form).serialize()); <-- Too hacky

            • Why is this code removing the shake css class, it never assigns it..

            • On the whole it seems Form.prototype.update could be written in a smarter way, but I would have to see an actual snippet working to get started on that






            share|improve this answer


























              0














              From a once over;





              • I would not split statements like



                this.$status.addClass('error')
                .text(message);


                over 2 lines, I would reserve that treatment for the longer statements.



              • You seem to cheat with the simpleness of your call, Form knows already the error messages with this.errors and the validation types with this.required. These should have come in as options.

              • I tend to comment all my regular expressions with my intent and an example, that might be overkill for your regular expressions

              • In Form.prototype.process it would have been simpler to have var self = this and use self instead of using $.proxy(. While using closures is not a panacea, it should also not be avoided at all costs.


              • You seem to like shortcuts, consider



                return filled = false;


                instead of



                filled = false;
                return false;



              • filled && this.submit($(this.$form).serialize()); <-- Too hacky

              • Why is this code removing the shake css class, it never assigns it..

              • On the whole it seems Form.prototype.update could be written in a smarter way, but I would have to see an actual snippet working to get started on that






              share|improve this answer
























                0












                0








                0






                From a once over;





                • I would not split statements like



                  this.$status.addClass('error')
                  .text(message);


                  over 2 lines, I would reserve that treatment for the longer statements.



                • You seem to cheat with the simpleness of your call, Form knows already the error messages with this.errors and the validation types with this.required. These should have come in as options.

                • I tend to comment all my regular expressions with my intent and an example, that might be overkill for your regular expressions

                • In Form.prototype.process it would have been simpler to have var self = this and use self instead of using $.proxy(. While using closures is not a panacea, it should also not be avoided at all costs.


                • You seem to like shortcuts, consider



                  return filled = false;


                  instead of



                  filled = false;
                  return false;



                • filled && this.submit($(this.$form).serialize()); <-- Too hacky

                • Why is this code removing the shake css class, it never assigns it..

                • On the whole it seems Form.prototype.update could be written in a smarter way, but I would have to see an actual snippet working to get started on that






                share|improve this answer












                From a once over;





                • I would not split statements like



                  this.$status.addClass('error')
                  .text(message);


                  over 2 lines, I would reserve that treatment for the longer statements.



                • You seem to cheat with the simpleness of your call, Form knows already the error messages with this.errors and the validation types with this.required. These should have come in as options.

                • I tend to comment all my regular expressions with my intent and an example, that might be overkill for your regular expressions

                • In Form.prototype.process it would have been simpler to have var self = this and use self instead of using $.proxy(. While using closures is not a panacea, it should also not be avoided at all costs.


                • You seem to like shortcuts, consider



                  return filled = false;


                  instead of



                  filled = false;
                  return false;



                • filled && this.submit($(this.$form).serialize()); <-- Too hacky

                • Why is this code removing the shake css class, it never assigns it..

                • On the whole it seems Form.prototype.update could be written in a smarter way, but I would have to see an actual snippet working to get started on that







                share|improve this answer












                share|improve this answer



                share|improve this answer










                answered Nov 4 '14 at 15:08









                konijn

                27k453235




                27k453235






























                    draft saved

                    draft discarded




















































                    Thanks for contributing an answer to Code Review Stack Exchange!


                    • Please be sure to answer the question. Provide details and share your research!

                    But avoid



                    • Asking for help, clarification, or responding to other answers.

                    • Making statements based on opinion; back them up with references or personal experience.


                    Use MathJax to format equations. MathJax reference.


                    To learn more, see our tips on writing great answers.





                    Some of your past answers have not been well-received, and you're in danger of being blocked from answering.


                    Please pay close attention to the following guidance:


                    • Please be sure to answer the question. Provide details and share your research!

                    But avoid



                    • Asking for help, clarification, or responding to other answers.

                    • Making statements based on opinion; back them up with references or personal experience.


                    To learn more, see our tips on writing great answers.




                    draft saved


                    draft discarded














                    StackExchange.ready(
                    function () {
                    StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f68817%2fmy-jquery-form-plugin%23new-answer', 'question_page');
                    }
                    );

                    Post as a guest















                    Required, but never shown





















































                    Required, but never shown














                    Required, but never shown












                    Required, but never shown







                    Required, but never shown

































                    Required, but never shown














                    Required, but never shown












                    Required, but never shown







                    Required, but never shown







                    Popular posts from this blog

                    Morgemoulin

                    Scott Moir

                    Souastre