要在Qt中调用谷歌拼音输入插件
编译谷歌拼音输入法源码:
可以通过编译谷歌拼音输入法的源码来实现在Qt中的应用。以下是一些步骤:
下载QtInputMethod_GooglePinyin源码,例如从Gitee获取。
使用MinGW64或MSVC2019_64构建套件编译源码。
编译完成后,将生成的tgtsmlInputContextPlugin.dll复制到Qt安装目录下的plugins/platforminputcontexts目录中。
在应用程序中设置环境变量QT_IM_MODULE为tgtsml,这样当光标聚焦在输入文本框时,会自动弹出虚拟键盘。
创建自定义输入法插件:
可以将谷歌拼音输入法编译成静态库或动态库,并在Qt中创建一个自定义的输入法插件。以下是一些关键步骤:
将谷歌拼音输入法的源码剥离出来,C++实现,并编译成静态库或动态库。
在Qt项目中添加静态库或动态库的链接,并包含相关的头文件和源文件。
实现一个自定义的输入法类,封装谷歌拼音输入法的调用接口,例如pinyin_im::init和pinyin_im::search函数。
使用QtInputMethod_GooglePinyin项目:
该项目提供了一个完整的解决方案,包括编译好的输入法插件和示例程序。以下是一些关键步骤:
下载QtInputMethod_GooglePinyin源码。
使用MinGW64或MSVC2019_64构建套件编译源码。
将编译生成的tgtsmlInputContextPlugin.dll复制到Qt安装目录下的plugins/platforminputcontexts目录中。
在应用程序中设置环境变量QT_IM_MODULE为tgtsml,这样当光标聚焦在输入文本框时,会自动弹出虚拟键盘。
#ifndef KEYBOARDFORM_H
#define KEYBOARDFORM_H
#include <QWidget>
class QPushButton;
class QLabel;
class KeyboardForm : public QWidget
{
Q_OBJECT
public:
KeyboardForm(QWidget *parent = 0);
void clearChineseCache();
private:
void chineseCharactersUpdatePrevious();
void chineseCharactersUpdateNext();
void chineseCharactersSelected();
void btnBackspaceClicked();
void btnEnterClicked();
void btnUpperClicked();
void btnSymbolsClicked();
void btnLanguageClicked();
void btnBlankspaceClicked();
void btnEmojiClicked();
void characterButtonClicked();
void updateKeyboard();
void updateButtonStateOfChineseCharacters();
void searchChineseCharacters(const int ¤tpage);
void hideKeyboard();
QList<QPushButton*> character_btns_list, chinese_characters_list, change_chinese_characters_page_list;
enum InputMode{zh, en, symb};
InputMode current_mode, last_mode;
QWidget *widget_keyboard, *widget_pinyin;
bool upper_mode;
QLabel *m_label_pinyin;
int m_symbol_page;
signals:
void sendKeyToFocusItem(const QString &keytext);
};
#endif // KEYBOARDFORM_H
#include "keyboardform.h"
#include <QLabel>
#include <QPushButton>
#include <QVBoxLayout>
#include <QHBoxLayout>
#include <QFontDatabase>
#include <QFile>
#include <QApplication>
#include "pinyinime.h"
using namespace ime_pinyin;
#define chinesecharacters_number 7
const char *keyboard_characters = "qwertyuiopasdfghjklzxcvbnm,.?";
const QString keyboard_symbols[] = {"1", "2", "3", "4", "5", "6", "7", "8", "9", "0",
"@", "#", "_", "\"", "“", "”", ",", ",", ".", "。",
";", ";", ":", ":", "'", "’", "、", "!", "!",
"~", "~", "+", "-", "*", "/", "=", "÷", "×", "√",
"`", "?", "^", "&&", "%", "|", "(", ")", "(", ")",
"[", "]", "【", "】", "{", "}", "<", ">", "《",
"》", "$", "€", "£", "¢", "¥", "§", "—", "/", "\",
"·", "……", "——", "→", "←", "↑", "↓", "■", "□", "●",
"○", "『", "』", "「", "」", "★", "☆", "◆", "◇"}; //29*3
KeyboardForm::KeyboardForm(QWidget *parent)
: QWidget(parent)
{
character_btns_list.clear();
current_mode = InputMode::en;
upper_mode = false;
m_symbol_page = 0;
this->setFixedSize(800,250);
int keyboard_btn_width = this->width()/11.5;
int keyboard_btn_height = this->height()/5.0;
//设置主窗体样式
this->setAttribute(Qt::WA_TranslucentBackground);
this->setWindowFlags(Qt::Tool | \
Qt::FramelessWindowHint | \
Qt::WindowStaysOnTopHint | \
Qt::WindowDoesNotAcceptFocus);
//加载QSS样式表
QFile qss(":/res/stylesheet.qss");
if(false == qss.open(QFile::ReadOnly))return;
this->setStyleSheet(qss.readAll());
qss.close();
//图标字体
int fontId = QFontDatabase::addApplicationFont(":/res/FontAwesome.otf");
QString fontName = QFontDatabase::applicationFontFamilies(fontId).at(0);
QFont btnicofont(fontName);
btnicofont.setPixelSize(10);
//单行布局
QHBoxLayout *hb[6];
for(int i=0; i<6; i++)
{
hb[i] = new QHBoxLayout();
hb[i]->setMargin(0);
i == 1 ? hb[i]->setSpacing(2) : hb[i]->setSpacing(0);
}
widget_pinyin = new QWidget(this);
widget_pinyin->setFixedHeight(keyboard_btn_height);
//拼音缓存
m_label_pinyin = new QLabel(this);
m_label_pinyin->setFixedHeight(keyboard_btn_height*0.4);
hb[0]->addWidget(m_label_pinyin);
hb[0]->addStretch(1);
//汉子缓存
for(int i=0; i<chinesecharacters_number; i++)
{
QPushButton *btn = new QPushButton(this);
btn->setFixedHeight(keyboard_btn_height*0.6);
hb[1]->addWidget(btn);
if(i != chinesecharacters_number - 1) hb[1]->addStretch(1);
if (i == 0 || i == chinesecharacters_number-1)
{
change_chinese_characters_page_list.append(btn);
btn->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed);
btn->setFont(btnicofont);
btn->setText(i == 0 ? QString(QChar(0xf0d9)) : QString(QChar(0xf0da)));
btn->setObjectName("hanzichangepage");
i == 0 ? \
connect(btn, &QPushButton::clicked, this, &KeyboardForm::chineseCharactersUpdatePrevious) :
connect(btn, &QPushButton::clicked, this, &KeyboardForm::chineseCharactersUpdateNext);
}
else
{
chinese_characters_list.append(btn);
btn->setObjectName("hanzicandidates");
connect(btn, &QPushButton::clicked, this, &KeyboardForm::chineseCharactersSelected);
}
}
QVBoxLayout *vb_pinyin = new QVBoxLayout(widget_pinyin);
vb_pinyin->addLayout(hb[0]);
vb_pinyin->addLayout(hb[1]);
vb_pinyin->setMargin(0);
vb_pinyin->setSpacing(0);
widget_keyboard = new QWidget(this);
widget_keyboard->setFixedHeight(keyboard_btn_height*4.0);
//键盘
for(int i=0; i<29; i++)
{
QPushButton *btn = new QPushButton(QChar(keyboard_characters[i]),this);
btn->setFixedSize(keyboard_btn_width, keyboard_btn_height);
character_btns_list.append(btn);
connect(btn, &QPushButton::clicked, this, &KeyboardForm::characterButtonClicked);
}
//第一排字母:0-9
for(int i=0; i<10; i++)
{
hb[2]->addWidget(character_btns_list.at(i));
}
QPushButton *btn_backspace = new QPushButton(QChar(0xf060));
btn_backspace->setFont(btnicofont);
btn_backspace->setFixedSize(keyboard_btn_width*1.5, keyboard_btn_height);
btn_backspace->setObjectName("function_button");
hb[2]->addWidget(btn_backspace);
connect(btn_backspace, &QPushButton::clicked, this, &KeyboardForm::btnBackspaceClicked);
//第二排字母:10-18
hb[3]->addStretch(1);
for(int i=10; i<19; i++)
{
hb[3]->addWidget(character_btns_list.at(i));
}
QPushButton *btn_enter = new QPushButton("Enter");
btn_enter->setFixedSize(keyboard_btn_width*1.5, keyboard_btn_height);
btn_enter->setObjectName("function_button");
hb[3]->addWidget(btn_enter);
hb[3]->addStretch(1);
connect(btn_enter, &QPushButton::clicked, this, &KeyboardForm::btnEnterClicked);
//第三排字母:20-26
QPushButton *btn_upper = new QPushButton(QChar(0xf062));
btn_upper->setFixedSize(keyboard_btn_width*1.5, keyboard_btn_height);
btn_upper->setFont(btnicofont);
btn_upper->setObjectName("function_button");
hb[4]->addWidget(btn_upper);
connect(btn_upper, &QPushButton::clicked, this, &KeyboardForm::btnUpperClicked);
for(int i=19; i<29; i++)
{
hb[4]->addWidget(character_btns_list.at(i));
}
character_btns_list.append(btn_upper);
//第四排功能键
QPushButton *btn_symbols = new QPushButton(".?123");
btn_symbols->setFixedSize(keyboard_btn_width*1.5, keyboard_btn_height);
btn_symbols->setObjectName("function_button");
hb[5]->addWidget(btn_symbols);
connect(btn_symbols, &QPushButton::clicked, this, &KeyboardForm::btnSymbolsClicked);
QPushButton *btn_language = new QPushButton(QChar(0xf0ac));
btn_language->setFixedSize(keyboard_btn_width, keyboard_btn_height);
btn_language->setFont(btnicofont);
btn_language->setObjectName("function_button");
hb[5]->addWidget(btn_language);
connect(btn_language, &QPushButton::clicked, this, &KeyboardForm::btnLanguageClicked);
QPushButton *btn_blankspace = new QPushButton("English");
btn_blankspace->setFixedHeight(keyboard_btn_height);
hb[5]->addWidget(btn_blankspace);
character_btns_list.append(btn_blankspace);
connect(btn_blankspace, &QPushButton::clicked, this, &KeyboardForm::btnBlankspaceClicked);
QPushButton *btn_emoji = new QPushButton(QChar(0xf118));
btn_emoji->setFixedSize(keyboard_btn_width, keyboard_btn_height);
btn_emoji->setFont(btnicofont);
btn_emoji->setObjectName("emoji");
hb[5]->addWidget(btn_emoji);
connect(btn_emoji, &QPushButton::clicked, this, &KeyboardForm::btnEmojiClicked);
QPushButton *btn_hidekeyboard = new QPushButton(QString(QChar(0xf11c)).append(QChar(0xf103)));
btn_hidekeyboard->setFixedSize(keyboard_btn_width*1.5, keyboard_btn_height);
btn_hidekeyboard->setFont(btnicofont);
btn_hidekeyboard->setObjectName("function_button");
hb[5]->addWidget(btn_hidekeyboard);
connect(btn_hidekeyboard, &QPushButton::clicked, this, &KeyboardForm::hideKeyboard);
QVBoxLayout *vb_keyboard = new QVBoxLayout(widget_keyboard);
vb_keyboard->setMargin(0);
vb_keyboard->setSpacing(0);
for(int i=2; i<6; i++)
{
vb_keyboard->addLayout(hb[i]);
}
QVBoxLayout *vb_system = new QVBoxLayout(this);
vb_system->setMargin(0);
vb_system->setSpacing(0);
vb_system->addStretch(1);
vb_system->addWidget(widget_pinyin);
vb_system->addWidget(widget_keyboard);
widget_pinyin->hide();
updateButtonStateOfChineseCharacters();
}
void KeyboardForm::updateButtonStateOfChineseCharacters()
{
if(m_label_pinyin->text().isEmpty())
{
m_label_pinyin->setHidden(true);
change_chinese_characters_page_list.at(0)->setHidden(true);
change_chinese_characters_page_list.at(1)->setHidden(true);
}
else
{
m_label_pinyin->setHidden(false);
change_chinese_characters_page_list.at(0)->setHidden(false);
change_chinese_characters_page_list.at(1)->setHidden(false);
}
}
void KeyboardForm::chineseCharactersUpdatePrevious()
{
searchChineseCharacters(-1);
}
void KeyboardForm::chineseCharactersUpdateNext()
{
searchChineseCharacters(1);
}
void KeyboardForm::chineseCharactersSelected()
{
emit sendKeyToFocusItem(((QPushButton*)sender())->text());
clearChineseCache();
}
void KeyboardForm::btnBackspaceClicked()
{
if(current_mode != InputMode::zh || m_label_pinyin->text().isEmpty())
{
emit sendKeyToFocusItem("\x7F");
}
else
{
m_label_pinyin->setText(m_label_pinyin->text().left(m_label_pinyin->text().length()-1));
if(m_label_pinyin->text().isEmpty())
{
clearChineseCache();
}
else
{
searchChineseCharacters(0);
}
}
}
void KeyboardForm::btnEnterClicked()
{
if(current_mode != InputMode::zh || m_label_pinyin->text().isEmpty())
{
emit sendKeyToFocusItem("\n");
}
else
{
emit sendKeyToFocusItem(m_label_pinyin->text());
clearChineseCache();
}
}
void KeyboardForm::btnUpperClicked()
{
if(current_mode == InputMode::en)
{
upper_mode = !upper_mode;
}
else if(current_mode == InputMode::zh)
{
if(!m_label_pinyin->text().isEmpty() && m_label_pinyin->text().right(1).compare("'"))
{
m_label_pinyin->setText(m_label_pinyin->text().append("'"));
}
}
else
{
if(m_symbol_page == 0)
{
m_symbol_page = 1;
character_btns_list.at(character_btns_list.length()-2)->setText("2/3");
}
else if(m_symbol_page == 1)
{
m_symbol_page = 2;
character_btns_list.at(character_btns_list.length()-2)->setText("3/3");
}
else
{
m_symbol_page = 0;
character_btns_list.at(character_btns_list.length()-2)->setText("1/3");
}
}
updateKeyboard();
}
void KeyboardForm::btnSymbolsClicked()
{
if(current_mode != InputMode::symb)
{
widget_pinyin->setHidden(true);
if(current_mode == InputMode::en)
{
character_btns_list.at(character_btns_list.length()-1)->setText("Symbols");
}
else if(current_mode == InputMode::zh)
{
character_btns_list.at(character_btns_list.length()-1)->setText("符号");
}
((QPushButton*)sender())->setText("abc");
last_mode = current_mode;
current_mode = InputMode::symb;
character_btns_list.at(character_btns_list.length()-2)->setText("1/3");
}
else
{
((QPushButton*)sender())->setText(".?123");
current_mode = last_mode;
m_symbol_page = 0;
}
upper_mode = false;
updateKeyboard();
}
void KeyboardForm::btnLanguageClicked()
{
upper_mode = false;
if(current_mode == InputMode::zh)
{
current_mode = InputMode::en;
}
else if(current_mode == InputMode::en)
{
current_mode = InputMode::zh;
}
if(current_mode != InputMode::symb)
{
last_mode = current_mode;
updateKeyboard();
}
}
void KeyboardForm::clearChineseCache()
{
m_label_pinyin->setText("");
for(int i=0; i<chinese_characters_list.length(); i++)
{
chinese_characters_list.at(i)->setText("");
}
updateButtonStateOfChineseCharacters();
}
void KeyboardForm::hideKeyboard()
{
clearChineseCache();
this->hide();
}
void KeyboardForm::updateKeyboard()
{
if(current_mode != InputMode::zh)
{
clearChineseCache();
}
if(current_mode == InputMode::symb)
{
character_btns_list.at(character_btns_list.length()-2)->setCheckable(false);
for(int i=0; i<29; i++)
{
character_btns_list.at(i)->setText(keyboard_symbols[i + m_symbol_page*29]);
}
}
else
{
if(true == upper_mode && current_mode == InputMode::en)
{
character_btns_list.at(character_btns_list.length()-2)->setCheckable(true);
character_btns_list.at(character_btns_list.length()-2)->setChecked(true);
for(int i=0; i<26; i++)
{
character_btns_list.at(i)->setText(QChar(keyboard_characters[i]).toUpper());
}
}
else
{
for(int i=0; i<26; i++)
{
character_btns_list.at(i)->setText(QChar(keyboard_characters[i]));
}
}
if(current_mode == InputMode::en)
{
widget_pinyin->setHidden(true);
character_btns_list.at(character_btns_list.length()-5)->setText(",");
character_btns_list.at(character_btns_list.length()-4)->setText(".");
character_btns_list.at(character_btns_list.length()-3)->setText("?");
character_btns_list.at(character_btns_list.length()-2)->setText(QChar(0xf062));
character_btns_list.at(character_btns_list.length()-1)->setText("English");
}
else if(current_mode == InputMode::zh)
{
character_btns_list.at(character_btns_list.length()-2)->setCheckable(false);
widget_pinyin->setHidden(false);
character_btns_list.at(character_btns_list.length()-5)->setText(",");
character_btns_list.at(character_btns_list.length()-4)->setText("。");
character_btns_list.at(character_btns_list.length()-3)->setText("?");
character_btns_list.at(character_btns_list.length()-2)->setText("分词");
character_btns_list.at(character_btns_list.length()-1)->setText("拼音");
}
}
}
void KeyboardForm::btnBlankspaceClicked()
{
if(current_mode != InputMode::zh || m_label_pinyin->text().isEmpty())
{
emit sendKeyToFocusItem(" ");
}
else
{
emit sendKeyToFocusItem(chinese_characters_list.at(0)->text());
clearChineseCache();
}
}
void KeyboardForm::btnEmojiClicked()
{
emit sendKeyToFocusItem("::)");
}
void KeyboardForm::characterButtonClicked()
{
if(current_mode == InputMode::zh)
{
if(((QPushButton*)sender())->text() == "," || ((QPushButton*)sender())->text() == "。" || ((QPushButton*)sender())->text() == "?")
{
emit sendKeyToFocusItem(((QPushButton*)sender())->text());
}
else
{
if(m_label_pinyin->text().length()<15)
{
m_label_pinyin->setText(m_label_pinyin->text().append(((QPushButton*)sender())->text()));
searchChineseCharacters(0);
updateButtonStateOfChineseCharacters();
}
}
}
else
{
emit sendKeyToFocusItem(((QPushButton*)sender())->text());
}
}
void KeyboardForm::searchChineseCharacters(const int ¤tpage)
{
const int max_spelling_length = 32;
const int max_decoded_length = 32;
const int max_single_hanzi = 20;
static unsigned int page_change_times = 0;
QString app_dir(qApp->applicationDirPath()+"/dict");
im_open_decoder(QString("%1/dict_pinyin.dat").arg(app_dir).toLocal8Bit().data(),
QString("%1/dict_pinyin_user.dat").arg(app_dir).toLocal8Bit().data());
im_set_max_lens(max_spelling_length, max_decoded_length);
im_reset_search();
QByteArray bytearray(m_label_pinyin->text().toUtf8());
char *pinyin(bytearray.data());
size_t cand_num = im_search(pinyin, bytearray.size());
size_t decode_len;
im_get_sps_str(&decode_len);
if (decode_len == 1)
{
if (cand_num > 10) cand_num = 10;
}
else
{
size_t single = 0;
size_t multi = 0;
char16 *cand_buf = new char16[max_decoded_length];
for(size_t i = 0; i < cand_num; i++)
{
im_get_candidate(i, cand_buf, max_decoded_length);
if (strlen((char *)cand_buf) > 2)
{
multi++;
}
else
{
single++;
if (single > max_single_hanzi) break;
}
}
cand_num = multi + single;
delete cand_buf;
}
switch(currentpage)
{
case 1:
if(cand_num > chinese_characters_list.length() && page_change_times < cand_num - chinese_characters_list.length())
page_change_times++;
break;
case -1:
if(page_change_times > 0) page_change_times--;
break;
default:
page_change_times = 0;
break;
}
if(0 == page_change_times)
change_chinese_characters_page_list.at(0)->setEnabled(false);
else
change_chinese_characters_page_list.at(0)->setEnabled(true);
if(page_change_times == cand_num - chinese_characters_list.length())
change_chinese_characters_page_list.at(1)->setEnabled(false);
else
change_chinese_characters_page_list.at(1)->setEnabled(true);
char16 *cand_buf = new char16[max_decoded_length];
char16 *cand;
QString cand_str;
for (unsigned i = 0; i < cand_num; i++)
{
cand = im_get_candidate(i, cand_buf, max_decoded_length);
if (cand)
{
cand_str = QString::fromUtf16(cand);
if (i == 0) cand_str.remove(0, im_get_fixed_len());
}
else
{
cand_str = "";
}
int tmpindex = i - page_change_times;
if(tmpindex >= 0 && tmpindex < chinese_characters_list.length())
{
switch(currentpage)
{
case 1:
chinese_characters_list.at(tmpindex)->setText(cand_str);
break;
case -1:
chinese_characters_list.at(tmpindex)->setText(cand_str);
break;
default:
chinese_characters_list.at(tmpindex)->setText(cand_str);
break;
}
}
}
delete cand_buf;
}