1999-10-02 16:21:10 +00:00
|
|
|
#include <config.h>
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
#ifdef __GNUG__
|
|
|
|
#pragma implementation "trans_mgr.h"
|
|
|
|
#endif
|
|
|
|
|
1999-10-07 18:44:17 +00:00
|
|
|
//#include <cassert>
|
1999-09-27 18:44:28 +00:00
|
|
|
#include "trans_mgr.h"
|
|
|
|
#include "trans.h"
|
|
|
|
#include "lyxtext.h"
|
|
|
|
#include "LString.h"
|
1999-10-07 18:44:17 +00:00
|
|
|
#include "debug.h"
|
1999-09-27 18:44:28 +00:00
|
|
|
#include "chset.h"
|
|
|
|
#include "insets/insetlatexaccent.h"
|
|
|
|
#include "BufferView.h"
|
|
|
|
#include "buffer.h"
|
|
|
|
#include "lyxrc.h"
|
1999-10-02 16:21:10 +00:00
|
|
|
#include "support/lstrings.h"
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
extern LyXRC* lyxrc;
|
1999-10-02 16:21:10 +00:00
|
|
|
extern string DoAccent(const string&,tex_accent);
|
1999-09-27 18:44:28 +00:00
|
|
|
extern void InsertCorrectQuote();
|
1999-10-02 16:21:10 +00:00
|
|
|
extern string DoAccent(char,tex_accent);
|
1999-09-27 18:44:28 +00:00
|
|
|
extern BufferView* current_view;
|
|
|
|
|
|
|
|
|
|
|
|
// TransFSMData
|
|
|
|
TransFSMData::TransFSMData()
|
|
|
|
{
|
|
|
|
deadkey_=deadkey2_=0;
|
|
|
|
deadkey_info_.accent=deadkey2_info_.accent=TEX_NOACCENT;
|
1999-10-02 16:21:10 +00:00
|
|
|
comb_info_=0;
|
1999-09-27 18:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// TransState
|
|
|
|
const char TransState::TOKEN_SEP=4;
|
|
|
|
|
|
|
|
|
|
|
|
TransState::~TransState()
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
|
|
// TransInitState
|
|
|
|
TransInitState::TransInitState()
|
|
|
|
{
|
|
|
|
init_state_=this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
string TransInitState::normalkey(char c,char *t)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
1999-10-02 16:21:10 +00:00
|
|
|
string res;
|
|
|
|
if (t!=0)
|
1999-09-27 18:44:28 +00:00
|
|
|
res=t;
|
|
|
|
else
|
|
|
|
res=c;
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//bool TransInitState::backspace()
|
|
|
|
//{
|
|
|
|
// return true;
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
string TransInitState::deadkey(char c,KmodInfo d)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
|
|
|
deadkey_=c;
|
|
|
|
deadkey_info_=d;
|
|
|
|
currentState=deadkey_state_;
|
1999-10-02 16:21:10 +00:00
|
|
|
return string();
|
1999-09-27 18:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// TransDeadkeyState
|
|
|
|
TransDeadkeyState::TransDeadkeyState()
|
|
|
|
{
|
|
|
|
deadkey_state_=this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
string TransDeadkeyState::normalkey(char c,char *trans)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
1999-10-02 16:21:10 +00:00
|
|
|
string res;
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
// Check if it is an exception
|
|
|
|
KmodException l=deadkey_info_.exception_list;
|
1999-10-02 16:21:10 +00:00
|
|
|
while(l!=0) {
|
1999-09-27 18:44:28 +00:00
|
|
|
if (l->c==c) {
|
|
|
|
res=l->data;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
l=l->next;
|
|
|
|
}
|
1999-10-02 16:21:10 +00:00
|
|
|
if (l==0) {
|
1999-09-27 18:44:28 +00:00
|
|
|
// Not an exception. Check if it allowed
|
|
|
|
if (current_view->currentBuffer()->params.allowAccents==true ||
|
1999-10-02 16:21:10 +00:00
|
|
|
countChar(deadkey_info_.allowed, c) > 0) {
|
1999-09-27 18:44:28 +00:00
|
|
|
res=DoAccent(c,deadkey_info_.accent);
|
|
|
|
} else {
|
|
|
|
// Not allowed
|
|
|
|
if (deadkey_!=0)
|
|
|
|
res=deadkey_;
|
|
|
|
res+=TOKEN_SEP;
|
|
|
|
res+=trans;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
currentState=init_state_;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//bool TransDeadkeyState::backspace()
|
|
|
|
//{
|
|
|
|
// currentState=init_state_;
|
|
|
|
// return false;
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
string TransDeadkeyState::deadkey(char c,KmodInfo d)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
1999-10-02 16:21:10 +00:00
|
|
|
string res;
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
// Check if the same deadkey was typed twice
|
|
|
|
if (deadkey_==c) {
|
|
|
|
res=deadkey_;
|
|
|
|
deadkey_=0;
|
|
|
|
deadkey_info_.accent=TEX_NOACCENT;
|
|
|
|
currentState=init_state_;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if it is a combination or an exception
|
|
|
|
KmodException l;
|
|
|
|
l=deadkey_info_.exception_list;
|
|
|
|
|
|
|
|
while(l) {
|
|
|
|
if (l->combined==true && l->accent==d.accent) {
|
|
|
|
deadkey2_=c;
|
|
|
|
deadkey2_info_=d;
|
|
|
|
comb_info_=l;
|
|
|
|
currentState=combined_state_;
|
1999-10-02 16:21:10 +00:00
|
|
|
return string();
|
1999-09-27 18:44:28 +00:00
|
|
|
}
|
|
|
|
if (l->c==c) {
|
|
|
|
res=l->data;
|
|
|
|
deadkey_=0;
|
|
|
|
deadkey_info_.accent=TEX_NOACCENT;
|
|
|
|
currentState=init_state_;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
l=l->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Not a combination or an exception.
|
|
|
|
// Output deadkey1 and keep deadkey2
|
|
|
|
|
|
|
|
if (deadkey_!=0)
|
|
|
|
res=deadkey_;
|
|
|
|
deadkey_=c;
|
|
|
|
deadkey_info_=d;
|
|
|
|
currentState=deadkey_state_;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TransCombinedState::TransCombinedState()
|
|
|
|
{
|
|
|
|
combined_state_=this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
string TransCombinedState::normalkey(char c,char *trans)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
1999-10-02 16:21:10 +00:00
|
|
|
string res;
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
// Check if the key is allowed on the combination
|
1999-10-02 16:21:10 +00:00
|
|
|
if (countChar(comb_info_->data, c) > 0) {
|
|
|
|
string temp;
|
1999-09-27 18:44:28 +00:00
|
|
|
temp=DoAccent(c,deadkey2_info_.accent);
|
|
|
|
res=DoAccent(temp,deadkey_info_.accent);
|
|
|
|
currentState=init_state_;
|
|
|
|
} else {
|
|
|
|
// Not allowed. Output deadkey1 and check deadkey2 + c
|
|
|
|
if (deadkey_!=0)
|
|
|
|
res+=deadkey_;
|
|
|
|
res+=TOKEN_SEP;
|
|
|
|
deadkey_=deadkey2_;
|
|
|
|
deadkey_info_=deadkey2_info_;
|
|
|
|
// Call deadkey state and leave it to setup the FSM
|
|
|
|
res+=deadkey_state_->normalkey(c,trans);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// bool TransCombinedState::backspace()
|
|
|
|
// {
|
|
|
|
// // cancel the second deadkey
|
|
|
|
// deadkey2_=0;
|
|
|
|
// deadkey2_info_.accent=TEX_NOACCENT;
|
|
|
|
// currentState=deadkey_state_;
|
|
|
|
|
|
|
|
// return false;
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
string TransCombinedState::deadkey(char c,KmodInfo d)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
|
|
|
// Third key in a row. Output the first one and
|
|
|
|
// reenter with shifted deadkeys
|
1999-10-02 16:21:10 +00:00
|
|
|
string res;
|
1999-09-27 18:44:28 +00:00
|
|
|
if (deadkey_!=0)
|
|
|
|
res=deadkey_;
|
|
|
|
res+=TOKEN_SEP;
|
|
|
|
deadkey_=deadkey2_;
|
|
|
|
deadkey_info_=deadkey2_info_;
|
|
|
|
res+=deadkey_state_->deadkey(c,d);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// TransFSM
|
|
|
|
TransFSM::TransFSM():
|
|
|
|
TransFSMData(),
|
|
|
|
TransInitState(),
|
|
|
|
TransDeadkeyState(),
|
|
|
|
TransCombinedState()
|
|
|
|
{
|
|
|
|
currentState=init_state_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// TransManager
|
|
|
|
|
|
|
|
TransManager::TransManager()
|
1999-10-02 16:21:10 +00:00
|
|
|
: active_(0),t1_(new Trans),t2_(new Trans),chset_(new CharacterSet)
|
1999-09-27 18:44:28 +00:00
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
|
|
Trans* TransManager::default_=new Trans;
|
|
|
|
|
|
|
|
|
|
|
|
TransManager::~TransManager()
|
|
|
|
{
|
|
|
|
delete t1_;
|
|
|
|
delete t2_;
|
|
|
|
delete chset_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
int TransManager::SetPrimary(string const & language)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
|
|
|
if (t1_->GetName()==language)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return t1_->Load(language);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
int TransManager::SetSecondary(string const & language)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
|
|
|
if (t2_->GetName()==language)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return t2_->Load(language);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool TransManager::setCharset(const char *set)
|
|
|
|
{
|
|
|
|
return chset_->loadFile(set);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TransManager::EnablePrimary()
|
|
|
|
{
|
|
|
|
if (t1_->IsDefined())
|
|
|
|
active_=t1_;
|
|
|
|
|
1999-10-07 18:44:17 +00:00
|
|
|
lyxerr[Debug::KBMAP] << "Enabling primary keymap" << endl;
|
1999-09-27 18:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TransManager::EnableSecondary()
|
|
|
|
{
|
|
|
|
if (t2_->IsDefined( ))
|
|
|
|
active_=t2_;
|
1999-10-07 18:44:17 +00:00
|
|
|
lyxerr[Debug::KBMAP] << "Enabling secondary keymap" << endl;
|
1999-09-27 18:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TransManager::DisableKeymap()
|
|
|
|
{
|
|
|
|
active_=default_;
|
1999-10-07 18:44:17 +00:00
|
|
|
lyxerr[Debug::KBMAP] << "Disabling keymap" << endl;
|
1999-09-27 18:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TransManager::TranslateAndInsert(char c,LyXText *text)
|
|
|
|
{
|
1999-10-02 16:21:10 +00:00
|
|
|
string res;
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
res=active_->process(c,*this);
|
|
|
|
|
|
|
|
// Process with tokens
|
1999-10-02 16:21:10 +00:00
|
|
|
string temp;
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
while(res.length()>0) {
|
1999-10-02 16:21:10 +00:00
|
|
|
res=split(res, temp,TransState::TOKEN_SEP);
|
1999-09-27 18:44:28 +00:00
|
|
|
insert(temp,text);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
void TransManager::insertVerbatim(const string& str,LyXText *text)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
|
|
|
int l=str.length();
|
|
|
|
|
|
|
|
for (int i=0;i<l;i++){
|
|
|
|
if (str[i]=='\"'
|
|
|
|
&& text->GetFont(text->cursor.par,
|
|
|
|
text->cursor.pos).latex() == LyXFont::OFF)
|
|
|
|
InsertCorrectQuote();
|
|
|
|
else
|
|
|
|
text->InsertChar(str[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
void TransManager::insert(string str,LyXText *text)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
1999-10-02 16:21:10 +00:00
|
|
|
string t=str;
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
// Go through the character encoding only if the current
|
|
|
|
// encoding (chset_->name()) matches the current font_norm
|
|
|
|
// (lyrxc->font_norm
|
|
|
|
|
|
|
|
if (chset_->getName()!=lyxrc->font_norm ||
|
|
|
|
chset_->encodeString(str)==false) {
|
|
|
|
// Could not find an encoding
|
|
|
|
InsetLatexAccent ins(str);
|
|
|
|
if (ins.CanDisplay()) {
|
|
|
|
text->InsertInset(new InsetLatexAccent(ins));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
insertVerbatim(str,text);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//bool TransManager::backspace()
|
|
|
|
//{
|
|
|
|
// return trans_fsm_.currentState->backspace();
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
|
|
void TransManager::deadkey(char c,tex_accent accent,LyXText *t)
|
|
|
|
{
|
|
|
|
KmodInfo i;
|
1999-10-02 16:21:10 +00:00
|
|
|
string res;
|
1999-09-27 18:44:28 +00:00
|
|
|
|
|
|
|
if (c==0 && active_!=default_) {
|
|
|
|
// A deadkey was pressed that cannot be printed
|
|
|
|
// or a accent command was typed in the minibuffer
|
|
|
|
|
|
|
|
if (active_->isAccentDefined(accent,i)==true) {
|
|
|
|
res=trans_fsm_.currentState->deadkey(c,i);
|
|
|
|
insert(res,t);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}// else {
|
|
|
|
// Fall through to the next case
|
|
|
|
//}
|
|
|
|
|
|
|
|
if (active_==default_ || c==0) {
|
|
|
|
i.accent=accent;
|
|
|
|
i.allowed=lyx_accent_table[accent].native;
|
1999-10-19 20:59:27 +00:00
|
|
|
i.data.clear();
|
1999-10-02 16:21:10 +00:00
|
|
|
i.exception_list=0;
|
1999-09-27 18:44:28 +00:00
|
|
|
|
1999-10-02 16:21:10 +00:00
|
|
|
string res=trans_fsm_.currentState->deadkey(c,i);
|
1999-09-27 18:44:28 +00:00
|
|
|
insert(res,t);
|
|
|
|
} else {
|
|
|
|
// Go through the translation
|
|
|
|
TranslateAndInsert(c,t);
|
|
|
|
}
|
|
|
|
}
|