微信小程序入门

目录
    1. 新建项目
        1.1 项目结构说明
        1.2 程序运行程序流程
     2. 基础组件
     3. 常用API

微信公众平台
小程序开发工具下载

申请账号
    邮箱(账号不能是 微信开放平台 或 微信公众平台)
    密码
查看AppID 和 AppSercet 
    设置|开发设置

打开工具 | 填写AppID创建项目 
    | 点击编译(在左侧模拟器看效果)
    | 点击预览(生成二维码 用微信扫描)
1. 新建项目
新建项目

1.1 项目结构说明

.wxml  : 页面显示文件
.js    : 脚本文件
.wxss  : 样式文件
.json  : 配置文件

json

    app.json 做全局的配置
    页面.json 做单独页面的配置
app.json
    全局页面配置(小程序的所有页面路径、界面表现、网络超时时间、底部 tab)

{
  "pages":[            所有页面路径
    "pages/index/index",    第一个即首页
    "pages/logs/logs"
  ],
  "window":{        所有页面的顶部窗口 文本样式、背景色、标题、标题样式
    "backgroundTextStyle":"light",
    "navigationBarBackgroundColor": "#fff",
    "navigationBarTitleText": "WeChat",
    "navigationBarTextStyle":"black"
  }
}
project.config.json
    开发工具个人喜好配置

{
    "description": "项目配置文件。",
    "setting": {
        "urlCheck": true,
        "es6": true,
        "postcss": true,
        "minified": true,
        "newFeature": true
    },
    "compileType": "miniprogram",
    "libVersion": "1.9.91",
    "appid": "wx8ed79e757ebebea1",
    "projectname": "Test",
    "isGameTourist": false,
    "condition": {
        "search": {
            "current": -1,
            "list": []
        },
        "conversation": {
            "current": -1,
            "list": []
        },
        "game": {
            "currentL": -1,
            "list": []
        },
        "miniprogram": {
            "current": -1,
            "list": []
        }
    }
}
page.json
    单个页面配置


{
  "navigationBarTitleText": "hello"
}

WXML

    类似于HTML的角色(显示控件)
wxml:(控件)
<text>{{ssbb}}</text>
js:(内容)
const app = getApp()
Page({
  data: {
    ssbb: 'Hello World',
  }
})
页面.wxml

<!--页面.wxml-->
<view class="container">
  <view class="userinfo">
    <button wx:if="{{!hasUserInfo && canIUse}}" open-type="getUserInfo" bindgetuserinfo="getUserInfo"> 获取头像昵称 </button>
    <block wx:else>
      <image bindtap="bindViewTap" class="userinfo-avatar" src="{{userInfo.avatarUrl}}" background-size="cover"></image>
      <text class="userinfo-nickname">{{userInfo.nickName}}</text>
    </block>
  </view>
  <view class="usermotto">
    <text class="user-motto">{{motto}}</text>
  </view>
</view>

WXSS

    类似于css的角色(修饰控件)    
app.wxss 
    全局样式


/**app.wxss**/
.container {
  height: 100%;
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: space-between;
  padding: 200rpx 0;
  box-sizing: border-box;
} 
页面.wxss
    单独页面样式


/**页面.wxss**/
.userinfo {
  display: flex;
  flex-direction: column;
  align-items: center;
}

.userinfo-avatar {
  width: 128rpx;
  height: 128rpx;
  margin: 20rpx;
  border-radius: 50%;
}

.userinfo-nickname {
  color: #aaa;
}

.usermotto {
  margin-top: 200px;
}

JS

    脚本文件(处理交互)
wxml:(控件)
<text>{{ssbb}}</text>
<button bindtap="clickMe">点击我</button>

js:(内容)
const app = getApp()
Page({
  data: {
    ssbb: 'Hello World',
  },
  clickMe: function() {
    this.setData({ ssbb: "Hello World !!!" })
  }
})
 app.js
    处理全局页面

//app.js
App({
  onLaunch: function () {
    // 展示本地存储能力
    var logs = wx.getStorageSync('logs') || []
    logs.unshift(Date.now())
    wx.setStorageSync('logs', logs)

    // 登录
    wx.login({
      success: res => {
        // 发送 res.code 到后台换取 openId, sessionKey, unionId
      }
    })
    // 获取用户信息
    wx.getSetting({
      success: res => {
        if (res.authSetting['scope.userInfo']) {
          // 已经授权,可以直接调用 getUserInfo 获取头像昵称,不会弹框
          wx.getUserInfo({
            success: res => {
              // 可以将 res 发送给后台解码出 unionId
              this.globalData.userInfo = res.userInfo

              // 由于 getUserInfo 是网络请求,可能会在 Page.onLoad 之后才返回
              // 所以此处加入 callback 以防止这种情况
              if (this.userInfoReadyCallback) {
                this.userInfoReadyCallback(res)
              }
            }
          })
        }
      }
    })
  },
  globalData: {
    userInfo: null
  }
})
页面.js
    处理单独页面
    
// 页面.js
// 获取应用实例
const app = getApp()

Page({
  data: {
    motto: 'Hello World',
    userInfo: {},
    hasUserInfo: false,
    canIUse: wx.canIUse('button.open-type.getUserInfo')
  },
  //事件处理函数
  bindViewTap: function() {
    wx.navigateTo({
      url: '../logs/logs'
    })
  },
  onLoad: function () {
    if (app.globalData.userInfo) {
      this.setData({
        userInfo: app.globalData.userInfo,
        hasUserInfo: true
      })
    } else if (this.data.canIUse){
      // 由于 getUserInfo 是网络请求,可能会在 Page.onLoad 之后才返回
      // 所以此处加入 callback 以防止这种情况
      app.userInfoReadyCallback = res => {
        this.setData({
          userInfo: res.userInfo,
          hasUserInfo: true
        })
      }
    } else {
      // 在没有 open-type=getUserInfo 版本的兼容处理
      wx.getUserInfo({
        success: res => {
          app.globalData.userInfo = res.userInfo
          this.setData({
            userInfo: res.userInfo,
            hasUserInfo: true
          })
        }
      })
    }
  },
  getUserInfo: function(e) {
    console.log(e)
    app.globalData.userInfo = e.detail.userInfo
    this.setData({
      userInfo: e.detail.userInfo,
      hasUserInfo: true
    })
  }
})

1.2 程序运行程序流程

App({
  // 程序启动后调用
  onLaunch: function () { 
  }
})

Page({
  data: { // 初始数据
  },
  // 页面加载完毕后调用(显示完初始数据后)
  onLoad: function () {
  }
})
2. 基础组件
    一个组件通常包括开始标签和结束标签,属性用来修饰这个组件,内容在两个标签之内。
    所有组件与属性都是小写,以连字符-连接


所有组件都有的属性
    | id               | String       | 组件的唯一标识(唯一)
    | class            | String       | 组件的样式类 (对应的 WXSS文件 中定义的样式类)
    | style            | String       | 组件的内联样式 
    | hidden           | Boolean      | 组件是否显示(默认显示)
    | data-*           | Any          | 自定义属性(组件上触发的事件时,会发送给事件处理函数)
    | bind* / catch*   | EventHandler | 组件的事件 


属性类型
    | Boolean       | 布尔值  有该属性则永为true
    | Number        | 数字    1, 2.5 
    | String        | 字符串  "string"
    | Array         | 数组     [ 1, "string" ]
    | Object        | 对象     { key: value }
    | EventHandler  | 事件处理函数名 
    | Any           | 任意属性 

2.1 视图容器

view视图

<view class="hello">
</view>

属性
    | hover-class             | String 默认:none (按下去时的样式class)
    | hover-stop-propagation  | Boolean 默认:false (是否阻止本节点的祖先节点出现点击状态) 
    | hover-start-time        | Number 默认:50 (按住后多久出现点击状态,单位ms)
    | hover-stay-time         | Number 默认:400 (手指松开后点击状态保留时间,单位ms)

scroll-view滚动视图

<scroll-view scroll-y style="height: 200px;" bindscrolltoupper="upper" bindscrolltolower="lower" bindscroll="scroll" scroll-into-view="{{toView}}" scroll-top="{{scrollTop}}">
</scroll-view>

属性
scroll-x            Boolean 默认:false    是否允许横向滚动
scroll-y            Boolean 默认:false    是否允许纵向滚动
upper-threshold     Number  默认:50   距顶部/左边多远时(单位px),触发 scrolltoupper 事件
lower-threshold     Number  默认:50   距底部/右边多远时(单位px),触发 scrolltolower 事件
scroll-top          Number      设置竖向滚动条位置
scroll-left         Number      设置横向滚动条位置
scroll-into-view        String      滚动到某元素,值为某子元素id(id不能以数字开头)。
scroll-with-animation   Boolean 默认:false    在设置滚动条位置时使用动画过渡
enable-back-to-top      Boolean 默认:false    iOS点击顶部状态栏、安卓双击标题栏时,滚动条返回顶部(只支持竖向)
bindscrolltoupper       EventHandle     滚动到顶部/左边,会触发 scrolltoupper 事件
bindscrolltolower       EventHandle     滚动到底部/右边,会触发 scrolltolower 事件
bindscroll              EventHandle     滚动时触发,event.detail = {scrollLeft, scrollTop, scrollHeight, scrollWidth, deltaX, deltaY}

注意:
    1. 不能在 scroll-view 中使用 textarea、map、canvas、video 组件
    2. scroll-into-view 的优先级高于 scroll-top
    3. 在滚动 scroll-view 时会阻止页面回弹,所以在 scroll-view 中滚动,是无法触发 onPullDownRefresh
    4. 若要使用下拉刷新,请使用页面的滚动,而不是 scroll-view ,这样也能通过点击顶部状态栏回到页面顶部
var order = ['red', 'yellow', 'blue', 'green', 'red']
Page({
  data: {
    toView: 'red',
    scrollTop: 100
  },
  upper: function(e) {
    console.log(e)
  },
  lower: function(e) {
    console.log(e)
  },
  scroll: function(e) {
    console.log(e)
  },
  tap: function(e) {
    for (var i = 0; i < order.length; ++i) {
      if (order[i] === this.data.toView) {
        this.setData({
          toView: order[i + 1]
        })
        break
      }
    }
  },
  tapMove: function(e) {
    this.setData({
      scrollTop: this.data.scrollTop + 10
    })
  }
})

canvas画布

<canvas style="width: 400px; height: 500px;" canvas-id="secondCanvas" binderror="canvasIdErrorCallback">
</canvas>

canvas.js
Page({
  canvasIdErrorCallback: function (e) {
    console.error(e.detail.errMsg)
  },
  onReady: function (e) {
    // 使用 wx.createContext 获取绘图上下文 context
    var context = wx.createCanvasContext('firstCanvas')

    context.setStrokeStyle("#00ff00")
    context.setLineWidth(5)
    context.rect(0, 0, 200, 200)
    context.stroke()
    context.setStrokeStyle("#ff0000")
    context.setLineWidth(2)
    context.moveTo(160, 100)
    context.arc(100, 100, 60, 0, 2 * Math.PI, true)
    context.moveTo(140, 100)
    context.arc(100, 100, 40, 0, Math.PI, false)
    context.moveTo(85, 80)
    context.arc(80, 80, 5, 0, 2 * Math.PI, true)
    context.moveTo(125, 80)
    context.arc(120, 80, 5, 0, 2 * Math.PI, true)
    context.stroke()
    context.draw()
  }
})

注意:
    1.canvas 标签默认宽度300px、高度225px。
    2.同一页面中的 canvas-id 不可重复,否则不可见。
    3.canvas 组件是由客户端创建的原生组件,它的层级是最高的,不能通过 z-index 控制层级
    4.不能在 scroll-view、swiper、picker-view、movable-view 中使用 canvas 组件
    5.css 动画对 canvas 组件无效
    6.避免设置过大的宽高,在安卓下会有crash的问题


属性
canvas-id       String          唯一标识符
disable-scroll  Boolean         默认:false当在 canvas 中移动时且有绑定手势事件时,禁止屏幕滚动以及下拉刷新
bindtouchstart  EventHandle     手指触摸开始时调用
bindtouchmove   EventHandle     手指触摸后移动后调用
bindtouchend    EventHandle     手指触摸动作结束后调用
bindtouchcancel EventHandle     手指触摸动作被打断时调用(如来电提醒,弹窗)
bindlongtap     EventHandle     手指长按 500ms 之后调用(触发了长按事件后进行移动不会触发屏幕的滚动)
binderror       EventHandle     当发生错误时触发 error 事件后调用,detail = {errMsg: 'something wrong'}

map 地图

处理点击事件、经度、纬度
<map bindmarkertap="markertap" longitude="经度" latitude="纬度"></map>


属性
    | longitude         | Number  中心点经度 
    | latitude          | Number  中心点纬度 
    | scale             | Number  默认:16 缩放级别(5-18)
    | markers           | Array   标记点 
    | polyline          | Array   路线 
    | circles           | Array   圆 
    | controls          | Array   控件 
    | include-points    | Array   缩放视野以包含所有给定的坐标点
    | show-location     | Boolean 显示带有方向的当前定位点 
    | bindmarkertap     | EventHandle 点击标记点时调用
    | bindcallouttap    | EventHandle 点击标记点对应的气泡时调用
    | bindcontroltap    | EventHandle 点击控件时调用
    | bindregionchange  | EventHandle 视野发生变化时调用
    | bindtap           | EventHandle 点击地图时调用
    | bindupdated       | EventHandle 在地图渲染更新完成时调用



markers 属性
    | id        | 标记点id  Numbermarker 点击事件回调会返回此id (非必填,建议填)
    | latitude  | 纬度  Number 浮点数(-90 ~ 90)(必填)
    | longitude | 经度 Number 浮点数(-180 ~ 180)(必填)
    | title     | 标注点名 String(非必填)
    | iconPath  | 显示的图标 String 支持相对路径、根目录(必填)
    | rotate    | 旋转角度 Number 顺时针旋转的角度,默认为 0( 0 ~ 360)(非必填)
    | alpha     | 标注的透明度 Number 默认1,无透明
    | width     | 标注图标宽度 Number  默认为图片实际宽度(非必填)
    | height    | 标注图标高度 Number 默认为图片实际高度(非必填)
    | callout   | 自定义标记点上方的气泡窗口  Object(非必填)
    | label     | 为标记点旁边增加标签 Object (非必填)
    | anchor    | 经纬度在标注图标的锚点,默认底边中点 Object {x, y},x表示横向(0-1),y表示竖向(0-1)(非必填)


气泡 callout 属性
    | content     | 文本  String
    | color       | 文本颜色  String 
    | fontSize    | 文字大小  Number 
    | borderRadius| callout边框圆角 Number 
    | bgColor     | 背景色  String 
    | padding     | 文本边缘留白 Number 
    | display     | 'BYCLICK':点击显示; 'ALWAYS':常显  String 
    | textAlign   | 文本对齐方式 String (有效值: left, right, center)


marker 上的气泡 label
    | content     | 文本 | String 
    | color       | 文本颜色 | String 
    | fontSize    | 文字大小 | Number 
    | x           | label的坐标,原点是 marker 对应的经纬度 | Number
    | y           | label的坐标,原点是 marker 对应的经纬度 | Number 
    | borderWidth | 边框宽度 | Number
    | borderColor | 边框颜色 | String 
    | borderRadius| 边框圆角 | Number 
    | bgColor     | 背景色 | String 
    | padding     | 文本边缘留白 | Number
    | textAlign   | 文本对齐方式。(有效值: left, right, center)String 



polyline坐标点 属性
    | points      | 经纬度数组 | Array (必填)  [{latitude: 0, longitude: 0}] 
    | color       | 线的颜色 | String (非必填) 8位十六进制表示,后两位表示alpha值,如:#000000AA 
    | width       | 线的宽度 | Number(非必填)
    | dottedLine  | 是否虚线 | Boolean(非必填) 默认:false 
    | arrowLine   | 带箭头的线 | Boolean(非必填)| 默认:false,开发者工具暂不支持该属性
    | arrowIconPath | 更换箭头图标 | String(非必填) | 在arrowLine为true时生效
    | borderColor | 线的边框颜色 | String(非必填)
    | borderWidth | 线的厚度 | Number(非必填) 


circles 圆覆盖物 属性
    latitude    纬度  Number(必填)  浮点数,范围 -90 ~ 90
    longitude   经度  Number(必填)  浮点数,范围 -180 ~ 180
    color       边框色 String(非必填) 8位十六进制表示,后两位表示alpha值,如:#000000AA
    fillColor   填充色 String(非必填) 8位十六进制表示,后两位表示alpha值,如:#000000AA
    radius      半径  Number(必填)  
    strokeWidth 边框宽 Number(非必填) 


controls控件(不随着地图移动)
    id          控件id    Number(非必填)在控件点击事件回调会返回此id
    position    控件在地图的位置 Object(必填) 控件相对地图位置
    iconPath    显示的图标 String(必填)支持相对路径、绝对路径
    clickable   是否可点击 Boolean(非必填)默认不可点击


position 属性
    left    距离地图的左边界多远  Number(非必填)默认为0
    top 距离地图的上边界多远  Number(非必填)默认为0
    width   控件宽度    Number(非必填)默认为图片宽度
    height  控件高度    Number(非必填)默认为图片高度


<map id="map" longitude="113.324520" latitude="23.099994" scale="14" controls="{{controls}}" bindcontroltap="controltap" markers="{{markers}}" bindmarkertap="markertap" polyline="{{polyline}}" bindregionchange="regionchange" show-location style="width: 100%; height: 300px;"></map>

Page({
  data: {
    markers: [{
      iconPath: "/resources/others.png",
      id: 0,
      latitude: 23.099994,
      longitude: 113.324520,
      width: 50,
      height: 50
    }],
    polyline: [{
      points: [{
        longitude: 113.3245211,
        latitude: 23.10229
      }, {
        longitude: 113.324520,
        latitude: 23.21229
      }],
      color:"#FF0000DD",
      width: 2,
      dottedLine: true
    }],
    controls: [{
      id: 1,
      iconPath: '/resources/location.png',
      position: {
        left: 0,
        top: 300 - 50,
        width: 50,
        height: 50
      },
      clickable: true
    }]
  },
  regionchange(e) {
    console.log(e.type)
  },
  markertap(e) {
    console.log(e.markerId)
  },
  controltap(e) {
    console.log(e.controlId)
  }
})

audio音频


属性
    id          String  唯一标识符
    src         String  音频地址
    loop        Boolean 是否循环播放(默认:false)
    controls    Boolean 是否显示默认控件(默认:false)
    poster      String  默认控件上的音频封面的图片资源地址,如果 controls 属性值为 false 则设置 poster 无效
    name        String  默认控件上的音频名字,如果 controls 属性值为 false 则设置 name 无效
    author      String  默认控件上的作者名字,如果 controls 属性值为 false 则设置 author 无效
    binderror   EventHandle 当发生错误时触发 error 事件,detail = {errMsg: MediaError.code}
    bindplay    EventHandle 当开始/继续播放时触发play事件
    bindpause   EventHandle 当暂停播放时触发 pause 事件
    bindtimeupdate  EventHandle 当播放进度改变时触发 timeupdate 事件,detail = {currentTime, duration}
    bindended   EventHandle 当播放到末尾时触发 ended 事件

MediaError.code
  1 获取资源被用户禁止
  2 网络错误
  3 解码错误
  4 不合适资源


<audio poster="{{poster}}" name="{{name}}" author="{{author}}" src="{{src}}" id="myAudio" controls loop></audio>

<button type="primary" bindtap="audioPlay">播放</button>
<button type="primary" bindtap="audioPause">暂停</button>
<button type="primary" bindtap="audio14">设置当前播放时间为14秒</button>
<button type="primary" bindtap="audioStart">回到开头</button>

Page({
  onReady: function (e) {
    // 使用 wx.createAudioContext 获取 audio 上下文 context
    this.audioCtx = wx.createAudioContext('myAudio')
  },
  data: {
    poster: 'http://y.gtimg.cn/music/photo_new/T002R300x300M000003rsKF44GyaSk.jpg?max_age=2592000',
    name: '此时此刻',
    author: '许巍',
    src: 'http://ws.stream.qqmusic.qq.com/M500001VfvsJ21xFqb.mp3?guid=ffffffff82def4af4b12b3cd9337d5e7&uin=346897220&vkey=6292F51E1E384E06DCBDC9AB7C49FD713D632D313AC4858BACB8DDD29067D3C601481D36E62053BF8DFEAF74C0A5CCFADD6471160CAF3E6A&fromtag=46',
  },
  audioPlay: function () {
    this.audioCtx.play()
  },
  audioPause: function () {
    this.audioCtx.pause()
  },
  audio14: function () {
    this.audioCtx.seek(14)
  },
  audioStart: function () {
    this.audioCtx.seek(0)
  }
})

image图片


属性
    | src       | String  图片地址
    | mode      | String 'scaleToFill' | 图片裁剪、缩放的模式
    | lazy-load | Boolean (默认:false) | 图片懒加载。只针对page与scroll-view下的image有效 | [1.5.0]
    | binderror | HandleEvent 当错误发生时,发布到 AppService 的事件名,事件对象event.detail = {errMsg: 'something wrong'} 
    | bindload  | HandleEvent 当图片载入完毕时,发布到 AppService 的事件名,事件对象event.detail = {height:'图片高度px', width:'图片宽度px'} |

注意:
    1.image组件默认宽度300px、高度225px


mode 有 13 种模式,其中 4 种是缩放模式,9 种是裁剪模式。
    缩放  scaleToFill 不按比例缩放填满
    缩放  aspectFit   按比例缩放(不裁剪)
    缩放  aspectFill  按比例缩放(可能裁剪)
    缩放  widthFix    宽度不变,高度自动变化,保持原图宽高比不变
    裁剪  top     不缩放图片 距上
    裁剪  bottom  不缩放图片 距下
    裁剪  center  不缩放图片 距中
    裁剪  left    不缩放图片 距左
    裁剪  right   不缩放图片 距右
    裁剪  top left    不缩放图片 距左上
    裁剪  top right   不缩放图片 距右上
    裁剪  bottom left 不缩放图片 距左下
    裁剪  bottom right    不缩放图片 距右下


<view class="page">
  <view class="page__hd">
    <text class="page__title">image</text>
    <text class="page__desc">图片</text>
  </view>
  <view class="page__bd">
    <view class="section section_gap" wx:for="{{array}}" wx:for-item="item">
      <view class="section__title">{{item.text}}</view>
      <view class="section__ctn">
        <image style="width: 200px; height: 200px; background-color: #eeeeee;" mode="{{item.mode}}" src="{{src}}"></image>
      </view>
    </view>
  </view>
</view>

Page({
  data: {
    array: [{
      mode: 'scaleToFill',
      text: 'scaleToFill:不保持纵横比缩放图片,使图片完全适应'
    }, { 
      mode: 'aspectFit',
      text: 'aspectFit:保持纵横比缩放图片,使图片的长边能完全显示出来'
    }, { 
      mode: 'aspectFill',
      text: 'aspectFill:保持纵横比缩放图片,只保证图片的短边能完全显示出来'
    }, { 
      mode: 'top',
      text: 'top:不缩放图片,只显示图片的顶部区域' 
    }, {      
      mode: 'bottom',
      text: 'bottom:不缩放图片,只显示图片的底部区域'
    }, { 
      mode: 'center',
      text: 'center:不缩放图片,只显示图片的中间区域'
    }, { 
      mode: 'left',
      text: 'left:不缩放图片,只显示图片的左边区域'
    }, { 
      mode: 'right',
      text: 'right:不缩放图片,只显示图片的右边边区域'
    }, { 
      mode: 'top left',
      text: 'top left:不缩放图片,只显示图片的左上边区域' 
    }, { 
      mode: 'top right',
      text: 'top right:不缩放图片,只显示图片的右上边区域'
    }, { 
      mode: 'bottom left',
      text: 'bottom left:不缩放图片,只显示图片的左下边区域'
    }, { 
      mode: 'bottom right',
      text: 'bottom right:不缩放图片,只显示图片的右下边区域'
    }],
    src: '../../resources/cat.jpg'
  },
  imageError: function(e) {
    console.log('image3发生error事件,携带值为', e.detail.errMsg)
  }
})

视频


属性
    | src             | String 视频地址 
    | initial-time    | Number 初始播放位置 [1.6.0]
    | duration        | Number 指定视频时长 [1.1.0]
    | controls        | Boolean (默认:true) 是否显示默认播放控件(播放/暂停按钮、播放进度、时间) 
    | danmu-list      | Object Array 弹幕列表 
    | danmu-btn       | Boolean(默认: false )是否显示弹幕按钮,只在初始化时有效,不能动态变更
    | enable-danmu    | Boolean(默认: false )是否展示弹幕,只在初始化时有效,不能动态变更
    | autoplay        | Boolean(默认: false )是否自动播放
    | loop            | Boolean(默认: false )是否循环播放[1.4.0]
    | muted           | Boolean(默认: false )是否静音播放[1.4.0]
    | page-gesture    | Boolean(默认: false )在非全屏模式下,是否开启亮度与音量调节手势 [1.6.0]
    | direction       | Number 设置全屏时视频的方向,不指定则根据宽高比自动判断。有效值为 0(正常竖向), 90(屏幕逆时针90度), -90(屏幕顺时针90度) [1.7.0]
    | show-progress   | Boolean(默认: true)若不设置,宽度大于240时才会显示 [1.9.0]
    | show-fullscreen-btn     | Boolean(默认: true)是否显示全屏按钮 [1.9.0]
    | show-play-btn           | Boolean(默认: true)是否显示视频底部控制栏的播放按钮 [1.9.0]
    | show-center-play-btn    | Boolean(默认: true)是否显示视频中间的播放按钮 [1.9.0]
    | enable-progress-gesture | Boolean(默认: true)是否开启控制进度的手势 | [1.9.0]
    | objectFit        | String | contain | 当视频大小与 video 容器大小不一致时,视频的表现形式。contain:包含,fill:填充,cover:覆盖 
    | poster           | String |  | 视频封面的图片网络资源地址,如果 controls 属性值为 false 则设置 poster 无效
    | bindplay         | EventHandle |  | 当开始/继续播放时触发play事件 
    | bindpause        | EventHandle |  | 当暂停播放时触发 pause 事件
    | bindended        | EventHandle |  | 当播放到末尾时触发 ended 事件 
    | bindtimeupdate   | EventHandle |  | 播放进度变化时触发,event.detail = {currentTime, duration} 。触发频率 250ms 一次 
    | bindfullscreenchange | EventHandle |  | 当视频进入和退出全屏是触发,event.detail = {fullScreen, direction},direction取为 vertical 或 horizontal | [1.4.0]
    | bindwaiting      | EventHandle |  | 视频出现缓冲时触发[1.7.0]
    | binderror        | EventHandle |  | 视频播放出错时触发



<view class="section tc">
  <video src="{{src}}"   controls ></video>
  <view class="btn-area">
    <button bindtap="bindButtonTap">获取视频</button>
  </view>
</view>

<view class="section tc">
  <video id="myVideo" src="http://wxsnsdy.tc.qq.com/105/20210/snsdyvideodownload?filekey=30280201010421301f0201690402534804102ca905ce620b1241b726bc41dcff44e00204012882540400&bizid=1023&hy=SH&fileparam=302c020101042530230204136ffd93020457e3c4ff02024ef202031e8d7f02030f42400204045a320a0201000400" danmu-list="{{danmuList}}" enable-danmu danmu-btn controls></video>
  <view class="btn-area">
    <button bindtap="bindButtonTap">获取视频</button>
    <input bindblur="bindInputBlur"/>
    <button bindtap="bindSendDanmu">发送弹幕</button>
  </view>
</view>

function getRandomColor () {
  let rgb = []
  for (let i = 0 ; i < 3; ++i){
    let color = Math.floor(Math.random() * 256).toString(16)
    color = color.length == 1 ? '0' + color : color
    rgb.push(color)
  }
  return '#' + rgb.join('')
}

Page({
  onReady: function (res) {
    this.videoContext = wx.createVideoContext('myVideo')
  },
  inputValue: '',
    data: {
        src: '',
    danmuList: [
      {
        text: '第 1s 出现的弹幕',
        color: '#ff0000',
        time: 1
      },
      {
        text: '第 3s 出现的弹幕',
        color: '#ff00ff',
        time: 3
    }]
    },
  bindInputBlur: function(e) {
    this.inputValue = e.detail.value
  },
  bindButtonTap: function() {
    var that = this
    wx.chooseVideo({
      sourceType: ['album', 'camera'],
      maxDuration: 60,
      camera: ['front','back'],
      success: function(res) {
        that.setData({
          src: res.tempFilePath
        })
      }
    })
  },
  bindSendDanmu: function () {
    this.videoContext.sendDanmu({
      text: this.inputValue,
      color: getRandomColor()
    })
  }
})

注意:
    1.video 组件是由客户端创建的原生组件,它的层级是最高的,不能通过 z-index 控制层级。
    2.请勿在 scroll-view、swiper、picker-view、movable-view 中使用 video 组件。
    3. css 动画对 video 组件无效。

相机

需要[用户授权] scope.camera


属性
    device-position String  back    前置或后置,值为front, back
    flash           String  auto    闪光灯,值为auto, on, off
    bindstop        EventHandle     摄像头在非正常终止时触发,如退出后台等情况
    binderror       EventHandle     用户不允许使用摄像头时触发



注意:
    1.camera 组件是由客户端创建的原生组件,它的层级是最高的,不能通过 z-index 控制层级。可使用 cover-view cover-image覆盖在上面。
    2.同一页面只能插入一个 camera 组件。
    3.请勿在 scroll-view、swiper、picker-view、movable-view 中使用 camera 组件。


<!-- camera.wxml -->
<camera device-position="back" flash="off" binderror="error" style="width: 100%; height: 300px;"></camera>
<button type="primary" bindtap="takePhoto">拍照</button>
<view>预览</view>
<image mode="widthFix" src="{{src}}"></image>

Page({
    takePhoto() {
        const ctx = wx.createCameraContext()
        ctx.takePhoto({
            quality: 'high',
            success: (res) => {
                this.setData({
                    src: res.tempImagePath
                })
            }
        })
    },
    error(e) {
        console.log(e.detail)
    }
})

live-player实时音视频播放

暂只针对如下类目开放,需要先通过类目审核,再在小程序管理后台,“设置”-“接口设置”中自助开通该组件权限
    社交      直播
    教育      在线教育
    医疗      互联网医院,公立医院
    政务民生   所有二级类目
    金融      基金、信托、保险、银行、证券/期货、非金融机    构自营小额贷款、征信业务、消费金融


属性
    src         String  音视频地址(目前仅支持 flv, rtmp)  
    mode        String  live    live(直播),RTC(实时通话)  
    autoplay    Boolean(默认false)是否自动播放  
    muted       Boolean(默认false)是否静音    
    orientation String (默认vertical)画面方向(可选值有 vertical,horizontal)   
    object-fit  String(默认contain)填充模式(可选值有 contain,fillCrop)    
    background-mute     Boolean (默认false)进入后台时是否静音  
    min-cache            Number(默认1)最小缓冲区,单位s   
    max-cache            Number(默认3)最大缓冲区,单位s   
    bindstatechange      EventHandle   播放状态变化事件,detail = {code} 
    bindfullscreenchange EventHandle   全屏变化事件,detail = {direction, fullScreen}  
    bindnetstatus        EventHandle   网络状态通知,detail = {info}


注意:
    1. <live-player /> 默认宽度300px、高度225px,可通过wxss设置宽高。
    2. 开发者工具上暂不支持
    3.live-player 组件是由客户端创建的原生组件,它的层级是最高的,不能通过 z-index 控制层级。可使用 cover-view cover-image覆盖在上面。
    4.请勿在 scroll-view、swiper、picker-view、movable-view 中使用 live-player 组件。
    5.css 动画对 live-player 组件无效。

状态码
2001    已经连接服务器
2002    已经连接服务器,开始拉流
2003    网络接收到首个视频数据包(IDR)
2004    视频播放开始
2005    视频播放进度
2006    视频播放结束
2007    视频播放Loading
2008    解码器启动
2009    视频分辨率改变
-2301   网络断连,且经多次重连抢救无效,更多重试请自行重启播放
-2302   获取加速拉流地址失败
2101    当前视频帧解码失败
2102    当前音频帧解码失败
2103    网络断连, 已启动自动重连
2104    网络来包不稳:可能是下行带宽不足,或由于主播端出流不均匀
2105    当前视频播放出现卡顿
2106    硬解启动失败,采用软解
2107    当前视频帧不连续,可能丢帧
2108    当前流硬解第一个I帧失败,SDK自动切软解
3001    RTMP -DNS解析失败
3002    RTMP服务器连接失败
3003    RTMP服务器握手失败
3005    RTMP 读/写失败


网络状态数据
videoBitrate    当前视频编/码器输出的比特率,单位 kbps
audioBitrate    当前音频编/码器输出的比特率,单位 kbps
videoFPS    当前视频帧率
videoGOP    当前视频 GOP,也就是每两个关键帧(I帧)间隔时长,单位 s
netSpeed    当前的发送/接收速度
netJitter   网络抖动情况,抖动越大,网络越不稳定
videoWidth  视频画面的宽度
videoHeight 视频画面的高度



<live-player src="https://domain/pull_stream" mode="RTC" autoplay bindstatechange="statechange" binderror="error" style="width: 300px; height: 225px;" />
Page({
  statechange(e) {
    console.log('live-player code:', e.detail.code)
  },
  error(e) {
      console.error('live-player error:', e.detail.errMsg)
  }
})

live-pusher实时音频录制

需要[用户授权]scope.camera、scope.record
暂只支持同上


属性
    url                 String  推流地址(目前仅支持 flv, rtmp 格式)    
    mode                String  RTC SD(标清), HD(高清), FHD(超清), RTC(实时通话)  
    autopush            Boolean false   自动推流    
    muted               Boolean false   是否静音    
    enable-camera       Boolean true    开启摄像头   
    auto-focus          Boolean true    自动聚集    
    orientation         String  vertical    vertical,horizontal 
    beauty              Number  0   美颜  
    whiteness           Number  0   美白  
    aspect              String  9:16    宽高比,可选值有 3:4, 9:16  
    min-bitrate         Number  200 最小码率    
    max-bitrate         Number  1000    最大码率    
    waiting-image       String  进入后台时推流的等待画面    
    waiting-image-md5   String  等待画面资源的MD5值 
    background-mute     Boolean(默认false)进入后台时是否静音   
    bindstatechange     EventHandle 状态变化事件,detail = {code}  
    bindnetstatus       EventHandle 网络状态通知,detail = {info}


状态码
1001    已经连接推流服务器
1002    已经与服务器握手完毕,开始推流
1003    打开摄像头成功
1004    录屏启动成功
1005    推流动态调整分辨率
1006    推流动态调整码率
1007    首帧画面采集完成
1008    编码器启动
-1301   打开摄像头失败
-1302   打开麦克风失败
-1303   视频编码失败
-1304   音频编码失败
-1305   不支持的视频分辨率
-1306   不支持的音频采样率
-1307   网络断连,且经多次重连抢救无效,更多重试请自行重启推流
-1308   开始录屏失败,可能是被用户拒绝
-1309   录屏失败,不支持的Android系统版本,需要5.0以上的系统
-1310   录屏被其他应用打断了
-1311   Android Mic打开成功,但是录不到音频数据
-1312   录屏动态切横竖屏失败
1101    网络状况不佳:上行带宽太小,上传数据受阻
1102    网络断连, 已启动自动重连
1103    硬编码启动失败,采用软编码
1104    视频编码失败
1105    新美颜软编码启动失败,采用老的软编码
1106    新美颜软编码启动失败,采用老的软编码
3001    RTMP -DNS解析失败
3002    RTMP服务器连接失败
3003    RTMP服务器握手失败
3004    RTMP服务器主动断开,请检查推流地址的合法性或防盗链有效期
3005    RTMP 读/写失败


网络状态数据
videoBitrate    当前视频编/码器输出的比特率,单位 kbps
audioBitrate    当前音频编/码器输出的比特率,单位 kbps
videoFPS    当前视频帧率
videoGOP    当前视频 GOP,也就是每两个关键帧(I帧)间隔时长,单位 s
netSpeed    当前的发送/接收速度
netJitter   网络抖动情况,抖动越大,网络越不稳定
videoWidth  视频画面的宽度
videoHeight 视频画面的高度


 <live-pusher url="https://domain/push_stream" mode="RTC" autopush bindstatechange="statechange" style="width: 300px; height: 225px;" />
Page({
  statechange(e) {
    console.log('live-pusher code:', e.detail.code)
  }
})


注意:
    1.<live-player /> 默认宽度为100%、无默认高度,请通过wxss设置宽高。
    2.开发者工具上暂不支持。
    3.live-pusher 组件是由客户端创建的原生组件,它的层级是最高的,不能通过 z-index 控制层级。可使用 cover-view cover-image覆盖在上面。
    4.请勿在 scroll-view、swiper、picker-view、movable-view 中使用 live-pusher 组件。
    5.css 动画对 live-pusher 组件无效。
3. 常用API
js中使用

获取用户经纬度
wx.getLocation({
  type: 'wgs84',
  success: (res) => {
    var latitude = res.latitude // 经度
    var longitude = res.longitude // 纬度
  }
})

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

推荐阅读更多精彩内容