1
0
mirror of https://gitee.com/koogua/course-tencent-cloud.git synced 2025-06-21 19:22:45 +08:00
2020-09-04 17:33:16 +08:00

559 lines
16 KiB
PHP

<?php
namespace App\Http\Admin\Services;
use App\Builders\CourseList as CourseListBuilder;
use App\Caches\Course as CourseCache;
use App\Caches\CourseCategoryList as CourseCategoryListCache;
use App\Caches\CourseRelatedList as CourseRelatedListCache;
use App\Caches\CourseTeacherList as CourseTeacherListCache;
use App\Library\Paginator\Query as PagerQuery;
use App\Models\Course as CourseModel;
use App\Models\CourseCategory as CourseCategoryModel;
use App\Models\CourseRating as CourseRatingModel;
use App\Models\CourseRelated as CourseRelatedModel;
use App\Models\CourseUser as CourseUserModel;
use App\Models\ImGroup as ImGroupModel;
use App\Repos\Category as CategoryRepo;
use App\Repos\Chapter as ChapterRepo;
use App\Repos\Course as CourseRepo;
use App\Repos\CourseCategory as CourseCategoryRepo;
use App\Repos\CourseRelated as CourseRelatedRepo;
use App\Repos\CourseUser as CourseUserRepo;
use App\Repos\User as UserRepo;
use App\Services\Syncer\CourseIndex as CourseIndexSyncer;
use App\Validators\Course as CourseValidator;
class Course extends Service
{
public function getCourses()
{
$pagerQuery = new PagerQuery();
$params = $pagerQuery->getParams();
if (!empty($params['xm_category_ids'])) {
$xmCategoryIds = explode(',', $params['xm_category_ids']);
$params['category_id'] = count($xmCategoryIds) > 1 ? $xmCategoryIds : $xmCategoryIds[0];
}
if (!empty($params['xm_teacher_ids'])) {
$xmTeacherIds = explode(',', $params['xm_teacher_ids']);
$params['teacher_id'] = count($xmTeacherIds) > 1 ? $xmTeacherIds : $xmTeacherIds[0];
}
$params['deleted'] = $params['deleted'] ?? 0;
$sort = $pagerQuery->getSort();
$page = $pagerQuery->getPage();
$limit = $pagerQuery->getLimit();
$courseRepo = new CourseRepo();
$pager = $courseRepo->paginate($params, $sort, $page, $limit);
return $this->handleCourses($pager);
}
public function getCourse($id)
{
return $this->findOrFail($id);
}
public function createCourse()
{
$post = $this->request->getPost();
$validator = new CourseValidator();
$model = $validator->checkModel($post['model']);
$title = $validator->checkTitle($post['title']);
try {
$this->db->begin();
$course = new CourseModel();
$course->model = $model;
$course->title = $title;
if ($course->create() === false) {
throw new \RuntimeException('Create Course Failed');
}
$courseRating = new CourseRatingModel();
$courseRating->course_id = $course->id;
if ($courseRating->create() === false) {
throw new \RuntimeException('Create CourseRating Failed');
}
$imGroup = new ImGroupModel();
$imGroup->course_id = $course->id;
$imGroup->name = $course->title;
$imGroup->about = $course->summary;
$imGroup->published = 1;
if ($imGroup->create() === false) {
throw new \RuntimeException('Create ImGroup Failed');
}
$this->db->commit();
return $course;
} catch (\Exception $e) {
$this->db->rollback();
$logger = $this->getLogger();
$logger->error('Create Course Error ' . kg_json_encode([
'code' => $e->getCode(),
'message' => $e->getMessage(),
]));
throw new \RuntimeException('sys.trans_rollback');
}
}
public function updateCourse($id)
{
$course = $this->findOrFail($id);
$post = $this->request->getPost();
$validator = new CourseValidator();
$data = [];
if (isset($post['title'])) {
$data['title'] = $validator->checkTitle($post['title']);
}
if (isset($post['cover'])) {
$data['cover'] = $validator->checkCover($post['cover']);
}
if (isset($post['keywords'])) {
$data['keywords'] = $validator->checkKeywords($post['keywords']);
}
if (isset($post['summary'])) {
$data['summary'] = $validator->checkSummary($post['summary']);
}
if (isset($post['details'])) {
$data['details'] = $validator->checkDetails($post['details']);
}
if (isset($post['level'])) {
$data['level'] = $validator->checkLevel($post['level']);
}
if (isset($post['price_mode'])) {
if ($post['price_mode'] == 'free') {
$data['market_price'] = 0;
$data['vip_price'] = 0;
} else {
$data['market_price'] = $validator->checkMarketPrice($post['market_price']);
$data['vip_price'] = $validator->checkVipPrice($post['vip_price']);
$validator->checkComparePrice($post['market_price'], $post['vip_price']);
$data['study_expiry'] = $validator->checkStudyExpiry($post['study_expiry']);
$data['refund_expiry'] = $validator->checkRefundExpiry($post['refund_expiry']);
}
}
if (isset($post['published'])) {
$data['published'] = $validator->checkPublishStatus($post['published']);
if ($post['published'] == 1) {
$validator->checkPublishAbility($course);
}
}
if (isset($post['xm_category_ids'])) {
$this->saveCategories($course, $post['xm_category_ids']);
}
if (isset($post['xm_teacher_ids'])) {
$this->saveTeachers($course, $post['xm_teacher_ids']);
}
if (isset($post['xm_course_ids'])) {
$this->saveRelatedCourses($course, $post['xm_course_ids']);
}
$course->update($data);
return $course;
}
public function deleteCourse($id)
{
$course = $this->findOrFail($id);
$course->deleted = 1;
$course->update();
return $course;
}
public function restoreCourse($id)
{
$course = $this->findOrFail($id);
$course->deleted = 0;
$course->update();
return $course;
}
public function getStudyExpiryOptions()
{
return CourseModel::studyExpiryOptions();
}
public function getRefundExpiryOptions()
{
return CourseModel::refundExpiryOptions();
}
public function getXmCategories($id)
{
$categoryRepo = new CategoryRepo();
$allCategories = $categoryRepo->findAll(['deleted' => 0]);
if ($allCategories->count() == 0) {
return [];
}
$courseCategoryIds = [];
if ($id > 0) {
$courseRepo = new CourseRepo();
$courseCategories = $courseRepo->findCategories($id);
if ($courseCategories->count() > 0) {
foreach ($courseCategories as $category) {
$courseCategoryIds[] = $category->id;
}
}
}
$list = [];
foreach ($allCategories as $category) {
if ($category->level == 1) {
$list[$category->id] = [
'name' => $category->name,
'value' => $category->id,
'children' => [],
];
}
}
foreach ($allCategories as $category) {
$selected = in_array($category->id, $courseCategoryIds);
$parentId = $category->parent_id;
if ($category->level == 2) {
$list[$parentId]['children'][] = [
'id' => $category->id,
'name' => $category->name,
'selected' => $selected,
];
}
}
return array_values($list);
}
public function getXmTeachers($id)
{
$userRepo = new UserRepo();
$allTeachers = $userRepo->findTeachers();
if ($allTeachers->count() == 0) {
return [];
}
$courseTeacherIds = [];
if ($id > 0) {
$courseRepo = new CourseRepo();
$courseTeachers = $courseRepo->findTeachers($id);
if ($courseTeachers->count() > 0) {
foreach ($courseTeachers as $teacher) {
$courseTeacherIds[] = $teacher->id;
}
}
}
$list = [];
foreach ($allTeachers as $teacher) {
$selected = in_array($teacher->id, $courseTeacherIds);
$list[] = [
'id' => $teacher->id,
'name' => $teacher->name,
'selected' => $selected,
];
}
return $list;
}
public function getXmCourses($id)
{
$courseRepo = new CourseRepo();
$courses = $courseRepo->findRelatedCourses($id);
$list = [];
if ($courses->count() > 0) {
foreach ($courses as $course) {
$list[] = [
'id' => $course->id,
'title' => $course->title,
'selected' => true,
];
}
}
return $list;
}
public function getChapters($id)
{
$course = $this->findOrFail($id);
$deleted = $this->request->getQuery('deleted', 'int', 0);
$chapterRepo = new ChapterRepo();
return $chapterRepo->findAll([
'parent_id' => 0,
'course_id' => $course->id,
'deleted' => $deleted,
]);
}
protected function findOrFail($id)
{
$validator = new CourseValidator();
return $validator->checkCourse($id);
}
protected function rebuildCourseCache(CourseModel $course)
{
$cache = new CourseCache();
$cache->rebuild($course->id);
}
protected function rebuildCourseIndex(CourseModel $course)
{
$syncer = new CourseIndexSyncer();
$syncer->addItem($course->id);
}
protected function saveTeachers(CourseModel $course, $teacherIds)
{
$courseRepo = new CourseRepo();
$courseTeachers = $courseRepo->findTeachers($course->id);
$originTeacherIds = [];
if ($courseTeachers->count() > 0) {
foreach ($courseTeachers as $teacher) {
$originTeacherIds[] = $teacher->id;
}
}
$newTeacherIds = $teacherIds ? explode(',', $teacherIds) : [];
$addedTeacherIds = array_diff($newTeacherIds, $originTeacherIds);
if ($addedTeacherIds) {
foreach ($addedTeacherIds as $teacherId) {
$courseTeacher = new CourseUserModel();
$courseTeacher->create([
'course_id' => $course->id,
'user_id' => $teacherId,
'role_type' => CourseUserModel::ROLE_TEACHER,
'source_type' => CourseUserModel::SOURCE_IMPORT,
'expiry_time' => strtotime('+10 years'),
]);
}
}
$deletedTeacherIds = array_diff($originTeacherIds, $newTeacherIds);
if ($deletedTeacherIds) {
$courseUserRepo = new CourseUserRepo();
foreach ($deletedTeacherIds as $teacherId) {
$courseTeacher = $courseUserRepo->findCourseTeacher($course->id, $teacherId);
if ($courseTeacher) {
$courseTeacher->delete();
}
}
}
$teacherId = $newTeacherIds[0] ?? 0;
if ($teacherId) {
$course->teacher_id = $teacherId;
$course->update();
}
$cache = new CourseTeacherListCache();
$cache->rebuild($course->id);
}
protected function saveCategories(CourseModel $course, $categoryIds)
{
$courseRepo = new CourseRepo();
$courseCategories = $courseRepo->findCategories($course->id);
$originCategoryIds = [];
if ($courseCategories->count() > 0) {
foreach ($courseCategories as $category) {
$originCategoryIds[] = $category->id;
}
}
$newCategoryIds = $categoryIds ? explode(',', $categoryIds) : [];
$addedCategoryIds = array_diff($newCategoryIds, $originCategoryIds);
if ($addedCategoryIds) {
foreach ($addedCategoryIds as $categoryId) {
$courseCategory = new CourseCategoryModel();
$courseCategory->create([
'course_id' => $course->id,
'category_id' => $categoryId,
]);
}
}
$deletedCategoryIds = array_diff($originCategoryIds, $newCategoryIds);
if ($deletedCategoryIds) {
$courseCategoryRepo = new CourseCategoryRepo();
foreach ($deletedCategoryIds as $categoryId) {
$courseCategory = $courseCategoryRepo->findCourseCategory($course->id, $categoryId);
if ($courseCategory) {
$courseCategory->delete();
}
}
}
$categoryId = $newCategoryIds[0] ?? 0;
if ($categoryId) {
$course->category_id = $categoryId;
$course->update();
}
$cache = new CourseCategoryListCache();
$cache->rebuild($course->id);
}
protected function saveRelatedCourses(CourseModel $course, $courseIds)
{
$courseRepo = new CourseRepo();
$relatedCourses = $courseRepo->findRelatedCourses($course->id);
$originRelatedIds = [];
if ($relatedCourses->count() > 0) {
foreach ($relatedCourses as $relatedCourse) {
$originRelatedIds[] = $relatedCourse->id;
}
}
$newRelatedIds = $courseIds ? explode(',', $courseIds) : [];
$addedRelatedIds = array_diff($newRelatedIds, $originRelatedIds);
$courseRelatedRepo = new CourseRelatedRepo();
/**
* 双向关联
*/
if ($addedRelatedIds) {
foreach ($addedRelatedIds as $relatedId) {
if ($relatedId != $course->id) {
$record = $courseRelatedRepo->findCourseRelated($course->id, $relatedId);
if (!$record) {
$courseRelated = new CourseRelatedModel();
$courseRelated->create([
'course_id' => $course->id,
'related_id' => $relatedId,
]);
}
$record = $courseRelatedRepo->findCourseRelated($relatedId, $course->id);
if (!$record) {
$courseRelated = new CourseRelatedModel();
$courseRelated->create([
'course_id' => $relatedId,
'related_id' => $course->id,
]);
}
}
}
}
$deletedRelatedIds = array_diff($originRelatedIds, $newRelatedIds);
/**
* 单向删除
*/
if ($deletedRelatedIds) {
$courseRelatedRepo = new CourseRelatedRepo();
foreach ($deletedRelatedIds as $relatedId) {
$courseRelated = $courseRelatedRepo->findCourseRelated($course->id, $relatedId);
if ($courseRelated) {
$courseRelated->delete();
}
}
}
$cache = new CourseRelatedListCache();
$cache->rebuild($course->id);
}
protected function handleCourses($pager)
{
if ($pager->total_items > 0) {
$builder = new CourseListBuilder();
$pipeA = $pager->items->toArray();
$pipeB = $builder->handleCategories($pipeA);
$pipeC = $builder->handleTeachers($pipeB);
$pipeD = $builder->objects($pipeC);
$pager->items = $pipeD;
}
return $pager;
}
}