我无法弄清楚为什么这个程序失败了。
#!/usr/bin/env python
from __future__ import division, print_function
from future_builtins import *
import types
import libui as ui
from PyQt4 import QtCore
import sip
p = ui.QPoint()
q = QtCore.QPoint()
def _q_getattr(self, attr):
print("get %s" % attr)
value = getattr(sip.wrapinstance(self.myself(), QtCore.QPoint), attr)
print("get2 %s returned %s" % (attr, value))
return value
p.__getattr__ = types.MethodType(_q_getattr, p)
print(p.__getattr__('x')()) # Works! Prints "0"
print(p.x()) # AttributeError: 'QPoint' object has no attribute 'x'
我使用Boost.Python来创建libui,它暴露了类QPoint。我包括PyQt4,它有一个sip暴露的QPoint。我正在尝试完成两种类型之间的映射。
我检查过p
是一个新风格的类,为什么__getattr__
没有调用p.x()
?
答案 0 :(得分:5)
这与昨天的问题someone else has encountered有些相似。简而言之,似乎特殊方法(如__getattr__
,__str__
,__repr__
,__call__
等)在新式类实例中不可覆盖,即您只能在其类型中定义它们。
这是我对该问题的解决方案的改编,希望对你的问题有用:
def _q_getattr(self, attr):
print("get %s" % attr)
return getattr(self, 'x')
def override(p, methods):
oldType = type(p)
newType = type(oldType.__name__ + "_Override", (oldType,), methods)
p.__class__ = newType
override(p, { '__getattr__': _q_getattr})
print(p.__getattr__('x')()) # Works! Prints "0"
print(p.x()) # Should work!
答案 1 :(得分:1)
我建议您不要尝试在boost python中公开QPoint。你应该能够通过boost注册到python的转换器,它将使用SIP api函数将QPoint从/转换为python作为sip对象。
我已经完成了,但最近还不足以提供更多细节。
答案 2 :(得分:1)
这是如何集成PyQt4和boost :: python
的示例首先我们必须定义wrap / unwrap函数来处理裸指针
long int unwrap(QObject* ptr) {
return reinterpret_cast<long int>(ptr);
}
template <typename T>
T* wrap(long int ptr) {
return reinterpret_cast<T*>(ptr);
}
之后我们必须注册我们想要整合的所有类
class_<QObject, QObject*, boost::noncopyable>("QObject", no_init)
.def("unwrap", unwrap)
.def("wrap", make_function( wrap<QObject>, return_value_policy<return_by_value>() ))
.staticmethod("wrap");
class_<QWidget, bases<QObject>, QWidget*, boost::noncopyable>("QWidget")
.def("wrap", make_function( wrap<QWidget>, return_value_policy<return_by_value>() ))
.staticmethod("wrap");
class_<QFrame, bases<QWidget>, QFrame*, boost::noncopyable>("QFrame")
.def("wrap", make_function( wrap<QFrame>, return_value_policy<return_by_value>() ))
.staticmethod("wrap");
class_<QLabel, bases<QFrame>, QLabel*, boost::noncopyable>("QLabel")
.def("wrap", make_function( wrap<QLabel>, return_value_policy<return_by_value>() ))
.staticmethod("wrap");
例如,我们的课程适用于.. QLabel:
class worker: public QObject {
...
void add_label(QLabel*);
};
我们也必须将这个类暴露给python:
class_<worker, bases<QObject>, worker*, boost::noncopyable>("worker")
.def("add_label", &worker::add_label);
现在我们准备互动, 在C ++上 - 大小做这样的事情
worker* w = new worker;
main_namespace["worker"] = boost::ref(w);
蟒:
from PyQt4.Qt import *
import sip
import mylib as MyLib
#...
#If you are using QApplication on C++-size you don't need to create another one
lb = QLabel("label from PyQt4!")
lb_ptr = sip.unwrapinstance(f)
my_lb = MyLib.QLabel.wrap(lb_ptr)
worker.add_label(my_lb)
在其他情况下,如果您不想将自己的Q-object发送给PyQt4:
QLabel* lb = new QLabel("C++ label");
main_namespace["lb"] = boost::ref(lb);
蟒:
from PyQt4.Qt import *
import sip
import mylib as MyLib
#...
my_lb_ptr = lb.unwrap()
qt_lb = sip.wrapinstance(my_lb_ptr, QLabel)
这是我真正的小帮手:
from PyQt4.Qt import *
import sip
def toQt(object, type):
ptr = object.unwrap()
return sip.wrapinstance(ptr, type)
def fromQt(object, type):
ptr = sip.unwrapinstance(object)
return type.wrap(ptr)