تعلم الجافاسكربت المعاصرة قد يكون أمرا معقدا خاصة للقادمين الجدد الذين لم يشهدوا التغييرات والتطورات المتعددة التي طرأت على النظام البيئي (Ecosystem) للغة البرمجة جافاسكربت في المدة الأخيرة.
في بداياتي مع تطوير المواقع، وأنا مطور واجهات أمامية، كان كل ما نحتاجه هو تعلم جافاسكربت، CSS وطبعا HTML. أما الآن فنتحدث عن Node ،Npm ،Babel ،Webpack ،Typescript إلخ….
المبتدؤون يقولون: “نحن لا نفهم كل هذا الهراء ؟! لماذا كل هذه التعقيدات لمجرد صناعة صفحة ويب ليست بالضرورة بهذه الدرجة من التعقيد ؟”
هدفي من هذا الموضوع هو القيام بسرد تاريخي لتطور بيئة الجافاسكريبت حتى أضحت على ما هي عليه الآن في 2018، مع اكتشاف وتبيان الإشكاليات التي جاءت كل هذه الأدوات لحلها.
سنبدأ بمثال صغير عن كيفية تطوير الواجهة الأمامية لموقع ويب بالطريقة القديمة التي بدأت بها مسيرتي المهنية. ثم سنتدرج ونصعد شيئا فشيئا في السلم ونقوم بإدراج الأدوات التي ذكرناها أعلاه، أداة بأداة. أي أنني سأقوم بمحاكاة السياق التاريخي لتطور النظام البيئي ل JavaScript. في الأخير، أما متأكد بأنك ستفهم دور كل Npm ،Webpack ،Babel وأخواتهم وستقرر بعد ذلك إن كانت فعلا تستحق الإستعانة بها أم لا.
لنفترض أنه لدينا ملف script.js يحتوي على أكواد الجافاسكربت التي قمنا بكتابتها، ليكن محتوى هذا الملف ببساطة هو كالتالي :
// script.js console.log("مرحبا في ملف الجافاسكربت!");
1
2
// script.js
console.log("مرحبا في ملف الجافاسكربت!");
الطريقة القديمة في إضافة الجافاسكربت لصفحات html تتمثل في استدعاء ملف جافاسكربت script.js من الصفحة index.html باستخدام الوسم <script> كما يلي :
<!-- index.html --> <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>مثال لجافاسكربت</title> <script src="script.js"></script> </head> <body> <h1>مرحبا بكم!</h1> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>مثال لجافاسكربت</title>
<script src="script.js"></script>
</head>
<body>
<h1>مرحبا بكم!</h1>
</body>
</html>
هذا كل ما نحتاجه لإضافة أكواد جافاسكربت لموقعنا!
لنقل بأننا نريد بعد ذلك إضافة مكتبة جافاسكربت خارجية، مثلا Moment.js (هذه المكتبة تقوم بتحويل التواريخ البرمجية إلى تواريخ يستطيع الإنسان العادي قراءتها). على سبيل المثال يمكننا استخدامها كما يلي :
moment("20111031", "YYYYMMDD").fromNow(); // منذ ٦ أعوام
1
moment("20111031", "YYYYMMDD").fromNow(); // منذ ٦ أعوام
بطبيعة الحال لن تستطيع استخدام هذه المكتبة إلا إذا قمت باستدعائها من الصفحة. في الصفحة الرئيسية ل moment.js هناك تعليمات، كما في الصورة أسفله، لمساعدتنا على إضافة هذه المكتبة لمشروعنا :
ما كل هذه الأكواد الغريبة ؟! عفوا نسيت بأننا ما نزال مع مدرسة جافاسكربت القديمة. سنعود لهذه الأكواد فيما بعد، ولكن دعونا الآن فقط نقوم بتحميل مكتبة Moment.js يدويا ثم نقوم بإضافة الملف moment.min.js لمشروعنا واستدعاؤه في صفحة index.html كما فعلنا مع الملف الأول الخاص بنا :
<!-- index.html --> <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>مثال لجافاسكربت</title> <link rel="stylesheet" href="style.css"> <script src="moment.min.js"></script> <script src="script.js"></script> </head> <body> <h1>مرحبا بكم!</h1> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>مثال لجافاسكربت</title>
<link rel="stylesheet" href="style.css">
<script src="moment.min.js"></script>
<script src="script.js"></script>
</head>
<body>
<h1>مرحبا بكم!</h1>
</body>
</html>
لاحظوا بأننا قمنا بإضافة الملف moment.min.js قبل الملف script.js حتى نستطيع استخدام الدالة moment في شفرتنا البرمجية. هذا الترتيب ضروري وإلا سيظهر لنا الخطأ بعد التنفيذ.
// script.js console.log("مرحبا في ملف الجافاسكربت!"); console.log(moment().startOf('day').fromNow());
1
2
3
// script.js
console.log("مرحبا في ملف الجافاسكربت!");
console.log(moment().startOf('day').fromNow());
هكذا كنا نستخدم مكتبات الجافاسكريبت في مشاريعنا. الجميل في هذه الطريقة أنها سهلة الفهم والإستيعاب، والسيء فيها أن كل شيء يتم بشكل يدوي. فمثلا ماذا لو أردنا تحديث المكتبات التي نقوم باستخدامها في حال توفر تحديثات جديدة لها ؟ طبعا كنا نقوم بإعادة تحميل النسخة الأحدث ونعوض بها النسخة القديمة، يدوياًّ!
استخدام مدير الحزم الخاص بجافاسكريبت (npm)
بداية من عام 2010، بدأت مجموعة من ما يسمى مدراء الحزم الخاصة بلغة البرمجة جافاسكريبت بالظهور، هدفها كان واضحا : تحميل وتحديث حزم (مكتبات) الجافاسكريبت من مكان واحد. مدير الحزم Bower كان هو الأكثر شهرة في 2013، ولكن سرعان ما انتزع منه npm مكانة الريادة. أما في الأشهر الأخيرة، فقد ظهر مدير حزم جديد اسمه yarn (طورته شركة فيسبوك) ويبدو أنه يلقى قبولا حسنا لدى المطورين ولو أنه في الأول والأخير يعتمد على npm خلف الكواليس، حيث يقوم بتحميل الحزم من مستودع Node Package Manager (أو npm).
تجدر الإشارة إلى أن مدير الحزم npm صمم في الأول خصيصا ل Node.js، تقنية تمكننا من تشغيل الجافاسكريبت خارج المتصفحات (في الخوادم). ولكن بعد ذلك أصبح مديرا للحزم يهم لغة الجافاسكريبت بصفة عامة، سواءً جافاسكربت المتصفحات أو جافاسكريبت Node.js.
اقرأ أيضا: ما هو Node.js وما هي مميزاته ؟
يتم التعامل مع npm من نافذة الأوامر السطرية (command line)، وهي النافذة التي لم نكن قط نفتحها كمطوري واجهات أمامية. وهنا لا بد أن أنصح جميع المطورين المبتدئين بضرورة تعلم أساسيات التعامل مع الأوامر السطرية لأنها تستخدم بشكل كبير جدا في جميع مجالات البرمجة في وقتنا الحالي.
لنرى كيفية استخدام مدير الحزم npm لتحميل مكتبة Moment.js عوض تحميلها ذلك بشكل يدوي كما فعلنا سابقا. يجب تثبيت Node.js على حاسوبك، أتوماتيكيا سيتم تثبيت مدير الحزم npm كذلك. بعد ذلك سنفتح نافذة الأوامر السطرية، وانطلاقا منها ندخل للمجلد الذي يضم مشروعنا (حيث يوجد الملف index.html) ثم نقوم بتنفيذ الأمر التالي :
npm init
1
npm init
بعد التنفيذ، سيُطلَب منك الإجابة على عدد من الأسئلة (يمكنك فقط الضغط على الزر Enter عند كل سؤال)، وبعدها سيتم توليد ملف اسمه package.json داخل نفس المجلد. هذا الملف يستخدمه مدير الحزم npm للقيام بالعديد من المهام المتعلقة بهذا المشروع. المحتوى الإفتراضي لهذا الملف يكون على الشكل التالي :
{ "name": "your-project-name", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1" }, "author": "", "license": "ISC" }
1
2
3
4
5
6
7
8
9
10
11
{
"name": "your-project-name",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC"
}
الآن لتنزيل المكتبة moment.js، نقوم باستخدام الأمر السطري التالي :
npm install moment --save
1
npm install moment --save
هذا الأمر يقوم بشيئين اثنين :
1
2
3
4
5
6
7
8
9
10
11
12
13
14
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.19.1"
}
}
هذه الطريقة مفيدة، خاصة إذا أردنا فيما بعد مشاركة المشروع مع الآخرين، حيث نقوم بإرسال مجلد المشروع مع الملف package.json من دون المجلد node_modules الذي قد يكون حجمه كبيرا بحسب حجم المشروع وعدد الحزم التي يستخدمها. بعد ذلك يكفي أن يقوم الآخر بتنفيذ الأمر npm install داخل المجلد وسيتم تحميل جميع حزم وتبعيات المشروع بشكل أوتوماتيكي. هذا بالضبط ما يقوم به جميع أصحاب المشاريع المشاريع على منصة Github. حيث يُطلب من مدير النسخ Git إهمال هذا المجلد عن طريق الملف “.gitignore”. في النهاية الملف package.json سيمكننا من تحميل المجلد node_modules والحزم الخاصة بهذا المشروع بعد تحميله من Github.
من الآن لن نقوم مجدد بتحميل مكتبات الجافاسكريبت بشكل يدوي، مدير الحزم npm يقوم بالمهمة بشكل جيد نيابة عنا.
بالنظر في داخل المجلد node_modules سنجد الملف moment.min.js في المسار node_modules/moment/min. إذن سنقوم باستدعاء moment.min.js في ملف index.html كما يلي :
<!-- index.html --> <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>مثال لجافاسكربت</title> <link rel="stylesheet" href="style.css"> <script src="node_modules/moment/min/moment.min.js"></script> <script src="script.js"></script> </head> <body> <h1>مرحبا بكم!</h1> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>مثال لجافاسكربت</title>
<link rel="stylesheet" href="style.css">
<script src="node_modules/moment/min/moment.min.js"></script>
<script src="script.js"></script>
</head>
<body>
<h1>مرحبا بكم!</h1>
</body>
</html>
هذا جميل، استطعنا تحميل مكتبتنا بالإستعانة بمدير الحزم npm. ولكن مازلنا مطالبين بالدخول للمجلد node_modules والبحث عن ملف المكتبة ثم استدعاؤه بشكل يدوي في ملف html. نريد حلا يمكننا من استدعاء المكتبة بشكل أوتوماتيكي حينما نحتاجها
الإستعانة بمُجَمِّع لوحدات جافاسكريبت Module Bundler
معظم لغات البرمجة تتمتع بإمكانية استدعاء الشفرة البرمجية من ملف لملف آخر. لغة البرمجة JavaScript لم تكن مصممة لهذا الغرض، لأنها في البداية كان معدة لتشتغل داخل المتصفح فقط، وبالتالي لم يكن بالإمكان الوصول لملفات الحاسوب وذلك راجع بطبيعة الحال لأسباب تتعلق بالأمن والحماية.
لوقت طويل كان كل ما نفعله هو مشاركة محتوى ملفات الجافاسكربت في مشروعنا على شكل متغيرات عامة (Global Variables). هذا بالذات ما فعلناه أعلاه، حيث أن استدعاءنا لمكتبة moment.min.js يعني بأن جميع المكتبات والملفات الأخرى التي سيتم استدعاؤها أسفلها (كما هو الحال بالنسبة ل script.js) سيكون بإمكانها الوصول للدالة moment() حتى ولو كانت في غير حاجة إليها.
في عام 2009، ظهر مشروع طموح اسمه CommonJs هدفه إتاحة نظام بيئي لجافاسكربت خارج المتصفحات. الجانب الكبير والأهم من مشروع CommonJs كان ذلك الجزء المتعلق باستيراد وتصدير (Import & Export Modules) الشفرة البرمجية عبر الملفات، والتخلص من إشكالية المتغيرات العامة. التنفيذ والتطبيق (Implementation) الأشهر لمشروع وحدات CommonJs هو Node.js.
كما ذكرنا سابقا، Node.js هو بيئة جافاسكربت (Javascript Runtime) مصممة للإشتغال في الخوادم. سنرى فيما يلي كيف نعيد صياغة استدعاء مكتبة Moment باستخدام طريقة الوحدات في Node.js. عوض استخدام الوسم <script> لإستدعاء المكتبة داخل ملف index.html، سنقوم باستدعائها مباشرة من داخل ملف الجافاسكريبت script.js :
// script.js var moment = require('moment'); console.log("Hello from JavaScript!"); console.log(moment().startOf('day').fromNow());
1
2
3
4
// script.js
var moment = require('moment');
console.log("Hello from JavaScript!");
console.log(moment().startOf('day').fromNow());
هكذا تعمل الوحدات في Node.js، طبعا لأن الأخير يشتغل في الخوادم ويستطيع الوصول لنظام الملفات بالحاسوب ويعلم جيدا أين يجد الوحدة التي نقوم باستدعائها. فعوضا عن كتابة require('./node_modules/moment/min/moment.min.js) ، نقوم فقط بكتابة require('moment') .
هذا جيدا ورائع مع Node.js، ولكن ليس مع المتصفح :'( الأخير لا يستطيع، كما أسلفنا، الوصول إلى نظام الملفات بالحاسوب، وبالتالي فتحميل الوحدات أمر في غير المتناول والدالة require غير معرفة (Not Defined) أساسا في البيئة التشغيلية للمتصفح.
أول مُجَمِّع للوحدات Module Bundler كان هو Browserify، ظهر في عام 2011 وكان الحل الذي مكن مطوري الواجهات الأمامية من استخدام طريقة Node.js في التعامل مع الوحدات. وكان له الفضل كذلك في أن يصبح مدير الحزم npm خيارا قويا لدى مطوري الواجهات الأمامية بعدما كان دائما حكرا على مطوري Node.js. ومنذ عام 2015 تقريبا، أصبح Webpack هو مجمع الوحدات الأكثر استخداما، مستفيدا من الشعبية الجارفة لمكتبة React.js التي اختار مطوروها Webpack ليكون مجمع وحداتها الرسمي.
لنأخذ نظرة عن كيفية الإستعانة ب Webpack حتى نجعل المثال أعلاه ( require('moment')) يعمل في المتصفح. أولا علينا تثبيت Webpack في مشروعنا. Webpack نفسه عبارة عن حزمة npm، لذلك سنقوم بتحميله وإضافته للمشروع تماما كما فعلنا مع مكتبة moment.js قبل قليل :
npm install webpack --save-dev
1
npm install webpack --save-dev
لاحظ أننا استخدمنا البارامتر –save-dev عوض –save الذي استخدمناه سابقا مع moment.js. السبب أننا نريد حفظ Webpack كتبعية خاصة فقط بمرحلة التطوير في ملف package.json وليس تبعية للمشروع في مرحلة الإنتاج (Production stage). ملف package.json تم تحديثه تلقائيا بعد تحميل Webpack وأصبح على هذا النحو :
{ "name": "modern-javascript-example", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1" }, "author": "", "license": "ISC", "dependencies": { "moment": "^2.19.1" }, "devDependencies": { "webpack": "^3.7.1" } }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.19.1"
},
"devDependencies": {
"webpack": "^3.7.1"
}
}
الآن وقد قمنا بتثبيت Webpack بنجاح وتحميله على مجلد الحزم node_modules الخاص بمشروعنا، سيمكننا استخدامه من نافذة الأوامر السطرية بهذه الكيفية :
./node_modules/.bin/webpack script.js bundle.js
1
./node_modules/.bin/webpack script.js bundle.js
هذا الأمر يقول لأداة Webpack (الموجودة داخل المجلد node_modules ) : خذ الملف script.js وابحث فيه عن جميع التعابير التي تحتوي على دالة الإستيراد require وقم باستبدالها بالشفرة البرمجية للوحدة التي تم استيرادها، ثم احفظ المخرج الجديد(Output) في ملف جديد اسمه bundle.js. يعني هذا أننا لن نستخدم من جديد الملف script.js في متصفحنا، بل نقوم عوضا عن ذلك باستدعاء الملف bundle.js المتوافق مع إمكانيات المتصفحات، أما script.js فنقوم باستخدامه فقط في مرحلة البرمجة والتطوير ولا يظهر في بيئة الإنتاج النهائية.
<!-- index.html --> <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>مثال لجافاسكربت</title> <script src="bundle.js"></script> </head> <body> <h1>مرحبا بكم!</h1> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>مثال لجافاسكربت</title>
<script src="bundle.js"></script>
</head>
<body>
<h1>مرحبا بكم!</h1>
</body>
</html>
بعد إعادة تحميل الصفحة على المتصفح، سنرى بأن كل شيء يعمل جيدا كما في السابق.
يجب علينا إعادة تنفيذ الأمر السابق في كل مرة نقوم فيها بالتعديل على ملف script.js. هذا الأمر قد يكون مملا وغير عملي، خاصة عندما تزداد درجة تعقيد مهام Webpack في مشروعنا، فما رأيناه أعلاه يمكن اعتباره أبسط ما يمكن ل Webpack القيام به.
ربما يجدر بنا الآن اكتشاف الملف webpack.config.js الذي يعتبر المكان الذي نقوم فيه بإضافة إعدادات Webpack الخاصة بالمشروع. لنقم بإنشاء هذا الملف ونضيف إليه الإعدادات التالية :
// webpack.config.js module.exports = { entry: './script.js', output: { filename: 'bundle.js' } };
1
2
3
4
5
6
7
// webpack.config.js
module.exports = {
entry: './script.js',
output: {
filename: 'bundle.js'
}
};
الآن يمكننا تنفيذ نفس المهمة السابقة دون تحديد أسماء الملفين (script.js و bundle.js ) من الأمر السطري. يكفي تنفيذ ما يلي وسيقوم Webpack بمعرفة ما يجب عليه فعله بالإعتماد على التوجيهات في ملف webpack.config.js :
./node_modules/.bin/webpack
1
./node_modules/.bin/webpack
تقدمنا خطوة إلى الأمام، ولكن لا يزال هذا مضجرا لأننا مازلنا مطالبين بتنفيذ الأمر في كل مرة نعدل فيها ملف script.js.
سنواصل التقدم وإضافة مميزات أخرى قوية لبيئتنا التطويرية.
استخدام الميزات الجديدة في جافاسكريبت (Babel)
مع التطور الكبير في لغة البرمجة جافاسكريبت في الأعوام الأخيرة، أصبح من الصعب على المتصفحات مواكبة هذه التطورات ودمج كل هذه الميزات الجديدة في بيئاتها التشغيلية (Runtime). فمثلا هناك عدد كبير جدا من ميزات إصدار الجافاسكريبت ES6 ليست مدعومة بعد من جميع المتصفحات الكبيرة، هذا يمنع المطورين من استخدامها والإستفادة من الإمكانية العظيمة التي تتيحها.
لحسن الحظ، ظهرت في الفترة الماضية أدوات تعرف باسم Javascript Transpilers هدفها كتابة الجافاسكريبت بطريقة أكثر حداثة وانتاجية، وعند الإنتهاء من التكويد تتم عملية Transpiling التي تحول الشفرة البرمجية التي كتبها المطور إلى كود جافاسكريبت اعتيادي تفهمه وتدعمه جميع المتصفحات.
CoffeeScript كان أول هذه الأدوات التي لقيت نجاحا كبيرا حيث يمكن من كتابة أكواد جافاسكريبت بطريقة حديثة وأكثر انتاجية. ولكن في الفترة الأخيرة خفت نجم CoffeeScript وبرز في الساحة لاعبين جديدين استحوذا على أكبر نصيب من الكعكعة، نتحدث عن Babel و Typescript.
Typescript طورته شركة مايكروسوفت ويعتبر بمثابة لغة جديدة استلهمت العديد من الأشياء المميزة في لغة البرمجة سي شارب #C. يعني أنه يمكن للمطورين كتابة لغة جديدة والإستفادة من بعض نقاط قوة #C وبعد الإنتهاء من كتابة الكود يتم تحويله لجافاسكريبت اعتيادي يفهمه المتصفح.
يعتبر إطار العمل Angular أشهر التقنيات التي تعتمد على Typescript بشكل افتراضي، وكان له الفضل الكبير في الشهرة التي حققها الأخير في الآونة الأخيرة.
أما Babel فلايمكن اعتباره لغة برمجة، ولكن فقط أداة تمكننا من كتابة أكواد الجافاسكريبت على الطريقة الحديثة (ES6) وبعد ذلك تحويلها لأكواد جافاسكريبت مدعومة من جميع المتصفحات الكبيرة.
معظم المطورين يفضلون Babel لأنه كما قلنا ليس لغة برمجة جديدة يجب تعلمها بل فقط أداة تجعل الجافاسكريبت الحديث متوافق مع قدرات المتصفحات الحالية.
لنكتشف معا كيف يمكننا استخدام Babel في مشروعنا مع مجمع الوحدات Webpack. أولا علينا تحميل Babel من مدير الحزم npm بالطريقة التالية المعتادة (أصبحت معتادة الآن ) :
npm install babel-core babel-preset-env babel-loader --save-dev
1
npm install babel-core babel-preset-env babel-loader --save-dev
لاحظ أننا قمنا بتحميل 3 حزم دفعة واحدة كتبعيات خاصة بمرحلة التطوير (save-dev) :
// webpack.config.js module.exports = { entry: './script.js', output: { filename: 'bundle.js' }, module: { rules: [ { test: /\.js$/, exclude: /node_modules/, use: { loader: 'babel-loader', options: { presets: ['env'] } } } ] } };
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
// webpack.config.js
module.exports = {
entry: './script.js',
output: {
filename: 'bundle.js'
},
module: {
rules: [
{
test: /\.js$/,
exclude: /node_modules/,
use: {
loader: 'babel-loader',
options: {
presets: ['env']
}
}
}
]
}
};
الشكل الجديد لهذا الملف قد يكون مربكا أو غير مفهوم بالنسبة لك، هذا طبيعي كونها المرة الأولى التي تتعامل فيها مع Webpack.
ببساطة قلنا لصديقنا Webpack : أنظر لجميع الملفات التي امتدادها js. باستثناء الملفات الموجودة داخل المجلد node_modules، واستعن بالحزمتين babel-loader و babel-preset-env للتمكن من عملية تحويل (Transpiling) أكواد الجافاسكريبت.
الآن بتنا نستطيع البدء في كتابة أكواد الجافاسكريبت مع أحدث الميزات الجديدة في هذه اللغة. هذا مثال لكتابة كلاس جافاسكريبت على طريقة ES2015 في ملف script.js :
// script.js var moment = require('moment'); console.log("Hello from JavaScript!"); console.log(moment().startOf('day').fromNow()); class User { constructor(name) { this.name = name; } sayHi() { alert(this.name); } } let user = new User("Aissa"); user.sayHi();
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// script.js
var moment = require('moment');
console.log("Hello from JavaScript!");
console.log(moment().startOf('day').fromNow());
class User {
constructor(name) {
this.name = name;
}
sayHi() {
alert(this.name);
}
}
let user = new User("Aissa");
user.sayHi();
نستطيع كذلك استخدام طريقة ES2015 في استيراد الوحدات عوض معيارية CommonJs التي رأيناها في السابق. هذه الطريقة هي التي تستعمل في جل المشاريع في الوقت الحالي :
// script.js import moment from 'moment'; console.log("Hello from JavaScript!"); console.log(moment().startOf('day').fromNow()); class User { constructor(name) { this.name = name; } sayHi() { alert(this.name); } } let user = new User("Aissa"); user.sayHi();
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// script.js
import moment from 'moment';
console.log("Hello from JavaScript!");
console.log(moment().startOf('day').fromNow());
class User {
constructor(name) {
this.name = name;
}
sayHi() {
alert(this.name);
}
}
let user = new User("Aissa");
user.sayHi();
طريقة import الخاصة بإصدارات جافاسكريبت الحديثة لا تختلف كثيرا عن require التي استنبطها Node.js من مشروع CommonJs، غير أن الأولى — أي import — تعتبر أكثر مرونة وسلاسة خاصة في الحالات المتقدمة.
بما أننا قمنا بالتعديل على الملف script.js فعلينا من جديد أن نطلب من Webpack أن يقوم بعملية التجميع (Bundling) :
./node_modules/.bin/webpack
1
./node_modules/.bin/webpack
الآن يمكنك إعادة تحميل صفحة index.html، وسترى بأنها تعمل وفق المتوقع من دون مشاكل حتى في المتصفحات القديمة التي لا تدعم نهائيا ميزة الكلاسات في جافاسكريبت مثل انترنت إكسبلورر IE9.
وإذا كنت فضوليا أكثر (مبروك إذا كان الحال كذلك) فستذهب للملف bundle.js وتبحث عن الشكل الجديد للكلاس الذي قمت بكتابته سابقا في ملف script.js، هكذا أصبح حاله :
//bundle.js //... var _createClass = function() { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function(Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var User = function() { function User(name) { _classCallCheck(this, User); this.name = name; } _createClass(User, [{ key: "sayHi", value: function sayHi() { alert(this.name); } }]); return User; }(); var user = new User("Aissa"); user.sayHi(); //...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
//bundle.js
//...
var _createClass = function() {
function defineProperties(target, props) {
for (var i = 0; i < props.length; i++) {
var descriptor = props;
descriptor.enumerable = descriptor.enumerable || false;
descriptor.configurable = true;
if ("value" in descriptor) descriptor.writable = true;
Object.defineProperty(target, descriptor.key, descriptor);
}
}
return function(Constructor, protoProps, staticProps) {
if (protoProps) defineProperties(Constructor.prototype, protoProps);
if (staticProps) defineProperties(Constructor, staticProps);
return Constructor;
};
}();
function _classCallCheck(instance, Constructor) {
if (!(instance instanceof Constructor)) {
throw new TypeError("Cannot call a class as a function");
}
}
var User = function() {
function User(name) {
_classCallCheck(this, User);
this.name = name;
}
_createClass(User, [{
key: "sayHi",
value: function sayHi() {
alert(this.name);
}
}]);
return User;
}();
var user = new User("Aissa");
user.sayHi();
//...
نعم أعلم أن هذا الكود معقد أكثر وأقل جمالية، Babel قام باستخدام الجافاسكريبت القديم (استخدام الدوال، تحويل let إلى var) حتى نحصل في الأخير على الشفرة البرمجية التي تفهمها جميع المتصفحات المعروفة.
مثال الكلاس بسيط بالمقارنة مع كل مزايا جافاسكريبت الحديثة والتي يدعمها Babel. الأخير تستعين به React.js لتحويل أكواد JSX خاصتها إلى أكواد جافاسكريبت اعتيادية، لذلك صدقني أخي العزيز إن قلت لك أنه بعد تقدمك في الجافاسكريبت ستجد بابل في كل مكان.
الآن تتضح الأمور أكثر ولم يتبق لنا إلا بعض اللمسات الأخيرة. أول هذه اللمسات هي ضغط (Minify) الملفات المجمعة (في حالتنا bundle.js) حتى نقلل من حجمه قدر المستطاع. ثم بعد ذلك سنضيف إمكانية التشغيل التلقائي ل Webpack كلما قمنا بالتعديل على ملف script.js حتى لا نطلب منه ذلك في كل مرة من واجهة الأوامر السطرية.
الإستعانة بمشغلات المهام (Task Runner)
الآن وقد تمكننا بنجاح من تجميع وحداتنا البرمجية بعد كتابتها بأحدث المعايير والطرق المتبعة، سيكون من المنطقي أتمتة ما نستطيع من مختلف المهام التي تشكل أجزاء مهمة من عملية التطوير، نتحدث عن مهام من قبيل ضغط ملفات الأكواد (جافاسكريبت و css)، تحسين الصور، تنفيذ إختبارات الشفرة البرمجية Unit Tests، إلخ…
في فترة من الفترات (حوالي 2013)، كان GruntJs هو مشغل المهام الأشهر، وبعده جاء Gulp بقوة وتقاسما معا الكعكة. كلاهما يعتمدان على الإضافات حيث كل إضافة تقوم بمهمة محددة.
ولكن في الآونة الأخيرة، وبالخصوص منذ انفجار شعبية React.js، بات المطورون يستخدمون Webpack (الذي هو في الأصل مجمع وحدات Module Bundler وليس مشغل مهام Task Runner) في تنفيذ المهام بالإعتماد على أوامر NPM نفسه. هذه الأوامر تعرف باسم NPM Scripts ونصرح بها في ملف package.json :
{ "name": "modern-javascript-example", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1", "build": "webpack -p", "watch": "webpack --watch" }, "author": "", "license": "ISC", "dependencies": { "moment": "^2.19.1" }, "devDependencies": { "babel-core": "^6.26.0", "babel-loader": "^7.1.2", "babel-preset-env": "^1.6.1", "webpack": "^3.7.1" } }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"build": "webpack -p",
"watch": "webpack --watch"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.19.1"
},
"devDependencies": {
"babel-core": "^6.26.0",
"babel-loader": "^7.1.2",
"babel-preset-env": "^1.6.1",
"webpack": "^3.7.1"
}
}
في هذا المثال، حددنا سكريبتين : build و watch. لتنفيذ السكريبت نقوم بإدخال الأمر npm run متبوع باسم السكريبت.
على سبيل المثال، لتنفيذ السكريبت build نقوم بإدخال الأمر التالي في Command Line :
npm run build
1
npm run build
هذا السكريبت يقوم بتشغيل Webpack بالإعتماد على إعدادات ملف webpack.config.js، وأضفنا البارامتر p- الذي يقوم ل Webpack بأننا نريد تجميع الكود لمرحلة الإنتاج (Production)، أي ضغطه (Minify code).
npm run watch
1
npm run watch
أما السكريبت watch فيشغل Webpack في كل مرة نقوم فيها بعملية بالتعديل. يعني لن نظطر من الآن لكتابة الأمر في واجهة الأوامر السطرية بعد كل تعديل. خطوة عملية للغاية في مرحلة التطوير
لاحظ بأنه في سكريبتات npm لا نقوم بتعيين المسار الكامل ./node_modules/.bin/webpack ، لأن Node.js يعلم جيدا أين يجد كل حزمة بالإعتماد على اسمها فقط.
هذا رائع! يمكننا أن جعله أكثر روعة بتثبيت أداة أخرى رائعة : Webpack-dev-server. هذه الحزمة تعتمد كما نرى من اسمها على Webpack وتضع بين يدي المطور خادم ويب مع خاصية Live Reloading التي تعطينا إمكانية رؤية التغييرات في المتصفح دون القيام بإعادة تحميل الصفحة كل مرة، يعني بمجرد تعديل وحفظ ملف script.js سنرى النتيجة في المتصفح في ظرف أجزاء بالمئة من الثانية
الآن تعلمون كيف تقومون بتحميل هذه الأداة، أحسنتم، الطريقة هي :
npm install webpack-dev-server --save-dev
1
npm install webpack-dev-server --save-dev
بعد ذلك قم بإضافة سكريبت جديد لملف package.json بهذه الكيفية :
{ "name": "modern-javascript-example", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1", "build": "webpack -p", "watch": "webpack --watch", "server": "webpack-dev-server --open" }, "author": "", "license": "ISC", "dependencies": { "moment": "^2.19.1" }, "devDependencies": { "babel-core": "^6.26.0", "babel-loader": "^7.1.2", "babel-preset-env": "^1.6.1", "webpack": "^3.7.1", "webpack-dev-server": "^2.7.1" } }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"build": "webpack -p",
"watch": "webpack --watch",
"server": "webpack-dev-server --open"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.19.1"
},
"devDependencies": {
"babel-core": "^6.26.0",
"babel-loader": "^7.1.2",
"babel-preset-env": "^1.6.1",
"webpack": "^3.7.1",
"webpack-dev-server": "^2.7.1"
}
}
الآن يمكنك بدء تشغيل خادم التطوير الخاص بمشروعك بهذه الطريقة :
npm run server
1
npm run server
هذه الأمر يقوم بفتح الصفحة index.html في متصفحك مع عنوان محلي افتراضي localhost:8080، وكلما عدلت على الملف script.js يقوم webpack-dev-server بإعادة جمع ال Bundle وتحديث حالة الصفحة في المتصفح بشكل تلقائي. هذا يوفر وقتا ثمينا على المطورين ما يجعلهم يركزون أكثر على عملهم في كتابة كود جيد عوض الإنتقال بين المتصفح والمحرر بعد كل تعديل.
هذه أمثلة بسيطة فقط مما يمكن القيام به مع Webpack و Webpack-dev-server. تستطيعون استغلال سكريبتات npm لتشغيل مهام أخرى مثل تحويل أكواد Sass إلى css، ضغط الصور وغيرها …
أدعوكم لزيارة الموقع الرسمي لكل منهما لاكتشاف المزيد من مميزاتهما.
في بداياتي مع تطوير المواقع، وأنا مطور واجهات أمامية، كان كل ما نحتاجه هو تعلم جافاسكربت، CSS وطبعا HTML. أما الآن فنتحدث عن Node ،Npm ،Babel ،Webpack ،Typescript إلخ….
المبتدؤون يقولون: “نحن لا نفهم كل هذا الهراء ؟! لماذا كل هذه التعقيدات لمجرد صناعة صفحة ويب ليست بالضرورة بهذه الدرجة من التعقيد ؟”
هدفي من هذا الموضوع هو القيام بسرد تاريخي لتطور بيئة الجافاسكريبت حتى أضحت على ما هي عليه الآن في 2018، مع اكتشاف وتبيان الإشكاليات التي جاءت كل هذه الأدوات لحلها.
سنبدأ بمثال صغير عن كيفية تطوير الواجهة الأمامية لموقع ويب بالطريقة القديمة التي بدأت بها مسيرتي المهنية. ثم سنتدرج ونصعد شيئا فشيئا في السلم ونقوم بإدراج الأدوات التي ذكرناها أعلاه، أداة بأداة. أي أنني سأقوم بمحاكاة السياق التاريخي لتطور النظام البيئي ل JavaScript. في الأخير، أما متأكد بأنك ستفهم دور كل Npm ،Webpack ،Babel وأخواتهم وستقرر بعد ذلك إن كانت فعلا تستحق الإستعانة بها أم لا.
سيكون هذا المقال طويلا جدا، أنصحك بالجلوس في مكان مريح وإعداد فنجان قهوة أو كأس شاي قبل بدء القراءة
الطريقة القديمة في إضافة الجافاسكربت للصفحةلنفترض أنه لدينا ملف script.js يحتوي على أكواد الجافاسكربت التي قمنا بكتابتها، ليكن محتوى هذا الملف ببساطة هو كالتالي :
// script.js console.log("مرحبا في ملف الجافاسكربت!");
1
2
// script.js
console.log("مرحبا في ملف الجافاسكربت!");
الطريقة القديمة في إضافة الجافاسكربت لصفحات html تتمثل في استدعاء ملف جافاسكربت script.js من الصفحة index.html باستخدام الوسم <script> كما يلي :
<!-- index.html --> <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>مثال لجافاسكربت</title> <script src="script.js"></script> </head> <body> <h1>مرحبا بكم!</h1> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>مثال لجافاسكربت</title>
<script src="script.js"></script>
</head>
<body>
<h1>مرحبا بكم!</h1>
</body>
</html>
هذا كل ما نحتاجه لإضافة أكواد جافاسكربت لموقعنا!
لنقل بأننا نريد بعد ذلك إضافة مكتبة جافاسكربت خارجية، مثلا Moment.js (هذه المكتبة تقوم بتحويل التواريخ البرمجية إلى تواريخ يستطيع الإنسان العادي قراءتها). على سبيل المثال يمكننا استخدامها كما يلي :
moment("20111031", "YYYYMMDD").fromNow(); // منذ ٦ أعوام
1
moment("20111031", "YYYYMMDD").fromNow(); // منذ ٦ أعوام
بطبيعة الحال لن تستطيع استخدام هذه المكتبة إلا إذا قمت باستدعائها من الصفحة. في الصفحة الرئيسية ل moment.js هناك تعليمات، كما في الصورة أسفله، لمساعدتنا على إضافة هذه المكتبة لمشروعنا :
ما كل هذه الأكواد الغريبة ؟! عفوا نسيت بأننا ما نزال مع مدرسة جافاسكربت القديمة. سنعود لهذه الأكواد فيما بعد، ولكن دعونا الآن فقط نقوم بتحميل مكتبة Moment.js يدويا ثم نقوم بإضافة الملف moment.min.js لمشروعنا واستدعاؤه في صفحة index.html كما فعلنا مع الملف الأول الخاص بنا :
<!-- index.html --> <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>مثال لجافاسكربت</title> <link rel="stylesheet" href="style.css"> <script src="moment.min.js"></script> <script src="script.js"></script> </head> <body> <h1>مرحبا بكم!</h1> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>مثال لجافاسكربت</title>
<link rel="stylesheet" href="style.css">
<script src="moment.min.js"></script>
<script src="script.js"></script>
</head>
<body>
<h1>مرحبا بكم!</h1>
</body>
</html>
لاحظوا بأننا قمنا بإضافة الملف moment.min.js قبل الملف script.js حتى نستطيع استخدام الدالة moment في شفرتنا البرمجية. هذا الترتيب ضروري وإلا سيظهر لنا الخطأ بعد التنفيذ.
// script.js console.log("مرحبا في ملف الجافاسكربت!"); console.log(moment().startOf('day').fromNow());
1
2
3
// script.js
console.log("مرحبا في ملف الجافاسكربت!");
console.log(moment().startOf('day').fromNow());
هكذا كنا نستخدم مكتبات الجافاسكريبت في مشاريعنا. الجميل في هذه الطريقة أنها سهلة الفهم والإستيعاب، والسيء فيها أن كل شيء يتم بشكل يدوي. فمثلا ماذا لو أردنا تحديث المكتبات التي نقوم باستخدامها في حال توفر تحديثات جديدة لها ؟ طبعا كنا نقوم بإعادة تحميل النسخة الأحدث ونعوض بها النسخة القديمة، يدوياًّ!
استخدام مدير الحزم الخاص بجافاسكريبت (npm)
بداية من عام 2010، بدأت مجموعة من ما يسمى مدراء الحزم الخاصة بلغة البرمجة جافاسكريبت بالظهور، هدفها كان واضحا : تحميل وتحديث حزم (مكتبات) الجافاسكريبت من مكان واحد. مدير الحزم Bower كان هو الأكثر شهرة في 2013، ولكن سرعان ما انتزع منه npm مكانة الريادة. أما في الأشهر الأخيرة، فقد ظهر مدير حزم جديد اسمه yarn (طورته شركة فيسبوك) ويبدو أنه يلقى قبولا حسنا لدى المطورين ولو أنه في الأول والأخير يعتمد على npm خلف الكواليس، حيث يقوم بتحميل الحزم من مستودع Node Package Manager (أو npm).
تجدر الإشارة إلى أن مدير الحزم npm صمم في الأول خصيصا ل Node.js، تقنية تمكننا من تشغيل الجافاسكريبت خارج المتصفحات (في الخوادم). ولكن بعد ذلك أصبح مديرا للحزم يهم لغة الجافاسكريبت بصفة عامة، سواءً جافاسكربت المتصفحات أو جافاسكريبت Node.js.
اقرأ أيضا: ما هو Node.js وما هي مميزاته ؟
يتم التعامل مع npm من نافذة الأوامر السطرية (command line)، وهي النافذة التي لم نكن قط نفتحها كمطوري واجهات أمامية. وهنا لا بد أن أنصح جميع المطورين المبتدئين بضرورة تعلم أساسيات التعامل مع الأوامر السطرية لأنها تستخدم بشكل كبير جدا في جميع مجالات البرمجة في وقتنا الحالي.
لنرى كيفية استخدام مدير الحزم npm لتحميل مكتبة Moment.js عوض تحميلها ذلك بشكل يدوي كما فعلنا سابقا. يجب تثبيت Node.js على حاسوبك، أتوماتيكيا سيتم تثبيت مدير الحزم npm كذلك. بعد ذلك سنفتح نافذة الأوامر السطرية، وانطلاقا منها ندخل للمجلد الذي يضم مشروعنا (حيث يوجد الملف index.html) ثم نقوم بتنفيذ الأمر التالي :
npm init
1
npm init
بعد التنفيذ، سيُطلَب منك الإجابة على عدد من الأسئلة (يمكنك فقط الضغط على الزر Enter عند كل سؤال)، وبعدها سيتم توليد ملف اسمه package.json داخل نفس المجلد. هذا الملف يستخدمه مدير الحزم npm للقيام بالعديد من المهام المتعلقة بهذا المشروع. المحتوى الإفتراضي لهذا الملف يكون على الشكل التالي :
{ "name": "your-project-name", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1" }, "author": "", "license": "ISC" }
1
2
3
4
5
6
7
8
9
10
11
{
"name": "your-project-name",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC"
}
الآن لتنزيل المكتبة moment.js، نقوم باستخدام الأمر السطري التالي :
npm install moment --save
1
npm install moment --save
هذا الأمر يقوم بشيئين اثنين :
- أولا، يقوم بتحميل الحزمة moment.js داخل مجلد اسمه node_modules يتم إنشاؤه تلقائيا.
- ثانيا، يقوم بتحديث الملف package.json وذلك بإضافة اسم الحزمة moment في جزء dependencies حيث يتم تسجيل كافة الحزم التي يحتاجها هذا المشروع.
أظنك الآن فهمت (ولو قليلا) الأكواد الموجودة في الصفحة الرئيسية لمكتبة moment.js عن كيفية تثبيتها (الصورة أعلاه)
{ "name": "modern-javascript-example", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1" }, "author": "", "license": "ISC", "dependencies": { "moment": "^2.19.1" } }1
2
3
4
5
6
7
8
9
10
11
12
13
14
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.19.1"
}
}
هذه الطريقة مفيدة، خاصة إذا أردنا فيما بعد مشاركة المشروع مع الآخرين، حيث نقوم بإرسال مجلد المشروع مع الملف package.json من دون المجلد node_modules الذي قد يكون حجمه كبيرا بحسب حجم المشروع وعدد الحزم التي يستخدمها. بعد ذلك يكفي أن يقوم الآخر بتنفيذ الأمر npm install داخل المجلد وسيتم تحميل جميع حزم وتبعيات المشروع بشكل أوتوماتيكي. هذا بالضبط ما يقوم به جميع أصحاب المشاريع المشاريع على منصة Github. حيث يُطلب من مدير النسخ Git إهمال هذا المجلد عن طريق الملف “.gitignore”. في النهاية الملف package.json سيمكننا من تحميل المجلد node_modules والحزم الخاصة بهذا المشروع بعد تحميله من Github.
من الآن لن نقوم مجدد بتحميل مكتبات الجافاسكريبت بشكل يدوي، مدير الحزم npm يقوم بالمهمة بشكل جيد نيابة عنا.
بالنظر في داخل المجلد node_modules سنجد الملف moment.min.js في المسار node_modules/moment/min. إذن سنقوم باستدعاء moment.min.js في ملف index.html كما يلي :
<!-- index.html --> <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>مثال لجافاسكربت</title> <link rel="stylesheet" href="style.css"> <script src="node_modules/moment/min/moment.min.js"></script> <script src="script.js"></script> </head> <body> <h1>مرحبا بكم!</h1> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>مثال لجافاسكربت</title>
<link rel="stylesheet" href="style.css">
<script src="node_modules/moment/min/moment.min.js"></script>
<script src="script.js"></script>
</head>
<body>
<h1>مرحبا بكم!</h1>
</body>
</html>
هذا جميل، استطعنا تحميل مكتبتنا بالإستعانة بمدير الحزم npm. ولكن مازلنا مطالبين بالدخول للمجلد node_modules والبحث عن ملف المكتبة ثم استدعاؤه بشكل يدوي في ملف html. نريد حلا يمكننا من استدعاء المكتبة بشكل أوتوماتيكي حينما نحتاجها
الإستعانة بمُجَمِّع لوحدات جافاسكريبت Module Bundler
معظم لغات البرمجة تتمتع بإمكانية استدعاء الشفرة البرمجية من ملف لملف آخر. لغة البرمجة JavaScript لم تكن مصممة لهذا الغرض، لأنها في البداية كان معدة لتشتغل داخل المتصفح فقط، وبالتالي لم يكن بالإمكان الوصول لملفات الحاسوب وذلك راجع بطبيعة الحال لأسباب تتعلق بالأمن والحماية.
لوقت طويل كان كل ما نفعله هو مشاركة محتوى ملفات الجافاسكربت في مشروعنا على شكل متغيرات عامة (Global Variables). هذا بالذات ما فعلناه أعلاه، حيث أن استدعاءنا لمكتبة moment.min.js يعني بأن جميع المكتبات والملفات الأخرى التي سيتم استدعاؤها أسفلها (كما هو الحال بالنسبة ل script.js) سيكون بإمكانها الوصول للدالة moment() حتى ولو كانت في غير حاجة إليها.
في عام 2009، ظهر مشروع طموح اسمه CommonJs هدفه إتاحة نظام بيئي لجافاسكربت خارج المتصفحات. الجانب الكبير والأهم من مشروع CommonJs كان ذلك الجزء المتعلق باستيراد وتصدير (Import & Export Modules) الشفرة البرمجية عبر الملفات، والتخلص من إشكالية المتغيرات العامة. التنفيذ والتطبيق (Implementation) الأشهر لمشروع وحدات CommonJs هو Node.js.
كما ذكرنا سابقا، Node.js هو بيئة جافاسكربت (Javascript Runtime) مصممة للإشتغال في الخوادم. سنرى فيما يلي كيف نعيد صياغة استدعاء مكتبة Moment باستخدام طريقة الوحدات في Node.js. عوض استخدام الوسم <script> لإستدعاء المكتبة داخل ملف index.html، سنقوم باستدعائها مباشرة من داخل ملف الجافاسكريبت script.js :
// script.js var moment = require('moment'); console.log("Hello from JavaScript!"); console.log(moment().startOf('day').fromNow());
1
2
3
4
// script.js
var moment = require('moment');
console.log("Hello from JavaScript!");
console.log(moment().startOf('day').fromNow());
هكذا تعمل الوحدات في Node.js، طبعا لأن الأخير يشتغل في الخوادم ويستطيع الوصول لنظام الملفات بالحاسوب ويعلم جيدا أين يجد الوحدة التي نقوم باستدعائها. فعوضا عن كتابة require('./node_modules/moment/min/moment.min.js) ، نقوم فقط بكتابة require('moment') .
هذا جيدا ورائع مع Node.js، ولكن ليس مع المتصفح :'( الأخير لا يستطيع، كما أسلفنا، الوصول إلى نظام الملفات بالحاسوب، وبالتالي فتحميل الوحدات أمر في غير المتناول والدالة require غير معرفة (Not Defined) أساسا في البيئة التشغيلية للمتصفح.
هنا سيتدخل مجمع الوحدات لحل هذه المعضلة.
مجمع وحدات الجافاسكربت (Javascript Module Bundler) هو أداة تعتمد على إمكانيات Node.js (الوصول لنظام الملفات) لإنشاء مخرجات نهائية تكون متوافقة مع قدرات المتصفح (عدم الوصول لنظام الملفات). في حالتنا، نحتاج Module Bundler يكون قادرا على إيجاد جميع دوال require (الغير متوافقة مع المتصفح) في شفرتنا البرمجية واستبدالها بمحتوى الملفات التي تقوم باستدعائها. النتيجة النهائية ستكون عبارة عن ملف جافاسكربت واحد خالٍ من دوال require ومتوافق بشكل كامل مع بيئة المتصفح.أول مُجَمِّع للوحدات Module Bundler كان هو Browserify، ظهر في عام 2011 وكان الحل الذي مكن مطوري الواجهات الأمامية من استخدام طريقة Node.js في التعامل مع الوحدات. وكان له الفضل كذلك في أن يصبح مدير الحزم npm خيارا قويا لدى مطوري الواجهات الأمامية بعدما كان دائما حكرا على مطوري Node.js. ومنذ عام 2015 تقريبا، أصبح Webpack هو مجمع الوحدات الأكثر استخداما، مستفيدا من الشعبية الجارفة لمكتبة React.js التي اختار مطوروها Webpack ليكون مجمع وحداتها الرسمي.
لنأخذ نظرة عن كيفية الإستعانة ب Webpack حتى نجعل المثال أعلاه ( require('moment')) يعمل في المتصفح. أولا علينا تثبيت Webpack في مشروعنا. Webpack نفسه عبارة عن حزمة npm، لذلك سنقوم بتحميله وإضافته للمشروع تماما كما فعلنا مع مكتبة moment.js قبل قليل :
npm install webpack --save-dev
1
npm install webpack --save-dev
لاحظ أننا استخدمنا البارامتر –save-dev عوض –save الذي استخدمناه سابقا مع moment.js. السبب أننا نريد حفظ Webpack كتبعية خاصة فقط بمرحلة التطوير في ملف package.json وليس تبعية للمشروع في مرحلة الإنتاج (Production stage). ملف package.json تم تحديثه تلقائيا بعد تحميل Webpack وأصبح على هذا النحو :
{ "name": "modern-javascript-example", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1" }, "author": "", "license": "ISC", "dependencies": { "moment": "^2.19.1" }, "devDependencies": { "webpack": "^3.7.1" } }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.19.1"
},
"devDependencies": {
"webpack": "^3.7.1"
}
}
الآن وقد قمنا بتثبيت Webpack بنجاح وتحميله على مجلد الحزم node_modules الخاص بمشروعنا، سيمكننا استخدامه من نافذة الأوامر السطرية بهذه الكيفية :
./node_modules/.bin/webpack script.js bundle.js
1
./node_modules/.bin/webpack script.js bundle.js
هذا الأمر يقول لأداة Webpack (الموجودة داخل المجلد node_modules ) : خذ الملف script.js وابحث فيه عن جميع التعابير التي تحتوي على دالة الإستيراد require وقم باستبدالها بالشفرة البرمجية للوحدة التي تم استيرادها، ثم احفظ المخرج الجديد(Output) في ملف جديد اسمه bundle.js. يعني هذا أننا لن نستخدم من جديد الملف script.js في متصفحنا، بل نقوم عوضا عن ذلك باستدعاء الملف bundle.js المتوافق مع إمكانيات المتصفحات، أما script.js فنقوم باستخدامه فقط في مرحلة البرمجة والتطوير ولا يظهر في بيئة الإنتاج النهائية.
<!-- index.html --> <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>مثال لجافاسكربت</title> <script src="bundle.js"></script> </head> <body> <h1>مرحبا بكم!</h1> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>مثال لجافاسكربت</title>
<script src="bundle.js"></script>
</head>
<body>
<h1>مرحبا بكم!</h1>
</body>
</html>
بعد إعادة تحميل الصفحة على المتصفح، سنرى بأن كل شيء يعمل جيدا كما في السابق.
يجب علينا إعادة تنفيذ الأمر السابق في كل مرة نقوم فيها بالتعديل على ملف script.js. هذا الأمر قد يكون مملا وغير عملي، خاصة عندما تزداد درجة تعقيد مهام Webpack في مشروعنا، فما رأيناه أعلاه يمكن اعتباره أبسط ما يمكن ل Webpack القيام به.
ربما يجدر بنا الآن اكتشاف الملف webpack.config.js الذي يعتبر المكان الذي نقوم فيه بإضافة إعدادات Webpack الخاصة بالمشروع. لنقم بإنشاء هذا الملف ونضيف إليه الإعدادات التالية :
// webpack.config.js module.exports = { entry: './script.js', output: { filename: 'bundle.js' } };
1
2
3
4
5
6
7
// webpack.config.js
module.exports = {
entry: './script.js',
output: {
filename: 'bundle.js'
}
};
الآن يمكننا تنفيذ نفس المهمة السابقة دون تحديد أسماء الملفين (script.js و bundle.js ) من الأمر السطري. يكفي تنفيذ ما يلي وسيقوم Webpack بمعرفة ما يجب عليه فعله بالإعتماد على التوجيهات في ملف webpack.config.js :
./node_modules/.bin/webpack
1
./node_modules/.bin/webpack
تقدمنا خطوة إلى الأمام، ولكن لا يزال هذا مضجرا لأننا مازلنا مطالبين بتنفيذ الأمر في كل مرة نعدل فيها ملف script.js.
سنجعل هذه المهمة أكثر سلاسة وتلقائية فيما بعد، قليلا من الصبر
لا بأس بما قمنا به لحد الساعدة، لم نعد نقوم باستدعاء المكتبات والسكريبتات الخارجية على شكل متغيرات ودوال عامة من خلال الوسم <script>. نحن الآن نعتمد على فلسفة الوحدات (Modules)، دون أن ننسى كذلك أننا في النهاية نحصل على ملف جافاسكريبت واحد وهذه نقطة إيجابية أخرى لتحسين أداء التطبيق.سنواصل التقدم وإضافة مميزات أخرى قوية لبيئتنا التطويرية.
استخدام الميزات الجديدة في جافاسكريبت (Babel)
مع التطور الكبير في لغة البرمجة جافاسكريبت في الأعوام الأخيرة، أصبح من الصعب على المتصفحات مواكبة هذه التطورات ودمج كل هذه الميزات الجديدة في بيئاتها التشغيلية (Runtime). فمثلا هناك عدد كبير جدا من ميزات إصدار الجافاسكريبت ES6 ليست مدعومة بعد من جميع المتصفحات الكبيرة، هذا يمنع المطورين من استخدامها والإستفادة من الإمكانية العظيمة التي تتيحها.
لحسن الحظ، ظهرت في الفترة الماضية أدوات تعرف باسم Javascript Transpilers هدفها كتابة الجافاسكريبت بطريقة أكثر حداثة وانتاجية، وعند الإنتهاء من التكويد تتم عملية Transpiling التي تحول الشفرة البرمجية التي كتبها المطور إلى كود جافاسكريبت اعتيادي تفهمه وتدعمه جميع المتصفحات.
CoffeeScript كان أول هذه الأدوات التي لقيت نجاحا كبيرا حيث يمكن من كتابة أكواد جافاسكريبت بطريقة حديثة وأكثر انتاجية. ولكن في الفترة الأخيرة خفت نجم CoffeeScript وبرز في الساحة لاعبين جديدين استحوذا على أكبر نصيب من الكعكعة، نتحدث عن Babel و Typescript.
Typescript طورته شركة مايكروسوفت ويعتبر بمثابة لغة جديدة استلهمت العديد من الأشياء المميزة في لغة البرمجة سي شارب #C. يعني أنه يمكن للمطورين كتابة لغة جديدة والإستفادة من بعض نقاط قوة #C وبعد الإنتهاء من كتابة الكود يتم تحويله لجافاسكريبت اعتيادي يفهمه المتصفح.
يعتبر إطار العمل Angular أشهر التقنيات التي تعتمد على Typescript بشكل افتراضي، وكان له الفضل الكبير في الشهرة التي حققها الأخير في الآونة الأخيرة.
أما Babel فلايمكن اعتباره لغة برمجة، ولكن فقط أداة تمكننا من كتابة أكواد الجافاسكريبت على الطريقة الحديثة (ES6) وبعد ذلك تحويلها لأكواد جافاسكريبت مدعومة من جميع المتصفحات الكبيرة.
معظم المطورين يفضلون Babel لأنه كما قلنا ليس لغة برمجة جديدة يجب تعلمها بل فقط أداة تجعل الجافاسكريبت الحديث متوافق مع قدرات المتصفحات الحالية.
لنكتشف معا كيف يمكننا استخدام Babel في مشروعنا مع مجمع الوحدات Webpack. أولا علينا تحميل Babel من مدير الحزم npm بالطريقة التالية المعتادة (أصبحت معتادة الآن ) :
npm install babel-core babel-preset-env babel-loader --save-dev
1
npm install babel-core babel-preset-env babel-loader --save-dev
لاحظ أننا قمنا بتحميل 3 حزم دفعة واحدة كتبعيات خاصة بمرحلة التطوير (save-dev) :
- babel-core : هذه الحزمة هي النواة الأساسية لتقنية Babel.
- babel-preset-env : الحزمة التي نحدد بها ما هي المميزات التي نريد الوصول إليها. babel-preset-env يعني أننا نريد استخدام جميع مميزات جافاسكريبت الحديثة التي يدعمها Babel. عوضا عنها يمكننا استخدام مثلا babel-preset-es2015 حتى نكون دقيقين أكثر، ولكن هذا لن يشمل مميزات es2016 و es2017. بالتالي يفضل دائما استخدام babel-preset-env لأنها تشمل الكل.
- babel-loader : أما هذه الحزمة فهي التي تجعل Babel متوافقا مع Webpack.
// webpack.config.js module.exports = { entry: './script.js', output: { filename: 'bundle.js' }, module: { rules: [ { test: /\.js$/, exclude: /node_modules/, use: { loader: 'babel-loader', options: { presets: ['env'] } } } ] } };
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
// webpack.config.js
module.exports = {
entry: './script.js',
output: {
filename: 'bundle.js'
},
module: {
rules: [
{
test: /\.js$/,
exclude: /node_modules/,
use: {
loader: 'babel-loader',
options: {
presets: ['env']
}
}
}
]
}
};
الشكل الجديد لهذا الملف قد يكون مربكا أو غير مفهوم بالنسبة لك، هذا طبيعي كونها المرة الأولى التي تتعامل فيها مع Webpack.
ببساطة قلنا لصديقنا Webpack : أنظر لجميع الملفات التي امتدادها js. باستثناء الملفات الموجودة داخل المجلد node_modules، واستعن بالحزمتين babel-loader و babel-preset-env للتمكن من عملية تحويل (Transpiling) أكواد الجافاسكريبت.
الآن بتنا نستطيع البدء في كتابة أكواد الجافاسكريبت مع أحدث الميزات الجديدة في هذه اللغة. هذا مثال لكتابة كلاس جافاسكريبت على طريقة ES2015 في ملف script.js :
// script.js var moment = require('moment'); console.log("Hello from JavaScript!"); console.log(moment().startOf('day').fromNow()); class User { constructor(name) { this.name = name; } sayHi() { alert(this.name); } } let user = new User("Aissa"); user.sayHi();
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// script.js
var moment = require('moment');
console.log("Hello from JavaScript!");
console.log(moment().startOf('day').fromNow());
class User {
constructor(name) {
this.name = name;
}
sayHi() {
alert(this.name);
}
}
let user = new User("Aissa");
user.sayHi();
نستطيع كذلك استخدام طريقة ES2015 في استيراد الوحدات عوض معيارية CommonJs التي رأيناها في السابق. هذه الطريقة هي التي تستعمل في جل المشاريع في الوقت الحالي :
// script.js import moment from 'moment'; console.log("Hello from JavaScript!"); console.log(moment().startOf('day').fromNow()); class User { constructor(name) { this.name = name; } sayHi() { alert(this.name); } } let user = new User("Aissa"); user.sayHi();
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// script.js
import moment from 'moment';
console.log("Hello from JavaScript!");
console.log(moment().startOf('day').fromNow());
class User {
constructor(name) {
this.name = name;
}
sayHi() {
alert(this.name);
}
}
let user = new User("Aissa");
user.sayHi();
طريقة import الخاصة بإصدارات جافاسكريبت الحديثة لا تختلف كثيرا عن require التي استنبطها Node.js من مشروع CommonJs، غير أن الأولى — أي import — تعتبر أكثر مرونة وسلاسة خاصة في الحالات المتقدمة.
بما أننا قمنا بالتعديل على الملف script.js فعلينا من جديد أن نطلب من Webpack أن يقوم بعملية التجميع (Bundling) :
./node_modules/.bin/webpack
1
./node_modules/.bin/webpack
الآن يمكنك إعادة تحميل صفحة index.html، وسترى بأنها تعمل وفق المتوقع من دون مشاكل حتى في المتصفحات القديمة التي لا تدعم نهائيا ميزة الكلاسات في جافاسكريبت مثل انترنت إكسبلورر IE9.
وإذا كنت فضوليا أكثر (مبروك إذا كان الحال كذلك) فستذهب للملف bundle.js وتبحث عن الشكل الجديد للكلاس الذي قمت بكتابته سابقا في ملف script.js، هكذا أصبح حاله :
//bundle.js //... var _createClass = function() { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function(Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var User = function() { function User(name) { _classCallCheck(this, User); this.name = name; } _createClass(User, [{ key: "sayHi", value: function sayHi() { alert(this.name); } }]); return User; }(); var user = new User("Aissa"); user.sayHi(); //...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
//bundle.js
//...
var _createClass = function() {
function defineProperties(target, props) {
for (var i = 0; i < props.length; i++) {
var descriptor = props;
descriptor.enumerable = descriptor.enumerable || false;
descriptor.configurable = true;
if ("value" in descriptor) descriptor.writable = true;
Object.defineProperty(target, descriptor.key, descriptor);
}
}
return function(Constructor, protoProps, staticProps) {
if (protoProps) defineProperties(Constructor.prototype, protoProps);
if (staticProps) defineProperties(Constructor, staticProps);
return Constructor;
};
}();
function _classCallCheck(instance, Constructor) {
if (!(instance instanceof Constructor)) {
throw new TypeError("Cannot call a class as a function");
}
}
var User = function() {
function User(name) {
_classCallCheck(this, User);
this.name = name;
}
_createClass(User, [{
key: "sayHi",
value: function sayHi() {
alert(this.name);
}
}]);
return User;
}();
var user = new User("Aissa");
user.sayHi();
//...
نعم أعلم أن هذا الكود معقد أكثر وأقل جمالية، Babel قام باستخدام الجافاسكريبت القديم (استخدام الدوال، تحويل let إلى var) حتى نحصل في الأخير على الشفرة البرمجية التي تفهمها جميع المتصفحات المعروفة.
مثال الكلاس بسيط بالمقارنة مع كل مزايا جافاسكريبت الحديثة والتي يدعمها Babel. الأخير تستعين به React.js لتحويل أكواد JSX خاصتها إلى أكواد جافاسكريبت اعتيادية، لذلك صدقني أخي العزيز إن قلت لك أنه بعد تقدمك في الجافاسكريبت ستجد بابل في كل مكان.
الآن تتضح الأمور أكثر ولم يتبق لنا إلا بعض اللمسات الأخيرة. أول هذه اللمسات هي ضغط (Minify) الملفات المجمعة (في حالتنا bundle.js) حتى نقلل من حجمه قدر المستطاع. ثم بعد ذلك سنضيف إمكانية التشغيل التلقائي ل Webpack كلما قمنا بالتعديل على ملف script.js حتى لا نطلب منه ذلك في كل مرة من واجهة الأوامر السطرية.
الإستعانة بمشغلات المهام (Task Runner)
الآن وقد تمكننا بنجاح من تجميع وحداتنا البرمجية بعد كتابتها بأحدث المعايير والطرق المتبعة، سيكون من المنطقي أتمتة ما نستطيع من مختلف المهام التي تشكل أجزاء مهمة من عملية التطوير، نتحدث عن مهام من قبيل ضغط ملفات الأكواد (جافاسكريبت و css)، تحسين الصور، تنفيذ إختبارات الشفرة البرمجية Unit Tests، إلخ…
في فترة من الفترات (حوالي 2013)، كان GruntJs هو مشغل المهام الأشهر، وبعده جاء Gulp بقوة وتقاسما معا الكعكة. كلاهما يعتمدان على الإضافات حيث كل إضافة تقوم بمهمة محددة.
ولكن في الآونة الأخيرة، وبالخصوص منذ انفجار شعبية React.js، بات المطورون يستخدمون Webpack (الذي هو في الأصل مجمع وحدات Module Bundler وليس مشغل مهام Task Runner) في تنفيذ المهام بالإعتماد على أوامر NPM نفسه. هذه الأوامر تعرف باسم NPM Scripts ونصرح بها في ملف package.json :
{ "name": "modern-javascript-example", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1", "build": "webpack -p", "watch": "webpack --watch" }, "author": "", "license": "ISC", "dependencies": { "moment": "^2.19.1" }, "devDependencies": { "babel-core": "^6.26.0", "babel-loader": "^7.1.2", "babel-preset-env": "^1.6.1", "webpack": "^3.7.1" } }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"build": "webpack -p",
"watch": "webpack --watch"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.19.1"
},
"devDependencies": {
"babel-core": "^6.26.0",
"babel-loader": "^7.1.2",
"babel-preset-env": "^1.6.1",
"webpack": "^3.7.1"
}
}
في هذا المثال، حددنا سكريبتين : build و watch. لتنفيذ السكريبت نقوم بإدخال الأمر npm run متبوع باسم السكريبت.
على سبيل المثال، لتنفيذ السكريبت build نقوم بإدخال الأمر التالي في Command Line :
npm run build
1
npm run build
هذا السكريبت يقوم بتشغيل Webpack بالإعتماد على إعدادات ملف webpack.config.js، وأضفنا البارامتر p- الذي يقوم ل Webpack بأننا نريد تجميع الكود لمرحلة الإنتاج (Production)، أي ضغطه (Minify code).
npm run watch
1
npm run watch
أما السكريبت watch فيشغل Webpack في كل مرة نقوم فيها بعملية بالتعديل. يعني لن نظطر من الآن لكتابة الأمر في واجهة الأوامر السطرية بعد كل تعديل. خطوة عملية للغاية في مرحلة التطوير
لاحظ بأنه في سكريبتات npm لا نقوم بتعيين المسار الكامل ./node_modules/.bin/webpack ، لأن Node.js يعلم جيدا أين يجد كل حزمة بالإعتماد على اسمها فقط.
هذا رائع! يمكننا أن جعله أكثر روعة بتثبيت أداة أخرى رائعة : Webpack-dev-server. هذه الحزمة تعتمد كما نرى من اسمها على Webpack وتضع بين يدي المطور خادم ويب مع خاصية Live Reloading التي تعطينا إمكانية رؤية التغييرات في المتصفح دون القيام بإعادة تحميل الصفحة كل مرة، يعني بمجرد تعديل وحفظ ملف script.js سنرى النتيجة في المتصفح في ظرف أجزاء بالمئة من الثانية
الآن تعلمون كيف تقومون بتحميل هذه الأداة، أحسنتم، الطريقة هي :
npm install webpack-dev-server --save-dev
1
npm install webpack-dev-server --save-dev
بعد ذلك قم بإضافة سكريبت جديد لملف package.json بهذه الكيفية :
{ "name": "modern-javascript-example", "version": "1.0.0", "description": "", "main": "index.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1", "build": "webpack -p", "watch": "webpack --watch", "server": "webpack-dev-server --open" }, "author": "", "license": "ISC", "dependencies": { "moment": "^2.19.1" }, "devDependencies": { "babel-core": "^6.26.0", "babel-loader": "^7.1.2", "babel-preset-env": "^1.6.1", "webpack": "^3.7.1", "webpack-dev-server": "^2.7.1" } }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"build": "webpack -p",
"watch": "webpack --watch",
"server": "webpack-dev-server --open"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.19.1"
},
"devDependencies": {
"babel-core": "^6.26.0",
"babel-loader": "^7.1.2",
"babel-preset-env": "^1.6.1",
"webpack": "^3.7.1",
"webpack-dev-server": "^2.7.1"
}
}
الآن يمكنك بدء تشغيل خادم التطوير الخاص بمشروعك بهذه الطريقة :
npm run server
1
npm run server
هذه الأمر يقوم بفتح الصفحة index.html في متصفحك مع عنوان محلي افتراضي localhost:8080، وكلما عدلت على الملف script.js يقوم webpack-dev-server بإعادة جمع ال Bundle وتحديث حالة الصفحة في المتصفح بشكل تلقائي. هذا يوفر وقتا ثمينا على المطورين ما يجعلهم يركزون أكثر على عملهم في كتابة كود جيد عوض الإنتقال بين المتصفح والمحرر بعد كل تعديل.
هذه أمثلة بسيطة فقط مما يمكن القيام به مع Webpack و Webpack-dev-server. تستطيعون استغلال سكريبتات npm لتشغيل مهام أخرى مثل تحويل أكواد Sass إلى css، ضغط الصور وغيرها …
أدعوكم لزيارة الموقع الرسمي لكل منهما لاكتشاف المزيد من مميزاتهما.