中软实习培训记录十(0730)
今天主要来完成剩余的新闻管理的删除功能
删除功能
删除功能实现
1、Service接口中定义删除方法,通过id删除数据项
//删除
void deleteNew(Long id);
在NewServiceImpl中实现方法:
@Override
public void deleteNew(Long id) {
newRepository.deleteById(id);
}
2、到NewController中,进行删除操作,同时进行操作成功提示。
@GetMapping("/news/{id}/delete")
public String delete(@PathVariable Long id,RedirectAttributes attributes){
newService.deleteNew(id);
attributes.addFlashAttribute("message","删除成功");
return REDIRECT_LIST;
}
删除功能演示
点击新闻对应的“删除”按钮
删除成功,同时进行提示:
用户的新闻浏览系统
前面我们已经实现了一个新闻的后台管理系统,现在开始设计用户的新闻浏览系统
界面设计
根据上面的界面,可以看出,我们希望实现主页上首先能展示分类列表、标签列表、新闻列表、最新推荐等内容,同时点击之后能进行相应的跳转。
主页新闻列表实现
1、我们首先来实现主页上的新闻列表罗列,同时新闻的展示有分页的效果,因此首先需要在NewService中进行定义:
//主页显示新闻列表
Page<News> ListNew(Pageable pageable);
在ServiceImpl中进行实现,只要调用findAll即可
@Override
public Page<News> ListNew(Pageable pageable) {
return newRepository.findAll(pageable);
}
2、Controller部分,我们创建IndexController,注意该类放在web文件夹下,与之前的admin并级
@Controller
public class IndexController {
@Autowired
private TypeService typeService;
@Autowired
private TagService tagService;
@Autowired
private NewService newService;
@GetMapping("/")
public String index(@PageableDefault(size = 3,sort = {"updateTime"},direction = Sort.Direction.DESC)
Pageable pageable, Model model){
model.addAttribute("page",newService.ListNew(pageable));
return "index";
}
}
主页新闻列表显示
成功实现了新闻列表的显示,因为数据库中新闻数据较少,所以暂时无法看到分页效果。
新闻列表中包含了新闻标题、新闻描述、发布作者、时间以及所属类别
主页类别列表实现
类别在显示的时候,需要根据类别下文章数进行排序,因此我们就需要进行统计
1、在TypeService定义方法:
//主页显示文章数排前几的类别
List<Type> listTypeTop(Integer size);
在TypeServiceImpl进行实现
@Override
public List<Type> listTypeTop(Integer size) {
Sort sort=Sort.by(Sort.Direction.DESC,"news.size");//根据新闻篇数降序排列
Pageable pageable= PageRequest.of(0,size,sort);
return typeRepository.findTop(pageable);
}
**注意,**因为返回值需要一个findTop的内容,同时使得找到的内容根据pageable设定的显示,因此在TypeRepository中新增一条方法
@Query("select t from Type t")
List<Type> findTop(Pageable pageable);
2、回到IndexController,添加操作,在index方法中添加属性
@GetMapping("/")
public String index(@PageableDefault(size = 3,sort = {"updateTime"},direction = Sort.Direction.DESC)
Pageable pageable, Model model){
model.addAttribute("page",newService.ListNew(pageable));
model.addAttribute("types",typeService.listTypeTop(3));
return "index";
}
主页类别列表显示
主页标签列表实现
1、跟类别Type一样处理,TagService处定义方法
List<Tag> listTagTop(Integer size);
在TagServiceImpl中进行实现:
@Override
public List<Tag> listTagTop(Integer size) {
Sort sort=Sort.by(Sort.Direction.DESC,"newsList.size");
Pageable pageable= PageRequest.of(0,size,sort);
return tagRepository.findTop(pageable);
}
**注意,**同样需要在TagRepository中新增一条方法:
@Query("select t from Tag t")
List<Tag> findTop(Pageable pageable);
2、回到IndexController,添加操作,在index方法中添加属性
@GetMapping("/")
public String index(@PageableDefault(size = 3,sort = {"updateTime"},direction = Sort.Direction.DESC)
Pageable pageable, Model model){
model.addAttribute("page",newService.ListNew(pageable));
model.addAttribute("types",typeService.listTypeTop(3));
model.addAttribute("tags",tagService.listTagTop(3));
return "index";
}
主页标签列表显示
主页最新推荐实现
1、在NewService中定义方法
//主页最新推荐新闻列表
List<News> listRecommendNewTop(Integer size);
在NewServiceImpl中进行实现,要求推荐的新闻按照更新时间进行降序排列:
@Override
public List<News> listRecommendNewTop(Integer size) {
Sort sort=Sort.by(Sort.Direction.DESC,"updateTime");
Pageable pageable= PageRequest.of(0,size,sort);
return newRepository.findTop(pageable);
}
**注意,**同样需要在NewRepository中新增一条方法,进行更新:
@Query("select n from News n where n.recommend=true")
List<News> findTop(Pageable pageable);
2、回到IndexController,添加操作,更新,在index方法中添加属性
@GetMapping("/")
public String index(@PageableDefault(size = 3,sort = {"updateTime"},direction = Sort.Direction.DESC)
Pageable pageable, Model model){
model.addAttribute("page",newService.ListNew(pageable));
model.addAttribute("types",typeService.listTypeTop(3));
model.addAttribute("tags",tagService.listTagTop(3));
model.addAttribute("recommendNews",newService.listRecommendNewTop(3));
return "index";
}
主页最新推荐显示
主页搜索实现
在主页的搜索框中输入字段,希望能够通过模糊查询找出标题或者新闻主体内容【注意主页上显示的是标题和描述、并非主体内容】包含该字段的所有新闻
1、在NewService中新定义接口,参数除了分页外还有查询条件:
//全局搜索
Page<News> listNew(String query,Pageable pageable);
在NewServiceImpl中进行实现:
@Override
public Page<News> listNew(String query, Pageable pageable) {
return newRepository.findByQuery(query,pageable);
}
**注意,**同样需要在NewRepository中新增一条方法,实现根据新闻标题和内容中是否符合查询条件,来得到对应的新闻:
@Query("select n from News n where n.title like ?1 or n.content like ?1")
Page<News> findByQuery(String query, Pageable pageable);
2、回到IndexController,添加操作,不再是像前面Type、Tag等显示在index方法中添加属性,而是另起一个方法:
@PostMapping("/search")
public String search(@PageableDefault(size = 3,sort = {"updateTime"},direction = Sort.Direction.DESC) Pageable pageable,
@RequestParam String query,Model model){
model.addAttribute("page",newService.listNew("%"+query+"%", pageable));
model.addAttribute("query",query);
return "search";
}
主页搜索演示
在主页右上角搜索框中输入查询条件,得到一个新的查询结果页面,对于搜索出来的两条数据,的确符合搜索要求,第一条数据显而易见标题就已符合“12”条件,第二条数据则是因为文章主体内容为“1234567”,包含了“12”而搜索得到。
搜索功能成功!
新闻详情页实现
我们在主页上点击新闻列表中的某一篇新闻,进入新闻详情页,查看新闻的具体内容
1、在Service中,我们已经定义了getNew方法
News getNew(Long id);
因此我们可以直接进行调用,不需要额外写Service方法了
2、IndexController中新建一个news方法,对详情页部分进行处理:
@RequestMapping("/news/{id}")
public String news(@PathVariable Long id,Model model){
model.addAttribute("news",newService.getNew(id));
return "new";
}
新闻详情页展示
存在问题
我们在管理系统进行新闻编辑的时候,是实现了文本的MarkDown编辑功能,如下:
但是我们在用户查看新闻详情页的时候,这个设置的格式却没有出现,而是直接打出了#,与预期不符。
因此针对这个问题,我们对引入一个工具类进而对代码进行修改。
代码更新
1、引入工具包
package com.zr0726.news.util;
import org.commonmark.Extension;
import org.commonmark.ext.gfm.tables.TableBlock;
import org.commonmark.ext.gfm.tables.TablesExtension;
import org.commonmark.ext.heading.anchor.HeadingAnchorExtension;
import org.commonmark.node.Link;
import org.commonmark.node.Node;
import org.commonmark.parser.Parser;
import org.commonmark.renderer.html.AttributeProvider;
import org.commonmark.renderer.html.AttributeProviderContext;
import org.commonmark.renderer.html.AttributeProviderFactory;
import org.commonmark.renderer.html.HtmlRenderer;
import java.util.*;
public class MarkdownUtils {
/**
* markdown格式转换成HTML格式
* @param markdown
* @return
*/
public static String markdownToHtml(String markdown) {
Parser parser = Parser.builder().build();
Node document = parser.parse(markdown);
HtmlRenderer renderer = HtmlRenderer.builder().build();
return renderer.render(document);
}
/**
* 增加扩展[标题锚点,表格生成]
* Markdown转换成HTML
* @param markdown
* @return
*/
public static String markdownToHtmlExtensions(String markdown) {
//h标题生成id
Set<Extension> headingAnchorExtensions = Collections.singleton(HeadingAnchorExtension.create());
//转换table的HTML
List<Extension> tableExtension = Arrays.asList(TablesExtension.create());
Parser parser = Parser.builder()
.extensions(tableExtension)
.build();
Node document = parser.parse(markdown);
HtmlRenderer renderer = HtmlRenderer.builder()
.extensions(headingAnchorExtensions)
.extensions(tableExtension)
.attributeProviderFactory(new AttributeProviderFactory() {
public AttributeProvider create(AttributeProviderContext context) {
return new CustomAttributeProvider();
}
})
.build();
return renderer.render(document);
}
/**
* 处理标签的属性
*/
static class CustomAttributeProvider implements AttributeProvider {
@Override
public void setAttributes(Node node, String tagName, Map<String, String> attributes) {
//改变a标签的target属性为_blank
if (node instanceof Link) {
attributes.put("target", "_blank");
}
if (node instanceof TableBlock) {
attributes.put("class", "ui celled table");
}
}
}
public static void main(String[] args) {
String table = "| hello | hi | 哈哈哈 |\n" +
"| ----- | ---- | ----- |\n" +
"| 斯维尔多 | 士大夫 | f啊 |\n" +
"| 阿什顿发 | 非固定杆 | 撒阿什顿发 |\n" +
"\n";
String a = "[imCoding 爱编程](http://www.lirenmi.cn)";
System.out.println(markdownToHtmlExtensions(a));
}
}
2、在NewService中重定义一个方法,对新闻内容进行markdown到html的转换
//处理markdown
News getAndConvert(Long id);
NewServiceImpl中进行实现:
@Override
public News getAndConvert(Long id) {
News news=newRepository.findById(id).orElse(null);
if(news==null){
System.out.println("该新闻不存在");
}
News news1=new News();
BeanUtils.copyProperties(news,news1);
String content=news1.getContent();
news1.setContent(MarkdownUtils.markdownToHtmlExtensions(content));
return news1;
}
3、IndexController的news方法中我们原本调用了getNews方法,现在利用工具进行convert之后,我们修改调用新定义的getAndConvert方法
@RequestMapping("/news/{id}")
public String news(@PathVariable Long id,Model model){
model.addAttribute("news",newService.getAndConvert(id));
return "new";
}
4、效果演示:
注意在编辑新闻的时候,设置格式时,#与内容要隔一个空,不然也会被直接当作内容输入。