Navigationsunterpunkte automatisch schliessen bei klick auf einen Hauptpunkt

paprika

New member
Hallo zusammen :)

Ich habe auf selfhtml ein super Skript für eine ausklappbare Navigation gefunden. Die Anpassung per CSS klappte super. Nur leider bleibt die Unternavigation offen, ausser man klickt wieder auf den selben Hauptmenüpunkt. Es wäre genial, wenn die Unternavigation schliessen würde, sobald man auf irgendeinen anderen Hauptmenüpunkt klicken würde... so, dass immer nur eine Unternavigation offen ist. Da ich mich sehr schlecht mit JS auskenne und meine Versuchungen bisher erfolglos waren, hoffe ich ihr könnt mir weiterhelfen...

Hier noch das original Skript von selfhtml:

Code:
<html>
<head>
<title>Beispiel</title>
<script type="text/javascript">
  /*
   * Fügt den Listeneinträgen Eventhandler und CSS Klassen hinzu,
   * um die Menüpunkte am Anfang zu schließen.
   *
   * menu: Referenz auf die Liste.
   * data: String, der die Nummern aufgeklappter Menüpunkte enthält.
   */
  function treeMenu_init(menu, data) {
    var array = new Array(0);
    if(data != null && data != "") {
      array = data.match(/\d+/g);
    }
    var items = menu.getElementsByTagName("li");
    for(var i = 0; i < items.length; i++) {
      items[i].onclick = treeMenu_handleClick;
      if(!treeMenu_contains(treeMenu_getClasses(items[i]), "treeMenu_opened")
          && items[i].getElementsByTagName("ul").length
            + items[i].getElementsByTagName("ol").length > 0) {
        var classes = treeMenu_getClasses(items[i]);
        if(array.length > 0 && array[0] == i) {
          classes.push("treeMenu_opened")
        }
        else {
          classes.push("treeMenu_closed")
        }
        items[i].className = classes.join(" ");
        if(array.length > 0 && array[0] == i) {
          array.shift();
        }
      }
    }
  }

  /*
   * Ändert die Klasse eines angeclickten Listenelements, sodass
   * geöffnete Menüpunkte geschlossen und geschlossene geöffnet
   * werden.
   *
   * event: Das Event Objekt, dass der Browser übergibt.
   */
  function treeMenu_handleClick(event) {
    if(event == null) { //Workaround für die fehlenden DOM Eigenschaften im IE
      event = window.event;
      event.currentTarget = event.srcElement;
      while(event.currentTarget.nodeName.toLowerCase() != "li") {
        event.currentTarget = event.currentTarget.parentNode;
      }
      event.cancelBubble = true;
    }
    else {
      event.stopPropagation();
    }
    var array = treeMenu_getClasses(event.currentTarget);
    for(var i = 0; i < array.length; i++) {
      if(array[i] == "treeMenu_closed") {
        array[i] = "treeMenu_opened";
      }
      else if(array[i] == "treeMenu_opened") {
        array[i] = "treeMenu_closed"
      }
    }
    event.currentTarget.className = array.join(" ");
  }

  /*
   * Gibt alle Klassen zurück, die einem HTML-Element zugeordnet sind.
   *
   * element: Das HTML-Element
   * return: Die zugeordneten Klassen.
   */
  function treeMenu_getClasses(element) {
    if(element.className) {
      return element.className.match(/[^ \t\n\r]+/g);
    }
    else {
      return new Array(0);
    }
  }

  /*
   * Überprüft, ob ein Array ein bestimmtes Element enthält.
   *
   * array: Das Array
   * element: Das Element
   * return: true, wenn das Array das Element enthält.
   */
  function treeMenu_contains(array, element) {
    for(var i = 0; i < array.length; i++) {
      if(array[i] == element) {
        return true;
      }
    }
    return false;
  }

  /*
   * Gibt einen String zurück, indem die Nummern aller geöffneten
   * Menüpunkte stehen.
   *
   * menu: Referenz auf die Liste
   * return: Der String
   */
  function treeMenu_store(menu) {
    var result = new Array();;
    var items = menu.getElementsByTagName("li");
    for(var i = 0; i < items.length; i++) {
      if(treeMenu_contains(treeMenu_getClasses(items[i]), "treeMenu_opened")) {
        result.push(i);
      }
    }
    return result.join(" ");
  }
</script>
<style type="text/css">
  li.treeMenu_opened ul {
    display: block;
  }
  li.treeMenu_closed ul {
    display: none;
  }
</style>
<body onload="treeMenu_init(document.getElementById('menu'), '')">
<ul id="menu">
  <li>erstens
    <ul>
      <li>A</li>
      <li>B</li>
    </ul>
  </li>
  <li>zweitens
    <ul>
      <li>a</li>
      <li>b</li>
    </ul>
  </li>
</ul>
</body>
</html>
 
Einen Code habe ich leider noch nicht :disturbed: das Script ist im Moment nur voller Kommentare - ich bin schon froh, wenn ich den Code grösstenteils nachvollziehen kann. Eigentlich sollte man, dann keinen fremden Code kopieren... :eek: aber vielleicht kannst du mir trotzdem bei einem Ansatz helfen - bzw. mir sagen ob meine Überlegungen überhaupt Sinn ergeben/möglich sind:

Bei dem untenstehenden Code wird ja abgefragt, ob die Klasse treeMenu_cosed oder treeMenu_opened besteht und anschliessend geändert. Wäre es möglich, dass die Abfrage so lauten würde:
Wenn die aktuell ausgewählte Klasse gleich treeMenu_closed, dann schliesse alle Klassen mit dem Inhalt treeMenu und öffne das aktuell ausgewählte Element, wenn treeMenu_opened, dann auf treeMenu_closed wechseln.

Code:
    var array = treeMenu_getClasses(event.currentTarget);
    for(var i = 0; i < array.length; i++) {
      if(array[i] == "treeMenu_closed") {
        array[i] = "treeMenu_opened";
      }
      else if(array[i] == "treeMenu_opened") {
        array[i] = "treeMenu_closed"
      }
    }

Könnte man
Code:
 var items = navigation.getElementsByTagName("li");
dafür verwenden?

Wie du siehst, ich habe leider wirklich keine Ahnung :S
 
Ja, du könntest durch alle Einträge in items durchgehen und prüfen, ob eines offen ist.

Mein Ansatz wäre, dass du dir beim Öffnen einfach irgendwo merkst, welches Element du gerade geöffnet hast und beim nächsten Öffnen schließt du dieses Element und merkst dir das Neue...
 
Vielen Dank :) ich werde es in der nächsten Zeit versuchen. In der Zwischenzeit habe ich festgestellt, dass es mit jquery viel kürzer geht...
Code:
	$("#navigation li").click(function(){
		$(".menu").removeClass("treeMenu_opened");
		$(".menu").toggleClass("treeMenu_closed");
		$(this).removeClass("treeMenu_closed");
		$(this).addClass("treeMenu_opened");
	});

});
wahrscheinlich noch optimierungsbedürftig aber für diese Nacht reichts ;)

Gute Nacht.
 
Zurück
Oben