元素加载后 JavaScript 就会绑定(bind)元素事件

标签 javascript jquery jquery-events

我想让 DOM 元素在加载后立即发生(在本网站或 Google 上搜索没有找到任何内容,但很难用几句话来解释这一点)。例如,当加载大页面时,我想隐藏/添加 onlick 事件等。一旦元素出现在用户屏幕上,而不是出现在 $(document).ready() 上,就立即对其进行处理。我写了一个简单的类(只是为了训练,闭包对我来说是新的,所以可能会有很多错误),它可以实现我想要的功能,但我想使用更好的商业用途(在我帮助开发的网站上)。这是我的源代码:

function MY_loader() {
    function container() {
        this.add_event = function(new_event,obj_selector,settings) {
            if(typeof(settings)!='object') {
                settings={};
            }
            if(typeof(settings.event_id)!='string') {
                settings.event_id=gen_new_event_id();
            }
            settings=$.extend({},default_settings,settings);
            settings.obj_selector=obj_selector;
            settings.event=new_event;
            events[settings.event_id]=settings;
        }

        this.execute_events = function(if_force) {
            if(typeof(if_force)=='undefined') {
                if_force=false;
            }
            if(html_is_loaded&&!if_force) {
                return;
            }
            var temp_obj;
            for(var event_name in events) {
                temp_obj=$(events[event_name].obj_selector);
                if(temp_obj.length || (html_is_loaded && events[event_name].if_force)) {
                    temp_obj.each(function() {
                        if(events[event_name].expect_multiple) {
                            if($(this).data('my_loader_'+events[event_name].event_id+'_executed')) {
                                
                                return;
                            }
                        }
                        if(events[event_name].event_type!='func') {
                            $(this).bind(events[event_name].event_type+'.'+events[event_name].event_id,events[event_name].event);
                        }
                        else {
                            events[event_name].event($(this));
                        }
                        if(events[event_name].expect_multiple) {
                            alert('here');
                            $(this).data('my_loader_'+events[event_name].event_id+'_executed',1);
                        }
                    });
                       //alert(events[event_name].obj_selector+' '+events[event_name].event_type);
                    if(!events[event_name].expect_multiple) {
                        delete events[event_name];
                    }
                }
            }
            if(!html_is_loaded) {
               var cur_time=new Date().getTime();
               setTimeout('MY_loader().execute_events();',Math.max(Math.min(tick_time-(cur_time-last_tick_time),tick_time),0,min_tick_diff));
               last_tick_time=cur_time;
            }
        }

        this.html_is_loaded_set=function(if_html_is_loaded) {
            html_is_loaded=if_html_is_loaded?true:false;
        };

        this.html_is_loaded_get=function() {
            return html_is_loaded?true:false;
        };
        
        return this;
    }
    function instance(if_strat) {
        if(typeof(class_is_loaded)=='undefined'||!class_is_loaded) {
            load_class();
        }
        return container(if_strat);
    }

    var load_class=function() {
        this.class_is_loaded=true;
        this.events = {};
        this.allowed_event_id_chars='abcdefghijklmnopqrstuvwxyz0123456789';
        this.default_settings={
            'event_type':'click',
            'if_force':false,
            'expect_multiple':false
        };
        this.tick_time=500;
        this.min_tick_diff=100;
        this.last_tick_time=0;
        this.html_is_loaded=false;
        MY_loader().execute_events();
        $(document).ready(function(){
            MY_loader().html_is_loaded_set(true);
            MY_loader().execute_events(true);
        });
    }

    var gen_new_event_id=function() {
        for(var new_id=gen_random_id();typeof(events[new_id])!='undefined';new_id=gen_random_id());
        return new_id;
    }

    var gen_random_id=function() {
        var allowed_event_id_chars_size=allowed_event_id_chars.length;
        var new_id='';
        for(var i=0;i<10;i++) {
            new_id+=allowed_event_id_chars[get_random_int(0,allowed_event_id_chars_size-1)];
        }
        return new_id;
    }

    function get_random_int(min, max)
    {
        return Math.floor(Math.random() * (max - min + 1)) + min;
    }

    return new instance();
}
//Add click event to #some_selector (once and dont check after it)
MY_loader().add_event(function() {
    alert($(this).val());
}, '#some_selector');
//Hide elements as soon as they are loaded.
//We expect multiple elements with this selector, so it will
//check for this elements untill document is loaded, but this function
//will be applied only one time for each element.
MY_loader().add_event(function(obj) {
    obj.hide();
    alert('This should happen before DOM is completely loaded!');
}, '.some_other_selector',{'event_type':'func','expect_multiple':true});
//This alert should be outputted the last
$(document).ready(function(){
    alert('Document is fully loaded!');
});

UPD:为了让这个问题变得更有趣,因为它看起来太具体了,我必须添加:大多数浏览器在完全加载之前开始页面渲染(由于某种原因这似乎并不为人所知),在这里有几个链接:

  1. http://en.wikipedia.org/wiki/Incremental_rendering
  2. When do browsers start to render partially transmitted HTML?
  3. http://www.vbulletin.org/forum/showthread.php?t=161099

所以,我的问题应该更广为人知,因为这种增量页面加载给开发人员带来了很多问题:用户可以在页面加载之前单击不起作用的按钮,加载时 JavaScript 隐藏的一些内容可能会显示出来,并且它们可能很难看,等等(这是我的问题的例子)。其他开发者会忽略这个问题吗?如果我错了,请纠正我。

最佳答案

如果您担心不希望用户在最终页面之前看到该页面,您可能可以直接进行 DOM 修改,例如:

<a href="#" id="test">Test</a>
<script type="text/javascript" src="enhancePage.js"></script>

由于在解析脚本之前相关节点已存在于 DOM 中,因此您可以使用它。

关于元素加载后 JavaScript 就会绑定(bind)元素事件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6857148/

相关文章:

javascript - 如何将事件监听器添加到 jEditable 输入?

javascript - Div contenteditable 并在 FF 上按 "enter"

JQuery 可在溢出内拖动 : auto div is confined within the div

javascript - 删除在一个下拉列表字段中选择的选项,并且不显示在该行中存在的其他下拉列表中不工作 - jquery

javascript - 简单的变量赋值不起作用

javascript - jQuery 单击/更改事件在 IE7/8 中无法正常工作

javascript - 如何在 react 中获取对象中的嵌套数组?

javascript - jQuery Img 切换 Ajax 帖子

Javascript - 加载多个函数 onLoad

javascript - 检查元素上是否存在事件