2024-08-12 10:30:59 +01:00
|
|
|
# typed: true # rubocop:todo Sorbet/StrictSigil
|
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.
|
2016-09-27 22:23:13 +02:00
|
|
|
class Locale
|
2020-10-20 12:03:48 +02: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
|
2024-01-18 22:18:42 +00:00
|
|
|
LANGUAGE_REGEX = /(?:[a-z]{2,3})/
|
2020-08-12 00:04:20 +02:00
|
|
|
private_constant :LANGUAGE_REGEX
|
|
|
|
|
|
|
|
# ISO 15924
|
2024-01-18 22:18:42 +00:00
|
|
|
SCRIPT_REGEX = /(?:[A-Z][a-z]{3})/
|
2020-08-12 00:04:20 +02:00
|
|
|
private_constant :SCRIPT_REGEX
|
2016-09-27 22:23:13 +02:00
|
|
|
|
2023-04-13 11:20:28 +08:00
|
|
|
# ISO 3166-1 or UN M.49
|
2024-01-18 22:18:42 +00:00
|
|
|
REGION_REGEX = /(?:[A-Z]{2}|\d{3})/
|
2023-04-13 11:20:28 +08:00
|
|
|
private_constant :REGION_REGEX
|
|
|
|
|
2024-01-18 22:18:42 +00:00
|
|
|
LOCALE_REGEX = /\A((?:#{LANGUAGE_REGEX}|#{REGION_REGEX}|#{SCRIPT_REGEX})(?:-|$)){1,3}\Z/
|
2020-08-12 00:04:20 +02:00
|
|
|
private_constant :LOCALE_REGEX
|
2016-09-27 22:23:13 +02:00
|
|
|
|
|
|
|
def self.parse(string)
|
2021-02-12 18:33:37 +05:30
|
|
|
if (locale = try_parse(string))
|
2020-08-12 00:04:20 +02:00
|
|
|
return locale
|
|
|
|
end
|
|
|
|
|
|
|
|
raise ParserError, "'#{string}' cannot be parsed to a #{self}"
|
|
|
|
end
|
|
|
|
|
2020-10-20 12:03:48 +02:00
|
|
|
sig { params(string: String).returns(T.nilable(T.attached_class)) }
|
2020-08-12 00:04:20 +02:00
|
|
|
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
|
|
|
|
2021-02-12 18:33:37 +05:30
|
|
|
if (language = scanner.scan(LANGUAGE_REGEX))
|
2023-12-14 02:52:30 +00:00
|
|
|
sep = scanner.scan("-")
|
2020-08-12 00:04:20 +02:00
|
|
|
return if (sep && scanner.eos?) || (sep.nil? && !scanner.eos?)
|
2016-09-27 22:23:13 +02:00
|
|
|
end
|
|
|
|
|
2023-03-17 20:15:42 +08:00
|
|
|
if (script = scanner.scan(SCRIPT_REGEX))
|
2023-12-14 02:52:30 +00:00
|
|
|
sep = scanner.scan("-")
|
2020-08-12 00:04:20 +02:00
|
|
|
return if (sep && scanner.eos?) || (sep.nil? && !scanner.eos?)
|
|
|
|
end
|
|
|
|
|
2023-03-17 20:15:42 +08:00
|
|
|
region = scanner.scan(REGION_REGEX)
|
2020-08-12 00:04:20 +02:00
|
|
|
|
|
|
|
return unless scanner.eos?
|
2016-10-07 20:03:50 +02:00
|
|
|
|
2023-04-13 11:20:28 +08:00
|
|
|
new(language, script, region)
|
2016-09-27 22:23:13 +02:00
|
|
|
end
|
|
|
|
|
2023-04-13 11:20:28 +08:00
|
|
|
attr_reader :language, :script, :region
|
2016-09-27 22:23:13 +02:00
|
|
|
|
2023-04-13 11:20:28 +08:00
|
|
|
def initialize(language, script, region)
|
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
|
|
|
|
|
|
|
{
|
2024-03-07 16:20:20 +00:00
|
|
|
language:,
|
|
|
|
script:,
|
|
|
|
region:,
|
2016-09-27 22:23:13 +02:00
|
|
|
}.each do |key, value|
|
|
|
|
next if value.nil?
|
|
|
|
|
2023-12-18 09:34:01 -08:00
|
|
|
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
|
|
|
|
2023-04-13 11:20:28 +08:00
|
|
|
[:language, :script, :region].all? do |var|
|
2025-02-28 09:59:32 +00:00
|
|
|
next true if other.public_send(var).nil?
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2023-04-13 11:20:28 +08:00
|
|
|
[:language, :script, :region].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
|
|
|
|
|
2020-10-20 12:03:48 +02:00
|
|
|
sig { returns(String) }
|
2016-09-27 22:23:13 +02:00
|
|
|
def to_s
|
2023-04-13 11:20:28 +08:00
|
|
|
[@language, @script, @region].compact.join("-")
|
2016-09-27 22:23:13 +02:00
|
|
|
end
|
|
|
|
end
|