| 
									
										
										
										
											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-10-20 12:03:48 +02:00
										 |  |  |   extend T::Sig | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # 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) | 
					
						
							| 
									
										
										
										
											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)) | 
					
						
							| 
									
										
										
										
											2020-08-12 00:04:20 +02:00
										 |  |  |       sep = scanner.scan(/-/) | 
					
						
							|  |  |  |       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)) | 
					
						
							| 
									
										
										
										
											2020-08-12 00:04:20 +02:00
										 |  |  |       sep = scanner.scan(/-/) | 
					
						
							|  |  |  |       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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-20 12:03:48 +02:00
										 |  |  |   sig { returns(String) } | 
					
						
							| 
									
										
										
										
											2016-09-27 22:23:13 +02:00
										 |  |  |   def to_s | 
					
						
							|  |  |  |     [@language, @region, @script].compact.join("-") | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end |