/* This file is part of * ====================================================== * * LyX, The Document Processor * * Copyright 1995 Matthias Ettrich * Copyright 1995-2001 The LyX Team. * * ====================================================== */ #include #ifdef __GNUG__ #pragma implementation #endif #include "lyxfunc.h" #include "version.h" #include "kbmap.h" #include "lyxrow.h" #include "bufferlist.h" #include "BufferView.h" #include "ColorHandler.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 "layout.h" #include "bufferview_funcs.h" #include "minibuffer.h" #include "vspace.h" #include "LyXView.h" #include "FloatList.h" #include "converter.h" #include "exporter.h" #include "importer.h" #include "FontLoader.h" #include "TextCache.h" #include "lyxfind.h" #include "undo_funcs.h" #include "ParagraphParameters.h" #include "insets/inseturl.h" #include "insets/insetlatexaccent.h" #include "insets/insettoc.h" #include "insets/insetref.h" #include "insets/insetparent.h" #include "insets/insetindex.h" #include "insets/insetinclude.h" #include "insets/insetbib.h" #include "insets/insetcite.h" #include "insets/insettext.h" #include "insets/insetert.h" #include "insets/insetexternal.h" #include "insets/insetgraphics.h" #include "insets/insetfoot.h" #include "insets/insetmarginal.h" #include "insets/insetminipage.h" #include "insets/insetfloat.h" #if 0 #include "insets/insetlist.h" #include "insets/insettheorem.h" #endif #include "insets/insettabular.h" #include "insets/insetcaption.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 #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; extern void show_symbols_form(LyXFunc *); extern LyXAction lyxaction; // (alkis) extern tex_accent_struct get_accent(kb_action action); extern void ShowLatexLog(); #if 0 /// class MiniBufferController : public SigC::Object { public: /// MiniBufferController() { minibuffer .cmdReady .connect(slot(this, &MiniBufferController::receiveCommand)); minibuffer .argReady .connect(slot(this, &MiniBufferController::receiveArg)); } /// void receiveCmd(string const & cmd) {} /// void receiveArg(string const & arg) {} private: }; namespace { MiniBufferController mb_ctrl; } #endif /* === globals =========================================================== */ LyXFunc::LyXFunc(LyXView * o) : owner(o), keyseq(toplevel_keymap.get(), toplevel_keymap.get()), cancel_meta_seq(toplevel_keymap.get(), toplevel_keymap.get()) { meta_fake_bit = 0; lyx_dead_action = LFUN_NOACTION; lyx_calling_dead_action = LFUN_NOACTION; } inline LyXText * LyXFunc::TEXT(bool flag = true) const { if (flag) return owner->view()->text; return owner->view()->getLyXText(); } // I changed this func slightly. I commented out the ...FinishUndo(), // this means that all places that used to have a moveCursorUpdate, now // have a ...FinishUndo() as the preceeding statement. I have also added // a moveCursorUpdate to some of the functions that updated the cursor, but // that did not show its new position. inline void LyXFunc::moveCursorUpdate(bool flag, bool selecting) { if (selecting || TEXT(flag)->selection.mark()) { TEXT(flag)->setSelection(owner->view()); if (TEXT(flag)->bv_owner) owner->view()->toggleToggle(); } owner->view()->update(TEXT(flag), BufferView::SELECT|BufferView::FITCUR); owner->view()->showCursor(); /* ---> Everytime the cursor is moved, show the current font state. */ // should this too me moved out of this func? owner->view()->setState(); } void LyXFunc::handleKeyFunc(kb_action action) { char c = keyseq.getiso(); if (keyseq.length() > 1) { c = 0; } owner->getIntl()->getTrans() .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 owner->view()->update(TEXT(false), BufferView::SELECT|BufferView::FITCUR|BufferView::CHANGE); TEXT(false)->selection.cursor = TEXT(false)->cursor; } void LyXFunc::processKeySym(KeySym keysym, unsigned int state) { string argument; if (lyxerr.debugging(Debug::KEY)) { char const * tmp = XKeysymToString(keysym); string const stm = (tmp ? tmp : ""); lyxerr << "KeySym is " << stm << "[" << keysym << "] State is [" << state << "]" << endl; } // Do nothing if we have nothing (JMarc) if (keysym == NoSymbol) { lyxerr[Debug::KEY] << "Empty kbd action (probably composing)" << endl; return; } // Can we be sure that this will work for all X Window // implementations? (Lgb) // This code snippet makes lyx ignore some keys. Perhaps // all of them should be explictly mentioned? if ((keysym >= XK_Shift_L && keysym <= XK_Hyper_R) || keysym == XK_Mode_switch || keysym == 0x0) { return; } // 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 &(ShiftMask|ControlMask |Mod1Mask)); if (lyxerr.debugging(Debug::KEY)) { lyxerr << "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 = 0. RVDK_PATCH_5. if ((action != LFUN_CANCEL) && (action != LFUN_META_FAKE)) { if (lyxerr.debugging(Debug::KEY)) { lyxerr << "meta_fake_bit is [" << meta_fake_bit << "]" << endl; } // remove Caps Lock and Mod2 as a modifiers action = keyseq.addkey(keysym, (state | meta_fake_bit) &(ShiftMask|ControlMask |Mod1Mask)); if (lyxerr.debugging(Debug::KEY)) { lyxerr << "action now set to [" << action << "]" << endl; } } // Dont remove this unless you know what you are doing. meta_fake_bit = 0; // 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()); } if (action == LFUN_UNKNOWN_ACTION) { // It is unknown, but what if we remove all // the modifiers? (Lgb) action = keyseq.addkey(keysym, 0); if (lyxerr.debugging(Debug::KEY)) { lyxerr << "Removing modifiers...\n" << "Action now set to [" << action << "]" << endl; } if (action == LFUN_UNKNOWN_ACTION) { owner->message(_("Unknown function.")); return; } } if (action == LFUN_SELFINSERT) { // This is very X dependent. unsigned int c = keysym; string argument; c = kb_keymap::getiso(c); if (c > 0) argument = static_cast(c); dispatch(LFUN_SELFINSERT, argument); lyxerr[Debug::KEY] << "SelfInsert arg[`" << argument << "']" << endl; } else verboseDispatch(action, false); } FuncStatus LyXFunc::getStatus(int ac) const { kb_action action; string argument; action = lyxaction.retrieveActionArg(ac, argument); return getStatus(action, argument); } FuncStatus LyXFunc::getStatus(kb_action action, string const & argument) const { FuncStatus flag; Buffer * buf = owner->buffer(); if (action == LFUN_NOACTION) { setStatusMessage(N_("Nothing to do")); return flag.disabled(true); } if (action == LFUN_UNKNOWN_ACTION) { setStatusMessage(N_("Unknown action")); return flag.unknown(true); } // the default error message if we disable the command setStatusMessage(N_("Command disabled")); // Check whether we need a buffer if (!lyxaction.funcHasFlag(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(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 = owner->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 (action) { case LFUN_MENUPRINT: disable = !Exporter::IsExportable(buf, "dvi") || lyxrc.print_command == "none"; break; case LFUN_EXPORT: disable = argument == "fax" && !Exporter::IsExportable(buf, argument); break; case LFUN_UNDO: disable = buf->undostack.empty(); break; case LFUN_REDO: disable = buf->redostack.empty(); break; case LFUN_SPELLCHECK: disable = lyxrc.isp_command == "none"; break; #ifndef HAVE_LIBAIKSAURUS case LFUN_THESAURUS_ENTRY: disable = true; break; #endif case LFUN_RUNCHKTEX: disable = lyxrc.chktex_command == "none"; break; case LFUN_BUILDPROG: disable = !Exporter::IsExportable(buf, "program"); 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_TABULAR_FEATURE: disable = true; if (tli) { FuncStatus ret; //ret.disabled(true); if (tli->lyxCode() == Inset::TABULAR_CODE) { ret = static_cast(tli) ->getStatus(argument); } else if (tli->getFirstLockingInsetOfType(Inset::TABULAR_CODE)) { ret = static_cast (tli->getFirstLockingInsetOfType(Inset::TABULAR_CODE)) ->getStatus(argument); } flag |= ret; disable = false; } else { static InsetTabular inset(*owner->buffer(), 1, 1); FuncStatus ret; disable = true; ret = inset.getStatus(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_BOOKMARK_GOTO: disable = !owner->view()-> isSavedPosition(strToUnsignedInt(argument)); break; case LFUN_INSET_TOGGLE: { LyXText * lt = owner->view()->getLyXText(); disable = !(isEditableInset(lt->getInset()) || (lt->inset_owner && lt->inset_owner->owner() && lt->inset_owner->owner()->isOpen())); break; } case LFUN_MATH_VALIGN: if (mathcursor) { char align = mathcursor->valign(); if (align == '\0') { disable = true; break; } if (argument.empty()) { flag.clear(); break; } if (!contains("tcb", argument[0])) { disable = true; break; } flag.setOnOff(argument[0] == align); } else disable = true; break; case LFUN_MATH_HALIGN: if (mathcursor) { char align = mathcursor->halign(); if (align == '\0') { disable = true; break; } if (argument.empty()) { flag.clear(); break; } if (!contains("lcr", argument[0])) { disable = true; break; } flag.setOnOff(argument[0] == align); } else disable = true; break; case LFUN_MATH_MUTATE: if (tli && (tli->lyxCode() == Inset::MATH_CODE)) { MathInsetTypes type = mathcursor->formula()->getType(); if (argument == "inline") { flag.setOnOff(type == LM_OT_SIMPLE); } else if (argument == "display") { flag.setOnOff(type == LM_OT_EQUATION); } else if (argument == "eqnarray") { flag.setOnOff(type == LM_OT_EQNARRAY); } else if (argument == "align") { flag.setOnOff(type == LM_OT_ALIGN); } else if (argument == "alignat") { flag.setOnOff(type == LM_OT_ALIGNAT); } else if (argument == "xalignat") { flag.setOnOff(type == LM_OT_XALIGNAT); } else if (argument == "xxalignat") { flag.setOnOff(type == LM_OT_XXALIGNAT); } else { disable = true; } } 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: disable = !mathcursor; break; // we need to be math mode and a math array for that // Hack: halign produces non-zero result iff we are in a math array case LFUN_MATH_ROW_INSERT: case LFUN_MATH_ROW_DELETE: case LFUN_MATH_COLUMN_INSERT: case LFUN_MATH_COLUMN_DELETE: disable = !mathcursor || !mathcursor->halign(); break; default: break; } // the functions which insert insets Inset::Code code = Inset::NO_CODE; switch (action) { case LFUN_INSET_ERT: code = Inset::ERT_CODE; break; case LFUN_INSET_GRAPHICS: code = Inset::GRAPHICS_CODE; break; case LFUN_INSET_FOOTNOTE: code = Inset::FOOT_CODE; break; case LFUN_DIALOG_TABULAR_INSERT: case LFUN_INSET_TABULAR: code = Inset::TABULAR_CODE; break; case LFUN_INSET_EXTERNAL: code = Inset::EXTERNAL_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_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::IGNORE_CODE; break; case LFUN_INSERT_LABEL: code = Inset::LABEL_CODE; break; case LFUN_REF_INSERT: code = Inset::REF_CODE; break; case LFUN_CITATION_CREATE: case LFUN_CITATION_INSERT: code = Inset::CITE_CODE; break; case LFUN_INSERT_BIBTEX: code = Inset::BIBTEX_CODE; break; case LFUN_INDEX_INSERT: case LFUN_INDEX_INSERT_LAST: case LFUN_INDEX_CREATE: code = Inset::INDEX_CODE; break; case LFUN_INDEX_PRINT: code = Inset::INDEX_PRINT_CODE; break; case LFUN_CHILD_INSERT: code = Inset::INCLUDE_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: case LFUN_INSERT_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 (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 (argument == buf->fileName()) flag.setOnOff(true); break; default: break; } // the font related toggles if (!mathcursor) { LyXFont const & font = TEXT(false)->real_current_font; switch (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 { MathTextCodes tc = mathcursor->getLastCode(); switch (action) { case LFUN_BOLD: flag.setOnOff(tc == LM_TC_BF); break; case LFUN_SANS: flag.setOnOff(tc == LM_TC_SF); break; case LFUN_EMPH: flag.setOnOff(tc == LM_TC_CAL); break; case LFUN_ROMAN: flag.setOnOff(tc == LM_TC_RM); break; case LFUN_CODE: flag.setOnOff(tc == LM_TC_TT); break; case LFUN_NOUN: flag.setOnOff(tc == LM_TC_BB); break; case LFUN_DEFAULT: flag.setOnOff(tc == LM_TC_VAR); break; default: break; } } return flag; } // temporary dispatch method void LyXFunc::miniDispatch(string const & s) { string s2(frontStrip(strip(s))); if (!s2.empty()) { verboseDispatch(s2, true); } } void LyXFunc::verboseDispatch(string const & s, bool show_sc) { int action = lyxaction.LookupFunc(frontStrip(s)); if (action == LFUN_UNKNOWN_ACTION) { string const msg = string(_("Unknown function (")) + s + ")"; owner->message(msg); } else { verboseDispatch(action, show_sc); } } void LyXFunc::verboseDispatch(int ac, bool show_sc) { string argument; kb_action action; // get the real action and argument action = lyxaction.retrieveActionArg(ac, argument); verboseDispatch(action, argument, show_sc); } void LyXFunc::verboseDispatch(kb_action action, string const & argument, bool show_sc) { string res = dispatch(action, argument); commandshortcut.erase(); if (lyxrc.display_shortcuts && show_sc) { if (action != LFUN_SELFINSERT) { // Put name of command and list of shortcuts // for it in minibuffer string comname = lyxaction.getActionName(action); int pseudoaction = action; bool argsadded = false; if (!argument.empty()) { // the pseudoaction is useful for the bindings pseudoaction = lyxaction.searchActionArg(action, argument); if (pseudoaction == LFUN_UNKNOWN_ACTION) { pseudoaction = action; } else { comname += " " + argument; argsadded = true; } } string const shortcuts = toplevel_keymap->findbinding(pseudoaction); if (!shortcuts.empty()) { comname += ": " + shortcuts; } else if (!argsadded && !argument.empty()) { comname += " " + argument; } if (!comname.empty()) { comname = strip(comname); commandshortcut = "(" + comname + ')'; } } } if (res.empty()) { if (!commandshortcut.empty()) { owner->getMiniBuffer()->addSet(commandshortcut); } } else { owner->getMiniBuffer()->addSet(' ' + commandshortcut); } } string const LyXFunc::dispatch(kb_action action, string argument) { lyxerr[Debug::ACTION] << "LyXFunc::Dispatch: action[" << action <<"] arg[" << 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 (owner->view()->available()) owner->view()->hideCursor(); // We cannot use this function here if (getStatus(action, argument).disabled()) { lyxerr[Debug::ACTION] << "LyXFunc::Dispatch: " << lyxaction.getActionName(action) << " [" << action << "] is disabled at this location" << endl; setErrorMessage(getStatusMessage()); goto exit_with_message; } if (owner->view()->available() && owner->view()->theLockingInset()) { UpdatableInset::RESULT result; if ((action > 1) || ((action == LFUN_UNKNOWN_ACTION) && (!keyseq.deleted()))) { UpdatableInset * inset = owner->view()->theLockingInset(); #if 1 int inset_x; int dummy_y; inset->getCursorPos(owner->view(), inset_x, dummy_y); #endif if ((action == LFUN_UNKNOWN_ACTION) && argument.empty()) { argument = keyseq.getiso(); } // Undo/Redo is a bit tricky for insets. if (action == LFUN_UNDO) { owner->view()->menuUndo(); goto exit_with_message; } else if (action == LFUN_REDO) { owner->view()->menuRedo(); goto exit_with_message; } else if (((result=inset-> // Hand-over to inset's own dispatch: localDispatch(owner->view(), action, argument)) == UpdatableInset::DISPATCHED) || (result == UpdatableInset::DISPATCHED_NOUPDATE)) goto exit_with_message; // If UNDISPATCHED, just soldier on else if (result == UpdatableInset::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 == UpdatableInset::FINISHED_RIGHT) { TEXT()->cursorRight(owner->view()); moveCursorUpdate(true, false); owner->showState(); goto exit_with_message; } else if (result == UpdatableInset::FINISHED_UP) { if (TEXT()->cursor.irow()->previous()) { #if 1 TEXT()->setCursorFromCoordinates( owner->view(), TEXT()->cursor.ix() + inset_x, TEXT()->cursor.iy() - TEXT()->cursor.irow()->baseline() - 1); TEXT()->cursor.x_fix(TEXT()->cursor.x()); #else TEXT()->cursorUp(owner->view()); #endif moveCursorUpdate(true, false); owner->showState(); } else { owner->view()->update(TEXT(), BufferView::SELECT|BufferView::FITCUR); } goto exit_with_message; } else if (result == UpdatableInset::FINISHED_DOWN) { if (TEXT()->cursor.irow()->next()) { #if 1 TEXT()->setCursorFromCoordinates( owner->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(owner->view()); #endif } else { TEXT()->cursorRight(owner->view()); } moveCursorUpdate(true, false); owner->showState(); 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(owner->view()); owner->view()->setState(); owner->showState(); break; case LFUN_RIGHT: if (!TEXT()->cursor.par()->isRightToLeftPar(owner->buffer()->params)) { TEXT()->cursorRight(owner->view()); moveCursorUpdate(true, false); owner->showState(); } goto exit_with_message; case LFUN_LEFT: if (TEXT()->cursor.par()->isRightToLeftPar(owner->buffer()->params)) { TEXT()->cursorRight(owner->view()); moveCursorUpdate(true, false); owner->showState(); } goto exit_with_message; case LFUN_DOWN: if (TEXT()->cursor.row()->next()) TEXT()->cursorDown(owner->view()); else TEXT()->cursorRight(owner->view()); moveCursorUpdate(true, false); owner->showState(); goto exit_with_message; default: break; } } } } switch (action) { case LFUN_ESCAPE: { if (!owner->view()->available()) break; // this function should be used always [asierra060396] UpdatableInset * tli = owner->view()->theLockingInset(); if (tli) { UpdatableInset * lock = tli->getLockingInset(); if (tli == lock) { owner->view()->unlockInset(tli); TEXT()->cursorRight(owner->view()); moveCursorUpdate(true, false); owner->showState(); } else { tli->unlockInsetInInset(owner->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_WORDFINDBACKWARD); if (!searched_string.empty()) { lyxfind::LyXFind(owner->view(), searched_string, fw); } // owner->view()->showCursor(); } break; case LFUN_PREFIX: { if (owner->view()->available() && !owner->view()->theLockingInset()) { owner->view()->update(TEXT(), BufferView::SELECT|BufferView::FITCUR); } owner->message(keyseq.printOptions()); } break; // --- Misc ------------------------------------------- case LFUN_EXEC_COMMAND: { vector allCmds; transform(lyxaction.func_begin(), lyxaction.func_end(), back_inserter(allCmds), lyx::firster()); static vector hist; owner->getMiniBuffer()->getString(MiniBuffer::spaces, allCmds, hist); } break; case LFUN_CANCEL: // RVDK_PATCH_5 keyseq.reset(); meta_fake_bit = 0; if (owner->view()->available()) // cancel any selection dispatch(LFUN_MARK_OFF); setMessage(N_("Cancel")); break; case LFUN_META_FAKE: // RVDK_PATCH_5 { meta_fake_bit = Mod1Mask; 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. owner->view()->center(); break; // --- Menus ----------------------------------------------- case LFUN_MENUNEW: menuNew(false); break; case LFUN_MENUNEWTMPLT: menuNew(true); break; case LFUN_CLOSEBUFFER: closeBuffer(); break; case LFUN_MENUWRITE: if (!owner->buffer()->isUnnamed()) { ostringstream s1; s1 << _("Saving document") << ' ' << MakeDisplayPath(owner->buffer()->fileName() + "..."); owner->message(s1.str().c_str()); MenuWrite(owner->view(), owner->buffer()); } else WriteAs(owner->view(), owner->buffer()); break; case LFUN_WRITEAS: WriteAs(owner->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: #if 0 case LFUN_LOFVIEW: case LFUN_LOTVIEW: case LFUN_LOAVIEW: #endif { InsetCommandParams p; #if 0 if (action == LFUN_TOCVIEW) #endif p.setCmdName("tableofcontents"); #if 0 else if (action == LFUN_LOAVIEW) p.setCmdName("listof{algorithm}{List of Algorithms}"); else if (action == LFUN_LOFVIEW) p.setCmdName("listoffigures"); else p.setCmdName("listoftables"); #endif owner->getDialogs()->createTOC(p.getAsString()); break; } case LFUN_DIALOG_TABULAR_INSERT: owner->getDialogs()->showTabularCreate(); break; case LFUN_AUTOSAVE: AutoSave(owner->view()); break; case LFUN_UNDO: owner->view()->menuUndo(); break; case LFUN_REDO: owner->view()->menuRedo(); break; case LFUN_MENUSEARCH: owner->getDialogs()->showSearch(); break; case LFUN_REMOVEERRORS: if (owner->view()->removeAutoInsets()) { owner->view()->redraw(); owner->view()->fitCursor(); } break; case LFUN_DEPTH_MIN: changeDepth(owner->view(), TEXT(false), -1); break; case LFUN_DEPTH_PLUS: changeDepth(owner->view(), TEXT(false), 1); break; case LFUN_FREE: owner->getDialogs()->setUserFreeFont(); break; case LFUN_RECONFIGURE: Reconfigure(owner->view()); break; #if 0 case LFUN_FLOATSOPERATE: if (argument == "openfoot") owner->view()->allFloats(1,0); else if (argument == "closefoot") owner->view()->allFloats(0,0); else if (argument == "openfig") owner->view()->allFloats(1,1); else if (argument == "closefig") owner->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; } owner->prohibitInput(); string const fname = i18nLibFileSearch("doc", arg, "lyx"); if (fname.empty()) { lyxerr << "LyX: unable to find documentation file `" << arg << "'. Bad installation?" << endl; owner->allowInput(); break; } ostringstream str; str << _("Opening help file") << ' ' << MakeDisplayPath(fname) << "..."; owner->message(str.str().c_str()); owner->view()->buffer(bufferlist.loadLyXFile(fname, false)); owner->allowInput(); 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: owner->view()->buffer(bufferlist.getBuffer(argument)); break; case LFUN_FILE_NEW: { // servercmd: argument must be :