Mercurial > emacs
diff src/keyboard.c @ 10861:655e3daa560c
(cmd_error): Use clear_prefix_arg.
(internal_last_event_frame, Vlast_event_frame): Normal vars again.
All uses changed.
(Quniversal_argument, Qdigit_argument, Qnegative_argument): Declare.
(clear_prefix_arg, finalize_prefix_arg, describe_prefix_arg): New fns.
(command_loop_1): Handle digits and minus specially, when they're
part of a prefix arg.
Handle universal-argument and digit-argument and negative-argument
bindings here, rather than doing I/O in the Lisp code.
(read_char): When reading switch-frame events from the side queue,
set internal_last_event_frame.
(readable_events): Return non-zero if a side queue has data.
(kbd_buffer_get_event): Don't abort if event has no associated frame.
(read_key_sequence): Improve behavior when there's no current display.
(init_perdisplay): Initialize the new members.
author | Karl Heuer <kwzh@gnu.org> |
---|---|
date | Wed, 01 Mar 1995 04:27:37 +0000 |
parents | 06d6b2e17987 |
children | ff1b5efecdb0 |
line wrap: on
line diff
--- a/src/keyboard.c Wed Mar 01 03:45:45 1995 +0000 +++ b/src/keyboard.c Wed Mar 01 04:27:37 1995 +0000 @@ -259,6 +259,20 @@ /* The buffer that was current when the last command was started. */ Lisp_Object last_point_position_buffer; +#ifdef MULTI_FRAME +/* The frame in which the last input event occurred, or Qmacro if the + last event came from a macro. We use this to determine when to + generate switch-frame events. This may be cleared by functions + like Fselect_frame, to make sure that a switch-frame event is + generated by the next character. */ +Lisp_Object internal_last_event_frame; +#endif + +/* A user-visible version of the above, intended to allow users to + figure out where the last event came from, if the event doesn't + carry that information itself (i.e. if it was a character). */ +Lisp_Object Vlast_event_frame; + /* The timestamp of the last input event we received from the X server. X Windows wants this for selection ownership. */ unsigned long last_event_timestamp; @@ -267,6 +281,7 @@ Lisp_Object Qforward_char; Lisp_Object Qbackward_char; Lisp_Object Qundefined; +Lisp_Object Quniversal_argument, Qdigit_argument, Qnegative_argument; /* read_key_sequence stores here the command definition of the key sequence that it reads. */ @@ -753,9 +768,7 @@ Vstandard_output = Qt; Vstandard_input = Qt; Vexecuting_macro = Qnil; - if (!current_perdisplay) - abort (); - current_perdisplay->Vprefix_arg = Qnil; + clear_prefix_arg (); cancel_echoing (); cmd_error_internal (data, 0); @@ -943,6 +956,43 @@ error ("No recursive edit is in progress"); } +void +clear_prefix_arg () +{ + if (!current_perdisplay) + abort (); + current_perdisplay->prefix_factor = Qnil; + current_perdisplay->prefix_value = Qnil; + current_perdisplay->prefix_sign = 1; + current_perdisplay->prefix_partial = 0; + Vprefix_arg = Qnil; +} + +static void +finalize_prefix_arg () +{ + if (!NILP (current_perdisplay->prefix_factor)) + Vprefix_arg = Fcons (current_perdisplay->prefix_factor, Qnil); + else if (NILP (current_perdisplay->prefix_value)) + Vprefix_arg = (current_perdisplay->prefix_sign > 0 ? Qnil : Qminus); + else if (current_perdisplay->prefix_sign > 0) + Vprefix_arg = current_perdisplay->prefix_value; + else + XSETINT (Vprefix_arg, -XINT (current_perdisplay->prefix_value)); + current_perdisplay->prefix_partial = 0; +} + +static void +describe_prefix_arg () +{ + if (INTEGERP (Vprefix_arg)) + message ("Arg: %d", Vprefix_arg); + else if (CONSP (Vprefix_arg)) + message ("Arg: [%d]", XCONS (Vprefix_arg)->car); + else if (EQ (Vprefix_arg, Qminus)) + message ("Arg: -"); +} + /* This is the actual command reading loop, sans error-handling encapsulation. */ @@ -1096,6 +1146,16 @@ last_point_position = PT; XSETBUFFER (last_point_position_buffer, prev_buffer); + /* If we're building a prefix argument, override minus and digits. */ + if (current_perdisplay->prefix_partial && i == 1 && NATNUMP (keybuf[0])) + { + if (XFASTINT (keybuf[0]) == '-' + && NILP (current_perdisplay->prefix_value)) + cmd = Qnegative_argument; + else if (XFASTINT (keybuf[0]) >= '0' && XFASTINT (keybuf[0]) <= '9') + cmd = Qdigit_argument; + } + /* Execute the command. */ this_command = cmd; @@ -1110,12 +1170,56 @@ bitch_at_user (); defining_kbd_macro = 0; update_mode_lines = 1; - current_perdisplay->Vprefix_arg = Qnil; - + clear_prefix_arg (); } else { - if (NILP (current_perdisplay->Vprefix_arg) && ! no_direct) + if (EQ (cmd, Quniversal_argument)) + { + if (!current_perdisplay->prefix_partial) + { + /* First C-u */ + XSETFASTINT (current_perdisplay->prefix_factor, 4); + current_perdisplay->prefix_value = Qnil; + current_perdisplay->prefix_sign = 1; + current_perdisplay->prefix_partial = 1; + } + else if (!NILP (current_perdisplay->prefix_factor)) + { + /* Subsequent C-u */ + XSETINT (current_perdisplay->prefix_factor, + XINT (current_perdisplay->prefix_factor) * 4); + } + else + { + /* Terminating C-u */ + finalize_prefix_arg (); + describe_prefix_arg (); + } + goto directly_done; + } + else if (EQ (cmd, Qnegative_argument)) + { + current_perdisplay->prefix_factor = Qnil; + current_perdisplay->prefix_sign *= -1; + current_perdisplay->prefix_partial = 1; + goto directly_done; + } + else if (EQ (cmd, Qdigit_argument) && INTEGERP (keybuf[0])) + { + current_perdisplay->prefix_factor = Qnil; + if (NILP (current_perdisplay->prefix_value)) + XSETFASTINT (current_perdisplay->prefix_value, 0); + XSETINT (current_perdisplay->prefix_value, + (XINT (current_perdisplay->prefix_value) * 10 + + (XINT (keybuf[0]) & 0177) - '0')); + current_perdisplay->prefix_partial = 1; + goto directly_done; + } + if (current_perdisplay->prefix_partial) + finalize_prefix_arg (); + + if (NILP (Vprefix_arg) && ! no_direct) { /* Recognize some common commands in common situations and do them directly. */ @@ -1238,7 +1342,7 @@ /* Here for a command that isn't executed directly */ nonundocount = 0; - if (NILP (current_perdisplay->Vprefix_arg)) + if (NILP (Vprefix_arg)) Fundo_boundary (); Fcommand_execute (this_command, Qnil); @@ -1262,7 +1366,7 @@ 3) we want to leave this_command_key_count non-zero, so that read_char will realize that it is re-reading a character, and not echo it a second time. */ - if (NILP (current_perdisplay->Vprefix_arg)) + if (NILP (Vprefix_arg)) { last_command = this_command; cancel_echoing (); @@ -1283,7 +1387,7 @@ finalize: /* Install chars successfully executed in kbd macro. */ - if (defining_kbd_macro && NILP (current_perdisplay->Vprefix_arg)) + if (defining_kbd_macro && NILP (Vprefix_arg)) finalize_kbd_macro_chars (); #ifdef MULTI_PERDISPLAY @@ -1546,10 +1650,7 @@ internal_last_event_frame after each command is read, but events read from a macro should never cause a new frame to be selected. */ - if (!current_perdisplay) - abort (); - current_perdisplay->internal_last_event_frame = Qmacro; - current_perdisplay->Vlast_event_frame = Qmacro; + Vlast_event_frame = internal_last_event_frame = Qmacro; #endif /* Exit the macro if we are at the end. @@ -1600,10 +1701,8 @@ { XSETINT (c, quit_char); #ifdef MULTI_FRAME - XSETFRAME (current_perdisplay->internal_last_event_frame, - selected_frame); - current_perdisplay->Vlast_event_frame - = current_perdisplay->internal_last_event_frame; + XSETFRAME (internal_last_event_frame, selected_frame); + Vlast_event_frame = internal_last_event_frame; #endif /* If we report the quit char as an event, don't do so more than once. */ @@ -1766,6 +1865,13 @@ { c = XCONS (perd->kbd_queue)->car; perd->kbd_queue = XCONS (perd->kbd_queue)->cdr; + input_pending = readable_events (); +#ifdef MULTI_FRAME + if (EVENT_HAS_PARAMETERS (c) + && EQ (EVENT_HEAD_KIND (EVENT_HEAD (c)), Qswitch_frame)) + internal_last_event_frame = XCONS (XCONS (c)->cdr)->car; + Vlast_event_frame = internal_last_event_frame; +#endif } else { @@ -2081,6 +2187,18 @@ if (FRAMEP (do_mouse_tracking) && mouse_moved) return 1; #endif + if (display_locked) + { + if (CONSP (current_perdisplay->kbd_queue)) + return 1; + } + else + { + PERDISPLAY *perd; + for (perd = all_perdisplays; perd; perd = perd->next_perdisplay) + if (CONSP (perd->kbd_queue)) + return 1; + } return 0; } @@ -2118,14 +2236,12 @@ will set Vlast_event_frame again, so this is safe to do. */ { Lisp_Object focus; - PERDISPLAY *perd; focus = FRAME_FOCUS_FRAME (XFRAME (event->frame_or_window)); if (NILP (focus)) focus = event->frame_or_window; - perd = get_perdisplay (XFRAME (focus)); - perd->internal_last_event_frame = focus; - perd->Vlast_event_frame = focus; + internal_last_event_frame = focus; + Vlast_event_frame = focus; } #endif @@ -2261,9 +2377,12 @@ frame = XCONS (frame)->car; else if (WINDOWP (frame)) frame = WINDOW_FRAME (XWINDOW (frame)); + /* There are still some events that don't set this field. + For now, just ignore the problem. */ if (!FRAMEP (frame)) - abort (); - *perdp = get_perdisplay (XFRAME (frame)); + *perdp = all_perdisplays; + else + *perdp = get_perdisplay (XFRAME (frame)); } obj = Qnil; @@ -2357,10 +2476,10 @@ if (! NILP (focus)) frame = focus; - if (! EQ (frame, (*perdp)->internal_last_event_frame) + if (! EQ (frame, internal_last_event_frame) && XFRAME (frame) != selected_frame) obj = make_lispy_switch_frame (frame); - (*perdp)->internal_last_event_frame = frame; + internal_last_event_frame = frame; #endif /* MULTI_FRAME */ /* If we didn't decide to make a switch-frame event, go ahead @@ -2412,10 +2531,10 @@ if (NILP (frame)) XSETFRAME (frame, f); - if (! EQ (frame, (*perdp)->internal_last_event_frame) + if (! EQ (frame, internal_last_event_frame) && XFRAME (frame) != selected_frame) obj = make_lispy_switch_frame (frame); - (*perdp)->internal_last_event_frame = frame; + internal_last_event_frame = frame; } #endif @@ -2433,7 +2552,7 @@ input_pending = readable_events (); #ifdef MULTI_FRAME - (*perdp)->Vlast_event_frame = (*perdp)->internal_last_event_frame; + Vlast_event_frame = internal_last_event_frame; #endif return (obj); @@ -4801,7 +4920,7 @@ /* Record the initial state of the echo area and this_command_keys; we will need to restore them if we replay a key sequence. */ if (INTERACTIVE) - echo_start = echo_length (); + echo_start = (current_perdisplay ? echo_length () : 0); keys_start = this_command_key_count; #if defined (GOBBLE_FIRST_EVENT) @@ -4875,7 +4994,7 @@ /* These are no-ops the first time through, but if we restart, they revert the echo area and this_command_keys to their original state. */ this_command_key_count = keys_start; - if (INTERACTIVE && t < mock_input) + if (INTERACTIVE && t < mock_input && current_perdisplay) echo_truncate (echo_start); /* If the best binding for the current key sequence is a keymap, or @@ -5687,9 +5806,9 @@ struct backtrace backtrace; extern int debug_on_next_call; - prefixarg = current_perdisplay->Vprefix_arg; - current_perdisplay->Vprefix_arg = Qnil; - current_perdisplay->Vcurrent_prefix_arg = prefixarg; + prefixarg = Vprefix_arg; + clear_prefix_arg (); + Vcurrent_prefix_arg = prefixarg; debug_on_next_call = 0; if (SYMBOLP (cmd)) @@ -5805,7 +5924,7 @@ UNGCPRO; function = Fintern (function, Qnil); - current_perdisplay->Vprefix_arg = prefixarg; + Vprefix_arg = prefixarg; this_command = function; return Fcommand_execute (function, Qt); @@ -6183,15 +6302,9 @@ abort (); #endif #ifdef MULTI_FRAME - { - Lisp_Object frame; - - if (!current_perdisplay) - abort (); - frame = current_perdisplay->internal_last_event_frame; - if (FRAMEP (frame) && XFRAME (frame) != selected_frame) - Fhandle_switch_frame (make_lispy_switch_frame (frame)); - } + if (FRAMEP (internal_last_event_frame) + && XFRAME (internal_last_event_frame) != selected_frame) + Fhandle_switch_frame (make_lispy_switch_frame (internal_last_event_frame)); #endif _longjmp (getcjmp, 1); @@ -6289,15 +6402,11 @@ init_perdisplay (perd) PERDISPLAY *perd; { - perd->Vprefix_arg = Qnil; - perd->Vcurrent_prefix_arg = Qnil; -#ifdef MULTI_FRAME - /* This means that command_loop_1 won't try to select anything the first - time through. */ - perd->internal_last_event_frame = Qnil; -#endif + perd->prefix_factor = Qnil; + perd->prefix_value = Qnil; + perd->prefix_sign = 1; + perd->prefix_partial = 0; perd->kbd_queue = Qnil; - perd->Vlast_event_frame = Qnil; perd->immediate_echo = 0; perd->echoptr = perd->echobuf; perd->echo_after_prompt = -1; @@ -6335,6 +6444,13 @@ #endif input_pending = 0; +#ifdef MULTI_FRAME + /* This means that command_loop_1 won't try to select anything the first + time through. */ + internal_last_event_frame = Qnil; + Vlast_event_frame = internal_last_event_frame; +#endif + #ifndef MULTI_PERDISPLAY if (initialized) wipe_perdisplay (&the_only_perdisplay); @@ -6424,6 +6540,15 @@ Qundefined = intern ("undefined"); staticpro (&Qundefined); + Quniversal_argument = intern ("universal-argument"); + staticpro (&Quniversal_argument); + + Qdigit_argument = intern ("digit-argument"); + staticpro (&Qdigit_argument); + + Qnegative_argument = intern ("negative-argument"); + staticpro (&Qnegative_argument); + Qpre_command_hook = intern ("pre-command-hook"); staticpro (&Qpre_command_hook); @@ -6641,6 +6766,11 @@ "Number of complete keys read from the keyboard so far."); num_input_keys = 0; + DEFVAR_LISP ("last-event-frame", &Vlast_event_frame, + "The frame in which the most recently read event occurred.\n\ +If the last event came from a keyboard macro, this is set to `macro'."); + Vlast_event_frame = Qnil; + DEFVAR_LISP ("help-char", &Vhelp_char, "Character to recognize as meaning Help.\n\ When it is read, do `(eval help-form)', and display result if it's a string.\n\ @@ -6781,28 +6911,6 @@ This function is called with no arguments after each command\n\ whenever `deferred-action-list' is non-nil."); Vdeferred_action_function = Qnil; - - DEFVAR_DISPLAY ("prefix-arg", Vprefix_arg, - "The value of the prefix argument for the next editing command.\n\ -It may be a number, or the symbol `-' for just a minus sign as arg,\n\ -or a list whose car is a number for just one or more C-U's\n\ -or nil if no argument has been specified.\n\ -\n\ -You cannot examine this variable to find the argument for this command\n\ -since it has been set to nil by the time you can look.\n\ -Instead, you should use the variable `current-prefix-arg', although\n\ -normally commands can get this prefix argument with (interactive \"P\")."); - - DEFVAR_DISPLAY ("current-prefix-arg", Vcurrent_prefix_arg, - "The value of the prefix argument for this editing command.\n\ -It may be a number, or the symbol `-' for just a minus sign as arg,\n\ -or a list whose car is a number for just one or more C-U's\n\ -or nil if no argument has been specified.\n\ -This is what `(interactive \"P\")' returns."); - - DEFVAR_DISPLAY ("last-event-frame", Vlast_event_frame, - "The frame in which the most recently read event occurred.\n\ -If the last event came from a keyboard macro, this is set to `macro'."); } keys_of_keyboard ()