require 'erb' require 'rabbit/gtk' require 'rabbit/dependency-canvas' require 'rabbit/renderer/display/drawing-area-view-only' require 'rabbit/renderer/display/hook-handler' require 'rabbit/renderer/display/key-handler' require 'rabbit/renderer/display/button-handler' require 'rabbit/renderer/display/scroll-handler' require 'rabbit/renderer/display/menu' module Rabbit class InfoWindow include ERB::Util include GetText include Renderer::Display::HookHandler include Renderer::Display::KeyHandler include Renderer::Display::ButtonHandler include Renderer::Display::ScrollHandler include Renderer::Display::Menu def initialize(canvas) @canvas = canvas @window = nil @timer_id = nil @note_area = nil init_hook_handler init_key_handler init_button_handler init_scroll_handler end def show(width=nil, height=nil) init_gui(width, height) @window.show_all update_source toggle_index_mode if @canvas.index_mode? adjust_slide end def hide return unless showing? detach_menu(@window) detach_key(@window) each do |canvas| canvas.detach end @window.signal_handler_disconnect(@window_destroy_id) @window.destroy @window = @window_destroy_id = nil @canvas_widgets = @outer_box = nil GLib::Source.remove(@timer_id) if @timer_id @timer_id = nil @previous_canvas = @current_canvas = @next_canvas = nil end def showing? !@window.nil? end def moved(index) return unless showing? update(index) end def parsed return unless showing? update_source update end def index_mode_on return unless showing? toggle_index_mode end def index_mode_off return unless showing? toggle_index_mode end private def init_gui(width, height) init_canvas init_window(width, height) end def init_canvas @current_canvas = make_canvas @previous_canvas = make_canvas @next_canvas = make_canvas end def make_canvas DependencyCanvas.new(@canvas, @canvas.logger, Renderer::Display::DrawingAreaViewOnly) end def init_window(width, height) @window = Gtk::Window.new @window_destroy_id = @window.signal_connect("destroy") do @canvas.activate("ToggleInfoWindow") end @window.title = _("%s: Information window") % @canvas.title @window.set_default_size(width, height) if width and height if on_note_mode? init_widgets_on_note_mode(width, height) else init_widgets(width, height) end init_menu attach_key(@window) attach_menu(@window) event_mask = Gdk::EventMask::BUTTON_PRESS_MASK event_mask |= Gdk::EventMask::BUTTON_RELEASE_MASK event_mask |= Gdk::EventMask::BUTTON1_MOTION_MASK event_mask |= Gdk::EventMask::BUTTON2_MOTION_MASK event_mask |= Gdk::EventMask::BUTTON3_MOTION_MASK @window.add_events(event_mask) set_button_event(@window) set_scroll_event(@window) @window.add(@outer_box) end def init_widgets(width, height) init_timer_label(width * (1.0 / 3.0), height * (1.0 / 3.0)) @outer_box = Gtk::Box.new(:vertical) current_box = Gtk::Box.new(:horizontal) @current_canvas.attach_to(nil, @window, current_box) do |container, widget| widget.set_size_request(width * (2.0 / 3.0), height * (2.0 / 3.0)) container.pack_start(widget, :expand => true, :fill => false) end @outer_box.pack_start(current_box, :expand => true, :fill => false) bottom_box = Gtk::Box.new(:horizontal) @previous_canvas.attach_to(nil, @window, bottom_box) do |container, widget| widget.set_size_request(width * (1.0 / 3.0), height * (1.0 / 3.0)) container.pack_start(widget, :expand => false, :fill => false) end bottom_box.pack_start(@timer_label, :expand => true, :fill => false) @next_canvas.attach_to(nil, @window, bottom_box) do |container, widget| widget.set_size_request(width * (1.0 / 3.0), height * (1.0 / 3.0)) container.pack_end(widget, :expand => false, :fill => false) end @outer_box.pack_end(bottom_box, :expand => false, :fill => false) @outer_box.show end def init_widgets_on_note_mode(width, height) init_timer_label(width * (1.0 / 5.0), height * (2.0 / 5.0)) init_note_area @outer_box = Gtk::Box.new(:vertical) current_box = Gtk::Box.new(:horizontal) current_box.pack_start(@timer_label, :expand => false, :fill => false) @previous_canvas.attach_to(nil, @window, current_box) do |container, widget| widget.set_size_request(width * (1.0 / 5.0), height * (2.0 / 5.0)) container.pack_start(widget, :expand => true, :fill => true, :padding => 10) end @current_canvas.attach_to(nil, @window, current_box) do |container, widget| widget.set_size_request(width * (2.0 / 5.0), height * (2.0 / 5.0)) container.pack_start(widget, :expand => true, :fill => true) end @next_canvas.attach_to(nil, @window, current_box) do |container, widget| widget.set_size_request(width * (1.0 / 5.0), height * (2.0 / 5.0)) container.pack_end(widget, :expand => true, :fill => true, :padding => 10) end @outer_box.pack_start(current_box, :expand => false, :fill => false) bottom_box = Gtk::Box.new(:horizontal) bottom_box.pack_start(@note_area, :expand => true, :fill => true, :padding => 20) @outer_box.pack_start(bottom_box, :expand => true, :fill => true, :padding => 20) @outer_box.show end def init_canvas_widgets @canvas_widgets = Gtk::Box.new(:horizontal) @current_canvas.attach_to(nil, @window, @canvas_widgets) @next_canvas.attach_to(nil, @window, @canvas_widgets) end def init_timer_label(width, height) @timer_label = Gtk::Label.new @timer_label.justify = :center @timer_label.markup = markupped_timer_label(width, height) end def init_note_area @note_area = Gtk::DrawingArea.new if @note_area.class.signals.include?("expose-event") @note_area.signal_connect("expose-event") do |area, event| context = area.window.create_cairo_context draw_text_as_large_as_possible(area, context, note_text) Gdk::Event::PROPAGATE end else @note_area.signal_connect("draw") do |area, context| draw_text_as_large_as_possible(area, context, note_text) Gdk::Event::PROPAGATE end end end def update(index=nil) start_timer if @timer_id.nil? @note_area.queue_draw if @note_area adjust_slide(index) end def note_text note = @canvas.current_slide["note"] return note if note.nil? note.gsub(/\\n/, "\n") end def draw_text_as_large_as_possible(area, context, markupped_text) return if markupped_text.nil? area_width, area_height = area.window.size layout = context.create_pango_layout layout.context.resolution = @canvas.font_resolution attributes, text = Pango.parse_markup(markupped_text) layout.text = text layout.attributes = attributes layout.width = area_width * Pango::SCALE layout.wrap = :word_char set_as_large_as_font_description(layout, area_height) context.update_pango_layout(layout) context.show_pango_layout(layout) end def set_as_large_as_font_description(layout, max_height) family = "Sans" size = 14 last_font_description = nil loop do font_description = Pango::FontDescription.new("#{family} #{size}") layout.font_description = font_description layout_height = layout.pixel_size[1] break if layout_height > max_height last_font_description = font_description size = [size * 1.2, size + 5].min end last_font_description ||= Pango::FontDescription.new("#{family} #{size}") layout.font_description = last_font_description end def start_timer @timer_id = GLib::Timeout.add(1000) do @timer_label.markup = markupped_timer_label if showing? if showing? and @canvas.rest_time GLib::Source::CONTINUE else @timer_id = nil GLib::Source::REMOVE end end end def markupped_timer_label(width=nil, height=nil) width ||= @window.size[0] * (1.0 / 3.0) height ||= @window.size[1] * (1.0 / 3.0) attrs = {} font_size = on_note_mode? ? 100 : 200 attrs["font_desc"] = ((height * font_size) / Pango::SCALE).to_s rest_time = @canvas.rest_time attrs["foreground"] = "red" if rest_time and rest_time < 0 "#{h timer_label}" end def timer_label rest_time = @canvas.rest_time || @canvas.allotted_time if rest_time "%s%02d:%02d" % Utils.split_number_to_minute_and_second(rest_time) else _("unlimited") end end def markupped_note_text(width=nil, height=nil) height ||= @window.size[1] * (3.0 / 5.0) if @canvas.current_slide["note"] text = @canvas.current_slide["note"].gsub(/\\n/, "\n") else text = "" end attrs = {} attrs["font_desc"] = ((height * 40) / Pango::SCALE).to_s "#{text}" end def update_source each do |canvas| source = Source::Memory.new("UTF-8", @canvas.logger) @canvas.source_force_modified(true) do |original_source| source.source = original_source.read source.base = original_source.base end canvas.parse(source) end end def adjust_slide(base_index=nil) base_index ||= @canvas.current_index @previous_canvas.move_to_if_can([base_index - 1, 0].max) @current_canvas.move_to_if_can(base_index) @next_canvas.move_to_if_can([base_index + 1, @canvas.slide_size - 1].min) end def toggle_index_mode each do |canvas| canvas.toggle_index_mode end end def each(&block) [@previous_canvas, @current_canvas, @next_canvas].each(&block) end def on_note_mode? @canvas.slides.any? {|slide| slide["note"]} end end end