Commit 8fcec1c1 by 尹烁恒

完成首页和选择页面的开发

parent e43ec9e2
......@@ -213,7 +213,6 @@
"version": "5.5.2",
"resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz",
"integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=",
"dev": true,
"requires": {
"co": "4.6.0",
"fast-deep-equal": "1.1.0",
......@@ -1958,8 +1957,7 @@
"co": {
"version": "4.6.0",
"resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz",
"integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=",
"dev": true
"integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ="
},
"coa": {
"version": "1.0.4",
......@@ -3655,14 +3653,12 @@
"fast-deep-equal": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz",
"integrity": "sha1-wFNHeBfIa1HaqFPIHgWbcz0CNhQ=",
"dev": true
"integrity": "sha1-wFNHeBfIa1HaqFPIHgWbcz0CNhQ="
},
"fast-json-stable-stringify": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz",
"integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=",
"dev": true
"integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I="
},
"fast-levenshtein": {
"version": "2.0.6",
......@@ -3833,6 +3829,95 @@
"readable-stream": "2.3.6"
}
},
"flyio": {
"version": "0.5.9",
"resolved": "https://registry.npmjs.org/flyio/-/flyio-0.5.9.tgz",
"integrity": "sha512-qClVDZQH1OsizhpqeLbdOy1OYsFct4m2fvlmQatdXcDL79TjR7HGWGEU4AJCYVVo8GnqFTRpsGeMDEwaCIW6yg==",
"requires": {
"request": "2.87.0"
},
"dependencies": {
"assert-plus": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz",
"integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU="
},
"aws-sign2": {
"version": "0.7.0",
"resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz",
"integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg="
},
"caseless": {
"version": "0.12.0",
"resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz",
"integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw="
},
"form-data": {
"version": "2.3.2",
"resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.2.tgz",
"integrity": "sha1-SXBJi+YEwgwAXU9cI67NIda0kJk=",
"requires": {
"asynckit": "0.4.0",
"combined-stream": "1.0.6",
"mime-types": "2.1.18"
}
},
"har-validator": {
"version": "5.0.3",
"resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.0.3.tgz",
"integrity": "sha1-ukAsJmGU8VlW7xXg/PJCmT9qff0=",
"requires": {
"ajv": "5.5.2",
"har-schema": "2.0.0"
}
},
"http-signature": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz",
"integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=",
"requires": {
"assert-plus": "1.0.0",
"jsprim": "1.4.1",
"sshpk": "1.14.1"
}
},
"request": {
"version": "2.87.0",
"resolved": "https://registry.npmjs.org/request/-/request-2.87.0.tgz",
"integrity": "sha512-fcogkm7Az5bsS6Sl0sibkbhcKsnyon/jV1kF3ajGmF0c8HrttdKTPRT9hieOaQHA5HEq6r8OyWOo/o781C1tNw==",
"requires": {
"aws-sign2": "0.7.0",
"aws4": "1.7.0",
"caseless": "0.12.0",
"combined-stream": "1.0.6",
"extend": "3.0.1",
"forever-agent": "0.6.1",
"form-data": "2.3.2",
"har-validator": "5.0.3",
"http-signature": "1.2.0",
"is-typedarray": "1.0.0",
"isstream": "0.1.2",
"json-stringify-safe": "5.0.1",
"mime-types": "2.1.18",
"oauth-sign": "0.8.2",
"performance-now": "2.1.0",
"qs": "6.5.1",
"safe-buffer": "5.1.2",
"tough-cookie": "2.3.4",
"tunnel-agent": "0.6.0",
"uuid": "3.2.1"
}
},
"tunnel-agent": {
"version": "0.6.0",
"resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz",
"integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=",
"requires": {
"safe-buffer": "5.1.2"
}
}
}
},
"follow-redirects": {
"version": "1.5.0",
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.5.0.tgz",
......@@ -4707,6 +4792,11 @@
"pify": "3.0.0"
}
},
"har-schema": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz",
"integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI="
},
"har-validator": {
"version": "2.0.6",
"resolved": "https://registry.npmjs.org/har-validator/-/har-validator-2.0.6.tgz",
......@@ -5545,8 +5635,7 @@
"json-schema-traverse": {
"version": "0.3.1",
"resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz",
"integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A=",
"dev": true
"integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A="
},
"json-stable-stringify-without-jsonify": {
"version": "1.0.1",
......@@ -6056,6 +6145,11 @@
"integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==",
"dev": true
},
"minapp-api-promise": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/minapp-api-promise/-/minapp-api-promise-1.0.2.tgz",
"integrity": "sha512-wQnKABIFYK0fKFz2ajsM05j51d+TlK5Cfw040yhyAKDGIQG3nRUUWpbHy8fw2AXNv6Idg3cVGntNzWgtugckMw=="
},
"minimalistic-assert": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz",
......@@ -6885,6 +6979,11 @@
"sha.js": "2.4.11"
}
},
"performance-now": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz",
"integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns="
},
"pify": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz",
......@@ -7796,8 +7895,7 @@
"qs": {
"version": "6.5.1",
"resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz",
"integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==",
"dev": true
"integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A=="
},
"query-string": {
"version": "4.3.4",
......
......@@ -11,6 +11,8 @@
"lint": "eslint --ext .js,.vue src"
},
"dependencies": {
"flyio": "^0.5.9",
"minapp-api-promise": "^1.0.2",
"mpvue": "^1.0.11",
"node-sass": "^4.9.0",
"sass-loader": "^7.0.2",
......
{
"description": "项目配置文件。",
"setting": {
"urlCheck": true,
"urlCheck": false,
"es6": false,
"postcss": true,
"minified": true,
......@@ -9,7 +9,7 @@
},
"miniprogramRoot": "./dist/",
"compileType": "miniprogram",
"appid": "wxa8fde6597493ec39",
"appid": "wx5c22f66dec164454",
"projectname": "mpvue_quickstart",
"condition": {
"search": {
......@@ -20,13 +20,24 @@
"current": -1,
"list": []
},
"plugin": {
"current": -1,
"list": []
},
"game": {
"currentL": -1,
"list": []
},
"miniprogram": {
"current": -1,
"list": []
"list": [
{
"id": -1,
"name": "counter",
"pathName": "pages/counter/main",
"query": ""
}
]
}
}
}
\ No newline at end of file
<template>
<div class="option-item">
<img />
{{text}}
</div>
</template>
......@@ -14,11 +15,20 @@ export default {
},
methods: {
handleTap (index) {
console.log('card.vue', index)
this.$emit('click', index)
}
}
}
</script>
<style>
.option-item {
width: 80%;
border: 2rpx solid #eee;
border-radius: 12rpx;
height: 100rpx;
line-height: 100rpx;
text-align: center;
margin: 20rpx auto;
}
</style>
import Vue from 'vue'
import App from './App'
Vue.config.productionTip = false
App.mpType = 'app'
const app = new Vue(App)
app.$mount()
......@@ -14,8 +12,8 @@ export default {
pages: ['pages/logs/main', '^pages/index/main'],
window: {
backgroundTextStyle: 'light',
navigationBarBackgroundColor: '#fff',
navigationBarTitleText: 'WeChat',
navigationBarBackgroundColor: '#cbbdab',
navigationBarTitleText: '我的手机',
navigationBarTextStyle: 'black'
},
debug: true
......
<template>
<div class="counter-page">
<scroll-view
class="scroll-left"
scroll-y
:style="{'height': height + 'rpx'}"
>
<card v-for="(item, index) in List" :key="index" :text="item.name" :list="item.list"></card>
</scroll-view>
<img src="../../assets/img/index-bg.jpg" class="counter-bg"/>
<!-- <div class="search-container">
<img src="../../assets/img/search_bg.png" class="search-bg" mode="aspectFit"/>
<div class="search-placeholder" :style="{opacity: hasFoucs || searchValue ? 0 : 1}">
<img src="../../assets/img/ic_search.png" class="search-icon">
<text>搜索</text>
</div>
<input class="search-input" type="text" @focus="handleInputFoucs" @blur="handleInputBlur" confirm-type="搜索" v-model.lazy="searchValue"/>
</div> -->
<div class="content-main">
<scroll-view
class="scroll-left"
scroll-y>
<div
v-for="(item, index) in brandList"
:key="index"
class="option-item" :class="{'is-active': index === p_index }" @click="handleOptionItem(index, item.name)">
<p class="option-badge" :style="{display: item.total ? 'block' : 'none' }"><span>{{item.total}}</span></p>
<img class="option-item-image" src="../../assets/img/index-option-bg-active.png" :style="{ display: index === p_index ? 'block' : 'none'} "/>
<img class="option-item-image" src="../../assets/img/index-option-bg.png" :style="{ display: index === p_index ? 'none' : 'block'} "/>
<p class="option-text">{{item.name}}</p>
</div>
</scroll-view>
<div class="scroll-right-container">
<scroll-view
class="scroll-right"
@scrolltolower="handleScrollBottom"
:lower-threshold="40"
scroll-y>
<div class="content-wrapper">
<div class="item-cell" v-for="(item, itemIndex) in itemList" :key="itemIndex" @click="handleItemCell(itemIndex)">
<!-- <img class="item-cell-image" src="../../assets/img/sel_box.png" :style="{ display: item.is_active ? 'none' : 'block'} "/> -->
<img class="item-cell-image" src="../../assets/img/sel_box_sel.png" :style="{ display: item.is_active ? 'block' : 'none'} "/>
<img :src="item.img" class="item-img" mode="aspectFit" :lazy-load="true"/>
<text class="item-text" :style="{color: item.is_active ? '#fff' : '#282828'}">{{item.name}}</text>
</div>
</div>
</scroll-view>
</div>
</div>
<div class="submit-field">
<button class="button-submit" open-type="getUserInfo" @getuserinfo="handleGetUserInfo"></button>
</div>
</div>
</template>
<script>
// Use Vuex
import card from '@/components/card'
import store from '@/store'
import fly, { setAuthorization } from '@/request'
export default {
computed: {
count () {
return store.state.counterPage.count
},
log () {
return store.state.logPage.log
brand () {
return this.brandList[this.p_index] && this.brandList[this.p_index].name
}
},
components: {
card
},
data() {
// 性能触发.所有的list只存key
return {
height: 400,
List: [{
name: '小米',
list: ['小米1', '小米2', '小米3']
}]
p_index: 1,
itemList: [],
brandList: [],
hasFoucs: false,
searchValue: '',
searchPage: 1
}
},
created() {
this.initData()
this.getConfig()
this.login()
},
methods: {
increment () {
store.commit('increment')
initData() {
this._cache = {}
},
login() {
const that = this
wx.login({
success(res) {
that._jscode = res.code
wx.setStorageSync('hasLogin', true)
},
fail() {
that.login()
}
})
},
handleInputFoucs() {
this.hasFoucs = true
},
handleInputBlur() {
console.log(this.searchValue)
this.hasFoucs = false
// 是否进行条件过滤?
this.fetchListByWord(this.searchValue, this.searchPage)
},
// 滑动到底端
handleScrollBottom() {
console.log(this._cache, this.brand)
const obj = this._cache[this.brand]
if (obj.loaded) return
this.fetchDetail(this.brand, obj.page)
},
handleGetUserInfo(e) {
console.log(e)
if (e.target.errMsg === 'getUserInfo:ok') {
this.getUserInfo(e.target)
} else {
wx.showModal({
title: '提示',
content: '亲,需要您授权才能提交哦',
showCancel: false
})
}
},
async getUserInfo(res) {
if (wx.getStorageSync('hasLogin')) {
this.handleSubmit()
return
}
fly.post('auth_login', {
encrypted_data: res.encryptedData,
iv: res.iv,
js_code: this._jscode
})
.then(res => {
console.log(res)
setAuthorization(res.data.token)
// 调用提交接口
this.handleSubmit()
})
.catch(async res => {
await this.login()
this.getUserInfo(res)
})
},
getConfig() {
fly.get('config?version=1.0.0')
.then(res => {
this.brandList = res.data.brands
this.brandList = this.brandList.map(item => {
return {
name: item,
total: 0
}
})
for (let { name } of this.brandList) {
this._cache[name] = {}
}
console.log(this._cache)
this.fetchDetail(this.brandList[1].name)
})
.catch(res => {
console.log(res)
wx.showToast({
title: '请更新微信'
})
})
},
// 提交
handleSubmit() {
console.log('handleSubmit', this._cache)
let array = []
Object.values(this._cache).forEach(phoneType => {
if (phoneType.list) {
let phoneArray = phoneType.list.filter(item => item.is_active).map(item => item.id)
array = array.concat(...phoneArray)
}
})
console.log(array, 'submit array')
console.log(JSON.stringify(array), 'json stringify array')
},
decrement () {
store.commit('decrement')
// 获取详情
fetchDetail(brand = '', page = 1) {
fly.get(`mobiles?brand=${brand}&page=${page}`)
.then(res => {
page++
this._cache[brand].page = page
// 是否加载结束
if (res.data.total_page <= page) {
this._cache[brand].loaded = true
}
this.itemList = this.itemList.concat(...res.data.mobiles)
this._cache[brand].list = this.itemList
})
.catch(res => {
console.log('warning')
})
},
// 根据条件获取手机列表
fetchListByWord(word = '', page = 1, count = 20) {
fly.get(`search?word=${word}&page=${page}&per_page=${count}`)
.then(res => {
this.itemList = res.data
})
},
handleOptionItem(index, brand = '') {
console.log(this.brand, brand, this._cache)
// 如果当前的已有list,就直接切换
if (this._cache[brand].list) {
this.itemList = this._cache[brand].list
this.p_index = index
return
}
// 清空当前列表
this.itemList = []
this.fetchDetail(brand)
this.p_index = index
},
// 点击右侧
handleItemCell(itemIndex) {
let total = 0
this.itemList = this.itemList.map((item, index) => {
if (item.is_active) {
total++
}
if (itemIndex === index) {
total++
return {
...item,
is_active: !item.is_active
}
}
return item
})
this._cache[this.brand].list = this.itemList
this.brandList[this.p_index].total = total
}
}
}
</script>
<style>
<style lang="scss">
.counter-page {
width: 100%;
background: #fff;
background: #c9baa8;
height: 100vh;
display: flex;
flex-flow: column nowrap;
.search-container {
position: relative;
margin: 0 30rpx;
flex: 0 0 80rpx;
line-height: 40rpx;
}
.search-placeholder {
position: absolute;
width: 100%;
height: 100%;
display: flex;
justify-content: center;
align-items: center;
color: #d9d9d9;
font-size: 30rpx;
.search-icon {
margin-right: 12rpx;
width: 31rpx;
height: 32rpx;
}
}
.search-bg {
position: absolute;
width: 100%;
height: 80rpx;
top: 0;
left: 0;
}
.search-input {
width: 100%;
height: 80rpx;
color: #d9d9d9;
font-size: 30rpx;;
padding: 0 40rpx
}
}
.scroll-left {
width: 300rpx;
height: 100vh;
background: red;
width: 200rpx;
height: 100%;
color: white;
}
.scroll-right-container {
width: calc(100% - 200rpx);
display: flex;
flex-flow: column nowrap;
margin: 20rpx 30rpx 0 40rpx;
}
.scroll-right {
height: 100%;
color: white;
}
.content-wrapper {
width: 100%;
height: 100%;
display: flex;
flex-flow: row wrap;
justify-content: space-between;
.item-cell {
position: relative;
width: 230rpx;
height: 150rpx;
margin-bottom: 20rpx;
text-align: center;
}
.item-cell-image {
position: absolute;
width: 100%;
height: 100%;
top: 0;
left: 0;;
}
.item-img {
position: absolute;
width: 100rpx;
height: 100rpx;
z-index: 3;
left: 50%;
top: 10rpx;
transform: translate3d(-50%, 0, 0);
}
.item-text {
position: absolute;
width: 100%;
height: 40rpx;
z-index: 3;
left: 50%;
bottom: 0;
transform: translate3d(-50%, 0, 0);
font-size: 20rpx;
margin: 0 10rpx;
@include ellipsis;
}
}
.content-main {
display: flex;
justify-content: center;
flex: 1;
}
.option-item {
position: relative;
width: 166rpx;
height: 72rpx;
line-height: 72rpx;
text-align: center;
margin: 20rpx auto 20rpx 30rpx;
}
.option-item-image {
position: absolute;
width: 100%;
height: 100%;
left: 0;
top: 0;
}
.counter-bg {
position: absolute;
width: 100%;
height: 100%;
top: 0;
left: 0;
}
.option-text {
position: absolute;
z-index: 1;
width: 100%;
height: 100%;
color: #dbd1c0;
font-size: 32rpx;
}
.option-badge {
position: absolute;
z-index: 1;
width: 32rpx;
height: 32rpx;
line-height: 32rpx;
border-radius: 50%;
background: orange;
font-size: 22rpx;
color: #fff;
top: 0;
right: 0;
}
.submit-field {
flex: 0 0 200rpx;
}
.button-submit {
width: 489rpx;
height: 123rpx;
margin: 40rpx auto;
background-color: transparent;
background-repeat: no-repeat;
background-origin: center center;
background-size: contain;
// 使用base64.避免很复杂的布局。因为必须是button。同时还需要具有用户授权的功能.
background-image: url('');
}
</style>
......@@ -3,3 +3,11 @@ import App from './index'
const app = new Vue(App)
app.$mount()
// 这里才会导出page.json
export default {
config: {
navigationBarTitleText: '布里吉',
disableScroll: true
}
}
<template>
<div class="index-page">
<button class="start-button" @click="handleStart">开始</button>
<img src="../../assets/img/index-bg.jpg" class="index-bg"/>
<img src="../../assets/img/index-bg-cover.jpg" class="index-bg-cover" />
<img src="../../assets/img/index-start-button.jpg" class="index-start-button" @click="handleStart"/>
</div>
</template>
......@@ -45,16 +47,44 @@ export default {
created () {
// 调用应用实例的方法获取全局数据
this.getUserInfo()
// this.getUserInfo()
}
}
</script>
<style scoped lang="scss">
.index-page {
@include ellipsis;
position: relative;
width: 100%;
height: 100vh;
background: #fff;
background: #c9baa8;
text-align: center;
}
.index-bg {
position: absolute;
width: 100%;
height: 100%;
top: 0;
left: 0;
z-index: 1;
}
.index-bg-cover {
position: absolute;
width: 659rpx;
height: 798rpx;
top: 30rpx;
left: 50%;
transform: translate3d(-50%, 0 ,0);
z-index: 2;
}
.index-start-button {
position: absolute;
width: 489rpx;
height: 123rpx;
bottom: 180rpx;
left: 50%;
transform: translate3d(-50%, 0, 0);
z-index: 3;
}
</style>
......@@ -4,7 +4,7 @@ import App from './index'
const app = new Vue(App)
app.$mount()
// 这里才会导出json
// 这里才会导出page.json
export default {
config: {
navigationBarTitleText: '布里吉',
......
......@@ -11,7 +11,7 @@ function setAuthorization (header) {
Authorization = header
}
const log = console.log
fly.config.baseURL = ''
fly.config.baseURL = 'https://test-api.wxagame.com/game-mobile/api/v1/'
// add token
fly.interceptors.request.use((config, promise) => {
......@@ -62,3 +62,6 @@ fly.interceptors.response.use((config, promise) => {
}
})
export default fly
export {
setAuthorization
}
......@@ -31,10 +31,21 @@ const logsModule = {
}
}
}
const indexModule = {
state: {
config: {}
},
mutations: {
setConfig (state, payload) {
state.config = payload
}
}
}
const store = new Vuex.Store({
modules: {
counterPage: counterModule,
logPage: logsModule
logPage: logsModule,
indexPage: indexModule
}
})
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment