. /** * @package MantisBT * @copyright Copyright (C) 2000 - 2002 Kenzaburo Ito - kenito@300baud.org * @copyright Copyright (C) 2002 - 2014 MantisBT Team - mantisbt-dev@lists.sourceforge.net * @link http://www.mantisbt.org */ /** * MantisBT Core API's */ require_once( 'core.php' ); require_once( 'email_api.php' ); auth_reauthenticate(); html_page_top( lang_get( 'manage_workflow_config' ) ); print_manage_menu( 'adm_permissions_report.php' ); print_manage_config_menu( 'manage_config_workflow_page.php' ); $t_access = current_user_get_access_level(); $t_project = helper_get_current_project(); $t_can_change_workflow = $t_access >= config_get_access( 'status_enum_workflow' ); $t_can_change_flags = $t_can_change_workflow; $t_overrides = array(); /** * Returns a string to define the background color attribute depending * on the level where it's overridden * @param int $p_level_file config file's access level * @param int $p_level_global all projects' access level * @param int $p_level_project current project's access level * @return string bgcolor attribute, or '' if no color */ function set_colour_override( $p_level_file, $p_level_global, $p_level_project ) { global $t_colour_global, $t_colour_project; if ( $p_level_project != $p_level_global ) { $t_colour = $t_colour_project; } else if( $p_level_global != $p_level_file ) { $t_colour = $t_colour_global; } else { return ''; } return ' bgcolor="' . $t_colour . '" '; } function set_overrides( $p_config ) { global $t_overrides; if( !in_array( $p_config, $t_overrides ) ) { $t_overrides[] = $p_config; } } function parse_workflow( $p_enum_workflow ) { $t_status_arr = MantisEnum::getAssocArrayIndexedByValues( config_get( 'status_enum_string' ) ); if ( 0 == count( $p_enum_workflow ) ) { # workflow is not set, default it to all transitions foreach ( $t_status_arr as $t_status => $t_label ) { $t_temp_workflow = array(); foreach ( $t_status_arr as $t_next => $t_next_label ) { if ( $t_status <> $t_next ) { $t_temp_workflow[] = $t_next . ':' . $t_next_label; } } $p_enum_workflow[$t_status] = implode( ',', $t_temp_workflow ); } } $t_entry = array(); $t_exit = array(); # prepopulate new bug state (bugs go from nothing to here) $t_submit_status_array = config_get( 'bug_submit_status' ); $t_new_label = MantisEnum::getLabel( lang_get( 'status_enum_string' ), config_get( 'bug_submit_status' ) ); if ( is_array( $t_submit_status_array ) ) { # @@@ (thraxisp) this is not implemented in bug_api.php foreach ($t_submit_status_array as $t_access => $t_status ) { $t_entry[$t_status][0] = $t_new_label; $t_exit[0][$t_status] = $t_new_label; } } else { $t_status = $t_submit_status_array; $t_entry[$t_status][0] = $t_new_label; $t_exit[0][$t_status] = $t_new_label; } # add user defined arcs $t_default = array(); foreach ( $t_status_arr as $t_status => $t_status_label ) { if ( isset( $p_enum_workflow[$t_status] ) ) { $t_next_arr = MantisEnum::getAssocArrayIndexedByValues( $p_enum_workflow[$t_status] ); foreach ( $t_next_arr as $t_next => $t_next_label) { if ( !isset( $t_default[$t_status] ) ) { $t_default[$t_status] = $t_next; } $t_exit[$t_status][$t_next] = ''; $t_entry[$t_next][$t_status] = ''; } } else { $t_exit[$t_status] = array(); } if ( !isset( $t_entry[$t_status] ) ) { $t_entry[$t_status] = array(); } } return array( 'entry' => $t_entry, 'exit' => $t_exit, 'default' => $t_default ); } # Get the value associated with the specific action and flag. function show_flag( $p_from_status_id, $p_to_status_id ) { global $t_can_change_workflow, $t_overrides, $t_file_workflow, $t_global_workflow, $t_project_workflow, $t_resolved_status, $t_reopen_status, $t_reopen_label; if ( $p_from_status_id <> $p_to_status_id ) { $t_file = isset( $t_file_workflow['exit'][$p_from_status_id][$p_to_status_id] ) ? 1 : 0 ; $t_global = isset( $t_global_workflow['exit'][$p_from_status_id][$p_to_status_id] ) ? 1 : 0 ; $t_project = isset( $t_project_workflow['exit'][$p_from_status_id][$p_to_status_id] ) ? 1 : 0; $t_colour = set_colour_override( $t_file, $t_global, $t_project ); if ( $t_can_change_workflow && $t_colour != '' ) { set_overrides( 'status_enum_workflow' ); } $t_value = ''; $t_flag = ( 1 == $t_project ); if ( $t_can_change_workflow ) { $t_flag_name = $p_from_status_id . ':' . $p_to_status_id; $t_set = $t_flag ? "checked=\"checked\"" : ""; $t_value .= ""; } else { $t_value .= $t_flag ? 'X' : ' '; } # Add 'reopened' label if ( $p_from_status_id >= $t_resolved_status && $p_to_status_id == $t_reopen_status ) { $t_value .= "
($t_reopen_label)"; } } else { $t_value = ' '; } $t_value .= ''; return $t_value; } function section_begin( $p_section_name ) { $t_enum_statuses = MantisEnum::getValues( config_get( 'status_enum_string' ) ); echo ''; echo '' . "\n"; echo ''; echo ''; echo "\n"; foreach( $t_enum_statuses as $t_status ) { echo ''; } echo ''; echo '' . "\n"; } function capability_row( $p_from_status ) { global $t_file_workflow, $t_global_workflow, $t_project_workflow, $t_can_change_workflow; $t_enum_status = MantisEnum::getAssocArrayIndexedByValues( config_get( 'status_enum_string' ) ); echo ''; foreach ( $t_enum_status as $t_to_status_id => $t_to_status_label ) { echo show_flag( $p_from_status, $t_to_status_id ); } $t_file = isset( $t_file_workflow['default'][$p_from_status] ) ? $t_file_workflow['default'][$p_from_status] : 0 ; $t_global = isset( $t_global_workflow['default'][$p_from_status] ) ? $t_global_workflow['default'][$p_from_status] : 0 ; $t_project = isset( $t_project_workflow['default'][$p_from_status] ) ? $t_project_workflow['default'][$p_from_status] : 0; $t_colour = set_colour_override( $t_file, $t_global, $t_project ); if ( $t_can_change_workflow && $t_colour != '' ) { set_overrides( 'status_enum_workflow' ); } echo ''; echo '' . "\n"; } function section_end() { echo '
' . $p_section_name . '
' . lang_get( 'current_status' ) . '' . lang_get( 'next_status' ) . '
 ' . string_no_break( MantisEnum::getLabel( lang_get( 'status_enum_string' ), $t_status ) ) . ' ' . lang_get( 'custom_field_default_value' ) . '
' . string_no_break( MantisEnum::getLabel( lang_get( 'status_enum_string' ), $p_from_status ) ) . ''; if ( $t_can_change_workflow ) { echo ''; } else { echo MantisEnum::getLabel( lang_get( 'status_enum_string' ), $t_project ); } echo '

' . "\n"; } function threshold_begin( $p_section_name ) { echo ''; echo '' . "\n"; echo ''; echo ''; echo ''; echo "\n"; } function threshold_row( $p_threshold ) { global $t_access, $t_can_change_flags; $t_file = config_get_global( $p_threshold ); $t_global = config_get( $p_threshold, null, null, ALL_PROJECTS ); $t_project = config_get( $p_threshold ); $t_can_change_threshold = $t_access >= config_get_access( $p_threshold ); $t_colour = set_colour_override( $t_file, $t_global, $t_project ); if ( $t_can_change_threshold && $t_colour != '' ) { set_overrides( $p_threshold ); } echo ''; if ( $t_can_change_threshold ) { echo ' '; echo ''; $t_can_change_flags = true; } else { echo '' . MantisEnum::getLabel( lang_get( 'status_enum_string' ), $t_project ) . ' '; echo ''; } echo '' . "\n"; } function threshold_end() { echo '
' . $p_section_name . '
' . lang_get( 'threshold' ) . '' . lang_get( 'status_level' ) . '' . lang_get( 'alter_level' ) . '
' . lang_get( 'desc_' . $p_threshold ) . ' ' . MantisEnum::getLabel( lang_get( 'access_levels_enum_string' ), config_get_access( $p_threshold ) ) . ' 

' . "\n"; } function access_begin( $p_section_name ) { echo ''; echo '' . "\n"; echo ''; } function access_row() { global $t_access, $t_can_change_flags; $t_enum_status = MantisEnum::getAssocArrayIndexedByValues( config_get( 'status_enum_string' ) ); $t_file_new = config_get_global( 'report_bug_threshold' ); $t_global_new = config_get( 'report_bug_threshold', null, null, ALL_PROJECTS ); $t_project_new = config_get( 'report_bug_threshold' ); $t_file_set = config_get_global( 'set_status_threshold' ); $t_global_set = config_get( 'set_status_threshold', null, null, ALL_PROJECTS ); $t_project_set = config_get( 'set_status_threshold' ); $t_submit_status = config_get( 'bug_submit_status' ); # Print the table rows foreach( $t_enum_status as $t_status => $t_status_label ) { echo ''; if( $t_status == $t_submit_status ) { # 'NEW' status $t_level_project = $t_project_new; $t_can_change = ( $t_access >= config_get_access( 'report_bug_threshold' ) ); $t_colour = set_colour_override( $t_file_new, $t_global_new, $t_project_new ); if( $t_can_change && $t_colour != '' ) { set_overrides( 'report_bug_threshold' ); } } else { # Other statuses # File level: fallback if set_status_threshold is not defined if( isset( $t_file_set[$t_status] ) ) { $t_level_file = $t_file_set[$t_status]; } else { $t_level_file = config_get_global('update_bug_status_threshold'); } $t_level_global = isset( $t_global_set[$t_status] ) ? $t_global_set[$t_status] : $t_level_file; $t_level_project = isset( $t_project_set[$t_status] ) ? $t_project_set[$t_status] : $t_level_global; $t_can_change = ( $t_access >= config_get_access( 'set_status_threshold' ) ); $t_colour = set_colour_override( $t_level_file, $t_level_global, $t_level_project ); if( $t_can_change && $t_colour != '' ) { set_overrides( 'set_status_threshold' ); } } if ( $t_can_change ) { echo ' '; $t_can_change_flags = true; } else { echo ''; } echo '' . "\n"; } } # end function access_row echo '

'; # count arcs in and out of each status $t_enum_status = config_get( 'status_enum_string' ); $t_status_arr = MantisEnum::getAssocArrayIndexedByValues( $t_enum_status ); $t_extra_enum_status = '0:non-existent,' . $t_enum_status; $t_lang_enum_status = '0:' . lang_get( 'non_existent' ) . ',' . lang_get( 'status_enum_string' ); $t_all_status = explode( ',', $t_extra_enum_status); # gather all versions of the workflow $t_file_workflow = parse_workflow( config_get_global( 'status_enum_workflow' ) ); $t_global_workflow = parse_workflow( config_get( 'status_enum_workflow', null, null, ALL_PROJECTS ) ); $t_project_workflow = parse_workflow( config_get( 'status_enum_workflow' ) ); # validate the project workflow $t_validation_result = ''; foreach ( $t_status_arr as $t_status => $t_label ) { if ( isset( $t_project_workflow['exit'][$t_status][$t_status] ) ) { $t_validation_result .= ''; } } # check for entry == 0 without exit == 0, unreachable state foreach ( $t_status_arr as $t_status => $t_status_label) { if ( ( 0 == count( $t_project_workflow['entry'][$t_status] ) ) && ( 0 < count( $t_project_workflow['exit'][$t_status] ) ) ){ $t_validation_result .= ''; } } # check for exit == 0 without entry == 0, unleaveable state foreach ( $t_status_arr as $t_status => $t_status_label ) { if ( ( 0 == count( $t_project_workflow['exit'][$t_status] ) ) && ( 0 < count( $t_project_workflow['entry'][$t_status] ) ) ){ $t_validation_result .= ''; } } # check for exit == 0 and entry == 0, isolated state foreach ( $t_status_arr as $t_status => $t_status_label ) { if ( ( 0 == count( $t_project_workflow['exit'][$t_status] ) ) && ( 0 == count( $t_project_workflow['entry'][$t_status] ) ) ){ $t_validation_result .= ''; } } $t_colour_project = config_get( 'colour_project' ); $t_colour_global = config_get( 'colour_global' ); echo "\n"; echo form_security_field( 'manage_config_workflow_set' ); if ( ALL_PROJECTS == $t_project ) { $t_project_title = lang_get( 'config_all_projects' ); } else { $t_project_title = sprintf( lang_get( 'config_project' ) , string_display( project_get_name( $t_project ) ) ); } echo '

' . $t_project_title . '

' . "\n"; echo '

' . lang_get( 'colour_coding' ) . '
'; if ( ALL_PROJECTS <> $t_project ) { echo '' . lang_get( 'colour_project' ) .'
'; } echo '' . lang_get( 'colour_global' ) . '

'; # show the settings used to derive the table threshold_begin( lang_get( 'workflow_thresholds' ) ); if ( !is_array( config_get( 'bug_submit_status' ) ) ) { threshold_row( 'bug_submit_status' ); } threshold_row( 'bug_resolved_status_threshold' ); threshold_row( 'bug_reopen_status' ); threshold_end(); echo '
'; if ( '' <> $t_validation_result ) { echo '
' . $p_section_name . '
' . lang_get( 'access_change' ) . '
' . string_no_break( MantisEnum::getLabel( lang_get( 'status_enum_string' ), $t_status ) ) . '' . MantisEnum::getLabel( lang_get( 'access_levels_enum_string' ), $t_level_project ) . '
' . MantisEnum::getLabel( $t_lang_enum_status, $t_status ) . '' . lang_get( 'superfluous' ) . '
' . MantisEnum::getLabel( $t_lang_enum_status, $t_status ) . '' . lang_get( 'unreachable' ) . '
' . MantisEnum::getLabel( $t_lang_enum_status, $t_status ) . '' . lang_get( 'no_exit' ) . '
' . MantisEnum::getLabel( $t_lang_enum_status, $t_status ) . '' . lang_get( 'unreachable' ) . '
' . lang_get( 'no_exit' ) . '
'; echo '' . "\n"; echo ''; echo ''; echo "\n"; echo $t_validation_result; echo '
' . lang_get( 'validation' ) . '
' . lang_get( 'status' ) . '' . lang_get( 'comment' ) . '


'; } # Initialization for 'reopened' label handling $t_resolved_status = config_get( 'bug_resolved_status_threshold' ); $t_reopen_status = config_get( 'bug_reopen_status' ); $t_reopen_label = MantisEnum::getLabel( lang_get( 'resolution_enum_string' ), config_get( 'bug_reopen_resolution' ) ); # display the graph as a matrix section_begin( lang_get( 'workflow' ) ); foreach ( $t_status_arr as $t_from_status => $t_from_label) { capability_row( $t_from_status ); } section_end(); if ( $t_can_change_workflow ) { echo '

' . lang_get( 'workflow_change_access' ) . ':'; echo '


'; } # display the access levels required to move an issue access_begin( lang_get( 'access_levels' ) ); access_row(); section_end(); if ( $t_access >= config_get_access( 'set_status_threshold' ) ) { echo '

' . lang_get( 'access_change_access' ) . ':'; echo '


'; } if ( $t_can_change_flags ) { echo "\n"; echo "\n"; if ( 0 < count( $t_overrides ) ) { echo "
\n"; echo form_security_field( 'manage_config_revert' ); echo ""; echo ""; echo ""; echo "\n"; echo "
\n"; } } else { echo "\n"; } html_page_bottom();