2024-04-29 00:17:15 +02:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2018-03-06 13:44:29 +01:00
|
|
|
class Champ < ApplicationRecord
|
2023-01-04 13:09:14 +01:00
|
|
|
include ChampConditionalConcern
|
2024-11-26 22:05:26 +01:00
|
|
|
include ChampValidateConcern
|
|
|
|
include ChampRevisionConcern
|
2023-01-04 13:09:14 +01:00
|
|
|
|
2024-09-27 15:44:57 +02:00
|
|
|
self.ignored_columns += [:type_de_champ_id, :parent_id]
|
2024-07-01 15:31:32 +02:00
|
|
|
|
2024-09-20 14:54:26 +02:00
|
|
|
attr_readonly :stable_id
|
|
|
|
|
2022-03-09 10:27:43 +01:00
|
|
|
belongs_to :dossier, inverse_of: false, touch: true, optional: false
|
2022-10-20 10:14:47 +02:00
|
|
|
has_many_attached :piece_justificative_file
|
2016-03-15 17:17:56 +01:00
|
|
|
|
2019-07-11 10:28:44 +02:00
|
|
|
# We declare champ specific relationships (Champs::CarteChamp, Champs::SiretChamp and Champs::RepetitionChamp)
|
2018-11-08 14:36:53 +01:00
|
|
|
# here because otherwise we can't easily use includes in our queries.
|
2021-05-25 16:10:01 +02:00
|
|
|
has_many :geo_areas, -> { order(:created_at) }, dependent: :destroy, inverse_of: :champ
|
2020-07-20 16:06:03 +02:00
|
|
|
belongs_to :etablissement, optional: true, dependent: :destroy
|
2018-11-08 14:36:53 +01:00
|
|
|
|
2022-11-30 10:53:51 +01:00
|
|
|
delegate :procedure, to: :dossier
|
|
|
|
|
2024-07-01 15:31:32 +02:00
|
|
|
def type_de_champ
|
|
|
|
@type_de_champ ||= dossier.revision
|
|
|
|
.types_de_champ
|
|
|
|
.find(-> { raise "Type De Champ #{stable_id} not found in Revision #{dossier.revision_id}" }) { _1.stable_id == stable_id }
|
|
|
|
end
|
|
|
|
|
2020-06-18 13:27:55 +02:00
|
|
|
delegate :libelle,
|
|
|
|
:type_champ,
|
|
|
|
:description,
|
2024-09-13 17:22:08 +02:00
|
|
|
:drop_down_options,
|
2022-08-02 12:57:41 +02:00
|
|
|
:drop_down_other?,
|
2024-09-19 17:11:14 +02:00
|
|
|
:drop_down_options_with_other,
|
2021-10-19 18:47:28 +02:00
|
|
|
:drop_down_secondary_libelle,
|
|
|
|
:drop_down_secondary_description,
|
2022-11-07 10:05:27 +01:00
|
|
|
:collapsible_explanation_enabled?,
|
|
|
|
:collapsible_explanation_text,
|
2023-03-24 11:08:31 +01:00
|
|
|
:header_section_level_value,
|
2023-03-24 10:31:01 +01:00
|
|
|
:current_section_level,
|
2020-06-18 13:27:55 +02:00
|
|
|
:exclude_from_export?,
|
|
|
|
:exclude_from_view?,
|
2022-07-21 17:59:20 +02:00
|
|
|
:non_fillable?,
|
2022-09-08 11:25:39 +02:00
|
|
|
:fillable?,
|
2022-10-17 10:16:30 +02:00
|
|
|
:mandatory?,
|
2022-12-19 12:32:09 +01:00
|
|
|
:prefillable?,
|
2023-01-17 15:28:41 +01:00
|
|
|
:refresh_after_update?,
|
2023-04-28 23:28:13 +02:00
|
|
|
:character_limit?,
|
|
|
|
:character_limit,
|
2023-09-29 23:09:41 +02:00
|
|
|
:expression_reguliere,
|
|
|
|
:expression_reguliere_exemple_text,
|
|
|
|
:expression_reguliere_error_message,
|
2020-06-18 13:27:55 +02:00
|
|
|
to: :type_de_champ
|
2016-03-15 17:17:56 +01:00
|
|
|
|
2024-11-07 12:38:03 +01:00
|
|
|
delegate(*TypeDeChamp.type_champs.values.map { "#{_1}?".to_sym }, to: :type_de_champ)
|
|
|
|
delegate :piece_justificative_or_titre_identite?, :any_drop_down_list?, to: :type_de_champ
|
|
|
|
|
2023-02-22 19:32:25 +01:00
|
|
|
delegate :to_typed_id, :to_typed_id_for_query, to: :type_de_champ, prefix: true
|
2023-02-15 12:15:34 +01:00
|
|
|
|
2023-01-26 17:57:57 +01:00
|
|
|
delegate :revision, to: :dossier, prefix: true
|
|
|
|
|
2017-10-05 16:10:00 +02:00
|
|
|
scope :updated_since?, -> (date) { where('champs.updated_at > ?', date) }
|
2022-12-01 12:00:21 +01:00
|
|
|
scope :prefilled, -> { where(prefilled: true) }
|
2017-10-05 16:10:00 +02:00
|
|
|
|
2022-10-11 15:11:51 +02:00
|
|
|
before_save :cleanup_if_empty
|
2022-08-29 10:20:05 +02:00
|
|
|
before_save :normalize
|
2022-10-11 11:52:59 +02:00
|
|
|
after_update_commit :fetch_external_data_later
|
2019-04-18 16:55:35 +02:00
|
|
|
|
2018-02-09 17:38:30 +01:00
|
|
|
def public?
|
|
|
|
!private?
|
|
|
|
end
|
|
|
|
|
2022-06-13 22:54:57 +02:00
|
|
|
def child?
|
2024-11-26 22:05:26 +01:00
|
|
|
row_id.present? && !is_type?(TypeDeChamp.type_champs.fetch(:repetition))
|
|
|
|
end
|
|
|
|
|
|
|
|
def row?
|
|
|
|
row_id.present? && is_type?(TypeDeChamp.type_champs.fetch(:repetition))
|
2022-06-13 22:54:57 +02:00
|
|
|
end
|
|
|
|
|
2022-10-17 10:05:59 +02:00
|
|
|
# used for the `required` html attribute
|
|
|
|
# check visibility to avoid hidden required input
|
|
|
|
# which prevent the form from being sent.
|
|
|
|
def required?
|
|
|
|
type_de_champ.mandatory? && visible?
|
|
|
|
end
|
|
|
|
|
2022-10-17 10:38:23 +02:00
|
|
|
def mandatory_blank?
|
2024-10-21 12:24:12 +02:00
|
|
|
type_de_champ.mandatory_blank?(self)
|
2019-02-19 12:38:33 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
def blank?
|
2024-12-10 11:20:29 +01:00
|
|
|
# FIXME: temporary fix to avoid breaking validation
|
|
|
|
in_dossier_revision? ? type_de_champ.champ_blank?(self) : value.blank?
|
2017-03-29 13:37:07 +02:00
|
|
|
end
|
|
|
|
|
2024-11-04 11:41:24 +01:00
|
|
|
def used_by_routing_rules?
|
|
|
|
procedure.used_by_routing_rules?(type_de_champ)
|
|
|
|
end
|
|
|
|
|
2018-07-25 19:34:06 +02:00
|
|
|
def search_terms
|
2018-12-24 17:29:46 +01:00
|
|
|
[to_s]
|
2018-07-25 19:34:06 +02:00
|
|
|
end
|
|
|
|
|
2024-04-23 10:03:46 +02:00
|
|
|
def to_s
|
2024-10-21 11:51:34 +02:00
|
|
|
type_de_champ.champ_value(self)
|
2017-10-26 16:54:10 +02:00
|
|
|
end
|
|
|
|
|
2024-10-21 11:51:34 +02:00
|
|
|
def last_write_type_champ
|
|
|
|
TypeDeChamp::CHAMP_TYPE_TO_TYPE_CHAMP.fetch(type)
|
2019-09-11 16:04:42 +02:00
|
|
|
end
|
|
|
|
|
2024-11-06 22:29:59 +01:00
|
|
|
def is_type?(type_champ)
|
|
|
|
last_write_type_champ == type_champ
|
2024-10-29 16:38:17 +01:00
|
|
|
end
|
|
|
|
|
2018-06-21 15:23:24 +02:00
|
|
|
def main_value_name
|
|
|
|
:value
|
|
|
|
end
|
2019-04-18 16:55:35 +02:00
|
|
|
|
2024-03-21 10:21:19 +01:00
|
|
|
def champ_descriptor_id
|
|
|
|
type_de_champ.to_typed_id
|
|
|
|
end
|
|
|
|
|
2019-08-28 11:25:41 +02:00
|
|
|
def to_typed_id
|
2022-12-16 15:07:26 +01:00
|
|
|
if row_id.present?
|
|
|
|
GraphQL::Schema::UniqueWithinType.encode('Champ', "#{stable_id}|#{row_id}")
|
2022-07-21 14:54:14 +02:00
|
|
|
else
|
|
|
|
type_de_champ.to_typed_id
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.decode_typed_id(typed_id)
|
|
|
|
_, stable_id_with_maybe_row = GraphQL::Schema::UniqueWithinType.decode(typed_id)
|
|
|
|
stable_id_with_maybe_row.split('|')
|
2019-08-28 11:25:41 +02:00
|
|
|
end
|
|
|
|
|
2020-03-23 16:50:59 +01:00
|
|
|
def html_label?
|
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2023-07-03 11:16:49 +02:00
|
|
|
def legend_label?
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2023-07-10 14:48:33 +02:00
|
|
|
def single_checkbox?
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2022-01-05 11:32:05 +01:00
|
|
|
def input_group_id
|
2024-03-14 15:13:13 +01:00
|
|
|
html_id
|
2022-01-05 11:32:05 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
def input_id
|
|
|
|
"#{html_id}-input"
|
|
|
|
end
|
|
|
|
|
2022-04-28 15:03:07 +02:00
|
|
|
# A predictable string to use when generating an input name for this champ.
|
|
|
|
#
|
2022-11-10 22:21:14 +01:00
|
|
|
# Rail's FormBuilder can auto-generate input names, using the form "dossier[champs_public_attributes][5]",
|
2022-04-28 15:03:07 +02:00
|
|
|
# where [5] is the index of the field in the form.
|
|
|
|
# However the field index makes it difficult to render a single field, independent from the ordering of the others.
|
|
|
|
#
|
|
|
|
# Luckily, this is only used to make the name unique, but the actual value is ignored when Rails parses nested
|
2024-04-22 14:22:52 +02:00
|
|
|
# attributes. So instead of the field index, this method uses the champ public_id; which gives us an independent and
|
2022-04-28 15:03:07 +02:00
|
|
|
# predictable input name.
|
|
|
|
def input_name
|
2022-11-29 11:30:06 +01:00
|
|
|
if private?
|
2024-04-22 14:22:52 +02:00
|
|
|
"dossier[champs_private_attributes][#{public_id}]"
|
2022-04-28 15:03:07 +02:00
|
|
|
else
|
2024-04-22 14:22:52 +02:00
|
|
|
"dossier[champs_public_attributes][#{public_id}]"
|
2022-04-28 15:03:07 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-01-05 11:32:05 +01:00
|
|
|
def labelledby_id
|
|
|
|
"#{html_id}-label"
|
|
|
|
end
|
|
|
|
|
|
|
|
def describedby_id
|
2024-04-09 16:19:54 +02:00
|
|
|
"#{html_id}-describedby_id"
|
2022-01-05 11:32:05 +01:00
|
|
|
end
|
|
|
|
|
2021-02-04 19:24:52 +01:00
|
|
|
def log_fetch_external_data_exception(exception)
|
2023-10-27 14:16:29 +02:00
|
|
|
update_column(:fetch_external_data_exceptions, [exception.inspect])
|
2021-02-04 19:24:52 +01:00
|
|
|
end
|
|
|
|
|
2021-02-09 12:35:23 +01:00
|
|
|
def fetch_external_data?
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2023-05-30 11:39:53 +02:00
|
|
|
def poll_external_data?
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2023-10-27 14:16:29 +02:00
|
|
|
def fetch_external_data_error?
|
|
|
|
fetch_external_data_exceptions.present? && self.external_id.present?
|
|
|
|
end
|
|
|
|
|
2023-05-30 11:39:53 +02:00
|
|
|
def fetch_external_data_pending?
|
2023-10-27 14:16:29 +02:00
|
|
|
fetch_external_data? && poll_external_data? && external_id.present? && data.nil? && !fetch_external_data_error?
|
2023-05-30 11:39:53 +02:00
|
|
|
end
|
|
|
|
|
2021-02-09 12:35:23 +01:00
|
|
|
def fetch_external_data
|
|
|
|
raise NotImplemented.new(:fetch_external_data)
|
|
|
|
end
|
|
|
|
|
2023-02-28 13:55:52 +01:00
|
|
|
def update_with_external_data!(data:)
|
|
|
|
update!(data: data)
|
|
|
|
end
|
|
|
|
|
2023-03-21 15:59:03 +01:00
|
|
|
def clone(fork = false)
|
2024-09-27 15:44:57 +02:00
|
|
|
champ_attributes = [:private, :row_id, :type, :stable_id, :stream]
|
2023-03-21 15:59:03 +01:00
|
|
|
value_attributes = fork || !private? ? [:value, :value_json, :data, :external_id] : []
|
|
|
|
relationships = fork || !private? ? [:etablissement, :geo_areas] : []
|
2022-12-16 15:43:36 +01:00
|
|
|
|
2023-07-26 11:44:36 +02:00
|
|
|
deep_clone(only: champ_attributes + value_attributes, include: relationships, validate: !fork) do |original, kopy|
|
2024-03-18 20:04:53 +01:00
|
|
|
if original.is_a?(Champ)
|
|
|
|
kopy.write_attribute(:stable_id, original.stable_id)
|
|
|
|
kopy.write_attribute(:stream, 'main')
|
|
|
|
end
|
2024-05-21 09:50:27 +02:00
|
|
|
ClonePiecesJustificativesService.clone_attachments(original, kopy) if fork || !private?
|
2022-11-08 19:00:15 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-02-27 15:33:25 +01:00
|
|
|
def focusable_input_id
|
|
|
|
input_id
|
|
|
|
end
|
|
|
|
|
2023-03-21 18:24:39 +01:00
|
|
|
def forked_with_changes?
|
|
|
|
public? && dossier.champ_forked_with_changes?(self)
|
|
|
|
end
|
|
|
|
|
2024-03-21 12:01:55 +01:00
|
|
|
def public_id
|
|
|
|
if row_id.blank?
|
|
|
|
stable_id.to_s
|
|
|
|
else
|
|
|
|
"#{stable_id}-#{row_id}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-01-05 11:32:05 +01:00
|
|
|
def html_id
|
2024-09-20 16:21:12 +02:00
|
|
|
type_de_champ.html_id(row_id)
|
2022-01-05 11:32:05 +01:00
|
|
|
end
|
|
|
|
|
2021-02-09 12:35:23 +01:00
|
|
|
def cleanup_if_empty
|
2024-02-20 17:49:07 +01:00
|
|
|
if fetch_external_data? && persisted? && external_id_changed?
|
2021-02-09 12:35:23 +01:00
|
|
|
self.data = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def fetch_external_data_later
|
|
|
|
if fetch_external_data? && external_id.present? && data.nil?
|
2023-10-27 14:16:29 +02:00
|
|
|
update_column(:fetch_external_data_exceptions, [])
|
2021-03-11 12:32:08 +01:00
|
|
|
ChampFetchExternalDataJob.perform_later(self, external_id)
|
2021-02-09 12:35:23 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-08-29 10:20:05 +02:00
|
|
|
def normalize
|
|
|
|
return if value.nil?
|
2022-12-27 11:43:34 +01:00
|
|
|
return if value.present? && !value.include?("\u0000")
|
2022-08-29 10:20:05 +02:00
|
|
|
|
2024-05-07 11:25:37 +02:00
|
|
|
write_attribute(:value, value.delete("\u0000"))
|
2022-08-29 10:20:05 +02:00
|
|
|
end
|
|
|
|
|
2021-02-09 12:35:23 +01:00
|
|
|
class NotImplemented < ::StandardError
|
|
|
|
def initialize(method)
|
|
|
|
super(":#{method} not implemented")
|
|
|
|
end
|
|
|
|
end
|
2015-11-03 10:48:40 +01:00
|
|
|
end
|