微信小程序实现聊天室


Posted in Javascript onAugust 21, 2020

本文实例为大家分享了微信小程序实现聊天室的具体代码,供大家参考,具体内容如下

微信小程序实现聊天室

正文:

<!-- <button bindtap='close'>关闭</button><button bindtap='open'>打开</button> -->
<view wx:if='{{login}}' class='login_zz'></view>
<button wx:if='{{login}}' class='login' bindgetuserinfo='userinfo' open-type="getUserInfo">登录</button>
<view class='page_bg' wx:if='{{block}}' bindtap='hide_bg' />
<view class='btn_bg' wx:if='{{block}}'>
 <view wx:for="{{link_list}}" wx:key='index'>
  <button class="sp_tit" id='{{index}}' bindtap='list_item'>查看详情 {{item}} </button>
 </view>
</view>
<scroll-view class="history" scroll-y="true" scroll-with-animation scroll-top="{{scrollTop}}">
 <block wx:key="{{index}}" wx:for="{{allContentList}}">
  <block wx:if="{{item.is_my}}">
   <view class='my_right new_txt'>
    <view class='time' wx:if='{{item.messageTime&&item.messageTime!=0}}'> {{item.messageTime}} </view>
    <view class='p_r page_r' style='margin-right: 25rpx;' wx:if='{{item.text}}'>
     <view class='new_txt'>
      <view class='new_txt_my'>
       <view class='arrow'>
        <em></em>
        <span></span>
       </view>
       <text decode="true">{{item.text}}</text> </view>
     </view>
     <image class='new_img' src='{{item.avatarUrl}}'></image>
    </view>
    <view class='p_r page_r' style='margin-right: 25rpx;' wx:if='{{item.audio}}' bindtap='my_audio_click' data-id='{{index}}'>
     <view class='new_txt_my_2' style=' width:{{item.length}}px'>
      <image class='my_audio' src='/images/yuyin_icon.png'></image>
     </view>
     <span class='_span'></span>
     <open-data class='new_img' type="userAvatarUrl"></open-data>
    </view>
   </view>
  </block>
  <!-- <view class='you_left' id='id_{{allContentList.length}}'> -->
  <block wx:if="{{item.is_ai&&item.is_ai!=''}}">
   <view class='you_left' style='width:100%;' id='id_{{allContentList.length}}' wx:key="{{index}}">
    <view class='time' wx:if='{{item.messageTime}}'> {{item.messageTime}} </view>
    <view class='p_r' style='margin-left: 20rpx;'>
     <image class='new_img' src='{{item.avatarUrl}}'></image>
     <view class='new_txt'>
      <view class='new_txt_ai'>
       <view class='arrow'>
        <em></em>
        <span></span>
       </view>
       <view class='ai_content'>
        <!-- <text wx:if='{{item.type=="1"}}' decode="true">{{item.text}}</text> -->
        <text decode="true">{{item.text}}</text>
        <!-- <image wx:if='{{item.type=="2"}}' style='width:160rpx;height:160rpx;' src='{{item.src}}'></image> -->
       </view>
       <view class='yes_problem_log' wx:if="{{item.yse_problem&&item.solve_show}}" style=''>感谢您的反馈,我们会再接再厉!</view>
       <view class='yes_problem_log' style='color:#32CF3C' wx:if="{{item.no_problem&&item.solve_show}}" bindtap='phone_click'>拨打人工客服</view>
      </view>
     </view>
    </view>
   </view>
  </block>
 </block>
</scroll-view>
<!-- 遮罩 -->
<view class='zezhao' wx:if='{{cross}}' bindtap='add_icon_click' id='2'></view>
<!-- 输入框 -->
<view class='{{cross?"in_voice_icon":""}}'>
 <view class="sendmessage" style='bottom:{{input_bottom}}px'>
  <input type="text" style='{{focus?"border-bottom: 1px solid #88DD4B;":""}}' adjust-position='{{false}}' cursor-spacing='5' bindinput="bindKeyInput" value='{{inputValue}}' focus='{{focus}}' bindblur='no_focus' bindfocus="focus" confirm-type="done" placeholder="请输入您要咨询的问题"
  />
  <button wx:if='{{if_send&&inputValue!=""}}' bindtap="submitTo" class='user_input_text'>发送</button>
  <image class='add_icon' bindtap='add_icon_click' id='1' wx:if='{{add&&!if_send&&inputValue==""}}' src='/images/jia_img.png'></image>
  <image class='add_icon' bindtap='add_icon_click' id='2' wx:if='{{cross}}' src='/images/audio/cross37.png'></image>
 </view>
 <view wx:if='{{cross}}' class='item' bindtap='phone_click'>
  <image class='img' src='/images/yuyin_icon.png'></image>
  <view class='text'>人工客服</view>
 </view>
</view>
<!-- <view class='zezhao' wx:if='{{add_icon_click}}' bindtap='add_icon_click'></view> -->
<!-- <view class='in_voice_icon'>
 <view class="sendmessage_2">
  <input type="text" bindinput="bindKeyInput" adjust-position='{{false}}' value='{{inputValue}}' focus='{{focus}}' bindfocus="focus" confirm-type="done" placeholder="" />
  <image class='add_icon' bindtap='add_icon_click' src='/images/audio/cross37.png'></image>
 </view>
 <view class='item' bindtap='phone_click'>
  <image class='img' src='/images/yuyin_icon.png'></image>
  <view class='text'>人工客服</view>
 </view>
</view> -->
page {
 background-color: #f2f2f2;
 height: 100%;
 padding: 0 auto;
 margin: 0 auto;
}
 
.login {
 z-index: 999;
 position: fixed;
 top: 300rpx;
 height: 100rpx;
 text-align: center;
 line-height: 100rpx;
 width: 400rpx;
 left: 50%;
 margin-left: -200rpx;
 background: white;
}
 
.Phone {
 top: 500rpx;
 width: 400rpx;
}
 
.login_zz {
 position: fixed;
 left: 0;
 top: 0;
 width: 100%;
 height: 100%;
 z-index: 998;
 background-color: rgba(0, 0, 0, 0.5);
}
 
swiper {
 height: 180rpx;
}
 
swiper swiper-item .slide-image {
 width: 100%;
 height: 180rpx;
}
 
.jia_img {
 height: 80rpx;
 width: 90rpx;
}
 
.time {
 text-align: center;
 padding: 5rpx 20rpx 5rpx 20rpx;
 border-radius: 10rpx;
 display: block;
 height: 38rpx;
 line-height: 38rpx;
 position: relative;
 margin: 0 auto;
 margin-bottom: 20rpx;
 width: 90rpx;
 color: white;
 font-size: 26rpx;
 background-color: #dedede;
}
 
.tab {
 bottom: 120rpx;
}
 
.tab_1 {
 position: fixed;
 bottom: 50rpx;
 width: 200rpx;
 font-size: 26rpx;
 left: 50%;
 margin-left: -45rpx;
 height: 100rpx;
}
 
.tab_2 {
 right: 30rpx;
 position: fixed;
}
 
/* 聊天 */
 
.my_right {
 float: right;
 margin-top: 30rpx;
 position: relative;
}
 
.my_audio {
 height: 60rpx;
 width: 60rpx;
 z-index: 2;
 position: relative;
 top: 10rpx;
 left: 20rpx;
}
 
.you_left {
 margin-top: 30rpx;
 float: left;
 position: relative;
 padding-left: 5rpx;
}
 
.new_img {
 width: 85rpx;
 height: 85rpx;
 overflow: hidden;
}
 
.page_r {
 float: right;
}
 
.new_txt {
 min-width: 380rpx;
 width: 460rpx;
 word-break: break-all;
}
 
.new_txt_my {
 border-radius: 7rpx;
 background: #9fe75a;
 position: relative;
 right: 30rpx;
 padding: 17rpx 30rpx 17rpx 30rpx;
 float: right;
 border: 1px solid #d0d0d0;
}
 
.new_txt_my .arrow {
 position: absolute;
 z-index: 2;
 width: 40rpx;
 right: -38rpx;
}
 
.new_txt_my .arrow em {
 position: absolute;
 border-style: solid;
 border-width: 15rpx;
 border-color: transparent transparent transparent #d0d0d0;
 top: 1rpx;
}
 
.new_txt_my .arrow span {
 position: absolute;
 top: 5rpx;
 border-style: solid;
 border-width: 15rpx;
 border-color: transparent transparent transparent #9fe75a;
}
 
.new_txt_my_2 {
 word-break: break-all;
 border-radius: 7rpx;
 background: #9fe75a;
 min-width: 330rpx;
 max-width: 530rpx;
 padding: 17rpx 30rpx 17rpx 30rpx;
 float: right;
}
 
.new_txt_ai {
 border-radius: 7rpx;
 left: 20rpx;
 background-color: #fff;
 position: relative;
 border: 1px solid #d0d0d0;
 float: left;
}
 
.new_txt_ai .arrow {
 position: relative;
 width: 40rpx;
 left: -30rpx;
}
 
.new_txt_ai .arrow em {
 position: absolute;
 border-style: solid;
 border-width: 15rpx;
 top: 20rpx;
 border-color: transparent #d0d0d0 transparent transparent;
}
 
.new_txt_ai .arrow span {
 position: absolute;
 top: 20rpx;
 border-style: solid;
 border-width: 15rpx;
 border-color: transparent #fff transparent transparent;
 left: 2rpx;
}
 
.ai_content {
 word-break: break-all;
 padding: 17rpx 30rpx 17rpx 30rpx;
}
 
.sanjiao {
 top: 25rpx;
 position: relative;
 width: 0px;
 height: 0px;
 border-width: 15rpx;
 border-style: solid;
}
 
.my {
 border-color: transparent transparent transparent #9fe75a;
}
 
.you {
 border-color: transparent #fff transparent transparent;
}
 
._span {
 border-color: #fff transparent transparent;
 top: -17px;
}
 
.is_ai_btn {
 border-radius: 0 0 7px 7px;
 border-top: 1px solid #d0d0d0;
 background: white;
 position: relative;
 bottom: 0;
 left: 0;
 width: 100%;
 height: 80rpx;
 line-height: 80rpx;
 display: flex;
 flex-direction: row;
 text-align: center;
}
 
.is_ai_btn view {
 width: 50%;
}
 
.is_ai_btn image {
 width: 32rpx;
 position: relative;
 top: 4rpx;
 height: 32rpx;
}
 
.is_ai_btn .two {
 border-left: 1px solid #d0d0d0;
}
 
.yes_problem_log {
 border-top: 1px solid #d0d0d0;
 height: 80rpx;
 text-align: center;
 line-height: 80rpx;
}
 
.voice_icon {
 width: 60rpx;
 height: 60rpx;
 margin: 0 auto;
 padding: 10rpx 10rpx 10rpx 10rpx;
}
 
.add_icon {
 width: 70rpx;
 height: 70rpx;
 margin: 0 auto;
 padding: 20rpx 10rpx 10rpx 15rpx;
}
 
.voice_ing {
 width: 90%;
 height: 75rpx;
 line-height: 85rpx;
 text-align: center;
 border-radius: 15rpx;
 border: 1px solid #d0d0d0;
}
 
.zezhao {
 height: 100%;
 position: absolute;
 top: 0;
 left: 0;
 z-index: 2;
 width: 100%;
 background: rgba(0, 0, 0, 0.5);
}
 
.in_voice_icon {
 z-index: 3;
 left: 0;
 bottom: 0;
 width: 100%;
 position: absolute;
 height: 500rpx;
 background: #f8f8f8;
}
 
.in_voice_icon .item {
 position: relative;
 margin-top: 50rpx;
 margin-left: 50rpx;
 text-align: center;
 width: 140rpx;
}
 
.in_voice_icon .img {
 width: 80rpx;
 height: 80rpx;
 border-radius: 15rpx;
}
 
.in_voice_icon .text {
 font-size: 32rpx;
 margin-top: 20rpx;
}
 
.sendmessage {
 width: 100%;
 z-index: 2;
 display: flex;
 position: fixed;
 bottom: 0px;
 background-color: #f8f8f8;
 flex-direction: row;
 height: 100rpx;
}
 
.sendmessage input {
 width: 78%;
 height: 80rpx;
 line-height: 80rpx;
 font-size: 28rpx;
 margin-top: 10rpx;
 margin-left: 20rpx;
 border-bottom: 1px solid #d0d0d0;
 padding-left: 20rpx;
}
 
.sendmessage button {
 border: 1px solid white;
 width: 18%;
 height: 80rpx;
 background: #0c0;
 color: white;
 line-height: 80rpx;
 margin-top: 10rpx;
 font-size: 28rpx;
}
 
.hei {
 height: 20rpx;
}
 
.history {
 height: 88%;
 display: flex;
 font-size: 14px;
 line-height: 50rpx;
 position: relative;
 top: 20rpx;
}
 
.icno_kf {
 position: fixed;
 bottom: 160rpx;
 margin: 0 auto;
 text-align: center;
 left: 50%;
 margin-left: -40rpx;
 width: 100rpx;
 height: 100rpx;
 border-radius: 50%;
}
// pages/index/to_news/to_news.js 
var app = getApp();
var util = require("../../utils/util.js");
var socketOpen = false;
var uuid = '',
 time_ = "1";
var recorder = wx.getRecorderManager();
const innerAudioContext = wx.createInnerAudioContext() //获取播放对象var
frameBuffer_Data, session, SocketTask, string_base64;
Page({
 data: {
  login: false,
  listCustmerServiceBanner: [],
  indicatorDots: false,
  autoplay: false,
  interval: 5000,
  duration: 1000,
  user_input_text: '', //用户输入文字  
  inputValue: '',
  time: '',
  returnValue: '',
  if_send: false,
  add: true,
  cross: false, // is_my: true, text: '12432'  
  allContentList: [{}, {
   is_ai: []
  }],
  num: 0
 }, // 页面加载 
 onLoad: function(e) {
  var that = this;
  if (wx, wx.getStorageSync('openid')) {
   this.setData({
    login: false
   })
  } else {
   this.setData({
    login: true
   })
  }
  let url = app.url + '/show.php'
  util.request(url, 'post', {}, '正在加载···', function(res) {
   for (var i = 0; i < res.data.length; i++) {
    console.log(i, res.data[i].iv, wx.getStorageSync('openid'))
    if (res.data[i].iv == wx.getStorageSync('openid')) {
     res.data[i].is_my = true
    } else {
     res.data[i].is_ai = true
    }
   }
   that.setData({
    allContentList: res.data,
    inputValue: ''
   })
   that.bottom()
  })
 }, // 获取用户信息并且登录,获取openid 
 userinfo: function(e) {
  var that = this;
  var nickName = JSON.parse(e.detail.rawData).nickName;
  var avatarUrl = JSON.parse(e.detail.rawData).avatarUrl;
  console.log('nickName:', nickName, 'avatarUrl:', avatarUrl)
  wx.setStorageSync('nickName', nickName)
  wx.setStorageSync('avatarUrl', avatarUrl)
  wx.login({
   success: function(res) {
    let url = app.url + '/login.php'
    if (res.code) {
     util.request(url, 'post', {
      code: res.code
     }, '正在登录···', function(res) {
      console.log(res.data)
      if (res.data.openid) {
       var openid = res.data.openid;
       wx.setStorageSync('openid', openid);
       if (avatarUrl && openid) {
        wx.showToast({
         title: '登录成功!',
        })
        that.setData({
         login: false
        })
       }
      }
     })
    }
   }
  });
 }, // 页面加载完成 
 onReady: function() {
  var that = this;
  this.on_recorder();
  this.bottom()
 }, // 提交文字 
 submitTo: function(e) {
  console.log('提交文字')
  let that = this;
  if (that.data.inputValue == "") {
   return;
  }
  var url = app.url + '/up_text.php'
  var data = {
   avatarUrl: wx.getStorageSync('avatarUrl'),
   iv: wx.getStorageSync('openid'),
   inputValue: that.data.inputValue,
   time: (new Date()).getTime()
  }
  console.log('提交文字data:', data)
  util.request(url, 'post', data, '', function(res) { // res.data = res.data.reverse();  
   for (var i = 0; i < res.data.length; i++) {
    console.log(i, res.data[i].iv, wx.getStorageSync('openid'))
    if (res.data[i].iv == wx.getStorageSync('openid')) {
     res.data[i].is_my = true
    } else {
     res.data[i].is_ai = true
    }
   }
   that.setData({
    allContentList: res.data,
    inputValue: ''
   })
   that.bottom()
  }, function(res) {})
 }, // 点击加号 
 add_icon_click: function(e) {
  console.log(e.target.id) // e.target.id == 1 点击加号  ==2 点击 X  
  if (e.target.id == 2) {
   this.setData({
    add: true,
    cross: false,
    input_bottom: 0
   })
  } else if (e.target.id == 1) {
   this.setData({
    add: false,
    cross: true,
    input_bottom: 240
   })
  }
 }, // 输入框 
 bindKeyInput: function(e) {
  console.log(e.detail.value)
  if (e.detail.value == "") {
   this.setData({
    if_send: false,
    inputValue: e.detail.value
   })
  } else {
   this.setData({
    if_send: true,
    inputValue: e.detail.value
   })
  }
 }, // 获取到焦点 
 focus: function(e) {
  var that = this;
  console.log(e.detail.height)
  this.setData({
   focus: true,
   add: true,
   cross: false,
   input_bottom: e.detail.height
  })
 }, // 失去焦点 
 no_focus: function(e) {
  if (this.data.cross) {
   this.setData({
    focus: false,
    input_bottom: 240,
   })
  } else {
   this.setData({
    focus: false,
    input_bottom: 0
   })
  }
 },
 onHide: function() {}, // 获取hei的id节点然后屏幕焦点调转到这个节点  
 bottom: function() {
  var that = this;
  this.setData({
   scrollTop: 100000
  })
 },
 hide_bg: function() {
  this.setData({
   block: false
  })
 }, // 点击录音事件 
 my_audio_click: function(e) {
  console.log('my_audio_click执行了', e)
  var index = e.currentTarget.dataset.id;
  console.log('url地址', this.data.allContentList[index].audio);
  innerAudioContext.src = this.data.allContentList[index].audio
  innerAudioContext.seek(0);
  innerAudioContext.play();
 }, // 手指点击录音 
 voice_ing_start: function() {
  var that = this;
  this.setData({
   voice_ing_start_date: new Date().getTime(), //记录开始点击的时间  
  })
  const options = {
   duration: 10000, //指定录音的时长,单位 ms  
   sampleRate: 16000, //采样率  
   numberOfChannels: 1, //录音通道数 
   encodeBitRate: 24000, //编码码率  
   format: 'mp3', //音频格式,有效值 aac/mp3   
   frameSize: 12, //指定帧大小,单位 KB  
  }
  recorder.start(options) //开始录音  
  this.animation = wx.createAnimation({
   duration: 1200,
  }) //播放按钮动画  
  that.animation.scale(0.8, 0.8); //还原 
  that.setData({
   spreakingAnimation: that.animation.export()
  })
 }, // 录音监听事件 
 on_recorder: function() {
  recorder.onStart((res) => {
   console.log('开始录音');
  })
  recorder.onStop((res) => {
   console.log('停止录音,临时路径', res.tempFilePath); // _tempFilePath = res.tempFilePath;   
   var x = new Date().getTime() - this.data.voice_ing_start_date
   if (x > 1000) {
    that.data.allContentList.push({
     is_my: true,
     audio: res.tempFilePath,
     length: x / 1000 * 30
    });
    that.setData({
     allContentList: that.data.allContentList
    })
   }
  })
  recorder.onFrameRecorded((res) => {
   var x = new Date().getTime() - this.data.voice_ing_start_date
   if (x > 1000) {
    console.log('onFrameRecorded res.frameBuffer', res.frameBuffer);
    string_base64 = wx.arrayBufferToBase64(res.frameBuffer) // console.log('string_base64--', wx.arrayBufferToBase64(string_base64))  
    if (res.isLastFrame) {
     var data = {
      audioType: 3,
      cmd: 1,
      type: 2,
      signType: 'BASE64',
      session: session,
      body: string_base64,
     }
     console.log('that.data.allContentList', that.data.allContentList) // 进行下一步操作   
    } else {
     var data = {
      cmd: 1,
      audioType: 2,
      type: 2,
      signType: 'BASE64',
      session: session,
      body: string_base64
     }
     console.log('录音上传的data:', data)
    }
   }
  })
 }, // 手指松开录音 
 voice_ing_end: function() {
  var that = this;
  that.setData({
   voice_icon_click: false,
   animationData: {}
  })
  this.animation = "";
  var x = new Date().getTime() - this.data.voice_ing_start_date
  if (x < 1000) {
   console.log('录音停止,说话小于1秒!')
   wx.showModal({
    title: '提示',
    content: '说话要大于1秒!',
   })
   recorder.stop();
  } else { // 录音停止,开始上传  
   recorder.stop();
  }
 }, // 点击语音图片 
 voice_icon_click: function() {
  this.setData({
   voice_icon_click: !this.data.voice_icon_click
  })
 },
 
})

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
在Javascript中定义对象类别
Dec 22 Javascript
利用try-catch判断变量是已声明未声明还是未赋值
Mar 12 Javascript
jquery鼠标放上去显示悬浮层即弹出定位的div层
Apr 25 Javascript
浅谈JavaScript字符串拼接
Jun 25 Javascript
jQuery实现的网页左侧在线客服效果代码
Oct 23 Javascript
jQuery数据类型小结(14个)
Jan 08 Javascript
如何理解Vue的v-model指令的使用方法
Jul 19 Javascript
angularJS1 url中携带参数的获取方法
Oct 09 Javascript
javascript中this的用法实践分析
Jul 29 Javascript
vue2.0 获取从http接口中获取数据,组件开发,路由配置方式
Nov 04 Javascript
JS异步宏队列与微队列原理区别详解
Jul 02 Javascript
vue 实现setInterval 创建和销毁实例
Jul 21 Javascript
vue实现折线图 可按时间查询
Aug 21 #Javascript
Vue按时间段查询数据组件使用详解
Aug 21 #Javascript
js绘制一条直线并旋转45度
Aug 21 #Javascript
AJAX XMLHttpRequest对象创建使用详解
Aug 20 #Javascript
基于vue.js仿淘宝收货地址并设置默认地址的案例分析
Aug 20 #Javascript
微信小程序以7天为周期连续签到7天功能效果的示例代码
Aug 20 #Javascript
微信小程序连续签到7天积分获得功能的示例代码
Aug 20 #Javascript
You might like
简要剖析PHP的Yii框架的组件化机制的基本知识
2016/03/17 PHP
Yii操作数据库实现动态获取表名的方法
2016/03/29 PHP
Js中的onblur和onfocus事件应用介绍
2013/08/27 Javascript
Node.js事件驱动
2015/06/18 Javascript
理解和运用JavaScript的闭包机制
2015/08/13 Javascript
JavaScript学习笔记之取数组中最大值和最小值
2016/03/23 Javascript
非常酷炫的Bootstrap图片轮播动画
2016/05/27 Javascript
Angularjs---项目搭建图文教程
2016/07/08 Javascript
AngularJS中指令的四种基本形式实例分析
2016/11/22 Javascript
angularjs使用directive实现分页组件的示例
2017/02/07 Javascript
jQuery实现在新增加的元素上添加事件方法案例分析
2017/02/09 Javascript
js输入框使用正则表达式校验输入内容的实例
2017/02/12 Javascript
vue2.0实战之基础入门(1)
2017/03/27 Javascript
BootStrap Table 后台数据绑定、特殊列处理、排序功能
2017/05/27 Javascript
vue mint-ui 实现省市区街道4级联动示例(仿淘宝京东收货地址4级联动)
2017/10/16 Javascript
[49:27]LGD vs OG 2018国际邀请赛小组赛BO2 第二场 8.16
2018/08/17 DOTA
python模拟登录百度代码分享(获取百度贴吧等级)
2013/12/27 Python
python文件和目录操作函数小结
2014/07/11 Python
Python脚本实现集群检测和管理功能
2015/03/06 Python
PYTHON 中使用 GLOBAL引发的一系列问题
2016/10/12 Python
Django实现一对多表模型的跨表查询方法
2018/12/18 Python
简单了解Python matplotlib线的属性
2019/06/29 Python
Python3远程监控程序的实现方法
2019/07/15 Python
详细整理python 字符串(str)与列表(list)以及数组(array)之间的转换方法
2019/08/30 Python
Iconfont(矢量图标)+iconmoon(图标svg互转)配合javascript实现社交分享系统
2020/04/21 Python
pyspark给dataframe增加新的一列的实现示例
2020/04/24 Python
python 爬取百度文库并下载(免费文章限定)
2020/12/04 Python
HTML5 Canvas 实现K线图的示例代码
2019/12/23 HTML / CSS
HTML5 直播疯狂点赞动画实现代码 附源码
2020/04/14 HTML / CSS
美国生日蛋糕店:Bake Me A Wish!
2017/02/08 全球购物
COACH德国官方网站:纽约现代奢侈品牌,1941年
2018/06/09 全球购物
Yahoo-PHP面试题4
2012/05/05 面试题
大学生职业生涯规划范文
2014/01/22 职场文书
高中生职业规划范文
2014/03/09 职场文书
出纳担保书范文
2014/04/02 职场文书
企业安全生产检查制度
2015/08/06 职场文书