Skip to content

Commit

Permalink
feat: ユーザー追加機能の追加
Browse files Browse the repository at this point in the history
  • Loading branch information
testusuke committed May 20, 2024
1 parent d273466 commit 5f77e93
Show file tree
Hide file tree
Showing 3 changed files with 165 additions and 4 deletions.
8 changes: 5 additions & 3 deletions app/(authenticated)/teams/[id]/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,8 @@ import {classFactory} from "@/src/models/ClassModel";
export default async function TeamDetailPage({ params }: { params: { id: string } }) {
const teamId = parseInt(params.id, 10)
const teamInfo = await teamFactory().show(teamId)
const classes = await classFactory().show(teamInfo.classId)
const classModel = await classFactory().show(teamInfo.classId)
const classUsers = await classFactory().getUsers(teamInfo.classId)
const teamUsers = await teamFactory().getTeamUsers(teamId);

return (
Expand All @@ -17,7 +18,7 @@ export default async function TeamDetailPage({ params }: { params: { id: string
<Link underline="hover" color="inherit" href="/">
管理者のダッシュボード
</Link>
<Link underline="hover" color="inherit" href="/teams/">
<Link underline="hover" color="inherit" href={"/teams/"}>
チーム管理
</Link>
<Typography color="text.primary">{teamInfo.name}</Typography>
Expand All @@ -27,7 +28,8 @@ export default async function TeamDetailPage({ params }: { params: { id: string
<TeamEditor
team={teamInfo}
teamUser={teamUsers}
class={classes}
class={classModel}
classUsers={classUsers}
/>
</CardBackground>
</Stack>
Expand Down
131 changes: 131 additions & 0 deletions components/teams/addTeamMemberDialog.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
'use client'
import {AgGridReact} from 'ag-grid-react';
import "ag-grid-community/styles/ag-grid.css"; // Mandatory CSS required by the grid
import "ag-grid-community/styles/ag-theme-quartz.css"; // Optional Theme applied to the grid
import {ColDef, GridApi, GridReadyEvent, ModuleRegistry} from 'ag-grid-community';
import {ClientSideRowModelModule} from 'ag-grid-community';
import {Button, Dialog, DialogActions, DialogContent, DialogTitle, Typography} from "@mui/material";
import {User} from "@/src/models/UserModel";
import {Team, teamFactory} from "@/src/models/TeamModel";
import {useState} from "react";
import {useRouter} from "next/navigation";

ModuleRegistry.registerModules([ClientSideRowModelModule]);

export type AddTeamMemberDialogProps = {
isOpen: boolean
setClose: () => void
team: Team
users: User[]
}

// Row Data Interface
type IRow = {
id: number,
username: string,
gender: string,
emailAccountName: string,
}

export default function AddTeamMemberDialog(props: AddTeamMemberDialogProps) {
const router = useRouter()
const height = 'calc(100vh - 230px)';
const [gridApi, setGridApi] = useState<GridApi<IRow> | null>(null)
// Column Definitions: Defines & controls grid columns.
const colDefs: ColDef<IRow>[] = [
{
field: "id",
headerName: "ID",
headerCheckboxSelection: true,
checkboxSelection: true,
showDisabledCheckboxes: true,
},
{field: "username", headerName: "ユーザー名"},
{field: "gender", headerName: "性別"},
{field: "emailAccountName", headerName: "学籍番号"},
]

const rowData: IRow[] = props.users.map((user) => {
return {
id: user.id,
username: user.name,
gender: user.gender == "male" ? "男" : "女",
emailAccountName: user.email.split("@")[0]
} as IRow
})

const handleGridReady = (params: GridReadyEvent) => {
setGridApi(params.api)
}

const handleSubmit = async () => {
if (!gridApi) {
alert("エラーが発生しました")
return
}

const selectedRowIds = gridApi
.getSelectedRows()
.map((row) => row.id)

if (selectedRowIds.length === 0) {
alert("ユーザーを選択してください")
return
}

// add team user
await teamFactory().addTeamUsers(props.team.id, selectedRowIds)

// refresh
router.refresh()
// close
props.setClose()
}

return (
<Dialog
open={props.isOpen}
onClose={props.setClose}
maxWidth={"lg"}
fullWidth
>
<DialogTitle>
<Typography>
チームメンバーの追加
</Typography>
</DialogTitle>
<DialogContent>
<div
className={"ag-theme-quartz"}
style={{
width: '100%',
height: height,
borderRadius: "10px"
}}
>
<AgGridReact
rowData={rowData}
columnDefs={colDefs}
rowSelection={"multiple"}
onGridReady={handleGridReady}
/>
</div>
</DialogContent>
<DialogActions>
<Button
variant={"outlined"}
onClick={props.setClose}
>
キャンセル
</Button>
<Button
type={"submit"}
variant={"contained"}
onClick={handleSubmit}
>
追加
</Button>
</DialogActions>
</Dialog>
)
}
30 changes: 29 additions & 1 deletion components/teams/teamEditor.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,23 +6,26 @@ import {
Stack, Table, TableBody, TableCell, TableContainer, TableHead, TableRow, TextField,
Typography
} from "@mui/material";
import {HiCheck, HiMiniTrash} from "react-icons/hi2";
import {HiCheck, HiMiniTrash, HiPlus} from "react-icons/hi2";
import React, {useState} from "react";
import {useRouter} from "next/navigation";
import {Team, teamFactory} from "@/src/models/TeamModel";
import {Class} from "@/src/models/ClassModel";
import {User} from "@/src/models/UserModel";
import TeamDelete from "@/components/teams/teamDelete";
import AddTeamMemberDialog from "@/components/teams/addTeamMemberDialog";

type TeamEditorProps = {
class: Class;
team: Team;
teamUser: User[];
classUsers: User[];
}

export default function TeamEditor(props: TeamEditorProps) {
const router = useRouter()
const [teamName, setTeamName] = useState<string>(props.team.name)
const [isOpenTeamMemberAddDialog, setIsOpenTeamMemberAddDialog] = useState<boolean>(false)

const handleSubmit = async () => {
await teamFactory().update(props.team.id, {
Expand All @@ -43,6 +46,10 @@ export default function TeamEditor(props: TeamEditorProps) {
router.refresh()
}

const notTeamUsers = props.classUsers.filter((user) => {
return !props.teamUser.some((teamUser) => teamUser.id === user.id)
})

return (
<>
<Stack mx={0} my={2} spacing={2} direction={"column"}>
Expand Down Expand Up @@ -153,6 +160,27 @@ export default function TeamEditor(props: TeamEditorProps) {
</TableContainer>
</FormControl>

<AddTeamMemberDialog
isOpen={isOpenTeamMemberAddDialog}
setClose={() => {
setIsOpenTeamMemberAddDialog(false)
}}
team={props.team}
users={notTeamUsers}
/>

<Button
variant={"contained"}
color={"primary"}
sx={{flexGrow: 3}}
startIcon={<HiPlus/>}
onClick={() => {
setIsOpenTeamMemberAddDialog(true)
}}
>
チームメンバー追加
</Button>

<Stack
direction={"row"}
my={0.5}
Expand Down

0 comments on commit 5f77e93

Please sign in to comment.