;; Copyright (c) 1987-2002 Franz Inc, Berkeley, Ca.
;;
;; Permission is granted to any individual or institution to use, copy,
;; modify, and distribute this software, and to distribute modified
;; versions, provided that this complete copyright and permission notice is
;; maintained, intact, in all copies and supporting documentation.
;;
;; Franz Incorporated provides this software "as is" without
;; express or implied warranty.
;; $Id: fi-rlogin.el,v 3.1 2004/01/16 19:27:49 layer Exp $
(defvar fi:rlogin-mode-map nil
"The rlogin major-mode keymap.")
(defvar fi:rlogin-mode-super-key-map nil
"Used for super-key processing in rlogin mode.")
(defvar fi:rlogin-image-name "rlogin"
"*Default remote-login image to invoke from (fi:rlogin). If the value
is a string then it names the image file or image path that
`fi:rlogin' invokes. Otherwise, the value of this variable is given
to funcall, the result of which should yield a string which is the image
name or path.")
(defvar fi:rlogin-image-arguments nil
"*Default remote-login image arguments when invoked from (fi:rlogin).")
(defvar fi:rlogin-prompt-pattern
"^[-_.a-zA-Z0-9]*[#$%>] *"
"*Regexp used by Newline command in rlogin mode to match subshell prompts.
Anything from beginning of line up to the end of what this pattern matches
is deemed to be prompt, and is not re-executed.")
(defvar fi:rlogin-initial-input "stty -echo nl\n"
"*The initial input sent to the rlogin subprocess, after the first prompt
is seen.")
(defun fi:rlogin-mode (&optional mode-hook)
"Major mode for interacting with an inferior rlogin.
The keymap for this mode is bound to fi:rlogin-mode-map:
\\{fi:rlogin-mode-map}
Entry to this mode runs the following hooks:
fi:subprocess-mode-hook
fi:rlogin-mode-hook
in the above order.
When calling from a program, argument is MODE-HOOK,
which is funcall'd just after killing all local variables but before doing
any other mode setup."
(interactive)
(fi::kill-all-local-variables)
(if mode-hook (funcall mode-hook))
(setq major-mode 'fi:rlogin-mode)
(setq mode-name "Rlogin")
(fi::initialize-mode-map 'fi:rlogin-mode-map 'fi:rlogin-super-key-map
'rlogin)
(use-local-map fi:rlogin-mode-map)
(setq fi:shell-popd-regexp nil)
(setq fi:shell-pushd-regexp nil)
(setq fi:shell-cd-regexp nil)
(run-hooks 'fi:subprocess-mode-hook 'fi:rlogin-mode-hook))
(defun fi:rlogin (&optional buffer-number host user)
"Start an rlogin in a buffer whose name is determined from the optional
prefix argument BUFFER-NUMBER and the HOST. Rlogin buffer names start with
`*HOST*' and end with an optional \"\". If BUFFER-NUMBER is not given
it defaults to 1. If BUFFER-NUMBER is 1, then the trailing \"<1>\" is
omited. If BUFFER-NUMBER is < 0, then the first available buffer name is
chosen (a buffer with no process attached to it.
The host name is read from the minibuffer.
The rlogin image file and image arguments are taken from the variables
`fi:rlogin-image-name' and `fi:rlogin-image-arguments'."
(interactive "p\nsRemote login to host: ")
(let ((fi:subprocess-env-vars
'(("TERM" . "dumb")
;; these two aren't passed, actually (bummer!)
("EMACS" . "t")
("DISPLAY" . (getenv "DISPLAY")))))
(fi::make-subprocess nil
host
buffer-number
default-directory
'fi:rlogin-mode
fi:rlogin-prompt-pattern
fi:rlogin-image-name
(append fi:rlogin-image-arguments
(list host)
(when user
(list "-l" user)))
'fi::rlogin-filter
(function (lambda (&rest ignore) (cd "~/"))))))
(defun fi:rlogin-new-user (&optional buffer-number host user)
(interactive "p\nsRemote login to host: \nsUser name: ")
(fi:rlogin buffer-number host user))
(defun fi::rlogin-filter (process output)
"Filter for `fi:rlogin' subprocess buffers.
Watch for the first shell prompt from the remote login, then send the
string bound to fi:rlogin-initial-input, and turn ourself off."
(let ((old-buffer (fi::subprocess-filter process output t)))
(if (save-excursion (beginning-of-line)
(looking-at fi::prompt-pattern))
(progn
(set-process-filter process 'fi::subprocess-filter)
(process-send-string process fi:rlogin-initial-input)))
(if old-buffer
(set-buffer old-buffer))))