package com.loaderman.toastdemo;

import android.content.Context;
import android.graphics.PixelFormat;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager;
import android.widget.TextView; /**
*
* 自定义归属地吐司
* <p/>
* 让窗口布局响应触摸事件:
* 1. 去掉FLAG_NOT_TOUCHABLE的标记
* 2. 调整显示级别为TYPE_PHONE, 可以支持触摸
* 3. 加权限: <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
*/
public class AddressToast { private WindowManager mWM;
private View mView;
private final WindowManager.LayoutParams mParams;
private final TextView tvAddress;
private int startX;
private int startY; private Context mContext;
private int[] mIcons = new int[]{R.drawable.shape_address_normal, R.drawable
.shape_address_orange, R.drawable.shape_address_blue, R.drawable.shape_address_gray,
R.drawable.shape_address_green};
public AddressToast(Context ctx) {
mContext = ctx; //窗口管理器
//窗口: android系统中最顶级的布局, 任何的界面和组件都显示在窗口中, Activity, 状态栏, dialog
//当需要在第三方app中显示布局时, 由于没有activity可以依托,此时可以直接使用WindowManager在窗口上添加布局
mWM = (WindowManager) ctx.getSystemService(Context.WINDOW_SERVICE); //初始化布局参数, 定义当前布局的宽高,位置,显示方式等信息
mParams = new WindowManager.LayoutParams();
mParams.height = WindowManager.LayoutParams.WRAP_CONTENT;//高度包裹内容
mParams.width = WindowManager.LayoutParams.WRAP_CONTENT;//宽度包裹内容
mParams.format = PixelFormat.TRANSLUCENT;//显示格式,默认就行
//params.windowAnimations = com.android.internal.R.style.Animation_Toast;//窗口动画
mParams.type = WindowManager.LayoutParams.TYPE_PHONE;//类型, 吐司类型(不支持触摸)
//params.setTitle("Toast");//标题
//标记, 保持屏幕常亮, 没有焦点, 不可触摸
mParams.flags = WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON
| WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
// | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE; // mParams.x;
// mParams.y; //初始化布局
// mView = new TextView(ctx);
// mView.setText(text);
// mView.setTextColor(Color.RED);
// mView.setTextSize(25);
mView = View.inflate(ctx, R.layout.toast_address, null);
tvAddress = (TextView) mView.findViewById(R.id.tv_address);
//设置窗口布局的触摸事件
mView.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
System.out.println("按下...");
//1.记录起点坐标
startX = (int) event.getRawX();
startY = (int) event.getRawY();
break;
case MotionEvent.ACTION_MOVE:
System.out.println("移动...");
//2. 记录移动后的坐标
int moveX = (int) event.getRawX();
int moveY = (int) event.getRawY();
//3. 计算偏移量
int dx = moveX - startX;
int dy = moveY - startY;
//4. 根据偏移量更新布局位置
mParams.x += dx;
mParams.y += dy;
mWM.updateViewLayout(mView, mParams);//更新控件位置 //5. 重新初始化起点坐标
startX = moveX;
startY = moveY;
break;
case MotionEvent.ACTION_UP:
System.out.println("抬起...");
break;
default:
break;
}
return true;//消费掉此事件
}
});
} //显示布局
public void show(String text) {
//mView.setText(text);
tvAddress.setText(text);
//获取当前选中的样式
tvAddress.setBackgroundResource(mIcons[0]);//重新设置背景
//给窗口添加布局对象
mWM.addView(mView, mParams);
} //隐藏布局
public void hide() {
if (mWM != null && mView != null) {
//移除窗口布局
//布局没有添加给窗口时, 从窗口移除布局就会出异常
try {
mWM.removeView(mView);
} catch (Exception e) {
e.printStackTrace();
}
}
} }
package com.loaderman.toastdemo;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View; public class MainActivity extends AppCompatActivity {
private AddressToast mToast;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mToast = new AddressToast(this);
}
public void display(View view){
mToast.show("自定义的哦");
}
public void hide(View view){
mToast.hide();
}
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.loaderman.toastdemo.MainActivity"> <Button
android:id="@+id/btn_show"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="display"
android:text="显示"/>
<Button
android:id="@+id/btn_hide"
android:onClick="hide"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="隐藏"/>
</LinearLayout>

toast_address.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical"> <TextView
android:id="@+id/tv_address"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="@drawable/shape_address_normal"
android:drawableLeft="@drawable/location"
android:text="未知号码"
android:padding="8dp"
android:textColor="@color/white"
android:textSize="18sp"/>
</LinearLayout>

在res/drawable下

shape_address_blue.xml shape_address_gray.xml shape_address_green.xml shape_normal_blue.xml shape_address_orange.xml只需要把下面代码需要以下color即可

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle">
<!--颜色-->
<solid android:color="@color/global_blue"/>
<!--给矩形加圆角-->
<corners android:radius="6dp"/>
</shape>

在res/values/colors.xml添加需求颜色值即可如

<?xml version="1.0" encoding="utf-8"?>
<resources>
<color name="colorPrimary">#3F51B5</color>
<color name="colorPrimaryDark">#303F9F</color>
<color name="colorAccent">#FF4081</color>
<color name="red">#F00</color>
<color name="black">#000</color>
<color name="gray">#a000</color>
<color name="global_blue">#459FD7</color>
<color name="white">#fff</color>
<color name="light_gray">#cccccc</color>
<color name="light_green">#00ffa1</color>
</resources>

添加权限:

 <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>

效果图:

自定义可拖动的Toast的更多相关文章

  1. 自定义带动画的Toast

    一.style样式: 1.  // 移动和透明渐变结合的动画 <style name="anim_view">        <item name="@ ...

  2. Android特效专辑(三)——自定义不一样的Toast

    Android特效专辑(三)--自定义不一样的Toast 大家都知道,Android的控件有时候很难满足我们的需求,所以我们需要自定义View.自定义的方式很多,有继承原生控件也有直接自定义View的 ...

  3. ToastCustom【自定义显示风格的Toast】

    版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 基于系统Toast的自定义显示风格的Toast. 效果图 代码分析 ToastCustom类基于系统Toast,不是继承Toast, ...

  4. Android自定义Seekbar拖动条式样

    SeekBar拖动条可以由用户控制,进行拖动操作.比如,应用程序中用户需要对音量进行控制,就可以使用拖动条来实现. 1.SeekBar控件的使用 1.1SeekBar常用属性 SeekBar的常用属性 ...

  5. 自定义垂直拖动的seekbar进度条

    系统自定义的seekbar为横向拖动的样式,需要纵向的时则需要自己定义,网上很多说了重写系统SeekBar中onDraw()的方法,但是我使用的时候不知道为什么拖动条和点偏离了,不在一条直线上,好气. ...

  6. Android自定义可拖动的悬浮按钮

    在页面上随意拖动的按钮 public class MoveScaleRotateView extends RelativeLayout { private Context mContext; //默认 ...

  7. vue2.0 自定义 提示框(Toast)组件

    1.自定义 提示框 组件 src / components / Toast / index.js /** * 自定义 提示框( Toast )组件 */ var Toast = {}; var sho ...

  8. vue 自定义 提示框(Toast)组件

    1.自定义 提示框 组件 src / components / Toast / index.js /** * 自定义 提示框( Toast )组件 */ var Toast = {}; var sho ...

  9. vue自定义指令拖动div

    钩子函数一个指令定义对象可以提供如下几个钩子函数:bind:只掉用一次,指令第一次绑定到元素是调用,在这里可以进行一次性的初始化设置inserted:被绑定元素插入父节点时调用(仅保证父节点存在,但不 ...

随机推荐

  1. 访问kubernetes api

    kubernetes api介绍 作用: 将各种资源对象的数据都通过该api接口被提交到后端的持久化存储etcd中; 一个api的顶层元素由kind丶apiVersion丶metadata丶spec和 ...

  2. 数据防泄密(DLP)

    数据防泄密(DLP)近几年已经成为国内非常热门的关键词. 目前市场上DLP产品主要可以分为三大类,各类产品都来源于不同的技术体系,防护效果也各有优缺点. 第一类,以监控审计为主,对进出的数据进行过滤, ...

  3. python 之多线程join()

    join()其实就是阻塞线程,控制线程的执行,从而控制住代码的执行顺序. 参照这篇文章:python3对多线程join的理解 通常都是,线程join()后,顺序执行join()后面的代码,如下面的例子 ...

  4. 逻辑卷管理(LVM)

    LVM:Logical Volume Management 逻辑卷管理 LVM是建立在硬盘和分区之上的一个逻辑层,来提高磁盘分区管理的灵活性. 传统磁盘管理:我们上层是直接访问文件系统,从而对底层的物 ...

  5. Web UI开发推荐!Kendo UI for jQuery自定义小部件——使用MVVM

    Kendo UI for jQuery最新试用版下载 Kendo UI目前最新提供Kendo UI for jQuery.Kendo UI for Angular.Kendo UI Support f ...

  6. 抓取腾讯招聘python岗位

    # -*- coding: utf-8 -*- """ @author: Dell Created on Mon Dec 23 17:55:06 2019 "& ...

  7. linux高级技巧:heartbeat+lvs

    linux高级技巧:heartbeat+lvs   Heartbeat 项目是 Linux-HA 工程的一个组成部分,它实现了一个高可用集群系统.心跳服务和集群通信是高可用集群的两个关键组件,在 He ...

  8. 部署LVS-NAT模式调度器

    创建集群服务器 [root@proxy ~]# yum -y install ipvsadm [root@proxy ~]# ipvsadm -A -t -s wrr 添加真实服务器 [root@pr ...

  9. AcWing:167. 木棒(dfs + 剪枝)

    乔治拿来一组等长的木棒,将它们随机地砍断,使得每一节木棍的长度都不超过50个长度单位. 然后他又想把这些木棍恢复到为裁截前的状态,但忘记了初始时有多少木棒以及木棒的初始长度. 请你设计一个程序,帮助乔 ...

  10. R-ets()

    前情需知 指数预测模型 指数模型是用来预测时序未来值的最常用模型.这类模型相对比较简单,但是实践证明它们的短期预测能力较好.不同指数模型建模时选用的因子可能不同.比如 单指数模型(simple/sin ...