| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  | # typed: strict | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |   sig { params(string: String).returns(T.attached_class) } | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |   sig { returns(T.nilable(String)) } | 
					
						
							|  |  |  |   attr_reader :language | 
					
						
							| 
									
										
										
										
											2016-09-27 22:23:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |   sig { returns(T.nilable(String)) } | 
					
						
							|  |  |  |   attr_reader :script | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sig { returns(T.nilable(String)) } | 
					
						
							|  |  |  |   attr_reader :region | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sig { params(language: T.nilable(String), script: T.nilable(String), region: T.nilable(String)).void } | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |     unless language.nil? | 
					
						
							|  |  |  |       regex = LANGUAGE_REGEX | 
					
						
							|  |  |  |       raise ParserError, "'language' does not match #{regex}" unless language.match?(regex) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       @language = T.let(language, T.nilable(String)) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-27 22:23:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |     unless script.nil? | 
					
						
							|  |  |  |       regex = SCRIPT_REGEX | 
					
						
							|  |  |  |       raise ParserError, "'script' does not match #{regex}" unless script.match?(regex) | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |       @script = T.let(script, T.nilable(String)) | 
					
						
							| 
									
										
										
										
											2016-09-27 22:23:13 +02:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return if region.nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     regex = REGION_REGEX | 
					
						
							|  |  |  |     raise ParserError, "'region' does not match #{regex}" unless region.match?(regex) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @region = T.let(region, T.nilable(String)) | 
					
						
							| 
									
										
										
										
											2016-09-27 22:23:13 +02:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |   sig { params(other: T.any(String, Locale)).returns(T::Boolean) } | 
					
						
							| 
									
										
										
										
											2016-09-27 22:23:13 +02:00
										 |  |  |   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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |   sig { params(other: T.any(String, Locale)).returns(T::Boolean) } | 
					
						
							| 
									
										
										
										
											2016-09-27 22:23:13 +02:00
										 |  |  |   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? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 19:33:50 +02:00
										 |  |  |   sig { | 
					
						
							|  |  |  |     params( | 
					
						
							|  |  |  |       locale_groups: T::Enumerable[T::Enumerable[T.any(String, Locale)]], | 
					
						
							|  |  |  |     ).returns( | 
					
						
							|  |  |  |       T.nilable(T::Enumerable[T.any(String, Locale)]), | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											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 |