File: hypertext.rb

package info (click to toggle)
ruby-gnome2 3.1.0-1%2Bdeb9u1
  • links: PTS, VCS
  • area: main
  • in suites: stretch
  • size: 16,072 kB
  • ctags: 17,433
  • sloc: ansic: 93,621; ruby: 62,273; xml: 335; sh: 246; makefile: 25
file content (190 lines) | stat: -rw-r--r-- 5,364 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
# Copyright (c) 2016 Ruby-GNOME2 Project Team
# This program is licenced under the same licence as Ruby-GNOME2.
#
=begin
=  Text View/Hypertext

 Usually, tags modify the appearance of text in the view, e.g. making it
 bold or colored or underlined. But tags are not restricted to appearance.
 They can also affect the behavior of mouse and key presses, as this demo
 shows.
=end
class HypertextDemo
  def initialize(main_window)
    initialize_window(main_window)
    initialize_cursors(main_window)
    initialize_view

    @view.signal_connect "key-press-event" do |_widget, event|
      case event.keyval
      when Gdk::Keyval::KEY_Return, Gdk::Keyval::KEY_KP_Enter
        iter = @buffer.get_iter_at_mark(@buffer.get_mark("insert"))
        follow_if_link(iter) if iter
      end

      false
    end

    # Links can also be activated by clicking or tapping.
    @view.signal_connect "event-after" do |widget, event|
      if event.is_a?(Gdk::EventButton) && event.button == 1
        buffer = widget.buffer

        # we shouldn't follow a link if the user has selected something
        range = buffer.selection_bounds
        return false if range && range[0].offset != range[1].offset

        x, y = widget.window_to_buffer_coords(:widget, event.x, event.y)
        iter = widget.get_iter_at_location(x, y)
        follow_if_link(iter) if iter
      else
        false
      end
    end

    @view.signal_connect "motion-notify-event" do |widget, event|
      x, y = widget.window_to_buffer_coords(:widget, event.x, event.y)
      set_cursor_if_appropriate(widget, x, y)
      widget.window.pointer

      false
    end

    sw = Gtk::ScrolledWindow.new
    sw.set_policy(:automatic, :automatic)
    @window.add(sw)
    sw.add(@view)
    show_page(1)
    sw.show_all
  end

  def run
    if !@window.visible?
      @window.show_all
    else
      @window.destroy
    end
    @window
  end

  private

  def initialize_window(main_window)
    @window = Gtk::Window.new(:toplevel)
    @window.screen = main_window.screen
    @window.title = "Hypertext"
    @window.set_default_size(450, 450)
    @window.border_width = 0
  end

  def initialize_cursors(main_window)
    display = main_window.display
    @hand_cursor = Gdk::Cursor.new("pointer", :display => display)
    @regular_cursor = Gdk::Cursor.new("text", :display => display)
  end

  def initialize_view
    @view = Gtk::TextView.new
    @view.wrap_mode = :word
    @view.left_margin = 20
    @view.right_margin = 20
    @buffer = @view.buffer
  end

  # Fills the buffer with text and interspersed links. In any real
  # hypertext app, this method would parse a file to identify the links.
  def show_page(page)
    @buffer.text = ""
    case page
    when 1
      generate_page_1
    when 2
      generate_page_2
    when 3
      generate_page_3
    end
  end

  def generate_page_1
    iter = @buffer.get_iter_at(:offset => 0)
    @buffer.insert(iter, "Some text to show that simple")
    insert_link(iter, "hyper text", 3)
    @buffer.insert(iter, " can easily be realized with ")
    insert_link(iter, "tags", 2)
    @buffer.insert(iter, ".")
  end

  def generate_page_2
    iter = @buffer.get_iter_at(:offset => 0)
    @buffer.insert(iter, <<-EOF)
A tag is an attribute that can be applied to some range of text.
For example, a tag might be called "bold" and make the text inside
the tag bold. However, the tag concept is more general than that;
tags don't have to affect appearance. They can instead affect the
behavior of mouse and key presses, "lock" a range of text so the
user can't edit it, or countless other things.\n
EOF
    insert_link(iter, "Goback", 1)
  end

  def generate_page_3
    iter = @buffer.get_iter_at(:offset => 0)
    tag = @buffer.create_tag(nil,
                             "weight" => Pango::FontDescription::WEIGHT_BOLD)
    @buffer.insert(iter, "hypertext:\n", :tags => [tag])
    @buffer.insert(iter, <<-EOF)
machine-readable text that is not sequential but is organized
so that related items of information are connected
EOF
    insert_link(iter, "Go back", 1)
  end

  def insert_link(iter, text, page)
    tag = @buffer.create_tag(nil,
                             "foreground" => "blue",
                             "underline" => Pango::AttrUnderline::SINGLE)
    tag.page = page
    @buffer.insert(iter, text, :tags => [tag])
  end

  # Looks at all tags covering the position of iter in the text view,
  # and if one of them is a link, follow it by showing the page identified
  # by the data attached to it.
  def follow_if_link(iter)
    tags = iter.tags
    tags.each do |tag|
      if tag.page
        show_page(tag.page)
        break
      end
    end
  end

  # Looks at all tags covering the position (x, y) in the text view,
  # and if one of them is a link, change the cursor to the "hands" cursor
  # typically used by web browsers.
  def set_cursor_if_appropriate(text_view, x, y)
    iter = text_view.get_iter_at_location(x, y)
    return unless iter
    hovering = false
    tags = iter.tags
    tags.each do |tag|
      if tag.page
        hovering = true
        break
      end
    end

    if hovering != @hovering
      @hovering = hovering
      window = text_view.get_window(:text)
      window.cursor = @hovering ? @hand_cursor : @regular_cursor
    end
  end
end

module Gtk
  class TextTag
    attr_accessor :page
  end
end