提交 e504e13c 编写于 作者: 璃白.'s avatar 璃白. 🌻

feat:初始化编辑器

上级 c97c8733
{ {
"scripts": { "scripts": {
"dev": "webpack serve --mode=development", "start": "webpack serve --mode=development",
"build": "webpack --mode=production" "build": "webpack --mode=production"
}, },
"devDependencies": { "devDependencies": {
......
...@@ -3,7 +3,10 @@ ...@@ -3,7 +3,10 @@
<markdown-header /> <markdown-header />
<markdownPreview v-if="showPreview" /> <markdownPreview v-if="showPreview" />
<markdown-editor v-else /> <markdown-editor v-else />
<markdown-footer v-if="!showPreview" /> <markdown-footer
:can-attach-file="canAttachFile"
v-if="!showPreview && canAttachFile"
/>
</div> </div>
</template> </template>
<script> <script>
...@@ -19,8 +22,9 @@ export default { ...@@ -19,8 +22,9 @@ export default {
markdownEditor, markdownEditor,
markdownPreview markdownPreview
}, },
computed: { computed: {
...mapState(["showPreview", "isFocus"]) ...mapState(["showPreview", "isFocus", "canAttachFile"])
} }
}; };
</script> </script>
...@@ -28,13 +32,13 @@ export default { ...@@ -28,13 +32,13 @@ export default {
.md_container { .md_container {
width: 1000px; width: 1000px;
margin: 200px auto; margin: 200px auto;
border: 1px solid var(--md-editor-theme-color); border: 1px solid var(--md-editor-border-color);
border-radius: 4px; border-radius: 4px;
padding: 10px 16px; padding: 10px 16px;
box-sizing: border-box; box-sizing: border-box;
transition: border 0.3s; transition: border 0.3s;
&.active { &.active {
border: 1px solid var(--md-editor-theme-color-active); border: 1px solid var(--md-editor-border-color-active);
} }
} }
</style> </style>
// 获取选中文本信息
export function getSelectionInfo(selectorId) { export function getSelectionInfo(selectorId) {
const selector = document.getElementById(selectorId); const selector = document.getElementById(selectorId);
// const selection = window.getSelection(); // const selection = window.getSelection();
...@@ -9,6 +10,7 @@ export function getSelectionInfo(selectorId) { ...@@ -9,6 +10,7 @@ export function getSelectionInfo(selectorId) {
}; };
} }
// 工具栏格式化文本
export function formatText(text, selectionInfo, startStr = "", endStr = "") { export function formatText(text, selectionInfo, startStr = "", endStr = "") {
if (!selectionInfo) return text + startStr + endStr; if (!selectionInfo) return text + startStr + endStr;
return ( return (
...@@ -19,3 +21,41 @@ export function formatText(text, selectionInfo, startStr = "", endStr = "") { ...@@ -19,3 +21,41 @@ export function formatText(text, selectionInfo, startStr = "", endStr = "") {
text.slice(selectionInfo.selectionEnd) text.slice(selectionInfo.selectionEnd)
); );
} }
// 初始化样式
export function initStyle({
borderColor,
borderColorActive,
textColor,
textColorActive
}) {
if (borderColor) {
document.documentElement.style.setProperty(
"--md-editor-border-color",
borderColor
);
}
if (borderColorActive) {
document.documentElement.style.setProperty(
"--md-editor-border-color-active",
borderColorActive
);
}
if (textColor) {
document.documentElement.style.setProperty(
"--md-editor-text-color",
textColor
);
}
if (textColorActive) {
document.documentElement.style.setProperty(
"--md-editor-text-color-active",
textColorActive
);
}
}
//
export function isNotEmpty(val) {
return val !== null && val !== undefined;
}
@import "./variable.less"; @import "./variable.less";
@import "./iconfont.less"; @import "./iconfont.less";
@import './markdown.less';
* { * {
padding: 0; padding: 0;
......
.htmledit_views address,
.htmledit_views cite,
.htmledit_views dfn,
.htmledit_views em,
.htmledit_views i,
.htmledit_views span[lang],
.htmledit_views var {
font-style: italic;
}
.htmledit_views {
font-family: -apple-system, SF UI Text, Arial, PingFang SC, Hiragino Sans GB,
Microsoft YaHei, WenQuanYi Micro Hei, sans-serif, SimHei, SimSun;
}
.cke_editable {
margin: 24px;
word-wrap: break-word;
}
.htmledit_views img.right {
border: 1px solid #ccc;
float: right;
margin-left: 15px;
padding: 5px;
}
.htmledit_views img.left {
border: 1px solid #ccc;
float: left;
margin-right: 15px;
padding: 5px;
}
.htmledit_views .marker {
background-color: #ff0;
}
.htmledit_views figure {
text-align: center;
margin: 0 24px;
display: inline-block;
}
.htmledit_views figure > figcaption {
text-align: center;
margin: 8px;
color: #999;
font-size: 14px;
display: block;
}
.htmledit_views a > img {
padding: 1px;
margin: 1px;
border: none;
outline: #0782c1 solid 1px;
}
.htmledit_views .code-featured {
border: 5px solid red;
}
.htmledit_views .math-featured {
padding: 20px;
box-shadow: 0 0 2px rgba(200, 0, 0, 1);
background-color: rgba(255, 0, 0, 0.05);
margin: 10px;
}
.htmledit_views .image-clean {
border: 0;
background: 0 0;
padding: 0;
}
.htmledit_views .image-clean > figcaption {
font-size: 0.9em;
text-align: right;
}
.htmledit_views .image-grayscale {
background-color: #fff;
color: #666;
}
.htmledit_views .image-grayscale img,
.htmledit_views img.image-grayscale {
filter: grayscale(100%);
}
.htmledit_views .embed-240p {
max-width: 426px;
max-height: 240px;
margin: 0 auto;
}
.htmledit_views .embed-360p {
max-width: 640px;
max-height: 360px;
margin: 0 auto;
}
.htmledit_views .embed-480p {
max-width: 854px;
max-height: 480px;
margin: 0 auto;
}
.htmledit_views .embed-720p {
max-width: 1280px;
max-height: 720px;
margin: 0 auto;
}
.htmledit_views .embed-1080p {
max-width: 1920px;
max-height: 1080px;
margin: 0 auto;
}
.htmledit_views p {
font-size: 18px;
color: #4d4d4d;
font-weight: 400;
line-height: 26px !important;
margin: 0 0 16px;
overflow-x: auto;
overflow-y: hidden;
}
p[align="center"] {
text-align: center;
}
.htmledit_views img {
max-width: 100%;
height: auto;
}
.htmledit_views strong,
.htmledit_views strong span {
font-weight: 700;
}
.htmledit_views * {
box-sizing: border-box;
}
.htmledit_views h1,
.htmledit_views h2,
.htmledit_views h3,
.htmledit_views h4,
.htmledit_views h5,
.htmledit_views h6 {
color: #4f4f4f;
margin: 8px 0 16px;
font-weight: 700;
}
.htmledit_views ol,
.htmledit_views ul {
margin: 0 0 24px;
padding: 0;
font-size: 16px;
}
.htmledit_views ul ol {
margin: 0 0 24px 32px;
}
.htmledit_views ul li {
list-style-type: disc;
margin: 8px 0 0 32px;
}
.htmledit_views ol li {
list-style-type: decimal;
margin-left: 40px;
margin-top: 8px;
}
.htmledit_views h1 {
font-size: 28px;
line-height: 36px;
}
.htmledit_views h2 {
font-size: 24px;
line-height: 32px;
}
.htmledit_views h3 {
font-size: 22px;
line-height: 30px;
}
.htmledit_views h4 {
font-size: 20px;
line-height: 28px;
}
.htmledit_views h5 {
font-size: 18px;
line-height: 26px;
}
.htmledit_views h6 {
font-size: 16px;
line-height: 24px;
}
.htmledit_views blockquote {
display: block;
padding: 16px 16px 0;
margin: 0 0 24px;
border-left: 8px solid #dddfe4;
background: #eef0f4;
overflow: auto;
overflow-scrolling: touch;
word-wrap: normal;
word-break: normal;
}
.htmledit_views blockquote ol,
.htmledit_views blockquote ul {
margin-bottom: 16px;
padding: 0;
font-size: 16px;
line-height: 24px;
}
.htmledit_views blockquote ol li,
.htmledit_views blockquote ul li {
margin-bottom: 0;
}
.htmledit_views blockquote p {
font-size: 16px;
line-height: 26px;
font-weight: 400;
margin-bottom: 16px;
color: #4f4f4f;
}
.htmledit_views hr {
margin: 24px 0;
border: none;
border-bottom: solid #ccc 1px;
}
.htmledit_views table tr {
border: 0;
border-top: 1px solid #ddd;
background-color: #fff;
}
.htmledit_views table {
border-collapse: collapse;
display: table;
width: 100%;
text-align: left;
margin-bottom: 24px;
margin-left: auto;
margin-right: auto;
}
.htmledit_views tbody {
border: 0;
}
.htmledit_views table tr:nth-child(2n) {
background-color: #f7f7f7;
}
.htmledit_views table tr td,
.htmledit_views table tr th {
border: 1px solid #ddd;
font-size: 14px;
color: #4f4f4f;
line-height: 22px;
padding: 8px;
text-align: left;
}
.htmledit_views table tr td p,
.htmledit_views table tr th p {
font-size: 14px;
color: #4f4f4f;
margin: 0;
padding: 0;
text-align: left;
line-height: 22px;
}
.htmledit_views table tr th p {
font-weight: 700;
}
.htmledit_views table tr td code,
.htmledit_views table tr th code {
white-space: normal;
word-break: break-word;
}
.htmledit_views table tr th {
font-weight: 700;
background-color: #eff3f5;
}
.htmledit_views dl {
margin: 24px;
}
.htmledit_views dl dt {
margin: 8px;
font-weight: 700;
}
.htmledit_views dl dt dd {
margin: 8px;
}
.htmledit_views abbr[data-original-title],
.htmledit_views abbr[title] {
cursor: help;
border-bottom: 1px dotted #999;
}
.htmledit_views .initialism {
font-size: 90%;
text-transform: uppercase;
}
.htmledit_views pre {
white-space: pre-wrap;
word-wrap: break-word;
margin: 0 0 24px;
overflow-x: auto;
padding: 8px;
}
.marker {
background-color: #ff0;
}
htmledit_views img.mathcode {
margin: 0 8px;
}
.htmledit_views kbd,
.htmledit_views pre,
.htmledit_views samp {
font-family: Consolas, Inconsolata, Courier, monospace;
font-size: 14px;
line-height: 22px;
color: #000;
}
.htmledit_views code ol li div.hljs-ln-code,
.htmledit_views code ol li div.hljs-ln-numbers,
.htmledit_views pre code,
.htmledit_views pre code div,
.htmledit_views pre code span {
font-family: "Source Code Pro", "DejaVu Sans Mono", "Ubuntu Mono",
"Anonymous Pro", "Droid Sans Mono", Menlo, Monaco, Consolas, Inconsolata,
Courier, monospace, "PingFang SC", "Microsoft YaHei", sans-serif;
}
.htmledit_views code {
border-radius: 4px;
}
.htmledit_views a {
color: #4ea1db;
text-decoration: none;
}
.htmledit_views a:focus,
.htmledit_views a:hover {
color: #ca0c16;
}
.htmledit_views a:visited {
color: #6795b5;
}
.htmledit_views .hot-keyword:visited {
color: #3399ea;
}
.htmledit_views .footnote {
vertical-align: top;
position: relative;
top: -4px;
font-size: 12px;
}
.htmledit_views .flow-chart,
.htmledit_views .sequence-diagram {
text-align: center;
margin-bottom: 24px;
font-size: 14px !important;
}
.htmledit_views .flow-chart [fill="#000"],
.htmledit_views .flow-chart [fill="#000000"],
.htmledit_views .flow-chart [fill="black"],
.htmledit_views .sequence-diagram [fill="#000"],
.htmledit_views .sequence-diagram [fill="#000000"],
.htmledit_views .sequence-diagram [fill="black"] {
fill: #4f4f4f;
}
.htmledit_views .flow-chart [stroke="#000000"],
.htmledit_views .sequence-diagram [stroke="#000000"] {
stroke: #4f4f4f;
}
.htmledit_views .MathJax_SVG_Display {
text-align: center;
margin: 24px 0;
font-size: 18px;
font-weight: 400;
color: #4f4f4f;
position: relative;
text-indent: 0;
max-width: none;
max-height: none;
min-width: 0;
min-height: 0;
width: 100%;
}
.htmledit_views pre code {
display: block;
line-height: 22px;
overflow-x: auto;
white-space: pre;
word-wrap: normal;
border-radius: 4px;
padding: 8px;
tab-size: 4;
}
.htmledit_views pre code code.hljs {
padding: 0;
}
.htmledit_views pre code:not(.hljs) {
background-color: #f3f4f5;
}
.htmledit_views pre code,
.htmledit_views pre code div,
.htmledit_views pre code span {
font-size: 14px;
}
.htmledit_views .prettyprint,
.htmledit_views pre.prettyprint {
margin: 0 0 24px;
padding: 8px 16px 4px 56px;
background-color: #f6f8fa;
border: none;
}
.htmledit_views .prettyprint .pre-numbering {
position: absolute;
width: 48px;
background-color: #eef0f4;
top: 0;
left: 0;
margin: 0;
padding: 8px 0;
list-style: none;
text-align: right;
}
.htmledit_views .pre-numbering li {
padding: 0 8px;
list-style: none;
margin: 0;
}
.htmledit_views dl dd {
margin: 0 0 8px 40px;
}
.htmledit_views kbd {
padding: 2px 8px;
border: 1px solid rgba(63, 63, 63, 0.25);
box-shadow: 0 1px 0 rgba(63, 63, 63, 0.25);
background-color: #fff;
color: #333;
border-radius: 4px;
display: inline-block;
margin: 0 2px;
white-space: nowrap;
}
.htmledit_views .dp-highlighter {
font-size: 12px;
text-align: left;
margin: 0 0 24px;
}
.htmledit_views code ol {
margin: 0;
overflow: hidden;
}
.htmledit_views code ol li {
list-style-type: none;
margin-left: 0;
margin-top: 0;
height: 22px;
}
.htmledit_views code ol li div.hljs-ln-code {
margin-left: 8px;
}
.htmledit_views code ol li div.hljs-ln-code,
.htmledit_views code ol li div.hljs-ln-numbers {
float: left;
height: 22px;
}
.htmledit_views code ol li div.hljs-ln-numbers {
width: 24px;
border-right: 1px solid #c5c5c5;
}
.htmledit_views code ol li div.hljs-ln-numbers .hljs-ln-line {
word-wrap: normal;
}
.htmledit_views code ol.hundred li div.hljs-ln-numbers {
width: 30px;
}
.htmledit_views code ol.thousand li div.hljs-ln-numbers {
width: 36px;
}
.htmledit_views .csdn-data-video {
width: 200px;
}
.htmledit_views .csdn-data-video img {
margin: 0;
display: block;
width: 200px;
height: 112px;
}
.htmledit_views .csdn-data-video p {
margin-top: 10px;
margin-bottom: 0;
font-size: 14px;
}
.htmledit_views pre[data-from="code-for-outside"] {
overflow: hidden;
}
.htmledit_views pre[data-from="code-for-outside"] code {
overflow-x: auto;
overflow-y: hidden;
}
.htmledit_views pre[data-from="code-for-outside"] code * {
overflow: visible;
overflow-wrap: break-word;
}
.htmledit_views pre[data-from="code-for-outside"] p code {
padding: 0;
}
:root { :root {
--md-editor-theme-color: #dbdbdb; --md-editor-border-color: #dbdbdb;
--md-editor-theme-color-active: #409eff; --md-editor-border-color-active: #409eff;
--md-editor-text-color: #303030; --md-editor-text-color: #303030;
--md-editor-text-color-active: #000; --md-editor-text-color-active: #000;
} }
<template> <template>
<div class="md_preview"> <div class="htmledit_views">
<div v-html="html"></div> <div v-html="html"></div>
</div> </div>
</template> </template>
...@@ -32,7 +32,7 @@ export default { ...@@ -32,7 +32,7 @@ export default {
}; };
</script> </script>
<style lang="less" scoped> <style lang="less" scoped>
.md_preview { .htmledit_views {
min-height: 170px; min-height: 170px;
padding: 10px 0; padding: 10px 0;
box-sizing: border-box; box-sizing: border-box;
......
<template> <template>
<div :class="['md_footer', { active: isFocus }]"> <div :class="['md_footer', { active: isFocus }]">
<div class="doc"></div> <div class="doc"></div>
<upload-files /> <upload-files v-if="canAttachFile" />
</div> </div>
</template> </template>
<script> <script>
...@@ -9,11 +9,8 @@ import uploadFiles from "./upload-files"; ...@@ -9,11 +9,8 @@ import uploadFiles from "./upload-files";
import { mapState } from "vuex"; import { mapState } from "vuex";
export default { export default {
components: { uploadFiles }, components: { uploadFiles },
data() {
return {};
},
computed: { computed: {
...mapState(["isFocus"]) ...mapState(["isFocus", "canAttachFile"])
} }
}; };
</script> </script>
...@@ -22,14 +19,14 @@ export default { ...@@ -22,14 +19,14 @@ export default {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
border-top: 1px solid var(--md-editor-theme-color); border-top: 1px solid var(--md-editor-border-color);
padding-top: 10px; padding-top: 10px;
box-sizing: border-box; box-sizing: border-box;
font-size: 14px; font-size: 14px;
color: var(--md-editor-text-color); color: var(--md-editor-text-color);
transition: border-top 0.3s; transition: border-top 0.3s;
&.active { &.active {
border-top: 1px solid var(--md-editor-theme-color-active); border-top: 1px solid var(--md-editor-border-color-active);
} }
} }
</style> </style>
<template> <template>
<div class="upload_files" @click="upload"> <div class="upload_files" @click="$refs.mdUploadFile.click()">
<span class="icon iconfont icon-tupian"></span> <span class="icon iconfont icon-tupian"></span>
<input
ref="mdUploadFile"
class="md_upload"
type="file"
name="md-upload-file"
id="md-upload-file"
@change="upload"
multiple
/>
<span>添加附件</span> <span>添加附件</span>
</div> </div>
</template> </template>
<script> <script>
import { mapState, mapMutations } from "vuex";
import { formatText } from "@/assets/js/utils";
export default { export default {
data() { data() {
return {}; return {};
}, },
computed: {
...mapState(["text", "selectionInfo"])
},
methods: { methods: {
upload() {} ...mapMutations(["setText"]),
upload(e) {
const fileList = Array.from(e.target.files);
console.log(fileList);
fileList.forEach(item => {
console.log(item);
// const formData = new formData();
// formData.append("file", item);
this.updateText(
"![",
"]()"
);
});
},
updateText(startStr, endStr) {
const selectionInfo = this.selectionInfo;
const originalText = this.text;
const newText = formatText(originalText, selectionInfo, startStr, endStr);
if (!newText) return;
this.setText(newText);
}
} }
}; };
</script> </script>
...@@ -23,9 +57,12 @@ export default { ...@@ -23,9 +57,12 @@ export default {
cursor: pointer; cursor: pointer;
display: inline-block; display: inline-block;
vertical-align: bottom; vertical-align: bottom;
&:hover { }
color: var(--md-editor-text-color-active); &:hover {
} color: var(--md-editor-text-color-active);
}
.md_upload {
display: none;
} }
} }
</style> </style>
...@@ -43,9 +43,9 @@ export default { ...@@ -43,9 +43,9 @@ export default {
align-items: center; align-items: center;
height: 32px; height: 32px;
transition: border-bottom 0.3s; transition: border-bottom 0.3s;
border-bottom: 1px solid var(--md-editor-theme-color); border-bottom: 1px solid var(--md-editor-border-color);
&.active { &.active {
border-bottom: 1px solid var(--md-editor-theme-color-active); border-bottom: 1px solid var(--md-editor-border-color-active);
} }
.header_tabs { .header_tabs {
display: flex; display: flex;
...@@ -76,7 +76,7 @@ export default { ...@@ -76,7 +76,7 @@ export default {
color: var(--md-editor-text-color-active); color: var(--md-editor-text-color-active);
&::after { &::after {
width: 100%; width: 100%;
background: var(--md-editor-theme-color); background: var(--md-editor-border-color);
} }
} }
&.active { &.active {
...@@ -84,7 +84,7 @@ export default { ...@@ -84,7 +84,7 @@ export default {
font-weight: 700; font-weight: 700;
&::after { &::after {
width: 100%; width: 100%;
background: var(--md-editor-theme-color-active); background: var(--md-editor-border-color-active);
} }
} }
& + .tab_item { & + .tab_item {
......
import Vue from "vue"; import Vue from "vue";
import App from "./App"; import App from "./App";
import store from "./store"; import store from "./store";
import { initStyle, isNotEmpty } from "@/assets/js/utils";
import "@/assets/style/global.less"; import "@/assets/style/global.less";
function initStyle(val) { function initMdEditor(obj) {
document.documentElement.style.setProperty( const { el, onChange, canAttachFile, themeOptions } = obj;
"--md-editor-theme-color-active",
val
);
}
function MdEditor(obj) {
const { el, onChange, themeActive } = obj;
if (!el || !document.querySelector(el)) throw new Error("请指定容器"); if (!el || !document.querySelector(el)) throw new Error("请指定容器");
initStyle(themeActive); if (isNotEmpty(themeOptions)) initStyle(themeOptions);
if (isNotEmpty(canAttachFile))
store.commit("setCanAttachFile", canAttachFile);
new Vue({ new Vue({
store, store,
...@@ -28,4 +24,4 @@ function MdEditor(obj) { ...@@ -28,4 +24,4 @@ function MdEditor(obj) {
}).$mount(el); }).$mount(el);
} }
window.MdEditor = MdEditor; window.MdEditor = initMdEditor;
...@@ -52,9 +52,10 @@ export default new Vuex.Store({ ...@@ -52,9 +52,10 @@ export default new Vuex.Store({
} }
], ],
ulNum: 1, ulNum: 1,
text: "4564564564657567\n3456456456456", text: "# 456456456\n## 4657567\n3456456456456",
selectionInfo: "", selectionInfo: "",
html: "" html: "",
canAttachFile: true
}, },
mutations: { mutations: {
setFullScreen(state, val) { setFullScreen(state, val) {
...@@ -68,6 +69,7 @@ export default new Vuex.Store({ ...@@ -68,6 +69,7 @@ export default new Vuex.Store({
}, },
setText(state, val) { setText(state, val) {
state.text = val; state.text = val;
state.selectionInfo = "";
}, },
setSelectionInfo(state, val) { setSelectionInfo(state, val) {
state.selectionInfo = val; state.selectionInfo = val;
...@@ -77,6 +79,9 @@ export default new Vuex.Store({ ...@@ -77,6 +79,9 @@ export default new Vuex.Store({
}, },
setUlNum(state, val) { setUlNum(state, val) {
state.ulNum = val; state.ulNum = val;
},
setCanAttachFile(state, val) {
state.canAttachFile = val;
} }
} }
}); });
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册