mirror of
https://git.lyx.org/repos/lyx.git
synced 2024-11-07 02:28:35 +00:00
0b6dc43b18
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@1374 a592a061-630c-0410-9148-cb99ea01b6c8
403 lines
7.5 KiB
C
403 lines
7.5 KiB
C
#include <config.h>
|
|
|
|
#ifdef __GNUG__
|
|
#pragma implementation "trans.h"
|
|
#endif
|
|
|
|
#include "LyXView.h"
|
|
#include "trans.h"
|
|
#include "support/filetools.h"
|
|
#include "tex-strings.h"
|
|
#include "lyxlex.h"
|
|
#include "debug.h"
|
|
#include "trans_mgr.h"
|
|
|
|
using std::endl;
|
|
|
|
|
|
// KmodInfo
|
|
KmodInfo::KmodInfo()
|
|
{
|
|
exception_list = 0;
|
|
}
|
|
|
|
|
|
// Default Trans
|
|
bool DefaultTrans::init_ = false;
|
|
|
|
|
|
DefaultTrans::DefaultTrans()
|
|
{
|
|
if (init_ == false) {
|
|
// Do initialization
|
|
init_ = true;
|
|
}
|
|
}
|
|
|
|
|
|
string const DefaultTrans::process(char c, TransManager & k)
|
|
{
|
|
char dummy[2] = "?";
|
|
dummy[0] = c;
|
|
|
|
return k.normalkey(c, dummy);
|
|
}
|
|
|
|
|
|
// Trans class
|
|
|
|
Trans::Trans()
|
|
{
|
|
for (int i = 0; i < TEX_MAX_ACCENT + 1; ++i)
|
|
kmod_list_[i] = 0;
|
|
}
|
|
|
|
|
|
Trans::~Trans()
|
|
{
|
|
FreeKeymap();
|
|
}
|
|
|
|
|
|
void Trans::InsertException(Trans::keyexc & exclist, char c,
|
|
string const & data, bool flag, tex_accent accent)
|
|
{
|
|
keyexc p = new Keyexc;
|
|
p->next = exclist;
|
|
p->c = c;
|
|
|
|
p->data = data;
|
|
p->combined = flag;
|
|
p->accent = accent;
|
|
|
|
exclist = p;
|
|
}
|
|
|
|
|
|
void Trans::FreeException(Trans::keyexc & exclist)
|
|
{
|
|
Trans::keyexc p = exclist;
|
|
while (p) {
|
|
p = exclist->next;
|
|
delete exclist;
|
|
exclist = p;
|
|
}
|
|
}
|
|
|
|
|
|
void Trans::FreeKeymap()
|
|
{
|
|
for (int i = 0; i < 256; ++i)
|
|
if (!keymap_[i].empty()) {
|
|
keymap_[i].erase();
|
|
}
|
|
for (int i = 0; i < TEX_MAX_ACCENT + 1; ++i)
|
|
if (kmod_list_[i]) {
|
|
FreeException(kmod_list_[i]->exception_list);
|
|
delete kmod_list_[i];
|
|
kmod_list_[i] = 0;
|
|
}
|
|
}
|
|
|
|
|
|
bool Trans::IsDefined() const
|
|
{
|
|
return !name_.empty();
|
|
}
|
|
|
|
|
|
string const & Trans::GetName() const
|
|
{
|
|
return name_;
|
|
}
|
|
|
|
|
|
enum kmaptags_ {
|
|
KCOMB = 1,
|
|
KMOD,
|
|
KMAP,
|
|
KXMOD,
|
|
K_LAST
|
|
};
|
|
|
|
|
|
struct keyword_item kmapTags[K_LAST - 1] = {
|
|
{"\\kcomb", KCOMB },
|
|
{ "\\kmap", KMAP },
|
|
{ "\\kmod", KMOD },
|
|
{ "\\kxmod", KXMOD }
|
|
};
|
|
|
|
|
|
tex_accent getkeymod(string const &);
|
|
|
|
|
|
void Trans::AddDeadkey(tex_accent accent, string const & keys,
|
|
string const & allowed)
|
|
{
|
|
if (kmod_list_[accent]) {
|
|
FreeException(kmod_list_[accent]->exception_list);
|
|
|
|
delete kmod_list_[accent];
|
|
}
|
|
|
|
kmod_list_[accent] = new kmod_list_decl;
|
|
kmod_list_[accent]->data = keys;
|
|
kmod_list_[accent]->accent = accent;
|
|
if (allowed == "native") {
|
|
kmod_list_[accent]->allowed= lyx_accent_table[accent].native;
|
|
} else {
|
|
kmod_list_[accent]->allowed = allowed;
|
|
}
|
|
|
|
for (string::size_type i = 0; i < keys.length(); ++i) {
|
|
string & temp =
|
|
keymap_[static_cast<unsigned char>(keys[i])];
|
|
if (!temp.empty())
|
|
temp.erase();
|
|
|
|
// But the question remains: "Should we be allowed
|
|
// to change bindings, without unbinding first?"
|
|
// Lgb
|
|
temp += char(0);
|
|
temp += char(accent);
|
|
}
|
|
kmod_list_[accent]->exception_list = 0;
|
|
}
|
|
|
|
|
|
int Trans::Load(LyXLex & lex)
|
|
{
|
|
bool error = false;
|
|
|
|
while (lex.IsOK() && !error) {
|
|
switch (lex.lex()) {
|
|
case KMOD:
|
|
{
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "KMOD:\t" << lex.text() << endl;
|
|
if (lex.next(true)) {
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "key\t`" << lex.text()
|
|
<< "'" << endl;
|
|
} else
|
|
return -1;
|
|
|
|
string keys = lex.GetString();
|
|
|
|
if (lex.next(true)) {
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "accent\t`" << lex.text()
|
|
<< "'" << endl;
|
|
} else
|
|
return -1;
|
|
|
|
tex_accent accent = getkeymod(lex.GetString());
|
|
|
|
if (accent == TEX_NOACCENT)
|
|
return -1;
|
|
|
|
if (lex.next(true)) {
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "allowed\t`" << lex.text()
|
|
<< "'" << endl;
|
|
} else
|
|
return -1;
|
|
|
|
string allowed = lex.GetString();
|
|
|
|
AddDeadkey(accent, keys, allowed);
|
|
break;
|
|
}
|
|
case KCOMB: {
|
|
string str;
|
|
|
|
lyxerr[Debug::KBMAP] << "KCOMB:" << endl;
|
|
if (lex.next(true)) {
|
|
str= lex.text();
|
|
lyxerr[Debug::KBMAP] << str << endl;
|
|
} else
|
|
return -1;
|
|
|
|
tex_accent accent_1 = getkeymod(str);
|
|
if (accent_1 == TEX_NOACCENT) return -1;
|
|
|
|
if (lex.next(true)) {
|
|
str = lex.text();
|
|
lyxerr[Debug::KBMAP] << str << endl;
|
|
} else
|
|
return -1;
|
|
|
|
tex_accent accent_2= getkeymod(str);
|
|
if (accent_2 == TEX_NOACCENT) return -1;
|
|
|
|
if (kmod_list_[accent_1] == 0
|
|
|| kmod_list_[accent_2] == 0)
|
|
return -1;
|
|
|
|
// Find what key accent_2 is on - should
|
|
// check about accent_1 also
|
|
int key = 0;
|
|
for (; key < 256; ++key) {
|
|
if (!keymap_[key].empty()
|
|
&& keymap_[key][0] == 0
|
|
&& keymap_[key][1] == accent_2)
|
|
break;
|
|
}
|
|
string allowed;
|
|
|
|
if (lex.next()) {
|
|
allowed = lex.GetString();
|
|
lyxerr[Debug::KBMAP] << "allowed: "
|
|
<< allowed << endl;
|
|
} else
|
|
return -1;
|
|
|
|
InsertException(kmod_list_[accent_1]->exception_list,
|
|
static_cast<char>(key), allowed,
|
|
true, accent_2);
|
|
}
|
|
break;
|
|
case KMAP: {
|
|
unsigned char key_from;
|
|
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "KMAP:\t" << lex.text() << endl;
|
|
if (lex.next(true)) {
|
|
key_from = lex.text()[0];
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "\t`" << lex.text() << "'"
|
|
<< endl;
|
|
} else
|
|
return -1;
|
|
|
|
if (lex.next(true)) {
|
|
string string_to = lex.text();
|
|
//char * string_to =
|
|
// strcpy(new char[strlen(t)+1], t);
|
|
keymap_[key_from] = string_to;
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "\t`" << string_to << "'"
|
|
<< endl;
|
|
} else
|
|
return -1;
|
|
|
|
break;
|
|
}
|
|
case KXMOD: {
|
|
tex_accent accent;
|
|
char key;
|
|
string str;
|
|
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "KXMOD:\t" << lex.text() << endl;
|
|
if (lex.next(true)) {
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "\t`" << lex.text() << "'"
|
|
<< endl;
|
|
accent = getkeymod(lex.GetString());
|
|
} else
|
|
return -1;
|
|
|
|
if (lex.next(true)) {
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "\t`" << lex.text() << "'"
|
|
<< endl;
|
|
key = lex.text()[0];
|
|
} else
|
|
return -1;
|
|
|
|
if (lex.next(true)) {
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "\t`" << lex.text() << "'"
|
|
<< endl;
|
|
str = lex.text();
|
|
} else
|
|
return -1;
|
|
|
|
InsertException(kmod_list_[accent]->exception_list,
|
|
key, str);
|
|
break;
|
|
}
|
|
case LyXLex::LEX_FEOF:
|
|
lyxerr[Debug::PARSER] << "End of parsing" << endl;
|
|
break;
|
|
default:
|
|
lex.printError("ParseKeymapFile: "
|
|
"Unknown tag: `$$Token'");
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
bool Trans::isAccentDefined(tex_accent accent, KmodInfo & i) const
|
|
{
|
|
if (kmod_list_[accent] != 0) {
|
|
i = *kmod_list_[accent];
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
string const Trans::process(char c, TransManager & k)
|
|
{
|
|
string dummy("?");
|
|
string dt = dummy;
|
|
string const t = Match(static_cast<unsigned char>(c));
|
|
|
|
if (t.empty() && c != 0) {
|
|
dt[0] = c;
|
|
return k.normalkey(c, dt);
|
|
} else if (!t.empty() && t[0] != char(0)) {
|
|
dt = t;
|
|
return k.normalkey(c, dt);
|
|
} else {
|
|
return k.deadkey(c,
|
|
*kmod_list_[static_cast<tex_accent>(t[1])]);
|
|
}
|
|
}
|
|
|
|
|
|
int Trans::Load(string const & language)
|
|
{
|
|
string const filename = LibFileSearch("kbd", language, "kmap");
|
|
if (filename.empty())
|
|
return -1;
|
|
|
|
FreeKeymap();
|
|
LyXLex lex(kmapTags, K_LAST-1);
|
|
lex.setFile(filename);
|
|
|
|
int const res = Load(lex);
|
|
|
|
if (res == 0) {
|
|
name_ = language;
|
|
} else
|
|
name_.erase();
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
tex_accent getkeymod(string const & p)
|
|
/* return modifier - decoded from p and update p */
|
|
{
|
|
for (int i = 1; i <= TEX_MAX_ACCENT; ++i) {
|
|
if (lyxerr.debugging(Debug::KBMAP))
|
|
lyxerr << "p = " << p
|
|
<< ", lyx_accent_table[" << i
|
|
<< "].name = `" << lyx_accent_table[i].name
|
|
<< "'" << endl;
|
|
|
|
if (lyx_accent_table[i].name
|
|
&& contains(p, lyx_accent_table[i].name)) {
|
|
lyxerr[Debug::KBMAP] << "Found it!" << endl;
|
|
return static_cast<tex_accent>(i);
|
|
}
|
|
}
|
|
return TEX_NOACCENT;
|
|
}
|