Android直播间购物袋动画,Android直播送礼物发消息页面(仿印客直播)

本文主要是介绍Android直播间购物袋动画,Android直播送礼物发消息页面(仿印客直播),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

简介

年关将至,给大家拜个早年,祝大家鸡年大吉吧 ~ 哈哈 有污的同学面壁去吧!闲话不多说,今天实现的效果是直播的时候浮层View,不知道直播的sdk是不是都提供UI这方面的东西。先看一张效果图:

3bd95054bb8fcd42979019ad3e5581b4.gif

实现效果图

主要实现效果:

(当然直播的时候视频是主要的,这里不关心视频,重点是View,这里只是随意播放了一个视频。)

首先是顶部横线滑动的ListView显示房间的成员;

ListView 自动滚动到底部;

自定义实现刷礼物的View,过段时间自动消失;

送礼物的时候弹出礼物的DialogFragment+ViewPager+GridView;

点击评论的时候监听弹出键盘的事件;

直播底部显示送心的特效(这里用了第三方的 tyrant:heartlayout);

VedioView播放视频;

实现过程

首先需要清楚视频是出于最底部的,在视频上面浮着View,这里我用了ViewPager来实现右滑”隐藏“View,其实是切换。

很简单布局:

android:layout_width="match_parent"

android:layout_height="match_parent"

android:background="@drawable/wether_bg"

android:orientation="vertical">

android:id="@+id/video"

android:layout_width="match_parent"

android:layout_height="match_parent" />

android:id="@+id/view_pager"

android:background="@null"

android:layout_width="match_parent"

android:layout_height="match_parent">

接下来实现2个Fragment,一个是空白的Fragment(NoFragment ),一个是带有各种特效的Fragment(ChatFragment ),再填充到ViewPager里面去,当然这里还需要去实现视频的播放,很简单的VedioView,这里就不再多说~

//从Raw资源文件读取视频

final String uri = ("android.resource://" + this.getPackageName() + "/raw/vedio");

videoView.setVideoURI(Uri.parse(uri));

videoView.start();

//视频可以循环播放

videoView.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {

@Override

public void onCompletion(MediaPlayer mp) {

videoView.setVideoURI(Uri.parse(uri));

videoView.start();

}

});

接下来看主要的ChatFragment :

package cm.wzh.live.ui;

import android.content.Context;

import android.graphics.Color;

import android.net.Uri;

import android.os.Bundle;

import android.os.Handler;

import android.support.annotation.Nullable;

import android.support.v4.app.Fragment;

import android.view.LayoutInflater;

import android.view.View;

import android.view.ViewGroup;

import android.view.inputmethod.InputMethodManager;

import android.widget.AdapterView;

import android.widget.EditText;

import android.widget.ListView;

import android.widget.VideoView;

import java.util.ArrayList;

import java.util.Random;

import java.util.Timer;

import cm.wzh.live.R;

import cm.wzh.live.adapter.MemberAdapter;

import cm.wzh.live.adapter.MessageAdapter;

import cm.wzh.live.entity.Gift;

import cm.wzh.live.entity.Member;

import cm.wzh.live.entity.Message;

import cm.wzh.live.utils.CharUtils;

import cm.wzh.live.view.FragmentDialog;

import cm.wzh.live.view.FragmentGiftDialog;

import cm.wzh.live.view.GiftItemView;

import cm.wzh.live.view.HorizontialListView;

import cm.wzh.live.view.MyVideoView;

import tyrantgit.widget.HeartLayout;

/**

* author:Administrator on 2016/12/26 09:35

* description:文件说明

* version:版本

*/

public class ChatFragment extends Fragment implements View.OnClickListener, View.OnLayoutChangeListener {

private HorizontialListView listview ;

private ListView messageList ;

private GiftItemView giftView ;

private MemberAdapter mAdapter ;

private MessageAdapter messageAdapter ;

private ArrayList members ;

private ArrayList messages ;

private ArrayList gifts ;

private HeartLayout heartLayout ;

private Random mRandom ;

private Timer mTimer = new Timer();

private View sendView,menuView ,topView;

private EditText sendEditText ;

//屏幕高度

private int screenHeight = 0;

//软件盘弹起后所占高度阀值

private int keyHeight = 0;

private View rootView ;

@Nullable

@Override

public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

super.onCreateView(inflater, container, savedInstanceState);

View view = LayoutInflater.from(getActivity()).inflate(R.layout.fragment_chat,null,false);

initView(view);

initData();

return view;

}

private void initView(View view) {

mRandom = new Random();

listview = (HorizontialListView) view.findViewById(R.id.list);

mAdapter = new MemberAdapter(getActivity());

listview.setAdapter(mAdapter);

listview.setOnItemClickListener(new AdapterView.OnItemClickListener() {

@Override

public void onItemClick(AdapterView> adapterView, View view, int i, long l) {

showDialog(mAdapter.datas.get(i));

}

});

messageList = (ListView) view.findViewById(R.id.list_message);

messageAdapter = new MessageAdapter(getActivity());

messageList.setAdapter(messageAdapter);

giftView = (GiftItemView) view.findViewById(R.id.gift_item_first);

heartLayout = (HeartLayout)view.findViewById(R.id.heart_layout);

handler.postDelayed(runnable, 2000);//每5秒执行一次runnable.

view.findViewById(R.id.send_message).setOnClickListener(this);

view.findViewById(R.id.gift).setOnClickListener(this);

sendView = view.findViewById(R.id.layout_send_message);

menuView = view.findViewById(R.id.layout_bottom_menu);

topView = view.findViewById(R.id.layout_top);

sendEditText = (EditText) view.findViewById(R.id.send_edit);

//获取屏幕高度

screenHeight = getActivity().getWindowManager().getDefaultDisplay().getHeight();

//阀值设置为屏幕高度的1/3

keyHeight = screenHeight/3;

rootView = view.findViewById(R.id.activity_main);

rootView.addOnLayoutChangeListener(this);

}

private void showDialog(Member m) {

FragmentDialog.newInstance(m.name, m.sig, "确定", "取消",-1,false, new FragmentDialog.OnClickBottomListener() {

@Override

public void onPositiveClick() {

}

@Override

public void onNegtiveClick() {

}

}).show(getChildFragmentManager(),"dialog");

}

Handler handler=new Handler();

Runnable runnable=new Runnable() {

@Override

public void run() {

if (messages!=null){

Message m = new Message();

m.img = "http://v1.qzone.cc/avatar/201503/06/18/27/54f981200879b698.jpg%21200x200.jpg" ;

m.name=CharUtils.getRandomString(8) ;

m.level = (int)(Math.random()*100+1) ;

m.message= CharUtils.getRandomString(20);

messages.add(m);

messageAdapter.notifyDataSetChanged();

messageList.setSelection(messageAdapter.getCount()-1);

}

handler.postDelayed(this, 1000);

}

};

Handler heartHandler=new Handler();

Runnable heartRunnable=new Runnable() {

@Override

public void run() {

heartLayout.post(new Runnable() {

@Override

public void run() {

heartLayout.addHeart(randomColor());

}

});

heartHandler.postDelayed(this, 1000);

}

};

@Override

public void onPause() {

super.onPause();

heartHandler.removeCallbacks(heartRunnable);

}

@Override

public void onResume() {

super.onResume();

heartHandler.postDelayed(heartRunnable, 2000);

}

private int randomColor() {

return Color.rgb(mRandom.nextInt(255), mRandom.nextInt(255), mRandom.nextInt(255));

}

/**

* 添加一些数据

*/

private void initData() {

members = new ArrayList<>();

for (int i=0;i<18;i++){

Member m = new Member();

m.img = "http://www.ld12.com/upimg358/allimg/c150808/143Y5Q9254240-11513_lit.png" ;

m.name="Baby" ;

m.sig = "这个家伙很懒,什么都没留下!";

members.add(m);

}

mAdapter.setDatas(members);

messages = new ArrayList<>();

for (int i=0;i<18;i++){

Message m = new Message();

m.img = "http://www.ld12.com/upimg358/allimg/c150808/143Y5Q9254240-11513_lit.png" ;

m.name="Baby" ;

m.level = i ;

m.message="掘金是中国质量最高的技术分享社区,邀请稀土用户作为 Co-Editor 来分享优质的技术干货" ;

messages.add(m);

}

messageAdapter.setDatas(messages);

gifts = new ArrayList<>();

}

@Override

public void onDestroy() {

super.onDestroy();

mTimer.cancel();

handler.removeCallbacks(runnable);

}

@Override

public void onClick(View v) {

int id = v.getId() ;

if (id==R.id.send_message){

sendView.setVisibility(View.VISIBLE);

menuView.setVisibility(View.GONE);

topView.setVisibility(View.GONE);

sendEditText.requestFocus();

InputMethodManager inputManager =

(InputMethodManager)sendEditText.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);

inputManager.showSoftInput(sendEditText, 0);

}else if (id==R.id.gift){

FragmentGiftDialog.newInstance().setOnGridViewClickListener(new FragmentGiftDialog.OnGridViewClickListener() {

@Override

public void click(Gift gift) {

gift.name="文人骚客";

gift.giftName = "送你一个小礼物" ;

if (!gifts.contains(gift)){

gifts.add(gift);

giftView.setGift(gift);

}

giftView.addNum(1);

}

}).show(getChildFragmentManager(),"dialog");

}

}

@Override

public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {

//现在认为只要控件将Activity向上推的高度超过了1/3屏幕高,就认为软键盘弹起

if(oldBottom != 0 && bottom != 0 &&(oldBottom - bottom > keyHeight)){

sendView.setVisibility(View.VISIBLE);

menuView.setVisibility(View.GONE);

topView.setVisibility(View.GONE);

// Toast.makeText(MainActivity.getActivity(), "监听到软键盘弹起...", Toast.LENGTH_SHORT).show();

}else if(oldBottom != 0 && bottom != 0 &&(bottom - oldBottom > keyHeight)){

sendView.setVisibility(View.GONE);

menuView.setVisibility(View.VISIBLE);

topView.setVisibility(View.VISIBLE);

// Toast.makeText(MainActivity.getActivity(), "监听到软件盘关闭...", Toast.LENGTH_SHORT).show();

}

}

}

这里可以看到,已经把刷礼物的View封装成了 GiftItemView 。

package cm.wzh.live.view;

import android.animation.Animator;

import android.animation.AnimatorListenerAdapter;

import android.animation.AnimatorSet;

import android.animation.ObjectAnimator;

import android.content.Context;

import android.os.Handler;

import android.text.TextUtils;

import android.util.AttributeSet;

import android.view.LayoutInflater;

import android.view.View;

import android.view.ViewGroup;

import android.view.animation.LinearInterpolator;

import android.widget.ImageView;

import android.widget.LinearLayout;

import android.widget.TextView;

import com.bumptech.glide.Glide;

import java.util.ArrayList;

import cm.wzh.live.R;

import cm.wzh.live.entity.Gift;

/**

* author:Administrator on 2016/12/27 09:34

* description:文件说明

* version:版本

*/

public class GiftItemView extends LinearLayout {

private ImageView avatar ;

private TextView name ;

private TextView giftName ;

private TextView giftNumTv ;

private ImageView giftIv ;

private Gift gift ;

private int giftNum = 1 ;

private boolean isShow = false ;

public GiftItemView(Context context) {

this(context,null);

}

public GiftItemView(Context context, AttributeSet attrs) {

this(context, attrs,0);

}

public GiftItemView(Context context, AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

init();

}

private void init() {

setOrientation(VERTICAL);

setVisibility(INVISIBLE);

LayoutParams lp = new LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);

setLayoutParams(lp);

View convertView = LayoutInflater.from(getContext()).inflate(R.layout.item_gift_message,null,false);

avatar = (ImageView) convertView.findViewById(R.id.avatar);

giftIv = (ImageView) convertView.findViewById(R.id.gift_type);

name = (TextView) convertView.findViewById(R.id.name);

giftName = (TextView) convertView.findViewById(R.id.gift_name);

giftNumTv = (TextView) convertView.findViewById(R.id.gift_num);

addView(convertView);

}

public void setGift(Gift gift) {

this.gift = gift;

refreshView();

}

/**

* 设置礼物数量放大和复原的View

* @param view

* @param duration

*/

public void scaleView(View view,long duration){

AnimatorSet animatorSet = new AnimatorSet();//组合动画

ObjectAnimator scaleX = ObjectAnimator.ofFloat(view, "scaleX", 2f, 1f);

ObjectAnimator scaleY = ObjectAnimator.ofFloat(view, "scaleY", 2f, 1f);

animatorSet.setDuration(duration);

animatorSet.setInterpolator(new LinearInterpolator());

animatorSet.play(scaleY).with(scaleX);//两个动画同时开始

animatorSet.start();

animatorSet.addListener(new AnimatorListenerAdapter() {

@Override

public void onAnimationEnd(Animator animation) {

super.onAnimationEnd(animation);

if (onAnimatorListener!=null){

onAnimatorListener.onAnimationEnd(gift);

}

}

@Override

public void onAnimationStart(Animator animation) {

super.onAnimationStart(animation);

if (onAnimatorListener!=null){

onAnimatorListener.onAnimationStart(animation);

}

}

});

}

/**

* 刷新view

*/

public void refreshView(){

if (gift==null){

return;

}

giftNum = gift.num ;

if (!TextUtils.isEmpty(gift.img)){

Glide.with(getContext()).load(gift.img).placeholder(R.drawable.default_head).into(avatar);

}else {

avatar.setImageResource(R.drawable.default_head);

}

name.setText(gift.name);

giftName.setText(gift.giftName);

giftNumTv.setText("x"+gift.num);

giftIv.setImageResource(gift.giftType);

scaleView(giftNumTv,200);

}

/**

* 连续点击送礼物的时候数字缩放效果

* @param num

*/

public void addNum(int num){

giftNum += num ;

giftNumTv.setText("x"+giftNum);

scaleView(giftNumTv,200);

handler.removeCallbacks(runnable);

if (!isShow()){

show();

}

handler.postDelayed(runnable, 3000);

}

Handler handler=new Handler();

Runnable runnable=new Runnable() {

@Override

public void run() {

isShow = false ;

giftNum = 0;

setVisibility(INVISIBLE);

}

};

/**

* 显示view,并开启定时器

*/

public void show(){

isShow = true ;

setVisibility(VISIBLE);

handler.postDelayed(runnable, 3000);

}

public boolean isShow() {

return isShow;

}

private OnAnimatorListener onAnimatorListener ;

public void setOnAnimatorListener(OnAnimatorListener onAnimatorListener) {

this.onAnimatorListener = onAnimatorListener;

}

public interface OnAnimatorListener{

public void onAnimationEnd(Gift gift);

public void onAnimationStart(Animator animation);

}

}

主要是Handler+Runnable实现定时任务,当连续送礼物的时候再重新计时,用属性动画ObjectAnimator对送的数字缩放动画。

对于ListView自动滚动是定时任务每来一个Item就加到ListView集合里面,然后用listview.setSelection(int adapter.getCount())定位到最后一个Item。

有兴趣的可以看一下源代码: github下载地址 github上有apk文件可以下载体验下。

如果有什么问题,希望大家指出来,谢谢~

这篇关于Android直播间购物袋动画,Android直播送礼物发消息页面(仿印客直播)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/394474

相关文章

最好用的WPF加载动画功能

《最好用的WPF加载动画功能》当开发应用程序时,提供良好的用户体验(UX)是至关重要的,加载动画作为一种有效的沟通工具,它不仅能告知用户系统正在工作,还能够通过视觉上的吸引力来增强整体用户体验,本文给... 目录前言需求分析高级用法综合案例总结最后前言当开发应用程序时,提供良好的用户体验(UX)是至关重要

基于Python实现PDF动画翻页效果的阅读器

《基于Python实现PDF动画翻页效果的阅读器》在这篇博客中,我们将深入分析一个基于wxPython实现的PDF阅读器程序,该程序支持加载PDF文件并显示页面内容,同时支持页面切换动画效果,文中有详... 目录全部代码代码结构初始化 UI 界面加载 PDF 文件显示 PDF 页面页面切换动画运行效果总结主

使用JavaScript将PDF页面中的标注扁平化的操作指南

《使用JavaScript将PDF页面中的标注扁平化的操作指南》扁平化(flatten)操作可以将标注作为矢量图形包含在PDF页面的内容中,使其不可编辑,DynamsoftDocumentViewer... 目录使用Dynamsoft Document Viewer打开一个PDF文件并启用标注添加功能扁平化

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

SpringBoot如何访问jsp页面

《SpringBoot如何访问jsp页面》本文介绍了如何在SpringBoot项目中进行Web开发,包括创建项目、配置文件、添加依赖、控制层修改、测试效果以及在IDEA中进行配置的详细步骤... 目录SpringBoot如何访问JSP页python面简介实现步骤1. 首先创建的项目一定要是web项目2. 在

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

Qt QWidget实现图片旋转动画

《QtQWidget实现图片旋转动画》这篇文章主要为大家详细介绍了如何使用了Qt和QWidget实现图片旋转动画效果,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 一、效果展示二、源码分享本例程通过QGraphicsView实现svg格式图片旋转。.hpjavascript

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

如何在页面调用utility bar并传递参数至lwc组件

1.在app的utility item中添加lwc组件: 2.调用utility bar api的方式有两种: 方法一,通过lwc调用: import {LightningElement,api ,wire } from 'lwc';import { publish, MessageContext } from 'lightning/messageService';import Ca

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo