1 /*
2  * Copyright (C) 2012 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.recovery_l10n;
18 
19 import android.app.Activity;
20 import android.content.Context;
21 import android.content.Intent;
22 import android.content.res.AssetManager;
23 import android.content.res.Configuration;
24 import android.content.res.Resources;
25 import android.graphics.Bitmap;
26 import android.os.Bundle;
27 import android.os.RemoteException;
28 import android.util.DisplayMetrics;
29 import android.util.Log;
30 import android.view.View;
31 import android.widget.Button;
32 import android.widget.TextView;
33 import android.widget.Spinner;
34 import android.widget.ArrayAdapter;
35 import android.widget.AdapterView;
36 
37 import java.io.FileOutputStream;
38 import java.io.IOException;
39 import java.util.ArrayList;
40 import java.util.Arrays;
41 import java.util.Comparator;
42 import java.util.HashMap;
43 import java.util.Locale;
44 
45 /**
46  * This activity assists in generating the specially-formatted bitmaps
47  * of text needed for recovery's localized text display.  Each image
48  * contains all the translations of a single string; above each
49  * translation is a "header row" that encodes that subimage's width,
50  * height, and locale using pixel values.
51  *
52  * To use this app to generate new translations:
53  *
54  *   - Update the string resources in res/values-*
55  *
56  *   - Build and run the app.  Select the string you want to
57  *     translate, and press the "Go" button.
58  *
59  *   - Wait for it to finish cycling through all the strings, then
60  *     pull /data/data/com.android.recovery_l10n/files/text-out.png
61  *     from the device.
62  *
63  *   - "pngcrush -c 0 text-out.png output.png"
64  *
65  *   - Put output.png in bootable/recovery/res/images/ (renamed
66  *     appropriately).
67  *
68  * Recovery expects 8-bit 1-channel images (white text on black
69  * background).  pngcrush -c 0 will convert the output of this program
70  * to such an image.  If you use any other image handling tools,
71  * remember that they must be lossless to preserve the exact values of
72  * pixels in the header rows; don't convert them to jpeg or anything.
73  */
74 
75 public class Main extends Activity {
76     private static final String TAG = "RecoveryL10N";
77 
78     HashMap<Locale, Bitmap> savedBitmaps;
79     TextView mText;
80     int mStringId = R.string.recovery_installing;
81 
82     public class TextCapture implements Runnable {
83         private Locale nextLocale;
84         private Locale thisLocale;
85         private Runnable next;
86 
TextCapture(Locale thisLocale, Locale nextLocale, Runnable next)87         TextCapture(Locale thisLocale, Locale nextLocale, Runnable next) {
88             this.nextLocale = nextLocale;
89             this.thisLocale = thisLocale;
90             this.next = next;
91         }
92 
run()93         public void run() {
94             Bitmap b = mText.getDrawingCache();
95             savedBitmaps.put(thisLocale, b.copy(Bitmap.Config.ARGB_8888, false));
96 
97             if (nextLocale != null) {
98                 switchTo(nextLocale);
99             }
100 
101             if (next != null) {
102                 mText.postDelayed(next, 200);
103             }
104         }
105     }
106 
switchTo(Locale locale)107     private void switchTo(Locale locale) {
108         Resources standardResources = getResources();
109         AssetManager assets = standardResources.getAssets();
110         DisplayMetrics metrics = standardResources.getDisplayMetrics();
111         Configuration config = new Configuration(standardResources.getConfiguration());
112         config.locale = locale;
113         Resources defaultResources = new Resources(assets, metrics, config);
114 
115         mText.setText(mStringId);
116 
117         mText.setDrawingCacheEnabled(false);
118         mText.setDrawingCacheEnabled(true);
119         mText.setDrawingCacheQuality(View.DRAWING_CACHE_QUALITY_HIGH);
120     }
121 
122     @Override
onCreate(Bundle savedInstance)123     public void onCreate(Bundle savedInstance) {
124         super.onCreate(savedInstance);
125         setContentView(R.layout.main);
126 
127         savedBitmaps = new HashMap<Locale, Bitmap>();
128 
129         Spinner spinner = (Spinner) findViewById(R.id.which);
130         ArrayAdapter<CharSequence> adapter = ArrayAdapter.createFromResource(
131             this, R.array.string_options, android.R.layout.simple_spinner_item);
132         adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
133         spinner.setAdapter(adapter);
134         spinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
135             @Override
136             public void onItemSelected(AdapterView parent, View view,
137                                        int pos, long id) {
138                 switch (pos) {
139                     case 0: mStringId = R.string.recovery_installing; break;
140                     case 1: mStringId = R.string.recovery_erasing; break;
141                     case 2: mStringId = R.string.recovery_no_command; break;
142                     case 3: mStringId = R.string.recovery_error; break;
143                     case 4: mStringId = R.string.recovery_installing_security; break;
144                 }
145             }
146             @Override public void onNothingSelected(AdapterView parent) { }
147             });
148 
149         mText = (TextView) findViewById(R.id.text);
150 
151         String[] localeNames = getAssets().getLocales();
152         Arrays.sort(localeNames, new Comparator<String>() {
153         // Override the string comparator so that en is sorted behind en_US.
154         // As a result, en_US will be matched first in recovery.
155             @Override
156             public int compare(String s1, String s2) {
157                 if (s1.equals(s2)) {
158                     return 0;
159                 } else if (s1.startsWith(s2)) {
160                     return -1;
161                 } else if (s2.startsWith(s1)) {
162                     return 1;
163                 }
164                 return s1.compareTo(s2);
165             }
166         });
167 
168         ArrayList<Locale> locales = new ArrayList<Locale>();
169         for (String localeName : localeNames) {
170             Log.i(TAG, "locale = " + localeName);
171             if (!localeName.isEmpty()) {
172                 locales.add(Locale.forLanguageTag(localeName));
173             }
174         }
175 
176         final Runnable seq = buildSequence(locales.toArray(new Locale[0]));
177 
178         Button b = (Button) findViewById(R.id.go);
179         b.setOnClickListener(new View.OnClickListener() {
180             @Override
181             public void onClick(View ignore) {
182                 mText.post(seq);
183             }
184             });
185     }
186 
buildSequence(final Locale[] locales)187     private Runnable buildSequence(final Locale[] locales) {
188         Runnable head = new Runnable() { public void run() { mergeBitmaps(locales); } };
189         Locale prev = null;
190         for (Locale loc : locales) {
191             head = new TextCapture(loc, prev, head);
192             prev = loc;
193         }
194         final Runnable fhead = head;
195         final Locale floc = prev;
196         return new Runnable() { public void run() { startSequence(fhead, floc); } };
197     }
198 
199     private void startSequence(Runnable firstRun, Locale firstLocale) {
200         savedBitmaps.clear();
201         switchTo(firstLocale);
202         mText.postDelayed(firstRun, 200);
203     }
204 
205     private void saveBitmap(Bitmap b, String filename) {
206         try {
207             FileOutputStream fos = openFileOutput(filename, 0);
208             b.compress(Bitmap.CompressFormat.PNG, 100, fos);
209             fos.close();
210         } catch (IOException e) {
211             Log.i(TAG, "failed to write PNG", e);
212         }
213     }
214 
215     private int colorFor(byte b) {
216         return 0xff000000 | (b<<16) | (b<<8) | b;
217     }
218 
219     private int colorFor(int b) {
220         return 0xff000000 | (b<<16) | (b<<8) | b;
221     }
222 
223     private void mergeBitmaps(final Locale[] locales) {
224         HashMap<String, Integer> countByLanguage = new HashMap<String, Integer>();
225 
226         int height = 2;
227         int width = 10;
228         int maxHeight = 0;
229         for (Locale loc : locales) {
230             Bitmap b = savedBitmaps.get(loc);
231             int h = b.getHeight();
232             int w = b.getWidth();
233             height += h+1;
234             if (h > maxHeight) maxHeight = h;
235             if (w > width) width = w;
236 
237             String lang = loc.getLanguage();
238             if (countByLanguage.containsKey(lang)) {
239                 countByLanguage.put(lang, countByLanguage.get(lang)+1);
240             } else {
241                 countByLanguage.put(lang, 1);
242             }
243         }
244 
245         Log.i(TAG, "output bitmap is " + width + " x " + height);
246         Bitmap out = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
247         out.eraseColor(0xff000000);
248         int[] pixels = new int[maxHeight * width];
249 
250         int p = 0;
251         for (Locale loc : locales) {
252             Bitmap bm = savedBitmaps.get(loc);
253             int h = bm.getHeight();
254             int w = bm.getWidth();
255 
256             bm.getPixels(pixels, 0, w, 0, 0, w, h);
257 
258             // Find the rightmost and leftmost columns with any
259             // nonblack pixels; we'll copy just that region to the
260             // output image.
261 
262             int right = w;
263             while (right > 1) {
264                 boolean all_black = true;
265                 for (int j = 0; j < h; ++j) {
266                     if (pixels[j*w+right-1] != 0xff000000) {
267                         all_black = false;
268                         break;
269                     }
270                 }
271                 if (all_black) {
272                     --right;
273                 } else {
274                     break;
275                 }
276             }
277 
278             int left = 0;
279             while (left < right-1) {
280                 boolean all_black = true;
281                 for (int j = 0; j < h; ++j) {
282                     if (pixels[j*w+left] != 0xff000000) {
283                         all_black = false;
284                         break;
285                     }
286                 }
287                 if (all_black) {
288                     ++left;
289                 } else {
290                     break;
291                 }
292             }
293 
294             // Make the last country variant for a given language be
295             // the catch-all for that language (because recovery will
296             // take the first one that matches).
297             String lang = loc.getLanguage();
298             if (countByLanguage.get(lang) > 1) {
299                 countByLanguage.put(lang, countByLanguage.get(lang)-1);
300                 lang = loc.toString();
301             }
302             int tw = right - left;
303             Log.i(TAG, "encoding \"" + loc + "\" as \"" + lang + "\": " + tw + " x " + h);
304             byte[] langBytes = lang.getBytes();
305             out.setPixel(0, p, colorFor(tw & 0xff));
306             out.setPixel(1, p, colorFor(tw >>> 8));
307             out.setPixel(2, p, colorFor(h & 0xff));
308             out.setPixel(3, p, colorFor(h >>> 8));
309             out.setPixel(4, p, colorFor(langBytes.length));
310             int x = 5;
311             for (byte b : langBytes) {
312                 out.setPixel(x, p, colorFor(b));
313                 x++;
314             }
315             out.setPixel(x, p, colorFor(0));
316 
317             p++;
318 
319             out.setPixels(pixels, left, w, 0, p, tw, h);
320             p += h;
321         }
322 
323         // if no languages match, suppress text display by using a
324         // single black pixel as the image.
325         out.setPixel(0, p, colorFor(1));
326         out.setPixel(1, p, colorFor(0));
327         out.setPixel(2, p, colorFor(1));
328         out.setPixel(3, p, colorFor(0));
329         out.setPixel(4, p, colorFor(0));
330         p++;
331 
332         saveBitmap(out, "text-out.png");
333         Log.i(TAG, "wrote text-out.png");
334     }
335 }
336