help-gnu-emacs
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

RE: how to save a face in Lisp?


From: Drew Adams
Subject: RE: how to save a face in Lisp?
Date: Wed, 3 Oct 2007 00:41:13 -0700

> >>> In Lisp, how can I save a face to the user's custom-file after
> >>> it has been changed (or to cause it to be saved when Emacs is quit)?
> >>
> >> The function custom-set-faces works for me.
> >
> > It requires the face's SPEC also as input:
> >
> >   "The arguments should be a list where each entry has the form:
> >     (FACE SPEC [NOW [COMMENT]])
> >    SPEC is stored as the saved value for FACE..."
> >
> > I'm looking for something that takes just the face as arg and
> > saves it. IOW, I don't want to define the face; I just want to save it.

I could have added that, anyway, `custom-set-face' does not write to the
custom-file and save it; `custom-set-face' is just the form that needs to be
written to (or updated in) the file.

> You don't have to define the face manually.  But you have to write Elisp
> code to find the current values of the attributes of an existing face,
> and write those values to a custom-file which can be read back the next
> time you start Emacs.
>
> Example:  If you execute the following in a Lisp-Interaction buffer:
> (describe-face 'default)
> you will see all the default face's attributes whose *current* values
> you have to save to a file:  :family, :width, :height, :slant,
> :foreground, :background, etc.
>
> A function to inspect each attribute in turn is called 'face-attribute
> So again, for the default face, you can get each value:
> (face-attribute 'default :family)
> (face-attribute 'default :height)
> (face-attribute 'default :background)
>
> And so on.  Get the value of each attribute of the face you're
> interested in, and then write those values to a file and you can read
> them back in at any time.

I want to save an _arbitrary_ existing face with a Lisp function:

(defun save-face (face)
  "Update user's `custom-file' with current definition of FACE."
 ...)

Is there an existing function to do that? If not, how to go about it?

What you suggest would save the current attributes of the face. That would
be like writing `custom-set-face' with this spec: (list (list t
(custom-face-attributes-get face nil))).

What I want is to save the face spec as customized by the user (or as
defined by defface if never customized), but updated to reflect the face's
current attributes (what `custom-face-attributes-get' provides). IOW, just
what would be written in the `custom-set-faces' sexp if the face had been
modified by the user in Customize and saved by the user in Customize.

The function I'm looking for would do what `custom-save-all' does with
`custom-save-faces', but for only one face. It would of course update any
existing `custom-set-faces' in the custom-file, not add another one (there
should be only one). And it would write out the face regardless of whether
Customize thought the face had been customized - for example, even if it
were changed by `modify-face'.

There is function `custom-face-save', but it takes a widget as arg, not a
face, and it operates strictly in the context of Customize. Based on its
code, I guess I could just put nil as the face's `customized-face' property,
put its current definition as its `saved-face' property, and then call
`custom-save-all'. IIUC, that is also (analogous to) how
`customize-save-variable' works.

But how can I get that complete current definition (spec) to use for
property `saved-face'? Using `face-user-default-spec' I can get the user's
saved definition of the face (or the defface definition if there is no
custom definition). And using `face-attr-construct' I can get the face's
current global definition (which is only part of a spec, not a complete
spec). But how do I merge those two to get a complete defface-style spec
that reflects the current attribute values? That is what needs to be written
into the `custom-set-faces' sexp, AFAIK.

For example, suppose this is the definition of face `foo':

(defface foo
    '((((background dark)) (:foreground "DarkGreen"))
      (t (:foreground "Red")))
  "...")

And suppose this has been evaluated, temporarily "customizing" the face
outside of Customize:

(set-face-attribute 'foo nil :foreground "Yellow")

And suppose that the background mode is not `dark'. Then the spec that needs
to be written if this face is saved is this:

 '((((background dark)) (:foreground "DarkGreen"))
   (t (:foreground "Yellow")))

Getting that updated spec seems to be the stumbling block. (Is there a
function that gives that?) The Customize code has that ready to hand,
because of its code that handles the face change by the user.

(Also, I'd want to add some code to only save the face if it had actually
changed. And ideally I'd like something that works also with Emacs 20, which
doesn't have function `face-user-default-spec'.)

The custom code is difficult for me to follow (widgetry leaves me numb), and
this all seems terribly complicated. For a user variable, it's simple: just
call `customize-save-variable'. Why is there no `customize-save-face'? Of
course, for `customize-save-variable' too you need to pass the variable's
value to save, and the hard part for the face seems to be getting a complete
updated spec for it.







reply via email to

[Prev in Thread] Current Thread [Next in Thread]