Merge pull request #112 from sebastianludwig/formatter_modularization_v2
Formatter modularization v2
This commit is contained in:
commit
b8291de63c
12 changed files with 256 additions and 224 deletions
|
@ -132,12 +132,13 @@ module Twine
|
|||
end
|
||||
|
||||
def format_file(strings, lang)
|
||||
result = format_header(lang) + "\n"
|
||||
header = format_header(lang)
|
||||
result = ""
|
||||
result += header + "\n" if header
|
||||
result += format_sections(strings, lang)
|
||||
end
|
||||
|
||||
def format_header(lang)
|
||||
raise NotImplementedError.new("You must implement format_header in your formatter class.")
|
||||
end
|
||||
|
||||
def format_sections(strings, lang)
|
||||
|
@ -165,21 +166,24 @@ module Twine
|
|||
result += rows.join
|
||||
end
|
||||
|
||||
def format_row(row, lang)
|
||||
value = row.translated_string_for_lang(lang)
|
||||
|
||||
return nil unless value
|
||||
|
||||
result = ""
|
||||
if row.comment
|
||||
comment = format_comment(row.comment)
|
||||
result += comment + "\n" if comment
|
||||
end
|
||||
|
||||
result += key_value_pattern % { key: format_key(row.key.dup), value: format_value(value.dup) }
|
||||
def row_pattern
|
||||
"%{comment}%{key_value}"
|
||||
end
|
||||
|
||||
def format_comment(comment)
|
||||
def format_row(row, lang)
|
||||
return nil unless row.translated_string_for_lang(lang)
|
||||
|
||||
result = row_pattern.scan(/%\{([a-z_]+)\}/).flatten
|
||||
result.map! { |element| send("format_#{element}".to_sym, row, lang) }
|
||||
result.flatten.join
|
||||
end
|
||||
|
||||
def format_comment(row, lang)
|
||||
end
|
||||
|
||||
def format_key_value(row, lang)
|
||||
value = row.translated_string_for_lang(lang)
|
||||
key_value_pattern % { key: format_key(row.key.dup), value: format_value(value.dup) }
|
||||
end
|
||||
|
||||
def key_value_pattern
|
||||
|
@ -194,6 +198,10 @@ module Twine
|
|||
value
|
||||
end
|
||||
|
||||
def escape_quotes(text)
|
||||
text.gsub('"', '\\\\"')
|
||||
end
|
||||
|
||||
def write_file(path, lang)
|
||||
encoding = @options[:output_encoding] || 'UTF-8'
|
||||
|
||||
|
|
|
@ -105,8 +105,8 @@ module Twine
|
|||
"\t<!-- SECTION: #{section.name} -->"
|
||||
end
|
||||
|
||||
def format_comment(comment)
|
||||
"\t<!-- #{comment.gsub('--', '—')} -->"
|
||||
def format_comment(row, lang)
|
||||
"\t<!-- #{row.comment.gsub('--', '—')} -->\n" if row.comment
|
||||
end
|
||||
|
||||
def key_value_pattern
|
||||
|
@ -114,11 +114,10 @@ module Twine
|
|||
end
|
||||
|
||||
def format_value(value)
|
||||
value = value.dup
|
||||
# Android enforces the following rules on the values
|
||||
# 1) apostrophes and quotes must be escaped with a backslash
|
||||
value = escape_quotes(value)
|
||||
value.gsub!("'", "\\\\'")
|
||||
value.gsub!('"', '\\\\"')
|
||||
# 2) HTML escape the string
|
||||
value = CGI.escapeHTML(value)
|
||||
# 3) fix substitutions (e.g. %s/%@)
|
||||
|
|
|
@ -98,8 +98,8 @@ module Twine
|
|||
"\"%{key}\" = \"%{value}\";\n"
|
||||
end
|
||||
|
||||
def format_comment(comment)
|
||||
"/* #{comment.gsub('*/', '* /')} */"
|
||||
def format_comment(row, lang)
|
||||
"/* #{row.comment.gsub('*/', '* /')} */\n" if row.comment
|
||||
end
|
||||
|
||||
def format_key(key)
|
||||
|
@ -109,11 +109,6 @@ module Twine
|
|||
def format_value(value)
|
||||
escape_quotes(value)
|
||||
end
|
||||
|
||||
def escape_quotes(text)
|
||||
text.gsub('"', '\\\\"')
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -90,53 +90,43 @@ module Twine
|
|||
end
|
||||
end
|
||||
|
||||
def write_file(path, lang)
|
||||
default_lang = @strings.language_codes[0]
|
||||
encoding = @options[:output_encoding] || 'UTF-8'
|
||||
File.open(path, "w:#{encoding}") do |f|
|
||||
f.puts "##\n # Django Strings File\n # Generated by Twine #{Twine::VERSION}\n # Language: #{lang}\n "
|
||||
@strings.sections.each do |section|
|
||||
printed_section = false
|
||||
section.rows.each do |row|
|
||||
if row.matches_tags?(@options[:tags], @options[:untagged])
|
||||
f.puts ''
|
||||
if !printed_section
|
||||
if section.name && section.name.length > 0
|
||||
f.print "#--------- #{section.name} ---------#\n\n"
|
||||
end
|
||||
printed_section = true
|
||||
end
|
||||
|
||||
basetrans = row.translated_string_for_lang(default_lang)
|
||||
def format_file(strings, lang)
|
||||
@default_lang = strings.language_codes[0]
|
||||
super
|
||||
end
|
||||
|
||||
key = row.key
|
||||
key = key.gsub('"', '\\\\"')
|
||||
def format_header(lang)
|
||||
"##\n # Django Strings File\n # Generated by Twine #{Twine::VERSION}\n # Language: #{lang}\n"
|
||||
end
|
||||
|
||||
value = row.translated_string_for_lang(lang, default_lang)
|
||||
if value
|
||||
value = value.gsub('"', '\\\\"')
|
||||
def format_section_header(section)
|
||||
"#--------- #{section.name} ---------#\n"
|
||||
end
|
||||
|
||||
comment = row.comment
|
||||
def row_pattern
|
||||
"%{comment}%{base_translation}%{key_value}"
|
||||
end
|
||||
|
||||
if comment
|
||||
comment = comment.gsub('"', '\\\\"')
|
||||
end
|
||||
def format_base_translation(row, lang)
|
||||
base_translation = row.translations[@default_lang]
|
||||
"# base translation: \"#{base_translation}\"\n" if base_translation
|
||||
end
|
||||
|
||||
if comment && comment.length > 0
|
||||
f.print "#. #{comment} \n"
|
||||
end
|
||||
def key_value_pattern
|
||||
"msgid \"%{key}\"\n" +
|
||||
"msgstr \"%{value}\"\n"
|
||||
end
|
||||
|
||||
if basetrans && basetrans.length > 0
|
||||
f.print "# base translation: \"#{basetrans}\"\n"
|
||||
end
|
||||
def format_comment(row, lang)
|
||||
"#. #{escape_quotes(row.comment)}\n" if row.comment
|
||||
end
|
||||
|
||||
f.print "msgid \"#{key}\"\n"
|
||||
f.print "msgstr \"#{value}\"\n"
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
def format_key(key)
|
||||
escape_quotes(key)
|
||||
end
|
||||
|
||||
def format_value(value)
|
||||
escape_quotes(value)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -51,7 +51,7 @@ module Twine
|
|||
match = /((?:[^"\\]|\\.)+)\s*=\s*((?:[^"\\]|\\.)*)/.match(line)
|
||||
if match
|
||||
key = match[1]
|
||||
value = match[2]
|
||||
value = match[2].strip
|
||||
value.gsub!(/\{[0-9]\}/, '%@')
|
||||
set_translation_for_key(key, lang, value)
|
||||
if last_comment
|
||||
|
@ -70,40 +70,25 @@ module Twine
|
|||
end
|
||||
end
|
||||
|
||||
def write_file(path, lang)
|
||||
default_lang = @strings.language_codes[0]
|
||||
encoding = @options[:output_encoding] || 'UTF-8'
|
||||
File.open(path, "w:#{encoding}") do |f|
|
||||
f.puts "## Flash Strings File\n## Generated by Twine #{Twine::VERSION}\n## Language: #{lang}\n"
|
||||
@strings.sections.each do |section|
|
||||
printed_section = false
|
||||
section.rows.each do |row|
|
||||
if row.matches_tags?(@options[:tags], @options[:untagged])
|
||||
f.puts ''
|
||||
if !printed_section
|
||||
if section.name && section.name.length > 0
|
||||
f.print "## #{section.name} ##\n\n"
|
||||
end
|
||||
printed_section = true
|
||||
end
|
||||
def format_header(lang)
|
||||
"## Flash Strings File\n## Generated by Twine #{Twine::VERSION}\n## Language: #{lang}"
|
||||
end
|
||||
|
||||
key = row.key
|
||||
value = row.translated_string_for_lang(lang, default_lang)
|
||||
if value
|
||||
placeHolderNumber = -1
|
||||
value = value.gsub(/%[d@]/) { placeHolderNumber += 1; '{%d}' % placeHolderNumber }
|
||||
|
||||
comment = row.comment
|
||||
if comment && comment.length > 0
|
||||
f.print "# #{comment}\n"
|
||||
end
|
||||
def format_section_header(section)
|
||||
"## #{section.name} ##\n"
|
||||
end
|
||||
|
||||
f.print "#{key}=#{value}"
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
def format_comment(row, lang)
|
||||
"# #{row.comment}\n" if row.comment
|
||||
end
|
||||
|
||||
def key_value_pattern
|
||||
"%{key}=%{value}"
|
||||
end
|
||||
|
||||
def format_value(value)
|
||||
placeHolderNumber = -1
|
||||
value.gsub(/%[d@]/) { placeHolderNumber += 1; '{%d}' % placeHolderNumber }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -60,50 +60,43 @@ module Twine
|
|||
end
|
||||
end
|
||||
|
||||
def write_file(path, lang)
|
||||
default_lang = @strings.language_codes[0]
|
||||
encoding = @options[:output_encoding] || 'UTF-8'
|
||||
File.open(path, "w:#{encoding}") do |f|
|
||||
f.puts "msgid \"\"\nmsgstr \"\"\n\"Language: #{lang}\\n\"\n\"X-Generator: Twine #{Twine::VERSION}\\n\"\n\n"
|
||||
@strings.sections.each do |section|
|
||||
printed_section = false
|
||||
section.rows.each do |row|
|
||||
if row.matches_tags?(@options[:tags], @options[:untagged])
|
||||
if !printed_section
|
||||
f.puts ''
|
||||
if section.name && section.name.length > 0
|
||||
section_name = section.name.gsub('--', '—')
|
||||
f.puts "# SECTION: #{section_name}"
|
||||
end
|
||||
printed_section = true
|
||||
end
|
||||
def format_file(strings, lang)
|
||||
@default_lang = strings.language_codes[0]
|
||||
super
|
||||
end
|
||||
|
||||
basetrans = row.translated_string_for_lang(default_lang)
|
||||
def format_header(lang)
|
||||
"msgid \"\"\nmsgstr \"\"\n\"Language: #{lang}\\n\"\n\"X-Generator: Twine #{Twine::VERSION}\\n\"\n"
|
||||
end
|
||||
|
||||
if basetrans
|
||||
key = row.key
|
||||
key = key.gsub('"', '\\\\"')
|
||||
def format_section_header(section)
|
||||
"# SECTION: #{section.name}"
|
||||
end
|
||||
|
||||
comment = row.comment
|
||||
if comment
|
||||
comment = comment.gsub('"', '\\\\"')
|
||||
end
|
||||
def row_pattern
|
||||
"%{comment}%{key}%{base_translation}%{value}"
|
||||
end
|
||||
|
||||
if comment && comment.length > 0
|
||||
f.print "#. \"#{comment}\"\n"
|
||||
end
|
||||
def format_row(row, lang)
|
||||
return nil unless row.translated_string_for_lang(@default_lang)
|
||||
|
||||
f.print "msgctxt \"#{key}\"\nmsgid \"#{basetrans}\"\n"
|
||||
value = row.translated_string_for_lang(lang)
|
||||
if value
|
||||
value = value.gsub('"', '\\\\"')
|
||||
end
|
||||
f.print "msgstr \"#{value}\"\n\n"
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
super
|
||||
end
|
||||
|
||||
def format_comment(row, lang)
|
||||
"#. \"#{escape_quotes(row.comment)}\"\n" if row.comment
|
||||
end
|
||||
|
||||
def format_key(row, lang)
|
||||
"msgctxt \"#{row.key.dup}\"\n"
|
||||
end
|
||||
|
||||
def format_base_translation(row, lang)
|
||||
"msgid \"#{row.translations[@default_lang]}\"\n"
|
||||
end
|
||||
|
||||
def format_value(row, lang)
|
||||
"msgstr \"#{row.translated_string_for_lang(lang)}\"\n"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -45,46 +45,45 @@ module Twine
|
|||
end
|
||||
end
|
||||
|
||||
def format_file(strings, lang)
|
||||
"{\n#{super}\n}"
|
||||
end
|
||||
|
||||
def format_sections(strings, lang)
|
||||
sections = strings.sections.map { |section| format_section(section, lang) }
|
||||
sections.join(",\n\n")
|
||||
end
|
||||
|
||||
def format_section_header(section)
|
||||
end
|
||||
|
||||
def format_section(section, lang)
|
||||
rows = section.rows.dup
|
||||
|
||||
rows.map! { |row| format_row(row, lang) }
|
||||
rows.compact! # remove nil entries
|
||||
rows.join(",\n")
|
||||
end
|
||||
|
||||
def key_value_pattern
|
||||
"\"%{key}\":\"%{value}\""
|
||||
end
|
||||
|
||||
def format_key(key)
|
||||
escape_quotes(key)
|
||||
end
|
||||
|
||||
def format_value(value)
|
||||
escape_quotes(value)
|
||||
end
|
||||
|
||||
def write_file(path, lang)
|
||||
begin
|
||||
require "json"
|
||||
rescue LoadError
|
||||
raise Twine::Error.new "You must run 'gem install json' in order to read or write jquery-localize files."
|
||||
end
|
||||
|
||||
printed_string = false
|
||||
default_lang = @strings.language_codes[0]
|
||||
encoding = @options[:output_encoding] || 'UTF-8'
|
||||
File.open(path, "w:#{encoding}") do |f|
|
||||
f.print "{"
|
||||
|
||||
@strings.sections.each_with_index do |section, si|
|
||||
printed_section = false
|
||||
section.rows.each_with_index do |row, ri|
|
||||
if row.matches_tags?(@options[:tags], @options[:untagged])
|
||||
if printed_string
|
||||
f.print ",\n"
|
||||
end
|
||||
|
||||
if !printed_section
|
||||
f.print "\n"
|
||||
printed_section = true
|
||||
end
|
||||
|
||||
key = row.key
|
||||
key = key.gsub('"', '\\\\"')
|
||||
|
||||
value = row.translated_string_for_lang(lang, default_lang)
|
||||
value = value.gsub('"', '\\\\"')
|
||||
|
||||
f.print "\"#{key}\":\"#{value}\""
|
||||
printed_string = true
|
||||
end
|
||||
end
|
||||
end
|
||||
f.puts "\n}"
|
||||
|
||||
end
|
||||
super
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -111,64 +111,45 @@ module Twine
|
|||
end
|
||||
end
|
||||
|
||||
def write_file(path, lang)
|
||||
default_lang = nil
|
||||
if DEFAULT_LANG_CODES.has_key?(lang)
|
||||
default_lang = DEFAULT_LANG_CODES[lang]
|
||||
end
|
||||
File.open(path, 'w:UTF-8') do |f|
|
||||
f.puts "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<!-- Tizen Strings File -->\n<!-- Generated by Twine #{Twine::VERSION} -->\n<!-- Language: #{lang} -->"
|
||||
f.write '<string_table Bversion="2.0.0.201311071819" Dversion="20120315">'
|
||||
@strings.sections.each do |section|
|
||||
printed_section = false
|
||||
section.rows.each do |row|
|
||||
if row.matches_tags?(@options[:tags], @options[:untagged])
|
||||
if !printed_section
|
||||
f.puts ''
|
||||
if section.name && section.name.length > 0
|
||||
section_name = section.name.gsub('--', '—')
|
||||
f.puts "\t<!-- SECTION: #{section_name} -->"
|
||||
end
|
||||
printed_section = true
|
||||
end
|
||||
def format_header(lang)
|
||||
"<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<!-- Tizen Strings File -->\n<!-- Generated by Twine #{Twine::VERSION} -->\n<!-- Language: #{lang} -->"
|
||||
end
|
||||
|
||||
key = row.key
|
||||
def format_sections(strings, lang)
|
||||
result = '<string_table Bversion="2.0.0.201311071819" Dversion="20120315">'
|
||||
|
||||
result += super + "\n"
|
||||
|
||||
value = row.translated_string_for_lang(lang, default_lang)
|
||||
if !value && !@options[:exclude_untranslated]
|
||||
value = row.translated_string_for_lang(@strings.language_codes[0])
|
||||
end
|
||||
result += '</string_table>'
|
||||
end
|
||||
|
||||
if value # if values is nil, there was no appropriate translation, so let Tizen handle the defaulting
|
||||
value = String.new(value) # use a copy to prevent modifying the original
|
||||
def format_section_header(section)
|
||||
"\t<!-- SECTION: #{section.name} -->"
|
||||
end
|
||||
|
||||
# Tizen enforces the following rules on the values
|
||||
# 1) apostrophes and quotes must be escaped with a backslash
|
||||
value.gsub!('\'', '\\\\\'')
|
||||
value.gsub!('"', '\\\\"')
|
||||
# 2) HTML escape the string
|
||||
value = CGI.escapeHTML(value)
|
||||
# 3) fix substitutions (e.g. %s/%@)
|
||||
value = androidify_substitutions(value)
|
||||
# 4) replace beginning and end spaces with \0020. Otherwise Tizen strips them.
|
||||
value.gsub!(/\A *| *\z/) { |spaces| '\u0020' * spaces.length }
|
||||
def format_comment(row, lang)
|
||||
"\t<!-- #{row.comment.gsub('--', '—')} -->\n" if row.comment
|
||||
end
|
||||
|
||||
comment = row.comment
|
||||
if comment
|
||||
comment = comment.gsub('--', '—')
|
||||
end
|
||||
def key_value_pattern
|
||||
"\t<text id=\"IDS_%{key}\">%{value}</text>"
|
||||
end
|
||||
|
||||
if comment && comment.length > 0
|
||||
f.puts "\t<!-- #{comment} -->\n"
|
||||
end
|
||||
f.puts "\t<text id=\"IDS_#{key.upcase}\">#{value}</text>"
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
def format_key(key)
|
||||
key.upcase
|
||||
end
|
||||
|
||||
f.puts '</string_table>'
|
||||
end
|
||||
def format_value(value)
|
||||
value = escape_quotes(value)
|
||||
# Tizen enforces the following rules on the values
|
||||
# 1) apostrophes and quotes must be escaped with a backslash
|
||||
value.gsub!("'", "\\\\'")
|
||||
# 2) HTML escape the string
|
||||
value = CGI.escapeHTML(value)
|
||||
# 3) fix substitutions (e.g. %s/%@)
|
||||
value = androidify_substitutions(value)
|
||||
# 4) replace beginning and end spaces with \0020. Otherwise Tizen strips them.
|
||||
value.gsub(/\A *| *\z/) { |spaces| '\u0020' * spaces.length }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
28
test/fixtures/formatter_django.po
vendored
Normal file
28
test/fixtures/formatter_django.po
vendored
Normal file
|
@ -0,0 +1,28 @@
|
|||
##
|
||||
# Django Strings File
|
||||
# Generated by Twine 0.7.0
|
||||
# Language: en
|
||||
|
||||
|
||||
#--------- Section 1 ---------#
|
||||
|
||||
#. comment key1
|
||||
# base translation: "value1-english"
|
||||
msgid "key1"
|
||||
msgstr "value1-english"
|
||||
|
||||
# base translation: "value2-english"
|
||||
msgid "key2"
|
||||
msgstr "value2-english"
|
||||
|
||||
|
||||
#--------- Section 2 ---------#
|
||||
|
||||
# base translation: "value3-english"
|
||||
msgid "key3"
|
||||
msgstr "value3-english"
|
||||
|
||||
#. comment key4
|
||||
# base translation: "value4-english"
|
||||
msgid "key4"
|
||||
msgstr "value4-english"
|
15
test/fixtures/formatter_flash.properties
vendored
Normal file
15
test/fixtures/formatter_flash.properties
vendored
Normal file
|
@ -0,0 +1,15 @@
|
|||
## Flash Strings File
|
||||
## Generated by Twine 0.7.0
|
||||
## Language: en
|
||||
|
||||
## Section 1 ##
|
||||
|
||||
# comment key1
|
||||
key1=value1-english
|
||||
key2=value2-english
|
||||
|
||||
## Section 2 ##
|
||||
|
||||
key3=value3-english
|
||||
# comment key4
|
||||
key4=value4-english
|
1
test/fixtures/formatter_gettext.po
vendored
1
test/fixtures/formatter_gettext.po
vendored
|
@ -24,4 +24,3 @@ msgstr "value3-english"
|
|||
msgctxt "key4"
|
||||
msgid "value4-english"
|
||||
msgstr "value4-english"
|
||||
|
||||
|
|
|
@ -200,3 +200,43 @@ class TestTizenFormatter < FormatterTest
|
|||
end
|
||||
|
||||
end
|
||||
|
||||
class TestDjangoFormatter < FormatterTest
|
||||
def setup
|
||||
super Twine::Formatters::Django
|
||||
end
|
||||
|
||||
def test_read_file_format
|
||||
@formatter.read_file fixture('formatter_django.po'), 'en'
|
||||
|
||||
1.upto(4) do |i|
|
||||
assert_equal "value#{i}-english", @strings.strings_map["key#{i}"].translations['en']
|
||||
end
|
||||
end
|
||||
|
||||
def test_write_file_output_format
|
||||
formatter = Twine::Formatters::Django.new @twine_file, {}
|
||||
formatter.write_file @output_path, 'en'
|
||||
assert_equal content('formatter_django.po'), output_content
|
||||
end
|
||||
end
|
||||
|
||||
class TestFlashFormatter < FormatterTest
|
||||
def setup
|
||||
super Twine::Formatters::Flash
|
||||
end
|
||||
|
||||
def test_read_file_format
|
||||
@formatter.read_file fixture('formatter_flash.properties'), 'en'
|
||||
|
||||
1.upto(4) do |i|
|
||||
assert_equal "value#{i}-english", @strings.strings_map["key#{i}"].translations['en']
|
||||
end
|
||||
end
|
||||
|
||||
def test_write_file_output_format
|
||||
formatter = Twine::Formatters::Flash.new @twine_file, {}
|
||||
formatter.write_file @output_path, 'en'
|
||||
assert_equal content('formatter_flash.properties'), output_content
|
||||
end
|
||||
end
|
||||
|
|
Reference in a new issue