Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(167)

Side by Side Diff: third_party/libaddressinput/chromium/cpp/src/rule.cc

Issue 106763007: [rac] Parse postal code formats and required fields in libaddressinput. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Fix compile on windows. Created 7 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (C) 2013 Google Inc. 1 // Copyright (C) 2013 Google Inc.
2 // 2 //
3 // Licensed under the Apache License, Version 2.0 (the "License"); 3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License. 4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at 5 // You may obtain a copy of the License at
6 // 6 //
7 // http://www.apache.org/licenses/LICENSE-2.0 7 // http://www.apache.org/licenses/LICENSE-2.0
8 // 8 //
9 // Unless required by applicable law or agreed to in writing, software 9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS, 10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and 12 // See the License for the specific language governing permissions and
13 // limitations under the License. 13 // limitations under the License.
14 14
15 #include "rule.h" 15 #include "rule.h"
16 16
17 #include <libaddressinput/address_field.h> 17 #include <libaddressinput/address_field.h>
18 #include <libaddressinput/util/scoped_ptr.h> 18 #include <libaddressinput/util/scoped_ptr.h>
19 19
20 #include <map> 20 #include <cassert>
21 #include <cstddef>
21 #include <string> 22 #include <string>
22 #include <utility> 23 #include <vector>
23 24
24 #include "address_field_util.h"
25 #include "grit.h" 25 #include "grit.h"
26 #include "messages.h" 26 #include "messages.h"
27 #include "util/json.h" 27 #include "util/json.h"
28 #include "util/string_split.h" 28 #include "util/string_split.h"
29 29
30 namespace i18n { 30 namespace i18n {
31 namespace addressinput { 31 namespace addressinput {
32 32
33 namespace { 33 namespace {
34 34
35 typedef std::map<std::string, int> NameMessageIdMap; 35 bool ParseToken(char c, AddressField* field) {
36 36 assert(field != NULL);
37 const char kAdminAreaNameTypeKey[] = "state_name_type"; 37 switch (c) {
38 const char kFormatKey[] = "fmt"; 38 case 'R':
39 const char kLanguageKey[] = "lang"; 39 *field = COUNTRY;
40 const char kLanguagesKey[] = "languages"; 40 return true;
41 const char kPostalCodeNameTypeKey[] = "zip_name_type"; 41 case 'S':
42 const char kSubKeysKey[] = "sub_keys"; 42 *field = ADMIN_AREA;
43 43 return true;
44 // Used as a separator in a list of items. For example, the list of supported 44 case 'C':
45 // languages can be "de~fr~it". 45 *field = LOCALITY;
46 const char kSeparator = '~'; 46 return true;
47 47 case 'D':
48 NameMessageIdMap InitAdminAreaMessageIds() { 48 *field = DEPENDENT_LOCALITY;
49 NameMessageIdMap message_ids; 49 return true;
50 message_ids.insert(std::make_pair( 50 case 'X':
51 "area", IDS_LIBADDRESSINPUT_I18N_AREA)); 51 *field = SORTING_CODE;
52 message_ids.insert(std::make_pair( 52 return true;
53 "county", IDS_LIBADDRESSINPUT_I18N_COUNTY_LABEL)); 53 case 'Z':
54 message_ids.insert(std::make_pair( 54 *field = POSTAL_CODE;
55 "department", IDS_LIBADDRESSINPUT_I18N_DEPARTMENT)); 55 return true;
56 message_ids.insert(std::make_pair( 56 case 'A':
57 "district", IDS_LIBADDRESSINPUT_I18N_DEPENDENT_LOCALITY_LABEL)); 57 *field = STREET_ADDRESS;
58 message_ids.insert(std::make_pair( 58 return true;
59 "do_si", IDS_LIBADDRESSINPUT_I18N_DO_SI)); 59 case 'O':
60 message_ids.insert(std::make_pair( 60 *field = ORGANIZATION;
61 "emirate", IDS_LIBADDRESSINPUT_I18N_EMIRATE)); 61 return true;
62 message_ids.insert(std::make_pair( 62 case 'N':
63 "island", IDS_LIBADDRESSINPUT_I18N_ISLAND)); 63 *field = RECIPIENT;
64 message_ids.insert(std::make_pair( 64 return true;
65 "parish", IDS_LIBADDRESSINPUT_I18N_PARISH)); 65 default:
66 message_ids.insert(std::make_pair( 66 return false;
67 "prefecture", IDS_LIBADDRESSINPUT_I18N_PREFECTURE)); 67 }
68 message_ids.insert(std::make_pair(
69 "province", IDS_LIBADDRESSINPUT_I18N_PROVINCE));
70 message_ids.insert(std::make_pair(
71 "state", IDS_LIBADDRESSINPUT_I18N_STATE_LABEL));
72 return message_ids;
73 } 68 }
74 69
75 const NameMessageIdMap& GetAdminAreaMessageIds() { 70 // Clears |fields|, parses |format|, and adds the format address fields to
76 static const NameMessageIdMap kAdminAreaMessageIds(InitAdminAreaMessageIds()); 71 // |fields|.
77 return kAdminAreaMessageIds; 72 //
73 // For example, the address format in Switzerland is "%O%n%N%n%A%nAX-%Z
74 // %C%nÅLAND". It includes the allowed fields prefixed with %, newlines denoted
75 // %n, and the extra text that should be included on an envelope. This function
76 // parses only the tokens denoted % to determine how an address input form
77 // should be laid out.
78 //
79 // The format string "%O%n%N%n%A%nAX-%Z%C%nÅLAND" is parsed into
80 // {{ORGANIZATION}, {RECIPIENT}, {STREET_ADDRESS}, {POSTAL_CODE, LOCALITY}, {}}.
81 void ParseAddressFieldsFormat(const std::string& format,
82 std::vector<std::vector<AddressField> >* fields) {
83 assert(fields != NULL);
84 fields->clear();
85 fields->resize(1);
86 std::vector<std::string> format_parts;
87 SplitString(format, '%', &format_parts);
88 for (std::vector<std::string>::size_type i = 1;
Evan Stade 2013/12/21 00:27:13 what's wrong with size_t?
please use gerrit instead 2014/01/02 19:49:39 You're right, size_t is better in this case. size_
89 i < format_parts.size(); ++i) {
90 AddressField field = COUNTRY;
91 if (ParseToken(format_parts[i][0], &field)) {
92 fields->back().push_back(field);
93 } else if (format_parts[i][0] == 'n') {
94 fields->push_back(std::vector<AddressField>());
95 }
96 }
78 } 97 }
79 98
80 NameMessageIdMap InitPostalCodeMessageIds() { 99 // Clears |fields|, parses |required|, and adds the required fields to |fields|.
81 NameMessageIdMap message_ids; 100 // For example, parses "SCDX" into {ADMIN_AREA, LOCALITY, DEPENDENT_LOCALITY,
82 message_ids.insert(std::make_pair( 101 // SORTING_CODE}.
83 "postal", IDS_LIBADDRESSINPUT_I18N_POSTAL_CODE_LABEL)); 102 void ParseAddressFieldsRequired(const std::string& required,
84 message_ids.insert(std::make_pair( 103 std::vector<AddressField>* fields) {
85 "zip", IDS_LIBADDRESSINPUT_I18N_ZIP_CODE_LABEL)); 104 assert(fields != NULL);
86 return message_ids; 105 fields->clear();
106 for (std::string::const_iterator token = required.begin();
Evan Stade 2013/12/21 00:27:13 nit: more idiomatic/easier to read imo: for (size
please use gerrit instead 2014/01/02 19:49:39 Done.
107 token != required.end(); ++token) {
108 AddressField field = COUNTRY;
109 if (ParseToken(*token, &field)) {
110 fields->push_back(field);
111 }
112 }
87 } 113 }
88 114
89 const NameMessageIdMap& GetPostalCodeMessageIds() { 115 int GetAdminAreaMessageId(const std::string& admin_area_type) {
90 static const NameMessageIdMap kPostalCodeMessageIds( 116 if (admin_area_type == "area") {
91 InitPostalCodeMessageIds()); 117 return IDS_LIBADDRESSINPUT_I18N_AREA;
92 return kPostalCodeMessageIds; 118 }
119 if (admin_area_type == "county") {
120 return IDS_LIBADDRESSINPUT_I18N_COUNTY_LABEL;
121 }
122 if (admin_area_type == "department") {
123 return IDS_LIBADDRESSINPUT_I18N_DEPARTMENT;
124 }
125 if (admin_area_type == "district") {
126 return IDS_LIBADDRESSINPUT_I18N_DEPENDENT_LOCALITY_LABEL;
127 }
128 if (admin_area_type == "do_si") {
129 return IDS_LIBADDRESSINPUT_I18N_DO_SI;
130 }
131 if (admin_area_type == "emirate") {
132 return IDS_LIBADDRESSINPUT_I18N_EMIRATE;
133 }
134 if (admin_area_type == "island") {
135 return IDS_LIBADDRESSINPUT_I18N_ISLAND;
136 }
137 if (admin_area_type == "parish") {
138 return IDS_LIBADDRESSINPUT_I18N_PARISH;
139 }
140 if (admin_area_type == "prefecture") {
141 return IDS_LIBADDRESSINPUT_I18N_PREFECTURE;
142 }
143 if (admin_area_type == "province") {
144 return IDS_LIBADDRESSINPUT_I18N_PROVINCE;
145 }
146 if (admin_area_type == "state") {
147 return IDS_LIBADDRESSINPUT_I18N_STATE_LABEL;
148 }
149 return INVALID_MESSAGE_ID;
93 } 150 }
94 151
95 int GetMessageIdFromName(const std::string& name, 152 int GetPostalCodeMessageId(const std::string& postal_code_type) {
96 const NameMessageIdMap& message_ids) { 153 if (postal_code_type == "postal") {
97 NameMessageIdMap::const_iterator it = message_ids.find(name); 154 return IDS_LIBADDRESSINPUT_I18N_POSTAL_CODE_LABEL;
98 return it != message_ids.end() ? it->second : INVALID_MESSAGE_ID; 155 }
156 if (postal_code_type == "zip") {
157 return IDS_LIBADDRESSINPUT_I18N_ZIP_CODE_LABEL;
158 }
159 return INVALID_MESSAGE_ID;
99 } 160 }
100 161
101 } // namespace 162 } // namespace
102 163
103 Rule::Rule() 164 Rule::Rule()
104 : format_(), 165 : format_(),
166 required_(),
105 sub_keys_(), 167 sub_keys_(),
106 languages_(), 168 languages_(),
107 language_(), 169 language_(),
170 postal_code_format_(),
108 admin_area_name_message_id_(INVALID_MESSAGE_ID), 171 admin_area_name_message_id_(INVALID_MESSAGE_ID),
109 postal_code_name_message_id_(INVALID_MESSAGE_ID) {} 172 postal_code_name_message_id_(INVALID_MESSAGE_ID) {}
110 173
111 Rule::~Rule() {} 174 Rule::~Rule() {}
112 175
113 void Rule::CopyFrom(const Rule& rule) { 176 void Rule::CopyFrom(const Rule& rule) {
114 format_ = rule.format_; 177 format_ = rule.format_;
178 required_ = rule.required_;
115 sub_keys_ = rule.sub_keys_; 179 sub_keys_ = rule.sub_keys_;
116 languages_ = rule.languages_; 180 languages_ = rule.languages_;
117 language_ = rule.language_; 181 language_ = rule.language_;
182 postal_code_format_ = rule.postal_code_format_;
118 admin_area_name_message_id_ = rule.admin_area_name_message_id_; 183 admin_area_name_message_id_ = rule.admin_area_name_message_id_;
119 postal_code_name_message_id_ = rule.postal_code_name_message_id_; 184 postal_code_name_message_id_ = rule.postal_code_name_message_id_;
120 } 185 }
121 186
122 bool Rule::ParseSerializedRule(const std::string& serialized_rule) { 187 bool Rule::ParseSerializedRule(const std::string& serialized_rule) {
123 scoped_ptr<Json> json(Json::Build()); 188 scoped_ptr<Json> json(Json::Build());
124 if (!json->ParseObject(serialized_rule)) { 189 if (!json->ParseObject(serialized_rule)) {
125 return false; 190 return false;
126 } 191 }
127 192
193 static const char kFormatKey[] = "fmt";
128 if (json->HasStringValueForKey(kFormatKey)) { 194 if (json->HasStringValueForKey(kFormatKey)) {
129 ParseAddressFieldsFormat(json->GetStringValueForKey(kFormatKey), &format_); 195 ParseAddressFieldsFormat(json->GetStringValueForKey(kFormatKey), &format_);
130 } 196 }
131 197
198 static const char kRequiredKey[] = "require";
199 if (json->HasStringValueForKey(kRequiredKey)) {
200 ParseAddressFieldsRequired(
201 json->GetStringValueForKey(kRequiredKey), &required_);
202 }
203
Evan Stade 2013/12/21 00:27:13 extra newline
204
205 // Used as a separator in a list of items. For example, the list of supported
206 // languages can be "de~fr~it".
207 static const char kSeparator = '~';
208 static const char kSubKeysKey[] = "sub_keys";
132 if (json->HasStringValueForKey(kSubKeysKey)) { 209 if (json->HasStringValueForKey(kSubKeysKey)) {
133 SplitString( 210 SplitString(
134 json->GetStringValueForKey(kSubKeysKey), kSeparator, &sub_keys_); 211 json->GetStringValueForKey(kSubKeysKey), kSeparator, &sub_keys_);
135 } 212 }
136 213
214 static const char kLanguagesKey[] = "languages";
137 if (json->HasStringValueForKey(kLanguagesKey)) { 215 if (json->HasStringValueForKey(kLanguagesKey)) {
138 SplitString( 216 SplitString(
139 json->GetStringValueForKey(kLanguagesKey), kSeparator, &languages_); 217 json->GetStringValueForKey(kLanguagesKey), kSeparator, &languages_);
140 } 218 }
141 219
220 static const char kLanguageKey[] = "lang";
142 if (json->HasStringValueForKey(kLanguageKey)) { 221 if (json->HasStringValueForKey(kLanguageKey)) {
143 language_ = json->GetStringValueForKey(kLanguageKey); 222 language_ = json->GetStringValueForKey(kLanguageKey);
144 } 223 }
145 224
146 if (json->HasStringValueForKey(kAdminAreaNameTypeKey)) { 225 static const char kPostalCodeFormatKey[] = "zip";
147 admin_area_name_message_id_ = 226 if (json->HasStringValueForKey(kPostalCodeFormatKey)) {
148 GetMessageIdFromName(json->GetStringValueForKey(kAdminAreaNameTypeKey), 227 postal_code_format_ = json->GetStringValueForKey(kPostalCodeFormatKey);
149 GetAdminAreaMessageIds());
150 } 228 }
151 229
230 static const char kAdminAreaNameTypeKey[] = "state_name_type";
231 if (json->HasStringValueForKey(kAdminAreaNameTypeKey)) {
232 admin_area_name_message_id_ = GetAdminAreaMessageId(
233 json->GetStringValueForKey(kAdminAreaNameTypeKey));
234 }
235
236 static const char kPostalCodeNameTypeKey[] = "zip_name_type";
152 if (json->HasStringValueForKey(kPostalCodeNameTypeKey)) { 237 if (json->HasStringValueForKey(kPostalCodeNameTypeKey)) {
153 postal_code_name_message_id_ = 238 postal_code_name_message_id_ = GetPostalCodeMessageId(
154 GetMessageIdFromName(json->GetStringValueForKey(kPostalCodeNameTypeKey), 239 json->GetStringValueForKey(kPostalCodeNameTypeKey));
155 GetPostalCodeMessageIds());
156 } 240 }
157 241
158 return true; 242 return true;
159 } 243 }
160 244
161 } // namespace addressinput 245 } // namespace addressinput
162 } // namespace i18n 246 } // namespace i18n
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698