require 'prawn/measurement_extensions' def default_margin 10 end def maybe_start_new_page(pdf, size) if pdf.cursor < size + default_margin pdf.start_new_page end end def clean_string(str) str.tr(' ', ' ') # replace non breaking space, which are invalid in pdf end def text_box(pdf, text, x, width) box = ::Prawn::Text::Box.new(text.to_s, document: pdf, width: width, overflow: :expand, at: [x, pdf.cursor]) box.render box.height end def format_in_2_lines(pdf, label, text) min_height = [ label.present? ? pdf.height_of_formatted([{ text: label, style: :bold, size: 12 }]) : nil, text.present? ? pdf.height_of_formatted([{ text: text }]) : nil ].compact.sum maybe_start_new_page(pdf, min_height) pdf.pad_bottom(2) do pdf.font 'marianne', style: :bold, size: 12 do pdf.text label end end pdf.pad_bottom(default_margin) do pdf.text text end end def format_in_2_columns(pdf, label, text) min_height = [ label.present? ? pdf.height_of_formatted([{ text: label }]) : nil, text.present? ? pdf.height_of_formatted([{ text: text }]) : nil ].compact.max maybe_start_new_page(pdf, min_height) pdf.pad_bottom(default_margin) do height = [ text_box(pdf, label, 0, 150), text_box(pdf, ':', 150, 10), text_box(pdf, text, 160, pdf.bounds.width - 160) ].max pdf.move_down height end end def add_title(pdf, title) maybe_start_new_page(pdf, 100) pdf.pad(default_margin) do pdf.font 'marianne', style: :bold, size: 20 do pdf.text title end end end def add_section_title(pdf, title) maybe_start_new_page(pdf, 100) pdf.pad_bottom(default_margin) do pdf.font 'marianne', style: :bold, size: 14 do pdf.text title end end end def add_identite_individual(pdf, individual) pdf.pad_bottom(default_margin) do format_in_2_columns(pdf, "Civilité", individual.gender) format_in_2_columns(pdf, "Nom", individual.nom) format_in_2_columns(pdf, "Prénom", individual.prenom) if individual.birthdate.present? format_in_2_columns(pdf, "Date de naissance", try_format_date(individual.birthdate)) end end end def add_identite_etablissement(pdf, etablissement) pdf.pad_bottom(default_margin) do format_in_2_columns(pdf, "SIRET", etablissement.siret) format_in_2_columns(pdf, "SIRET du siège social", etablissement.entreprise.siret_siege_social) if etablissement.entreprise.siret_siege_social.present? format_in_2_columns(pdf, "Dénomination", raison_sociale_or_name(etablissement)) format_in_2_columns(pdf, "Forme juridique ", etablissement.entreprise_forme_juridique) if etablissement.entreprise_capital_social.present? format_in_2_columns(pdf, "Capital social ", pretty_currency(etablissement.entreprise_capital_social)) end format_in_2_columns(pdf, "Libellé NAF ", etablissement.libelle_naf) format_in_2_columns(pdf, "Code NAF ", etablissement.naf) format_in_2_columns(pdf, "Date de création ", try_format_date(etablissement.entreprise.date_creation)) if etablissement.entreprise_etat_administratif.present? format_in_2_columns(pdf, "État administratif", humanized_entreprise_etat_administratif(etablissement)) end if @include_infos_administration if etablissement.entreprise_effectif_mensuel.present? format_in_2_columns(pdf, "Effectif mensuel #{try_format_mois_effectif(etablissement)} (URSSAF) ", number_with_delimiter(etablissement.entreprise_effectif_mensuel.to_s)) end if etablissement.entreprise_effectif_annuel_annee.present? format_in_2_columns(pdf, "Effectif moyen annuel #{etablissement.entreprise_effectif_annuel_annee} (URSSAF) ", number_with_delimiter(etablissement.entreprise_effectif_annuel.to_s)) end end format_in_2_columns(pdf, "Effectif (ISPF) ", effectif(etablissement)) format_in_2_columns(pdf, "Code effectif ", etablissement.entreprise.code_effectif_entreprise) if etablissement.entreprise.numero_tva_intracommunautaire.present? format_in_2_columns(pdf, "Numéro de TVA intracommunautaire ", etablissement.entreprise.numero_tva_intracommunautaire) end format_in_2_columns(pdf, "Adresse ", etablissement.adresse) if etablissement.association? format_in_2_columns(pdf, "Numéro RNA ", etablissement.association_rna) format_in_2_columns(pdf, "Titre ", etablissement.association_titre) format_in_2_columns(pdf, "Objet ", etablissement.association_objet) format_in_2_columns(pdf, "Date de création ", try_format_date(etablissement.association_date_creation)) format_in_2_columns(pdf, "Date de publication ", try_format_date(etablissement.association_date_publication)) format_in_2_columns(pdf, "Date de déclaration ", try_format_date(etablissement.association_date_declaration)) end end end def add_single_champ(pdf, champ) tdc = champ.type_de_champ return if champ.conditional? && !champ.visible? case champ.type when 'Champs::PieceJustificativeChamp', 'Champs::TitreIdentiteChamp' return when 'Champs::HeaderSectionChamp' libelle = if @dossier.auto_numbering_section_headers_for?(champ) "#{@dossier.index_for_section_header(champ)}. #{champ.libelle}" else champ.libelle end add_section_title(pdf, libelle) when 'Champs::ExplicationChamp' format_in_2_lines(pdf, tdc.libelle, strip_tags(tdc.description)) when 'Champs::CarteChamp' pdf.pad_bottom(4) do pdf.font 'marianne', style: :bold, size: 12 do pdf.text tdc.libelle end pdf.indent(default_margin) do champ.geo_areas.each do |area| pdf.text "- #{clean_string(area.label)}" end end end when 'Champs::SiretChamp' pdf.font 'marianne', style: :bold do pdf.text tdc.libelle end if champ.etablissement.present? add_identite_etablissement(pdf, champ.etablissement) end when 'Champs::NumberChamp' value = champ.blank? ? 'Non communiqué' : number_with_delimiter(champ.to_s) format_in_2_lines(pdf, tdc.libelle, value) when 'Champs::CommuneChamp' value = champ.blank? ? 'Non communiqué' : champ.to_s format_in_2_lines(pdf, tdc.libelle, value) pdf.text "Département : #{champ.departement_code_and_name}" if champ.departement? else value = champ.blank? ? 'Non communiqué' : champ.to_s format_in_2_lines(pdf, tdc.libelle, value) end end def add_champs(pdf, champs) champs.each do |champ| if champ.type == 'Champs::RepetitionChamp' champ.rows.each do |row| row.each do |inner_champ| add_single_champ(pdf, inner_champ) end end else add_single_champ(pdf, champ) end end end def add_message(pdf, message) sender = message.redacted_email if message.sent_by_system? sender = 'Email automatique' elsif message.sent_by?(@dossier.user) sender = @dossier.user_email_for(:display) end format_in_2_lines(pdf, "#{sender}, #{try_format_date(message.created_at)}", ActionView::Base.full_sanitizer.sanitize(message.body)) end def add_avis(pdf, avis) format_in_2_lines(pdf, "Avis de #{avis.email_to_display}#{avis.confidentiel? ? ' (confidentiel)' : ''}", avis.answer || 'En attente de réponse') if [true, false].include? avis.question_answer format_in_2_columns(pdf, "Réponse oui/non ", t("question_answer.#{avis.question_answer}", scope: 'helpers.label')) end end def add_etat_dossier(pdf, dossier) pdf.pad_bottom(default_margin) do pending_correction = dossier.pending_correction? ? " (en attente de correction)" : nil pdf.text "Ce dossier est #{clean_string(dossier_display_state(dossier, lower: true))}#{pending_correction}.", inline_format: true end end def add_etats_dossier(pdf, dossier) if dossier.depose_at.present? format_in_2_columns(pdf, "Déposé le", try_format_date(dossier.depose_at)) end if dossier.pending_correction? format_in_2_columns(pdf, "Correction demandée le", try_format_date(dossier.pending_correction.created_at)) end if dossier.en_instruction_at.present? format_in_2_columns(pdf, "En instruction le", try_format_date(dossier.en_instruction_at)) end if dossier.processed_at.present? format_in_2_columns(pdf, "Décision le", try_format_date(dossier.processed_at)) end end prawn_document(page_size: "A4") do |pdf| pdf.font_families.update('marianne' => { normal: Rails.root.join('lib/prawn/fonts/marianne/marianne-regular.ttf'), bold: Rails.root.join('lib/prawn/fonts/marianne/marianne-bold.ttf') }) pdf.font 'marianne' pdf.pad_bottom(40) do pdf.image DOSSIER_PDF_EXPORT_LOGO_SRC, width: 300, position: :center end format_in_2_columns(pdf, 'Dossier Nº', @dossier.id.to_s) format_in_2_columns(pdf, 'Démarche', @dossier.procedure.libelle) format_in_2_columns(pdf, 'Organisme', @dossier.procedure.organisation_name) add_etat_dossier(pdf, @dossier) if @dossier.motivation.present? format_in_2_columns(pdf, "Motif de la décision", @dossier.motivation) end add_title(pdf, 'Historique') add_etats_dossier(pdf, @dossier) add_title(pdf, "Identité du demandeur") if @dossier.france_connect_information.present? format_in_2_columns(pdf, 'Informations FranceConnect', france_connect_informations(@dossier.france_connect_information)) end format_in_2_columns(pdf, "Email", @dossier.user_email_for(:display)) if @dossier.individual.present? add_identite_individual(pdf, @dossier.individual) elsif @dossier.etablissement.present? add_identite_etablissement(pdf, @dossier.etablissement) end add_title(pdf, 'Formulaire') add_champs(pdf, @dossier.champs_public) if @include_infos_administration && @dossier.champs_private.present? add_title(pdf, "Annotations privées") add_champs(pdf, @dossier.champs_private) end if @include_infos_administration && @dossier.avis.present? add_title(pdf, "Avis") @dossier.avis.each do |avis| add_avis(pdf, avis) end end if @include_avis_for_expert && @dossier.avis.present? add_title(pdf, "Avis") @dossier.avis_for_expert(@include_avis_for_expert).each do |avis| add_avis(pdf, avis) end end if @dossier.commentaires.present? add_title(pdf, 'Messagerie') @dossier.commentaires.each do |commentaire| add_message(pdf, commentaire) end end pdf.number_pages ' / ', at: [pdf.bounds.right - 80, pdf.bounds.bottom], align: :right, size: 10 end