功能优化和新增 (#956)

* feat: 首页项目拖拽排序功能

* feat: 增加首页项目拖拽排序增加只能管理员进行, 排序失败元素回到原本位置

* perf: 新建文章以后直接进入到编辑文章页面

* perf: 优化文档打开时或刷新时样式闪动问题

* perf: 优化表格样式

* feat: 支持上传视频功能

* feat: 视频样式调整

* feat: 直接粘贴视频上传功能

* perf: 优化markdown目录显示
pull/968/head
zhanzhenping 2024-07-05 15:31:34 +08:00 committed by GitHub
parent 710d5bcf50
commit 1ea922106d
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
12 changed files with 243 additions and 131 deletions

View File

@ -79,7 +79,7 @@ avatar=/static/images/headimgurl.jpg
token_size=12 token_size=12
#上传文件的后缀,如果不限制后缀可以设置为 * #上传文件的后缀,如果不限制后缀可以设置为 *
upload_file_ext=txt|doc|docx|xls|xlsx|ppt|pptx|pdf|7z|rar|jpg|jpeg|png|gif upload_file_ext=txt|doc|docx|xls|xlsx|ppt|pptx|pdf|7z|rar|jpg|jpeg|png|gif|mp4|webm|avi
#上传的文件大小限制 #上传的文件大小限制
# - 如果不填写, 则默认1GB如果希望超过1GB必须带单位 # - 如果不填写, 则默认1GB如果希望超过1GB必须带单位

View File

@ -106,9 +106,9 @@ func GetDefaultCover() string {
return URLForWithCdnImage(web.AppConfig.DefaultString("cover", "/static/images/book.jpg")) return URLForWithCdnImage(web.AppConfig.DefaultString("cover", "/static/images/book.jpg"))
} }
// 获取允许的商城文件的类型. // 获取允许的上传文件的类型.
func GetUploadFileExt() []string { func GetUploadFileExt() []string {
ext := web.AppConfig.DefaultString("upload_file_ext", "png|jpg|jpeg|gif|txt|doc|docx|pdf") ext := web.AppConfig.DefaultString("upload_file_ext", "png|jpg|jpeg|gif|txt|doc|docx|pdf|mp4")
temp := strings.Split(ext, "|") temp := strings.Split(ext, "|")
@ -201,7 +201,7 @@ func GetExportOutputPath() string {
return exportOutputPath return exportOutputPath
} }
// 判断是否是允许商城的文件类型. // 判断是否是允许上传的文件类型.
func IsAllowUploadFileExt(ext string) bool { func IsAllowUploadFileExt(ext string) bool {
if strings.HasPrefix(ext, ".") { if strings.HasPrefix(ext, ".") {

View File

@ -7,6 +7,7 @@ import (
"html/template" "html/template"
"image/png" "image/png"
"io" "io"
"mime/multipart"
"net/http" "net/http"
"net/url" "net/url"
"os" "os"
@ -486,41 +487,23 @@ func (c *DocumentController) Upload() {
c.JsonResult(6001, i18n.Tr(c.Lang, "message.param_error")) c.JsonResult(6001, i18n.Tr(c.Lang, "message.param_error"))
} }
name := "editormd-file-file" names := []string{"editormd-file-file", "editormd-image-file", "file", "editormd-resource-file"}
var files []*multipart.FileHeader
for _, name := range names {
file, err := c.GetFiles(name)
if err != nil {
continue
}
if len(file) > 0 && err == nil {
files = append(files, file...)
}
}
// file, moreFile, err := c.GetFile(name) if len(files) == 0 {
// if err == http.ErrMissingFile || moreFile == nil {
// name = "editormd-image-file"
// file, moreFile, err = c.GetFile(name)
// if err == http.ErrMissingFile || moreFile == nil {
// c.JsonResult(6003, i18n.Tr(c.Lang, "message.upload_file_empty"))
// return
// }
// }
// ****3xxx
files, err := c.GetFiles(name)
if err == http.ErrMissingFile {
name = "editormd-image-file"
files, err = c.GetFiles(name)
if err == http.ErrMissingFile {
// c.JsonResult(6003, i18n.Tr(c.Lang, "message.upload_file_empty"))
// return
name = "file"
files, err = c.GetFiles(name)
// logs.Info(files)
if err == http.ErrMissingFile {
c.JsonResult(6003, i18n.Tr(c.Lang, "message.upload_file_empty")) c.JsonResult(6003, i18n.Tr(c.Lang, "message.upload_file_empty"))
return return
} }
}
}
// if err != nil {
// http.Error(w, err.Error(), http.StatusNoContent)
// return
// }
// jMap := make(map[string]interface{})
// s := []map[int]interface{}{}
result2 := []map[string]interface{}{} result2 := []map[string]interface{}{}
var result map[string]interface{} var result map[string]interface{}
for i, _ := range files { for i, _ := range files {
@ -528,24 +511,6 @@ func (c *DocumentController) Upload() {
file, err := files[i].Open() file, err := files[i].Open()
defer file.Close() defer file.Close()
// if err != nil {
// http.Error(w, err.Error(), http.StatusInternalServerError)
// return
// }
// //create destination file making sure the path is writeable.
// dst, err := os.Create("upload/" + files[i].Filename)
// defer dst.Close()
// if err != nil {
// http.Error(w, err.Error(), http.StatusInternalServerError)
// return
// }
// //copy the uploaded file to the destination file
// if _, err := io.Copy(dst, file); err != nil {
// http.Error(w, err.Error(), http.StatusInternalServerError)
// return
// }
// }
// ****
if err != nil { if err != nil {
c.JsonResult(6002, err.Error()) c.JsonResult(6002, err.Error())
@ -619,19 +584,25 @@ func (c *DocumentController) Upload() {
filePath := filepath.Join(conf.WorkingDirectory, "uploads", identify) filePath := filepath.Join(conf.WorkingDirectory, "uploads", identify)
//将图片和文件分开存放 //将图片和文件分开存放
// if filetil.IsImageExt(moreFile.Filename) { attachment := models.NewAttachment()
var strategy filetil.FileTypeStrategy
if filetil.IsImageExt(files[i].Filename) { if filetil.IsImageExt(files[i].Filename) {
filePath = filepath.Join(filePath, "images", fileName+ext) strategy = filetil.ImageStrategy{}
attachment.ResourceType = "image"
} else if filetil.IsVideoExt(files[i].Filename) {
strategy = filetil.VideoStrategy{}
attachment.ResourceType = "video"
} else { } else {
filePath = filepath.Join(filePath, "files", fileName+ext) strategy = filetil.DefaultStrategy{}
attachment.ResourceType = "file"
} }
filePath = strategy.GetFilePath(filePath, fileName, ext)
path := filepath.Dir(filePath) path := filepath.Dir(filePath)
_ = os.MkdirAll(path, os.ModePerm) _ = os.MkdirAll(path, os.ModePerm)
// err = c.SaveToFile(name, filePath) // frome beego controller.go: savetofile it only operates the first one of mutil-upload form file field.
//copy the uploaded file to the destination file //copy the uploaded file to the destination file
dst, err := os.Create(filePath) dst, err := os.Create(filePath)
defer dst.Close() defer dst.Close()
@ -640,12 +611,6 @@ func (c *DocumentController) Upload() {
c.JsonResult(6005, i18n.Tr(c.Lang, "message.failed")) c.JsonResult(6005, i18n.Tr(c.Lang, "message.failed"))
} }
// if err != nil {
// logs.Error("保存文件失败 -> ", err)
// c.JsonResult(6005, i18n.Tr(c.Lang, "message.failed"))
// }
attachment := models.NewAttachment()
attachment.BookId = bookId attachment.BookId = bookId
// attachment.FileName = moreFile.Filename // attachment.FileName = moreFile.Filename
attachment.FileName = files[i].Filename attachment.FileName = files[i].Filename
@ -662,8 +627,7 @@ func (c *DocumentController) Upload() {
attachment.DocumentId = docId attachment.DocumentId = docId
} }
// if filetil.IsImageExt(moreFile.Filename) { if filetil.IsImageExt(files[i].Filename) || filetil.IsVideoExt(files[i].Filename) {
if filetil.IsImageExt(files[i].Filename) {
attachment.HttpPath = "/" + strings.Replace(strings.TrimPrefix(filePath, conf.WorkingDirectory), "\\", "/", -1) attachment.HttpPath = "/" + strings.Replace(strings.TrimPrefix(filePath, conf.WorkingDirectory), "\\", "/", -1)
if strings.HasPrefix(attachment.HttpPath, "//") { if strings.HasPrefix(attachment.HttpPath, "//") {
attachment.HttpPath = conf.URLForWithCdnImage(string(attachment.HttpPath[1:])) attachment.HttpPath = conf.URLForWithCdnImage(string(attachment.HttpPath[1:]))
@ -697,11 +661,12 @@ func (c *DocumentController) Upload() {
"alt": attachment.FileName, "alt": attachment.FileName,
"is_attach": isAttach, "is_attach": isAttach,
"attach": attachment, "attach": attachment,
"resource_type": attachment.ResourceType,
} }
result2 = append(result2, result) result2 = append(result2, result)
} }
if name == "file" { if len(files) == 1 {
// froala单图片上传 // froala单文件上传
c.Ctx.Output.JSON(result, true, false) c.Ctx.Output.JSON(result, true, false)
} else { } else {
c.Ctx.Output.JSON(result2, true, false) c.Ctx.Output.JSON(result2, true, false)

View File

@ -33,6 +33,7 @@ type Attachment struct {
FileExt string `orm:"column(file_ext);size(50);description(文件后缀)" json:"file_ext"` FileExt string `orm:"column(file_ext);size(50);description(文件后缀)" json:"file_ext"`
CreateTime time.Time `orm:"type(datetime);column(create_time);auto_now_add;description(创建时间)" json:"create_time"` CreateTime time.Time `orm:"type(datetime);column(create_time);auto_now_add;description(创建时间)" json:"create_time"`
CreateAt int `orm:"column(create_at);type(int);description(创建人id)" json:"create_at"` CreateAt int `orm:"column(create_at);type(int);description(创建人id)" json:"create_at"`
ResourceType string `orm:"-" json:"resource_type"`
} }
// TableName 获取对应上传附件数据库表名. // TableName 获取对应上传附件数据库表名.

View File

@ -20,7 +20,8 @@
width: 100%; width: 100%;
overflow: auto; overflow: auto;
border-bottom: none; border-bottom: none;
line-height: 1.5 line-height: 1.5;
display: table;
} }
.editormd-preview-container table td,.editormd-preview-container table th { .editormd-preview-container table td,.editormd-preview-container table th {
@ -50,30 +51,43 @@
width: 100%; width: 100%;
} }
.whole-article-wrap {
display: flex;
flex-direction: column;
}
.article-body .markdown-toc{ .article-body .markdown-toc{
position: fixed; position: fixed;
right: 0; right: 50px;
width: 260px; width: 260px;
font-size: 12px; font-size: 12px;
margin-top: -70px;
overflow: auto; overflow: auto;
margin-right: 50px; border: 1px solid #e8e8e8;
border-radius: 6px;
} }
.markdown-toc ul{ .markdown-toc ul{
list-style:none; list-style:none;
} }
.markdown-toc-list {
padding:20px 0 !important;
margin-bottom: 0 !important;
}
.markdown-toc .markdown-toc-list>li{ .markdown-toc .markdown-toc-list>li{
padding: 3px 10px 3px 16px; padding: 3px 10px 3px 16px;
line-height: 18px; line-height: 18px;
border-left: 2px solid #e8e8e8; /*border-left: 2px solid #e8e8e8;*/
color: #595959; color: #595959;
margin-left: -2px;
} }
.markdown-toc .markdown-toc-list>li.active{ .markdown-toc .markdown-toc-list>li.active{
border-right: 2px solid #25b864; border-right: 2px solid #25b864;
} }
.article-body .markdown-article{ .article-body .markdown-article{
margin-right: 250px; width: calc(100% - 260px);
/*margin-right: 250px;*/
} }
.article-body.content .markdown-toc{ .article-body.content .markdown-toc{
position: relative; position: relative;
@ -86,7 +100,7 @@
.markdown-toc-list .directory-item { .markdown-toc-list .directory-item {
padding: 3px 10px 3px 16px; padding: 3px 10px 3px 16px;
line-height: 18px; line-height: 18px;
border-left: 2px solid #e8e8e8; /*border-left: 2px solid #e8e8e8;*/
color: #595959; color: #595959;
} }
.markdown-toc-list .directory-item-link { .markdown-toc-list .directory-item-link {

View File

@ -3594,7 +3594,7 @@
background-color: #f8f8f8; background-color: #f8f8f8;
} }
.markdown-body img { .markdown-body img, .markdown-body video {
max-width: 100%; max-width: 100%;
-moz-box-sizing: border-box; -moz-box-sizing: border-box;
box-sizing: border-box; box-sizing: border-box;

View File

@ -2878,7 +2878,8 @@
background-color: #f8f8f8; background-color: #f8f8f8;
} }
.markdown-body img {
.markdown-body img, .markdown-body video {
max-width: 100%; max-width: 100%;
-moz-box-sizing: border-box; -moz-box-sizing: border-box;
box-sizing: border-box; box-sizing: border-box;

View File

@ -437,6 +437,85 @@ function uploadImage($id, $callback) {
}); });
} }
function uploadResource($id, $callback) {
locales = {
'zh-CN': {
unsupportType: '/',
uploadFailed: '/'
},
'en': {
unsupportType: 'Unsupport image/video type',
uploadFailed: 'Upload image/video failed'
}
}
/** 粘贴上传的资源 **/
document.getElementById($id).addEventListener('paste', function (e) {
if (e.clipboardData && e.clipboardData.items) {
var clipboard = e.clipboardData;
for (var i = 0, len = clipboard.items.length; i < len; i++) {
if (clipboard.items[i].kind === 'file' || clipboard.items[i].type.indexOf('image') > -1) {
var resource = clipboard.items[i].getAsFile();
var fileName = String((new Date()).valueOf());
console.log(resource.type)
switch (resource.type) {
case "image/png" :
fileName += ".png";
break;
case "image/jpg" :
fileName += ".jpg";
break;
case "image/jpeg" :
fileName += ".jpeg";
break;
case "image/gif" :
fileName += ".gif";
break;
case "video/mp4":
fileName += ".mp4";
break;
case "video/webm":
fileName += ".webm";
break;
default :
layer.msg(locales[lang].unsupportType);
return;
}
var form = new FormData();
form.append('editormd-resource-file', resource, fileName);
var layerIndex = 0;
$.ajax({
url: window.imageUploadURL,
type: "POST",
dataType: "json",
data: form,
processData: false,
contentType: false,
beforeSend: function () {
layerIndex = $callback('before');
},
error: function () {
layer.close(layerIndex);
$callback('error');
layer.msg(locales[lang].uploadFailed);
},
success: function (data) {
layer.close(layerIndex);
$callback('success', data);
}
});
e.preventDefault();
}
}
}
});
}
/** /**
* *
*/ */

View File

@ -143,9 +143,7 @@ function renderPage($data) {
$("#doc_id").val($data.doc_id); $("#doc_id").val($data.doc_id);
if ($data.page) { if ($data.page) {
loadComment($data.page, $data.doc_id); loadComment($data.page, $data.doc_id);
} else {
}
else {
pageClicked(-1, $data.doc_id); pageClicked(-1, $data.doc_id);
} }
@ -156,6 +154,7 @@ function renderPage($data) {
$("#view_container").removeClass("theme__dark theme__green theme__light theme__red theme__default") $("#view_container").removeClass("theme__dark theme__green theme__light theme__red theme__default")
$("#view_container").addClass($data.markdown_theme) $("#view_container").addClass($data.markdown_theme)
} }
checkMarkdownTocElement();
} }
/*** /***
@ -230,6 +229,7 @@ function initHighlighting() {
} }
$(function () { $(function () {
checkMarkdownTocElement();
$(".view-backtop").on("click", function () { $(".view-backtop").on("click", function () {
$('.manual-right').animate({ scrollTop: '0px' }, 200); $('.manual-right').animate({ scrollTop: '0px' }, 200);
}); });
@ -280,7 +280,7 @@ $(function () {
$(window).resize(function (e) { $(window).resize(function (e) {
var h = $(".manual-catalog").innerHeight() - 20; var h = $(".manual-catalog").innerHeight() - 50;
$(".markdown-toc").height(h); $(".markdown-toc").height(h);
}).resize(); }).resize();
@ -418,3 +418,18 @@ function loadCopySnippets() {
Prism.highlightElement(snippet); Prism.highlightElement(snippet);
}); });
} }
function checkMarkdownTocElement() {
console.log(111)
let toc = $(".markdown-toc-list");
let articleComment = $("#articleComment");
if (toc.length) {
$(".wiki-bottom-left").css("width", "calc(100% - 260px)");
articleComment.css("width", "calc(100% - 260px)");
articleComment.css("margin", "30px 0 70px 0");
} else {
$(".wiki-bottom-left").css("width", "100%");
articleComment.css("width", "100%");
articleComment.css("margin", "30px auto 70px auto;");
}
}

View File

@ -245,18 +245,22 @@ $(function () {
//如果没有选中节点则选中默认节点 //如果没有选中节点则选中默认节点
openLastSelectedNode(); openLastSelectedNode();
uploadImage("docEditor", function ($state, $res) { uploadResource("docEditor", function ($state, $res) {
if ($state === "before") { if ($state === "before") {
return layer.load(1, { return layer.load(1, {
shade: [0.1, '#fff'] // 0.1 透明度的白色背景 shade: [0.1, '#fff'] // 0.1 透明度的白色背景
}); });
} else if ($state === "success") { } else if ($state === "success") {
// if ($res.errcode === 0) { if ($res.errcode === 0) {
// var value = '![](' + $res.url + ')'; if ($res.resource_type === 'video') {
// 3xxx 20240602 let value = `<video controls><source src="${$res.url}" type="video/mp4"></video>`;
if ($res[0].errcode === 0) {
var value = '![](' + $res[0].url + ')';
window.editor.insertValue(value); window.editor.insertValue(value);
} else {
let value = '![](' + $res.url + ')';
window.editor.insertValue(value);
}
} else {
layer.msg("上传失败:" + $res.message);
} }
} }
}); });

View File

@ -1,20 +1,42 @@
package filetil package filetil
import ( import (
"bytes"
"fmt"
"io"
"io/ioutil"
"math"
"os" "os"
"path/filepath" "path/filepath"
"strings" "strings"
"io"
"fmt"
"math"
"io/ioutil"
"bytes"
) )
//================================== //==================================
//更多文件和目录的操作使用filepath包和os包 //更多文件和目录的操作使用filepath包和os包
//================================== //==================================
type FileTypeStrategy interface {
GetFilePath(filePath, fileName, ext string) string
}
type ImageStrategy struct{}
func (i ImageStrategy) GetFilePath(filePath, fileName, ext string) string {
return filepath.Join(filePath, "images", fileName+ext)
}
type VideoStrategy struct{}
func (v VideoStrategy) GetFilePath(filePath, fileName, ext string) string {
return filepath.Join(filePath, "videos", fileName+ext)
}
type DefaultStrategy struct{}
func (d DefaultStrategy) GetFilePath(filePath, fileName, ext string) string {
return filepath.Join(filePath, "files", fileName+ext)
}
// 返回的目录扫描结果 // 返回的目录扫描结果
type FileList struct { type FileList struct {
IsDir bool //是否是目录 IsDir bool //是否是目录
@ -66,7 +88,6 @@ func CopyFile(source string, dst string) (err error) {
} }
} }
destFile, err := os.Create(dst) destFile, err := os.Create(dst)
if err != nil { if err != nil {
return err return err
@ -224,6 +245,7 @@ func HasFileOfExt(path string,exts []string) bool {
return err == os.ErrExist return err == os.ErrExist
} }
// IsImageExt 判断是否是图片后缀 // IsImageExt 判断是否是图片后缀
func IsImageExt(filename string) bool { func IsImageExt(filename string) bool {
ext := filepath.Ext(filename) ext := filepath.Ext(filename)
@ -236,6 +258,19 @@ func IsImageExt(filename string) bool {
strings.EqualFold(ext, ".bmp") || strings.EqualFold(ext, ".bmp") ||
strings.EqualFold(ext, ".webp") strings.EqualFold(ext, ".webp")
} }
// IsImageExt 判断是否是视频后缀
func IsVideoExt(filename string) bool {
ext := filepath.Ext(filename)
return strings.EqualFold(ext, ".mp4") ||
strings.EqualFold(ext, ".webm") ||
strings.EqualFold(ext, ".ogg") ||
strings.EqualFold(ext, ".avi") ||
strings.EqualFold(ext, ".flv") ||
strings.EqualFold(ext, ".mov")
}
// 忽略字符串中的BOM头 // 忽略字符串中的BOM头
func ReadFileAndIgnoreUTF8BOM(filename string) ([]byte, error) { func ReadFileAndIgnoreUTF8BOM(filename string) ([]byte, error) {
@ -251,6 +286,5 @@ func ReadFileAndIgnoreUTF8BOM(filename string) ([]byte,error) {
return data[3:], err return data[3:], err
} }
return data, nil return data, nil
} }

View File

@ -508,7 +508,6 @@
} }
}).on("uploadSuccess",function (file, res) { }).on("uploadSuccess",function (file, res) {
for(var index in window.vueApp.lists){ for(var index in window.vueApp.lists){
var item = window.vueApp.lists[index]; var item = window.vueApp.lists[index];
if(item.attachment_id === file.id){ if(item.attachment_id === file.id){