Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/harjot-oberai/long-shadows

Long Shadows for android!
https://github.com/harjot-oberai/long-shadows

android concave-hull library longshadows

Last synced: 6 days ago
JSON representation

Long Shadows for android!

Awesome Lists containing this project

README

        

#

Long-Shadows

A library for efficiently generating and rendering beautiful long shadows in Android.



Platform


API


License: MIT


# Table of contents

* [Features](#features)
* [Usage](#usage)
* [Examples](#examples)
* [Example 1 (Basic)](#example-1-basic)
* [Example 2 (Shadow Angle)](#example-2-shadow-angle)
* [Example 3 (Shadow Length)](#example-3-shadow-length)
* [Example 4 (Shadow Blur) ](#example-4-shadow-blur)
* [Example 5 (Shadow Color)](#example-5-shadow-color)
* [Example 6 (Single View, Multiple Shadows)](#example-6-single-view-multiple-shadows)
* [Example 7 (Non-Transparent background)](#example-7-non-transparent-background)
* [Dynamically updating shadows](#dynamically-updating-shadows)
* [Custom Views and ViewGroups](#using-long-shadows-with-custom-views-and-custom-viewgroups)
* [How does this work ?](#how-does-this-work-)
* [Documentation](#documentation)
* [Limitations](#limitations)
* [Credits](#credits)
* [License](#license)

# Features

- Any Type of views : Long-Shadows can render shadow for any type of view, even custom views.
- Heavy calculation in Native : Long-Shadows shifts the load of heavy calculations to native, to reduce overhead in Java. Thus the rendering is very fast and efficient.
- Precise control over shadow of every View : The library supports fine control over shadows of a view, so two views in same ViewGroup can have different shadow parameters.
- Multiple Shadows for a View : The library support rendering multiple long shadows for a single view, simulating multiple light sources.
- Async Calculations : The library allows the operations to be asynchronous to avoid blocking the UI thread for long calculations.
- Shadow animations : The library supports fade out animation while rendering the shadow.

# Usage
Just add the following dependency in your app's `build.gradle`
```groovy
dependencies {
implementation 'com.sdsmdg.harjot:longshadows:1.0.1'
}
```

# Examples
Here are some examples to provide you a head start with using this library. The examples are selected to demonstrate almost all the basic features provided by the library. So lets start !

## Example 1 (Basic)

LongShadowsImageView

```xml

```

LongShadowsTextView

```xml

```

## Example 2 (Shadow Angle)

#### XML
```xml

```

#### Result

## Example 3 (Shadow Length)

#### XML
```xml

```

#### Result

## Example 4 (Shadow Blur)

#### XML
```xml

```

#### Result

## Example 5 (Shadow Color)

Shadows are filled by a linear gradient, so shadow color is built of two compontents `shadow_start_color`, which defines the start color of the gradient and `shadow_end_color`, which defines the end color of the gradient.

> **Tip** : To get a solid color shadow, just put the same value in `shadow_start_color` and `shadow_end_color` attributes.

#### XML
```xml

```

#### Result

## Example 6 (Single View, Multiple Shadows)

The library offers multiple long shadows for a single view. To use multiple shadows, just pass a comma separated string of shadow angles of you choice in the `shadow_angle` attribute.

> **e.g.** `app:shadow_angle="45, 90, 135"`. This will create 3 shadows at angle *45*, *90* and *135* degrees.

> You can also specify different lengths for each shadow in a similar way, by passing a comma separated string of shadow lengths in the `shadow_length` attribute.

**e.g.**
`app:shadow_angle="45, 90, 135"`
`app:shadow_length="200, 300, 400"`

This will create 3 shadows at angles *45*, *90* and *135*, with shadows lengths, *200*, *300* and *400* respectively.

#### XML
```xml

```

#### Result

## Example 7 (Non-Transparent background)

So far we have seen that the library generates long shadows for all non-transparent parts in an image, but what if we want the image to have a non-transparent background, but only create shadow for what is inside the image and not the background. This is where `background_transparent` and `background_color_to_ignore` attributes come into play.

If your image has say a white background, with some character in the middle and we only want the shadow for the middle character and completely avoid shadow for the white background, you can set `background_transparent` as `false`, indicating the background contains some color other that 0 (transparent), and `background_color_to_ignore` as `#FFFFFF`, indicating that we need to treat all white pixels as background and not generate shadow for them.

> **Note** : Transparent pixels are always counted as background and never generate any shadow of their own. This is useful in cases where the background of your image's background is a circle or a rounded-rectangle with white color, where we have to count both transparent pixels and white pixel as background.

**e.g.** Lets say we want to display the google icon which has a white rectangle that acts as the background of the letter `G`.
Lets see how can we generate long shadow only for the letter and avoid building the shadow for the rectangular white background.

> For this example we will create a rectangular drawable `rectangular_white_background.xml` which will act as the background to icon vector `ic_google_icon_colored.xml`. You can find these resource files [here](/app/src/main/res/drawable/)

#### XML
```xml

```

#### Result

> **Note** : As much as I want people to use this feature, I should warn that this won't work in several cases as android internally applies anti alias on images/bitmaps while scaling and this leads to some random colors around the boundaries. Since the algorithm checks for only one particular color, these new colors will be treated as contours and shadow will be generated, leading to unexpected results.


So if you want to have a background, I would suggest to draw the background in a different View and overlay this View with `LongShadowsView`/`LongShadowsImageView`/`LongShadowsTextView`, containing only the part you want to generate the shadow for.


If you are creating a custom view and want to draw a bitmap and use this feature, you may set `Paint#setFilterBitmap()` to `false`. This will disable anti aliasing for the bitmap.

# Dynamically updating shadows

All the shadow parameters can be updated dynamically through java. Here is a small example :

```java

LongShadowsImageView longShadowsImageView = (LongShadowsImageView) findViewById(R.id.long_shadows_image_view);

longShadowsImageView.setShadowAngle("135");

// Set various paramters here

// Call the update method at the end, for the changes to take effect
longShadowsImageView.update();

```

> **Note** : You need to call `update()` method at the end for the changes to take effect. This is done to avoid updating views again and again as the parameters are changed. This approach ensures that the views are updated only when the user is done with changing the parameters.

# Custom Views and ViewGroups

### Custom View
If you want your own custom view to be able to generate long shadows, just extend `LongShadowsView` instead of `View` while creating the custom view.
That's it :)
```java
public class CustomLongShadowsView extends LongShadowsView {

public CustomLongShadowsView(Context context) {
super(context);
}

public CustomLongShadowsView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}

public CustomLongShadowsView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}

@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);

/*
* Draw whatever you want.
*/

}
}
```

> **Note** : The custom view must be inside `LongShadowsWrapper` or `LongShadowsFrameLayoutWrapper` for the shadow to be generated.

### Custom View Group

`LongShadowsWrapper` and `LongShadowsFrameLayoutWrapper` are two custom view groups provided by library that extend `RelativeLayout` and `FrameLayout` respectively. These custom view groups recursively search for eligible views for long shadow generation, even inside nested view groups.

So if you have a custom view group, inside which you want to place your `LongShadowsImageView` or `LongShadowsTextView`, then you can just make this view group a direct child of `LongShadowsWrapper` or `LongShadowsFrameLayoutWrapper` and the library will take care of everything else.

See the example below :

```xml

```

# How does this work ?

The `LongShadowsWrapper` is an extension of `Relative Layout`. The `LongShadowsFrameLayoutWrapper` is an extension of `FrameLayout`. Use any one of them as per your convenience.

Firstly, eligible child views (`LongShadowsImageView`, `LongShadowsTextView` or extensions of `LongShadowsView`) are found using a recursive strategy. Once an eligible view is found it goes through the following process to generate the long shadow.

1. First a bitmap is generated from the drawing cache of the View. The bitmap is converted into an integer array of pixels and passed on to the native function `getShadowPaths(...)` along with width, height and some other parameters.

2. The native function first generates contours using the `contours` function. The `contours` function returns vector points which are on the boundary of an object. This is achieved by doing BFS(Breath First Search) on the grid and marking the points which are on the boundary and were encountered in the BFS run.

3. Each of the contour is then passed to `getShadowPathsFromContour` function, which internally calls `boundaryPath` function. The `boundaryPath` function returns the **Front Polar Path** of the object. **Front Polar Path** is the set of points which will be directly visible from the Light Source, in polar order. We need the points in Polar Order, so that we can remove darkening of some regions due to overlapping, which will result in some dark regions during rendering due to alpha blending.

3.1. To find the **Front polar order**, we first extract the outer boundary of an object because for objects like 'O', there are two boundaries. This is done through `getOuterBoundary` function. In this function, we do BFS travesal from a point on outer boundary. So all the points which are not visited/marked after the BFS travesal belong to the inner boundary.

3.2. Now we have the Outer Boundary of the object. Now we will seperate the Boundary which will be visible by the Light source(lets name this boundary as Front Path) from the Overall Boundary. Also, notice that the Front Path will begin from the Upper tangent to the Lower tangent (If tangents are drawn from the Light Source to the Object). So we again apply BFS from Upper Tangent to Lower Tangent. This is done in the function `getPath`.

3.3. We still cannot be sure that the path returned by getPath is the Front Path, as there are 2 paths from the Upper Tangent to the Lower Tangent. So we compare the Area formed by the Light Source and the 2 Paths. The one with the less area will be the Front Path. `getArea` function calculates the area of the figure formed by the Light Source and the path by using Shoelace Formulae.

3.4. Corner Cases handling - We have used `atan` function to get the angle, inorder to sort the points in Polar Order. But `atan` function is discountinuos. So we have carefully handled the discountinuity of the function in the `getPolarOrder` function.

4. Once we have the **Front Polar Path**, we translate its end points according to the `shadow length` and `shadow angle`.

5. Now all these points are collected and converted into a structure `ShadowPath`. The `getShadowPathsFromContour` functions returns one `ShadowPath` object for every contour.

6. All `ShadowPath` objects are passed back to Java, where `Path` objects are created using the points.

7. Finally the `Path` object created is rendered and filled with a `LinearGradient` with starting and end points as defined in the `ShadowPath` object.

**P.S.** : All the algorithms used in the native code have been implemented by my friend [Piyush Mehrotra](https://github.com/hm98).

# Documentation

Attributes for `LongShadowsWrapper` and `LongShadowsFrameLayoutWrapper`

|XML attribute |Java set methods |Description | Default Value |
|----------------------|-------------------------------|--------------------------------------------|-------------------|
|shouldClipChildren |setShouldClipChildren(...) |Set whether to clip children to bounds |false |
|shouldClipToPadding |setShouldClipToPadding(...) |Set whether to clip children to padding |false |
|calculateAsync |setShouldCalculateAsync(...) |Set the flag for async shadow calculations. |true |
|showWhenAllReady |setShowShadowsWhenAllReady(...)|Set the flag for showing all shadows after all calculations are over|true|
|animateShadow |setShouldAnimateShadow(...) |Set the flag for shadow animation |true |
|animationDuration |setAnimationDuration(...) |Set the value of shadow animation duration. |300ms |

Attributes for `LongShadowsImageView`, `LongShadowsTextView` and `LongShadowsView`

|XML attribute |Java set methods |Description | Type | Default Value |
|----------------------|------------------------------|-------------------------------------------|---------------|-------------------|
|shadow_angle |setShadowAngle(...) |Set the shadow angle for the view |String |"45.0f" |
|shadow_length |setShadowLength(...) |Set the shadow length for the view |String |"400" |
|shadow_startColor |setShadowStartColor(...) |Set the start color of the shadow gradient |Color |#88000000 |
|shadow_endColor |setShadowEndColor(...) |Set the end color of the shadow gradient |Color |#00000000 |
|shadow_blur_enabled |setShadowBlurEnabled(...) |Set the flag for enabling shadow blur |boolean |true |
|shadow_blur_radius |setShadowBlurRadius(...) |Set the shadow blur radius |int |1 |
|shadow_alpha |setShadowAlpha(...) |Set the shadow alpha |int |255 |
|background_transparent |setBackgroundTransparent(...)|Set the flag for background transparency |boolean |true |
|background_color_to_ignore|setBackgroundColorToIgnore(...)|Set the background color to ignore |Color |#00000000 |

# Limitations

1. The library uses native code to make calculations faster. This will result in a larger apk size if `unversalApk` is built (APK for all architectures). So to reduce apk size, you may need to build separate apks for different architecture using ABI filters in `build.gradle` of your app. See [build.gradle](/app/build.gradle) for more details.

# Credits

1. **Piyush Mehrotra** ( [Codeforces](http://codeforces.com/profile/hm_98), [CodeChef](https://www.codechef.com/users/hm_98), [Github](https://github.com/hm98) ) : Implementation of algorithm to find the shadow paths from pixel array. This library would not have been possible without his talent and hard work.

2. [Yaroslav](https://github.com/yarolegovich) : Implementation of asynchronous calculations and shadow animations in [MaterialShadows](https://github.com/harjot-oberai/MaterialShadows), which was directly reused in this library.

3. [Feather Icons](https://feathericons.com/) : For providing the vectors used in examples.

4. [WorldVectorLogo](https://worldvectorlogo.com/) : For providing some more vectors for the examples.

5. [StackOverflow](https://stackoverflow.com/) : Nobody knows everything :)

# License
Long-Shadows is licensed under `MIT license`. View [license](LICENSE.md).