بناء التنقل الديناميكي للتذييل في Angular
أثناء تقدمك في رحلتك Angular، ستواجه حتمًا تحديات، خاصة عند محاولة إنشاء ميزات ديناميكية مثل التنقل. إحدى هذه الميزات هي إنشاء تنقل ديناميكي للتذييل يعكس شريط التنقل الرئيسي لتطبيقك. غالبًا ما يكون هذا مطلوبًا عندما تريد رابطًا سريعًا لأهم الصفحات في موقعك، ولكن بطريقة أكثر دقة وغير مزعجة، ويتم وضعها عادةً في أسفل الصفحة. 💡
ومع ذلك، أثناء تنفيذ هذه الميزة، قد يواجه المطورون مشكلات تتعلق بأخطاء كتابة TypeScript. قد يكون هذا محبطًا، خاصة بالنسبة لأولئك الذين ما زالوا يتعلمون Angular. في هذه المقالة سوف نستكشف خطأ محددًا غالبًا ما ينشأ عند بناء أنظمة الملاحة الديناميكية وكيفية حله. يعود الأمر كله إلى آلية التحقق من النوع في TypeScript، والتي تضمن تطابق الأنواع كما هو متوقع بين التنقل الرئيسي وتنقل التذييل الديناميكي. 🚀
يرتبط الخطأ الذي قد تواجهه بأنواع غير متطابقة بين خاصية "العنوان" في تكوين المسار والنوع المتوقع في المكون الخاص بك. المشكلة الرئيسية هنا هي أن TypeScript غير قادر على تحليل نوع "العنوان" كسلسلة، لأنه قد يكون أيضًا نوعًا أو وظيفة أو بنية أخرى، مما يؤدي إلى إنشاء تعارض. لا تقلق، فهذا حجر عثرة شائع، والإصلاح أسهل مما تظن!
في القسم التالي، سنتعمق أكثر في خطأ TypeScript المحدد (TS2322) الذي تواجهه، ونراجع رمز مكون Angular، ونرشدك إلى كيفية حل هذه المشكلة خطوة بخطوة. مع القليل من الفهم لكيفية عمل TypeScript وكيفية تعامل Angular مع التوجيه، ستتمكن من التغلب على هذه العقبة وإنشاء نظام تنقل ديناميكي يعمل بسلاسة عبر تطبيقك. 😊
يأمر | مثال للاستخدام |
---|---|
@Component | يتم استخدام الديكور @Component لتحديد مكون Angular. فهو يحدد بيانات التعريف الخاصة بالمكون، مثل محدده ونموذجه ونمطه. في هذه الحالة، يتم استخدامه لإنشاء مكون "التنقل في التذييل" لشريط التنقل في التذييل. |
RouterModule | RouterModule هي وحدة Angular تتيح التنقل بين طرق العرض. إنه ضروري لميزات التوجيه مثل "routerLink"، و"routerLinkActive"، والتحميل البطيء للمكونات في تطبيق Angular. |
this.router.config | هذه مصفوفة تحتوي على تكوين مسار التطبيق بالكامل. باستخدام هذه الخاصية، يمكنك الوصول إلى المسارات المحددة وتصفيتها لمكونات التنقل الديناميكي مثل التنقل في التذييل. |
filter() | يتم استخدام طريقة التصفية لإنشاء مصفوفة جديدة تحتوي على جميع العناصر التي اجتازت الاختبار الذي تم تنفيذه بواسطة وظيفة رد الاتصال المتوفرة. في هذه الحالة، يقوم بتصفية المسارات التي لا تحتوي على مسار أو عنوان أو بيانات، مما يضمن تضمين المسارات الصالحة فقط في التنقل الديناميكي. |
map() | يقوم الأسلوب Map بإنشاء مصفوفة جديدة مليئة بنتائج استدعاء دالة متوفرة على كل عنصر في المصفوفة. يتم استخدامه لتحويل كل مسار إلى كائن يحتوي فقط على المسار والعنوان، وهو أمر مطلوب لعرض التنقل الديناميكي. |
typeof | يتم استخدام عامل التشغيل typeof للتحقق من نوع بيانات القيمة. هنا، يتم استخدامه للتحقق مما إذا كان عنوان المسار عبارة عن سلسلة قبل تعيينه لخاصية العنوان في التنقل الديناميكي، مما يضمن تعيين النوع المناسب. |
isString() (Type Guard) | isString هي وظيفة حماية من النوع المخصص. يتم استخدام حراس الكتابة في TypeScript لتضييق نطاق الأنواع. في هذه الحالة، يتم استخدامه للتحقق بأمان مما إذا كان العنوان عبارة عن سلسلة قبل محاولة تعيينه لخاصية عنوان التنقل الديناميكي. |
! (Non-null Assertion Operator) | يُخبر عامل التأكيد غير الفارغ (!)، المُستخدم بعد خصائص المسار والعنوان، TypeScript بأن هذه الخصائص لن تكون أبدًا فارغة أو غير محددة في وقت التشغيل، حتى لو تمت كتابتها على أنها فارغة. يساعد هذا في تجنب أخطاء TypeScript عند الوصول إلى خصائص المسار. |
expect() (Jasmine/Unit Test) | يتم استخدام الدالة المتوقعة في اختبارات الوحدة للتأكد من أن القيمة تستوفي شروطًا معينة. في هذه الحالة، يتم استخدامه للتحقق مما إذا تم إنشاء المكون بنجاح وما إذا تم تصفية مسارات التنقل الديناميكي بشكل صحيح. |
استكشاف خطأ TypeScript والحل في التنقل الديناميكي الزاوي
في Angular، يعد التوجيه ميزة أساسية تسمح لك بإنشاء تنقل ديناميكي داخل التطبيق الخاص بك. في هذا السيناريو، تظهر المشكلة عند محاولة إنشاء تنقل ديناميكي بالتذييل يعكس التنقل الرئيسي. يحدث الخطأ عندما يتم اكتشاف عدم تطابق نوع TypeScript لخاصية "العنوان" للمسارات. تشير رسالة الخطأ إلى أن النوع المتوقع عبارة عن سلسلة، ولكن وجدت TypeScript أن خاصية "العنوان" يمكن أن تكون أيضًا "نوعًا"
ال `@عنصريتم استخدام الديكور في Angular لتحديد البيانات الوصفية للمكون. في هذه الحالة، يتم الإعلان عن مكون `footer-nav`، الذي يتولى عرض التنقل الديناميكي للتذييل. يحتوي المكون على خصائص مهمة مثل `templateUrl` و`styleUrls` التي تشير إلى ملفات HTML وCSS الخاصة بالمكون، على التوالي. من خلال إدخال خدمة "جهاز التوجيه" في مُنشئ المكون، يمكننا الوصول إلى تكوين المسار ويمكننا ملء روابط التنقل ديناميكيًا. تحتوي مصفوفة "المسارات" الموجودة في المكون على البيانات اللازمة لإنشاء التنقل في التذييل، حيث يحتوي كل مسار على "مسار" و"عنوان" لعرضهما في واجهة المستخدم.
في البرنامج النصي، نستخدم "this.router.config" للوصول إلى تكوينات المسار من التطبيق الرئيسي. يتم بعد ذلك استخدام الأسلوب `filter()` لتحديد المسارات الصالحة فقط، أي تلك التي لها `مسار` و`عنوان`. يتم استخدام وظيفة `map()` لتحويل المسارات التي تمت تصفيتها إلى تنسيق جديد، مع التأكد من أن كل كائن مسار يحتوي على خصائص `المسار` و`العنوان` المطلوبة. الأهم من ذلك، أن استخدام التأكيدات غير الخالية (مثل `route.path!`) يهدف إلى السماح لـ TypeScript بمعرفة أن خصائص المسار والعنوان ستحتوي دائمًا على قيم، على الرغم من أنه قد يتم وضع علامة عليها على أنها "غير محددة" أو `فارغة`. . ومع ذلك، يجب توخي الحذر عند استخدام هذا العامل، لأنه يتجاوز عمليات التحقق من سلامة النوع في TypeScript.
على الرغم من أن أسلوب التنقل الديناميكي يعد طريقة جيدة لإنشاء مكونات قابلة لإعادة الاستخدام للتذييل، إلا أن أحد الجوانب المهمة هو التأكد من أن تكوين المسار آمن من حيث النوع. يحدث الخطأ لأن TypeScript تتوقع أن يكون المسار `title` عبارة عن سلسلة بسيطة، ولكنه قد يكون أيضًا نوعًا معقدًا (مثل وظيفة `Resolve` أو `Type`). لحل هذه المشكلة، يمكنك تعديل منطق التصفية أو إضافة حراس النوع لضمان تعيين العناوين الصالحة فقط للتنقل الديناميكي. في النهاية، سيؤدي هذا إلى قائمة تنقل آمنة ومولدة ديناميكيًا تتصرف تمامًا مثل التنقل الرئيسي، ولكنها موضوعة في التذييل. يساعد هذا الأسلوب في الحفاظ على التطبيق معياريًا ونظيفًا، وسيكون بمثابة إضافة قيمة إلى مجموعة أدوات تطوير Angular الخاصة بك! 😊
إصلاح عدم تطابق نوع TypeScript في التنقل الديناميكي الزاوي للتذييل
TypeScript، الزاوي، التنقل الديناميكي
import { Component, inject } from '@angular/core';
import { RouterModule, Router, NavigationEnd } from '@angular/router';
@Component({
selector: 'footer-nav',
standalone: true,
imports: [RouterModule],
templateUrl: './footer-nav.component.html',
styleUrl: './footer-nav.component.scss'
})
export class FooterNavComponent {
routes: { path: string; title: string; }[] = [];
constructor(private router: Router) {
this.routes = this.router.config.filter(route => route.path !== '' && route.data && route.title)
.map(route => ({ path: route.path!, title: route.title as string! }));
}
}
النهج البديل: معالجة أخطاء TypeScript للتنقل الديناميكي مع المسارات المعقدة
TypeScript، Angular، معالجة الأخطاء، التنقل الديناميكي
import { Component, inject } from '@angular/core';
import { RouterModule, Router, NavigationEnd } from '@angular/router';
@Component({
selector: 'footer-nav',
standalone: true,
imports: [RouterModule],
templateUrl: './footer-nav.component.html',
styleUrl: './footer-nav.component.scss'
})
export class FooterNavComponent {
routes: { path: string; title: string; }[] = [];
constructor(private router: Router) {
this.routes = this.router.config.filter(route => route.path !== '' && route.data && route.title)
.map(route => ({ path: route.path!, title: typeof route.title === 'string' ? route.title : 'Default Title' }));
}
}
استخدام حراس الكتابة لمنع أخطاء كتابة TypeScript في التنقل الديناميكي
TypeScript، Angular، Type Guard، التنقل
import { Component, inject } from '@angular/core';
import { RouterModule, Router, NavigationEnd } from '@angular/router';
function isString(value: string | Type<Resolve<string>> | ResolveFn<string>): value is string {
return typeof value === 'string';
}
@Component({
selector: 'footer-nav',
standalone: true,
imports: [RouterModule],
templateUrl: './footer-nav.component.html',
styleUrl: './footer-nav.component.scss'
})
export class FooterNavComponent {
routes: { path: string; title: string; }[] = [];
constructor(private router: Router) {
this.routes = this.router.config.filter(route => route.path !== '' && route.data && route.title)
.map(route => ({ path: route.path!, title: isString(route.title) ? route.title : 'Fallback Title' }));
}
}
مثال على اختبار الوحدة لمكون التنقل الديناميكي الزاوي
الزاوي، اختبار الوحدة، الدعابة، الياسمين
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { RouterModule, Router } from '@angular/router';
import { FooterNavComponent } from './footer-nav.component';
describe('FooterNavComponent', () => {
let component: FooterNavComponent;
let fixture: ComponentFixture<FooterNavComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
imports: [RouterModule],
declarations: [FooterNavComponent]
}).compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(FooterNavComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create the component', () => {
expect(component).toBeTruthy();
});
it('should filter routes correctly', () => {
const filteredRoutes = component.routes;
expect(filteredRoutes.length).toBeGreaterThan(0);
});
});
الجدول: شرح الأوامر المحددة المستخدمة في حل التنقل الديناميكي الزاوي
فهم التنقل الديناميكي وإصلاحات خطأ TypeScript في Angular
في Angular، يمكن أن يكون إنشاء نظام تنقل ديناميكي طريقة رائعة لتحسين تجربة المستخدم، خاصة عندما تريد تكرار أو تكرار عناصر تنقل معينة في أجزاء مختلفة من تطبيقك. من الأمثلة الشائعة على ذلك إنشاء شريط تنقل ديناميكي في التذييل، مشابه للشريط الموجود في الرأس. يحدث الخطأ الذي واجهته، TS2322، بسبب عدم تطابق النوع في TypeScript، حيث من المتوقع أن يكون "عنوان" المسارات عبارة عن سلسلة بسيطة، ولكنه قد يكون أيضًا أكثر تعقيدًا بسبب استخدام أدوات الحل أو أساليب جلب البيانات الديناميكية مثل "Resolve" أو "ResolveFn". تسمح هذه الميزات المتقدمة للمسارات بجلب البيانات قبل التنقل ولكنها قد تسبب مشاكل عندما يتعذر على TypeScript استنتاج النوع المناسب لخصائص المسار مثل "العنوان".
لحل هذه المشكلة، تحتاج إلى ضبط التعليمات البرمجية الخاصة بك للتعامل مع الأنواع المعقدة بشكل مناسب. تتمثل إحدى الطرق في تعديل تكوينات المسار الخاص بك للتأكد من أن خاصية `title` الخاصة بكل مسار محددة بشكل واضح على أنها سلسلة. يمكن القيام بذلك عن طريق استخدام تأكيدات النوع أو حراس النوع للتحقق مما إذا كان "العنوان" عبارة عن سلسلة قبل تعيينه إلى مصفوفة "المسارات" في المكون الخاص بك. بالإضافة إلى ذلك، إذا كانت مساراتك تستخدم وحدات الحل لجلب العناوين الديناميكية، فتأكد من جلب البيانات وتعيينها بشكل صحيح قبل أن يحاول مكون التنقل الوصول إلى "العنوان". سيضمن هذا أن نظام الكتابة الخاص بـ TypeScript يمكنه التحقق من صحة خصائص المسار بشكل صحيح.
جانب آخر مهم هو التأكد من تكوين خدمات "RouterModule" و"Router" بشكل صحيح في الوحدة النمطية Angular الخاصة بك. توفر هذه الخدمات الوظائف اللازمة لتصفية تكوين المسار وتعيينه ديناميكيًا للتنقل في التذييل. من خلال إدخال خدمة "جهاز التوجيه" في مُنشئ المكون الخاص بك والوصول إلى "this.router.config"، يمكنك تصفية المسارات المتاحة وإنشاء مجموعة جديدة من المسارات خصيصًا للتذييل. يضمن هذا أن يتم إنشاء التنقل في التذييل ديناميكيًا من نفس تكوين المسار مثل التنقل الرئيسي، مما يجعل تطبيقك معياريًا وسهل الصيانة.
الأسئلة المتداولة حول التنقل الديناميكي وأخطاء TypeScript في Angular
- كيف أقوم بإصلاح خطأ TS2322 المتعلق بعنوان المسار في Angular؟
- يحدث الخطأ لأن TypeScript تتوقع أن يكون المسار "title" عبارة عن سلسلة، ولكنه قد يكون أيضًا نوعًا أكثر تعقيدًا مثل "Resolve". لإصلاح ذلك، يمكنك إما استخدام تأكيد النوع للتأكد من معاملة العنوان كسلسلة، أو تحديث تكوين المسار الخاص بك للتأكد من أن `العنوان` هو دائمًا سلسلة عند الوصول إليه في المكون الخاص بك. Example: `العنوان: المسار.العنوان كسلسلة`.
- ما هي أفضل طريقة لإنشاء التنقل في التذييل ديناميكيًا في Angular؟
- يمكنك إنشاء التنقل في التذييل ديناميكيًا باستخدام Angular's `RouterModuleخدمات "و"جهاز التوجيه"". أولاً، تحتاج إلى إدخال "جهاز التوجيه" في المكون الخاص بك، والوصول إلى "this.router.config"، وتصفية المسارات وتعيينها، ثم عرضها باستخدام "*ngFor" في القالب الخاص بك.
- كيف يمكنني التأكد من أن التنقل الديناميكي يعمل للمسارات البطيئة التحميل؟
- لا تتوفر المسارات المحملة البطيئة على الفور في تكوين المسار. للتأكد من تضمينها في التنقل الديناميكي، يجب عليك أولاً التأكد من تحديد المسارات بشكل صحيح باستخدام `loadChildren` أو `loadComponent` في وحدة التوجيه الخاصة بك. بعد ذلك، استخدم خدمة "جهاز التوجيه" للوصول إلى المسارات المحملة ديناميكيًا وإدراجها في شريط التنقل في التذييل.
- هل يمكنني استخدام أدوات حل المسار لتحميل البيانات لعناوين التنقل؟
- نعم، تعد أدوات حل المسار مثالية لتحميل البيانات قبل التنقل. يمكنك استخدام محلل لجلب العناوين الديناميكية لمساراتك، ولكن يجب عليك التأكد من توفر العنوان قبل محاولة تعيين المسارات إلى مكون التنقل الديناميكي الخاص بك. يجب أن يكون العنوان سلسلة عند الوصول إليه.
- ما هو دور `map()` في تصفية بيانات المسار وتعديلها؟
- ال `map()يتم استخدام الدالة لتحويل البيانات من تكوين المسار. فهو يسمح لك باختيار وتحويل خصائص محددة (مثل "المسار" و"العنوان") من كائن المسار، وإنشاء مصفوفة جديدة من كائنات المسار المبسطة لاستخدامها في المكون. وهذا يجعل البيانات أكثر قابلية للإدارة ويضمن تمرير البيانات ذات الصلة فقط إلى شريط التنقل في التذييل.
- هل يمكن للوضع الصارم لـ TypeScript أن يسبب مشكلات في التنقل الديناميكي؟
- نعم، تم تصميم الوضع الصارم لـ TypeScript لاكتشاف عدم تطابق الكتابة والأخطاء مبكرًا. يمكن أن يكون هذا مفيدًا، ولكنه يعني أيضًا أنك بحاجة إلى أن تكون واضحًا بشأن أنواعك. إذا كنت تستخدم أنواعًا معقدة مثل `Resolve` أو `ResolveFn`، فتأكد من التعامل معها بشكل صحيح، إما عن طريق تأكيد النوع أو حماية النوع، لتجنب الأخطاء في منطق التنقل.
- كيف يعمل "routerLinkActive" الخاص بـ Angular في التنقل الديناميكي؟
- `routerLinkActive` هو توجيه يُستخدم لإضافة فئة CSS إلى الارتباط النشط في التنقل. في التنقل الديناميكي بالتذييل، يساعد في تمييز المسار النشط حاليًا. يمكنك ضبطه على "نشط" لتصميم الرابط عندما يكون المسار نشطًا، مما يوفر إشارة مرئية للمستخدم حول قسم الموقع الذي يشاهده حاليًا.
- لماذا لا يتم تحديث التنقل الديناميكي الخاص بي عندما أقوم بالتنقل؟
- إذا لم يتم تحديث التنقل الديناميكي، فقد يكون ذلك بسبب عدم اكتشاف المكون للتغييرات. لإصلاح ذلك، تأكد من أن مكون التنقل يستمع إلى أحداث جهاز التوجيه ويقوم بتحديث قائمة المسارات وفقًا لذلك. يمكنك استخدام Angular `Router.events`للاشتراك في تغييرات المسار وتحديث قائمة المسارات النشطة ديناميكيًا.
- هل يمكنني تطبيق نفس منطق التوجيه الديناميكي على كل من الرأس والتذييل؟
- نعم، يعمل منطق إنشاء التنقل الديناميكي مع كل من الرأس والتذييل. يمكنك إعادة استخدام تصفية المسار وتعليمات التعيين البرمجية في كلا المكونين، طالما أنهما يصلان إلى نفس تكوين المسار ويقومان بإنشاء الروابط ديناميكيًا.
عند العمل باستخدام التنقل الديناميكي في Angular، غالبًا ما يواجه المطورون أخطاء مثل TS2322 بسبب عدم تطابق النوع بين الأنواع المتوقعة والفعلية في تكوينات المسار. في هذه المقالة، تناولنا كيفية معالجة مشكلات TypeScript المتعلقة طريق خصائص، وخاصة عنوان. يتضمن الحل ضمان الكتابة المتسقة للمسارات، سواء كنت تتعامل مع الوحدات النمطية المحملة ببطء أو تستخدمها الحلول للمحتوى الديناميكي. لقد ناقشنا أيضًا أفضل الممارسات لإنشاء تنقل ديناميكي للتذييل، استنادًا إلى تكوين المسار الرئيسي لديك.
فهم إصلاح أخطاء TypeScript في التنقل الديناميكي
تسمح مرونة Angular للمطورين بإعداد التنقل الديناميكي بسهولة لمختلف المكونات، بما في ذلك التذييل. ومع ذلك، عند العمل مع المسارات المحملة ببطء والمحتوى الديناميكي، قد يؤدي TypeScript إلى ظهور أخطاء مثل TS2322 بسبب عدم تطابق النوع. تتضمن المشكلة الأكثر شيوعًا تكوينات المسار، حيث من المتوقع أن يكون عنوان المسار بسيطًا خيط، ولكن يمكن أن يكون في بعض الأحيان أكثر تعقيدًا بسبب الحلول أو طرق جلب البيانات غير المتزامنة. مفتاح حل هذه المشكلة هو ضمان الكتابة المتسقة والصحيحة في مساراتك.
أحد أفضل الحلول هو تحديث تكوين المسار للتأكد من كتابة عنوان كل مسار بشكل واضح كسلسلة. يمكن القيام بذلك عن طريق استخدام تأكيدات النوع أو عمليات التحقق البسيطة ضمن منطق تعيين المسار الخاص بك. إذا تم حل خاصية العنوان ديناميكيًا عبر محلل، فيجب عليك التأكد من توفر البيانات وكتابتها بشكل صحيح قبل تمريرها إلى مكون التذييل للعرض. من خلال القيام بذلك، سيقوم TypeScript بالتحقق من صحة البيانات بشكل صحيح، مما يمنع الأخطاء عندما يحاول مكون التنقل في التذييل الوصول إلى عنوان المسار.
علاوة على ذلك، لتعزيز قابلية التوسع لتطبيقك، يجب أن تفكر في إعادة استخدام منطق التنقل الرئيسي الخاص بك في أجزاء أخرى من التطبيق، مثل التذييل. يمكن تحقيق ذلك بسهولة عن طريق الوصول إلى المسارات المحددة في وحدة التوجيه الخاصة بتطبيقك، وتصفية البيانات الضرورية، وتمريرها إلى شريط التنقل في التذييل. عن طريق الحقن جهاز التوجيه الخدمة وباستخدام طرق التوجيه الخاصة بـ Angular، يمكنك إنشاء نظام تنقل ديناميكي معياري يعمل بشكل متسق عبر أقسام مختلفة من الموقع.
خاتمة:
في الختام، حل أخطاء TypeScript المتعلقة بالتنقل الديناميكي في Angular يعود إلى إدارة أنواع المسارات بشكل صحيح. من خلال التأكد من كتابة الخصائص بشكل متسق، يمكن للمطورين تجنب الأخطاء الشائعة مثل خطأ TS2322. بالإضافة إلى ذلك، يمكن أن يؤدي إنشاء تنقل ديناميكي قابل لإعادة الاستخدام إلى تبسيط إدارة التنقل عبر المكونات المختلفة في تطبيقك.
من خلال اتباع أفضل ممارسات التحميل البطيء وجلب بيانات المسار ونمطية المكونات، يمكنك إنشاء نظام تنقل ديناميكي فعال وخالي من الأخطاء. سيؤدي تبني هذه المفاهيم إلى جعل تطبيقات Angular الخاصة بك أكثر قابلية للصيانة ومرونة وسهولة في الاستخدام. 🚀
المراجع والمواد المصدرية
- يوفر نظرة ثاقبة لفهم أخطاء TypeScript وحلول التنقل الديناميكي في Angular. لمزيد من المعلومات التفصيلية، قم بزيارة التوثيق الزاوي .
- يناقش تكوين المسار وتوافق نوع TypeScript، والذي يرتبط مباشرة بالخطأ TS2322 الذي واجهه في التعليمات البرمجية. مرجع: الوثائق الرسمية لـ TypeScript .
- يشرح التحميل البطيء في Angular وكيفية التعامل مع بيانات المسار للتنقل الديناميكي. يمكن العثور على مزيد من القراءة على دليل التحميل البطيء الزاوي .