免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1541 | 回复: 0
打印 上一主题 下一主题

kde4下KParts入门 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2008-01-15 19:48 |只看该作者 |倒序浏览

kde4刚刚方布,KParts的基本原理没有变化,但是接口已经大大简化,尤其是Part工厂的编写工作少了很多,kde的techbase上也没有更新这部分的文档,甚至在kde4的实现代码中也没有完全迁移到新的接口(例如有些标记为KDE_DEPRECATED的接口还被使用)。废话不说了,下面进入正题。最基本的原理我就不说了,重点关注一下kde4中的变化,下文中我就以一个例子做说明。
一个有Part的kde程序大概会有几部分组成:一个Shell,作为所有Part的容器,一组Part(可由KParts::PartManager管理),和许多的KPart::Plugin。Part和Plugin都是KXMLGUIClient的间接子类,这就意味着这些部件的GUI是可以动态融合的(主要是菜单项和工具栏),Shell程序没有这个要求。
一个Part应该包含一个继承自KParts::Part接口(或其子类)的类定义和定义其GUI部分的XML文件。先来看notepad_part.h:
class NotepadPart: public KParts::ReadWritePart
{
    Q_OBJECT
        public:
    NotepadPart(QWidget* parentWidget, QObject *parent, const QStringList& args);
    ~NotepadPart();
    void setReadWrite(bool rw);
    static KAboutData *createAboutData();
   
protected:
    bool saveFile();
    bool openFile();
    protected slots:
    void slotSelectAll();
protected:
    QTextEdit *m_edit;
    KComponentData *m_instance;
};
我从KParts::ReadWritePart继承, 然后实现了saveFile和openFile两个纯虚函数。
NotepadPart(QWidget* parentWidget, QObject *parent, const QStringList& args)
为了使用新的Part构建方法,构造函数必须定义成上面的形式。
static KAboutData *createAboutData()
这个方法是Part工厂要求实现的,后面将会看到。
另一个区别是kde4中将原来的KInstance类改名为KComponentData,以使其功能更明确。
现在看实现部分 notepad_part.cpp
typedef KParts::GenericFactory NotepadPartFactory;
K_EXPORT_COMPONENT_FACTORY( libnotepad_part, NotepadPartFactory )
这两行用来定义一个NotepadPart的工厂类,并且生成合适的构造函数和初始化函数。这样工厂类就可以在合适的时候知道如何生成NotepadPart对象。上面notepad_part.h中定义的构造函数声明和createAboutData将被工厂调用。这比以前的方式要简洁很多,你也可以不使用上面的方法定义工厂类,而使用kde3的方式。
NotepadPart::NotepadPart(QWidget* parentWidget, QObject *parent, const QStringList& args)
    : KParts::ReadWritePart(parent)
{
    setComponentData( NotepadPartFactory::componentData() );
    QString rcfile = KStandardDirs::locate( "data", "notepad_part/notepad_part.rc" );
    if ( !rcfile.isEmpty() )
        setXMLFile( rcfile );
   
    m_edit = new QTextEdit( parentWidget );
    m_edit->setFocus();
    setWidget( m_edit );
    KAction *actSel = new KAction(QString("Select All"), this );
    actionCollection()->addAction("selectAll", actSel);
    connect( actSel, SIGNAL(triggered()), this, SLOT(slotSelectAll()) );
   
    setReadWrite( true );   
}
在NotepadPart的构造函数中,应该使用setComponentData初始化Part的ComponentData对象,工厂类负责对象的ComponentData、KAboutData等的处理,我们可以重新实现其createComponentData接口来提供新的信息。createAboutData的实现如下:
KAboutData* NotepadPart::createAboutData()
{
    KAboutData* aboutData = new KAboutData("notepad_part", 0, ki18n("NotepadPart"), "0.1");
    aboutData->addAuthor( ki18n("sycao"), KLocalizedString(), "yinshuiboy@gmail.com" );
    return aboutData;
}
一个part需要一个rc文件来定义其GUI,notepad_part.rc如下:
  
   
      Edit
      
     
  
下面再定义一个NotepadPart的插件,插件从KParts::Plugin继承,定义如下:
class SpellCheck: public KParts::Plugin
{
    Q_OBJECT
        public:
    SpellCheck(QObject *parent = 0,
               const QStringList& args = QStringList());
    ~SpellCheck();
    public slots:
    void slotSpellCheck();
};
这里值得注意的是,构造函数的声明方式由其工厂类的需要决定的。插件的工厂可以KGenericFactory模板定义如下:
typedef KGenericFactory SpellCheckFactory;
K_EXPORT_COMPONENT_FACTORY( libspellcheck, SpellCheckFactory("spellcheck") )
插件的构造函数如下:
SpellCheck::SpellCheck(QObject *parent, const QStringList& args)
    : KParts::Plugin( parent )
{
    //    setComponentData( SpellCheckFactory::componentData() );
    KAction *actSC = new KAction( "spell check", this );
    connect( actSC, SIGNAL(triggered()),
             this, SLOT(slotSpellCheck()) );
    actionCollection()->addAction("spell_check", actSC );
}
通常情况下,我们把插件放在相关part的子目录kpartplugins目录下,那么该part被载入时,会自动寻找和融合plugin的界面。这里不需要在构造函数里调用setXMLFile,但是同样要提供一个rc文件和一个desktop服务描述文件。desktop文件如下:
[Desktop Entry]
Encoding=UTF-8
Icon=mypluginicon
Type=Service
ServiceTypes=KPluginInfo
Name=spellCheck
Comment=Description of what the plugin does
X-KDE-PluginInfo-EnabledByDefault=true
X-KDE-PluginInfo-Author=sycao
X-KDE-PluginInfo-Email=sycao@gmail.com
X-KDE-PluginInfo-Name=internalname
X-KDE-PluginInfo-Version=1.0
X-KDE-PluginInfo-Website=http://www.plugin.org/
X-KDE-PluginInfo-Category=demo
X-KDE-PluginInfo-License=GPL
这里定义了该插件的类型、库等信息。
接下来是Shell部分,我定义了一个MainShell类,从这里载入多个part,并且由KParts::PartManager来帮助管理多个part。其构造函数如下:
MainShell::MainShell()
    : KParts::MainWindow( )
{
    setXMLFile( "tutor1/main_shell.rc" );
    KAction *actFileOpen = new KAction( QString("file open"), this );
    actionCollection()->addAction( "file_open", actFileOpen );
    connect( actFileOpen, SIGNAL(triggered()),
             this, SLOT(slotFileOpen()) );
    KAction *actFileQuit = new KAction( QString("file &quit"), this );
    connect( actFileQuit, SIGNAL(triggered()),
             this, SLOT(slotFileQuit()) );
    actionCollection()->addAction( "file_quit", actFileQuit );
   
    KAction *actSwitch = new KAction( QString("switch part"), this );
    connect( actSwitch, SIGNAL(triggered()),
             this, SLOT(slotSwitch()) );
    actionCollection()->addAction( "part_switch", actSwitch );
    m_sa = new QScrollArea( this );
    KPluginFactory *factory = KLibLoader::self()->factory("libnotepad_part");   
    if ( factory ) {
        m_part_editor =
            factory->create(this, this);
        if ( !m_part_editor )
            kapp->quit();
        m_part_editor->widget()->hide();
    } else
        kDebug() lastErrorMessage();
    factory = KPluginLoader("libkonsolepart").factory();
    if ( factory ) {
        m_part_terminal =
            factory->create(this);
        if ( !m_part_terminal )
            kapp->quit();
        TerminalInterface *ti =
            qobject_cast( m_part_terminal );
        ti->startProgram( "bash", QStringList() );
        ti->sendInput( "ls\n" );
        m_part_terminal->widget()->hide();
    } else
        kDebug() addPart( m_part_editor, true );
    m_pm->addPart( m_part_terminal, false );
}
在kde4中,你可以用KLibLoader、KPluginLoader来载入Part和插件。KPluginLoader是推荐的方法。当你编译的时候,你会发现系统有warning,提示你参数中不要出现前缀lib。比如
KPluginFactory *factory = KLibLoader::self()->factory("libnotepad_part");
会被要求写成
KPluginFactory *factory = KLibLoader::self()->factory("notepad_part");
但是如果你不加lib前缀,你会发现系统没法找到你所需要的组件,除非使用全路径格式。可能是由于目前的kde4代码没有完全迁移到新的接口的原因。
其他代码就不具体说明了,有兴趣的可以试试,大家可以研究一下。
你要编译代码,必须有可用的QT4和kde4环境,以及cmake。


       
        文件:kparts.demo.tar
        大小:50KB
        下载:
下载
       


本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u1/58649/showart_465085.html
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP