Class: Textbringer::Window
- Inherits:
-
Object
- Object
- Textbringer::Window
- Defined in:
- lib/textbringer/window.rb
Direct Known Subclasses
Constant Summary collapse
- KEY_NAMES =
{}
- HAVE_GET_KEY_MODIFIERS =
defined?(Curses.get_key_modifiers)
- ALT_NUMBER_BASE =
Curses::ALT_0 - ?0.ord
- ALT_ALPHA_BASE =
Curses::ALT_A - ?a.ord
- @@started =
false- @@list =
[]
- @@current =
nil- @@echo_area =
nil- @@has_colors =
false
Instance Attribute Summary collapse
-
#bottom_of_window ⇒ Object
readonly
Returns the value of attribute bottom_of_window.
-
#buffer ⇒ Object
Returns the value of attribute buffer.
-
#columns ⇒ Object
readonly
Returns the value of attribute columns.
-
#lines ⇒ Object
readonly
Returns the value of attribute lines.
-
#mode_line ⇒ Object
readonly
Returns the value of attribute mode_line.
-
#top_of_window ⇒ Object
readonly
Returns the value of attribute top_of_window.
-
#window ⇒ Object
readonly
Returns the value of attribute window.
-
#x ⇒ Object
readonly
Returns the value of attribute x.
-
#y ⇒ Object
readonly
Returns the value of attribute y.
Class Method Summary collapse
- .beep ⇒ Object
- .colors ⇒ Object
- .columns ⇒ Object
- .current ⇒ Object
- .current=(window) ⇒ Object
- .delete_other_windows ⇒ Object
- .delete_window ⇒ Object
- .echo_area ⇒ Object
- .has_colors=(value) ⇒ Object
- .has_colors? ⇒ Boolean
- .lines ⇒ Object
- .list(include_echo_area: false) ⇒ Object
- .load_faces ⇒ Object
- .other_window ⇒ Object
- .redisplay ⇒ Object
- .redraw ⇒ Object
- .resize ⇒ Object
- .set_default_colors(fg, bg) ⇒ Object
- .start ⇒ Object
- .update ⇒ Object
Instance Method Summary collapse
- #active? ⇒ Boolean
- #close ⇒ Object
- #current? ⇒ Boolean
- #delete ⇒ Object
- #deleted? ⇒ Boolean
- #echo_area? ⇒ Boolean
- #enlarge(n) ⇒ Object
- #has_input? ⇒ Boolean
- #highlight ⇒ Object
-
#initialize(lines, columns, y, x) ⇒ Window
constructor
A new instance of Window.
- #move(y, x) ⇒ Object
- #read_event ⇒ Object
- #read_event_nonblock ⇒ Object
- #recenter ⇒ Object
- #recenter_if_needed ⇒ Object
- #redisplay ⇒ Object
- #redraw ⇒ Object
- #resize(lines, columns) ⇒ Object
- #restore_point ⇒ Object
- #save_point ⇒ Object
- #scroll_down ⇒ Object
- #scroll_up ⇒ Object
- #shrink(n) ⇒ Object
- #shrink_if_larger_than_buffer ⇒ Object
- #split ⇒ Object
- #wait_input(msecs) ⇒ Object
Constructor Details
#initialize(lines, columns, y, x) ⇒ Window
Returns a new instance of Window.
217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 |
# File 'lib/textbringer/window.rb', line 217 def initialize(lines, columns, y, x) @lines = lines @columns = columns @y = y @x = x initialize_window(lines, columns, y, x) @window.keypad = true @window.scrollok(false) @window.idlok(true) @buffer = nil @top_of_window = nil @bottom_of_window = nil @point_mark = nil @deleted = false @raw_key_buffer = [] @key_buffer = [] end |
Instance Attribute Details
#bottom_of_window ⇒ Object (readonly)
Returns the value of attribute bottom_of_window.
215 216 217 |
# File 'lib/textbringer/window.rb', line 215 def bottom_of_window @bottom_of_window end |
#buffer ⇒ Object
Returns the value of attribute buffer.
214 215 216 |
# File 'lib/textbringer/window.rb', line 214 def buffer @buffer end |
#columns ⇒ Object (readonly)
Returns the value of attribute columns.
214 215 216 |
# File 'lib/textbringer/window.rb', line 214 def columns @columns end |
#lines ⇒ Object (readonly)
Returns the value of attribute lines.
214 215 216 |
# File 'lib/textbringer/window.rb', line 214 def lines @lines end |
#mode_line ⇒ Object (readonly)
Returns the value of attribute mode_line.
214 215 216 |
# File 'lib/textbringer/window.rb', line 214 def mode_line @mode_line end |
#top_of_window ⇒ Object (readonly)
Returns the value of attribute top_of_window.
215 216 217 |
# File 'lib/textbringer/window.rb', line 215 def top_of_window @top_of_window end |
#window ⇒ Object (readonly)
Returns the value of attribute window.
214 215 216 |
# File 'lib/textbringer/window.rb', line 214 def window @window end |
#x ⇒ Object (readonly)
Returns the value of attribute x.
214 215 216 |
# File 'lib/textbringer/window.rb', line 214 def x @x end |
#y ⇒ Object (readonly)
Returns the value of attribute y.
214 215 216 |
# File 'lib/textbringer/window.rb', line 214 def y @y end |
Class Method Details
.beep ⇒ Object
210 211 212 |
# File 'lib/textbringer/window.rb', line 210 def self.beep Curses.beep end |
.colors ⇒ Object
103 104 105 |
# File 'lib/textbringer/window.rb', line 103 def self.colors Curses.colors end |
.columns ⇒ Object
183 184 185 |
# File 'lib/textbringer/window.rb', line 183 def self.columns Curses.cols end |
.current ⇒ Object
32 33 34 |
# File 'lib/textbringer/window.rb', line 32 def self.current @@current end |
.current=(window) ⇒ Object
36 37 38 39 40 41 42 43 44 |
# File 'lib/textbringer/window.rb', line 36 def self.current=(window) if window.deleted? window = @@list.first end @@current.save_point if @@current && !@@current.deleted? @@current = window @@current.restore_point Buffer.current = window.buffer end |
.delete_other_windows ⇒ Object
66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 |
# File 'lib/textbringer/window.rb', line 66 def self.delete_other_windows if @@current.echo_area? raise EditorError, "Can't expand the echo area to full screen" end @@list.delete_if do |window| if window.current? || window.echo_area? false else window.delete true end end @@current.move(0, 0) @@current.resize(Window.lines - 1, @@current.columns) end |
.delete_window ⇒ Object
46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 |
# File 'lib/textbringer/window.rb', line 46 def self.delete_window if @@current.echo_area? raise EditorError, "Can't delete the echo area" end if @@list.size == 2 raise EditorError, "Can't delete the sole window" end i = @@list.index(@@current) if i == 0 window = @@list[1] window.move(0, 0) else window = @@list[i - 1] end window.resize(@@current.lines + window.lines, window.columns) @@current.delete @@list.delete_at(i) self.current = window end |
.echo_area ⇒ Object
91 92 93 |
# File 'lib/textbringer/window.rb', line 91 def self.echo_area @@echo_area end |
.has_colors=(value) ⇒ Object
95 96 97 |
# File 'lib/textbringer/window.rb', line 95 def self.has_colors=(value) @@has_colors = value end |
.has_colors? ⇒ Boolean
99 100 101 |
# File 'lib/textbringer/window.rb', line 99 def self.has_colors? @@has_colors end |
.lines ⇒ Object
179 180 181 |
# File 'lib/textbringer/window.rb', line 179 def self.lines Curses.lines end |
.list(include_echo_area: false) ⇒ Object
24 25 26 27 28 29 30 |
# File 'lib/textbringer/window.rb', line 24 def self.list(include_echo_area: false) if include_echo_area @@list.dup else @@list.reject(&:echo_area?) end end |
.load_faces ⇒ Object
112 113 114 115 |
# File 'lib/textbringer/window.rb', line 112 def self.load_faces require_relative "faces/basic" require_relative "faces/programming" end |
.other_window ⇒ Object
82 83 84 85 86 87 88 89 |
# File 'lib/textbringer/window.rb', line 82 def self.other_window i = @@list.index(@@current) begin i += 1 window = @@list[i % @@list.size] end while !window.active? self.current = window end |
.redisplay ⇒ Object
157 158 159 160 161 162 163 164 165 |
# File 'lib/textbringer/window.rb', line 157 def self.redisplay return if Controller.current.executing_keyboard_macro? return if Window.current.has_input? @@list.each do |window| window.redisplay unless window.current? end current.redisplay update end |
.redraw ⇒ Object
167 168 169 170 171 172 173 |
# File 'lib/textbringer/window.rb', line 167 def self.redraw @@list.each do |window| window.redraw unless window.current? end current.redraw update end |
.resize ⇒ Object
187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 |
# File 'lib/textbringer/window.rb', line 187 def self.resize @@list.delete_if do |window| if !window.echo_area? && window.y > Window.lines - CONFIG[:window_min_height] window.delete true else false end end @@list.each_with_index do |window, i| unless window.echo_area? if i < @@list.size - 2 window.resize(window.lines, Window.columns) else window.resize(Window.lines - 1 - window.y, Window.columns) end end end @@echo_area.move(Window.lines - 1, 0) @@echo_area.resize(1, Window.columns) end |
.set_default_colors(fg, bg) ⇒ Object
107 108 109 110 |
# File 'lib/textbringer/window.rb', line 107 def self.set_default_colors(fg, bg) Curses.assume_default_colors(Color[fg], Color[bg]) Window.redraw end |
.start ⇒ Object
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 |
# File 'lib/textbringer/window.rb', line 117 def self.start if @@started raise EditorError, "Already started" end Curses.init_screen Curses.noecho Curses.raw Curses.nonl self.has_colors = Curses.has_colors? if has_colors? Curses.start_color Curses.use_default_colors load_faces end begin window = Textbringer::Window.new(Window.lines - 1, Window.columns, 0, 0) window.buffer = Buffer.new_buffer("*scratch*") @@list.push(window) Window.current = window @@echo_area = Textbringer::EchoArea.new(1, Window.columns, Window.lines - 1, 0) Buffer.minibuffer.keymap = MINIBUFFER_LOCAL_MAP @@echo_area.buffer = Buffer.minibuffer @@list.push(@@echo_area) @@started = true yield ensure @@list.each do |win| win.close end @@list.clear Curses.echo Curses.noraw Curses.nl Curses.close_screen @@started = false end end |
.update ⇒ Object
175 176 177 |
# File 'lib/textbringer/window.rb', line 175 def self.update Curses.doupdate end |
Instance Method Details
#active? ⇒ Boolean
239 240 241 |
# File 'lib/textbringer/window.rb', line 239 def active? true end |
#close ⇒ Object
258 259 260 |
# File 'lib/textbringer/window.rb', line 258 def close @window.close end |
#current? ⇒ Boolean
288 289 290 |
# File 'lib/textbringer/window.rb', line 288 def current? self == @@current end |
#delete ⇒ Object
247 248 249 250 251 252 253 254 255 256 |
# File 'lib/textbringer/window.rb', line 247 def delete unless @deleted if current? Window.current = @@list.first end delete_marks @window.close @deleted = true end end |
#deleted? ⇒ Boolean
243 244 245 |
# File 'lib/textbringer/window.rb', line 243 def deleted? @deleted end |
#echo_area? ⇒ Boolean
235 236 237 |
# File 'lib/textbringer/window.rb', line 235 def echo_area? false end |
#enlarge(n) ⇒ Object
554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 |
# File 'lib/textbringer/window.rb', line 554 def enlarge(n) if n > 0 max_height = Window.lines - CONFIG[:window_min_height] * (@@list.size - 2) - 1 new_lines = [lines + n, max_height].min needed_lines = new_lines - lines resize(new_lines, columns) i = @@list.index(self) indices = (i + 1).upto(@@list.size - 2).to_a + (i - 1).downto(0).to_a indices.each do |j| break if needed_lines == 0 window = @@list[j] extended_lines = [ window.lines - CONFIG[:window_min_height], needed_lines ].min window.resize(window.lines - extended_lines, window.columns) needed_lines -= extended_lines end y = 0 @@list.each do |win| win.move(y, win.x) y += win.lines end elsif n < 0 && @@list.size > 2 new_lines = [lines + n, CONFIG[:window_min_height]].max diff = lines - new_lines resize(new_lines, columns) i = @@list.index(self) if i < @@list.size - 2 window = @@list[i + 1] window.move(window.y - diff, window.x) else window = @@list[i - 1] move(self.y + diff, self.x) end window.resize(window.lines + diff, window.columns) end end |
#has_input? ⇒ Boolean
335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 |
# File 'lib/textbringer/window.rb', line 335 def has_input? if !@raw_key_buffer.empty? || !@key_buffer.empty? return true end @window.nodelay = true begin c = @window.get_char if c @raw_key_buffer.push(c) end !c.nil? ensure @window.nodelay = false end end |
#highlight ⇒ Object
351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 |
# File 'lib/textbringer/window.rb', line 351 def highlight @highlight_on = {} @highlight_off = {} return if !@@has_colors || !CONFIG[:syntax_highlight] || @buffer.binary? syntax_table = @buffer.mode.syntax_table || DEFAULT_SYNTAX_TABLE if @buffer.bytesize < CONFIG[:highlight_buffer_size_limit] base_pos = @buffer.point_min s = @buffer.to_s else base_pos = @buffer.point len = columns * (lines - 1) / 2 * 3 s = @buffer.substring(@buffer.point, @buffer.point + len).scrub("") end return if !s.valid_encoding? re_str = syntax_table.map { |name, re| "(?<#{name}>#{re})" }.join("|") re = Regexp.new(re_str) names = syntax_table.keys s.scan(re) do b = base_pos + $`.bytesize e = b + $&.bytesize if b < @buffer.point && @buffer.point < e b = @buffer.point end name = names.find { |n| $~[n] } attributes = Face[name]&.attributes if attributes @highlight_on[b] = attributes @highlight_off[e] = attributes end end end |
#move(y, x) ⇒ Object
486 487 488 489 490 491 |
# File 'lib/textbringer/window.rb', line 486 def move(y, x) @y = y @x = x @window.move(y, x) @mode_line.move(y + @window.maxy, x) end |
#read_event ⇒ Object
292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 |
# File 'lib/textbringer/window.rb', line 292 def read_event key = get_char if key.is_a?(Integer) if HAVE_GET_KEY_MODIFIERS if Curses::ALT_0 <= key && key <= Curses::ALT_9 @key_buffer.push((key - ALT_NUMBER_BASE).chr) return "\e" elsif Curses::ALT_A <= key && key <= Curses::ALT_Z @key_buffer.push((key - ALT_ALPHA_BASE).chr) return "\e" end end KEY_NAMES[key] || key else key&.encode(Encoding::UTF_8) end end |
#read_event_nonblock ⇒ Object
310 311 312 313 314 315 316 317 |
# File 'lib/textbringer/window.rb', line 310 def read_event_nonblock @window.nodelay = true begin read_event ensure @window.nodelay = false end end |
#recenter ⇒ Object
501 502 503 504 505 506 507 508 509 510 511 512 |
# File 'lib/textbringer/window.rb', line 501 def recenter @buffer.save_point do |saved| max = (lines - 1) / 2 count = beginning_of_line_and_count(max) while count < max break if @buffer.point == 0 @buffer.backward_char count += beginning_of_line_and_count(max - count - 1) + 1 end @buffer.mark_to_point(@top_of_window) end end |
#recenter_if_needed ⇒ Object
514 515 516 517 518 519 |
# File 'lib/textbringer/window.rb', line 514 def recenter_if_needed if @buffer.point_before_mark?(@top_of_window) || @buffer.point_after_mark?(@bottom_of_window) recenter end end |
#redisplay ⇒ Object
385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 |
# File 'lib/textbringer/window.rb', line 385 def redisplay return if @buffer.nil? redisplay_mode_line @buffer.save_point do |saved| if current? point = saved else point = @point_mark @buffer.point_to_mark(@point_mark) end framer y = x = 0 @buffer.point_to_mark(@top_of_window) highlight @window.erase @window.setpos(0, 0) @window.attrset(0) if current? && @buffer.visible_mark && @buffer.point_after_mark?(@buffer.visible_mark) @window.attron(Curses::A_REVERSE) end while !@buffer.end_of_buffer? cury, curx = @window.cury, @window.curx if @buffer.point_at_mark?(point) y, x = cury, curx if current? && @buffer.visible_mark if @buffer.point_after_mark?(@buffer.visible_mark) @window.attroff(Curses::A_REVERSE) elsif @buffer.point_before_mark?(@buffer.visible_mark) @window.attron(Curses::A_REVERSE) end end end if current? && @buffer.visible_mark && @buffer.point_at_mark?(@buffer.visible_mark) if @buffer.point_after_mark?(point) @window.attroff(Curses::A_REVERSE) elsif @buffer.point_before_mark?(point) @window.attron(Curses::A_REVERSE) end end if attrs = @highlight_off[@buffer.point] @window.attroff(attrs) end if attrs = @highlight_on[@buffer.point] @window.attron(attrs) end c = @buffer.char_after if c == "\n" @window.clrtoeol break if cury == lines - 2 # lines include mode line @window.setpos(cury + 1, 0) @buffer.forward_char next elsif c == "\t" n = calc_tab_width(curx) c = " " * n else c = escape(c) end if curx < columns - 4 newx = nil else newx = curx + Buffer.display_width(c) if newx > columns if cury == lines - 2 break else @window.clrtoeol @window.setpos(cury + 1, 0) end end end @window.addstr(c) break if newx == columns && cury == lines - 2 @buffer.forward_char end if current? && @buffer.visible_mark @window.attroff(Curses::A_REVERSE) end @buffer.mark_to_point(@bottom_of_window) if @buffer.point_at_mark?(point) y, x = @window.cury, @window.curx end if x == columns - 1 c = @buffer.char_after(point.location) if c && Buffer.display_width(c) > 1 y += 1 x = 0 end end @window.setpos(y, x) @window.noutrefresh end end |
#redraw ⇒ Object
481 482 483 484 |
# File 'lib/textbringer/window.rb', line 481 def redraw @window.redraw @mode_line.redraw end |
#resize(lines, columns) ⇒ Object
493 494 495 496 497 498 499 |
# File 'lib/textbringer/window.rb', line 493 def resize(lines, columns) @lines = lines @columns = columns @window.resize(lines - 1, columns) @mode_line.move(@y + lines - 1, @x) @mode_line.resize(1, columns) end |
#restore_point ⇒ Object
284 285 286 |
# File 'lib/textbringer/window.rb', line 284 def restore_point @buffer.point_to_mark(@point_mark) end |
#save_point ⇒ Object
276 277 278 279 280 281 282 |
# File 'lib/textbringer/window.rb', line 276 def save_point @buffer[:top_of_window] ||= @buffer.new_mark @buffer[:top_of_window].location = @top_of_window.location @buffer[:bottom_of_window] ||= @buffer.new_mark @buffer[:bottom_of_window].location = @bottom_of_window.location @buffer.mark_to_point(@point_mark) end |
#scroll_down ⇒ Object
531 532 533 534 535 536 537 538 539 |
# File 'lib/textbringer/window.rb', line 531 def scroll_down if @top_of_window.location == @buffer.point_min raise RangeError, "Beginning of buffer" end @buffer.point_to_mark(@top_of_window) @buffer.next_line @buffer.beginning_of_line @top_of_window.location = 0 end |
#scroll_up ⇒ Object
521 522 523 524 525 526 527 528 529 |
# File 'lib/textbringer/window.rb', line 521 def scroll_up if @bottom_of_window.location == @buffer.point_max raise RangeError, "End of buffer" end @buffer.point_to_mark(@bottom_of_window) @buffer.previous_line @buffer.beginning_of_line @buffer.mark_to_point(@top_of_window) end |
#shrink(n) ⇒ Object
595 596 597 |
# File 'lib/textbringer/window.rb', line 595 def shrink(n) enlarge(-n) end |
#shrink_if_larger_than_buffer ⇒ Object
599 600 601 602 603 604 605 606 607 608 609 610 611 612 |
# File 'lib/textbringer/window.rb', line 599 def shrink_if_larger_than_buffer @buffer.save_point do @buffer.end_of_buffer @buffer.skip_re_backward(/\s/) count = beginning_of_line_and_count(Window.lines) + 1 while !@buffer.beginning_of_buffer? @buffer.backward_char count += beginning_of_line_and_count(Window.lines) + 1 end if lines - 1 > count shrink(lines - 1 - count) end end end |
#split ⇒ Object
541 542 543 544 545 546 547 548 549 550 551 552 |
# File 'lib/textbringer/window.rb', line 541 def split old_lines = lines new_lines = (old_lines / 2.0).ceil if new_lines < CONFIG[:window_min_height] raise EditorError, "Window too small" end resize(new_lines, columns) new_window = Window.new(old_lines - new_lines, columns, y + new_lines, x) new_window.buffer = buffer i = @@list.index(self) @@list.insert(i + 1, new_window) end |
#wait_input(msecs) ⇒ Object
319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 |
# File 'lib/textbringer/window.rb', line 319 def wait_input(msecs) if !@raw_key_buffer.empty? || !@key_buffer.empty? return @raw_key_buffer.first || @key_buffer.first end @window.timeout = msecs begin c = @window.get_char if c @raw_key_buffer.push(c) end c ensure @window.timeout = -1 end end |