1 /* This file is part of
2 * ======================================================
4 * LyX, The Document Processor
6 * Copyright 1995 Matthias Ettrich
7 * Copyright 1995-1999 The LyX Team.
9 * ====================================================== */
18 #pragma implementation
21 #include "lyxlookup.h"
24 #include "bufferlist.h"
25 #include "lyxserver.h"
30 #include "LyXAction.h"
32 #include "insets/insetlatex.h"
34 #include "insets/inseturl.h"
35 #include "insets/insetlatexaccent.h"
36 #include "insets/insettoc.h"
37 #include "insets/insetlof.h"
38 #include "insets/insetloa.h"
39 #include "insets/insetlot.h"
40 #include "insets/insetref.h"
41 #include "insets/insetparent.h"
42 #include "insets/insetindex.h"
43 #include "insets/insetinclude.h"
44 #include "insets/insetbib.h"
45 #include "mathed/formulamacro.h"
47 #include "spellchecker.h" // RVDK_PATCH_5
48 #include "minibuffer.h"
52 #include "lyx_gui_misc.h"
53 #include "support/filetools.h"
54 #include "support/FileInfo.h"
55 #include "support/syscall.h"
56 #include "support/lstrings.h"
57 #include "support/path.h"
58 #include "lyxscreen.h"
63 #include "trans_mgr.h"
64 #include "ImportLaTeX.h"
65 #include "ImportNoweb.h"
68 extern bool cursor_follows_scrollbar;
70 extern void InsertAsciiFile(string const &, bool);
71 extern void math_insert_symbol(char const *);
72 extern Bool math_insert_greek(char const); // why "Bool"?
73 extern BufferList bufferlist;
74 extern LyXServer * lyxserver;
75 extern short greek_kb_flag;
76 extern FD_form_toc * fd_form_toc;
77 extern bool selection_possible;
79 extern kb_keymap * toplevel_keymap;
81 extern void MenuWrite(Buffer *);
82 extern void MenuWriteAs(Buffer *);
83 extern int MenuRunLaTeX(Buffer *);
84 extern int MenuBuildProg(Buffer *);
85 extern int MenuRunChktex(Buffer *);
86 extern bool CreatePostscript(Buffer *, bool);
87 extern void MenuPrint(Buffer *);
88 extern void MenuSendto();
89 extern void QuitLyX();
90 extern void MenuFax(Buffer *);
91 extern void MenuExport(Buffer *, string const &);
92 extern void MenuPasteSelection(char at);
93 extern LyXAction lyxaction;
95 extern tex_accent_struct get_accent(kb_action action);
97 extern void AutoSave();
98 extern void MenuSearch();
99 extern void SetUpdateTimer(float timer = 0.3);
100 extern void FreeUpdateTimer();
101 extern bool PreviewDVI(Buffer *);
102 extern bool PreviewPostscript(Buffer *);
103 extern void MenuInsertLabel(char const *);
104 extern void MenuInsertRef();
105 extern void MenuLayoutCharacter();
106 extern void MenuLayoutParagraph();
107 extern void MenuLayoutDocument();
108 extern void MenuLayoutPaper();
109 extern void MenuLayoutTable(int flag);
110 extern void MenuLayoutQuotes();
111 extern void MenuLayoutPreamble();
112 extern void MenuLayoutSave();
113 extern void bulletForm();
115 extern Buffer * NewLyxFile(string const &);
116 extern void LoadLyXFile(string const &);
117 extern void Reconfigure();
119 extern int current_layout;
120 extern int getISOCodeFromLaTeX(char *);
122 //extern int UnlockInset(UpdatableInset *);
124 extern void ShowLatexLog();
126 extern void UpdateInset(Inset * inset, bool mark_dirty = true);
128 /* === globals =========================================================== */
130 bool LyXFunc::show_sc = true;
133 LyXFunc::LyXFunc(LyXView * o)
137 lyx_dead_action = LFUN_NOACTION;
138 lyx_calling_dead_action = LFUN_NOACTION;
143 // I changed this func slightly. I commented out the ...FinishUndo(),
144 // this means that all places that used to have a moveCursorUpdate, now
145 // have a ...FinishUndo() as the preceeding statement. I have also added
146 // a moveCursorUpdate to some of the functions that updated the cursor, but
147 // that did not show its new position.
149 void LyXFunc::moveCursorUpdate(bool selecting)
151 if (selecting || owner->view()->text->mark_set) {
152 owner->view()->text->SetSelection();
153 owner->view()->getScreen()->ToggleToggle();
154 owner->view()->update(0);
156 owner->view()->update(-2); // this IS necessary
160 owner->view()->getScreen()->ShowCursor();
162 /* ---> Everytime the cursor is moved, show the current font state. */
163 // should this too me moved out of this func?
164 //owner->getMiniBuffer()->Set(CurrentState());
168 int LyXFunc::processKeyEvent(XEvent * ev)
173 XKeyEvent * keyevent = &ev->xkey;
174 KeySym keysym_return;
176 int num_bytes = LyXLookupString(ev, s_r, 10, &keysym_return);
178 if (lyxerr.debugging(Debug::KEY)) {
179 lyxerr << "KeySym is "
180 << XKeysymToString(keysym_return)
182 << keysym_return << "]"
183 << " and num_bytes is "
185 << " the string returned is \""
186 << s_r << '\"' << endl;
188 // Do nothing if we have nothing (JMarc)
189 if (num_bytes == 0 && keysym_return == NoSymbol) {
190 lyxerr[Debug::KEY] << "Empty kbd action (probably composing)"
195 // this function should be used always [asierra060396]
196 if (owner->view()->available() &&
197 owner->view()->the_locking_inset &&
198 keysym_return == XK_Escape) {
199 owner->view()->unlockInset(owner->view()->the_locking_inset);
200 owner->view()->text->CursorRight();
204 // Can we be sure that this will work for all X-Windows
205 // implementations? (Lgb)
206 // This code snippet makes lyx ignore some keys. Perhaps
207 // all of them should be explictly mentioned?
208 if((keysym_return >= XK_Shift_L && keysym_return <= XK_Hyper_R)
209 || keysym_return == XK_Mode_switch || keysym_return == 0x0)
212 // Do a one-deep top-level lookup for
213 // cancel and meta-fake keys. RVDK_PATCH_5
214 cancel_meta_seq.reset();
216 int action = cancel_meta_seq.addkey(keysym_return, keyevent->state
217 &(ShiftMask|ControlMask
220 // When not cancel or meta-fake, do the normal lookup.
221 // Note how the meta_fake Mod1 bit is OR-ed in and reset afterwards.
222 // Mostly, meta_fake_bit = 0. RVDK_PATCH_5.
223 if ( (action != LFUN_CANCEL) && (action != LFUN_META_FAKE) ) {
225 // remove Caps Lock and Mod2 as a modifiers
226 action = keyseq.addkey(keysym_return,
227 (keyevent->state | meta_fake_bit)
228 &(ShiftMask|ControlMask
231 // Dont remove this unless you know what you are doing.
234 if (action == 0) action = LFUN_PREFIX;
236 if (lyxerr.debugging(Debug::KEY)) {
242 << num_bytes << "]" << endl;
245 // already here we know if it any point in going further
246 // why not return already here if action == -1 and
247 // num_bytes == 0? (Lgb)
249 if(keyseq.length > 1 || keyseq.length < -1){
252 owner->getMiniBuffer()->Set(buf);
256 if (keyseq.length < -1) { // unknown key sequence...
260 owner->getMiniBuffer()->Set(_("Unknown sequence:"), buf);
264 char isochar = keyseq.getiso();
265 if (!(keyevent->state&ControlMask) &&
266 !(keyevent->state&Mod1Mask) &&
267 (isochar && keysym_return < 0xF000)) {
270 if (argument.empty()) {
271 lyxerr.debug() << "Empty argument!" << endl;
272 // This can`t possibly be of any use
273 // so we`ll skip the dispatch.
277 if (action == LFUN_SELFINSERT) {
281 bool tmp_sc = show_sc;
283 Dispatch(action, argument.c_str());
290 LyXFunc::func_status LyXFunc::getStatus(int ac) const
293 func_status flag = LyXFunc::OK;
295 Buffer * buf = owner->buffer();
297 if (lyxaction.isPseudoAction(ac))
298 action = lyxaction.retrieveActionArg(ac, argument);
300 action = static_cast<kb_action>(ac);
302 if (action == LFUN_UNKNOWN_ACTION) {
303 setErrorMessage(N_("Unknown action"));
304 return LyXFunc::Unknown;
307 // Check whether we need a buffer
308 if (!lyxaction.funcHasFlag(action, LyXAction::NoBuffer)) {
309 // Yes we need a buffer, do we have one?
312 // Can we use a readonly buffer?
313 if (buf->isReadonly() &&
314 !lyxaction.funcHasFlag(action,
315 LyXAction::ReadOnly)) {
317 setErrorMessage(N_("Document is read-only"));
318 flag |= LyXFunc::Disabled;
322 setErrorMessage(N_("Command not allowed with"
323 "out any document open"));
324 flag |= LyXFunc::Disabled;
328 if (flag & LyXFunc::Disabled)
331 // I would really like to avoid having this switch and rather try to
332 // encode this in the function itself.
333 static bool noLaTeX = lyxrc->latex_command == "none";
334 bool disable = false;
337 disable = noLaTeX || lyxrc->view_dvi_command == "none";
340 disable = noLaTeX || lyxrc->view_ps_command == "none";
347 disable = noLaTeX || lyxrc->print_command == "none";
350 disable = noLaTeX || lyxrc->fax_command == "none";
353 if (argument == "latex")
354 disable = lyxrc->relyx_command == "none";
357 if (argument == "dvi" || argument == "postscript")
359 if (argument == "html")
360 disable = lyxrc->html_command == "none";
363 disable = buf->undostack.empty();
366 disable = buf->redostack.empty();
368 case LFUN_SPELLCHECK:
369 disable = lyxrc->isp_command == "none";
372 disable = lyxrc->chktex_command == "none";
374 case LFUN_LAYOUT_TABLE:
375 disable = ! owner->view()->text->cursor.par->table;
381 flag |= LyXFunc::Disabled;
384 func_status box = LyXFunc::ToggleOff;
385 LyXFont font = owner->view()->text->real_current_font;
388 if (font.emph() == LyXFont::ON)
389 box = LyXFunc::ToggleOn;
392 if (font.noun() == LyXFont::ON)
393 box = LyXFunc::ToggleOn;
396 if (font.series() == LyXFont::BOLD_SERIES)
397 box = LyXFunc::ToggleOn;
400 if (font.latex() == LyXFont::ON)
401 box = LyXFunc::ToggleOn;
414 string LyXFunc::Dispatch(string const & s)
416 // Split command string into command and argument
417 string cmd, line = frontStrip(s);
418 string arg = strip(frontStrip(split(line, cmd, ' ')));
420 return Dispatch(lyxaction.LookupFunc(cmd.c_str()), arg.c_str());
424 string LyXFunc::Dispatch(int ac,
425 char const * do_not_use_this_arg)
430 FL_OBJECT * ob = 0; // This will disapear soon
432 // we have not done anything wrong yet.
434 dispatch_buffer.clear();
436 // if action is a pseudo-action, we need the real action
437 if (lyxaction.isPseudoAction(ac)) {
439 action = static_cast<kb_action>
440 (lyxaction.retrieveActionArg(ac, tmparg));
444 action = static_cast<kb_action>(ac);
445 if (do_not_use_this_arg)
446 argument = do_not_use_this_arg; // except here
449 selection_possible = false;
451 if (owner->view()->available()
452 && owner->view()->getScreen())
453 owner->view()->getScreen()->HideCursor();
455 // We cannot use this function here
456 if (getStatus(action) & Disabled)
457 goto exit_with_message;
459 commandshortcut.clear();
461 if (lyxrc->display_shortcuts && show_sc) {
462 if (action != LFUN_SELFINSERT) {
463 // Put name of command and list of shortcuts
464 // for it in minibuffer
465 string comname = lyxaction.getActionName(action);
467 int pseudoaction = action;
468 bool argsadded = false;
470 if (!argument.empty()) {
471 // If we have the command with argument,
474 lyxaction.searchActionArg(action,
477 if (pseudoaction == -1) {
478 pseudoaction = action;
480 comname += " " + argument;
485 string shortcuts = toplevel_keymap->findbinding(pseudoaction);
487 if (!shortcuts.empty()) {
488 comname += ": " + shortcuts;
489 } else if (!argsadded) {
490 comname += " " + argument;
493 if (!comname.empty()) {
494 comname = strip(comname);
495 commandshortcut = "(" + comname + ')';
496 owner->getMiniBuffer()->Set(commandshortcut);
497 // Here we could even add a small pause,
498 // to annoy the user and make him learn
500 // No! That will just annoy, not teach
501 // anything. The user will read the messages
502 // if they are interested. (Asger)
507 // If in math mode pass the control to
508 // the math inset [asierra060396]
509 if (owner->view()->available() &&
510 owner->view()->the_locking_inset) {
512 || (action == LFUN_UNKNOWN_ACTION
513 && keyseq.length >= -1)) {
514 if (action == LFUN_UNKNOWN_ACTION
515 && argument.empty()) {
516 argument = keyseq.getiso();
518 // Undo/Redo pre 0.13 is a bit tricky for insets.
519 if (action == LFUN_UNDO) {
521 UpdatableInset * inset =
522 owner->view()->the_locking_inset;
523 inset->GetCursorPos(slx, sly);
524 owner->view()->unlockInset(inset);
525 owner->view()->menuUndo();
526 inset = static_cast<UpdatableInset*>(owner->view()->text->cursor.par->GetInset(owner->view()->text->cursor.pos));
528 inset->Edit(slx, sly);
531 if (action == LFUN_REDO) {
533 UpdatableInset * inset = owner->view()->the_locking_inset;
534 inset->GetCursorPos(slx, sly);
535 owner->view()->unlockInset(inset);
536 owner->view()->menuRedo();
537 inset = static_cast<UpdatableInset*>(owner->view()->text->cursor.par->GetInset(owner->view()->text->cursor.pos));
539 inset->Edit(slx, sly);
542 if (owner->view()->the_locking_inset->LocalDispatch(action, argument.c_str()))
545 setMessage(N_("Text mode"));
546 if (action == LFUN_RIGHT || action == -1)
547 owner->view()->text->CursorRight();
548 if (action == LFUN_LEFT || action == LFUN_RIGHT)
555 // --- Misc -------------------------------------------
556 case LFUN_WORDFINDFORWARD :
557 case LFUN_WORDFINDBACKWARD : {
558 static string last_search;
559 string searched_string;
561 if (!argument.empty()) {
562 last_search = argument;
563 searched_string = argument;
565 searched_string = last_search;
568 LyXText * ltCur = owner->view()->text ;
570 if (!searched_string.empty() &&
571 ((action == LFUN_WORDFINDBACKWARD) ?
572 ltCur->SearchBackward(searched_string.c_str()) :
573 ltCur->SearchForward(searched_string.c_str()))) {
575 // ??? What is that ???
576 owner->view()->update(-2);
579 // clear the selection (if there is any)
580 owner->view()->getScreen()->ToggleSelection();
581 owner->view()->text->ClearSelection();
583 // Move cursor so that successive C-s 's will not stand in place.
584 if( action == LFUN_WORDFINDFORWARD )
585 owner->view()->text->CursorRightOneWord();
586 owner->view()->text->FinishUndo();
587 moveCursorUpdate(false);
590 // set the new selection
591 // SetSelectionOverLenChars(owner->view()->currentBuffer()->text, iLenSelected);
592 owner->view()->getScreen()->ToggleSelection(false);
596 // REMOVED : if (owner->view()->getWorkArea()->focus)
597 owner->view()->getScreen()->ShowCursor();
603 if (owner->view()->available()
604 && owner->view()->getScreen()) {
605 owner->view()->update(-2);
608 keyseq.print(buf, true);
609 owner->getMiniBuffer()->Set(buf, string(), string(), 1);
613 // --- Misc -------------------------------------------
614 case LFUN_EXEC_COMMAND:
615 owner->getMiniBuffer()->ExecCommand();
618 case LFUN_CANCEL: // RVDK_PATCH_5
621 if(owner->view()->available())
622 // cancel any selection
623 Dispatch(LFUN_MARK_OFF, 0);
624 setMessage(N_("Cancel"));
627 case LFUN_META_FAKE: // RVDK_PATCH_5
629 meta_fake_bit = Mod1Mask;
631 keyseq.print(buf, true);
632 string res = string("M-") + buf;
633 setMessage(buf); // RVDK_PATCH_5
637 case LFUN_READ_ONLY_TOGGLE:
638 if (owner->buffer()->lyxvc.inUse()) {
639 owner->buffer()->lyxvc.toggleReadOnly();
641 owner->buffer()->setReadonly(
642 !owner->buffer()->isReadonly());
646 case LFUN_CENTER: // this is center and redraw.
647 owner->view()->beforeChange();
648 if (owner->view()->text->cursor.y >
649 owner->view()->getWorkArea()->h / 2) {
650 owner->view()->getScreen()->
651 Draw(owner->view()->text->cursor.y -
652 owner->view()->getWorkArea()->h / 2);
654 owner->view()->getScreen()->
657 owner->view()->update(0);
658 owner->view()->redraw();
662 if (owner->view()->available()) {
663 owner->view()->text->toggleAppendix();
664 owner->view()->update(1);
668 // --- Menus -----------------------------------------------
673 case LFUN_MENUNEWTMPLT:
681 case LFUN_CLOSEBUFFER:
686 MenuWrite(owner->buffer());
689 case LFUN_MENUWRITEAS:
690 MenuWriteAs(owner->buffer());
693 case LFUN_MENURELOAD:
698 PreviewDVI(owner->buffer());
702 PreviewPostscript(owner->buffer());
706 MenuRunLaTeX(owner->buffer());
710 MenuBuildProg(owner->buffer());
714 MenuRunChktex(owner->buffer());
718 CreatePostscript(owner->buffer(), false);
722 MenuPrint(owner->buffer());
726 MenuFax(owner->buffer());
730 MenuExport(owner->buffer(), argument);
735 //needs argument as string
736 string imtyp = argument;
739 if (imtyp == "latex") {
740 doImportLaTeX(false);
743 else if (imtyp == "ascii") {
744 doImportASCII(false);
745 } else if (imtyp == "asciiparagraph") {
748 } else if (imtyp == "noweb") {
751 setErrorMessage(string(N_("Unknown import type: "))
763 if (fd_form_toc->form_toc->visible) {
764 fl_raise_form(fd_form_toc->form_toc);
766 static int ow = -1, oh;
767 fl_show_form(fd_form_toc->form_toc,
769 FL_FREE_SIZE, FL_FULLBORDER,
770 _("Table of Contents"));
772 ow = fd_form_toc->form_toc->w;
773 oh = fd_form_toc->form_toc->h;
775 fl_set_form_minsize(fd_form_toc->form_toc, ow, oh);
779 case LFUN_TOC_INSERT:
781 Inset * new_inset = new InsetTOC(owner->buffer());
782 owner->view()->insertInset(new_inset, "Standard", true);
786 case LFUN_LOF_INSERT:
788 Inset * new_inset = new InsetLOF(owner->buffer());
789 owner->view()->insertInset(new_inset, "Standard", true);
793 case LFUN_LOA_INSERT:
795 Inset * new_inset = new InsetLOA(owner->buffer());
796 owner->view()->insertInset(new_inset, "Standard", true);
800 case LFUN_LOT_INSERT:
802 Inset * new_inset = new InsetLOT(owner->buffer());
803 owner->view()->insertInset(new_inset, "Standard", true);
820 owner->view()->menuUndo();
824 owner->view()->menuRedo();
827 case LFUN_MENUSEARCH:
832 owner->view()->paste();
835 case LFUN_PASTESELECTION:
838 if (argument == "paragraph") asPara = true;
839 MenuPasteSelection(asPara);
844 owner->view()->cut();
848 owner->view()->copy();
851 case LFUN_LAYOUT_COPY:
852 owner->view()->copyEnvironment();
855 case LFUN_LAYOUT_PASTE:
856 owner->view()->pasteEnvironment();
860 owner->view()->gotoError();
863 case LFUN_REMOVEERRORS:
864 if (owner->view()->removeAutoInsets()) {
865 owner->view()->redraw();
866 owner->view()->fitCursor();
867 owner->view()->updateScrollbar();
872 owner->view()->gotoNote();
876 owner->view()->openStuff();
879 case LFUN_HYPHENATION:
880 owner->view()->hyphenationPoint();
884 owner->view()->ldots();
887 case LFUN_END_OF_SENTENCE:
888 owner->view()->endOfSentenceDot();
891 case LFUN_MENU_SEPARATOR:
892 owner->view()->menuSeparator();
896 owner->view()->hfill();
907 case LFUN_DEPTH_PLUS:
923 case LFUN_RECONFIGURE:
928 if (owner->view()->available()
929 && !owner->view()->text->selection
930 && owner->view()->text->cursor.par->footnoteflag
931 != LyXParagraph::NO_FOOTNOTE)
932 { // only melt footnotes with FOOTMELT, not margins etc
933 if(owner->view()->text->cursor.par->footnotekind == LyXParagraph::FOOTNOTE)
940 case LFUN_MARGINMELT:
941 if (owner->view()->available()
942 && !owner->view()->text->selection
943 && owner->view()->text->cursor.par->footnoteflag
944 != LyXParagraph::NO_FOOTNOTE) {
946 if(owner->view()->text->cursor.par->footnotekind == LyXParagraph::MARGIN)
953 // --- version control -------------------------------
954 case LFUN_VC_REGISTER:
956 if (!owner->buffer()->lyxvc.inUse())
957 owner->buffer()->lyxvc.registrer();
961 case LFUN_VC_CHECKIN:
963 if (owner->buffer()->lyxvc.inUse()
964 && !owner->buffer()->isReadonly())
965 owner->buffer()->lyxvc.checkIn();
969 case LFUN_VC_CHECKOUT:
971 if (owner->buffer()->lyxvc.inUse()
972 && owner->buffer()->isReadonly())
973 owner->buffer()->lyxvc.checkOut();
979 owner->buffer()->lyxvc.revert();
985 owner->buffer()->lyxvc.undoLast();
989 case LFUN_VC_HISTORY:
991 owner->buffer()->lyxvc.showLog();
995 // --- buffers ----------------------------------------
996 case LFUN_PREVBUFFER:
998 #warning fix this please
1000 // it is the LyXView or the BufferView that should
1001 // remember the previous buffer, not bufferlist.
1002 // if (owner->view()->available()){
1003 // owner->view()->beforeChange();
1004 // owner->buffer()->update(-2);
1006 // owner->view()->setBuffer(bufferlist.prev());
1009 // resizeCurrentBufferPseudoExpose();
1012 case LFUN_FILE_INSERT:
1014 MenuInsertLyXFile(argument);
1018 case LFUN_FILE_INSERT_ASCII:
1020 bool asPara = (argument == "paragraph");
1021 InsertAsciiFile(string(), asPara);
1027 // servercmd: argument must be <file>:<template>
1028 Buffer * tmpbuf = NewLyxFile(argument);
1030 owner->view()->buffer(tmpbuf);
1034 case LFUN_FILE_OPEN:
1035 owner->view()->buffer(bufferlist.loadLyXFile(argument));
1038 case LFUN_LATEX_LOG:
1044 lyxerr.debug() << "LFUN_LAYOUTNO: (arg) " << argument << endl;
1045 int sel = strToInt(argument);
1046 lyxerr.debug() << "LFUN_LAYOUTNO: (sel) "<< sel << endl;
1048 // Should this give a setMessage instead?
1050 return string(); // illegal argument
1052 --sel; // sel 1..., but layout 0...
1054 // Pretend we got the name instead.
1055 Dispatch(int(LFUN_LAYOUT),
1056 textclasslist.NameOfLayout(owner->view()->
1065 lyxerr.debug() << "LFUN_LAYOUT: (arg) "
1066 << argument << endl;
1068 // Derive layout number from given argument (string)
1069 // and current buffer's textclass (number). */
1071 textclasslist.NumberOfLayout(owner->
1077 // see if we found the layout number:
1078 if (layoutno == -1) {
1079 setErrorMessage(string(N_("Layout ")) + argument +
1084 if (current_layout != layoutno) {
1085 owner->view()->getScreen()->HideCursor();
1086 current_layout = layoutno;
1087 owner->view()->update(-2);
1088 owner->view()->text->
1089 SetLayout(layoutno);
1090 owner->getToolbar()->combox->
1091 select(owner->view()->
1094 owner->view()->update(1);
1099 case LFUN_LAYOUT_DOCUMENT:
1100 MenuLayoutDocument();
1103 case LFUN_LAYOUT_PARAGRAPH:
1104 MenuLayoutParagraph();
1107 case LFUN_LAYOUT_CHARACTER:
1108 MenuLayoutCharacter();
1111 case LFUN_LAYOUT_TABLE:
1114 if (argument == "true") flag = 1;
1115 MenuLayoutTable(flag);
1119 case LFUN_LAYOUT_PAPER:
1123 case LFUN_LAYOUT_QUOTES:
1127 case LFUN_LAYOUT_PREAMBLE:
1128 MenuLayoutPreamble();
1131 case LFUN_LAYOUT_SAVE_DEFAULT:
1135 case LFUN_DROP_LAYOUTS_CHOICE:
1136 owner->getToolbar()->combox->Show();
1167 case LFUN_UNDERLINE:
1171 case LFUN_FONT_SIZE:
1172 FontSizeCB(argument);
1175 case LFUN_FONT_STATE:
1176 setMessage(CurrentState());
1179 case LFUN_UPCASE_WORD:
1180 owner->view()->update(-2);
1182 owner->view()->text->ChangeWordCase(LyXText::text_uppercase);
1183 owner->view()->update(1);
1187 case LFUN_LOWCASE_WORD:
1188 owner->view()->update(-2);
1190 owner->view()->text->ChangeWordCase(LyXText::text_lowercase);
1191 owner->view()->update(1);
1195 case LFUN_CAPITALIZE_WORD:
1196 owner->view()->update(-2);
1198 owner->view()->text->ChangeWordCase(LyXText::text_capitalization);
1199 owner->view()->update(1);
1203 case LFUN_INSERT_LABEL:
1204 MenuInsertLabel(argument.c_str());
1207 case LFUN_INSERT_REF:
1211 case LFUN_REFTOGGLE:
1214 static_cast<InsetRef*>(getInsetByCode(Inset::REF_CODE));
1216 if (inset->getFlag() == InsetRef::REF)
1217 inset->setFlag(InsetRef::PAGE_REF);
1219 inset->setFlag(InsetRef::REF);
1222 setErrorMessage(N_("No cross-reference to toggle"));
1229 owner->view()->restorePosition();
1235 string label(argument);
1236 if (label.empty()) {
1238 static_cast<InsetRef*>(getInsetByCode(Inset::REF_CODE));
1240 label = inset->getContents();
1243 if (!label.empty()) {
1244 owner->view()->savePosition();
1245 owner->view()->gotoLabel(label.c_str());
1250 case LFUN_MENU_OPEN_BY_NAME:
1251 owner->getMenus()->openByName(argument);
1252 break; // RVDK_PATCH_5
1254 case LFUN_SPELLCHECK:
1255 if (lyxrc->isp_command != "none")
1257 break; // RVDK_PATCH_5
1259 // --- Cursor Movements -----------------------------
1262 LyXText * tmptext = owner->view()->text;
1263 if(!tmptext->mark_set)
1264 owner->view()->beforeChange();
1265 owner->view()->update(-2);
1266 if (tmptext->cursor.pos < tmptext->cursor.par->Last()
1267 && tmptext->cursor.par->GetChar(tmptext->cursor.pos)
1268 == LyXParagraph::META_INSET
1269 && tmptext->cursor.par->GetInset(tmptext->cursor.pos)
1270 && tmptext->cursor.par->GetInset(tmptext->cursor.pos)->Editable() == 2){
1271 Inset * tmpinset = tmptext->cursor.par->GetInset(tmptext->cursor.pos);
1272 setMessage(tmpinset->EditMessage());
1273 tmpinset->Edit(0, 0);
1276 tmptext->CursorRight();
1277 owner->view()->text->FinishUndo();
1278 moveCursorUpdate(false);
1279 owner->getMiniBuffer()->Set(CurrentState());
1285 // This is soooo ugly. Isn`t it possible to make
1286 // it simpler? (Lgb)
1287 LyXText * txt = owner->view()->text;
1288 if(!txt->mark_set) owner->view()->beforeChange();
1289 owner->view()->update(-2);
1291 if (txt->cursor.pos < txt->cursor.par->Last()
1292 && txt->cursor.par->GetChar(txt->cursor.pos)
1293 == LyXParagraph::META_INSET
1294 && txt->cursor.par->GetInset(txt->cursor.pos)
1295 && txt->cursor.par->GetInset(txt->cursor.pos)->Editable() == 2) {
1296 Inset * tmpinset = txt->cursor.par->GetInset(txt->cursor.pos);
1297 setMessage(tmpinset->EditMessage());
1298 tmpinset->Edit(tmpinset->Width(txt->GetFont(txt->cursor.par,
1299 txt->cursor.pos)), 0);
1302 owner->view()->text->FinishUndo();
1303 moveCursorUpdate(false);
1304 owner->getMiniBuffer()->Set(CurrentState());
1309 if(!owner->view()->text->mark_set) owner->view()->beforeChange();
1310 owner->view()->update(-3);
1311 owner->view()->text->CursorUp();
1312 owner->view()->text->FinishUndo();
1313 moveCursorUpdate(false);
1314 owner->getMiniBuffer()->Set(CurrentState());
1318 if(!owner->view()->text->mark_set)
1319 owner->view()->beforeChange();
1320 owner->view()->update(-3);
1321 owner->view()->text->CursorDown();
1322 owner->view()->text->FinishUndo();
1323 moveCursorUpdate(false);
1324 owner->getMiniBuffer()->Set(CurrentState());
1327 case LFUN_UP_PARAGRAPH:
1328 if(!owner->view()->text->mark_set)
1329 owner->view()->beforeChange();
1330 owner->view()->update(-3);
1331 owner->view()->text->CursorUpParagraph();
1332 owner->view()->text->FinishUndo();
1333 moveCursorUpdate(false);
1334 owner->getMiniBuffer()->Set(CurrentState());
1337 case LFUN_DOWN_PARAGRAPH:
1338 if(!owner->view()->text->mark_set)
1339 owner->view()->beforeChange();
1340 owner->view()->update(-3);
1341 owner->view()->text->CursorDownParagraph();
1342 owner->view()->text->FinishUndo();
1343 moveCursorUpdate(false);
1344 owner->getMiniBuffer()->Set(CurrentState());
1348 if(!owner->view()->text->mark_set)
1349 owner->view()->beforeChange();
1350 owner->view()->update(-3);
1351 owner->view()->cursorPrevious();
1352 owner->view()->text->FinishUndo();
1353 moveCursorUpdate(false);
1354 owner->getMiniBuffer()->Set(CurrentState());
1358 if(!owner->view()->text->mark_set)
1359 owner->view()->beforeChange();
1360 owner->view()->update(-3);
1361 owner->view()->cursorNext();
1362 owner->view()->text->FinishUndo();
1363 moveCursorUpdate(false);
1364 owner->getMiniBuffer()->Set(CurrentState());
1368 if(!owner->view()->text->mark_set)
1369 owner->view()->beforeChange();
1370 owner->view()->update(-2);
1371 owner->view()->text->CursorHome();
1372 owner->view()->text->FinishUndo();
1373 moveCursorUpdate(false);
1374 owner->getMiniBuffer()->Set(CurrentState());
1378 if(!owner->view()->text->mark_set)
1379 owner->view()->beforeChange();
1380 owner->view()->update(-2);
1381 owner->view()->text->CursorEnd();
1382 owner->view()->text->FinishUndo();
1383 moveCursorUpdate(false);
1384 owner->getMiniBuffer()->Set(CurrentState());
1388 if(!owner->view()->text->mark_set)
1389 owner->view()->beforeChange();
1390 owner->view()->update(-2);
1391 owner->view()->text->CursorTab();
1392 owner->view()->text->FinishUndo();
1393 moveCursorUpdate(false);
1394 owner->getMiniBuffer()->Set(CurrentState());
1397 case LFUN_WORDRIGHT:
1398 if(!owner->view()->text->mark_set)
1399 owner->view()->beforeChange();
1400 owner->view()->update(-2);
1401 owner->view()->text->CursorRightOneWord();
1402 owner->view()->text->FinishUndo();
1403 moveCursorUpdate(false);
1404 owner->getMiniBuffer()->Set(CurrentState());
1408 if(!owner->view()->text->mark_set)
1409 owner->view()->beforeChange();
1410 owner->view()->update(-2);
1411 owner->view()->text->CursorLeftOneWord();
1412 owner->view()->text->FinishUndo();
1413 moveCursorUpdate(false);
1414 owner->getMiniBuffer()->Set(CurrentState());
1417 case LFUN_BEGINNINGBUF:
1418 if(!owner->view()->text->mark_set)
1419 owner->view()->beforeChange();
1420 owner->view()->update(-2);
1421 owner->view()->text->CursorTop();
1422 owner->view()->text->FinishUndo();
1423 moveCursorUpdate(false);
1424 owner->getMiniBuffer()->Set(CurrentState());
1428 if(!owner->view()->text->mark_set)
1429 owner->view()->beforeChange();
1430 owner->view()->update(-2);
1431 owner->view()->text->CursorBottom();
1432 owner->view()->text->FinishUndo();
1433 moveCursorUpdate(false);
1434 owner->getMiniBuffer()->Set(CurrentState());
1438 /* cursor selection ---------------------------- */
1440 owner->view()->update(-2);
1441 owner->view()->text->CursorRight();
1442 owner->view()->text->FinishUndo();
1443 moveCursorUpdate(true);
1444 owner->getMiniBuffer()->Set(CurrentState());
1448 owner->view()->update(-2);
1449 owner->view()->text->CursorLeft();
1450 owner->view()->text->FinishUndo();
1451 moveCursorUpdate(true);
1452 owner->getMiniBuffer()->Set(CurrentState());
1456 owner->view()->update(-2);
1457 owner->view()->text->CursorUp();
1458 owner->view()->text->FinishUndo();
1459 moveCursorUpdate(true);
1460 owner->getMiniBuffer()->Set(CurrentState());
1464 owner->view()->update(-2);
1465 owner->view()->text->CursorDown();
1466 owner->view()->text->FinishUndo();
1467 moveCursorUpdate(true);
1468 owner->getMiniBuffer()->Set(CurrentState());
1471 case LFUN_UP_PARAGRAPHSEL:
1472 owner->view()->update(-2);
1473 owner->view()->text->CursorUpParagraph();
1474 owner->view()->text->FinishUndo();
1475 moveCursorUpdate(true);
1476 owner->getMiniBuffer()->Set(CurrentState());
1479 case LFUN_DOWN_PARAGRAPHSEL:
1480 owner->view()->update(-2);
1481 owner->view()->text->CursorDownParagraph();
1482 owner->view()->text->FinishUndo();
1483 moveCursorUpdate(true);
1484 owner->getMiniBuffer()->Set(CurrentState());
1488 owner->view()->update(-2);
1489 owner->view()->cursorPrevious();
1490 owner->view()->text->FinishUndo();
1491 moveCursorUpdate(true);
1492 owner->getMiniBuffer()->Set(CurrentState());
1496 owner->view()->update(-2);
1497 owner->view()->cursorNext();
1498 owner->view()->text->FinishUndo();
1499 moveCursorUpdate(true);
1500 owner->getMiniBuffer()->Set(CurrentState());
1504 owner->view()->update(-2);
1505 owner->view()->text->CursorHome();
1506 owner->view()->text->FinishUndo();
1507 moveCursorUpdate(true);
1508 owner->getMiniBuffer()->Set(CurrentState());
1512 owner->view()->update(-2);
1513 owner->view()->text->CursorEnd();
1514 owner->view()->text->FinishUndo();
1515 moveCursorUpdate(true);
1516 owner->getMiniBuffer()->Set(CurrentState());
1519 case LFUN_WORDRIGHTSEL:
1520 owner->view()->update(-2);
1521 owner->view()->text->CursorRightOneWord();
1522 owner->view()->text->FinishUndo();
1523 moveCursorUpdate(true);
1524 owner->getMiniBuffer()->Set(CurrentState());
1527 case LFUN_WORDLEFTSEL:
1528 owner->view()->update(-2);
1529 owner->view()->text->CursorLeftOneWord();
1530 owner->view()->text->FinishUndo();
1531 moveCursorUpdate(true);
1532 owner->getMiniBuffer()->Set(CurrentState());
1535 case LFUN_BEGINNINGBUFSEL:
1536 owner->view()->update(-2);
1537 owner->view()->text->CursorTop();
1538 owner->view()->text->FinishUndo();
1539 moveCursorUpdate(true);
1540 owner->getMiniBuffer()->Set(CurrentState());
1543 case LFUN_ENDBUFSEL:
1544 owner->view()->update(-2);
1545 owner->view()->text->CursorBottom();
1546 owner->view()->text->FinishUndo();
1547 moveCursorUpdate(true);
1548 owner->getMiniBuffer()->Set(CurrentState());
1551 // --- text changing commands ------------------------
1552 case LFUN_BREAKLINE:
1553 owner->view()->beforeChange();
1554 owner->view()->text->InsertChar(LyXParagraph::META_NEWLINE);
1555 owner->view()->smallUpdate(1);
1556 SetUpdateTimer(0.01);
1557 moveCursorUpdate(false);
1560 case LFUN_PROTECTEDSPACE:
1561 owner->view()->beforeChange();
1562 owner->view()->text->
1563 InsertChar(LyXParagraph::META_PROTECTED_SEPARATOR);
1564 owner->view()->smallUpdate(1);
1566 moveCursorUpdate(false);
1570 if(owner->view()->text->mark_set) {
1571 owner->view()->beforeChange();
1572 owner->view()->update(0);
1573 setMessage(N_("Mark removed"));
1575 owner->view()->beforeChange();
1576 owner->view()->text->mark_set = 1;
1577 owner->view()->update(0);
1578 setMessage(N_("Mark set"));
1580 owner->view()->text->sel_cursor =
1581 owner->view()->text->cursor;
1586 if (!owner->view()->text->selection) {
1587 owner->view()->text->Delete();
1588 owner->view()->text->sel_cursor =
1589 owner->view()->text->cursor;
1590 owner->view()->smallUpdate(1);
1591 // It is possible to make it a lot faster still
1592 // just comment out the lone below...
1593 owner->view()->getScreen()->ShowCursor();
1595 owner->view()->cut();
1600 case LFUN_DELETE_SKIP:
1602 // Reverse the effect of LFUN_BREAKPARAGRAPH_SKIP.
1604 LyXCursor cursor = owner->view()->text->cursor;
1607 if (!owner->view()->text->selection) {
1608 if (cursor.pos == cursor.par->Last()) {
1609 owner->view()->text->CursorRight();
1610 cursor = owner->view()->text->cursor;
1612 && !(cursor.par->added_space_top
1613 == VSpace (VSpace::NONE))) {
1614 owner->view()->text->SetParagraph
1615 (cursor.par->line_top,
1616 cursor.par->line_bottom,
1617 cursor.par->pagebreak_top,
1618 cursor.par->pagebreak_bottom,
1619 VSpace(VSpace::NONE),
1620 cursor.par->added_space_bottom,
1622 cursor.par->labelwidthstring, 0);
1623 owner->view()->text->CursorLeft();
1624 owner->view()->update (1);
1626 owner->view()->text->CursorLeft();
1627 owner->view()->text->Delete();
1628 owner->view()->text->sel_cursor =
1629 owner->view()->text->cursor;
1630 owner->view()->smallUpdate(1);
1633 owner->view()->text->Delete();
1634 owner->view()->text->sel_cursor =
1635 owner->view()->text->cursor;
1636 owner->view()->smallUpdate(1);
1639 owner->view()->cut();
1645 /* -------> Delete word forward. */
1646 case LFUN_DELETE_WORD_FORWARD:
1647 owner->view()->update(-2);
1649 owner->view()->text->DeleteWordForward();
1650 owner->view()->update( 1 );
1652 moveCursorUpdate(false);
1655 /* -------> Delete word backward. */
1656 case LFUN_DELETE_WORD_BACKWARD:
1657 owner->view()->update(-2);
1659 owner->view()->text->DeleteWordBackward();
1660 owner->view()->update( 1 );
1662 moveCursorUpdate(false);
1665 /* -------> Kill to end of line. */
1666 case LFUN_DELETE_LINE_FORWARD:
1668 owner->view()->update(-2);
1669 owner->view()->text->DeleteLineForward();
1670 owner->view()->update( 1 );
1672 moveCursorUpdate(false);
1675 /* -------> Set mark off. */
1677 owner->view()->beforeChange();
1678 owner->view()->update(0);
1679 owner->view()->text->sel_cursor =
1680 owner->view()->text->cursor;
1681 setMessage(N_("Mark off"));
1684 /* -------> Set mark on. */
1686 owner->view()->beforeChange();
1687 owner->view()->text->mark_set = 1;
1688 owner->view()->update( 0 );
1689 owner->view()->text->sel_cursor =
1690 owner->view()->text->cursor;
1691 setMessage(N_("Mark on"));
1694 case LFUN_BACKSPACE:
1697 if (!owner->view()->text->selection) {
1698 if (owner->getIntl()->getTrans()->backspace()) {
1699 owner->view()->text->Backspace();
1700 owner->view()->text->sel_cursor =
1701 owner->view()->text->cursor;
1702 owner->view()->smallUpdate(1);
1703 // It is possible to make it a lot faster still
1704 // just comment out the lone below...
1705 owner->view()->getScreen()->ShowCursor();
1708 owner->view()->cut();
1714 case LFUN_BACKSPACE_SKIP:
1716 // Reverse the effect of LFUN_BREAKPARAGRAPH_SKIP.
1718 LyXCursor cursor = owner->view()->text->cursor;
1721 if (!owner->view()->text->selection) {
1723 && !(cursor.par->added_space_top
1724 == VSpace (VSpace::NONE))) {
1725 owner->view()->text->SetParagraph
1726 (cursor.par->line_top,
1727 cursor.par->line_bottom,
1728 cursor.par->pagebreak_top,
1729 cursor.par->pagebreak_bottom,
1730 VSpace(VSpace::NONE), cursor.par->added_space_bottom,
1732 cursor.par->labelwidthstring, 0);
1733 owner->view()->update (1);
1735 owner->view()->text->Backspace();
1736 owner->view()->text->sel_cursor
1738 owner->view()->smallUpdate (1);
1741 owner->view()->cut();
1746 case LFUN_BREAKPARAGRAPH:
1748 owner->view()->beforeChange();
1749 owner->view()->text->BreakParagraph(0);
1750 owner->view()->smallUpdate(1);
1751 SetUpdateTimer(0.01);
1752 owner->view()->text->sel_cursor =
1753 owner->view()->text->cursor;
1757 case LFUN_BREAKPARAGRAPHKEEPLAYOUT:
1759 owner->view()->beforeChange();
1760 owner->view()->text->BreakParagraph(1);
1761 owner->view()->smallUpdate(1);
1762 SetUpdateTimer(0.01);
1763 owner->view()->text->sel_cursor =
1764 owner->view()->text->cursor;
1768 case LFUN_BREAKPARAGRAPH_SKIP:
1770 // When at the beginning of a paragraph, remove
1771 // indentation and add a "defskip" at the top.
1772 // Otherwise, do the same as LFUN_BREAKPARAGRAPH.
1774 LyXCursor cursor = owner->view()->text->cursor;
1776 owner->view()->beforeChange();
1777 if (cursor.pos == 0) {
1778 if (cursor.par->added_space_top == VSpace(VSpace::NONE)) {
1779 owner->view()->text->SetParagraph
1780 (cursor.par->line_top,
1781 cursor.par->line_bottom,
1782 cursor.par->pagebreak_top,
1783 cursor.par->pagebreak_bottom,
1784 VSpace(VSpace::DEFSKIP), cursor.par->added_space_bottom,
1786 cursor.par->labelwidthstring, 1);
1787 owner->view()->update(1);
1791 owner->view()->text->BreakParagraph(0);
1792 owner->view()->smallUpdate(1);
1794 SetUpdateTimer(0.01);
1795 owner->view()->text->sel_cursor = cursor;
1800 owner->view()->beforeChange();
1801 owner->view()->text->InsertChar('\"'); // This " matches the single quote in the code
1802 owner->view()->smallUpdate(1);
1804 moveCursorUpdate(false);
1810 InsetCommand * new_inset;
1811 if (action == LFUN_HTMLURL)
1812 new_inset = new InsetUrl("htmlurl", "", "");
1814 new_inset = new InsetUrl("url", "", "");
1815 owner->view()->insertInset(new_inset);
1816 new_inset->Edit(0, 0);
1820 // --- lyxserver commands ----------------------------
1822 case LFUN_CHARATCURSOR:
1824 LyXParagraph::size_type pos =
1825 owner->view()->text->cursor.pos;
1826 if(pos < owner->view()->text->cursor.par->size())
1827 dispatch_buffer = owner->view()->text->
1828 cursor.par->text[pos];
1830 dispatch_buffer = "EOF";
1836 tostr(owner->view()->text->cursor.x) + ' '
1837 + tostr(owner->view()->text->cursor.y);
1844 sscanf(argument.c_str(), " %d %ld", &x, &y);
1845 owner->view()->text->SetCursorFromCoordinates(x, y);
1849 case LFUN_GETLAYOUT:
1851 tostr(owner->view()->text->cursor.par->layout);
1856 LyXFont * font = &(owner->view()->text->current_font);
1857 if(font->shape() == LyXFont::ITALIC_SHAPE)
1858 dispatch_buffer = 'E';
1859 else if(font->shape() == LyXFont::SMALLCAPS_SHAPE)
1860 dispatch_buffer = 'N';
1862 dispatch_buffer = '0';
1869 LyXFont * font = &(owner->view()->text->current_font);
1870 if(font->latex() == LyXFont::ON)
1871 dispatch_buffer = 'L';
1873 dispatch_buffer = '0';
1878 setMessage(owner->buffer()->fileName());
1879 lyxerr.debug() << "FNAME["
1880 << owner->buffer()->fileName()
1888 dispatch_buffer = buf;
1889 lyxserver->notifyClient(dispatch_buffer);
1893 case LFUN_GOTOFILEROW:
1895 char file_name[100];
1897 sscanf(argument.c_str(), " %s %d", file_name, &row);
1899 // Must replace extension of the file to be .lyx and get full path
1900 string s = ChangeExtension(string(file_name), ".lyx", false);
1902 // Either change buffer or load the file
1903 if (bufferlist.exists(s))
1904 owner->view()->buffer(bufferlist.getBuffer(s));
1906 owner->view()->buffer(bufferlist.loadLyXFile(s));
1909 owner->view()->setCursorFromRow(row);
1912 owner->view()->beforeChange();
1913 if (owner->view()->text->cursor.y >
1914 owner->view()->getWorkArea()->h / 2) {
1915 owner->view()->getScreen()->
1916 Draw(owner->view()->text->cursor.y -
1917 owner->view()->getWorkArea()->h/2);
1919 owner->view()->getScreen()->
1922 owner->view()->update(0);
1923 owner->view()->redraw();
1930 int qa = lyxaction.LookupFunc(argument.c_str());
1931 setMessage(lyxaction.helpText(static_cast<kb_action>(qa)));
1935 // --- accented characters ---------------------------
1938 case LFUN_CIRCUMFLEX:
1948 case LFUN_SPECIAL_CARON:
1951 case LFUN_HUNG_UMLAUT:
1957 if (keyseq.length == -1 && keyseq.getiso() != 0)
1958 c = keyseq.getiso();
1962 owner->getIntl()->getTrans()->
1963 deadkey(c, get_accent(action).accent,
1964 owner->view()->text);
1966 // Need to reset, in case the minibuffer calls these
1971 // copied verbatim from do_accent_char
1972 owner->view()->smallUpdate(1);
1974 owner->view()->text->sel_cursor =
1975 owner->view()->text->cursor;
1979 // --- toolbar ----------------------------------
1980 case LFUN_PUSH_TOOLBAR:
1982 int nth = strToInt(argument);
1983 if (lyxerr.debugging(Debug::TOOLBAR)) {
1984 lyxerr << "LFUN_PUSH_TOOLBAR: argument = `"
1985 << argument << "'\n"
1986 << "LFUN_PUSH_TOOLBAR: nth = `"
1987 << nth << "'" << endl;
1992 setErrorMessage(N_("Push-toolbar needs argument > 0"));
1994 owner->getToolbar()->push(nth);
1999 case LFUN_ADD_TO_TOOLBAR:
2001 if (lyxerr.debugging(Debug::TOOLBAR)) {
2002 lyxerr << "LFUN_ADD_TO_TOOLBAR:"
2003 "argument = `" << argument << '\'' << endl;
2005 string tmp(argument);
2006 //lyxerr <<string("Argument: ") + argument);
2007 //lyxerr <<string("Tmp : ") + tmp);
2010 setErrorMessage(N_("Usage: toolbar-add-to <LyX command>"));
2012 owner->getToolbar()->add(argument, false);
2013 owner->getToolbar()->set();
2018 // --- insert characters ----------------------------------------
2020 case LFUN_INSERT_INSET_LATEX:
2022 Inset * new_inset = new InsetLatex(argument);
2023 owner->buffer()->insertInset(new_inset);
2027 // --- Mathed stuff. If we are here, there is no locked inset yet.
2032 if (!greek_kb_flag) {
2034 setMessage(N_("Math greek mode on"));
2041 case LFUN_GREEK_TOGGLE:
2043 greek_kb_flag = greek_kb_flag ? 0 : 2;
2044 if (greek_kb_flag) {
2045 setMessage(N_("Math greek keyboard on"));
2047 setMessage(N_("Math greek keyboard off"));
2052 case LFUN_MATH_DELIM:
2053 case LFUN_INSERT_MATRIX:
2055 if (owner->view()->available()) {
2057 open_new_inset(new InsetFormula(false));
2059 the_locking_inset->LocalDispatch(action, argument.c_str());
2064 case LFUN_INSERT_MATH:
2066 math_insert_symbol(argument.c_str());
2070 case LFUN_MATH_DISPLAY:
2072 if (owner->view()->available())
2073 owner->view()->open_new_inset(new InsetFormula(true));
2077 case LFUN_MATH_MACRO:
2079 if (owner->view()->available()) {
2082 setErrorMessage(N_("Missing argument"));
2084 string s1 = token(s, ' ', 1);
2085 int na = s1.empty() ? 0: atoi(s1.c_str());
2087 open_new_inset(new InsetFormulaMacro(token(s, ' ', 0), na));
2093 case LFUN_MATH_MODE: // Open or create a math inset
2096 if (owner->view()->available())
2097 owner->view()->open_new_inset(new InsetFormula);
2098 setMessage(N_("Math editor mode"));
2102 case LFUN_MATH_NUMBER:
2103 case LFUN_MATH_LIMITS:
2105 setErrorMessage(N_("This is only allowed in math mode!"));
2109 case LFUN_INSERT_CITATION:
2111 InsetCitation * new_inset = new InsetCitation();
2113 // The note, if any, must be after the key, delimited
2114 // by a | so both key and remark can have spaces.
2115 if (!argument.empty()) {
2116 string lsarg(argument);
2117 if (contains(lsarg, "|")) {
2118 new_inset->setContents(token(lsarg, '|', 0));
2119 new_inset->setOptions(token(lsarg, '|', 1));
2121 new_inset->setContents(lsarg);
2122 owner->view()->insertInset(new_inset);
2124 owner->view()->insertInset(new_inset);
2125 new_inset->Edit(0, 0);
2130 case LFUN_INSERT_BIBTEX:
2132 // ale970405+lasgoutt970425
2133 // The argument can be up to two tokens separated
2134 // by a space. The first one is the bibstyle.
2135 string lsarg(argument);
2136 string bibstyle = token(lsarg, ' ', 1);
2137 if (bibstyle.empty())
2139 InsetBibtex * new_inset
2140 = new InsetBibtex(token(lsarg, ' ', 0),
2144 owner->view()->insertInset(new_inset);
2145 if (lsarg.empty()) {
2146 new_inset->Edit(0, 0);
2151 // BibTeX data bases
2152 case LFUN_BIBDB_ADD:
2154 InsetBibtex * inset =
2155 static_cast<InsetBibtex*>(getInsetByCode(Inset::BIBTEX_CODE));
2157 inset->addDatabase(argument);
2162 case LFUN_BIBDB_DEL:
2164 InsetBibtex * inset =
2165 static_cast<InsetBibtex*>(getInsetByCode(Inset::BIBTEX_CODE));
2167 inset->delDatabase(argument);
2172 case LFUN_BIBTEX_STYLE:
2174 InsetBibtex * inset =
2175 static_cast<InsetBibtex*>(getInsetByCode(Inset::BIBTEX_CODE));
2177 inset->setOptions(argument);
2182 case LFUN_INDEX_INSERT:
2183 case LFUN_INDEX_INSERT_LAST:
2185 // Can't do that at the beginning of a paragraph.
2186 if (owner->view()->text->cursor.pos - 1 < 0)
2189 InsetIndex * new_inset = new InsetIndex();
2190 if (!argument.empty()) {
2191 string lsarg(argument);
2192 new_inset->setContents(lsarg);
2193 owner->view()->insertInset(new_inset);
2196 //get the current word for an argument
2197 LyXParagraph::size_type lastpos =
2198 owner->view()->text->cursor.pos - 1;
2199 // Get the current word. note that this must be done
2200 // before inserting the inset, or the inset will
2202 string curstring(owner->view()
2203 ->text->cursor.par->GetWord(lastpos));
2205 //make the new inset and write the current word into it
2206 InsetIndex * new_inset = new InsetIndex();
2208 new_inset->setContents(curstring);
2210 //don't edit it if the call was to INSERT_LAST
2211 if(action != LFUN_INDEX_INSERT_LAST) {
2212 new_inset->Edit(0, 0);
2214 //it looks blank on the screen unless
2215 //we do something. put it here.
2217 // move the cursor to the returned value of lastpos
2218 // but only for the auto-insert
2219 owner->view()->text->cursor.pos = lastpos;
2222 //put the new inset into the buffer.
2223 // there should be some way of knowing the user
2224 //cancelled & avoiding this, but i don't know how
2225 owner->view()->insertInset(new_inset);
2230 case LFUN_INDEX_PRINT:
2232 Inset * new_inset = new InsetPrintIndex(owner->buffer());
2233 owner->view()->insertInset(new_inset, "Standard", true);
2237 case LFUN_PARENTINSERT:
2239 lyxerr << "arg " << argument << endl;
2240 Inset * new_inset = new InsetParent(argument, owner->buffer());
2241 owner->view()->insertInset(new_inset, "Standard", true);
2245 case LFUN_CHILDINSERT:
2247 Inset * new_inset = new InsetInclude(argument,
2249 owner->view()->insertInset(new_inset, "Standard", true);
2250 new_inset->Edit(0, 0);
2254 case LFUN_CHILDOPEN:
2257 MakeAbsPath(argument,
2258 OnlyPath(owner->buffer()->fileName()));
2259 setMessage(N_("Opening child document ") +
2260 MakeDisplayPath(filename) + "...");
2261 owner->view()->savePosition();
2262 if (bufferlist.exists(filename))
2263 owner->view()->buffer(bufferlist.getBuffer(filename));
2265 owner->view()->buffer(bufferlist.loadLyXFile(filename));
2269 case LFUN_INSERT_NOTE:
2270 owner->view()->insertNote();
2273 case LFUN_INSERTFOOTNOTE:
2275 LyXParagraph::footnote_kind kind;
2276 if (argument == "footnote")
2277 { kind = LyXParagraph::FOOTNOTE; }
2278 else if (argument == "margin")
2279 { kind = LyXParagraph::MARGIN; }
2280 else if (argument == "figure")
2281 { kind = LyXParagraph::FIG; }
2282 else if (argument == "table")
2283 { kind = LyXParagraph::TAB; }
2284 else if (argument == "wide-fig")
2285 { kind = LyXParagraph::WIDE_FIG; }
2286 else if (argument == "wide-tab")
2287 { kind = LyXParagraph::WIDE_TAB; }
2288 else if (argument == "algorithm")
2289 { kind = LyXParagraph::ALGORITHM; }
2291 setErrorMessage(N_("Unknown kind of footnote"));
2294 owner->view()->text->InsertFootnoteEnvironment(kind);
2295 owner->view()->update(1);
2299 case LFUN_BUFFERBULLETSSELECT:
2303 case LFUN_TOGGLECURSORFOLLOW:
2304 cursor_follows_scrollbar = !cursor_follows_scrollbar;
2307 case LFUN_KMAP_OFF: // keymap off
2308 owner->getIntl()->KeyMapOn(false);
2311 case LFUN_KMAP_PRIM: // primary keymap
2312 owner->getIntl()->KeyMapPrim();
2315 case LFUN_KMAP_SEC: // secondary keymap
2316 owner->getIntl()->KeyMapSec();
2319 case LFUN_KMAP_TOGGLE: // toggle keymap
2320 owner->getIntl()->ToggleKeyMap();
2323 case LFUN_SELFINSERT:
2325 for (string::size_type i = 0; i < argument.length(); ++i) {
2326 owner->view()->text->InsertChar(argument[i]);
2327 // This needs to be in the loop, or else we
2328 // won't break lines correctly. (Asger)
2329 owner->view()->smallUpdate(1);
2332 owner->view()->text->sel_cursor =
2333 owner->view()->text->cursor;
2334 moveCursorUpdate(false);
2340 // argument contains ';'-terminated commands
2341 while (argument.find(';') != string::npos) {
2343 argument = split(argument, first, ';');
2349 case LFUN_SAVEPREFERENCES:
2351 Path p(user_lyxdir);
2352 lyxrc->write("preferences");
2356 case LFUN_UNKNOWN_ACTION:
2358 if(!owner->buffer()) {
2360 setErrorMessage(N_("No document open"));
2364 if (owner->buffer()->isReadonly()) {
2366 setErrorMessage(N_("Document is read only"));
2370 if (!argument.empty()) {
2372 /* Automatically delete the currently selected
2373 * text and replace it with what is being
2374 * typed in now. Depends on lyxrc settings
2375 * "auto_region_delete", which defaults to
2378 if ( lyxrc->auto_region_delete ) {
2379 if (owner->view()->text->selection){
2380 owner->view()->text->CutSelection(false);
2381 owner->view()->update(-1);
2385 owner->view()->beforeChange();
2386 for (string::size_type i = 0;
2387 i < argument.length(); ++i) {
2388 if (greek_kb_flag) {
2389 if (!math_insert_greek(argument[i]))
2390 owner->getIntl()->getTrans()->TranslateAndInsert(argument[i], owner->view()->text);
2392 owner->getIntl()->getTrans()->TranslateAndInsert(argument[i], owner->view()->text);
2395 owner->view()->smallUpdate(1);
2398 owner->view()->text->sel_cursor =
2399 owner->view()->text->cursor;
2400 moveCursorUpdate(false);
2403 // why is an "Unknown action" with empty
2404 // argument even dispatched in the first
2405 // place? I`ll probably change that. (Lgb)
2407 setErrorMessage(N_("Unknown action"));
2411 lyxerr << "A truly unknown func!" << endl;
2417 string res = getMessage();
2420 if (!commandshortcut.empty()) {
2421 string newbuf = owner->getMiniBuffer()->GetText();
2422 if (newbuf != commandshortcut) {
2423 owner->getMiniBuffer()->Set(newbuf
2429 owner->getMiniBuffer()->Set(string(_(res.c_str()))
2430 + " " + commandshortcut);
2437 void LyXFunc::setupLocalKeymap()
2439 keyseq.stdmap = keyseq.curmap = toplevel_keymap;
2440 cancel_meta_seq.stdmap = cancel_meta_seq.curmap = toplevel_keymap;
2444 void LyXFunc::MenuNew(bool fromTemplate)
2446 string fname, initpath = lyxrc->document_path;
2449 if (owner->view()->available()) {
2450 string trypath = owner->buffer()->filepath;
2451 // If directory is writeable, use this as default.
2452 if (IsDirWriteable(trypath) == 1)
2457 fileDlg.SetButton(0, _("Documents"), lyxrc->document_path);
2458 fileDlg.SetButton(1, _("Templates"), lyxrc->template_path);
2459 fname = fileDlg.Select(_("Enter Filename for new document"),
2460 initpath, "*.lyx", _("newfile"));
2463 if (fname.empty()) {
2464 owner->getMiniBuffer()->Set(_("Canceled."));
2465 lyxerr.debug() << "New Document Cancelled." << endl;
2469 // get absolute path of file and make sure the filename ends
2471 string s = MakeAbsPath(fname);
2472 if (!IsLyXFilename(s))
2475 // Check if the document already is open
2476 if (bufferlist.exists(s)){
2477 switch(AskConfirmation(_("Document is already open:"),
2478 MakeDisplayPath(s, 50),
2479 _("Do you want to close that document now?\n"
2480 "('No' will just switch to the open version)")))
2482 case 1: // Yes: close the document
2483 if (!bufferlist.close(bufferlist.getBuffer(s)))
2484 // If close is canceled, we cancel here too.
2487 case 2: // No: switch to the open document
2488 owner->view()->buffer(bufferlist.getBuffer(s));
2490 case 3: // Cancel: Do nothing
2491 owner->getMiniBuffer()->Set(_("Canceled."));
2496 // Check whether the file already exists
2497 if (IsLyXFilename(s)) {
2499 if (fi.readable() &&
2500 AskQuestion(_("File already exists:"),
2501 MakeDisplayPath(s, 50),
2502 _("Do you want to open the document?"))) {
2504 owner->getMiniBuffer()->Set(_("Opening document"),
2505 MakeDisplayPath(s), "...");
2507 owner->view()->buffer(
2508 bufferlist.loadLyXFile(s));
2509 owner->getMiniBuffer()->Set(_("Document"),
2516 // The template stuff
2520 fname = fileDlg.Select(_("Choose template"),
2521 lyxrc->template_path,
2527 // find a free buffer
2528 lyxerr.debug() << "Find a free buffer." << endl;
2529 owner->view()->buffer(bufferlist.newFile(s, templname));
2533 void LyXFunc::MenuOpen()
2535 string initpath = lyxrc->document_path;
2538 if (owner->view()->available()) {
2539 string trypath = owner->buffer()->filepath;
2540 // If directory is writeable, use this as default.
2541 if (IsDirWriteable(trypath) == 1)
2547 fileDlg.SetButton(0, _("Documents"), lyxrc->document_path);
2548 fileDlg.SetButton(1, _("Examples"),
2549 AddPath(system_lyxdir, "examples"));
2550 string filename = fileDlg.Select(_("Select Document to Open"),
2554 // check selected filename
2555 if (filename.empty()) {
2556 owner->getMiniBuffer()->Set(_("Canceled."));
2560 // get absolute path of file and make sure the filename ends
2562 filename = MakeAbsPath(filename);
2563 if (!IsLyXFilename(filename))
2567 owner->getMiniBuffer()->Set(_("Opening document"),
2568 MakeDisplayPath(filename), "...");
2569 Buffer * openbuf = bufferlist.loadLyXFile(filename);
2571 owner->view()->buffer(openbuf);
2572 owner->getMiniBuffer()->Set(_("Document"),
2573 MakeDisplayPath(filename),
2576 owner->getMiniBuffer()->Set(_("Could not open document"),
2577 MakeDisplayPath(filename));
2582 void LyXFunc::doImportASCII(bool linorpar)
2584 string initpath = lyxrc->document_path;
2587 if (owner->view()->available()) {
2588 string trypath = owner->buffer()->filepath;
2589 // If directory is writeable, use this as default.
2590 if (IsDirWriteable(trypath) == 1)
2596 fileDlg.SetButton(0, _("Documents"), lyxrc->document_path);
2597 fileDlg.SetButton(1, _("Examples"),
2598 AddPath(system_lyxdir, "examples"));
2599 string filename = fileDlg.Select(_("Select ASCII file to Import"),
2603 // check selected filename
2604 if (filename.empty()) {
2605 owner->getMiniBuffer()->Set(_("Canceled."));
2609 // get absolute path of file
2610 filename = MakeAbsPath(filename);
2612 string s = ChangeExtension(filename, ".lyx", false);
2614 // Check if the document already is open
2615 if (bufferlist.exists(s)) {
2616 switch(AskConfirmation(_("Document is already open:"),
2617 MakeDisplayPath(s, 50),
2618 _("Do you want to close that document now?\n"
2619 "('No' will just switch to the open version)")))
2621 case 1: // Yes: close the document
2622 if (!bufferlist.close(bufferlist.getBuffer(s)))
2623 // If close is canceled, we cancel here too.
2626 case 2: // No: switch to the open document
2627 owner->view()->buffer(bufferlist.getBuffer(s));
2629 case 3: // Cancel: Do nothing
2630 owner->getMiniBuffer()->Set(_("Canceled."));
2635 // Check if a LyX document by the same root exists in filesystem
2636 FileInfo f(s, true);
2637 if (f.exist() && !AskQuestion(_("A document by the name"),
2639 _("already exists. Overwrite?"))) {
2640 owner->getMiniBuffer()->Set(_("Canceled."));
2644 owner->view()->buffer(bufferlist.newFile(s, string()));
2645 owner->getMiniBuffer()->Set(_("Importing ASCII file"),
2646 MakeDisplayPath(filename), "...");
2647 // Insert ASCII file
2648 InsertAsciiFile(filename, linorpar);
2649 owner->getMiniBuffer()->Set(_("ASCII file "),
2650 MakeDisplayPath(filename),
2655 void LyXFunc::doImportLaTeX(bool isnoweb)
2657 string initpath = lyxrc->document_path;
2660 if (owner->view()->available()) {
2661 string trypath = owner->buffer()->filepath;
2662 // If directory is writeable, use this as default.
2663 if (IsDirWriteable(trypath) == 1)
2669 fileDlg.SetButton(0, _("Documents"), lyxrc->document_path);
2670 fileDlg.SetButton(1, _("Examples"),
2671 AddPath(system_lyxdir, "examples"));
2674 filename = fileDlg.Select(_("Select Noweb file to Import"),
2677 filename = fileDlg.Select(_("Select LaTeX file to Import"),
2683 // check selected filename
2684 if (filename.empty()) {
2685 owner->getMiniBuffer()->Set(_("Canceled."));
2689 // get absolute path of file
2690 filename = MakeAbsPath(filename);
2692 // Check if the document already is open
2693 string LyXfilename = ChangeExtension(filename, ".lyx", false);
2694 if (bufferlist.exists(LyXfilename)){
2695 switch(AskConfirmation(_("Document is already open:"),
2696 MakeDisplayPath(LyXfilename, 50),
2697 _("Do you want to close that document now?\n"
2698 "('No' will just switch to the open version)")))
2700 case 1: // Yes: close the document
2701 if (!bufferlist.close(bufferlist.getBuffer(LyXfilename)))
2702 // If close is canceled, we cancel here too.
2705 case 2: // No: switch to the open document
2706 owner->view()->buffer(
2707 bufferlist.getBuffer(LyXfilename));
2709 case 3: // Cancel: Do nothing
2710 owner->getMiniBuffer()->Set(_("Canceled."));
2715 // Check if a LyX document by the same root exists in filesystem
2716 FileInfo f(LyXfilename, true);
2717 if (f.exist() && !AskQuestion(_("A document by the name"),
2718 MakeDisplayPath(LyXfilename),
2719 _("already exists. Overwrite?"))) {
2720 owner->getMiniBuffer()->Set(_("Canceled."));
2727 owner->getMiniBuffer()->Set(_("Importing LaTeX file"),
2728 MakeDisplayPath(filename), "...");
2729 ImportLaTeX myImport(filename);
2730 openbuf = myImport.run();
2732 owner->getMiniBuffer()->Set(_("Importing Noweb file"),
2733 MakeDisplayPath(filename), "...");
2734 ImportNoweb myImport(filename);
2735 openbuf = myImport.run();
2738 owner->view()->buffer(openbuf);
2739 owner->getMiniBuffer()->Set(isnoweb ?
2740 _("Noweb file ") : _("LateX file "),
2741 MakeDisplayPath(filename),
2744 owner->getMiniBuffer()->Set(isnoweb ?
2745 _("Could not import Noweb file") :
2746 _("Could not import LaTeX file"),
2747 MakeDisplayPath(filename));
2752 void LyXFunc::MenuInsertLyXFile(string const & filen)
2754 string filename = filen;
2756 if (filename.empty()) {
2757 // Launch a file browser
2758 string initpath = lyxrc->document_path;
2761 if (owner->view()->available()) {
2762 string trypath = owner->buffer()->filepath;
2763 // If directory is writeable, use this as default.
2764 if (IsDirWriteable(trypath) == 1)
2770 fileDlg.SetButton(0, _("Documents"), lyxrc->document_path);
2771 fileDlg.SetButton(1, _("Examples"),
2772 AddPath(system_lyxdir, "examples"));
2773 filename = fileDlg.Select(_("Select Document to Insert"),
2777 // check selected filename
2778 if (filename.empty()) {
2779 owner->getMiniBuffer()->Set(_("Canceled."));
2784 // get absolute path of file and make sure the filename ends
2786 filename = MakeAbsPath(filename);
2787 if (!IsLyXFilename(filename))
2791 owner->getMiniBuffer()->Set(_("Inserting document"),
2792 MakeDisplayPath(filename), "...");
2793 bool res = owner->view()->insertLyXFile(filename);
2795 owner->getMiniBuffer()->Set(_("Document"),
2796 MakeDisplayPath(filename),
2799 owner->getMiniBuffer()->Set(_("Could not insert document"),
2800 MakeDisplayPath(filename));
2805 void LyXFunc::reloadBuffer()
2807 string fn = owner->buffer()->fileName();
2808 if (bufferlist.close(owner->buffer()))
2809 owner->view()->buffer(bufferlist.loadLyXFile(fn));
2813 void LyXFunc::CloseBuffer()
2815 if (bufferlist.close(owner->buffer()) && !quitting) {
2816 if (bufferlist.empty()) {
2817 // need this otherwise SEGV may occur while trying to
2818 // set variables that don't exist
2819 // since there's no current buffer
2820 CloseAllBufferRelatedPopups();
2823 owner->view()->buffer(bufferlist.first());
2829 Inset * LyXFunc::getInsetByCode(Inset::Code code)
2833 LyXCursor cursor = owner->view()->text->cursor;
2834 LyXParagraph::size_type pos = cursor.pos;
2835 LyXParagraph * par = cursor.par;
2837 while (par && !found) {
2838 while ((inset = par->ReturnNextInsetPointer(pos))){
2839 if (inset->LyxCode() == code) {
2847 return found ? inset : 0;
2851 // Each "owner" should have it's own message method. lyxview and
2852 // the minibuffer would use the minibuffer, but lyxserver would
2853 // send an ERROR signal to its client. Alejandro 970603
2854 // This func is bit problematic when it comes to NLS, to make the
2855 // lyx servers client be language indepenent we must not translate
2856 // strings sent to this func.
2857 void LyXFunc::setErrorMessage(string const & m) const
2859 dispatch_buffer = m;
2864 void LyXFunc::setMessage(string const & m)
2866 dispatch_buffer = m;