更新
This commit is contained in:
parent
608bff9df1
commit
05a76e265f
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
node_modules/
|
@ -39,55 +39,11 @@ if (Helper::options()->JEditor !== 'off') {
|
|||||||
class Editor
|
class Editor
|
||||||
{
|
{
|
||||||
public static function Edit()
|
public static function Edit()
|
||||||
{ ?>
|
{
|
||||||
<script>
|
?>
|
||||||
var JoeUploadURL = '<?php Helper::security()->index('/action/upload'); ?>';
|
<link rel="stylesheet" href="<?php Helper::options()->themeUrl('typecho/write/css/joe.write.min.css') ?>">
|
||||||
</script>
|
<script src="https://cdn.jsdelivr.net/npm/hyperdown@2.4.10/Parser.min.js"></script>
|
||||||
|
<script api="<?php Helper::security()->index('/action/upload'); ?>" src="<?php Helper::options()->themeUrl('typecho/write/js/joe.write.chunk.js') ?>"></script>
|
||||||
<?php
|
<?php
|
||||||
/* 编辑器核心 */
|
|
||||||
echo '<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/lib/codemirror.min.css">';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/lib/codemirror.min.js"></script>';
|
|
||||||
/* 编辑器语法高亮 */
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/markdown/markdown.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/javascript/javascript.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/css/css.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/xml/xml.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/gfm/gfm.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/pug/pug.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/sass/sass.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/stylus/stylus.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/clike/clike.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/htmlmixed/htmlmixed.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/vue/vue.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/swift/swift.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/sql/sql.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/shell/shell.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/python/python.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/php/php.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/go/go.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/yaml/yaml.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/yaml-frontmatter/yaml-frontmatter.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/mode/meta.min.js"></script>';
|
|
||||||
/* 编辑器附加功能 */
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/edit/continuelist.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/selection/active-line.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/edit/matchbrackets.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/edit/closebrackets.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/selection/selection-pointer.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/display/fullscreen.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/edit/closetag.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/mode/overlay.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/edit/matchtags.min.js"></script>';
|
|
||||||
/* 代码折行 */
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/fold/foldcode.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/fold/foldgutter.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/fold/xml-fold.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/fold/markdown-fold.min.js"></script>';
|
|
||||||
echo '<script src="https://cdn.jsdelivr.net/npm/codemirror@5.59.4/addon/fold/brace-fold.min.js"></script>';
|
|
||||||
/* 引入本地文件 */
|
|
||||||
echo '<link rel="stylesheet" href="/usr/themes/Joe/typecho/editor/css/joe.editor.min.css?v=2021314">';
|
|
||||||
echo '<script src="/usr/themes/Joe/typecho/editor/js/joe.editor.min.js?v=2021314"></script>';
|
|
||||||
echo '<script src="/usr/themes/Joe/typecho/editor/js/joe.constructor.min.js?v=2021314"></script>';
|
|
||||||
echo '<script src="/usr/themes/Joe/typecho/editor/js/joe.instance.min.js?v=2021314"></script>';
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
/* 获取主题当前版本号 */
|
/* 获取主题当前版本号 */
|
||||||
function _getVersion()
|
function _getVersion()
|
||||||
{
|
{
|
||||||
return "6.1.0";
|
return "6.1.1";
|
||||||
};
|
};
|
||||||
|
|
||||||
/* 判断是否是手机 */
|
/* 判断是否是手机 */
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "typecho-joe-next",
|
"name": "typecho-joe-next",
|
||||||
"version": "6.1.0",
|
"version": "6.1.1",
|
||||||
"description": "A Theme Of Typecho",
|
"description": "A Theme Of Typecho",
|
||||||
"main": "index.php",
|
"main": "index.php",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
1
typecho/editor/css/joe.editor.min.css
vendored
1
typecho/editor/css/joe.editor.min.css
vendored
File diff suppressed because one or more lines are too long
@ -1,343 +0,0 @@
|
|||||||
input[name="fields[keywords]"] {
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
textarea[name="fields[description]"],
|
|
||||||
textarea[name="fields[abstract]"],
|
|
||||||
textarea[name="fields[thumb]"],
|
|
||||||
textarea[name="fields[video]"] {
|
|
||||||
width: 100%;
|
|
||||||
height: 80px;
|
|
||||||
}
|
|
||||||
|
|
||||||
textarea[id="text"],
|
|
||||||
span[class="resize"] {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.CodeMirror {
|
|
||||||
&-progress {
|
|
||||||
display: none;
|
|
||||||
position: absolute;
|
|
||||||
z-index: 6688;
|
|
||||||
right: 100%;
|
|
||||||
top: 0;
|
|
||||||
width: 100%;
|
|
||||||
height: 2px;
|
|
||||||
border-radius: 1px;
|
|
||||||
background: linear-gradient(to right, #4cd964, #5ac8fa, #007aff);
|
|
||||||
transition: right 1s linear;
|
|
||||||
}
|
|
||||||
&-fullscreen {
|
|
||||||
position: fixed;
|
|
||||||
top: 35px;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
bottom: 0;
|
|
||||||
height: auto;
|
|
||||||
z-index: 999;
|
|
||||||
}
|
|
||||||
&-foldmarker {
|
|
||||||
cursor: pointer;
|
|
||||||
padding-left: 3px;
|
|
||||||
font-family: "Helvetica Neue", Helvetica, "PingFang SC", "Hiragino Sans GB",
|
|
||||||
"Microsoft YaHei", "微软雅黑", Arial, sans-serif;
|
|
||||||
transition: color 0.25s;
|
|
||||||
user-select: none;
|
|
||||||
}
|
|
||||||
&-foldgutter {
|
|
||||||
width: 12px;
|
|
||||||
user-select: none;
|
|
||||||
&-open,
|
|
||||||
&-folded {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
width: 100%;
|
|
||||||
height: 22px;
|
|
||||||
cursor: pointer;
|
|
||||||
&::before {
|
|
||||||
content: "";
|
|
||||||
display: block;
|
|
||||||
width: 11px;
|
|
||||||
height: 11px;
|
|
||||||
background: url();
|
|
||||||
background-size: 100% 100%;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
&-folded {
|
|
||||||
&::before {
|
|
||||||
transform: rotate(-90deg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
&.cm-s-joe {
|
|
||||||
font-size: 15px;
|
|
||||||
line-height: 1.5;
|
|
||||||
font-family: Consolas, Monaco, "Andale Mono", "Ubuntu Mono", monospace;
|
|
||||||
background: #2d2d2d;
|
|
||||||
color: #ccc;
|
|
||||||
/* 占位符 */
|
|
||||||
.CodeMirror-placeholder {
|
|
||||||
color: #666;
|
|
||||||
}
|
|
||||||
/* 选择文字时候的颜色 */
|
|
||||||
.CodeMirror-selected {
|
|
||||||
background: rgba(221, 240, 255, 0.2);
|
|
||||||
}
|
|
||||||
/* 行号 */
|
|
||||||
.CodeMirror-gutters {
|
|
||||||
background: #2d2d2d;
|
|
||||||
border-right: none;
|
|
||||||
padding: 0 3px;
|
|
||||||
}
|
|
||||||
/* 行号字体色 */
|
|
||||||
.CodeMirror-linenumber {
|
|
||||||
color: #8f938f;
|
|
||||||
}
|
|
||||||
/* 光标颜色 */
|
|
||||||
.CodeMirror-cursor {
|
|
||||||
border-left: 1px solid #a7a7a7;
|
|
||||||
}
|
|
||||||
/* 当前行高亮 */
|
|
||||||
.CodeMirror-activeline-background {
|
|
||||||
background: rgba(255, 255, 255, 0.031);
|
|
||||||
}
|
|
||||||
/* 括号匹配 */
|
|
||||||
.CodeMirror-matchingbracket {
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.25);
|
|
||||||
color: #8f938f;
|
|
||||||
margin: -1px -1px 0 -1px;
|
|
||||||
}
|
|
||||||
/* 1 - 6级标题样式 */
|
|
||||||
.cm-header-1,
|
|
||||||
.cm-header-2,
|
|
||||||
.cm-header-3,
|
|
||||||
.cm-header-4,
|
|
||||||
.cm-header-5,
|
|
||||||
.cm-header-6 {
|
|
||||||
font-size: 18px;
|
|
||||||
color: #569cd6 !important;
|
|
||||||
}
|
|
||||||
/* 加粗/倾斜 */
|
|
||||||
.cm-strong,
|
|
||||||
.cm-em {
|
|
||||||
color: #569cd6;
|
|
||||||
}
|
|
||||||
/* 代码高亮样式 */
|
|
||||||
.cm-keyword {
|
|
||||||
color: #cc99cd;
|
|
||||||
}
|
|
||||||
.cm-string {
|
|
||||||
color: #7ec699;
|
|
||||||
}
|
|
||||||
.cm-operator {
|
|
||||||
color: #67cdcc;
|
|
||||||
}
|
|
||||||
.cm-number {
|
|
||||||
color: #f08d49;
|
|
||||||
}
|
|
||||||
.cm-def {
|
|
||||||
color: #f08d49;
|
|
||||||
}
|
|
||||||
.cm-tag {
|
|
||||||
color: #e2777a;
|
|
||||||
}
|
|
||||||
.cm-attribute {
|
|
||||||
color: #e2777a;
|
|
||||||
}
|
|
||||||
.cm-bracket {
|
|
||||||
color: #ccc;
|
|
||||||
}
|
|
||||||
.cm-qualifier {
|
|
||||||
color: #cc99cd;
|
|
||||||
}
|
|
||||||
.cm-property {
|
|
||||||
color: #f8c555;
|
|
||||||
}
|
|
||||||
.cm-comment {
|
|
||||||
color: #666;
|
|
||||||
}
|
|
||||||
/* 文本引用 */
|
|
||||||
.cm-quote {
|
|
||||||
color: #ae81ff;
|
|
||||||
}
|
|
||||||
/* 列表 */
|
|
||||||
.cm-variable-2 {
|
|
||||||
color: #de8e30;
|
|
||||||
}
|
|
||||||
/* 超链接 */
|
|
||||||
.cm-image,
|
|
||||||
.cm-link {
|
|
||||||
color: #f56c6c;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
&-means {
|
|
||||||
display: flex;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
background: #222;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
&.fullscreen {
|
|
||||||
position: fixed;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
z-index: 1000;
|
|
||||||
}
|
|
||||||
&-item {
|
|
||||||
position: relative;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
list-style: none;
|
|
||||||
width: 40px;
|
|
||||||
height: 35px;
|
|
||||||
cursor: pointer;
|
|
||||||
border-right: 1px solid #2d2d2d;
|
|
||||||
border-bottom: 1px solid #2d2d2d;
|
|
||||||
box-sizing: border-box;
|
|
||||||
&:hover {
|
|
||||||
background: #2d2d2d;
|
|
||||||
}
|
|
||||||
&.active {
|
|
||||||
background: #2d2d2d;
|
|
||||||
.CodeMirror-means__dropdown {
|
|
||||||
visibility: visible;
|
|
||||||
opacity: 1;
|
|
||||||
transform: translateX(-50%) rotateX(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
&__dropdown {
|
|
||||||
position: absolute;
|
|
||||||
top: 42px;
|
|
||||||
left: 50%;
|
|
||||||
z-index: 1000;
|
|
||||||
color: #bfbfbf;
|
|
||||||
background: #222;
|
|
||||||
padding: 5px 0;
|
|
||||||
border-radius: 4px;
|
|
||||||
visibility: hidden;
|
|
||||||
transform-origin: top;
|
|
||||||
opacity: 0;
|
|
||||||
transform: translateX(-50%) rotateX(-90deg);
|
|
||||||
transition: visibility 0.25s, transform 0.25s, opacity 0.25s;
|
|
||||||
&::before {
|
|
||||||
content: "";
|
|
||||||
position: absolute;
|
|
||||||
top: -7px;
|
|
||||||
left: 50%;
|
|
||||||
transform: translateX(-50%);
|
|
||||||
width: 0;
|
|
||||||
height: 0;
|
|
||||||
border-left: 7px solid transparent;
|
|
||||||
border-right: 7px solid transparent;
|
|
||||||
border-bottom: 7px solid #222;
|
|
||||||
}
|
|
||||||
&-item {
|
|
||||||
padding: 0 20px;
|
|
||||||
line-height: 34px;
|
|
||||||
transition: background 0.25s;
|
|
||||||
&:hover {
|
|
||||||
color: #f2f2f2;
|
|
||||||
background: #2d2d2d;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
&-dialog {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
position: fixed;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
bottom: 0;
|
|
||||||
z-index: 6666;
|
|
||||||
background: rgba(0, 0, 0, 0);
|
|
||||||
visibility: hidden;
|
|
||||||
transition: background 0.25s, visibility 0.25s;
|
|
||||||
&__wrapper {
|
|
||||||
width: 380px;
|
|
||||||
background: #fff;
|
|
||||||
color: #606266;
|
|
||||||
border-radius: 4px;
|
|
||||||
opacity: 0;
|
|
||||||
transform: translateY(-30%);
|
|
||||||
transition: opacity 0.25s, transform 0.25s;
|
|
||||||
&-header {
|
|
||||||
padding: 12px 20px;
|
|
||||||
border-bottom: 1px solid #e4e7ed;
|
|
||||||
}
|
|
||||||
&-bodyer {
|
|
||||||
padding: 20px;
|
|
||||||
.fitem {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
margin-bottom: 15px;
|
|
||||||
&:last-child {
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
label {
|
|
||||||
margin-right: 10px;
|
|
||||||
}
|
|
||||||
input {
|
|
||||||
width: auto;
|
|
||||||
flex: 1;
|
|
||||||
-webkit-appearance: none;
|
|
||||||
outline: none;
|
|
||||||
border: 1px solid #dcdfe6;
|
|
||||||
height: 34px;
|
|
||||||
box-sizing: border-box;
|
|
||||||
padding: 0 10px;
|
|
||||||
border-radius: 3px;
|
|
||||||
color: #606266;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
&-footer {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: flex-end;
|
|
||||||
padding: 0 20px 10px;
|
|
||||||
&--cancle,
|
|
||||||
&--confirm {
|
|
||||||
cursor: pointer;
|
|
||||||
border: none;
|
|
||||||
outline: none;
|
|
||||||
margin-left: 5px;
|
|
||||||
background: #fff;
|
|
||||||
color: #606266;
|
|
||||||
-webkit-appearance: none;
|
|
||||||
padding: 7px 15px;
|
|
||||||
border-radius: 3px;
|
|
||||||
}
|
|
||||||
&--confirm {
|
|
||||||
color: #fff;
|
|
||||||
background: #409eff;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
&.active {
|
|
||||||
visibility: visible;
|
|
||||||
background: rgba(0, 0, 0, 0.75);
|
|
||||||
.CodeMirror-dialog__wrapper {
|
|
||||||
transform: translateY(0);
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
.CodeMirror {
|
|
||||||
&-dialog {
|
|
||||||
&__wrapper {
|
|
||||||
width: 90%;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,380 +0,0 @@
|
|||||||
class JoeInstance {
|
|
||||||
constructor() {
|
|
||||||
this.init();
|
|
||||||
}
|
|
||||||
|
|
||||||
init() {
|
|
||||||
$(".CodeMirror-wrap").before('<ul class="CodeMirror-means"></ul>');
|
|
||||||
$(".CodeMirror-wrap").append('<div class="CodeMirror-progress"></div>');
|
|
||||||
$("body").append(`
|
|
||||||
<div class="CodeMirror-dialog">
|
|
||||||
<div class="CodeMirror-dialog__wrapper">
|
|
||||||
<div class="CodeMirror-dialog__wrapper-header"></div>
|
|
||||||
<div class="CodeMirror-dialog__wrapper-bodyer"></div>
|
|
||||||
<div class="CodeMirror-dialog__wrapper-footer">
|
|
||||||
<button class="CodeMirror-dialog__wrapper-footer--cancle">取消</button>
|
|
||||||
<button class="CodeMirror-dialog__wrapper-footer--confirm">确定</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`);
|
|
||||||
$(".CodeMirror-dialog__wrapper-footer--cancle").on("click", () => {
|
|
||||||
this.options.cancel();
|
|
||||||
$(".CodeMirror-dialog").removeClass("active");
|
|
||||||
$("body").css("overflow", "");
|
|
||||||
});
|
|
||||||
$(".CodeMirror-dialog__wrapper-footer--confirm").on("click", () => {
|
|
||||||
this.options.confirm();
|
|
||||||
$(".CodeMirror-dialog").removeClass("active");
|
|
||||||
$("body").css("overflow", "");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 打开弹窗 */
|
|
||||||
__OpenTheDialog(options = {}) {
|
|
||||||
const _options = {
|
|
||||||
title: "提示",
|
|
||||||
innerHtml: "内容",
|
|
||||||
cancel: () => {},
|
|
||||||
confirm: () => {},
|
|
||||||
};
|
|
||||||
this.options = Object.assign(_options, options);
|
|
||||||
$(".CodeMirror-dialog__wrapper-header").html(this.options.title);
|
|
||||||
$(".CodeMirror-dialog__wrapper-bodyer").html(this.options.innerHtml);
|
|
||||||
$(".CodeMirror-dialog").addClass("active");
|
|
||||||
$("body").css("overflow", "hidden");
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 初始化插入标题功能 - 已测试 √ */
|
|
||||||
initTitleText(tool) {
|
|
||||||
const item = $(`
|
|
||||||
<li class="CodeMirror-means-item" title="${tool.title}">
|
|
||||||
${tool.icon}
|
|
||||||
<div class="CodeMirror-means__dropdown">
|
|
||||||
<div class="CodeMirror-means__dropdown-item" data-text="# "> H1 </div>
|
|
||||||
<div class="CodeMirror-means__dropdown-item" data-text="## "> H2 </div>
|
|
||||||
<div class="CodeMirror-means__dropdown-item" data-text="### "> H3 </div>
|
|
||||||
<div class="CodeMirror-means__dropdown-item" data-text="#### "> H4 </div>
|
|
||||||
<div class="CodeMirror-means__dropdown-item" data-text="##### "> H5 </div>
|
|
||||||
<div class="CodeMirror-means__dropdown-item" data-text="###### "> H6 </div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
`);
|
|
||||||
item.on("click", function (e) {
|
|
||||||
e.stopPropagation();
|
|
||||||
$(this).toggleClass("active");
|
|
||||||
});
|
|
||||||
item.on("click", ".CodeMirror-means__dropdown-item", function (e) {
|
|
||||||
e.stopPropagation();
|
|
||||||
const text = $(this).attr("data-text");
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
if (cursor.ch === 0) JoeEditor.replaceSelection(text);
|
|
||||||
else JoeEditor.replaceSelection("\n\n" + text);
|
|
||||||
item.removeClass("active");
|
|
||||||
JoeEditor.focus();
|
|
||||||
});
|
|
||||||
$(document).on("click", () => item.removeClass("active"));
|
|
||||||
$(".CodeMirror-means").append(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 全屏/取消全屏 - 已测试 √ */
|
|
||||||
handleFullscreen(tool) {
|
|
||||||
const toolsHeight = $(".CodeMirror-means").height();
|
|
||||||
const item = $(
|
|
||||||
`<li class="CodeMirror-means-item" title="${tool.title}">${tool.icon}</li>`
|
|
||||||
);
|
|
||||||
item.on("click", function () {
|
|
||||||
const bool = JoeEditor.getOption("fullScreen");
|
|
||||||
if (bool) {
|
|
||||||
JoeEditor.setOption("fullScreen", false);
|
|
||||||
$(".CodeMirror-wrap").css("top", "");
|
|
||||||
$(".CodeMirror-means").removeClass("fullscreen");
|
|
||||||
$(item).removeClass("active");
|
|
||||||
} else {
|
|
||||||
JoeEditor.setOption("fullScreen", true);
|
|
||||||
$(".CodeMirror-wrap").css("top", toolsHeight);
|
|
||||||
$(".CodeMirror-means").addClass("fullscreen");
|
|
||||||
$(item).addClass("active");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
$(".CodeMirror-means").append(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 加粗 - 已测试 √ */
|
|
||||||
insertBoldText() {
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const selection = JoeEditor.getSelection();
|
|
||||||
JoeEditor.replaceSelection("**" + selection + "**");
|
|
||||||
if (selection === "") JoeEditor.setCursor(cursor.line, cursor.ch + 2);
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 倾斜 - 已测试 √ */
|
|
||||||
insertItalicText() {
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const selection = JoeEditor.getSelection();
|
|
||||||
JoeEditor.replaceSelection("*" + selection + "*");
|
|
||||||
if (selection === "") JoeEditor.setCursor(cursor.line, cursor.ch + 1);
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 删除线 - 已测试 √ */
|
|
||||||
insertDeleteText() {
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const selection = JoeEditor.getSelection();
|
|
||||||
JoeEditor.replaceSelection("~~" + selection + "~~");
|
|
||||||
if (selection === "") JoeEditor.setCursor(cursor.line, cursor.ch + 2);
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 引用 - 已测试 √ */
|
|
||||||
insertQuoteText() {
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const selection = JoeEditor.getSelection();
|
|
||||||
if (cursor.ch === 0) {
|
|
||||||
JoeEditor.replaceSelection("> " + selection);
|
|
||||||
} else {
|
|
||||||
JoeEditor.setCursor(cursor.line, 0);
|
|
||||||
JoeEditor.replaceSelection("> " + selection);
|
|
||||||
JoeEditor.setCursor(cursor.line, cursor.ch + 2);
|
|
||||||
}
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 空格 - 已测试 √ */
|
|
||||||
insertSpaceText() {
|
|
||||||
JoeEditor.replaceSelection(" ");
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 行内代码 - 已测试 √ */
|
|
||||||
insertCodeInlineText() {
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const selection = JoeEditor.getSelection();
|
|
||||||
JoeEditor.replaceSelection("`" + selection + "`");
|
|
||||||
if (selection === "") JoeEditor.setCursor(cursor.line, cursor.ch + 1);
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 横线 - 已测试 √ */
|
|
||||||
insertHrText() {
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
if (cursor.ch === 0) JoeEditor.replaceSelection("\n------------\n\n");
|
|
||||||
else JoeEditor.replaceSelection("\n\n------------\n\n");
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 无序列表 - 已测试 √ */
|
|
||||||
insertUnorderedListText() {
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const selection = JoeEditor.getSelection();
|
|
||||||
if (cursor.ch === 0) {
|
|
||||||
if (selection === "") {
|
|
||||||
JoeEditor.replaceSelection("- ");
|
|
||||||
} else {
|
|
||||||
const selectionText = selection.split("\n");
|
|
||||||
for (let i = 0, len = selectionText.length; i < len; i++) {
|
|
||||||
selectionText[i] =
|
|
||||||
selectionText[i] === "" ? "" : "- " + selectionText[i];
|
|
||||||
}
|
|
||||||
JoeEditor.replaceSelection(selectionText.join("\n"));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (selection === "") {
|
|
||||||
JoeEditor.replaceSelection("\n- ");
|
|
||||||
} else {
|
|
||||||
const selectionText = selection.split("\n");
|
|
||||||
for (let i = 0, len = selectionText.length; i < len; i++) {
|
|
||||||
selectionText[i] =
|
|
||||||
selectionText[i] === "" ? "" : "- " + selectionText[i];
|
|
||||||
}
|
|
||||||
JoeEditor.replaceSelection("\n" + selectionText.join("\n"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 有序列表 - 已测试 √ */
|
|
||||||
insertOrderedListText() {
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const selection = JoeEditor.getSelection();
|
|
||||||
if (cursor.ch === 0) {
|
|
||||||
if (selection === "") {
|
|
||||||
JoeEditor.replaceSelection("1. ");
|
|
||||||
} else {
|
|
||||||
const selectionText = selection.split("\n");
|
|
||||||
for (let i = 0, len = selectionText.length; i < len; i++) {
|
|
||||||
selectionText[i] =
|
|
||||||
selectionText[i] === ""
|
|
||||||
? ""
|
|
||||||
: i + 1 + ". " + selectionText[i];
|
|
||||||
}
|
|
||||||
JoeEditor.replaceSelection(selectionText.join("\n"));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (selection === "") {
|
|
||||||
JoeEditor.replaceSelection("\n\n1. ");
|
|
||||||
} else {
|
|
||||||
const selectionText = selection.split("\n");
|
|
||||||
for (let i = 0, len = selectionText.length; i < len; i++) {
|
|
||||||
selectionText[i] =
|
|
||||||
selectionText[i] === ""
|
|
||||||
? ""
|
|
||||||
: i + 1 + ". " + selectionText[i];
|
|
||||||
}
|
|
||||||
JoeEditor.replaceSelection("\n" + selectionText.join("\n"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 插入时间 - 已测试 √ */
|
|
||||||
insertTimeText() {
|
|
||||||
const time = new Date();
|
|
||||||
const _Year = time.getFullYear();
|
|
||||||
const _Month = String(time.getMonth() + 1).padStart(2, 0);
|
|
||||||
const _Date = String(time.getDate()).padStart(2, 0);
|
|
||||||
const _Hours = String(time.getHours()).padStart(2, 0);
|
|
||||||
const _Minutes = String(time.getMinutes()).padStart(2, 0);
|
|
||||||
const _Seconds = String(time.getSeconds()).padStart(2, 0);
|
|
||||||
const _Day = [
|
|
||||||
"星期日",
|
|
||||||
"星期一",
|
|
||||||
"星期二",
|
|
||||||
"星期三",
|
|
||||||
"星期四",
|
|
||||||
"星期五",
|
|
||||||
"星期六",
|
|
||||||
][time.getDay()];
|
|
||||||
const _time = `${_Year}-${_Month}-${_Date} ${_Hours}:${_Minutes}:${_Seconds} ${_Day}`;
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
if (cursor.ch === 0) JoeEditor.replaceSelection(_time);
|
|
||||||
else JoeEditor.replaceSelection("\n" + _time);
|
|
||||||
JoeEditor.focus();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 插入URL链接 - 已测试 √ */
|
|
||||||
insertLinkText() {
|
|
||||||
this.__OpenTheDialog({
|
|
||||||
title: "插入链接",
|
|
||||||
innerHtml: `
|
|
||||||
<div class="fitem">
|
|
||||||
<label>链接标题</label>
|
|
||||||
<input autocomplete="off" name="title" placeholder="请输入链接标题"/>
|
|
||||||
</div>
|
|
||||||
<div class="fitem">
|
|
||||||
<label>链接地址</label>
|
|
||||||
<input autocomplete="off" name="url" placeholder="请输入链接地址"/>
|
|
||||||
</div>
|
|
||||||
`,
|
|
||||||
confirm() {
|
|
||||||
const title = $(".CodeMirror-dialog input[name='title']").val();
|
|
||||||
const url = $(".CodeMirror-dialog input[name='url']").val();
|
|
||||||
JoeEditor.replaceSelection(
|
|
||||||
`[${title || "默认标题"}](${url || "默认地址"})`
|
|
||||||
);
|
|
||||||
JoeEditor.focus();
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 插入图片 - 已测试 √ */
|
|
||||||
insertImageText() {
|
|
||||||
this.__OpenTheDialog({
|
|
||||||
title: "插入图片",
|
|
||||||
innerHtml: `
|
|
||||||
<div class="fitem">
|
|
||||||
<label>图片名称</label>
|
|
||||||
<input autocomplete="off" name="title" placeholder="请输入图片名称"/>
|
|
||||||
</div>
|
|
||||||
<div class="fitem">
|
|
||||||
<label>图片地址</label>
|
|
||||||
<input autocomplete="off" name="url" placeholder="请输入图片地址"/>
|
|
||||||
</div>
|
|
||||||
`,
|
|
||||||
confirm() {
|
|
||||||
const title = $(".CodeMirror-dialog input[name='title']").val();
|
|
||||||
const url = $(".CodeMirror-dialog input[name='url']").val();
|
|
||||||
JoeEditor.replaceSelection(
|
|
||||||
`![${title || "默认图片"}](${url || "默认地址"})`
|
|
||||||
);
|
|
||||||
JoeEditor.focus();
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 插入表格 - 已测试 √ */
|
|
||||||
insertTableText() {
|
|
||||||
this.__OpenTheDialog({
|
|
||||||
title: "插入表格",
|
|
||||||
innerHtml: `
|
|
||||||
<div class="fitem">
|
|
||||||
<label>表格行</label>
|
|
||||||
<input style="width: 50px; flex: none; margin-right: 10px;" value="3" autocomplete="off" name="row"/>
|
|
||||||
<label>表格列</label>
|
|
||||||
<input style="width: 50px; flex: none;" value="3" autocomplete="off" name="column"/>
|
|
||||||
</div>
|
|
||||||
`,
|
|
||||||
confirm() {
|
|
||||||
let row = $(".CodeMirror-dialog input[name='row']").val();
|
|
||||||
let column = $(".CodeMirror-dialog input[name='column']").val();
|
|
||||||
if (isNaN(row)) row = 3;
|
|
||||||
if (isNaN(column)) column = 3;
|
|
||||||
let rowStr = "";
|
|
||||||
let rangeStr = "";
|
|
||||||
let columnlStr = "";
|
|
||||||
for (let i = 0; i < column; i++) {
|
|
||||||
rowStr += "| 表头 ";
|
|
||||||
rangeStr += "| :--: ";
|
|
||||||
}
|
|
||||||
for (let i = 0; i < row; i++) {
|
|
||||||
for (let j = 0; j < column; j++) columnlStr += "| 表格 ";
|
|
||||||
columnlStr += "|\n";
|
|
||||||
}
|
|
||||||
const htmlStr = `${rowStr}|\n${rangeStr}|\n${columnlStr}\n`;
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
if (cursor.ch === 0) JoeEditor.replaceSelection(htmlStr);
|
|
||||||
else JoeEditor.replaceSelection("\n\n" + htmlStr);
|
|
||||||
JoeEditor.focus();
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 插入代码块 - 已测试 √ */
|
|
||||||
insertCodeBlockText() {
|
|
||||||
this.__OpenTheDialog({
|
|
||||||
title: "插入代码块",
|
|
||||||
innerHtml: `
|
|
||||||
<div class="fitem">
|
|
||||||
<label>语言类型</label>
|
|
||||||
<input autocomplete="off" name="type" placeholder="请输入语言类型(英文)"/>
|
|
||||||
</div>
|
|
||||||
`,
|
|
||||||
confirm() {
|
|
||||||
const type = $(".CodeMirror-dialog input[name='type']").val();
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const htmlStr = `\`\`\`${
|
|
||||||
type || "html"
|
|
||||||
}\ncode here...\n\`\`\`\n\n`;
|
|
||||||
if (cursor.ch === 0) JoeEditor.replaceSelection(htmlStr);
|
|
||||||
else JoeEditor.replaceSelection("\n\n" + htmlStr);
|
|
||||||
JoeEditor.focus();
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 关于 */
|
|
||||||
handleAbout() {
|
|
||||||
this.__OpenTheDialog({
|
|
||||||
title: "关于",
|
|
||||||
innerHtml: `
|
|
||||||
<ul>
|
|
||||||
<li>文件读取教程(将文件推拽至编辑器即可)</li>
|
|
||||||
<li>图片粘贴上传教程(不支持本地文件复制粘贴,支持网络图片复制、截图等)</li>
|
|
||||||
<li>本编辑器仅为Joe主题使用,未经允许不得移植至其他主题!</li>
|
|
||||||
</ul>
|
|
||||||
`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
window.JoeInstance = JoeInstance;
|
|
1
typecho/editor/js/joe.constructor.min.js
vendored
1
typecho/editor/js/joe.constructor.min.js
vendored
File diff suppressed because one or more lines are too long
@ -1,134 +0,0 @@
|
|||||||
$(function () {
|
|
||||||
$("head").append(
|
|
||||||
`<meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, shrink-to-fit=no, viewport-fit=cover">`
|
|
||||||
);
|
|
||||||
|
|
||||||
window.JoeEditor = CodeMirror.fromTextArea(
|
|
||||||
document.querySelector("textarea#text"),
|
|
||||||
{
|
|
||||||
theme: "joe",
|
|
||||||
mode: "gfm",
|
|
||||||
/* Tab大小 */
|
|
||||||
indentUnit: 4,
|
|
||||||
/* 显示行号 */
|
|
||||||
lineNumbers: true,
|
|
||||||
/* 超出宽度自动换行 */
|
|
||||||
lineWrapping: true,
|
|
||||||
/* 当前行背景高亮 */
|
|
||||||
styleActiveLine: true,
|
|
||||||
/* 匹配括号 */
|
|
||||||
matchBrackets: true,
|
|
||||||
/* 自动闭合括号 */
|
|
||||||
autoCloseBrackets: true,
|
|
||||||
/* 自动闭合标签 */
|
|
||||||
autoCloseTags: true,
|
|
||||||
/* 折行 */
|
|
||||||
foldGutter: true,
|
|
||||||
gutters: ["CodeMirror-linenumbers", "CodeMirror-foldgutter"],
|
|
||||||
/* 匹配标签 */
|
|
||||||
matchTags: { bothTags: true },
|
|
||||||
/* 按键 */
|
|
||||||
extraKeys: {
|
|
||||||
/* 换行时自动联想输入的类型 */
|
|
||||||
Enter: "newlineAndIndentContinueMarkdownList",
|
|
||||||
/* 匹配标签 */
|
|
||||||
"Ctrl-J": "toMatchingTag",
|
|
||||||
},
|
|
||||||
/* 双击鼠标选中后,可以拖拽被选择的文本 */
|
|
||||||
selectionPointer: true,
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
/* 设置高度 */
|
|
||||||
JoeEditor.setSize("100%", "550");
|
|
||||||
|
|
||||||
/* 实时预览 */
|
|
||||||
JoeEditor.on("change", function () {
|
|
||||||
// console.log(JoeEditor.getValue());
|
|
||||||
});
|
|
||||||
|
|
||||||
/* 节流,防止傻逼疯狂ctrl + v上传 */
|
|
||||||
let _flag = false;
|
|
||||||
/* 实现粘贴上传图片 */
|
|
||||||
JoeEditor.on("paste", function (editor, event) {
|
|
||||||
let clipboardData =
|
|
||||||
event.clipboardData ||
|
|
||||||
window.clipboardData ||
|
|
||||||
event.originalEvent.clipboardData;
|
|
||||||
if (!clipboardData || !clipboardData.items) return;
|
|
||||||
let items = clipboardData.items;
|
|
||||||
let file = null;
|
|
||||||
if (items.length === 0) return;
|
|
||||||
for (let i = 0; i < items.length; i++) {
|
|
||||||
if (items[i].kind === "file" && items[i].type.match(/^image/)) {
|
|
||||||
event.preventDefault(), (file = items[i].getAsFile());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!file) return;
|
|
||||||
/* 节流,防止傻逼疯狂ctrl + v上传 */
|
|
||||||
if (_flag) return;
|
|
||||||
_flag = true;
|
|
||||||
let uploadUrl = JoeUploadURL;
|
|
||||||
let cid = $('input[name="cid"]').val();
|
|
||||||
cid && (uploadUrl = uploadUrl + "&cid=" + cid);
|
|
||||||
let random = Date.now().toString(36);
|
|
||||||
let fileName = random + ".png";
|
|
||||||
let formData = new FormData();
|
|
||||||
formData.append("name", fileName);
|
|
||||||
formData.append("file", file, fileName);
|
|
||||||
$.ajax({
|
|
||||||
url: uploadUrl,
|
|
||||||
method: "post",
|
|
||||||
data: formData,
|
|
||||||
contentType: false,
|
|
||||||
processData: false,
|
|
||||||
xhr: () => {
|
|
||||||
let xhr = $.ajaxSettings.xhr();
|
|
||||||
if (!xhr.upload) return;
|
|
||||||
$(".CodeMirror-progress").show();
|
|
||||||
xhr.upload.addEventListener(
|
|
||||||
"progress",
|
|
||||||
(e) => {
|
|
||||||
let percent = (e.loaded / e.total) * 100;
|
|
||||||
$(".CodeMirror-progress").css("right", -percent + "%");
|
|
||||||
},
|
|
||||||
false
|
|
||||||
);
|
|
||||||
return xhr;
|
|
||||||
},
|
|
||||||
success(res) {
|
|
||||||
const text = `![${res[1].title}](${res[0]})`;
|
|
||||||
let timer = setTimeout(function () {
|
|
||||||
$(".CodeMirror-progress").hide();
|
|
||||||
$(".CodeMirror-progress").css("right", "100%");
|
|
||||||
/* 节流,防止傻逼疯狂ctrl + v上传 */
|
|
||||||
_flag = false;
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
if (cursor.ch === 0) JoeEditor.replaceSelection(text);
|
|
||||||
else JoeEditor.replaceSelection("\n\n" + text);
|
|
||||||
JoeEditor.focus();
|
|
||||||
clearTimeout(timer);
|
|
||||||
}, 1000);
|
|
||||||
},
|
|
||||||
error() {
|
|
||||||
let timer = setTimeout(function () {
|
|
||||||
$(".CodeMirror-progress").hide();
|
|
||||||
$(".CodeMirror-progress").css("right", "100%");
|
|
||||||
/* 节流,防止傻逼疯狂ctrl + v上传 */
|
|
||||||
_flag = false;
|
|
||||||
clearTimeout(timer);
|
|
||||||
}, 1000);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
Typecho.insertFileToEditor = function (file, url, isImage) {
|
|
||||||
const htmlStr = isImage
|
|
||||||
? "![" + file + "](" + url + ")"
|
|
||||||
: "[" + file + "](" + url + ")";
|
|
||||||
const cursor = JoeEditor.getCursor();
|
|
||||||
const n = cursor.ch === 0 ? "" : "\n";
|
|
||||||
JoeEditor.replaceSelection(n + htmlStr);
|
|
||||||
JoeEditor.focus();
|
|
||||||
};
|
|
||||||
});
|
|
1
typecho/editor/js/joe.editor.min.js
vendored
1
typecho/editor/js/joe.editor.min.js
vendored
@ -1 +0,0 @@
|
|||||||
$(function(){$("head").append('<meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, shrink-to-fit=no, viewport-fit=cover">'),window.JoeEditor=CodeMirror.fromTextArea(document.querySelector("textarea#text"),{theme:"joe",mode:"gfm",indentUnit:4,lineNumbers:!0,lineWrapping:!0,styleActiveLine:!0,matchBrackets:!0,autoCloseBrackets:!0,autoCloseTags:!0,foldGutter:!0,gutters:["CodeMirror-linenumbers","CodeMirror-foldgutter"],matchTags:{bothTags:!0},extraKeys:{Enter:"newlineAndIndentContinueMarkdownList","Ctrl-J":"toMatchingTag"},selectionPointer:!0}),JoeEditor.setSize("100%","550"),JoeEditor.on("change",function(){});let e=!1;JoeEditor.on("paste",function(t,o){let r=o.clipboardData||window.clipboardData||o.originalEvent.clipboardData;if(!r||!r.items)return;let i=r.items,n=null;if(0===i.length)return;for(let e=0;e<i.length;e++)"file"===i[e].kind&&i[e].type.match(/^image/)&&(o.preventDefault(),n=i[e].getAsFile());if(!n)return;if(e)return;e=!0;let a=JoeUploadURL,s=$('input[name="cid"]').val();s&&(a=a+"&cid="+s);let d=Date.now().toString(36),l=d+".png",c=new FormData;c.append("name",l),c.append("file",n,l),$.ajax({url:a,method:"post",data:c,contentType:!1,processData:!1,xhr:()=>{let e=$.ajaxSettings.xhr();if(e.upload)return $(".CodeMirror-progress").show(),e.upload.addEventListener("progress",e=>{let t=e.loaded/e.total*100;$(".CodeMirror-progress").css("right",-t+"%")},!1),e},success(t){const o=`![${t[1].title}](${t[0]})`;let r=setTimeout(function(){$(".CodeMirror-progress").hide(),$(".CodeMirror-progress").css("right","100%"),e=!1;const t=JoeEditor.getCursor();0===t.ch?JoeEditor.replaceSelection(o):JoeEditor.replaceSelection("\n\n"+o),JoeEditor.focus(),clearTimeout(r)},1e3)},error(){let t=setTimeout(function(){$(".CodeMirror-progress").hide(),$(".CodeMirror-progress").css("right","100%"),e=!1,clearTimeout(t)},1e3)}})}),Typecho.insertFileToEditor=function(e,t,o){const r=o?"!["+e+"]("+t+")":"["+e+"]("+t+")",i=JoeEditor.getCursor(),n=0===i.ch?"":"\n";JoeEditor.replaceSelection(n+r),JoeEditor.focus()}});
|
|
@ -1,199 +0,0 @@
|
|||||||
$(function () {
|
|
||||||
const Joe = new JoeInstance();
|
|
||||||
[
|
|
||||||
{
|
|
||||||
type: "bold",
|
|
||||||
title: "文本加粗",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M341.333 469.333h192a106.667 106.667 0 1 0 0-213.333h-192v213.333zm426.667 192a192 192 0 0 1-192 192H256V170.667h277.333a192 192 0 0 1 138.923 324.522A191.915 191.915 0 0 1 768 661.333zM341.333 554.667V768H576a106.667 106.667 0 1 0 0-213.333H341.333z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "italic",
|
|
||||||
title: "文本倾斜",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M640 853.333H298.667V768h124.885l90.283-512H384v-85.333h341.333V256H600.448l-90.283 512H640z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "delete",
|
|
||||||
title: "文本删除",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M731.904 597.333c9.813 22.016 14.763 46.507 14.763 73.387 0 57.259-22.358 102.059-67.03 134.272-44.757 32.213-106.496 48.341-185.301 48.341-69.973 0-139.221-16.256-207.787-48.81v-96.256c64.854 37.418 131.2 56.149 199.083 56.149 108.843 0 163.413-31.232 163.797-93.739a94.293 94.293 0 0 0-27.648-68.394l-5.12-4.992H128v-85.334h768v85.334H731.904zm-173.995-128H325.504a174.336 174.336 0 0 1-20.523-22.272c-18.432-23.808-27.648-52.565-27.648-86.442 0-52.736 19.883-97.579 59.606-134.528 39.808-36.95 101.29-55.424 184.533-55.424 62.763 0 122.837 13.994 180.139 41.984v91.818c-51.2-29.312-107.307-43.946-168.363-43.946-105.813 0-158.677 33.365-158.677 100.096 0 17.92 9.301 33.536 27.904 46.89 18.602 13.355 41.557 23.979 68.821 32 26.453 7.68 55.339 17.664 86.613 29.824z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "space",
|
|
||||||
title: "空格",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M170.667 384v170.667h682.666V384h85.334v213.333A42.667 42.667 0 0 1 896 640H128a42.667 42.667 0 0 1-42.667-42.667V384h85.334z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "code-inline",
|
|
||||||
title: "行内代码",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M723.2 361.13l60.33-60.33L994.73 512l-211.2 211.2-60.33-60.33L874.027 512 723.2 361.13zm-422.4 0L149.973 512 300.8 662.87l-60.33 60.33L29.27 512l211.2-211.2 60.33 60.33z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "hr",
|
|
||||||
title: "横线",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="22" height="22"><path d="M85.333 469.333h85.334v85.334H85.333v-85.334zm170.667 0h512v85.334H256v-85.334zm597.333 0h85.334v85.334h-85.334v-85.334z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "quote",
|
|
||||||
title: "文本引用",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M195.541 739.03C151.595 692.351 128 640 128 555.135c0-149.333 104.832-283.179 257.28-349.355l38.101 58.795c-142.293 76.97-170.112 176.853-181.205 239.83 22.912-11.862 52.907-16 82.304-13.27 76.97 7.125 137.643 70.315 137.643 148.864a149.333 149.333 0 0 1-149.334 149.333 165.163 165.163 0 0 1-117.248-50.304zm426.667 0c-43.947-46.678-67.541-99.03-67.541-183.894 0-149.333 104.832-283.179 257.28-349.355l38.101 58.795c-142.293 76.97-170.112 176.853-181.205 239.83 22.912-11.862 52.906-16 82.304-13.27 76.97 7.125 137.642 70.315 137.642 148.864a149.333 149.333 0 0 1-149.333 149.333 165.163 165.163 0 0 1-117.248-50.304z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "link",
|
|
||||||
title: "超链接",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M879.2 131.6c-103-95.5-264.1-88-361.4 11.2L474.7 184c-13.1 13.1-3.7 35.6 13.1 37.5 26.2 1.9 52.4 7.5 78.7 15 7.5 1.9 16.9 0 22.5-5.6l9.4-9.4c54.3-54.3 142.3-59.9 198.5-11.2 63.7 54.3 65.5 151.7 7.5 209.7L662 562.3c-18.7 18.7-41.2 30-63.7 37.5-30 7.5-61.8 5.6-89.9-5.6-16.9-7.5-33.7-16.9-48.7-31.8-7.5-7.5-13.1-15-18.7-24.3-7.5-13.1-24.3-15-33.7-3.7l-52.4 52.4c-7.5 7.5-7.5 18.7-1.9 28.1 7.5 11.2 16.9 20.6 26.2 30 13.1 13.1 30 26.2 44.9 35.6 26.2 16.9 56.2 28.1 86.1 33.7 58.1 11.2 121.7 1.9 174.2-26.2 20.6-11.2 41.2-26.2 58.1-43.1l142.3-142.3c104.9-103.2 101.2-271.8-5.6-371zM534.7 803.9l-39.3-5.6s-26.2-5.6-39.3-11.2c-7.5-1.9-16.9 0-22.5 5.6l-9.4 9.4c-54.3 54.3-142.3 59.9-198.5 11.2-63.7-54.3-65.5-151.7-7.5-209.7l142.3-142.3c18.7-18.7 41.2-30 63.7-37.5 30-7.5 61.8-5.6 89.9 5.6 16.9 7.5 33.7 16.9 48.7 31.8 7.5 7.5 13.1 15 18.7 24.3 7.5 13.1 24.3 15 33.7 3.7l52.4-52.4c7.5-7.5 7.5-18.7 1.9-28.1-7.5-11.2-16.9-20.6-26.2-30-13.1-13.1-28.1-26.2-44.9-35.6-26.2-16.9-56.2-28.1-88-33.7-58.1-11.2-121.7-1.9-174.2 26.2-20.6 11.2-41.2 26.2-58.1 43.1L141.4 515.5c-99.3 99.3-106.7 260.3-11.2 361.4C229.5 985.5 398 987.4 501 884.4l46.8-46.8c13.1-9.4 3.7-31.9-13.1-33.7z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "image",
|
|
||||||
title: "插入图片",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M46.545 977.455V46.545h930.91v930.91zm93.091-186.182v93.09h744.728V717.918l-214.11-228.305zm0-107.055l548.585-311.854 196.143 209.454V139.636H139.636zm99.282-376.227a82.758 82.758 0 0 1 82.758-82.758 82.758 82.758 0 0 1 82.757 82.758 82.758 82.758 0 0 1-82.757 82.805 82.804 82.804 0 0 1-82.758-82.898z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "ordered-list",
|
|
||||||
title: "有序列表",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M341.333 170.667H896V256H341.333v-85.333zm-128-42.667v128H256v42.667H128V256h42.667v-85.333H128V128h85.333zM128 597.333V490.667h85.333v-21.334H128v-42.666h128v106.666h-85.333v21.334H256v42.666H128zM213.333 832H128v-42.667h85.333V768H128v-42.667h128V896H128v-42.667h85.333V832zm128-362.667H896v85.334H341.333v-85.334zm0 298.667H896v85.333H341.333V768z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "title",
|
|
||||||
title: "标题",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="22" height="22"><path d="M256 213.333h104.875v267.094h324.48V213.333h104.874v640H685.355V570.07h-324.48v283.264H256z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "unordered-list",
|
|
||||||
title: "无序列表",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M341.333 170.667H896V256H341.333v-85.333zM192 277.333a64 64 0 1 1 0-128 64 64 0 0 1 0 128zM192 576a64 64 0 1 1 0-128 64 64 0 0 1 0 128zm0 294.4a64 64 0 1 1 0-128 64 64 0 0 1 0 128zm149.333-401.067H896v85.334H341.333v-85.334zm0 298.667H896v85.333H341.333V768z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
type: "undo",
|
|
||||||
title: "撤销",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M76 463.7l294.8 294.9c19.5 19.4 52.8 5.6 52.8-21.9V561.5c202.5-8.2 344.1 59.5 501.6 338.3 8.5 15 31.5 7.9 30.6-9.3-30.5-554.7-453-571.4-532.3-569.6v-174c0-27.5-33.2-41.3-52.7-21.8L75.9 420c-12 12.1-12 31.6.1 43.7z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "redo",
|
|
||||||
title: "回退",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M946.8 420L651.9 125.1c-19.5-19.5-52.7-5.7-52.7 21.8v174c-79.3-1.8-501.8 14.9-532.3 569.6-.9 17.2 22.1 24.3 30.6 9.3C255 621 396.6 553.3 599.1 561.5v175.2c0 27.5 33.3 41.3 52.8 21.9l294.8-294.9c12.1-12.1 12.1-31.6.1-43.7z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
type: "table",
|
|
||||||
title: "表格",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M892.8 108h-763c-26.2 0-47.5 21.3-47.5 47.5v714c0 26.2 21.3 47.5 47.5 47.5h763c26.2 0 47.5-21.3 47.5-47.5v-714c0-26.2-21.3-47.5-47.5-47.5zm-291 294.7v172.1H411.7V402.7h190.1zm76 0h186.5v172.1H677.8V402.7zM864.3 184v142.7h-706V184h706zM158.4 841V402.7h177.3v172.1H159.8v76h175.9v188.9h76V650.8h190.1v188.9h76V650.8h186.5V841H158.4z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "code-block",
|
|
||||||
title: "代码块",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="18" height="18"><path d="M958.17 447.4L760.69 249.92l-65.82 65.83 197.47 197.47L694.87 710.7l65.82 65.82 197.48-197.47 65.83-65.83zM263.3 249.92L65.82 447.4 0 513.22l65.82 65.83L263.3 776.52l65.82-65.82-197.47-197.48 197.47-197.47zm79.947 699.563L590.96 52.19l89.72 24.768-247.713 897.295z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "time",
|
|
||||||
title: "当前时间",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M512.163 1023.69a511.838 511.838 0 1 1 511.527-511.528 512.148 512.148 0 0 1-511.527 511.528zm0-930.05a418.833 418.833 0 1 0 418.522 418.522A419.143 419.143 0 0 0 512.163 93.641z" fill="#bfbfbf"/><path d="M509.683 233.148q46.502 0 46.502 46.503v279.014q0 46.502-46.502 46.502-46.503 0-46.503-46.502V279.651q0-46.503 46.503-46.503z" fill="#bfbfbf"/><path d="M674.172 734.953q-32.883 32.882-65.765 0l-131.529-131.53q-32.882-32.88 0-65.764 32.882-32.882 65.765 0l131.53 131.53q32.881 32.881 0 65.764z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "clean",
|
|
||||||
title: "清屏",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="18" height="18"><path d="M479.3 863.6L899.9 443c37.4-37.4 37.4-98.3 0-135.8L716.7 124.1C698.5 106 674.4 96 648.7 96c-25.8 0-50.4 10.8-68.6 29l-455 455c-18.2 18.2-29 42.8-29 68.6 0 25.7 9.9 49.9 28.1 68l183.1 183.2c18.1 18.1 42.2 28.1 67.9 28.1 3 0 5.9-.1 8.8-.4v.1h512c17.7 0 32-14.3 32-32s-14.3-32-32-32H479.3zm-126.8-9L169.4 671.5c-6-6-9.4-14.1-9.4-22.6 0-8.5 3.3-16.6 9.4-22.6l104.9-104.9 228.4 228.4-104.9 104.8c-6 6-14.1 9.4-22.6 9.4-8.6 0-16.6-3.3-22.7-9.4z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "fullscreen",
|
|
||||||
title: "全屏/取消全屏",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="18" height="18"><path d="M181.2 803.4V637H98v208c0 45.9 37.2 83.2 83.2 83.2h208V845H222.8c-23 0-41.6-18.6-41.6-41.6zM844.8 98.6h-208v83.2h166.4c23 0 41.6 18.6 41.6 41.6v166.4H928v-208c0-46-37.3-83.2-83.2-83.2zm-746.6 83v208h83.2V223.2c0-23 18.6-41.6 41.6-41.6h166.4V98.4h-208c-46 0-83.2 37.3-83.2 83.2zm746.4 455.6v166.4c0 23-18.6 41.6-41.6 41.6H636.6v83.2h208c45.9 0 83.2-37.2 83.2-83.2v-208h-83.2z" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "about",
|
|
||||||
title: "关于",
|
|
||||||
icon:
|
|
||||||
'<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M881.6 512.1c0 203.7-165.7 369.5-369.5 369.5S142.6 715.8 142.6 512.1s165.7-369.5 369.5-369.5 369.5 165.8 369.5 369.5m77.7 0c0-246.6-200.6-447.2-447.2-447.2S64.9 265.5 64.9 512.1s200.6 447.2 447.2 447.2 447.2-200.6 447.2-447.2M582.5 318.2c9-9 14.2-21.6 14.2-34.3 0-12.8-5.2-25.4-14.2-34.4s-21.6-14.2-34.4-14.2c-12.7 0-25.3 5.2-34.3 14.2-9.1 9-14.3 21.6-14.3 34.4 0 12.7 5.2 25.3 14.3 34.3 9 9 21.5 14.3 34.3 14.3s25.3-5.3 34.4-14.3m-96.6 464.3c-7.6 0-15.2-2.6-21.3-7.5-9.9-7.9-14.5-20.7-12.1-33.1l47.9-243.1-26.4 14.8c-16.4 9.2-37.1 3.4-46.3-13-9.2-16.4-3.4-37.1 12.9-46.3l90.7-51.1c11.6-6.5 25.9-5.7 36.6 2.1 10.8 7.8 16 21.1 13.4 34.1l-49.6 251.9 40.7-17.7c17.2-7.5 37.3.4 44.8 17.6 7.5 17.2-.4 37.3-17.6 44.8l-100.2 43.7c-4.3 1.9-8.9 2.8-13.5 2.8" fill="#bfbfbf"/></svg>',
|
|
||||||
},
|
|
||||||
].forEach((tool) => {
|
|
||||||
if (tool.type === "title") {
|
|
||||||
Joe.initTitleText(tool);
|
|
||||||
} else if (tool.type === "fullscreen") {
|
|
||||||
Joe.handleFullscreen(tool);
|
|
||||||
} else {
|
|
||||||
const item = $(
|
|
||||||
`<li class="CodeMirror-means-item" title="${tool.title}">${tool.icon}</li>`
|
|
||||||
);
|
|
||||||
item.on("click", () => {
|
|
||||||
switch (tool.type) {
|
|
||||||
case "bold":
|
|
||||||
Joe.insertBoldText();
|
|
||||||
break;
|
|
||||||
case "italic":
|
|
||||||
Joe.insertItalicText();
|
|
||||||
break;
|
|
||||||
case "delete":
|
|
||||||
Joe.insertDeleteText();
|
|
||||||
break;
|
|
||||||
case "quote":
|
|
||||||
Joe.insertQuoteText();
|
|
||||||
break;
|
|
||||||
case "space":
|
|
||||||
Joe.insertSpaceText();
|
|
||||||
break;
|
|
||||||
case "code-inline":
|
|
||||||
Joe.insertCodeInlineText();
|
|
||||||
break;
|
|
||||||
case "hr":
|
|
||||||
Joe.insertHrText();
|
|
||||||
break;
|
|
||||||
case "unordered-list":
|
|
||||||
Joe.insertUnorderedListText();
|
|
||||||
break;
|
|
||||||
case "ordered-list":
|
|
||||||
Joe.insertOrderedListText();
|
|
||||||
break;
|
|
||||||
case "undo":
|
|
||||||
JoeEditor.undo();
|
|
||||||
JoeEditor.focus();
|
|
||||||
break;
|
|
||||||
case "redo":
|
|
||||||
JoeEditor.redo();
|
|
||||||
JoeEditor.focus();
|
|
||||||
break;
|
|
||||||
case "clean":
|
|
||||||
JoeEditor.setValue("");
|
|
||||||
JoeEditor.focus();
|
|
||||||
break;
|
|
||||||
case "time":
|
|
||||||
Joe.insertTimeText();
|
|
||||||
break;
|
|
||||||
case "link":
|
|
||||||
Joe.insertLinkText();
|
|
||||||
break;
|
|
||||||
case "image":
|
|
||||||
Joe.insertImageText();
|
|
||||||
break;
|
|
||||||
case "table":
|
|
||||||
Joe.insertTableText();
|
|
||||||
break;
|
|
||||||
case "code-block":
|
|
||||||
Joe.insertCodeBlockText();
|
|
||||||
break;
|
|
||||||
case "about":
|
|
||||||
Joe.handleAbout();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
$(".CodeMirror-means").append(item);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
1
typecho/editor/js/joe.instance.min.js
vendored
1
typecho/editor/js/joe.instance.min.js
vendored
File diff suppressed because one or more lines are too long
1
typecho/write/css/joe.write.min.css
vendored
Normal file
1
typecho/write/css/joe.write.min.css
vendored
Normal file
File diff suppressed because one or more lines are too long
526
typecho/write/css/joe.write.min.scss
Normal file
526
typecho/write/css/joe.write.min.scss
Normal file
@ -0,0 +1,526 @@
|
|||||||
|
input[name='fields[keywords]'] {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
textarea[name='fields[description]'],
|
||||||
|
textarea[name='fields[abstract]'],
|
||||||
|
textarea[name='fields[thumb]'],
|
||||||
|
textarea[name='fields[video]'] {
|
||||||
|
width: 100%;
|
||||||
|
height: 80px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#text,
|
||||||
|
#text ~ .resize {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
body.fullscreen {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: 550px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
z-index: 6666;
|
||||||
|
content-visibility: auto;
|
||||||
|
background: #222;
|
||||||
|
&.fullscreen {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
* {
|
||||||
|
outline: none !important;
|
||||||
|
}
|
||||||
|
.cm-tools {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
background-color: #303133;
|
||||||
|
padding: 5px 0 0 5px;
|
||||||
|
.cm-tools-item {
|
||||||
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-radius: 2px;
|
||||||
|
padding: 5px;
|
||||||
|
margin: 0 5px 5px 0;
|
||||||
|
transition: background 0.25s;
|
||||||
|
&:hover {
|
||||||
|
background: #393939;
|
||||||
|
svg {
|
||||||
|
fill: #efefef;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&.active {
|
||||||
|
background: #393939;
|
||||||
|
svg {
|
||||||
|
fill: #efefef;
|
||||||
|
}
|
||||||
|
.cm-tools__dropdown {
|
||||||
|
visibility: visible;
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateX(-50%) rotateX(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
svg {
|
||||||
|
vertical-align: middle;
|
||||||
|
fill: #bfbfbf;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&__dropdown {
|
||||||
|
position: absolute;
|
||||||
|
top: 42px;
|
||||||
|
left: 50%;
|
||||||
|
z-index: 1000;
|
||||||
|
background: #313335;
|
||||||
|
padding: 5px 0;
|
||||||
|
border-radius: 4px;
|
||||||
|
visibility: hidden;
|
||||||
|
transform-origin: top;
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateX(-50%) rotateX(-90deg);
|
||||||
|
transition: visibility 0.25s, transform 0.25s, opacity 0.25s;
|
||||||
|
&-item {
|
||||||
|
padding: 0 20px;
|
||||||
|
line-height: 32px;
|
||||||
|
transition: background 0.25s;
|
||||||
|
user-select: none;
|
||||||
|
font-size: 15px;
|
||||||
|
color: #bfbfbf;
|
||||||
|
font-weight: 600;
|
||||||
|
&:hover {
|
||||||
|
background: #393939;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: -7px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
width: 0;
|
||||||
|
height: 0;
|
||||||
|
border-left: 7px solid transparent;
|
||||||
|
border-right: 7px solid transparent;
|
||||||
|
border-bottom: 7px solid #313335;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.cm-mainer {
|
||||||
|
flex: 1;
|
||||||
|
min-height: 0;
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
.cm-wrap {
|
||||||
|
flex-shrink: 0;
|
||||||
|
height: 100%;
|
||||||
|
flex: 1;
|
||||||
|
min-width: 0;
|
||||||
|
font-size: 14px;
|
||||||
|
.cm-scroller {
|
||||||
|
line-height: 1.5;
|
||||||
|
font-family: Consolas, Monaco, 'Andale Mono', 'Ubuntu Mono', monospace;
|
||||||
|
overscroll-behavior: none;
|
||||||
|
&::-webkit-scrollbar {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
::selection {
|
||||||
|
background-color: rgba(221, 240, 255, 0.2);
|
||||||
|
}
|
||||||
|
.cm-content {
|
||||||
|
white-space: pre-wrap;
|
||||||
|
word-break: break-all;
|
||||||
|
word-wrap: break-word;
|
||||||
|
color: #ccc;
|
||||||
|
caret-color: #ccc;
|
||||||
|
padding: 15px 13px 15px 11px;
|
||||||
|
}
|
||||||
|
.cm-matchingBracket {
|
||||||
|
color: #8f938f;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.25);
|
||||||
|
margin: -1px;
|
||||||
|
}
|
||||||
|
.cm-nonmatchingBracket {
|
||||||
|
color: #ff5627;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.cm-resize {
|
||||||
|
flex-shrink: 0;
|
||||||
|
height: 100%;
|
||||||
|
position: relative;
|
||||||
|
width: 16px;
|
||||||
|
background: #303133;
|
||||||
|
cursor: col-resize;
|
||||||
|
&::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
width: 4px;
|
||||||
|
height: 40px;
|
||||||
|
border-radius: 2px;
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.cm-preview {
|
||||||
|
position: relative;
|
||||||
|
flex-shrink: 0;
|
||||||
|
width: 0;
|
||||||
|
height: 100%;
|
||||||
|
background: #fff;
|
||||||
|
overflow: auto;
|
||||||
|
box-sizing: border-box;
|
||||||
|
overscroll-behavior: none;
|
||||||
|
&::-webkit-scrollbar-track {
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
&::-webkit-scrollbar {
|
||||||
|
width: 6px;
|
||||||
|
height: 6px;
|
||||||
|
}
|
||||||
|
&::-webkit-scrollbar-thumb {
|
||||||
|
border-radius: 3px;
|
||||||
|
background: #c0c4cc;
|
||||||
|
}
|
||||||
|
.cm-preview-content {
|
||||||
|
padding: 20px;
|
||||||
|
font-size: 14px;
|
||||||
|
font-family: 'Helvetica Neue', Helvetica, 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', '微软雅黑', Arial, sans-serif;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
word-break: break-all;
|
||||||
|
word-wrap: break-word;
|
||||||
|
color: #606266;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
-moz-osx-font-smoothing: grayscale;
|
||||||
|
* {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
h1,
|
||||||
|
h2,
|
||||||
|
h3,
|
||||||
|
h4,
|
||||||
|
h5,
|
||||||
|
h6 {
|
||||||
|
color: #303133;
|
||||||
|
font-size: 18px;
|
||||||
|
line-height: 24px;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
h1 {
|
||||||
|
padding: 0 15px;
|
||||||
|
&::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 8.5px;
|
||||||
|
left: 0;
|
||||||
|
height: 7px;
|
||||||
|
width: 7px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: #409eff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
h2 {
|
||||||
|
padding: 0 15px;
|
||||||
|
&::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 10%;
|
||||||
|
bottom: 10%;
|
||||||
|
left: 0;
|
||||||
|
width: 4px;
|
||||||
|
border-radius: 2px;
|
||||||
|
background: #409eff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
h3 {
|
||||||
|
padding: 0 15px 0 20px;
|
||||||
|
&::before {
|
||||||
|
content: '#';
|
||||||
|
color: #409eff;
|
||||||
|
font-weight: 700;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
line-height: 24px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
h4 {
|
||||||
|
&::before {
|
||||||
|
content: '「';
|
||||||
|
color: #409eff;
|
||||||
|
font-weight: 600;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
&::after {
|
||||||
|
content: '」';
|
||||||
|
color: #409eff;
|
||||||
|
font-weight: 600;
|
||||||
|
margin-left: 5px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
h5 {
|
||||||
|
padding: 0 15px 0 28px;
|
||||||
|
&::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 2px;
|
||||||
|
left: 0;
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
background-image: url('');
|
||||||
|
background-size: 100% 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
h6 {
|
||||||
|
padding: 0 15px 0 28px;
|
||||||
|
&::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 2px;
|
||||||
|
left: 0;
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
background-image: url('');
|
||||||
|
background-size: 100% 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
hr {
|
||||||
|
border: none;
|
||||||
|
height: 1px;
|
||||||
|
background-color: #e4e7ed;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
}
|
||||||
|
p {
|
||||||
|
line-height: 26px;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
}
|
||||||
|
blockquote {
|
||||||
|
line-height: 26px;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
background: #ecf8ff;
|
||||||
|
border-left: 5px solid #50bfff;
|
||||||
|
color: #50bfff;
|
||||||
|
padding: 8px 15px;
|
||||||
|
border-radius: 0 4px 4px 0;
|
||||||
|
p {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
code {
|
||||||
|
display: inline-block;
|
||||||
|
min-height: 26px;
|
||||||
|
line-height: 26px;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 12px;
|
||||||
|
background: #fdf6ec;
|
||||||
|
padding: 0 8px;
|
||||||
|
color: #e6a23c;
|
||||||
|
vertical-align: top;
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
background: #f0f0ec;
|
||||||
|
padding: 15px;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
}
|
||||||
|
pre code {
|
||||||
|
display: unset;
|
||||||
|
vertical-align: unset;
|
||||||
|
min-height: unset;
|
||||||
|
line-height: unset;
|
||||||
|
border-radius: unset;
|
||||||
|
font-size: unset;
|
||||||
|
background: unset;
|
||||||
|
padding: unset;
|
||||||
|
color: unset;
|
||||||
|
}
|
||||||
|
ol,
|
||||||
|
ul {
|
||||||
|
margin-bottom: 15px;
|
||||||
|
padding-left: 36px;
|
||||||
|
li {
|
||||||
|
line-height: 26px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ol li {
|
||||||
|
list-style: decimal;
|
||||||
|
}
|
||||||
|
ul li {
|
||||||
|
list-style: disc;
|
||||||
|
}
|
||||||
|
table {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 100%;
|
||||||
|
table-layout: fixed;
|
||||||
|
color: #909399;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
font-size: 13px;
|
||||||
|
border-top: 1px solid #ebeef5;
|
||||||
|
border-left: 1px solid #ebeef5;
|
||||||
|
td,
|
||||||
|
th {
|
||||||
|
padding: 8px;
|
||||||
|
border-bottom: 1px solid #ebeef5;
|
||||||
|
border-right: 1px solid #ebeef5;
|
||||||
|
}
|
||||||
|
thead {
|
||||||
|
th {
|
||||||
|
font-weight: 500;
|
||||||
|
background: #ebeef5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tbody {
|
||||||
|
tr {
|
||||||
|
transition: background 0.35s;
|
||||||
|
&:hover {
|
||||||
|
background: #ebeef5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
img:not(.owo_image) {
|
||||||
|
display: block;
|
||||||
|
max-width: 100%;
|
||||||
|
border-radius: 4px;
|
||||||
|
transition: transform 0.35s, box-shadow 0.35s;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
.owo_image {
|
||||||
|
max-height: 26px;
|
||||||
|
vertical-align: top;
|
||||||
|
}
|
||||||
|
a {
|
||||||
|
display: inline-block;
|
||||||
|
line-height: 26px;
|
||||||
|
color: #409eff;
|
||||||
|
position: relative;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.cm-progress-left {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 50%;
|
||||||
|
width: 0;
|
||||||
|
height: 3px;
|
||||||
|
background-color: #ff5627;
|
||||||
|
transition: width 0.5s;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
.cm-progress-right {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 50%;
|
||||||
|
width: 0;
|
||||||
|
height: 3px;
|
||||||
|
background-color: #ff5627;
|
||||||
|
transition: width 0.5s;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-modal {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
z-index: 9999;
|
||||||
|
background: rgba(0, 0, 0, 0);
|
||||||
|
visibility: hidden;
|
||||||
|
transition: background 0.25s, visibility 0.25s;
|
||||||
|
will-change: background, visibility;
|
||||||
|
&__wrapper {
|
||||||
|
width: 380px;
|
||||||
|
background: #fff;
|
||||||
|
color: #606266;
|
||||||
|
border-radius: 4px;
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(-30%);
|
||||||
|
transition: opacity 0.25s, transform 0.25s;
|
||||||
|
&-header {
|
||||||
|
padding: 12px 20px;
|
||||||
|
border-bottom: 1px solid #e4e7ed;
|
||||||
|
}
|
||||||
|
&-bodyer {
|
||||||
|
padding: 20px;
|
||||||
|
.fitem {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
&:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
label {
|
||||||
|
margin-right: 10px;
|
||||||
|
}
|
||||||
|
input {
|
||||||
|
width: auto;
|
||||||
|
flex: 1;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
outline: none;
|
||||||
|
border: 1px solid #dcdfe6;
|
||||||
|
height: 34px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding: 0 10px;
|
||||||
|
border-radius: 3px;
|
||||||
|
color: #606266;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-footer {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-end;
|
||||||
|
padding: 0 20px 10px;
|
||||||
|
&--cancle,
|
||||||
|
&--confirm {
|
||||||
|
cursor: pointer;
|
||||||
|
border: none;
|
||||||
|
outline: none;
|
||||||
|
margin-left: 5px;
|
||||||
|
background: #fff;
|
||||||
|
color: #606266;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
padding: 7px 15px;
|
||||||
|
border-radius: 3px;
|
||||||
|
}
|
||||||
|
&--confirm {
|
||||||
|
color: #fff;
|
||||||
|
background: #409eff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&.active {
|
||||||
|
visibility: visible;
|
||||||
|
background: rgba(0, 0, 0, 0.75);
|
||||||
|
.cm-modal__wrapper {
|
||||||
|
transform: translateY(0);
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.cm-modal__wrapper {
|
||||||
|
width: 90%;
|
||||||
|
}
|
||||||
|
}
|
14126
typecho/write/js/joe.write.chunk.js
Normal file
14126
typecho/write/js/joe.write.chunk.js
Normal file
File diff suppressed because it is too large
Load Diff
659
typecho/write/js/joe.write.js
Normal file
659
typecho/write/js/joe.write.js
Normal file
@ -0,0 +1,659 @@
|
|||||||
|
import { EditorView, keymap } from '@codemirror/view';
|
||||||
|
import { EditorState } from '@codemirror/state';
|
||||||
|
import { bracketMatching } from '@codemirror/matchbrackets';
|
||||||
|
import { closeBrackets, closeBracketsKeymap } from '@codemirror/closebrackets';
|
||||||
|
import { defaultKeymap, defaultTabBinding } from '@codemirror/commands';
|
||||||
|
import { history, historyKeymap, undo, redo } from '@codemirror/history';
|
||||||
|
import { classHighlightStyle } from '@codemirror/highlight';
|
||||||
|
|
||||||
|
class JoeAction {
|
||||||
|
constructor() {
|
||||||
|
$('body').append('<div class="cm-modal"><div class="cm-modal__wrapper"><div class="cm-modal__wrapper-header"></div><div class="cm-modal__wrapper-bodyer"></div><div class="cm-modal__wrapper-footer"><button class="cm-modal__wrapper-footer--cancle">取消</button><button class="cm-modal__wrapper-footer--confirm">确定</button></div></div></div>');
|
||||||
|
$('.cm-modal__wrapper-footer--cancle').on('click', () => {
|
||||||
|
this.options.cancel();
|
||||||
|
$('.cm-modal').removeClass('active');
|
||||||
|
});
|
||||||
|
$('.cm-modal__wrapper-footer--confirm').on('click', () => {
|
||||||
|
this.options.confirm();
|
||||||
|
$('.cm-modal').removeClass('active');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
_openModal(options = {}) {
|
||||||
|
const _options = {
|
||||||
|
title: '提示',
|
||||||
|
innerHtml: '内容',
|
||||||
|
cancel: () => {},
|
||||||
|
confirm: () => {}
|
||||||
|
};
|
||||||
|
this.options = Object.assign(_options, options);
|
||||||
|
$('.cm-modal__wrapper-header').html(this.options.title);
|
||||||
|
$('.cm-modal__wrapper-bodyer').html(this.options.innerHtml);
|
||||||
|
$('.cm-modal').addClass('active');
|
||||||
|
}
|
||||||
|
_getLineCh(cm) {
|
||||||
|
const head = cm.state.selection.main.head;
|
||||||
|
const line = cm.state.doc.lineAt(head);
|
||||||
|
return head - line.from;
|
||||||
|
}
|
||||||
|
_replaceSelection(cm, str) {
|
||||||
|
cm.dispatch(cm.state.replaceSelection(str));
|
||||||
|
}
|
||||||
|
_setCursor(cm, pos) {
|
||||||
|
cm.dispatch({ selection: { anchor: pos } });
|
||||||
|
}
|
||||||
|
_getSelection(cm) {
|
||||||
|
return cm.state.sliceDoc(cm.state.selection.main.from, cm.state.selection.main.to);
|
||||||
|
}
|
||||||
|
_insetAmboText(cm, str) {
|
||||||
|
const cursor = cm.state.selection.main.head;
|
||||||
|
const selection = this._getSelection(cm);
|
||||||
|
this._replaceSelection(cm, ` ${str + selection + str} `);
|
||||||
|
if (selection === '') this._setCursor(cm, cursor + str.length + 1);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleFullScreen(cm, el) {
|
||||||
|
el.toggleClass('active');
|
||||||
|
$('body').toggleClass('fullscreen');
|
||||||
|
$('.cm-container').toggleClass('fullscreen');
|
||||||
|
$('.cm-preview').width(0);
|
||||||
|
}
|
||||||
|
handlePublish() {
|
||||||
|
$('#btn-submit').click();
|
||||||
|
}
|
||||||
|
handleUndo(cm) {
|
||||||
|
undo(cm);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleRedo(cm) {
|
||||||
|
redo(cm);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleIndent(cm) {
|
||||||
|
this._replaceSelection(cm, ' ');
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleTime(cm) {
|
||||||
|
const time = new Date();
|
||||||
|
const _Year = time.getFullYear();
|
||||||
|
const _Month = String(time.getMonth() + 1).padStart(2, 0);
|
||||||
|
const _Date = String(time.getDate()).padStart(2, 0);
|
||||||
|
const _Hours = String(time.getHours()).padStart(2, 0);
|
||||||
|
const _Minutes = String(time.getMinutes()).padStart(2, 0);
|
||||||
|
const _Seconds = String(time.getSeconds()).padStart(2, 0);
|
||||||
|
const _Day = ['星期日', '星期一', '星期二', '星期三', '星期四', '星期五', '星期六'][time.getDay()];
|
||||||
|
const _time = `${this._getLineCh(cm) ? '\n' : ''}${_Year}-${_Month}-${_Date} ${_Hours}:${_Minutes}:${_Seconds} ${_Day}\n`;
|
||||||
|
this._replaceSelection(cm, _time);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleHr(cm) {
|
||||||
|
const str = `${this._getLineCh(cm) ? '\n' : ''}\n------------\n\n`;
|
||||||
|
this._replaceSelection(cm, str);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleClean(cm) {
|
||||||
|
cm.dispatch({ changes: { from: 0, to: cm.state.doc.length, insert: '' } });
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleOrdered(cm) {
|
||||||
|
const selection = this._getSelection(cm);
|
||||||
|
if (selection === '') {
|
||||||
|
const str = (this._getLineCh(cm) ? '\n\n' : '') + '1. ';
|
||||||
|
this._replaceSelection(cm, str);
|
||||||
|
} else {
|
||||||
|
const selectionText = selection.split('\n');
|
||||||
|
for (let i = 0, len = selectionText.length; i < len; i++) {
|
||||||
|
selectionText[i] = selectionText[i] === '' ? '' : i + 1 + '. ' + selectionText[i];
|
||||||
|
}
|
||||||
|
const str = (this._getLineCh(cm) ? '\n' : '') + selectionText.join('\n');
|
||||||
|
this._replaceSelection(cm, str);
|
||||||
|
}
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleUnordered(cm) {
|
||||||
|
const selection = this._getSelection(cm);
|
||||||
|
if (selection === '') {
|
||||||
|
const str = (this._getLineCh(cm) ? '\n' : '') + '- ';
|
||||||
|
this._replaceSelection(cm, str);
|
||||||
|
} else {
|
||||||
|
const selectionText = selection.split('\n');
|
||||||
|
for (let i = 0, len = selectionText.length; i < len; i++) {
|
||||||
|
selectionText[i] = selectionText[i] === '' ? '' : '- ' + selectionText[i];
|
||||||
|
}
|
||||||
|
const str = (this._getLineCh(cm) ? '\n' : '') + selectionText.join('\n');
|
||||||
|
this._replaceSelection(cm, str);
|
||||||
|
}
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleQuote(cm) {
|
||||||
|
const selection = this._getSelection(cm);
|
||||||
|
if (selection === '') {
|
||||||
|
this._replaceSelection(cm, `${this._getLineCh(cm) ? '\n' : ''}> `);
|
||||||
|
} else {
|
||||||
|
const selectionText = selection.split('\n');
|
||||||
|
for (let i = 0, len = selectionText.length; i < len; i++) {
|
||||||
|
selectionText[i] = selectionText[i] === '' ? '' : '> ' + selectionText[i];
|
||||||
|
}
|
||||||
|
const str = (this._getLineCh(cm) ? '\n' : '') + selectionText.join('\n');
|
||||||
|
this._replaceSelection(cm, str);
|
||||||
|
}
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
handleDownload(cm) {
|
||||||
|
const title = $('#title').val() || '新文章';
|
||||||
|
const aTag = document.createElement('a');
|
||||||
|
let blob = new Blob([cm.state.doc.toString()]);
|
||||||
|
aTag.download = title + '.md';
|
||||||
|
aTag.href = URL.createObjectURL(blob);
|
||||||
|
aTag.click();
|
||||||
|
URL.revokeObjectURL(blob);
|
||||||
|
}
|
||||||
|
handleTitle(cm, tool) {
|
||||||
|
const item = $(`
|
||||||
|
<div class="cm-tools-item" title="${tool.title}">
|
||||||
|
${tool.innerHTML}
|
||||||
|
<div class="cm-tools__dropdown">
|
||||||
|
<div class="cm-tools__dropdown-item" data-text="# "> H1 </div>
|
||||||
|
<div class="cm-tools__dropdown-item" data-text="## "> H2 </div>
|
||||||
|
<div class="cm-tools__dropdown-item" data-text="### "> H3 </div>
|
||||||
|
<div class="cm-tools__dropdown-item" data-text="#### "> H4 </div>
|
||||||
|
<div class="cm-tools__dropdown-item" data-text="##### "> H5 </div>
|
||||||
|
<div class="cm-tools__dropdown-item" data-text="###### "> H6 </div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
item.on('click', function (e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
$(this).toggleClass('active');
|
||||||
|
});
|
||||||
|
const _this = this;
|
||||||
|
item.on('click', '.cm-tools__dropdown-item', function (e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
const text = $(this).attr('data-text');
|
||||||
|
if (_this._getLineCh(cm)) _this._replaceSelection(cm, '\n\n' + text);
|
||||||
|
else _this._replaceSelection(cm, text);
|
||||||
|
item.removeClass('active');
|
||||||
|
cm.focus();
|
||||||
|
});
|
||||||
|
$(document).on('click', () => item.removeClass('active'));
|
||||||
|
$('.cm-tools').append(item);
|
||||||
|
}
|
||||||
|
handleLink(cm) {
|
||||||
|
this._openModal({
|
||||||
|
title: '插入链接',
|
||||||
|
innerHtml: `
|
||||||
|
<div class="fitem">
|
||||||
|
<label>链接标题</label>
|
||||||
|
<input autocomplete="off" name="title" placeholder="请输入链接标题"/>
|
||||||
|
</div>
|
||||||
|
<div class="fitem">
|
||||||
|
<label>链接地址</label>
|
||||||
|
<input autocomplete="off" name="url" placeholder="请输入链接地址"/>
|
||||||
|
</div>
|
||||||
|
`,
|
||||||
|
confirm: () => {
|
||||||
|
const title = $(".cm-modal input[name='title']").val() || 'Test';
|
||||||
|
const url = $(".cm-modal input[name='url']").val() || 'http://';
|
||||||
|
this._replaceSelection(cm, ` [${title}](${url}) `);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
handleImage(cm) {
|
||||||
|
this._openModal({
|
||||||
|
title: '插入图片',
|
||||||
|
innerHtml: `
|
||||||
|
<div class="fitem">
|
||||||
|
<label>图片名称</label>
|
||||||
|
<input autocomplete="off" name="title" placeholder="请输入图片名称"/>
|
||||||
|
</div>
|
||||||
|
<div class="fitem">
|
||||||
|
<label>图片地址</label>
|
||||||
|
<input autocomplete="off" name="url" placeholder="请输入图片地址"/>
|
||||||
|
</div>
|
||||||
|
`,
|
||||||
|
confirm: () => {
|
||||||
|
const title = $(".cm-modal input[name='title']").val() || 'Test';
|
||||||
|
const url = $(".cm-modal input[name='url']").val() || 'http://';
|
||||||
|
this._replaceSelection(cm, ` ![${title}](${url}) `);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
handleTable(cm) {
|
||||||
|
this._openModal({
|
||||||
|
title: '插入表格',
|
||||||
|
innerHtml: `
|
||||||
|
<div class="fitem">
|
||||||
|
<label>表格行</label>
|
||||||
|
<input style="width: 50px; flex: none; margin-right: 10px;" value="3" autocomplete="off" name="row"/>
|
||||||
|
<label>表格列</label>
|
||||||
|
<input style="width: 50px; flex: none;" value="3" autocomplete="off" name="column"/>
|
||||||
|
</div>
|
||||||
|
`,
|
||||||
|
confirm: () => {
|
||||||
|
let row = $(".cm-modal input[name='row']").val();
|
||||||
|
let column = $(".cm-modal input[name='column']").val();
|
||||||
|
if (isNaN(row)) row = 3;
|
||||||
|
if (isNaN(column)) column = 3;
|
||||||
|
let rowStr = '';
|
||||||
|
let rangeStr = '';
|
||||||
|
let columnlStr = '';
|
||||||
|
for (let i = 0; i < column; i++) {
|
||||||
|
rowStr += '| 表头 ';
|
||||||
|
rangeStr += '| :--: ';
|
||||||
|
}
|
||||||
|
for (let i = 0; i < row; i++) {
|
||||||
|
for (let j = 0; j < column; j++) columnlStr += '| 表格 ';
|
||||||
|
columnlStr += '|\n';
|
||||||
|
}
|
||||||
|
const htmlStr = `${rowStr}|\n${rangeStr}|\n${columnlStr}\n`;
|
||||||
|
if (this._getLineCh(cm)) this._replaceSelection(cm, '\n\n' + htmlStr);
|
||||||
|
else this._replaceSelection(cm, htmlStr);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
handleCodeBlock(cm) {
|
||||||
|
this._openModal({
|
||||||
|
title: '插入代码块',
|
||||||
|
innerHtml: `
|
||||||
|
<div class="fitem">
|
||||||
|
<label>语言类型</label>
|
||||||
|
<input autocomplete="off" name="type" placeholder="请输入语言类型(英文)"/>
|
||||||
|
</div>
|
||||||
|
`,
|
||||||
|
confirm: () => {
|
||||||
|
const type = $(".cm-modal input[name='type']").val() || 'html';
|
||||||
|
const htmlStr = `\`\`\`${type}\ncode here...\n\`\`\``;
|
||||||
|
if (this._getLineCh(cm)) this._replaceSelection(cm, '\n\n' + htmlStr);
|
||||||
|
else this._replaceSelection(cm, htmlStr);
|
||||||
|
cm.focus();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
handleAbout() {
|
||||||
|
this._openModal({
|
||||||
|
title: '关于',
|
||||||
|
innerHtml: `
|
||||||
|
<ul>
|
||||||
|
<li>短代码功能正在开发中...</li>
|
||||||
|
<li>仅支持网络图片粘贴上传(截图等)</li>
|
||||||
|
<li>本编辑器仅供Joe主题使用,未经允许不得移植至其他主题!</li>
|
||||||
|
</ul>
|
||||||
|
`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class Joe extends JoeAction {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
this.tools = [
|
||||||
|
{
|
||||||
|
type: 'undo',
|
||||||
|
title: '撤销',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="18" height="18"><path d="M76 463.7l294.8 294.9c19.5 19.4 52.8 5.6 52.8-21.9V561.5c202.5-8.2 344.1 59.5 501.6 338.3 8.5 15 31.5 7.9 30.6-9.3-30.5-554.7-453-571.4-532.3-569.6v-174c0-27.5-33.2-41.3-52.7-21.8L75.9 420c-12 12.1-12 31.6.1 43.7z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'redo',
|
||||||
|
title: '重做',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="18" height="18"><path d="M946.8 420L651.9 125.1c-19.5-19.5-52.7-5.7-52.7 21.8v174c-79.3-1.8-501.8 14.9-532.3 569.6-.9 17.2 22.1 24.3 30.6 9.3C255 621 396.6 553.3 599.1 561.5v175.2c0 27.5 33.3 41.3 52.8 21.9l294.8-294.9c12.1-12.1 12.1-31.6.1-43.7z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'bold',
|
||||||
|
title: '加粗',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M341.333 469.333h192a106.667 106.667 0 1 0 0-213.333h-192v213.333zm426.667 192a192 192 0 0 1-192 192H256V170.667h277.333a192 192 0 0 1 138.923 324.522A191.915 191.915 0 0 1 768 661.333zM341.333 554.667V768H576a106.667 106.667 0 1 0 0-213.333H341.333z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'italic',
|
||||||
|
title: '倾斜',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M640 853.333H298.667V768h124.885l90.283-512H384v-85.333h341.333V256H600.448l-90.283 512H640z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'delete',
|
||||||
|
title: '删除',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M731.904 597.333c9.813 22.016 14.763 46.507 14.763 73.387 0 57.259-22.358 102.059-67.03 134.272-44.757 32.213-106.496 48.341-185.301 48.341-69.973 0-139.221-16.256-207.787-48.81v-96.256c64.854 37.418 131.2 56.149 199.083 56.149 108.843 0 163.413-31.232 163.797-93.739a94.293 94.293 0 0 0-27.648-68.394l-5.12-4.992H128v-85.334h768v85.334H731.904zm-173.995-128H325.504a174.336 174.336 0 0 1-20.523-22.272c-18.432-23.808-27.648-52.565-27.648-86.442 0-52.736 19.883-97.579 59.606-134.528 39.808-36.95 101.29-55.424 184.533-55.424 62.763 0 122.837 13.994 180.139 41.984v91.818c-51.2-29.312-107.307-43.946-168.363-43.946-105.813 0-158.677 33.365-158.677 100.096 0 17.92 9.301 33.536 27.904 46.89 18.602 13.355 41.557 23.979 68.821 32 26.453 7.68 55.339 17.664 86.613 29.824z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'code-inline',
|
||||||
|
title: '行内代码',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M416.328 512.336a96 96 0 1 0 192 0 96 96 0 1 0-192 0zM336.328 860c-12.288 0-24.568-6-33.944-17.992l-224-286.584c-18.744-23.984-18.744-62.856 0-86.84l224-286.592c18.744-23.976 49.136-23.976 67.88 0 18.744 23.984 18.744 62.864 0 86.848L180.208 512l190.056 243.168c18.744 23.968 18.744 62.856 0 86.832-9.368 12-21.648 18-33.936 18zm352 0c12.28 0 24.568-6 33.936-17.992l224-286.584c18.752-23.984 18.752-62.856 0-86.84l-224-286.592c-18.744-23.976-49.136-23.976-67.872 0-18.752 23.984-18.752 62.864 0 86.848L844.448 512 654.392 755.168c-18.752 23.968-18.752 62.856 0 86.832 9.376 12 21.656 18 33.936 18z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'indent',
|
||||||
|
title: '缩进',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M692.019 192c-22.087 0-39.992 17.935-39.992 40.059v277.192c-.638-9.212-4.471-18.244-11.522-25.286l-280.74-280.353c-15.504-15.483-40.643-15.483-56.147 0-15.505 15.483-15.505 40.587 0 56.07L556.282 512 303.617 764.316c-15.505 15.482-15.505 40.587 0 56.07 15.504 15.484 40.643 15.483 56.146 0l280.74-280.352a40.074 40.074 0 0 0 2.726-3.012c5.322-6.517 8.247-14.329 8.797-22.275V791.82c0 22.122 17.905 40.057 39.992 40.057s39.993-17.935 39.993-40.057V232.059c.001-22.124-17.906-40.059-39.992-40.059z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'hr',
|
||||||
|
title: '横线',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M896 469.333H128c-25.6 0-42.667 17.067-42.667 42.667S102.4 554.667 128 554.667h768c25.6 0 42.667-17.067 42.667-42.667S921.6 469.333 896 469.333z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'quote',
|
||||||
|
title: '引用',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M195.541 739.03C151.595 692.351 128 640 128 555.135c0-149.333 104.832-283.179 257.28-349.355l38.101 58.795c-142.293 76.97-170.112 176.853-181.205 239.83 22.912-11.862 52.907-16 82.304-13.27 76.97 7.125 137.643 70.315 137.643 148.864a149.333 149.333 0 0 1-149.334 149.333 165.163 165.163 0 0 1-117.248-50.304zm426.667 0c-43.947-46.678-67.541-99.03-67.541-183.894 0-149.333 104.832-283.179 257.28-349.355l38.101 58.795c-142.293 76.97-170.112 176.853-181.205 239.83 22.912-11.862 52.906-16 82.304-13.27 76.97 7.125 137.642 70.315 137.642 148.864a149.333 149.333 0 0 1-149.333 149.333 165.163 165.163 0 0 1-117.248-50.304z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'title',
|
||||||
|
title: '标题',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="22" height="22"><path d="M256 213.333h104.875v267.094h324.48V213.333h104.874v640H685.355V570.07h-324.48v283.264H256z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'ordered-list',
|
||||||
|
title: '有序列表',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M341.333 170.667H896V256H341.333v-85.333zm-128-42.667v128H256v42.667H128V256h42.667v-85.333H128V128h85.333zM128 597.333V490.667h85.333v-21.334H128v-42.666h128v106.666h-85.333v21.334H256v42.666H128zM213.333 832H128v-42.667h85.333V768H128v-42.667h128V896H128v-42.667h85.333V832zm128-362.667H896v85.334H341.333v-85.334zm0 298.667H896v85.333H341.333V768z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'unordered-list',
|
||||||
|
title: '无序列表',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M341.333 170.667H896V256H341.333v-85.333zM192 277.333a64 64 0 1 1 0-128 64 64 0 0 1 0 128zM192 576a64 64 0 1 1 0-128 64 64 0 0 1 0 128zm0 294.4a64 64 0 1 1 0-128 64 64 0 0 1 0 128zm149.333-401.067H896v85.334H341.333v-85.334zm0 298.667H896v85.333H341.333V768z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'link',
|
||||||
|
title: '超链接',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="15" height="15"><path d="M879.2 131.6c-103-95.5-264.1-88-361.4 11.2L474.7 184c-13.1 13.1-3.7 35.6 13.1 37.5 26.2 1.9 52.4 7.5 78.7 15 7.5 1.9 16.9 0 22.5-5.6l9.4-9.4c54.3-54.3 142.3-59.9 198.5-11.2 63.7 54.3 65.5 151.7 7.5 209.7L662 562.3c-18.7 18.7-41.2 30-63.7 37.5-30 7.5-61.8 5.6-89.9-5.6-16.9-7.5-33.7-16.9-48.7-31.8-7.5-7.5-13.1-15-18.7-24.3-7.5-13.1-24.3-15-33.7-3.7l-52.4 52.4c-7.5 7.5-7.5 18.7-1.9 28.1 7.5 11.2 16.9 20.6 26.2 30 13.1 13.1 30 26.2 44.9 35.6 26.2 16.9 56.2 28.1 86.1 33.7 58.1 11.2 121.7 1.9 174.2-26.2 20.6-11.2 41.2-26.2 58.1-43.1l142.3-142.3c104.9-103.2 101.2-271.8-5.6-371zM534.7 803.9l-39.3-5.6s-26.2-5.6-39.3-11.2c-7.5-1.9-16.9 0-22.5 5.6l-9.4 9.4c-54.3 54.3-142.3 59.9-198.5 11.2-63.7-54.3-65.5-151.7-7.5-209.7l142.3-142.3c18.7-18.7 41.2-30 63.7-37.5 30-7.5 61.8-5.6 89.9 5.6 16.9 7.5 33.7 16.9 48.7 31.8 7.5 7.5 13.1 15 18.7 24.3 7.5 13.1 24.3 15 33.7 3.7l52.4-52.4c7.5-7.5 7.5-18.7 1.9-28.1-7.5-11.2-16.9-20.6-26.2-30-13.1-13.1-28.1-26.2-44.9-35.6-26.2-16.9-56.2-28.1-88-33.7-58.1-11.2-121.7-1.9-174.2 26.2-20.6 11.2-41.2 26.2-58.1 43.1L141.4 515.5c-99.3 99.3-106.7 260.3-11.2 361.4C229.5 985.5 398 987.4 501 884.4l46.8-46.8c13.1-9.4 3.7-31.9-13.1-33.7z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'image',
|
||||||
|
title: '插入图片',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M46.545 977.455V46.545h930.91v930.91zm93.091-186.182v93.09h744.728V717.918l-214.11-228.305zm0-107.055l548.585-311.854 196.143 209.454V139.636H139.636zm99.282-376.227a82.758 82.758 0 0 1 82.758-82.758 82.758 82.758 0 0 1 82.757 82.758 82.758 82.758 0 0 1-82.757 82.805 82.804 82.804 0 0 1-82.758-82.898z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'table',
|
||||||
|
title: '表格',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="18" height="18"><path d="M892.8 108h-763c-26.2 0-47.5 21.3-47.5 47.5v714c0 26.2 21.3 47.5 47.5 47.5h763c26.2 0 47.5-21.3 47.5-47.5v-714c0-26.2-21.3-47.5-47.5-47.5zm-291 294.7v172.1H411.7V402.7h190.1zm76 0h186.5v172.1H677.8V402.7zM864.3 184v142.7h-706V184h706zM158.4 841V402.7h177.3v172.1H159.8v76h175.9v188.9h76V650.8h190.1v188.9h76V650.8h186.5V841H158.4z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'code-block',
|
||||||
|
title: '代码块',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M902.4 454.4l-144-144a40.704 40.704 0 0 0-57.6 57.6l144 144-144 144a40.704 40.704 0 0 0 57.6 57.6l144-144a81.472 81.472 0 0 0 0-115.2zm-636.8-144l-144 144a81.472 81.472 0 0 0 0 115.2l144 144a40.704 40.704 0 0 0 57.6-57.6l-144-144 144-144a40.704 40.704 0 0 0-57.6-57.6zm109.568 544.064l195.072-706.56a40.704 40.704 0 0 1 78.528 21.632l-195.072 706.56a40.704 40.704 0 0 1-78.528-21.696z" /></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'time',
|
||||||
|
title: '当前时间',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="18" height="18"><path d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm0 820c-205.4 0-372-166.6-372-372s166.6-372 372-372 372 166.6 372 372-166.6 372-372 372z"/><path d="M686.7 638.6L544.1 535.5V288c0-4.4-3.6-8-8-8H488c-4.4 0-8 3.6-8 8v275.4c0 2.6 1.2 5 3.3 6.5l165.4 120.6c3.6 2.6 8.6 1.8 11.2-1.7l28.6-39c2.6-3.7 1.8-8.7-1.8-11.2z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'clean',
|
||||||
|
title: '清屏',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M479.3 863.6L899.9 443c37.4-37.4 37.4-98.3 0-135.8L716.7 124.1C698.5 106 674.4 96 648.7 96c-25.8 0-50.4 10.8-68.6 29l-455 455c-18.2 18.2-29 42.8-29 68.6 0 25.7 9.9 49.9 28.1 68l183.1 183.2c18.1 18.1 42.2 28.1 67.9 28.1 3 0 5.9-.1 8.8-.4v.1h512c17.7 0 32-14.3 32-32s-14.3-32-32-32H479.3zm-126.8-9L169.4 671.5c-6-6-9.4-14.1-9.4-22.6 0-8.5 3.3-16.6 9.4-22.6l104.9-104.9 228.4 228.4-104.9 104.8c-6 6-14.1 9.4-22.6 9.4-8.6 0-16.6-3.3-22.7-9.4z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'download',
|
||||||
|
title: '下载',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M64.3 874.8v62.6c0 4.8 3.9 8.7 8.7 8.7h878c4.8 0 8.7-3.9 8.7-8.7v-62.6c0-4.8-3.9-8.7-8.7-8.7H73c-4.8 0-8.7 3.9-8.7 8.7zm418.9-99.9c3.2 2.9 6.8 5.3 10.9 7.1 5.2 2.3 10.7 3.4 16.1 3.4 9.8 0 19.5-3.6 27-10.5l291.4-270.4c3.5-3.2 3.7-8.7.5-12.2l-42.3-46.2c-3.3-3.6-8.8-3.8-12.3-.5L550.2 654.5v-528c0-4.8-3.9-8.7-8.7-8.7h-62.6c-4.8 0-8.7 3.9-8.7 8.7v527.9L239.6 442.9c-3.5-3.2-9-3-12.3.5L185 489.7c-3.2 3.5-3 9 .5 12.3l297.7 272.9z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'fullScreen',
|
||||||
|
title: '全屏/取消全屏',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M181.2 803.4V637H98v208c0 45.9 37.2 83.2 83.2 83.2h208V845H222.8c-23 0-41.6-18.6-41.6-41.6zM844.8 98.6h-208v83.2h166.4c23 0 41.6 18.6 41.6 41.6v166.4H928v-208c0-46-37.3-83.2-83.2-83.2zm-746.6 83v208h83.2V223.2c0-23 18.6-41.6 41.6-41.6h166.4V98.4h-208c-46 0-83.2 37.3-83.2 83.2zm746.4 455.6v166.4c0 23-18.6 41.6-41.6 41.6H636.6v83.2h208c45.9 0 83.2-37.2 83.2-83.2v-208h-83.2z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'publish',
|
||||||
|
title: '发布文章',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M128 554.667h768a42.667 42.667 0 0 0 0-85.334H128a42.667 42.667 0 0 0 0 85.334z"/><path d="M469.333 128v768a42.667 42.667 0 0 0 85.334 0V128a42.667 42.667 0 0 0-85.334 0z"/></svg>'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'about',
|
||||||
|
title: '关于',
|
||||||
|
innerHTML: '<svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="20" height="20"><path d="M881.6 512.1c0 203.7-165.7 369.5-369.5 369.5S142.6 715.8 142.6 512.1s165.7-369.5 369.5-369.5 369.5 165.8 369.5 369.5m77.7 0c0-246.6-200.6-447.2-447.2-447.2S64.9 265.5 64.9 512.1s200.6 447.2 447.2 447.2 447.2-200.6 447.2-447.2M582.5 318.2c9-9 14.2-21.6 14.2-34.3 0-12.8-5.2-25.4-14.2-34.4s-21.6-14.2-34.4-14.2c-12.7 0-25.3 5.2-34.3 14.2-9.1 9-14.3 21.6-14.3 34.4 0 12.7 5.2 25.3 14.3 34.3 9 9 21.5 14.3 34.3 14.3s25.3-5.3 34.4-14.3m-96.6 464.3c-7.6 0-15.2-2.6-21.3-7.5-9.9-7.9-14.5-20.7-12.1-33.1l47.9-243.1-26.4 14.8c-16.4 9.2-37.1 3.4-46.3-13-9.2-16.4-3.4-37.1 12.9-46.3l90.7-51.1c11.6-6.5 25.9-5.7 36.6 2.1 10.8 7.8 16 21.1 13.4 34.1l-49.6 251.9 40.7-17.7c17.2-7.5 37.3.4 44.8 17.6 7.5 17.2-.4 37.3-17.6 44.8l-100.2 43.7c-4.3 1.9-8.9 2.8-13.5 2.8"/></svg>'
|
||||||
|
}
|
||||||
|
];
|
||||||
|
this.plugins = [classHighlightStyle, history(), bracketMatching(), closeBrackets()];
|
||||||
|
this.keymaps = [defaultTabBinding, ...defaultKeymap, ...historyKeymap, ...closeBracketsKeymap];
|
||||||
|
this.parser = new HyperDown();
|
||||||
|
this._isPasting = false;
|
||||||
|
this.init_ViewPort();
|
||||||
|
this.init_Editor();
|
||||||
|
this.init_Preview();
|
||||||
|
this.init_Tools();
|
||||||
|
this.init_Insert();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 已测 √ */
|
||||||
|
init_ViewPort() {
|
||||||
|
try {
|
||||||
|
if ($('meta[name="viewport"]').length > 0) $('meta[name="viewport"]').attr('content', 'width=device-width, user-scalable=no, initial-scale=1.0, shrink-to-fit=no, viewport-fit=cover');
|
||||||
|
else $('head').append('<meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, shrink-to-fit=no, viewport-fit=cover">');
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Init_ViewPort Error: ' + e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 已测 √ */
|
||||||
|
init_Editor() {
|
||||||
|
try {
|
||||||
|
$('#text').before(`
|
||||||
|
<div class="cm-container">
|
||||||
|
<div class="cm-tools"></div>
|
||||||
|
<div class="cm-mainer">
|
||||||
|
<div class="cm-resize"></div>
|
||||||
|
<div class="cm-preview">
|
||||||
|
<div class="cm-preview-content">${this.parser.makeHtml($('#text').val())}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="cm-progress-left"></div>
|
||||||
|
<div class="cm-progress-right"></div>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
const cm = new EditorView({
|
||||||
|
state: EditorState.create({
|
||||||
|
doc: $('#text').val(),
|
||||||
|
extensions: [
|
||||||
|
...this.plugins,
|
||||||
|
keymap.of([...this.keymaps]),
|
||||||
|
EditorView.updateListener.of(update => {
|
||||||
|
if (!update.docChanged) return;
|
||||||
|
$('.cm-preview-content').html(this.parser.makeHtml(update.state.doc.toString()));
|
||||||
|
}),
|
||||||
|
EditorView.domEventHandlers({
|
||||||
|
paste: e => {
|
||||||
|
const clipboardData = e.clipboardData;
|
||||||
|
if (!clipboardData || !clipboardData.items) return;
|
||||||
|
const items = clipboardData.items;
|
||||||
|
if (!items.length) return;
|
||||||
|
let blob = null;
|
||||||
|
for (let i = 0; i < items.length; i++) {
|
||||||
|
if (items[i].type.indexOf('image') !== -1) {
|
||||||
|
e.preventDefault();
|
||||||
|
blob = items[i].getAsFile();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!blob) return;
|
||||||
|
let api = $('script[api]').attr('api');
|
||||||
|
if (!api) return;
|
||||||
|
const cid = $('input[name="cid"]').val();
|
||||||
|
cid && (api = api + '&cid=' + cid);
|
||||||
|
if (this._isPasting) return;
|
||||||
|
this._isPasting = true;
|
||||||
|
const fileName = Date.now().toString(36) + '.png';
|
||||||
|
let formData = new FormData();
|
||||||
|
formData.append('name', fileName);
|
||||||
|
formData.append('file', blob, fileName);
|
||||||
|
$.ajax({
|
||||||
|
url: api,
|
||||||
|
method: 'post',
|
||||||
|
data: formData,
|
||||||
|
contentType: false,
|
||||||
|
processData: false,
|
||||||
|
xhr: () => {
|
||||||
|
const xhr = $.ajaxSettings.xhr();
|
||||||
|
if (!xhr.upload) return;
|
||||||
|
xhr.upload.addEventListener(
|
||||||
|
'progress',
|
||||||
|
e => {
|
||||||
|
let percent = (e.loaded / e.total) * 100;
|
||||||
|
$('.cm-progress-left').width(percent / 2 + '%');
|
||||||
|
$('.cm-progress-right').width(percent / 2 + '%');
|
||||||
|
},
|
||||||
|
false
|
||||||
|
);
|
||||||
|
return xhr;
|
||||||
|
},
|
||||||
|
success: res => {
|
||||||
|
$('.cm-progress-left').width(0);
|
||||||
|
$('.cm-progress-right').width(0);
|
||||||
|
this._isPasting = false;
|
||||||
|
const str = `${super._getLineCh(cm) ? '\n' : ''}![${res[1].title}](${res[0]})\n`;
|
||||||
|
super._replaceSelection(cm, str);
|
||||||
|
cm.focus();
|
||||||
|
},
|
||||||
|
error: () => {
|
||||||
|
$('.cm-progress-left').width(0);
|
||||||
|
$('.cm-progress-right').width(0);
|
||||||
|
this._isPasting = false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
]
|
||||||
|
})
|
||||||
|
});
|
||||||
|
$('.cm-mainer').prepend(cm.dom);
|
||||||
|
$('form[name="write_post"]').on('submit', () => $('#text').val(cm.state.doc.toString()));
|
||||||
|
this.cm = cm;
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Init_Editor Error: ' + e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 已测 √ */
|
||||||
|
init_Preview() {
|
||||||
|
const move = (nowClientX, nowWidth, clientX) => {
|
||||||
|
let moveX = nowClientX - clientX;
|
||||||
|
let moveWidth = nowWidth + moveX;
|
||||||
|
if (moveWidth <= 0) moveWidth = 0;
|
||||||
|
if (moveWidth >= $('.cm-mainer').outerWidth() - 16) moveWidth = $('.cm-mainer').outerWidth() - 16;
|
||||||
|
$('.cm-preview').width(moveWidth);
|
||||||
|
};
|
||||||
|
$('.cm-resize').on({
|
||||||
|
mousedown: e => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
const nowWidth = $('.cm-preview').outerWidth();
|
||||||
|
const nowClientX = e.clientX;
|
||||||
|
document.onmousemove = _e => {
|
||||||
|
if (window.requestAnimationFrame) requestAnimationFrame(() => move(nowClientX, nowWidth, _e.clientX));
|
||||||
|
else move(nowClientX, nowWidth, _e.clientX);
|
||||||
|
};
|
||||||
|
document.onmouseup = () => {
|
||||||
|
document.onmousemove = null;
|
||||||
|
document.onmouseup = null;
|
||||||
|
};
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
touchstart: e => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
const nowWidth = $('.cm-preview').outerWidth();
|
||||||
|
const nowClientX = e.originalEvent.targetTouches[0].clientX;
|
||||||
|
document.ontouchmove = _e => {
|
||||||
|
if (window.requestAnimationFrame) requestAnimationFrame(() => move(nowClientX, nowWidth, _e.targetTouches[0].clientX));
|
||||||
|
else move(nowClientX, nowWidth, _e.targetTouches[0].clientX);
|
||||||
|
};
|
||||||
|
document.ontouchend = () => {
|
||||||
|
document.ontouchmove = null;
|
||||||
|
document.ontouchend = null;
|
||||||
|
};
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 已测 √ */
|
||||||
|
init_Tools() {
|
||||||
|
this.tools.forEach(item => {
|
||||||
|
if (item.type === 'title') {
|
||||||
|
super.handleTitle(this.cm, item);
|
||||||
|
} else {
|
||||||
|
const el = $(`<div class="cm-tools-item" title="${item.title}">${item.innerHTML}</div>`);
|
||||||
|
el.on('click', e => {
|
||||||
|
e.preventDefault();
|
||||||
|
switch (item.type) {
|
||||||
|
case 'fullScreen':
|
||||||
|
super.handleFullScreen(this.cm, el);
|
||||||
|
break;
|
||||||
|
case 'publish':
|
||||||
|
super.handlePublish();
|
||||||
|
break;
|
||||||
|
case 'undo':
|
||||||
|
super.handleUndo(this.cm);
|
||||||
|
break;
|
||||||
|
case 'redo':
|
||||||
|
super.handleRedo(this.cm);
|
||||||
|
break;
|
||||||
|
case 'time':
|
||||||
|
super.handleTime(this.cm);
|
||||||
|
break;
|
||||||
|
case 'bold':
|
||||||
|
super._insetAmboText(this.cm, '**');
|
||||||
|
break;
|
||||||
|
case 'italic':
|
||||||
|
super._insetAmboText(this.cm, '*');
|
||||||
|
break;
|
||||||
|
case 'delete':
|
||||||
|
super._insetAmboText(this.cm, '~~');
|
||||||
|
break;
|
||||||
|
case 'code-inline':
|
||||||
|
super._insetAmboText(this.cm, '`');
|
||||||
|
break;
|
||||||
|
case 'indent':
|
||||||
|
super.handleIndent(this.cm);
|
||||||
|
break;
|
||||||
|
case 'hr':
|
||||||
|
super.handleHr(this.cm);
|
||||||
|
break;
|
||||||
|
case 'clean':
|
||||||
|
super.handleClean(this.cm);
|
||||||
|
break;
|
||||||
|
case 'ordered-list':
|
||||||
|
super.handleOrdered(this.cm);
|
||||||
|
break;
|
||||||
|
case 'unordered-list':
|
||||||
|
super.handleUnordered(this.cm);
|
||||||
|
break;
|
||||||
|
case 'quote':
|
||||||
|
super.handleQuote(this.cm);
|
||||||
|
break;
|
||||||
|
case 'download':
|
||||||
|
super.handleDownload(this.cm);
|
||||||
|
break;
|
||||||
|
case 'link':
|
||||||
|
super.handleLink(this.cm);
|
||||||
|
break;
|
||||||
|
case 'image':
|
||||||
|
super.handleImage(this.cm);
|
||||||
|
break;
|
||||||
|
case 'table':
|
||||||
|
super.handleTable(this.cm);
|
||||||
|
break;
|
||||||
|
case 'code-block':
|
||||||
|
super.handleCodeBlock(this.cm);
|
||||||
|
break;
|
||||||
|
case 'about':
|
||||||
|
super.handleAbout();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$('.cm-tools').append(el);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 已测 √ */
|
||||||
|
init_Insert() {
|
||||||
|
try {
|
||||||
|
Typecho.insertFileToEditor = (file, url, isImage) => {
|
||||||
|
const str = `${super._getLineCh(this.cm) ? '\n' : ''}${isImage ? '!' : ''}[${file}](${url})\n`;
|
||||||
|
super._replaceSelection(this.cm, str);
|
||||||
|
this.cm.focus();
|
||||||
|
};
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Init_Insert Error: ' + e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', () => new Joe());
|
47
typecho/write/package.json
Normal file
47
typecho/write/package.json
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
{
|
||||||
|
"scripts": {
|
||||||
|
"serve": "rollup -c -w"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@codemirror/autocomplete": "^0.18.3",
|
||||||
|
"@codemirror/basic-setup": "^0.18.0",
|
||||||
|
"@codemirror/buildhelper": "^0.1.3",
|
||||||
|
"@codemirror/closebrackets": "^0.18.0",
|
||||||
|
"@codemirror/collab": "^0.18.1",
|
||||||
|
"@codemirror/commands": "^0.18.0",
|
||||||
|
"@codemirror/comment": "^0.18.0",
|
||||||
|
"@codemirror/fold": "^0.18.0",
|
||||||
|
"@codemirror/gutter": "^0.18.0",
|
||||||
|
"@codemirror/highlight": "^0.18.3",
|
||||||
|
"@codemirror/history": "^0.18.0",
|
||||||
|
"@codemirror/lang-cpp": "^0.18.0",
|
||||||
|
"@codemirror/lang-css": "^0.18.0",
|
||||||
|
"@codemirror/lang-html": "^0.18.0",
|
||||||
|
"@codemirror/lang-java": "^0.18.0",
|
||||||
|
"@codemirror/lang-javascript": "^0.18.0",
|
||||||
|
"@codemirror/lang-json": "^0.18.0",
|
||||||
|
"@codemirror/lang-lezer": "^0.18.0",
|
||||||
|
"@codemirror/lang-markdown": "^0.18.0",
|
||||||
|
"@codemirror/lang-python": "^0.18.0",
|
||||||
|
"@codemirror/lang-rust": "^0.18.0",
|
||||||
|
"@codemirror/lang-sql": "^0.18.0",
|
||||||
|
"@codemirror/lang-xml": "^0.18.0",
|
||||||
|
"@codemirror/language": "^0.18.0",
|
||||||
|
"@codemirror/language-data": "^0.18.0",
|
||||||
|
"@codemirror/legacy-modes": "^0.18.0",
|
||||||
|
"@codemirror/lint": "^0.18.1",
|
||||||
|
"@codemirror/matchbrackets": "^0.18.0",
|
||||||
|
"@codemirror/panel": "^0.18.1",
|
||||||
|
"@codemirror/rangeset": "^0.18.0",
|
||||||
|
"@codemirror/rectangular-selection": "^0.18.0",
|
||||||
|
"@codemirror/search": "^0.18.2",
|
||||||
|
"@codemirror/state": "^0.18.2",
|
||||||
|
"@codemirror/stream-parser": "^0.18.1",
|
||||||
|
"@codemirror/text": "^0.18.0",
|
||||||
|
"@codemirror/theme-one-dark": "^0.18.0",
|
||||||
|
"@codemirror/tooltip": "^0.18.4",
|
||||||
|
"@codemirror/view": "^0.18.3",
|
||||||
|
"@rollup/plugin-node-resolve": "^11.2.0",
|
||||||
|
"rollup-plugin-uglify": "^6.0.4"
|
||||||
|
}
|
||||||
|
}
|
10
typecho/write/rollup.config.js
Normal file
10
typecho/write/rollup.config.js
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
import { nodeResolve } from '@rollup/plugin-node-resolve';
|
||||||
|
import { uglify } from 'rollup-plugin-uglify';
|
||||||
|
export default {
|
||||||
|
input: './js/joe.write.js',
|
||||||
|
output: {
|
||||||
|
file: './js/joe.write.chunk.js',
|
||||||
|
format: 'iife'
|
||||||
|
},
|
||||||
|
plugins: [nodeResolve()]
|
||||||
|
};
|
Loading…
Reference in New Issue
Block a user