Android 2021版本
1、工程结构介绍
项目工程文件介绍:
.gradle和.idea:项目自动生成的一些文件,一般打包的时候会将这两个包删掉然后去发给别人
.app:项目中的资源和代码都会在这里面,工作里面的核心目录
gradle:是构建器,里面有一个wrapper文件,
gitignore:版本控制的时候,指定把我们的文件排除在版本控制里面
gradlew:用在Mac或者linux系统上的gradle执行指令的时候所用的文件
gradlew.bat:windows里面执行gradle所用文件。
local.properties:指定sdk的路径
settings.gradle:指定我们项目中所有引入的模块
下面是app内部的目录结构:
- build:这个文件就是通过build>make project去生成的。
一般在压缩的时候也会将这个删除去减少我们压缩包的大小。
-
libs:当我们项目中使用了第三方的jar包的时候,就需要把jar包放到这个目录中去。
-
src:
- androidTest:编写测试用例的
- main
- java:java代码
-
res:
- AndroidManifest.xml:四大组件的注册和权限等。
- test:单元测试
-
gitignore:这个内部的gitignore和外部的作用一样,只不过外面是管理整个工程的,而这个是管理我们这个app的。
-
build.gradle:app模块的构建脚本,一般一些第三方库的依赖就是在这里
-
properties:指定项目代码的混淆规则
2、TextView
2.1基础属性:
layout_width 、layout_height:设置组件的宽高
-
match_parent:父容器有多宽,这个组件就有多宽
-
wrap_content:内容有多宽,这个组件就有多宽
-
或者直接输入数值
id: 设置一个组件的id
text:
-
在布局文件中设置组件的文本内容:
android:text="hello"
-
也可以在java代码当中设置文本内容,如果两个都设置的话,java代码中设置的文本内容会把xml布局文件中设置的文本内容给覆盖掉。
TextView tv_one = findViewById(R.id.tv_one); tv_one.setText("leo");
两个都设置了之后,最终的显示结果:
textColor:设置字体的颜色
颜色一共有八位。
- 最前面这两位设置咱们的透明度。00表示纯透明,ff表示不透明。
- 剩下六位依次是:红、绿、蓝
textStyle:设置字体的风格
如果属性忘记有什么了,可以直接点进来看。
textSize:字体大小,一般用的是sp
android:textSize="30sp"
background: 控件的背景颜色,可以理解为填充整个控件的颜色,可以是图片
android:background="#FFFF0000"
gravity: 设置控件中内容的对齐方向,TextView中 是文字,ImageView中是图片等等。
android:gravity="center"
上面的这些可能不太规范,一般都是放到values这个文件夹下的文件里面
比如设置字符串名字就是string.xml里面写
<resources>
<string name="tv_one" >123456</string>
</resources>
就可以在layout里面引用了
android:text="@string/tv_one"
2.2带阴影的TextView
-
android:shadowColor:设置阴影颜色需要与shadowRadius-起使用
-
android:shadowRadius:设置阴影的模糊程度设为0.1就变成字体颜色了,建议使用3.0
-
android:shadowDx:设 置阴影在水平方向的偏移就是水平方向阴影开始的横坐标位置
-
android:shadowDy:设置阴影在竖直方向的偏移就是竖直方向阴影开始的纵坐标位置
<TextView
android:id="@+id/tv_one"
android:layout_width="200dp"
android:layout_height="200dp"
android:text="@string/tv_one"
android:textColor="@color/design_default_color_background"
android:textStyle="italic"
android:textSize="30sp"
android:shadowColor="@color/red"
android:shadowRadius="3"
android:shadowDx="10.0"
android:shadowDy="10.0"
android:gravity="center"/>
显示效果如图所示:
2.3跑马灯效果的TextView
- android:singleLine:内容单行显示
- android:focusable:是否可以获取焦点
- android:focusablelnTouchMode;用于控制视图在触摸模式下是否可以聚焦
- android:ellipsize="marquee"在哪里省略文本
- android:marqueeRepeatLimit:字幕动画重复的次数
<TextView
android:id="@+id/tv_two"
android:layout_width="200dp"
android:layout_height="200dp"
android:text="@string/circle"
android:textColor="@color/design_default_color_background"
android:textStyle="italic"
android:textSize="30sp"
android:shadowColor="@color/red"
android:shadowRadius="3"
android:shadowDx="10.0"
android:shadowDy="10.0"
android:gravity="center"
android:singleLine="true"
android:ellipsize="marquee"
android:marqueeRepeatLimit="marquee_forever"
android:focusable="true"
android:focusableInTouchMode="true"/>
设置完这些属性还是没有动起来:
-
设置可点击:
android:clickable="true"
点击之后显示跑马灯效果。
-
自定义MyTextView继承TextView(5.0之后是
androidx.appcompat.widget.AppCompatTextView
),重写isFocused()方法@Override public boolean isFocused() { return true; }
然后更改我们的组件的名字把TextView换成我们自己的组件
com.android.myapplication.MyTextView
-
或者直接在
<TextView>
标签中设置一个<requestFocus/>
先从第二问开始吧,
AppCompatTextView
继承自TextView
,是对TextView
的一种扩展,因为在5.0中首次推出了MaterialDesign
这种设计风格,但是众所周知,5.0推出不可能所有的设备全部都更新到最新版本,为了在早期版本上实现新的功能(这些新功能比如从源码注释中解读到backgroundTint
属性,根据文本内容自适应大小等),即为了新特性同样可以兼容老版本,framework在创建TextView
实例的时候,自动帮我们进行了替换。其它的AppCompatXXX与XXX的关系也是如此。
3.Button
3.1StateListDrawable
设置background:
设置按钮颜色的时候在老版本设置按钮颜色的时候是显示正常的,但是新版本需要设置一下
在themes文件下添加.Bridge
新建一个drawable资源中,设置按钮按下和不按下的时候的区别
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:drawable="@drawable/ic_baseline_account_balance_24" android:state_pressed="true"></item>
<item android:drawable="@drawable/ic_baseline_accessibility_24" android:state_pressed="false"></item>
</selector>
设置backgroundTint
在res下新建一个color文件夹,新建一个draw资源文件中
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:color="ff0000" android:state_pressed="true"/>
<item android:color="@color/black" />
</selector>
在布局文件中:
android:backgroundTint="@color/btn_color_seletor"
设置前景色foreground
android:foreground="#50667788"
事件监听:
- 首先现在oncreate方法里面找到btn
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button btn = findViewById(R.id.btn);
}
可以添加各种监听事件:
- 点击事件
//点击事件
btn.setOnLongClickListener(new View.OnLongClickListener() {
@Override
public boolean onLongClick(View view) {
Log.e(TAG,"ONCLICK:");
return false;
}
});
当返回值为true的时候,没有打印onclick方法。先执行ontouch-->onlongclick(true),由于条件判断导致onclick这个方法没有执行。
- 触摸事件
//触摸事件
btn.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View view, MotionEvent motionEvent) {
Log.e(TAG,"ontouch:"+motionEvent.getAction());
return false;
}
});
鼠标按下去是 ontouch0
鼠标长按之后左右move是2
鼠标抬起来是ontouch1
onclick事件发生在ontouch 0之后
- 长按事件
//长按事件
btn.setOnLongClickListener(new View.OnLongClickListener(){
@Override
public boolean onLongClick(View view) {
Log.e(TAG,"onLongClick:");
return false;
}
});
当我们设置这个方法的返回值为true的时候,就不再显示其他事件了。
事件就被ontouch消费了,就不再传递给onclick和onlongclick了
你要点击按钮,肯定事先得触摸按钮,所以才有触摸事件,然后才有点击事件,然后点击按钮又会有按下和按钮弹起这些时间。
长按事件处理除了这个内部类监听之外还有可以添加自定义方法:
public void helloClick(View view) {
Log.e(TAG,"ONCLICK:");
}
在layout中只需要写这个方法的名字即可
android:onClick="helloClick"
4、EditText
属性
- android:hint输入提示
android:hint="请输入用户名:"
-
andrid:textColorHint:输入提示文字的颜色
-
android:nputType:输入类型
比如说设置为phone
android:inputType="phone"
- android:drawablexXxx :在输入框的指定方位添加图片
比如说我们添加一个左侧的图标:
android:drawableLeft="@drawable/ic_baseline_person_24"
-
android:drawablePadding:设置图片与输入内容的间距
图标距离文字太近了,可以设置一定的边距:
android:drawablePadding="20dp"
-
android:paddingXxx:设置内容与边框的间距
android:paddingLeft="20dp"
-
android:background:背景色
获取输入框内容:
设置两个组件的id分别为et和btn
public class MainActivity extends AppCompatActivity { private EditText et; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); //分别获取两个id Button btn = findViewById(R.id.btn); et = findViewById(R.id.et); //设置监听事件 btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { //输出 String text = et.getText().toString(); Log.e("leo","輸入的内容"+text); } }); } }
这里一直获取不出来,是因为选择错了模拟器!
5、imageView
属性
- scaleType
android:scaleType="matrix"
缩放类型:
-
fitStart:保持宽高比缩放图片,直到较长的边与Image的边长相等缩放完成后将图片放在ImageView的左上角
-
fitCenter:默认值,同上,缩放后放于中间
-
fitEnd:同上,缩放后放于右下角
-
fitXY:对图像的横纵方向进行独立缩放,使得该图片完全适应lmageView,但是图片的宽高比可能会发生改变
-
center保持原图的大小,显示在ImageView的中心。 当原图的size大于ImageView的size,超过部分裁剪处理。
并不会对图片进行一个拉伸
-
centerCrop保持宽高比缩放图片,直到完全覆盖ImageView,可能会出现图片的显示不完全
-
centerInside保持宽高比缩放图片,直到ImageView能够完全地显示图片
centerInside最主要的功能是把这个图片显示在imageView里面,如果你的图片小于这个imageView的话,就直接显示在最中间
如果你的图片大于imageView的话,就将其缩放。
如果图片大于imageView的话,centerInside和fitCenter显示效果相同。但是如果图片小的话,显示效果是不同的。
- matrix不改变原图的大小,从ImageView的左上角开始绘制原图,原图超过ImageView的部分作裁剪处理
imageView很少有情况和我们的imageView的长宽一样。
所以我们需要设置
maxWidth和maxHeight:
android:maxHeight="200dp"
android:maxWidth="200dp"
添加之后并没有什么变化,设置一下adjustViewBounds为true:
android:adjustViewBounds="true"
6、Progressbar
属性
-
android:max:进度条的最大值
-
anndroid:progress: 进度条已完成进度值。通过这个值来获取当前进度条的值
- 给button加上一个click方法load:
android:onClick="load"
-
progressbar设置id
android:id="@+id/pb2"
-
progressbar获取id
private ProgressBar pb2; //oncreate方法: pb2 = findViewById(R.id.pb2);
-
编写方法load
public void load(View view) { //设置进度条的值 int progress = pb2.getProgress(); //每次click的时候这个数值都要增加10 progress+=10; //重新设置progress的数值 pb2.setProgress(progress); }
-
android:indeterminate:如果设置成true,则进度条不精确显示进度,类似于我们之前的圆形进度条,一直在滚动。不固定。
-
style=" ?android:attr/progressBarStyleHorizontal":水平进度条
style="?android:attr/progressBarStyleHorizontal"
设置进度条显示或者隐藏:
private ProgressBar pb;
//在oncreate方法里面
pb = findViewById(R.id.pb);
然后创建事件监听:
public void helloClick(View view) {
if (pb.getVisibility() == View.GONE){
pb.setVisibility(View.VISIBLE);
}else{
pb.setVisibility(View.GONE);
}
}
View.VISIBLE--->可见
View.INVISIBLE--->不可见,但这个View仍然会占用在xml文件中所分配的布局空间,不重新layout
View.GONE---->不可见,但这个View在ViewGroup中不保留位置,会重新layout,不再占用空间,那后面的view就会取代他的位置,
7、notification
创建一个NotificationManager
NotificationManager类是一个通知管理器类,这个对象是由系统维护的服务,是以单例模式的方式获得,所以一般并不直接实例化这个对象。
-
使用Activity. getSystemService(String)方法获取NotificationManager对象
manager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
Activity.getSystemSericelString)方法可以通过Android系统级服务的句柄,返回对应的对象。在这里需要返NotificationManager,所以直接传递Context.NOTIFICATION_ SERVICE即可。
-
使用Builder构造器来创建Notification对象
使用NotificationCompat类的Builder构造器来创建Notification对象,可以保证程序在所有的版本上都能正常工作。Android8.0新增 了通知渠道(channelId)这个概念,如果没有设置,则通知无法Android8.0的机器上显示Notification build = new NotificationCompat.Builder(this,"hello").build();
这里builder的第二个参数就是ChannelId
NotificationChannel
通知渠道: Android 8.0引入了通知渠道,其允许您为要显示的每种通知类型创建用户可自定
义的渠道。
所以我们需要判断一下版本:
if (Build.VERSION.SDK_INT>=Build.VERSION_CODES.O){
NotificationChannel channel = new NotificationChannel("hello", "测试通知", NotificationManager.IMPORTANCE_HIGH);
//把创建的放入这里面
manager.createNotificationChannel(channel);
}
NotificationChannel
public NotificationChannel(String id, CharSequence name, int importance) {
throw new RuntimeException("Stub!");
}
-
id:实际上就是我们的channelId,这里必须要和我们notification里面的channelId是一样的!
-
CharSequence name:我们后期用户可以看懂即可
-
int importance:
通知重要程度:
- IMPORTANCE_ NONE关闭通知
- IMPORTANCE_ MIN 开启通知,不会弹出,但没有提示音,状态栏中无显示
- IMPORTANCE_ Low开启通知,不会弹出,不发出提示音,状态栏中显示
- IMPORTANCE_ DEFAULT开启通知,不会弹出,发出提示音,状态栏中显示
- IMPORTANCE_HIGH 开启通知,会弹出,发出提示音,状态栏中显示
设置通知的一些属性
- setContentTitle 设置通知的标题
- setContentText设置通知的内容
- setSmallIcon()
Notification notification = new NotificationCompat.Builder(this,"hello")
.setContentTitle("官方通知")//设置通知的标题
.setContentText("世界那么大,想去走走吗?")//设置通知的内容
.setSmallIcon(R.drawable.ic_baseline_person_24)//设置通知的图标
.build();
public Builder setSmallIcon(int icon)
返回值为int型,所以只能不能是有颜色的图
Android从5.0系统开始,对于通知栏图标的设计进行了修改现在Google要求,所有应用程序的通知栏图标,应该只使用alpha图层来进行绘制,而不应该包括RGB图层。简而言之来说,就是图片不能带颜色!
//通过发送通知来触发通知
public void sendNotification(View view) {
//这里的第一个参数
manager.notify(1,notification);
}
-
id 指定通知的 id,要保证每个通知的 id 是不同的;
-
Notification 对象,传入之前创建好的即可。
显示效果:
其他的一些属性:
-
setLargelcon(Bitmap icon)设置通知的大图标
.setLargeIcon(BitmapFactory.decodeResource(getResources(),R.drawable.test1))
这里的Icon的格式是bitmap所以可以设置彩色的图片
但是需要转换一下:
BitmapFactory.decodeResource(getResources(),R.drawable.test1)
-
setColor(int argb)设置小图标的颜色
这里的参数是int型的rgb,我们可以调用Color.parseColor(),把这个十六进制的颜色转化为int型的数值
-
setContentIntent(PendingIntent intent)设置点击通知后的跳转意图
需要先创建Intent:
Intent intent = new Intent(this, NotifictionActivity.class); PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
然后将这个pendingintent放入到setContentIntent方法中
PendingIntent.getActivity中的四个方法:
context:上下文对象。
requstCode:请求码,发件人的私人请求代码(当前未使用)。
intent:请求意图。用于要指明要启动的类以及数据的传递;
flags:这是一个关键的标志位:
-
setAutoCanelboolean boolean)设置点击通知后自动清除通知
.setAutoCancel(true)
设置取消通知:
public void cancelNotification(View view) { manager.cancel(1); }
注意这里的cancel传入的参数id是要和我们刚开始manager.notify(1,notification);这里面的id要一致!
-
setWhen(long when)设置通知被创建的时间
8、toolbar
在res>values>themes.xml文件中可以看到:
我们可以将其修改为NoActionBar
之后可以看出这里的toolbar没有了
属性:
特别注意这里的这个Toolbar应该是androidx.appcompat.widget.Toolbar!
-
android:layout_width="match_parent"
-
android:layout_height="?attr/actionBarSize"
<androidx.appcompat.widget.Toolbar
android:background="#ffff00"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"/>
注意此时的linearlayout的宽高都是match_parent
-
设置标题颜色 边距:
app:title="标题"
app:titleTextColor="#ff0000"
app:titleMarginStart="90dp" -
设置子标题的颜色
app:subtitle="子标题"
app:subtitleTextColor="@color/purple_200" -
设置logo和navigationicon
app:logo="@mipmap/ic_launcher"
app:navigationIcon="@drawable/ic_baseline_arrow_back_24"
app:title="标题"
app:titleTextColor="#ff0000"
app:titleMarginStart="90dp"
app:subtitle="子标题"
app:subtitleTextColor="@color/purple_200"
app:logo="@mipmap/ic_launcher"
app:navigationIcon="@drawable/ic_baseline_arrow_back_24"
效果:
同时也可以通过java里面的类直接设置:
Toolbar toolbar2 = findViewById(R.id.tb2);
toolbar2.setNavigationIcon(R.drawable.ic_baseline_arrow_back_24);
toolbar2.setTitle("TOOLBAR2");
toolbar2.setSubtitle("TOOLBAR2副标题");
toolbar2.setNavigationIcon(R.mipmap.ic_launcher);
Log.e("hello","onclick:toolbar2被点击了~");
设置toolbar内容居中:
<androidx.appcompat.widget.Toolbar
android:id="@+id/tb3"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:layout_marginTop="10dp"
android:background="#ffff00"
>
<TextView
android:text="标题"
android:layout_gravity="center"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
</androidx.appcompat.widget.Toolbar>
可以直接在Toolbar里面添加一个TextView,这个TextView中设置 android:layout_gravity="center"
。
9、alertDialog
AlertDialog. Builder builder = new AlertDialog Bider(context);构建Dialog的各种参数
public void showDialog(View view) {
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setIcon(R.mipmap.ic_launcher)
.setTitle("我是对话框")
.setMessage("今天天气怎么样")
.create()
.show();
}
这些set方法返回都是一个builder,所以我们可以用链式结构,但是create返回值是alertdialog,show又是dialog这个类里面的方法,所以我们把show放在create后面。
添加按钮:
//设置确定按钮
.setPositiveButton("确定", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialogInterface, int i) {
Log.e("hello","onclick:点击了确定");
}
})
//设置取消按钮
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialogInterface, int i) {
Log.e("hello","onclick:点击了取消");
}
})
//设置中间按钮
.setNeutralButton("中间", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialogInterface, int i) {
Log.e("hello","onclick:点击了中间");
}
})
设置布局:
新建一个dialog_view.xml文件
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal"
xmlns:android="http://schemas.android.com/apk/res/android"
android:background="@color/purple_200">
<ImageView
android:src="@mipmap/ic_launcher"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="哈哈,天气很好~"/>
</LinearLayout>
将其设置为这个dialog的布局:
View dialogview = getLayoutInflater().inflate(R.layout.dialog_view,null);
.setView(dialogview)