opf/openproject

View on GitHub
lib/redmine/menu_manager/menu_item.rb

Summary

Maintainability
A
2 hrs
Test Coverage
#-- copyright
# OpenProject is an open source project management software.
# Copyright (C) 2012-2024 the OpenProject GmbH
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
# Copyright (C) 2006-2013 Jean-Philippe Lang
# Copyright (C) 2010-2013 the ChiliProject Team
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program 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 General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
#
# See COPYRIGHT and LICENSE files for more details.
#++

class Redmine::MenuManager::MenuItem < Redmine::MenuManager::TreeNode
  include Redmine::I18n
  attr_reader :name,
              :param,
              :icon_after,
              :context,
              :condition,
              :parent,
              :child_menus,
              :last,
              :partial,
              :engine,
              :enterprise_feature

  def initialize(name, url, options)
    raise ArgumentError, "Invalid option :if for menu item '#{name}'" if options[:if] && !options[:if].respond_to?(:call)
    raise ArgumentError, "Invalid option :html for menu item '#{name}'" if options[:html] && !options[:html].is_a?(Hash)
    raise ArgumentError, "Cannot set the :parent to be the same as this item" if options[:parent] == name.to_sym

    if options[:children] && !options[:children].respond_to?(:call)
      raise ArgumentError,
            "Invalid option :children for menu item '#{name}'"
    end

    @name = name
    @url = url
    @condition = options[:if]
    @param = options[:param] || :project_id
    @icon = options[:icon]
    @icon_after = options[:icon_after]
    @enterprise_feature = options[:enterprise_feature]
    @caption = options[:caption]
    @context = options[:context]
    @html_options = options[:html].nil? ? {} : options[:html].dup
    # Adds a unique class to each menu item based on its name
    @html_options[:class] = [
      @html_options[:class], "#{@name.to_s.dasherize}-menu-item"
    ].compact.join(" ")
    @parent = options[:parent]
    @child_menus = options[:children]
    @last = options[:last] || false
    @partial = options[:partial]
    @badge = options[:badge]
    @engine = options[:engine]
    @allow_deeplink = options[:allow_deeplink]
    @skip_permissions_check = !!options[:skip_permissions_check]
    super(@name.to_sym)
  end

  def caption(project = nil)
    if @caption.is_a?(Proc)
      c = @caption.call(project).to_s
      c = @name.to_s.humanize if c.blank?
      c
    elsif @caption.nil?
      l_or_humanize(name, prefix: "label_")
    else
      @caption.is_a?(Symbol) ? I18n.t(@caption) : @caption
    end
  end

  def caption=(new_caption)
    @caption = new_caption
  end

  def icon(project = nil)
    if @icon.is_a?(Proc)
      @icon.call(project).to_s
    else
      @icon
    end
  end

  def icon=(new_icon)
    @icon = new_icon
  end

  def badge(project = nil)
    if @badge.is_a?(Proc)
      @badge.call(project).to_s
    else
      @badge
    end
  end

  def badge=(new_badge)
    @badge = new_badge
  end

  def url(project = nil)
    if @url.is_a?(Proc)
      @url.call(project)
    else
      @url
    end
  end

  def url=(new_url)
    @url = new_url
  end

  # Allow special case that the user is not allowed to see the parent node but at least one children.
  # In that case, parent and the children are shown.
  # The parent's url is then changed to the children's url.
  def allow_deeplink?
    @allow_deeplink
  end

  def allow_deeplink=(allow_deeplink)
    @allow_deeplink = allow_deeplink
  end

  def skip_permissions_check?
    @skip_permissions_check
  end

  def html_options(options = {})
    if options[:selected]
      o = @html_options.dup
      o[:class] += " selected"
      o
    else
      @html_options
    end
  end

  def add_condition(new_condition)
    raise ArgumentError, "Condition needs to be callable" unless new_condition.respond_to?(:call)

    old_condition = @condition
    @condition = if old_condition.respond_to?(:call)
                   ->(project) { old_condition.call(project) && new_condition.call(project) }
                 else
                   ->(project) { new_condition.call(project) }
                 end
  end
end