class Form::Subsection attr_accessor :id, :label, :section, :pages, :depends_on, :form def initialize(id, hsh, section) @id = id @section = section if hsh @label = hsh["label"] @depends_on = hsh["depends_on"] @displayed_in_tasklist_from_json = hsh["displayed_in_tasklist"] @pages = hsh["pages"].map { |s_id, p| Form::Page.new(s_id, p, self) } end end delegate :form, to: :section def questions @questions ||= pages.flat_map(&:questions) end def enabled?(log) return true unless depends_on form.depends_on_met(depends_on, log) end def status(log) return :cannot_start_yet unless enabled?(log) qs = applicable_questions(log) qs_optional_removed = qs.reject { |q| log.optional_fields.include?(q.id) } return :not_started if qs.count.positive? && qs.all? { |question| question.unanswered?(log) || question.read_only? || question.derived?(log) } return :completed if qs_optional_removed.all? { |question| question.completed?(log) } :in_progress end def complete?(log) status(log) == :completed end def is_incomplete?(log) %i[not_started in_progress].include?(status(log)) end def is_started?(log) %i[in_progress completed].include?(status(log)) end def applicable_questions(log) questions.select do |q| (q.displayed_to_user?(log) && !q.derived?(log)) || q.is_derived_or_has_inferred_check_answers_value?(log) end end def displayed_in_tasklist?(log) return true unless @displayed_in_tasklist_from_json @displayed_in_tasklist_from_json.any? do |conditions| conditions.all? do |method, expected_return_value| log.send(method) == expected_return_value end end end def not_displayed_in_tasklist?(log) !displayed_in_tasklist?(log) end end