١٦ مايو ٢٠٢٣

المشي في الـ DOM

يسمح لنا الـ DOM بالقيام بأي شيء مع العناصر ومحتوياتها، ولكن أولاً يجب علينا الوصول إلى كائن DOM المقابل.

تبدأ جميع العمليات على الـ DOM مع كائن document. هذه هي النقطة الرئيسية للدخول إلى DOM. من خلاله يمكننا الوصول إلى أي عقدة.

هذه صورة لروابط تسمح بالسفر بين عقد DOM:

دعونا نناقشهم بمزيد من التفصيل.

في الأعلى: documentElement و body

تتوفر العقد الشجرية العليا مباشرة كخصائص document:

html> = document.documentElement>
العقدة الوثيقة العليا هي document.documentElement. هذه هي عقدة DOM للعلامة <html>.
body> = document.body>
عقدة DOM أخرى مستخدمة على نطاق واسع هي عنصر body>document.body>.
head> = document.head>
تتوفر علامة <head> كـ document.head.
هناك مشكلة: يمكن أن يكون document.bodyقيمتهnull

لا يمكن للبرنامج النصي الوصول إلى عنصر غير موجود في لحظة التشغيل.

على وجه الخصوص، إذا كان البرنامج النصي داخل <head>، فإن document.body غير متاح، لأن المتصفح لم يقرأه بعد.

لذلك، في المثال أدناه يظهر التنبيه الأول null:

<html>

  <head>
    <script>
          alert( "من HEAD: " + document.body ); // null، لا يوجد <body> بعد
    </script>
  </head>

  <body>

    <script>
      alert("من BODY: " + document.body); // HTMLBodyElement، الآن يوجد
    </script>

  </body>
</html>
فى عالم ال null DOM تعنى “غير موجود”

في ال DOM، تعني قيمة null “غير موجود” أو “لا يوجد مثل هذه العقدة”.

الأطفال: childNodes، firstChild، lastChild

هناك مصطلحان سنستخدمهما من الآن فصاعدًا:

  • عقد الأطفال (أو الأطفال) – العناصر التي هي أطفال مباشرين. بعبارة أخرى، هم متداخلون بالضبط في المعطى. على سبيل المثال، <head> و <body> هما أطفال عنصر <html>.
  • السلالة – جميع العناصر المتداخلة في المعطى، بما في ذلك الأطفال وأطفالهم وهكذا.

على سبيل المثال، هنا <body> لديه أطفال <div> و <ul> (وبعض العقد النصية الفارغة):

<html>
<body>
  <div>بداية</div>

  <ul>
    <li>
      <b>معلومات</b>
    </li>
  </ul>
</body>
</html>

…وسلالة <body> ليست فقط الأطفال المباشرين <div>، <ul> ولكن أيضًا العناصر المتداخلة بشكل أعمق، مثل <li> (طفل <ul>) و <b> (طفل <li>) – الشجرة الفرعية بأكملها.

تدرج مجموعة childNodes جميع عقد الأطفال، بما في ذلك العقد النصية.

يوضح المثال أدناه أطفال document.body:

<html>
<body>
  <div>بداية</div>

  <ul>
    <li>معلومات</li>
  </ul>

  <div>نهاية</div>

  <script>
    for (let i = 0; i < document.body.childNodes.length; i++) {
      alert( document.body.childNodes[i] ); // Text, DIV, Text, UL, ..., SCRIPT
    }
  </script>
  ...المزيد من المحتوى...
</body>
</html>

يرجى ملاحظة تفصيل مثير للاهتمام هنا. إذا قمنا بتشغيل المثال أعلاه ، فإن آخر عنصر يظهر هو <script>. في الواقع ، يحتوي المستند على المزيد من المحتوى أدناه ، ولكن في لحظة تنفيذ البرنامج النصي لم يقرأ المتصفح بعد ، لذلك لا يرى البرنامج النصي.

توفر الخصائص firstChild و lastChild إمكانية الوصول السريع إلى الأطفال الأول والأخير.*

إنهم مجرد اختصارات. إذا كانت هناك عقد فرعية ، فإن التالي صحيح دائمًا:

elem.childNodes[0] === elem.firstChild
elem.childNodes[elem.childNodes.length - 1] === elem.lastChild

هناك أيضًا وظيفة خاصة elem.hasChildNodes() للتحقق مما إذا كانت هناك أي عقد فرعية.

مجموعات DOM

كما نرى ، يبدو childNodes كمصفوفة. ولكن في الواقع ليس مجموعة ، بل مجموعة – كائن قابل للتكرار يشبه المصفوفة.

هناك نتيجتان مهمتان:

  1. يمكننا استخدام for..of للتكرار عليه:
for (let node of document.body.childNodes) {
  alert(node); // يظهر جميع العقد من المجموعة
}

هذا لأنه قابل للتكرار (يوفر خاصية Symbol.iterator ، كما هو مطلوب).

  1. لن تعمل طرق المصفوفة ، لأنها ليست مصفوفة:
alert(document.body.childNodes.filter); // undefined (لا يوجد طريقة تصفية!)

الشيء الأول جميل. الثانية محتملة ، لأنه يمكن استخدام Array.from لإنشاء مصفوفة “حقيقية” من المجموعة ، إذا كانت نرغب في استخدام طرق المصفوفات:

alert( Array.from(document.body.childNodes).filter ); // function
مجموعات DOM للقراءة فقط

مجموعات DOM ، وحتى أكثر – جميع خصائص التنقل المدرجة في هذا الفصل للقراءة فقط.

لا يمكننا استبدال الطفل بشيء آخر عن طريق تعيين childNodes[i] = ....

تغيير DOM يحتاج إلى طرق أخرى. سنراهم في الفصل التالي.

مجموعات DOM حية

تكاد تكون جميع مجموعات DOM بدون استثناءات قليلة حية. بعبارة أخرى ، فهي تعكس الحالة الحالية لـ DOM.

إذا قمنا بالاحتفاظ بإشارة إلى elem.childNodes ، وإضافة / إزالة العقد في DOM ، فإنها تظهر في المجموعة تلقائيًا.

لا تستخدم for..in للتكرار على المجموعات

تكون المجموعات قابلة للتكرار باستخدام for..of. في بعض الأحيان يحاول الناس استخدام for..in لذلك.

من فضلك، لا. يتكرر حلقة for..in على جميع الخصائص القابلة للتعداد. والمجموعات لديها بعض الخصائص “الإضافية” التي نادرًا ما نستخدمها والتي عادةً ما لا نرغب في الحصول عليها:

<body>
<script>
  // يظهر 0 و 1 والطول والعنصر والقيم والمزيد.
  for (let prop in document.body.childNodes) alert(prop);
</script>
</body>

الأشقاء والوالد

الأشقاء هم العقد التي هي أطفال من نفس الوالد.

على سبيل المثال، هنا <head> و <body> هم أشقاء:

<html>
  <head>...</head><body>...</body>
</html>
  • يقال أن <body> هو الشقيق “التالي” أو “الأيمن” لـ <head>,
  • يقال أن <head> هو الشقيق “السابق” أو “الأيسر” لـ <body>.

الشقيق التالي موجود في خاصية nextSibling، والسابق – في previousSibling.

الوالد متاح كـ parentNode.

على سبيل المثال:

// والد <body> هو <html>
alert(document.body.parentNode === document.documentElement); // صحيح

// بعد <head> يأتي <body>
alert(document.head.nextSibling); // HTMLBodyElement

// قبل <body> يأتي <head>
alert(document.body.previousSibling); // HTMLHeadElement

التنقل الخاص بالعناصر فقط

تشير خصائص التنقل المذكورة أعلاه إلى جميع العقد. على سبيل المثال، في childNodes يمكننا رؤية كل من عقد النصوص وعقد العناصر وحتى عقد التعليقات إذا كانت موجودة.

ولكن للعديد من المهام لا نريد عقد النصوص أو التعليقات. نريد التلاعب بعقد العناصر التي تمثل العلامات وتشكل هيكل الصفحة.

لذلك دعونا نرى المزيد من روابط التنقل التي تأخذ عقد العناصر فقط في الاعتبار:

الروابط مشابهة لتلك المعطاة أعلاه، فقط مع كلمة Element بداخلها:

  • children – فقط تلك الأطفال التي هي عقد عناصر.
  • firstElementChild، lastElementChild – أول وآخر أطفال العناصر.
  • previousElementSibling، nextElementSibling – عناصر الجوار.
  • parentElement – عنصر الوالد.
لماذا parentElement؟ هل يمكن أن يكون الوالد ليس عنصرًا؟

تعيد خاصية parentElementالوالد “العنصر”، بينما تعيدparentNode الوالد “أي عقدة”. هذه الخصائص عادة متشابهة: كلاهما يحصل على الوالد.

مع استثناء واحد لـ document.documentElement:

alert(document.documentElement.parentNode); // document
alert(document.documentElement.parentElement); // null

السبب هو أن العقدة الجذر document.documentElement (<html>) لديها document كوالدها. لكن document ليس عقدة عنصر، لذلك تعيد parentNode ولا تعيد parentElement.

قد يكون هذا التفصيل مفيدًا عندما نريد السفر من عنصر عشوائي elem إلى <html>، ولكن ليس إلى document:

while ((elem = elem.parentElement)) { // اذهب لأعلى حتى <html>
  alert(elem);
}

دعونا نعدل أحد الأمثلة أعلاه: استبدل childNodes بـ children. الآن يظهر العناصر فقط:

<html>
<body>
  <div>بداية</div>

  <ul>
    <li>معلومات</li>
  </ul>

  <div>نهاية</div>

  <script>
    for (let elem of document.body.children) {
      alert(elem); // DIV، UL، DIV، SCRIPT
    }
  </script>
  ...
</body>
</html>

المزيد من الروابط: الجداول

حتى الآن قمنا بوصف خصائص التنقل الأساسية.

قد توفر أنواع معينة من عناصر DOM خصائص إضافية، محددة لنوعها، للراحة.

الجداول هي مثال رائع على ذلك، وتمثل حالة مهمة بشكل خاص:

عنصر <table> يدعم (بالإضافة إلى المعطى أعلاه) هذه الخصائص:

  • table.rows – مجموعة عناصر <tr> في الجدول.
  • table.caption/tHead/tFoot – مراجع لعناصر <caption>، <thead>، <tfoot>.
  • table.tBodies – مجموعة عناصر <tbody> (يمكن أن يكون هناك العديد وفقًا للمعيار، ولكن سيكون هناك دائمًا واحد على الأقل – حتى لو لم يكن في HTML المصدر، سيضعه المتصفح في DOM).

عناصر <thead> و <tfoot> و <tbody> توفر خاصية rows:

  • tbody.rows – مجموعة <tr> بداخلها.

<tr>:

  • tr.cells – مجموعة الخلايا <td> و <th> داخل <tr> المعطى.
  • tr.sectionRowIndex – الموضع (الفهرس) لـ <tr> المعطى داخل <thead>/<tbody>/<tfoot> المحيط.
  • tr.rowIndex – رقم <tr> في الجدول ككل (بما في ذلك جميع صفوف الجدول).

<td> و <th>:

  • td.cellIndex – عدد الخلايا داخل التغليف<tr>.

مثال على الاستخدام:

<table id="table">
  <tr>
    <td>واحد</td><td>اثنان</td>
  </tr>
  <tr>
    <td>ثلاثة</td><td>أربعة</td>
  </tr>
</table>

<script>
  // احصل على td مع "اثنان" (الصف الأول، العمود الثاني)
  let td = table.rows[0].cells[1];
  td.style.backgroundColor = "red"; // قم بتسليط الضوء عليه
</script>

المواصفات: بيانات جدولية.

هناك أيضًا خصائص تنقل إضافية لنماذج HTML. سننظر إليهم لاحقًا عندما نبدأ في العمل مع النماذج.

ملخص

بالنظر إلى عقدة DOM، يمكننا الذهاب إلى جيرانها المباشرين باستخدام خصائص التنقل.

هناك مجموعتان رئيسيتان منهم:

  • لجميع العقد: parentNode، childNodes، firstChild، lastChild، previousSibling، nextSibling.
  • لعقد العنصر فقط: parentElement، children، firstElementChild، lastElementChild، previousElementSibling، nextElementSibling.

توفر بعض أنواع عناصر DOM، مثل الجداول، خصائص ومجموعات إضافية للوصول إلى محتوياتها.

مهمه

الأهمية: 5

انظر إلى هذه الصفحة:

<html>
  <body>
    <div>المستخدمون:</div>
    <ul>
      <li>جون</li>
      <li>بيت</li>
    </ul>
  </body>
</html>

لكل من التالي، أعطِ طريقة واحدة على الأقل لكيفية الوصول إليها:

  • عقدة <DOM <div؟
  • عقدة <DOM <ul؟
  • العنصر <li> الثاني (مع Pete)؟

هناك العديد من الطرق، على سبيل المثال:

عقدة <DOM <div:

document.body.firstElementChild;
// أو
document.body.children[0];
// أو (العقدة الأولى هي المسافة، لذلك نأخذ الثانية)
document.body.childNodes[1];

عقدة <DOM <ul:

document.body.lastElementChild;
// أو
document.body.children[1];

العنصر <li> الثاني (مع Pete):

// احصل على <ul>، ثم احصل على آخر عنصر فرعي له
document.body.lastElementChild.lastElementChild;
الأهمية: 5

إذا كان elem – هو عقدة DOM عنصرية عشوائية…

  • هل صحيح أن elem.lastChild.nextSibling هو دائمًا null؟
  • هل صحيح أن elem.children[0].previousSibling هو دائمًا null؟
  1. نعم، صحيح. العنصر elem.lastChild هو دائمًا الأخير، ولا يوجد له nextSibling.
  2. لا، خطأ، لأن elem.children[0] هو الطفل الأول بين العناصر. ولكن قد توجد عقد غير عنصرية قبله. لذلك قد يكون previousSibling عقدة نصية.

يرجى الملاحظة: في كلا الحالتين إذا لم يكن هناك أطفال، فسيكون هناك خطأ.

إذا لم يكن هناك أطفال، فإن elem.lastChild هو null، لذلك لا يمكننا الوصول إلى elem.lastChild.nextSibling. ومجموعة elem.children فارغة (مثل مصفوفة فارغة []).

الأهمية: 5

اكتب الكود لتلوين كل الخلايا القطرية في الجدول باللون الأحمر.

سوف تحتاج إلى الحصول على كل <td> قطرية من <table> وتلوينها باستخدام الكود:

// يجب أن يكون td مرجعًا لخلية الجدول
td.style.backgroundColor = 'red';

يجب أن يكون النتيجة:

افتح sandbox للمهمه.

سنستخدم خصائص rows و cells للوصول إلى الخلايا القطرية في الجدول.

افتح الحل في sandbox.

خريطة الدورة التعليمية

التعليقات

إقرأ هذا قبل أن تضع تعليقًا…
  • إذا كان لديك اقتراحات أو تريد تحسينًا - من فضلك من فضلك إفتح موضوعًا فى جيتهاب أو شارك بنفسك بدلًا من التعليقات.
  • إذا لم تستطع أن تفهم شيئّا فى المقال - وضّح ماهو.
  • إذا كنت تريد عرض كود استخدم عنصر <code> ، وللكثير من السطور استخدم <pre>، ولأكثر من 10 سطور استخدم (plnkr, JSBin, codepen…)