/* This file is part of * ====================================================== * * LyX, The Document Processor * * Copyright 1995 Matthias Ettrich * Copyright 1995-2001 The LyX Team. * * ====================================================== */ #include #include "lyxfunc.h" #include "version.h" #include "kbmap.h" #include "lyxrow.h" #include "bufferlist.h" #include "BufferView.h" #include "lyxserver.h" #include "intl.h" #include "lyx_main.h" #include "lyx_cb.h" #include "LyXAction.h" #include "debug.h" #include "lyxrc.h" #include "lyxtext.h" #include "gettext.h" #include "Lsstream.h" #include "trans_mgr.h" #include "encoding.h" #include "layout.h" #include "bufferview_funcs.h" #include "frontends/LyXView.h" #include "frontends/lyx_gui.h" #include "vspace.h" #include "FloatList.h" #include "format.h" #include "exporter.h" #include "importer.h" #include "TextCache.h" #include "lyxfind.h" #include "undo_funcs.h" #include "ParagraphParameters.h" #include "insets/insetcommand.h" #include "insets/inseterror.h" #include "insets/insetert.h" #include "insets/insettabular.h" #include "mathed/formulamacro.h" #include "mathed/math_cursor.h" #include "mathed/math_inset.h" #include "frontends/FileDialog.h" #include "frontends/Dialogs.h" #include "frontends/Toolbar.h" #include "frontends/Menubar.h" #include "frontends/Alert.h" #include "graphics/GraphicsCache.h" #include "support/lyxalgo.h" #include "support/LAssert.h" #include "support/filetools.h" #include "support/FileInfo.h" #include "support/forkedcontr.h" #include "support/lstrings.h" #include "support/path.h" #include "support/lyxfunctional.h" #include "BoostFormat.h" #include #include #include #include #include #include using std::pair; using std::make_pair; using std::endl; using std::find_if; using std::vector; using std::transform; using std::back_inserter; extern BufferList bufferlist; extern LyXServer * lyxserver; extern bool selection_possible; extern boost::scoped_ptr toplevel_keymap; // (alkis) extern tex_accent_struct get_accent(kb_action action); extern void ShowLatexLog(); LyXFunc::LyXFunc(LyXView * o) : owner(o), encoded_last_key(0), keyseq(toplevel_keymap.get(), toplevel_keymap.get()), cancel_meta_seq(toplevel_keymap.get(), toplevel_keymap.get()), meta_fake_bit(key_modifier::none) { } inline LyXText * LyXFunc::TEXT(bool flag = true) const { if (flag) return view()->text; return view()->getLyXText(); } inline void LyXFunc::moveCursorUpdate(bool flag, bool selecting) { if (selecting || TEXT(flag)->selection.mark()) { TEXT(flag)->setSelection(view()); if (TEXT(flag)->isTopLevel()) view()->toggleToggle(); } view()->update(TEXT(flag), BufferView::SELECT|BufferView::FITCUR); view()->showCursor(); view()->switchKeyMap(); } void LyXFunc::handleKeyFunc(kb_action action) { char c = encoded_last_key; if (keyseq.length()) { c = 0; } owner->getIntl().getTransManager() .deadkey(c, get_accent(action).accent, TEXT(false)); // Need to clear, in case the minibuffer calls these // actions keyseq.clear(); // copied verbatim from do_accent_char view()->update(TEXT(false), BufferView::SELECT|BufferView::FITCUR|BufferView::CHANGE); TEXT(false)->selection.cursor = TEXT(false)->cursor; } void LyXFunc::processKeySym(LyXKeySymPtr keysym, key_modifier::state state) { string argument; if (lyxerr.debugging(Debug::KEY)) { lyxerr << "KeySym is " << keysym->getSymbolName() << endl; } // Do nothing if we have nothing (JMarc) if (!keysym->isOK()) { lyxerr[Debug::KEY] << "Empty kbd action (probably composing)" << endl; return; } if (keysym->isModifier()) { lyxerr[Debug::KEY] << "isModifier true" << endl; return; } Encoding const * encoding = view()->getEncoding(); encoded_last_key = keysym->getISOEncoded(encoding ? encoding->Name() : ""); // Do a one-deep top-level lookup for // cancel and meta-fake keys. RVDK_PATCH_5 cancel_meta_seq.reset(); int action = cancel_meta_seq.addkey(keysym, state); lyxerr[Debug::KEY] << "action first set to [" << action << ']' << endl; // When not cancel or meta-fake, do the normal lookup. // Note how the meta_fake Mod1 bit is OR-ed in and reset afterwards. // Mostly, meta_fake_bit = key_modifier::none. RVDK_PATCH_5. if ((action != LFUN_CANCEL) && (action != LFUN_META_FAKE)) { // remove Caps Lock and Mod2 as a modifiers action = keyseq.addkey(keysym, (state | meta_fake_bit)); lyxerr[Debug::KEY] << "action now set to [" << action << ']' << endl; } // Dont remove this unless you know what you are doing. meta_fake_bit = key_modifier::none; // can this happen now ? if (action == LFUN_NOACTION) { action = LFUN_PREFIX; } if (lyxerr.debugging(Debug::KEY)) { lyxerr << "Key [action=" << action << "][" << keyseq.print() << ']' << endl; } // already here we know if it any point in going further // why not return already here if action == -1 and // num_bytes == 0? (Lgb) if (keyseq.length() > 1) { owner->message(keyseq.print()); } // Maybe user can only reach the key via holding down shift. // Let's see. But only if shift is the only modifier if (action == LFUN_UNKNOWN_ACTION && state == key_modifier::shift) { lyxerr[Debug::KEY] << "Trying without shift" << endl; action = keyseq.addkey(keysym, key_modifier::none); lyxerr[Debug::KEY] << "Action now " << action << endl; } if (action == LFUN_UNKNOWN_ACTION) { // Hmm, we didn't match any of the keysequences. See // if it's normal insertable text not already covered // by a binding if (keysym->isText() && keyseq.length() == 1) { lyxerr[Debug::KEY] << "isText() is true, inserting." << endl; action = LFUN_SELFINSERT; } else { lyxerr[Debug::KEY] << "Unknown, !isText() - giving up" << endl; owner->message(_("Unknown function.")); return; } } if (action == LFUN_SELFINSERT) { if (encoded_last_key != 0) { string arg; arg += encoded_last_key; dispatch(FuncRequest(view(), LFUN_SELFINSERT, arg)); lyxerr[Debug::KEY] << "SelfInsert arg[`" << argument << "']" << endl; } } else { dispatch(action); } } FuncStatus LyXFunc::getStatus(int ac) const { return getStatus(lyxaction.retrieveActionArg(ac)); } FuncStatus LyXFunc::getStatus(FuncRequest const & ev) const { FuncStatus flag; Buffer * buf = owner->buffer(); if (ev.action == LFUN_NOACTION) { setStatusMessage(N_("Nothing to do")); return flag.disabled(true); } switch (ev.action) { case LFUN_UNKNOWN_ACTION: #ifndef HAVE_LIBAIKSAURUS case LFUN_THESAURUS_ENTRY: #endif flag.unknown(true); break; default: flag |= lyx_gui::getStatus(ev); } if (flag.unknown()) { setStatusMessage(N_("Unknown action")); return flag; } // the default error message if we disable the command setStatusMessage(N_("Command disabled")); // Check whether we need a buffer if (!lyxaction.funcHasFlag(ev.action, LyXAction::NoBuffer)) { // Yes we need a buffer, do we have one? if (buf) { // yes // Can we use a readonly buffer? if (buf->isReadonly() && !lyxaction.funcHasFlag(ev.action, LyXAction::ReadOnly)) { // no setStatusMessage(N_("Document is read-only")); flag.disabled(true); } } else { // no setStatusMessage(N_("Command not allowed with" "out any document open")); return flag.disabled(true); } } UpdatableInset * tli = view()->theLockingInset(); // I would really like to avoid having this switch and rather try to // encode this in the function itself. bool disable = false; switch (ev.action) { case LFUN_MENUPRINT: disable = !Exporter::IsExportable(buf, "dvi") || lyxrc.print_command == "none"; break; case LFUN_EXPORT: disable = ev.argument != "custom" && !Exporter::IsExportable(buf, ev.argument); break; case LFUN_UNDO: disable = buf->undostack.empty(); break; case LFUN_REDO: disable = buf->redostack.empty(); break; case LFUN_CUT: case LFUN_COPY: if (tli) { UpdatableInset * in = tli; if (in->lyxCode() != Inset::TABULAR_CODE) { in = tli->getFirstLockingInsetOfType(Inset::TABULAR_CODE); } if (in && static_cast(in)->hasSelection()) { disable = false; break; } } disable = !mathcursor && !view()->getLyXText()->selection.set(); break; case LFUN_RUNCHKTEX: disable = lyxrc.chktex_command == "none"; break; case LFUN_BUILDPROG: disable = !Exporter::IsExportable(buf, "program"); break; case LFUN_LAYOUT_CHARACTER: disable = tli && tli->lyxCode() == Inset::ERT_CODE; break; case LFUN_LAYOUT_TABULAR: disable = !tli || (tli->lyxCode() != Inset::TABULAR_CODE && !tli->getFirstLockingInsetOfType(Inset::TABULAR_CODE)); break; case LFUN_LAYOUT: case LFUN_LAYOUT_PARAGRAPH: { Inset * inset = TEXT(false)->cursor.par()->inInset(); disable = inset && inset->forceDefaultParagraphs(inset); break; } case LFUN_INSET_OPTARG: disable = (TEXT(false)->cursor.par()->layout()->optionalargs == 0); break; case LFUN_TABULAR_FEATURE: if (mathcursor) { #if 0 // FIXME: check temporarily disabled // valign code char align = mathcursor->valign(); if (align == '\0') { disable = true; break; } if (ev.argument.empty()) { flag.clear(); break; } if (!contains("tcb", ev.argument[0])) { disable = true; break; } flag.setOnOff(ev.argument[0] == align); } else disable = true; char align = mathcursor->halign(); if (align == '\0') { disable = true; break; } if (ev.argument.empty()) { flag.clear(); break; } if (!contains("lcr", ev.argument[0])) { disable = true; break; } flag.setOnOff(ev.argument[0] == align); #endif disable = !mathcursor->halign(); break; } if (tli) { FuncStatus ret; //ret.disabled(true); if (tli->lyxCode() == Inset::TABULAR_CODE) { ret = static_cast(tli) ->getStatus(ev.argument); } else if (tli->getFirstLockingInsetOfType(Inset::TABULAR_CODE)) { ret = static_cast (tli->getFirstLockingInsetOfType(Inset::TABULAR_CODE)) ->getStatus(ev.argument); } flag |= ret; disable = false; } else { static InsetTabular inset(*owner->buffer(), 1, 1); FuncStatus ret; disable = true; ret = inset.getStatus(ev.argument); if (ret.onoff(true) || ret.onoff(false)) flag.setOnOff(false); } break; case LFUN_VC_REGISTER: disable = buf->lyxvc.inUse(); break; case LFUN_VC_CHECKIN: disable = !buf->lyxvc.inUse() || buf->isReadonly(); break; case LFUN_VC_CHECKOUT: disable = !buf->lyxvc.inUse() || !buf->isReadonly(); break; case LFUN_VC_REVERT: case LFUN_VC_UNDO: case LFUN_VC_HISTORY: disable = !buf->lyxvc.inUse(); break; case LFUN_MENURELOAD: disable = buf->isUnnamed() || buf->isClean(); break; case LFUN_BOOKMARK_GOTO: disable = !view()-> isSavedPosition(strToUnsignedInt(ev.argument)); break; case LFUN_MERGE_CHANGES: case LFUN_ACCEPT_CHANGE: case LFUN_REJECT_CHANGE: case LFUN_ACCEPT_ALL_CHANGES: case LFUN_REJECT_ALL_CHANGES: disable = !buf->params.tracking_changes; break; case LFUN_INSET_TOGGLE: { LyXText * lt = view()->getLyXText(); disable = !(isEditableInset(lt->getInset()) || (lt->inset_owner && lt->inset_owner->owner() && lt->inset_owner->owner()->isOpen())); break; } case LFUN_LATEX_LOG: disable = !IsFileReadable(buf->getLogName().second); break; case LFUN_MATH_MUTATE: if (mathcursor) //flag.setOnOff(mathcursor->formula()->hullType() == ev.argument); flag.setOnOff(false); else disable = true; break; // we just need to be in math mode to enable that case LFUN_MATH_SIZE: case LFUN_MATH_SPACE: case LFUN_MATH_LIMITS: case LFUN_MATH_NONUMBER: case LFUN_MATH_NUMBER: case LFUN_MATH_EXTERN: disable = !mathcursor; break; default: break; } // the functions which insert insets Inset::Code code = Inset::NO_CODE; switch (ev.action) { case LFUN_DIALOG_SHOW_NEW_INSET: if (ev.argument == "bibitem") code = Inset::BIBITEM_CODE; else if (ev.argument == "bibtex") code = Inset::BIBTEX_CODE; else if (ev.argument == "citation") code = Inset::CITE_CODE; else if (ev.argument == "ert") code = Inset::ERT_CODE; else if (ev.argument == "external") code = Inset::EXTERNAL_CODE; else if (ev.argument == "float") code = Inset::FLOAT_CODE; else if (ev.argument == "graphics") code = Inset::GRAPHICS_CODE; else if (ev.argument == "include") code = Inset::INCLUDE_CODE; else if (ev.argument == "index") code = Inset::INDEX_CODE; else if (ev.argument == "label") code = Inset::LABEL_CODE; else if (ev.argument == "minipage") code = Inset::MINIPAGE_CODE; else if (ev.argument == "ref") code = Inset::REF_CODE; else if (ev.argument == "toc") code = Inset::TOC_CODE; else if (ev.argument == "url") code = Inset::URL_CODE; else if (ev.argument == "wrap") code = Inset::WRAP_CODE; break; case LFUN_INSET_ERT: code = Inset::ERT_CODE; break; case LFUN_INSET_FOOTNOTE: code = Inset::FOOT_CODE; break; case LFUN_TABULAR_INSERT: code = Inset::TABULAR_CODE; break; case LFUN_INSET_MARGINAL: code = Inset::MARGIN_CODE; break; case LFUN_INSET_MINIPAGE: code = Inset::MINIPAGE_CODE; break; case LFUN_INSET_FLOAT: case LFUN_INSET_WIDE_FLOAT: code = Inset::FLOAT_CODE; break; case LFUN_INSET_WRAP: code = Inset::WRAP_CODE; break; case LFUN_FLOAT_LIST: code = Inset::FLOAT_LIST_CODE; break; #if 0 case LFUN_INSET_LIST: code = Inset::LIST_CODE; break; case LFUN_INSET_THEOREM: code = Inset::THEOREM_CODE; break; #endif case LFUN_INSET_CAPTION: code = Inset::CAPTION_CODE; break; case LFUN_INSERT_NOTE: code = Inset::NOTE_CODE; break; case LFUN_INSERT_LABEL: code = Inset::LABEL_CODE; break; case LFUN_INSET_OPTARG: code = Inset::OPTARG_CODE; break; case LFUN_INDEX_INSERT: code = Inset::INDEX_CODE; break; case LFUN_INDEX_PRINT: code = Inset::INDEX_PRINT_CODE; break; case LFUN_TOC_INSERT: code = Inset::TOC_CODE; break; case LFUN_PARENTINSERT: code = Inset::PARENT_CODE; break; case LFUN_HTMLURL: case LFUN_URL: code = Inset::URL_CODE; break; case LFUN_QUOTE: // always allow this, since we will inset a raw quote // if an inset is not allowed. break; case LFUN_HYPHENATION: case LFUN_LIGATURE_BREAK: case LFUN_HFILL: case LFUN_MENU_SEPARATOR: case LFUN_LDOTS: case LFUN_END_OF_SENTENCE: code = Inset::SPECIALCHAR_CODE; break; case LFUN_PROTECTEDSPACE: // slight hack: we know this is allowed in math mode if (!mathcursor) code = Inset::SPECIALCHAR_CODE; break; default: break; } if (code != Inset::NO_CODE && tli && !tli->insetAllowed(code)) disable = true; if (disable) flag.disabled(true); // A few general toggles switch (ev.action) { case LFUN_TOOLTIPS_TOGGLE: flag.setOnOff(owner->getDialogs().tooltipsEnabled()); break; case LFUN_READ_ONLY_TOGGLE: flag.setOnOff(buf->isReadonly()); break; case LFUN_APPENDIX: flag.setOnOff(TEXT(false)->cursor.par()->params().startOfAppendix()); break; case LFUN_SWITCHBUFFER: // toggle on the current buffer, but do not toggle off // the other ones (is that a good idea?) if (ev.argument == buf->fileName()) flag.setOnOff(true); break; case LFUN_TRACK_CHANGES: flag.setOnOff(buf->params.tracking_changes); break; default: break; } // the font related toggles if (!mathcursor) { LyXFont const & font = TEXT(false)->real_current_font; switch (ev.action) { case LFUN_EMPH: flag.setOnOff(font.emph() == LyXFont::ON); break; case LFUN_NOUN: flag.setOnOff(font.noun() == LyXFont::ON); break; case LFUN_BOLD: flag.setOnOff(font.series() == LyXFont::BOLD_SERIES); break; case LFUN_SANS: flag.setOnOff(font.family() == LyXFont::SANS_FAMILY); break; case LFUN_ROMAN: flag.setOnOff(font.family() == LyXFont::ROMAN_FAMILY); break; case LFUN_CODE: flag.setOnOff(font.family() == LyXFont::TYPEWRITER_FAMILY); break; default: break; } } else { string tc = mathcursor->getLastCode(); switch (ev.action) { case LFUN_BOLD: flag.setOnOff(tc == "mathbf"); break; case LFUN_SANS: flag.setOnOff(tc == "mathsf"); break; case LFUN_EMPH: flag.setOnOff(tc == "mathcal"); break; case LFUN_ROMAN: flag.setOnOff(tc == "mathrm"); break; case LFUN_CODE: flag.setOnOff(tc == "mathtt"); break; case LFUN_NOUN: flag.setOnOff(tc == "mathbb"); break; case LFUN_DEFAULT: flag.setOnOff(tc == "mathnormal"); break; default: break; } } // this one is difficult to get right. As a half-baked // solution, we consider only the first action of the sequence if (ev.action == LFUN_SEQUENCE) { // argument contains ';'-terminated commands flag = getStatus(lyxaction.LookupFunc(token(ev.argument, ';', 0))); } return flag; } void LyXFunc::dispatch(string const & s, bool verbose) { int const action = lyxaction.LookupFunc(s); if (action == LFUN_UNKNOWN_ACTION) { #if USE_BOOST_FORMAT boost::format fmt(_("Unknown function (%1$s)")); fmt % s; owner->message(fmt.str()); #else string const msg = string(_("Unknown function (")) + s + ')'; owner->message(msg); #endif return; } dispatch(action, verbose); } void LyXFunc::dispatch(int ac, bool verbose) { dispatch(lyxaction.retrieveActionArg(ac), verbose); } void LyXFunc::dispatch(FuncRequest const & ev, bool verbose) { lyxerr[Debug::ACTION] << "LyXFunc::dispatch: action[" << ev.action <<"] arg[" << ev.argument << ']' << endl; // we have not done anything wrong yet. errorstat = false; dispatch_buffer.erase(); #ifdef NEW_DISPATCHER // We try do call the most specific dispatcher first: // 1. the lockinginset's dispatch // 2. the bufferview's dispatch // 3. the lyxview's dispatch #endif selection_possible = false; if (view()->available()) view()->hideCursor(); string argument = ev.argument; kb_action action = ev.action; // We cannot use this function here if (getStatus(ev).disabled()) { lyxerr[Debug::ACTION] << "LyXFunc::dispatch: " << lyxaction.getActionName(action) << " [" << action << "] is disabled at this location" << endl; setErrorMessage(getStatusMessage()); goto exit_with_message; } if (view()->available() && view()->theLockingInset()) { Inset::RESULT result; if ((action > 1) || ((action == LFUN_UNKNOWN_ACTION) && (!keyseq.deleted()))) { UpdatableInset * inset = view()->theLockingInset(); #if 1 int inset_x; int dummy_y; inset->getCursorPos(view(), inset_x, dummy_y); #endif if ((action == LFUN_UNKNOWN_ACTION) && argument.empty()) { argument = encoded_last_key; } // Undo/Redo is a bit tricky for insets. if (action == LFUN_UNDO) { view()->undo(); goto exit_with_message; } else if (action == LFUN_REDO) { view()->redo(); goto exit_with_message; } else if (((result=inset-> // Hand-over to inset's own dispatch: localDispatch(FuncRequest(view(), action, argument))) == DISPATCHED) || (result == DISPATCHED_NOUPDATE)) goto exit_with_message; // If UNDISPATCHED, just soldier on else if (result == FINISHED) { goto exit_with_message; // We do not need special RTL handling here: // FINISHED means that the cursor should be // one position after the inset. } else if (result == FINISHED_RIGHT) { TEXT()->cursorRight(view()); moveCursorUpdate(true, false); owner->view_state_changed(); goto exit_with_message; } else if (result == FINISHED_UP) { if (TEXT()->cursor.irow()->previous()) { #if 1 TEXT()->setCursorFromCoordinates( view(), TEXT()->cursor.ix() + inset_x, TEXT()->cursor.iy() - TEXT()->cursor.irow()->baseline() - 1); TEXT()->cursor.x_fix(TEXT()->cursor.x()); #else TEXT()->cursorUp(view()); #endif moveCursorUpdate(true, false); owner->view_state_changed(); } else { view()->update(TEXT(), BufferView::SELECT|BufferView::FITCUR); } goto exit_with_message; } else if (result == FINISHED_DOWN) { if (TEXT()->cursor.irow()->next()) { #if 1 TEXT()->setCursorFromCoordinates( view(), TEXT()->cursor.ix() + inset_x, TEXT()->cursor.iy() - TEXT()->cursor.irow()->baseline() + TEXT()->cursor.irow()->height() + 1); TEXT()->cursor.x_fix(TEXT()->cursor.x()); #else TEXT()->cursorDown(view()); #endif } else { TEXT()->cursorRight(view()); } moveCursorUpdate(true, false); owner->view_state_changed(); goto exit_with_message; } #warning I am not sure this is still right, please have a look! (Jug 20020417) else { // result == UNDISPATCHED //setMessage(N_("Text mode")); switch (action) { case LFUN_UNKNOWN_ACTION: case LFUN_BREAKPARAGRAPH: case LFUN_BREAKLINE: TEXT()->cursorRight(view()); view()->switchKeyMap(); owner->view_state_changed(); break; case LFUN_RIGHT: if (!TEXT()->cursor.par()->isRightToLeftPar(owner->buffer()->params)) { TEXT()->cursorRight(view()); moveCursorUpdate(true, false); owner->view_state_changed(); } goto exit_with_message; case LFUN_LEFT: if (TEXT()->cursor.par()->isRightToLeftPar(owner->buffer()->params)) { TEXT()->cursorRight(view()); moveCursorUpdate(true, false); owner->view_state_changed(); } goto exit_with_message; case LFUN_DOWN: if (TEXT()->cursor.row()->next()) TEXT()->cursorDown(view()); else TEXT()->cursorRight(view()); moveCursorUpdate(true, false); owner->view_state_changed(); goto exit_with_message; default: break; } } } } switch (action) { case LFUN_ESCAPE: { if (!view()->available()) break; // this function should be used always [asierra060396] UpdatableInset * tli = view()->theLockingInset(); if (tli) { UpdatableInset * lock = tli->getLockingInset(); if (tli == lock) { view()->unlockInset(tli); TEXT()->cursorRight(view()); moveCursorUpdate(true, false); owner->view_state_changed(); } else { tli->unlockInsetInInset(view(), lock, true); } finishUndo(); // Tell the paragraph dialog that we changed paragraph owner->getDialogs().updateParagraph(); } } break; // --- Misc ------------------------------------------- case LFUN_WORDFINDFORWARD : case LFUN_WORDFINDBACKWARD : { static string last_search; string searched_string; if (!argument.empty()) { last_search = argument; searched_string = argument; } else { searched_string = last_search; } bool fw = (action == LFUN_WORDFINDFORWARD); if (!searched_string.empty()) { lyxfind::LyXFind(view(), searched_string, fw); } // view()->showCursor(); } break; case LFUN_PREFIX: { if (view()->available() && !view()->theLockingInset()) { view()->update(TEXT(), BufferView::SELECT|BufferView::FITCUR); } owner->message(keyseq.printOptions()); } break; // --- Misc ------------------------------------------- case LFUN_EXEC_COMMAND: owner->focus_command_buffer(); break; case LFUN_CANCEL: // RVDK_PATCH_5 keyseq.reset(); meta_fake_bit = key_modifier::none; if (view()->available()) // cancel any selection dispatch(LFUN_MARK_OFF); setMessage(N_("Cancel")); break; case LFUN_META_FAKE: // RVDK_PATCH_5 { meta_fake_bit = key_modifier::alt; setMessage(keyseq.print()); } break; case LFUN_READ_ONLY_TOGGLE: if (owner->buffer()->lyxvc.inUse()) { owner->buffer()->lyxvc.toggleReadOnly(); } else { owner->buffer()->setReadonly( !owner->buffer()->isReadonly()); } break; case LFUN_CENTER: // this is center and redraw. view()->center(); break; // --- Menus ----------------------------------------------- case LFUN_MENUNEW: menuNew(argument, false); break; case LFUN_MENUNEWTMPLT: menuNew(argument, true); break; case LFUN_CLOSEBUFFER: closeBuffer(); break; case LFUN_MENUWRITE: if (!owner->buffer()->isUnnamed()) { ostringstream s1; #if USE_BOOST_FORMAT s1 << boost::format(_("Saving document %1$s...")) % MakeDisplayPath(owner->buffer()->fileName()); #else s1 << _("Saving document ") << MakeDisplayPath(owner->buffer()->fileName()) << _("..."); #endif owner->message(STRCONV(s1.str())); MenuWrite(view(), owner->buffer()); s1 << _(" done."); owner->message(STRCONV(s1.str())); } else WriteAs(view(), owner->buffer()); break; case LFUN_WRITEAS: WriteAs(view(), owner->buffer(), argument); break; case LFUN_MENURELOAD: reloadBuffer(); break; case LFUN_UPDATE: Exporter::Export(owner->buffer(), argument, true); break; case LFUN_PREVIEW: Exporter::Preview(owner->buffer(), argument); break; case LFUN_BUILDPROG: Exporter::Export(owner->buffer(), "program", true); break; case LFUN_RUNCHKTEX: MenuRunChktex(owner->buffer()); break; case LFUN_MENUPRINT: owner->getDialogs().showPrint(); break; case LFUN_EXPORT: if (argument == "custom") owner->getDialogs().showSendto(); else Exporter::Export(owner->buffer(), argument, false); break; case LFUN_IMPORT: doImport(argument); break; case LFUN_QUIT: QuitLyX(); break; case LFUN_TOCVIEW: { InsetCommandParams p("tableofcontents"); string const data = InsetCommandMailer::params2string("toc", p); owner->getDialogs().show("toc", data, 0); break; } case LFUN_AUTOSAVE: AutoSave(view()); break; case LFUN_UNDO: view()->undo(); break; case LFUN_REDO: view()->redo(); break; case LFUN_MENUSEARCH: owner->getDialogs().showSearch(); break; case LFUN_REMOVEERRORS: if (view()->removeAutoInsets()) { #warning repaint() or update() or nothing ? view()->repaint(); view()->fitCursor(); } break; case LFUN_DEPTH_MIN: changeDepth(view(), TEXT(false), -1); break; case LFUN_DEPTH_PLUS: changeDepth(view(), TEXT(false), 1); break; case LFUN_FREE: owner->getDialogs().setUserFreeFont(); break; case LFUN_RECONFIGURE: Reconfigure(view()); break; #if 0 case LFUN_FLOATSOPERATE: if (argument == "openfoot") view()->allFloats(1,0); else if (argument == "closefoot") view()->allFloats(0,0); else if (argument == "openfig") view()->allFloats(1,1); else if (argument == "closefig") view()->allFloats(0,1); break; #else #ifdef WITH_WARNINGS //#warning Find another implementation here (or another lyxfunc)! #endif #endif case LFUN_HELP_ABOUTLYX: owner->getDialogs().showAboutlyx(); break; case LFUN_HELP_TEXINFO: owner->getDialogs().showTexinfo(); break; case LFUN_HELP_OPEN: { string const arg = argument; if (arg.empty()) { setErrorMessage(N_("Missing argument")); break; } string const fname = i18nLibFileSearch("doc", arg, "lyx"); if (fname.empty()) { lyxerr << "LyX: unable to find documentation file `" << arg << "'. Bad installation?" << endl; break; } ostringstream str; #if USE_BOOST_FORMAT str << boost::format(_("Opening help file %1$s...")) % MakeDisplayPath(fname); #else str << _("Opening help file ") << MakeDisplayPath(fname) << _("..."); #endif owner->message(STRCONV(str.str())); view()->buffer(bufferlist.loadLyXFile(fname, false)); break; } // --- version control ------------------------------- case LFUN_VC_REGISTER: { if (!owner->buffer()->lyxvc.inUse()) owner->buffer()->lyxvc.registrer(); } break; case LFUN_VC_CHECKIN: { if (owner->buffer()->lyxvc.inUse() && !owner->buffer()->isReadonly()) owner->buffer()->lyxvc.checkIn(); } break; case LFUN_VC_CHECKOUT: { if (owner->buffer()->lyxvc.inUse() && owner->buffer()->isReadonly()) owner->buffer()->lyxvc.checkOut(); } break; case LFUN_VC_REVERT: { owner->buffer()->lyxvc.revert(); } break; case LFUN_VC_UNDO: { owner->buffer()->lyxvc.undoLast(); } break; case LFUN_VC_HISTORY: { owner->getDialogs().showVCLogFile(); break; } // --- buffers ---------------------------------------- case LFUN_SWITCHBUFFER: view()->buffer(bufferlist.getBuffer(argument)); break; case LFUN_FILE_NEW: { // servercmd: argument must be :