forked from AuroraMiddleware/gtk
a4762fbff4
2000-07-25 Havoc Pennington <hp@redhat.com> * gtk/testtext.c, gtk/testtextbuffer.c: update to reflect text widget changes. * gtk/gtktextview.h: To be consistent with usage of "line" throughout the API to mean "newline-terminated thingy", change MOVEMENT_LINE to be MOVEMENT_WRAPPED_LINE, and MOVEMENT_PARAGRAPH to MOVEMENT_LINE. (GtkTextView): Add flags for default editability, and whether to show the cursor. Add functions to get/set that. Add (gtk_text_view_get_iter_location): new function * gtk/gtktexttypes.h: Move GtkTextLine typedef from here to gtktextlayout.h (g_convert): Add g_convert temporarily, will go in glib in a bit * gtk/gtktexttagtable.h: include gtktexttag.h, and define GtkTextTagTableForeach instead of brokenly using GHFunc. Change gtk_text_tag_table_foreach() so it doesn't use GHFunc. * gtk/gtktexttagprivate.h: Remove GtkTextStyleValues from here, moved to public header. * gtk/gtktexttag.h: Rename the "elide" attribute of tags to "invisible", since "elide" was a bad name. (gtk_text_tag_get_priority): Added (GtkTextStyleValues): put this in public header, along with functions to use it. * gtk/gtktextmarkprivate.h: Include more headers, since we no longer include gtktextbtree.h. * gtk/gtktextmark.h: Add gtk_text_mark_ref, gtk_text_mark_unref, gtk_text_mark_deleted * gtk/gtktextlayout.h: Don't include the "really private" headers, only buffer/iter. Forward declare GtkTextLIne and GtkTextLineData to make this possible. Now we only need to install gtktextlayout.h, not gtktextbtree.h and gtktext*private.h. (However the Makefile.am isn't changed yet because of the logistics of merging gtk-hp-patches piecemeal) * gtk/gtktextiterprivate.h: include btree header, so it compiles; rename gtk_text_iter_get_line to gtk_text_iter_get_text_line since gtk_text_iter_get_line is now used in the public API for a different purpose. * gtk/gtktextiter.h: Clean up function names to be more consistent. Always call char offset "offset" and byte index "index". A "line" is always a line number. (gtk_text_iter_is_last): new function, more efficient than the existing way to check (gtk_text_iter_is_first): new function, also more efficient (gtk_text_iter_up_lines, gtk_text_iter_down_lines): Remove these (gtk_text_iter_next_char, gtk_text_iter_prev_char): Renamed from gtk_text_iter_forward_char, etc. (gtk_text_iter_forward_to_tag_toggle): Renamed from forward_find_tag_toggle, since this isn't a linear search (GtkTextCharPredicate): rename from GtkTextViewCharPredicate (gtk_text_iter_forward_search, gtk_text_iter_backward_search): New functions, search for a buffer substring. * gtk/gtktextbuffer.h: Add fields to store whether a paste is interactive and default editable (since we need to store that info until we receive the selection data). Remove all the _at_char and at_line etc. versions of functions; only have iterator versions. Add _interactive() versions of functions, that consider the editability of text. (FIXME add interactive flag to the insert/delete signals per Darin's suggestion) (gtk_text_buffer_get_tag_table): new function, demand-creates the tag table if necessary Remove declaration of gtk_text_buffer_get_iter_from_string (_gtk_text_buffer_get_btree): private/internal function, added. * gtk/gtktextbtree.h: Remove forward decl of GtkTextLineData. (gtk_text_line_is_last): new function
376 lines
9.4 KiB
C
376 lines
9.4 KiB
C
/*
|
||
* tkTextMark.c --
|
||
*
|
||
* This file contains the procedure that implement marks for
|
||
* text widgets.
|
||
*
|
||
* Copyright (c) 1994 The Regents of the University of California.
|
||
* Copyright (c) 1994-1997 Sun Microsystems, Inc.
|
||
*
|
||
* See the file "license.terms" for information on usage and redistribution
|
||
* of this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
||
*
|
||
* RCS: @(#) $Id$
|
||
*/
|
||
|
||
#include "gtktextbtree.h"
|
||
|
||
gboolean
|
||
gtk_text_mark_is_visible(GtkTextMark *mark)
|
||
{
|
||
GtkTextLineSegment *seg;
|
||
|
||
seg = (GtkTextLineSegment*)mark;
|
||
|
||
return seg->body.mark.visible;
|
||
}
|
||
|
||
char *
|
||
gtk_text_mark_get_name (GtkTextMark *mark)
|
||
{
|
||
GtkTextLineSegment *seg;
|
||
|
||
seg = (GtkTextLineSegment*)mark;
|
||
|
||
return g_strdup (seg->body.mark.name);
|
||
}
|
||
|
||
|
||
GtkTextMark *
|
||
gtk_text_mark_ref (GtkTextMark *mark)
|
||
{
|
||
GtkTextLineSegment *seg;
|
||
|
||
seg = (GtkTextLineSegment*)mark;
|
||
|
||
mark_segment_ref (seg);
|
||
|
||
return mark;
|
||
}
|
||
|
||
void
|
||
gtk_text_mark_unref (GtkTextMark *mark)
|
||
{
|
||
GtkTextLineSegment *seg;
|
||
|
||
seg = (GtkTextLineSegment*)mark;
|
||
|
||
mark_segment_unref (seg);
|
||
}
|
||
|
||
gboolean
|
||
gtk_text_mark_deleted (GtkTextMark *mark)
|
||
{
|
||
GtkTextLineSegment *seg;
|
||
|
||
g_return_val_if_fail (mark != NULL, FALSE);
|
||
|
||
seg = (GtkTextLineSegment*)mark;
|
||
|
||
return seg->body.mark.tree == NULL;
|
||
}
|
||
|
||
/*
|
||
* Macro that determines the size of a mark segment:
|
||
*/
|
||
|
||
#define MSEG_SIZE ((unsigned) (G_STRUCT_OFFSET(GtkTextLineSegment, body) \
|
||
+ sizeof(GtkTextMarkBody)))
|
||
|
||
|
||
GtkTextLineSegment*
|
||
mark_segment_new(GtkTextBTree *tree,
|
||
gboolean left_gravity,
|
||
const gchar *name)
|
||
{
|
||
GtkTextLineSegment *mark;
|
||
|
||
mark = (GtkTextLineSegment *) g_malloc0(MSEG_SIZE);
|
||
mark->body.mark.name = g_strdup(name);
|
||
|
||
if (left_gravity)
|
||
mark->type = >k_text_left_mark_type;
|
||
else
|
||
mark->type = >k_text_right_mark_type;
|
||
|
||
mark->byte_count = 0;
|
||
mark->char_count = 0;
|
||
|
||
mark->body.mark.tree = tree;
|
||
mark->body.mark.line = NULL;
|
||
mark->next = NULL;
|
||
|
||
mark->body.mark.refcount = 1;
|
||
|
||
mark->body.mark.visible = FALSE;
|
||
mark->body.mark.not_deleteable = FALSE;
|
||
|
||
return mark;
|
||
}
|
||
|
||
void
|
||
mark_segment_ref(GtkTextLineSegment *mark)
|
||
{
|
||
g_return_if_fail(mark != NULL);
|
||
g_return_if_fail(mark->type == >k_text_right_mark_type ||
|
||
mark->type == >k_text_left_mark_type);
|
||
g_return_if_fail(mark->body.mark.refcount > 0);
|
||
|
||
mark->body.mark.refcount += 1;
|
||
}
|
||
|
||
void
|
||
mark_segment_unref(GtkTextLineSegment *mark)
|
||
{
|
||
g_return_if_fail(mark != NULL);
|
||
g_return_if_fail(mark->type == >k_text_right_mark_type ||
|
||
mark->type == >k_text_left_mark_type);
|
||
g_return_if_fail(mark->body.mark.refcount > 0);
|
||
|
||
mark->body.mark.refcount -= 1;
|
||
|
||
if (mark->body.mark.refcount == 0)
|
||
{
|
||
g_free(mark->body.mark.name);
|
||
|
||
g_free(mark);
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Forward references for procedures defined in this file:
|
||
*/
|
||
|
||
static int mark_segment_delete_func (GtkTextLineSegment *segPtr,
|
||
GtkTextLine *line,
|
||
int treeGone);
|
||
static GtkTextLineSegment *mark_segment_cleanup_func (GtkTextLineSegment *segPtr,
|
||
GtkTextLine *line);
|
||
static void mark_segment_check_func (GtkTextLineSegment *segPtr,
|
||
GtkTextLine *line);
|
||
|
||
/*
|
||
* The following structures declare the "mark" segment types.
|
||
* There are actually two types for marks, one with left gravity
|
||
* and one with right gravity. They are identical except for
|
||
* their gravity property.
|
||
*/
|
||
|
||
GtkTextLineSegmentClass gtk_text_right_mark_type = {
|
||
"mark", /* name */
|
||
FALSE, /* leftGravity */
|
||
(GtkTextLineSegmentSplitFunc) NULL, /* splitFunc */
|
||
mark_segment_delete_func, /* deleteFunc */
|
||
mark_segment_cleanup_func, /* cleanupFunc */
|
||
(GtkTextLineSegmentLineChangeFunc) NULL, /* lineChangeFunc */
|
||
mark_segment_check_func /* checkFunc */
|
||
};
|
||
|
||
GtkTextLineSegmentClass gtk_text_left_mark_type = {
|
||
"mark", /* name */
|
||
TRUE, /* leftGravity */
|
||
(GtkTextLineSegmentSplitFunc) NULL, /* splitFunc */
|
||
mark_segment_delete_func, /* deleteFunc */
|
||
mark_segment_cleanup_func, /* cleanupFunc */
|
||
(GtkTextLineSegmentLineChangeFunc) NULL, /* lineChangeFunc */
|
||
mark_segment_check_func /* checkFunc */
|
||
};
|
||
|
||
|
||
/*
|
||
*--------------------------------------------------------------
|
||
*
|
||
* mark_segment_delete_func --
|
||
*
|
||
* This procedure is invoked by the text B-tree code whenever
|
||
* a mark lies in a range of characters being deleted.
|
||
*
|
||
* Results:
|
||
* Returns 1 to indicate that deletion has been rejected.
|
||
*
|
||
* Side effects:
|
||
* None (even if the whole tree is being deleted we don't
|
||
* free up the mark; it will be done elsewhere).
|
||
*
|
||
*--------------------------------------------------------------
|
||
*/
|
||
|
||
/* ARGSUSED */
|
||
static int
|
||
mark_segment_delete_func(segPtr, line, treeGone)
|
||
GtkTextLineSegment *segPtr; /* Segment being deleted. */
|
||
GtkTextLine *line; /* Line containing segment. */
|
||
int treeGone; /* Non-zero means the entire tree is
|
||
* being deleted, so everything must
|
||
* get cleaned up. */
|
||
{
|
||
return 1;
|
||
}
|
||
|
||
/*
|
||
*--------------------------------------------------------------
|
||
*
|
||
* mark_segment_cleanup_func --
|
||
*
|
||
* This procedure is invoked by the B-tree code whenever a
|
||
* mark segment is moved from one line to another.
|
||
*
|
||
* Results:
|
||
* None.
|
||
*
|
||
* Side effects:
|
||
* The line field of the segment gets updated.
|
||
*
|
||
*--------------------------------------------------------------
|
||
*/
|
||
|
||
static GtkTextLineSegment *
|
||
mark_segment_cleanup_func(markPtr, line)
|
||
GtkTextLineSegment *markPtr; /* Mark segment that's being moved. */
|
||
GtkTextLine *line; /* Line that now contains segment. */
|
||
{
|
||
markPtr->body.mark.line = line;
|
||
return markPtr;
|
||
}
|
||
|
||
#if 0
|
||
|
||
|
||
/*
|
||
*--------------------------------------------------------------
|
||
*
|
||
* GtkTextInsertDisplayFunc --
|
||
*
|
||
* This procedure is called to display the insertion
|
||
* cursor.
|
||
*
|
||
* Results:
|
||
* None.
|
||
*
|
||
* Side effects:
|
||
* Graphics are drawn.
|
||
*
|
||
*--------------------------------------------------------------
|
||
*/
|
||
|
||
/* ARGSUSED */
|
||
void
|
||
GtkTextInsertDisplayFunc(chunkPtr, x, y, height, baseline, display, dst, screenY)
|
||
GtkTextDisplayChunk *chunkPtr; /* Chunk that is to be drawn. */
|
||
int x; /* X-position in dst at which to
|
||
* draw this chunk (may differ from
|
||
* the x-position in the chunk because
|
||
* of scrolling). */
|
||
int y; /* Y-position at which to draw this
|
||
* chunk in dst (x-position is in
|
||
* the chunk itself). */
|
||
int height; /* Total height of line. */
|
||
int baseline; /* Offset of baseline from y. */
|
||
Display *display; /* Display to use for drawing. */
|
||
Drawable dst; /* Pixmap or window in which to draw
|
||
* chunk. */
|
||
int screenY; /* Y-coordinate in text window that
|
||
* corresponds to y. */
|
||
{
|
||
GtkTextView *tkxt = (GtkTextView *) chunkPtr->clientData;
|
||
int halfWidth = tkxt->insertWidth/2;
|
||
|
||
if ((x + halfWidth) < 0) {
|
||
/*
|
||
* The insertion cursor is off-screen. Just return.
|
||
*/
|
||
|
||
return;
|
||
}
|
||
|
||
/*
|
||
* As a special hack to keep the cursor visible on mono displays
|
||
* (or anywhere else that the selection and insertion cursors
|
||
* have the same color) write the default background in the cursor
|
||
* area (instead of nothing) when the cursor isn't on. Otherwise
|
||
* the selection might hide the cursor.
|
||
*/
|
||
|
||
if (tkxt->flags & INSERT_ON) {
|
||
Tk_Fill3DRectangle(tkxt->tkwin, dst, tkxt->insertBorder,
|
||
x - tkxt->insertWidth/2, y, tkxt->insertWidth,
|
||
height, tkxt->insertBorderWidth, TK_RELIEF_RAISED);
|
||
} else if (tkxt->selBorder == tkxt->insertBorder) {
|
||
Tk_Fill3DRectangle(tkxt->tkwin, dst, tkxt->border,
|
||
x - tkxt->insertWidth/2, y, tkxt->insertWidth,
|
||
height, 0, TK_RELIEF_FLAT);
|
||
}
|
||
}
|
||
|
||
/*
|
||
*--------------------------------------------------------------
|
||
*
|
||
* InsertUndisplayFunc --
|
||
*
|
||
* This procedure is called when the insertion cursor is no
|
||
* longer at a visible point on the display. It does nothing
|
||
* right now.
|
||
*
|
||
* Results:
|
||
* None.
|
||
*
|
||
* Side effects:
|
||
* None.
|
||
*
|
||
*--------------------------------------------------------------
|
||
*/
|
||
|
||
/* ARGSUSED */
|
||
static void
|
||
InsertUndisplayFunc(tkxt, chunkPtr)
|
||
GtkTextView *tkxt; /* Overall information about text
|
||
* widget. */
|
||
GtkTextDisplayChunk *chunkPtr; /* Chunk that is about to be freed. */
|
||
{
|
||
return;
|
||
}
|
||
|
||
#endif
|
||
/*
|
||
*--------------------------------------------------------------
|
||
*
|
||
* mark_segment_check_func --
|
||
*
|
||
* This procedure is invoked by the B-tree code to perform
|
||
* consistency checks on mark segments.
|
||
*
|
||
* Results:
|
||
* None.
|
||
*
|
||
* Side effects:
|
||
* The procedure panics if it detects anything wrong with
|
||
* the mark.
|
||
*
|
||
*--------------------------------------------------------------
|
||
*/
|
||
|
||
static void
|
||
mark_segment_check_func(markPtr, line)
|
||
GtkTextLineSegment *markPtr; /* Segment to check. */
|
||
GtkTextLine *line; /* Line containing segment. */
|
||
{
|
||
if (markPtr->body.mark.line != line)
|
||
g_error("mark_segment_check_func: markPtr->body.mark.line bogus");
|
||
|
||
/* No longer do this because we don't have access to btree
|
||
struct members */
|
||
#if 0
|
||
/*
|
||
* Make sure that the mark is still present in the text's mark
|
||
* hash table.
|
||
*/
|
||
for (hPtr = Tcl_FirstHashEntry(&markPtr->body.mark.tkxt->markTable,
|
||
&search); hPtr != markPtr->body.mark.hPtr;
|
||
hPtr = Tcl_NextHashEntry(&search)) {
|
||
if (hPtr == NULL) {
|
||
panic("mark_segment_check_func couldn't find hash table entry for mark");
|
||
}
|
||
}
|
||
#endif
|
||
}
|