Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Confirm to leave the page when editing a form with jQuery

Tags:

jquery

I am coding a form and I need a function like stackoverflow have: "You have started writing or editing a post.".

I've looked through the code of stackoverflow to see how they do it but I don't get it at all to apply to my simple form. This is what they have on question.js:

function initNavPrevention(b) {
    if (b.which == "undefined") {
        return
    }
    var a = $("#wmd-input");
    a.unbind("keypress", initNavPrevention);
    setConfirmUnload("You have started writing or editing a post.", a)
}

One of the functions triggers this (I don't know what c is):

if (c) {
    e.keypress(initNavPrevention)
}

And finally they have setConfirmUnload(null); to disable it I suppose.

My case is simple. I have a page where I load a <form /> via AJAX and I want to prevent when this form is loaded to leave the page without click on the SUBMIT button or if the user clicks CANCEL the form is disabled and the prevention will not popup.

Having this code, can someone tell me how I can include it on my website?

This is my AJAX function to load the form:

$("#projects_historics_edit_part_1").click(function(){
    $("#ajax-loader").fadeIn('normal');

    $.ajax({
        url: BASE_URL + 'projects/ajax/get_historics_edit_part_1/' + PROJECT_ID,
        type: "POST",
        success: function(data) {
            $("div#projects_historics_part_1").html(data);

            $("#client").autocomplete(client_autcomplete_options);

            var inicofin = $("#initdate, #findate").datepicker(initdate_datepicker_options);
        }
    });

    $("#ajax-loader").fadeOut("normal");

    return false;    
});

Thank you in advance!

like image 374
udexter Avatar asked Feb 24 '11 09:02

udexter


3 Answers

You can do this using window.onbeforeunload like this:

window.onbeforeunload = function() {
    return 'Are you sure you want to navigate away from this page?';
};

So in your code, you would put that inside the success callback of your $.ajax call.

To unset the confirmation, you can do this:

window.onbeforeunload = null;

Also, see this answer: How can I override the OnBeforeUnload dialog and replace it with my own?

like image 88
betamax Avatar answered Nov 04 '22 11:11

betamax


This is how in JQuery

$('#form').data('serialize',$('#form').serialize());
  // On load save form current state

$(window).bind('beforeunload', function(e){
    if($('#form').serialize()!=$('#form').data('serialize'))return true;
    else e=null;
    // i.e; if form state change show box not.
});

You can Google JQuery Form Serialize function, this will collect all form inputs and save it in array. I guess this explain is enough :)

like image 30
Wasim A. Avatar answered Nov 04 '22 10:11

Wasim A.


I made a general function for this based on Wasim's excellent answer:

var confirm_nav_ignore = [];
function confirm_nav() {
    function save_form_state($form) {
        var old_state = $form.serialize();
        $form.data('old_state', old_state);
    }

    // On load, save form current state
    $('form').each(function() {
        save_form_state($(this));
    });

    // On submit, save form current state
    $('form').submit(function() {
        save_form_state($(this));
    });

    // strip fields that we should ignore
    function strip(form_data) {
        for (var i=0; i<confirm_nav_ignore.length; i++) {
            var field = confirm_nav_ignore[i];
            form_data = form_data.replace(new RegExp("\\b"+field+"=[^&]*"), '');
        }   
        return form_data;
    }

    // Before unload, confirm navigation if any field has been edited
    $(window).on('beforeunload', function(e) {
        var rv;
        $('form').each(function() {
            var $form = $(this);
            var old_state = strip($form.data('old_state'));
            var new_state = strip($form.serialize());
            if (new_state != old_state) {
                rv = '';
            }   
        }); 
        return rv;
    }); 
}

// I call it at the end of the on-load handler:
$(function() {
    confirm_nav();
});

Changes:

  1. Allows to submit a form without confirmation.
  2. Works with multiple forms, using $('form').each(...)
  3. Does not depend on the form having id="form"
  4. Returns '' as the custom confirmation message, instead of true which displays "true" in Chrome.
  5. For difficult pages, it can be told to ignore certain form fields.
  6. Run it from the end of the on-load handler... seems like a good idea.
  7. It's generic enough, so we can use it on every page in a site.

It might look over-engineered, but that's what I had to do to make it work correctly in a real web app.

Thanks to Wasim, for the great idea to use JQuery $(form).serialize().

like image 3
Sam Watkins Avatar answered Nov 04 '22 11:11

Sam Watkins