目录

123.会议室预订系统会议室管理模块-用户端前端开发

目录

这节来写用户端的会议室列表:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/5a6f760e036d49bab8ff8191b8c5a8ff~tplv-k3u1fbpfcp-watermark.image?

现在,用户端首页是这样的:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/0fcfd2a1c94b4a2299380e024f721234~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1858&h=1090&s=79837&e=png&b=ffffff

需要在 / 下添加一个二级路由:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/2686beddeddc481a928da0208d7bd2b1~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=864&h=1082&s=137849&e=png&b=202020

{
    path: '/',
    element: <Menu/>,
    children: [
      {
        path: '/',
        element: <MeetingRoomList/>
      },
      {
        path: 'meeting_room_list',
        element: <MeetingRoomList/>
      },
      {
        path: 'booking_history',
        element: <BookingHistory/>
      }
    ]
}

然后分别实现这三个组件:

src/page/menu/Menu.tsx

import { Outlet, useLocation } from "react-router-dom";
import { Menu as AntdMenu, MenuProps } from 'antd';
import './menu.css';
import { MenuClickEventHandler } from "rc-menu/lib/interface";
import { router } from "../..";

const items: MenuProps['items'] = [
    {
        key: '1',
        label: "会议室列表"
    },
    {
        key: '2',
        label: "预定历史"
    }
];

const handleMenuItemClick: MenuClickEventHandler = (info) => {
    let path = '';
    switch(info.key) {
        case '1':
            path = '/meeting_room_list';
            break;
        case '2':
            path = '/booking_history';
            break;              
    }
    router.navigate(path);
}


export function Menu() {

    const location = useLocation();

    function getSelectedKeys() {
        if(location.pathname === '/meeting_room_list') {
            return ['1']
        } else if(location.pathname === '/booking_history') {
            return ['2']
        } else {
            return ['1']
        }
    }

    return <div id="menu-container">
        <div className="menu-area">
            <AntdMenu
                defaultSelectedKeys={getSelectedKeys()}
                items={items}
                onClick={handleMenuItemClick}
            />
        </div>
        <div className="content-area">
            <Outlet></Outlet>
        </div>
    </div>
}

引入 antd 的 Menu 实现菜单。

渲染的时候根据 useLocation 拿到的 pathname 来设置选中的菜单项。

点击菜单项的时候用 router.push 修改路径。

这里用到的 router 需要在 index.tsx 导出:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/9ca0768b34944f0998bf1ac57f3a2d4f~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=912&h=474&s=92107&e=png&b=1f1f1f

这些我们前面写过一遍。

menu.css 如下:

#menu-container {
    display: flex;
    flex-direction: row;
}
#menu-container .menu-area {
    width: 200px;
}

然后是 src/pages/meeting_room_list/MeetingRoomList.tsx

export function MeetingRoomList() {
    return <div>MeetingRoomList</div>
}

还有 src/pages/booking_history/BookingHistory.tsx

export function BookingHistory() {
    return <div>BookingHistory</div>
}

在 index.tsx 里导入这些组件后,我们跑起来看看:

npm run start:dev

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/446fb7e3ff6a47ecacba283b2a8a1a96~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1262&h=612&s=179976&e=gif&f=28&b=fefefe

点击菜单项的路由切换,以及刷新选中对应菜单项,都没问题。

然后来写下列表页面,其实这个和管理端的会议室列表差不多:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/5a6f760e036d49bab8ff8191b8c5a8ff~tplv-k3u1fbpfcp-watermark.image?

我们把那个复制过来改改。

首先,在 interfaces.ts 添加用到的接口:

export async function searchMeetingRoomList(name: string, capacity: number, equipment: string, pageNo: number, pageSize: number) {
    return await axiosInstance.get('/meeting-room/list', {
        params: {
            name,
            capacity,
            equipment,
            pageNo,
            pageSize
        }
    });
}

然后写下列表:

import { Badge, Button, Form, Input, Popconfirm, Table, message } from "antd";
import { useCallback, useEffect, useMemo, useState } from "react";
import './meeting_room_list.css';
import { ColumnsType } from "antd/es/table";
import { useForm } from "antd/es/form/Form";
import { searchMeetingRoomList } from "../../interface/interfaces";

interface SearchMeetingRoom {
    name: string;
    capacity: number;
    equipment: string;
}

interface MeetingRoomSearchResult {
    id: number,
    name: string;
    capacity: number;
    location: string;
    equipment: string;
    description: string;
    isBooked: boolean;
    createTime: Date;
    updateTime: Date;
}

export function MeetingRoomList() {
    const [pageNo, setPageNo] = useState<number>(1);
    const [pageSize, setPageSize] = useState<number>(10);

    const [meetingRoomResult, setMeetingRoomResult] = useState<Array<MeetingRoomSearchResult>>([]);

    const columns: ColumnsType<MeetingRoomSearchResult> = useMemo(() => [
        {
            title: '名称',
            dataIndex: 'name'
        },
        {
            title: '容纳人数',
            dataIndex: 'capacity',
        },
        {
            title: '位置',
            dataIndex: 'location'
        },
        {
            title: '设备',
            dataIndex: 'equipment'
        },
        {
            title: '描述',
            dataIndex: 'description'
        },
        {
            title: '添加时间',
            dataIndex: 'createTime'
        },
        {
            title: '上次更新时间',
            dataIndex: 'updateTime'
        },
        {
            title: '预定状态',
            dataIndex: 'isBooked',
            render: (_, record) => (
                record.isBooked ? <Badge status="error">已被预订</Badge> : <Badge status="success">可预定</Badge>
            )
        },
        {
            title: '操作',
            render: (_, record) => (
                <div>
                    <a href="#">预定</a>
                </div>
            )
        }
    ], []);

    const searchMeetingRoom = useCallback(async (values: SearchMeetingRoom) => {
        const res = await searchMeetingRoomList(values.name, values.capacity, values.equipment, pageNo, pageSize);

        const { data } = res.data;
        if(res.status === 201 || res.status === 200) {
            setMeetingRoomResult(data.meetingRooms.map((item: MeetingRoomSearchResult) => {
                return {
                    key: item.id,
                    ...item
                }
            }))
        } else {
            message.error(data || '系统繁忙,请稍后再试');
        }
    }, []);

    const [form ]  = useForm();

    useEffect(() => {
        searchMeetingRoom({
            name: form.getFieldValue('name'),
            capacity: form.getFieldValue('capacity'),
            equipment: form.getFieldValue('equipment')
        });
    }, [pageNo, pageSize]);

    const changePage = useCallback(function(pageNo: number, pageSize: number) {
        setPageNo(pageNo);
        setPageSize(pageSize);
    }, []);

    return <div id="meetingRoomList-container">
        <div className="meetingRoomList-form">
            <Form
                form={form}
                onFinish={searchMeetingRoom}
                name="search"
                layout='inline'
                colon={false}
            >
                <Form.Item label="会议室名称" name="name">
                    <Input />
                </Form.Item>

                <Form.Item label="容纳人数" name="capacity">
                    <Input />
                </Form.Item>

                <Form.Item label="设备" name="equipment">
                    <Input/>
                </Form.Item>

                <Form.Item label=" ">
                    <Button type="primary" htmlType="submit">
                        搜索会议室
                    </Button>
                </Form.Item>
            </Form>
        </div>
        <div className="meetingRoomList-table">
            <Table columns={columns} dataSource={meetingRoomResult} pagination={ {
                current: pageNo,
                pageSize: pageSize,
                onChange: changePage
            }}/>
        </div>
    </div>
}

上面是 form、下面是 table。

调用搜索接口来搜索列表数据,然后设置到 table 的 dataSource。

每次分页变化的时候重新搜索。

然后 css 部分如下:

#meetingRoomList-container {
    padding: 20px;
}
#meetingRoomList-container .meetingRoomList-form {
    margin-bottom: 40px;
}

这样,列表页就完成了:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e03f191acba548c080c6df79d6ba12ad~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2282&h=996&s=369175&e=gif&f=28&b=fdfdfd

其实写这个模块的时候偷懒了,应该是写完后端接口,还要写 swager 文档。

然后前端根据 swagger 接口文档才能知道传啥参数,有啥返回值。

当时我们没写 swagger 文档,现在补一下:

打开后端项目,在 MeetingRoomController 里加一下 swagger 相关的装饰器:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f0a9c5aea2d54195a5782aca77447257~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=876&h=488&s=82012&e=png&b=1f1f1f

首先加一下 delete 接口的:

@ApiParam({
    name: 'id',
    type: Number,
    description: 'id'
})
@ApiResponse({
    status: HttpStatus.OK,
    description: 'success'
})

访问 http://localhost:3005/api-doc 可以看到这个接口的文档:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/a8b963c048bf443e960e6bf9a57ed815~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=730&h=920&s=57129&e=png&b=f9eeed

其实会议室的接口都是需要登录才能访问的,当时为了测试方便没有加,现在加一下:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/120d7fdaeeda4067af9c813137c518f2~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=772&h=600&s=83483&e=png&b=1f1f1f

添加 @RequireLogin 装饰器,标识接口需要登录。

并且添加对应的 @ApiBearerAuth 的 swagger 装饰器,代表需要添加 Bearer 的 header。

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/b2e9f334c1dc4d49b597ae390b8ed7d8~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1768&h=808&s=210740&e=gif&f=28&b=fbf0ed

我们现在 postman 里测试下:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e2647600e3294667988d07e60f10a32d~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1008&h=710&s=73376&e=png&b=fcfcfc

这时候直接调用 delete 接口就会提示需要先登录了。

然后我们登录下,拿到 token。

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/1878903a2706490ab7549b401dac94b8~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1280&h=964&s=192034&e=png&b=fdfdfd

把它复制到 swagger 文档这里:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e84ac199636344dda2096a6781adea5b~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1880&h=708&s=98312&e=png&b=312f2e

然后点击这个 try it out:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/c710fcb942834328ad86aefbe95abdcf~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1966&h=630&s=58947&e=png&b=f9eeec

数据库中现在有 3 条记录:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/94cfeff6d31b4265abb2a106531b8591~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=994&h=262&s=85140&e=png&b=f9f9f9

把 id 为 10 那条删掉。

点击 execute: https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/72aeb9819eb54d2987759687846b06c3~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1584&h=730&s=60955&e=png&b=f9eeed

swagger 会发送请求,下面会打印响应:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/a4b1c8cba1fb48bdb045d18c14ca6aa3~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1548&h=1018&s=172362&e=png&b=333333

这时数据库里就没有这条记录了: https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/0fd6d061c25244e6b180dfd939c05adf~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=772&h=282&s=71192&e=png&b=f7f7f7

可以直接在 swagger 文档里测试接口,不用 postman 也行。

然后继续写下个接口的 swagger 文档:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/20d1107d67f743139c2c218b7a90194d~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=820&h=156&s=31995&e=png&b=1f1f1f

这个接口的参数也是用 @ApiParam 标识,但它的响应不是 string,而是 MeetingRoom。

而我们现在并没有 vo,没地方标识属性:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/b87f13f1aafb4e5a8af14f89143f37bc~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=634&h=200&s=25174&e=png&b=1f1f1f

所以要创建个 vo:

新建 src/meeting-room/vo/meeting-room.vo.ts

import { ApiProperty } from "@nestjs/swagger";

export class MeetingRoomVo {
    
    @ApiProperty()
    id: number;

    @ApiProperty()
    name: string;

    @ApiProperty()
    capacity: number;

    @ApiProperty()
    location: string;

    @ApiProperty()
    equipment: string;

    @ApiProperty()
    description: string;

    @ApiProperty()
    isBooked: boolean;

    @ApiProperty()
    createTime: Date;

    @ApiProperty()
    updateTime: Date;
}

然后加一下 swagger 的装饰器:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/403bf1d50ff2435cb173313a7e33db2a~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=884&h=526&s=86522&e=png&b=1f1f1f

@ApiBearerAuth()
@ApiParam({
    name: 'id',
    type: Number,
})
@ApiResponse({
    status: HttpStatus.OK,
    description: 'success',
    type: MeetingRoomVo
})

试一下:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/37136786e63d4977b0de06c030d9a91a~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1200&h=1430&s=159801&e=png&b=eef4fa

接下来是 update 接口:

他有两种响应:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/be0633b63e4f4b81971024b773fdb6b2~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=896&h=900&s=169239&e=png&b=1f1f1f

分别写一下: https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/ef54451d60544fd7b16a6cf32db46bd1~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=954&h=608&s=111879&e=png&b=1f1f1f

@ApiBearerAuth()
@ApiBody({
    type: UpdateMeetingRoomDto,
})
@ApiResponse({
    status: HttpStatus.BAD_REQUEST,
    description: '会议室不存在'
})
@ApiResponse({
    status: HttpStatus.OK,
    description: 'success'
})

然后在 dto 里标注下属性:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e0912339cf2d4a3a84550b6e1f64483e~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1224&h=468&s=109922&e=png&b=1f1f1f

因为 update 的 dto 继承了 create 的 dto,所以那里也要加一下:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e565e7676c00472b842aeb9ce9b36623~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=996&h=1268&s=205871&e=png&b=1f1f1f

这样 swagger 文档就对了:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/4c2adc8415da42868092c8ee1b4fed81~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=952&h=1296&s=110208&e=png&b=faf4ee

然后是 create 接口:

postman 里调用下是这样的:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/5339becd99cd4d6ebe127cef968e1c8c~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=916&h=1128&s=146290&e=png&b=fdfdfd

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/df30ee02c0cb46509d4adcee93299775~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=858&h=878&s=96535&e=png&b=fcfcfc

所以 swagger 装饰器这样写:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/7cd2bb760ccd4d0f8ada447247032729~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=978&h=620&s=113340&e=png&b=1f1f1f

@ApiBearerAuth()
@ApiBody({
    type: CreateMeetingRoomDto,
})
@ApiResponse({
    status: HttpStatus.BAD_REQUEST,
    description: '会议室名字已存在'
})
@ApiResponse({
    status: HttpStatus.OK,
    type: MeetingRoomVo
})

这样 swagger 文档显示的就对了:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/22530084774640bc95d5bddce063347b~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=968&h=1484&s=169359&e=png&b=eff7f3

然后还有最后一个 list 接口:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/786fa3b2fb2c4245a782810e3c853f2f~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1342&h=410&s=117414&e=png&b=1f1f1f

它的响应是这样的:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/4e5d0a05f5ba42c5bcb165a5284042bd~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1196&h=1280&s=152984&e=png&b=fdfdfd

首先创建响应数据的 vo:

src/meeting-room/vo/meeting-room-list.vo.ts

import { ApiProperty } from "@nestjs/swagger";
import { MeetingRoomVo } from "./meeting-room.vo";

export class MeetingRoomListVo {

    @ApiProperty({
        type: [MeetingRoomVo]
    })
    users: Array<MeetingRoomVo>;

    @ApiProperty()
    totalCount: number;
}

然后加一下 swagger 的装饰器:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/675a8cca8ac94b0089e77dbc4cab6cd2~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1170&h=1174&s=171865&e=png&b=1f1f1f

@ApiBearerAuth()
@ApiQuery({
    name: 'pageNo',
    type: Number,
    required: false
})
@ApiQuery({
    name: 'pageSize',
    type: Number,
    required: false
})
@ApiQuery({
    name: 'name',
    type: String,
    required: false
})
@ApiQuery({
    name: 'capacity',
    type: String,
    required: false
})
@ApiQuery({
    name: 'equipment',
    type: String,
    required: false
})
@ApiResponse({
    type: MeetingRoomListVo
})

有同学说,不用把 service 里的返回值改成 MeetingRoomListVo 对象么?

不用,只要结构对上就行。

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/00ce922e32a7473096d86a179df7bdbc~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=938&h=934&s=81975&e=png&b=eef4fa

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/1707ae9b1ad9415e8f62f5eaa10670c2~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=834&h=882&s=103636&e=png&b=eef4fa

最后,在 controller 上加上个 @ApiTags,把下面的接口分到单独一组:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/a8ffeb13d85243c0b69015d3a0241ed0~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=626&h=170&s=41061&e=png&b=1f1f1f

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/31ce4c320acc4e4f86df0cc125983104~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=780&h=608&s=59113&e=png&b=f8f4f4

这样,用户端的会议室列表页面,swagger 文档就都完成了。

案例代码上传了小册仓库:

前端代码

后端代码

总结

这节我们写了用户端的会议室列表页,并且补了 swagger 文档。

用户端列表页就是调用 list 接口,通过 form 来填写参数,通过 table 展示结果。

swagger 文档部分就是分别通过 @ApiPram @ApiQuery @ApiBody @ApiResponse 标识接口,通过 @ApiProperty 标识 dto 和 vo 的属性。

这样,会议室模块的前端后端就都完成了。