84 |
85 |
86 |
87 |
88 |
89 |
--------------------------------------------------------------------------------
/library/build.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
7 |
8 |
9 |
29 |
30 |
31 |
40 |
41 |
42 |
43 |
47 |
48 |
49 |
51 |
63 |
64 |
82 |
83 |
84 |
85 |
86 |
90 |
91 |
92 |
93 |
--------------------------------------------------------------------------------
/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 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # 最火Android开源项目SystemBarTint使用
2 | ---
3 | 开源地址:[https://github.com/open-android/SystemBarTint](https://github.com/open-android/SystemBarTint "开源项目地址")
4 |
5 | PS:如果觉得文章太长,你也可观看该课程的[视频](https://www.boxuegu.com/web/html/video.html?courseId=172§ionId=8a2c9bed5a3a4c7e015a3a69cb6e0072&chapterId=8a2c9bed5a3a4c7e015a3a6a2fce0073&vId=8a2c9bed5a3a4c7e015a3a6a95dd0074&videoId=A85FCED9A07D49719C33DC5901307461),亲,里面还有高清,无码的福利喔
6 |
7 |
8 | 修改状态栏颜色,可保持与actionbar 一致
9 |
10 |
11 | # 运行效果
12 |  
13 |
14 |
15 |
16 | * 爱生活,爱学习,更爱做代码的搬运工,分类查找更方便请下载黑马助手app
17 |
18 | 
19 |
20 |
21 | ## 使用步骤
22 |
23 | ### 1. 在project的build.gradle添加如下代码(如下图)
24 |
25 | allprojects {
26 | repositories {
27 | ...
28 | maven { url "https://jitpack.io" }
29 | }
30 | }
31 |
32 | 
33 |
34 |
35 | ### 2. 在Module的build.gradle添加依赖
36 |
37 | compile 'com.github.open-android:SystemBarTint:v1.0.0'
38 |
39 |
40 | ### 3. 复制如下代码到xml
41 |
42 |
46 |
52 |
53 |
54 |
55 | ### 4. 拷贝以下内容到 res/string.xml中
56 |
57 | > 也可以自己定义自己的字符串。
58 |
59 | Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam ligula sapien, fermentum vitae aliquam sed, lacinia in nibh. Pellentesque euismod diam eget justo interdum, sit amet elementum ipsum vulputate. Aliquam sed purus urna. Vivamus luctus nisi sem, a fringilla velit iaculis id. Donec nec vulputate justo. Fusce vulputate sodales tellus, in fringilla ante posuere non. Vivamus vulputate nisl non augue dignissim, consectetur rutrum risus facilisis. Sed consectetur aliquam dolor, sed pulvinar mi tristique vitae. Donec quis ligula quis velit consequat dictum id eu felis. Mauris ac congue ante, sed elementum eros.\n\nSed eu euismod dolor, a vestibulum lorem. Phasellus euismod volutpat risus sit amet pellentesque. Proin sed massa eget risus malesuada dignissim et in ipsum..\n\nNullam risus felis, dictum et varius eget, rutrum non nunc. Vestibulum lorem nulla, porttitor ac sapien interdum, porta tristique turpis. Quisque ut dui vitae urna congue scelerisque. Nulla eu commodo felis. Nulla aliquam magna a arcu elementum, vitae tincidunt risus semper. Sed consectetur diam vel magna mattis, sed congue nisi fringilla. Ut in facilisis elit. Etiam aliquet orci urna. Nam interdum nunc fringilla iaculis cursus.
60 |
61 | ### 5. 拷贝以下内容到 res/color.xml中
62 |
63 | > 也可以自己定义颜色。
64 |
65 | #FF0099EE
66 | #FF0099cc
67 |
68 | ### 6. 拷贝以下内容到res/style.xml 中
69 |
70 |
73 |
74 |
77 |
78 | ###7. 拷贝以下内容到activity中
79 | > 注意 : 应该继承的是Activity ,而不是AppCompatActivity
80 |
81 | @Override
82 | protected void onCreate(Bundle savedInstanceState) {
83 | super.onCreate(savedInstanceState);
84 | setContentView(R.layout.activity_main);
85 |
86 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
87 | setTranslucentStatus(true);
88 | }
89 |
90 | SystemBarTintManager tintManager = new SystemBarTintManager(this);
91 | tintManager.setStatusBarTintEnabled(true);
92 | //设置状态栏颜色
93 | tintManager.setStatusBarTintResource(R.color.statusbar_bg);
94 |
95 | }
96 |
97 | @TargetApi(19)
98 | private void setTranslucentStatus(boolean on) {
99 | Window win = getWindow();
100 | WindowManager.LayoutParams winParams = win.getAttributes();
101 | final int bits = WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS;
102 | if (on) {
103 | winParams.flags |= bits;
104 | } else {
105 | winParams.flags &= ~bits;
106 | }
107 | win.setAttributes(winParams);
108 | }
109 |
110 | ###8 . 注册activity的时候,请指定主题样式为之前定义的主题
111 |
112 |
113 |
115 |
116 |
117 |
118 |
119 |
120 |
121 |
122 | * 注意细节
123 |
124 | 1. 如果想实现上面右图效果,呈现透明色状态栏的。只需要修改布局的根标签的fitsSystemWindows 为false,并且搭配透明的颜色即可。
125 |
126 | //此处修改为false
130 |
131 | 2. 配合透明颜色即可
132 |
133 | #330000ff
134 | #33ff0000
135 |
136 | 3. 界面应该继承的是Activity ,而不是AppCompatActivity
137 |
138 |
139 | * 详细的使用方法在DEMO里面都演示啦,如果你觉得这个库还不错,请赏我一颗star吧~~~
140 |
141 | * 欢迎关注微信公众号
142 |
143 | 
144 |
--------------------------------------------------------------------------------
/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | SystemBarTint
3 |
4 | Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam ligula sapien, fermentum vitae aliquam sed, lacinia in nibh. Pellentesque euismod diam eget justo interdum, sit amet elementum ipsum vulputate. Aliquam sed purus urna. Vivamus luctus nisi sem, a fringilla velit iaculis id. Donec nec vulputate justo. Fusce vulputate sodales tellus, in fringilla ante posuere non. Vivamus vulputate nisl non augue dignissim, consectetur rutrum risus facilisis. Sed consectetur aliquam dolor, sed pulvinar mi tristique vitae. Donec quis ligula quis velit consequat dictum id eu felis. Mauris ac congue ante, sed elementum eros.\n\nSed eu euismod dolor, a vestibulum lorem. Phasellus euismod volutpat risus sit amet pellentesque. Integer sit amet nisl malesuada, posuere orci sodales, euismod metus. Suspendisse potenti. Nullam mollis tristique accumsan. Vestibulum sit amet dictum nisl. Sed rhoncus lacus purus, vitae scelerisque urna venenatis at.\n\nUt ornare sapien sed condimentum varius. Nulla ut varius nibh. Nam consequat ligula sem, interdum hendrerit diam commodo sed. Vestibulum imperdiet mauris quis mauris vestibulum lobortis. Nunc bibendum, nunc ut porttitor tristique, turpis mauris cursus magna, ac molestie risus hendrerit odio. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Cras pulvinar odio vestibulum ligula mollis, vel venenatis orci semper. Duis consectetur molestie justo, ut pulvinar nisi semper vel. Cras quis fringilla augue, ut vehicula turpis. Praesent nec imperdiet dui. Morbi quis pharetra lectus. Interdum et malesuada fames ac ante ipsum primis in faucibus. Sed luctus mauris vitae augue rhoncus, vel interdum nisl pellentesque. In tempor, mi et vestibulum condimentum, nisl mauris cursus diam, aliquet vestibulum risus erat sit amet diam.\n\nInteger suscipit eros at diam tempor, ac condimentum turpis pharetra. Vestibulum pharetra risus in lacus semper, at bibendum ligula condimentum. Proin sed massa eget risus malesuada dignissim et in ipsum. Donec convallis tellus vel lectus lobortis, non lobortis libero faucibus. Sed nec posuere mi. Pellentesque sed sem sit amet erat lobortis commodo nec ut nisl. Pellentesque rhoncus risus id massa accumsan, hendrerit posuere mauris pulvinar.\n\nNullam risus felis, dictum et varius eget, rutrum non nunc. Vestibulum lorem nulla, porttitor ac sapien interdum, porta tristique turpis. Quisque ut dui vitae urna congue scelerisque. Nulla eu commodo felis. Nulla aliquam magna a arcu elementum, vitae tincidunt risus semper. Sed consectetur diam vel magna mattis, sed congue nisi fringilla. Ut in facilisis elit. Etiam aliquet orci urna. Nam interdum nunc fringilla iaculis cursus.\n\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Nam ligula sapien, fermentum vitae aliquam sed, lacinia in nibh. Pellentesque euismod diam eget justo interdum, sit amet elementum ipsum vulputate. Aliquam sed purus urna. Vivamus luctus nisi sem, a fringilla velit iaculis id. Donec nec vulputate justo. Fusce vulputate sodales tellus, in fringilla ante posuere non. Vivamus vulputate nisl non augue dignissim, consectetur rutrum risus facilisis. Sed consectetur aliquam dolor, sed pulvinar mi tristique vitae. Donec quis ligula quis velit consequat dictum id eu felis. Mauris ac congue ante, sed elementum eros.\n\nSed eu euismod dolor, a vestibulum lorem. Phasellus euismod volutpat risus sit amet pellentesque. Integer sit amet nisl malesuada, posuere orci sodales, euismod metus. Suspendisse potenti. Nullam mollis tristique accumsan. Vestibulum sit amet dictum nisl. Sed rhoncus lacus purus, vitae scelerisque urna venenatis at.\n\nUt ornare sapien sed condimentum varius. Nulla ut varius nibh. Nam consequat ligula sem, interdum hendrerit diam commodo sed. Vestibulum imperdiet mauris quis mauris vestibulum lobortis. Nunc bibendum, nunc ut porttitor tristique, turpis mauris cursus magna, ac molestie risus hendrerit odio. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Cras pulvinar odio vestibulum ligula mollis, vel venenatis orci semper. Duis consectetur molestie justo, ut pulvinar nisi semper vel. Cras quis fringilla augue, ut vehicula turpis. Praesent nec imperdiet dui. Morbi quis pharetra lectus. Interdum et malesuada fames ac ante ipsum primis in faucibus. Sed luctus mauris vitae augue rhoncus, vel interdum nisl pellentesque. In tempor, mi et vestibulum condimentum, nisl mauris cursus diam, aliquet vestibulum risus erat sit amet diam.\n\nInteger suscipit eros at diam tempor, ac condimentum turpis pharetra. Vestibulum pharetra risus in lacus semper, at bibendum ligula condimentum. Proin sed massa eget risus malesuada dignissim et in ipsum. Donec convallis tellus vel lectus lobortis, non lobortis libero faucibus. Sed nec posuere mi. Pellentesque sed sem sit amet erat lobortis commodo nec ut nisl. Pellentesque rhoncus risus id massa accumsan, hendrerit posuere mauris pulvinar.\n\nNullam risus felis, dictum et varius eget, rutrum non nunc. Vestibulum lorem nulla, porttitor ac sapien interdum, porta tristique turpis. Quisque ut dui vitae urna congue scelerisque. Nulla eu commodo felis. Nulla aliquam magna a arcu elementum, vitae tincidunt risus semper. Sed consectetur diam vel magna mattis, sed congue nisi fringilla. Ut in facilisis elit. Etiam aliquet orci urna. Nam interdum nunc fringilla iaculis cursus.\n\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Nam ligula sapien, fermentum vitae aliquam sed, lacinia in nibh. Pellentesque euismod diam eget justo interdum, sit amet elementum ipsum vulputate. Aliquam sed purus urna. Vivamus luctus nisi sem, a fringilla velit iaculis id. Donec nec vulputate justo. Fusce vulputate sodales tellus, in fringilla ante posuere non. Vivamus vulputate nisl non augue dignissim, consectetur rutrum risus facilisis. Sed consectetur aliquam dolor, sed pulvinar mi tristique vitae. Donec quis ligula quis velit consequat dictum id eu felis. Mauris ac congue ante, sed elementum eros.\n\nSed eu euismod dolor, a vestibulum lorem. Phasellus euismod volutpat risus sit amet pellentesque. Integer sit amet nisl malesuada, posuere orci sodales, euismod metus. Suspendisse potenti. Nullam mollis tristique accumsan. Vestibulum sit amet dictum nisl. Sed rhoncus lacus purus, vitae scelerisque urna venenatis at.\n\nUt ornare sapien sed condimentum varius. Nulla ut varius nibh. Nam consequat ligula sem, interdum hendrerit diam commodo sed. Vestibulum imperdiet mauris quis mauris vestibulum lobortis. Nunc bibendum, nunc ut porttitor tristique, turpis mauris cursus magna, ac molestie risus hendrerit odio. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Cras pulvinar odio vestibulum ligula mollis, vel venenatis orci semper. Duis consectetur molestie justo, ut pulvinar nisi semper vel. Cras quis fringilla augue, ut vehicula turpis. Praesent nec imperdiet dui. Morbi quis pharetra lectus. Interdum et malesuada fames ac ante ipsum primis in faucibus. Sed luctus mauris vitae augue rhoncus, vel interdum nisl pellentesque. In tempor, mi et vestibulum condimentum, nisl mauris cursus diam, aliquet vestibulum risus erat sit amet diam.\n\nInteger suscipit eros at diam tempor, ac condimentum turpis pharetra. Vestibulum pharetra risus in lacus semper, at bibendum ligula condimentum. Proin sed massa eget risus malesuada dignissim et in ipsum. Donec convallis tellus vel lectus lobortis, non lobortis libero faucibus. Sed nec posuere mi. Pellentesque sed sem sit amet erat lobortis commodo nec ut nisl. Pellentesque rhoncus risus id massa accumsan, hendrerit posuere mauris pulvinar.\n\nNullam risus felis, dictum et varius eget, rutrum non nunc. Vestibulum lorem nulla, porttitor ac sapien interdum, porta tristique turpis. Quisque ut dui vitae urna congue scelerisque. Nulla eu commodo felis. Nulla aliquam magna a arcu elementum, vitae tincidunt risus semper. Sed consectetur diam vel magna mattis, sed congue nisi fringilla. Ut in facilisis elit. Etiam aliquet orci urna. Nam interdum nunc fringilla iaculis cursus.\n\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Nam ligula sapien, fermentum vitae aliquam sed, lacinia in nibh. Pellentesque euismod diam eget justo interdum, sit amet elementum ipsum vulputate. Aliquam sed purus urna. Vivamus luctus nisi sem, a fringilla velit iaculis id. Donec nec vulputate justo. Fusce vulputate sodales tellus, in fringilla ante posuere non. Vivamus vulputate nisl non augue dignissim, consectetur rutrum risus facilisis. Sed consectetur aliquam dolor, sed pulvinar mi tristique vitae. Donec quis ligula quis velit consequat dictum id eu felis. Mauris ac congue ante, sed elementum eros.\n\nSed eu euismod dolor, a vestibulum lorem. Phasellus euismod volutpat risus sit amet pellentesque. Integer sit amet nisl malesuada, posuere orci sodales, euismod metus. Suspendisse potenti. Nullam mollis tristique accumsan. Vestibulum sit amet dictum nisl. Sed rhoncus lacus purus, vitae scelerisque urna venenatis at.\n\nUt ornare sapien sed condimentum varius. Nulla ut varius nibh. Nam consequat ligula sem, interdum hendrerit diam commodo sed. Vestibulum imperdiet mauris quis mauris vestibulum lobortis. Nunc bibendum, nunc ut porttitor tristique, turpis mauris cursus magna, ac molestie risus hendrerit odio. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Cras pulvinar odio vestibulum ligula mollis, vel venenatis orci semper. Duis consectetur molestie justo, ut pulvinar nisi semper vel. Cras quis fringilla augue, ut vehicula turpis. Praesent nec imperdiet dui. Morbi quis pharetra lectus. Interdum et malesuada fames ac ante ipsum primis in faucibus. Sed luctus mauris vitae augue rhoncus, vel interdum nisl pellentesque. In tempor, mi et vestibulum condimentum, nisl mauris cursus diam, aliquet vestibulum risus erat sit amet diam.\n\nInteger suscipit eros at diam tempor, ac condimentum turpis pharetra. Vestibulum pharetra risus in lacus semper, at bibendum ligula condimentum. Proin sed massa eget risus malesuada dignissim et in ipsum. Donec convallis tellus vel lectus lobortis, non lobortis libero faucibus. Sed nec posuere mi. Pellentesque sed sem sit amet erat lobortis commodo nec ut nisl. Pellentesque rhoncus risus id massa accumsan, hendrerit posuere mauris pulvinar.\n\nNullam risus felis, dictum et varius eget, rutrum non nunc. Vestibulum lorem nulla, porttitor ac sapien interdum, porta tristique turpis. Quisque ut dui vitae urna congue scelerisque. Nulla eu commodo felis. Nulla aliquam magna a arcu elementum, vitae tincidunt risus semper. Sed consectetur diam vel magna mattis, sed congue nisi fringilla. Ut in facilisis elit. Etiam aliquet orci urna. Nam interdum nunc fringilla iaculis cursus.\n\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Nam ligula sapien, fermentum vitae aliquam sed, lacinia in nibh. Pellentesque euismod diam eget justo interdum, sit amet elementum ipsum vulputate. Aliquam sed purus urna. Vivamus luctus nisi sem, a fringilla velit iaculis id. Donec nec vulputate justo. Fusce vulputate sodales tellus, in fringilla ante posuere non. Vivamus vulputate nisl non augue dignissim, consectetur rutrum risus facilisis. Sed consectetur aliquam dolor, sed pulvinar mi tristique vitae. Donec quis ligula quis velit consequat dictum id eu felis. Mauris ac congue ante, sed elementum eros.\n\nSed eu euismod dolor, a vestibulum lorem. Phasellus euismod volutpat risus sit amet pellentesque. Integer sit amet nisl malesuada, posuere orci sodales, euismod metus. Suspendisse potenti. Nullam mollis tristique accumsan. Vestibulum sit amet dictum nisl. Sed rhoncus lacus purus, vitae scelerisque urna venenatis at.\n\nUt ornare sapien sed condimentum varius. Nulla ut varius nibh. Nam consequat ligula sem, interdum hendrerit diam commodo sed. Vestibulum imperdiet mauris quis mauris vestibulum lobortis. Nunc bibendum, nunc ut porttitor tristique, turpis mauris cursus magna, ac molestie risus hendrerit odio. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Cras pulvinar odio vestibulum ligula mollis, vel venenatis orci semper. Duis consectetur molestie justo, ut pulvinar nisi semper vel. Cras quis fringilla augue, ut vehicula turpis. Praesent nec imperdiet dui. Morbi quis pharetra lectus. Interdum et malesuada fames ac ante ipsum primis in faucibus. Sed luctus mauris vitae augue rhoncus, vel interdum nisl pellentesque. In tempor, mi et vestibulum condimentum, nisl mauris cursus diam, aliquet vestibulum risus erat sit amet diam.\n\nInteger suscipit eros at diam tempor, ac condimentum turpis pharetra. Vestibulum pharetra risus in lacus semper, at bibendum ligula condimentum. Proin sed massa eget risus malesuada dignissim et in ipsum. Donec convallis tellus vel lectus lobortis, non lobortis libero faucibus. Sed nec posuere mi. Pellentesque sed sem sit amet erat lobortis commodo nec ut nisl. Pellentesque rhoncus risus id massa accumsan, hendrerit posuere mauris pulvinar.\n\nNullam risus felis, dictum et varius eget, rutrum non nunc. Vestibulum lorem nulla, porttitor ac sapien interdum, porta tristique turpis. Quisque ut dui vitae urna congue scelerisque. Nulla eu commodo felis. Nulla aliquam magna a arcu elementum, vitae tincidunt risus semper. Sed consectetur diam vel magna mattis, sed congue nisi fringilla. Ut in facilisis elit. Etiam aliquet orci urna. Nam interdum nunc fringilla iaculis cursus.
5 |
6 |
7 |
--------------------------------------------------------------------------------
/library/src/com/itheima/systembartint/SystemBarTintManager.java:
--------------------------------------------------------------------------------
1 | /*
2 | * Copyright (C) 2013 readyState Software Ltd
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 |
17 | package com.itheima.systembartint;
18 |
19 | import android.annotation.SuppressLint;
20 | import android.annotation.TargetApi;
21 | import android.app.Activity;
22 | import android.content.Context;
23 | import android.content.res.Configuration;
24 | import android.content.res.Resources;
25 | import android.content.res.TypedArray;
26 | import android.graphics.drawable.Drawable;
27 | import android.os.Build;
28 | import android.util.DisplayMetrics;
29 | import android.util.TypedValue;
30 | import android.view.Gravity;
31 | import android.view.View;
32 | import android.view.ViewConfiguration;
33 | import android.view.ViewGroup;
34 | import android.view.Window;
35 | import android.view.WindowManager;
36 | import android.widget.FrameLayout.LayoutParams;
37 |
38 | import java.lang.reflect.Method;
39 |
40 | /**
41 | * Class to manage status and navigation bar tint effects when using KitKat
42 | * translucent system UI modes.
43 | *
44 | */
45 | public class SystemBarTintManager {
46 |
47 | static {
48 | // Android allows a system property to override the presence of the navigation bar.
49 | // Used by the emulator.
50 | // See https://github.com/android/platform_frameworks_base/blob/master/policy/src/com/android/internal/policy/impl/PhoneWindowManager.java#L1076
51 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
52 | try {
53 | Class c = Class.forName("android.os.SystemProperties");
54 | Method m = c.getDeclaredMethod("get", String.class);
55 | m.setAccessible(true);
56 | sNavBarOverride = (String) m.invoke(null, "qemu.hw.mainkeys");
57 | } catch (Throwable e) {
58 | sNavBarOverride = null;
59 | }
60 | }
61 | }
62 |
63 |
64 | /**
65 | * The default system bar tint color value.
66 | */
67 | public static final int DEFAULT_TINT_COLOR = 0x99000000;
68 |
69 | private static String sNavBarOverride;
70 |
71 | private final SystemBarConfig mConfig;
72 | private boolean mStatusBarAvailable;
73 | private boolean mNavBarAvailable;
74 | private boolean mStatusBarTintEnabled;
75 | private boolean mNavBarTintEnabled;
76 | private View mStatusBarTintView;
77 | private View mNavBarTintView;
78 |
79 | /**
80 | * Constructor. Call this in the host activity onCreate method after its
81 | * content view has been set. You should always create new instances when
82 | * the host activity is recreated.
83 | *
84 | * @param activity The host activity.
85 | */
86 | @TargetApi(19)
87 | public SystemBarTintManager(Activity activity) {
88 |
89 | Window win = activity.getWindow();
90 | ViewGroup decorViewGroup = (ViewGroup) win.getDecorView();
91 |
92 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
93 | // check theme attrs
94 | int[] attrs = {android.R.attr.windowTranslucentStatus,
95 | android.R.attr.windowTranslucentNavigation};
96 | TypedArray a = activity.obtainStyledAttributes(attrs);
97 | try {
98 | mStatusBarAvailable = a.getBoolean(0, false);
99 | mNavBarAvailable = a.getBoolean(1, false);
100 | } finally {
101 | a.recycle();
102 | }
103 |
104 | // check window flags
105 | WindowManager.LayoutParams winParams = win.getAttributes();
106 | int bits = WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS;
107 | if ((winParams.flags & bits) != 0) {
108 | mStatusBarAvailable = true;
109 | }
110 | bits = WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION;
111 | if ((winParams.flags & bits) != 0) {
112 | mNavBarAvailable = true;
113 | }
114 | }
115 |
116 | mConfig = new SystemBarConfig(activity, mStatusBarAvailable, mNavBarAvailable);
117 | // device might not have virtual navigation keys
118 | if (!mConfig.hasNavigtionBar()) {
119 | mNavBarAvailable = false;
120 | }
121 |
122 | if (mStatusBarAvailable) {
123 | setupStatusBarView(activity, decorViewGroup);
124 | }
125 | if (mNavBarAvailable) {
126 | setupNavBarView(activity, decorViewGroup);
127 | }
128 |
129 | }
130 |
131 | /**
132 | * Enable tinting of the system status bar.
133 | *
134 | * If the platform is running Jelly Bean or earlier, or translucent system
135 | * UI modes have not been enabled in either the theme or via window flags,
136 | * then this method does nothing.
137 | *
138 | * @param enabled True to enable tinting, false to disable it (default).
139 | */
140 | public void setStatusBarTintEnabled(boolean enabled) {
141 | mStatusBarTintEnabled = enabled;
142 | if (mStatusBarAvailable) {
143 | mStatusBarTintView.setVisibility(enabled ? View.VISIBLE : View.GONE);
144 | }
145 | }
146 |
147 | /**
148 | * Enable tinting of the system navigation bar.
149 | *
150 | * If the platform does not have soft navigation keys, is running Jelly Bean
151 | * or earlier, or translucent system UI modes have not been enabled in either
152 | * the theme or via window flags, then this method does nothing.
153 | *
154 | * @param enabled True to enable tinting, false to disable it (default).
155 | */
156 | public void setNavigationBarTintEnabled(boolean enabled) {
157 | mNavBarTintEnabled = enabled;
158 | if (mNavBarAvailable) {
159 | mNavBarTintView.setVisibility(enabled ? View.VISIBLE : View.GONE);
160 | }
161 | }
162 |
163 | /**
164 | * Apply the specified color tint to all system UI bars.
165 | *
166 | * @param color The color of the background tint.
167 | */
168 | public void setTintColor(int color) {
169 | setStatusBarTintColor(color);
170 | setNavigationBarTintColor(color);
171 | }
172 |
173 | /**
174 | * Apply the specified drawable or color resource to all system UI bars.
175 | *
176 | * @param res The identifier of the resource.
177 | */
178 | public void setTintResource(int res) {
179 | setStatusBarTintResource(res);
180 | setNavigationBarTintResource(res);
181 | }
182 |
183 | /**
184 | * Apply the specified drawable to all system UI bars.
185 | *
186 | * @param drawable The drawable to use as the background, or null to remove it.
187 | */
188 | public void setTintDrawable(Drawable drawable) {
189 | setStatusBarTintDrawable(drawable);
190 | setNavigationBarTintDrawable(drawable);
191 | }
192 |
193 | /**
194 | * Apply the specified alpha to all system UI bars.
195 | *
196 | * @param alpha The alpha to use
197 | */
198 | public void setTintAlpha(float alpha) {
199 | setStatusBarAlpha(alpha);
200 | setNavigationBarAlpha(alpha);
201 | }
202 |
203 | /**
204 | * Apply the specified color tint to the system status bar.
205 | *
206 | * @param color The color of the background tint.
207 | */
208 | public void setStatusBarTintColor(int color) {
209 | if (mStatusBarAvailable) {
210 | mStatusBarTintView.setBackgroundColor(color);
211 | }
212 | }
213 |
214 | /**
215 | * Apply the specified drawable or color resource to the system status bar.
216 | *
217 | * @param res The identifier of the resource.
218 | */
219 | public void setStatusBarTintResource(int res) {
220 | if (mStatusBarAvailable) {
221 | mStatusBarTintView.setBackgroundResource(res);
222 | }
223 | }
224 |
225 | /**
226 | * Apply the specified drawable to the system status bar.
227 | *
228 | * @param drawable The drawable to use as the background, or null to remove it.
229 | */
230 | @SuppressWarnings("deprecation")
231 | public void setStatusBarTintDrawable(Drawable drawable) {
232 | if (mStatusBarAvailable) {
233 | mStatusBarTintView.setBackgroundDrawable(drawable);
234 | }
235 | }
236 |
237 | /**
238 | * Apply the specified alpha to the system status bar.
239 | *
240 | * @param alpha The alpha to use
241 | */
242 | @TargetApi(11)
243 | public void setStatusBarAlpha(float alpha) {
244 | if (mStatusBarAvailable && Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
245 | mStatusBarTintView.setAlpha(alpha);
246 | }
247 | }
248 |
249 | /**
250 | * Apply the specified color tint to the system navigation bar.
251 | *
252 | * @param color The color of the background tint.
253 | */
254 | public void setNavigationBarTintColor(int color) {
255 | if (mNavBarAvailable) {
256 | mNavBarTintView.setBackgroundColor(color);
257 | }
258 | }
259 |
260 | /**
261 | * Apply the specified drawable or color resource to the system navigation bar.
262 | *
263 | * @param res The identifier of the resource.
264 | */
265 | public void setNavigationBarTintResource(int res) {
266 | if (mNavBarAvailable) {
267 | mNavBarTintView.setBackgroundResource(res);
268 | }
269 | }
270 |
271 | /**
272 | * Apply the specified drawable to the system navigation bar.
273 | *
274 | * @param drawable The drawable to use as the background, or null to remove it.
275 | */
276 | @SuppressWarnings("deprecation")
277 | public void setNavigationBarTintDrawable(Drawable drawable) {
278 | if (mNavBarAvailable) {
279 | mNavBarTintView.setBackgroundDrawable(drawable);
280 | }
281 | }
282 |
283 | /**
284 | * Apply the specified alpha to the system navigation bar.
285 | *
286 | * @param alpha The alpha to use
287 | */
288 | @TargetApi(11)
289 | public void setNavigationBarAlpha(float alpha) {
290 | if (mNavBarAvailable && Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
291 | mNavBarTintView.setAlpha(alpha);
292 | }
293 | }
294 |
295 | /**
296 | * Get the system bar configuration.
297 | *
298 | * @return The system bar configuration for the current device configuration.
299 | */
300 | public SystemBarConfig getConfig() {
301 | return mConfig;
302 | }
303 |
304 | /**
305 | * Is tinting enabled for the system status bar?
306 | *
307 | * @return True if enabled, False otherwise.
308 | */
309 | public boolean isStatusBarTintEnabled() {
310 | return mStatusBarTintEnabled;
311 | }
312 |
313 | /**
314 | * Is tinting enabled for the system navigation bar?
315 | *
316 | * @return True if enabled, False otherwise.
317 | */
318 | public boolean isNavBarTintEnabled() {
319 | return mNavBarTintEnabled;
320 | }
321 |
322 | private void setupStatusBarView(Context context, ViewGroup decorViewGroup) {
323 | mStatusBarTintView = new View(context);
324 | LayoutParams params = new LayoutParams(LayoutParams.MATCH_PARENT, mConfig.getStatusBarHeight());
325 | params.gravity = Gravity.TOP;
326 | if (mNavBarAvailable && !mConfig.isNavigationAtBottom()) {
327 | params.rightMargin = mConfig.getNavigationBarWidth();
328 | }
329 | mStatusBarTintView.setLayoutParams(params);
330 | mStatusBarTintView.setBackgroundColor(DEFAULT_TINT_COLOR);
331 | mStatusBarTintView.setVisibility(View.GONE);
332 | decorViewGroup.addView(mStatusBarTintView);
333 | }
334 |
335 | private void setupNavBarView(Context context, ViewGroup decorViewGroup) {
336 | mNavBarTintView = new View(context);
337 | LayoutParams params;
338 | if (mConfig.isNavigationAtBottom()) {
339 | params = new LayoutParams(LayoutParams.MATCH_PARENT, mConfig.getNavigationBarHeight());
340 | params.gravity = Gravity.BOTTOM;
341 | } else {
342 | params = new LayoutParams(mConfig.getNavigationBarWidth(), LayoutParams.MATCH_PARENT);
343 | params.gravity = Gravity.RIGHT;
344 | }
345 | mNavBarTintView.setLayoutParams(params);
346 | mNavBarTintView.setBackgroundColor(DEFAULT_TINT_COLOR);
347 | mNavBarTintView.setVisibility(View.GONE);
348 | decorViewGroup.addView(mNavBarTintView);
349 | }
350 |
351 | /**
352 | * Class which describes system bar sizing and other characteristics for the current
353 | * device configuration.
354 | *
355 | */
356 | public static class SystemBarConfig {
357 |
358 | private static final String STATUS_BAR_HEIGHT_RES_NAME = "status_bar_height";
359 | private static final String NAV_BAR_HEIGHT_RES_NAME = "navigation_bar_height";
360 | private static final String NAV_BAR_HEIGHT_LANDSCAPE_RES_NAME = "navigation_bar_height_landscape";
361 | private static final String NAV_BAR_WIDTH_RES_NAME = "navigation_bar_width";
362 | private static final String SHOW_NAV_BAR_RES_NAME = "config_showNavigationBar";
363 |
364 | private final boolean mTranslucentStatusBar;
365 | private final boolean mTranslucentNavBar;
366 | private final int mStatusBarHeight;
367 | private final int mActionBarHeight;
368 | private final boolean mHasNavigationBar;
369 | private final int mNavigationBarHeight;
370 | private final int mNavigationBarWidth;
371 | private final boolean mInPortrait;
372 | private final float mSmallestWidthDp;
373 |
374 | private SystemBarConfig(Activity activity, boolean translucentStatusBar, boolean traslucentNavBar) {
375 | Resources res = activity.getResources();
376 | mInPortrait = (res.getConfiguration().orientation == Configuration.ORIENTATION_PORTRAIT);
377 | mSmallestWidthDp = getSmallestWidthDp(activity);
378 | mStatusBarHeight = getInternalDimensionSize(res, STATUS_BAR_HEIGHT_RES_NAME);
379 | mActionBarHeight = getActionBarHeight(activity);
380 | mNavigationBarHeight = getNavigationBarHeight(activity);
381 | mNavigationBarWidth = getNavigationBarWidth(activity);
382 | mHasNavigationBar = (mNavigationBarHeight > 0);
383 | mTranslucentStatusBar = translucentStatusBar;
384 | mTranslucentNavBar = traslucentNavBar;
385 | }
386 |
387 | @TargetApi(14)
388 | private int getActionBarHeight(Context context) {
389 | int result = 0;
390 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
391 | TypedValue tv = new TypedValue();
392 | context.getTheme().resolveAttribute(android.R.attr.actionBarSize, tv, true);
393 | result = TypedValue.complexToDimensionPixelSize(tv.data, context.getResources().getDisplayMetrics());
394 | }
395 | return result;
396 | }
397 |
398 | @TargetApi(14)
399 | private int getNavigationBarHeight(Context context) {
400 | Resources res = context.getResources();
401 | int result = 0;
402 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
403 | if (hasNavBar(context)) {
404 | String key;
405 | if (mInPortrait) {
406 | key = NAV_BAR_HEIGHT_RES_NAME;
407 | } else {
408 | key = NAV_BAR_HEIGHT_LANDSCAPE_RES_NAME;
409 | }
410 | return getInternalDimensionSize(res, key);
411 | }
412 | }
413 | return result;
414 | }
415 |
416 | @TargetApi(14)
417 | private int getNavigationBarWidth(Context context) {
418 | Resources res = context.getResources();
419 | int result = 0;
420 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
421 | if (hasNavBar(context)) {
422 | return getInternalDimensionSize(res, NAV_BAR_WIDTH_RES_NAME);
423 | }
424 | }
425 | return result;
426 | }
427 |
428 | @TargetApi(14)
429 | private boolean hasNavBar(Context context) {
430 | Resources res = context.getResources();
431 | int resourceId = res.getIdentifier(SHOW_NAV_BAR_RES_NAME, "bool", "android");
432 | if (resourceId != 0) {
433 | boolean hasNav = res.getBoolean(resourceId);
434 | // check override flag (see static block)
435 | if ("1".equals(sNavBarOverride)) {
436 | hasNav = false;
437 | } else if ("0".equals(sNavBarOverride)) {
438 | hasNav = true;
439 | }
440 | return hasNav;
441 | } else { // fallback
442 | return !ViewConfiguration.get(context).hasPermanentMenuKey();
443 | }
444 | }
445 |
446 | private int getInternalDimensionSize(Resources res, String key) {
447 | int result = 0;
448 | int resourceId = res.getIdentifier(key, "dimen", "android");
449 | if (resourceId > 0) {
450 | result = res.getDimensionPixelSize(resourceId);
451 | }
452 | return result;
453 | }
454 |
455 | @SuppressLint("NewApi")
456 | private float getSmallestWidthDp(Activity activity) {
457 | DisplayMetrics metrics = new DisplayMetrics();
458 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
459 | activity.getWindowManager().getDefaultDisplay().getRealMetrics(metrics);
460 | } else {
461 | // TODO this is not correct, but we don't really care pre-kitkat
462 | activity.getWindowManager().getDefaultDisplay().getMetrics(metrics);
463 | }
464 | float widthDp = metrics.widthPixels / metrics.density;
465 | float heightDp = metrics.heightPixels / metrics.density;
466 | return Math.min(widthDp, heightDp);
467 | }
468 |
469 | /**
470 | * Should a navigation bar appear at the bottom of the screen in the current
471 | * device configuration? A navigation bar may appear on the right side of
472 | * the screen in certain configurations.
473 | *
474 | * @return True if navigation should appear at the bottom of the screen, False otherwise.
475 | */
476 | public boolean isNavigationAtBottom() {
477 | return (mSmallestWidthDp >= 600 || mInPortrait);
478 | }
479 |
480 | /**
481 | * Get the height of the system status bar.
482 | *
483 | * @return The height of the status bar (in pixels).
484 | */
485 | public int getStatusBarHeight() {
486 | return mStatusBarHeight;
487 | }
488 |
489 | /**
490 | * Get the height of the action bar.
491 | *
492 | * @return The height of the action bar (in pixels).
493 | */
494 | public int getActionBarHeight() {
495 | return mActionBarHeight;
496 | }
497 |
498 | /**
499 | * Does this device have a system navigation bar?
500 | *
501 | * @return True if this device uses soft key navigation, False otherwise.
502 | */
503 | public boolean hasNavigtionBar() {
504 | return mHasNavigationBar;
505 | }
506 |
507 | /**
508 | * Get the height of the system navigation bar.
509 | *
510 | * @return The height of the navigation bar (in pixels). If the device does not have
511 | * soft navigation keys, this will always return 0.
512 | */
513 | public int getNavigationBarHeight() {
514 | return mNavigationBarHeight;
515 | }
516 |
517 | /**
518 | * Get the width of the system navigation bar when it is placed vertically on the screen.
519 | *
520 | * @return The width of the navigation bar (in pixels). If the device does not have
521 | * soft navigation keys, this will always return 0.
522 | */
523 | public int getNavigationBarWidth() {
524 | return mNavigationBarWidth;
525 | }
526 |
527 | /**
528 | * Get the layout inset for any system UI that appears at the top of the screen.
529 | *
530 | * @param withActionBar True to include the height of the action bar, False otherwise.
531 | * @return The layout inset (in pixels).
532 | */
533 | public int getPixelInsetTop(boolean withActionBar) {
534 | return (mTranslucentStatusBar ? mStatusBarHeight : 0) + (withActionBar ? mActionBarHeight : 0);
535 | }
536 |
537 | /**
538 | * Get the layout inset for any system UI that appears at the bottom of the screen.
539 | *
540 | * @return The layout inset (in pixels).
541 | */
542 | public int getPixelInsetBottom() {
543 | if (mTranslucentNavBar && isNavigationAtBottom()) {
544 | return mNavigationBarHeight;
545 | } else {
546 | return 0;
547 | }
548 | }
549 |
550 | /**
551 | * Get the layout inset for any system UI that appears at the right of the screen.
552 | *
553 | * @return The layout inset (in pixels).
554 | */
555 | public int getPixelInsetRight() {
556 | if (mTranslucentNavBar && !isNavigationAtBottom()) {
557 | return mNavigationBarWidth;
558 | } else {
559 | return 0;
560 | }
561 | }
562 |
563 | }
564 |
565 | }
566 |
--------------------------------------------------------------------------------