gpt4 book ai didi

javascript - Vanilla Javascript 在窗口调整大小时操作元素

转载 作者:塔克拉玛干 更新时间:2023-11-02 20:42:33 24 4
gpt4 key购买 nike

我正在玩一个 Javascript 选项卡系统。

当窗口宽度低于某个断点时,选项卡内容将插入 Accordion 布局的链接之后,并且当您增加窗口宽度时,该过程相反。

当您减小窗口大小时,一切都按预期工作,但随着您增加窗口宽度而中断。以下代码导致了问题 tabBody.appendChild( tabBodyItems[k] );

3 个 .c-Tabs_BodyItem 元素中只有 2 个追加???

HTML

<div class="c-Tabs" id="js-Tabs">
<div class="o-Container">
<div class="o-Row c-Tabs_Head">
<div class="o-Col-sm-4 c-Tabs_HeadItem">
<a href="" class="c-Tabs_Link">Link 1</a>
</div>
<div class="o-Col-sm-4 c-Tabs_HeadItem is-active">
<a href="" class="c-Tabs_Link">Link 2</a>
</div>
<div class="o-Col-sm-4 c-Tabs_HeadItem">
<a href="" class="c-Tabs_Link">Link 3</a>
</div>
</div>
<div class="c-Tabs_Body">
<div class="c-Tabs_BodyItem">
<ul>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
</ul>
</div>
<div class="c-Tabs_BodyItem is-active">
<ul>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
</ul>
</div>
<div class="c-Tabs_BodyItem">
<ul>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
</ul>
</div>
</div>
</div>
</div>

Javascript

我不会显示实际标签系统的 Javascript;这不是导致问题的原因。

function insertAfter( el, referenceNode )
{
referenceNode.parentNode.insertBefore( el, referenceNode.nextSibling );
}

function debounce( func, wait, immediate )
{
var timeout;

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

var tabHeadItems = document.getElementsByClassName( 'c-Tabs_HeadItem' );
var tabBody = document.querySelector( '.c-Tabs_Body' );
var tabBodyItems = document.getElementsByClassName( 'c-Tabs_BodyItem' );
var isTabs = true;

function setTab()
{
if ( window.matchMedia( '(min-width: 768px)' ).matches && isTabs === false )
{
for( var k = 0, lenK = tabBodyItems.length; k < lenK; k++ )
{
tabBody.appendChild( tabBodyItems[k] );
}

isTabs = true;
}
else if ( window.matchMedia( '(max-width: 767px)' ).matches && isTabs === true )
{
for( var l = 0, lenL = tabBodyItems.length; l < lenL; l++ )
{
insertAfter( tabBodyItems[l], tabHeadItems[l] );
}

isTabs = false;
}
}

setTab();

var debounceSetTab = debounce(function() {
setTab();
}, 250 );

window.addEventListener( 'resize', debounceSetTab );

最佳答案

原因是 getElementsByClassName 返回一个live NodeList。这意味着随着 DOM 的变化,NodeList 的内容也会发生变化以反射(reflect)它。元素 tabBodyItems 的顺序基于它们在 DOM 中的位置,因此当您在 DOM 中移动元素时,它们在 tabBodyItems 中的索引会发生变化以反射(reflect)这一点。当您在 for 循环中遍历集合时会发生这种情况,因此您将跳过元素,因为它们会重新编号。

解决此问题的最简单方法是使用 document.querySelectorAll 而不是 document.getElementsByClassName。这将返回一个静态的 NodeList

function insertAfter( el, referenceNode )
{
referenceNode.parentNode.insertBefore( el, referenceNode.nextSibling );
}

function debounce( func, wait, immediate )
{
var timeout;

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

var tabHeadItems = document.querySelectorAll( '.c-Tabs_HeadItem' );
var tabBody = document.querySelector( '.c-Tabs_Body' );
var tabBodyItems = document.querySelectorAll( '.c-Tabs_BodyItem' );
var isTabs = true;

function setTab()
{
if ( window.matchMedia( '(min-width: 768px)' ).matches && isTabs === false )
{
for( var k = 0, lenK = tabBodyItems.length; k < lenK; k++ )
{
tabBody.appendChild( tabBodyItems[k] );
}

isTabs = true;
}
else if ( window.matchMedia( '(max-width: 767px)' ).matches && isTabs === true )
{
for( var l = 0, lenL = tabBodyItems.length; l < lenL; l++ )
{
insertAfter( tabBodyItems[l], tabHeadItems[l] );
}

isTabs = false;
}
}

setTab();

var debounceSetTab = debounce(function() {
setTab();
}, 250 );

window.addEventListener( 'resize', debounceSetTab );
<div class="c-Tabs" id="js-Tabs">
<div class="o-Container">
<div class="o-Row c-Tabs_Head">
<div class="o-Col-sm-4 c-Tabs_HeadItem">
<a href="" class="c-Tabs_Link">Link 1</a>
</div>
<div class="o-Col-sm-4 c-Tabs_HeadItem is-active">
<a href="" class="c-Tabs_Link">Link 2</a>
</div>
<div class="o-Col-sm-4 c-Tabs_HeadItem">
<a href="" class="c-Tabs_Link">Link 3</a>
</div>
</div>
<div class="c-Tabs_Body">
<div class="c-Tabs_BodyItem" id="body1">
<ul>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
</ul>
</div>
<div class="c-Tabs_BodyItem is-active" id="body2">
<ul>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
</ul>
</div>
<div class="c-Tabs_BodyItem"id="body3">
<ul>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
<li>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</li>
</ul>
</div>
</div>
</div>
</div>

关于javascript - Vanilla Javascript 在窗口调整大小时操作元素,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39282619/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com