journey
title आपकी गेम एनीमेशन यात्रा
section मूवमेंट के मूल सिद्धांत
गति के सिद्धांत समझें: 3: Student
निर्देशांक अपडेट सीखें: 4: Student
बुनियादी मूवमेंट लागू करें: 4: Student
section खिलाड़ी नियंत्रण
कीबोर्ड इवेंट्स हैंडल करें: 4: Student
डिफ़ॉल्ट व्यवहार रोकें: 5: Student
उत्तरदायी नियंत्रण बनाएं: 5: Student
section गेम सिस्टम
गेम लूप बनाएं: 5: Student
ऑब्जेक्ट जीवनचक्र प्रबंधित करें: 5: Student
पब/सब पैटर्न लागू करें: 5: Student
अपने पसंदीदा खेलों के बारे में सोचें – जो चीज़ उन्हें आकर्षक बनाती है वह सिर्फ सुंदर ग्राफिक्स नहीं है, बल्कि सब कुछ कैसे चलता है और आपके क्रियाओं पर प्रतिक्रिया करता है। इस समय, आपका अंतरिक्ष खेल एक सुंदर चित्र की तरह है, लेकिन हम गति जोड़ने वाले हैं जो इसे जीवन्त बना देगा।
जब नासा के इंजीनियरों ने अपोलो मिशनों के लिए गाइडेंस कंप्यूटर प्रोग्राम किया, तो उन्हें एक समान चुनौती का सामना करना पड़ा: आप अंतरिक्ष यान को कैसे पायलट के इनपुट का जवाब देते हुए स्वचालित रूप से पाठ्यक्रम सुधार बनाए रख सकते हैं? आज हम जो सिद्धांत सीखेंगे वे उन्हीं अवधारणाओं को दोहराते हैं – खिलाड़ियों द्वारा नियंत्रित गति को स्वचालित सिस्टम व्यवहार के साथ प्रबंधित करना।
इस पाठ में, आप सीखेंगे कि कैसे अंतरिक्ष यान स्क्रीन पर ग्लाइड करते हैं, खिलाड़ी के आदेशों पर प्रतिक्रिया करते हैं, और चिकनी गति पैटर्न बनाते हैं। हम सब कुछ सरल अवधारणाओं में तोड़ेंगे जो स्वाभाविक रूप से एक-दूसरे पर आधारित हैं।
अंत तक, आपके खिलाड़ी अपने हीरो जहाज को स्क्रीन पर उड़ाते हुए देखेंगे जबकि दुश्मन जहाज ऊपर से गश्त करते रहेंगे। इससे भी महत्वपूर्ण, आप उन मूल सिद्धांतों को समझेंगे जो खेल गति प्रणालियों को शक्ति देते हैं।
mindmap
root((गेम एनिमेशन))
Movement Types
Player Controlled [खिलाड़ी नियंत्रित]
Automatic Motion [स्वचालित गति]
Physics Based [भौतिकी आधारित]
Scripted Paths [स्क्रिप्टेड पथ]
Event Handling
Keyboard Input [कीबोर्ड इनपुट]
Mouse Events [माउस ईवेंट]
Touch Controls [टच नियंत्रण]
Default Prevention [डिफ़ॉल्ट रोकथाम]
Game Loop
Update Logic [लॉजिक अपडेट करें]
Render Frame [फ्रेम रेंडर करें]
Clear Canvas [कैनवास साफ़ करें]
Frame Rate Control [फ्रेम दर नियंत्रण]
Object Management
Position Updates [स्थिति अपडेट]
Collision Detection [मुठभेड़ का पता लगाना]
Lifecycle Management [जीवनचक्र प्रबंधन]
State Tracking [स्थिति ट्रैकिंग]
Communication
Pub/Sub Pattern [पब/सब पैटर्न]
Event Emitters [ईवेंट एमिटर]
Message Passing [संदेश पासिंग]
Loose Coupling [ढीला कनेक्शन]
जब चीज़ें चलने लगती हैं तो खेल जीवित हो उठते हैं, और यह मूल रूप से दो तरीकों से होता है:
- खिलाड़ी नियंत्रित गति: जब आप कोई कुंजी दबाते हैं या माउस क्लिक करते हैं, तो कुछ चलता है। यह आप और आपके खेल की दुनिया के बीच सीधा संपर्क है।
- स्वचालित गति: जब खेल खुद चीज़ें चलाने का निर्णय लेता है – जैसे वे दुश्मन जहाज जो स्क्रीन पर गश्त करते हैं चाहे आप कुछ भी कर रहे हों या नहीं।
कंप्यूटर स्क्रीन पर वस्तुओं को चलाना जितना आप सोचते हैं उससे कहीं आसान है। क्या आपको गणित की क्लास में x और y निर्देशांक याद हैं? वही हम यहाँ उपयोग कर रहे हैं। जब गेलिलियो ने 1610 में बृहस्पति के चंद्रमाओं को ट्रैक किया था, वे मूल रूप से वही कर रहे थे – गति पैटर्न को समझने के लिए समय के साथ स्थिति का प्लॉटिंग।
स्क्रीन पर वस्तुओं को चलाना फ्लिपबुक एनीमेशन बनाने जैसा है – आपको इन तीन सरल चरणों का पालन करना होता है:
flowchart LR
A["फ़्रेम N"] --> B["स्थिति अपडेट करें"]
B --> C["कैनवस साफ़ करें"]
C --> D["आब्जेक्ट ड्रॉ करें"]
D --> E["फ़्रेम N+1"]
E --> F{जारी रखें?}
F -->|हाँ| B
F -->|नहीं| G["खेल खत्म"]
subgraph "एनिमेशन चक्र"
H["1. नई स्थिति की गणना करें"]
I["2. पिछला फ़्रेम मिटाएं"]
J["3. नया फ़्रेम रेंडर करें"]
end
style B fill:#e1f5fe
style C fill:#ffebee
style D fill:#e8f5e8
- स्थिति अपडेट करें – अपने ऑब्जेक्ट को जिस जगह होना चाहिए उसे बदलें (शायद इसे दाईं ओर 5 पिक्सेल स्थानांतरित करें)
- पुराना फ्रेम मिटाएं – स्क्रीन को साफ़ करें ताकि कहीं भी भूतिया ट्रेल न दिखे
- नया फ्रेम बनाएं – अपने ऑब्जेक्ट को नई जगह पर बनाएं
यह सब तेजी से करें, और धमाकेदार! आपके पास चिकनी गति होगी जो खिलाड़ियों को प्राकृतिक लगेगी।
यहाँ कोड में इसका क्या रूप हो सकता है:
// नायक के स्थान को सेट करें
hero.x += 5;
// आयत को साफ करें जो नायक को होस्ट करता है
ctx.clearRect(0, 0, canvas.width, canvas.height);
// खेल की पृष्ठभूमि और नायक को पुनः चित्रित करें
ctx.fillRect(0, 0, canvas.width, canvas.height);
ctx.fillStyle = "black";
ctx.drawImage(heroImg, hero.x, hero.y);यह कोड क्या करता है:
- नायक के x-निर्देशांक को क्षैतिज रूप से 5 पिक्सेल आगे बढ़ाता है
- पिछले फ्रेम को हटाने के लिए पूरे कैनवास क्षेत्र को साफ़ करता है
- कैनवास को काले पृष्ठभूमि रंग से भरता है
- नायक की छवि को उसकी नई स्थिति पर पुनः बनाता है
✅ क्या आप सोच सकते हैं कि अपने नायक को प्रति सेकंड कई फ्रेम्स में पुनः बनाने से प्रदर्शन पर क्या प्रभाव पड़ सकता है? इस पैटर्न के वैकल्पिक तरीकों के बारे में पढ़ें।
यहाँ हम खिलाड़ी के इनपुट को खेल की क्रिया से जोड़ते हैं। जब कोई स्पेसबार दबाकर लेजर फायर करता है या ऐरो की दबाकर ऐस्टिरॉयड से बचता है, तो आपका खेल उस इनपुट का पता लगाना और प्रतिक्रिया देना चाहिए।
कीबोर्ड इवेंट्स विंडो स्तर पर होते हैं, जिसका मतलब है कि आपका पूरा ब्राउज़र विंडो उन की प्रेस पर सुनता है। वहीं, माउस क्लिक विशेष तत्वों से जुड़े हो सकते हैं (जैसे बटन पर क्लिक करना)। हमारे अंतरिक्ष खेल के लिए, हम कीबोर्ड नियंत्रणों पर ध्यान देंगे क्योंकि यही खिलाड़ियों को क्लासिक आर्केड अनुभव देता है।
यह मुझे 1800 के दशक के टेलीग्राफ ऑपरेटरों की याद दिलाता है, जिन्हें मोर्स कोड इनपुट को सार्थक संदेशों में अनुवाद करना होता था – हम भी कुछ इसी तरह कर रहे हैं, की प्रेस को खेल कमांड में बदलना।
इवेंट को संभालने के लिए आपको विंडो की addEventListener() विधि का उपयोग करना होगा और इसे दो इनपुट पैरामीटर देना होगा। पहला पैरामीटर इवेंट का नाम होता है, जैसे keyup। दूसरा पैरामीटर वह फ़ंक्शन होता है जो इवेंट होने पर कॉल किया जाएगा।
यहाँ एक उदाहरण है:
window.addEventListener('keyup', (evt) => {
// evt.key = कुंजी का स्ट्रिंग प्रतिनिधित्व
if (evt.key === 'ArrowUp') {
// कुछ करें
}
});यहाँ क्या होता है:
- पूरे विंडो पर कीबोर्ड इवेंट सुनता है
- इवेंट ऑब्जेक्ट को पकड़ता है जिसमें यह जानकारी होती है कि कौन सी कुंजी दबाई गई थी
- जांचता है कि दबाई गई कुंजी किसी विशिष्ट कुंजी से मेल खाती है (इस मामले में, ऊपर की एरो)
- जब शर्त पूरी होती है तो कोड चलाता है
कीबोर्ड इवेंट्स के लिए इवेंट में दो प्रॉपर्टीज़ होती हैं जिन्हें आप देख सकते हैं कि कौन सी कुंजी दबाई गई:
key- यह दबाई गई कुंजी का स्ट्रिंग प्रतिनिधित्व होता है, जैसे'ArrowUp'keyCode- यह एक संख्या होती है, जैसे37, जोArrowLeftको दर्शाती है
✅ कीबोर्ड इवेंट का उपयोग खेल विकास के बाहर भी उपयोगी है। आप इस तकनीक के और किन उपयोगों के बारे में सोच सकते हैं?
sequenceDiagram
participant User
participant Browser
participant EventSystem
participant GameLogic
participant Hero
User->>Browser: एरोअप कुंजी दबाएं
Browser->>EventSystem: keydown इवेंट
EventSystem->>EventSystem: preventDefault()
EventSystem->>GameLogic: emit('KEY_EVENT_UP')
GameLogic->>Hero: hero.y -= 5
Hero->>Hero: स्थिति अपडेट करें
Note over Browser,GameLogic: इवेंट फ्लो ब्राउज़र डिफ़ॉल्ट्स को रोکتا है
Note over GameLogic,Hero: पब/सब पैटर्न साफ संचार सक्षम करता है
कुछ कुंजियाँ ब्राउज़र की अंतर्निहित क्रियाएँ करती हैं जो आपके खेल में बाधा डाल सकती हैं। ऐरो की पेज को स्क्रॉल करते हैं और स्पेसबार नीचे कूदता है – यह व्यवहार आप तब नहीं चाहते जब कोई अपने अंतरिक्ष यान को नियंत्रित कर रहा हो।
हम इन डिफ़ॉल्ट व्यवहारों को रोक सकते हैं और हमारे खेल को इनपुट को नियंत्रित करने दे सकते हैं। यह उसी तरह है जैसे प्रारंभिक कंप्यूटर प्रोग्रामरों ने सिस्टम इंटरप्ट्स को ओवरराइड करके कस्टम व्यवहार बनाए थे – हम बस इसे ब्राउज़र स्तर पर कर रहे हैं। यह तरीका है:
const onKeyDown = function (e) {
console.log(e.keyCode);
switch (e.keyCode) {
case 37:
case 39:
case 38:
case 40: // तीर कुंजी
case 32:
e.preventDefault();
break; // स्पेस
default:
break; // अन्य कुंजीओं को अवरुद्ध न करें
}
};
window.addEventListener('keydown', onKeyDown);इस रोकथाम कोड को समझना:
- ऐसे विशिष्ट की कोड की जांच करता है जो अवांछित ब्राउज़र व्यवहार उत्पन्न कर सकते हैं
- ऐरो की और स्पेसबार के लिए डिफ़ॉल्ट ब्राउज़र क्रिया को रोकता है
- दूसरी कुंजियों को सामान्य रूप से काम करने देता है
- ब्राउज़र के अंतर्निहित व्यवहार को रोकने के लिए
e.preventDefault()का उपयोग करता है
इवेंट हैंडलिंग समझ: स्वचालित गति पर जाने से पहले, सुनिश्चित करें कि आप:
- ✅
keydownऔरkeyupइवेंट्स में अंतर समझा सकते हैं - ✅ समझते हैं कि क्यों हम ब्राउज़र के डिफ़ॉल्ट व्यवहार को रोकते हैं
- ✅ वर्णन कर सकते हैं कि इवेंट लिस्नर्स कैसे उपयोगकर्ता इनपुट को खेल लॉजिक से जोड़ते हैं
- ✅ पहचान सकते हैं कि कौन सी कुंजियाँ खेल नियंत्रणों में बाधा डाल सकती हैं
त्वरित स्व-परीक्षा: अगर आपने ऐरो की के लिए डिफ़ॉल्ट व्यवहार न रोका तो क्या होगा? उत्तर: ब्राउज़र पेज को स्क्रॉल करेगा, जो खेल की गति में बाधा पैदा करेगा
इवेंट सिस्टम आर्किटेक्चर: अब आप समझते हैं:
- विंडो-स्तरीय सुनना: ब्राउज़र स्तर पर इवेंट को पकड़ना
- इवेंट ऑब्जेक्ट प्रॉपर्टीज़:
keyस्ट्रिंग बनामkeyCodeसंख्या - डिफ़ॉल्ट रोकथाम: अवांछित ब्राउज़र व्यवहार रोकना
- शर्तीय लॉजिक: विशिष्ट कुंजी संयोजनों पर प्रतिक्रिया देना
अब आइए उन वस्तुओं के बारे में बात करें जो बिना खिलाड़ी के इनपुट के चलती हैं। जैसे स्क्रीन पर गश्त करते हुए दुश्मन जहाज, सीधे रेखाओं में उड़ती हुई गोलियाँ, या पीछे बादल बहते हुए। यह स्वायत्त गति आपके खेल की दुनिया को जीवंत महसूस कराती है, भले ही कोई कंट्रोल छू भी न रहा हो।
हम जावास्क्रिप्ट के अंतर्निहित टाइमर का उपयोग करते हैं जो नियमित अंतराल पर स्थिति अपडेट करते हैं। यह अवधारणा उस तरह है जैसे पेंडुलम घड़ी काम करती है – एक नियमित यंत्र जो निरंतर, समयबद्ध क्रियाएँ चलाता है। यह बहुत सरल हो सकता है:
const id = setInterval(() => {
// दुश्मन को y अक्ष पर स्थानांतरित करें
enemy.y += 10;
}, 100);यह गति कोड क्या करता है:
- हर 100 मिलीसेकंड में एक टाइमर बनाता है
- हर बार दुश्मन के y-निर्देशांक को 10 पिक्सेल बढ़ाता है
- अंतराल आईडी स्टोर करता है जिससे हमें जरूरत पड़ने पर इसे रोकना होगा
- स्वचालित रूप से दुश्मन को स्क्रीन पर नीचे की ओर ले जाता है
यह वह अवधारणा है जो सब कुछ जोड़ती है – गेम लूप। यदि आपका खेल एक फिल्म होता, तो गेम लूप फिल्म प्रोजेक्टर होता जो तेज़ी से फ्रेम दर फ्रेम दिखाता है ताकि सब कुछ चिकनी गति से चले।
हर खेल का यह लूप पृष्ठभूमि में चलता है। यह एक फ़ंक्शन है जो सभी खेल वस्तुओं को अपडेट करता है, स्क्रीन को पुनः बनाता है, और इस प्रक्रिया को लगातार दोहराता है। यह आपके हीरो, सभी दुश्मनों, और उड़ती लेजर की पूरी स्थिति को ट्रैक करता है।
यह सिद्धांत मुझे याद दिलाता है कि कैसे शुरुआती फिल्म एनीमेटर वॉल्ट डिज़्नी को चरित्रों को फ्रेम दर फ्रेम पुनः बनाना पड़ता था ताकि गति का भ्रम उत्पन्न हो। हम भी ऐसा ही कर रहे हैं, बस कोड से पेंसिल के बजाय।
यहाँ गेम लूप का एक सामान्य रूप कोड में दिखाया गया है:
flowchart TD
A["खेल लूप शुरू करें"] --> B["कैनवास साफ़ करें"]
B --> C["पृष्ठभूमि भरें"]
C --> D["खेल वस्तुओं को अपडेट करें"]
D --> E["नायक को चित्रित करें"]
E --> F["दुश्मनों को चित्रित करें"]
F --> G["यूआई तत्वों को चित्रित करें"]
G --> H["अगले फ्रेम की प्रतीक्षा करें"]
H --> I{खेल चल रहा है?}
I -->|हाँ| B
I -->|नहीं| J["खेल समाप्त करें"]
subgraph "फ़्रेम रेट नियंत्रण"
K["60 FPS = 16.67ms"]
L["30 FPS = 33.33ms"]
M["10 FPS = 100ms"]
end
style B fill:#ffebee
style D fill:#e1f5fe
style E fill:#e8f5e8
style F fill:#e8f5e8
const gameLoopId = setInterval(() => {
function gameLoop() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
ctx.fillStyle = "black";
ctx.fillRect(0, 0, canvas.width, canvas.height);
drawHero();
drawEnemies();
drawStaticObjects();
}
gameLoop();
}, 200);गेम लूप संरचना को समझना:
- पिछले फ्रेम को हटाने के लिए पूरे कैनवास को साफ करता है
- पृष्ठभूमि को एक ठोस रंग से भरता है
- सभी गेम वस्तुओं को उनकी वर्तमान स्थिति में बनाता है
- इस प्रक्रिया को हर 200 मिलीसेकंड में दोहराता है ताकि चिकनी एनिमेशन बन सके
- अंतराल समय का नियंत्रण करके फ्रेम दर प्रबंधित करता है
अब हम गतिहीन दृश्य में गति जोड़ेंगे जिसे आपने पहले बनाया था। हम इसे एक स्क्रीनशॉट से एक इंटरैक्टिव अनुभव में बदलेंगे। हम इसे चरण दर चरण काम करेंगे ताकि हर हिस्सा पिछले वाले पर निर्भर हो।
पिछले पाठ से कोड लेकर शुरू करें (या अगर आपको ताज़ा शुरुआत चाहिए तो Part II- starter फ़ोल्डर में कोड लेकर शुरू करें)।
आज हम जो बना रहे हैं:
- हीरो नियंत्रण: एरो की आपके अंतरिक्ष जहाज को स्क्रीन पर चलाएंगी
- दुश्मन की गति: वे विदेशी जहाज अपने आक्रमण शुरू करेंगे
चलिए इन फीचर्स को लागू करना शुरू करते हैं।
your-work उपफ़ोल्डर में बनाए गए फाइलों को खोजें। इसमें निम्नलिखित होंगे:
-| assets
-| enemyShip.png
-| player.png
-| index.html
-| app.js
-| package.jsonआप अपने प्रोजेक्ट को your-work फ़ोल्डर में इस कमांड से शुरू करते हैं:
cd your-work
npm startयह कमांड क्या करता है:
- आपके प्रोजेक्ट डायरेक्टरी पर नेविगेट करता है
http://localhost:5000पते पर HTTP सर्वर शुरू करता है- आपके खेल की फाइलों को सेवा करता है ताकि आप उन्हें ब्राउज़र में टेस्ट कर सकें
ऊपर बताया गया HTTP सर्वर पता http://localhost:5000 पर शुरू हो जाएगा। एक ब्राउज़र खोलिए और यह पता दर्ज कीजिए, अब यह नायक और सभी दुश्मनों को रेंडर कर रहा होगा; अभी कुछ भी नहीं चल रहा है!
-
heroऔरenemyऔरgame objectके लिए समर्पित ऑब्जेक्ट जोड़ें, इनमेंxऔरyगुण होने चाहिए। (याद रखें Inheritance or composition भाग)।संकेत
game objectवह होना चाहिए जिसमेंxऔरyहों और जो खुद को कैनवास पर ड्रॉ कर सके।टिप: एक नया
GameObjectक्लास जोड़ना शुरू करें, जिसका कंस्ट्रक्टर नीचे दिखाए अनुसार हो, और फिर इसे कैनवास पर ड्रॉ करें:class GameObject { constructor(x, y) { this.x = x; this.y = y; this.dead = false; this.type = ""; this.width = 0; this.height = 0; this.img = undefined; } draw(ctx) { ctx.drawImage(this.img, this.x, this.y, this.width, this.height); } }
इस बेस क्लास को समझना:
- उन सामान्य गुणों को परिभाषित करता है जो सभी गेम ऑब्जेक्ट साझा करते हैं (स्थिति, आकार, इमेज)
- एक
deadफ्लैग शामिल करता है ताकि पता लगाया जा सके कि ऑब्जेक्ट को हटाया जाना चाहिए या नहीं - एक
draw()मेथड प्रदान करता है जो ऑब्जेक्ट को कैनवास पर रेंडर करता है - सभी गुणों के लिए डिफ़ॉल्ट मान सेट करता है जिन्हें चाइल्ड क्लास ओवरराइड कर सकते हैं
classDiagram
class GameObject {
+x: संख्या
+y: संख्या
+dead: बूलियन
+type: स्ट्रिंग
+width: संख्या
+height: संख्या
+img: इमेज
+draw(ctx)
}
class Hero {
+speed: संख्या
+type: "हीरो"
+width: 98
+height: 75
}
class Enemy {
+type: "दुश्मन"
+width: 98
+height: 50
+setInterval()
}
GameObject <|-- Hero
GameObject <|-- Enemy
class EventEmitter {
+listeners: ऑब्जेक्ट
+on(message, listener)
+emit(message, payload)
}
अब, इस `GameObject` को विस्तार देते हुए `Hero` और `Enemy` बनाएँ:
```javascript
class Hero extends GameObject {
constructor(x, y) {
super(x, y);
this.width = 98;
this.height = 75;
this.type = "Hero";
this.speed = 5;
}
}
```
```javascript
class Enemy extends GameObject {
constructor(x, y) {
super(x, y);
this.width = 98;
this.height = 50;
this.type = "Enemy";
const id = setInterval(() => {
if (this.y < canvas.height - this.height) {
this.y += 5;
} else {
console.log('Stopped at', this.y);
clearInterval(id);
}
}, 300);
}
}
```
**इन क्लासों की प्रमुख अवधारणाएँ:**
- `GameObject` से `extends` कीवर्ड का उपयोग कर वे उत्तराधिकार में हैं
- पैरेंट कंस्ट्रक्टर को `super(x, y)` के साथ कॉल करते हैं
- प्रत्येक ऑब्जेक्ट प्रकार के लिए विशिष्ट आयाम और गुण सेट करते हैं
- दुश्मनों के लिए `setInterval()` का उपयोग कर स्वचालित गति लागू करते हैं
-
की-बोर्ड नेविगेशन को संभालने के लिए की-इवेंट हैंडलर जोड़ें (हीरो को ऊपर/नीचे बाएं/दाएं ले जाने के लिए)
याद रखें यह कार्डिनल सिस्टम है, टॉप-लेफ्ट
0,0होता है। साथ ही डिफ़ॉल्ट व्यवहार को रोकने के लिए कोड जोड़ें।टिप: अपनी
onKeyDownफ़ंक्शन बनाएं और इसे विंडो पर संलग्न करें:const onKeyDown = function (e) { console.log(e.keyCode); // डिफ़ॉल्ट व्यवहार को रोकने के लिए ऊपर दिए गए पाठ से कोड जोड़ें switch (e.keyCode) { case 37: case 39: case 38: case 40: // तीर कुंजी case 32: e.preventDefault(); break; // स्पेस default: break; // अन्य कुंजीओं को ब्लॉक न करें } }; window.addEventListener("keydown", onKeyDown);
यह इवेंट हैंडलर क्या करता है:
- पूरे विंडो पर कीडाउन इवेंट सुनता है
- यह डिबगिंग के लिए कुंजी कोड को लॉग करता है कि कौन-कौन सी कुंजियाँ दबाई जा रही हैं
- ऐरो की और स्पेसबार के लिए डिफ़ॉल्ट ब्राउज़र व्यवहार को रोकता है
- अन्य कुंजियों को सामान्य रूप से कार्य करने देता है
इस समय अपने ब्राउज़र कंसोल को देखें और ध्वनियां दर्ज होते देखें।
-
बचे हुए भागों का पालन करते हुए अपना कोड साफ रखने के लिए Pub sub pattern लागू करें।
प्रकाशन-सदस्यता पैटर्न आपके कोड को व्यवस्थित करता है जिससे इवेंट का पता चलना और संभालना अलग हो जाता है। इससे आपका कोड मॉड्यूलर और रखरखाव में आसान बन जाता है।
अंतिम भाग करने के लिए, आप:
-
विंडो पर एक इवेंट लिस्नर जोड़ें:
window.addEventListener("keyup", (evt) => { if (evt.key === "ArrowUp") { eventEmitter.emit(Messages.KEY_EVENT_UP); } else if (evt.key === "ArrowDown") { eventEmitter.emit(Messages.KEY_EVENT_DOWN); } else if (evt.key === "ArrowLeft") { eventEmitter.emit(Messages.KEY_EVENT_LEFT); } else if (evt.key === "ArrowRight") { eventEmitter.emit(Messages.KEY_EVENT_RIGHT); } });
यह इवेंट सिस्टम क्या करता है:
- कीबोर्ड इनपुट पकड़ता है और उसे कस्टम गेम इवेंट्स में बदलता है
- इनपुट डिटेक्शन और गेम लॉजिक को अलग करता है
- नियंत्रणों को बाद में आसानी से बदलने देता है बिना गेम कोड प्रभावित किए
- एक ही इनपुट पर कई सिस्टम को प्रतिक्रिया देने देता है
-
flowchart TD
A["कीबोर्ड इनपुट"] --> B["विंडो इवेंट लिस्नर"]
B --> C["इवेंट इमिटर"]
C --> D["KEY_EVENT_UP"]
C --> E["KEY_EVENT_DOWN"]
C --> F["KEY_EVENT_LEFT"]
C --> G["KEY_EVENT_RIGHT"]
D --> H["हीरो मूवमेंट"]
D --> I["ध्वनि प्रणाली"]
D --> J["दृश्य प्रभाव"]
E --> H
F --> H
G --> H
style A fill:#e1f5fe
style C fill:#e8f5e8
style H fill:#fff3e0
-
मैसेज प्रकाशित और सदस्यता के लिए एक EventEmitter क्लास बनाएँ:
class EventEmitter { constructor() { this.listeners = {}; } on(message, listener) { if (!this.listeners[message]) { this.listeners[message] = []; } this.listeners[message].push(listener); }
-
कॉन्स्टेंट्स जोड़ें और EventEmitter सेटअप करें:
const Messages = { KEY_EVENT_UP: "KEY_EVENT_UP", KEY_EVENT_DOWN: "KEY_EVENT_DOWN", KEY_EVENT_LEFT: "KEY_EVENT_LEFT", KEY_EVENT_RIGHT: "KEY_EVENT_RIGHT", }; let heroImg, enemyImg, laserImg, canvas, ctx, gameObjects = [], hero, eventEmitter = new EventEmitter();
यह सेटअप क्या करता है:
- मैसेज कॉन्स्टेंट्स को परिभाषित करता है ताकि टाइपो से बचा जा सके और रीफैक्टरिंग आसान हो
- इमेज, कैनवास संदर्भ, और गेम स्टेट के लिए वेरिएबल्स घोषित करता है
- प्रकाशन-सदस्यता प्रणाली के लिए एक वैश्विक इवेंट एमिटर बनाता है
- सभी गेम ऑब्जेक्ट्स को रखने के लिए एक एरे इनिशियलाइज़ करता है
-
गेम को इनिशियलाइज़ करें
function initGame() { gameObjects = []; createEnemies(); createHero(); eventEmitter.on(Messages.KEY_EVENT_UP, () => { hero.y -= 5; }); eventEmitter.on(Messages.KEY_EVENT_DOWN, () => { hero.y += 5; }); eventEmitter.on(Messages.KEY_EVENT_LEFT, () => { hero.x -= 5; });
-
गेम लूप सेटअप करें
window.onloadफ़ंक्शन को रीफ़ैक्टर करें ताकि वह गेम को इनिशियलाइज़ करे और एक उचित अंतराल पर गेम लूप सेट करे। आप एक लेजर बीम भी जोड़ेंगे:window.onload = async () => { canvas = document.getElementById("canvas"); ctx = canvas.getContext("2d"); heroImg = await loadTexture("assets/player.png"); enemyImg = await loadTexture("assets/enemyShip.png"); laserImg = await loadTexture("assets/laserRed.png"); initGame(); const gameLoopId = setInterval(() => { ctx.clearRect(0, 0, canvas.width, canvas.height); ctx.fillStyle = "black"; ctx.fillRect(0, 0, canvas.width, canvas.height); drawGameObjects(ctx); }, 100); };
गेम सेटअप को समझना:
- पेज के पूरी तरह लोड होने का इंतजार करता है शुरू करने से पहले
- कैनवास एलिमेंट और उसका 2D रेंडरिंग कॉन्टेक्सट प्राप्त करता है
- सभी इमेज असिंक्रोनस रूप से
awaitका उपयोग करके लोड करता है - गेम लूप को 100ms के अंतराल पर (10 FPS) चलाता है
- हर फ़्रेम स्क्रीन को क्लियर और फिर से ड्रॉ करता है
-
कोड जोड़ें जिससे दुश्मन एक निश्चित अंतराल पर हिलें
createEnemies()फ़ंक्शन को रीफ़ैक्टर करें ताकि वह दुश्मनों को बनाए और उन्हें नए gameObjects क्लास में पुश करे:function createEnemies() { const MONSTER_TOTAL = 5; const MONSTER_WIDTH = MONSTER_TOTAL * 98; const START_X = (canvas.width - MONSTER_WIDTH) / 2; const STOP_X = START_X + MONSTER_WIDTH; for (let x = START_X; x < STOP_X; x += 98) { for (let y = 0; y < 50 * 5; y += 50) { const enemy = new Enemy(x, y); enemy.img = enemyImg; gameObjects.push(enemy); } } }
दुश्मन निर्माण जो करता है:
- स्थिति की गणना करता है ताकि दुश्मन स्क्रीन के केंद्र में हों
- नेस्टेड लूप्स का उपयोग करके दुश्मनों का ग्रिड बनाता है
- प्रत्येक दुश्मन ऑब्जेक्ट को दुश्मन की इमेज असाइन करता है
- प्रत्येक दुश्मन को ग्लोबल गेम ऑब्जेक्ट्स एरे में जोड़ता है
और एक
createHero()फ़ंक्शन जोड़ें जो हीरो के लिए समान प्रक्रिया करता है।function createHero() { hero = new Hero( canvas.width / 2 - 45, canvas.height - canvas.height / 4 ); hero.img = heroImg; gameObjects.push(hero); }
हीरो निर्माण जो करता है:
- हीरो को स्क्रीन के नीचे केंद्र में स्थित करता है
- हीरो ऑब्जेक्ट को हीरो इमेज असाइन करता है
- रेंडरिंग के लिए हीरो को गेम ऑब्जेक्ट्स एरे में जोड़ता है
और अंत में, ड्राइंग शुरू करने के लिए एक
drawGameObjects()फ़ंक्शन जोड़ें:function drawGameObjects(ctx) { gameObjects.forEach(go => go.draw(ctx)); }
ड्राइंग फ़ंक्शन को समझना:
- एरे में सभी गेम ऑब्जेक्ट्स पर इटरेट करता है
- प्रत्येक ऑब्जेक्ट पर
draw()मेथड कॉल करता है - ऐसे ऑब्जेक्ट्स को रेंडरिंग के लिए कैनवास कॉन्टेक्स्ट पास करता है
पूर्ण गेम सिस्टम समझ: अपनी सम्पूर्ण आर्किटेक्चर की महारथ सत्यापित करें:
- ✅ विरासत कैसे हीरो और दुश्मन को सामान्य GameObject गुण साझा करने देता है?
- ✅ प्रकाशक/सदस्य पैटर्न आपके कोड को अधिक रखरखाव योग्य क्यों बनाता है?
- ✅ गेम लूप स्मूथ एनिमेशन बनाने में क्या भूमिका निभाता है?
- ✅ इवेंट लिस्नर कैसे यूजर इनपुट को गेम ऑब्जेक्ट व्यवहार से जोड़ते हैं?
सिस्टम इंटीग्रेशन: आपका गेम अब दिखाता है:
- ऑब्जेक्ट-ओरिएंटेड डिज़ाइन: बेस क्लासेस के साथ विशेषज्ञ विरासत
- इवेंट-ड्रिवन आर्किटेक्चर: ढीली जोड़ी के लिए पब/सदस्य पैटर्न
- एनिमेशन फ्रेमवर्क: निरंतर फ्रेम अपडेट के साथ गेम लूप
- इनपुट हैंडलिंग: कीबोर्ड इवेंट्स विथ डिफ़ॉल्ट प्रिवेंशन
- एसेट प्रबंधन: इमेज लोडिंग और स्प्राइट रेंडरिंग
प्रोफेशनल पैटर्न्स: आपने कार्यान्वित किया है:
- जिम्मेदारियों का पृथक्करण: इनपुट, लॉजिक, और रेंडरिंग अलग-अलग रखना
- पॉलीमॉर्फिज़्म: सभी गेम ऑब्जेक्ट्स सामान्य ड्राइंग इंटरफ़ेस साझा करते हैं
- मैसेज पासिंग: घटकों के बीच साफ़ संवाद
- संसाधन प्रबंधन: कुशल स्प्राइट और एनिमेशन हैंडलिंग
आपके दुश्मन अब आपके हीरो स्पेसशिप की ओर बढ़ना शुरू कर देंगे! } }
and add a `createHero()` function to do a similar process for the hero. ```javascript function createHero() { hero = new Hero( canvas.width / 2 - 45, canvas.height - canvas.height / 4 ); hero.img = heroImg; gameObjects.push(hero); }और अंत में, ड्राइंग शुरू करने के लिए एक
drawGameObjects()फ़ंक्शन जोड़ें:function drawGameObjects(ctx) { gameObjects.forEach(go => go.draw(ctx)); }
आपके दुश्मन अब आपके हीरो स्पेसशिप की ओर बढ़ना शुरू कर देंगे!
यहाँ एक चुनौती है जो आपके गेम की परिष्करण को बढ़ाएगी: सीमाएं और स्मूथ कंट्रोल जोड़ना। वर्तमान में, आपका हीरो स्क्रीन से बाहर उड़ सकता है, और मूवमेंट में फ्रेम-कूदपन हो सकता है।
आपका मिशन: अपने स्पेसशिप को अधिक यथार्थवादी महसूस कराने के लिए स्क्रीन सीमाओं और तरल मूवमेंट को लागू करें। यह NASA के उड़ान नियंत्रण प्रणालियों की तरह है जो अंतरिक्ष यान को सुरक्षित ऑपरेशन सीमाओं से बाहर जाने से रोकती हैं।
यहाँ क्या बनाना है: एक ऐसा सिस्टम बनाएं जो आपके हीरो स्पेसशिप को स्क्रीन पर बनाए रखे, और नियंत्रणों को स्मूथ महसूस कराए। जब खिलाड़ी किसी एरो की दबाए रखते हैं, तो जहाज निरंतर ग्लाइड करे, न कि अलग-अलग कदमों में चले। स्क्रीन की सीमाओं पर पहुँचने पर एक सूक्ष्म प्रभाव जोड़ने पर विचार करें ताकि खेल क्षेत्र के किनारे का संकेत मिले।
agent mode के बारे में यहाँ और जानें।
जैसे-जैसे प्रोजेक्ट बड़े होते हैं, कोड संगठन की महत्ता बढ़ती है। आपने देखा होगा कि आपकी फ़ाइल में फ़ंक्शन, वेरिएबल्स, और क्लासेस घुल-मिल गए हैं। यह मुझे उस तरह याद दिलाता है जैसे अपोलो मिशन कोड संगठन की जिम्मेदारी अभियंताओं की थी, जिन्होंने स्पष्ट, रखरखाव योग्य सिस्टम बनाए जिन पर कई टीमें एक साथ काम कर सकें।
आपका मिशन: एक सॉफ्टवेयर आर्किटेक्ट की तरह सोचें। आप अपना कोड कैसे व्यवस्थित करेंगे ताकि छह महीने बाद आप (या कोई सहयोगी) इसे समझ सके? भले ही सब एक फ़ाइल में हो, आप बेहतर संगठन कर सकते हैं:
- जुड़े फ़ंक्शंस को साफ़ कमेंट हेडर के साथ समूहित करना
- जिम्मेदारियों का पृथक्करण — गेम लॉजिक को रेंडरिंग से अलग रखना
- वेरिएबल्स और फ़ंक्शंस के लिए सुसंगत नामकरण बनाना
- गेम के विभिन्न पक्षों को व्यवस्थित करने के लिए मॉड्यूल या नामस्थान बनाना
- प्रत्येक मुख्य अनुभाग का उद्देश्य व्याख्यायित करने वाला दस्तावेज़ जोड़ना
प्रतिबिंब प्रश्न:
- जब आप वापस आते हैं, तो आपके कोड के कौन से हिस्से समझने में सबसे कठिन होते हैं?
- आप अपना कोड कैसे व्यवस्थित करेंगे ताकि कोई और भी आसानी से योगदान दे सके?
- यदि आप नए फीचर्स जैसे पावर-अप या अलग दुश्मन प्रकार जोड़ना चाहें तो क्या होगा?
हमने सब कुछ स्क्रैच से बनाया है, जो सीखने के लिए शानदार है, लेकिन यहाँ एक छोटा रहस्य है – कुछ अद्भुत जावास्क्रिप्ट फ्रेमवर्क्स हैं जो आपके लिए बहुत सारा भारी काम कर सकते हैं। एक बार जब आप हमने कवर किए मुख्य सिद्धांतों से सहज हो जाएं, तो वहाँ उपलब्ध विकल्पों का पता लगाना उपयोगी है exploring what's available।
फ्रेमवर्क्स को इस तरह समझें जैसे आपके पास एक अच्छी तरह से स्टॉक्ड टूलबॉक्स हो बजाय हर उपकरण को हाथ से बनाने के। वे उन कोड संगठन की चुनौतियों को हल कर सकते हैं जिनकी हमने बात की, साथ ही ऐसी सुविधाएँ प्रदान करते हैं जिन्हें खुद बनाने में हफ्ते लग सकते हैं।
खोजने योग्य बातें:
- गेम इंजन कोड को कैसे व्यवस्थित करते हैं – आप उनकी चतुर पैटर्न देखकर चकित रह जाएंगे
- कैनवास गेम्स को बटर-स्मूद चलाने के लिए प्रदर्शन ट्रिक्स
- आधुनिक जावास्क्रिप्ट सुविधाएँ जो आपके कोड को साफ़ और अधिक रखरखाव योग्य बना सकती हैं
- गेम ऑब्जेक्ट्स और उनके संबंधों को प्रबंधित करने के विभिन्न दृष्टिकोण
timeline
title गेम एनीमेशन और इंटरैक्शन सीखने की प्रगति
section मूवमेंट फंडामेंटल्स (20 मिनट)
एनीमेशन सिद्धांत: फ्रेम आधारित एनीमेशन
: स्थिति अपडेट्स
: निर्देशांक प्रणाली
: सुचारू गति
section इवेंट सिस्टम्स (25 मिनट)
उपयोगकर्ता इनपुट: कीबोर्ड इवेंट हैंडलिंग
: डिफ़ॉल्ट व्यवहार रोकना
: इवेंट ऑब्जेक्ट गुण
: विंडो-स्तरीय सुनना
section गेम आर्किटेक्चर (30 मिनट)
ऑब्जेक्ट डिज़ाइन: इनहेरिटेंस पैटर्न
: बेस क्लास निर्माण
: विशेष व्यवहार
: पॉलीमॉर्फिक इंटरफेस
section कम्युनिकेशन पैटर्न्स (35 मिनट)
पब/सब कार्यान्वयन: इवेंट इमिटर्स
: संदेश स्थिरांक
: कमजोर जुड़ाव
: सिस्टम एकीकरण
section गेम लूप मास्टरी (40 मिनट)
रीयल-टाइम सिस्टम्स: फ्रेम दर नियंत्रण
: अपडेट/रेंडर चक्र
: स्थिति प्रबंधन
: प्रदर्शन अनुकूलन
section उन्नत तकनीकें (45 मिनट)
पेशेवर विशेषताएं: टक्कर पता लगाना
: भौतिकी सिमुलेशन
: स्थिति मशीनें
: घटक सिस्टम
section गेम इंजन अवधारणाएं (1 सप्ताह)
फ्रेमवर्क समझना: एंटिटी-घटक सिस्टम
: दृश्य ग्राफ़
: एसेट पाइपलाइनों
: प्रदर्शन प्रोफाइलिंग
section उत्पादन कौशल (1 महीना)
पेशेवर विकास: कोड संगठन
: टीम सहयोग
: परीक्षण रणनीतियाँ
: डिप्लॉयमेंट अनुकूलन
इस पाठ को पूरा करने के बाद, आपने अब मास्टरी हासिल कर ली है:
- एनिमेशन सिद्धांत: फ्रेम-आधारित मूवमेंट और स्मूथ ट्रांजिशंस
- इवेंट-ड्रिवन प्रोग्रामिंग: सही इवेंट प्रबंधन के साथ कीबोर्ड इनपुट हैंडलिंग
- ऑब्जेक्ट-ओरिएंटेड डिज़ाइन: विरासत और पॉलीमॉर्फिक इंटरफ़ेस
- कम्युनिकेशन पैटर्न्स: पब/सब आर्किटेक्चर फॉर मैंटेनेबल कोड
- गेम लूप आर्किटेक्चर: रियल-टाइम अपडेट और रेंडरिंग साइकिल्स
- इनपुट सिस्टम्स: डिफ़ॉल्ट व्यवहार रोकने के साथ उपयोगकर्ता नियंत्रण मैपिंग
- एसेट मैनेजमेंट: स्प्राइट लोडिंग और कुशल रेंडरिंग तकनीकें
- ब्राउज़र कंसोल खोलें और
addEventListener('keydown', console.log)आज़माएँ ताकि कीबोर्ड इवेंट देखें - एक सरल डिव एलिमेंट बनाएं और उसे एरो कीज़ से मूव करें
-
setIntervalके साथ निरंतर मूवमेंट बनाएँ -
event.preventDefault()के साथ डिफ़ॉल्ट व्यवहार रोकने का प्रयोग करें
- पोस्ट-लेक्चर क्विज़ पूरा करें और इवेंट-ड्रिवन प्रोग्रामिंग समझें
- मूविंग हीरो स्पेसशिप बनाएं पूरी कीबोर्ड नियंत्रण के साथ
- स्मूथ दुश्मन मूवमेंट पैटर्न लागू करें
- गेम ऑब्जेक्ट्स के स्क्रीन छोड़ने से रोकने के लिए सीमाएं जोड़ें
- बेसिक कॉलीजन डिटेक्शन बनाएं गेम ऑब्जेक्ट्स के बीच
- पूरा स्पेस गेम बनाएं, परिष्कृत मूवमेंट और इंटरेक्शन के साथ
- एडवांस मूवमेंट पैटर्न, जैसे कर्व्स, एक्सेलेरेशन और फिजिक्स जोड़ें
- स्मूथ ट्रांजिशंस और ईज़िंग फ़ंक्शन लागू करें
- पार्टिकल इफेक्ट्स और विज़ुअल फीडबैक सिस्टम बनाएं
- 60fps स्मूथ गेमप्ले के लिए गेम प्रदर्शन अनुकूलित करें
- मोबाइल टच कंट्रोल और उत्तरदायी डिज़ाइन जोड़ें
- एडवांस्ड एनिमेशन सिस्टम के साथ जटिल इंटरैक्टिव एप्लिकेशन बनाएं
- GSAP जैसी एनिमेशन लाइब्रेरी सीखें या अपना खुद का एनिमेशन इंजन बनाएं
- ओपन सोर्स गेम डेवलपमेंट और एनिमेशन प्रोजेक्ट्स में योगदान दें
- ग्राफिक्स-गहन एप्लिकेशन के लिए प्रदर्शन अनुकूलन में महारत हासिल करें
- गेम डेवलपमेंट और एनिमेशन पर शैक्षिक सामग्री बनाएं
- एडवांस्ड इंटरैक्टिव प्रोग्रामिंग कौशल दिखाने वाला पोर्टफोलियो बनाएं
वास्तविक दुनिया के अनुप्रयोग: आपकी गेम एनिमेशन कौशल सीधे लागू होती है:
- इंटरएक्टिव वेब एप्लिकेशन: गतिशील डैशबोर्ड और रीयल-टाइम इंटरफेस
- डेटा विज़ुअलाइज़ेशन: एनिमेटेड चार्ट्स और इंटरैक्टिव ग्राफिक्स
- शैक्षिक सॉफ्टवेयर: इंटरैक्टिव सिमुलेशन्स और सीखने के उपकरण
- मोबाइल डेवलपमेंट: टच-आधारित गेम्स और जेस्चर हैंडलिंग
- डेस्कटॉप एप्लिकेशन: इलेक्ट्रॉन ऐप्स स्मूद एनिमेशन के साथ
- वेब एनिमेशन: CSS और जावास्क्रिप्ट एनिमेशन लाइब्रेरीज़
प्रोफेशनल स्किल्स पाए: आप अब कर सकते हैं:
- आर्किटेक्ट जटिलता के साथ स्केल करने वाले इवेंट-ड्रिवन सिस्टम
- इंप्लीमेंट स्मूथ एनिमेशन गणितीय सिद्धांतों के साथ
- डीबग जटिल इंटरेक्शन सिस्टम ब्राउज़र डेवलपर टूल्स से
- ऑप्टिमाइज़ गेम प्रदर्शन विभिन्न डिवाइस और ब्राउज़र के लिए
- डिजाइन रखरखाव योग्य कोड संरचनाएं सिद्ध पॅटर्न्स के साथ
गेम डेवलपमेंट कॉन्सेप्ट्स मास्टर्ड:
- फ्रेम रेट प्रबंधन: FPS और टाइमिंग कंट्रोल समझना
- इनपुट हैंडलिंग: क्रॉस-प्लेटफ़ॉर्म कीबोर्ड और इवेंट सिस्टम
- ऑब्जेक्ट लाइफसायकल: निर्माण, अपडेट, और विनाश पैटर्न
- स्टेट सिंक्रोनाइज़ेशन: फ्रेम्स के बीच गेम स्टेट को सुसंगत रखना
- इवेंट आर्किटेक्चर: गेम सिस्टम्स के बीच decoupled संवाद
अगला स्तर: आप तैयार हैं कोलीजन डिटेक्शन, स्कोरिंग सिस्टम, साउंड इफेक्ट्स जोड़ने के लिए, या आधुनिक गेम फ्रेमवर्क्स जैसे Phaser या Three.js की खोज के लिए!
🌟 उपलब्धि प्राप्त! आपने एक पूरा इंटरैक्टिव गेम सिस्टम प्रोफेशनल आर्किटेक्चर पैटर्न्स के साथ बनाया है!
अस्वीकरण: इस दस्तावेज़ का अनुवाद AI अनुवाद सेवा Co-op Translator का उपयोग करके किया गया है। हालांकि हम सटीकता के लिए प्रयासरत हैं, कृपया ध्यान रखें कि स्वचालित अनुवाद में त्रुटियाँ या गलतियाँ हो सकती हैं। मूल दस्तावेज़ अपनी मूल भाषा में अधिकारिक स्रोत माना जाना चाहिए। महत्वपूर्ण जानकारी के लिए, पेशेवर मानव अनुवाद की सलाह दी जाती है। इस अनुवाद के उपयोग से उत्पन्न किसी भी गलतफहमी या मिसइंटरप्रिटेशन के लिए हम जिम्मेदार नहीं हैं।