demarches-normaliennes/app/services/procedure_export_service.rb

95 lines
2.8 KiB
Ruby
Raw Normal View History

class ProcedureExportService
2019-06-25 15:46:10 +02:00
attr_reader :dossiers
2019-06-25 15:46:10 +02:00
def initialize(procedure, dossiers)
@procedure = procedure
2019-09-18 16:51:45 +02:00
@dossiers = dossiers.downloadable_sorted
2019-06-25 15:46:10 +02:00
@tables = [:dossiers, :etablissements, :avis] + champs_repetables_options
end
def to_csv
2019-06-25 15:46:10 +02:00
SpreadsheetArchitect.to_csv(options_for(:dossiers, :csv))
end
def to_xlsx
2019-06-25 15:46:10 +02:00
# We recursively build multi page spreadsheet
@tables.reduce(nil) do |package, table|
SpreadsheetArchitect.to_axlsx_package(options_for(table, :xlsx), package)
end.to_stream.read
end
def to_ods
2019-06-25 15:46:10 +02:00
# We recursively build multi page spreadsheet
@tables.reduce(nil) do |spreadsheet, table|
SpreadsheetArchitect.to_rodf_spreadsheet(options_for(table, :ods), spreadsheet)
end.bytes
end
private
2019-06-25 15:46:10 +02:00
def etablissements
@etablissements ||= dossiers.flat_map do |dossier|
2018-12-28 17:54:53 +01:00
[dossier.champs, dossier.champs_private]
.flatten
2019-09-12 11:26:22 +02:00
.filter { |champ| champ.is_a?(Champs::SiretChamp) }
end.filter_map(&:etablissement) + dossiers.filter_map(&:etablissement)
end
2019-06-25 15:46:10 +02:00
def avis
@avis ||= dossiers.flat_map(&:avis)
end
2019-06-25 15:46:10 +02:00
def champs_repetables
@champs_repetables ||= dossiers.flat_map do |dossier|
[dossier.champs, dossier.champs_private]
.flatten
.filter { |champ| champ.is_a?(Champs::RepetitionChamp) }
end.group_by(&:libelle_for_export)
end
2019-06-25 15:46:10 +02:00
def champs_repetables_options
champs_repetables.map do |libelle, champs|
2018-12-28 17:54:53 +01:00
[
2019-06-25 15:46:10 +02:00
libelle,
champs.flat_map(&:rows_for_export)
]
end
end
2019-06-25 15:46:10 +02:00
DEFAULT_STYLES = {
header_style: { background_color: "000000", color: "FFFFFF", font_size: 12, bold: true },
row_style: { background_color: nil, color: "000000", font_size: 12 }
}
2019-06-25 15:46:10 +02:00
def options_for(table, format)
options = case table
2019-06-25 15:46:10 +02:00
when :dossiers
{ instances: dossiers.to_a, sheet_name: 'Dossiers', spreadsheet_columns: spreadsheet_columns(format) }
2019-06-25 15:46:10 +02:00
when :etablissements
{ instances: etablissements.to_a, sheet_name: 'Etablissements' }
when :avis
{ instances: avis.to_a, sheet_name: 'Avis' }
when Array
{ instances: table.last, sheet_name: table.first }
end.merge(DEFAULT_STYLES)
2020-09-29 15:56:38 +02:00
# transliterate: convert to ASCII characters
# to ensure truncate respects 30 bytes
2020-09-29 15:56:38 +02:00
# /\*?[] are invalid Excel worksheet characters
options[:sheet_name] = I18n.transliterate(options[:sheet_name], locale: :en)
.delete('/\*?[]')
.truncate(30, omission: '')
options
end
def spreadsheet_columns(format)
types_de_champ = @procedure.types_de_champ_for_export
types_de_champ_private = @procedure.types_de_champ_private_for_export
Proc.new do |instance|
instance.send(:"spreadsheet_columns_#{format}", types_de_champ: types_de_champ, types_de_champ_private: types_de_champ_private)
end
end
end