讓angularjs支援瀏覽器自動填表

來源:互聯網
上載者:User

 最近有很多前端同學都抱怨說,登陸表單都不能記錄自己的帳號。這對於用單頁面(single-page applications)和使用ajax比較多的網頁來說,還是一個蠻普遍的問題。

UserApp是一個使用angularjs構建的WebApp,但是他一直未能支援瀏覽器的"save password"特性。
下面就列出了一些發現的問題:

表單不能用js動態插入DOM。
表單必須真正的發出一個POST請求。(不能擷取表單內容,然後用ajax發出請求)
當瀏覽器自動填表後,$scope並不能獲得更新後的資料。
Firefox相對來說簡單一點,只要表單元素有name屬性,觸發提交事件之後,它就會自動提醒使用者是否記錄資料。

 

複製代碼 代碼如下:
<form name="login-form" ng-submit="login()">
    <input id="login" name="login" type="text" ng-model="user.login">
    <input id="password" name="password" type="password" ng-model="user.password">
</form>

 

firefox記錄資料的要求比較簡單

但是firefox有一個問題,在自動填表之後,$scope裡的資料並不會更新。所以我google了一下,發現了一些針對此問題的hacks。但是總感覺這些解決方案沒必要,因為我需要的只是在提交表單的時候將資料帶上,而不是一些很溜逼的資料雙向繫結技術。所以我就採用了很簡單的辦法:在提交表單的時候擷取表單元素的值。

 

複製代碼 代碼如下:
$scope.login = function() {
    $scope.user = {
        login: $("#login").val(),
        password: $("#password").val()
    };
    ...
    return false;
};

 

OK,現在firefox沒問題了,但是chrome怎麼辦呢?

chrome只有在form表單真正發起POST請求的時候才會提示使用者是否儲存密碼,但這樣的話就不能用Ajax操作了。

下面是解決辦法:

當表單發出Post請求時,用ng-submit截取,返回false將其阻止,同時用ajax提交資料。
當ajax成功返回時,將session儲存在cookies裡,再將表單重新提交一邊。
頁面重載的時候會發現已經認證過了,就將其重新導向到首頁。
這會讓頁面重新整理一次,但也就是登入的時候需要重新整理而已,確保頁面返回時是同一個地址就行了。
但如果表單是被動態添加進DOM的時候,這個方法仍然不行。解決方案就是在index.html中添加一個隱藏的表單,需要提交資料的時候就將其它表單攜帶的資料複製到隱藏表單裡。

我將它打包成了一個directive:

 

代碼如下:
app.directive("ngLoginSubmit", function(){
return {
    restrict: "A",
    scope: {
        onSubmit: "=ngLoginSubmit"
    },
    link: function(scope, element, attrs) {
        $(element)[0].onsubmit = function() {
            $("#login-login").val($("#login", element).val());
            $("#login-password").val($("#password", element).val());

 

            scope.onSubmit(function() {
                $("#login-form")[0].submit();
            });
            return false;
        };
    }
};
});

 

在index.html裡隱藏的表單:

 

代碼如下:
<form name="login-form" id="login-form" method="post" action="" style="display: none;">
    <input name="login" id="login-login" type="text">
    <input name="password" id="login-password" type="password">
</form>

 

臨時登入表單

 

代碼如下:
<form name="login-form" autocomplete="on" ng-login-submit="login">
    <input id="login" name="login" type="text" autocomplete="on">
    <input id="password" name="password" type="password" autocomplete="on">
</form>

 

登入用的controller:

 

代碼如下:
$scope.login = function(submit) {
    $scope.user = {
        login: $("#login").val(),
        password: $("#password").val()
    };

 

    function ajaxCallback() {
        submit();
    }  

    return false;
};

 

重新整理時會提示是否重新提交表單

現在這個問題解決了,但是每當你按下f5的時候,瀏覽器都會提醒你要不要重新提交表單。這確實有點蛋疼,所以我添加了一個pre-login.html檔案,隱藏的表單會提交資料到這裡,然後再重新導向到index.html。

現在OK了~

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.