├── app
├── .gitignore
├── src
│ ├── main
│ │ ├── res
│ │ │ ├── values
│ │ │ │ ├── strings.xml
│ │ │ │ ├── colors.xml
│ │ │ │ ├── styles.xml
│ │ │ │ └── attrs.xml
│ │ │ ├── mipmap-hdpi
│ │ │ │ ├── ic_launcher.png
│ │ │ │ └── ic_launcher_round.png
│ │ │ ├── mipmap-mdpi
│ │ │ │ ├── ic_launcher.png
│ │ │ │ └── ic_launcher_round.png
│ │ │ ├── mipmap-xhdpi
│ │ │ │ ├── ic_launcher.png
│ │ │ │ └── ic_launcher_round.png
│ │ │ ├── mipmap-xxhdpi
│ │ │ │ ├── ic_launcher.png
│ │ │ │ └── ic_launcher_round.png
│ │ │ ├── mipmap-xxxhdpi
│ │ │ │ ├── ic_launcher.png
│ │ │ │ └── ic_launcher_round.png
│ │ │ └── layout
│ │ │ │ └── activity_main.xml
│ │ ├── AndroidManifest.xml
│ │ └── java
│ │ │ └── com
│ │ │ └── tape_hencoder
│ │ │ ├── MainActivity.java
│ │ │ └── TapeView.java
│ ├── test
│ │ └── java
│ │ │ └── com
│ │ │ └── tape_hencoder
│ │ │ └── ExampleUnitTest.java
│ └── androidTest
│ │ └── java
│ │ └── com
│ │ └── tape_hencoder
│ │ └── ExampleInstrumentedTest.java
├── proguard-rules.pro
└── build.gradle
├── settings.gradle
├── ezgif.com-video-to-gif.gif
├── .idea
├── copyright
│ └── profiles_settings.xml
├── encodings.xml
├── modules.xml
├── runConfigurations.xml
├── gradle.xml
├── compiler.xml
└── misc.xml
├── gradle
└── wrapper
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── .gitignore
├── README.md
├── gradle.properties
├── tapeview
└── src
│ └── main
│ ├── res
│ └── values
│ │ └── attrs.xml
│ └── java
│ └── com
│ └── tapeview
│ └── TapeView.java
├── gradlew.bat
└── gradlew
/app/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/settings.gradle:
--------------------------------------------------------------------------------
1 | include ':app', ':tapeview'
2 |
--------------------------------------------------------------------------------
/ezgif.com-video-to-gif.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/ezgif.com-video-to-gif.gif
--------------------------------------------------------------------------------
/.idea/copyright/profiles_settings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
--------------------------------------------------------------------------------
/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | Tape_Hencoder
3 |
4 |
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-hdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-mdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/A-Heavy-Rain/Tape_Hencoder/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | *.iml
2 | .gradle
3 | /local.properties
4 | /.idea/workspace.xml
5 | /.idea/libraries
6 | .DS_Store
7 | /build
8 | /captures
9 | .externalNativeBuild
10 |
--------------------------------------------------------------------------------
/.idea/encodings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
--------------------------------------------------------------------------------
/app/src/main/res/values/colors.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | #3F51B5
4 | #303F9F
5 | #FF4081
6 |
7 |
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Tue Oct 17 09:36:54 CST 2017
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-3.3-all.zip
7 |
--------------------------------------------------------------------------------
/app/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
10 |
11 |
12 |
--------------------------------------------------------------------------------
/app/src/test/java/com/tape_hencoder/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package com.tape_hencoder;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * Example local unit test, which will execute on the development machine (host).
9 | *
10 | * @see Testing documentation
11 | */
12 | public class ExampleUnitTest {
13 | @Test
14 | public void addition_isCorrect() throws Exception {
15 | assertEquals(4, 2 + 2);
16 | }
17 | }
--------------------------------------------------------------------------------
/.idea/modules.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
--------------------------------------------------------------------------------
/.idea/runConfigurations.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
11 |
12 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | 
2 | # Tape_Hencoder
3 | ## 高度自定义刻度尺,HenCoder仿写系列。
4 | 我所实现的几个重要点:
5 | 1. 支持惯性滑动。
6 | 2. 支持滑动停止后,指示器调整到刻度线之下。
7 | 3. 可以设置初始值、最大值、最小值。
8 | 4. 支持以下自定义属性:
9 | - gap_width 最小刻度所显示出来的宽度
10 | - init_value 初始值
11 | - number_margin_top 数字上边距
12 | - number_margin_bottom 数字下边距
13 | - tape_height 刻度高度
14 | - tape_color 刻度颜色
15 | - tape_width 刻度宽度
16 | - indicator_color 指示器颜色
17 | - indicator_width 指示器宽度
18 | - indicator_height 指示器高度
19 | - number_color 数字颜色
20 | - number_size 数字字体大小
21 | - max_value 最大值
22 | - min_value 最小值
23 | ### 觉得还不错的给个start,哈哈。
24 |
--------------------------------------------------------------------------------
/.idea/gradle.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
--------------------------------------------------------------------------------
/.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 |
--------------------------------------------------------------------------------
/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 | org.gradle.jvmargs=-Xmx1536m
13 |
14 | # When configured, Gradle will run in incubating parallel mode.
15 | # This option should only be used with decoupled projects. More details, visit
16 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
17 | # org.gradle.parallel=true
18 |
--------------------------------------------------------------------------------
/app/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
--------------------------------------------------------------------------------
/app/src/main/java/com/tape_hencoder/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.tape_hencoder;
2 |
3 | import android.support.v7.app.AppCompatActivity;
4 | import android.os.Bundle;
5 | import android.widget.TextView;
6 |
7 | public class MainActivity extends AppCompatActivity {
8 |
9 | @Override
10 | protected void onCreate(Bundle savedInstanceState) {
11 | super.onCreate(savedInstanceState);
12 | setContentView(R.layout.activity_main);
13 | TapeView tapeView= (TapeView) findViewById(R.id.tape_view);
14 | final TextView tv_value= (TextView) findViewById(R.id.tv_value);
15 | tv_value.setText(tapeView.getCurrentValue()+"");
16 | tapeView.setValueListener(new TapeView.ValueListener() {
17 | @Override
18 | public void OnValue(int value) {
19 | tv_value.setText(value+"");
20 | }
21 | });
22 | }
23 | }
24 |
--------------------------------------------------------------------------------
/app/src/androidTest/java/com/tape_hencoder/ExampleInstrumentedTest.java:
--------------------------------------------------------------------------------
1 | package com.tape_hencoder;
2 |
3 | import android.content.Context;
4 | import android.support.test.InstrumentationRegistry;
5 | import android.support.test.runner.AndroidJUnit4;
6 |
7 | import org.junit.Test;
8 | import org.junit.runner.RunWith;
9 |
10 | import static org.junit.Assert.*;
11 |
12 | /**
13 | * Instrumentation test, which will execute on an Android device.
14 | *
15 | * @see Testing documentation
16 | */
17 | @RunWith(AndroidJUnit4.class)
18 | public class ExampleInstrumentedTest {
19 | @Test
20 | public void useAppContext() throws Exception {
21 | // Context of the app under test.
22 | Context appContext = InstrumentationRegistry.getTargetContext();
23 |
24 | assertEquals("com.tape_hencoder", appContext.getPackageName());
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/app/src/main/res/values/attrs.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
--------------------------------------------------------------------------------
/tapeview/src/main/res/values/attrs.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
--------------------------------------------------------------------------------
/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 /Users/chineseskill/Library/Android/sdk/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 |
19 | # Uncomment this to preserve the line number information for
20 | # debugging stack traces.
21 | #-keepattributes SourceFile,LineNumberTable
22 |
23 | # If you keep the line number information, uncomment this to
24 | # hide the original source file name.
25 | #-renamesourcefileattribute SourceFile
26 |
--------------------------------------------------------------------------------
/app/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 |
3 | android {
4 | compileSdkVersion 26
5 | buildToolsVersion "26.0.1"
6 | defaultConfig {
7 | applicationId "com.tape_hencoder"
8 | minSdkVersion 14
9 | targetSdkVersion 26
10 | versionCode 1
11 | versionName "1.0"
12 | testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
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 | androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
25 | exclude group: 'com.android.support', module: 'support-annotations'
26 | })
27 | compile 'com.android.support:appcompat-v7:26.+'
28 | compile 'com.android.support.constraint:constraint-layout:1.0.2'
29 | testCompile 'junit:junit:4.12'
30 | }
31 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/activity_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
10 |
11 |
15 |
16 |
23 |
24 |
31 |
32 |
33 |
42 |
43 |
--------------------------------------------------------------------------------
/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 |
--------------------------------------------------------------------------------
/.idea/misc.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 | 1.8
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
--------------------------------------------------------------------------------
/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 | # Attempt to set APP_HOME
46 | # Resolve links: $0 may be a link
47 | PRG="$0"
48 | # Need this for relative symlinks.
49 | while [ -h "$PRG" ] ; do
50 | ls=`ls -ld "$PRG"`
51 | link=`expr "$ls" : '.*-> \(.*\)$'`
52 | if expr "$link" : '/.*' > /dev/null; then
53 | PRG="$link"
54 | else
55 | PRG=`dirname "$PRG"`"/$link"
56 | fi
57 | done
58 | SAVED="`pwd`"
59 | cd "`dirname \"$PRG\"`/" >/dev/null
60 | APP_HOME="`pwd -P`"
61 | cd "$SAVED" >/dev/null
62 |
63 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
64 |
65 | # Determine the Java command to use to start the JVM.
66 | if [ -n "$JAVA_HOME" ] ; then
67 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
68 | # IBM's JDK on AIX uses strange locations for the executables
69 | JAVACMD="$JAVA_HOME/jre/sh/java"
70 | else
71 | JAVACMD="$JAVA_HOME/bin/java"
72 | fi
73 | if [ ! -x "$JAVACMD" ] ; then
74 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
75 |
76 | Please set the JAVA_HOME variable in your environment to match the
77 | location of your Java installation."
78 | fi
79 | else
80 | JAVACMD="java"
81 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
82 |
83 | Please set the JAVA_HOME variable in your environment to match the
84 | location of your Java installation."
85 | fi
86 |
87 | # Increase the maximum file descriptors if we can.
88 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
89 | MAX_FD_LIMIT=`ulimit -H -n`
90 | if [ $? -eq 0 ] ; then
91 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
92 | MAX_FD="$MAX_FD_LIMIT"
93 | fi
94 | ulimit -n $MAX_FD
95 | if [ $? -ne 0 ] ; then
96 | warn "Could not set maximum file descriptor limit: $MAX_FD"
97 | fi
98 | else
99 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
100 | fi
101 | fi
102 |
103 | # For Darwin, add options to specify how the application appears in the dock
104 | if $darwin; then
105 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
106 | fi
107 |
108 | # For Cygwin, switch paths to Windows format before running java
109 | if $cygwin ; then
110 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
111 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
112 | JAVACMD=`cygpath --unix "$JAVACMD"`
113 |
114 | # We build the pattern for arguments to be converted via cygpath
115 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
116 | SEP=""
117 | for dir in $ROOTDIRSRAW ; do
118 | ROOTDIRS="$ROOTDIRS$SEP$dir"
119 | SEP="|"
120 | done
121 | OURCYGPATTERN="(^($ROOTDIRS))"
122 | # Add a user-defined pattern to the cygpath arguments
123 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
124 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
125 | fi
126 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
127 | i=0
128 | for arg in "$@" ; do
129 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
130 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
131 |
132 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
133 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
134 | else
135 | eval `echo args$i`="\"$arg\""
136 | fi
137 | i=$((i+1))
138 | done
139 | case $i in
140 | (0) set -- ;;
141 | (1) set -- "$args0" ;;
142 | (2) set -- "$args0" "$args1" ;;
143 | (3) set -- "$args0" "$args1" "$args2" ;;
144 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
145 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
146 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
147 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
148 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
149 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
150 | esac
151 | fi
152 |
153 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
154 | function splitJvmOpts() {
155 | JVM_OPTS=("$@")
156 | }
157 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
158 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
159 |
160 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
161 |
--------------------------------------------------------------------------------
/tapeview/src/main/java/com/tapeview/TapeView.java:
--------------------------------------------------------------------------------
1 | package com.tapeview;
2 |
3 | import android.content.Context;
4 | import android.content.res.TypedArray;
5 | import android.graphics.Canvas;
6 | import android.graphics.Color;
7 | import android.graphics.Paint;
8 | import android.graphics.Rect;
9 | import android.support.annotation.Nullable;
10 | import android.text.TextPaint;
11 | import android.util.AttributeSet;
12 | import android.util.Log;
13 | import android.view.MotionEvent;
14 | import android.view.VelocityTracker;
15 | import android.view.View;
16 | import android.view.ViewConfiguration;
17 | import android.widget.OverScroller;
18 |
19 | /**
20 | * Created by zhaozhibo on 2017/10/16.
21 | */
22 |
23 | public class TapeView extends View {
24 | private int gapWidth;
25 | private int width;
26 | private int height;
27 | private int tapeHeight;
28 | /**
29 | * 刻度值宽度
30 | */
31 | private int tapeWidth;
32 | /**
33 | * 这个是指示器距离左边那个刻度的距离(0,tapeWidth)
34 | * 绘制的时候要注意这个数值
35 | */
36 | private int gapOffset;
37 | private int numberMarginTop;
38 | private int numberMarginBottom;
39 | private int numberHeight;
40 | private int numberSize;
41 | private int indicatorWidth;
42 | private int indicatorHeight;
43 |
44 |
45 | private int currentValue;
46 | private int initValue;
47 | private int maxValue;
48 | private int minValue;
49 |
50 | private int tapeColor;
51 | private int indicatorColor;
52 | private int numberColor;
53 |
54 | private Paint tapePaint;
55 | private Paint textPaint;
56 | private Paint indicatorPaint;
57 |
58 | private Rect textRect;
59 |
60 |
61 |
62 | private VelocityTracker velocityTracker;
63 | private OverScroller scroller;
64 | private float maxVelocity;
65 | private float minVelocity;
66 |
67 |
68 | private float lastX;
69 | private float moveX;
70 | /**
71 | * 防止move是触发computeScroll中的else
72 | */
73 | private boolean isCanScroll;
74 | /**
75 | * 防止循环调用
76 | */
77 | private boolean isCanJustScroll;
78 |
79 | public TapeView(Context context) {
80 | this(context, null);
81 | }
82 |
83 | public TapeView(Context context, @Nullable AttributeSet attrs) {
84 | this(context, attrs, 0);
85 | }
86 |
87 | public TapeView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
88 | super(context, attrs, defStyleAttr);
89 | TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.TapeView, 0, 0);
90 | try {
91 | initValue = a.getInteger(R.styleable.TapeView_init_value, 50);
92 | gapWidth = a.getDimensionPixelOffset(R.styleable.TapeView_gap_width, 35);
93 | numberMarginTop = a.getDimensionPixelOffset(R.styleable.TapeView_number_margin_top, 80);
94 | tapeHeight = a.getDimensionPixelOffset(R.styleable.TapeView_tape_height, 60);
95 | tapeColor = a.getColor(R.styleable.TapeView_tape_color, Color.parseColor("#E6E8E5"));
96 | indicatorColor = a.getColor(R.styleable.TapeView_indicator_color, Color.parseColor("#6DB67C"));
97 | numberColor = a.getColor(R.styleable.TapeView_number_color, Color.parseColor("#555855"));
98 | numberSize = a.getDimensionPixelOffset(R.styleable.TapeView_number_size, 48);
99 | indicatorWidth = a.getDimensionPixelOffset(R.styleable.TapeView_indicator_width, 10);
100 | indicatorHeight = a.getDimensionPixelOffset(R.styleable.TapeView_indicator_height, 130);
101 | tapeWidth = a.getDimensionPixelOffset(R.styleable.TapeView_tape_width, 5);
102 | minValue = a.getInteger(R.styleable.TapeView_min_value, -100);
103 | maxValue = a.getInteger(R.styleable.TapeView_max_value, 100);
104 | numberMarginBottom = a.getDimensionPixelOffset(R.styleable.TapeView_number_margin_bottom, 80);
105 | } finally {
106 | a.recycle();
107 | }
108 | init();
109 | }
110 |
111 | private void init() {
112 |
113 |
114 | initPaint();
115 | currentValue = initValue;
116 |
117 |
118 | textRect = new Rect();
119 |
120 | ViewConfiguration vc = ViewConfiguration.get(getContext());
121 | maxVelocity = vc.getScaledMaximumFlingVelocity();
122 | minVelocity = vc.getScaledMinimumFlingVelocity();
123 |
124 | scroller = new OverScroller(getContext());
125 |
126 | //直接获取一下数字高度
127 | textPaint.getTextBounds("1", 0, 1, textRect);
128 | numberHeight = textRect.height();
129 | }
130 |
131 | private void initPaint() {
132 | tapePaint = new Paint();
133 | tapePaint.setColor(tapeColor);
134 | tapePaint.setAntiAlias(true);
135 | tapePaint.setStyle(Paint.Style.FILL);
136 | tapePaint.setStrokeWidth(tapeWidth);
137 |
138 |
139 | textPaint = new TextPaint();
140 | textPaint.setTextSize(numberSize);
141 | textPaint.setAntiAlias(true);
142 | textPaint.setColor(numberColor);
143 |
144 | indicatorPaint = new Paint();
145 | indicatorPaint.setColor(indicatorColor);
146 | indicatorPaint.setStrokeWidth(indicatorWidth);
147 | indicatorPaint.setAntiAlias(true);
148 | indicatorPaint.setStrokeCap(Paint.Cap.ROUND);
149 | }
150 |
151 | @Override
152 | protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
153 | super.onMeasure(widthMeasureSpec, heightMeasureSpec);
154 | width = MeasureSpec.getSize(widthMeasureSpec);
155 | int heightModel = MeasureSpec.getMode(heightMeasureSpec);
156 | int heightSize = MeasureSpec.getSize(heightMeasureSpec);
157 | if (heightModel == MeasureSpec.AT_MOST) {
158 | height = tapeHeight * 2 + numberHeight + numberMarginTop + numberMarginBottom;
159 | } else {
160 | height = heightSize;
161 | }
162 | setMeasuredDimension(width, height);
163 |
164 | }
165 |
166 | @Override
167 | protected void onDraw(Canvas canvas) {
168 | super.onDraw(canvas);
169 | if (gapOffset < 0 || gapOffset > gapWidth) {
170 | throw new IllegalStateException("gapOffset is illegalState");
171 | }
172 | drawLeft(canvas);
173 | drawRight(canvas);
174 | drawIndicator(canvas);
175 |
176 | }
177 |
178 | private void drawIndicator(Canvas canvas) {
179 | canvas.drawLine(width / 2, 0, width / 2, indicatorHeight, indicatorPaint);
180 | }
181 |
182 | private void drawLeft(Canvas canvas) {
183 | int drawLength = gapOffset;
184 | int currentValue = this.currentValue;
185 | int totalWidth = width / 2;
186 | //多加一个gapWidth 防止数字消失太突然
187 | while (drawLength < width / 2 + gapWidth) {
188 | if (currentValue < minValue)
189 | break;
190 | if (currentValue % 10 == 0) {
191 | canvas.drawLine(totalWidth - drawLength, 0, totalWidth - drawLength, tapeHeight * 2, tapePaint);
192 | drawText(canvas, currentValue, totalWidth - drawLength);
193 | } else {
194 | canvas.drawLine(totalWidth - drawLength, 0, totalWidth - drawLength, tapeHeight, tapePaint);
195 | }
196 | drawLength += gapWidth;
197 | currentValue--;
198 | }
199 |
200 | }
201 |
202 | private void drawRight(Canvas canvas) {
203 | int drawLength = gapWidth - gapOffset;
204 | int currentValue = this.currentValue + 1;
205 | int totalWidth = width / 2;
206 | //多加一个gapWidth 防止数字消失太突然
207 | while (drawLength < width / 2 + gapWidth) {
208 | if (currentValue > maxValue)
209 | break;
210 | if (currentValue % 10 == 0) {
211 | canvas.drawLine(totalWidth + drawLength, 0, totalWidth + drawLength, tapeHeight * 2, tapePaint);
212 | drawText(canvas, currentValue, totalWidth + drawLength);
213 | } else {
214 | canvas.drawLine(totalWidth + drawLength, 0, totalWidth + drawLength, tapeHeight, tapePaint);
215 | }
216 | drawLength += gapWidth;
217 | currentValue++;
218 | }
219 |
220 | }
221 |
222 | private void drawText(Canvas canvas, int value, int x) {
223 | String drawStr = value + "";
224 | textPaint.getTextBounds(drawStr, 0, drawStr.length(), textRect);
225 | float width = textPaint.measureText(drawStr);
226 | canvas.drawText(drawStr, x - width / 2, tapeHeight * 2 + numberHeight + numberMarginTop, textPaint);
227 | }
228 |
229 |
230 | @Override
231 | public boolean onTouchEvent(MotionEvent event) {
232 | acquireVelocityTracker(event);
233 |
234 | switch (event.getAction()) {
235 | case MotionEvent.ACTION_DOWN:
236 | scroller.forceFinished(true);
237 | isCanScroll = false;
238 | break;
239 | case MotionEvent.ACTION_MOVE:
240 | float dx = event.getX() - lastX;
241 | moveX += dx;
242 |
243 | //根据moveX的值判断是否超出最大最小值范围,分情况处理
244 | if ((moveX > (initValue - minValue) * gapWidth && dx > 0)) {
245 | moveX = (initValue - minValue) * gapWidth;
246 | currentValue = minValue;
247 | gapOffset = 0;
248 | } else if ((moveX < -(maxValue - initValue) * gapWidth && dx < 0)) {
249 | moveX = -(maxValue - initValue) * gapWidth;
250 | currentValue = maxValue;
251 | gapOffset = 0;
252 | } else {
253 | float moveValue = initValue - (moveX / gapWidth);
254 | //处理负数情况
255 | if (moveValue > 0) {
256 | currentValue = (int) moveValue;
257 | } else {
258 | currentValue = (int) moveValue - 1;
259 | }
260 | //根据左右移动判断gapOffset大小
261 | if (moveX > 0) {
262 | gapOffset = (int) (gapWidth - Math.abs(moveX % gapWidth));
263 | } else {
264 | gapOffset = (int) Math.abs(moveX % gapWidth);
265 | }
266 | }
267 | if (valueListener != null) {
268 | if (gapOffset > gapWidth / 2) {
269 | valueListener.OnValue(currentValue + 1);
270 | } else {
271 | valueListener.OnValue(currentValue);
272 |
273 | }
274 | }
275 |
276 |
277 | invalidate();
278 | isCanScroll = false;
279 | break;
280 | case MotionEvent.ACTION_UP:
281 | case MotionEvent.ACTION_CANCEL:
282 | velocityTracker.computeCurrentVelocity(1000, maxVelocity);
283 | float xVelocity = velocityTracker.getXVelocity();
284 |
285 | isCanJustScroll = true;
286 | isCanScroll = true;
287 |
288 | if (Math.abs(xVelocity) > 2 * minVelocity) {
289 |
290 | scroller.fling((int) event.getX(), 0, (int) xVelocity, 0, Integer.MIN_VALUE,
291 | Integer.MAX_VALUE, 0, 0);
292 | invalidate();
293 |
294 | } else {
295 | Log.i("gapOffset", gapOffset + "");
296 | if (gapOffset > gapWidth / 2) {
297 |
298 | adjustScroll((int) event.getX(), -(gapWidth - gapOffset));
299 | } else {
300 | adjustScroll((int) event.getX(), gapOffset);
301 | }
302 | }
303 |
304 | break;
305 | default:
306 | break;
307 | }
308 | lastX = event.getX();
309 | return true;
310 |
311 | }
312 |
313 | private void acquireVelocityTracker(final MotionEvent event) {
314 | if (null == velocityTracker) {
315 | velocityTracker = VelocityTracker.obtain();
316 | }
317 | velocityTracker.addMovement(event);
318 | }
319 |
320 |
321 | @Override
322 | public void computeScroll() {
323 | super.computeScroll();
324 | if (scroller.computeScrollOffset()) {
325 | float dx = scroller.getCurrX() - lastX;
326 | moveX += dx;
327 | //根据moveX的值判断是否超出最大最小值范围,分情况处理
328 | if ((moveX > (initValue - minValue) * gapWidth && dx > 0)) {
329 | moveX = (initValue - minValue) * gapWidth;
330 | currentValue = minValue;
331 | gapOffset = 0;
332 | } else if ((moveX < -(maxValue - initValue) * gapWidth && dx < 0)) {
333 | moveX = -(maxValue - initValue) * gapWidth;
334 | currentValue = maxValue;
335 | gapOffset = 0;
336 | } else {
337 | float moveValue = initValue - (moveX / gapWidth);
338 | //处理负数情况
339 | if (moveValue > 0) {
340 | currentValue = (int) moveValue;
341 | } else {
342 | currentValue = (int) moveValue - 1;
343 | }
344 | //根据左右移动判断gapOffset大小
345 | if (moveX > 0) {
346 | gapOffset = (int) (gapWidth - Math.abs(moveX % gapWidth));
347 | } else {
348 | gapOffset = (int) Math.abs(moveX % gapWidth);
349 | }
350 | if (valueListener != null) {
351 | if (gapOffset > gapWidth / 2) {
352 | valueListener.OnValue(currentValue + 1);
353 | } else {
354 | valueListener.OnValue(currentValue);
355 |
356 | }
357 | }
358 | }
359 | lastX = scroller.getCurrX();
360 | if (valueListener != null) {
361 | if (gapOffset > gapWidth / 2) {
362 | valueListener.OnValue(currentValue + 1);
363 | } else {
364 | valueListener.OnValue(currentValue);
365 |
366 | }
367 | }
368 |
369 | invalidate();
370 |
371 | } else if (isCanScroll && isCanJustScroll) {
372 | isCanJustScroll = false;
373 |
374 | if (gapOffset > gapWidth / 2) {
375 | adjustScroll(scroller.getFinalX(), -(gapWidth - gapOffset));
376 | } else {
377 | adjustScroll(scroller.getFinalX(), gapOffset);
378 | }
379 |
380 | }
381 | }
382 |
383 | /**
384 | * 用于滑动后 指示器没有刚好指到刻度值上 进行微调
385 | *
386 | * @param start 起始X坐标
387 | * @param distance 移动差值
388 | */
389 | private void adjustScroll(int start, int distance) {
390 | scroller.startScroll(start, 0, distance, 0);
391 | invalidate();
392 | }
393 |
394 | private ValueListener valueListener;
395 |
396 | public interface ValueListener {
397 | void OnValue(int value);
398 | }
399 |
400 | /**
401 | * 监听刻度值
402 | *
403 | * @param valueListener
404 | */
405 | public void setValueListener(ValueListener valueListener) {
406 | this.valueListener = valueListener;
407 | }
408 |
409 | public int getCurrentValue() {
410 | return this.currentValue;
411 | }
412 | }
413 |
--------------------------------------------------------------------------------
/app/src/main/java/com/tape_hencoder/TapeView.java:
--------------------------------------------------------------------------------
1 | package com.tape_hencoder;
2 |
3 | import android.content.Context;
4 | import android.content.res.TypedArray;
5 | import android.graphics.Canvas;
6 | import android.graphics.Color;
7 | import android.graphics.Paint;
8 | import android.graphics.Rect;
9 | import android.support.annotation.Nullable;
10 | import android.text.TextPaint;
11 | import android.util.AttributeSet;
12 | import android.util.Log;
13 | import android.view.MotionEvent;
14 | import android.view.VelocityTracker;
15 | import android.view.View;
16 | import android.view.ViewConfiguration;
17 | import android.widget.OverScroller;
18 |
19 | /**
20 | * Created by zhaozhibo on 2017/10/16.
21 | */
22 |
23 | public class TapeView extends View {
24 | private int gapWidth;
25 | private int width;
26 | private int height;
27 | private int tapeHeight;
28 | /**
29 | * 刻度值宽度
30 | */
31 | private int tapeWidth;
32 | /**
33 | * 这个是指示器距离左边那个刻度的距离(0,tapeWidth)
34 | * 绘制的时候要注意这个数值
35 | */
36 | private int gapOffset;
37 | private int numberMarginTop;
38 | private int numberMarginBottom;
39 | private int numberHeight;
40 | private int numberSize;
41 | private int indicatorWidth;
42 | private int indicatorHeight;
43 |
44 |
45 | private int currentValue;
46 | private int initValue;
47 | private int maxValue;
48 | private int minValue;
49 |
50 | private int tapeColor;
51 | private int indicatorColor;
52 | private int numberColor;
53 |
54 | private Paint tapePaint;
55 | private Paint textPaint;
56 | private Paint indicatorPaint;
57 |
58 | private Rect textRect;
59 |
60 |
61 |
62 | private VelocityTracker velocityTracker;
63 | private OverScroller scroller;
64 | private float maxVelocity;
65 | private float minVelocity;
66 |
67 |
68 | private float lastX;
69 | private float moveX;
70 | /**
71 | * 防止move是触发computeScroll中的else
72 | */
73 | private boolean isCanScroll;
74 | /**
75 | * 防止循环调用
76 | */
77 | private boolean isCanJustScroll;
78 |
79 | public TapeView(Context context) {
80 | this(context, null);
81 | }
82 |
83 | public TapeView(Context context, @Nullable AttributeSet attrs) {
84 | this(context, attrs, 0);
85 | }
86 |
87 | public TapeView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
88 | super(context, attrs, defStyleAttr);
89 | TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.TapeView, 0, 0);
90 | try {
91 | initValue = a.getInteger(R.styleable.TapeView_init_value, 50);
92 | gapWidth = a.getDimensionPixelOffset(R.styleable.TapeView_gap_width, 35);
93 | numberMarginTop = a.getDimensionPixelOffset(R.styleable.TapeView_number_margin_top, 80);
94 | tapeHeight = a.getDimensionPixelOffset(R.styleable.TapeView_tape_height, 60);
95 | tapeColor = a.getColor(R.styleable.TapeView_tape_color, Color.parseColor("#E6E8E5"));
96 | indicatorColor = a.getColor(R.styleable.TapeView_indicator_color, Color.parseColor("#6DB67C"));
97 | numberColor = a.getColor(R.styleable.TapeView_number_color, Color.parseColor("#555855"));
98 | numberSize = a.getDimensionPixelOffset(R.styleable.TapeView_number_size, 48);
99 | indicatorWidth = a.getDimensionPixelOffset(R.styleable.TapeView_indicator_width, 10);
100 | indicatorHeight = a.getDimensionPixelOffset(R.styleable.TapeView_indicator_height, 130);
101 | tapeWidth = a.getDimensionPixelOffset(R.styleable.TapeView_tape_width, 5);
102 | minValue = a.getInteger(R.styleable.TapeView_min_value, -100);
103 | maxValue = a.getInteger(R.styleable.TapeView_max_value, 100);
104 | numberMarginBottom = a.getDimensionPixelOffset(R.styleable.TapeView_number_margin_bottom, 80);
105 | } finally {
106 | a.recycle();
107 | }
108 | init();
109 | }
110 |
111 | private void init() {
112 |
113 |
114 | initPaint();
115 | currentValue = initValue;
116 |
117 |
118 | textRect = new Rect();
119 |
120 | ViewConfiguration vc = ViewConfiguration.get(getContext());
121 | maxVelocity = vc.getScaledMaximumFlingVelocity();
122 | minVelocity = vc.getScaledMinimumFlingVelocity();
123 |
124 | scroller = new OverScroller(getContext());
125 |
126 | //直接获取一下数字高度
127 | textPaint.getTextBounds("1", 0, 1, textRect);
128 | numberHeight = textRect.height();
129 | }
130 |
131 | private void initPaint() {
132 | tapePaint = new Paint();
133 | tapePaint.setColor(tapeColor);
134 | tapePaint.setAntiAlias(true);
135 | tapePaint.setStyle(Paint.Style.FILL);
136 | tapePaint.setStrokeWidth(tapeWidth);
137 |
138 |
139 | textPaint = new TextPaint();
140 | textPaint.setTextSize(numberSize);
141 | textPaint.setAntiAlias(true);
142 | textPaint.setColor(numberColor);
143 |
144 | indicatorPaint = new Paint();
145 | indicatorPaint.setColor(indicatorColor);
146 | indicatorPaint.setStrokeWidth(indicatorWidth);
147 | indicatorPaint.setAntiAlias(true);
148 | indicatorPaint.setStrokeCap(Paint.Cap.ROUND);
149 | }
150 |
151 | @Override
152 | protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
153 | super.onMeasure(widthMeasureSpec, heightMeasureSpec);
154 | width = MeasureSpec.getSize(widthMeasureSpec);
155 | int heightModel = MeasureSpec.getMode(heightMeasureSpec);
156 | int heightSize = MeasureSpec.getSize(heightMeasureSpec);
157 | if (heightModel == MeasureSpec.AT_MOST) {
158 | height = tapeHeight * 2 + numberHeight + numberMarginTop + numberMarginBottom;
159 | } else {
160 | height = heightSize;
161 | }
162 | setMeasuredDimension(width, height);
163 |
164 | }
165 |
166 | @Override
167 | protected void onDraw(Canvas canvas) {
168 | super.onDraw(canvas);
169 | if (gapOffset < 0 || gapOffset > gapWidth) {
170 | throw new IllegalStateException("gapOffset is illegalState");
171 | }
172 | drawLeft(canvas);
173 | drawRight(canvas);
174 | drawIndicator(canvas);
175 |
176 | }
177 |
178 | private void drawIndicator(Canvas canvas) {
179 | canvas.drawLine(width / 2, 0, width / 2, indicatorHeight, indicatorPaint);
180 | }
181 |
182 | private void drawLeft(Canvas canvas) {
183 | int drawLength = gapOffset;
184 | int currentValue = this.currentValue;
185 | int totalWidth = width / 2;
186 | //多加一个gapWidth 防止数字消失太突然
187 | while (drawLength < width / 2 + gapWidth) {
188 | if (currentValue < minValue)
189 | break;
190 | if (currentValue % 10 == 0) {
191 | canvas.drawLine(totalWidth - drawLength, 0, totalWidth - drawLength, tapeHeight * 2, tapePaint);
192 | drawText(canvas, currentValue, totalWidth - drawLength);
193 | } else {
194 | canvas.drawLine(totalWidth - drawLength, 0, totalWidth - drawLength, tapeHeight, tapePaint);
195 | }
196 | drawLength += gapWidth;
197 | currentValue--;
198 | }
199 |
200 | }
201 |
202 | private void drawRight(Canvas canvas) {
203 | int drawLength = gapWidth - gapOffset;
204 | int currentValue = this.currentValue + 1;
205 | int totalWidth = width / 2;
206 | //多加一个gapWidth 防止数字消失太突然
207 | while (drawLength < width / 2 + gapWidth) {
208 | if (currentValue > maxValue)
209 | break;
210 | if (currentValue % 10 == 0) {
211 | canvas.drawLine(totalWidth + drawLength, 0, totalWidth + drawLength, tapeHeight * 2, tapePaint);
212 | drawText(canvas, currentValue, totalWidth + drawLength);
213 | } else {
214 | canvas.drawLine(totalWidth + drawLength, 0, totalWidth + drawLength, tapeHeight, tapePaint);
215 | }
216 | drawLength += gapWidth;
217 | currentValue++;
218 | }
219 |
220 | }
221 |
222 | private void drawText(Canvas canvas, int value, int x) {
223 | String drawStr = value + "";
224 | textPaint.getTextBounds(drawStr, 0, drawStr.length(), textRect);
225 | float width = textPaint.measureText(drawStr);
226 | canvas.drawText(drawStr, x - width / 2, tapeHeight * 2 + numberHeight + numberMarginTop, textPaint);
227 | }
228 |
229 |
230 | @Override
231 | public boolean onTouchEvent(MotionEvent event) {
232 | acquireVelocityTracker(event);
233 |
234 | switch (event.getAction()) {
235 | case MotionEvent.ACTION_DOWN:
236 | scroller.forceFinished(true);
237 | isCanScroll = false;
238 | break;
239 | case MotionEvent.ACTION_MOVE:
240 | float dx = event.getX() - lastX;
241 | moveX += dx;
242 |
243 | //根据moveX的值判断是否超出最大最小值范围,分情况处理
244 | if ((moveX > (initValue - minValue) * gapWidth && dx > 0)) {
245 | moveX = (initValue - minValue) * gapWidth;
246 | currentValue = minValue;
247 | gapOffset = 0;
248 | } else if ((moveX < -(maxValue - initValue) * gapWidth && dx < 0)) {
249 | moveX = -(maxValue - initValue) * gapWidth;
250 | currentValue = maxValue;
251 | gapOffset = 0;
252 | } else {
253 | float moveValue = initValue - (moveX / gapWidth);
254 | //处理负数情况
255 | if (moveValue > 0) {
256 | currentValue = (int) moveValue;
257 | } else {
258 | currentValue = (int) moveValue - 1;
259 | }
260 | //根据左右移动判断gapOffset大小
261 | if (moveX > 0) {
262 | gapOffset = (int) (gapWidth - Math.abs(moveX % gapWidth));
263 | } else {
264 | gapOffset = (int) Math.abs(moveX % gapWidth);
265 | }
266 | }
267 | if (valueListener != null) {
268 | if (gapOffset > gapWidth / 2) {
269 | valueListener.OnValue(currentValue + 1);
270 | } else {
271 | valueListener.OnValue(currentValue);
272 |
273 | }
274 | }
275 |
276 |
277 | invalidate();
278 | isCanScroll = false;
279 | break;
280 | case MotionEvent.ACTION_UP:
281 | case MotionEvent.ACTION_CANCEL:
282 | velocityTracker.computeCurrentVelocity(1000, maxVelocity);
283 | float xVelocity = velocityTracker.getXVelocity();
284 |
285 | isCanJustScroll = true;
286 | isCanScroll = true;
287 |
288 | if (Math.abs(xVelocity) > 2 * minVelocity) {
289 |
290 | scroller.fling((int) event.getX(), 0, (int) xVelocity, 0, Integer.MIN_VALUE,
291 | Integer.MAX_VALUE, 0, 0);
292 | invalidate();
293 |
294 | } else {
295 | Log.i("gapOffset", gapOffset + "");
296 | if (gapOffset > gapWidth / 2) {
297 |
298 | adjustScroll((int) event.getX(), -(gapWidth - gapOffset));
299 | } else {
300 | adjustScroll((int) event.getX(), gapOffset);
301 | }
302 | }
303 |
304 | break;
305 | default:
306 | break;
307 | }
308 | lastX = event.getX();
309 | return true;
310 |
311 | }
312 |
313 | private void acquireVelocityTracker(final MotionEvent event) {
314 | if (null == velocityTracker) {
315 | velocityTracker = VelocityTracker.obtain();
316 | }
317 | velocityTracker.addMovement(event);
318 | }
319 |
320 |
321 | @Override
322 | public void computeScroll() {
323 | super.computeScroll();
324 | if (scroller.computeScrollOffset()) {
325 | float dx = scroller.getCurrX() - lastX;
326 | moveX += dx;
327 | //根据moveX的值判断是否超出最大最小值范围,分情况处理
328 | if ((moveX > (initValue - minValue) * gapWidth && dx > 0)) {
329 | moveX = (initValue - minValue) * gapWidth;
330 | currentValue = minValue;
331 | gapOffset = 0;
332 | } else if ((moveX < -(maxValue - initValue) * gapWidth && dx < 0)) {
333 | moveX = -(maxValue - initValue) * gapWidth;
334 | currentValue = maxValue;
335 | gapOffset = 0;
336 | } else {
337 | float moveValue = initValue - (moveX / gapWidth);
338 | //处理负数情况
339 | if (moveValue > 0) {
340 | currentValue = (int) moveValue;
341 | } else {
342 | currentValue = (int) moveValue - 1;
343 | }
344 | //根据左右移动判断gapOffset大小
345 | if (moveX > 0) {
346 | gapOffset = (int) (gapWidth - Math.abs(moveX % gapWidth));
347 | } else {
348 | gapOffset = (int) Math.abs(moveX % gapWidth);
349 | }
350 | if (valueListener != null) {
351 | if (gapOffset > gapWidth / 2) {
352 | valueListener.OnValue(currentValue + 1);
353 | } else {
354 | valueListener.OnValue(currentValue);
355 |
356 | }
357 | }
358 | }
359 | lastX = scroller.getCurrX();
360 | if (valueListener != null) {
361 | if (gapOffset > gapWidth / 2) {
362 | valueListener.OnValue(currentValue + 1);
363 | } else {
364 | valueListener.OnValue(currentValue);
365 |
366 | }
367 | }
368 |
369 | invalidate();
370 |
371 | } else if (isCanScroll && isCanJustScroll) {
372 | isCanJustScroll = false;
373 |
374 | if (gapOffset > gapWidth / 2) {
375 | adjustScroll(scroller.getFinalX(), -(gapWidth - gapOffset));
376 | } else {
377 | adjustScroll(scroller.getFinalX(), gapOffset);
378 | }
379 |
380 | }
381 | }
382 |
383 | /**
384 | * 用于滑动后 指示器没有刚好指到刻度值上 进行微调
385 | *
386 | * @param start 起始X坐标
387 | * @param distance 移动差值
388 | */
389 | private void adjustScroll(int start, int distance) {
390 | scroller.startScroll(start, 0, distance, 0);
391 | invalidate();
392 | }
393 |
394 | private ValueListener valueListener;
395 |
396 | public interface ValueListener {
397 | void OnValue(int value);
398 | }
399 |
400 | /**
401 | * 监听刻度值
402 | *
403 | * @param valueListener
404 | */
405 | public void setValueListener(ValueListener valueListener) {
406 | this.valueListener = valueListener;
407 | }
408 |
409 | public int getCurrentValue() {
410 | return this.currentValue;
411 | }
412 | }
413 |
--------------------------------------------------------------------------------