# frozen_string_literal: true require "relaton_bib/contributor" module RelatonBib class << self def person_hash_to_bib(c) Person.new( name: fullname_hash_to_bib(c), affiliation: affiliation_hash_to_bib(c), contacts: contacts_hash_to_bib(c), identifiers: person_identifiers_hash_to_bib(c), ) end def fullname_hash_to_bib(c) n = c[:name] FullName.new( forenames: array(n[:forenames])&.map { |f| localname(f, c) }, initials: array(n[:initials])&.map { |f| localname(f, c) }, additions: array(n[:additions])&.map { |f| localname(f, c) }, prefix: array(n[:prefix])&.map { |f| localname(f, c) }, surname: localname(n[:surname], c), completename: localname(n[:completename], c), ) end def person_identifiers_hash_to_bib(c) array(c[:identifiers])&.map do |a| PersonIdentifier.new(a[:type], a[:id]) end end end # Person's full name class FullName # @return [Array] attr_accessor :forenames # @return [Array] attr_accessor :initials # @return [RelatonBib::LocalizedString] attr_accessor :surname # @return [Array] attr_accessor :additions # @return [Array] attr_accessor :prefix # @return [RelatonBib::LocalizedString] attr_reader :completename # @param surname [RelatonBib::LocalizedString] # @param forenames [Array] # @param initials [Array] # @param additions [Array] # @param prefix [Array] # @param completename [RelatonBib::LocalizedString] def initialize(**args) unless args[:surname] || args[:completename] raise ArgumentError, "Should be given :surname or :completename" end @surname = args[:surname] @forenames = args.fetch :forenames, [] @initials = args.fetch :initials, [] @additions = args.fetch :additions, [] @prefix = args.fetch :prefix, [] @completename = args[:completename] end # @param builder [Nokogiri::XML::Builder] def to_xml(builder) builder.name do if completename builder.completename { completename.to_xml builder } else prefix.each { |p| builder.prefix { p.to_xml builder } } initials.each { |i| builder.initial { i.to_xml builder } } additions.each { |a| builder.addition { a.to_xml builder } } builder.surname { surname.to_xml builder } forenames.each { |f| builder.forename { f.to_xml builder } } end end end end # Person identifier type. module PersonIdentifierType ISNI = "isni" URI = "uri" # Checks type. # @param type [String] # @raise [ArgumentError] if type isn't "isni" or "uri" def self.check(type) unless [ISNI, URI].include? type raise ArgumentError, 'Invalid type. It should be "isni" or "uri".' end end end # Person identifier. class PersonIdentifier # @return [RelatonBib::PersonIdentifierType::ISNI, RelatonBib::PersonIdentifierType::URI] attr_accessor :type # @return [String] attr_accessor :value # @param type [RelatonBib::PersonIdentifierType::ISNI, RelatonBib::PersonIdentifierType::URI] # @param value [String] def initialize(type, value) PersonIdentifierType.check type @type = type @value = value end # @param builser [Nokogiri::XML::Builder] def to_xml(builder) builder.identifier value, type: type end end # Person class. class Person < Contributor # @return [RelatonBib::FullName] attr_accessor :name # @return [Array] attr_accessor :affiliation # @return [Array] attr_accessor :identifiers # @param name [RelatonBib::FullName] # @param affiliation [Array] # @param contacts [Array] # @param identifiers [Array] def initialize(name:, affiliation: [], contacts: [], identifiers: []) super(contacts: contacts) @name = name @affiliation = affiliation @identifiers = identifiers end # @param builder [Nokogiri::XML::Builder] def to_xml(builder) builder.person do name.to_xml builder affiliation.each { |a| a.to_xml builder } identifiers.each { |id| id.to_xml builder } contacts.each { |contact| contact.to_xml builder } end end end end