From e10619ef737d3adc60e19bf5fb224dc4c0d54055 Mon Sep 17 00:00:00 2001 From: Robert Papp Date: Sat, 14 May 2016 20:30:54 +0200 Subject: [PATCH] bumptech/glide#556 Data Uri / Base64 from JSON --- .../github/_556_data_uri/TestFragment.java | 40 -------- .../FirebaseGlideModule.java | 4 +- .../FirebaseImage.java | 2 +- .../FirebaseModelLoader.java | 2 +- .../_556_data_uri_firebase/TestFragment.java | 23 +++++ .../Base64StreamDataFetcher.java | 36 +++++++ .../DataUriGlideModule.java | 18 ++++ .../DataUriModelLoader.java | 38 ++++++++ .../TestFragment.java | 24 +++++ .../github/_556_data_uri_via_POST/Image.java | 34 +++++++ .../ImageModelLoader.java | 26 +++++ .../JSONImageFetcher.java | 97 +++++++++++++++++++ .../RemoteImageGlideModule.java | 18 ++++ .../_556_data_uri_via_POST/TestFragment.java | 23 +++++ src/main/res/values/strings.xml | 3 + 15 files changed, 344 insertions(+), 44 deletions(-) delete mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/TestFragment.java rename src/glide3/java/com/bumptech/glide/supportapp/github/{_556_data_uri => _556_data_uri_firebase}/FirebaseGlideModule.java (84%) rename src/glide3/java/com/bumptech/glide/supportapp/github/{_556_data_uri => _556_data_uri_firebase}/FirebaseImage.java (95%) rename src/glide3/java/com/bumptech/glide/supportapp/github/{_556_data_uri => _556_data_uri_firebase}/FirebaseModelLoader.java (97%) create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/TestFragment.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/Base64StreamDataFetcher.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/DataUriGlideModule.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/DataUriModelLoader.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/TestFragment.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/Image.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/ImageModelLoader.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/JSONImageFetcher.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/RemoteImageGlideModule.java create mode 100644 src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/TestFragment.java diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/TestFragment.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/TestFragment.java deleted file mode 100644 index b4a18dc..0000000 --- a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/TestFragment.java +++ /dev/null @@ -1,40 +0,0 @@ -package com.bumptech.glide.supportapp.github._556_data_uri; - -import java.io.InputStream; - -import android.content.Context; -import android.os.AsyncTask; - -import com.bumptech.glide.Glide; -import com.bumptech.glide.load.engine.DiskCacheStrategy; -import com.bumptech.glide.supportapp.*; - -public class TestFragment extends GlideImageFragment { - @Override protected void load(final Context context) { - //noinspection unchecked - new AsyncTask() { - @Override protected Object doInBackground(Object[] params) { - Glide.get(context).clearDiskCache(); - return null; - } - @Override protected void onPostExecute(Object result) { - loadImage(context); - } - }.execute(); - } - - private void loadImage(Context context) { - //Glide.setup(new GlideBuilder(context).setBitmapPool(new BitmapPoolAdapter())); - Glide.get(context).register(FirebaseImage.class, InputStream.class, new FirebaseModelLoader.Factory(null)); - Glide - .with(context) - .load(new FirebaseImage( - "")) - .placeholder(R.drawable.glide_placeholder) - .error(R.drawable.glide_error) - .diskCacheStrategy(DiskCacheStrategy.NONE) - .skipMemoryCache(true) - .into(imageView); - } -} - diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseGlideModule.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseGlideModule.java similarity index 84% rename from src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseGlideModule.java rename to src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseGlideModule.java index 223ff1b..a2ccae7 100644 --- a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseGlideModule.java +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseGlideModule.java @@ -1,4 +1,4 @@ -package com.bumptech.glide.supportapp.github._556_data_uri; +package com.bumptech.glide.supportapp.github._556_data_uri_firebase; import java.io.InputStream; @@ -9,7 +9,7 @@ import com.firebase.client.Firebase; // TODO https://github.com/bumptech/glide/wiki/Configuration#creating-a-glidemodule -class FirebaseGlideModule implements GlideModule { +public class FirebaseGlideModule implements GlideModule { @Override public void applyOptions(Context context, GlideBuilder builder) { } diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseImage.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseImage.java similarity index 95% rename from src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseImage.java rename to src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseImage.java index 34409c6..c6d9fdf 100644 --- a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseImage.java +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseImage.java @@ -1,4 +1,4 @@ -package com.bumptech.glide.supportapp.github._556_data_uri; +package com.bumptech.glide.supportapp.github._556_data_uri_firebase; import java.util.concurrent.CountDownLatch; import java.util.concurrent.atomic.AtomicReference; diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseModelLoader.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseModelLoader.java similarity index 97% rename from src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseModelLoader.java rename to src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseModelLoader.java index 70c6a4f..6fdb9ab 100644 --- a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri/FirebaseModelLoader.java +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/FirebaseModelLoader.java @@ -1,4 +1,4 @@ -package com.bumptech.glide.supportapp.github._556_data_uri; +package com.bumptech.glide.supportapp.github._556_data_uri_firebase; import java.io.*; diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/TestFragment.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/TestFragment.java new file mode 100644 index 0000000..63b92f7 --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_firebase/TestFragment.java @@ -0,0 +1,23 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_firebase; + +import android.content.Context; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.load.engine.DiskCacheStrategy; +import com.bumptech.glide.supportapp.*; + +public class TestFragment extends GlideImageFragment { + @Override protected void load(Context context) { + //Glide.setup(new GlideBuilder(context).setBitmapPool(new BitmapPoolAdapter())); + //Glide.get(context).register(FirebaseImage.class, InputStream.class, new FirebaseModelLoader.Factory(null)); + Glide + .with(context) + .load(new FirebaseImage("data:image/jpeg;base64," + getString(R.string.glide_base64))) + .placeholder(R.drawable.glide_placeholder) + .error(R.drawable.glide_error) + .diskCacheStrategy(DiskCacheStrategy.NONE) + .skipMemoryCache(true) + .into(imageView); + } +} + diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/Base64StreamDataFetcher.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/Base64StreamDataFetcher.java new file mode 100644 index 0000000..dd2ae38 --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/Base64StreamDataFetcher.java @@ -0,0 +1,36 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_from_string; + +import java.io.*; + +import android.util.Base64; + +import com.bumptech.glide.Priority; +import com.bumptech.glide.load.data.DataFetcher; + +class Base64StreamDataFetcher implements DataFetcher { + private final String base64; + public Base64StreamDataFetcher(String base64) { + this.base64 = base64; + } + @Override public InputStream loadData(Priority priority) throws Exception { + // depending on how you encoded it, the below is just a guess: + // here the full base64 is decoded into bytes and the bytes will be parsed by Glide + // TODO match the flags based on your possible inputs: e.g. add | Base64.URL_SAFE + byte[] raw = Base64.decode(base64, Base64.NO_WRAP); + return new ByteArrayInputStream(raw); + // ---- alternative + // if you don't want to delay decoding you can use something like: + // Base64InputStream (http://stackoverflow.com/a/19981216/253468) + // here the base64 bytes are passed to Base64InputStream and that to Glide + // so base64 decoding will be done later when Glide reads from the stream. + //return new Base64InputStream(new ByteArrayInputStream(base64.getBytes("utf-8")), Base64.NO_WRAP | Base64.URL_SAFE); + } + @Override public String getId() { + // not well-cacheable, I suggest to use .diskCacheStrategy(NONE) + return base64; + } + @Override public void cancel() { + } + @Override public void cleanup() { + } +} diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/DataUriGlideModule.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/DataUriGlideModule.java new file mode 100644 index 0000000..0ac951d --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/DataUriGlideModule.java @@ -0,0 +1,18 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_from_string; + +import java.io.InputStream; + +import android.content.Context; + +import com.bumptech.glide.*; +import com.bumptech.glide.module.GlideModule; + +// TODO https://github.com/bumptech/glide/wiki/Configuration#creating-a-glidemodule +public class DataUriGlideModule implements GlideModule { + @Override public void applyOptions(Context context, GlideBuilder builder) { + + } + @Override public void registerComponents(Context context, Glide glide) { + glide.register(String.class, InputStream.class, new DataUriModelLoader.Factory()); + } +} diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/DataUriModelLoader.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/DataUriModelLoader.java new file mode 100644 index 0000000..e81c112 --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/DataUriModelLoader.java @@ -0,0 +1,38 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_from_string; + +import java.io.InputStream; + +import android.content.Context; + +import com.bumptech.glide.load.data.DataFetcher; +import com.bumptech.glide.load.model.*; +import com.bumptech.glide.load.model.stream.*; + +// TODO read https://github.com/bumptech/glide/wiki/Downloading-custom-sizes-with-Glide +class DataUriModelLoader implements StreamModelLoader { + private final ModelLoader defaultLoader; + public DataUriModelLoader(ModelLoader defaultLoader) { + this.defaultLoader = defaultLoader; + } + + private static final String PREFIX = "data:image/png;base64,"; + @Override public DataFetcher getResourceFetcher(final String model, int width, int height) { + if (model.startsWith(PREFIX)) { // TODO better matching based on your needs + return new Base64StreamDataFetcher(model.substring(PREFIX.length())); + } else { + return defaultLoader.getResourceFetcher(model, width, height); + } + } + + public static class Factory implements ModelLoaderFactory { + public Factory() { + } + @Override public ModelLoader build(Context context, GenericLoaderFactory factories) { + // StreamStringLoader is hardcoded, because we've already overwritten String -> InputStream loader + // This will be better handled in Glide v4. + return new DataUriModelLoader(new StreamStringLoader(context)); + } + @Override public void teardown() { + } + } +} diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/TestFragment.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/TestFragment.java new file mode 100644 index 0000000..e809ed8 --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_from_string/TestFragment.java @@ -0,0 +1,24 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_from_string; + +import android.content.Context; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.load.engine.DiskCacheStrategy; +import com.bumptech.glide.load.resource.drawable.GlideDrawable; +import com.bumptech.glide.supportapp.*; +import com.bumptech.glide.supportapp.utils.LoggingListener; + +public class TestFragment extends GlideImageFragment { + @Override protected void load(final Context context) { + Glide + .with(context) + .load("data:image/png;base64," + getString(R.string.glide_base64)) + .placeholder(R.drawable.glide_placeholder) + .error(R.drawable.glide_error) + .diskCacheStrategy(DiskCacheStrategy.NONE) + .skipMemoryCache(false) + .listener(new LoggingListener()) + .into(imageView); + } +} + diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/Image.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/Image.java new file mode 100644 index 0000000..d9ff737 --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/Image.java @@ -0,0 +1,34 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_via_POST; + +import java.io.IOException; + +import org.json.*; + +public class Image { + private final String key; + public Image(String key) { + this.key = key; + } + + public String getKey() { + return key; + } + + public String getUri() throws IOException { + return "http://httpbin.org/post"; + } + + public JSONObject getPayload() throws IOException { + try { + JSONObject object = new JSONObject(); + object.put("imageKey", key); + return object; + } catch (JSONException e) { + throw new IOException("Invalid implementation", e); + } + } + + @Override public String toString() { + return "Image for key=" + key; + } +} diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/ImageModelLoader.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/ImageModelLoader.java new file mode 100644 index 0000000..041af3e --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/ImageModelLoader.java @@ -0,0 +1,26 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_via_POST; + +import java.io.InputStream; + +import android.content.Context; + +import com.bumptech.glide.load.data.DataFetcher; +import com.bumptech.glide.load.model.*; +import com.bumptech.glide.load.model.stream.StreamModelLoader; + +// TODO read https://github.com/bumptech/glide/wiki/Downloading-custom-sizes-with-Glide +class ImageModelLoader implements StreamModelLoader { + @Override public DataFetcher getResourceFetcher(final Image model, int width, int height) { + return new JSONImageFetcher(model); + } + + public static class Factory implements ModelLoaderFactory { + public Factory() { + } + @Override public ModelLoader build(Context context, GenericLoaderFactory factories) { + return new ImageModelLoader(); + } + @Override public void teardown() { + } + } +} diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/JSONImageFetcher.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/JSONImageFetcher.java new file mode 100644 index 0000000..7afa0fc --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/JSONImageFetcher.java @@ -0,0 +1,97 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_via_POST; + +import java.io.*; +import java.net.*; + +import javax.net.ssl.HttpsURLConnection; + +import org.json.*; + +import android.util.Base64; + +import com.bumptech.glide.Priority; +import com.bumptech.glide.load.data.DataFetcher; +import com.bumptech.glide.supportapp.*; + +class JSONImageFetcher implements DataFetcher { + private final Image model; + + public JSONImageFetcher(Image model) { + this.model = model; + } + + private HttpURLConnection conn; + private InputStream response; + + @Override public InputStream loadData(Priority priority) throws Exception { + // This implementation is totally up to you + // TODO produce an InputStream that contains the raw image bytes that can be decoded (below is just an example) + URL url = new URL(model.getUri()); + conn = (HttpURLConnection)url.openConnection(); + try { + conn.setReadTimeout(15000); + conn.setConnectTimeout(15000); + conn.setRequestMethod("POST"); + conn.setDoInput(true); + conn.setDoOutput(true); + conn.setRequestProperty("Content-Type", "application/json"); + + JSONObject payload = model.getPayload(); + // TODO remove this, only for testing with httpbin to have a proper response + payload.put("imageData", "data:image/png,base64," + App.getInstance().getString(R.string.glide_base64)); + OutputStream request = conn.getOutputStream(); + try { + request.write(payload.toString().getBytes("UTF-8")); + } finally { + request.close(); + } + int responseCode = conn.getResponseCode(); + if (responseCode == HttpsURLConnection.HTTP_OK) { + response = conn.getInputStream(); + try { + JSONObject object = readJSON(response); + // TODO remove this, only needed for testing with httpbin + object = new JSONObject(object.getString("data")); + String dataUri = object.getString("imageData"); + String base64 = dataUri.substring("data:image/png,base64,".length()); + byte[] raw = Base64.decode(base64, Base64.NO_WRAP); + return new ByteArrayInputStream(raw); + } finally { + response.close(); + } + } else { + throw new IOException("Invalid response: " + responseCode); + } + } finally { + conn.disconnect(); + } + } + + private JSONObject readJSON(InputStream stream) throws IOException, JSONException { + BufferedReader reader = new BufferedReader(new InputStreamReader(stream, "UTF-8")); + StringBuilder builder = new StringBuilder(); + String line; + while ((line = reader.readLine()) != null) { + builder.append(line); + } + return new JSONObject(builder.toString()); + } + @Override public void cleanup() { + if (response != null) { + try { + response.close(); + } catch (IOException e) { + // Ignore + } + } + if (conn != null) { + conn.disconnect(); + } + } + @Override public String getId() { + return model.getKey(); + } + @Override public void cancel() { + + } +} diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/RemoteImageGlideModule.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/RemoteImageGlideModule.java new file mode 100644 index 0000000..75972c9 --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/RemoteImageGlideModule.java @@ -0,0 +1,18 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_via_POST; + +import java.io.InputStream; + +import android.content.Context; + +import com.bumptech.glide.*; +import com.bumptech.glide.module.GlideModule; + +// TODO https://github.com/bumptech/glide/wiki/Configuration#creating-a-glidemodule +public class RemoteImageGlideModule implements GlideModule { + @Override public void applyOptions(Context context, GlideBuilder builder) { + + } + @Override public void registerComponents(Context context, Glide glide) { + glide.register(Image.class, InputStream.class, new ImageModelLoader.Factory()); + } +} diff --git a/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/TestFragment.java b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/TestFragment.java new file mode 100644 index 0000000..cfdd8f9 --- /dev/null +++ b/src/glide3/java/com/bumptech/glide/supportapp/github/_556_data_uri_via_POST/TestFragment.java @@ -0,0 +1,23 @@ +package com.bumptech.glide.supportapp.github._556_data_uri_via_POST; + +import android.content.Context; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.load.resource.drawable.GlideDrawable; +import com.bumptech.glide.supportapp.*; +import com.bumptech.glide.supportapp.utils.LoggingListener; + +public class TestFragment extends GlideImageFragment { + @Override protected void load(final Context context) { + Glide + .with(context) + .load(new Image("identifier_referencing_an_image")) + .placeholder(R.drawable.glide_placeholder) + .error(R.drawable.glide_error) + //.diskCacheStrategy(DiskCacheStrategy.NONE) + //.skipMemoryCache(true) + .listener(new LoggingListener()) + .into(imageView); + } +} + diff --git a/src/main/res/values/strings.xml b/src/main/res/values/strings.xml index 89e2d2b..33c7159 100644 --- a/src/main/res/values/strings.xml +++ b/src/main/res/values/strings.xml @@ -4,4 +4,7 @@ Glide Support Glide Quick + + + iVBORw0KGgoAAAANSUhEUgAAAMgAAADICAYAAACtWK6eAAAORElEQVR42u2dXWwc1RmG35lZe/23OMS/67gucRI3gfw4IUBBDrRFjchVxY2rqgqV6AWoFQgVJBBERUUhooKiCFQ1uQChctNGqkDqBVXVShVClVpBhRBtQwgxitI4SaMkTfzvtacXMJu1Z72e2Tlnzplz3ufKcez1zpx553u/b77vLEAIIYQQQgghhBBCCCGEEEIIIYQQQgghhBBCCCGEaIfDU5Ae/S893Rp83bx980TU3yudv/j5/PmLWwHgzBOHJnkmKZBMi6Chp/PjXE/nTWn+3emPjrdRPBSIdqx/8+WxtMVA0VAgFAQFQ4FkhQ1vHfXd1majjmn21OkDpx8+8DxXlwKhKBhZKBBRCXac6pJpLE5O47P7H+I1QYGYk1cwqlAgtFEUCgVCYVAoFAiFQaFQIMwxbODTvQ84FIgBbPrjr31eznKwoepl7MHRTtF2USBVsP1ZBm0XBcKooTGmtbAYIxDmGsxNKJAqsEJFy0WB0FLRclEgtFS0XPrhUhz68Lc7voPv9Q4ad1xuazM2vHXUp0AkM3Dk4DOmimN//xBc38fjN21HsbHZSJFkce3cLIkjPzhw0ERx9OVb8dTQbszNzyPneXhx0+3G2q2siSQTAtnw1lHfVHEAwKvb9qDB8wDfh+842NRcMNJqZVEkbhbEYXKlan//ELbd2AUAWFhchON+kcs+0r/FSKuVNZG4FId6awUAk5NftDI5XxYWPTh4dnAnTCYLInEpDg2sFYCFhQUAQO7LfwPAzra1+PbaPoqEAlnK+jdfHjNdHPd1DZStFQDMzs5eXxT3+rL8dP0wTEdnkWgnkIEjB5+xoXXkha13Lfn34uJi+Wsvlyt/3ei4OLL5LoqEAvmiVd3kalXAGzvvRUuuYcn3Aou1PILYYrV0FYlWArFhjuO+rgHc1Rm+2CsFUu0qscFqBbknBZLRioYMawUApVIJvn/98D0vvCyNjosXN95m/PlxW5ux/s2XxygQC8XxytaRkLUCgKmpqaWL4lRflj1rejBcWGv8ecr1dN5U+VkqVgskq01scdnV3om9vdVrD6VSacm/Had646sD4Bcbb7fhdGljt5UKpP+lp1ttmec4suObcFe48CsrWOW7aEUlq5I2L2eF1dLFWbi8S6Rjrdbkm1b8/7m5udD3vIoHhrZaLR0chmvz3UG1tQqoTNDL33NWni+yyWq5rc1QmY8oEcjAkYPPwBIOb92zorUKqCzxlkXg1v6dNi+HH/dvYT5iokBseBgIAI+t347elraaPxM0KYajxOoTqvt7Nxjd8auD1UpdILZYq758Kx7esH3Vn6sWPYClTYu1rNZrN4/QapkiEF1q22nwm917V7VWwNImxdDiuKsvT0cuT6tlikBsqVpFsVYB1Uq8Ad4KpV6brVbaT9lTE4gtDwSjWqvVLFbUCBJYrV9uvtMKgaTd6Z2aQGx5IPjqtj2RrFUUgcS5o6xrbDF6jl1VHuuadkAqqZwvj8LyJsWQxfLiLY/pc+xKigM8BeKsVTBfHpXlTYqhxXHiLY8Hx+gtg1TcdF1TDkQHa9UQoTS7PILUzC2c+Lt1mr5lUCVpVEUZQRRYq4BaFaxyUhqxkmWj1UqjKipVILZEj8c37arr96o1KYbzEC/269qwZRBzkIxQbb48KrUS9PLPOPVtis45ds0FYsNzj5Xmy6NSq8RbzkPc+j81wJY59kwKxIbnHtXmy6OyUpNiSCAJPsLFli2DZN6MpQhEp6F7Ha1V1OgBRGtatN1qybwZSxGI6Ru/7WrvxNc7ioleo1aTYmiR3GTL9NRXt8N0ZN2UmaTXQa358qhEKfEGeHWUeiuxYY5d1k1ZuEBML+2uNl8u2mKJiCCAXXPsjCAKrdVq8+UyBCLijmPDHLuMZF2oQEyfNRdhrYDVmxRDFssTs0ymWy0ZybpQgZg8a/7c124XYq2A1ZsUQ4vkiFsmWi1aLOH05Vsx+pUhYa+3WpNiyB454j5i3AFwaMOttFkUiDiizpdHJU4FKyCXsJJViclz7KJtljCBmFq9ijNfHpUoTYrhPMQT+h5smmNnBJForeLMl0clToJe/h2BESywWqZuGSTyoSEFUoPXd35LqLUKiFPiLV/Qrvj3YarVEvnQUIhATCzv7u8fwmBhjfDXjdqkGL7jO3KOk1ZLvkBMK+/WM18uM3oAyZsWa1ktW+bYabEEUc98eVTiNCmGFsuVs1wmzrGLcjUUSBVrVc98eVTqKfEGeAJLvcsxbY5dlKtJLBDT9tuVZa2SWiyZEQSwa8ugVAXS0NP5sSkn442d90qzViIEIvtBk01bBqUmEFOGo5LOl0chbpNi6C7vyXfEj1iyUzxzkJgkmS+PStwmxdBiOfKXy4NjzBy7CPtPgSD5fHmcCJIEx3FSOR+mzLGL2FjOeoGkYa0CklSwypZWYiWrEm4ZRIEAAJ7bckdqf6ueJsVwHuKl8l5t2TJIqkCyXuIVNV8elSQJevk1UrJZJlktZQLJcolX5Hx5VJKUeMt5iOuk+p5t2DJImkCyXOIVNV8elXqbFEMCQboCsWHLIOYgiq2VqOgByGtarIXNc+zWCaQv35q6tQKSNSmGFs1Nd9myvGVQ0jzZOoGIni+PiogSb4CXUqmXVssygciYL0/bYqmIIJVWy7bhKmsEImu+XIVAVO2OYfIcu/UCUWWtgORNiiGL5albNpO3DLJWIPv7h5RZKyB5k2Jo0Ry1y2bTHLvxApE5Xx4nggi1OooioY1Wy3iByJwvj4rIClZATkEly0arZbRAZM+XR0VEk2I4D/GUH9f3eweNt1rGCkQHaxUgMkEvv6ZimwXYMcdurEB0sFYBIku85TzAdbQ4NtPn2I0UyH1dA1pYK0Bck2I4UXa0Od+mbRlkvEDSmC9XGT0ANU2LWbRaZ544NEmBVJDWfHlURDYphhbP1Wf5TLVaic5w6fzFz3WzVmnNl0dFRom3fOdWXOpdzo/WbaZAKpk/f3ErrZUai6VbBAHMnGNPdIaT+juRvLJ1RCtrlYZAdPxIL9Pm2I3IQVTMl0eyoIKbFEMWy9Nz+UzaMsgIgaQ9Xx4V0U2KocVz9Fw+k6xWLusH8Gj/Fni+j2sz09q9t9nSvNTXdxwH0wvzWq7L5uYCNjYXcHL6mrL3MP3R8TarBVLwcrh89RJ+9cn7Wr6/Nq8B+9r74ElyWWNXLuLt/3yq7fpsml3ESYV/X0SOnDhGqyz13rbQgHyTvk9wJxbm8f7UJWnRSWdxAMC6zi7ckmuxOwdRVeodgIfB7h7tT/DYzDX8d0F8N++fzp7KxAW2q6WAgpezVyAqSr2Njot72rszc5L/fHUcCwJrCGNXLuLEtUuZOPZ8UzNG/Ly9AlHBiJ9HPp+tky7KamXBWulgtWZPnT5gpUCyYq1kWa2sWKuw1boBjSmWpU8/fOB5bQQiSq2mWSvRVuv4pfOZsVZhq9WUSavl6qTW1bit5GXOWomyWlNzs3hn/FSmj32wuwcD8OwTSBoUG5uwpbgOWWds5hrOlmZi/Y7v+/j9mRMwgXvau6VbLZGPHjIhkEbHxYhnzsexv3vtPEoxrNYnly/g7PSEEceez+elW62x/T9Zr51APt37gLRmqB2lHNoLBZjEuxMXgAj9YyZYqyxbLe0jSLGxCTuK5n0M2IW5aZydn7bGWi3njuY1UqzW4uS0PQIxzVrFtVomWavltBcK2FES/4T9s/sfcrQViOhy781+g3HWKqrVujIzZZy1ClnnYh+KjU1av0ehAhFZ7u3INWJ3dxGmU81q+b6P353+N2xgxGsVZrVE2yutLVaW+3eSWq2/jn+Oq/NzVhy7SKsl2l5JEYiIatYw8ujq6IBN/OF/Z8vW6u+Xz1l17DuKfejINZpvsWit6mdiYR7/nLpijbUS7RhkzSVJEUiSN2uTtVrOB1cvWGOtltPV0YFh1L/2Ih8OShdIvW/2llyLddaKXGd3d7EuqyUjOZduseK+6YKXw66WAq8Sy6nHQchIzqULJO6bHvHzWs+Xk2xYLSOT9FtyLVjX2cWrgwAAtt2wJvIcu8weQOkCifLmaa3IcnSaY1ceQWitSDWizLHLjh6pCKTWQQzAo7UiK6LDlkHKIkjW58uJWquVRvRITSDVDiaLW/cQNVZro6vuOkktglQ+F8nq1j1EDXcW1i7p+E0reqQqkOC5CK0ViW21KubY094LOtUcZPqj420mbN1D0ieYY5fVc6WFQM48cWjShK17iBq6T5wZTftvKvlYpp9/+K7P5SZxmJmYxLMj+1K/XpWUecdPnOSKk1ioEIcygRwefdDhkpOonPrgw2Oq/rbSC5VWi+hqrZRGEB3uDITWSvsIAgA/e+8dv6mtlVcCCfHk8N3Kr0/l3byq7xCEeYfWEYT5CFnO5fFzeGHfqEOBUCREs6RcO4vFpJ3obLm18/+PHXvdLw5t5JXCpJwCoUiIzuLQViAAy78UBwVCkRCtxaG9QCgSioMCoUgoDo3JxMdAPzuyz2GLPMXBCLIKrG5RHIwgNTg8+qDzr7+8t4uXWfaYmZjMnDgyF0EqYVtKtsSR1aZUN6sn/cnhu52ZiUlefZozfuJkpju2M99q/tQ7x/wbi728EplvMIJU44V9o8xLmG8wgkSBz0vUc+qDD48d/eGj3zXleIyb5nvotVd+O3jr8CgvVVoqCoTRhFGDAqmfHxw+tPPmb4z8g5ewvFzD9D0FrNgwgZUu2ikKhLaLdooCoVAoDAqEQqEwKBDmKBQGBZIxbH+GYkNVigKh/WK0oEAolqSMnzjJz2mhQJivMFJQIBQMBUGBZCnRv7GvdzRt0VAMFIgR4gm+jlMtuzx+DpfPnjsGABQBIYQQQgghhBBCCCGEEEIIIYQQQgghhBBCCCGEEGI9/wd+pgKyVJt0vgAAAABJRU5ErkJggg==