IT_Programming/Android_Java

Branding the EdgeEffect aka Hit the Wall with Your Own Color

JJun ™ 2015. 1. 1. 20:19

 출처: http://evendanan.net/android/branding/2013/12/09/branding-edge-effect/


 


Lately, I've become sort of branding obsessed; colors, font sizes and types, UI language, they all seem so important to me, I can hardly look at an App these days without pulling on my hairs and scream "God, oh God, why would they use inconsistent, non-brand colors!" Ya, but that's my problem. Your is another: how to brand the EdgeEffect glow, ah?

 

TL;DR; Use the gist here.

 

It is usually pretty easy to change colors, sizes, etc. for widgets which are in your App's real estate (e.i., not the OS's chrome: notification bar icons, for example), but, weird as it sounds, Android does not provide any method to change (or brand) the hitting-the-wall feedback glow effect in its widget (ListView, GridView, ScrollView, etc.), and will only use the OS's color. This even gets weirder, when Android 2.3 has a green edge-effect, while Android 3+ uses a blueish (Holo) color, and KitKat (v4.4) uses a gray glow effect (ironic fact: Android team changed the colors of the chrome in KitKat to a nuetral gray color, so it will not collide with your branding, although, a gray Edge-Effect inside your App's UI is clearly colliding, IMHO).

 

 

The Cause


The EdgeEffect class uses a Drawable resource (an OS's resource, not yours, hint, hint, Google), 

which provides the glow effect, this resource is loaded internally by the EdgeEffect class:


public EdgeEffect(Context context) {
    final Resources res = context.getResources();
    mEdge = res.getDrawable(R.drawable.overscroll_edge);
    mGlow = res.getDrawable(R.drawable.overscroll_glow); 

view rawEdgeEffect_ctor.java hosted with ❤ by GitHub


And the size of the glow (how much the user over-pulled the list) is calculated during draw(Canvas) 

call:

 public boolean draw(Canvas canvas) {
...
    int glowBottom = (int) Math.min(
            mGlowHeight * mGlowScaleY * mGlowHeight / mGlowWidth * 0.6f,
            mGlowHeight * MAX_GLOW_HEIGHT);
    if (mWidth < mMinWidth) {
        // Center the glow and clip it.
        int glowLeft = (mWidth - mMinWidth)/2;
        mGlow.setBounds(glowLeft, 0, mWidth - glowLeft, glowBottom);
    } else {
        // Stretch the glow to fit.
        mGlow.setBounds(0, 0, mWidth, glowBottom);
    }
    mGlow.draw(canvas);
...       

view rawEdgeEffect_draw.java hosted with ❤ by GitHub

There is no way to change the drawable resource (hint one, Google.), or apply a filter on the drawable (hint two, Google.). A shame.

 

 

Solution - Hacker style


What can be done? I'm going to hack my way into branding this drawable. I'll take advantage of the
fact that the Drawable instance is shared, and not mutate (which makes sense, right?)
and I'll load that Drawable in my code, and apply a filter on it:

 

static void brandGlowEffect(Context context, int brandColor) {
    int glowDrawableId = context.getResources().getIdentifier("overscroll_glow", "drawable"

                                   , "android");
    Drawable androidGlow = context.getResources().getDrawable(glowDrawableId);
    androidGlow.setColorFilter(brandColor, PorterDuff.Mode.SRC_IN);
} 

view rawbrandGlowEffect.java hosted with ❤ by GitHub

 

 

What just happened?


The name of the glow resource has never changed in Android's history (so far), (line 1) so I'll get its identification, (line 2) load the Drawable, which is shared instance across my Application, thus the same one EdgeEffect is using, (line 3) and apply a MULTIPLY filter on it, with the brand's color.

 

 

 

What about the overscroll_edge resource

The edge-effect is broken to two drawables: the glow (overscroll_glow), and the edge (overscroll_edge). This method allows the EdgeEffect class to stretch just the glow, and keep the edge crisp. To brand the edge (you'll probably want to do that too), use this revised method, which brand both the glow and the edge:

 static void brandGlowEffect(Context context, int brandColor) {
      //glow
      int glowDrawableId = context.getResources().getIdentifier("overscroll_glow", "drawable"

                                    , "android");
      Drawable androidGlow = context.getResources().getDrawable(glowDrawableId);
      androidGlow.setColorFilter(brandColor, PorterDuff.Mode.SRC_IN);
      //edge
      int edgeDrawableId = context.getResources().getIdentifier("overscroll_edge", "drawable"

                                   , "android");
      Drawable androidEdge = context.getResources().getDrawable(edgeDrawableId);
      androidEdge.setColorFilter(brandColor, PorterDuff.Mode.SRC_IN);
}

view rawbrandGlowEffect_full.java hosted with ❤ by GitHub

 

 

 

Pitfalls

Android may decide to change the name of the resource, in which case you'll need to change the call to getIdentifier


The shared instance may be unloaded from memory, so remember to call the brandGlowEffect after each creation of EdgeEffect (after creation of ListView, GridView, ScrollView, or your own instance of EdgeEffect), best place will be in the Activity's onCreate, right after setting the layout.
I'll emphasis the first point (since this is the core of the hack, and it's a reflection kind of solution, thus dangerous): make sure this method works on the target Android version you had in mind. I take no responsibility!

 

 


Branding - The Technical Challenge

An App should be useful, easy to use, effective, beautiful, stable. It should also be unique, and represent its creator. This is where Branding comes in - a way to express and show yourself in the App UI: colors, Type-face and font sizes, language, behavior, icons and graphics, etc.

I'll try to post more about this subject, but not from a marketing, design, or decision (reason) standpoint. Instead I'll show you how, technically, to brand your App. E.g., in this post I've shown how to change the color of Android's build-in Edge-Effect, you can use this method to change the glow color to your brand's main (or secondary) color! What ever that color maybe (which is a design decision).