scratch/edwin/macros.scm

265 lines
8.2 KiB
Scheme
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#| -*-Scheme-*-
Copyright (C) 1986, 1987, 1988, 1989, 1990, 1991, 1992, 1993, 1994,
1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005,
2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016,
2017, 2018, 2019, 2020 Massachusetts Institute of Technology
This file is part of MIT/GNU Scheme.
MIT/GNU Scheme is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or (at
your option) any later version.
MIT/GNU Scheme is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with MIT/GNU Scheme; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301,
USA.
|#
;;;; Editor Macros
;; Upwards compatibility:
(define edwin-syntax-table (->environment '(EDWIN)))
(define-syntax define-editor-alias
(sc-macro-transformer
(lambda (form env)
env
(if (syntax-match? '(symbol symbol symbol) (cdr form))
(let ((type (cadr form))
(new (caddr form))
(old (cadddr form)))
(receive (table name-map)
(case type
((MODE)
(values 'editor-modes mode-name->scheme-name))
((command)
(values 'editor-commands command-name->scheme-name))
((variable)
(values 'editor-variables variable-name->scheme-name))
(else
(error "Unknown alias type:" type)))
`(BEGIN
(DEFINE ,(name-map new) ,(name-map old))
(STRING-TABLE-PUT! ,table
,(symbol->string new)
,(name-map old)))))
(ill-formed-syntax form)))))
(define-syntax define-command
(rsc-macro-transformer
(lambda (form environment)
(capture-syntactic-environment
(lambda (instance-environment)
(if (syntax-match? '(symbol expression expression expression)
(cdr form))
(let ((name (list-ref form 1))
(description (list-ref form 2))
(interactive (list-ref form 3))
(procedure (list-ref form 4)))
(let ((scheme-name (command-name->scheme-name name)))
`(,(close-syntax 'define environment)
,scheme-name
(,(close-syntax 'make-command environment)
',name
,description
,interactive
,(if (and (pair? procedure)
(identifier=?
instance-environment (car procedure)
environment 'lambda)
(pair? (cdr procedure)))
`(,(close-syntax 'named-lambda environment)
(,scheme-name ,@(cadr procedure))
,@(cddr procedure))
procedure)))))
(ill-formed-syntax form)))))))
(define-syntax ref-command-object
(sc-macro-transformer
(lambda (form environment)
(if (syntax-match? '(symbol) (cdr form))
(close-syntax (command-name->scheme-name (cadr form)) environment)
(ill-formed-syntax form)))))
(define (command-name->scheme-name name)
(symbol 'edwin-command$ name))
(define-syntax ref-command
(sc-macro-transformer
(lambda (form environment)
environment
(if (syntax-match? '(symbol) (cdr form))
`(COMMAND-PROCEDURE (REF-COMMAND-OBJECT ,(cadr form)))
(ill-formed-syntax form)))))
(define-syntax command-defined?
(sc-macro-transformer
(lambda (form environment)
environment
(if (syntax-match? '(symbol) (cdr form))
(let ((variable-name (command-name->scheme-name (cadr form))))
`(LET ((_ENV (->ENVIRONMENT '(EDWIN))))
(AND (ENVIRONMENT-BOUND? _ENV ',variable-name)
(ENVIRONMENT-ASSIGNED? _ENV ',variable-name))))
(ill-formed-syntax form)))))
(define-syntax define-variable
(rsc-macro-transformer
(lambda (form environment)
(expand-variable-definition form environment `#f))))
(define-syntax define-variable-per-buffer
(rsc-macro-transformer
(lambda (form environment)
(expand-variable-definition form environment `#t))))
(define (expand-variable-definition form environment buffer-local?)
(if (and (syntax-match? '(symbol + expression) (cdr form))
(<= (length form) 6))
`(,(close-syntax 'define environment)
,(variable-name->scheme-name (list-ref form 1))
(,(close-syntax 'make-variable environment)
',(list-ref form 1)
,(if (> (length form) 2) (list-ref form 2) '#f)
,(if (> (length form) 3) (list-ref form 3) '#f)
,buffer-local?
,(if (> (length form) 4) (list-ref form 4) '#f)
,(if (> (length form) 5) (list-ref form 5) '#f)))
(ill-formed-syntax form)))
(define-syntax ref-variable-object
(sc-macro-transformer
(lambda (form environment)
(if (syntax-match? '(symbol) (cdr form))
(close-syntax (variable-name->scheme-name (cadr form)) environment)
(ill-formed-syntax form)))))
(define (variable-name->scheme-name name)
(symbol 'edwin-variable$ name))
(define-syntax ref-variable
(sc-macro-transformer
(lambda (form environment)
(if (syntax-match? '(symbol ? expression) (cdr form))
(let ((name `(ref-variable-object ,(cadr form))))
(if (pair? (cddr form))
`(variable-local-value ,(close-syntax (caddr form) environment)
,name)
`(variable-value ,name)))
(ill-formed-syntax form)))))
(define-syntax set-variable!
(sc-macro-transformer
(lambda (form environment)
(expand-variable-assignment form environment
(lambda (name value buffer)
(if buffer
`(set-variable-local-value! ,buffer ,name ,value)
`(set-variable-value! ,name ,value)))))))
(define-syntax local-set-variable!
(sc-macro-transformer
(lambda (form environment)
(expand-variable-assignment form environment
(lambda (name value buffer)
`(define-variable-local-value! ,(or buffer `(current-buffer)) ,name
,value))))))
(define (expand-variable-assignment form environment generator)
(if (and (syntax-match? '(symbol * expression) (cdr form))
(<= (length form) 4))
(generator `(ref-variable-object ,(list-ref form 1))
(if (> (length form) 2)
(close-syntax (list-ref form 2) environment)
`#f)
(if (> (length form) 3)
(close-syntax (list-ref form 3) environment)
#f))
(ill-formed-syntax form)))
(define-syntax define-major-mode
(sc-macro-transformer
(let ((pattern
`(symbol ,(lambda (x) (or (not x) (symbol? x)))
,(lambda (x) (or (not x) (string? x)))
expression
? expression)))
(lambda (form environment)
(if (syntax-match? pattern (cdr form))
(let ((name (list-ref form 1))
(super-mode-name (list-ref form 2)))
(let ((scheme-name (mode-name->scheme-name name)))
`(define ,scheme-name
(make-mode ',name
#t
',(or (list-ref form 3)
(symbol->string name))
,(if super-mode-name
`(->mode ',super-mode-name)
`#f)
,(close-syntax (list-ref form 4) environment)
,(let ((initialization
(if (and (> (length form) 5)
(list-ref form 5))
(close-syntax (list-ref form 5)
environment)
#f)))
(if super-mode-name
`(lambda (buffer)
((mode-initialization
(mode-super-mode
,(close-syntax scheme-name
environment)))
buffer)
,@(if initialization
`((,initialization buffer))
`()))
(or initialization
`(lambda (buffer)
buffer
unspecific))))))))
(ill-formed-syntax form))))))
(define-syntax define-minor-mode
(sc-macro-transformer
(let ((pattern
`(symbol ,(lambda (x) (or (not x) (string? x)))
expression
? expression)))
(lambda (form environment)
(if (syntax-match? pattern (cdr form))
(let ((name (list-ref form 1)))
`(define ,(mode-name->scheme-name name)
(make-mode ',name
#f
',(or (list-ref form 2)
(symbol->string name))
#f
,(close-syntax (list-ref form 3) environment)
,(if (and (> (length form) 4)
(list-ref form 4))
(close-syntax (list-ref form 4) environment)
`(lambda (buffer) buffer unspecific)))))
(ill-formed-syntax form))))))
(define-syntax ref-mode-object
(sc-macro-transformer
(lambda (form environment)
(if (syntax-match? '(symbol) (cdr form))
(close-syntax (mode-name->scheme-name (cadr form)) environment)
(ill-formed-syntax form)))))
(define (mode-name->scheme-name name)
(symbol 'edwin-mode$ name))