2021-04-18 14:34:55 +03:00
# frozen_string_literal: true
2020-07-27 11:49:23 +03:00
class DashboardContent
include Redmine :: I18n
attr_accessor :user , :project
2020-08-04 20:09:18 +03:00
MAX_MULTIPLE_OCCURS = 8
2020-07-27 11:49:23 +03:00
DEFAULT_MAX_ENTRIES = 10
RENDER_ASYNC_CACHE_EXPIRES_IN = 30
class << self
def types
descendants . map { | dc | dc :: TYPE_NAME }
end
end
def with_chartjs?
false
end
def initialize ( attr = { } )
self . user = attr [ :user ] . presence || User . current
self . project = attr [ :project ] . presence
end
def groups
%w[ top left right bottom ]
end
def block_definitions
{
'issuequery' = > { label : l ( :label_query_with_name , l ( :label_issue_plural ) ) ,
permission : :view_issues ,
query_block : {
label : l ( :label_issue_plural ) ,
list_partial : 'issues/list' ,
class : IssueQuery ,
link_helper : '_project_issues_path' ,
count_method : 'issue_count' ,
entries_method : 'issues' ,
entities_var : :issues ,
with_project : true
} ,
2022-04-19 13:51:37 +03:00
max_occurs : MAX_MULTIPLE_OCCURS } ,
2020-11-06 12:12:25 +03:00
'text' = > { label : l ( :label_text_sync ) ,
2020-07-27 11:49:23 +03:00
max_occurs : MAX_MULTIPLE_OCCURS ,
partial : 'dashboards/blocks/text' } ,
2020-11-06 12:12:25 +03:00
'text_async' = > { label : l ( :label_text_async ) ,
max_occurs : MAX_MULTIPLE_OCCURS ,
async : { required_settings : % i [ text ] ,
partial : 'dashboards/blocks/text_async' } } ,
2020-07-27 11:49:23 +03:00
'news' = > { label : l ( :label_news_latest ) ,
2021-09-20 13:44:10 +03:00
permission : :view_news ,
if : proc { Redmine :: AccessControl . active_module? ( :news ) } } ,
2020-07-27 11:49:23 +03:00
'documents' = > { label : l ( :label_document_plural ) ,
2021-09-20 13:44:10 +03:00
permission : :view_documents ,
if : proc { Redmine :: AccessControl . active_module? ( :documents ) } } ,
2020-08-10 19:28:52 +03:00
'my_spent_time' = > { label : l ( :label_my_spent_time ) ,
2020-08-10 19:49:48 +03:00
permission : :log_time } ,
2020-07-27 11:49:23 +03:00
'feed' = > { label : l ( :label_additionals_feed ) ,
2021-12-17 09:16:45 +03:00
max_occurs : 20 ,
2020-07-27 11:49:23 +03:00
async : { required_settings : % i [ url ] ,
2020-08-10 17:19:03 +03:00
cache_expires_in : 600 ,
2020-07-27 11:49:23 +03:00
skip_user_id : true ,
partial : 'dashboards/blocks/feed' } }
}
end
# Returns the available blocks
def available_blocks
return @available_blocks if defined? @available_blocks
2021-03-24 18:20:33 +03:00
available_blocks = block_definitions . reject do | _block_name , block_specs |
2022-04-19 13:51:37 +03:00
block_specs . key? ( :permission ) && ! block_permission_allowed? ( block_specs [ :permission ] ) ||
2021-03-24 18:20:33 +03:00
block_specs . key? ( :admin_only ) && block_specs [ :admin_only ] && ! user . admin? ||
block_specs . key? ( :if ) && ! block_specs [ :if ] . call ( project )
2020-07-27 11:49:23 +03:00
end
@available_blocks = available_blocks . sort_by { | _k , v | v [ :label ] } . to_h
end
def block_options ( blocks_in_use = [ ] )
options = [ ]
available_blocks . each do | block , block_options |
indexes = blocks_in_use . map do | n |
Regexp . last_match ( 2 ) . to_i if n =~ / \ A #{ block } (__( \ d+))? \ z /
end
indexes . compact!
occurs = indexes . size
block_id = indexes . any? ? " #{ block } __ #{ indexes . max + 1 } " : block
disabled = ( occurs > = ( available_blocks [ block ] [ :max_occurs ] || 1 ) )
block_id = nil if disabled
options << [ block_options [ :label ] , block_id ]
end
options
end
def valid_block? ( block , blocks_in_use = [ ] )
block . present? && block_options ( blocks_in_use ) . map ( & :last ) . include? ( block )
end
def find_block ( block )
block . to_s =~ / \ A(.*?)(__ \ d+)? \ z /
2021-04-18 14:34:55 +03:00
name = Regexp . last_match 1
2020-07-27 11:49:23 +03:00
available_blocks . key? ( name ) ? available_blocks [ name ] . merge ( name : name ) : nil
end
# Returns the default layout for a new dashboard
def default_layout
{
'left' = > [ 'legacy_left' ] ,
'right' = > [ 'legacy_right' ]
}
end
2022-04-19 13:51:37 +03:00
private
2023-01-04 19:11:45 +03:00
def issues_cache_expires_in
2023-01-04 19:29:10 +03:00
open_issue_count = Rails . cache . fetch issues_cache_key , expires_in : 1 . hour do
query = if project
IssueQuery . new project : project , name : '_'
else
IssueQuery . new name : '_'
end
query . issue_count
2023-01-04 19:11:45 +03:00
end
2023-01-04 19:29:10 +03:00
if open_issue_count > 1_000
2023-01-04 19:11:45 +03:00
86_400 # 1 day
2023-01-04 19:29:10 +03:00
elsif open_issue_count > 500
14_400 # 4 hour
elsif open_issue_count > 200
2023-01-04 19:11:45 +03:00
3_600 # 1 hour
2023-01-04 19:29:10 +03:00
elsif open_issue_count > 100
1_800 # 30 minutes
2023-01-04 19:11:45 +03:00
else
60
end
end
def issues_cache_key
Digest :: SHA256 . hexdigest [ project & . id , User . current . id ] . compact . join ( '-' )
end
2022-04-19 13:51:37 +03:00
# if more the one permission is specified, all permissions are required
def block_permission_allowed? ( permission )
2023-06-04 10:26:19 +03:00
Array ( permission ) . all? { | p | user . allowed_to? p , project , global : true }
2022-04-19 13:51:37 +03:00
end
2020-07-27 11:49:23 +03:00
end