├── .gitignore ├── .idea ├── .name ├── compiler.xml ├── copyright │ └── profiles_settings.xml ├── encodings.xml ├── gradle.xml ├── misc.xml ├── modules.xml └── runConfigurations.xml ├── README.md ├── app ├── .gitignore ├── build.gradle ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── com │ │ └── example │ │ └── burge │ │ └── meizhunotification │ │ └── ApplicationTest.java │ ├── main │ ├── AndroidManifest.xml │ ├── java │ │ └── com │ │ │ └── example │ │ │ └── burge │ │ │ └── meizhunotification │ │ │ ├── activity │ │ │ └── MainActivity.java │ │ │ └── notification │ │ │ └── MeiZhuNotification.java │ └── res │ │ ├── anim │ │ ├── anim_in.xml │ │ └── anim_out.xml │ │ ├── drawable │ │ └── notify.png │ │ ├── layout │ │ ├── activity_main.xml │ │ └── layout_notification.xml │ │ ├── mipmap-hdpi │ │ └── ic_launcher.png │ │ ├── mipmap-mdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xhdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xxhdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xxxhdpi │ │ └── ic_launcher.png │ │ ├── values-w820dp │ │ └── dimens.xml │ │ └── values │ │ ├── colors.xml │ │ ├── dimens.xml │ │ ├── strings.xml │ │ └── styles.xml │ └── test │ └── java │ └── com │ └── example │ └── burge │ └── meizhunotification │ └── ExampleUnitTest.java ├── build.gradle ├── capture ├── GIF.gif ├── S60614-102422.jpg └── device-2016-06-15-092323.png ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat └── settings.gradle /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /local.properties 4 | /.idea/workspace.xml 5 | /.idea/libraries 6 | .DS_Store 7 | /build 8 | /captures 9 | -------------------------------------------------------------------------------- /.idea/.name: -------------------------------------------------------------------------------- 1 | MeiZhuNotification -------------------------------------------------------------------------------- /.idea/compiler.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /.idea/copyright/profiles_settings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | -------------------------------------------------------------------------------- /.idea/encodings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /.idea/gradle.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 22 | 23 | -------------------------------------------------------------------------------- /.idea/misc.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 19 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 46 | -------------------------------------------------------------------------------- /.idea/modules.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /.idea/runConfigurations.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 11 | 12 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | 2 | 3 | 仿魅族手机消息通知效果 4 | 5 | ![](capture/GIF.gif) -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | 3 | android { 4 | compileSdkVersion 23 5 | buildToolsVersion "23.0.3" 6 | 7 | defaultConfig { 8 | applicationId "com.example.burge.meizhunotification" 9 | minSdkVersion 14 10 | targetSdkVersion 23 11 | versionCode 1 12 | versionName "1.0" 13 | } 14 | buildTypes { 15 | release { 16 | minifyEnabled false 17 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 18 | } 19 | } 20 | } 21 | 22 | dependencies { 23 | compile fileTree(dir: 'libs', include: ['*.jar']) 24 | testCompile 'junit:junit:4.12' 25 | compile 'com.android.support:appcompat-v7:23.4.0' 26 | } 27 | -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in D:\ProgramFiles\Development\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 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/example/burge/meizhunotification/ApplicationTest.java: -------------------------------------------------------------------------------- 1 | package com.example.burge.meizhunotification; 2 | 3 | import android.app.Application; 4 | import android.test.ApplicationTestCase; 5 | 6 | /** 7 | * Testing Fundamentals 8 | */ 9 | public class ApplicationTest extends ApplicationTestCase { 10 | public ApplicationTest() { 11 | super(Application.class); 12 | } 13 | } -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 6 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /app/src/main/java/com/example/burge/meizhunotification/activity/MainActivity.java: -------------------------------------------------------------------------------- 1 | package com.example.burge.meizhunotification.activity; 2 | 3 | import android.os.Bundle; 4 | import android.support.v7.app.AppCompatActivity; 5 | import android.view.View; 6 | 7 | import com.example.burge.meizhunotification.notification.MeiZhuNotification; 8 | import com.example.burge.meizhunotification.R; 9 | 10 | public class MainActivity extends AppCompatActivity { 11 | 12 | @Override 13 | protected void onCreate(Bundle savedInstanceState) { 14 | super.onCreate(savedInstanceState); 15 | setContentView(R.layout.activity_main); 16 | final MeiZhuNotification notification = 17 | new MeiZhuNotification.Builder().setContext(MainActivity.this) 18 | .setTime(System.currentTimeMillis()) 19 | .setImgRes(R.drawable.notify) 20 | .setTitle("你收到了一条消息") 21 | .setContent("人丑就要多读书").build(); 22 | 23 | 24 | findViewById(R.id.btn_show_window).setOnClickListener(new View.OnClickListener() { 25 | @Override 26 | public void onClick(View v) { 27 | 28 | notification.show(); 29 | 30 | } 31 | }); 32 | 33 | findViewById(R.id.btn_hide_window).setOnClickListener(new View.OnClickListener() { 34 | @Override 35 | public void onClick(View v) { 36 | notification.dismiss(); 37 | } 38 | }); 39 | 40 | } 41 | } 42 | -------------------------------------------------------------------------------- /app/src/main/java/com/example/burge/meizhunotification/notification/MeiZhuNotification.java: -------------------------------------------------------------------------------- 1 | package com.example.burge.meizhunotification.notification; 2 | 3 | import android.animation.Animator; 4 | import android.animation.AnimatorListenerAdapter; 5 | import android.animation.IntEvaluator; 6 | import android.animation.ValueAnimator; 7 | import android.content.Context; 8 | import android.os.Handler; 9 | import android.os.Message; 10 | import android.text.TextUtils; 11 | import android.view.Gravity; 12 | import android.view.LayoutInflater; 13 | import android.view.MotionEvent; 14 | import android.view.View; 15 | import android.view.ViewGroup; 16 | import android.view.WindowManager; 17 | import android.widget.TextView; 18 | import android.widget.ImageView; 19 | 20 | import java.util.Date; 21 | import java.util.Locale; 22 | 23 | import com.example.burge.meizhunotification.R; 24 | 25 | import java.text.SimpleDateFormat; 26 | 27 | /** 28 | * Created by _SOLID 29 | * Date:2016/6/13 30 | * Time:11:25 31 | */ 32 | public class MeiZhuNotification implements View.OnTouchListener { 33 | 34 | private static final int DIRECTION_LEFT = -1; 35 | private static final int DIRECTION_NONE = 0; 36 | private static final int DIRECTION_RIGHT = 1; 37 | 38 | private static final int DISMISS_INTERVAL = 3000; 39 | 40 | private WindowManager mWindowManager; 41 | private WindowManager.LayoutParams mWindowParams; 42 | private View mContentView; 43 | private Context mContext; 44 | private int mScreenWidth = 0; 45 | private int mStatusBarHeight = 0; 46 | 47 | private boolean isShowing = false; 48 | private ValueAnimator restoreAnimator = null; 49 | private ValueAnimator dismissAnimator = null; 50 | private ImageView mIvIcon; 51 | private TextView mTvTitle; 52 | private TextView mTvContent; 53 | private TextView mTvTime; 54 | 55 | 56 | public MeiZhuNotification(Builder builder) { 57 | mContext = builder.getContext(); 58 | 59 | mStatusBarHeight = getStatusBarHeight(); 60 | mScreenWidth = mContext.getResources().getDisplayMetrics().widthPixels; 61 | 62 | mWindowManager = (WindowManager) 63 | mContext.getSystemService(Context.WINDOW_SERVICE); 64 | mWindowParams = new WindowManager.LayoutParams(); 65 | mWindowParams.type = WindowManager.LayoutParams.TYPE_TOAST;// 系统提示window 66 | mWindowParams.gravity = Gravity.LEFT | Gravity.TOP; 67 | mWindowParams.width = WindowManager.LayoutParams.MATCH_PARENT; 68 | mWindowParams.height = WindowManager.LayoutParams.WRAP_CONTENT; 69 | mWindowParams.flags = 70 | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | 71 | WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS; 72 | //设置进入和退出动画 73 | mWindowParams.windowAnimations = R.style.NotificationAnim; 74 | mWindowParams.x = 0; 75 | mWindowParams.y = -mStatusBarHeight; 76 | 77 | setContentView(mContext, builder); 78 | } 79 | 80 | 81 | private static final int HIDE_WINDOW = 0; 82 | 83 | private Handler mHandler = new Handler(new Handler.Callback() { 84 | @Override 85 | public boolean handleMessage(Message msg) { 86 | switch (msg.what) { 87 | case HIDE_WINDOW: 88 | dismiss(); 89 | break; 90 | } 91 | return false; 92 | } 93 | }); 94 | 95 | /*** 96 | * 设置内容视图 97 | * 98 | * @param context 99 | */ 100 | private void setContentView(Context context, Builder builder) { 101 | mContentView = LayoutInflater.from(context).inflate(R.layout.layout_notification, null); 102 | View v_state_bar = mContentView.findViewById(R.id.v_state_bar); 103 | ViewGroup.LayoutParams layoutParameter = v_state_bar.getLayoutParams(); 104 | layoutParameter.height = mStatusBarHeight; 105 | v_state_bar.setLayoutParams(layoutParameter); 106 | 107 | mIvIcon = (ImageView) mContentView.findViewById(R.id.iv_icon); 108 | mTvTitle = (TextView) mContentView.findViewById(R.id.tv_title); 109 | mTvContent = (TextView) mContentView.findViewById(R.id.tv_content); 110 | mTvTime = (TextView) mContentView.findViewById(R.id.tv_time); 111 | 112 | setIcon(builder.imgRes); 113 | setTitle(builder.title); 114 | setContent(builder.content); 115 | setTime(builder.time); 116 | 117 | mContentView.setOnTouchListener(this); 118 | } 119 | 120 | 121 | private int downX = 0; 122 | private int direction = DIRECTION_NONE; 123 | 124 | @Override 125 | public boolean onTouch(View v, MotionEvent event) { 126 | if (isAnimatorRunning()) { 127 | return false; 128 | } 129 | switch (event.getAction()) { 130 | case MotionEvent.ACTION_DOWN: 131 | downX = (int) event.getRawX(); 132 | break; 133 | case MotionEvent.ACTION_MOVE: 134 | //处于滑动状态就取消自动消失 135 | mHandler.removeMessages(HIDE_WINDOW); 136 | int moveX = (int) event.getRawX() - downX; 137 | //判断滑动方向 138 | if (moveX > 0) { 139 | direction = DIRECTION_RIGHT; 140 | } else { 141 | direction = DIRECTION_LEFT; 142 | } 143 | 144 | updateWindowLocation(moveX, mWindowParams.y); 145 | 146 | break; 147 | case MotionEvent.ACTION_UP: 148 | if (Math.abs(mWindowParams.x) > mScreenWidth / 2) { 149 | startDismissAnimator(direction); 150 | } else { 151 | startRestoreAnimator(); 152 | } 153 | break; 154 | } 155 | return true; 156 | } 157 | 158 | private void startRestoreAnimator() { 159 | restoreAnimator = new ValueAnimator().ofInt(mWindowParams.x, 0); 160 | restoreAnimator.setDuration(300); 161 | restoreAnimator.setEvaluator(new IntEvaluator()); 162 | 163 | restoreAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { 164 | @Override 165 | public void onAnimationUpdate(ValueAnimator animation) { 166 | System.out.println("onAnimationUpdate:" + animation.getAnimatedValue()); 167 | updateWindowLocation((Integer) animation.getAnimatedValue(), -mStatusBarHeight); 168 | } 169 | }); 170 | restoreAnimator.addListener(new AnimatorListenerAdapter() { 171 | @Override 172 | public void onAnimationEnd(Animator animation) { 173 | restoreAnimator = null; 174 | autoDismiss(); 175 | } 176 | }); 177 | restoreAnimator.start(); 178 | } 179 | 180 | private void startDismissAnimator(int direction) { 181 | if (direction == DIRECTION_LEFT) 182 | dismissAnimator = new ValueAnimator().ofInt(mWindowParams.x, -mScreenWidth); 183 | else { 184 | dismissAnimator = new ValueAnimator().ofInt(mWindowParams.x, mScreenWidth); 185 | } 186 | dismissAnimator.setDuration(300); 187 | dismissAnimator.setEvaluator(new IntEvaluator()); 188 | 189 | dismissAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { 190 | @Override 191 | public void onAnimationUpdate(ValueAnimator animation) { 192 | updateWindowLocation((Integer) animation.getAnimatedValue(), -mStatusBarHeight); 193 | } 194 | }); 195 | dismissAnimator.addListener(new AnimatorListenerAdapter() { 196 | @Override 197 | public void onAnimationEnd(Animator animation) { 198 | restoreAnimator = null; 199 | dismiss(); 200 | } 201 | }); 202 | dismissAnimator.start(); 203 | } 204 | 205 | private boolean isAnimatorRunning() { 206 | return (restoreAnimator != null && restoreAnimator.isRunning()) || (dismissAnimator != null && dismissAnimator.isRunning()); 207 | } 208 | 209 | public void updateWindowLocation(int x, int y) { 210 | if (isShowing) { 211 | mWindowParams.x = x; 212 | mWindowParams.y = y; 213 | mWindowManager.updateViewLayout(mContentView, mWindowParams); 214 | } 215 | 216 | } 217 | 218 | public void show() { 219 | if (!isShowing) { 220 | isShowing = true; 221 | mWindowManager.addView(mContentView, mWindowParams); 222 | autoDismiss(); 223 | } 224 | } 225 | 226 | public void dismiss() { 227 | if (isShowing) { 228 | resetState(); 229 | mWindowManager.removeView(mContentView); 230 | } 231 | } 232 | 233 | /** 234 | * 重置状态 235 | */ 236 | private void resetState() { 237 | isShowing = false; 238 | mWindowParams.x = 0; 239 | mWindowParams.y = -mStatusBarHeight; 240 | } 241 | 242 | /** 243 | * 自动隐藏通知 244 | */ 245 | private void autoDismiss() { 246 | mHandler.removeMessages(HIDE_WINDOW); 247 | mHandler.sendEmptyMessageDelayed(HIDE_WINDOW, DISMISS_INTERVAL); 248 | } 249 | 250 | /** 251 | * 获取状态栏的高度 252 | */ 253 | public int getStatusBarHeight() { 254 | int height = 0; 255 | int resId = mContext.getResources().getIdentifier("status_bar_height", "dimen", "android"); 256 | if (resId > 0) { 257 | height = mContext.getResources().getDimensionPixelSize(resId); 258 | } 259 | return height; 260 | } 261 | 262 | 263 | public void setIcon(int imgRes) { 264 | if (-1 != imgRes) { 265 | mIvIcon.setVisibility(View.VISIBLE); 266 | mIvIcon.setImageResource(imgRes); 267 | } 268 | } 269 | 270 | public void setTitle(String title) { 271 | if (!TextUtils.isEmpty(title)) { 272 | mTvTitle.setVisibility(View.VISIBLE); 273 | mTvTitle.setText(title); 274 | } 275 | } 276 | 277 | public void setContent(String content) { 278 | mTvContent.setText(content); 279 | } 280 | 281 | public void setTime(long time) { 282 | SimpleDateFormat formatDateTime = new SimpleDateFormat("HH:mm", Locale.getDefault()); 283 | mTvTime.setText(formatDateTime.format(new Date(time))); 284 | } 285 | 286 | public static class Builder { 287 | 288 | private Context context; 289 | private int imgRes = -1; 290 | private String title; 291 | private String content = "none"; 292 | private long time = System.currentTimeMillis(); 293 | 294 | public Context getContext() { 295 | return context; 296 | } 297 | 298 | public Builder setContext(Context context) { 299 | this.context = context; 300 | return this; 301 | } 302 | 303 | public Builder setImgRes(int imgRes) { 304 | this.imgRes = imgRes; 305 | return this; 306 | } 307 | 308 | 309 | public Builder setTitle(String title) { 310 | this.title = title; 311 | return this; 312 | } 313 | 314 | 315 | public Builder setContent(String content) { 316 | this.content = content; 317 | return this; 318 | } 319 | 320 | 321 | public Builder setTime(long time) { 322 | this.time = time; 323 | return this; 324 | } 325 | 326 | 327 | public MeiZhuNotification build() { 328 | 329 | if (null == context) 330 | throw new IllegalArgumentException("the context is required."); 331 | 332 | return new MeiZhuNotification(this); 333 | } 334 | 335 | 336 | } 337 | 338 | } 339 | -------------------------------------------------------------------------------- /app/src/main/res/anim/anim_in.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 9 | 13 | -------------------------------------------------------------------------------- /app/src/main/res/anim/anim_out.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 9 | 13 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/notify.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/burgessjp/MeiZhuNotification/4f7dedbbffb55c4f82399c0ddd16d4353580c3e4/app/src/main/res/drawable/notify.png -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 11 | 12 |