Skip to content

Commit

Permalink
Merge pull request #51 from DDD-Community/feat/#46
Browse files Browse the repository at this point in the history
[feat/#46] 카메라, 알림 권한 허용 여부에 따른 메세지 및 스타일 추가
  • Loading branch information
lkhoony authored Sep 9, 2024
2 parents b666f74 + d772584 commit a542392
Show file tree
Hide file tree
Showing 13 changed files with 220 additions and 312 deletions.
35 changes: 31 additions & 4 deletions src/components/Camera.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import React, { useRef, useEffect } from "react"
import { useCameraPermission } from "@/hooks/useCameraPermission" // 커스텀 훅을 가져옵니다.

interface CameraProps {
detectStart: (video: HTMLVideoElement) => void
Expand All @@ -9,6 +10,7 @@ export default function Camera(props: CameraProps): React.ReactElement {
const { detectStart, canvasRef } = props
const videoRef = useRef<HTMLVideoElement>(null)

// 비디오를 시작하는 함수
const startVideo = (): void => {
navigator.mediaDevices
.getUserMedia({
Expand Down Expand Up @@ -38,9 +40,34 @@ export default function Camera(props: CameraProps): React.ReactElement {
})
}

// 비디오를 중지하는 함수
const stopVideo = (): void => {
if (videoRef.current && videoRef.current.srcObject) {
const stream = videoRef.current.srcObject as MediaStream
const tracks = stream.getTracks()

tracks.forEach((track) => {
track.stop() // 모든 트랙 중지
})

videoRef.current.srcObject = null // 비디오 스트림 초기화
}
}

// 커스텀 훅을 사용해 권한 상태 확인
const { hasPermission, isPermissionDenied } = useCameraPermission()

useEffect(() => {
startVideo()
}, [])
if (hasPermission) {
startVideo()
} else if (isPermissionDenied) {
stopVideo()
}

return () => {
stopVideo() // 컴포넌트가 언마운트될 때 비디오 중지
}
}, [hasPermission, isPermissionDenied])

return (
<div
Expand All @@ -50,9 +77,9 @@ export default function Camera(props: CameraProps): React.ReactElement {
position: "relative",
}}
>
<div style={{ position: "relative", width: "100%", height: "100%" }}>
<div className="rounded-3xl bg-[#787870]/20" style={{ position: "relative", width: "100%", height: "100%" }}>
<video
className="rounded-lg"
className="rounded-3xl"
ref={videoRef}
style={{
position: "absolute",
Expand Down
306 changes: 33 additions & 273 deletions src/components/Crew/CrewList.tsx
Original file line number Diff line number Diff line change
@@ -1,177 +1,10 @@
// import { useState, useRef, useEffect, ReactElement } from "react"
// import CreateCrewIcon from "@assets/icons/crew-create-button-icon.svg?react"
// import SortCrewIcon from "@assets/icons/crew-sort-icon.svg?react"
// import PrivateCrewIcon from "@assets/icons/crew-private-icon.svg?react"
// import CrewUserIcon from "@assets/icons/crew-user-icon.svg?react"
// import EmptyCrewImage from "@/assets/images/crew-empty.png"
// import { useGetGroups } from "@/hooks/useGroupMutation"
// import { group, groupsReq, sort } from "@/api"
// import CrewItem from "@/components/Crew/CrewItem"
// import JoinCrewModal from "@/components/Modal/JoinCrewModal"
// import CreateCrewModal from "@/components/Modal/CreateCrewModal"
// import InviteCrewModal from "@/components/Modal/InviteCrewModal"
// import WithdrawCrewModal from "@/components/Modal/WithdrawCrewModal"

// const SORT_LIST = [
// { sort: "userCount,desc", label: "크루원 많은 순" },
// { sort: "createdAt,desc", label: "최신 생성 크루 순" },
// ]

// const CrewList = (): ReactElement => {
// const [sort, setSort] = useState<number>(0)
// const [isDropdownOpen, setIsDropdownOpen] = useState<boolean>(false)
// const [isJoinModalOpen, setIsJoinModalOpen] = useState<boolean>(false)
// const [isCreateModalOpen, setIsCreateModalOpen] = useState<boolean>(false)
// const [isInviteModalOpen, setIsInviteModalOpen] = useState<boolean>(false)

// const [params] = useState<groupsReq>({
// page: 0,
// size: 10,
// sort: "userCount,desc",
// })

// const { data, isLoading, isError } = useGetGroups(params)

// const openJoinModal = (): void => setIsJoinModalOpen(true)
// const closeJoinModal = (): void => setIsJoinModalOpen(false)

// const openCreateModal = (): void => setIsCreateModalOpen(true)
// const closeCreateModal = (): void => setIsCreateModalOpen(false)

// const openInviteModal = (): void => setIsInviteModalOpen(true)
// const closeInviteModal = (): void => setIsInviteModalOpen(false)

// const dropdownRef = useRef<HTMLDivElement>(null)

// const toggleDropdown = (): void => {
// setIsDropdownOpen((prev) => !prev)
// }

// const createSortList = (): JSX.Element[] => {
// return SORT_LIST.map((s, i) => (
// // eslint-disable-next-line max-len
// <div
// key={`sort-list-${s.sort}`}
// className="cursor-pointer text-[13px] font-medium leading-[24px] text-zinc-400"
// onClick={() => {
// setSort(i)
// setIsDropdownOpen(false)
// }}
// >
// {s.label}
// </div>
// ))
// }

// const createGroupList = (_groups: group[] | undefined): JSX.Element | null => {
// if (!_groups) return null

// if (_groups.length === 0) {
// return (
// <div className="flex flex-grow flex-col items-center justify-center">
// <img src={EmptyCrewImage} alt="empty crew" />
// <div className="text-center text-[14px] font-semibold leading-[22px]">{"만들어진 크루가 아직 없습니다."}</div>
// </div>
// )
// }
// return (
// <div className="flex flex-grow flex-col gap-[8px]">
// {_groups.map((g) => (
// <CrewItem key={`crew-item-${g.id}`} group={g} onClickDetail={openJoinModal} />
// ))}
// </div>
// )
// }

// useEffect(() => {
// const handleClickOutside = (event: MouseEvent): void => {
// if (dropdownRef.current && !dropdownRef.current.contains(event.target as HTMLElement)) {
// setIsDropdownOpen(false)
// }
// }

// document.addEventListener("mousedown", handleClickOutside)
// return () => {
// document.removeEventListener("mousedown", handleClickOutside)
// }
// }, [dropdownRef])

// return (
// <div className="flex h-full w-full flex-col">
// {/* header */}
// <div className="mb-[24px] flex w-full items-center">
// <div className="flex-grow text-[22px] font-bold text-zinc-900">전체크루(0)</div>
// <div
// className="flex w-[138px] cursor-pointer items-center justify-center gap-[10px] rounded-[33px] bg-zinc-800 p-[10px] text-sm font-semibold text-white"
// onClick={openCreateModal}
// >
// <CreateCrewIcon />
// <div>크루 만들기</div>
// </div>
// </div>

// {/* sort */}
// <div className="relative mb-[12px] text-sm font-medium text-zinc-500" ref={dropdownRef}>
// <div className="flex cursor-pointer items-center" onClick={toggleDropdown}>
// <SortCrewIcon />
// <div>{SORT_LIST[sort].label}</div>
// </div>

// {/* dropdown */}
// <div
// className={`absolute flex transform flex-col items-center justify-center gap-2 rounded-lg bg-white px-4 py-3 shadow-lg transition-all duration-300 ${
// isDropdownOpen
// ? "translate-y-0 scale-100 opacity-100"
// : "pointer-events-none -translate-y-2 scale-95 opacity-0"
// }`}
// >
// {createSortList()}
// </div>
// </div>

// {/* list */}
// <div className="flex flex-grow flex-col gap-[8px]">
// <div
// style={{
// borderRadius: "12px",
// border: "1px solid #E5E7EB",
// }}
// className="flex w-full items-center gap-[24px] bg-white px-[24px] py-[11px] text-[14px] font-semibold leading-[32px]"
// >
// {/* crew name */}
// <div className="flex items-center gap-[6px]">
// <PrivateCrewIcon />
// <div>{"개발자들 다 모여"}</div>
// </div>
// {/* crew user cnt */}
// <div className="flex flex-grow items-center gap-[6px]">
// <CrewUserIcon />
// <div>{`10/30명`}</div>
// </div>
// {/* detail button */}
// <div className="flex cursor-pointer rounded-full bg-[#1A75FF] px-[22px] py-[6px] text-sm font-semibold text-white">
// 크루 상세보기
// </div>
// </div>
// </div>

// {isLoading ? "로딩 중입니다..." : isError ? "데이터를 불러오는데 실패했습니다." : createGroupList(data?.data)}
// <div onClick={openJoinModal}>asas</div>
// <div onClick={openInviteModal}>asas</div>
// <JoinCrewModal isOpen={isJoinModalOpen} onClose={closeJoinModal} />
// <CreateCrewModal isOpen={isCreateModalOpen} onClose={closeCreateModal} />
// <WithdrawCrewModal isOpen={isInviteModalOpen} onClose={closeInviteModal} />
// </div>
// )
// }

// export default CrewList

import { useState, useRef, useEffect, ReactElement } from "react"
import CreateCrewIcon from "@assets/icons/crew-create-button-icon.svg?react"
import SortCrewIcon from "@assets/icons/crew-sort-icon.svg?react"
import PrivateCrewIcon from "@assets/icons/crew-private-icon.svg?react"
import CrewUserIcon from "@assets/icons/crew-user-icon.svg?react"
import EmptyCrewImage from "@/assets/images/crew-empty.png"
import { useGetGroups } from "@/hooks/useGroupMutation"
import { group, groupsReq } from "@/api"
import CrewItem from "@/components/Crew/CrewItem"
import JoinCrewModal from "@/components/Modal/JoinCrewModal"
import CreateCrewModal from "@/components/Modal/CreateCrewModal"

Expand All @@ -186,6 +19,14 @@ const CrewList = (): ReactElement => {
const [isJoinModalOpen, setIsJoinModalOpen] = useState<boolean>(false)
const [isCreateModalOpen, setIsCreateModalOpen] = useState<boolean>(false)

const [params] = useState<groupsReq>({
page: 0,
size: 10,
sort: "userCount,desc",
})

const { data, isLoading, isError } = useGetGroups(params)

const openJoinModal = (): void => setIsJoinModalOpen(true)
const closeJoinModal = (): void => setIsJoinModalOpen(false)

Expand Down Expand Up @@ -214,6 +55,26 @@ const CrewList = (): ReactElement => {
))
}

const createGroupList = (_groups: group[] | undefined): JSX.Element | null => {
if (!_groups) return null

if (_groups.length === 0) {
return (
<div className="flex flex-grow flex-col items-center justify-center">
<img src={EmptyCrewImage} alt="empty crew" />
<div className="text-center text-[14px] font-semibold leading-[22px]">{"만들어진 크루가 아직 없습니다."}</div>
</div>
)
}
return (
<div className="flex flex-grow flex-col gap-[8px]">
{_groups.map((g) => (
<CrewItem key={`crew-item-${g.id}`} group={g} onClickDetail={openJoinModal} />
))}
</div>
)
}

useEffect(() => {
const handleClickOutside = (event: MouseEvent): void => {
if (dropdownRef.current && !dropdownRef.current.contains(event.target as HTMLElement)) {
Expand Down Expand Up @@ -261,108 +122,7 @@ const CrewList = (): ReactElement => {
</div>

{/* list */}
<div className="flex flex-grow flex-col gap-[8px]">
<div
style={{
borderRadius: "12px",
border: "1px solid #E5E7EB",
}}
className="flex w-full items-center gap-[24px] bg-white px-[24px] py-[11px] text-[14px] font-semibold leading-[32px]"
>
{/* crew name */}
<div className="flex items-center gap-[6px]">
<PrivateCrewIcon />
<div>{"개발자들 다 모여"}</div>
</div>
{/* crew user cnt */}
<div className="flex flex-grow items-center gap-[6px]">
<CrewUserIcon />
<div>{`10/30명`}</div>
</div>
{/* detail button */}
<button
className="flex w-[114px] cursor-pointer justify-center rounded-full bg-[#1A75FF] py-[6px] text-sm font-semibold text-white"
onClick={openJoinModal}
>
크루 상세보기
</button>
</div>
<div
style={{
borderRadius: "12px",
border: "1px solid #E5E7EB",
}}
className="flex w-full items-center gap-[24px] bg-white px-[24px] py-[11px] text-[14px] font-semibold leading-[32px]"
>
{/* crew name */}
<div className="flex items-center gap-[6px]">
<PrivateCrewIcon />
<div>{"개발자들 다 모여"}</div>
</div>
{/* crew user cnt */}
<div className="flex flex-grow items-center gap-[6px]">
<CrewUserIcon />
<div>{`10/30명`}</div>
</div>
{/* detail button */}
<button
className="flex w-[114px] cursor-pointer justify-center rounded-full bg-[#1A75FF] py-[6px] text-sm font-semibold text-white"
onClick={openJoinModal}
>
크루 상세보기
</button>
</div>
<div
style={{
borderRadius: "12px",
border: "1px solid #E5E7EB",
}}
className="flex w-full items-center gap-[24px] bg-white px-[24px] py-[11px] text-[14px] font-semibold leading-[32px]"
>
{/* crew name */}
<div className="flex items-center gap-[6px]">
<PrivateCrewIcon />
<div>{"개발자들 다 모여"}</div>
</div>
{/* crew user cnt */}
<div className="flex flex-grow items-center gap-[6px]">
<CrewUserIcon />
<div>{`10/30명`}</div>
</div>
{/* detail button */}
<button
className="flex w-[114px] cursor-pointer justify-center rounded-full bg-zinc-800 py-[6px] text-sm font-semibold text-white"
onClick={openJoinModal}
>
나의 크루
</button>
</div>
<div
style={{
borderRadius: "12px",
border: "1px solid #E5E7EB",
}}
className="flex w-full items-center gap-[24px] bg-white px-[24px] py-[11px] text-[14px] font-semibold leading-[32px]"
>
{/* crew name */}
<div className="flex items-center gap-[6px]">
<PrivateCrewIcon />
<div>{"개발자들 다 모여"}</div>
</div>
{/* crew user cnt */}
<div className="flex flex-grow items-center gap-[6px]">
<CrewUserIcon />
<div>{`10/30명`}</div>
</div>
{/* detail button */}
<button
className="flex w-[114px] justify-center rounded-full bg-gray-200 py-[6px] text-sm font-semibold text-white"
disabled={true}
>
크루 상세보기
</button>
</div>
</div>
{isLoading ? "로딩 중입니다..." : isError ? "데이터를 불러오는데 실패했습니다." : createGroupList(data?.data)}
<JoinCrewModal isOpen={isJoinModalOpen} onClose={closeJoinModal} id={0} onSubmit={(): void => {}} />
<CreateCrewModal isOpen={isCreateModalOpen} onClose={closeCreateModal} onSubmit={(): void => {}} />
</div>
Expand Down
Loading

0 comments on commit a542392

Please sign in to comment.