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



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

我遇到了一个小问题,试图使用firebreath创建一个混合python/c++插件。到目前为止,我已经成功地使用boost/python.h集成了所有部分,但当我试图从python中启动一个事件时,出现了一个问题。我遇到了一个问题,必须将python函数与c++函数绑定在一起(使用BOOST_PITHON_MODULE)。首先,我尝试将python与我的JSAPI派生类fbtestconpythonAPI直接绑定,这种方法的问题似乎是缺乏对浏览器实例化的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;
}

快速查看,您可以设置类以进行导出,如下所示:

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

然后,您可以将对C++实例的引用传递给Python,允许您调用其上的函数,这些函数反过来可以触发事件。

相关内容

  • 没有找到相关文章

最新更新