如何使用boost和Firebreath正确地从python中触发浏览器事件

时间:2012-02-16 19:12:22

标签: c++ python boost boost-python firebreath

首先我必须说,作为一名Python程序员,我可能会从错误的角度看待这个问题,自从我在大学写完最后一个c ++代码以来已经过去很多年了。

我遇到了一些问题,尝试使用firebreath创建一个混合python / c ++插件。到目前为止,我已经成功地使用boost / python.h集成了所有部分,但是当我尝试从python中激活事件时出现了一个问题。我偶然发现了必须将python函数与c ++函数绑定在一起的问题(使用BOOST_PYTHON_MODULE)。首先我尝试直接使用我的JSAPI派生类fbtestconpythonAPI绑定python,这种方法的问题似乎是缺少对浏览器实例化的JSAPI对象的引用,给出了python函数和c ++等价物之间的各种签名不匹配问题。执行时间处理时间。

唯一让我解决这个问题的方法(我同意,这是一个丑陋的脏解决方案),就是使用一个全局指针,我用set_pluginPointer手工初始化。到目前为止,这实际上运作得很好,但我知道这不是正确的方法。我读到我不应该使用带有JSAPI对象的“原始”指针,但我不确定如何用这个特定实现的shared_ptr替换它。另一个问题是全局变量,它在所有实例中共享,导致例如在打开的最后一个选项卡/窗口上触发所有事件。解决后者的一种方法是创建某种类型的数组,索引是当前的窗口/线程id,这是我应该能够从我的JSAPI对象和python / c ++函数访问的东西。

当然我是开放的,并且会非常谨慎地提出任何建议,如何改进/修复这个特定的解决方法,或者更好的是,正确的方式来交流boost :: python和firebreath而不进行黑客攻击。

以下是插件代码的相关部分

// Global pointer to plugin instance
fbtestconpythonAPI *fbtestPtr;
void fbtestconpythonAPI::set_pluginPointer(const std::string& val){
    m_testString = val;
    fbtestPtr = this; //Global pointer initialization
}

void echo(std::string x){
    // Firing the echo event on the plugin instance using the global raw pointer
    fbtestPtr->fire_echo(x, 1);
}

BOOST_PYTHON_MODULE(Pointless) {
    def("echo", echo);
}

FB::variant fbtestconpythonAPI::echo(const FB::variant& msg){
    int result_value;
    Py_Initialize(); 

    try {
        initPointless(); // initialize Pointless

        PyRun_SimpleString("import Pointless");
        PyRun_SimpleString("Pointless.echo('hello world')");

        object module(handle<>(borrowed(PyImport_AddModule("__main__"))));
        object dictionary = module.attr("__dict__");
    } catch (error_already_set) {
        PyErr_Print();
    }

    Py_Finalize();
    return 0;
}

1 个答案:

答案 0 :(得分:1)

从快速浏览一下,您就可以设置您的课程,如下所示:

class_<YourAPI, boost::noncopyable>("YourAPI", no_init)
    .def("function", &YourAPI::function);

然后你可以将a reference传递给你的C ++实例到Python,允许你调用它上面的函数,这反过来可以触发事件。