require "rails_helper"

RSpec.describe Form::Lettings::Questions::SchemeId, type: :model do
  subject(:question) { described_class.new(question_id, question_definition, page) }

  let(:question_id) { nil }
  let(:question_definition) { nil }
  let(:page) { instance_double(Form::Page) }
  let(:subsection) { instance_double(Form::Subsection) }
  let(:form) { instance_double(Form, start_date: Time.zone.local(2023, 4, 1)) }

  before do
    allow(page).to receive(:subsection).and_return(subsection)
    allow(subsection).to receive(:form).and_return(form)
    allow(form).to receive(:start_year_after_2024?).and_return(false)
  end

  it "has correct page" do
    expect(question.page).to eq(page)
  end

  it "has the correct id" do
    expect(question.id).to eq("scheme_id")
  end

  it "has the correct header" do
    expect(question.header).to eq("What scheme is this log for?")
  end

  it "has the correct check_answer_label" do
    expect(question.check_answer_label).to eq("Scheme name")
  end

  it "has the correct type" do
    expect(question.type).to eq("select")
  end

  context "when 2023" do
    before do
      allow(form).to receive(:start_year_after_2024?).and_return(false)
    end

    it "has the correct hint_text" do
      expect(question.hint_text).to eq("Enter postcode or scheme name")
    end
  end

  context "when 2024" do
    before do
      allow(form).to receive(:start_year_after_2024?).and_return(true)
    end

    it "has the correct hint_text" do
      expect(question.hint_text).to eq("Enter postcode or scheme name.<br><br>
        A supported housing scheme provides shared or self-contained housing for a particular client group, for example younger or vulnerable people.")
    end
  end

  it "has the correct conditional_for" do
    expect(question.conditional_for).to be_nil
  end

  it "is not marked as derived" do
    expect(question.derived?(nil)).to be false
  end

  it "has the correct inferred_answers" do
    expect(question.inferred_answers).to eq({
      "location.name": {
        "scheme_has_multiple_locations?": false,
      },
    })
  end

  describe "has the correct get_extra_check_answer_value" do
    let(:scheme) { create(:scheme) }

    context "when locations are present but not inferred" do
      let(:lettings_log) { create(:lettings_log) }

      before do
        allow(lettings_log).to receive(:scheme_has_multiple_locations?).and_return(true)
      end

      it "returns nil" do
        expect(question.get_extra_check_answer_value(lettings_log)).to be_nil
      end
    end

    context "when location is present and inferred" do
      let!(:location) { create(:location, scheme:) }
      let!(:lettings_log) { create(:lettings_log, scheme:, location:) }
      let(:real_2022_2023_form) { Form.new("config/forms/2022_2023.json") }

      before do
        allow(lettings_log).to receive(:scheme_has_multiple_locations?).and_return(false)
        allow(lettings_log).to receive(:form).and_return(real_2022_2023_form)
      end

      it "returns the postcode" do
        expect(question.get_extra_check_answer_value(lettings_log)).to eq(location.postcode)
      end
    end
  end

  context "when a user is signed in" do
    let(:organisation) { FactoryBot.create(:organisation) }
    let(:organisation_2) { FactoryBot.create(:organisation) }
    let(:user) { FactoryBot.create(:user, organisation:) }
    let(:scheme) { FactoryBot.create(:scheme, owning_organisation: organisation) }
    let(:lettings_log) { FactoryBot.create(:lettings_log, assigned_to: user, needstype: 2) }

    before do
      FactoryBot.create(:scheme, owning_organisation: organisation_2)
    end

    context "when a scheme with at least 1 location exists" do
      context "when the location is active" do
        before do
          FactoryBot.create(:location, startdate: Time.zone.yesterday, scheme:)
        end

        it "has the correct answer_options based on the schemes the user's organisation owns or manages" do
          expected_answer = { "" => "Select an option", scheme.id.to_s => scheme }
          expect(question.displayed_answer_options(lettings_log)).to eq(expected_answer)
        end
      end

      context "when the location is activating soon" do
        before do
          FactoryBot.create(:location, startdate: Time.zone.tomorrow, scheme:)
        end

        it "has the correct answer_options based on the schemes the user's organisation owns or manages" do
          expected_answer = { "" => "Select an option", scheme.id.to_s => scheme }
          expect(question.displayed_answer_options(lettings_log)).to eq(expected_answer)
        end
      end

      context "when the location is activating more than 2 weeks in the future" do
        before do
          FactoryBot.create(:location, startdate: Time.zone.today + 3.weeks, scheme:)
        end

        it "has the correct answer_options based on the schemes the user's organisation owns or manages" do
          expected_answer = { "" => "Select an option" }
          expect(question.displayed_answer_options(lettings_log)).to eq(expected_answer)
        end
      end

      context "when the scheme is deleted" do
        let(:scheme) { FactoryBot.create(:scheme, owning_organisation: organisation, discarded_at: Time.zone.yesterday) }

        before do
          FactoryBot.create(:location, startdate: Time.zone.tomorrow, scheme:)
        end

        it "has the correct answer_options based on the schemes the user's organisation owns or manages" do
          expected_answer = { "" => "Select an option" }
          expect(question.displayed_answer_options(lettings_log)).to eq(expected_answer)
        end
      end
    end

    context "when there are no schemes with locations" do
      it "returns a hash with one empty option" do
        expect(question.displayed_answer_options(lettings_log)).to eq({ "" => "Select an option" })
      end
    end

    context "when the question is not answered" do
      it "returns 'select an option' as selected answer" do
        lettings_log.update!(scheme: nil)
        answers = question.displayed_answer_options(lettings_log).map do |key, value|
          OpenStruct.new(id: key, name: value.respond_to?(:service_name) ? value.service_name : nil, resource: value)
        end
        answers.each do |answer|
          if answer.resource == "Select an option"
            expect(question.answer_selected?(lettings_log, answer)).to eq(true)
          else
            expect(question.answer_selected?(lettings_log, answer)).to eq(false)
          end
        end
      end
    end

    context "when the question is answered" do
      before do
        FactoryBot.create(:location, scheme:)
      end

      it "returns scheme as selected answer" do
        lettings_log.update!(scheme:)
        answers = question.displayed_answer_options(lettings_log).map do |key, value|
          OpenStruct.new(id: key, name: value.respond_to?(:service_name) ? value.service_name : nil, resource: value)
        end
        answers.each do |answer|
          if answer.id.to_i == scheme.id
            expect(question.answer_selected?(lettings_log, answer)).to eq(true)
          else
            expect(question.answer_selected?(lettings_log, answer)).to eq(false)
          end
        end
      end
    end
  end
end