Matthias Scharwies: ToDo-Liste - Vorüberlegungen

Guten Abend!

Ich würde gerne im JavaScript und das DOM-Tutorial das Kapitel zur DOM-Manipulation neu schreiben. [1]

Als kontextualisiertes Beispiel hat mir @Rolf B eine ToDo-Liste als klassisches Beispiel empfohlen (schon länger her):[2]

  • Listeneinträge erzeugen und löschen
  • textContent verändern
  • Klassen und Attribute hinzufügen und entfernen.

Die Grundstruktur scheint klar:

<ul id="taskList">
  <li>Im Wiki aktiv werden <button>Close</button></li>
  <li aria-selected="true">Spende an SELFHTML e.V. <button>Close</button></li> 
  ...
</ul>

Schwierig wird's, wenn man Listenelemente anklicken will, um sie als erledigt zu markieren. Wie würdet ihr das machen? Am Liebsten wäre es mir, wenn das "0" da stünde, man aber den gesamten Text anklicken dürfte und das "✓" dann als Pseudoelement erschiene.

Ist dieser Versuch Overkill?

<ul id="taskList">
  <li contenteditable="true">
    <span>im Wiki aktiv werden</span>
    <button aria-label="Erledigt">O</button>
    <button aria-label="Schließen">X</button>
  </li>
  <li contenteditable="true" aria-selected="true">
    <span>Spende an SELFHTML e.V. </span>
    <button aria-label="Erledigt">O</button>
    <button aria-label="Schließen">X</button>
  </li>
</ul>

Ist es überhaupt sinnvoll, Einträge nachträglich ändern zu dürfen?

Sollte man in einem zweiten Schritt aus der ul eine ol machen und Einträge nach oben oder unten schieben können?

Es soll einerseits für Anfänger gedacht sein (createElement, setAttribute, classList), andererseits auch ein vollwertiges Beispiel sein.

Herzliche Grüße
Matthias Scharwies

--
Was ist eine Signatur?

  1. Der Bild-Wechsler war da passend, wurde in der Komfort-Variante aber zu komplex für ein Anfänger-Kapitel zu DOM-Manipulation. ↩︎

  2. Dieses Beispiel aus 2015 enthält klickbare li-Elemente und sollte dann depubliziert werden. ↩︎

  1. Hallo Matthias,

    war das meine Idee? Oh je, mein Gedächtnis...

    zu markieren. Wie würdet ihr das machen

    Mit einer Checkbox, die ein Label "Erledigt" hat. Gerne in Leserichtung davor, mit dem Label visuell versteckt.

    Dass der Klick auf das Todo-Item es als erledigt markiert, finde ich schlecht verständlich für jemanden, der Assistenztechnik nutzt.

    Bearbeitung bitte nicht im Dauermodus, sondern mit einem Bearbeiten-Button. Der kann den Eintrag dann durch ein input-Element ersetzen, mit "OK" und "Abbrechen" als Buttons. Das könnte auch ein Popup (ein Dialog) sein - aber das ist vermutlich zu komplex für den Einstieg. "Bearbeiten" könnte man für das Beispiel als progressive enhancement verkaufen. Wenn man Löschen und Hinzufügen kann, reicht das. Find ich.

    Die Todo-Liste ist eine Gratwanderung zwischen UX und Anfängerfreundlichkeit.

    Phase 1:

    • Hinzufügen durch Eintragen in ein Input und Button "Hinzufügen"
    • Als erledigt markieren durch einen eigenen Button "Erledigt". Der löscht dann auch sofort

    "Todo-Liste Speichern" haben wir - meine ich - beim localStorage. Das sollte mn als Funktion im Beispiel als Tool drin haben, für Erklärungen aber zum localStorage verweisen.

    Phase 2:

    • Trennen von Erledigt und Löschen. Hier kann man auf Eventhandling in einer Liste eingehen (d.h. Bubbling ausnutzen)
    • Up/Down Buttons

    Phase 3:

    • Bearbeiten per Dialog

    Das wäre mein Plan. Ob der ideal ist, überlasse ich didaktisch gebildeteren Leuten 😉. Pass nur auf, dass Du nicht versehentlich Hieroglyphen verwendest, alter Ägypter!

    Rolf

    --
    sumpsi - posui - obstruxi
    1. Servus!

      Das wäre mein Plan. Ob der ideal ist, überlasse ich didaktisch gebildeteren Leuten 😉.

      Ich hirne mal - mir bleibt auch gar nix anderes übrig, bei den schlechten Datenraten. :-)

      Pass nur auf, dass Du nicht versehentlich Hieroglyphen verwendest, alter Ägypter!

      Sind jetzt auf der Rückreise nilabwärts. Das Schiff ist doppelt so schnell wie stromaufwärts, der Wind bläst einen in die Kabine.

      Herzliche Grüße

      Matthias Scharwies

      --
      Was ist eine Signatur?
  2. @@Matthias Scharwies

    ToDo-Liste […]
    Schwierig wird's, wenn man Listenelemente anklicken will, um sie als erledigt zu markieren. Wie würdet ihr das machen?

    Ich würde als erstes nachschauen, wie es Heydon in seinen Inclusive Components gemacht hat, worin ja auch eine Todo-Liste zu finden ist.

    Kwakoni Yiquan

    --
    Ad astra per aspera
  3. Hi,

    Schwierig wird's, wenn man Listenelemente anklicken will, um sie als erledigt zu markieren. Wie würdet ihr das machen? Am Liebsten wäre es mir, wenn das "0" da stünde, man aber den gesamten Text anklicken dürfte und das "✓" dann als Pseudoelement erschiene.

    0 heißt "erledigt"??? Da kommt doch keiner drauf.

    Ich würde für Erledigt eine Checkbox erwarten.

    Ist es überhaupt sinnvoll, Einträge nachträglich ändern zu dürfen?

    sollen Tippfehler nicht korrigiert werden dürfen?

    Oder: Todo-Eintrag: "6 Eier kaufen". Dann ruft das Partner an und sagt, nimm lieber 10". Da würd ich dann lieber den einen Eintrag anpassen, als einen zweiten "4 Eier kaufen" anzulegen.

    Sollte man in einem zweiten Schritt aus der ul eine ol machen und Einträge nach oben oder unten schieben können?

    ja, die Prioritäten können sich ja ändern.

    cu,
    Andreas a/k/a MudGuard

    1. @@MudGuard

      Oder: Todo-Eintrag: "6 Eier kaufen".

      „Und wenn sie Eier haben, bring 6 Stück mit.“ 😆

      Kwakoni Yiquan

      --
      Ad astra per aspera
      1. Hi,

        @@MudGuard

        Oder: Todo-Eintrag: "6 Eier kaufen".

        „Und wenn sie Eier haben, bring 6 Stück mit.“ 😆

        genau das hatte ich im Hinterkopf …

        cu,
        Andreas a/k/a MudGuard

  4. Lieber Matthias,

    das DOM für so eine ToDo-Liste würde ich so angehen:

    <ul id="task-list">
      <li>
        <input id="task-1" name="task-1" type="checkbox">
        <button name="delete">löschen</button>
        <button name="edit">bearbeiten</button>
        <label for="task-1">im Wiki aktiv werden</label>
      </li>
      <li>
        <input id="task-2" name="task-2" type="checkbox">
        <button name="delete">löschen</button>
        <button name="edit">bearbeiten</button>
        <label for="task-2">Spende an SELFHTML e.V.</label>
      </li>
      <li>
        <button id="add">neuen Eintrag hinzufügen</button>
      </li>
    </ul>
    

    [jsFiddle]

    Mit diesem DOM kann man den „Checkbox-Hack“ verwenden, um erledigte Tasks entsprechend anders darzustellen. Auch die Buttons kann man dann (wahlweise) unsichtbar schalten.

    Für alles, was den Eintrag inhaltlich verändert, braucht es einen passenden Button. Das Abhaken sollte eine Checkbox sein.

    Das mit contenteditable würde bedeuten, dass man das DOM innerhalb eines Elements verändern kann. Das willst Du bei einer ToDo-Liste nicht, denn sonst könnte man das <li> mit beliebigen Kindelementen befüllen, die ihrerseits Listen, Tabellen und noch viel mehr sein können. Und nur damit der User etwas kursiv und fett formatieren kann... würde ich unbedingt davon abraten.

    Liebe Grüße

    Felix Riesterer

    1. Guten Morgen,

      es geht ja um ein DOM-Tutorial, in dem man anhand eines Beispiels die meisten dieser Punkte abarbeitet:

      1. Selecting Elements
      2. Creating and Inserting Elements
        document.createElement(tagName)
        parentElement.appendChild(childElement)
        element.insertAdjacentHTML(position, html)
      3. Changing Content
        element.textContent
      4. Changing Attributes
        element.setAttribute(attribute, value)
      5. Traversing the DOM

      Mit diesem DOM kann man den „Checkbox-Hack“ verwenden, um erledigte Tasks entsprechend anders darzustellen.

      Und genau so etwas würde ich mit [aria-selected]{ background: green;} realisieren.

      Auch die Buttons kann man dann (wahlweise) unsichtbar schalten.

      Das war meine Überlegung: Anstelle des li-Inhalts einen Button mit dem Tasttext, da gäbe es aber keinen Hinweis, dass man ihn anklicken solle, um ihn als erledigt abzuhaken.

      Für alles, was den Eintrag inhaltlich verändert, braucht es einen passenden Button. Das Abhaken sollte eine Checkbox sein.

      Genau. So mache ich es. Wobei ich das Editieren wohl weglasse, und links die Checkbox und rechts den Löschen-Button (frei nach Heydon Pickering) platziere.

      Das mit contenteditable würde bedeuten, dass man das DOM innerhalb eines Elements verändern kann. Das willst Du bei einer ToDo-Liste nicht, denn sonst könnte man das <li> mit beliebigen Kindelementen befüllen, die ihrerseits Listen, Tabellen und noch viel mehr sein können. Und nur damit der User etwas kursiv und fett formatieren kann... würde ich unbedingt davon abraten.

      Das könnte man ja auch mit Script um HTML bereinigen und nur in textContent schreiben (siehe 3.), wäre für ein Anfänger-Tutorial aber zu komplex.


      Im Original-Artikel wird erst ein Absatz dynamisch eingefügt → 1. Idee: Da mach' ich eine li draus.

      Leider kommt danach eine Helfer-Funktion, die auf Wunsch Buttons oder p-Absätze erzeugt; Das geht in einer Liste leider nicht.
      → li und ul>li erzeugen lassen? -> li und li>button erzeugen lassen? → beides zu komplex!

      Wsl. lass ich diesen Abschnitt drin und fang erst danach an mit dem Anwendungsbeispiel an. Oder ich lasse Buttons und li-Elemente in verschiedene Elternelemente → wäre aber unübersichtlich.

      Wäre aber ein Übergang zu removeElement.


      Ich überlege noch!

      Herzliche Grüße

      Matthias Scharwies

      --
      Was ist eine Signatur?
      1. Servus!

        Mit diesem DOM kann man den „Checkbox-Hack“ verwenden, um erledigte Tasks entsprechend anders darzustellen.

        Und genau so etwas würde ich mit [aria-selected]{ background: green;} realisieren.

        Nein, input ist ok! Das ist aber kein „Checkbox-Hack“, da es nur das label formatiert, aber keine anderen Elemente.

        Herzliche Grüße

        Matthias Scharwies

        --
        Was ist eine Signatur?
      2. Lieber Matthias,

        1. Selecting Elements
        2. Creating and Inserting Elements
          document.createElement(tagName)
          parentElement.appendChild(childElement)
          element.insertAdjacentHTML(position, html)
        3. Changing Content
          element.textContent
        4. Changing Attributes
          element.setAttribute(attribute, value)
        5. Traversing the DOM

        naja, in meinem Fiddle werden ja manche dieser Eigenschaften und Methoden verwendet. Es mag einen Sinn haben, zu zeigen, wie man diese im Zusammenhang mit innerHTML verwendet. Auch das zeigt mein Fiddle.

        Wenn Du auf die Verwendung von Buttons Wert legst, musst Du Ereignisse verarbeiten. Das kann hinsichtlich der DOM-Thematik eine sinnvolle Ergänzung in diesem Kapitel sein. In meinem Fiddle wirst Du sehen, wie ich von einem Button auf seine Elternknoten schließe und dann damit etwas tue.

        Was die Position der Buttons im DOM angeht, so hatte ich diese auch zuerst am Ende. Aber da die Länge des label-Elements vom Inhalt abhängt, ich die Buttons aber vertical untereinander haben wollte, hatte ich mich dazu entschlossen, diese links zu positionieren. Das mag sich mit der Checkbox beißen, aber wenn ich die Buttons ganz rechts haben wollte, hätte ich mir erst ein passendes Grid-Layout überlegen müssen, mit dem ich aber noch immer nicht so fit bin...

        Wobei ich das Editieren wohl weglasse, und links die Checkbox und rechts den Löschen-Button (frei nach Heydon Pickering) platziere.

        Warum das Editieren nicht ebenso hinein nehmen? Das ergibt die Möglichkeit, beim Reagieren auf Button-Klicks zwischen den Buttons zu unterscheiden.

        Das mit contenteditable [...]

        Das könnte man ja auch mit Script um HTML bereinigen und nur in textContent schreiben (siehe 3.), wäre für ein Anfänger-Tutorial aber zu komplex.

        In der Tat. Deshalb hatte ich mir auch erlaubt, das Editieren mit window.prompt zu implementieren. Natürlich könnte man ein schickes dialog-Teil bauen, mit Fokusfangen und allem Schnickschnack, aber wie Du selbst richtig schreibst: Overkill. Ist das dann aber ein sinnvoller Grund es ganz zu lassen?

        Im Original-Artikel [...]

        Wäre aber ein Übergang zu removeElement.

        Der löschen-Button kann ja nur dann seine Funktion sinnvoll erfüllen, wenn man da removeElement verwendet. Tut mein Fiddle auch.

        Liebe Grüße

        Felix Riesterer

  5. Hallo zusammen,

    Nur zum Vergleich; da waren wir 2012!

    Das bestehende Tutorial habe ich 2016-2019 geschrieben; allerdings basierend auf dem, was mir @Felix Riesterer von 2012-14 beigebracht hatte. (Vor)-Letztes Jahr hatten wir schon überlegt, was man verbessern könnte, sind aber steckengeblieben.

    Mittlerweile habe ich das Tutorial umbenannt:[1]

    In Kapitel 1 werden getElementById, querySelector, querySelectorAll und textContent eingeführt.

    In Kapitel 2 wird noch mal die Einbindung mit script und dann DOMContentLoaded erklärt.

    Jetzt zur ToDo-Liste: Sie ist der Aufhänger und rote Faden, Thema ist DOM-Manipulation! Und zwar mit ES6, var und getElementById fliegen raus!

    1. Elemente erzeugen und wieder löschen

    • createElement(), die Helferfunktion ist in den Anhang gerutscht
    • insertAdjacentHTML, da wir mehrere Elemente haben[2]
    • remove anstelle von removeChild.[3]

    Und das ist mir aufgefallen: mit den „neuen“ DOM-Methoden findet der Elementbaum und DOM-Traversal im Hintergrund statt.

    Deshalb habe ich einen zweiten Abschnitt in den Anhang gepackt: DOM-Traversal

    2. Attribute auslesen und setzen

    Hier hirne ich noch;[4] auch was ich mit dem halben Bildwechsler machen soll.

    3. Web Storage, wie im Eingangspost erwähnt

    Ziel wäre es, das Tutorial fertigzustellen und von den JS-Referenzseiten drauf verlinken.

    Profis wissen, dass es closest und remove gibt, andere vertrauen weiter auf parentNode und childNode. Diese neuen Methoden möchte ich hier vorstellen!

    Für die node-Methoden bräuchte es ein eigenes Tutorial - evtl mit XML?
    Hat jemand Ideen?

    Herzliche Grüße
    Matthias Scharwies

    --
    Was ist eine Signatur?

    1. Ich musste das schnell rüberziehen, weil @Klark einiger meiner neuen Links wieder zurückgesetzt hatte. ;-) ↩︎

    2. Fun Fact: Auf Mastodon ist innerHTML das Mittel der Wahl!
      Wie sagt man denen freundlich, dass es was Besseres gibt? ↩︎

    3. Das CSS räum' ich noch auf! ↩︎

    4. Heydon kümmert sich eher um Barrierefreiheit und diskutiert, ob ein "X" oder ein "🗑" zugänglicher wäre. Interessant, aber nicht das Thema hier. ↩︎

    1. Hallo zusammen,

      Ich komme irgendwie nicht weiter. Im Wunsch alle Möglichkeiten zu bedenken, wird mir das Ganze für ein Einsteiger-Tutorial zu kompliziert. Ich habe auch Felix' Beispiel noch mal angeschaut.

      Deshalb möchte ich das Zwischenergbenis am Mittwoch, den 29.01.[1] am Discord-Stammtisch diskutieren:

      1. Wie soll das form innerhalb von dialog aufgebaut sein?
      2. Gestaltung der buttons kommt ganz zum Schluss - trotzdem auch ein Thema
      3. Wo würdet ihr Klassen oder Attribute einsetzen?
      • Habe es jetzt bei der Priorität gemacht, möchte aber nicht zu bunt werden.
      • Sollte der eingegebene Wert/Zustand "erledigt" evtl. in eine Klasse übertragen werden, damit man ihn nicht einfach wieder anklicken kann?
      1. Wie kann man das Alles anfängerverständlich umsetzen?

      Für die node-Methoden bräuchte es ein eigenes Tutorial - evtl mit XML?
      Hat jemand Ideen?

      Herzliche Grüße

      Matthias Scharwies

      --
      Was ist eine Signatur?

      1. sorry, grad noch'n Termin reingekommen. Ab Februar machen wir das aber wieder regelmäßig! ↩︎

      1. Lieber Matthias,

        vielleicht ist das nur mein persönlicher Eindruck, aber eine ToDo-Liste ohne serverseitige Verarbeitung hat in meinen Augen überhaupt keinen Sinn. Klar, kann man localStorage zum Speichern verwenden, aber dann ist diese Liste nur auf dem jeweiligen Gerät in diesem speziellen Browser verfügbar.

        Ist eine ToDo-Liste ein geeignetes Beispiel für ein Tutorial, welches DOM-Methoden in einem semi-sinnvollen Anwendungsbeispiel demonstrieren will? Mein Bauch sagt mir: Eher nicht.

        Wie wäre es mit einem Anwendungsbeispiel, welches z.B. eine tabellarische Darstellung von Informationen interaktiv nutzbar macht? Etwas in der Art, wie wir auf unseren Profil-Seiten unser Postingaufkommen in einer Diagrammkurve dargestellt bekommen, vielleicht? Du hattest doch mal die Idee zu einem Vokabeltrainer...?

        Das Ergebnis der DOM-Manipulationen sollte etwas Interaktives an sich haben, mit dem Herumzuspielen Spaß macht. Vielleicht geht das Ganze dann schon in Richtung Mini-Browsergame?

        Liebe Grüße

        Felix Riesterer