// This is the SIP interface definition for the majority of the QMap and // QMultiMap based mapped types. // // Copyright (c) 2021 Riverbank Computing Limited // // This file is part of PyQt5. // // This file may be used under the terms of the GNU General Public License // version 3.0 as published by the Free Software Foundation and appearing in // the file LICENSE included in the packaging of this file. Please review the // following information to ensure the GNU General Public License version 3.0 // requirements will be met: http://www.gnu.org/copyleft/gpl.html. // // If you do not wish to use this file under the terms of the GPL version 3.0 // then you may purchase a commercial license. For more information contact // info@riverbankcomputing.com. // // This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE // WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. template<_TYPE1_, _TYPE2_> %MappedType QMap<_TYPE1_, _TYPE2_> /TypeHint="Dict[_TYPE1_, _TYPE2_]", TypeHintValue="{}"/ { %TypeHeaderCode #include %End %ConvertFromTypeCode PyObject *d = PyDict_New(); if (!d) return 0; QMap<_TYPE1_, _TYPE2_>::const_iterator it = sipCpp->constBegin(); QMap<_TYPE1_, _TYPE2_>::const_iterator end = sipCpp->constEnd(); while (it != end) { _TYPE1_ *k = new _TYPE1_(it.key()); PyObject *kobj = sipConvertFromNewType(k, sipType__TYPE1_, sipTransferObj); if (!kobj) { delete k; Py_DECREF(d); return 0; } _TYPE2_ *v = new _TYPE2_(it.value()); PyObject *vobj = sipConvertFromNewType(v, sipType__TYPE2_, sipTransferObj); if (!vobj) { delete v; Py_DECREF(kobj); Py_DECREF(d); return 0; } int rc = PyDict_SetItem(d, kobj, vobj); Py_DECREF(vobj); Py_DECREF(kobj); if (rc < 0) { Py_DECREF(d); return 0; } ++it; } return d; %End %ConvertToTypeCode if (!sipIsErr) return PyDict_Check(sipPy); QMap<_TYPE1_, _TYPE2_> *qm = new QMap<_TYPE1_, _TYPE2_>; Py_ssize_t pos = 0; PyObject *kobj, *vobj; while (PyDict_Next(sipPy, &pos, &kobj, &vobj)) { int kstate; _TYPE1_ *k = reinterpret_cast<_TYPE1_ *>( sipForceConvertToType(kobj, sipType__TYPE1_, sipTransferObj, SIP_NOT_NONE, &kstate, sipIsErr)); if (*sipIsErr) { PyErr_Format(PyExc_TypeError, "a dict key has type '%s' but '_TYPE1_' is expected", sipPyTypeName(Py_TYPE(kobj))); delete qm; return 0; } int vstate; _TYPE2_ *v = reinterpret_cast<_TYPE2_ *>( sipForceConvertToType(vobj, sipType__TYPE2_, sipTransferObj, SIP_NOT_NONE, &vstate, sipIsErr)); if (*sipIsErr) { PyErr_Format(PyExc_TypeError, "a dict value has type '%s' but '_TYPE2_' is expected", sipPyTypeName(Py_TYPE(vobj))); sipReleaseType(k, sipType__TYPE1_, kstate); delete qm; return 0; } qm->insert(*k, *v); sipReleaseType(v, sipType__TYPE2_, vstate); sipReleaseType(k, sipType__TYPE1_, kstate); } *sipCppPtr = qm; return sipGetState(sipTransferObj); %End }; template %MappedType QMap /TypeHint="Dict[int, _TYPE_]", TypeHintValue="{}"/ { %TypeHeaderCode #include %End %ConvertFromTypeCode PyObject *d = PyDict_New(); if (!d) return 0; QMap::const_iterator it = sipCpp->constBegin(); QMap::const_iterator end = sipCpp->constEnd(); while (it != end) { PyObject *kobj = SIPLong_FromLong(it.key()); if (!kobj) { Py_DECREF(d); return 0; } _TYPE_ *v = new _TYPE_(it.value()); PyObject *vobj = sipConvertFromNewType(v, sipType__TYPE_, sipTransferObj); if (!vobj) { delete v; Py_DECREF(kobj); Py_DECREF(d); return 0; } int rc = PyDict_SetItem(d, kobj, vobj); Py_DECREF(vobj); Py_DECREF(kobj); if (rc < 0) { Py_DECREF(d); return 0; } ++it; } return d; %End %ConvertToTypeCode if (!sipIsErr) return PyDict_Check(sipPy); QMap *qm = new QMap; Py_ssize_t pos = 0; PyObject *kobj, *vobj; while (PyDict_Next(sipPy, &pos, &kobj, &vobj)) { int k = sipLong_AsInt(kobj); if (PyErr_Occurred()) { if (PyErr_ExceptionMatches(PyExc_TypeError)) PyErr_Format(PyExc_TypeError, "a dict key has type '%s' but 'int' is expected", sipPyTypeName(Py_TYPE(kobj))); delete qm; *sipIsErr = 1; return 0; } int vstate; _TYPE_ *v = reinterpret_cast<_TYPE_ *>( sipForceConvertToType(vobj, sipType__TYPE_, sipTransferObj, SIP_NOT_NONE, &vstate, sipIsErr)); if (*sipIsErr) { PyErr_Format(PyExc_TypeError, "a dict value has type '%s' but '_TYPE_' is expected", sipPyTypeName(Py_TYPE(vobj))); delete qm; return 0; } qm->insert(k, *v); sipReleaseType(v, sipType__TYPE_, vstate); } *sipCppPtr = qm; return sipGetState(sipTransferObj); %End }; template<_TYPE1_, _TYPE2_> %MappedType QMultiMap<_TYPE1_, _TYPE2_> /TypeHintOut="Dict[_TYPE1_, List[_TYPE2_]]", TypeHintValue="{}"/ { %TypeHeaderCode #include %End %ConvertFromTypeCode PyObject *d = PyDict_New(); if (!d) return 0; QList<_TYPE1_> keys = sipCpp->keys(); QList<_TYPE1_>::const_iterator kit = keys.constBegin(); QList<_TYPE1_>::const_iterator kit_end = keys.constEnd(); while (kit != kit_end) { _TYPE1_ *k = new _TYPE1_(*kit); PyObject *kobj = sipConvertFromNewType(k, sipType__TYPE1_, sipTransferObj); if (!kobj) { delete k; Py_DECREF(d); return 0; } // Create a Python list as the dictionary value. QList<_TYPE2_> values = sipCpp->values(*kit); PyObject *vobj = PyList_New(values.count()); if (!vobj) { Py_DECREF(kobj); Py_DECREF(d); return 0; } QList<_TYPE2_>::const_iterator vit = values.constBegin(); QList<_TYPE2_>::const_iterator vit_end = values.constEnd(); for (int i = 0; vit != vit_end; ++i) { _TYPE2_ *sv = new _TYPE2_(*vit); PyObject *svobj = sipConvertFromNewType(sv, sipType__TYPE2_, sipTransferObj); if (!svobj) { delete sv; Py_DECREF(vobj); Py_DECREF(kobj); Py_DECREF(d); return 0; } PyList_SetItem(vobj, i, svobj); ++vit; } int rc = PyDict_SetItem(d, kobj, vobj); Py_DECREF(vobj); Py_DECREF(kobj); if (rc < 0) { Py_DECREF(d); return 0; } ++kit; } return d; %End %ConvertToTypeCode if (!sipIsErr) return PyDict_Check(sipPy); // Note that PyQt v5.1 contains an unused implementation that can be // restored if needed (although it will need updating to accept an iterable // rather than just a list of values). PyErr_SetString(PyExc_NotImplementedError, "converting to QMultiMap<_TYPE1_, _TYPE2_> is unsupported"); return 0; %End };