252 lines
6.9 KiB
C
252 lines
6.9 KiB
C
/* simple.c
|
|
* Copyright (C) 2017 Red Hat, Inc
|
|
* Author: Benjamin Otte
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "config.h"
|
|
#include <gtk/gtk.h>
|
|
|
|
|
|
const char text[] =
|
|
"This library is free software; you can redistribute it and/or\n"
|
|
"modify it under the terms of the GNU Library General Public\n"
|
|
"License as published by the Free Software Foundation; either\n"
|
|
"version 2 of the License, or (at your option) any later version.\n"
|
|
"\n"
|
|
"This library is distributed in the hope that it will be useful,\n"
|
|
"but WITHOUT ANY WARRANTY; without even the implied warranty of\n"
|
|
"MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n"
|
|
"Library General Public License for more details.\n"
|
|
"\n"
|
|
"You should have received a copy of the GNU Library General Public\n"
|
|
"License along with this library. If not, see <http://www.gnu.org/licenses/>.\n";
|
|
|
|
static GtkWidget *tv;
|
|
static GtkTextBuffer *buffer;
|
|
static int len;
|
|
|
|
static GtkTextMark **marks;
|
|
static guint marks_timeout;
|
|
|
|
static gboolean toggle_mark (gpointer data)
|
|
{
|
|
int pos;
|
|
GtkTextMark *mark;
|
|
|
|
pos = g_random_int_range (0, len);
|
|
mark = marks[pos];
|
|
|
|
gtk_text_mark_set_visible (mark, !gtk_text_mark_get_visible (mark));
|
|
|
|
return G_SOURCE_CONTINUE;
|
|
}
|
|
|
|
static void
|
|
toggle_marks (GtkToggleButton *button)
|
|
{
|
|
int i;
|
|
gboolean enable;
|
|
|
|
enable = gtk_toggle_button_get_active (button);
|
|
|
|
if (!marks)
|
|
{
|
|
marks = g_new (GtkTextMark*, len);
|
|
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
marks[i] = gtk_text_mark_new (NULL, TRUE);
|
|
gtk_text_mark_set_visible (marks[i], i % 2);
|
|
}
|
|
}
|
|
|
|
if (enable)
|
|
{
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
GtkTextIter iter;
|
|
|
|
gtk_text_buffer_get_iter_at_offset (buffer, &iter, i);
|
|
gtk_text_buffer_add_mark (buffer, marks[i], &iter);
|
|
}
|
|
|
|
marks_timeout = g_timeout_add (16, toggle_mark, NULL);
|
|
}
|
|
else
|
|
{
|
|
for (i = 0; i < len; i++)
|
|
gtk_text_buffer_delete_mark (buffer, marks[i]);
|
|
|
|
if (marks_timeout)
|
|
g_source_remove (marks_timeout);
|
|
marks_timeout = 0;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
move_insert (gpointer data)
|
|
{
|
|
GtkTextMark *mark;
|
|
GtkTextIter iter, start, end;
|
|
|
|
mark = gtk_text_buffer_get_insert (buffer);
|
|
gtk_text_buffer_get_iter_at_mark (buffer, &iter, mark);
|
|
gtk_text_buffer_get_bounds (buffer, &start, &end);
|
|
|
|
if (gtk_text_iter_equal (&iter, &end))
|
|
gtk_text_iter_assign (&iter, &start);
|
|
else
|
|
gtk_text_iter_forward_cursor_position (&iter);
|
|
|
|
gtk_text_buffer_place_cursor (buffer, &iter);
|
|
|
|
return G_SOURCE_CONTINUE;
|
|
}
|
|
|
|
static guint cursor_timeout;
|
|
|
|
static void
|
|
toggle_cursor (GtkToggleButton *button)
|
|
{
|
|
gboolean enable;
|
|
|
|
enable = gtk_toggle_button_get_active (button);
|
|
if (enable)
|
|
cursor_timeout = g_timeout_add (16, move_insert, NULL);
|
|
else
|
|
{
|
|
if (cursor_timeout)
|
|
g_source_remove (cursor_timeout);
|
|
cursor_timeout = 0;
|
|
}
|
|
}
|
|
|
|
static GtkTextMark *the_mark;
|
|
static GtkWidget *mark_check;
|
|
static GtkWidget *mark_visible;
|
|
static GtkWidget *position_spin;
|
|
|
|
static void
|
|
update_mark_exists (void)
|
|
{
|
|
int pos;
|
|
GtkTextIter iter;
|
|
|
|
pos = gtk_spin_button_get_value_as_int (GTK_SPIN_BUTTON (position_spin));
|
|
gtk_text_buffer_get_iter_at_offset (buffer, &iter, pos);
|
|
|
|
if (gtk_check_button_get_active (GTK_CHECK_BUTTON (mark_check)))
|
|
gtk_text_buffer_add_mark (buffer, the_mark, &iter);
|
|
else
|
|
gtk_text_buffer_delete_mark (buffer, the_mark);
|
|
}
|
|
|
|
static void
|
|
update_mark_visible (void)
|
|
{
|
|
gtk_text_mark_set_visible (the_mark, gtk_check_button_get_active (GTK_CHECK_BUTTON (mark_visible)));
|
|
}
|
|
|
|
static void
|
|
update_mark_position (void)
|
|
{
|
|
int pos;
|
|
GtkTextIter iter;
|
|
|
|
pos = gtk_spin_button_get_value_as_int (GTK_SPIN_BUTTON (position_spin));
|
|
gtk_text_buffer_get_iter_at_offset (buffer, &iter, pos);
|
|
|
|
gtk_text_buffer_move_mark (buffer, the_mark, &iter);
|
|
}
|
|
|
|
static void
|
|
quit_cb (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
gboolean *done = data;
|
|
|
|
*done = TRUE;
|
|
|
|
g_main_context_wakeup (NULL);
|
|
}
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
GtkWidget *window, *sw, *box, *box2, *button;
|
|
gboolean done = FALSE;
|
|
|
|
gtk_init ();
|
|
|
|
window = gtk_window_new ();
|
|
gtk_window_set_default_size (GTK_WINDOW (window), 600, 400);
|
|
g_signal_connect (window, "destroy", G_CALLBACK (quit_cb), &done);
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 10);
|
|
|
|
sw = gtk_scrolled_window_new ();
|
|
gtk_widget_set_hexpand (sw, TRUE);
|
|
gtk_widget_set_vexpand (sw, TRUE);
|
|
gtk_window_set_child (GTK_WINDOW (window), box);
|
|
gtk_box_append (GTK_BOX (box), sw);
|
|
|
|
tv = gtk_text_view_new ();
|
|
gtk_scrolled_window_set_child (GTK_SCROLLED_WINDOW (sw), tv);
|
|
|
|
buffer = gtk_text_buffer_new (NULL);
|
|
gtk_text_view_set_buffer (GTK_TEXT_VIEW (tv), buffer);
|
|
|
|
gtk_text_buffer_set_text (buffer, text, -1);
|
|
|
|
len = strlen (text);
|
|
|
|
box2 = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 10);
|
|
g_object_set (box, "margin-start", 10, "margin-end", 10, NULL);
|
|
gtk_box_append (GTK_BOX (box), box2);
|
|
|
|
the_mark = gtk_text_mark_new ("my mark", TRUE);
|
|
box2 = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 10);
|
|
gtk_box_append (GTK_BOX (box), box2);
|
|
mark_check = gtk_check_button_new_with_label ("Mark");
|
|
g_signal_connect (mark_check, "notify::active", G_CALLBACK (update_mark_exists), NULL);
|
|
gtk_box_append (GTK_BOX (box2), mark_check);
|
|
mark_visible = gtk_check_button_new_with_label ("Visible");
|
|
g_signal_connect (mark_visible, "notify::active", G_CALLBACK (update_mark_visible), NULL);
|
|
gtk_box_append (GTK_BOX (box2), mark_visible);
|
|
gtk_box_append (GTK_BOX (box2), gtk_label_new ("Position:"));
|
|
position_spin = gtk_spin_button_new_with_range (0, len, 1);
|
|
g_signal_connect (position_spin, "value-changed", G_CALLBACK (update_mark_position), NULL);
|
|
gtk_box_append (GTK_BOX (box2), position_spin);
|
|
|
|
box2 = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 10);
|
|
g_object_set (box, "margin-start", 10, "margin-end", 10, NULL);
|
|
gtk_box_append (GTK_BOX (box), box2);
|
|
|
|
button = gtk_toggle_button_new_with_label ("Random marks");
|
|
g_signal_connect (button, "notify::active", G_CALLBACK (toggle_marks), NULL);
|
|
gtk_box_append (GTK_BOX (box2), button);
|
|
|
|
button = gtk_toggle_button_new_with_label ("Wandering cursor");
|
|
g_signal_connect (button, "notify::active", G_CALLBACK (toggle_cursor), NULL);
|
|
gtk_box_append (GTK_BOX (box2), button);
|
|
|
|
gtk_window_present (GTK_WINDOW (window));
|
|
|
|
while (!done)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
return 0;
|
|
}
|