mirror of https://github.com/halo-dev/halo
Merge remote-tracking branch 'origin/v1' into v1
commit
3f8dc55ae8
|
@ -13,6 +13,8 @@ import lombok.Data;
|
||||||
@Data
|
@Data
|
||||||
public class CategoryOutputDTO implements OutputConverter<CategoryOutputDTO, Category> {
|
public class CategoryOutputDTO implements OutputConverter<CategoryOutputDTO, Category> {
|
||||||
|
|
||||||
|
private Integer id;
|
||||||
|
|
||||||
private String name;
|
private String name;
|
||||||
|
|
||||||
private String slugName;
|
private String slugName;
|
||||||
|
|
|
@ -0,0 +1,58 @@
|
||||||
|
package cc.ryanc.halo.model.params;
|
||||||
|
|
||||||
|
import cc.ryanc.halo.model.dto.base.InputConverter;
|
||||||
|
import cc.ryanc.halo.model.entity.Category;
|
||||||
|
import cc.ryanc.halo.utils.SlugUtils;
|
||||||
|
import lombok.Data;
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
|
|
||||||
|
import javax.validation.constraints.NotBlank;
|
||||||
|
import javax.validation.constraints.Size;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Category param.
|
||||||
|
*
|
||||||
|
* @author johnniang
|
||||||
|
* @date 3/21/19
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
public class CategoryParam implements InputConverter<Category> {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 分类名称
|
||||||
|
*/
|
||||||
|
@NotBlank(message = "Category name must not be blank")
|
||||||
|
@Size(max = 50, message = "Length of category name must not be more than {max}")
|
||||||
|
private String name;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 缩略名
|
||||||
|
*/
|
||||||
|
@Size(max = 50, message = "Length of category slug name must not be more than {max}")
|
||||||
|
private String slugName;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 描述
|
||||||
|
*/
|
||||||
|
@Size(max = 100, message = "Length of category description must not be more than {max}")
|
||||||
|
private String description;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 上级目录
|
||||||
|
*/
|
||||||
|
private Integer parentId;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Category convertTo() {
|
||||||
|
// Handle default value
|
||||||
|
if (StringUtils.isBlank(slugName)) {
|
||||||
|
slugName = SlugUtils.slugify(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parentId == null || parentId < 0) {
|
||||||
|
parentId = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return InputConverter.super.convertTo();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
package cc.ryanc.halo.model.vo;
|
||||||
|
|
||||||
|
import cc.ryanc.halo.model.dto.CategoryOutputDTO;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Category vo.
|
||||||
|
*
|
||||||
|
* @author johnniang
|
||||||
|
* @date 3/21/19
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
public class CategoryVO extends CategoryOutputDTO {
|
||||||
|
|
||||||
|
private List<CategoryVO> children;
|
||||||
|
}
|
|
@ -2,6 +2,7 @@ package cc.ryanc.halo.repository;
|
||||||
|
|
||||||
import cc.ryanc.halo.model.entity.Category;
|
import cc.ryanc.halo.model.entity.Category;
|
||||||
import cc.ryanc.halo.repository.base.BaseRepository;
|
import cc.ryanc.halo.repository.base.BaseRepository;
|
||||||
|
import org.springframework.lang.NonNull;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Category repository.
|
* Category repository.
|
||||||
|
@ -9,4 +10,20 @@ import cc.ryanc.halo.repository.base.BaseRepository;
|
||||||
* @author johnniang
|
* @author johnniang
|
||||||
*/
|
*/
|
||||||
public interface CategoryRepository extends BaseRepository<Category, Integer> {
|
public interface CategoryRepository extends BaseRepository<Category, Integer> {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Counts by category name.
|
||||||
|
*
|
||||||
|
* @param name category name must not be blank
|
||||||
|
* @return the count
|
||||||
|
*/
|
||||||
|
long countByName(@NonNull String name);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Counts by category id.
|
||||||
|
*
|
||||||
|
* @param id category id must not be null
|
||||||
|
* @return the count
|
||||||
|
*/
|
||||||
|
long countById(@NonNull Integer id);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,12 @@
|
||||||
package cc.ryanc.halo.service;
|
package cc.ryanc.halo.service;
|
||||||
|
|
||||||
import cc.ryanc.halo.model.entity.Category;
|
import cc.ryanc.halo.model.entity.Category;
|
||||||
|
import cc.ryanc.halo.model.vo.CategoryVO;
|
||||||
import cc.ryanc.halo.service.base.CrudService;
|
import cc.ryanc.halo.service.base.CrudService;
|
||||||
|
import org.springframework.data.domain.Sort;
|
||||||
|
import org.springframework.lang.NonNull;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Category service.
|
* Category service.
|
||||||
|
@ -15,5 +20,15 @@ public interface CategoryService extends CrudService<Category, Integer> {
|
||||||
*
|
*
|
||||||
* @param id id
|
* @param id id
|
||||||
*/
|
*/
|
||||||
void remove(Integer id);
|
@Deprecated
|
||||||
|
void remove(@NonNull Integer id);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lists as category tree.
|
||||||
|
*
|
||||||
|
* @param sort sort info must not be null
|
||||||
|
* @return a category tree
|
||||||
|
*/
|
||||||
|
@NonNull
|
||||||
|
List<CategoryVO> listAsTree(@NonNull Sort sort);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,22 @@
|
||||||
package cc.ryanc.halo.service.impl;
|
package cc.ryanc.halo.service.impl;
|
||||||
|
|
||||||
|
import cc.ryanc.halo.exception.AlreadyExistsException;
|
||||||
|
import cc.ryanc.halo.exception.NotFoundException;
|
||||||
import cc.ryanc.halo.model.entity.Category;
|
import cc.ryanc.halo.model.entity.Category;
|
||||||
|
import cc.ryanc.halo.model.vo.CategoryVO;
|
||||||
import cc.ryanc.halo.repository.CategoryRepository;
|
import cc.ryanc.halo.repository.CategoryRepository;
|
||||||
import cc.ryanc.halo.service.CategoryService;
|
import cc.ryanc.halo.service.CategoryService;
|
||||||
import cc.ryanc.halo.service.base.AbstractCrudService;
|
import cc.ryanc.halo.service.base.AbstractCrudService;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.springframework.data.domain.Sort;
|
||||||
|
import org.springframework.lang.NonNull;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
import org.springframework.util.Assert;
|
||||||
|
import org.springframework.util.CollectionUtils;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* CategoryService implementation class
|
* CategoryService implementation class
|
||||||
|
@ -12,6 +24,7 @@ import org.springframework.stereotype.Service;
|
||||||
* @author : RYAN0UP
|
* @author : RYAN0UP
|
||||||
* @date : 2019-03-14
|
* @date : 2019-03-14
|
||||||
*/
|
*/
|
||||||
|
@Slf4j
|
||||||
@Service
|
@Service
|
||||||
public class CategoryServiceImpl extends AbstractCrudService<Category, Integer> implements CategoryService {
|
public class CategoryServiceImpl extends AbstractCrudService<Category, Integer> implements CategoryService {
|
||||||
|
|
||||||
|
@ -31,4 +44,107 @@ public class CategoryServiceImpl extends AbstractCrudService<Category, Integer>
|
||||||
public void remove(Integer id) {
|
public void remove(Integer id) {
|
||||||
// TODO 删除分类,以及和文章的对应关系
|
// TODO 删除分类,以及和文章的对应关系
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Category create(Category category) {
|
||||||
|
Assert.notNull(category, "Category to create must not be null");
|
||||||
|
|
||||||
|
// Check the category name
|
||||||
|
long count = categoryRepository.countByName(category.getName());
|
||||||
|
|
||||||
|
if (count > 0) {
|
||||||
|
log.error("Category has exist already: [{}]", category);
|
||||||
|
throw new AlreadyExistsException("The category has exist already");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check parent id
|
||||||
|
if (category.getParentId() > 0) {
|
||||||
|
count = categoryRepository.countById(category.getParentId());
|
||||||
|
|
||||||
|
if (count == 0) {
|
||||||
|
log.error("Parent category with id: [{}] was not found, category: [{}]", category.getParentId(), category);
|
||||||
|
throw new NotFoundException("Parent category with id = " + category.getParentId() + " was not found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create it
|
||||||
|
return super.create(category);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<CategoryVO> listAsTree(Sort sort) {
|
||||||
|
Assert.notNull(sort, "Sort info must not be null");
|
||||||
|
|
||||||
|
// List all category
|
||||||
|
List<Category> categories = listAll(sort);
|
||||||
|
|
||||||
|
if (CollectionUtils.isEmpty(categories)) {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create top category
|
||||||
|
CategoryVO topLevelCategory = createTopLevelCategory();
|
||||||
|
|
||||||
|
// Concrete the tree
|
||||||
|
concreteTree(topLevelCategory, categories);
|
||||||
|
|
||||||
|
return topLevelCategory.getChildren();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Concrete category tree.
|
||||||
|
*
|
||||||
|
* @param parentCategory parent category vo must not be null
|
||||||
|
* @param categories a list of category
|
||||||
|
*/
|
||||||
|
private void concreteTree(CategoryVO parentCategory, List<Category> categories) {
|
||||||
|
Assert.notNull(parentCategory, "Parent category must not be null");
|
||||||
|
|
||||||
|
if (CollectionUtils.isEmpty(categories)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create children container for removing after
|
||||||
|
List<Category> children = new LinkedList<>();
|
||||||
|
|
||||||
|
categories.forEach(category -> {
|
||||||
|
if (parentCategory.getId().equals(category.getParentId())) {
|
||||||
|
// Save child category
|
||||||
|
children.add(category);
|
||||||
|
|
||||||
|
// Convert to child category vo
|
||||||
|
CategoryVO child = new CategoryVO().convertFrom(category);
|
||||||
|
|
||||||
|
// Init children if absent
|
||||||
|
if (parentCategory.getChildren() == null) {
|
||||||
|
parentCategory.setChildren(new LinkedList<>());
|
||||||
|
}
|
||||||
|
parentCategory.getChildren().add(child);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Remove all child categories
|
||||||
|
categories.removeAll(children);
|
||||||
|
|
||||||
|
// Foreach children vos
|
||||||
|
if (!CollectionUtils.isEmpty(parentCategory.getChildren())) {
|
||||||
|
parentCategory.getChildren().forEach(childCategory -> concreteTree(childCategory, categories));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a top level category.
|
||||||
|
*
|
||||||
|
* @return top level category with id 0
|
||||||
|
*/
|
||||||
|
@NonNull
|
||||||
|
private CategoryVO createTopLevelCategory() {
|
||||||
|
CategoryVO topCategory = new CategoryVO();
|
||||||
|
// Set default value
|
||||||
|
topCategory.setId(0);
|
||||||
|
topCategory.setChildren(new LinkedList<>());
|
||||||
|
topCategory.setParentId(-1);
|
||||||
|
|
||||||
|
return topCategory;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,48 @@
|
||||||
|
package cc.ryanc.halo.web.controller.admin.api;
|
||||||
|
|
||||||
|
import cc.ryanc.halo.model.dto.CategoryOutputDTO;
|
||||||
|
import cc.ryanc.halo.model.entity.Category;
|
||||||
|
import cc.ryanc.halo.model.params.CategoryParam;
|
||||||
|
import cc.ryanc.halo.model.vo.CategoryVO;
|
||||||
|
import cc.ryanc.halo.service.CategoryService;
|
||||||
|
import io.swagger.annotations.ApiOperation;
|
||||||
|
import org.springframework.data.domain.Sort;
|
||||||
|
import org.springframework.data.web.SortDefault;
|
||||||
|
import org.springframework.web.bind.annotation.*;
|
||||||
|
|
||||||
|
import javax.validation.Valid;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import static org.springframework.data.domain.Sort.Direction.ASC;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Category controller.
|
||||||
|
*
|
||||||
|
* @author johnniang
|
||||||
|
* @date 3/21/19
|
||||||
|
*/
|
||||||
|
@RestController
|
||||||
|
@RequestMapping("/admin/api/categories")
|
||||||
|
public class CategoryController {
|
||||||
|
|
||||||
|
private final CategoryService categoryService;
|
||||||
|
|
||||||
|
public CategoryController(CategoryService categoryService) {
|
||||||
|
this.categoryService = categoryService;
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping("tree")
|
||||||
|
@ApiOperation("List as category tree")
|
||||||
|
public List<CategoryVO> listAsTree(@SortDefault(sort = "name", direction = ASC) Sort sort) {
|
||||||
|
return categoryService.listAsTree(sort);
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping
|
||||||
|
public CategoryOutputDTO createBy(@Valid @RequestBody CategoryParam categoryParam) {
|
||||||
|
// Convert to category
|
||||||
|
Category category = categoryParam.convertTo();
|
||||||
|
|
||||||
|
// Save it
|
||||||
|
return new CategoryOutputDTO().convertFrom(categoryService.create(category));
|
||||||
|
}
|
||||||
|
}
|
|
@ -38,7 +38,7 @@ public class PostController {
|
||||||
}
|
}
|
||||||
|
|
||||||
@GetMapping("status/{status}")
|
@GetMapping("status/{status}")
|
||||||
@ApiOperation("")
|
@ApiOperation("Gets a page of post by post status")
|
||||||
public Page<PostSimpleOutputDTO> pageByStatus(@PathVariable(name = "status") PostStatus status,
|
public Page<PostSimpleOutputDTO> pageByStatus(@PathVariable(name = "status") PostStatus status,
|
||||||
@PageableDefault(sort = "updateTime", direction = DESC) Pageable pageable) {
|
@PageableDefault(sort = "updateTime", direction = DESC) Pageable pageable) {
|
||||||
return postService.pageSimpleDtoByStatus(status, PostType.POST, pageable);
|
return postService.pageSimpleDtoByStatus(status, PostType.POST, pageable);
|
||||||
|
|
Loading…
Reference in New Issue