rht
This commit is contained in:
parent
e505b77dcb
commit
32c0b1041c
|
@ -15,7 +15,7 @@ async function bootstrap() {
|
||||||
const trpc = app.get(TrpcRouter);
|
const trpc = app.get(TrpcRouter);
|
||||||
trpc.applyMiddleware(app);
|
trpc.applyMiddleware(app);
|
||||||
|
|
||||||
const port = process.env.SERVER_PORT || 3000;
|
const port = process.env.SERVER_PORT || 3001;
|
||||||
|
|
||||||
await app.listen(port);
|
await app.listen(port);
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,7 +22,7 @@ export class GenDevService {
|
||||||
terms: Record<TaxonomySlug, Term[]> = {
|
terms: Record<TaxonomySlug, Term[]> = {
|
||||||
[TaxonomySlug.CATEGORY]: [],
|
[TaxonomySlug.CATEGORY]: [],
|
||||||
[TaxonomySlug.TAG]: [],
|
[TaxonomySlug.TAG]: [],
|
||||||
[TaxonomySlug.LEVEL]: [],
|
[TaxonomySlug.LEVEL]: []
|
||||||
};
|
};
|
||||||
depts: Department[] = [];
|
depts: Department[] = [];
|
||||||
domains: Department[] = [];
|
domains: Department[] = [];
|
||||||
|
|
|
@ -0,0 +1,25 @@
|
||||||
|
import { Checkbox, List } from 'antd';
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
export function CoursePreviewTabmsg({data}){
|
||||||
|
|
||||||
|
|
||||||
|
const renderItem = (item) => (
|
||||||
|
<List.Item>
|
||||||
|
<List.Item.Meta
|
||||||
|
title={item.title}
|
||||||
|
description={item.description}
|
||||||
|
/>
|
||||||
|
</List.Item>
|
||||||
|
);
|
||||||
|
|
||||||
|
return(
|
||||||
|
<div className='my-2'>
|
||||||
|
<List
|
||||||
|
dataSource={data}
|
||||||
|
split={false}
|
||||||
|
renderItem={renderItem}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
|
@ -0,0 +1,11 @@
|
||||||
|
import type { MenuProps } from 'antd';
|
||||||
|
import { Menu } from 'antd';
|
||||||
|
|
||||||
|
type MenuItem = Required<MenuProps>['items'][number];
|
||||||
|
|
||||||
|
export function CourseCatalog(){
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
|
@ -1,7 +1,51 @@
|
||||||
export function coursePreviewAllmsg() {
|
import { useEffect } from 'react';
|
||||||
|
import { CoursePreviewMsg } from '@web/src/app/main/course/preview/type.ts';
|
||||||
|
import { Button , Tabs , Image, Skeleton } from 'antd';
|
||||||
|
import type { TabsProps } from 'antd';
|
||||||
|
import { PlayCircleOutlined } from "@ant-design/icons";
|
||||||
|
export function CoursePreviewAllmsg({previewMsg,items,isLoading}: {previewMsg?:CoursePreviewMsg,items:TabsProps['items'],isLoading:Boolean}){
|
||||||
|
useEffect(() => {
|
||||||
|
console.log(previewMsg)
|
||||||
|
})
|
||||||
|
const TapOnChange = (key: string) => {
|
||||||
|
console.log(key);
|
||||||
|
};
|
||||||
return (
|
return (
|
||||||
<div className="min-h-screen">
|
<div className="min-h-screen max-w-7xl mx-auto px-6 lg:px-8">
|
||||||
helloword
|
<div className="overflow-auto flex justify-around align-items-center w-full mx-auto my-8">
|
||||||
|
<div className="relative w-[600px] h-[340px] m-4 overflow-hidden flex justify-center items-center">
|
||||||
|
<Image
|
||||||
|
src={previewMsg.isLoading ? 'error' : previewMsg.videoPreview}
|
||||||
|
alt="example"
|
||||||
|
preview = {false}
|
||||||
|
className="w-full h-full object-cover z-0"
|
||||||
|
fallback=""
|
||||||
|
/>
|
||||||
|
<div className='w-[600px] h-[360px] absolute top-0 z-10 bg-black opacity-30 transition-opacity duration-300 ease-in-out hover:opacity-70 cursor-pointer'>
|
||||||
|
<PlayCircleOutlined
|
||||||
|
className="absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 text-white text-4xl z-10"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col justify-between w-2/5 content-start h-[340px] my-4 overflow-hidden">
|
||||||
|
{
|
||||||
|
isLoading ? <Skeleton className='my-5' active />
|
||||||
|
:(
|
||||||
|
<>
|
||||||
|
<span className="text-3xl font-bold my-3 ">{previewMsg.Title}</span>
|
||||||
|
<span className="text-xl font-semibold my-3 text-gray-700">{previewMsg.SubTitle}</span>
|
||||||
|
<span className="text-lg font-light my-3 text-gray-500 text-clip">{previewMsg.Description}</span>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
<Button block type="primary" size='large'> 查看课程 </Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="overflow-auto w-11/12 mx-auto my-8">
|
||||||
|
<Tabs defaultActiveKey="1" tabBarGutter={100} items={items} onChange={TapOnChange} />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
|
@ -1,7 +1,81 @@
|
||||||
|
import { Skeleton, type TabsProps } from 'antd';
|
||||||
|
import { CoursePreviewAllmsg } from "./components/coursePreviewAllmsg";
|
||||||
|
import { CoursePreviewTabmsg } from "./components/couresPreviewTabmsg";
|
||||||
|
import { CoursePreviewMsg } from "./type";
|
||||||
|
import { api } from '@nice/client'
|
||||||
|
import { useNavigate, useParams } from 'react-router-dom';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { courseDetailSelect, CourseDto } from '@nice/common';
|
||||||
|
|
||||||
export function CoursePreview(){
|
export function CoursePreview(){
|
||||||
|
const { id } = useParams()
|
||||||
|
const { data:course,isLoading:courseIsLoading}:{data:CourseDto,isLoading:boolean}= api.post.findFirst.useQuery({
|
||||||
|
where:{
|
||||||
|
id
|
||||||
|
},
|
||||||
|
select:courseDetailSelect
|
||||||
|
})
|
||||||
|
// course.sections[0].lectures[0]
|
||||||
|
// `/course/${course.id}/detail/${Lecture.id}`
|
||||||
|
useEffect(() => {
|
||||||
|
if(!courseIsLoading){
|
||||||
|
setPreviewMsg({
|
||||||
|
videoPreview: course?.meta?.thumbnail,
|
||||||
|
Title: course?.title,
|
||||||
|
SubTitle:course?.subTitle,
|
||||||
|
Description:course?.content,
|
||||||
|
ToCourseUrl:`/course/${id}`,
|
||||||
|
isLoading:courseIsLoading
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
},[courseIsLoading])
|
||||||
|
const [previewMsg,setPreviewMsg] = useState({
|
||||||
|
videoPreview: '',
|
||||||
|
Title: '',
|
||||||
|
SubTitle:'',
|
||||||
|
Description:'',
|
||||||
|
ToCourseUrl:'',
|
||||||
|
isLoading:courseIsLoading
|
||||||
|
})
|
||||||
|
const tapData = [
|
||||||
|
{
|
||||||
|
title: '掌握R语言的基本概念语法',
|
||||||
|
description: '学生将学习R语言和RStudio的基本知识',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '掌握R语言的基本概念语法',
|
||||||
|
description: '学生将学习R语言的变量、数据类型、循环和条件语句等',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '掌握R语言的数据导入管理',
|
||||||
|
description: '学生将学会如何将数据导入R环境,并且使用各种类型的数据',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '掌握R语言的基本数据清洗',
|
||||||
|
description: '学生将学会使用R语言进行数据清洗、整理和管理',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '掌握R语言的基本数据统计',
|
||||||
|
description: '学生将学会使用R语言进行基本的数据统计',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '掌握R语言的基本绘图功能',
|
||||||
|
description: '学生将学会使用R语言基本的绘图功能和ggplot2的应用',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const isLoading = false
|
||||||
|
const items: TabsProps['items'] = [
|
||||||
|
{
|
||||||
|
key: '1',
|
||||||
|
label: '课程学习目标',
|
||||||
|
children: isLoading ? <Skeleton className='my-5' active />: <CoursePreviewTabmsg data={tapData}/>,
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
return(
|
return(
|
||||||
<div className="min-h-screen">
|
<div className="min-h-screen">
|
||||||
helloword
|
<CoursePreviewAllmsg previewMsg= {previewMsg} isLoading={courseIsLoading } items = {items}></CoursePreviewAllmsg>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
export interface CoursePreviewMsg{
|
||||||
|
videoPreview: string;
|
||||||
|
Title: string;
|
||||||
|
SubTitle:string;
|
||||||
|
Description:string;
|
||||||
|
ToCourseUrl:string;
|
||||||
|
isLoading:Boolean
|
||||||
|
}
|
|
@ -1,8 +1,9 @@
|
||||||
import { Checkbox, Divider, Radio, Space , Spin} from 'antd';
|
import { Checkbox, Divider, Radio, Space , Spin} from 'antd';
|
||||||
import { categories, levels } from '../mockData';
|
import { categories, levels } from '../mockData';
|
||||||
import { TaxonomySlug, TermDto } from '@nice/common';
|
import { TaxonomySlug, TermDto } from '@nice/common';
|
||||||
import { GetTaxonomyProps, useGetTaxonomy } from '@web/src/hooks/useGetTaxonomy';
|
|
||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
|
import { api } from '@nice/client';
|
||||||
|
|
||||||
interface FilterSectionProps {
|
interface FilterSectionProps {
|
||||||
selectedCategory: string;
|
selectedCategory: string;
|
||||||
|
@ -11,6 +12,35 @@ interface FilterSectionProps {
|
||||||
onLevelChange: (level: string) => void;
|
onLevelChange: (level: string) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface GetTaxonomyProps {
|
||||||
|
categories: string[];
|
||||||
|
isLoading: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
function useGetTaxonomy({type}) : GetTaxonomyProps {
|
||||||
|
const {data,isLoading} :{data:TermDto[],isLoading:boolean}= api.term.findMany.useQuery({
|
||||||
|
where:{
|
||||||
|
taxonomy: {
|
||||||
|
//TaxonomySlug.CATEGORY
|
||||||
|
slug:type
|
||||||
|
}
|
||||||
|
},
|
||||||
|
include:{
|
||||||
|
children :true
|
||||||
|
},
|
||||||
|
take:10, // 只取前10个
|
||||||
|
orderBy: {
|
||||||
|
createdAt: 'desc', // 按创建时间降序排列
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const categories = useMemo(() => {
|
||||||
|
const allCategories = isLoading ? [] : data?.map((course) => course.name);
|
||||||
|
return [...Array.from(new Set(allCategories))];
|
||||||
|
}, [data]);
|
||||||
|
return {categories,isLoading}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
export default function FilterSection({
|
export default function FilterSection({
|
||||||
selectedCategory,
|
selectedCategory,
|
||||||
selectedLevel,
|
selectedLevel,
|
||||||
|
|
|
@ -11,7 +11,35 @@ import {
|
||||||
} from '@ant-design/icons';
|
} from '@ant-design/icons';
|
||||||
import { TaxonomySlug, TermDto } from '@nice/common';
|
import { TaxonomySlug, TermDto } from '@nice/common';
|
||||||
import { api } from '@nice/client';
|
import { api } from '@nice/client';
|
||||||
import { GetTaxonomyProps, useGetTaxonomy } from '@web/src/hooks/useGetTaxonomy';
|
|
||||||
|
interface GetTaxonomyProps {
|
||||||
|
categories: string[];
|
||||||
|
isLoading: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
function useGetTaxonomy({type}) : GetTaxonomyProps {
|
||||||
|
const {data,isLoading} :{data:TermDto[],isLoading:boolean}= api.term.findMany.useQuery({
|
||||||
|
where:{
|
||||||
|
taxonomy: {
|
||||||
|
//TaxonomySlug.CATEGORY
|
||||||
|
slug:type
|
||||||
|
}
|
||||||
|
},
|
||||||
|
include:{
|
||||||
|
children :true
|
||||||
|
},
|
||||||
|
take:10, // 只取前10个
|
||||||
|
orderBy: {
|
||||||
|
createdAt: 'desc', // 按创建时间降序排列
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const categories = useMemo(() => {
|
||||||
|
const allCategories = isLoading ? [] : data?.map((course) => course.name);
|
||||||
|
return [...Array.from(new Set(allCategories))];
|
||||||
|
}, [data]);
|
||||||
|
return {categories,isLoading}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
const { Title, Text } = Typography;
|
const { Title, Text } = Typography;
|
||||||
|
|
||||||
|
|
|
@ -106,19 +106,19 @@ const HomePage = () => {
|
||||||
return (
|
return (
|
||||||
<div className="min-h-screen">
|
<div className="min-h-screen">
|
||||||
<HeroSection />
|
<HeroSection />
|
||||||
<TusUploader></TusUploader>
|
|
||||||
<CoursesSection
|
<CoursesSection
|
||||||
title="推荐课程"
|
title="推荐课程"
|
||||||
description="最受欢迎的精品课程,助你快速成长"
|
description="最受欢迎的精品课程,助你快速成长"
|
||||||
courses={mockCourses}
|
courses={mockCourses}
|
||||||
/>
|
/>
|
||||||
<CoursesSection
|
{/* <CoursesSection
|
||||||
title="热门课程"
|
title="热门课程"
|
||||||
description="最受欢迎的精品课程,助你快速成长"
|
description="最受欢迎的精品课程,助你快速成长"
|
||||||
courses={mockCourses}
|
courses={mockCourses}
|
||||||
/>
|
/> */}
|
||||||
<CategorySection />
|
<CategorySection />
|
||||||
<FeaturedTeachersSection />
|
{/* <FeaturedTeachersSection /> */}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,31 +0,0 @@
|
||||||
import { api } from "@nice/client/"
|
|
||||||
import { TaxonomySlug, TermDto } from "@nice/common"
|
|
||||||
import { useMemo } from 'react';
|
|
||||||
|
|
||||||
export interface GetTaxonomyProps {
|
|
||||||
categories: string[];
|
|
||||||
isLoading: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useGetTaxonomy({type}) : GetTaxonomyProps {
|
|
||||||
const {data,isLoading} :{data:TermDto[],isLoading:boolean}= api.term.findMany.useQuery({
|
|
||||||
where:{
|
|
||||||
taxonomy: {
|
|
||||||
//TaxonomySlug.CATEGORY
|
|
||||||
slug:type
|
|
||||||
}
|
|
||||||
},
|
|
||||||
include:{
|
|
||||||
children :true
|
|
||||||
},
|
|
||||||
take:10, // 只取前10个
|
|
||||||
orderBy: {
|
|
||||||
createdAt: 'desc', // 按创建时间降序排列
|
|
||||||
},
|
|
||||||
})
|
|
||||||
const categories = useMemo(() => {
|
|
||||||
const allCategories = isLoading ? [] : data?.map((course) => course.name);
|
|
||||||
return [...Array.from(new Set(allCategories))];
|
|
||||||
}, [data]);
|
|
||||||
return {categories,isLoading}
|
|
||||||
}
|
|
|
@ -95,7 +95,7 @@ export const routes: CustomRouteObject[] = [
|
||||||
},
|
},
|
||||||
// 课程预览页面
|
// 课程预览页面
|
||||||
{
|
{
|
||||||
path: "coursePreview",
|
path: "coursePreview/:id?",
|
||||||
element:<CoursePreview></CoursePreview>
|
element:<CoursePreview></CoursePreview>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue