summaryrefslogtreecommitdiffstats
path: root/kiten/rad.cpp
diff options
context:
space:
mode:
authortoma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da>2009-11-25 17:56:58 +0000
committertoma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da>2009-11-25 17:56:58 +0000
commitce599e4f9f94b4eb00c1b5edb85bce5431ab3df2 (patch)
treed3bb9f5d25a2dc09ca81adecf39621d871534297 /kiten/rad.cpp
downloadtdeedu-ce599e4f9f94b4eb00c1b5edb85bce5431ab3df2.tar.gz
tdeedu-ce599e4f9f94b4eb00c1b5edb85bce5431ab3df2.zip
Copy the KDE 3.5 branch to branches/trinity for new KDE 3.5 features.
BUG:215923 git-svn-id: svn://anonsvn.kde.org/home/kde/branches/trinity/kdeedu@1054174 283d02a7-25f6-0310-bc7c-ecb5cbfe19da
Diffstat (limited to 'kiten/rad.cpp')
-rw-r--r--kiten/rad.cpp435
1 files changed, 435 insertions, 0 deletions
diff --git a/kiten/rad.cpp b/kiten/rad.cpp
new file mode 100644
index 00000000..79986301
--- /dev/null
+++ b/kiten/rad.cpp
@@ -0,0 +1,435 @@
+/**
+ This file is part of Kiten, a KDE Japanese Reference Tool...
+ Copyright (C) 2001 Jason Katz-Brown <jason@katzbrown.com>
+ (C) 2005 Paul Temple <paul.temple@gmx.net>
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
+ USA
+**/
+
+#include <kapplication.h>
+#include <kconfig.h>
+#include <kdebug.h>
+#include <kdialog.h>
+#include <klistbox.h>
+#include <klocale.h>
+#include <kmessagebox.h>
+#include <kpushbutton.h>
+#include <kstandarddirs.h>
+
+#include <qbuttongroup.h>
+#include <qcheckbox.h>
+#include <qfile.h>
+#include <qlabel.h>
+#include <qlayout.h>
+#include <qspinbox.h>
+#include <qtextcodec.h>
+#include <qtooltip.h>
+
+#include "kitenconfig.h"
+#include "rad.h"
+
+Rad::Rad() : QObject()
+{
+ loaded = false;
+}
+
+void Rad::load()
+{
+ if (loaded)
+ return;
+
+ KStandardDirs *dirs = KGlobal::dirs();
+ QString radkfile = dirs->findResource("data", "kiten/radkfile");
+ if (radkfile.isNull())
+ {
+ KMessageBox::error(0, i18n("Kanji radical information file not installed, so radical searching cannot be used."));
+ return;
+ }
+
+ QFile f(radkfile);
+
+ if (!f.open(IO_ReadOnly))
+ {
+ KMessageBox::error(0, i18n("Kanji radical information could not be loaded, so radical searching cannot be used."));
+ }
+
+ QTextStream t(&f);
+ t.setCodec(QTextCodec::codecForName("eucJP"));
+ Radical cur;
+ while (!t.eof())
+ {
+ QString s = t.readLine();
+
+ QChar first = s.at(0);
+ if (first == '#') // comment!
+ {
+ // nothing
+ }
+ else if (first == '$') // header
+ {
+ // save previous one
+ if( !cur.kanji().isNull() )
+ list.append(cur);
+
+ //first entry is trim(last 4 chars).. <rad><space><strokes>
+ unsigned int strokes = s.right(2).toUInt();
+ QString radical = QString(s.at(2));
+ cur = Radical(radical, strokes);
+ }
+ else // continuation
+ {
+ cur.addKanji(s);
+ }
+ }
+
+ // we gotta append the last one!!
+ // this nagged jasonkb for a bit wondering why fue wasn't showing up ;)
+ list.append(cur);
+
+ f.close();
+
+ loaded = true;
+}
+
+QStringList Rad::radByStrokes(unsigned int strokes)
+{
+ load();
+
+ QStringList ret;
+ bool hadOne = false;
+ QValueListIterator<Radical> it = list.begin();
+
+ do
+ {
+ if ((*it).strokes() == strokes)
+ {
+ ret.append((*it).radical());
+ hadOne = true;
+ }
+ else if(hadOne) // shortcut because it's a sorted list
+ {
+ return ret;
+ }
+ }
+ while (++it != list.end());
+
+ return ret;
+}
+
+QStringList Rad::kanjiByRad(const QString &text)
+{
+ //kdDebug() << "kanjiByRad, text is " << text << endl;
+ load();
+ QStringList ret;
+
+ QValueListIterator<Radical> it;
+ for (it = list.begin(); it != list.end() && (*it).radical() != text; ++it)
+ {
+ //kdDebug() << "kanjiByRad, looping, radical is " << (*it).radical() << endl;
+ }
+
+ QString kanji = (*it).kanji();
+ for (unsigned i = 0; i < kanji.length(); ++i)
+ {
+ //kdDebug() << "kanjiByRad, i is " << i << endl;
+ ret.append(QString(kanji.at(i)));
+ }
+
+ return ret;
+}
+
+QStringList Rad::kanjiByRad(const QStringList &list)
+{
+ //kdDebug() << "kanjiByRad (list version)\n";
+
+ QStringList ret;
+ QValueList<QStringList> lists;
+
+ for (QStringList::ConstIterator it = list.begin(); it != list.end(); ++it)
+ {
+ //kdDebug() << "loading radical " << *it << endl;
+ lists.append(kanjiByRad(*it));
+ }
+
+ QStringList first = lists.first();
+ lists.pop_front();
+
+ for (QStringList::Iterator kit = first.begin(); kit != first.end(); ++kit)
+ {
+ //kdDebug() << "kit is " << *kit << endl;
+ QValueList<bool> outcomes;
+ for (QValueList<QStringList>::Iterator it = lists.begin(); it != lists.end(); ++it)
+ {
+ //kdDebug() << "looping through lists\n";
+ outcomes.append((*it).contains(*kit) > 0);
+ }
+
+ const bool containsBool = false;
+ if ((outcomes.contains(containsBool) < 1))
+ {
+ //kdDebug() << "appending " << *kit << endl;
+ ret.append(*kit);
+ }
+ else
+ {
+ //kdDebug() << "not appending " << *kit << endl;
+ }
+ }
+
+ return ret;
+}
+
+Radical Rad::radByKanji(const QString &text)
+{
+ load();
+ QString ret;
+
+ QValueListIterator<Radical> it;
+ for (it = list.end(); it != list.begin() && (*it).kanji().find(text) == -1; --it);
+
+ return (*it);
+}
+
+unsigned int Rad::strokesByRad(const QString &text)
+{
+ load();
+ QValueListIterator<Radical> it;
+ for(it = list.begin(); it != list.end() && (*it).radical() != text; ++it);
+
+ return (*it).strokes();
+}
+
+Rad::~Rad()
+{
+}
+
+///////////////////////////////////////////////
+
+RadWidget::RadWidget(Rad *_rad, QWidget *parent, const char *name) : QWidget(parent, name)
+{
+ hotlistNum = 3;
+
+ rad = _rad;
+ QHBoxLayout *hlayout = new QHBoxLayout(this, KDialog::marginHint(), KDialog::spacingHint());
+ QVBoxLayout *vlayout = new QVBoxLayout(hlayout, KDialog::spacingHint());
+
+ hotlistGroup = new QButtonGroup(1, Horizontal, i18n("Hotlist"), this);
+ //hotlistGroup->setRadioButtonExclusive(true);
+ vlayout->addWidget(hotlistGroup);
+
+ Config* config = Config::self();
+
+ hotlist = config->hotlist();
+
+ while (hotlist.size() > hotlistNum)
+ hotlist.pop_front();
+
+ for (unsigned int i = 0; i < hotlistNum; ++i)
+ {
+ if (i >= hotlistNum)
+ break;
+
+ hotlistGroup->insert(new KPushButton(*hotlist.at(i), hotlistGroup), i);
+ }
+ connect(hotlistGroup, SIGNAL(clicked(int)), SLOT(hotlistClicked(int)));
+
+ QVBoxLayout *layout = new QVBoxLayout(vlayout, KDialog::spacingHint());
+
+ totalStrokes = new QCheckBox(i18n("Search by total strokes"), this);
+ connect(totalStrokes, SIGNAL(clicked()), this, SLOT(totalClicked()));
+ layout->addWidget(totalStrokes);
+
+ QHBoxLayout *strokesLayout = new QHBoxLayout(layout, KDialog::spacingHint());
+ totalSpin = new QSpinBox(1, 30, 1, this);
+ strokesLayout->addWidget(totalSpin);
+ strokesLayout->addStretch();
+ totalErrLabel = new QLabel(i18n("+/-"), this);
+ strokesLayout->addWidget(totalErrLabel);
+ totalErrSpin = new QSpinBox(0, 15, 1, this);
+ strokesLayout->addWidget(totalErrSpin);
+
+ ok = new KPushButton(i18n("&Look Up"), this);
+ ok->setEnabled(false);
+ connect(ok, SIGNAL(clicked()), SLOT(apply()));
+ layout->addWidget(ok);
+ cancel = new KPushButton( KStdGuiItem::cancel(), this );
+
+ connect(cancel, SIGNAL(clicked()), SLOT(close()));
+ layout->addWidget(cancel);
+
+ QVBoxLayout *middlevLayout = new QVBoxLayout(hlayout, KDialog::spacingHint());
+
+ strokesSpin = new QSpinBox(1, 17, 1, this);
+ QToolTip::add(strokesSpin, i18n("Show radicals having this number of strokes"));
+ middlevLayout->addWidget(strokesSpin);
+
+ List = new KListBox(this);
+ middlevLayout->addWidget(List);
+ connect(List, SIGNAL(executed(QListBoxItem *)), this, SLOT(executed(QListBoxItem *)));
+ connect(strokesSpin, SIGNAL(valueChanged(int)), this, SLOT(updateList(int)));
+
+ QVBoxLayout *rightvlayout = new QVBoxLayout(hlayout, KDialog::spacingHint());
+ selectedList = new KListBox(this);
+ rightvlayout->addWidget(selectedList);
+ connect(selectedList, SIGNAL(selectionChanged()), this, SLOT(selectionChanged()));
+
+ remove = new KPushButton(i18n("&Remove"), this);
+ rightvlayout->addWidget(remove);
+ connect(remove, SIGNAL(clicked()), this, SLOT(removeSelected()));
+ remove->setEnabled(false);
+
+ clear = new KPushButton(KStdGuiItem::clear(), this);
+ rightvlayout->addWidget(clear);
+ connect(clear, SIGNAL(clicked()), this, SLOT(clearSelected()));
+ clear->setEnabled(false);
+
+ setCaption(kapp->makeStdCaption(i18n("Radical Selector")));
+
+ strokesSpin->setValue(config->strokes());
+ strokesSpin->setFocus();
+
+ totalSpin->setValue(config->totalStrokes());
+ totalErrSpin->setValue(config->totalStrokesErrorMargin());
+ totalStrokes->setChecked(config->searchByTotal());
+
+ // make sure the right parts of the total stroke
+ // selection system are enabled
+ totalClicked();
+
+ // initially show the list of radicals to choose from
+ updateList(strokesSpin->value());
+}
+
+RadWidget::~RadWidget()
+{
+}
+
+void RadWidget::hotlistClicked(int num)
+{
+ addToSelected(*hotlist.at(num));
+}
+
+void RadWidget::executed(QListBoxItem *item)
+{
+ addToSelected(item->text());
+}
+
+void RadWidget::clearSelected()
+{
+ selectedList->clear();
+ selected.clear();
+ numChanged();
+}
+
+void RadWidget::removeSelected()
+{
+ int currentItem = selectedList->currentItem();
+ if (currentItem != -1)
+ {
+ selectedList->removeItem(currentItem);
+ selected.remove(selected.at(currentItem));
+
+ numChanged();
+ selectionChanged();
+ }
+}
+
+void RadWidget::numChanged()
+{
+ ok->setEnabled(selectedList->count() > 0);
+ clear->setEnabled(selectedList->count() > 0);
+}
+
+void RadWidget::addRadical(const QString &radical)
+{
+ addToSelected(radical);
+}
+
+void RadWidget::addToSelected(const QString &text)
+{
+ if (!text.isNull() && !selected.contains(text))
+ {
+ selectedList->insertItem(text);
+ selected.append(text);
+
+ numChanged();
+ selectionChanged();
+ }
+}
+
+void RadWidget::selectionChanged()
+{
+ //kdDebug() << "selectionChanged()" << endl;
+ remove->setEnabled(selectedList->currentItem() != -1);
+}
+
+void RadWidget::updateList(int strokes)
+{
+ List->clear();
+ List->insertStringList(rad->radByStrokes(static_cast<unsigned int>(strokes)));
+}
+
+void RadWidget::apply()
+{
+
+ if (selected.count() < 1)
+ return;
+
+ emit set(selected, totalStrokes->isChecked() ? totalSpin->value() : 0, totalErrSpin->value());
+
+ Config* config = Config::self();
+ config->setStrokes(strokesSpin->value());
+ config->setTotalStrokes(totalSpin->value());
+ config->setTotalStrokesErrorMargin(totalErrSpin->value());
+ config->setSearchByTotal(totalStrokes->isChecked());
+
+ for (QStringList::Iterator it = selected.begin(); it != selected.end(); ++it)
+ {
+ if (hotlist.find(*it) == hotlist.end())
+ {
+ if (hotlist.size() >= hotlistNum)
+ hotlist.pop_front(); // stupid stl functions in Qt .. ;)
+ hotlist.append(*it);
+
+ config->setHotlist(hotlist);
+ }
+ }
+ config->writeConfig();
+ close();
+}
+
+void RadWidget::totalClicked()
+{
+ bool enable = totalStrokes->isChecked();
+ totalSpin->setEnabled(enable);
+ totalErrSpin->setEnabled(enable);
+ totalErrLabel->setEnabled(enable);
+}
+
+//////////////////////////////////////////////
+
+Radical::Radical(QString text, unsigned int strokes)
+{
+ _Radical = text;
+ Strokes = strokes;
+}
+
+void Radical::addKanji(const QString &text)
+{
+ Kanji.append(text);
+}
+
+#include "rad.moc"