├── .gitignore ├── .idea ├── .name ├── compiler.xml ├── copyright │ └── profiles_settings.xml ├── encodings.xml ├── gradle.xml ├── misc.xml ├── modules.xml ├── runConfigurations.xml └── vcs.xml ├── HeartLayout.iml ├── README.md ├── app ├── .gitignore ├── app.iml ├── build.gradle ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── tyrantgit │ │ └── sample │ │ └── heartlayout │ │ └── ApplicationTest.java │ └── main │ ├── AndroidManifest.xml │ ├── java │ └── tyrantgit │ │ └── sample │ │ └── heartlayout │ │ └── MainActivity.java │ └── res │ ├── drawable-hdpi │ └── bg.jpg │ ├── layout │ └── activity_main.xml │ ├── mipmap-hdpi │ └── ic_launcher.png │ ├── mipmap-mdpi │ └── ic_launcher.png │ ├── mipmap-xhdpi │ └── ic_launcher.png │ ├── mipmap-xxhdpi │ └── ic_launcher.png │ ├── values-w820dp │ └── dimens.xml │ └── values │ ├── dimens.xml │ ├── strings.xml │ └── styles.xml ├── assets └── heartlayout.gif ├── build.gradle ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat ├── heartlayout ├── .gitignore ├── build.gradle ├── heartlayout.iml ├── proguard-rules.pro └── src │ └── main │ ├── AndroidManifest.xml │ ├── java │ └── tyrantgit │ │ └── widget │ │ ├── AbstractPathAnimator.java │ │ ├── HeartLayout.java │ │ ├── HeartView.java │ │ └── PathAnimator.java │ └── res │ ├── drawable │ ├── heart.png │ └── heart_border.png │ └── values │ ├── attrs.xml │ ├── dimens.xml │ ├── integers.xml │ └── strings.xml └── settings.gradle /.gitignore: -------------------------------------------------------------------------------- 1 | .gradle 2 | /local.properties 3 | /.idea/workspace.xml 4 | /.idea/libraries 5 | .DS_Store 6 | /build 7 | /captures 8 | -------------------------------------------------------------------------------- /.idea/.name: -------------------------------------------------------------------------------- 1 | HeartLayout -------------------------------------------------------------------------------- /.idea/compiler.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /.idea/copyright/profiles_settings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | -------------------------------------------------------------------------------- /.idea/encodings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | -------------------------------------------------------------------------------- /.idea/gradle.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 18 | 19 | -------------------------------------------------------------------------------- /.idea/misc.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 19 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 46 | -------------------------------------------------------------------------------- /.idea/modules.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | -------------------------------------------------------------------------------- /.idea/runConfigurations.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 11 | 12 | -------------------------------------------------------------------------------- /.idea/vcs.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /HeartLayout.iml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | [![Android Gems](http://www.android-gems.com/badge/tyrantgit/HeartLayout.svg?branch=master)](http://www.android-gems.com/lib/tyrantgit/HeartLayout) 2 | [![Android Arsenal](https://img.shields.io/badge/Android%20Arsenal-HeartLayout-brightgreen.svg?style=flat)](http://android-arsenal.com/details/3/2558) 3 | 4 | # HeartLayout 5 | periscope like heartlayout 6 | 7 | ![heartlayout.gif](assets/heartlayout.gif) 8 | 9 | ## Getting started 10 | 11 | In your `build.gradle`: 12 | 13 | ```gradle 14 | dependencies { 15 | compile 'tyrant:heartlayout:1.0.1' 16 | } 17 | ``` 18 | 19 | ```java 20 | HeartLayout heartLayout = ...; 21 | heartLayout.addHeart(color); 22 | heartLayout.addHeart(color,heartResId,heartBorderResId); 23 | ``` 24 | 25 | ## License 26 | 27 | Copyright 2015 tyrantgit 28 | 29 | Licensed under the Apache License, Version 2.0 (the "License"); 30 | you may not use this file except in compliance with the License. 31 | You may obtain a copy of the License at 32 | 33 | http://www.apache.org/licenses/LICENSE-2.0 34 | 35 | Unless required by applicable law or agreed to in writing, software 36 | distributed under the License is distributed on an "AS IS" BASIS, 37 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 38 | See the License for the specific language governing permissions and 39 | limitations under the License. 40 | -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /app/app.iml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 8 | 9 | 10 | 11 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | 50 | 51 | 52 | 53 | 54 | 55 | 56 | 57 | 58 | 59 | 60 | 61 | 62 | 63 | 64 | 65 | 66 | 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | 75 | 76 | 77 | 78 | 79 | 80 | 81 | 82 | 83 | 84 | 85 | 86 | 87 | 88 | 89 | 90 | 91 | 92 | 93 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | 3 | android { 4 | compileSdkVersion 22 5 | buildToolsVersion "23.0.0 rc2" 6 | 7 | defaultConfig { 8 | applicationId "tyrantgit.sample.heartlayout" 9 | minSdkVersion 14 10 | targetSdkVersion 22 11 | versionCode 1 12 | versionName "1.0" 13 | } 14 | buildTypes { 15 | release { 16 | minifyEnabled false 17 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 18 | } 19 | } 20 | } 21 | 22 | dependencies { 23 | compile fileTree(dir: 'libs', include: ['*.jar']) 24 | compile project(':heartlayout') 25 | } 26 | -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in D:\develop\AndroidSdk/tools/proguard/proguard-android.txt 4 | # You can edit the include path and order by changing the proguardFiles 5 | # directive in build.gradle. 6 | # 7 | # For more details, see 8 | # http://developer.android.com/guide/developing/tools/proguard.html 9 | 10 | # Add any project specific keep options here: 11 | 12 | # If your project uses WebView with JS, uncomment the following 13 | # and specify the fully qualified class name to the JavaScript interface 14 | # class: 15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 16 | # public *; 17 | #} 18 | -------------------------------------------------------------------------------- /app/src/androidTest/java/tyrantgit/sample/heartlayout/ApplicationTest.java: -------------------------------------------------------------------------------- 1 | package tyrantgit.sample.heartlayout; 2 | 3 | import android.app.Application; 4 | import android.test.ApplicationTestCase; 5 | 6 | /** 7 | * Testing Fundamentals 8 | */ 9 | public class ApplicationTest extends ApplicationTestCase { 10 | public ApplicationTest() { 11 | super(Application.class); 12 | } 13 | } -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 10 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /app/src/main/java/tyrantgit/sample/heartlayout/MainActivity.java: -------------------------------------------------------------------------------- 1 | package tyrantgit.sample.heartlayout; 2 | 3 | import android.app.Activity; 4 | import android.graphics.Color; 5 | import android.os.Bundle; 6 | 7 | import java.util.Random; 8 | import java.util.Timer; 9 | import java.util.TimerTask; 10 | 11 | import tyrantgit.widget.HeartLayout; 12 | 13 | 14 | public class MainActivity extends Activity { 15 | 16 | private Random mRandom = new Random(); 17 | private Timer mTimer = new Timer(); 18 | private HeartLayout mHeartLayout; 19 | 20 | @Override 21 | protected void onCreate(Bundle savedInstanceState) { 22 | super.onCreate(savedInstanceState); 23 | setContentView(R.layout.activity_main); 24 | 25 | mHeartLayout = (HeartLayout) findViewById(R.id.heart_layout); 26 | mTimer.scheduleAtFixedRate(new TimerTask() { 27 | @Override 28 | public void run() { 29 | mHeartLayout.post(new Runnable() { 30 | @Override 31 | public void run() { 32 | mHeartLayout.addHeart(randomColor()); 33 | } 34 | }); 35 | } 36 | }, 500, 200); 37 | 38 | } 39 | 40 | @Override 41 | protected void onDestroy() { 42 | super.onDestroy(); 43 | mTimer.cancel(); 44 | } 45 | 46 | private int randomColor() { 47 | return Color.rgb(mRandom.nextInt(255), mRandom.nextInt(255), mRandom.nextInt(255)); 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /app/src/main/res/drawable-hdpi/bg.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/app/src/main/res/drawable-hdpi/bg.jpg -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 11 | 12 | 17 | 18 | 19 | -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/values-w820dp/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 5 | 64dp 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 16dp 4 | 16dp 5 | 6 | -------------------------------------------------------------------------------- /app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | HeartLayout 3 | 4 | Hello world! 5 | Settings 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /assets/heartlayout.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/assets/heartlayout.gif -------------------------------------------------------------------------------- /build.gradle: -------------------------------------------------------------------------------- 1 | // Top-level build file where you can add configuration options common to all sub-projects/modules. 2 | 3 | buildscript { 4 | repositories { 5 | jcenter() 6 | } 7 | dependencies { 8 | classpath 'com.android.tools.build:gradle:1.2.3' 9 | 10 | // NOTE: Do not place your application dependencies here; they belong 11 | // in the individual module build.gradle files 12 | } 13 | } 14 | 15 | allprojects { 16 | repositories { 17 | jcenter() 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /gradle.properties: -------------------------------------------------------------------------------- 1 | # Project-wide Gradle settings. 2 | 3 | # IDE (e.g. Android Studio) users: 4 | # Gradle settings configured through the IDE *will override* 5 | # any settings specified in this file. 6 | 7 | # For more details on how to configure your build environment visit 8 | # http://www.gradle.org/docs/current/userguide/build_environment.html 9 | 10 | # Specifies the JVM arguments used for the daemon process. 11 | # The setting is particularly useful for tweaking memory settings. 12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m 13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8 14 | 15 | # When configured, Gradle will run in incubating parallel mode. 16 | # This option should only be used with decoupled projects. More details, visit 17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects 18 | # org.gradle.parallel=true -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | #Wed Apr 10 15:27:10 PDT 2013 2 | distributionBase=GRADLE_USER_HOME 3 | distributionPath=wrapper/dists 4 | zipStoreBase=GRADLE_USER_HOME 5 | zipStorePath=wrapper/dists 6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.2.1-all.zip 7 | -------------------------------------------------------------------------------- /gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | ############################################################################## 4 | ## 5 | ## Gradle start up script for UN*X 6 | ## 7 | ############################################################################## 8 | 9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 10 | DEFAULT_JVM_OPTS="" 11 | 12 | APP_NAME="Gradle" 13 | APP_BASE_NAME=`basename "$0"` 14 | 15 | # Use the maximum available, or set MAX_FD != -1 to use that value. 16 | MAX_FD="maximum" 17 | 18 | warn ( ) { 19 | echo "$*" 20 | } 21 | 22 | die ( ) { 23 | echo 24 | echo "$*" 25 | echo 26 | exit 1 27 | } 28 | 29 | # OS specific support (must be 'true' or 'false'). 30 | cygwin=false 31 | msys=false 32 | darwin=false 33 | case "`uname`" in 34 | CYGWIN* ) 35 | cygwin=true 36 | ;; 37 | Darwin* ) 38 | darwin=true 39 | ;; 40 | MINGW* ) 41 | msys=true 42 | ;; 43 | esac 44 | 45 | # For Cygwin, ensure paths are in UNIX format before anything is touched. 46 | if $cygwin ; then 47 | [ -n "$JAVA_HOME" ] && JAVA_HOME=`cygpath --unix "$JAVA_HOME"` 48 | fi 49 | 50 | # Attempt to set APP_HOME 51 | # Resolve links: $0 may be a link 52 | PRG="$0" 53 | # Need this for relative symlinks. 54 | while [ -h "$PRG" ] ; do 55 | ls=`ls -ld "$PRG"` 56 | link=`expr "$ls" : '.*-> \(.*\)$'` 57 | if expr "$link" : '/.*' > /dev/null; then 58 | PRG="$link" 59 | else 60 | PRG=`dirname "$PRG"`"/$link" 61 | fi 62 | done 63 | SAVED="`pwd`" 64 | cd "`dirname \"$PRG\"`/" >&- 65 | APP_HOME="`pwd -P`" 66 | cd "$SAVED" >&- 67 | 68 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 69 | 70 | # Determine the Java command to use to start the JVM. 71 | if [ -n "$JAVA_HOME" ] ; then 72 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 73 | # IBM's JDK on AIX uses strange locations for the executables 74 | JAVACMD="$JAVA_HOME/jre/sh/java" 75 | else 76 | JAVACMD="$JAVA_HOME/bin/java" 77 | fi 78 | if [ ! -x "$JAVACMD" ] ; then 79 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 80 | 81 | Please set the JAVA_HOME variable in your environment to match the 82 | location of your Java installation." 83 | fi 84 | else 85 | JAVACMD="java" 86 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 87 | 88 | Please set the JAVA_HOME variable in your environment to match the 89 | location of your Java installation." 90 | fi 91 | 92 | # Increase the maximum file descriptors if we can. 93 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then 94 | MAX_FD_LIMIT=`ulimit -H -n` 95 | if [ $? -eq 0 ] ; then 96 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 97 | MAX_FD="$MAX_FD_LIMIT" 98 | fi 99 | ulimit -n $MAX_FD 100 | if [ $? -ne 0 ] ; then 101 | warn "Could not set maximum file descriptor limit: $MAX_FD" 102 | fi 103 | else 104 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 105 | fi 106 | fi 107 | 108 | # For Darwin, add options to specify how the application appears in the dock 109 | if $darwin; then 110 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 111 | fi 112 | 113 | # For Cygwin, switch paths to Windows format before running java 114 | if $cygwin ; then 115 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 116 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 117 | 118 | # We build the pattern for arguments to be converted via cygpath 119 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 120 | SEP="" 121 | for dir in $ROOTDIRSRAW ; do 122 | ROOTDIRS="$ROOTDIRS$SEP$dir" 123 | SEP="|" 124 | done 125 | OURCYGPATTERN="(^($ROOTDIRS))" 126 | # Add a user-defined pattern to the cygpath arguments 127 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 128 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 129 | fi 130 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 131 | i=0 132 | for arg in "$@" ; do 133 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 134 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 135 | 136 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 137 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 138 | else 139 | eval `echo args$i`="\"$arg\"" 140 | fi 141 | i=$((i+1)) 142 | done 143 | case $i in 144 | (0) set -- ;; 145 | (1) set -- "$args0" ;; 146 | (2) set -- "$args0" "$args1" ;; 147 | (3) set -- "$args0" "$args1" "$args2" ;; 148 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;; 149 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 150 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 151 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 152 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 153 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 154 | esac 155 | fi 156 | 157 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules 158 | function splitJvmOpts() { 159 | JVM_OPTS=("$@") 160 | } 161 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS 162 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" 163 | 164 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" 165 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @if "%DEBUG%" == "" @echo off 2 | @rem ########################################################################## 3 | @rem 4 | @rem Gradle startup script for Windows 5 | @rem 6 | @rem ########################################################################## 7 | 8 | @rem Set local scope for the variables with windows NT shell 9 | if "%OS%"=="Windows_NT" setlocal 10 | 11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 12 | set DEFAULT_JVM_OPTS= 13 | 14 | set DIRNAME=%~dp0 15 | if "%DIRNAME%" == "" set DIRNAME=. 16 | set APP_BASE_NAME=%~n0 17 | set APP_HOME=%DIRNAME% 18 | 19 | @rem Find java.exe 20 | if defined JAVA_HOME goto findJavaFromJavaHome 21 | 22 | set JAVA_EXE=java.exe 23 | %JAVA_EXE% -version >NUL 2>&1 24 | if "%ERRORLEVEL%" == "0" goto init 25 | 26 | echo. 27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 28 | echo. 29 | echo Please set the JAVA_HOME variable in your environment to match the 30 | echo location of your Java installation. 31 | 32 | goto fail 33 | 34 | :findJavaFromJavaHome 35 | set JAVA_HOME=%JAVA_HOME:"=% 36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 37 | 38 | if exist "%JAVA_EXE%" goto init 39 | 40 | echo. 41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 42 | echo. 43 | echo Please set the JAVA_HOME variable in your environment to match the 44 | echo location of your Java installation. 45 | 46 | goto fail 47 | 48 | :init 49 | @rem Get command-line arguments, handling Windowz variants 50 | 51 | if not "%OS%" == "Windows_NT" goto win9xME_args 52 | if "%@eval[2+2]" == "4" goto 4NT_args 53 | 54 | :win9xME_args 55 | @rem Slurp the command line arguments. 56 | set CMD_LINE_ARGS= 57 | set _SKIP=2 58 | 59 | :win9xME_args_slurp 60 | if "x%~1" == "x" goto execute 61 | 62 | set CMD_LINE_ARGS=%* 63 | goto execute 64 | 65 | :4NT_args 66 | @rem Get arguments from the 4NT Shell from JP Software 67 | set CMD_LINE_ARGS=%$ 68 | 69 | :execute 70 | @rem Setup the command line 71 | 72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 73 | 74 | @rem Execute Gradle 75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% 76 | 77 | :end 78 | @rem End local scope for the variables with windows NT shell 79 | if "%ERRORLEVEL%"=="0" goto mainEnd 80 | 81 | :fail 82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 83 | rem the _cmd.exe /c_ return code! 84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 85 | exit /b 1 86 | 87 | :mainEnd 88 | if "%OS%"=="Windows_NT" endlocal 89 | 90 | :omega 91 | -------------------------------------------------------------------------------- /heartlayout/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /heartlayout/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.library' 2 | 3 | android { 4 | compileSdkVersion 22 5 | buildToolsVersion "23.0.0 rc2" 6 | 7 | defaultConfig { 8 | minSdkVersion 14 9 | targetSdkVersion 22 10 | versionCode 1 11 | versionName "1.0" 12 | } 13 | buildTypes { 14 | release { 15 | minifyEnabled false 16 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 17 | } 18 | } 19 | } 20 | 21 | dependencies { 22 | compile fileTree(dir: 'libs', include: ['*.jar']) 23 | } 24 | -------------------------------------------------------------------------------- /heartlayout/heartlayout.iml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 8 | 9 | 10 | 11 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | 50 | 51 | 52 | 53 | 54 | 55 | 56 | 57 | 58 | 59 | 60 | 61 | 62 | 63 | 64 | 65 | 66 | 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | 75 | 76 | 77 | 78 | 79 | 80 | 81 | 82 | 83 | 84 | 85 | 86 | 87 | 88 | 89 | 90 | 91 | 92 | 93 | -------------------------------------------------------------------------------- /heartlayout/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in D:\develop\AndroidSdk/tools/proguard/proguard-android.txt 4 | # You can edit the include path and order by changing the proguardFiles 5 | # directive in build.gradle. 6 | # 7 | # For more details, see 8 | # http://developer.android.com/guide/developing/tools/proguard.html 9 | 10 | # Add any project specific keep options here: 11 | 12 | # If your project uses WebView with JS, uncomment the following 13 | # and specify the fully qualified class name to the JavaScript interface 14 | # class: 15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 16 | # public *; 17 | #} 18 | -------------------------------------------------------------------------------- /heartlayout/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /heartlayout/src/main/java/tyrantgit/widget/AbstractPathAnimator.java: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2015 tyrantgit 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 | package tyrantgit.widget; 17 | 18 | import android.content.res.Resources; 19 | import android.content.res.TypedArray; 20 | import android.graphics.Path; 21 | import android.view.View; 22 | import android.view.ViewGroup; 23 | 24 | import java.util.Random; 25 | import java.util.concurrent.atomic.AtomicInteger; 26 | 27 | 28 | public abstract class AbstractPathAnimator { 29 | private final Random mRandom; 30 | protected final Config mConfig; 31 | 32 | 33 | public AbstractPathAnimator(Config config) { 34 | mConfig = config; 35 | mRandom = new Random(); 36 | } 37 | 38 | public float randomRotation() { 39 | return mRandom.nextFloat() * 28.6F - 14.3F; 40 | } 41 | 42 | public Path createPath(AtomicInteger counter, View view, int factor) { 43 | Random r = mRandom; 44 | int x = r.nextInt(mConfig.xRand); 45 | int x2 = r.nextInt(mConfig.xRand); 46 | int y = view.getHeight() - mConfig.initY; 47 | int y2 = counter.intValue() * 15 + mConfig.animLength * factor + r.nextInt(mConfig.animLengthRand); 48 | factor = y2 / mConfig.bezierFactor; 49 | x = mConfig.xPointFactor + x; 50 | x2 = mConfig.xPointFactor + x2; 51 | int y3 = y - y2; 52 | y2 = y - y2 / 2; 53 | Path p = new Path(); 54 | p.moveTo(mConfig.initX, y); 55 | p.cubicTo(mConfig.initX, y - factor, x, y2 + factor, x, y2); 56 | p.moveTo(x, y2); 57 | p.cubicTo(x, y2 - factor, x2, y3 + factor, x2, y3); 58 | return p; 59 | } 60 | 61 | public abstract void start(View child, ViewGroup parent); 62 | 63 | public static class Config { 64 | public int initX; 65 | public int initY; 66 | public int xRand; 67 | public int animLengthRand; 68 | public int bezierFactor; 69 | public int xPointFactor; 70 | public int animLength; 71 | public int heartWidth; 72 | public int heartHeight; 73 | public int animDuration; 74 | 75 | static Config fromTypeArray(TypedArray typedArray) { 76 | Config config = new Config(); 77 | Resources res = typedArray.getResources(); 78 | config.initX = (int) typedArray.getDimension(R.styleable.HeartLayout_initX, 79 | res.getDimensionPixelOffset(R.dimen.heart_anim_init_x)); 80 | config.initY = (int) typedArray.getDimension(R.styleable.HeartLayout_initY, 81 | res.getDimensionPixelOffset(R.dimen.heart_anim_init_y)); 82 | config.xRand = (int) typedArray.getDimension(R.styleable.HeartLayout_xRand, 83 | res.getDimensionPixelOffset(R.dimen.heart_anim_bezier_x_rand)); 84 | config.animLength = (int) typedArray.getDimension(R.styleable.HeartLayout_animLength, 85 | res.getDimensionPixelOffset(R.dimen.heart_anim_length)); 86 | config.animLengthRand = (int) typedArray.getDimension(R.styleable.HeartLayout_animLengthRand, 87 | res.getDimensionPixelOffset(R.dimen.heart_anim_length_rand)); 88 | config.bezierFactor = typedArray.getInteger(R.styleable.HeartLayout_bezierFactor, 89 | res.getInteger(R.integer.heart_anim_bezier_factor)); 90 | config.xPointFactor = (int) typedArray.getDimension(R.styleable.HeartLayout_xPointFactor, 91 | res.getDimensionPixelOffset(R.dimen.heart_anim_x_point_factor)); 92 | config.heartWidth = (int) typedArray.getDimension(R.styleable.HeartLayout_heart_width, 93 | res.getDimensionPixelOffset(R.dimen.heart_size_width)); 94 | config.heartHeight = (int) typedArray.getDimension(R.styleable.HeartLayout_heart_height, 95 | res.getDimensionPixelOffset(R.dimen.heart_size_height)); 96 | config.animDuration = typedArray.getInteger(R.styleable.HeartLayout_anim_duration, 97 | res.getInteger(R.integer.anim_duration)); 98 | return config; 99 | } 100 | 101 | 102 | } 103 | 104 | 105 | } 106 | 107 | -------------------------------------------------------------------------------- /heartlayout/src/main/java/tyrantgit/widget/HeartLayout.java: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2015 tyrantgit 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 | package tyrantgit.widget; 17 | 18 | import android.content.Context; 19 | import android.content.res.TypedArray; 20 | import android.util.AttributeSet; 21 | import android.widget.RelativeLayout; 22 | 23 | 24 | public class HeartLayout extends RelativeLayout { 25 | 26 | private AbstractPathAnimator mAnimator; 27 | 28 | public HeartLayout(Context context) { 29 | super(context); 30 | init(null, 0); 31 | } 32 | 33 | public HeartLayout(Context context, AttributeSet attrs) { 34 | super(context, attrs); 35 | init(attrs, 0); 36 | } 37 | 38 | public HeartLayout(Context context, AttributeSet attrs, int defStyleAttr) { 39 | super(context, attrs, defStyleAttr); 40 | init(attrs, defStyleAttr); 41 | } 42 | 43 | private void init(AttributeSet attrs, int defStyleAttr) { 44 | 45 | final TypedArray a = getContext().obtainStyledAttributes( 46 | attrs, R.styleable.HeartLayout, defStyleAttr, 0); 47 | 48 | mAnimator = new PathAnimator(AbstractPathAnimator.Config.fromTypeArray(a)); 49 | 50 | a.recycle(); 51 | } 52 | 53 | public AbstractPathAnimator getAnimator() { 54 | return mAnimator; 55 | } 56 | 57 | public void setAnimator(AbstractPathAnimator animator) { 58 | clearAnimation(); 59 | mAnimator = animator; 60 | } 61 | 62 | public void clearAnimation() { 63 | for (int i = 0; i < getChildCount(); i++) { 64 | getChildAt(i).clearAnimation(); 65 | } 66 | removeAllViews(); 67 | } 68 | 69 | public void addHeart(int color) { 70 | HeartView heartView = new HeartView(getContext()); 71 | heartView.setColor(color); 72 | mAnimator.start(heartView, this); 73 | } 74 | 75 | public void addHeart(int color, int heartResId, int heartBorderResId) { 76 | HeartView heartView = new HeartView(getContext()); 77 | heartView.setColorAndDrawables(color, heartResId, heartBorderResId); 78 | mAnimator.start(heartView, this); 79 | } 80 | 81 | } 82 | -------------------------------------------------------------------------------- /heartlayout/src/main/java/tyrantgit/widget/HeartView.java: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2015 tyrantgit 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 | package tyrantgit.widget; 17 | 18 | import android.content.Context; 19 | import android.graphics.Bitmap; 20 | import android.graphics.BitmapFactory; 21 | import android.graphics.Canvas; 22 | import android.graphics.Paint; 23 | import android.graphics.PorterDuff; 24 | import android.graphics.PorterDuffColorFilter; 25 | import android.graphics.drawable.BitmapDrawable; 26 | import android.util.AttributeSet; 27 | import android.widget.ImageView; 28 | 29 | 30 | public class HeartView extends ImageView { 31 | 32 | private static final Paint sPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG); 33 | private int mHeartResId = R.drawable.heart; 34 | private int mHeartBorderResId = R.drawable.heart_border; 35 | private static Bitmap sHeart; 36 | private static Bitmap sHeartBorder; 37 | private static final Canvas sCanvas = new Canvas(); 38 | 39 | public HeartView(Context context, AttributeSet attrs) { 40 | super(context, attrs); 41 | } 42 | 43 | public HeartView(Context context, AttributeSet attrs, int defStyleAttr) { 44 | super(context, attrs, defStyleAttr); 45 | } 46 | 47 | public HeartView(Context context) { 48 | super(context); 49 | } 50 | 51 | public void setColor(int color) { 52 | Bitmap heart = createHeart(color); 53 | setImageDrawable(new BitmapDrawable(getResources(), heart)); 54 | } 55 | 56 | public void setColorAndDrawables(int color, int heartResId, int heartBorderResId) { 57 | if (heartResId != mHeartResId) { 58 | sHeart = null; 59 | } 60 | if (heartBorderResId != mHeartBorderResId) { 61 | sHeartBorder = null; 62 | } 63 | mHeartResId = heartResId; 64 | mHeartBorderResId = heartBorderResId; 65 | setColor(color); 66 | } 67 | 68 | public Bitmap createHeart(int color) { 69 | if (sHeart == null) { 70 | sHeart = BitmapFactory.decodeResource(getResources(), mHeartResId); 71 | } 72 | if (sHeartBorder == null) { 73 | sHeartBorder = BitmapFactory.decodeResource(getResources(), mHeartBorderResId); 74 | } 75 | Bitmap heart = sHeart; 76 | Bitmap heartBorder = sHeartBorder; 77 | Bitmap bm = createBitmapSafely(heartBorder.getWidth(), heartBorder.getHeight()); 78 | if (bm == null) { 79 | return null; 80 | } 81 | Canvas canvas = sCanvas; 82 | canvas.setBitmap(bm); 83 | Paint p = sPaint; 84 | canvas.drawBitmap(heartBorder, 0, 0, p); 85 | p.setColorFilter(new PorterDuffColorFilter(color, PorterDuff.Mode.SRC_ATOP)); 86 | float dx = (heartBorder.getWidth() - heart.getWidth()) / 2f; 87 | float dy = (heartBorder.getHeight() - heart.getHeight()) / 2f; 88 | canvas.drawBitmap(heart, dx, dy, p); 89 | p.setColorFilter(null); 90 | canvas.setBitmap(null); 91 | return bm; 92 | } 93 | 94 | private static Bitmap createBitmapSafely(int width, int height) { 95 | try { 96 | return Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); 97 | } catch (OutOfMemoryError error) { 98 | error.printStackTrace(); 99 | } 100 | return null; 101 | } 102 | 103 | } 104 | -------------------------------------------------------------------------------- /heartlayout/src/main/java/tyrantgit/widget/PathAnimator.java: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (C) 2015 tyrantgit 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 | package tyrantgit.widget; 17 | 18 | import android.graphics.Matrix; 19 | import android.graphics.Path; 20 | import android.graphics.PathMeasure; 21 | import android.os.Handler; 22 | import android.os.Looper; 23 | import android.view.View; 24 | import android.view.ViewGroup; 25 | import android.view.animation.Animation; 26 | import android.view.animation.LinearInterpolator; 27 | import android.view.animation.Transformation; 28 | 29 | import java.util.concurrent.atomic.AtomicInteger; 30 | 31 | public class PathAnimator extends AbstractPathAnimator { 32 | private final AtomicInteger mCounter = new AtomicInteger(0); 33 | private Handler mHandler; 34 | 35 | public PathAnimator(Config config) { 36 | super(config); 37 | mHandler = new Handler(Looper.getMainLooper()); 38 | } 39 | 40 | @Override 41 | public void start(final View child, final ViewGroup parent) { 42 | parent.addView(child, new ViewGroup.LayoutParams(mConfig.heartWidth, mConfig.heartHeight)); 43 | FloatAnimation anim = new FloatAnimation(createPath(mCounter, parent, 2), randomRotation(), parent, child); 44 | anim.setDuration(mConfig.animDuration); 45 | anim.setInterpolator(new LinearInterpolator()); 46 | anim.setAnimationListener(new Animation.AnimationListener() { 47 | @Override 48 | public void onAnimationEnd(Animation animation) { 49 | mHandler.post(new Runnable() { 50 | @Override 51 | public void run() { 52 | parent.removeView(child); 53 | } 54 | }); 55 | mCounter.decrementAndGet(); 56 | } 57 | 58 | @Override 59 | public void onAnimationRepeat(Animation animation) { 60 | 61 | } 62 | 63 | @Override 64 | public void onAnimationStart(Animation animation) { 65 | mCounter.incrementAndGet(); 66 | } 67 | }); 68 | anim.setInterpolator(new LinearInterpolator()); 69 | child.startAnimation(anim); 70 | } 71 | 72 | static class FloatAnimation extends Animation { 73 | private PathMeasure mPm; 74 | private View mView; 75 | private float mDistance; 76 | private float mRotation; 77 | 78 | public FloatAnimation(Path path, float rotation, View parent, View child) { 79 | mPm = new PathMeasure(path, false); 80 | mDistance = mPm.getLength(); 81 | mView = child; 82 | mRotation = rotation; 83 | parent.setLayerType(View.LAYER_TYPE_HARDWARE, null); 84 | } 85 | 86 | @Override 87 | protected void applyTransformation(float factor, Transformation transformation) { 88 | Matrix matrix = transformation.getMatrix(); 89 | mPm.getMatrix(mDistance * factor, matrix, PathMeasure.POSITION_MATRIX_FLAG); 90 | mView.setRotation(mRotation * factor); 91 | float scale = 1F; 92 | if (3000.0F * factor < 200.0F) { 93 | scale = scale(factor, 0.0D, 0.06666667014360428D, 0.20000000298023224D, 1.100000023841858D); 94 | } else if (3000.0F * factor < 300.0F) { 95 | scale = scale(factor, 0.06666667014360428D, 0.10000000149011612D, 1.100000023841858D, 1.0D); 96 | } 97 | mView.setScaleX(scale); 98 | mView.setScaleY(scale); 99 | transformation.setAlpha(1.0F - factor); 100 | } 101 | } 102 | 103 | private static float scale(double a, double b, double c, double d, double e) { 104 | return (float) ((a - b) / (c - b) * (e - d) + d); 105 | } 106 | } 107 | 108 | -------------------------------------------------------------------------------- /heartlayout/src/main/res/drawable/heart.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/heartlayout/src/main/res/drawable/heart.png -------------------------------------------------------------------------------- /heartlayout/src/main/res/drawable/heart_border.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/tyrantgit/HeartLayout/8975c69758241e24206e73d7be8408196ff998a6/heartlayout/src/main/res/drawable/heart_border.png -------------------------------------------------------------------------------- /heartlayout/src/main/res/values/attrs.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | -------------------------------------------------------------------------------- /heartlayout/src/main/res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 40.0dp 5 | 50.0dp 6 | 25.0dp 7 | 100.0dp 8 | 150.0dp 9 | 30.0dp 10 | 11 | 27.3dp 12 | 32.5dp 13 | 14 | 15 | -------------------------------------------------------------------------------- /heartlayout/src/main/res/values/integers.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 6 5 | 3000 6 | 7 | -------------------------------------------------------------------------------- /heartlayout/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | HeartLayout 3 | 4 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | include ':app', ':heartlayout' 2 | --------------------------------------------------------------------------------