view man/lispref/extents.texi @ 4858:e1d8b6b0c589

Define CYGWIN_HEADERS in s/cygwin32.h and s/mingw32.h instead of syswindows.h, so they're available everywhere
author Ben Wing <ben@xemacs.org>
date Thu, 14 Jan 2010 02:21:17 -0600
parents c91543697b09
children 99f8ebc082d9
line wrap: on
line source

@c -*-texinfo-*-
@c This is part of the XEmacs Lisp Reference Manual.
@c Copyright (C) 1990, 1991, 1992, 1993 Free Software Foundation, Inc.
@c Copyright (C) 1996 Ben Wing.
@c See the file lispref.texi for copying conditions.
@setfilename ../../info/extents.info
@node Extents, Specifiers, Abbrevs, top
@chapter Extents
@cindex extent

  An @dfn{extent} is a region of text (a start position and an end
position) that is displayed in a particular face and can have certain
other properties such as being read-only.  Extents can overlap each
other.  XEmacs efficiently handles buffers with large numbers of
extents in them.

@defun extentp object
This returns @code{t} if @var{object} is an extent.
@end defun

@menu
* Intro to Extents::	   Extents are regions over a buffer or string.
* Creating and Modifying Extents::
			   Basic extent functions.
* Extent Endpoints::	   Accessing and setting the bounds of an extent.
* Finding Extents::	   Determining which extents are in an object.
* Mapping Over Extents::   More sophisticated functions for extent scanning.
* Extent Properties::	   Extents have built-in and user-definable properties.
* Detached Extents::	   Extents that are not in a buffer.
* Extent Parents::         Inheriting properties from another extent.
* Duplicable Extents::	   Extents can be marked to be copied into strings.
* Extents and Events::	   Extents can interact with the keyboard and mouse.
* Atomic Extents::	   Treating a block of text as a single entity.
@end menu

@node Intro to Extents
@section Introduction to Extents
@cindex extent priority
@cindex priority of an extent

  An extent is a region of text within a buffer or string that has
certain properties associated with it.  The properties of an extent
primarily affect the way the text contained in the extent is displayed.
Extents can freely overlap each other in a buffer or string.  Extents
are invisible to functions that merely examine the text of a buffer or
string.

  @emph{Please note:} An alternative way to add properties to a buffer or
string is to use text properties.  @xref{Text Properties}.

  An extent is logically a Lisp object consisting of a start position,
an end position, a buffer or string to which these positions refer, and
a property list.  As text is inserted into a buffer, the start and end
positions of the extent are automatically adjusted as necessary to keep
the extent referring to the same text in the buffer.  If text is
inserted at the boundary of an extent, the extent's @code{start-open}
and @code{end-open} properties control whether the text is included as
part of the extent.  If the text bounded by an extent is deleted, the
extent becomes @dfn{detached}; its start and end positions are no longer
meaningful, but it maintains all its other properties and can later be
reinserted into a buffer. (None of these considerations apply to strings,
because text cannot be inserted into or deleted from a string.)

  Each extent has a face or list of faces associated with it, which
controls the way in which the text bounded by the extent is displayed.
If an extent's face is @code{nil} or its properties are partially
undefined, the corresponding properties from the default face for the
frame is used.  If two or more extents overlap, or if a list of more
than one face is specified for a particular extent, the corresponding
faces are merged to determine the text's displayed properties.  Every
extent has a @dfn{priority} that determines which face takes precedence
if the faces conflict. (If two extents have the same priority, the one
that comes later in the display order takes precedence.  @xref{Extent
Endpoints, display order}.) Higher-numbered priority values correspond
to a higher priority, and priority values can be negative.  Every extent
is created with a priority of 0, but this can be changed with
@code{set-extent-priority}.  Within a single extent with a list of faces,
faces earlier in the list have a higher priority than faces later in
the list.

  Extents can be set to respond specially to key and mouse events within
the extent.  An extent's @code{keymap} property controls the effect of
key and mouse strokes within the extent's text, and the @code{mouse-face}
property controls whether the extent is highlighted when the mouse moves
over it.  @xref{Extents and Events}.

  An extent can optionally have a @dfn{begin-glyph} or @dfn{end-glyph}
associated with it.  A begin-glyph or end-glyph is a pixmap or string
that will be displayed either at the start or end of an extent or in the
margin of the line that the start or end of the extent lies in,
depending on the extent's layout policy.  Begin-glyphs and end-glyphs
are used to implement annotations, and you should use the annotation API
functions in preference to the lower-level extent functions.  For more
information, @xref{Annotations}.

  If an extent has its @code{detachable} property set, it will become
@dfn{detached} (i.e. no longer in the buffer) when all its text is
deleted.  Otherwise, it will simply shrink down to zero-length and
sit in the same place in the buffer.  By default, the @code{detachable}
property is set on newly-created extents.  @xref{Detached Extents}.

  If an extent has its @code{duplicable} property set, it will be
remembered when a string is created from text bounded by the extent.
When the string is re-inserted into a buffer, the extent will also
be re-inserted.  This mechanism is used in the kill, yank, and undo
commands.  @xref{Duplicable Extents}.

@node Creating and Modifying Extents
@section Creating and Modifying Extents

@defun make-extent from to &optional buffer-or-string
This function makes an extent for the range [@var{from}, @var{to}) in
@var{buffer-or-string} (a buffer or string).  @var{buffer-or-string}
defaults to the current buffer.  Insertions at point @var{to} will be
outside of the extent; insertions at @var{from} will be inside the
extent, causing the extent to grow (@pxref{Extent Endpoints}).  This is
the same way that markers behave.  The extent is initially detached if
both @var{from} and @var{to} are @code{nil}, and in this case
@var{buffer-or-string} defaults to @code{nil}, meaning the extent is in
no buffer or string (@pxref{Detached Extents}).
@end defun

@defun delete-extent extent
This function removes @var{extent} from its buffer and destroys it.
This does not modify the buffer's text, only its display properties.
The extent cannot be used thereafter.  To remove an extent in such
a way that it can be re-inserted later, use @code{detach-extent}.
@xref{Detached Extents}.
@end defun

@defun extent-object extent
This function returns the buffer or string that @var{extent} is in.  If
the return value is @code{nil}, this means that the extent is detached;
however, a detached extent will not necessarily return a value of
@code{nil}.
@end defun

@defun extent-live-p object
This function returns @code{t} if @var{object} is an extent that has not
been deleted, and @code{nil} otherwise.
@end defun

@node Extent Endpoints
@section Extent Endpoints
@cindex extent endpoint
@cindex extent start position
@cindex extent end position
@cindex zero-length extent
@cindex display order
@cindex extent order
@cindex order of extents

  Every extent has a start position and an end position, and logically
affects the characters between those positions.  Normally the start and
end positions must both be valid positions in the extent's buffer or
string.  However, both endpoints can be @code{nil}, meaning the extent
is detached.  @xref{Detached Extents}.

  Whether the extent overlaps its endpoints is governed by its
@code{start-open} and @code{end-open} properties.  Insertion of a
character at a closed endpoint will expand the extent to include that
character; insertion at an open endpoint will not.  Similarly, functions
such as @code{extent-at} that scan over all extents overlapping a
particular position will include extents with a closed endpoint at that
position, but not extents with an open endpoint.

  Note that the @code{start-closed} and @code{end-closed} properties are
equivalent to @code{start-open} and @code{end-open} with the opposite
sense.

  Both endpoints can be equal, in which case the extent includes no
characters but still exists in the buffer or string.  Zero-length
extents are used to represent annotations (@pxref{Annotations}) and can
be used as a more powerful form of a marker.  Deletion of all the
characters in an extent may or may not result in a zero-length extent;
this depends on the @code{detachable} property (@pxref{Detached
Extents}).  Insertion at the position of a zero-length extent expands
the extent if both endpoints are closed; goes before the extent if it
has the @code{start-open} property; and goes after the extent if it has
the @code{end-open} property.  Zero-length extents with both the
@code{start-open} and @code{end-open} properties are treated as if their
starting point were closed.  Deletion of a character on a side of a
zero-length extent whose corresponding endpoint is closed causes the
extent to be detached if its @code{detachable} property is set; if the
corresponding endpoint is open, the extent remains in the buffer, moving
as necessary.

  Extents are ordered within a buffer or string by increasing start
position, and then by decreasing end position (this is called the
@dfn{display order}).

@defun extent-start-position extent
This function returns the start position of @var{extent}.
@end defun

@defun extent-end-position extent
This function returns the end position of @var{extent}.
@end defun

@defun extent-length extent
This function returns the length of @var{extent} in characters.  If
the extent is detached, this returns @code{0}.  If the extent is not
detached, this is equivalent to
@example
(- (extent-end-position @var{extent}) (extent-start-position @var{extent}))
@end example
@end defun

@defun set-extent-endpoints extent start end &optional buffer-or-string
This function sets the start and end position of @var{extent} to
@var{start} and @var{end}.  If both are @code{nil}, this is equivalent
to @code{detach-extent}.

@var{buffer-or-string} specifies the new buffer or string that the
extent should be in, and defaults to @var{extent}'s buffer or
string. (If @code{nil}, and @var{extent} is in no buffer and no string,
it defaults to the current buffer.)

See documentation on @code{detach-extent} for a discussion of undo
recording.
@end defun

@node Finding Extents
@section Finding Extents
@cindex extents, locating

  The following functions provide a simple way of determining the
extents in a buffer or string.  A number of more sophisticated
primitives for mapping over the extents in a range of a buffer or string
are also provided (@pxref{Mapping Over Extents}).  When reading through
this section, keep in mind the way that extents are ordered
(@pxref{Extent Endpoints}).

@defun extent-list &optional buffer-or-string from to flags property value
This function returns a list of the extents in @var{buffer-or-string}.
@var{buffer-or-string} defaults to the current buffer if omitted.
@var{from} and @var{to} can be used to limit the range over which
extents are returned; if omitted, all extents in the buffer or string
are returned.

  More specifically, if a range is specified using @var{from} and
@var{to}, only extents that overlap the range (i.e. begin or end inside
of the range) are included in the list.  @var{from} and @var{to} default
to the beginning and end of @var{buffer-or-string}, respectively.

  @var{flags} controls how end cases are treated.  For a discussion of
this, and exactly what ``overlap'' means, see @code{map-extents}.

The optional arguments @var{property} and @var{value} can be used to
further restrict which extents are returned.  They have the same meaning
as for @code{map-extents}.

If you want to map a function over the extents in a buffer or string,
consider using @code{map-extents} or @code{mapcar-extents} instead.

See also the function @code{extents-at}.
@end defun

  Functions that create extents must be prepared for the possibility
that there are other extents in the same area, created by other
functions.  To deal with this, functions typically mark their own
extents by setting a particular property on them.  The following
function makes it easier to locate those extents.

@defun extent-at pos &optional object property before at-flag
This function finds the ``smallest'' extent (i.e., the last one in the
display order) at (i.e., overlapping) @var{pos} in @var{object} (a
buffer or string) having @var{property} set.  @var{object} defaults to
the current buffer.  @var{property} defaults to @code{nil}, meaning that
any extent will do.  Returns @code{nil} if there is no matching extent
at @var{pos}.  If the fourth argument @var{before} is not @code{nil}, it
must be an extent; any returned extent will precede that extent.  This
feature allows @code{extent-at} to be used by a loop over extents.

@var{at-flag} controls how end cases are handled (i.e. what ``at''
really means), and should be one of:

@table @code
@item nil
@item after
An extent is at @var{pos} if it covers the character after @var{pos}.
This is consistent with the way that text properties work.
@item before
An extent is at @var{pos} if it covers the character before @var{pos}.
@item at
An extent is at @var{pos} if it overlaps or abuts @var{pos}.  This
includes all zero-length extents at @var{pos}.
@end table

  Note that in all cases, the start-openness and end-openness of the
extents considered is ignored.  If you want to pay attention to those
properties, you should use @code{map-extents}, which gives you more
control.
@end defun

  The following low-level functions are provided for explicitly
traversing the extents in a buffer according to the display order.
These functions are mostly intended for debugging---in normal
operation, you should probably use @code{mapcar-extents} or
@code{map-extents}, or loop using the @var{before} argument to
@code{extent-at}, rather than creating a loop using @code{next-extent}.

@defun next-extent extent
Given an extent @var{extent}, this function returns the next extent in
the buffer or string's display order.  If @var{extent} is a buffer or
string, this returns the first extent in the buffer or string.
@end defun

@defun previous-extent extent
Given an extent @var{extent}, this function returns the previous extent
in the buffer or string's display order.  If @var{extent} is a buffer or
string, this returns the last extent in the buffer or string.
@end defun

@node Mapping Over Extents
@section Mapping Over Extents
@cindex extents, mapping

  The most basic and general function for mapping over extents is called
@code{map-extents}.  You should read through the definition of this
function to familiarize yourself with the concepts and optional
arguments involved.  However, in practice you may find it more
convenient to use the function @code{mapcar-extents} or to create a loop
using the @code{before} argument to @code{extent-at} (@pxref{Finding
Extents}).

@defun map-extents function &optional object from to maparg flags property value
  This function maps @var{function} over the extents which overlap a
region in @var{object}.  @var{object} is normally a buffer or string but
could be an extent (see below).  The region is normally bounded by
[@var{from}, @var{to}) (i.e. the beginning of the region is closed and
the end of the region is open), but this can be changed with the
@var{flags} argument (see below for a complete discussion).

  @var{function} is called with the arguments (extent, @var{maparg}).
The arguments @var{object}, @var{from}, @var{to}, @var{maparg}, and
@var{flags} are all optional and default to the current buffer, the
beginning of @var{object}, the end of @var{object}, @code{nil}, and
@code{nil}, respectively.  @code{map-extents} returns the first
non-@code{nil} result produced by @var{function}, and no more calls to
@var{function} are made after it returns non-@code{nil}.

  If @var{object} is an extent, @var{from} and @var{to} default to the
extent's endpoints, and the mapping omits that extent and its
predecessors.  This feature supports restarting a loop based on
@code{map-extents}.  Note: @var{object} must be attached to a buffer or
string, and the mapping is done over that buffer or string.

  An extent overlaps the region if there is any point in the extent that
is also in the region. (For the purpose of overlap, zero-length extents
and regions are treated as closed on both ends regardless of their
endpoints' specified open/closedness.) Note that the endpoints of an
extent or region are considered to be in that extent or region if and
only if the corresponding end is closed.  For example, the extent [5,7]
overlaps the region [2,5] because 5 is in both the extent and the
region.  However, (5,7] does not overlap [2,5] because 5 is not in the
extent, and neither [5,7] nor (5,7] overlaps the region [2,5) because 5
is not in the region.

  The optional @var{flags} can be a symbol or a list of one or more
symbols, modifying the behavior of @code{map-extents}.  Allowed symbols
are:

@table @code
@item end-closed
The region's end is closed.

@item start-open
The region's start is open.

@item all-extents-closed
Treat all extents as closed on both ends for the purpose of determining
whether they overlap the region, irrespective of their actual open- or
closedness.
@item all-extents-open
Treat all extents as open on both ends.
@item all-extents-closed-open
Treat all extents as start-closed, end-open.
@item all-extents-open-closed
Treat all extents as start-open, end-closed.

@item start-in-region
In addition to the above conditions for extent overlap, the extent's
start position must lie within the specified region.  Note that, for
this condition, open start positions are treated as if 0.5 was added to
the endpoint's value, and open end positions are treated as if 0.5 was
subtracted from the endpoint's value.
@item end-in-region
The extent's end position must lie within the region.
@item start-and-end-in-region
Both the extent's start and end positions must lie within the region.
@item start-or-end-in-region
Either the extent's start or end position must lie within the region.

@item negate-in-region
The condition specified by a @code{*-in-region} flag must @emph{not}
hold for the extent to be considered.
@end table

  At most one of @code{all-extents-closed}, @code{all-extents-open},
@code{all-extents-closed-open}, and @code{all-extents-open-closed} may
be specified.

  At most one of @code{start-in-region}, @code{end-in-region},
@code{start-and-end-in-region}, and @code{start-or-end-in-region} may be
specified.

  If optional arg @var{property} is non-@code{nil}, only extents with
that property set on them will be visited.  If optional arg @var{value}
is non-@code{nil}, only extents whose value for that property is
@code{eq} to @var{value} will be visited.
@end defun

  If you want to map over extents and accumulate a list of results,
the following function may be more convenient than @code{map-extents}.

@defun mapcar-extents function &optional predicate buffer-or-string from to flags property value
This function applies @var{function} to all extents which overlap a
region in @var{buffer-or-string}.  The region is delimited by
@var{from} and @var{to}.  @var{function} is called with one argument,
the extent.  A list of the values returned by @var{function} is
returned.  An optional @var{predicate} may be used to further limit the
extents over which @var{function} is mapped.  The optional arguments
@var{flags}, @var{property}, and @var{value} may also be used to control
the extents passed to @var{predicate} or @var{function}, and have the
same meaning as in @code{map-extents}.
@end defun

@defun map-extent-children function &optional object from to maparg flags property value
This function is similar to @code{map-extents}, but differs in that:

@itemize @bullet
@item
It only visits extents which start in the given region.
@item
After visiting an extent @var{e}, it skips all other extents which start
inside @var{e} but end before @var{e}'s end.
@end itemize

Thus, this function may be used to walk a tree of extents in a buffer:
@example
(defun walk-extents (buffer &optional ignore)
  (map-extent-children 'walk-extents buffer))
@end example
@end defun

@defun extent-in-region-p extent &optional from to flags
This function returns @code{t} if @code{map-extents} would visit
@var{extent} if called with the given arguments.
@end defun

@node Extent Properties
@section Properties of Extents
@cindex extent property
@cindex property of an extent

  Each extent has a property list associating property names with
values.  Some property names have predefined meanings, and can usually
only assume particular values.  Assigning other values to such a
property either cause the value to be converted into a legal value
(e.g., assigning anything but @code{nil} to a Boolean property will
cause the value of @code{t} to be assigned to the property) or will
cause an error.  Property names without predefined meanings can be
assigned any value.  An undefined property is equivalent to a property
with a value of @code{nil}, or with a particular default value in the
case of properties with predefined meanings.  Note that, when an extent
is created, the @code{end-open} and @code{detachable} properties are set
on it.

  If an extent has a parent, all of its properties actually derive
from that parent (or from the root ancestor if the parent in turn
has a parent), and setting a property of the extent actually sets
that property on the parent.  @xref{Extent Parents}.

@defun extent-property extent property &optional default
This function returns @var{extent}'s value for @var{property}, or
@var{default} if no such property exists.
@end defun

@defun extent-properties extent
This function returns a list of all of @var{extent}'s properties that do
not have the value of @code{nil} (or the default value, for properties
with predefined meanings).
@end defun

@defun set-extent-property extent property value
This function sets @var{property} to @var{value} in @var{extent}. (If
@var{property} has a predefined meaning, only certain values are
allowed, and some values may be converted to others before being
stored.)
@end defun

@defun set-extent-properties extent plist
Change some properties of @var{extent}.  @var{plist} is a property
list.  This is useful to change many extent properties at once.
@end defun

The following table lists the properties with predefined meanings, along
with their allowable values.

@table @code
@item detached
(Boolean) Whether the extent is detached.   Setting this is the same
as calling @code{detach-extent}.  @xref{Detached Extents}.

@item destroyed
(Boolean) Whether the extent has been deleted.  Setting this is the same
as calling @code{delete-extent}.

@item priority
(integer) The extent's redisplay priority.  Defaults to 0.  @xref{Intro
to Extents, priority}.  This property can also be set with
@code{set-extent-priority} and accessed with @code{extent-priority}.

@item start-open
(Boolean) Whether the start position of the extent is open, meaning that
characters inserted at that position go outside of the extent.
@xref{Extent Endpoints}.

@item start-closed
(Boolean) Same as @code{start-open} but with the opposite sense.  Setting
this property clears @code{start-open} and vice-versa.

@item end-open
(Boolean) Whether the end position of the extent is open, meaning that
characters inserted at that position go outside of the extent.  This is
@code{t} by default.
@xref{Extent Endpoints}.

@item end-closed
(Boolean) Same as @code{end-open} but with the opposite sense.  Setting
this property clears @code{end-open} and vice-versa.

@item read-only
(Boolean) Whether text within this extent will be unmodifiable.

@item face
(face, face name, list of faces or face names, or @code{nil}) The face
in which to display the extent's text.  This property can also be set
with @code{set-extent-face} and accessed with @code{extent-face}.
Note that if a list of faces is specified, the faces are merged together,
with faces earlier in the list having priority over faces later in the
list.

@item mouse-face
(face, face name, list of faces or face names, or @code{nil}) The face
used to display the extent when the mouse moves over it.  This property
can also be set with @code{set-extent-mouse-face} and accessed with
@code{extent-mouse-face}.  Note that if a list of faces is specified,
the faces are merged together, with faces earlier in the list having
priority over faces later in the list.  @xref{Extents and Events}.

@item pointer
(pointer glyph)  The glyph used as the pointer when the mouse moves over
the extent.  This takes precedence over the @code{text-pointer-glyph}
and @code{nontext-pointer-glyph} variables.  If for any reason this
glyph is an invalid pointer, the standard glyphs will be used as
fallbacks.  @xref{External Glyphs}.

@item detachable
(Boolean) Whether this extent becomes detached when all of the text it
covers is deleted.  This is @code{t} by default.  @xref{Detached
Extents}.

@item duplicable
(Boolean) Whether this extent should be copied into strings, so that
kill, yank, and undo commands will restore or copy it.  @xref{Duplicable
Extents}.

@item unique
(Boolean) Meaningful only in conjunction with @code{duplicable}.
When this is set, there may be only one instance of
this extent attached at a time.  @xref{Duplicable Extents}.

@item invisible
(Boolean) If @code{t}, text under this extent will not be displayed --
it will look as if the text and the begin-glyph is not there at all.
The end-glyph will still be displayed.

@item keymap
(keymap or @code{nil}) This keymap is consulted for mouse clicks on this
extent or keypresses made while @code{point} is within the extent.
@xref{Extents and Events}.

@item copy-function
This is a hook that is run when a duplicable extent is about to be
copied from a buffer to a string (or the kill ring).  @xref{Duplicable
Extents}.

@item paste-function
This is a hook that is run when a duplicable extent is about to be
copied from a string (or the kill ring) into a buffer.  @xref{Duplicable
Extents}.

@item begin-glyph
(glyph or @code{nil}) This extent's begin glyph.
@xref{Annotations}.

@item end-glyph
(glyph or @code{nil}) This extent's end glyph.
@xref{Annotations}.

@item begin-glyph-layout
(@code{text}, @code{whitespace}, @code{inside-margin}, or
@code{outside-margin}) The layout policy for this extent's begin glyph.
Defaults to @code{text}.  @xref{Annotations}.

@item end-glyph-layout
(@code{text}, @code{whitespace}, @code{inside-margin}, or
@code{outside-margin}) The layout policy for this extent's end glyph.
Defaults to @code{text}.  @xref{Annotations}.

@item initial-redisplay-function
(any funcallable object) The function to be called the first time (a
part of) the extent is redisplayed.  It will be called with the extent
as its argument.

This is used by @code{lazy-shot} to implement lazy font-locking.  The
functionality is still experimental, and may change without further
notice.
@end table

The following convenience functions are provided for accessing
particular properties of an extent.

@defun extent-face extent
This function returns the @code{face} property of @var{extent}.  This
might also return a list of face names.  Do not modify this list
directly!  Instead, use @code{set-extent-face}.

Note that you can use @code{eq} to compare lists of faces as returned
by @code{extent-face}.  In other words, if you set the face of two
different extents to two lists that are @code{equal} but not @code{eq},
then the return value of @code{extent-face} on the two extents will
return the identical list.
@end defun

@defun extent-mouse-face extent
This function returns the @code{mouse-face} property of @var{extent}.
This might also return a list of face names.  Do not modify this list
directly!  Instead, use @code{set-extent-mouse-face}.

Note that you can use @code{eq} to compare lists of faces as returned
by @code{extent-mouse-face}, just like for @code{extent-face}.
@end defun

@defun extent-priority extent
This function returns the @code{priority} property of @var{extent}.
@end defun

@defun extent-keymap extent
This function returns the @code{keymap} property of @var{extent}.
@end defun

@defun extent-begin-glyph-layout extent
This function returns the @code{begin-glyph-layout} property of
@var{extent}, i.e. the layout policy associated with the @var{extent}'s
begin glyph.
@end defun

@defun extent-end-glyph-layout extent
This function returns the @code{end-glyph-layout} property of
@var{extent}, i.e. the layout policy associated with the @var{extent}'s
end glyph.
@end defun

@defun extent-begin-glyph extent
This function returns the @code{begin-glyph} property of @var{extent},
i.e. the glyph object displayed at the beginning of @var{extent}.  If
there is none, @code{nil} is returned.
@end defun

@defun extent-end-glyph extent
This function returns the @code{end-glyph} property of @var{extent},
i.e. the glyph object displayed at the end of @var{extent}.  If
there is none, @code{nil} is returned.
@end defun

The following convenience functions are provided for setting particular
properties of an extent.

@defun set-extent-priority extent priority
This function sets the @code{priority} property of @var{extent} to
@var{priority}.
@end defun

@defun set-extent-face extent face
This function sets the @code{face} property of @var{extent} to
@var{face}.
@end defun

@defun set-extent-mouse-face extent face
This function sets the @code{mouse-face} property of @var{extent} to
@var{face}.
@end defun

@defun set-extent-keymap extent keymap
This function sets the @code{keymap} property of @var{extent} to
@var{keymap}.  @var{keymap} must be either a keymap object, or
@code{nil}.
@end defun

@defun set-extent-begin-glyph-layout extent layout
This function sets the @code{begin-glyph-layout} property of
@var{extent} to @var{layout}.
@end defun

@defun set-extent-end-glyph-layout extent layout
This function sets the @code{end-glyph-layout} property of
@var{extent} to @var{layout}.
@end defun

@defun set-extent-begin-glyph extent begin-glyph &optional layout
This function sets the @code{begin-glyph} and @code{glyph-layout}
properties of @var{extent} to @var{begin-glyph} and @var{layout},
respectively. (@var{layout} defaults to @code{text} if not specified.)
@end defun

@defun set-extent-end-glyph extent end-glyph &optional layout
This function sets the @code{end-glyph} and @code{glyph-layout}
properties of @var{extent} to @var{end-glyph} and @var{layout},
respectively. (@var{layout} defaults to @code{text} if not specified.)
@end defun

@defun set-extent-initial-redisplay-function extent function
This function sets the @code{initial-redisplay-function} property of the
extent to @var{function}.
@end defun

@node Detached Extents
@section Detached Extents
@cindex detached extent

A detached extent is an extent that is not attached to a buffer or
string but can be re-inserted.  Detached extents have a start position
and end position of @code{nil}.  Extents can be explicitly detached
using @code{detach-extent}.  An extent is also detached when all of its
characters are all killed by a deletion, if its @code{detachable}
property is set; if this property is not set, the extent becomes a
zero-length extent. (Zero-length extents with the @code{detachable}
property set behave specially.  @xref{Extent Endpoints, zero-length
extents}.)

@defun detach-extent extent
This function detaches @var{extent} from its buffer or string.  If
@var{extent} has the @code{duplicable} property, its detachment is
tracked by the undo mechanism.  @xref{Duplicable Extents}.
@end defun

@defun extent-detached-p extent
This function returns @code{nil} if @var{extent} is detached, and
@code{t} otherwise.
@end defun

@defun copy-extent extent &optional object
This function makes a copy of @var{extent}.  It is initially detached.
Optional argument @var{object} defaults to @var{extent}'s object
(normally a buffer or string, but could be @code{nil}).
@end defun

@defun insert-extent extent &optional start end no-hooks object
This function inserts @var{extent} from @var{start} to @var{end} in
@var{object} (a buffer or string).  If @var{extent} is detached from a
different buffer or string, or in most cases when @var{extent} is
already attached, the extent will first be copied as if with
@code{copy-extent}.  This function operates the same as if @code{insert}
were called on a string whose extent data calls for @var{extent} to be
inserted, except that if @var{no-hooks} is non-@code{nil},
@var{extent}'s @code{paste-function} will not be invoked.
@xref{Duplicable Extents}.
@end defun

@node Extent Parents
@section Extent Parents
@cindex extent parent
@cindex extent children
@cindex parent, of extent
@cindex children, of extent

  An extent can have a parent extent set for it.  If this is the case,
the extent derives all its properties from that extent and has no
properties of its own.  The only ``properties'' that the extent keeps
are the buffer or string it refers to and the start and end points.
(More correctly, the extent's own properties are shadowed.  If you
later change the extent to have no parent, its own properties will
become visible again.)

  It is possible for an extent's parent to itself have a parent,
and so on.  Through this, a whole tree of extents can be created,
all deriving their properties from one root extent.  Note, however,
that you cannot create an inheritance loop---this is explicitly
disallowed.

  Parent extents are used to implement the extents over the modeline.

@defun set-extent-parent extent parent
This function sets the parent of @var{extent} to @var{parent}.
If @var{parent} is @code{nil}, the extent is set to have no parent.
@end defun

@defun extent-parent extent
This function return the parents (if any) of @var{extent}, or
@code{nil}.
@end defun

@defun extent-children extent
This function returns a list of the children (if any) of @var{extent}.
The children of an extent are all those extents whose parent is that
extent.  This function does not recursively trace children of children.
@end defun

@defun extent-descendants extent
This function returns a list of all descendants of @var{extent},
including @var{extent}.  This recursively applies @code{extent-children}
to any children of @var{extent}, until no more children can be found.
@end defun

@node Duplicable Extents
@section Duplicable Extents
@cindex duplicable extent
@cindex unique extents
@cindex extent replica
@cindex extent, duplicable
@cindex extent, unique

  If an extent has the @code{duplicable} property, it will be copied into
strings, so that kill, yank, and undo commands will restore or copy it.

Specifically:

@itemize @bullet
@item
When a string is created using @code{buffer-substring} or
@code{buffer-string}, any duplicable extents in the region corresponding
to the string will be copied into the string (@pxref{Buffer
Contents}).  When the string is inserted into a buffer using
@code{insert}, @code{insert-before-markers}, @code{insert-buffer} or
@code{insert-buffer-substring}, the extents in the string will be copied
back into the buffer (@pxref{Insertion}).  The extents in a string can,
of course, be retrieved explicitly using the standard extent primitives
over the string.

@item
Similarly, when text is copied or cut into the kill ring, any duplicable
extents will be remembered and reinserted later when the text is pasted
back into a buffer.

@item
When @code{concat} is called on strings, the extents in the strings are
copied into the resulting string.

@item
When @code{substring} is called on a string, the relevant extents
are copied into the resulting string.

@item
When a duplicable extent is detached by @code{detach-extent} or string
deletion, or inserted by @code{insert-extent} or string insertion, the
action is recorded by the undo mechanism so that it can be undone later.
Note that if an extent gets detached and then a later undo causes the
extent to get reinserted, the new extent will not be `eq' to the original
extent.

@item
Extent motion, face changes, and attachment via @code{make-extent} are
not recorded by the undo mechanism.  This means that extent changes
which are to be undo-able must be performed by character editing, or by
insertion and detachment of duplicable extents.

@item
A duplicable extent's @code{copy-function} property, if non-@code{nil},
should be a function, and will be run when a duplicable extent is about
to be copied from a buffer to a string (or the kill ring).  It is called
with three arguments: the extent and the buffer positions within it
which are being copied.  If this function returns @code{nil}, then the
extent will not be copied; otherwise it will.

@item
A duplicable extent's @code{paste-function} property, if non-@code{nil},
should be a function, and will be run when a duplicable extent is about
to be copied from a string (or the kill ring) into a buffer.  It is
called with three arguments: the original extent and the buffer
positions which the copied extent will occupy. (This hook is run after
the corresponding text has already been inserted into the buffer.) Note
that the extent argument may be detached when this function is run.  If
this function returns @code{nil}, no extent will be inserted.
Otherwise, there will be an extent covering the range in question.

  Note: if the extent to be copied is already attached to the buffer and
overlaps the new range, the extent will simply be extended and the
@code{paste-function} will not be called.
@end itemize

@node Extents and Events
@section Interaction of Extents with Keyboard and Mouse Events

  If an extent has the @code{mouse-face} property set, it will be
highlighted when the mouse passes over it.  This includes the
begin-glyph, but not the end-glyph.  Highlighting is accomplished by
merging the extent's face with the face or faces specified by the
@code{mouse-face} property.  The effect is as if a pseudo-extent with
the @code{mouse-face} face were inserted after the extent in the display
order (@pxref{Extent Endpoints}, display order).

@defvar mouse-highlight-priority
This variable holds the priority to use when merging in the highlighting
pseudo-extent.  The default is 1000.  This is purposely set very high
so that the highlighting pseudo-extent shows up even if there are other
extents with various priorities at the same location.
@end defvar

  You can also explicitly cause an extent to be highlighted.  Only one
extent at a time can be highlighted in this fashion, and any other
highlighted extent will be de-highlighted.

@defun highlight-extent extent &optional highlight-p
This function highlights (if @var{highlight-p} is non-@code{nil}) or
de-highlights (if @var{highlight-p} is @code{nil}) @var{extent}, if
@var{extent} has the @code{mouse-face} property. (Nothing happens if
@var{extent} does not have the @code{mouse-face} property.)
@end defun

@defun force-highlight-extent extent &optional highlight-p
This function is similar to @code{highlight-extent} but highlights
or de-highlights the extent regardless of whether it has the
@code{mouse-face} property.
@end defun

  If an extent has a @code{keymap} property, this keymap will be
consulted for mouse clicks on the extent and keypresses made while
@code{point} is within the extent.  The behavior of mouse clicks and
keystrokes not defined in the keymap is as normal for the buffer.

@node Atomic Extents
@section Atomic Extents
@cindex atomic extent

  If the Lisp file @file{atomic-extents} is loaded, then the atomic
extent facility is available.  An @dfn{atomic extent} is an extent for
which @code{point} cannot be positioned anywhere within it.  This
ensures that when selecting text, either all or none of the extent is
selected.

  To make an extent atomic, set its @code{atomic} property.