mirror of
https://github.com/opf/openproject.git
synced 2026-06-14 03:30:14 +00:00
137 lines
4.4 KiB
Ruby
137 lines
4.4 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
#-- copyright
|
|
# OpenProject is an open source project management software.
|
|
# Copyright (C) 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.
|
|
#++
|
|
|
|
module Projects
|
|
class UpdateService < ::BaseServices::Update
|
|
prepend Projects::Concerns::UpdateDemoData
|
|
include Projects::Concerns::ManageMembershipsFromCustomFields
|
|
|
|
private
|
|
|
|
attr_accessor :memoized_changes
|
|
|
|
def set_attributes(params)
|
|
ret = super
|
|
|
|
# Because awesome_nested_set reloads the model after saving, we cannot rely
|
|
# on saved_changes.
|
|
self.memoized_changes = model.changes
|
|
|
|
ret
|
|
end
|
|
|
|
def before_perform(service_call)
|
|
super.tap do
|
|
update_calculated_value_custom_fields
|
|
end
|
|
end
|
|
|
|
def after_perform(service_call)
|
|
ret = super
|
|
|
|
touch_on_custom_values_update
|
|
update_semantic_ids_on_identifier_change if Setting::WorkPackageIdentifier.semantic?
|
|
notify_on_identifier_renamed
|
|
send_update_notification
|
|
update_wp_versions_on_parent_change
|
|
handle_archiving
|
|
|
|
ret
|
|
end
|
|
|
|
def touch_on_custom_values_update
|
|
model.touch if only_custom_values_updated?
|
|
end
|
|
|
|
def update_semantic_ids_on_identifier_change
|
|
return unless memoized_changes["identifier"]
|
|
|
|
old_identifier = memoized_changes["identifier"].first
|
|
model.handle_semantic_rename(old_identifier)
|
|
end
|
|
|
|
def notify_on_identifier_renamed
|
|
return unless memoized_changes["identifier"]
|
|
|
|
OpenProject::Notifications.send(OpenProject::Events::PROJECT_RENAMED, project: model)
|
|
end
|
|
|
|
def send_update_notification
|
|
OpenProject::Notifications.send(OpenProject::Events::PROJECT_UPDATED, project: model)
|
|
end
|
|
|
|
def only_custom_values_updated?
|
|
!model.saved_changes? && model.custom_values.any?(&:saved_changes?)
|
|
end
|
|
|
|
def update_wp_versions_on_parent_change
|
|
return unless memoized_changes["parent_id"]
|
|
|
|
WorkPackage.update_versions_from_hierarchy_change(model)
|
|
end
|
|
|
|
def update_calculated_value_custom_fields
|
|
changed_cf_ids = model.custom_values.select(&:changed?).map(&:custom_field_id)
|
|
|
|
# Using unscope(where: :admin_only) to fix an issue when non admin user
|
|
# edits a custom field which is used by an admin only calculated value
|
|
# field. Without this unscoping, admin only value and all fields
|
|
# referencing it (recursively) will not be recalculated and there will
|
|
# even be no place for that recalculatin to be triggered unless an admin
|
|
# edits same value again.
|
|
#
|
|
# This may need to be handled differently to make it work for other custom
|
|
# field containers, like WorkPackage. User custom fields also has
|
|
# admin_only check.
|
|
affected_cfs = model.available_custom_fields.unscope(where: :admin_only).affected_calculated_fields(changed_cf_ids)
|
|
|
|
model.calculate_custom_fields(affected_cfs)
|
|
end
|
|
|
|
def handle_archiving
|
|
return unless model.saved_change_to_active?
|
|
|
|
service_class =
|
|
if model.active?
|
|
# was unarchived
|
|
Projects::UnarchiveService
|
|
else
|
|
# was archived
|
|
Projects::ArchiveService
|
|
end
|
|
|
|
# EmptyContract is used because archive/unarchive conditions have
|
|
# already been checked in Projects::UpdateContract
|
|
service = service_class.new(user:, model:, contract_class: EmptyContract)
|
|
service.call
|
|
end
|
|
end
|
|
end
|