ChatGPT解决这个技术问题 Extra ChatGPT

JavaScript 窗口大小调整事件

如何挂钩浏览器窗口调整大小事件?

a jQuery way of listening for resize events,但我不希望仅出于这一要求将其带入我的项目。

感谢大家。我不在乎 IE,我更多的是考虑在手机上调整 Opera 的大小。

F
Frank Robert Anderson

最佳做法是附加到调整大小事件。

window.addEventListener('resize', function(event) {
    ...
}, true);

jQuery 只是包装了标准的 resize DOM 事件,例如。

window.onresize = function(event) {
    ...
};

jQuery 可能会做一些工作来确保在所有浏览器中一致地触发 resize 事件,但我不确定是否有任何浏览器不同,但我鼓励您在 Firefox、Safari 和 IE 中进行测试。


此方法的一个潜在问题是您正在覆盖事件,因此您无法将多个操作附加到事件。 addEventListener/attachEvent 组合是让您的 javascript 与可能在页面上执行的任何其他脚本友好播放的最佳方式。
var onresize = window.onresize; window.onresize = function(event) { if (typeof onresize === 'function') onresize(); /** ... */ }
window.addEventListener('resize', function(){}, true);
@SubOne 一个完美的例子,说明一个人永远不应该这样做。
ECMAScript 6window.onresize = () => { /* code */ }; 或更好:window.addEventListener('resize', () => { /* code */ });
J
Jondlm

首先,我知道上面的评论中提到了 addEventListener 方法,但我没有看到任何代码。由于它是首选方法,因此它是:

window.addEventListener('resize', function(event){
  // do stuff here
});

Here's a working sample


这是最直接的答案。
听起来像是最好的答案,因为您没有覆盖 window.onresize 事件:-)
许多人像您在此示例中所做的那样添加匿名事件侦听器,但这并不是一个很好的做法,因为这种方法会导致impossible稍后删除这些侦听器。这在过去没有问题,因为无论如何网页都是短暂的,但在当今 AJAX、RIA 和 SPA 的时代,它正在成为一体。我们需要一种管理事件监听器生命周期的方法。因此,最好将侦听器函数分解为一个单独的函数,以便以后可以使用 removeEventListener 将其删除。
为什么没有所有的绒毛,每个人都不能这样回答
我认为 Stijn de Witt 和 quemeful 的回答都是正确的。有时您关心删除事件侦听器,但在您不关心的情况下,添加上面示例中的所有额外代码是不必要的,并且会导致臃肿和可读性降低。在我看来,如果您没有想到需要删除侦听器的理由,那么这种方法是最好的解决方案。如有必要,您可以随时重写它。根据我的经验,这些需求只在特定情况下出现。
A
Alex V

永远不要覆盖 window.onresize 函数。

相反,创建一个函数以将事件侦听器添加到对象或元素。这会检查并以防监听器不起作用,然后它会覆盖对象的功能作为最后的手段。这是 jQuery 等库中使用的首选方法。

object:元素或窗口对象
type:调整大小、滚动(事件类型)
callback:函数引用

var addEvent = function(object, type, callback) {
    if (object == null || typeof(object) == 'undefined') return;
    if (object.addEventListener) {
        object.addEventListener(type, callback, false);
    } else if (object.attachEvent) {
        object.attachEvent("on" + type, callback);
    } else {
        object["on"+type] = callback;
    }
};

然后使用是这样的:

addEvent(window, "resize", function_reference);

或使用匿名函数:

addEvent(window, "resize", function(event) {
  console.log('resized');
});

这应该是公认的答案。覆盖 onresize 只是不好的做法。
哈,所有额外的空检查是怎么回事?试图在 IE 4.5 中工作还是什么?
您想知道如何调用此函数来接收窗口调整大小事件?方法如下: addEvent(window, "resize", function_reference); :)
请注意,从 IE 11 开始,不再支持 attachEvent。未来的首选方式是addEventListener
小心elem == undefined。 “未定义”可能(尽管不太可能)在本地定义为其他内容。 stackoverflow.com/questions/8175673/…
s
str

2018+ 解决方案:

您应该使用 ResizeObserver。它是一种浏览器原生解决方案,其性能比使用 resize 事件要好得多。此外,它不仅支持document上的事件,还支持任意elements上的事件。

var ro = new ResizeObserver( entries => { for (let entry of entries) { const cr = entry.contentRect; console.log('Element:', entry.target); console.log(`Element size: ${cr .width}px x ${cr.height}px`); console.log(`元素填充:${cr.top}px ; ${cr.left}px`); } }); // 观察一个或多个元素 ro.observe(someElement);

目前,Firefox, Chrome, Safari, and Edge support it。对于其他(和更旧的)浏览器,您必须使用 polyfill


f
frontsideup

永远不要直接使用 resize 事件,因为它会在我们调整大小时连续触发。

使用去抖动功能来减轻多余的调用。

window.addEventListener('resize',debounce(handler, delay, immediate),false);

这是一个常见的在网络上漂浮的反跳,尽管在 lodash 中寻找更高级的反跳。

const debounce = (func, wait, immediate) => {
    var timeout;
    return () => {
        const context = this, args = arguments;
        const later = function() {
            timeout = null;
            if (!immediate) func.apply(context, args);
        };
        const callNow = immediate && !timeout;
        clearTimeout(timeout);
        timeout = setTimeout(later, wait);
        if (callNow) func.apply(context, args);
    };
};

这可以像这样使用......

window.addEventListener('resize', debounce(() => console.log('hello'),
200, false), false);

它不会每 200 毫秒触发一次以上。

对于移动方向更改,请使用:

window.addEventListener('orientationchange', () => console.log('hello'), false);

这是我整理的一个small library,以巧妙地处理这个问题。


我很欣赏以下信息:去抖动(它现在似乎是一个非常宝贵的概念/功能,即使我以前拒绝以这种方式使用超时),lodash(虽然我个人不相信)并解决关于是否使用 addEvent 作为后备的歧义到 addEventListener (只是因为较旧的答案并没有真正明确地解决有关此的评论)
仅供参考,因为您使用的是 ES6 箭头函数表示法,因此内部函数需要具有 (...arguments) => {...} (或 ...args 以减少混淆),因为 arguments 变量不再适用于它们的范围。
我完全清楚,那个片段不是我的代码,它只是作为一个例子。
很好的答案,需要解决调整大小的性能问题!去抖动是一种选择,另一种是简单的节流(如果您需要 UI 以“步骤”调整大小,这可能是一种方法)。有关示例,请参见 bencentra.com/code/2015/02/27/optimizing-window-resize.html
W
WebWanderer

我确实相信@Alex V 已经提供了正确的答案,但答案确实需要一些现代化,因为它现在已经超过五年了。

主要有两个问题:

永远不要使用对象作为参数名称。这是一个保留词。话虽如此,@Alex V 提供的功能将无法在严格模式下工作。如果使用 addEventListener 方法,@Alex V 提供的 addEvent 函数不会返回事件对象。应将另一个参数添加到 addEvent 函数以允许这样做。

注意:addEvent 的新参数已设为可选,因此迁移到此新函数版本不会中断对该函数的任何先前调用。将支持所有旧用途。

以下是更新后的 addEvent 函数,其中包含这些更改:

/*
    function: addEvent

    @param: obj         (Object)(Required)

        -   The object which you wish
            to attach your event to.

    @param: type        (String)(Required)

        -   The type of event you
            wish to establish.

    @param: callback    (Function)(Required)

        -   The method you wish
            to be called by your
            event listener.

    @param: eventReturn (Boolean)(Optional)

        -   Whether you want the
            event object returned
            to your callback method.
*/
var addEvent = function(obj, type, callback, eventReturn)
{
    if(obj == null || typeof obj === 'undefined')
        return;

    if(obj.addEventListener)
        obj.addEventListener(type, callback, eventReturn ? true : false);
    else if(obj.attachEvent)
        obj.attachEvent("on" + type, callback);
    else
        obj["on" + type] = callback;
};

对新 addEvent 函数的示例调用:

var watch = function(evt)
{
    /*
        Older browser versions may return evt.srcElement
        Newer browser versions should return evt.currentTarget
    */
    var dimensions = {
        height: (evt.srcElement || evt.currentTarget).innerHeight,
        width: (evt.srcElement || evt.currentTarget).innerWidth
    };
};

addEvent(window, 'resize', watch, true);

我认为 attachEvent 在 2017 年不再相关。
@VladNicula 我同意,但这个答案已有两年历史了。
i
informatik01
window.onresize = function() {
    // your code
};

正如上面的许多评论所说,最好不要覆盖 onresize 函数;而是添加一个新事件。请参阅 Jondlm 的回答。
R
Rory O'Kane

以下博文可能对您有用:Fixing the window resize event in IE

它提供了以下代码:

Sys.Application.add_load(function(sender, args) { $addHandler(window, 'resize', window_resize); }); var resizeTimeoutId;函数 window_resize(e) { window.clearTimeout(resizeTimeoutId); resizeTimeoutId = window.setTimeout('doResizeCode();', 10); }


这一项仅适用于 ASP.NET 应用程序
r
rysama

如果您只想调整窗口和窗口的大小,上面已经提到的解决方案将起作用。但是,如果您想将调整大小传播到子元素,则需要自己传播事件。这是一些示例代码:

window.addEventListener("resize", function () {
  var recResizeElement = function (root) {
    Array.prototype.forEach.call(root.childNodes, function (el) {

      var resizeEvent = document.createEvent("HTMLEvents");
      resizeEvent.initEvent("resize", false, true);
      var propagate = el.dispatchEvent(resizeEvent);

      if (propagate)
        recResizeElement(el);
    });
  };
  recResizeElement(document.body);
});

请注意,子元素可以调用

 event.preventDefault();

在作为调整大小事件的第一个 Arg 传入的事件对象上。例如:

var child1 = document.getElementById("child1");
child1.addEventListener("resize", function (event) {
  ...
  event.preventDefault();
});

M
Mxyk
<script language="javascript">
    window.onresize = function() {
    document.getElementById('ctl00_ContentPlaceHolder1_Accordion1').style.height = '100%';
} 

</script>

T
THE AMAZING
var EM = new events_managment();

EM.addEvent(window, 'resize', function(win,doc, event_){
    console.log('resized');
    //EM.removeEvent(win,doc, event_);
});

function events_managment(){
    this.events = {};
    this.addEvent = function(node, event_, func){
        if(node.addEventListener){
            if(event_ in this.events){
                node.addEventListener(event_, function(){
                    func(node, event_);
                    this.events[event_](win_doc, event_);
                }, true);
            }else{
                node.addEventListener(event_, function(){
                    func(node, event_);
                }, true);
            }
            this.events[event_] = func;
        }else if(node.attachEvent){

            var ie_event = 'on' + event_;
            if(ie_event in this.events){
                node.attachEvent(ie_event, function(){
                    func(node, ie_event);
                    this.events[ie_event]();
                });
            }else{
                node.attachEvent(ie_event, function(){
                    func(node, ie_event);
                });
            }
            this.events[ie_event] = func;
        }
    }
    this.removeEvent = function(node, event_){
        if(node.removeEventListener){
            node.removeEventListener(event_, this.events[event_], true);
            this.events[event_] = null;
            delete this.events[event_];
        }else if(node.detachEvent){
            node.detachEvent(event_, this.events[event_]);
            this.events[event_] = null;
            delete this.events[event_];
        }
    }
}

K
Kamil Kiełczewski

您可以使用以下方法,即 ok for small projects

<body onresize="yourHandler(event)">

function yourHandler(e) { console.log('Resized:', e.target.innerWidth) } Content...(调整浏览器大小以查看)