使用uniapp
任务描述
根据客户给定的需求描述,利用跨平台开发工具或原生开发工具,编码实现任务需求功能设计。
任务1:实现用户登录界面(12分)
登录界面
设计智慧城市App,利用移动开发工具HbuilderX打开项目,编码实现智慧城市App的登录页面,包括App的标题、账户和密码、自动登录切换按钮、登录按钮,以及微信、QQ和sina微博快捷登录的图标,如图1所示。输入账户和密码后,点击登录按钮,进入主界面。
任务2:实现主界面(15分)
进入App主页面(主页),如图2所示。彰显智慧城市风采,打造智慧型服务大厅,科技赋能,创新智能,打造视、听、触体验具佳的业务服务大厅,同时供应多种智慧城市解决方案。
数据分析页面
智慧城市重要职能是采集、存储、分析、挖掘城市运行中的所承受的大数据,具备全面感知和全面分析的能力,同时能够展示和扩展。
智慧城市的数据分析建设,有利于解决城市发展问题,有利于提升城市信息管理水平,主要包括以下功能。
1.条形图:请分析id为50-53新闻信息的点赞数量,按照新闻类别进行分析,横坐标为新闻标题(显示标题前4个字,其余使用“..”代替),纵坐标为新闻点赞数;
2.折线图:请分析统计所有车牌照所在地变化曲线,横坐标为牌照所在地名称,纵坐标为牌照所在地统计数,如(横坐标为甘A、甘B、甘C,纵坐标为10,20,30)
3.饼状图:请分析各招聘岗位占总招聘岗位的比例。(图内显示相应岗位名称及比例值)
注:接口数据详见API接口文档(需要自行处理、合并封装、解析相应接口返回的数据)。
<template>
<view class="wrap">
<view class="bgwhite info">
<view class="item">
<view class="label">账号</view>
<input placeholder="请输入账号" v-model="account"/>
</view>
<view class="item">
<view class="label">密码</view>
<input placeholder="请输入密码" v-model="passwd" type="password"/>
</view>
</view>
<view class="bgwhite info auto">
<view class="item">
<view class="label">自动登录</view>
<switch :checked="autologin" @change="setAuto"/>
</view>
</view>
<button type="primary" class="login" @click="doLogin">登录</button>
<view class="reg">
<a href="#">注册账号</a>
<view class="line">|</view>
<a href="#">忘记密码</a>
</view>
<view class="footer">
<image src=""></image>
<image src=""></image>
<image src=""></image>
</view>
</view>
</template>
<script>
import store from '@/store/index.js';//vuex
export default {
data() {
return {
account:'',//存储账号信息
passwd:'',//存储密码
autologin:false//存储是否自动登录
}
},
onLoad() {
let token=uni.getStorageSync('token');//获取storage中的token,判断上次是否是自动登录保存了身份token
if(token){//自动登录
store.commit('setToken',token);//更新vuex
uni.redirectTo({url:'/pages/index/index'});//跳转到需要的页面,这里跳转首页
}
},
methods: {
setAuto(e){this.autologin=e.detail.value;},
doLogin(){
if(this.account==''){uni.showModal({content:'请输入账号!',showCancel:false});return false}
if(this.passwd==''){uni.showModal({content:'请输入密码!',showCancel:false});return false}
//显示加载等待层
uni.showLoading({mask:true,title:'正在验证信息...'});
//验证通过发送请求,un.request更多配置参考:https://uniapp.dcloud.net.cn/api/request/request.html#request
uni.request({
url:'http://124.93.196.45:10001/prod-api/api/login',
method:'POST',
data:{
username:this.account,
password:this.passwd
},
headers:{'content-type':'application/json'},
success:(res)=>{
if(res.data.code==200){//成功,跳转用户后台或者需要的页面
if(this.autologin){//自动登录,保存token到localStorage中
uni.setStorageSync('token',res.data.token);
store.commit('setToken',token);//更新vuex,以便其他页面使用
}
else uni.removeStorageSync('token');
uni.redirectTo({url:'/pages/index/index'});
}
//失败提示错误信息
else uni.showModal({content:res.data.msg,showCancel:false});
},
fail(res){
uni.showModal({content:'请求失败!',showCancel:false});
},
complete(){uni.hideLoading()}
})
}
}
}
</script>
<style>
page{background-color: #f0eff4;font-size: 16px;}
.wrap{max-width:480px;margin: 0 auto;}
.bgwhite{background-color: #fff;}
.info{margin-top: 25px;border:solid 1px #ccc;padding:0 20px;margin-bottom:10px;}
.info .item{display: flex;padding:10px 0;}
.info .item:first-of-type{border-bottom: solid 1px #eee;}
.info .item .label{margin-right:30px;}
input{flex-grow:1}
.auto{margin-top: 10px;}
.auto .item{line-height: 30px;justify-content: space-between}
.login{line-height: 55px;margin: 30px 0;font-size: 18px;}
.reg{text-align: center;}
.reg .line{padding:0 10px;display: inline-block;}
.reg a{color: #6096b9;text-decoration: none;}
.footer{position:absolute;bottom:40px;text-align:center;left:0;width:100%;}
.footer image{border-radius: 50%;border:solid 1px #eee;background-color: #ccc;width:60px;height: 60px;margin-right: 30px;display: inline-block;}
.footer image:last-of-type{margin-right: 0;}
</style>
设置pages.json中这个页面的style
{
"pages": [
{
"path" : "pages/loginpage/loginpage",
"style" :
{
"navigationBarTitleText": "智慧城市",
"enablePullDownRefresh": false,
"navigationBarBackgroundColor": "#f7f7f7"
}
}]
......其他配置
}
这个其实是css的知识,应该学习下css布局,这个可以用flex布局了解下
拖拖拽拽不就出来了
学习WEB前端,弹性布局、内边距、外边距、元素对齐方式 是必需掌握的知识点。
分析如下,供参考(在根view下,添加 4个子view,请根据实际做微调):
标题
表单
账号:账号+输入框
密码:密码+输入框
自动登录
登录按钮
注册/找回密码(垂直居中、文字居中)
注册
竖线分隔符(可以在 css 中指定 注册元素的的 ::after 追加)
找回密码
底部第三方平台登录
弹性布局 介绍这篇很详细:
该回答引用ChatGPT
<template>
<view class="login-wrapper">
<view class="login-box">
<form>
<view class="input-wrapper">
<view class="input-label">账号</view>
<input type="text" placeholder="请输入账号" class="input-field" />
</view>
<view class="input-wrapper">
<view class="input-label">密码</view>
<input type="password" placeholder="请输入密码" class="input-field" />
</view>
<button class="login-btn">登录</button>
</form>
</view>
</view>
</template>
<style lang="scss">
.login-wrapper {
height: 100%;
display: flex;
justify-content: center;
align-items: center;
background-image: url('../../static/login-bg.jpg');
background-size: 100% 100%;
}
.login-box {
width: 400rpx;
height: 350rpx;
background: rgba(255, 255, 255, 0.9);
border-radius: 10px;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
}
.input-wrapper {
display: flex;
margin-bottom: 20rpx;
}
.input-label {
width: 80rpx;
font-size: 28rpx;
line-height: 80rpx;
text-align: center;
}
.input-field {
width: 240rpx;
height: 80rpx;
border: none;
border-radius: 5rpx;
background-color: #F8F8F8;
font-size: 28rpx;
padding: 0 10rpx;
}
.login-btn {
width: 300rpx;
height: 80rpx;
border: none;
border-radius: 5rpx;
background-color: #FF3E3E;
color: white;
font-size: 32rpx;
}
</style>
<template>
<view>
<!-- 主体表单 -->
<view class="main1">
<view class="title">智慧城市</view>
<view class="item">
<image class="login_image" src="../../../static/login/userName.png"></image>
<input v-model="phoneData" class="main-input" placeholder="请输入账号" />
</view>
<view style="margin-top: 8px"></view>
<view class="item">
<image class="login_image" src="../../../static/login/password.png"></image>
<input v-model="passData" class="main-input" type="password" placeholder="请输入密码" />
</view>
<view class="item" style="display: flex;justify-content:space-between;">
<text>自动登录</text>
<switch v-model="isSwitch" />
</view>
</view>
<view class="btn_login" @click="startLogin">登录</view>
<!-- 底部信息 -->
<view class="footer" style="margin-top: 16px">
<view>
<navigator url="/pages/my/register/select" open-type="navigate">新用户注册</navigator>
</view>
<view style="margin-left: 32%;">
<navigator url="/pages/my/forget/person" open-type="navigate">忘记密码</navigator>
</view>
</view>
<view class="weixinLogin">
<view
style="border: 1upx inset;width:30%;float: left;margin-top: 2%;margin-left: 3%;border-color: #F8F8F8;">
</view>
第三方账号登录
<view
style="border: 1upx inset;width:30%;float: right;margin-top: 2%;margin-right: 3%;border-color: #F8F8F8;">
</view>
</view>
<!-- 其他登录 -->
<view class="other_login cuIcon">
<view class="login_icon_wx">
<image style="width: 50px;height: 50px;" src="../../../static/wx.png" @tap="login_weixin"></image>
</view>
<view class="login_icon_wx">
<image style="width: 50px;height: 50px;" src="../../../static/wx.png" @tap="login_weixin"></image>
</view>
<view class="login_icon_wx">
<image style="width: 50px;height: 50px;" src="../../../static/wx.png" @tap="login_weixin"></image>
</view>
</view>
</view>
</template>
<script>
var _this;
export default {
data() {
return {
//logo图片 base64
logoImage: '',
phoneData: '', //用户/电话
passData: '', //密码
isSwitch:false,//是否自动登录
isRotate: false //是否加载旋转
};
},
mounted() {
_this = this;
},
destroyed() {
uni.hideKeyboard();
},
methods: {
toLogin() {
uni.showLoading({
title:"登陆中..."
})
uni.request({
url:'',//你登录请求的路径,
data:{
loginName: this.phoneData,
password: this.passData,
isSwitch:this.isSwitch
},
method:'POST',
success:(res) => {
if (res.resultCode == 1) {
uni.setStorageSync('token', res.data);
this.getUserInfo();
} else {
uni.showToast({
title:res.errmsg
})
}
uni.hideLoading();
},
fail:() => {
uni.hideLoading();
}
})
},
getUserInfo() {
uni.request({
url:'',//你请求的路径,
method:'GET',
success:(res) => {
if (res.resultCode == 1) {
uni.setStorageSync('password', this.passData);
uni.setStorageSync('type', res.data.type);
uni.setStorageSync('id', res.data.id);
if (res.data.type == 1) {
uni.setStorageSync('userName', res.data.telNumber);
uni.setStorageSync('mobile', res.data.telNumber);
} else {
uni.setStorageSync('userName', res.data.usrMail);
uni.setStorageSync('mobile', res.data.usrMail);
}
uni.setStorageSync('mobile', res.data.userName);
this.loginJifen();
uni.navigateBack({});
} else {
uni.showToast({
title:res.errmsg,
icon:'fail'
})
}
uni.hideLoading();
},
fail:() => {
uni.hideLoading();
}
})
},
checkOpenid(openid) {
uni.showLoading({
title:"登陆中..."
})
uni.setStorageSync('openid', openid);
uni.request({
url:'',//你登录请求的路径,
data:{
openId: openid
},
method:'POST',
success:(res) => {
if (res.errmsg == '微信号未绑定') {
uni.navigateTo({
url: '',//账号绑定页面
});
} else {
uni.setStorageSync('token', res.data);
this.getUserInfo();
}
uni.hideLoading();
},
fail:() => {
uni.hideLoading();
}
})
},
startLogin() {
if (this.phoneData.length == 0) {
uni.showToast({
title:'请输入账号',
icon:'fail'
})
return;
}
if (this.passData.length < 1) {
uni.showToast({
title:'请输入密码',
icon:'fail'
})
return;
}
this.toLogin();
},
loginJifen() {
uni.request({
url:'',//你登录请求的路径,
data:{
scoreType: 0,
score: 2
},
method:'POST',
success:(res) => {
uni.hideLoading();
},
fail:() => {
uni.hideLoading();
}
})
},
login_weixin() {
let that = this;
uni.login({
provider: 'weixin',
success: function(loginRes) {
console.log(loginRes.authResult);
that.checkOpenid(loginRes.authResult.openid);
}
});
this.checkOpenid('123456');
}
}
};
</script>
<style>
@import url('../../../static/css/main.css');
@import url('../../../static/css/icon.css');
.type {
display: flex;
margin-left: 16px;
border-bottom: 1px solid #eeeeee;
width: 100%;
}
.title {
position: absolute;
width: 100%;
height: 45px;
top: 30px;
left: 0;
right: 0;
line-height: 45px;
font-size: 20px;
text-align: center;
}
.title_des {
font-weight: bold;
color: #0055b8;
font-size: 22px;
margin-bottom: 32px;
}
.other_login {
margin: 150px auto;
width: 300px;
display: flex;
justify-content: space-around;
align-items: center;
}
.login_icon_wx {
color: #999999;
border: none;
}
.weixinLogin {
color: #999999;
text-align: center;
font-size: 12px;
margin-top: 60px;
left: auto;
right: auto;
}
.footer_des {
color: #666666;
text-align: center;
font-size: 14px;
margin-top: 40px;
}
.getCode {
font-size: 14px;
margin-left: 40px;
color: #0055b8;
}
.item {
line-height: 40px;
display: flex;
}
.login_image {
margin-top: 8px;
width: 20px;
height: 25px;
}
.main-input {
font-size: 14px;
margin-left: 16px;
border-bottom: 1px solid #eeeeee;
height: 40px;
line-height: 40px;
}
.main-input-code {
width: 70%;
font-size: 14px;
height: 40px;
line-height: 40px;
}
.btn_login {
color: #ffffff;
font-size: 16px;
width: 260px;
height: 40px;
background: #0055b8;
border-radius: 8px;
line-height: 40px;
text-align: center;
margin-left: auto;
margin-right: auto;
margin-top: 45px;
}
.main-input {
flex: 1;
text-align: left;
font-size: 28 upx;
padding-right: 10 upx;
margin-left: 20 upx;
}
</style>
您好,我是有问必答小助手,您的问题已经有小伙伴帮您解答,感谢您对有问必答的支持与关注!博主你好 我看着你和另一个博客的交流评论之后和我的问题差不多 就是你改动后的代码是如上图的那个吗??我还想问接口地址一样、数格格式一样、请求类型一样、接口也一样 为什么访问不到里面的数据啊 最基本的照片也无法我访问到 浏览器显示“没有此账户”,那是接口文档的问题还是说自己敲代码的问题啊 是不是文档中应该提供一个账号和密码 才能登陆上去啊 这样的话 那就是文档的问题对吗?还是说出现这一bug, 另有其他的原因啊 在线等回复