/** * \file InsetBase.cpp * This file is part of LyX, the document processor. * Licence details can be found in the file COPYING. * * \author Alejandro Aguilar Sierra * \author Jürgen Vigna * \author Lars Gullik Bjønnes * \author Matthias Ettrich * \author André Pönitz * * Full author contact details are available in file CREDITS. */ #include #include "InsetBase.h" #include "Buffer.h" #include "BufferView.h" #include "Color.h" #include "CoordCache.h" #include "Cursor.h" #include "debug.h" #include "debug.h" #include "Dimension.h" #include "DispatchResult.h" #include "FuncRequest.h" #include "FuncStatus.h" #include "gettext.h" #include "LyXText.h" #include "MetricsInfo.h" #include "MetricsInfo.h" #include "frontends/Painter.h" #include #include #include namespace lyx { class InsetName { public: InsetName(std::string const & n, InsetBase::Code c) : name(n), code(c) {} std::string name; InsetBase::Code code; }; typedef std::map TranslatorMap; static TranslatorMap const build_translator() { InsetName const insetnames[] = { InsetName("toc", InsetBase::TOC_CODE), InsetName("quote", InsetBase::QUOTE_CODE), InsetName("ref", InsetBase::REF_CODE), InsetName("url", InsetBase::URL_CODE), InsetName("htmlurl", InsetBase::HTMLURL_CODE), InsetName("separator", InsetBase::SEPARATOR_CODE), InsetName("ending", InsetBase::ENDING_CODE), InsetName("label", InsetBase::LABEL_CODE), InsetName("note", InsetBase::NOTE_CODE), InsetName("accent", InsetBase::ACCENT_CODE), InsetName("math", InsetBase::MATH_CODE), InsetName("index", InsetBase::INDEX_CODE), InsetName("nomenclature", InsetBase::NOMENCL_CODE), InsetName("include", InsetBase::INCLUDE_CODE), InsetName("graphics", InsetBase::GRAPHICS_CODE), InsetName("bibitem", InsetBase::BIBITEM_CODE), InsetName("bibtex", InsetBase::BIBTEX_CODE), InsetName("text", InsetBase::TEXT_CODE), InsetName("ert", InsetBase::ERT_CODE), InsetName("foot", InsetBase::FOOT_CODE), InsetName("margin", InsetBase::MARGIN_CODE), InsetName("float", InsetBase::FLOAT_CODE), InsetName("wrap", InsetBase::WRAP_CODE), InsetName("specialchar", InsetBase::SPECIALCHAR_CODE), InsetName("tabular", InsetBase::TABULAR_CODE), InsetName("external", InsetBase::EXTERNAL_CODE), InsetName("caption", InsetBase::CAPTION_CODE), InsetName("mathmacro", InsetBase::MATHMACRO_CODE), InsetName("cite", InsetBase::CITE_CODE), InsetName("float_list", InsetBase::FLOAT_LIST_CODE), InsetName("index_print", InsetBase::INDEX_PRINT_CODE), InsetName("nomencl_print", InsetBase::NOMENCL_PRINT_CODE), InsetName("optarg", InsetBase::OPTARG_CODE), InsetName("environment", InsetBase::ENVIRONMENT_CODE), InsetName("hfill", InsetBase::HFILL_CODE), InsetName("newline", InsetBase::NEWLINE_CODE), InsetName("line", InsetBase::LINE_CODE), InsetName("branch", InsetBase::BRANCH_CODE), InsetName("box", InsetBase::BOX_CODE), InsetName("charstyle", InsetBase::CHARSTYLE_CODE), InsetName("vspace", InsetBase::VSPACE_CODE), InsetName("mathmacroarg", InsetBase::MATHMACROARG_CODE), }; std::size_t const insetnames_size = sizeof(insetnames) / sizeof(insetnames[0]); std::map data; for (std::size_t i = 0; i != insetnames_size; ++i) { InsetName const & var = insetnames[i]; data[var.name] = var.code; } return data; } /// pretty arbitrary dimensions InsetBase::InsetBase() : dim_(10, 10, 10), background_color_(Color::background) {} InsetBase::InsetBase(InsetBase const & inset) : dim_(inset.dim_), background_color_(inset.background_color_) {} std::auto_ptr InsetBase::clone() const { std::auto_ptr b = doClone(); BOOST_ASSERT(typeid(*b) == typeid(*this)); return b; } docstring InsetBase::getInsetName() const { return from_ascii("unknown"); } InsetBase::Code InsetBase::translate(std::string const & name) { static TranslatorMap const translator = build_translator(); TranslatorMap::const_iterator it = translator.find(name); return it == translator.end() ? NO_CODE : it->second; } void InsetBase::dispatch(Cursor & cur, FuncRequest & cmd) { cur.updateFlags(Update::Force | Update::FitCursor); cur.dispatched(); doDispatch(cur, cmd); } void InsetBase::doDispatch(Cursor & cur, FuncRequest &) { cur.noUpdate(); cur.undispatched(); } bool InsetBase::getStatus(Cursor &, FuncRequest const & cmd, FuncStatus & flag) const { // LFUN_INSET_APPLY is sent from the dialogs when the data should // be applied. This is either changed to LFUN_INSET_MODIFY (if the // dialog belongs to us) or LFUN_INSET_INSERT (if the dialog does // not belong to us, i. e. the dialog was open, and the user moved // the cursor in our inset) in LyXFunc::getStatus(). // Dialogs::checkStatus() ensures that the dialog is deactivated if // LFUN_INSET_APPLY is disabled. switch (cmd.action) { case LFUN_INSET_MODIFY: // Allow modification of our data. // This needs to be handled in the doDispatch method of our // instantiatable children. flag.enabled(true); return true; case LFUN_INSET_INSERT: // Don't allow insertion of new insets. // Every inset that wants to allow new insets from open // dialogs needs to override this. flag.enabled(false); return true; default: return false; } } void InsetBase::edit(Cursor &, bool) { LYXERR(Debug::INSETS) << BOOST_CURRENT_FUNCTION << ": edit left/right" << std::endl; } InsetBase * InsetBase::editXY(Cursor &, int x, int y) { LYXERR(Debug::INSETS) << BOOST_CURRENT_FUNCTION << ": x=" << x << " y= " << y << std::endl; return this; } InsetBase::idx_type InsetBase::index(row_type row, col_type col) const { if (row != 0) lyxerr << BOOST_CURRENT_FUNCTION << ": illegal row: " << row << std::endl; if (col != 0) lyxerr << BOOST_CURRENT_FUNCTION << ": illegal col: " << col << std::endl; return 0; } bool InsetBase::idxBetween(idx_type idx, idx_type from, idx_type to) const { return from <= idx && idx <= to; } bool InsetBase::idxUpDown(Cursor &, bool) const { return false; } int InsetBase::docbook(Buffer const &, odocstream &, OutputParams const &) const { return 0; } bool InsetBase::directWrite() const { return false; } InsetBase::EDITABLE InsetBase::editable() const { return NOT_EDITABLE; } bool InsetBase::autoDelete() const { return false; } docstring const InsetBase::editMessage() const { return _("Opened inset"); } void InsetBase::cursorPos(BufferView const & /*bv*/, CursorSlice const &, bool, int & x, int & y) const { lyxerr << "InsetBase::cursorPos called directly" << std::endl; x = 100; y = 100; } void InsetBase::metricsMarkers(Dimension & dim, int framesize) const { dim.wid += 2 * framesize; dim.asc += framesize; } void InsetBase::metricsMarkers2(Dimension & dim, int framesize) const { dim.wid += 2 * framesize; dim.asc += framesize; dim.des += framesize; } void InsetBase::drawMarkers(PainterInfo & pi, int x, int y) const { Color::color pen_color = editing(pi.base.bv)? Color::mathframe : Color::background; int const t = x + width() - 1; int const d = y + descent(); pi.pain.line(x, d - 3, x, d, pen_color); pi.pain.line(t, d - 3, t, d, pen_color); pi.pain.line(x, d, x + 3, d, pen_color); pi.pain.line(t - 3, d, t, d, pen_color); setPosCache(pi, x, y); } void InsetBase::drawMarkers2(PainterInfo & pi, int x, int y) const { Color::color pen_color = editing(pi.base.bv)? Color::mathframe : Color::background; drawMarkers(pi, x, y); int const t = x + width() - 1; int const a = y - ascent(); pi.pain.line(x, a + 3, x, a, pen_color); pi.pain.line(t, a + 3, t, a, pen_color); pi.pain.line(x, a, x + 3, a, pen_color); pi.pain.line(t - 3, a, t, a, pen_color); setPosCache(pi, x, y); } bool InsetBase::editing(BufferView * bv) const { return bv->cursor().isInside(this); } int InsetBase::xo(BufferView const & bv) const { return bv.coordCache().getInsets().x(this); } int InsetBase::yo(BufferView const & bv) const { return bv.coordCache().getInsets().y(this); } bool InsetBase::covers(BufferView const & bv, int x, int y) const { //lyxerr << "InsetBase::covers, x: " << x << " y: " << y // << " xo: " << xo(bv) << " yo: " << yo() // << " x1: " << xo(bv) << " x2: " << xo() + width() // << " y1: " << yo(bv) - ascent() << " y2: " << yo() + descent() // << std::endl; return bv.coordCache().getInsets().has(this) && x >= xo(bv) && x <= xo(bv) + width() && y >= yo(bv) - ascent() && y <= yo(bv) + descent(); } void InsetBase::dump() const { Buffer buf("foo", 1); write(buf, lyxerr); } void InsetBase::setBackgroundColor(Color_color color) { background_color_ = color; } Color_color InsetBase::backgroundColor() const { return Color::color(background_color_); } void InsetBase::setPosCache(PainterInfo const & pi, int x, int y) const { //lyxerr << "InsetBase:: position cache to " << x << " " << y << std::endl; pi.base.bv->coordCache().insets().add(this, x, y); } ///////////////////////////////////////// bool isEditableInset(InsetBase const * inset) { return inset && inset->editable(); } bool isHighlyEditableInset(InsetBase const * inset) { return inset && inset->editable() == InsetBase::HIGHLY_EDITABLE; } } // namespace lyx