# Copyright (c) 2007-2021 Andy Maleh # # Permission is hereby granted, free of charge, to any person obtaining # a copy of this software and associated documentation files (the # "Software"), to deal in the Software without restriction, including # without limitation the rights to use, copy, modify, merge, publish, # distribute, sublicense, and/or sell copies of the Software, and to # permit persons to whom the Software is furnished to do so, subject to # the following conditions: # # The above copyright notice and this permission notice shall be # included in all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. require 'glimmer/swt/widget_proxy' module Glimmer module SWT class TreeProxy < Glimmer::SWT::WidgetProxy include Glimmer attr_reader :tree_editor, :tree_editor_text_proxy attr_accessor :tree_properties def initialize(underscored_widget_name, parent, args) super @tree_editor = TreeEditor.new(swt_widget) @tree_editor.horizontalAlignment = SWTProxy[:left] @tree_editor.grabHorizontal = true @tree_editor.minimumHeight = 20 end # Performs depth first search for tree items matching block condition # If no condition block is passed, returns all tree items # Returns a Java TreeItem array to easily set as selection on org.eclipse.swt.Tree if needed def depth_first_search(&condition) found = [] first_item = DisplayProxy.instance.auto_exec { swt_widget.getItems.first } recursive_depth_first_search(first_item, found, &condition) found.to_java(TreeItem) end # Returns all tree items including descendants def all_tree_items depth_first_search end def edit_in_progress? !!@edit_in_progress end def edit_selected_tree_item(before_write: nil, after_write: nil, after_cancel: nil) edit_tree_item(swt_widget.getSelection.first, before_write: before_write, after_write: after_write, after_cancel: after_cancel) end def edit_tree_item(tree_item, before_write: nil, after_write: nil, after_cancel: nil) return if tree_item.nil? content { @tree_editor_text_proxy = text { focus true text tree_item.getText action_taken = false cancel = lambda { @tree_editor_text_proxy.swt_widget.dispose @tree_editor_text_proxy = nil after_cancel&.call @edit_in_progress = false } action = lambda { |event| begin if !action_taken && !@edit_in_progress action_taken = true @edit_in_progress = true new_text = @tree_editor_text_proxy.swt_widget.getText if new_text == tree_item.getText cancel.call else before_write&.call tree_item.setText(new_text) model = tree_item.getData model.send("#{tree_properties[:text]}=", new_text) # makes tree update itself, so must search for selected tree item again edited_tree_item = depth_first_search { |ti| ti.getData == model }.first swt_widget.showItem(edited_tree_item) @tree_editor_text_proxy.swt_widget.dispose @tree_editor_text_proxy = nil after_write&.call(edited_tree_item) @edit_in_progress = false end end rescue => e Glimmer::Config.logger.error {"Error encountered while editing tree item!\n#{e.full_message}"} end } on_focus_lost(&action) on_key_pressed { |key_event| if key_event.keyCode == swt(:cr) action.call(key_event) elsif key_event.keyCode == swt(:esc) cancel.call end } } @tree_editor_text_proxy.swt_widget.selectAll } @tree_editor.setEditor(@tree_editor_text_proxy.swt_widget, tree_item) end private def recursive_depth_first_search(tree_item, found, &condition) return if tree_item.nil? found << tree_item if condition.nil? || DisplayProxy.instance.auto_exec {condition.call(tree_item)} # return if found.any? && !has_style?(:multi) # TODO inspect if this is a good optimization when needed tree_items = DisplayProxy.instance.auto_exec {tree_item.getItems} tree_items.each do |child_tree_item| recursive_depth_first_search(child_tree_item, found, &condition) end end def property_type_converters super.merge({ selection: lambda do |value| if value.is_a?(Array) depth_first_search {|ti| value.include?(ti.getData) } else depth_first_search {|ti| ti.getData == value} end end, }) end end end end