recyclerview
1.setSpanLookup 可以根据不同的行规则去设置每行的item数以达到不规则排序的效果
2.StaggeredGridLayoutManager 实现最简单瀑布流
3.PagerSnapHelper() 像viewpage一样一页一页滚动 快速滚动LinearSnapHelper
4.禁止滚动
LinearLayoutManager layoutManager = new LinearLayoutManager(mContext) {
@Override
public boolean canScrollVertically() {
return false;
}
};
让iamgeview保持图片的比例
<ImageView
android:id="@+id/image"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:scaleType="fitXY"
android:adjustViewBounds="true"/>
最小距离
ViewConfiguration.getScaledTouchSlop();触发移动事件的最小距离,自定义View处理touch事件的时候,有的时候需要判断用户是否真的存在movie,系统提供了这样的方法。表示滑动的时候,手的移动要大于这个返回的距离值才开始移动控件。
int slop = ViewConfigurationCompat.getScaledPagingTouchSlop(ViewConfiguration.get(context));
DragFloatingActionButton + recyclerview
问题 滚动时背景变化导致layout 回原来位置
解决 1:
解决2 :
floatingactionbutton icon 显示太小:
在sdk28之前,需要设置
android:scaleType="center"
app:borderWidth="0dp"
app:fabSize="normal"
就可以正常显示,但sdk升级到28之后再次出现图片缩小问题,所以sdk28之后需设置属性
app:maxImageSize="@dimen/dp_50"
就可以正常显示了
app:backgroundTint - 设置FAB的背景颜色。
app:rippleColor - 设置FAB点击时的背景颜色。
app:borderWidth - 该属性尤为重要,如果不设置0dp,那么在4.1的sdk上FAB会显示为正方形,而且在5.0以后的sdk没有阴影效果。所以设置为borderWidth="0dp"。
app:elevation - 默认状态下FAB的阴影大小。
app:pressedTranslationZ - 点击时候FAB的阴影大小。
app:fabSize - 设置FAB的大小,该属性有两个值,分别为normal和mini,对应的FAB大小分别为56dp和40dp。
android:src - 设置FAB的图标,Google建议符合Design设计的该图标大小为24dp。
app:layout_anchor - 设置FAB的锚点,即以哪个控件为参照点设置位置。
app:layout_anchorGravity - 设置FAB相对锚点的位置,值有 bottom、center、right、left、top等。
View绘制分三个步骤,顺序是:onMeasure,onLayout,onDraw。经代码亲测,log输出显示:调用invalidate方法只会执行onDraw方法;调用requestLayout方法只会执行onMeasure方法和onLayout方法,并不会执行onDraw方法。
dispatch
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
getParent().requestDisallowInterceptTouchEvent(true);
return super.dispatchTouchEvent(ev);
}
lottie 错误
Android lottie java.lang.IllegalStateException: Missing values for keyframe
对接lottie时,根据Lottie用法,加载json,效果显示不出来,lottie官网上预览json显示正常,大量搜索后,发现lottie低版本,不适用现在Adobe After Effects和bodymovin插件,重新导出json后,正常显示。
Settings > Advanced > Export Old Json Format
软键盘收起
private void hideInput(EditText editText){
InputMethodManager manager = (InputMethodManager) getSystemService(INPUT_METHOD_SERVICE);
manager.hideSoftInputFromWindow(editText.getWindowToken(),0);
}
Android 软键盘的弹出及关闭
Android 软键盘的弹出及关闭2
/**
* 1、获取main在窗体的可视区域
* 2、获取main在窗体的不可视区域高度
* 3、判断不可视区域高度
* ①大于180:键盘显示 获取Scroll的窗体坐标,算出main需要滚动的高度,使scroll显示。
* ②小于180:键盘隐藏
* 该值根据屏幕可以做出修改,在大屏手机上可以适当的调大,不然会出现问题。
*
* @param main 根布局
* @param scroll 需要显示的最下方View
*/
public void addLayoutListener(final View main, final View scroll) {
main.getViewTreeObserver().addOnGlobalLayoutListener(() -> {
Rect rect = new Rect();
main.getWindowVisibleDisplayFrame(rect);
int mainInvisibleHeight = main.getRootView().getHeight() - rect.bottom;
if (mainInvisibleHeight > 180) {
int[] location = new int[2];
scroll.getLocationInWindow(location);
int scrollHeight = (location[1] + scroll.getHeight() + ((RelativeLayout.LayoutParams) scroll.getLayoutParams()).bottomMargin) - rect.bottom;
if (scrollHeight > 0) {
main.scrollTo(0, scrollHeight);
}
} else {
main.scrollTo(0, 0);
}
});
}
Recyclerview 设置ItemView的内嵌偏移长度(inset)
TextItemDecoration.class
@Override
public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
super.getItemOffsets(outRect, view, parent, state);
//只是添加下面这一行代码
outRect.set(50, 50, 50, 50);
}
键盘不弹出
android:focusable="true"
android:focusableInTouchMode="true"
清空焦点去掉竖线
edittext.clearfocuse();
editText.setFocusableInTouchMode(fase);
splashactivity theme 刘海屏适配
<style name="SplashTheme" parent="Theme.AppCompat.Light.NoActionBar">
<item name="android:windowBackground">@drawable/splash</item>
<item name="android:windowFullscreen">true</item>
<!--不让windowBackground延申到navigation bar区域-->
<item name="android:windowDrawsSystemBarBackgrounds">false</item>
<!--适配Android P刘海屏-->
<item name="android:windowLayoutInDisplayCutoutMode">shortEdges</item>
</style>
[lianjie]([https://blog.csdn.net/wds1181977/article/details/88095843](https://blog.csdn.net/wds1181977/article/details/88095843)
)
[Q 文件]([https://www.jb51.net/article/177201.htm](https://www.jb51.net/article/177201.htm)
)([https://www.anzhuoq.com/thread-4890.htm](https://www.anzhuoq.com/thread-4890.htm)
)
ViewGroup容器组件的绘制,当它没有背景时直接调用的是dispatchDraw()方法, 而绕过了draw()方法,当它有背景的时候就调用draw()方法,而draw()方法里包含了dispatchDraw()方法的调用。因此要在ViewGroup上绘制东西的时候往往重写的是dispatchDraw()方法而不是onDraw()方法,或者自定制一个Drawable,重写它的draw(Canvas c)和 getIntrinsicWidth(), getIntrinsicHeight()方法,然后设为背景。
常用SpannableStringBuilder
SpannableStringBuilder stringBuilder = new SpannableStringBuilder(mPrivacyTv.getText());
stringBuilder.setSpan(new ClickableSpan() {
@Override
public void onClick(@NonNull View widget) {
WebViewActivity.startThisActivity(getActivity(), getResources().getString(R.string.privacy_url), getResources().getString(R.string.policy));
}
@Override
public void updateDrawState(@NonNull TextPaint ds) {
super.updateDrawState(ds);
ds.setColor(getResources().getColor(R.color.textColorPrimary));
ds.setUnderlineText(true);
}
}, 0, mPrivacyTv.getText().length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
mPrivacyTv.setText(stringBuilder);
mPrivacyTv.setMovementMethod(LinkMovementMethod.getInstance());
private SpannableStringBuilder getSpannerText(String strColor, int color, String str){
SpannableStringBuilder stringBuilder = new SpannableStringBuilder(str);
int start = str.indexOf(strColor);
int end = start + strColor.length();
if (start >= 0){
stringBuilder.setSpan(new ForegroundColorSpan(getResources().getColor(color)),start,end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
}
return stringBuilder;
}
retrofit interceptor 获取respond的内容 但是这样会导致accpt收不到数据
// Logger.e(TAG, "intercept: respond:" + response.body().string());
profiler 拦截net网络请求 查看其参数
view.getLocationOnscren为0 猜测是因为view 没有完全展示在页面中
BigDecimal 用String.valueof() 精度最大
滚动冲突vertical recycler+horizatal recycler+coordinatorlayout.就是对内部嵌套的横向滑动的RecyclerView设置它的setNestedScrollingEnabled(false)
如果是整个xml中使用fragment(内部页面为 linearlyout包裹reccyler) 那么很有可能会导致其内部的recycler和外部的滚动控件造成滚动冲突
避免多次加载fragment
private View rootView;// 缓存Fragment view
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState)
{
Log.i(TAG, "onCreateView");
if (rootView == null)
{
rootView = inflater.inflate(R.layout.fragment_1, null);
}
// 缓存的rootView需要判断是否已经被加过parent,如果有parent需要从parent删除,要不然会发生这个rootview已经有parent的错误。
ViewGroup parent = (ViewGroup) rootView.getParent();
if (parent != null)
{
parent.removeView(rootView);
}
return rootView;
}
Android私有目录分享权限问题
在分享私有目录下的文件的时候失败(Uri.parser的Uri),11及以上的系统不允许访问其它app的私有目录,用FileProvider就可以访问
https://developer.android.google.cn/training/secure-file-sharing/share-file?hl=zh-cn