Skip to content
PhoneticContext.cc 7.46 KiB
Newer Older
/* vim:set et ts=4 sts=4:
 *
 * libpyzy - The Chinese PinYin and Bopomofo conversion library.
 *
 * Copyright (c) 2008-2010 Peng Huang <shawn.p.huang@gmail.com>
 *
Hiroshi Sumita's avatar
Hiroshi Sumita committed
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
Hiroshi Sumita's avatar
Hiroshi Sumita committed
 * This library 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
 * Lesser General Public License for more details.
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
Hiroshi Sumita's avatar
Hiroshi Sumita committed
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301
#include "PhoneticContext.h"

#include "Database.h"
#include "PhraseEditor.h"
#include "SimpTradConverter.h"

namespace PyZy {

Hiroshi Sumita's avatar
Hiroshi Sumita committed
PhoneticContext::PhoneticContext (PhoneticContext::Observer *observer)
    : m_phrase_editor (m_config),
      m_observer (observer)
{
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    resetContext ();
}

PhoneticContext::~PhoneticContext ()
{
}

bool
PhoneticContext::updateSpecialPhrases (void)
{
    size_t size = m_special_phrases.size ();
    m_special_phrases.clear ();

Hiroshi Sumita's avatar
Hiroshi Sumita committed
    if (!m_config.specialPhrases)
Hiroshi Sumita's avatar
Hiroshi Sumita committed
        return false;

    if (!m_selected_special_phrase.empty ())
Hiroshi Sumita's avatar
Hiroshi Sumita committed
        return false;
    size_t begin = m_phrase_editor.cursorInChar ();
    size_t end = m_cursor;

    if (begin < end) {
        SpecialPhraseTable::instance ().lookup (
            m_text.substr (begin, m_cursor - begin),
            m_special_phrases);
    }

    return size != m_special_phrases.size () || size != 0;
}

Hiroshi Sumita's avatar
Hiroshi Sumita committed
void
PhoneticContext::reset (void)
{
    resetContext ();
    update ();
Hiroshi Sumita's avatar
Hiroshi Sumita committed
void
PhoneticContext::commitText (const std::string & commit_text)
{
    m_observer->commitText (this, commit_text);
}

void
PhoneticContext::updateInputText (void)
{
    m_observer->inputTextChanged (this);
}

void
PhoneticContext::updateCursor (void)
{
    m_observer->cursorChanged (this);
}

PhoneticContext::updateCandidates (void)
    m_focused_candidate = 0;
    m_observer->candidatesChanged (this);
Hiroshi Sumita's avatar
Hiroshi Sumita committed
void
PhoneticContext::updateAuxiliaryText (void)
{
    m_observer->auxiliaryTextChanged (this);
}

void
PhoneticContext::updatePreeditText (void)
{
    m_observer->preeditTextChanged (this);
}

Hiroshi Sumita's avatar
Hiroshi Sumita committed
PhoneticContext::resetContext (void)
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    m_cursor = 0;
    m_focused_candidate = 0;
    m_pinyin.clear ();
    m_pinyin_len = 0;
    m_phrase_editor.reset ();
    m_special_phrases.clear ();
    m_selected_special_phrase.clear ();
    m_text.clear ();
    m_preedit_text.clear ();
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    m_auxiliary_text.clear ();
Hiroshi Sumita's avatar
Hiroshi Sumita committed
bool
PhoneticContext::focusCandidatePrevious ()
{
    if (G_UNLIKELY (m_focused_candidate == 0)) {
Hiroshi Sumita's avatar
Hiroshi Sumita committed
        return false;
    }
    return focusCandidate (m_focused_candidate - 1);
}

Hiroshi Sumita's avatar
Hiroshi Sumita committed
bool
PhoneticContext::focusCandidateNext ()
{
    if (G_UNLIKELY (!hasCandidate (m_focused_candidate + 1))) {
Hiroshi Sumita's avatar
Hiroshi Sumita committed
        return false;
    }
    return focusCandidate (m_focused_candidate + 1);
}

Hiroshi Sumita's avatar
Hiroshi Sumita committed
bool
PhoneticContext::focusCandidate (size_t i)
    if (G_UNLIKELY (!hasCandidate (i))) {
        g_warning ("Too big index. Can't focus to selected candidate.");
Hiroshi Sumita's avatar
Hiroshi Sumita committed
        return false;
    }
    m_focused_candidate = i;

    updatePreeditText();
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    return true;
}

void
PhoneticContext::update ()
{
    updateCandidates ();
    updatePreeditText ();
    updateAuxiliaryText ();
}

Hiroshi Sumita's avatar
Hiroshi Sumita committed
bool
PhoneticContext::selectCandidate (size_t i)
    if (!hasCandidate (i)) {
        g_warning ("selectCandidate(%zd): Too big index!\n", i);
    }

    if (i < m_special_phrases.size ()) {
        // select a special phrase
        m_selected_special_phrase = m_special_phrases[i];
        m_focused_candidate = 0;
        if (m_cursor == m_text.size ()) {
            commit ();
        }
        else {
            updateSpecialPhrases ();
            update ();
        }

Hiroshi Sumita's avatar
Hiroshi Sumita committed
        return true;
    }

    i -= m_special_phrases.size ();
    if (m_phrase_editor.selectCandidate (i)) {
        m_focused_candidate = 0;
        if (m_phrase_editor.pinyinExistsAfterCursor () ||
            *textAfterPinyin () != '\0') {
            updateSpecialPhrases ();
            update ();
        }
        else {
            commit ();
        }
Hiroshi Sumita's avatar
Hiroshi Sumita committed
        return true;
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    return false;
Hiroshi Sumita's avatar
Hiroshi Sumita committed
bool
PhoneticContext::resetCandidate (size_t i)
{
    if (i < m_special_phrases.size ()) {
Hiroshi Sumita's avatar
Hiroshi Sumita committed
        return false;
    }
    i -= m_special_phrases.size ();

    if (m_phrase_editor.resetCandidate (i)) {
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    return true;
Hiroshi Sumita's avatar
Hiroshi Sumita committed
bool
PhoneticContext::unselectCandidates ()
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    if (!m_phrase_editor.unselectCandidates ()) {
        return false;
    }
    updateSpecialPhrases ();
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    update ();
    return true;
bool
PhoneticContext::hasCandidate (size_t i)
{
    if (G_UNLIKELY (!m_selected_special_phrase.empty())) {
        return false;
    }

    while (true) {
        const size_t candidates_size =
            m_special_phrases.size () + m_phrase_editor.candidates ().size ();
        if (i < candidates_size) {
            break;
        }
        if (G_UNLIKELY (!m_phrase_editor.fillCandidates ())) {
            return false;
        }
    }

    return true;
}

bool
PhoneticContext::getCandidate (size_t i, Candidate & candidate)
{
    if (G_UNLIKELY (!hasCandidate (i))) {
        return false;
    }

    if (i < m_special_phrases.size ()) {
        candidate.text = m_special_phrases[i];
        candidate.type = SPECIAL_PHRASE;
        return true;
    }

    i -= m_special_phrases.size ();
Hiroshi Sumita's avatar
Hiroshi Sumita committed
    if (m_config.modeSimp) {
        candidate.text = m_phrase_editor.candidate (i).phrase;
    } else {
        String output;
        SimpTradConverter::simpToTrad (m_phrase_editor.candidate (i).phrase,
                                       output);
        candidate.text = output;
    }
    candidate.type = m_phrase_editor.candidateIsUserPhrase (i)
        ? USER_PHRASE : NORMAL_PHRASE;
    return true;
}

size_t
PhoneticContext::getPreparedCandidatesSize () const
{
    if (G_UNLIKELY (!m_selected_special_phrase.empty())) {
        return 0;
    }

    return m_special_phrases.size () + m_phrase_editor.candidates ().size ();
}

Hiroshi Sumita's avatar
Hiroshi Sumita committed
Variant
PhoneticContext::getProperty (PropertyName name) const
{
    switch (name) {
    case PROPERTY_CONVERSION_OPTION:
        return Variant::fromUnsignedInt (m_config.option);
    case PROPERTY_SPECIAL_PHRASE:
        return Variant::fromBool (m_config.specialPhrases);
    case PROPERTY_MODE_SIMP:
        return Variant::fromBool (m_config.modeSimp);
    default:
        return Variant::nullVariant ();
    }
}

bool
PhoneticContext::setProperty (PropertyName name, const Variant &variant)
{
    if (variant.getType () == Variant::TYPE_UNSIGNED_INT) {
        const unsigned int value = variant.getUnsignedInt ();

        switch (name) {
        case PROPERTY_CONVERSION_OPTION:
            m_config.option = value;
            return true;
        default:
            return false;
        }
    } else if (variant.getType () == Variant::TYPE_BOOL) {
        const bool value = variant.getBool ();

        switch (name) {
        case PROPERTY_SPECIAL_PHRASE:
            m_config.specialPhrases = value;
            return true;
        case PROPERTY_MODE_SIMP:
            m_config.modeSimp = value;
            return true;
        default:
            return false;
        }
    }

    return false;
}

};  // namespace PyZy