您的当前位置:首页正文

基于微信小程序的实验室预约系统的设计与开发

2024-11-28 来源:个人技术集锦

1、前言介绍

伴随着信息技术与互联网技术的不断发展,校园也进到了一个新的信息化时代,传统管理技术性没法高效率、容易地管理体育场预约信息内容。为了实现时代的发展必须,提升实验室预约高效率,各种各样实验室预约体系应时而生,实验室预约管理系统的实现是信息内容时代浪潮时代的产物之一。一切系统都要遵循系统设计的基本流程。它还需要通过市场调查、需求分析报告、汇总设计、详尽设计以及测试,根据Node语言表达设计,完成实验室预约管理系统。该系统根据B/S,即所谓的电脑浏览器/网络服务器方式,运用Node技术,前端采用uniapp微信小程序等技术实现,选用MySQL作为数据库。
本实验室预约管理系统采用前后端分离的方式来实现,前端Vue、Element组件编写前端代码为主、其特点可以实时渲染加载数据,不用像以前传统的页面,写完要重新启动项目才能加载数据。系统还采用的File文件组件上传和预览图片等,以及前端利用了第三方富文本编辑器框架技术实现了实验室详情介绍和系统公告等模块。这样比传统文本框只能填写文字又来更好更直观的页面呈现方式。不仅可以上传文字,还可以直接复制网页图片文字等。实现动态渲染页面样式和内容等。

2、开发技术简介

本节介绍场馆预约平台用到的一些技术和开发环境的简介,用到开发技术主要包括:
(1)前端用到Element UI组件库、Vue框架和微信小程序
(2)后端用到Node
(3)包管理器Npm
(4)中间件Express
(5)数据库MySQL
系统开发环境主要是:前端开发工具Vscode,Hbuilder、操作系统Win10、CPU i5-9300H、内存8G。

3、系统功能图

实验室预约系统主要由客户端(微信小程序和安卓手机App)和Web管理端(基于Web服务器Apache Tomcat,MySQL数据库)这两个子系统组成。这两个子系统之间通过Htpp的request请求和response响应以json形式的数据交互,共同实现了实验室预约平台的功能。
客户服务端主要包括三大主要的功能模块:登录、预约和扫码签到模块。Web管理端主要包括三大主要的功能模块:用户、动态和预约模块。这六大模块还有相应的子模块,通过对子模块的实现来实现整个客户端的功能。如图4-3所示。

4、功能实现

系统登录功能是程序必不可少的功能,在登录页面必填的数据有两项,一项就是账号,另一项数据就是密码,当管理员正确填写并提交这二者数据之后,管理员就可以进入系统后台功能操作区。下图就是管理员登录页面。

该界面实现对实验室的添加、删除、修改等一系列功能

该界面实现对实验室的动态的添加、删除、修改等一系列操作

实验室预约的流程:当用户选择自己需要的预约的项目后,则需要用户选择合法的日期并且本系统的合法日期为当前日期的下一天的六天合法日期。当用户进入预约详情页面时用户可以选择预约的时间和地点。预约按钮点击后会判断用户是否登录以及是否重复预约等。若预约信息合法则同request的库生成对应订单的二维码并将预约信息存入到数据库中






5、库表设计

程序设计是离不开对应数据库的设计操作的,这样的做法就是减少数据对程序的依赖性,所以数据库的设计也是需要花费大量的日常时间来进行设计的,在设计中对程序开发需要存储的数据信息进行实体划分,先确认实体,然后设计实体的属性等操作,这种设计就是数据库设计里面不能少的必须有的E-R模型设计。为了降低程序设计的对应的数据库设计难度,开发人员也可以使用相应的工具来进行E-R模型设计,现在市面上设计E-R模型的工具有PowerDesigner建模工具,Navicat制作工具,还有微软的Visio绘图工具。为了简便起见,本程序在设计E-R模型的时候,就选用了微软的Visio这款功能强大,操作便利的绘图工具。

1.预约实体及其附属图


2.管理员实体及其附属图


3.实验室实体及其附属图

6、性能测试

单元测试是软件开发中的一种测试方法,用于测试代码中的最小可测试单元(通常是函数或方法),以验证其行为是否符合预期[15]。单元测试可以检测代码中的缺陷和错误,及时发现和修复代码问题,提高代码的质量和稳定性,也可以帮助开发人员快速定位问题,并提供反馈和调试信息,提高开发效率和效果。总之,单元测试是软件开发中非常重要的一环,可以帮助开发人员提高代码质量和效率,提高软件产品的质量和稳定性。

1.from base.data_read import ReadData
2.from base.response import method_post
3.import pytest

4.# ====测试数据读取====
5.def data_read():
6.    # 1. 将yaml文件读取
7.    data_dict = ReadData('data.yaml').read_data()
8.    # 2. 从字典中取数据  url  accounts
9.    url = data_dict['test_user_login']['url']
10.    accounts = data_dict['test_user_login']['accounts']
11.    # 3. 返回数据 供测试用例调用
12.    return url, accounts
13.
14.
15.class TestLogin():
16.    @pytest.mark.parametrize('url', [data_read()[0]])
17.    @pytest.mark.parametrize('accounts', data_read()[1])
18.    def test_login(self, url, accounts):
19.        # 1. 注意这里的account是列表格式
20.        print('\n>>>', type(accounts))
21.        print(accounts)
22.        # 2. 发请求
23.        response = method_post(url, json=accounts)
24.        # 3. 判断是否请求成功
25.        assert response.status_code == 200
26.        # 4. 根据返回的code判断是否登录成功
27.        code = response.json().get('code')
28.        print(response.json())
29.        assert code == 0, '登录失败!'
30.# =============测试结果==============
31.========= 1 failed, 1 passed in 11.81s ==========
7、关键代码
//index.js
//获取应用实例
const app = getApp()
const {login} =require('../../api/login')
Page({
  data: {
    phone: '',
    password: '',
    clientHeight:'',
    array: ['学生', '教师', ],
    index: 0,
  },
  onLoad(){
    var that=this
    wx.getSystemInfo({ 
      success: function (res) { 
        console.log(res.windowHeight)
          that.setData({ 
              clientHeight:res.windowHeight
        }); 
      } 
    }) 
  },
  bindType: function (e) {
    this.setData({
      index: e.detail.value
});
  },
  //协议
  goxieyi(){
   wx.navigateTo({
     url: '/pages/oppoint/oppoint',
   })
  },
  //获取输入款内容
  phone(e){
   this.setData({
     phone:e.detail.value
   })
  },
  password(e){
    this.setData({
      password:e.detail.value
    })
  },
  //登录事件
  goadmin(){
    let flag = false  //表示账户是否存在,false为初始值
    if(this.data.phone=='')
    {
      wx.showToast({
        icon:'none',
        title: '账号不能为空',
      })
    }else if(this.data.password==''){
      wx.showToast({
        icon:'none',
        title: '密码不能为空',
      })
    }else{
      let data={
        phone:this.data.phone,
        password:this.data.password,
        isadmin:this.data.index
      }
     login(data).then(res=>{
       console.log('55',res.data[0])
       wx.setStorageSync('userInfo', res.data[0])
       //添加登录状态
       wx.setStorageSync('statusLogin', true)
       wx.showToast({
        title: res.msg,
        icon:'none'
      })
       if(res.msg=="登录成功"){
        wx.switchTab({
            url: '../my/my',
          })
       }
     })
    }
  },
  register(){
    wx.navigateTo({
      url: './register',
    })
  }
})
 

<template>
  <div class="messge">
    <el-dialog title="提示" :visible.sync="dialogVisible" width="40%">
      <!-- 弹窗信息 -->
      <el-form
        ref="form"
        :model="form"
        :rules="rules"
        :inline="true"
        :before-close="handleClose"
        label-width="80px"
      >
        <el-form-item label="名称" prop="name">
          <el-input v-model="form.name" placeholder="实验室名称"></el-input>
        </el-form-item>
        <el-form-item label="开放时间" prop="openTime">
          <el-input
            v-model="form.openTime"
            placeholder="实验室开发时间"
          ></el-input>
        </el-form-item>
        <el-form-item label="地址" prop="address">
          <el-input v-model="form.address" placeholder="实验室地址"></el-input>
        </el-form-item>
        <el-form-item label="简介" prop="desc">
          <el-input v-model="form.desc" placeholder="实验室简介"></el-input>
        </el-form-item>
        <el-form-item label="负责人" prop="id">
          <el-select v-model="form.adminId" placeholder="请选择">
            <span></span>
            <el-option
              v-for="item in options"
              :key="item.id"
              :label="item.username"
              :value="item.id"
            >
            </el-option>
          </el-select>
        </el-form-item>
        <el-form-item label="实验室logo">
          <commonUpload></commonUpload>
        </el-form-item>
      </el-form>
      <!-- 表单提交 -->
      <span slot="footer" class="dialog-footer">
        <el-button @click="cancle">取 消</el-button>
        <el-button type="primary" @click="submit">确 定</el-button>
      </span>
    </el-dialog>

    <div class="manage-hander">
      <!-- 添加按钮 -->
      <el-button type="primary" @click="handleAdd">+ 新增</el-button>
    </div>
    <!-- 表格 -->
    <div class="common-table">
      <el-table :data="tableData" stripe height="90%" style="width: 100%">
        <el-table-column prop="name" label="实验室名称"> </el-table-column>
        <el-table-column prop="openTime" label="实验室开放时间">
        </el-table-column>
        <el-table-column prop="address" label="实验室地址"> </el-table-column>
        <el-table-column prop="logo" label="logo">
          <template slot-scope="scope">
            <el-image
              style="width: 150px; height: 100px"
              :src="scope.row.logo"
              :preview-src-list="[scope.row.logo]"
            >
            </el-image>
          </template>
        </el-table-column>
        <el-form-item label="简介" prop="desc">
          <el-input v-model="form.desc" placeholder="实验室简介"></el-input>
        </el-form-item>
        <el-table-column label="负责人" prop="adminId">
          <template slot-scope="scope">
            <span v-for="item in options" :key="item.id">
              <span v-if="scope.row.adminId === item.id">
                {{ item.username }}
              </span>
            </span>
          </template>
        </el-table-column>
        <el-table-column prop="addr" label="操作">
          <template slot-scope="scope">
            <el-button
              type="primary "
              size="mini"
              @click="handleEdit(scope.row)"
              >编辑</el-button
            >
            <el-button
              type="danger"
              size="mini"
              @click="handleDelete(scope.row)"
              >删除</el-button
            >
          </template>
        </el-table-column>
      </el-table>
      <!-- 分页组件 -->
      <el-pagination
        class="page"
        layout="prev, pager, next"
        :total="total"
        @current-change="handlePage"
      >
      </el-pagination>
    </div>
  </div>
</template>

<script>
import commonUpload from "@/components/commonUpload.vue";
import { getPlace, addPlace, deletePlace, updatePlace } from "@/api/index";
import { allAdmin } from "@/api/reservation";
export default {
  name: "User",
  data() {
    return {
      dialogVisible: false,
      // 表单字段
      form: {
        name: "",
        openTime: "",
        address: "",
        logo: "",
        adminId: "",
        desc: "",
      },
      // 表單校驗
      rules: {
        name: [{ required: true, message: "请输入实验室名称" }],
        openTime: [{ required: true, message: "请输入实验室开发时间" }],
        address: [{ required: true, message: "请输入实验室地址" }],
        desc: [{ required: true, message: "请输入实验室简介" }],
      },
      options: [
        {
          id: "",
          username: "",
        },
      ],
      // 表单数据
      tableData: [],
      total: 0, //页面的总条数
      modelType: 0, //0表示是新增的弹窗 ,1表示的是编辑
      userFrom: {
        username: "",
      },
    };
  },
  components: {
    commonUpload,
  },
  methods: {
    // 提交
    submit() {
      if (this.modelType == 0) {
        this.$refs.form.validate((value) => {
          // 校验为真
          if (value) {
            console.log("value", this.form);
            this.form.logo = this.$store.state.upload.imgUrl;
            addPlace(this.form).then((res) => {
              // console.log(res);
              this.$message.success("添加成功");
            });
            this.$refs.form.resetFields();
            this.dialogVisible = false;
            this.getList();
          }
        });
      } else {
        this.$refs.form.validate((value) => {
          // 校验为真
          if (value) {
            console.log("value", this.form);
            updatePlace(this.form).then((res) => {
              // console.log(res);
              this.$message.success("修改成功");
            });
            this.$refs.form.resetFields();
            this.dialogVisible = false;
            this.getList();
          }
        });
      }
    },
    handle(e) {
      console.log();
      // let id = res.id;
      this.$confirm("要修改用户权限吗?", "提示", {
        confirmButtonText: "确定",
        cancelButtonText: "取消",
        type: "warning",
      }).then((res) => {
        privalUser(e).then((res1) => {
          console.log(res1.data.data);
          this.$message.success(res1.data.data.msg);
        });
      });
    },
    // 关闭弹窗
    handleClose() {
      this.$refs.form.resetFields();
      this.dialogVisible = false;
    },
    // 点击取消按钮
    cancle() {
      this.handleClose();
    },
    // 表单的编辑
    handleEdit(res) {
      this.modelType = 1;
      this.dialogVisible = true;
      // 注意到对当前的数据进行深度拷贝
      this.form = JSON.parse(JSON.stringify(res));
    },
    //表单的删除
    handleDelete(res) {
      // let id = res.id;
      this.$confirm("此操作将永久删除该文件, 是否继续?", "提示", {
        confirmButtonText: "确定",
        cancelButtonText: "取消",
        type: "warning",
      })
        .then(() => {
          // 调用删除接口
          var that = this;
          deletePlace(res).then((res) => {
            that.$message({
              type: "success",
              message: res.data.data.msg,
            });
          });
          this.getList();
        })
        .catch(() => {
          this.$message({
            type: "info",
            message: "已取消删除",
          });
        });
    },
    // 新增数
    handleAdd() {
      (this.modelType = 0), (this.dialogVisible = true);
      this.form = {};
    },
    getOpstions() {
      allAdmin().then((res) => {
        this.options = res.data.data.data;
      });
    },
    // 查询数据
    handleSearch() {
      searchUser(this.userFrom).then((res) => {
        console.log(res.data.data);
        this.tableData = res.data.data.data;
        // this.getList();
      });
    },
    //分页
    handlePage(val) {
      console.log(val);
    },
    // 获取数据
    getList() {
      // 采用封装的api接口方法
      getPlace().then((res) => {
        console.log("11", res.data.data.data);
        this.tableData = res.data.data.data;
      });
    },
  },
  // 挂载
  mounted() {
    this.getList();
    this.getOpstions();
  },
};
</script>

<style lang="less">
.messge {
  height: 90%;
  .common-table {
    position: relative;
    height: calc(100% - 62px);
    .page {
      position: absolute;
      bottom: 0;
      right: 20px;
    }
  }
  .manage-hander {
    display: flex;
    justify-content: space-between;
    align-items: center;
  }
}
.el-table .cell {
  img {
    height: 50rpx;
  }
}
</style>

8、源码获取

大家点赞、收藏、关注、评论啦 、查看??获取联系方式??

?写在最后

?伙伴们,如果你已经看到了这里,觉得这篇文章有帮助到你的话不妨点赞?或 Star ✨支持一下哦!手动码字,如有错误,欢迎在评论区指正?~

你的支持就是我更新的最大动力?~

显示全文