2025, Sep 26 19:32
ScreenManager के साथ Kivy पोंग में गलत टक्करें: कारण और फिक्स
Kivy पोंग में ScreenManager के बाद टक्करें ऑफसेट दिखें तो वजह विजेट-कैनवास साइज मिसमैच है। Orb व Bat पर size सेट कर बग बिना गेम लूप बदले ठीक करें, आसानी से.
ScreenManager के साथ एक मुख्य मेनू जोड़ने पर साधारण Kivy पोंग क्लोन में कभी-कभी अजीब विजुअल समस्या दिखती है: पैडल और बॉल देखने में तो ठीक केंद्र में लगते हैं, लेकिन टक्करें और किनारों की जांच ऐसे बर्ताव करती हैं मानो वे कहीं और हों। गेम चलता है, बॉल उछलती है, पर यह सब अदृश्य सीमाओं के भीतर होता है। Position और size hints काम नहीं आते, और pos: 0, 0 ज़बरदस्ती देने पर सब कुछ कोने से चिपक जाता है। नीचे उसी स्थिति का न्यूनतम, पुनरुत्पाद्य सेटअप और उसका सटीक समाधान दिया गया है।
समस्या की रूपरेखा
यह कोड मेनू और गेम स्क्रीन को जोड़ता है, गेम लूप शेड्यूल करता है, और KV कैनवास से बॉल व पैडल बनाता है। लॉजिक काम करता है, लेकिन टक्करें खिसकी हुई महसूस होती हैं और दायां पैडल स्क्रीन के किनारे से पूरी तरह नहीं मिलता।
मुख्य मॉड्यूल:
from kivy.app import App
from kivy.properties import NumericProperty, ReferenceListProperty, ObjectProperty
from kivy.uix.screenmanager import Screen, ScreenManager, NoTransition
from kivy.uix.widget import Widget
from kivy.vector import Vector
from kivy.clock import Clock
class Bat(Widget):
points = NumericProperty(0)
def deflect_orb(self, orb):
if self.collide_widget(orb):
vx, vy = orb.momentum
offset = (orb.center_y - self.center_y) / (self.height / 2)
reflected = Vector(-1 * vx, vy)
impulse = reflected * 1.1
orb.momentum = impulse.x, impulse.y + offset
class Orb(Widget):
speed_x = NumericProperty(0)
speed_y = NumericProperty(0)
momentum = ReferenceListProperty(speed_x, speed_y)
def advance(self):
self.pos = Vector(*self.momentum) + self.pos
class Arena(Widget):
orb = ObjectProperty(None)
left_bat = ObjectProperty(None)
right_bat = ObjectProperty(None)
def toss_orb(self, vel=(4, 0)):
self.orb.center = self.center
self.orb.momentum = vel
def step_frame(self, dt):
self.orb.advance()
self.left_bat.deflect_orb(self.orb)
self.right_bat.deflect_orb(self.orb)
if (self.orb.y < 0) or (self.orb.top > self.height):
self.orb.speed_y *= -1
if self.orb.x < self.x:
self.right_bat.points += 1
self.toss_orb(vel=(4,0))
if self.orb.x > self.width:
self.left_bat.points += 1
self.toss_orb(vel=(-4,0))
def on_touch_move(self, touch):
if touch.x < self.width / 3:
self.left_bat.center_y = touch.y
if touch.x > self.width - self.width / 3:
self.right_bat.center_y = touch.y
class ArenaScreen(Screen):
def on_enter(self, *args):
match = Arena()
self.add_widget(match)
match.toss_orb()
Clock.schedule_interval(match.step_frame, 1 / 60)
return match
class MainMenu(Screen):
pass
class PongSuite(App):
def build(self):
sm = ScreenManager(transition=NoTransition())
sm.add_widget(MainMenu(name="menu"))
sm.add_widget(ArenaScreen(name="game"))
return sm
if __name__ == "__main__":
PongSuite().run()
KV फ़ाइल:
#:kivy 2.3.1
<Orb>:
canvas:
Ellipse:
pos: self.pos
size: 50, 50
<Bat>:
canvas:
Rectangle:
pos: self.pos
size: 25, 200
<Arena>:
orb: pong_orb
left_bat: bat_left
right_bat: bat_right
canvas:
Rectangle:
pos: self.center_x - 5, 0
size: 10, self.height
Label:
font_size: 70
center_x: root.width / 4
top: root.top - 50
text: str(root.left_bat.points)
Label:
font_size: 70
center_x: root.width * 3 / 4
top: root.top - 50
text: str(root.right_bat.points)
Orb:
id: pong_orb
center: self.parent.center
Bat:
id: bat_left
x: self.parent.x
center_y: self.parent.center_y
Bat:
id: bat_right
x: self.parent.width - self.width
center_y: self.parent.center_y
<MainMenu>:
start_btn: action_start
BoxLayout:
orientation: 'vertical'
Button:
id: action_start
text: "[b]Start Game[/b]"
markup: True
width: 200
height: 200
on_press: root.manager.current = "game"
असल में गड़बड़ी क्या है
गेम में बॉल के लिए 50×50 की Ellipse और हर पैडल के लिए 25×200 का Rectangle ड्रॉ होता है। लेकिन जिन विजेट्स पर ये ड्रॉइंग्स हैं, वे अपनी डिफ़ॉल्ट साइज ही रखते हैं। मान जांचने पर बॉल विजेट 100×100 दिखता है, और पैडल विजेट भी अपनी ड्रॉइंग के आकार के मुताबिक नहीं है। इस असंगति से दिखाई देने वाले आकार और वास्तविक विजेट ज्योमेट्री अलग हो जाते हैं। नतीजतन, स्क्रीन किनारों पर प्लेसमेंट और टक्कर जांच ऐसे बर्ताव करते हैं मानो ऑब्जेक्ट बड़े और खिसके हुए हों।
पहली नज़र में यह लेआउट की समस्या लगती है, खासकर ScreenManager जोड़ने के बाद, लेकिन असल वजह यह है कि विजेट का आकार उसके कैनवास पर बने शेप से मेल नहीं खा रहा।
समाधान
विजेट्स पर ही स्पष्ट साइज सेट करें ताकि वे कैनवास प्रिमिटिव्स से मेल खाएं। ऐसा करते ही दायां पैडल बॉर्डर के साथ ठीक बैठता है और टक्करें विजुअल्स के अनुरूप हो जाती हैं।
<Orb>:
size: 50, 50
canvas:
Ellipse:
pos: self.pos
size: 50, 50
<Bat>:
size: 25, 200
canvas:
Rectangle:
pos: self.pos
size: 25, 200
इस अपडेट के बाद टक्करें सही दिखती हैं और दायां पैडल ठीक जगह पर आता है (कम से कम Linux पर)। विंडो का आकार बदलने पर भी टक्करें सही लगती रहती हैं और ऑब्जेक्ट का साइज वही रहता है, जबकि दूरियां बड़ी नजर आती हैं। यह प्रभाव चाहिए या नहीं, यह लक्षित व्यवहार पर निर्भर करेगा।
यह बात ध्यान में क्यों रखनी चाहिए
Kivy में canvas निर्देश तय करते हैं कि आप क्या देखते हैं, लेकिन वे विजेट का अपना आकार नहीं बदलते। विजेट ज्योमेट्री पर आधारित कोई भी लॉजिक—स्क्रीन के सापेक्ष पोजिशनिंग से लेकर एलिमेंट्स की परस्पर क्रिया तक—विजेट के आकार का ही उपयोग करता है। इसलिए विजुअल शेप और विजेट बाउंड्स को एक जैसा रखने से उलझाने वाले ऑफ़सेट्स और “अदृश्य” हिटबॉक्स से बचा जा सकता है।
निष्कर्ष
जब आप कैनवास पर गेम एलिमेंट्स को कस्टम-ड्रॉ करते हैं, तो विजेट की साइज को ड्रॉ किए गए शेप के साथ संरेखित रखें। इसी मामले में, बॉल के लिए size: 50, 50 और पैडल के लिए size: 25, 200 स्पष्ट रूप से सेट करने से ऑफ़सेट टक्करें और बॉर्डर अलाइनमेंट की समस्याएं बिना गेम लूप या ScreenManager सेटअप को छेड़े दूर हो जाती हैं।
यह लेख StackOverflow पर प्रश्न (लेखक: blindcat97) और furas द्वारा दिए गए उत्तर पर आधारित है।