📄 editfns.c
字号:
/* Lisp functions pertaining to editing. Copyright (C) 1985, 1986, 1987, 1990 Free Software Foundation, Inc.This file is part of GNU Emacs.GNU Emacs is free software; you can redistribute it and/or modifyit under the terms of the GNU General Public License as published bythe Free Software Foundation; either version 1, or (at your option)any later version.GNU Emacs is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY; without even the implied warranty ofMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See theGNU General Public License for more details.You should have received a copy of the GNU General Public Licensealong with GNU Emacs; see the file COPYING. If not, write tothe Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */#include "config.h"#ifdef hpux/* needed by <pwd.h> */#include <stdio.h>#undef NULL#endif#include <pwd.h>#include "lisp.h"#include "buffer.h"#include "window.h"#define min(a, b) ((a) < (b) ? (a) : (b))#define max(a, b) ((a) > (b) ? (a) : (b))/* Some static data, and a function to initialize it for each run */static Lisp_Object Vsystem_name;static Lisp_Object Vuser_real_name; /* login name of current user ID */static Lisp_Object Vuser_full_name; /* full name of current user */static Lisp_Object Vuser_name; /* user name from USER or LOGNAME. */voidinit_editfns (){ char *user_name; register unsigned char *p, *q; struct passwd *pw; /* password entry for the current user */ Lisp_Object tem; extern char *index (); /* Set up system_name even when dumping. */ Vsystem_name = build_string (get_system_name ()); p = XSTRING (Vsystem_name)->data; while (*p) { if (*p == ' ' || *p == '\t') *p = '-'; p++; }#ifndef CANNOT_DUMP /* Don't bother with this on initial start when just dumping out */ if (!initialized) return;#endif /* not CANNOT_DUMP */ pw = (struct passwd *) getpwuid (getuid ()); Vuser_real_name = build_string (pw ? pw->pw_name : "unknown"); user_name = (char *) getenv ("USER"); if (!user_name) user_name = (char *) getenv ("LOGNAME"); /* USG equivalent */ if (user_name) Vuser_name = build_string (user_name); else Vuser_name = Vuser_real_name; tem = Fstring_equal (Vuser_name, Vuser_real_name); if (!NULL (tem)) pw = (struct passwd *) getpwnam (user_name); p = (unsigned char *) (pw ? USER_FULL_NAME : "unknown"); q = (unsigned char *) index (p, ','); Vuser_full_name = make_string (p, q ? q - p : strlen (p));#ifdef AMPERSAND_FULL_NAME p = XSTRING (Vuser_full_name)->data; q = (unsigned char *) index (p, '&'); /* Substitute the login name for the &, upcasing the first character. */ if (q) { char *r = (char *) alloca (strlen (p) + XSTRING (Vuser_name)->size + 1); bcopy (p, r, q - p); r[q - p] = 0; strcat (r, XSTRING (Vuser_real_name)->data); r[q - p] = UPCASE (r[q - p]); strcat (r, q + 1); Vuser_full_name = build_string (r); }#endif /* AMPERSAND_FULL_NAME */}DEFUN ("char-to-string", Fchar_to_string, Schar_to_string, 1, 1, 0, "Convert arg CHAR to a string containing that character.") (n) Lisp_Object n;{ char c; CHECK_NUMBER (n, 0); c = XINT (n); return make_string (&c, 1);}DEFUN ("string-to-char", Fstring_to_char, Sstring_to_char, 1, 1, 0, "Convert arg STRING to a character, the first character of that string.") (str) register Lisp_Object str;{ register Lisp_Object val; register struct Lisp_String *p; CHECK_STRING (str, 0); p = XSTRING (str); if (p->size) XFASTINT (val) = ((unsigned char *) p->data)[0]; else XFASTINT (val) = 0; return val;}static Lisp_Objectbuildmark (val) int val;{ register Lisp_Object mark; mark = Fmake_marker (); Fset_marker (mark, make_number (val), Qnil); return mark;}DEFUN ("point", Fpoint, Spoint, 0, 0, 0, "Return value of point, as an integer.\n\Beginning of buffer is position (point-min)") (){ Lisp_Object temp; XFASTINT (temp) = point; return temp;}DEFUN ("point-marker", Fpoint_marker, Spoint_marker, 0, 0, 0, "Return value of point, as a marker object.") (){ return buildmark (point);}intin_accessible_range (loc) int loc;{ if (loc < BEGV) return BEGV; if (loc > ZV) return ZV; return loc;}DEFUN ("goto-char", Fgoto_char, Sgoto_char, 1, 1, "NGoto char: ", "One arg, a number. Set point to that number.\n\Beginning of buffer is position (point-min), end is (point-max).") (n) register Lisp_Object n;{ register int charno; CHECK_NUMBER_COERCE_MARKER (n, 0); charno = XINT (n); SET_PT (in_accessible_range (charno)); return n;}static Lisp_Objectregion_limit (beginningp) int beginningp;{ register Lisp_Object m; m = Fmarker_position (current_buffer->mark); if (NULL (m)) error ("There is no region now"); if ((point < XFASTINT (m)) == beginningp) return (make_number (point)); else return (m);}DEFUN ("region-beginning", Fregion_beginning, Sregion_beginning, 0, 0, 0, "Return position of beginning of region, as an integer.") (){ return (region_limit (1));}DEFUN ("region-end", Fregion_end, Sregion_end, 0, 0, 0, "Return position of end of region, as an integer.") (){ return (region_limit (0));}#if 0 /* now in lisp code */DEFUN ("mark", Fmark, Smark, 0, 0, 0, "Return this buffer's mark value as integer, or nil if no mark.\n\If you are using this in an editing command, you are most likely making\n\a mistake; see the documentation of `set-mark'.") (){ return Fmarker_position (current_buffer->mark);}#endif /* commented out code */DEFUN ("mark-marker", Fmark_marker, Smark_marker, 0, 0, 0, "Return this buffer's mark, as a marker object.\n\Watch out! Moving this marker changes the mark position.\n\The marker will not point anywhere if mark is not set.") (){ return current_buffer->mark;}#if 0 /* this is now in lisp code */DEFUN ("set-mark", Fset_mark, Sset_mark, 1, 1, 0, "Set this buffer's mark to POS. Don't use this function!\n\That is to say, don't use this function unless you want\n\the user to see that the mark has moved, and you want the previous\n\mark position to be lost.\n\\n\Normally, when a new mark is set, the old one should go on the stack.\n\This is why most applications should use push-mark, not set-mark.\n\\n\Novice programmers often try to use the mark for the wrong purposes.\n\The mark saves a location for the user's convenience.\n\Most editing commands should not alter the mark.\n\To remember a location for internal use in the Lisp program,\n\store it in a Lisp variable. Example:\n\\n\ (let ((beg (point))) (forward-line 1) (delete-region beg (point))).") (pos) Lisp_Object pos;{ if (NULL (pos)) { current_buffer->mark = Qnil; return Qnil; } CHECK_NUMBER_COERCE_MARKER (pos, 0); if (NULL (current_buffer->mark)) current_buffer->mark = Fmake_marker (); Fset_marker (current_buffer->mark, pos, Qnil); return pos;}#endif /* commented-out code */Lisp_Objectsave_excursion_save (){ register int visible = XBUFFER (XWINDOW (selected_window)->buffer) == current_buffer; return Fcons (Fpoint_marker (), Fcons (Fcopy_marker (current_buffer->mark), visible ? Qt : Qnil));}Lisp_Objectsave_excursion_restore (info) register Lisp_Object info;{ register Lisp_Object tem; tem = Fmarker_buffer (Fcar (info)); /* If buffer being returned to is now deleted, avoid error */ /* Otherwise could get error here while unwinding to top level and crash */ /* In that case, Fmarker_buffer returns nil now. */ if (NULL (tem)) return Qnil; Fset_buffer (tem); tem = Fcar (info); Fgoto_char (tem); unchain_marker (tem); tem = Fcar (Fcdr (info)); Fset_marker (current_buffer->mark, tem, Fcurrent_buffer ()); if (XMARKER (tem)->buffer) unchain_marker (tem); tem = Fcdr (Fcdr (info)); if (!NULL (tem) && current_buffer != XBUFFER (XWINDOW (selected_window)->buffer)) Fswitch_to_buffer (Fcurrent_buffer (), Qnil); return Qnil;}DEFUN ("save-excursion", Fsave_excursion, Ssave_excursion, 0, UNEVALLED, 0, "Save point (and mark), execute BODY, then restore point and mark.\n\Executes BODY just like PROGN. Point and mark values are restored\n\even in case of abnormal exit (throw or error).") (args) Lisp_Object args;{ register Lisp_Object val; int count = specpdl_ptr - specpdl; record_unwind_protect (save_excursion_restore, save_excursion_save ()); val = Fprogn (args); unbind_to (count); return val;}DEFUN ("buffer-size", Fbufsize, Sbufsize, 0, 0, 0, "Return the number of characters in the current buffer.") (){ Lisp_Object temp; XFASTINT (temp) = Z - BEG; return temp;}DEFUN ("point-min", Fpoint_min, Spoint_min, 0, 0, 0, "Return the minimum permissible value of point in the current buffer.\n\This is 1, unless a clipping restriction is in effect.") (){ Lisp_Object temp; XFASTINT (temp) = BEGV; return temp;}DEFUN ("point-min-marker", Fpoint_min_marker, Spoint_min_marker, 0, 0, 0, "Return a marker to the beginning of the currently visible part of the buffer.\n\This is the beginning, unless a clipping restriction is in effect.") (){ return buildmark (BEGV);}DEFUN ("point-max", Fpoint_max, Spoint_max, 0, 0, 0, "Return the maximum permissible value of point in the current buffer.\n\This is (1+ (buffer-size)), unless a clipping restriction is in effect,\n\in which case it is less.") (){ Lisp_Object temp; XFASTINT (temp) = ZV; return temp;}DEFUN ("point-max-marker", Fpoint_max_marker, Spoint_max_marker, 0, 0, 0, "Return a marker to the end of the currently visible part of the buffer.\n\This is the actual end, unless a clipping restriction is in effect.") (){ return buildmark (ZV);}DEFUN ("following-char", Ffollchar, Sfollchar, 0, 0, 0, "Return the character following point, as a number.") (){ Lisp_Object temp; XFASTINT (temp) = FETCH_CHAR (point); return temp;}DEFUN ("preceding-char", Fprevchar, Sprevchar, 0, 0, 0, "Return the character preceding point, as a number.") (){ Lisp_Object temp; if (point <= BEGV) XFASTINT (temp) = 0; else XFASTINT (temp) = FETCH_CHAR (point - 1); return temp;}DEFUN ("bobp", Fbobp, Sbobp, 0, 0, 0, "Return T if point is at the beginning of the buffer.\n\If the buffer is narrowed, this means the beginning of the narrowed part.") (){ if (point == BEGV) return Qt; return Qnil;}DEFUN ("eobp", Feobp, Seobp, 0, 0, 0, "Return T if point is at the end of the buffer.\n\If the buffer is narrowed, this means the end of the narrowed part.") (){ if (point == ZV) return Qt; return Qnil;}DEFUN ("bolp", Fbolp, Sbolp, 0, 0, 0, "Return T if point is at the beginning of a line.") (){ if (point == BEGV || FETCH_CHAR (point - 1) == '\n') return Qt; return Qnil;}DEFUN ("eolp", Feolp, Seolp, 0, 0, 0, "Return T if point is at the end of a line.\n\`End of a line' includes point being at the end of the buffer.") (){ if (point == ZV || FETCH_CHAR (point) == '\n') return Qt; return Qnil;}DEFUN ("char-after", Fchar_after, Schar_after, 1, 1, 0, "One arg, POS, a number. Return the character in the current buffer\n\at position POS.\n\If POS is out of range, the value is NIL.") (pos) Lisp_Object pos;{ register Lisp_Object val; register int n; CHECK_NUMBER_COERCE_MARKER (pos, 0); n = XINT (pos); if (n < BEGV || n >= ZV) return Qnil; XFASTINT (val) = FETCH_CHAR (n); return val;}DEFUN ("user-login-name", Fuser_login_name, Suser_login_name, 0, 0, 0, "Return the name under which user logged in, as a string.\n\This is based on the effective uid, not the real uid.") (){ return Vuser_name;}DEFUN ("user-real-login-name", Fuser_real_login_name, Suser_real_login_name, 0, 0, 0, "Return the name of the user's real uid, as a string.\n\Differs from user-login-name when running under su.") (){ return Vuser_real_name;}DEFUN ("user-uid", Fuser_uid, Suser_uid, 0, 0, 0, "Return the effective uid of Emacs, as an integer.") (){ return make_number (geteuid ());}DEFUN ("user-real-uid", Fuser_real_uid, Suser_real_uid, 0, 0, 0, "Return the real uid of Emacs, as an integer.") (){ return make_number (getuid ());}DEFUN ("user-full-name", Fuser_full_name, Suser_full_name, 0, 0, 0, "Return the full name of the user logged in, as a string.") (){ return Vuser_full_name;}DEFUN ("system-name", Fsystem_name, Ssystem_name, 0, 0, 0, "Return the name of the machine you are running on, as a string.") (){ return Vsystem_name;}DEFUN ("current-time-string", Fcurrent_time_string, Scurrent_time_string, 0, 0, 0, "Return the current time, as a human-readable string.") (){ long current_time = time (0); register char *tem = (char *) ctime (¤t_time); tem [24] = 0; return build_string (tem);}insert1 (arg) Lisp_Object arg;{ Finsert (1, &arg);}DEFUN ("insert", Finsert, Sinsert, 0, MANY, 0, "Any number of args, strings or chars. Insert them after point, moving point forward.") (nargs, args) int nargs; register Lisp_Object *args;{ register int argnum; register Lisp_Object tem; char str[1]; for (argnum = 0; argnum < nargs; argnum++) { tem = args[argnum]; retry: if (XTYPE (tem) == Lisp_Int) { str[0] = XINT (tem); insert (str, 1); } else if (XTYPE (tem) == Lisp_String) { insert (XSTRING (tem)->data, XSTRING (tem)->size); } else { tem = wrong_type_argument (Qchar_or_string_p, tem); goto retry; } } return Qnil;}DEFUN ("insert-before-markers", Finsert_before_markers, Sinsert_before_markers, 0, MANY, 0, "Any number of args, strings or chars. Insert them after point,\n\moving point forward. Also, any markers pointing at the insertion point\n\get relocated to point after the newly inserted text.") (nargs, args)
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -