ftp.nice.ch/NiCE/emacs-19-for-NeXTstep/Emacs_for_NeXTstep_4.1.src.NIHS.s.tar.gz#/emacs-19/src/nsfaces.m

This is nsfaces.m in view mode; [Download] [Up]

/* "Face" primitives.
   Copyright (C) 1993, 1994 Free Software Foundation.

This file is part of GNU Emacs.

GNU Emacs 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, 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 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 GNU Emacs; see the file COPYING.  If not, write to
the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */

/* This is derived from work by Lucid (some parts very loosely so).  */

#import <appkit/appkit.h>

#include <sys/types.h>
#include <sys/stat.h>

#include "config.h"
#include "lisp.h"

#ifdef HAVE_NS

#include "nsterm.h"
#include "buffer.h"
#include "dispextern.h"
#include "frame.h"
#include "blockinput.h"
#include "window.h"
#include "multi-frame.h"

/* The number of face-id's in use (same for all frames).  */
int ns_next_face_id;
static NXColor default_color;

int ns_face_name_id_number ( /* FRAME_PTR, Lisp_Object name */ );
void ns_recompute_basic_faces ( /* FRAME_PTR f */ );
static int new_computed_face (struct frame *f, struct ns_face *new_face);
static void ensure_face_ready (struct frame *f, int fid);
static int face_eql(const struct ns_face *f1,const struct ns_face *f2);

int ns_load_font(struct ns_face *f)
   {
   double shrink;
   float bbox1,bbox3,under;
   id font,screenfont;
   NXFontMetrics *m;

   if (!NUMBERP(ns_shrink_space)) return 4;
   shrink=XFLOATINT(ns_shrink_space);

   if (f->name==0 || f->size==-1) return 1;
   font=[Font newFont:f->name size:f->size];
   if (font==nil) return 2;

   if (screenfont=[font screenFont]) font=screenfont;

   m=[font metrics];
   if (!m->isFixedPitch) return 3;

   f->font=font;
   f->width=[font getWidthOf:"M"];

   bbox1=m->fontBBox[1];
   bbox3=m->fontBBox[3];
   under=m->underlinePosition;
   if (!m->isScreenFont)
      {
      bbox1=floor(f->size*bbox1);
      bbox3=ceil (f->size*bbox3);
      under=floor(f->size*under);
      }
   f->height=rint(shrink*bbox3) - bbox1;
   f->descender=-bbox1;
   f->underpos=under-bbox1;
   f->underwidth=rint(m->underlineThickness);
   return 0;
   }

/* Clear out face_vector and start anew.
   This should be done from time to time just to avoid
   keeping too many graphics contexts in face_vector
   that are no longer needed.  */

void ns_clear_face_vector (void)
   {
   /* Empty for NS */
   return;
   }

static int face_eql(const struct ns_face *f1,const struct ns_face *f2)
   {
   return(f1->size==f2->size &&
          f1->underline==f2->underline &&
          NXEqualColor(f1->foreground_color,f2->foreground_color) &&
          NXEqualColor(f1->background_color,f2->background_color) &&
          (f1->name==f2->name));
   }

/* Managing parameter face arrays for frames. */

void ns_init_frame_faces (struct frame *f)
   {
   ensure_face_ready (f, 0);
   ensure_face_ready (f, 1);

   f->display.ns->n_computed_faces=0;
   f->display.ns->computed_faces=0;

   new_computed_face (f, f->display.ns->param_faces[0]);
   new_computed_face (f, f->display.ns->param_faces[1]);
   ns_recompute_basic_faces(f);

      {
      int i;
      Lisp_Object tail, frame;

      FOR_EACH_FRAME (tail, frame)
         if (FRAME_NS_P (XFRAME (frame)) && XFRAME (frame) != f)
            for (i=2; i< XFRAME(frame)->display.ns->n_param_faces; i++)
               if (XFRAME(frame)->display.ns->param_faces[i])
                  ensure_face_ready (f,i);
      }
   }


void ns_free_frame_faces (struct frame *f)
   {
   int i;

   for(i=0; i< f->display.ns->n_param_faces; i++)
      if (f->display.ns->param_faces[i])
         xfree (f->display.ns->param_faces[i]);

   for(i=0; i< f->display.ns->n_computed_faces; i++)
      if (f->display.ns->computed_faces[i])
         xfree (f->display.ns->computed_faces[i]);

   xfree (f->display.ns->param_faces);
   f->display.ns->param_faces =0;
   f->display.ns->n_param_faces =0;

   xfree (f->display.ns->computed_faces);
   f->display.ns->computed_faces =0;
   f->display.ns->n_computed_faces =0;
   }

/* Interning faces in a frame's face array.  */
static int new_computed_face (struct frame *f, struct ns_face *new_face)
   {
   int i;

   if (f->display.ns->n_computed_faces >= f->display.ns->size_computed_faces)
      {
      int new_size = f->display.ns->n_computed_faces + 32;

      f->display.ns->computed_faces = (struct ns_face **)
         xrealloc (f->display.ns->computed_faces,
                   new_size * sizeof (struct ns_face *));
      f->display.ns->size_computed_faces = new_size;
      }

   i = f->display.ns->n_computed_faces++;
   f->display.ns->computed_faces[i] =
      (struct ns_face *) xmalloc (sizeof(struct ns_face));

   *(f->display.ns->computed_faces[i]) = *new_face;
   return i;
   }


/* Find a match for NEW_FACE in a FRAME's computed face array, and add
   it if we don't find one.  */
static int intern_computed_face (struct frame *f, struct ns_face *new_face)
   {
   int i;

   /* Search for a computed face already on F equivalent to FACE.  */
   for (i = 0; i < f->display.ns->n_computed_faces; i++)
      {
      if (! f->display.ns->computed_faces[i])
         abort ();
      if (face_eql(new_face,f->display.ns->computed_faces[i]))
         return i;
      }

   /* We didn't find one; add a new one.  */
   return new_computed_face (f, new_face);
   }

/* Make parameter face id ID valid on frame F.  */
static void ensure_face_ready (struct frame *f, int fid)
   {
   if (f->display.ns->n_param_faces <= fid)
      {
      int n = fid + 10;
      f->display.ns->param_faces = (struct ns_face **)
         xrealloc (f->display.ns->param_faces,sizeof (struct ns_face *) * n);
      bzero (f->display.ns->param_faces + f->display.ns->n_param_faces,
             (n - f->display.ns->n_param_faces) * sizeof (struct ns_face *));

      f->display.ns->n_param_faces = n;
      }

   if (f->display.ns->param_faces [fid] == 0)
      {
      struct ns_face *nface=(struct ns_face *) xmalloc(sizeof (*nface));
      nface->underline=-1;
      nface->font=0;
      nface->name=0;
      nface->size=-1;
      nface->foreground_color=default_color;
      nface->background_color=default_color;
      f->display.ns->param_faces [fid] = nface;
      }
   }


/* Return non-zero if FONT1 and FONT2 have the same width.
   We do not check the height, because we can now deal with
   different heights.
   We assume that they're both character-cell fonts.  */

int ns_same_size_fonts (struct ns_face *font1, struct ns_face *font2)
   {
   return (font1->width == font2->width);
   }

/* Update the line_height of frame F according to the biggest font in
   any face.  Return nonzero if if line_height changes.  */

int ns_frame_update_line_height (struct frame *f)
   {
   int i,shift;
   int biggest = (int)rint(f->display.ns->face->height);

   for (i = 0; i < f->display.ns->n_computed_faces; i++)
      if (f->display.ns->computed_faces[i]!=0)
         {
         if (f->display.ns->computed_faces[i]->font==0)
            ns_load_font(f->display.ns->computed_faces[i]);
         if ((f->display.ns->computed_faces[i]->height) > biggest)
            biggest = (int)rint(f->display.ns->computed_faces[i]->height);
         }

   if (biggest == f->display.ns->line_height)
      return 0;

   f->display.ns->line_height = biggest;
   return 1;
   }

/* Modify face TO by copying from FROM all properties which have
   nondefault settings.  */

static void merge_faces (struct ns_face *from, struct ns_face *to)
   {
   if (from->name && to->name!=from->name)
      {
      to->name=from->name;
      to->font=0;
      }
   if (from->size!=-1 && to->size!=from->size)
      {
      to->size=from->size;
      to->font=0;
      }
   if (from->underline != -1)
      to->underline = from->underline;
   if (!NXEqualColor(from->foreground_color,default_color))
      to->foreground_color = from->foreground_color;
   if (!NXEqualColor(from->background_color,default_color))
      to->background_color = from->background_color;
   }

/* Set up the basic set of facial parameters, based on the frame's
   data; all faces are deltas applied to this.  */

static void compute_base_face (struct frame *f, struct ns_face *face)
   {
   *face = *f->display.ns->face;
   }

/* Return the face ID to use to display a special glyph which selects
   FACE_CODE as the face ID, assuming that ordinarily the face would
   be CURRENT_FACE.  F is the frame.  */

int ns_compute_glyph_face (struct frame *f, int face_code, int current_face)
   {
   struct ns_face face;

   face = *f->display.ns->param_faces[current_face];

   if (face_code >= 0 && face_code < f->display.ns->n_param_faces
       && f->display.ns->param_faces[face_code] != 0)
      merge_faces (f->display.ns->param_faces[face_code], &face);

   return intern_computed_face (f, &face);
   }

/* Return the face ID to use to display a special glyph which selects
   FACE_CODE as the face ID, assuming that ordinarily the face would
   be CURRENT_FACE.  F is the frame.  */

int ns_compute_glyph_face_1 (struct frame *f, Lisp_Object face_name,
                             int current_face)
   {
   struct ns_face face;
   
   face = *f->display.ns->param_faces[current_face];
   
   if (!NILP (face_name))
      {
      int face_code = ns_face_name_id_number (f, face_name);
      
      if (face_code >= 0 && face_code < f->display.ns->n_param_faces
          && f->display.ns->param_faces[face_code] != 0)
         merge_faces (f->display.ns->param_faces[face_code], &face);
      }
   
   return intern_computed_face (f, &face);
   }

/* Return the face ID associated with a buffer position POS.
   Store into *ENDPTR the position at which a different face is needed.
   This does not take account of glyphs that specify their own face codes.
   F is the frame in use for display, and W is a window displaying
   the current buffer.

   REGION_BEG, REGION_END delimit the region, so it can be highlighted.

   LIMIT is a position not to scan beyond.  That is to limit
   the time this function can take.

   If MOUSE is nonzero, use the character's mouse-face, not its face.  */

int ns_compute_char_face (struct frame *f, struct window *w, int pos,
                          int region_beg, int region_end, int *endptr, int limit,
                          int mouse)
   {
   struct ns_face face;
   Lisp_Object prop, position;
   int i, j, noverlays;
   int facecode;
   Lisp_Object *overlay_vec;
   Lisp_Object frame;
   int endpos;
   Lisp_Object propname;

   /* W must display the current buffer.  We could write this function
      to use the frame and buffer of W, but right now it doesn't.  */
   if (XBUFFER (w->buffer) != current_buffer)
      abort ();

   XSET (frame, Lisp_Frame, f);

   endpos = ZV;
   if (pos < region_beg && region_beg < endpos)
      endpos = region_beg;

   XFASTINT (position) = pos;

   if (mouse)
      propname = Qmouse_face;
   else
      propname = Qface;

   prop = Fget_text_property (position, propname, w->buffer);
      {
      Lisp_Object limit1, end;

      XFASTINT (limit1) = (limit < endpos ? limit : endpos);
    end = Fnext_single_property_change (position, propname, w->buffer, limit1);
      if (INTEGERP (end))
         endpos = XINT (end);
      }

      {
      int next_overlay;
      int len;

      /* First try with room for 40 overlays.  */
      len = 40;
      overlay_vec = (Lisp_Object *) alloca (len * sizeof (Lisp_Object));

      noverlays = overlays_at (pos, 0, &overlay_vec, &len, &next_overlay);

      /* If there are more than 40,
         make enough space for all, and try again.  */
      if (noverlays > len)
         {
         len = noverlays;
         overlay_vec = (Lisp_Object *) alloca (len * sizeof (Lisp_Object));
         noverlays = overlays_at (pos, 0, &overlay_vec, &len, &next_overlay);
         }

      if (next_overlay < endpos)
         endpos = next_overlay;
      }

   *endptr = endpos;

   /* Optimize the default case.  */
   if (noverlays == 0 && NILP (prop)
       && !(pos >= region_beg && pos < region_end))
      return 0;

   compute_base_face (f, &face);

   if (!NILP (prop))
      {
      facecode = ns_face_name_id_number (f, prop);
      if (facecode >= 0 && facecode < f->display.ns->n_param_faces
	  && f->display.ns->param_faces [facecode] != 0)
         merge_faces (f->display.ns->param_faces [facecode], &face);
      }

   noverlays = sort_overlays (overlay_vec, noverlays, w);

   /* Now merge the overlay data in that order.  */
   for (i = 0; i < noverlays; i++)
      {
      prop = Foverlay_get (overlay_vec[i], propname);
      if (!NILP (prop))
         {
         Lisp_Object oend;
         int oendpos;

         facecode = ns_face_name_id_number (f, prop);
         if (facecode >= 0 && facecode < f->display.ns->n_param_faces
             && f->display.ns->param_faces[facecode] != 0)
            merge_faces (f->display.ns->param_faces[facecode], &face);

         oend = OVERLAY_END (overlay_vec[i]);
         oendpos = OVERLAY_POSITION (oend);
         if (oendpos < endpos)
            endpos = oendpos;
         }
      }

   if (pos >= region_beg && pos < region_end)
      {
      if (region_end < endpos)
         endpos = region_end;
      if (region_face >= 0 && region_face < ns_next_face_id)
         merge_faces (f->display.ns->param_faces[region_face], &face);
      }

   *endptr = endpos;

   return intern_computed_face (f, &face);
   }

/* Recompute the GC's for the default and modeline faces.
   We call this after changing frame parameters on which those GC's
   depend.  */

void ns_recompute_basic_faces (struct frame *f)
   {
   /* If the frame's faces haven't been initialized yet, don't worry about
      this stuff.  */
   if (f->display.ns->n_param_faces < 2)
      return;

   compute_base_face (f, f->display.ns->computed_faces[0]);
   compute_base_face (f, f->display.ns->computed_faces[1]);

   merge_faces (f->display.ns->param_faces[0],
                f->display.ns->computed_faces[0]);
   merge_faces (f->display.ns->param_faces[1],
                f->display.ns->computed_faces[1]);
   }

DEFUN ("ns-make-face-internal", Fns_make_face_internal, Sns_make_face_internal, 1, 1, 0,
  "Create face number FACE-ID on all frames.")
  (face_id)
     Lisp_Object face_id;
   {
   Lisp_Object rest;
   int fid = XINT (face_id);

   check_ns();
   CHECK_NUMBER (face_id, 0);
   if (fid < 0 || fid >= ns_next_face_id)
      error ("Face id out of range");

   for (rest = Vframe_list; CONSP (rest); rest = XCONS (rest)->cdr)
      {
      struct frame *f = XFRAME (XCONS (rest)->car);
      if (FRAME_NS_P (f))
         ensure_face_ready (f, fid);
      }
   return Qnil;
   }

DEFUN ("ns-set-face-attribute-internal", Fns_set_face_attribute_internal,
       Sns_set_face_attribute_internal, 4, 4, 0, "")
     (face_id, attr_name, attr_value, frame)
     Lisp_Object face_id, attr_name, attr_value, frame;
   {
   struct ns_face *face;
   struct frame *f;
   int fid;
   int garbaged=0;

   check_ns();
   CHECK_FRAME (frame, 0);
   CHECK_NUMBER (face_id, 0);
   CHECK_SYMBOL (attr_name, 0);

   f = XFRAME (frame);
   fid = XINT (face_id);
   if (fid < 0 || fid >= ns_next_face_id)
      error ("Face id out of range");

   if (! FRAME_NS_P (f))
      return;

   ensure_face_ready (f, fid);
   face = f->display.ns->param_faces[XINT (face_id)];

   if (EQ (attr_name, Qfont))
      {
      if (NILP(attr_value))
         {
         face->name=0;
         }
      else
         {
         CHECK_STRING(attr_value,0);
         face->name=NXUniqueString(XSTRING (attr_value)->data);
         }
      face->font=0;
      if (ns_frame_update_line_height (f))
         ns_set_window_size (f, 0, f->width, f->height);
      garbaged=1;
      }
   else if (EQ (attr_name, Qfontsize))
      {
      if (NILP(attr_value))
         face->size=-1;
      else
         {
         CHECK_NUMBER_OR_FLOAT (attr_value, 0);
         face->size=XFLOATINT(attr_value);
         if (face->size<=0)
            error("Facesize non-positive.");
         }
      face->font=0;
      if (ns_frame_update_line_height (f))
         ns_set_window_size (f, 0, f->width, f->height);
      garbaged=1;
      }
   else if (EQ (attr_name, intern ("foreground")))
      {
      if (NILP(attr_value))
         face->foreground_color=default_color;
      else if (ns_lisp_to_color(attr_value,&face->foreground_color))
         error("Unknown color.");
      garbaged=1;
      }
   else if (EQ (attr_name, intern ("background")))
      {
      if (NILP(attr_value))
         face->background_color=default_color;
      else if (ns_lisp_to_color(attr_value,&face->background_color))
         error("Unknown color.");
      garbaged=1;
      }
   else if (EQ (attr_name, Qunderline))
      {
      if (NILP(attr_value))
         face->underline=-1;
      else if (EQ(attr_value,Qt))
         face->underline=1;
      else
         face->underline=0;
      garbaged=1;
      }
   else
      error ("unknown face attribute");

   if (fid == 0 || fid == 1)
      ns_recompute_basic_faces (f);

   if (garbaged)
      SET_FRAME_GARBAGED (f);
   
   return Qnil;
   }

DEFUN ("ns-internal-next-face-id", Fns_internal_next_face_id,
  Sns_internal_next_face_id, 0, 0, 0, "")
  ()
   {
   check_ns();
   return make_number (ns_next_face_id++);
   }

DEFUN ("ns-convert-font-trait-internal",
  Fns_convert_font_trait_internal, Sns_convert_font_trait_internal,
  2, 2, 0, "")
  (Lisp_Object name, Lisp_Object attribute)
   {
   id fm;
   id font;
   check_ns();
   CHECK_STRING(name,0);
   CHECK_SYMBOL(attribute,0);
   fm=[FontManager new];
   font=[Font newFont:XSTRING(name)->data size:10.0];
   if (!fm || !font)
      error ("Unknown font.");

   if (EQ(attribute, intern ("italic")))
      font=[fm convert:font toHaveTrait:NX_ITALIC];
   else if (EQ(attribute, intern ("unitalic")))
      font=[fm convert:font toNotHaveTrait:NX_ITALIC];
   else if (EQ(attribute, intern ("bold")))
      font=[fm convert:font toHaveTrait:NX_BOLD];
   else if (EQ(attribute, intern ("unbold")))
      font=[fm convert:font toNotHaveTrait:NX_BOLD];
   else if (EQ(attribute, intern ("nonstandardcharset")))
      font=[fm convert:font toHaveTrait:NX_NONSTANDARDCHARSET];
   else if (EQ(attribute, intern ("unnonstandardcharset")))
      font=[fm convert:font toNotHaveTrait:NX_NONSTANDARDCHARSET];
   else if (EQ(attribute, intern ("narrow")))
      font=[fm convert:font toHaveTrait:NX_NARROW];
   else if (EQ(attribute, intern ("unnarrow")))
      font=[fm convert:font toNotHaveTrait:NX_NARROW];
   else if (EQ(attribute, intern ("expanded")))
      font=[fm convert:font toHaveTrait:NX_EXPANDED];
   else if (EQ(attribute, intern ("unexpanded")))
      font=[fm convert:font toNotHaveTrait:NX_EXPANDED];
   else if (EQ(attribute, intern ("condensed")))
      font=[fm convert:font toHaveTrait:NX_CONDENSED];
   else if (EQ(attribute, intern ("uncondensed")))
      font=[fm convert:font toNotHaveTrait:NX_CONDENSED];
   else if (EQ(attribute, intern ("smallcaps")))
      font=[fm convert:font toHaveTrait:NX_SMALLCAPS];
   else if (EQ(attribute, intern ("unsmallcaps")))
      font=[fm convert:font toNotHaveTrait:NX_SMALLCAPS];
   else if (EQ(attribute, intern ("poster")))
      font=[fm convert:font toHaveTrait:NX_POSTER];
   else if (EQ(attribute, intern ("unposter")))
      font=[fm convert:font toNotHaveTrait:NX_POSTER];
   else if (EQ(attribute, intern ("compressed")))
      font=[fm convert:font toHaveTrait:NX_COMPRESSED];
   else if (EQ(attribute, intern ("uncompressed")))
      font=[fm convert:font toNotHaveTrait:NX_COMPRESSED];
   return build_string([font name]);
   }

int ns_face_name_id_number (struct frame *f, Lisp_Object name)
   {
   Lisp_Object tem;

   tem= Fcdr (assq_no_quit (name, f->face_alist));
   if (NILP (tem)) return 0;

   CHECK_VECTOR (tem, 0);
   tem = XVECTOR (tem)->contents[2];
   CHECK_NUMBER (tem, 0);
   return XINT (tem);
   }

void syms_of_nsfaces ()
   {
   default_color=NX_COLORCLEAR;

   defsubr (&Sns_make_face_internal);
   defsubr (&Sns_set_face_attribute_internal);
   defsubr (&Sns_internal_next_face_id);
   defsubr (&Sns_convert_font_trait_internal);
   }

#endif

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.