#include "PySceneObject.h" #include "PyScene.h" #include "GameEngine.h" #include "McRFPy_API.h" #include "McRFPy_Doc.h" #include // Static map to store Python scene objects by name static std::map python_scenes; PyObject* PySceneClass::__new__(PyTypeObject* type, PyObject* args, PyObject* kwds) { PySceneObject* self = (PySceneObject*)type->tp_alloc(type, 0); if (self) { self->initialized = false; // Don't create C++ scene yet - wait for __init__ } return (PyObject*)self; } int PySceneClass::__init__(PySceneObject* self, PyObject* args, PyObject* kwds) { static const char* keywords[] = {"name", nullptr}; const char* name = nullptr; if (!PyArg_ParseTupleAndKeywords(args, kwds, "s", const_cast(keywords), &name)) { return -1; } // Check if scene with this name already exists if (python_scenes.count(name) > 0) { PyErr_Format(PyExc_ValueError, "Scene with name '%s' already exists", name); return -1; } self->name = name; // Create the C++ PyScene McRFPy_API::game->createScene(name); // Get reference to the created scene GameEngine* game = McRFPy_API::game; if (!game) { PyErr_SetString(PyExc_RuntimeError, "No game engine initialized"); return -1; } // Store this Python object in our registry python_scenes[name] = self; Py_INCREF(self); // Keep a reference // Create a Python function that routes to on_keypress // We'll register this after the object is fully initialized self->initialized = true; return 0; } void PySceneClass::__dealloc(PyObject* self_obj) { PySceneObject* self = (PySceneObject*)self_obj; // Remove from registry if (python_scenes.count(self->name) > 0 && python_scenes[self->name] == self) { python_scenes.erase(self->name); } // Call Python object destructor Py_TYPE(self)->tp_free(self); } PyObject* PySceneClass::__repr__(PySceneObject* self) { return PyUnicode_FromFormat("", self->name.c_str()); } PyObject* PySceneClass::activate(PySceneObject* self, PyObject* args) { // Call the static method from McRFPy_API PyObject* py_args = Py_BuildValue("(s)", self->name.c_str()); PyObject* result = McRFPy_API::_setScene(NULL, py_args); Py_DECREF(py_args); return result; } PyObject* PySceneClass::get_ui(PySceneObject* self, PyObject* args) { // Call the static method from McRFPy_API PyObject* py_args = Py_BuildValue("(s)", self->name.c_str()); PyObject* result = McRFPy_API::_sceneUI(NULL, py_args); Py_DECREF(py_args); return result; } PyObject* PySceneClass::register_keyboard(PySceneObject* self, PyObject* args) { PyObject* callable; if (!PyArg_ParseTuple(args, "O", &callable)) { return NULL; } if (!PyCallable_Check(callable)) { PyErr_SetString(PyExc_TypeError, "Argument must be callable"); return NULL; } // Store the callable Py_INCREF(callable); // Get the current scene and set its key_callable GameEngine* game = McRFPy_API::game; if (game) { // We need to be on the right scene first std::string old_scene = game->scene; game->scene = self->name; game->currentScene()->key_callable = std::make_unique(callable); game->scene = old_scene; } Py_DECREF(callable); Py_RETURN_NONE; } PyObject* PySceneClass::get_name(PySceneObject* self, void* closure) { return PyUnicode_FromString(self->name.c_str()); } PyObject* PySceneClass::get_active(PySceneObject* self, void* closure) { GameEngine* game = McRFPy_API::game; if (!game) { Py_RETURN_FALSE; } return PyBool_FromLong(game->scene == self->name); } // #118: Scene position getter static PyObject* PySceneClass_get_pos(PySceneObject* self, void* closure) { GameEngine* game = McRFPy_API::game; if (!game) { Py_RETURN_NONE; } // Get the scene by name using the public accessor auto scene = game->getScene(self->name); if (!scene) { Py_RETURN_NONE; } // Create a Vector object auto type = (PyTypeObject*)PyObject_GetAttrString(McRFPy_API::mcrf_module, "Vector"); if (!type) return NULL; PyObject* args = Py_BuildValue("(ff)", scene->position.x, scene->position.y); PyObject* result = PyObject_CallObject((PyObject*)type, args); Py_DECREF(type); Py_DECREF(args); return result; } // #118: Scene position setter static int PySceneClass_set_pos(PySceneObject* self, PyObject* value, void* closure) { GameEngine* game = McRFPy_API::game; if (!game) { PyErr_SetString(PyExc_RuntimeError, "No game engine"); return -1; } auto scene = game->getScene(self->name); if (!scene) { PyErr_SetString(PyExc_RuntimeError, "Scene not found"); return -1; } // Accept tuple or Vector float x, y; if (PyTuple_Check(value) && PyTuple_Size(value) == 2) { x = PyFloat_AsDouble(PyTuple_GetItem(value, 0)); y = PyFloat_AsDouble(PyTuple_GetItem(value, 1)); } else if (PyObject_HasAttrString(value, "x") && PyObject_HasAttrString(value, "y")) { PyObject* xobj = PyObject_GetAttrString(value, "x"); PyObject* yobj = PyObject_GetAttrString(value, "y"); x = PyFloat_AsDouble(xobj); y = PyFloat_AsDouble(yobj); Py_DECREF(xobj); Py_DECREF(yobj); } else { PyErr_SetString(PyExc_TypeError, "pos must be a tuple (x, y) or Vector"); return -1; } scene->position = sf::Vector2f(x, y); return 0; } // #118: Scene visible getter static PyObject* PySceneClass_get_visible(PySceneObject* self, void* closure) { GameEngine* game = McRFPy_API::game; if (!game) { Py_RETURN_TRUE; } auto scene = game->getScene(self->name); if (!scene) { Py_RETURN_TRUE; } return PyBool_FromLong(scene->visible); } // #118: Scene visible setter static int PySceneClass_set_visible(PySceneObject* self, PyObject* value, void* closure) { GameEngine* game = McRFPy_API::game; if (!game) { PyErr_SetString(PyExc_RuntimeError, "No game engine"); return -1; } auto scene = game->getScene(self->name); if (!scene) { PyErr_SetString(PyExc_RuntimeError, "Scene not found"); return -1; } if (!PyBool_Check(value)) { PyErr_SetString(PyExc_TypeError, "visible must be a boolean"); return -1; } scene->visible = PyObject_IsTrue(value); return 0; } // #118: Scene opacity getter static PyObject* PySceneClass_get_opacity(PySceneObject* self, void* closure) { GameEngine* game = McRFPy_API::game; if (!game) { return PyFloat_FromDouble(1.0); } auto scene = game->getScene(self->name); if (!scene) { return PyFloat_FromDouble(1.0); } return PyFloat_FromDouble(scene->opacity); } // #118: Scene opacity setter static int PySceneClass_set_opacity(PySceneObject* self, PyObject* value, void* closure) { GameEngine* game = McRFPy_API::game; if (!game) { PyErr_SetString(PyExc_RuntimeError, "No game engine"); return -1; } auto scene = game->getScene(self->name); if (!scene) { PyErr_SetString(PyExc_RuntimeError, "Scene not found"); return -1; } double opacity; if (PyFloat_Check(value)) { opacity = PyFloat_AsDouble(value); } else if (PyLong_Check(value)) { opacity = PyLong_AsDouble(value); } else { PyErr_SetString(PyExc_TypeError, "opacity must be a number"); return -1; } // Clamp to valid range if (opacity < 0.0) opacity = 0.0; if (opacity > 1.0) opacity = 1.0; scene->opacity = opacity; return 0; } // Lifecycle callbacks void PySceneClass::call_on_enter(PySceneObject* self) { PyObject* method = PyObject_GetAttrString((PyObject*)self, "on_enter"); if (method && PyCallable_Check(method)) { PyObject* result = PyObject_CallNoArgs(method); if (result) { Py_DECREF(result); } else { PyErr_Print(); } Py_DECREF(method); } else { // Clear AttributeError if method doesn't exist PyErr_Clear(); Py_XDECREF(method); } } void PySceneClass::call_on_exit(PySceneObject* self) { PyObject* method = PyObject_GetAttrString((PyObject*)self, "on_exit"); if (method && PyCallable_Check(method)) { PyObject* result = PyObject_CallNoArgs(method); if (result) { Py_DECREF(result); } else { PyErr_Print(); } Py_DECREF(method); } else { // Clear AttributeError if method doesn't exist PyErr_Clear(); Py_XDECREF(method); } } void PySceneClass::call_on_keypress(PySceneObject* self, std::string key, std::string action) { PyGILState_STATE gstate = PyGILState_Ensure(); PyObject* method = PyObject_GetAttrString((PyObject*)self, "on_keypress"); if (method && PyCallable_Check(method)) { PyObject* result = PyObject_CallFunction(method, "ss", key.c_str(), action.c_str()); if (result) { Py_DECREF(result); } else { PyErr_Print(); } Py_DECREF(method); } else { // Clear AttributeError if method doesn't exist PyErr_Clear(); Py_XDECREF(method); } PyGILState_Release(gstate); } void PySceneClass::call_update(PySceneObject* self, float dt) { PyObject* method = PyObject_GetAttrString((PyObject*)self, "update"); if (method && PyCallable_Check(method)) { PyObject* result = PyObject_CallFunction(method, "f", dt); if (result) { Py_DECREF(result); } else { PyErr_Print(); } Py_DECREF(method); } else { // Clear AttributeError if method doesn't exist PyErr_Clear(); Py_XDECREF(method); } } void PySceneClass::call_on_resize(PySceneObject* self, int width, int height) { PyObject* method = PyObject_GetAttrString((PyObject*)self, "on_resize"); if (method && PyCallable_Check(method)) { PyObject* result = PyObject_CallFunction(method, "ii", width, height); if (result) { Py_DECREF(result); } else { PyErr_Print(); } Py_DECREF(method); } else { // Clear AttributeError if method doesn't exist PyErr_Clear(); Py_XDECREF(method); } } // Properties PyGetSetDef PySceneClass::getsetters[] = { {"name", (getter)get_name, NULL, MCRF_PROPERTY(name, "Scene name (str, read-only). Unique identifier for this scene."), NULL}, {"active", (getter)get_active, NULL, MCRF_PROPERTY(active, "Whether this scene is currently active (bool, read-only). Only one scene can be active at a time."), NULL}, // #118: Scene-level UIDrawable-like properties {"pos", (getter)PySceneClass_get_pos, (setter)PySceneClass_set_pos, MCRF_PROPERTY(pos, "Scene position offset (Vector). Applied to all UI elements during rendering."), NULL}, {"visible", (getter)PySceneClass_get_visible, (setter)PySceneClass_set_visible, MCRF_PROPERTY(visible, "Scene visibility (bool). If False, scene is not rendered."), NULL}, {"opacity", (getter)PySceneClass_get_opacity, (setter)PySceneClass_set_opacity, MCRF_PROPERTY(opacity, "Scene opacity (0.0-1.0). Applied to all UI elements during rendering."), NULL}, {NULL} }; // Methods PyMethodDef PySceneClass::methods[] = { {"activate", (PyCFunction)activate, METH_NOARGS, MCRF_METHOD(SceneClass, activate, MCRF_SIG("()", "None"), MCRF_DESC("Make this the active scene."), MCRF_RETURNS("None") MCRF_NOTE("Deactivates the current scene and activates this one. Scene transitions and lifecycle callbacks are triggered.") )}, {"get_ui", (PyCFunction)get_ui, METH_NOARGS, MCRF_METHOD(SceneClass, get_ui, MCRF_SIG("()", "UICollection"), MCRF_DESC("Get the UI element collection for this scene."), MCRF_RETURNS("UICollection: Collection of UI elements (Frames, Captions, Sprites, Grids) in this scene") MCRF_NOTE("Use to add, remove, or iterate over UI elements. Changes are reflected immediately.") )}, {"register_keyboard", (PyCFunction)register_keyboard, METH_VARARGS, MCRF_METHOD(SceneClass, register_keyboard, MCRF_SIG("(callback: callable)", "None"), MCRF_DESC("Register a keyboard event handler function."), MCRF_ARGS_START MCRF_ARG("callback", "Function that receives (key: str, pressed: bool) when keyboard events occur") MCRF_RETURNS("None") MCRF_NOTE("Alternative to overriding on_keypress() method. Handler is called for both key press and release events.") )}, {NULL} }; // Helper function to trigger lifecycle events void McRFPy_API::triggerSceneChange(const std::string& from_scene, const std::string& to_scene) { // Call on_exit for the old scene if (!from_scene.empty() && python_scenes.count(from_scene) > 0) { PySceneClass::call_on_exit(python_scenes[from_scene]); } // Call on_enter for the new scene if (!to_scene.empty() && python_scenes.count(to_scene) > 0) { PySceneClass::call_on_enter(python_scenes[to_scene]); } } // Helper function to update Python scenes void McRFPy_API::updatePythonScenes(float dt) { GameEngine* game = McRFPy_API::game; if (!game) return; // Only update the active scene if (python_scenes.count(game->scene) > 0) { PySceneClass::call_update(python_scenes[game->scene], dt); } } // Helper function to trigger resize events on Python scenes void McRFPy_API::triggerResize(int width, int height) { GameEngine* game = McRFPy_API::game; if (!game) return; // Only notify the active scene if (python_scenes.count(game->scene) > 0) { PySceneClass::call_on_resize(python_scenes[game->scene], width, height); } }