วิธีการแสดงโหลดปินเนอร์ใน jQuery?


412

ในต้นแบบฉันสามารถแสดงภาพ "กำลังโหลด ... " ด้วยรหัสนี้:

var myAjax = new Ajax.Request( url, {method: 'get', parameters: pars, 
onLoading: showLoad, onComplete: showResponse} );

function showLoad () {
    ...
}

ในjQueryฉันสามารถโหลดหน้าเซิร์ฟเวอร์ลงในองค์ประกอบด้วยสิ่งนี้

$('#message').load('index.php?pg=ajaxFlashcard');

แต่ฉันจะแนบสปินเนอร์การโหลดไปยังคำสั่งนี้เหมือนกับที่ฉันทำใน Prototype ได้อย่างไร

คำตอบ:


793

มีสองวิธี วิธีที่ฉันชอบคือการแนบฟังก์ชั่นเข้ากับเหตุการณ์ ajaxStart / Stop ในองค์ประกอบนั้น

$('#loadingDiv')
    .hide()  // Hide it initially
    .ajaxStart(function() {
        $(this).show();
    })
    .ajaxStop(function() {
        $(this).hide();
    })
;

ฟังก์ชั่น ajaxStart / Stop จะเริ่มทำงานทุกครั้งที่คุณทำการโทร Ajax

อัปเดต : ตั้งแต่วันที่ jQuery 1.8 เอกสารระบุสถานะที่.ajaxStart/Stopควรแนบdocumentเท่านั้น สิ่งนี้จะแปลงตัวอย่างข้างต้นเป็น:

var $loading = $('#loadingDiv').hide();
$(document)
  .ajaxStart(function () {
    $loading.show();
  })
  .ajaxStop(function () {
    $loading.hide();
  });

48
นั่นอาจจะทั่วโลกเกินไปสำหรับการโหลดลงใน DIV
montrealist

363
ทั่วโลกด้วยเหรอ? คุณต้องการข้อความ "โปรดรอสักครู่" เท่าไร
nickf

23
วิธีนี้น่าเสียดายที่คุณไม่สามารถควบคุมการโหลดการวางตำแหน่ง div ในกรณีที่คุณไม่ต้องการแสดงหน้าต่างการโหลดแบบโมดัล แต่แสดงใกล้กับองค์ประกอบที่กำลังรอให้โหลดเนื้อหา ajax ใน ...
glaz666

10
ajaxStart และ ajaxStop เป็นกิจกรรม jQuery เพื่อให้คุณสามารถกำหนดชื่อพวกเขาได้: stackoverflow.com/questions/1191485/… docs.jquery.com/Namespaced_Events
David Xia

10
หากใช้ jQuery> = 1.9 ให้แนบเหตุการณ์ ajaxStart และ ajaxStop กับ $ (เอกสาร) jquery.com/upgrade-guide/1.9/…
Domenic

213

สำหรับ jQuery ฉันใช้

jQuery.ajaxSetup({
  beforeSend: function() {
     $('#loader').show();
  },
  complete: function(){
     $('#loader').hide();
  },
  success: function() {}
});

13
ได้ผลสำหรับฉัน html ควรมีลักษณะดังนี้: <div id = 'loader'> <img src = "spinner.gif" /> </div>
yigal

7
อันนี้สะอาดสำหรับฉัน แทนที่จะเป็น ajaxSetup ฉันวางไว้ก่อนส่ง: และสมบูรณ์: ภายในคำสั่ง $ ajax ({... คำสั่ง.
kenswdev

4
ไม่ทราบว่ามันมีค่าอะไร แต่ jQuery กล่าวถึงในเอกสารว่าไม่แนะนำให้ใช้. jaxSetup () ลิงก์
pec

หมายเหตุที่นี่ที่beforeSendถูกเรียกก่อนการโทรทุกครั้งในขณะที่ajaxStartถูกทริกเกอร์เมื่อเรียกเมธอด ajax แรกเท่านั้น ในทำนองเดียวกันajaxStopมันจะถูกเรียกเมื่อการโทร ajax ครั้งสุดท้ายเสร็จสิ้น หากคุณมีคำขอมากกว่าหนึ่งคำขอข้อมูลโค้ดในคำตอบนี้จะทำงานไม่ถูกต้อง
nickf

ไม่ทำงานสำหรับฉันหากการโทร ajax หมดเวลา (Firefox 28 สำหรับ Mac)
osa

48

คุณสามารถใช้.ajaxฟังก์ชั่นของ jQuery และใช้ตัวเลือกbeforeSendและกำหนดฟังก์ชั่นที่คุณสามารถแสดงบางอย่างเช่นตัวโหลดเดอร์และตัวเลือกความสำเร็จคุณสามารถซ่อนตัวโหลดเดอร์นั้นได้

jQuery.ajax({
    type: "POST",
    url: 'YOU_URL_TO_WHICH_DATA_SEND',
    data:'YOUR_DATA_TO_SEND',
    beforeSend: function() {
        $("#loaderDiv").show();
    },
    success: function(data) {
        $("#loaderDiv").hide();
    }
});

คุณสามารถมีรูปภาพ Spinning Gif นี่คือเว็บไซต์ที่เป็นตัวสร้าง AJAX Loader ที่ยอดเยี่ยมตามโทนสีของคุณ: http://ajaxload.info/


16
successจะไม่ถูกเรียกถ้ามีข้อผิดพลาด แต่ "คุณก็จะได้รับcompleteการติดต่อกลับแม้สำหรับการร้องขอการซิงโคร" ตามjQuery Ajax เหตุการณ์
LeeGee

23

คุณสามารถแทรกภาพเคลื่อนไหวใน DOM ได้ทันทีก่อนการโทร AJAX และทำฟังก์ชั่นอินไลน์เพื่อลบ ...

$("#myDiv").html('<img src="images/spinner.gif" alt="Wait" />');
$('#message').load('index.php?pg=ajaxFlashcard', null, function() {
  $("#myDiv").html('');
});

สิ่งนี้จะช่วยให้แน่ใจว่าแอนิเมชั่นของคุณจะเริ่มที่เฟรมเดียวกันกับคำขอที่ตามมา (หากมีความสำคัญ) โปรดทราบว่า IE เวอร์ชันเก่าอาจมีปัญหากับการเคลื่อนไหว

โชคดี!


13
เคล็ดลับยอดนิยม: โหลด spinner.gif ล่วงหน้าในชุดที่มีชุดการแสดงผลเป็นไม่ มิฉะนั้นคุณอาจได้รับเอฟเฟกต์สปินเนอร์ล่าช้าเนื่องจากสปินเนอร์ยังคงดาวน์โหลดอยู่
uriDium

เคล็ดลับดีง่ายกว่าการใช้ปลั๊กอินของคนอื่น
Gordon Thompson

ดี! แต่ที่รักมีอีกอย่างหนึ่งที่ฉันต้องการแสดงสปินเนอร์เป็นเวลา 2 วินาทีแม้หน้าเว็บจะโหลดไปแล้ว
ทั้ง FM

5
@BothFM เพราะผู้คนสนุกกับการรอสิ่งต่าง ๆ ... ?
Josh Stodola

21
$('#message').load('index.php?pg=ajaxFlashcard', null, showResponse);
showLoad();

function showResponse() {
    hideLoad();
    ...
}

http://docs.jquery.com/Ajax/load#urldatacallback


คุณอาจมีปัญหาเรื่องเวลาที่นั่นใช่ไหม
Tim Büthe

2
@ UltimateBrent ฉันคิดว่าคุณเข้าใจผิด ทำไม showLoad () ไม่ใช่การติดต่อกลับ JavaScript จะโหลดเนื้อหาแบบอะซิงโครนัส showLoad () จะทำงานก่อนที่เนื้อหาจะโหลดหากฉันแก้ไข
Jaseem

2
@Jaseem ฉันใช้วิธีที่คล้ายกันเช่นกันและจริง ๆ แล้วก็อาศัยการโทรแบบ async ฉันจะวางโชว์โหลดที่ไหนสักแห่งก่อนการโทร ajax แต่ประเด็นทั้งหมดคือการแสดงตัวหมุนให้ JS เริ่มการโทรและฆ่าตัวหมุนในการติดต่อกลับหลังจาก AJAX เสร็จสิ้น
Nenotlep

17

หากคุณกำลังใช้$.ajax()คุณสามารถใช้บางอย่างเช่นนี้:

$.ajax({
        url: "destination url",
        success: sdialog,
        error: edialog,
        // shows the loader element before sending.
        beforeSend: function () { $("#imgSpinner1").show(); },
        // hides the loader after completion of request, whether successfull or failor.             
        complete: function () { $("#imgSpinner1").hide(); },             
        type: 'POST', dataType: 'json'
    });  

11

ใช้ปลั๊กอินการโหลด: http://plugins.jquery.com/project/loading

$.loading.onAjax({img:'loading.gif'});

10
หนึ่งปลั๊กอินสำหรับงานง่ายๆนี้หรือไม่? นั่นไม่ใช่ทางออกที่ดีจริงๆเหรอ? ต้องการโหลด 100 สคริปต์ในหน้าใคร
Jaseem

2
ตกลง บีบอัดและต่อกันถ้าคุณต้องการประสิทธิภาพที่ดีขึ้น
Nathan Bubna

9
ไม่เห็นด้วย: แยกและทำเป็นโมดูลใช้รหัสซ้ำหากคุณต้องการรักษางานของคุณและให้ผู้อื่นทำเช่นนั้น โอกาสที่ผู้ใช้ของคุณมีการเชื่อมต่อที่ไม่ดีไคลเอ็นต์ของพวกเขาไม่สามารถโหลดปลั๊กอินและรหัสของคุณได้อย่างไร
LeeGee

@NathanBubna ลิงก์นั้นตาย
SpringLearner

"โอกาสที่ผู้ใช้ของคุณจะมีการเชื่อมต่อที่ไม่ดีไคลเอ็นต์ของพวกเขาไม่สามารถโหลดปลั๊กอินและรหัสของคุณได้" ค่อนข้างดีฉันจะบอกว่า
andrew oxenburgh

9

ตัวแปร: ฉันมีไอคอนที่มี id = "โลโก้" ที่ด้านบนซ้ายของหน้าหลัก จากนั้นสปินเนอร์ gif จะถูกซ้อนทับอยู่ด้านบน (ด้วยความโปร่งใส) เมื่ออาแจ็กซ์ทำงาน

jQuery.ajaxSetup({
  beforeSend: function() {
     $('#logo').css('background', 'url(images/ajax-loader.gif) no-repeat')
  },
  complete: function(){
     $('#logo').css('background', 'none')
  },
  success: function() {}
});

9

วันนี้ฉันมีสองการเปลี่ยนแปลงที่จะตอบกลับที่เดิม

  1. ในฐานะของ jQuery 1.8 ajaxStart และ ajaxStop documentควรจะยึดติดอยู่กับ สิ่งนี้ทำให้ยากต่อการกรองคำขอ ajax เพียงบางส่วนเท่านั้น ซู ...
  2. การสลับไปใช้ajaxSendและajaxCompleteทำให้สามารถตัดการร้องขอ ajax ปัจจุบันก่อนที่จะแสดงสปินเนอร์

นี่คือรหัสหลังจากการเปลี่ยนแปลงเหล่านี้:

$(document)
    .hide()  // hide it initially
    .ajaxSend(function(event, jqxhr, settings) {
        if (settings.url !== "ajax/request.php") return;
        $(".spinner").show();
    })
    .ajaxComplete(function(event, jqxhr, settings) {
        if (settings.url !== "ajax/request.php") return;
        $(".spinner").hide();
    })

วิธีแก้ปัญหาที่ดี ajaxCompleteดูเหมือนว่าจะยิงก่อนเวลาอันควรเมื่อมีการร้องขอลองใหม่อีกครั้งเพื่อให้ฉันใช้การรวมกันของajaxSendและajaxStopและการทำงานที่ดี
Mahn

@SchweizerSchoggi: มีหัวข้ออื่น ๆ ที่ตอบคำถามนั้นใช้การค้นหา!
Emil Stenström

8

ฉันต้องการมีส่วนร่วมในคำตอบนี้ ฉันกำลังมองหาสิ่งที่คล้ายกันใน jQuery และสิ่งนี้ในที่สุดฉันก็ใช้

ผมได้โหลดปั่นของฉันจากhttp://ajaxload.info/ ทางออกของฉันจะขึ้นอยู่กับคำตอบง่ายๆนี้ที่http://christierney.com/2011/03/23/global-ajax-loading-spinners/

โดยพื้นฐานมาร์กอัป HTML และ CSS ของคุณจะเป็นดังนี้:

<style>
     #ajaxSpinnerImage {
          display: none;
     }
</style>

<div id="ajaxSpinnerContainer">
     <img src="~/Content/ajax-loader.gif" id="ajaxSpinnerImage" title="working..." />
</div>

แล้วคุณรหัส jQuery จะมีลักษณะเช่นนี้:

<script>
     $(document).ready(function () {
          $(document)
          .ajaxStart(function () {
               $("#ajaxSpinnerImage").show();
          })
          .ajaxStop(function () {
               $("#ajaxSpinnerImage").hide();
          });

          var owmAPI = "http://api.openweathermap.org/data/2.5/weather?q=London,uk&APPID=YourAppID";
          $.getJSON(owmAPI)
          .done(function (data) {
               alert(data.coord.lon);
          })
          .fail(function () {
               alert('error');
          });
     });
</script>

มันเป็นเรื่องง่ายเหมือนที่ :)


8

คุณสามารถกำหนดอิมเมจโหลดเดอร์ให้กับแท็กเดียวกับที่คุณจะโหลดเนื้อหาโดยใช้การโทร Ajax:

$("#message").html('<span>Loading...</span>');

$('#message').load('index.php?pg=ajaxFlashcard');

คุณยังสามารถแทนที่แท็ก span ด้วยแท็กรูปภาพ


1
การจัดการ UI ด้วยวิธีนี้จะทำได้ยาก เราอาจต้องการสไตล์ที่แตกต่างอย่างสิ้นเชิงสำหรับข้อความ "การโหลด" และข้อความสุดท้าย เราสามารถใช้วิธีการโทรกลับที่กล่าวถึงข้างต้นเพื่อจัดการปัญหานี้
Jaseem

6

เช่นเดียวกับการตั้งค่าเริ่มต้นทั่วโลกสำหรับเหตุการณ์ ajax คุณสามารถกำหนดพฤติกรรมสำหรับองค์ประกอบเฉพาะ บางทีการเปลี่ยนชั้นเรียนของพวกเขาอาจจะเพียงพอหรือไม่

$('#myForm').ajaxSend( function() {
    $(this).addClass('loading');
});
$('#myForm').ajaxComplete( function(){
    $(this).removeClass('loading');
});

ตัวอย่าง CSS เพื่อซ่อน #myForm ด้วยสปินเนอร์:

.loading {
    display: block;
    background: url(spinner.gif) no-repeat center middle;
    width: 124px;
    height: 124px;
    margin: 0 auto;
}
/* Hide all the children of the 'loading' element */
.loading * {
    display: none;  
}

1
นี่ควรเป็นคำตอบที่เลือกในมุมมองของฉัน ขอบคุณ @LeeGee
Nam G VU

คำตอบที่ดีจริงๆ
Raskolnikov

4

โปรดทราบว่าคุณต้องใช้การเรียกแบบอะซิงโครนัสเพื่อให้สปินเนอร์ทำงาน (อย่างน้อยนั่นคือสิ่งที่ทำให้ฉันไม่แสดงจนกระทั่งหลังจากการโทร ajax และจากนั้นก็หายไปอย่างรวดเร็วขณะที่การโทรเสร็จสิ้นและนำสปินเนอร์ออก)

$.ajax({
        url: requestUrl,
        data: data,
        dataType: 'JSON',
        processData: false,
        type: requestMethod,
        async: true,                         <<<<<<------ set async to true
        accepts: 'application/json',
        contentType: 'application/json',
        success: function (restResponse) {
            // something here
        },
        error: function (restResponse) {
            // something here                
        }
    });

4
$('#loading-image').html('<img src="/images/ajax-loader.gif"> Sending...');

        $.ajax({
            url:  uri,
            cache: false,
            success: function(){
                $('#loading-image').html('');           
            },

           error:   function(jqXHR, textStatus, errorThrown) {
            var text =  "Error has occured when submitting the job: "+jqXHR.status+ " Contact IT dept";
           $('#loading-image').html('<span style="color:red">'+text +'  </span>');

            }
        });

ทางที่สั้นที่สุดจะใส่แท็กภาพที่มีภาพที่ดาวน์โหลดจากajaxload.infoกับพื้นหลังโปร่งใส
Izabela Skibinska

2

ฉันใช้สิ่งต่อไปนี้กับ jQuery UI Dialog (อาจใช้งานได้กับ ajax callbacks อื่น ๆ )

$('<div><img src="/i/loading.gif" id="loading" /></div>').load('/ajax.html').dialog({
    height: 300,
    width: 600,
    title: 'Wait for it...'
});

ประกอบด้วยการโหลด gif แบบเคลื่อนไหวจนกว่าเนื้อหาจะถูกแทนที่เมื่อการโทร ajax เสร็จสิ้น


2

นี่เป็นวิธีที่ดีที่สุดสำหรับฉัน:

jQuery :

$(document).ajaxStart(function() {
  $(".loading").show();
});

$(document).ajaxStop(function() {
  $(".loading").hide();
});

กาแฟ :

  $(document).ajaxStart ->
    $(".loading").show()

  $(document).ajaxStop ->
    $(".loading").hide()

Docs: ajaxStart , ajaxStop


ขวาและคุณสามารถใช้ร่วมกับปลั๊กอินที่กล่าวถึงในคำตอบนี้ถ้าคุณต้องการ ...
Matt


2

นี่เป็นปลั๊กอินที่เรียบง่ายและชาญฉลาดสำหรับวัตถุประสงค์เฉพาะนั้น: https://github.com/hekigan/is-loading


ดีจริงๆ! โดยเฉพาะอย่างยิ่งถ้าคุณดูการสาธิต ... ปิดการใช้งานองค์ประกอบ DOM, ในแท็ก, การซ้อนทับแบบเต็ม, การซ้อนทับองค์ประกอบ ... จะให้ทุกสิ่งที่คุณต้องการ และคุณสามารถรวมการซ้อนทับแบบเต็มกับคำตอบนี้เพื่อให้ทำงานได้ทันที
Matt

1

ฉันทำนี่:

var preloaderdiv = '<div class="thumbs_preloader">Loading...</div>';
           $('#detail_thumbnails').html(preloaderdiv);
             $.ajax({
                        async:true,
                        url:'./Ajaxification/getRandomUser?top='+ $(sender).css('top') +'&lef='+ $(sender).css('left'),
                        success:function(data){
                            $('#detail_thumbnails').html(data);
                        }
             });

1

ฉันคิดว่าคุณถูก. วิธีนี้เป็นสากลมากเกินไป ...

อย่างไรก็ตาม - มันเป็นค่าเริ่มต้นที่ดีเมื่อการโทร AJAX ของคุณไม่มีผลกระทบกับหน้า (บันทึกพื้นหลังเป็นตัวอย่าง) (คุณสามารถปิดการโทร ajax ได้ตลอดเวลาโดยผ่าน "global": false - ดูเอกสารที่jquery

เมื่อมีการโทร AJAX เพื่อรีเฟรชบางส่วนของหน้าฉันชอบภาพ "กำลังโหลด" ของฉันที่เฉพาะเจาะจงกับส่วนที่รีเฟรช ฉันต้องการดูว่าส่วนใดรีเฟรช

ลองนึกภาพว่าเท่ห์มากแค่ไหนถ้าคุณสามารถเขียนสิ่งที่ชอบ:

$("#component_to_refresh").ajax( { ... } ); 

และนี่จะแสดง "กำลังโหลด" ในส่วนนี้ ด้านล่างเป็นฟังก์ชั่นที่ผมเขียนที่จัดการกับการแสดง "กำลังโหลด" เช่นกัน แต่มันก็เฉพาะกับพื้นที่ที่คุณกำลังรีเฟรชในอาแจ็กซ์

ก่อนอื่นให้ฉันแสดงวิธีใช้

<!-- assume you have this HTML and you would like to refresh 
      it / load the content with ajax -->

<span id="email" name="name" class="ajax-loading">
</span>

<!-- then you have the following javascript --> 

$(document).ready(function(){
     $("#email").ajax({'url':"/my/url", load:true, global:false});
 })

และนี่คือฟังก์ชั่น - การเริ่มต้นขั้นพื้นฐานที่คุณสามารถปรับปรุงได้ตามที่คุณต้องการ มันยืดหยุ่นมาก

jQuery.fn.ajax = function(options)
{
    var $this = $(this);
    debugger;
    function invokeFunc(func, arguments)
    {
        if ( typeof(func) == "function")
        {
            func( arguments ) ;
        }
    }

    function _think( obj, think )
    {
        if ( think )
        {
            obj.html('<div class="loading" style="background: url(/public/images/loading_1.gif) no-repeat; display:inline-block; width:70px; height:30px; padding-left:25px;"> Loading ... </div>');
        }
        else
        {
            obj.find(".loading").hide();
        }
    }

    function makeMeThink( think )
    {
        if ( $this.is(".ajax-loading") )
        {
            _think($this,think);
        }
        else
        {
            _think($this, think);
        }
    }

    options = $.extend({}, options); // make options not null - ridiculous, but still.
    // read more about ajax events
    var newoptions = $.extend({
        beforeSend: function()
        {
            invokeFunc(options.beforeSend, null);
            makeMeThink(true);
        },

        complete: function()
        {
            invokeFunc(options.complete);
            makeMeThink(false);
        },
        success:function(result)
        {
            invokeFunc(options.success);
            if ( options.load )
            {
                $this.html(result);
            }
        }

    }, options);

    $.ajax(newoptions);
};


1

หากคุณวางแผนที่จะใช้ตัวโหลดทุกครั้งที่คุณทำการร้องขอเซิร์ฟเวอร์คุณสามารถใช้รูปแบบต่อไปนี้

 jTarget.ajaxloader(); // (re)start the loader
 $.post('/libs/jajaxloader/demo/service/service.php', function (content) {
     jTarget.append(content); // or do something with the content
 })
 .always(function () {
     jTarget.ajaxloader("stop");
 });

รหัสนี้ใช้ปลั๊กอิน jajaxloader (ซึ่งฉันเพิ่งสร้างขึ้น)

https://github.com/lingtalfi/JAjaxLoader/


1

รหัส ajax ของฉันมีลักษณะเช่นนี้ฉันเพิ่งได้คอมเม้นท์ async: false line และสปินเนอร์ปรากฏขึ้น

$.ajax({
        url: "@Url.Action("MyJsonAction", "Home")",
        type: "POST",
        dataType: "json",
        data: {parameter:variable},
        //async: false, 

        error: function () {
        },

        success: function (data) {
          if (Object.keys(data).length > 0) {
          //use data 
          }
          $('#ajaxspinner').hide();
        }
      });

ฉันแสดงตัวหมุนภายในฟังก์ชันก่อนโค้ด ajax:

$("#MyDropDownID").change(function () {
        $('#ajaxspinner').show();

สำหรับ Html ฉันใช้คลาสตัวอักษรที่ยอดเยี่ยม:

<i id="ajaxspinner" class="fas fa-spinner fa-spin fa-3x fa-fw" style="display:none"></i>

หวังว่าจะช่วยใครซักคน


0

คุณสามารถใช้ปลั๊กอิน jQuery Block UIซึ่งทำทุกอย่างให้คุณได้และมันยังบล็อกหน้าของอินพุตใด ๆ ในขณะที่ ajax กำลังโหลด ในกรณีที่ปลั๊กอินดูเหมือนว่าจะไม่ทำงานคุณสามารถอ่านเกี่ยวกับวิธีการใช้งานที่ถูกต้องในคำตอบนี้ ลองดูสิ

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.