2020-10-10 14:16:11 +02:00
|
|
|
# typed: true
|
2019-04-19 15:38:03 +09:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2020-08-12 00:04:20 +02:00
|
|
|
# Representation of a system locale.
|
|
|
|
#
|
2020-11-05 17:17:03 -05:00
|
|
|
# Used to compare the system language and languages defined using the cask `language` stanza.
|
2020-08-12 00:04:20 +02:00
|
|
|
#
|
|
|
|
# @api private
|
2016-09-27 22:23:13 +02:00
|
|
|
class Locale
|
2020-11-05 17:17:03 -05:00
|
|
|
# Error when a string cannot be parsed to a {Locale}.
|
2016-10-07 20:03:50 +02:00
|
|
|
class ParserError < StandardError
|
2016-09-27 22:23:13 +02:00
|
|
|
end
|
|
|
|
|
2020-08-12 00:04:20 +02:00
|
|
|
# ISO 639-1 or ISO 639-2
|
|
|
|
LANGUAGE_REGEX = /(?:[a-z]{2,3})/.freeze
|
|
|
|
private_constant :LANGUAGE_REGEX
|
|
|
|
|
|
|
|
# ISO 3166-1 or UN M.49
|
|
|
|
REGION_REGEX = /(?:[A-Z]{2}|\d{3})/.freeze
|
|
|
|
private_constant :REGION_REGEX
|
|
|
|
|
|
|
|
# ISO 15924
|
|
|
|
SCRIPT_REGEX = /(?:[A-Z][a-z]{3})/.freeze
|
|
|
|
private_constant :SCRIPT_REGEX
|
2016-09-27 22:23:13 +02:00
|
|
|
|
2020-06-02 09:49:23 +01:00
|
|
|
LOCALE_REGEX = /\A((?:#{LANGUAGE_REGEX}|#{REGION_REGEX}|#{SCRIPT_REGEX})(?:-|$)){1,3}\Z/.freeze
|
2020-08-12 00:04:20 +02:00
|
|
|
private_constant :LOCALE_REGEX
|
2016-09-27 22:23:13 +02:00
|
|
|
|
|
|
|
def self.parse(string)
|
2020-08-12 00:04:20 +02:00
|
|
|
if locale = try_parse(string)
|
|
|
|
return locale
|
|
|
|
end
|
|
|
|
|
|
|
|
raise ParserError, "'#{string}' cannot be parsed to a #{self}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.try_parse(string)
|
|
|
|
return if string.blank?
|
2016-09-27 22:23:13 +02:00
|
|
|
|
2020-08-12 00:04:20 +02:00
|
|
|
scanner = StringScanner.new(string)
|
2016-10-07 20:03:50 +02:00
|
|
|
|
2020-08-12 00:04:20 +02:00
|
|
|
if language = scanner.scan(LANGUAGE_REGEX)
|
|
|
|
sep = scanner.scan(/-/)
|
|
|
|
return if (sep && scanner.eos?) || (sep.nil? && !scanner.eos?)
|
2016-09-27 22:23:13 +02:00
|
|
|
end
|
|
|
|
|
2020-08-12 00:04:20 +02:00
|
|
|
if region = scanner.scan(REGION_REGEX)
|
|
|
|
sep = scanner.scan(/-/)
|
|
|
|
return if (sep && scanner.eos?) || (sep.nil? && !scanner.eos?)
|
|
|
|
end
|
|
|
|
|
|
|
|
script = scanner.scan(SCRIPT_REGEX)
|
|
|
|
|
|
|
|
return unless scanner.eos?
|
2016-10-07 20:03:50 +02:00
|
|
|
|
2016-09-27 22:23:13 +02:00
|
|
|
new(language, region, script)
|
|
|
|
end
|
|
|
|
|
|
|
|
attr_reader :language, :region, :script
|
|
|
|
|
|
|
|
def initialize(language, region, script)
|
2019-02-19 13:11:32 +00:00
|
|
|
raise ArgumentError, "#{self.class} cannot be empty" if language.nil? && region.nil? && script.nil?
|
2016-09-27 22:23:13 +02:00
|
|
|
|
|
|
|
{
|
|
|
|
language: language,
|
|
|
|
region: region,
|
|
|
|
script: script,
|
|
|
|
}.each do |key, value|
|
|
|
|
next if value.nil?
|
|
|
|
|
|
|
|
regex = self.class.const_get("#{key.upcase}_REGEX")
|
2019-10-13 10:01:31 +01:00
|
|
|
raise ParserError, "'#{value}' does not match #{regex}" unless value&.match?(regex)
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2016-09-27 22:23:13 +02:00
|
|
|
instance_variable_set(:"@#{key}", value)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def include?(other)
|
2020-08-12 00:04:20 +02:00
|
|
|
unless other.is_a?(self.class)
|
|
|
|
other = self.class.try_parse(other)
|
|
|
|
return false if other.nil?
|
|
|
|
end
|
2016-09-27 22:23:13 +02:00
|
|
|
|
2016-10-22 13:32:46 +01:00
|
|
|
[:language, :region, :script].all? do |var|
|
2016-09-27 22:23:13 +02:00
|
|
|
if other.public_send(var).nil?
|
|
|
|
true
|
|
|
|
else
|
|
|
|
public_send(var) == other.public_send(var)
|
|
|
|
end
|
2016-10-22 13:32:46 +01:00
|
|
|
end
|
2016-09-27 22:23:13 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
def eql?(other)
|
2020-08-12 00:04:20 +02:00
|
|
|
unless other.is_a?(self.class)
|
|
|
|
other = self.class.try_parse(other)
|
|
|
|
return false if other.nil?
|
|
|
|
end
|
|
|
|
|
2016-10-22 13:32:46 +01:00
|
|
|
[:language, :region, :script].all? do |var|
|
2016-09-27 22:23:13 +02:00
|
|
|
public_send(var) == other.public_send(var)
|
2016-10-22 13:32:46 +01:00
|
|
|
end
|
2016-09-27 22:23:13 +02:00
|
|
|
end
|
|
|
|
alias == eql?
|
|
|
|
|
2018-05-28 15:01:58 +01:00
|
|
|
def detect(locale_groups)
|
2018-09-02 20:14:54 +01:00
|
|
|
locale_groups.find { |locales| locales.any? { |locale| eql?(locale) } } ||
|
|
|
|
locale_groups.find { |locales| locales.any? { |locale| include?(locale) } }
|
2018-05-28 15:01:58 +01:00
|
|
|
end
|
|
|
|
|
2016-09-27 22:23:13 +02:00
|
|
|
def to_s
|
|
|
|
[@language, @region, @script].compact.join("-")
|
|
|
|
end
|
|
|
|
end
|