Problemas com vários setIntervals em execução simultaneamente
-
26-09-2019 - |
Pergunta
Meu primeiro post aqui. Quero fazer um menu horizontal com o submenu deslizando para baixo no MouseOver. Eu sei que poderia usar o jQuery, mas isso é para praticar minhas habilidades de JavaScript.
Eu uso o seguinte código:
var up = new Array()
var down = new Array()
var submenustart
function titleover(headmenu, inter)
{
submenu = headmenu.lastChild
up[inter] = window.clearInterval(up[inter])
down[inter] = window.setInterval("slidedown(submenu)",1)
}
function slidedown(submenu)
{
if(submenu.offsetTop < submenustart)
{
submenu.style.top = submenu.offsetTop + 1 + "px"
}
}
function titleout(headmenu, inter)
{
submenu = headmenu.lastChild
down[inter] = window.clearInterval(down[inter])
up[inter] = window.setInterval("slideup(submenu)", 1)
}
function slideup(submenu)
{
if(submenu.offsetTop > submenustart - submenu.clientHeight + 1)
{
submenu.style.top = submenu.offsetTop - 1 + "px"
}
}
A variável submenustart é nomeada um valor em outra função que não é relevante para minha pergunta.
HTML se parece com o seguinte:
<table class="hoofding" id="hoofding">
<tr>
<td onmouseover="titleover(this, 0)" onmouseout="titleout(this, 0)"><a href="#" class="hoofdinglink" id="hoofd1">AAAA</a>
<table class="menu">
<tr><td><a href="...">1111</a></td></tr>
<tr><td><a href="...">2222</a></td></tr>
<tr><td><a href="...">3333</a></td></tr>
</table></td>
<td onmouseover="titleover(this, 1)" onmouseout="titleout(this, 1)"><a href="#" class="hoofdinglink">BBBB</a>
<table class="menu">
<tr><td><a href="...">1111</a></td></tr>
<tr><td><a href="...">2222</a></td></tr>
<tr><td><a href="...">3333</a></td></tr>
<tr><td><a href="...">4444</a></td></tr>
<tr><td><a href="...">5555</a></td></tr>
</table></td>
...
</tr>
</table>
O que acontece é o seguinte:
Se eu passar e sair (para o menu ex) A funcionar bem. Se eu passar agora no menu B, o intervalo aplicado a A agora é aplicado a B. Agora existem 2 funções de intervalo aplicadas a B. Aque uma originalmente para A e um novo desencadeado pelo MouseOver em B. se eu fosse para um Todos os intervalos agora são aplicados a A.
Eu tenho procurado horas, mas estou completamente preso.
Desde já, obrigado.
Solução
Minha primeira sugestão é
- Pratique usar semicolons corretamente e não confiar na inserção de semicolon
- Não passe funções como cordas (que requer um implícito
eval
) - Use JavaScript para anexar eventos de uma maneira não obra
.
var up = [],
down = [],
submenustart;
function titleover(headmenu, inter) {
up[inter] = window.clearInterval(up[inter]);
down[inter] = window.setInterval(function() { slidedown(headmenu.lastChild); }, 1);
}
function slidedown(submenu) {
if ( submenu.offsetTop < submenustart ) {
submenu.style.top = submenu.offsetTop + 1 + "px";
}
}
function titleout(headmenu, inter) {
down[inter] = window.clearInterval(down[inter]);
up[inter] = window.setInterval(function() { slideup(headmenu.lastChild); }, 1);
}
function slideup(submenu) {
if ( submenu.offsetTop > submenustart - submenu.clientHeight + 1 ) {
submenu.style.top = submenu.offsetTop - 1 + "px";
}
}
Minha segunda sugestão é não usar tabelas para menus, pois não são dados tabulares. Em vez disso, use uma lista não classificada.
<ul class="hoofding" id="hoofding">
<li onmouseover="titleover(this, 0)" onmouseout="titleout(this, 0)">
<a href="#" class="hoofdinglink" id="hoofd1">AAAA</a>
<ul class="menu">
<li><a href="...">1111</a></li>
<li><a href="...">2222</a></li>
<li><a href="...">3333</a></li>
</ul>
</li>
<li onmouseover="titleover(this, 1)" onmouseout="titleout(this, 1)">
<a href="#" class="hoofdinglink">BBBB</a>
<ul class="menu">
<li><a href="...">1111</a></li>
<li><a href="...">2222</a></li>
<li><a href="...">3333</a></li>
</ul>
</li>
</ul>
Outras dicas
Afaik, (e de acordo com a referência de Gecko DOM) Window.ClearInterval () não retorna nada, o que talvez a razão que estrague as coisas.