React Native视频Video组件

参考

react-native-video

安装步骤

yarn add react-native-video
yarn add @types/react-native-video --dev 加类型声明文件

android/settings.gradle加入

include ':react-native-video'
project(':react-native-video').projectDir = new File(rootProject.projectDir, '../node_modules/react-native-video/android')

android/build.gradle加入

allprojects {
    repositories {
        ...
        jcenter() {
            content {
                includeModule("com.yqritc", "android-scalablevideoview")
            }
        }
    }
}

android/app/build.gradle加入---

dependencies {
    ...
    implementation project(':react-native-video')  //加这个
}

android/app/src/main/java/com/文件名/MainApplication.java加入

import com.brentvatne.react.ReactVideoPackage;  //加这个
...
        @Override
        protected List<ReactPackage> getPackages() {
          @SuppressWarnings("UnnecessaryLocalVariable")
          List<ReactPackage> packages = new PackageList(this).getPackages();
          // Packages that cannot be autolinked yet can be added manually here, for example:
          // packages.add(new MyReactNativePackage());
          packages.add(new ReactVideoPackage());   //加这个
          return packages;
        }

使用

注意: Video组件自带的controls不会跟着页面滚动,所以最好自己封装一个controls。

封装Video组件
import React from 'react';
import {
  ViewStyle,
  StyleProp,
} from 'react-native'
import { View, Text, TouchableOpacity } from '@/components'
import Loading from "@/components/loading";
import RNVideo, { VideoProperties } from 'react-native-video';

interface componentProps extends VideoProperties {
  style?: StyleProp<ViewStyle>
}

interface State {
  videoPaused?: boolean
  hideControl?: boolean
  duration: number  //总时长-秒
  currentTime: number  //当前进度-秒
  isLoading?: boolean  //加载中
}


const date = {
  prefixZero: (num, n) => {
    return (Array (n).join ('0') + num).slice(-n);
  },
  parseSeconds: (second:number) => {
    if (second > 0) {
      let hour = 0
      let minute = 0
      let seconds = 0
      let data:{
        hour?: number 
        minute?: number 
        seconds?: number 
      } = {}
      minute = Math.floor(second / 60)
      if (parseInt(String(minute)) > 60) {
        hour = parseInt(String(minute / 60))
        minute %= 60 //算出有多分钟
      }
      seconds = second%60;
      data.hour = hour;
      data.minute = minute;
      data.seconds = parseInt(String(seconds))
  
      return data
    }
  }
} as const;

export default class Video extends React.PureComponent<componentProps, State>{
  static defaultProps = {
    resizeMode: 'contain',
    controls: false,
    repeat: true,
    paused: false,
    // poster: '',
    // posterResizeMode: ''
  }

  public state:State = {
    currentTime: 0,
    duration: 0
  };

  render() {
    const { videoPaused, hideControl, currentTime, duration, isLoading } = this.state
    const { style, ...restProps } = this.props
    return (
      <TouchableOpacity activeOpacity={1} style={{
        position: 'relative'
      }} onPress={this.toggleHideControl}>
        <RNVideo style={[{
            backgroundColor: 'rgba(0,0,0,0.8)'
          }, style]} 
          {...restProps} 
          ref={this.handleBindRef}
          paused={videoPaused}
          onLoadStart={this.onVideoLoadStart}
          onLoad={this.onLoad}
          onEnd={this.onEnd}
          progressUpdateInterval={1000}
          onProgress={this.onProgress}
        />
        {hideControl? null: 
          <View style={styles.controlBox()}>
            <View style={[styles.controlBtns(), {flex: 1}]}>
              <TouchableOpacity style={styles.playBtn()} onPress={this.togglePause}>
                {videoPaused? 
                  <Text>{'播放'}</Text> :
                  <Text>{'暂停'}</Text>
                }
              </TouchableOpacity>
              <TouchableOpacity style={styles.towardBtn()} onPress={this.headBack}>
                <Text>{'<<'}</Text>
              </TouchableOpacity>
              <TouchableOpacity style={styles.towardBtn()} onPress={this.headForward}>
                <Text>{'>>'}</Text>
              </TouchableOpacity>
            </View>
            {(!isLoading && duration) ?
              <View style={{paddingHorizontal: 10}}>
                <Text style={{}}>{this.parseSeconds(currentTime)}/{this.parseSeconds(duration)}</Text>
              </View>: null
            }
          </View>
        }
        {isLoading? 
          <View style={styles.loadingBox()}>
            <Loading visible={true}></Loading>
          </View> : null
        }
      </TouchableOpacity>
    );
  };

  private refVideo:any = null;
  private handleBindRef = (e:any) => {
    this.refVideo = e
  }

  private parseSeconds = (second:number) => {
    const vals = date.parseSeconds(second)
    if(vals) {
      let hour = vals.hour? date.prefixZero(vals.hour, 2): '';
      let minute = date.prefixZero(vals.minute, 2);
      let seconds = date.prefixZero(vals.seconds, 2);
      return hour? (hour+':'+minute+':'+seconds): (minute+':'+seconds)
    }else {
      return ''
    }
  }

  private togglePause = () => {
    const { videoPaused } = this.state
    if(videoPaused) {
      this.autoHideControls()
    }
    this.setState({videoPaused: !videoPaused})
  }
  private toggleHideControl = () => {
    const { hideControl } = this.state
    if(hideControl) {
      this.autoHideControls()
    }
    this.setState({hideControl: !hideControl})
  }
  private autoHideTimeout:any = null;
  private autoHideControls = () => {
    if(this.autoHideTimeout) {
      clearTimeout(this.autoHideTimeout)
    }
    this.autoHideTimeout = setTimeout(() => {
      if(!this.state.videoPaused) {
        this.setState({hideControl: true})
      }
    }, 5000)
  }

  onVideoLoadStart = () => {
    this.setState({ isLoading: true })
  }
  private onLoad = (data:any) => {
    this.setState({ duration: data.duration, isLoading: false, });
    this.autoHideControls()
  }
  private onEnd = () => {
    // TODO:  //BUG: repeat=false的情况下,onEnd后重新播放,不执行onProgress;
    this.setState({
      videoPaused: true,
      currentTime: 0,
      duration: 0,
    })
  }
  private onProgress = (data: any) => {
    this.setState({ currentTime: data.currentTime })
  }

  private seekTo = (second:number) => {
    this.refVideo?.seek(second)
  }
  private stepNumber: number = 5;  //一次前进后退的秒数
  private headForward = () => {
    const { currentTime, duration } = this.state
    let target = currentTime+this.stepNumber;
    target = target > duration? duration: target;
    this.seekTo(target);
    this.autoHideControls()
  }
  private headBack = () => {
    const { currentTime, duration } = this.state
    let target = currentTime-this.stepNumber;
    target = target > 0? target: 0;
    this.seekTo(target);
    this.autoHideControls()
  }

}

调用

import Video from '@/components/Video';
...
        render() {
            ...
                <Video style={{
                  height: 500, width: 300
                }}
                repeat={true}
                source={{uri: 'https://test.mp4'}}
              />
            ...
        }

遗留问题

repeat=false的情况下,onEnd后重新播放,不执行onProgress

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容