blob: 66365b102d9821bcb51f4280cd5717e590b3ab90 [file] [log] [blame]
Bram Moolenaar071d4272004-06-13 20:20:40 +00001/* vi:set ts=8 sts=4 sw=4:
2 *
3 * VIM - Vi IMproved by Bram Moolenaar
4 * Visual Workshop integration by Gordon Prieur
5 *
6 * Do ":help uganda" in Vim to read copying and usage conditions.
7 * Do ":help credits" in Vim to see a list of people who contributed.
8 * See README.txt for an overview of the Vim source code.
9 */
10
11#ifdef HAVE_CONFIG_H
12# include "auto/config.h"
13#endif
14#include <stdio.h>
15#include <stdlib.h>
Bram Moolenaar071d4272004-06-13 20:20:40 +000016#include <fcntl.h>
Bram Moolenaardfccaf02004-12-31 20:56:11 +000017#include <sys/types.h>
Bram Moolenaar071d4272004-06-13 20:20:40 +000018#include <netdb.h>
19#include <netinet/in.h>
20#include <errno.h>
Bram Moolenaar071d4272004-06-13 20:20:40 +000021#include <sys/socket.h>
22#ifdef HAVE_LIBGEN_H
23# include <libgen.h>
24#endif
25#include <unistd.h>
26#include <string.h>
27#include <stdlib.h>
28#include <ctype.h>
29
30#include <X11/Intrinsic.h>
31#include <Xm/Xm.h>
32#include <Xm/PushB.h>
33
34#include "integration.h" /* <EditPlugin/integration.h> */
35
36#include "vim.h"
37#include "version.h"
38#include "gui_beval.h"
39#include "workshop.h"
40
41void workshop_hotkeys(Boolean);
42
43static Boolean isShowing(int);
44static win_T *get_window(buf_T *);
45#if 0
46static int get_buffer_number(buf_T *);
47#endif
48static void updatePriority(Boolean);
49static char *addUniqueMnemonic(char *, char *);
50static char *fixup(char *);
51static char *get_selection(buf_T *);
52static char *append_selection(int, char *, int *, int *);
53static void load_buffer_by_name(char *, int);
54#if 0
55static void load_buffer_by_number(int, int);
56#endif
57static void load_window(char *, int lnum);
58static void warp_to_pc(int);
59#ifdef FEAT_BEVAL
Bram Moolenaare4efc3b2005-03-07 23:16:51 +000060void workshop_beval_cb(BalloonEval *, int);
Bram Moolenaar071d4272004-06-13 20:20:40 +000061#endif
62static char *fixAccelText(char *);
63static void addMenu(char *, char *, char *);
64static char *lookupVerb(char *, int);
65static int computeIndex(int, char_u *, int);
66static void coloncmd(char *, Boolean);
67
68extern Widget vimShell;
69extern Widget textArea;
70extern XtAppContext app_context;
71
72static int tbpri; /* ToolBar priority */
73int usingSunWorkShop = 0; /* set if -ws flag is used */
74char curMenuName[BUFSIZ];
75char curMenuPriority[BUFSIZ];
Bram Moolenaar071d4272004-06-13 20:20:40 +000076
77static Boolean workshopInitDone = False;
78static Boolean workshopHotKeysEnabled = False;
79
80/*
81 * The following enum is from <gp_dbx/gp_dbx_common.h>. We can't include it
82 * here because its C++.
83 */
84enum
85{
86 GPLineEval_EVALUATE, /* evaluate expression */
87 GPLineEval_INDIRECT, /* evaluate *<expression> */
88 GPLineEval_TYPE /* type of expression */
89};
90
91/*
92 * Store each verb in the MenuMap. This lets us map from a verb to a menu.
93 * There may be multiple matches for a single verb in this table.
94 */
95#define MENU_INC 50 /* menuMap incremental size increases */
96typedef struct
97{
98 char *name; /* name of the menu */
99 char *accel; /* optional accelerator key */
100 char *verb; /* menu verb */
101} MenuMap;
102static MenuMap *menuMap; /* list of verb/menu mappings */
103static int menuMapSize; /* current size of menuMap */
104static int menuMapMax; /* allocated size of menuMap */
105static char *initialFileCmd; /* save command but defer doing it */
106
107
108 void
109workshop_init()
110{
111 char_u buf[64];
112 int is_dirty = FALSE;
113 int width, height;
114 XtInputMask mask;
115
116 /*
117 * Turn on MenuBar, ToolBar, and Footer.
118 */
119 STRCPY(buf, p_go);
120 if (vim_strchr(p_go, GO_MENUS) == NULL)
121 {
122 STRCAT(buf, "m");
123 is_dirty = TRUE;
124 }
125 if (vim_strchr(p_go, GO_TOOLBAR) == NULL)
126 {
127 STRCAT(buf, "T");
128 is_dirty = TRUE;
129 }
130 if (vim_strchr(p_go, GO_FOOTER) == NULL)
131 {
132 STRCAT(buf, "F");
133 is_dirty = TRUE;
134 }
135 if (is_dirty)
136 set_option_value((char_u *)"go", 0L, buf, 0);
137
138 /*
139 * Set size from workshop_get_width_height().
140 */
141 width = height = 0;
142 if (workshop_get_width_height(&width, &height))
143 {
144 XtVaSetValues(vimShell,
145 XmNwidth, width,
146 XmNheight, height,
147 NULL);
148 }
149
150 /*
151 * Now read in the initial messages from eserve.
152 */
153 while ((mask = XtAppPending(app_context))
154 && (mask & XtIMAlternateInput) && !workshopInitDone)
155 XtAppProcessEvent(app_context, (XtInputMask)XtIMAlternateInput);
156}
157
158 void
159workshop_postinit()
160{
161 do_cmdline_cmd((char_u *)initialFileCmd);
162 ALT_INPUT_LOCK_OFF;
163 free(initialFileCmd);
164 initialFileCmd = NULL;
165}
166
167 void
168ex_wsverb(exarg_T *eap)
169{
170 msg_clr_cmdline();
171 workshop_perform_verb((char *) eap->arg, NULL);
172}
173
174/*
175 * Editor name
176 * This string is recognized by eserve and should be all lower case.
177 * This is how the editor detects that it is talking to gvim instead
178 * of NEdit, for example, when the connection is initiated from the editor.
179 */
180 char *
181workshop_get_editor_name()
182{
183 return "gvim";
184}
185
186/*
187 * Version number of the editor.
188 * This number is communicated along with the protocol
189 * version to the application.
190 */
191 char *
192workshop_get_editor_version()
193{
194 return Version;
195}
196
197/*
198 * Answer functions: called by eserve
199 */
200
201/*
202 * Name:
203 * workshop_load_file
204 *
205 * Function:
206 * Load a given file into the WorkShop buffer.
207 */
208/*ARGSUSED*/
209 void
210workshop_load_file(
211 char *filename, /* the file to load */
212 int line, /* an optional line number (or 0) */
213 char *frameid) /* used for multi-frame support */
214{
215#ifdef WSDEBUG_TRACE
216 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
217 wstrace("workshop_load_file(%s, %d)\n", filename, line);
218#endif
219
220#ifdef FEAT_BEVAL
Bram Moolenaare4efc3b2005-03-07 23:16:51 +0000221 bevalServers |= BEVAL_WORKSHOP;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000222#endif
223
224 load_window(filename, line);
225}
226
227/*
228 * Reload the WorkShop buffer
229 */
230 void
231workshop_reload_file(
232 char *filename,
233 int line)
234{
235#ifdef WSDEBUG_TRACE
236 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
237 wstrace("workshop_reload_file(%s, %d)\n", filename, line);
238#endif
239 load_window(filename, line);
240}
241
242 void
243workshop_show_file(
244 char *filename)
245{
246#ifdef WSDEBUG_TRACE
247 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
248 wstrace("workshop_show_file(%s)\n", filename);
249#endif
250
251 load_window(filename, 0);
252}
253
254 void
255workshop_goto_line(
256 char *filename,
257 int lineno)
258{
259#ifdef WSDEBUG_TRACE
260 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
261 wstrace("workshop_goto_line(%s, %d)\n", filename, lineno);
262#endif
263
264 load_window(filename, lineno);
265}
266
267/*ARGSUSED*/
268 void
269workshop_front_file(
270 char *filename)
271{
272#ifdef WSDEBUG_TRACE
273 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
274 wstrace("workshop_front_file()\n");
275#endif
276 /*
277 * Assumption: This function will always be called after a call to
278 * workshop_show_file(), so the file is always showing.
279 */
280 if (vimShell != NULL)
281 XRaiseWindow(gui.dpy, XtWindow(vimShell));
282}
283
284 void
285workshop_save_file(
286 char *filename)
287{
288 char cbuf[BUFSIZ]; /* build vim command here */
289
290#ifdef WSDEBUG_TRACE
291 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
292 wstrace("workshop_save_file(%s)\n", filename);
293#endif
294
295 /* Save the given file */
Bram Moolenaar051b7822005-05-19 21:00:46 +0000296 vim_snprintf(cbuf, sizeof(cbuf), "w %s", filename);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000297 coloncmd(cbuf, TRUE);
298}
299
300 void
301workshop_save_files()
302{
303 /* Save the given file */
304#ifdef WSDEBUG_TRACE
305 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
306 wstrace("workshop_save_files()\n");
307#endif
308
309 add_to_input_buf((char_u *) ":wall\n", 6);
310}
311
312 void
313workshop_quit()
314{
315#ifdef WSDEBUG_TRACE
316 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
317 wstrace("workshop_quit()\n");
318#endif
319
320 add_to_input_buf((char_u *) ":qall\n", 6);
321}
322
323 void
324workshop_minimize()
325{
326#ifdef WSDEBUG_TRACE
327 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
328 wstrace("workshop_minimize()\n");
329#endif
330 workshop_minimize_shell(vimShell);
331}
332 void
333workshop_maximize()
334{
335#ifdef WSDEBUG_TRACE
336 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
337 wstrace("workshop_maximize()\n");
338#endif
339
340 workshop_maximize_shell(vimShell);
341}
342
343 void
344workshop_add_mark_type(
345 int idx,
346 char *colorspec,
347 char *sign)
348{
349 char gbuf[BUFSIZ]; /* buffer for sign name */
350 char cibuf[BUFSIZ]; /* color information */
351 char cbuf[BUFSIZ]; /* command buffer */
352 char *bp;
353
354#ifdef WSDEBUG_TRACE
355 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
356 {
357 char *cp;
358
359 cp = strrchr(sign, '/');
360 if (cp == NULL)
361 cp = sign;
362 else
363 cp++; /* skip '/' character */
364 wstrace("workshop_add_mark_type(%d, \"%s\", \"%s\")\n", idx,
365 colorspec && *colorspec ? colorspec : "<None>", cp);
366 }
367#endif
368
369 /*
370 * Isolate the basename of sign in gbuf. We will use this for the
371 * GroupName in the highlight command sent to vim.
372 */
373 STRCPY(gbuf, gettail((char_u *)sign));
374 bp = strrchr(gbuf, '.');
375 if (bp != NULL)
376 *bp = NUL;
377
378 if (gbuf[0] != '-' && gbuf[1] != NUL)
379 {
380 if (colorspec != NULL && *colorspec)
381 {
Bram Moolenaar051b7822005-05-19 21:00:46 +0000382 vim_snprintf(cbuf, sizeof(cbuf),
383 "highlight WS%s guibg=%s", gbuf, colorspec);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000384 coloncmd(cbuf, FALSE);
Bram Moolenaar051b7822005-05-19 21:00:46 +0000385 vim_snprintf(cibuf, sizeof(cibuf), "linehl=WS%s", gbuf);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000386 }
387 else
388 cibuf[0] = NUL;
389
Bram Moolenaar051b7822005-05-19 21:00:46 +0000390 vim_snprintf(cbuf, sizeof(cbuf),
391 "sign define %d %s icon=%s", idx, cibuf, sign);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000392 coloncmd(cbuf, TRUE);
393 }
394}
395
396 void
397workshop_set_mark(
398 char *filename, /* filename which gets the mark */
399 int lineno, /* line number which gets the mark */
400 int markId, /* unique mark identifier */
401 int idx) /* which mark to use */
402{
403 char cbuf[BUFSIZ]; /* command buffer */
404
405 /* Set mark in a given file */
406#ifdef WSDEBUG_TRACE
407 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
408 wstrace("workshop_set_mark(%s, %d (ln), %d (id), %d (idx))\n",
409 filename, lineno, markId, idx);
410#endif
411
Bram Moolenaar051b7822005-05-19 21:00:46 +0000412 vim_snprintf(cbuf, sizeof(cbuf), "sign place %d line=%d name=%d file=%s",
Bram Moolenaar071d4272004-06-13 20:20:40 +0000413 markId, lineno, idx, filename);
414 coloncmd(cbuf, TRUE);
415}
416
417 void
418workshop_change_mark_type(
419 char *filename, /* filename which gets the mark */
420 int markId, /* unique mark identifier */
421 int idx) /* which mark to use */
422{
423 char cbuf[BUFSIZ]; /* command buffer */
424
425 /* Change mark type */
426#ifdef WSDEBUG_TRACE
427 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
428 wstrace("workshop_change_mark_type(%s, %d, %d)\n",
429 filename, markId, idx);
430#endif
431
Bram Moolenaar051b7822005-05-19 21:00:46 +0000432 vim_snprintf(cbuf, sizeof(cbuf),
433 "sign place %d name=%d file=%s", markId, idx, filename);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000434 coloncmd(cbuf, TRUE);
435}
436
437/*
438 * Goto the given mark in a file (e.g. show it).
439 * If message is not null, display it in the footer.
440 */
441 void
442workshop_goto_mark(
443 char *filename,
444 int markId,
445 char *message)
446{
447 char cbuf[BUFSIZ]; /* command buffer */
448
449 /* Goto mark */
450#ifdef WSDEBUG_TRACE
451 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
452 wstrace("workshop_goto_mark(%s, %d (id), %s)\n",
453 filename, markId, message && *message &&
454 !(*message == ' ' && message[1] == NULL) ?
455 message : "<None>");
456#endif
457
Bram Moolenaar051b7822005-05-19 21:00:46 +0000458 vim_snprintf(cbuf, sizeof(cbuf), "sign jump %d file=%s", markId, filename);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000459 coloncmd(cbuf, TRUE);
460 if (message != NULL && *message != NUL)
461 gui_mch_set_footer((char_u *)message);
462}
463
464 void
465workshop_delete_mark(
466 char *filename,
467 int markId)
468{
469 char cbuf[BUFSIZ]; /* command buffer */
470
471 /* Delete mark */
472#ifdef WSDEBUG_TRACE
473 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
474 wstrace("workshop_delete_mark(%s, %d (id))\n",
475 filename, markId);
476#endif
477
Bram Moolenaar051b7822005-05-19 21:00:46 +0000478 vim_snprintf(cbuf, sizeof(cbuf),
479 "sign unplace %d file=%s", markId, filename);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000480 coloncmd(cbuf, TRUE);
481}
482
483#if 0 /* not used */
484 void
485workshop_delete_all_marks(
486 void *window,
487 Boolean doRefresh)
488{
489#ifdef WSDEBUG_TRACE
490 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
491 wstrace("workshop_delete_all_marks(%#x, %s)\n",
492 window, doRefresh ? "True" : "False");
493#endif
494
495 coloncmd("sign unplace *", TRUE);
496}
497#endif
498
499 int
500workshop_get_mark_lineno(
501 char *filename,
502 int markId)
503{
504 buf_T *buf; /* buffer containing filename */
505 int lineno; /* line number of filename in buf */
506
507 /* Get mark line number */
508#ifdef WSDEBUG_TRACE
509 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
510 wstrace("workshop_get_mark_lineno(%s, %d)\n",
511 filename, markId);
512#endif
513
514 lineno = 0;
515 buf = buflist_findname((char_u *)filename);
516 if (buf != NULL)
517 lineno = buf_findsign(buf, markId);
518
519 return lineno;
520}
521
522
523#if 0 /* not used */
524 void
525workshop_adjust_marks(Widget *window, int pos,
526 int inserted, int deleted)
527{
528#ifdef WSDEBUG_TRACE
529 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
530 wstrace("XXXworkshop_adjust_marks(%s, %d, %d, %d)\n",
531 window ? XtName(window) : "<None>", pos, inserted, deleted);
532#endif
533}
534#endif
535
536/*
537 * Are there any moved marks? If so, call workshop_move_mark on
538 * each of them now. This is how eserve can find out if for example
539 * breakpoints have moved when a program has been recompiled and
540 * reloaded into dbx.
541 */
542/*ARGSUSED*/
543 void
544workshop_moved_marks(char *filename)
545{
546#ifdef WSDEBUG_TRACE
547 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
548 wstrace("XXXworkshop_moved_marks(%s)\n", filename);
549#endif
550}
551
552 int
553workshop_get_font_height()
554{
555 XmFontList fontList; /* fontList made from gui.norm_font */
556 XmString str;
557 Dimension w;
558 Dimension h;
559
560#ifdef WSDEBUG_TRACE
561 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
562 wstrace("workshop_get_font_height()\n");
563#endif
564
565 /* Pick the proper signs for this font size */
566 fontList = gui_motif_create_fontlist((XFontStruct *)gui.norm_font);
567 h = 0;
568 if (fontList != NULL)
569 {
570 str = XmStringCreateLocalized("A");
571 XmStringExtent(fontList, str, &w, &h);
572 XmStringFree(str);
573 XmFontListFree(fontList);
574 }
575
576 return (int)h;
577}
578
579/*ARGSUSED*/
580 void
581workshop_footer_message(
582 char *message,
583 int severity) /* severity is currently unused */
584{
585#ifdef WSDEBUG_TRACE
586 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
587 wstrace("workshop_footer_message(%s, %d)\n", message, severity);
588#endif
589
590 gui_mch_set_footer((char_u *) message);
591}
592
593/*
594 * workshop_menu_begin() is passed the menu name. We determine its mnemonic
595 * here and store its name and priority.
596 */
597 void
598workshop_menu_begin(
599 char *label)
600{
601 vimmenu_T *menu; /* pointer to last menu */
602 int menuPriority = 0; /* priority of new menu */
603 char mnembuf[64]; /* store menubar mnemonics here */
604 char *name; /* label with a mnemonic */
605 char *p; /* used to find mnemonics */
606 int idx; /* index into mnembuf */
607
608#ifdef WSDEBUG_TRACE
609 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
610 wstrace("workshop_menu_begin()\n");
611#endif
612
613 /*
614 * Look through all existing (non-PopUp and non-Toolbar) menus
615 * and gather their mnemonics. Use this list to decide what
616 * mnemonic should be used for label.
617 */
618
619 idx = 0;
620 mnembuf[idx++] = 'H'; /* H is mnemonic for Help */
621 for (menu = root_menu; menu != NULL; menu = menu->next)
622 {
623 if (menu_is_menubar(menu->name))
624 {
625 p = strchr((char *)menu->name, '&');
626 if (p != NULL)
627 mnembuf[idx++] = *++p;
628 }
629 if (menu->next != NULL
630 && strcmp((char *) menu->next->dname, "Help") == 0)
631 {
632 menuPriority = menu->priority + 10;
633 break;
634 }
635 }
636 mnembuf[idx++] = NUL;
637 name = addUniqueMnemonic(mnembuf, label);
638
Bram Moolenaar051b7822005-05-19 21:00:46 +0000639 vim_snprintf(curMenuName, sizeof(curMenuName), "%s", name);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000640 sprintf(curMenuPriority, "%d.0", menuPriority);
641}
642
643/*
644 * Append the name and priority to strings to be used in vim menu commands.
645 */
646 void
647workshop_submenu_begin(
648 char *label)
649{
650#ifdef WSDEBUG_TRACE
651 if (ws_debug && ws_dlevel & WS_TRACE
652 && strncmp(curMenuName, "ToolBar", 7) != 0)
653 wstrace("workshop_submenu_begin(%s)\n", label);
654#endif
655
656 strcat(curMenuName, ".");
657 strcat(curMenuName, fixup(label));
658
659 updatePriority(True);
660}
661
662/*
663 * Remove the submenu name and priority from curMenu*.
664 */
665
666 void
667workshop_submenu_end()
668{
669 char *p;
670
671#ifdef WSDEBUG_TRACE
672 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE)
673 && strncmp(curMenuName, "ToolBar", 7) != 0)
674 wstrace("workshop_submenu_end()\n");
675#endif
676
677 p = strrchr(curMenuPriority, '.');
678 ASSERT(p != NULL);
679 *p = NUL;
680
681 p = strrchr(curMenuName, '.');
682 ASSERT(p != NULL);
683 *p = NUL;
684}
685
686/*
687 * This is where menus are really made. Each item will generate an amenu vim
688 * command. The globals curMenuName and curMenuPriority contain the name and
689 * priority of the parent menu tree.
690 */
691/*ARGSUSED*/
692 void
693workshop_menu_item(
694 char *label,
695 char *verb,
696 char *accelerator,
697 char *acceleratorText,
698 char *name,
699 char *filepos,
700 char *sensitive)
701{
702 char cbuf[BUFSIZ];
703 char namebuf[BUFSIZ];
704 char accText[BUFSIZ];
705
706#ifdef WSDEBUG_TRACE
707 if (WSDLEVEL(WS_TRACE_VERBOSE)
708 && strncmp(curMenuName, "ToolBar", 7) != 0)
709 {
710 if (ws_dlevel & WS_TRACE_VERBOSE)
711 wsdebug("workshop_menu_item(\n"
712 "\tlabel = \"%s\",\n"
713 "\tverb = %s,\n"
714 "\taccelerator = %s,\n"
715 "\tacceleratorText = \"%s\",\n"
716 "\tname = %s,\n"
717 "\tfilepos = %s,\n"
718 "\tsensitive = %s)\n",
719 label && *label ? label : "<None>",
720 verb && *verb ? verb : "<None>",
721 accelerator && *accelerator ?
722 accelerator : "<None>",
723 acceleratorText && *acceleratorText ?
724 acceleratorText : "<None>",
725 name && *name ? name : "<None>",
726 filepos && *filepos ? filepos : "<None>",
727 sensitive);
728 else if (ws_dlevel & WS_TRACE)
729 wstrace("workshop_menu_item(\"%s\", %s)\n",
730 label && *label ? label : "<None>",
731 verb && *verb ? verb : "<None>", sensitive);
732 }
733#endif
734#ifdef WSDEBUG_SENSE
735 if (ws_debug)
736 wstrace("menu: %-21.20s%-21.20s(%s)\n", label, verb,
737 *sensitive == '1' ? "Sensitive" : "Insensitive");
738#endif
739
740 if (acceleratorText != NULL)
Bram Moolenaar051b7822005-05-19 21:00:46 +0000741 vim_snprintf(accText, sizeof(accText), "<Tab>%s", acceleratorText);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000742 else
743 accText[0] = NUL;
744 updatePriority(False);
Bram Moolenaar051b7822005-05-19 21:00:46 +0000745 vim_snprintf(namebuf, sizeof(namebuf), "%s.%s", curMenuName, fixup(label));
746 vim_snprintf(cbuf, sizeof(cbuf), "amenu %s %s%s\t:wsverb %s<CR>",
Bram Moolenaar071d4272004-06-13 20:20:40 +0000747 curMenuPriority, namebuf, accText, verb);
748
749 coloncmd(cbuf, TRUE);
750 addMenu(namebuf, fixAccelText(acceleratorText), verb);
751
752 if (*sensitive == '0')
753 {
Bram Moolenaar051b7822005-05-19 21:00:46 +0000754 vim_snprintf(cbuf, sizeof(cbuf), "amenu disable %s", namebuf);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000755 coloncmd(cbuf, TRUE);
756 }
757}
758
759/*
760 * This function is called when a complete WorkShop menu description has been
761 * sent over from eserve. We do some menu cleanup.
762 */
763
764 void
765workshop_menu_end()
766{
767 Boolean using_tearoff; /* set per current option setting */
768
769#ifdef WSDEBUG_TRACE
770 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
771 wstrace("workshop_menu_end()\n");
772#endif
773
774 using_tearoff = vim_strchr(p_go, GO_TEAROFF) != NULL;
775 gui_mch_toggle_tearoffs(using_tearoff);
776}
777
778 void
779workshop_toolbar_begin()
780{
781#ifdef WSDEBUG_TRACE
782 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
783 wstrace("workshop_toolbar_begin()\n");
784#endif
785
786 coloncmd("aunmenu ToolBar", True);
787 tbpri = 10;
788}
789
790 void
791workshop_toolbar_end()
792{
793 char_u buf[64];
794
795#ifdef WSDEBUG_TRACE
796 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
797 {
798 wstrace("workshop_toolbar_end()\n");
799 }
800#endif
801
802 /*
803 * Turn on ToolBar.
804 */
805 STRCPY(buf, p_go);
806 if (vim_strchr(p_go, 'T') == NULL)
807 {
808 STRCAT(buf, "T");
809 set_option_value((char_u *)"go", 0L, buf, 0);
810 }
811 workshopInitDone = True;
812}
813
814/*ARGSUSED*/
815 void
816workshop_toolbar_button(
817 char *label,
818 char *verb,
819 char *senseVerb,
820 char *filepos,
821 char *help,
822 char *sense,
823 char *file,
824 char *left)
825{
826 char cbuf[BUFSIZ + MAXPATHLEN];
827 char namebuf[BUFSIZ];
828 static int tbid = 1;
829 char_u *p;
Bram Moolenaar051b7822005-05-19 21:00:46 +0000830 int len;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000831
832#ifdef WSDEBUG_TRACE
833 if (WSDLEVEL(WS_TRACE_VERBOSE))
834 wsdebug("workshop_toolbar_button(\"%s\", %s, %s,\n"
835 "\t%s, \"%s\", %s,\n\t\"%s\",\n\t<%s>)\n",
836 label && *label ? label : "<None>",
837 verb && *verb ? verb : "<None>",
838 senseVerb && *senseVerb ? senseVerb : "<None>",
839 filepos && *filepos ? filepos : "<None>",
840 help && *help ? help : "<None>",
841 sense && *sense ? sense : "<None>",
842 file && *file ? file : "<None>",
843 left && *left ? left : "<None>");
844 else if (WSDLEVEL(WS_TRACE))
845 wstrace("workshop_toolbar_button(\"%s\", %s)\n",
846 label && *label ? label : "<None>",
847 verb && *verb ? verb : "<None>");
848#endif
849#ifdef WSDEBUG_SENSE
850 if (ws_debug)
851 wsdebug("button: %-21.20s%-21.20s(%s)\n", label, verb,
852 *sense == '1' ? "Sensitive" : "Insensitive");
853#endif
854
855 if (left && *left && atoi(left) > 0)
856 {
857 /* Add a separator (but pass the width passed after the ':') */
858 sprintf(cbuf, "amenu 1.%d ToolBar.-sep%d:%s- <nul>",
859 tbpri - 5, tbid++, left);
860
861 coloncmd(cbuf, True);
862 }
863
864 p = vim_strsave_escaped((char_u *)label, (char_u *)"\\. ");
Bram Moolenaar051b7822005-05-19 21:00:46 +0000865 vim_snprintf(namebuf, sizeof(namebuf), "ToolBar.%s", p);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000866 vim_free(p);
867 STRCPY(cbuf, "amenu <silent> ");
868 if (file != NULL && *file != NUL)
869 {
870 p = vim_strsave_escaped((char_u *)file, (char_u *)" ");
Bram Moolenaar051b7822005-05-19 21:00:46 +0000871 len = STRLEN(cbuf);
872 vim_snprintf(cbuf + len, sizeof(cbuf) - len, "icon=%s ", p);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000873 vim_free(p);
874 }
Bram Moolenaar051b7822005-05-19 21:00:46 +0000875 len = STRLEN(cbuf);
876 vim_snprintf(cbuf + len, sizeof(cbuf) - len,"1.%d %s :wsverb %s<CR>",
Bram Moolenaar071d4272004-06-13 20:20:40 +0000877 tbpri, namebuf, verb);
878
879 /* Define the menu item */
880 coloncmd(cbuf, True);
881
882 if (*sense == '0')
883 {
884 /* If menu isn't sensitive at startup... */
Bram Moolenaar051b7822005-05-19 21:00:46 +0000885 vim_snprintf(cbuf, sizeof(cbuf), "amenu disable %s", namebuf);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000886 coloncmd(cbuf, True);
887 }
888
889 if (help && *help)
890 {
891 /* Do the tooltip */
Bram Moolenaar051b7822005-05-19 21:00:46 +0000892 vim_snprintf(cbuf, sizeof(cbuf), "tmenu %s %s", namebuf, help);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000893 coloncmd(cbuf, True);
894 }
895
896 addMenu(namebuf, NULL, verb);
897 tbpri += 10;
898}
899
900 void
901workshop_frame_sensitivities(
902 VerbSense *vs) /* list of verbs to (de)sensitize */
903{
904 VerbSense *vp; /* iterate through vs */
905 char *menu_name; /* used in menu lookup */
906 int cnt; /* count of verbs to skip */
907 int len; /* length of nonvariant part of command */
908 char cbuf[4096];
909
910#ifdef WSDEBUG_TRACE
911 if (WSDLEVEL(WS_TRACE_VERBOSE) || WSDLEVEL(4))
912 {
913 wsdebug("workshop_frame_sensitivities(\n");
914 for (vp = vs; vp->verb != NULL; vp++)
915 wsdebug("\t%-25s%d\n", vp->verb, vp->sense);
916 wsdebug(")\n");
917 }
918 else if (WSDLEVEL(WS_TRACE))
919 wstrace("workshop_frame_sensitivities()\n");
920#endif
921#ifdef WSDEBUG_SENSE
922 if (ws_debug)
923 for (vp = vs; vp->verb != NULL; vp++)
924 wsdebug("change: %-21.20s%-21.20s(%s)\n",
925 "", vp->verb, vp->sense == 1 ?
926 "Sensitive" : "Insensitive");
927#endif
928
929 /*
930 * Look for all matching menu entries for the verb. There may be more
931 * than one if the verb has both a menu and toolbar entry.
932 */
933 for (vp = vs; vp->verb != NULL; vp++)
934 {
935 cnt = 0;
936 strcpy(cbuf, "amenu");
937 strcat(cbuf, " ");
938 strcat(cbuf, vp->sense ? "enable" : "disable");
939 strcat(cbuf, " ");
940 len = strlen(cbuf);
941 while ((menu_name = lookupVerb(vp->verb, cnt++)) != NULL)
942 {
943 strcpy(&cbuf[len], menu_name);
944 coloncmd(cbuf, FALSE);
945 }
946 }
947 gui_update_menus(0);
948 gui_mch_flush();
949}
950
951 void
952workshop_set_option(
953 char *option, /* name of a supported option */
954 char *value) /* value to set option to */
955{
956 char cbuf[BUFSIZ]; /* command buffer */
957
958#ifdef WSDEBUG_TRACE
959 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
960 {
961 wstrace("workshop_set_option(%s, %s)\n", option, value);
962 }
963#endif
964
965 cbuf[0] = NUL;
966 switch (*option) /* switch on 1st letter */
967 {
968 case 's':
969 if (strcmp(option, "syntax") == 0)
Bram Moolenaar051b7822005-05-19 21:00:46 +0000970 vim_snprintf(cbuf, sizeof(cbuf), "syntax %s", value);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000971 else if (strcmp(option, "savefiles") == 0)
972 ; /* XXX - Not yet implemented */
973 break;
974
975 case 'l':
976 if (strcmp(option, "lineno") == 0)
977 sprintf(cbuf, "set %snu",
978 (strcmp(value, "on") == 0) ? "" : "no");
979 break;
980
981 case 'p':
982 if (strcmp(option, "parentheses") == 0)
983 sprintf(cbuf, "set %ssm",
984 (strcmp(value, "on") == 0) ? "" : "no");
985 break;
986
987 case 'w':
988 /* this option is set by a direct call */
989#ifdef WSDEBUG
990 wsdebug("workshop_set_option: "
991 "Got unexpected workshopkeys option");
992#endif
993 break;
994
995 case 'b': /* these options are set from direct calls */
996 if (option[7] == NUL && strcmp(option, "balloon") == 0)
997 {
998#ifdef WSDEBUG
999 /* set by direct call to workshop_balloon_mode */
1000 wsdebug("workshop_set_option: "
1001 "Got unexpected ballooneval option");
1002#endif
1003 }
1004 else if (strcmp(option, "balloondelay") == 0)
1005 {
1006#ifdef WSDEBUG
1007 /* set by direct call to workshop_balloon_delay */
1008 wsdebug("workshop_set_option: "
1009 "Got unexpected balloondelay option");
1010#endif
1011 }
1012 break;
1013 }
1014 if (cbuf[0] != NUL)
1015 coloncmd(cbuf, TRUE);
1016}
1017
1018
1019 void
1020workshop_balloon_mode(
1021 Boolean on)
1022{
1023 char cbuf[BUFSIZ]; /* command buffer */
1024
1025#ifdef WSDEBUG_TRACE
1026 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
1027 wstrace("workshop_balloon_mode(%s)\n", on ? "True" : "False");
1028#endif
1029
1030 sprintf(cbuf, "set %sbeval", on ? "" : "no");
1031 coloncmd(cbuf, TRUE);
1032}
1033
1034
1035 void
1036workshop_balloon_delay(
1037 int delay)
1038{
1039 char cbuf[BUFSIZ]; /* command buffer */
1040
1041#ifdef WSDEBUG_TRACE
1042 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
1043 wstrace("workshop_balloon_delay(%d)\n", delay);
1044#endif
1045
1046 sprintf(cbuf, "set bdlay=%d", delay);
1047 coloncmd(cbuf, TRUE);
1048}
1049
1050
1051 void
1052workshop_show_balloon_tip(
1053 char *tip)
1054{
1055#ifdef WSDEBUG_TRACE
1056 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
1057 wstrace("workshop_show_balloon_tip(%s)\n", tip);
1058#endif
1059
1060 if (balloonEval != NULL)
1061 gui_mch_post_balloon(balloonEval, (char_u *)tip);
1062}
1063
1064
1065 void
1066workshop_hotkeys(
1067 Boolean on)
1068{
1069 char cbuf[BUFSIZ]; /* command buffer */
1070 MenuMap *mp; /* iterate over menuMap entries */
1071
1072#ifdef WSDEBUG_TRACE
1073 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
1074 wstrace("workshop_hotkeys(%s)\n", on ? "True" : "False");
1075#endif
1076
1077 workshopHotKeysEnabled = on;
1078 if (workshopHotKeysEnabled)
1079 for (mp = menuMap; mp < &menuMap[menuMapSize]; mp++)
1080 {
1081 if (mp->accel != NULL)
1082 {
Bram Moolenaar051b7822005-05-19 21:00:46 +00001083 vim_snprintf(cbuf, sizeof(cbuf),
1084 "map %s :wsverb %s<CR>", mp->accel, mp->verb);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001085 coloncmd(cbuf, TRUE);
1086 }
1087 }
1088 else
1089 for (mp = menuMap; mp < &menuMap[menuMapSize]; mp++)
1090 {
1091 if (mp->accel != NULL)
1092 {
Bram Moolenaar051b7822005-05-19 21:00:46 +00001093 vim_snprintf(cbuf, sizeof(cbuf), "unmap %s", mp->accel);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001094 coloncmd(cbuf, TRUE);
1095 }
1096 }
1097}
1098
1099/*
1100 * A button in the toolbar has been pushed.
1101 */
1102/*ARGSUSED*/
1103 int
1104workshop_get_positions(
1105 void *clientData, /* unused */
1106 char **filename, /* output data */
1107 int *curLine, /* output data */
1108 int *curCol, /* output data */
1109 int *selStartLine, /* output data */
1110 int *selStartCol, /* output data */
1111 int *selEndLine, /* output data */
1112 int *selEndCol, /* output data */
1113 int *selLength, /* output data */
1114 char **selection) /* output data */
1115{
1116 static char ffname[MAXPATHLEN];
1117
1118#ifdef WSDEBUG_TRACE
1119 if (WSDLEVEL(WS_TRACE_VERBOSE | WS_TRACE))
1120 wstrace("workshop_get_positions(%#x, \"%s\", ...)\n",
1121 clientData, (curbuf && curbuf->b_sfname != NULL)
1122 ? (char *)curbuf->b_sfname : "<None>");
1123#endif
1124
1125 strcpy(ffname, (char *) curbuf->b_ffname);
1126 *filename = ffname; /* copy so nobody can change b_ffname */
1127 *curLine = curwin->w_cursor.lnum;
1128 *curCol = curwin->w_cursor.col;
1129
1130 if (curbuf->b_visual_mode == 'v' &&
1131 equalpos(curwin->w_cursor, curbuf->b_visual_end))
1132 {
1133 *selStartLine = curbuf->b_visual_start.lnum;
1134 *selStartCol = curbuf->b_visual_start.col;
1135 *selEndLine = curbuf->b_visual_end.lnum;
1136 *selEndCol = curbuf->b_visual_end.col;
1137 *selection = get_selection(curbuf);
1138 if (*selection)
1139 *selLength = strlen(*selection);
1140 else
1141 *selLength = 0;
1142 }
1143 else
1144 {
1145 *selStartLine = *selEndLine = -1;
1146 *selStartCol = *selEndCol = -1;
1147 *selLength = 0;
1148 *selection = "";
1149 }
1150
1151 return True;
1152}
1153
1154
1155
1156/************************************************************************
1157 * Utility functions
1158 ************************************************************************/
1159
1160 static char *
1161get_selection(
1162 buf_T *buf) /* buffer whose selection we want */
1163{
1164 pos_T *start; /* start of the selection */
1165 pos_T *end; /* end of the selection */
1166 char *lp; /* pointer to actual line data */
1167 int llen; /* length of actual line data */
1168 char *sp; /* pointer to selection buffer */
1169 int slen; /* string length in selection buffer */
1170 int size; /* size of selection buffer */
1171 char *new_sp; /* temp pointer to new sp */
1172 int lnum; /* line number we are appending */
1173
1174 if (buf->b_visual_mode == 'v')
1175 {
1176 start = &buf->b_visual_start;
1177 end = &buf->b_visual_end;
1178 if (start->lnum == end->lnum)
1179 {
1180 /* selection is all on one line */
1181 lp = (char *) ml_get_pos(start);
1182 llen = end->col - start->col + 1;
1183 sp = (char *) malloc(llen + 1);
1184 if (sp != NULL)
1185 {
1186 strncpy(sp, lp, llen);
1187 sp[llen] = NUL;
1188 }
1189 }
1190 else
1191 {
1192 /* multi-line selection */
1193 lp = (char *) ml_get_pos(start);
1194 llen = strlen(lp);
1195 sp = (char *) malloc(BUFSIZ + llen);
1196 if (sp != NULL)
1197 {
1198 size = BUFSIZ + llen;
1199 strcpy(sp, lp);
1200 sp[llen] = '\n';
1201 slen = llen + 1;
1202
1203 lnum = start->lnum + 1;
1204 while (lnum < end->lnum)
1205 sp = append_selection(lnum++, sp, &size, &slen);
1206
1207 lp = (char *) ml_get(end->lnum);
1208 llen = end->col + 1;
1209 if ((slen + llen) >= size)
1210 {
1211 new_sp = (char *)
1212 realloc(sp, slen + llen + 1);
1213 if (new_sp != NULL)
1214 {
1215 size += llen + 1;
1216 sp = new_sp;
1217 }
1218 }
1219 if ((slen + llen) < size)
1220 {
1221 strncpy(&sp[slen], lp, llen);
1222 sp[slen + llen] = NUL;
1223 }
1224
1225 }
1226 }
1227 }
1228 else
1229 sp = NULL;
1230
1231 return sp;
1232}
1233
1234 static char *
1235append_selection(
1236 int lnum, /* line number to append */
1237 char *sp, /* pointer to selection buffer */
1238 int *size, /* ptr to size of sp */
1239 int *slen) /* ptr to length of selection string */
1240{
1241 char *lp; /* line of data from buffer */
1242 int llen; /* strlen of lp */
1243 char *new_sp; /* temp pointer to new sp */
1244
1245 lp = (char *)ml_get((linenr_T)lnum);
1246 llen = strlen(lp);
1247
1248 if ((*slen + llen) <= *size)
1249 {
1250 new_sp = (char *) realloc((void *) sp, BUFSIZ + *slen + llen);
1251 if (*new_sp != NUL)
1252 {
1253 *size = BUFSIZ + *slen + llen;
1254 sp = new_sp;
1255 }
1256 }
1257 if ((*slen + llen) > *size)
1258 {
1259 strcat(&sp[*slen], lp);
1260 *slen += llen;
1261 sp[*slen++] = '\n';
1262 }
1263
1264 return sp;
1265}
1266
1267
1268
1269 static void
1270load_buffer_by_name(
1271 char *filename, /* the file to load */
1272 int lnum) /* an optional line number (or 0) */
1273{
1274 char lnumbuf[16]; /* make line number option for :e */
1275 char cbuf[BUFSIZ]; /* command buffer */
1276
1277 if (lnum > 0)
1278 sprintf(lnumbuf, "+%d", lnum);
1279 else
1280 lnumbuf[0] = NUL;
1281
Bram Moolenaar051b7822005-05-19 21:00:46 +00001282 vim_snprintf(cbuf, sizeof(cbuf), "e %s %s", lnumbuf, filename);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001283 coloncmd(cbuf, False);
1284}
1285
1286
1287 static void
1288load_window(
1289 char *filename, /* filename to load */
1290 int lnum) /* linenumber to go to */
1291{
1292 buf_T *buf; /* buffer filename is stored in */
1293 win_T *win; /* window filenme is displayed in */
1294
1295 /*
1296 * Make sure filename is displayed and is the current window.
1297 */
1298
1299 buf = buflist_findname((char_u *)filename);
1300 if (buf == NULL || (win = get_window(buf)) == NULL)
1301 {
1302 /* No buffer or buffer is not in current window */
1303 /* wsdebug("load_window: load_buffer_by_name(\"%s\", %d)\n",
1304 filename, lnum); */
1305 load_buffer_by_name(filename, lnum);
1306 }
1307 else
1308 {
1309 /* buf is in a window */
1310 if (win != curwin)
1311 {
1312 win_enter(win, False);
1313 /* wsdebug("load_window: window endter %s\n",
1314 win->w_buffer->b_sfname); */
1315 }
1316 if (lnum > 0 && win->w_cursor.lnum != lnum)
1317 {
1318 warp_to_pc(lnum);
1319 /* wsdebug("load_window: warp to %s[%d]\n",
1320 win->w_buffer->b_sfname, lnum); */
1321 }
1322 }
1323 out_flush();
1324}
1325
1326
1327
1328 static void
1329warp_to_pc(
1330 int lnum) /* line number to warp to */
1331{
1332 char lbuf[256]; /* build line comand here */
1333
1334 if (lnum > 0)
1335 {
1336 if (State & INSERT)
1337 add_to_input_buf((char_u *) "\033", 1);
1338 if (isShowing(lnum))
1339 sprintf(lbuf, "%dG", lnum);
1340 else
1341 sprintf(lbuf, "%dz.", lnum);
1342 add_to_input_buf((char_u *) lbuf, strlen(lbuf));
1343 }
1344}
1345
1346 static Boolean
1347isShowing(
1348 int lnum) /* tell if line number is showing */
1349{
1350 return lnum >= curwin->w_topline && lnum < curwin->w_botline;
1351}
1352
1353
1354
1355 static win_T *
1356get_window(
1357 buf_T *buf) /* buffer to find window for */
1358{
1359 win_T *wp = NULL; /* window filename is in */
1360
1361 for (wp = firstwin; wp != NULL; wp = W_NEXT(wp))
1362 if (buf == wp->w_buffer)
1363 break;
1364 return wp;
1365}
1366
1367
1368#if 0 /* not used */
1369 static int
1370get_buffer_number(
1371 buf_T *buf) /* buffer to get position of */
1372{
1373 buf_T *bp; /* iterate over buffer list */
1374 int pos; /* the position in the buffer list */
1375
1376 pos = 1;
1377 for (bp = firstbuf; bp != NULL; bp = bp->b_next)
1378 {
1379 if (bp == buf)
1380 return pos;
1381 pos++;
1382 }
1383
1384 return 1;
1385}
1386#endif
1387
1388 static void
1389updatePriority(
1390 Boolean subMenu) /* if True then start new submenu pri */
1391{
1392 int pri; /* priority of this menu/item */
1393 char *p;
1394
1395 p = strrchr(curMenuPriority, '.');
1396 ASSERT(p != NULL);
1397 *p++ = NUL;
1398
1399 pri = atoi(p) + 10; /* our new priority */
1400
1401 if (subMenu)
Bram Moolenaar051b7822005-05-19 21:00:46 +00001402 vim_snprintf(curMenuPriority, sizeof(curMenuPriority),
1403 "%s.%d.0", curMenuPriority, pri);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001404 else
Bram Moolenaar051b7822005-05-19 21:00:46 +00001405 vim_snprintf(curMenuPriority, sizeof(curMenuPriority),
1406 "%s.%d", curMenuPriority, pri);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001407}
1408
1409 static char *
1410addUniqueMnemonic(
1411 char *mnemonics, /* currently used mnemonics */
1412 char *label) /* label of menu needing mnemonic */
1413{
1414 static char name[BUFSIZ]; /* buffer for the updated name */
1415 char *p; /* pointer into label */
1416 char *found; /* pointer to possible mnemonic */
1417
1418 found = NULL;
1419 for (p = label; *p != NUL; p++)
1420 if (strchr(mnemonics, *p) == 0)
1421 if (found == NULL || (isupper((int)*p) && islower((int)*found)))
1422 found = p;
1423
1424 if (found != NULL)
1425 {
1426 strncpy(name, label, (found - label));
1427 strcat(name, "&");
1428 strcat(name, found);
1429 }
1430 else
1431 strcpy(name, label);
1432
1433 return name;
1434}
1435
1436/*
1437 * Some characters in a menu name must be escaped in vim. Since this is vim
1438 * specific, it must be done on this side.
1439 */
1440 static char *
1441fixup(
1442 char *label)
1443{
1444 static char buf[BUFSIZ];
1445 char *bp; /* pointer into buf */
1446 char *lp; /* pointer into label */
1447
1448 lp = label;
1449 bp = buf;
1450 while (*lp != NUL)
1451 {
1452 if (*lp == ' ' || *lp == '.')
1453 *bp++ = '\\';
1454 *bp++ = *lp++;
1455 }
1456 *bp = NUL;
1457
1458 return buf;
1459}
1460
1461
1462#ifdef NOHANDS_SUPPORT_FUNCTIONS
1463
1464/* For the NoHands test suite */
1465
1466 char *
1467workshop_test_getcurrentfile()
1468{
1469 char *filename, *selection;
1470 int curLine, curCol, selStartLine, selStartCol, selEndLine;
1471 int selEndCol, selLength;
1472
1473 if (workshop_get_positions(
1474 NULL, &filename, &curLine, &curCol, &selStartLine,
1475 &selStartCol, &selEndLine, &selEndCol, &selLength,
1476 &selection))
1477 return filename;
1478 else
1479 return NULL;
1480}
1481
1482 int
1483workshop_test_getcursorrow()
1484{
1485 return 0;
1486}
1487
1488 int
1489workshop_test_getcursorcol()
1490{
1491 char *filename, *selection;
1492 int curLine, curCol, selStartLine, selStartCol, selEndLine;
1493 int selEndCol, selLength;
1494
1495 if (workshop_get_positions(
1496 NULL, &filename, &curLine, &curCol, &selStartLine,
1497 &selStartCol, &selEndLine, &selEndCol, &selLength,
1498 &selection))
1499 return curCol;
1500 else
1501 return -1;
1502}
1503
1504 char *
1505workshop_test_getcursorrowtext()
1506{
1507 return NULL;
1508}
1509
1510 char *
1511workshop_test_getselectedtext()
1512{
1513 char *filename, *selection;
1514 int curLine, curCol, selStartLine, selStartCol, selEndLine;
1515 int selEndCol, selLength;
1516
1517 if (workshop_get_positions(
1518 NULL, &filename, &curLine, &curCol, &selStartLine,
1519 &selStartCol, &selEndLine, &selEndCol, &selLength,
1520 &selection))
1521 return selection;
1522 else
1523 return NULL;
1524}
1525
1526/*ARGSUSED*/
1527 void
1528workshop_save_sensitivity(char *filename)
1529{
1530}
1531
1532#endif
1533
1534 static char *
1535fixAccelText(
1536 char *ap) /* original acceleratorText */
1537{
1538 char buf[256]; /* build in temp buffer */
1539 char *shift; /* shift string of "" */
1540
1541 if (ap == NULL)
1542 return NULL;
1543
1544 /* If the accelerator is shifted use the vim form */
1545 if (strncmp("Shift+", ap, 6) == 0)
1546 {
1547 shift = "S-";
1548 ap += 6;
1549 }
1550 else
1551 shift = "";
1552
1553 if (*ap == 'F' && atoi(&ap[1]) > 0)
1554 {
Bram Moolenaar051b7822005-05-19 21:00:46 +00001555 vim_snprintf(buf, sizeof(buf), "<%s%s>", shift, ap);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001556 return strdup(buf);
1557 }
1558 else
1559 return NULL;
1560}
1561
1562#ifdef FEAT_BEVAL
Bram Moolenaare4efc3b2005-03-07 23:16:51 +00001563 void
1564workshop_beval_cb(
Bram Moolenaar071d4272004-06-13 20:20:40 +00001565 BalloonEval *beval,
1566 int state)
1567{
Bram Moolenaare4efc3b2005-03-07 23:16:51 +00001568 win_T *wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001569 char_u *text;
1570 int type;
Bram Moolenaare4efc3b2005-03-07 23:16:51 +00001571 linenr_T lnum;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001572 int col;
1573 int idx;
1574 char buf[MAXPATHLEN * 2];
1575 static int serialNo = -1;
1576
1577 if (!p_beval)
1578 return;
1579
Bram Moolenaare4efc3b2005-03-07 23:16:51 +00001580 if (get_beval_info(beval, FALSE, &wp, &lnum, &text, &col) == OK)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001581 {
1582 if (text && text[0])
1583 {
1584 /* Send debugger request */
1585 if (strlen((char *) text) > (MAXPATHLEN/2))
1586 {
1587 /*
1588 * The user has probably selected the entire
1589 * buffer or something like that - don't attempt
1590 * to evaluate it
1591 */
1592 return;
1593 }
1594
1595 /*
1596 * WorkShop expects the col to be a character index, not
1597 * a column number. Compute the index from col. Also set
1598 * line to 0 because thats what dbx expects.
1599 */
1600 idx = computeIndex(col, text, beval->ts);
1601 if (idx > 0)
1602 {
Bram Moolenaare4efc3b2005-03-07 23:16:51 +00001603 lnum = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001604
1605 /*
1606 * If successful, it will respond with a balloon cmd.
1607 */
1608 if (state & ControlMask)
1609 /* Evaluate *(expression) */
1610 type = (int)GPLineEval_INDIRECT;
1611 else if (state & ShiftMask)
1612 /* Evaluate type(expression) */
1613 type = (int)GPLineEval_TYPE;
1614 else
1615 /* Evaluate value(expression) */
1616 type = (int)GPLineEval_EVALUATE;
1617
1618 /* Send request to dbx */
Bram Moolenaar051b7822005-05-19 21:00:46 +00001619 vim_snprintf(buf, sizeof(buf), "toolVerb debug.balloonEval "
Bram Moolenaare4efc3b2005-03-07 23:16:51 +00001620 "%s %ld,0 %d,0 %d,%d %ld %s\n",
1621 (char *)wp->w_buffer->b_ffname,
1622 (long)lnum, idx, type, serialNo++,
1623 (long)strlen((char *)text), (char *)text);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001624 balloonEval = beval;
1625 workshop_send_message(buf);
1626 }
1627 }
1628 }
1629}
1630#endif
1631
1632
1633 static int
1634computeIndex(
1635 int wantedCol,
1636 char_u *line,
1637 int ts)
1638{
1639 int col = 0;
1640 int idx = 0;
1641
1642 while (line[idx])
1643 {
1644 if (line[idx] == '\t')
1645 col += ts - (col % ts);
1646 else
1647 col++;
1648 idx++;
1649 if (col >= wantedCol)
1650 return idx;
1651 }
1652
1653 return -1;
1654}
1655
1656 static void
1657addMenu(
1658 char *menu, /* menu name */
1659 char *accel, /* accelerator text (optional) */
1660 char *verb) /* WorkShop action-verb */
1661{
1662 MenuMap *newMap;
1663 char cbuf[BUFSIZ];
1664
1665 if (menuMapSize >= menuMapMax)
1666 {
1667 newMap = realloc(menuMap,
1668 sizeof(MenuMap) * (menuMapMax + MENU_INC));
1669 if (newMap != NULL)
1670 {
1671 menuMap = newMap;
1672 menuMapMax += MENU_INC;
1673 }
1674 }
1675 if (menuMapSize < menuMapMax)
1676 {
1677 menuMap[menuMapSize].name = strdup(menu);
1678 menuMap[menuMapSize].accel = accel && *accel ? strdup(accel) : NULL;
1679 menuMap[menuMapSize++].verb = strdup(verb);
1680 if (accel && workshopHotKeysEnabled)
1681 {
Bram Moolenaar051b7822005-05-19 21:00:46 +00001682 vim_snprintf(cbuf, sizeof(cbuf),
1683 "map %s :wsverb %s<CR>", accel, verb);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001684 coloncmd(cbuf, TRUE);
1685 }
1686 }
1687}
1688
1689 static char *
1690nameStrip(
1691 char *raw) /* menu name, possibly with & chars */
1692{
1693 static char buf[BUFSIZ]; /* build stripped name here */
1694 char *bp = buf;
1695
1696 while (*raw)
1697 {
1698 if (*raw != '&')
1699 *bp++ = *raw;
1700 raw++;
1701 }
1702 *bp = NUL;
1703 return buf;
1704}
1705
1706
1707 static char *
1708lookupVerb(
1709 char *verb,
1710 int skip) /* number of matches to skip */
1711{
1712 int i; /* loop iterator */
1713
1714 for (i = 0; i < menuMapSize; i++)
1715 if (strcmp(menuMap[i].verb, verb) == 0 && skip-- == 0)
1716 return nameStrip(menuMap[i].name);
1717
1718 return NULL;
1719}
1720
1721
1722 static void
1723coloncmd(
1724 char *cmd, /* the command to print */
1725 Boolean force) /* force cursor update */
1726{
1727 char_u *cpo_save = p_cpo;
1728
1729#ifdef WSDEBUG
1730 if (WSDLEVEL(WS_TRACE_COLONCMD))
1731 wsdebug("Cmd: %s\n", cmd);
1732#endif
1733
1734 p_cpo = empty_option;
1735
1736 ALT_INPUT_LOCK_ON;
1737 do_cmdline_cmd((char_u *)cmd);
1738 ALT_INPUT_LOCK_OFF;
1739
1740 p_cpo = cpo_save;
1741
1742 if (force)
1743 gui_update_screen();
1744}
1745
1746/*
1747 * setDollarVim - Given the run directory, search for the vim install
1748 * directory and set $VIM.
1749 *
1750 * We can be running out of SUNWspro/bin or out of
1751 * SUNWspro/contrib/contrib6/vim5.6/bin so we check
1752 * relative to both of these directories.
1753 */
1754 static void
1755setDollarVim(
1756 char *rundir)
1757{
1758 char buf[MAXPATHLEN];
1759 char *cp;
1760
1761 /*
1762 * First case: Running from <install-dir>/SUNWspro/bin
1763 */
1764 strcpy(buf, rundir);
1765 strcat(buf, "/../contrib/contrib6/vim" VIM_VERSION_SHORT "/share/vim/"
1766 VIM_VERSION_NODOT "/syntax/syntax.vim");
1767 if (access(buf, R_OK) == 0)
1768 {
1769 strcpy(buf, "SPRO_WSDIR=");
1770 strcat(buf, rundir);
1771 cp = strrchr(buf, '/');
1772 if (cp != NULL)
1773 strcpy(cp, "/WS6U2");
1774 putenv(strdup(buf));
1775
1776 strcpy(buf, "VIM=");
1777 strcat(buf, rundir);
1778 strcat(buf, "/../contrib/contrib6/vim" VIM_VERSION_SHORT "/share/vim/"
1779 VIM_VERSION_NODOT);
1780 putenv(strdup(buf));
1781 return;
1782 }
1783
1784 /*
1785 * Second case: Probably running from
1786 * <install-dir>/SUNWspro/contrib/contrib6/vim5.6/bin
1787 */
1788 strcpy(buf, rundir);
1789 strcat(buf, "/../../../contrib/contrib6/vim" VIM_VERSION_SHORT
1790 "/share/vim/" VIM_VERSION_NODOT "/syntax/syntax.vim");
1791 if (access(buf, R_OK) == 0)
1792 {
1793 strcpy(buf, "SPRO_WSDIR=");
1794 strcat(buf, rundir);
1795 cp = strrchr(buf, '/');
1796 if (cp != NULL)
1797 strcpy(cp, "../../../../WS6U2");
1798 putenv(strdup(buf));
1799
1800 strcpy(buf, "VIM=");
1801 strcat(buf, rundir);
1802 strcat(buf, "/../../../contrib/contrib6/vim" VIM_VERSION_SHORT
1803 "/share/vim/" VIM_VERSION_NODOT);
1804 putenv(strdup(buf));
1805 return;
1806 }
1807}
1808
1809/*
1810 * findYourself - Find the directory we are running from. This is used to
1811 * set $VIM. We need to set this because users can install
1812 * the package in a different directory than the compiled
1813 * directory. This is a Sun Visual WorkShop requirement!
1814 *
1815 * Note: We override a user's $VIM because it won't have the
1816 * WorkShop specific files. S/he may not like this but its
1817 * better than getting the wrong files (especially as the
1818 * user is likely to have $VIM set to 5.4 or later).
1819 */
1820 void
1821findYourself(
1822 char *argv0)
1823{
1824 char *runpath = NULL;
1825 char *path;
1826 char *pathbuf;
1827
1828 if (*argv0 == '/')
1829 runpath = strdup(argv0);
1830 else if (*argv0 == '.' || strchr(argv0, '/'))
1831 {
1832 runpath = (char *) malloc(MAXPATHLEN);
1833 getcwd(runpath, MAXPATHLEN);
1834 strcat(runpath, "/");
1835 strcat(runpath, argv0);
1836 }
1837 else
1838 {
1839 path = getenv("PATH");
1840 if (path != NULL)
1841 {
1842 runpath = (char *) malloc(MAXPATHLEN);
1843 pathbuf = strdup(path);
1844 path = strtok(pathbuf, ":");
1845 do
1846 {
1847 strcpy(runpath, path);
1848 strcat(runpath, "/");
1849 strcat(runpath, argv0);
1850 if (access(runpath, X_OK) == 0)
1851 break;
1852 } while ((path = strtok(NULL, ":")) != NULL);
1853 free(pathbuf);
1854 }
1855 }
1856
1857 if (runpath != NULL)
1858 {
1859 char runbuf[MAXPATHLEN];
1860
1861 /*
1862 * We found the run directory. Now find the install dir.
1863 */
1864 (void)vim_FullName((char_u *)runpath, (char_u *)runbuf, MAXPATHLEN, 1);
1865 path = strrchr(runbuf, '/');
1866 if (path != NULL)
1867 *path = NUL; /* remove the vim/gvim name */
1868 path = strrchr(runbuf, '/');
1869 if (path != NULL)
1870 {
1871 if (strncmp(path, "/bin", 4) == 0)
1872 setDollarVim(runbuf);
1873 else if (strncmp(path, "/src", 4) == 0)
1874 {
1875 *path = NUL; /* development tree */
1876 setDollarVim(runbuf);
1877 }
1878 }
1879 free(runpath);
1880 }
1881}