Added Placeholders module and added methods to convert from twine format to android and vice versa.
This commit is contained in:
parent
b2b4bdcc8a
commit
1a49852a6e
5 changed files with 148 additions and 21 deletions
|
@ -7,6 +7,7 @@ module Twine
|
|||
require 'twine/encoding'
|
||||
require 'twine/output_processor'
|
||||
require 'twine/formatters'
|
||||
require 'twine/placeholders'
|
||||
require 'twine/runner'
|
||||
require 'twine/stringsfile'
|
||||
require 'twine/version'
|
||||
|
|
|
@ -49,7 +49,7 @@ module Twine
|
|||
value = CGI.unescapeHTML(value)
|
||||
value.gsub!('\\\'', '\'')
|
||||
value.gsub!('\\"', '"')
|
||||
value = iosify_substitutions(value)
|
||||
value = Placeholders.from_android_to_twine(value)
|
||||
value.gsub!(/(\\u0020)*|(\\u0020)*\z/) { |spaces| ' ' * (spaces.length / 6) }
|
||||
super(key, lang, value)
|
||||
end
|
||||
|
@ -121,7 +121,7 @@ module Twine
|
|||
# 2) HTML escape the string
|
||||
value = CGI.escapeHTML(value)
|
||||
# 3) fix substitutions (e.g. %s/%@)
|
||||
value = androidify_substitutions(value)
|
||||
value = Placeholders.from_twine_to_android(value)
|
||||
# 4) replace beginning and end spaces with \0020. Otherwise Android strips them.
|
||||
value.gsub(/\A *| *\z/) { |spaces| '\u0020' * spaces.length }
|
||||
end
|
||||
|
|
52
lib/twine/placeholders.rb
Normal file
52
lib/twine/placeholders.rb
Normal file
|
@ -0,0 +1,52 @@
|
|||
module Twine
|
||||
module Placeholders
|
||||
PLACEHOLDER_FLAGS_WIDTH_PRECISION_LENGTH = '([-+ 0#])?(\d+|\*)?(\.(\d+|\*))?(hh?|ll?|L|z|j|t)?'
|
||||
PLACEHOLDER_PARAMETER_FLAGS_WIDTH_PRECISION_LENGTH = '(\d+\$)?' + PLACEHOLDER_FLAGS_WIDTH_PRECISION_LENGTH
|
||||
|
||||
# http://developer.android.com/guide/topics/resources/string-resource.html#FormattingAndStyling
|
||||
# http://stackoverflow.com/questions/4414389/android-xml-percent-symbol
|
||||
# https://github.com/mobiata/twine/pull/106
|
||||
def self.from_twine_to_android(input)
|
||||
placeholder_types = '[diufFeEgGxXoscpaA]'
|
||||
|
||||
# %@ -> %s
|
||||
value = input.gsub(/(%#{PLACEHOLDER_PARAMETER_FLAGS_WIDTH_PRECISION_LENGTH})@/, '\1s')
|
||||
|
||||
placeholder_syntax = PLACEHOLDER_PARAMETER_FLAGS_WIDTH_PRECISION_LENGTH + placeholder_types
|
||||
placeholder_regex = /%#{placeholder_syntax}/
|
||||
|
||||
number_of_placeholders = value.scan(placeholder_regex).size
|
||||
|
||||
return value if number_of_placeholders == 0
|
||||
|
||||
# got placeholders -> need to double single percent signs
|
||||
# % -> %% (but %% -> %%, %d -> %d)
|
||||
single_percent_regex = /([^%])(%)(?!(%|#{placeholder_syntax}))/
|
||||
value.gsub! single_percent_regex, '\1%%'
|
||||
|
||||
return value if number_of_placeholders < 2
|
||||
|
||||
# number placeholders
|
||||
non_numbered_placeholder_regex = /%(#{PLACEHOLDER_FLAGS_WIDTH_PRECISION_LENGTH}#{placeholder_types})/
|
||||
|
||||
number_of_non_numbered_placeholders = value.scan(non_numbered_placeholder_regex).size
|
||||
|
||||
return value if number_of_non_numbered_placeholders == 0
|
||||
|
||||
raise Twine::Error.new("The value \"#{input}\" contains numbered and non-numbered placeholders") if number_of_placeholders != number_of_non_numbered_placeholders
|
||||
|
||||
# %d -> %$1d
|
||||
index = 0
|
||||
value.gsub!(non_numbered_placeholder_regex) { "%#{index += 1}$#{$1}" }
|
||||
|
||||
value
|
||||
end
|
||||
|
||||
def self.from_android_to_twine(input)
|
||||
placeholder_regex = /(%#{PLACEHOLDER_PARAMETER_FLAGS_WIDTH_PRECISION_LENGTH})s/
|
||||
|
||||
# %s -> %@
|
||||
input.gsub(placeholder_regex, '\1@')
|
||||
end
|
||||
end
|
||||
end
|
|
@ -34,16 +34,21 @@ class TestAndroidFormatter < FormatterTest
|
|||
end
|
||||
end
|
||||
|
||||
def test_set_translation_transforms_leading_spaces
|
||||
def test_set_translation_converts_leading_spaces
|
||||
@formatter.set_translation_for_key 'key1', 'en', "\u0020value"
|
||||
assert_equal ' value', @strings.strings_map['key1'].translations['en']
|
||||
end
|
||||
|
||||
def test_set_translation_transforms_trailing_spaces
|
||||
def test_set_translation_coverts_trailing_spaces
|
||||
@formatter.set_translation_for_key 'key1', 'en', "value\u0020\u0020"
|
||||
assert_equal 'value ', @strings.strings_map['key1'].translations['en']
|
||||
end
|
||||
|
||||
def test_set_translation_converts_string_placeholders
|
||||
@formatter.set_translation_for_key 'key1', 'en', "value %s"
|
||||
assert_equal 'value %@', @strings.strings_map['key1'].translations['en']
|
||||
end
|
||||
|
||||
def test_write_file_output_format
|
||||
formatter = Twine::Formatters::Android.new @twine_file, {}
|
||||
formatter.write_file @output_path, 'en'
|
||||
|
@ -67,23 +72,6 @@ class TestAndroidFormatter < FormatterTest
|
|||
# http://stackoverflow.com/questions/18735608/cgiescapehtml-is-escaping-single-quote
|
||||
assert_equal "not \\'so\\' easy", @formatter.format_value("not 'so' easy")
|
||||
end
|
||||
|
||||
def test_format_value_transforms_string_placeholder
|
||||
assert_equal '%s', @formatter.format_value('%@')
|
||||
end
|
||||
|
||||
def test_format_value_transforms_ordered_string_placeholder
|
||||
assert_equal '%1s', @formatter.format_value('%1@')
|
||||
end
|
||||
|
||||
def test_format_value_transforming_ordered_placeholders_maintains_order
|
||||
assert_equal '%2s %1d', @formatter.format_value('%2@ %1d')
|
||||
end
|
||||
|
||||
def test_format_value_does_not_alter_double_percent
|
||||
assert_equal '%%d%%', @formatter.format_value('%%d%%')
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
class TestAppleFormatter < FormatterTest
|
||||
|
|
86
test/test_placeholders.rb
Normal file
86
test/test_placeholders.rb
Normal file
|
@ -0,0 +1,86 @@
|
|||
require 'twine_test_case'
|
||||
|
||||
class PlaceholderTestCase < TwineTestCase
|
||||
def assert_starts_with(prefix, value)
|
||||
msg = message(nil) { "Expected #{mu_pp(value)} to start with #{mu_pp(prefix)}" }
|
||||
assert value.start_with?(prefix), msg
|
||||
end
|
||||
|
||||
def placeholder(type = nil)
|
||||
# %[parameter][flags][width][.precision][length]type (see https://en.wikipedia.org/wiki/Printf_format_string#Format_placeholder_specification)
|
||||
lucky = lambda { rand > 0.5 }
|
||||
placeholder = '%'
|
||||
placeholder += (rand * 20).to_i.to_s + '$' if lucky.call
|
||||
placeholder += '-+ 0#'.chars.to_a.sample if lucky.call
|
||||
placeholder += (0.upto(20).map(&:to_s) << "*").sample if lucky.call
|
||||
placeholder += '.' + (0.upto(20).map(&:to_s) << "*").sample if lucky.call
|
||||
placeholder += %w(h hh l ll L z j t).sample if lucky.call
|
||||
placeholder += type || 'diufFeEgGxXocpaA'.chars.to_a.sample # this does not contain s or @ because strings are a special case
|
||||
end
|
||||
end
|
||||
|
||||
class PlaceholderTest < TwineTestCase
|
||||
class ToAndroid < PlaceholderTestCase
|
||||
def to_android(value)
|
||||
Twine::Placeholders.from_twine_to_android(value)
|
||||
end
|
||||
|
||||
def test_replaces_string_placeholder
|
||||
placeholder = placeholder('@')
|
||||
expected = placeholder
|
||||
expected[-1] = 's'
|
||||
assert_equal "some #{expected} value", to_android("some #{placeholder} value")
|
||||
end
|
||||
|
||||
def test_does_not_change_regular_at_signs
|
||||
input = "some @ more @@ signs @"
|
||||
assert_equal input, to_android(input)
|
||||
end
|
||||
|
||||
def test_does_not_modify_single_percent_signs
|
||||
assert_equal "some % value", to_android("some % value")
|
||||
end
|
||||
|
||||
def test_escapes_single_percent_signs_if_placeholder_present
|
||||
assert_starts_with "some %% v", to_android("some % value #{placeholder}")
|
||||
end
|
||||
|
||||
def test_does_not_modify_double_percent_signs
|
||||
assert_equal "some %% value", to_android("some %% value")
|
||||
end
|
||||
|
||||
def test_does_not_modify_double_percent_signs_if_placeholder_present
|
||||
assert_starts_with "some %% v", to_android("some %% value #{placeholder}")
|
||||
end
|
||||
|
||||
def test_does_not_modify_single_placeholder
|
||||
input = "some #{placeholder} text"
|
||||
assert_equal input, to_android(input)
|
||||
end
|
||||
|
||||
def test_numbers_multiple_placeholders
|
||||
assert_equal "first %1$d second %2$f", to_android("first %d second %f")
|
||||
end
|
||||
|
||||
def test_does_not_modify_numbered_placeholders
|
||||
input = "second %2$f first %1$d"
|
||||
assert_equal input, to_android(input)
|
||||
end
|
||||
|
||||
def test_raises_an_error_when_mixing_numbered_and_non_numbered_placeholders
|
||||
assert_raises Twine::Error do
|
||||
to_android("some %d second %2$f")
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
class FromAndroid < PlaceholderTestCase
|
||||
def from_android(value)
|
||||
Twine::Placeholders.from_android_to_twine(value)
|
||||
end
|
||||
|
||||
def test_replaces_string_placeholder
|
||||
assert_equal "some %@ value", from_android("some %s value")
|
||||
end
|
||||
end
|
||||
end
|
Reference in a new issue