我正在玩一个 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/