Added Placeholders module and added methods to convert from twine format to android and vice versa.

This commit is contained in:
Sebastian Ludwig 2015-12-05 15:33:21 +01:00
parent b2b4bdcc8a
commit 1a49852a6e
5 changed files with 148 additions and 21 deletions

View file

@ -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'

View file

@ -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
View 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

View file

@ -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
View 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