1.封装好的轮轮播滑动框架(AndroidImageSlider)
2.YouTubeMusic略微调整了网络和Android的深色主题
3.å®åViewFlipperç使ç¨è¯¦è§£
4.android使用ViewPager实现轮播效果教程
5.Android-自定义轮播Banner-Viewpaper2实现+贪吃蛇原点指示器+触摸暂停轮播处理
封装好的滑动框架(AndroidImageSlider)
广告轮播条在各类app中扮演重要角色。今天介绍一款轮播图开源项目,播源它将轮播所需ViewPager与计时器进行封装,码a码操作简便,图代支持Gradle在线依赖。轮轮播项目内含多种滑动动画,播源北海网站源码满足大多数需求。码a码 使用方法如下: 首先,图代在项目中添加以下三个库的轮轮播依赖:Picasso:高效加载库
nineoldandroids:动画库
AndroidImageSlider:轮播图库
在AndroidManifest.xml中加入网络访问权限,以支持从URL加载。播源 在activity_main.xml布局文件中设置轮播图高度为dp,码a码并包含SliderLayout和PagerIndicator。图代PagerIndicator用于指示当前页面。轮轮播duilib网盘源码 在MainActivity.java中初始化控件,播源将描述和URL存入Map。码a码遍历Map,为每个页面创建TextSliderView,设置描述、URL、缩放类型等参数。调用SliderLayout的addSlider方法添加页面,并自定义滑动动画、指示器样式、描述显示动画和滚动时间。 实现页面点击监听,丰源码头在哪处理ViewPager中的页面改变事件。 滑动动画通过SliderLayout的setPresetTransformer方法设置,传入Transformer对象枚举值,可选择多种动画效果。 自定义指示器属性,如颜色、大小等,增加视觉效果。 下载源码进行实验,推荐从开源项目作者的GitHub仓库获取demo。 了解更多信息和项目源码,请访问以下链接:AndroidImageSlider项目地址:github.com/daimajia/AndroidImageSlider
项目源码下载:github.com/ansen/AndroidImageSlider
为了获取更多更新的星外主机源码技术文章和动态,建议关注公众号,获取Android开发、最新动态、开源项目等信息。YouTubeMusic略微调整了网络和Android的深色主题
YouTubeMusic悄然演绎色彩美学:深色主题微调
如同Reddit用户的洞察与9to5Google的深度剖析,YouTubeMusic的网络和Android应用程序正悄然进行着微妙的视觉革新。初版上线时,深色主题即为其标志,如今,细微的调色板调整正悄然上演。原本界面以黑与灰为主调,如今,奇兔源码社区一些深邃的灰色地带已被悄然着上更深的黑色,形成更加和谐的视觉层次。 在Android终端上,状态栏的暗度进一步提升,这一微妙的变化在9to5Google的揭示中尤为明显,库选项卡中的“最近活动”轮播现在与背景形成鲜明对比,更加醒目。值得注意的是,“探索”选项卡、搜索结果以及播放列表页面的界面整合,顶部设计更加统一,提升了整体的观感体验。 尽管这些改动看似微小,却透露出一种精心设计的统一感。据观察,这些调整似乎预示着YouTubeMusic正在逐步向浅色主题靠拢,以迎合用户对于更舒适视觉体验的需求。几天前,这些改变已经开始逐步应用,为用户的使用体验增添了一份贴心的考虑。 不难想象,深色主题的微调或许正是向全面支持浅色模式的过渡,这种策略旨在确保无论在何种环境下,都能为用户提供最理想的观感体验,让音乐欣赏的过程更加沉浸和享受。YouTubeMusic正以用户的视角出发,不断优化设计细节,期待着更多创新的色彩碰撞与融合。å®åViewFlipperç使ç¨è¯¦è§£
ViewFlipperçä»ç»å使ç¨
ViewFlipperæ¯Androidèªå¸¦çä¸ä¸ªå¤é¡µé¢ç®¡çæ§ä»¶ï¼ä¸å¯ä»¥èªå¨ææ¾ï¼åViewPagerä¸åï¼ViewPageræ¯ä¸é¡µé¡µçï¼èViewFlipperåæ¯ä¸å±å±çï¼åViewPagerä¸æ ·ï¼å¾å¤æ¶åï¼ç¨æ¥å®ç°è¿å ¥åºç¨åçå¼å¯¼é¡µï¼æè ç¨äºå¾çè½®æã
ä¸ã常ç¨çå±æ§åæ¹æ³å±æ§ï¼
android:inAnimation//设置ViewæImageViewè¿å ¥å±å¹æ¶ä½¿ç¨çå¨ç»android:outAnimation//设置ViewæImageViewéåºå±å¹æ¶ä½¿ç¨çå¨ç»android:flipInterval//设置ViewæImageViewä¹é´åæ¢çæ¶é´é´é常ç¨æ¹æ³ï¼
setInAnimationï¼è®¾ç½®ViewæImageViewè¿å ¥å±å¹æ¶ä½¿ç¨çå¨ç»
setOutAnimationï¼è®¾ç½®ViewæImageViewéåºå±å¹æ¶ä½¿ç¨çå¨ç»
showNextï¼è°ç¨è¯¥æ¹æ³æ¥æ¾ç¤ºViewFlipperéçä¸ä¸ä¸ªViewæImageView
showPreviousï¼è°ç¨è¯¥æ¹æ³æ¥æ¾ç¤ºViewFlipperçä¸ä¸ä¸ªViewæImageView
setFilpIntervalï¼è®¾ç½®ViewæImageViewä¹é´åæ¢çæ¶é´é´é
startFlippingï¼ä½¿ç¨ä¸é¢è®¾ç½®çæ¶é´é´éæ¥å¼å§åæ¢ææçViewæImageViewï¼åæ¢ä¼å¾ªç¯è¿è¡
stopFlippingï¼åæ¢ViewæImageViewåæ¢
äºãéæå¯¼å ¥è¿è¡ææï¼
å®ç°è¿ä¸ªææé常çç®å
ç¼åviewpfilper.xmlå¸å±æ件
<?xmlversion="1.0"encoding="utf-8"?><LinearLayoutxmlns:android=".edu.huse.banner_re.util;import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
/
*** 加载的工具类
* @author xray
*
*/
public class ImageLoader {
/
*** 加载完成的回调接口
* 添加URL参数,用于做错位判断
*/
public interface OnImageLoadListener{
//完成加载
void onImageLoadComplete(String url, Bitmap bitmap);
}
private OnImageLoadListener mListener;
/
*** 启动加载任务
* @param urlStr
* @param listener
*/
public void loadImage(String urlStr,OnImageLoadListener listener){
this.mListener = listener;
new ImageLoadTask().execute(urlStr);
}
/
*** 加载任务
* @author xray
*
*/
class ImageLoadTask extends AsyncTask
@Override
protected UrlAndBitmap doInBackground(String... params) {
try {
//创建URL,指定地址
URL url = new URL(params[0]);
//打开连接获得HttpURLConnection对象
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
//获得文件输入流
InputStream stream = conn.getInputStream();
//把输入流转换为
Bitmap bmp = BitmapFactory.decodeStream(stream);
//关闭流
stream.close();
return new UrlAndBitmap(params[0],bmp);
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
//包装地址和
class UrlAndBitmap{
String url;
Bitmap bitmap;
public UrlAndBitmap(String url, Bitmap bitmap) {
this.url = url;
this.bitmap = bitmap;
}
}
@Override
protected void onPostExecute(UrlAndBitmap result) {
//进行接口回调
if(mListener != null){
mListener.onImageLoadComplete(result.url,result.bitmap);
}
}
}
}
JSONLoader.java
//读取服务器json
package cn.edu.huse.banner_re.util;
import android.os.AsyncTask;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
/
*** 读取服务器JSON的工具类
* @author xray
*
*/
public class JSONLoader {
public interface OnJSONLoadListener{
void onJSONLoadComplete(String json);
}
private OnJSONLoadListener mListener;
public void loadJSON(String urlStr,OnJSONLoadListener listener){
this.mListener = listener;
new JSONLoadTask().execute(urlStr);
}
class JSONLoadTask extends AsyncTask
@Override
protected String doInBackground(String... params) {
try {
URL url = new URL(params[0]);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
InputStream inputStream = conn.getInputStream();
byte[] buffer = new byte[];
int len = 0;
//StringBuffer 线程安全,性能低 VS StringBuilder 线程不安全,性能高
StringBuilder strB = new StringBuilder();
while((len = inputStream.read(buffer)) != -1){
//将字节码转换为String
String str = new String(buffer,0,len);
strB.append(str);
}
inputStream.close();
return strB.toString();
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
@Override
protected void onPostExecute(String result) {
if(mListener != null){
mListener.onJSONLoadComplete(result);
}
}
}
}
MyViewPagerAdapter.java
package cn.edu.huse.banner_re.adapter;
import java.util.List;
import android.support.v4.view.PagerAdapter;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
/
*** 轮播的适配器
* @author Lenovo
*
*/
public class MyViewPagerAdapter extends PagerAdapter {
private List mData;
public MyViewPagerAdapter(List mData) {
this.mData = mData;
}
//返回视图的数量
@Override
public int getCount() {
return mData.size();
}
@Override
public boolean isViewFromObject(View arg0, Object arg1) {
return arg0 == arg1;
}
//销毁
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
container.removeView(mData.get(position));
}
//添加
@Override
public Object instantiateItem(ViewGroup container, int position) {
container.addView(mData.get(position));
return mData.get(position);
}
}
MainActivity.java
package cn.edu.huse.banner_re;
import java.util.ArrayList;
import java.util.List;
import org.json.JSONArray;
import org.json.JSONObject;
import android.app.Activity;
import android.graphics.Bitmap;
import android.os.Bundle;
import android.os.Handler;
import android.support.v4.view.ViewPager;
import android.support.v4.view.ViewPager.OnPageChangeListener;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.ArrayAdapter;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.ListView;
import android.widget.RadioButton;
import cn.edu.huse.banner_re.adapter.MyViewPagerAdapter;
import cn.edu.huse.banner_re.util.ImageLoader;
import cn.edu.huse.banner_re.util.ImageLoader.OnImageLoadListener;
import cn.edu.huse.banner_re.util.JSONLoader;
import cn.edu.huse.banner_re.util.JSONLoader.OnJSONLoadListener;
public class MainActivity extends Activity implements OnClickListener{
public static final String URL_GIFT = "/majax.action?method=getGiftListpageno=1";
public static final String URL_BASE = "";
private ListView mLv;
//放置圆点的布局
private LinearLayout mLayout;
private ViewPager mPager;
//保存viewpager上的集合
private List mImageViews;
//保存圆点的集合
private List mDotViews;
//当前
private ImageView mDotView;
//当前下标
private int mIndex;
private Handler mHandler = new Handler();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//初始化viewpager
initViewPager();
}
private void initViewPager() {
new JSONLoader().loadJSON(URL_GIFT, new OnJSONLoadListener() {
@Override
public void onJSONLoadComplete(String json) {
//得到的地址
List list = josnPare(json);
//初始化主控件
initView();
//将显示到viewpager中
initImageViews(list);
//为ViewPager设置适配器
mPager.setAdapter(new MyViewPagerAdapter(mImageViews));
//加载圆点
initDot(mImageViews.size());
//实现的轮播
mPager.postDelayed(new Runnable() {
@Override
public void run() {
//下标自动++
mIndex++;
//当下标到最后时,就重新来过
if(mIndex = mImageViews.size()){
mIndex = 0;
}
//设置
mPager.setCurrentItem(mIndex);
//重复调用
mHandler.postDelayed(this, );
}
}, );
//添加页码滑动的监听,实现圆点的切换
mPager.setOnPageChangeListener(new OnPageChangeListener() {
//页选中后调用
@Override
public void onPageSelected(int arg0) {
//改变原来的圆点的状态
mDotView.setEnabled(true);
//更新当前的圆点
mDotView = mDotViews.get(arg0);
//将当前选中的圆点设为选中状态
mDotView.setEnabled(false);
//更新下标
mIndex = arg0;
}
//滑动中途调用
@Override
public void onPageScrolled(int arg0, float arg1, int arg2) {
}
//页面滑动状态修改
@Override
public void onPageScrollStateChanged(int arg0) {
}
});
}
});
}
/**获取圆点函数
* @param count viewpager上的数量
*/
private void initDot(int count){
mDotViews = new ArrayList();
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(
LinearLayout.LayoutParams.WRAP_CONTENT,
LinearLayout.LayoutParams.WRAP_CONTENT);
//设置边距
params.setMargins(5, 0, 5, 0);
for(int i = 0; i count; i++){
ImageView image = new ImageView(this);
//设置圆点
image.setImageResource(R.drawable.mydot);
//设置的位置,方便后面的取值
image.setTag(i);
//设置监听
image.setOnClickListener(this);
//添加到布局中
mLayout.addView(image, params);
//将圆点保存到集合中
mDotViews.add(image);
}
mDotView = mDotViews.get(0);
mDotView.setEnabled(false);
}
/
***显示的函数
*/
private void initImageViews(List list) {
mImageViews = new ArrayList();
for(String str : list){
final ImageView imageView = new ImageView(this);
//设置默认
imageView.setImageResource(R.drawable.ic_launcher);
new ImageLoader().loadImage(URL_BASE+str, new OnImageLoadListener() {
@Override
public void onImageLoadComplete(String url, Bitmap bitmap) {
if(bitmap != null){
imageView.setImageBitmap(bitmap);
}
}
});
//将ImageView保存到集合中
mImageViews.add(imageView);
}
}
/
*** JSON解析获取的地址
* @param url
* @return
*/
private List josnPare(String url){
List list = new ArrayList();
try {
JSONObject jobj = new JSONObject(url);
JSONArray array = jobj.getJSONArray("ad");
for(int i = 0; i array.length(); i++){
JSONObject obj = (JSONObject) array.get(i);
list.add(obj.getString("iconurl"));
}
} catch (Exception e) {
e.printStackTrace();
}
return list;
}
/
*** 初始化控件
*/
private void initView() {
//找到控件
mLv = (ListView) findViewById(R.id.lv_list);
View view = LayoutInflater.from(this).inflate(R.layout.activity_banner, null);
mPager = (ViewPager)view.findViewById(R.id.vp_image);
//圆点布局
mLayout = (LinearLayout)view.findViewById(R.id.layout);
//模拟数据
List mList = new ArrayList();
for(int i = 0; i ; i++){
mList.add("条目--"+i);
}
//创建适配器
ArrayAdapter mAdapter = new ArrayAdapter(this,
android.R.layout.simple_list_item_1,
android.R.id.text1,
mList);
//设置头部视图
mLv.addHeaderView(view);
//设置适配器
mLv.setAdapter(mAdapter);
/*//圆点布局
mLayout = (LinearLayout)findViewById(R.id.layout);*/
}
/
*** 圆点的监听事件
*/
@Override
public void onClick(View arg0) {
//获取该圆点的位置
int index = Integer.parseInt(arg0.getTag().toString());
//将viewpager的显示成下标为index的
mPager.setCurrentItem(index);
//将原来的设置成默认值
mDotView.setEnabled(true);
//保存新的值
mDotView = (ImageView) arg0;
//将新的值改变成false
mDotView.setEnabled(false);
//更新下标
mIndex = index;
}
}
注意:因为这里用到了json解析,需要用到联网操作,一定要写上上网的权限:
运行效果:
Android-自定义轮播Banner-Viewpaper2实现+贪吃蛇原点指示器+触摸暂停轮播处理
在上一篇文章《MonkeyLei:更全的ViewPager2使用方式总结与Glide加载问题》的基础上,我们继续优化自定义轮播Banner组件,详情已添加至GitHub项目github.com/FanChael/MVV...中的lib_banner模块。 展示的轮播效果已经相当完善,注重了内存优化,并可能还有待进一步完善细节和修复小问题。代码库中包含简洁的实现和注释,便于查阅。 实现的关键点包括:自定义ViewGroup以添加圆角效果,通过测量布局调整内部Viewpaper2的margin,确保内容在圆角内显示。重写dispatchDraw方法完成最终绘制。
设计指示器绘制,考虑位置和颜色变化,结合ViewPager2的OnPageChangeCallback,实现贪吃蛇效果,通过判断位置和滚动距离动态调整。
触控暂停轮播:利用Viewpaper2提供的回调机制,当手指按下时停止轮播,避免直接处理TouchEvent,因为ViewPaper2不支持继承。
虽然目前满足项目需求,但仍有提升空间。你可以根据项目需求进一步扩展为组件库,甚至开源分享。我最近忙于面试准备,所以暂未继续深化这个组件。我的MVVM工程中已经自定义了两个组件,未来会继续完善它们。 以上内容可供参考,期待你的创新与改进。