mirror of
https://gitee.com/baidu/amis.git
synced 2024-11-29 18:48:45 +08:00
chore: 调整公式输入交互并补充编辑器模版文档 (#10034)
This commit is contained in:
parent
ec1cf05e39
commit
1a6a680020
@ -18,6 +18,15 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"path": "./packages/office-viewer"
|
"path": "./packages/office-viewer"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "./packages/amis-editor-core"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "./packages/amis-editor"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "./packages/amis-theme-editor-helper"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
765
examples/doc.css
765
examples/doc.css
@ -1,751 +1,17 @@
|
|||||||
/* csshint-disable */
|
|
||||||
@font-face {
|
|
||||||
font-family: octicons-link;
|
|
||||||
src: url(data:font/woff;charset=utf-8;base64,d09GRgABAAAAAAZwABAAAAAACFQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABEU0lHAAAGaAAAAAgAAAAIAAAAAUdTVUIAAAZcAAAACgAAAAoAAQAAT1MvMgAAAyQAAABJAAAAYFYEU3RjbWFwAAADcAAAAEUAAACAAJThvmN2dCAAAATkAAAABAAAAAQAAAAAZnBnbQAAA7gAAACyAAABCUM+8IhnYXNwAAAGTAAAABAAAAAQABoAI2dseWYAAAFsAAABPAAAAZwcEq9taGVhZAAAAsgAAAA0AAAANgh4a91oaGVhAAADCAAAABoAAAAkCA8DRGhtdHgAAAL8AAAADAAAAAwGAACfbG9jYQAAAsAAAAAIAAAACABiATBtYXhwAAACqAAAABgAAAAgAA8ASm5hbWUAAAToAAABQgAAAlXu73sOcG9zdAAABiwAAAAeAAAAME3QpOBwcmVwAAAEbAAAAHYAAAB/aFGpk3jaTY6xa8JAGMW/O62BDi0tJLYQincXEypYIiGJjSgHniQ6umTsUEyLm5BV6NDBP8Tpts6F0v+k/0an2i+itHDw3v2+9+DBKTzsJNnWJNTgHEy4BgG3EMI9DCEDOGEXzDADU5hBKMIgNPZqoD3SilVaXZCER3/I7AtxEJLtzzuZfI+VVkprxTlXShWKb3TBecG11rwoNlmmn1P2WYcJczl32etSpKnziC7lQyWe1smVPy/Lt7Kc+0vWY/gAgIIEqAN9we0pwKXreiMasxvabDQMM4riO+qxM2ogwDGOZTXxwxDiycQIcoYFBLj5K3EIaSctAq2kTYiw+ymhce7vwM9jSqO8JyVd5RH9gyTt2+J/yUmYlIR0s04n6+7Vm1ozezUeLEaUjhaDSuXHwVRgvLJn1tQ7xiuVv/ocTRF42mNgZGBgYGbwZOBiAAFGJBIMAAizAFoAAABiAGIAznjaY2BkYGAA4in8zwXi+W2+MjCzMIDApSwvXzC97Z4Ig8N/BxYGZgcgl52BCSQKAA3jCV8CAABfAAAAAAQAAEB42mNgZGBg4f3vACQZQABIMjKgAmYAKEgBXgAAeNpjYGY6wTiBgZWBg2kmUxoDA4MPhGZMYzBi1AHygVLYQUCaawqDA4PChxhmh/8ODDEsvAwHgMKMIDnGL0x7gJQCAwMAJd4MFwAAAHjaY2BgYGaA4DAGRgYQkAHyGMF8NgYrIM3JIAGVYYDT+AEjAwuDFpBmA9KMDEwMCh9i/v8H8sH0/4dQc1iAmAkALaUKLgAAAHjaTY9LDsIgEIbtgqHUPpDi3gPoBVyRTmTddOmqTXThEXqrob2gQ1FjwpDvfwCBdmdXC5AVKFu3e5MfNFJ29KTQT48Ob9/lqYwOGZxeUelN2U2R6+cArgtCJpauW7UQBqnFkUsjAY/kOU1cP+DAgvxwn1chZDwUbd6CFimGXwzwF6tPbFIcjEl+vvmM/byA48e6tWrKArm4ZJlCbdsrxksL1AwWn/yBSJKpYbq8AXaaTb8AAHja28jAwOC00ZrBeQNDQOWO//sdBBgYGRiYWYAEELEwMTE4uzo5Zzo5b2BxdnFOcALxNjA6b2ByTswC8jYwg0VlNuoCTWAMqNzMzsoK1rEhNqByEyerg5PMJlYuVueETKcd/89uBpnpvIEVomeHLoMsAAe1Id4AAAAAAAB42oWQT07CQBTGv0JBhagk7HQzKxca2sJCE1hDt4QF+9JOS0nbaaYDCQfwCJ7Au3AHj+LO13FMmm6cl7785vven0kBjHCBhfpYuNa5Ph1c0e2Xu3jEvWG7UdPDLZ4N92nOm+EBXuAbHmIMSRMs+4aUEd4Nd3CHD8NdvOLTsA2GL8M9PODbcL+hD7C1xoaHeLJSEao0FEW14ckxC+TU8TxvsY6X0eLPmRhry2WVioLpkrbp84LLQPGI7c6sOiUzpWIWS5GzlSgUzzLBSikOPFTOXqly7rqx0Z1Q5BAIoZBSFihQYQOOBEdkCOgXTOHA07HAGjGWiIjaPZNW13/+lm6S9FT7rLHFJ6fQbkATOG1j2OFMucKJJsxIVfQORl+9Jyda6Sl1dUYhSCm1dyClfoeDve4qMYdLEbfqHf3O/AdDumsjAAB42mNgYoAAZQYjBmyAGYQZmdhL8zLdDEydARfoAqIAAAABAAMABwAKABMAB///AA8AAQAAAAAAAAAAAAAAAAABAAAAAA==)
|
|
||||||
format('woff');
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown {
|
|
||||||
padding: 25px 45px 25px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body {
|
.markdown-body {
|
||||||
-ms-text-size-adjust: 100%;
|
|
||||||
-webkit-text-size-adjust: 100%;
|
|
||||||
color: #333;
|
|
||||||
font-family: 'PingFang SC', 'Microsoft YaHei', 'Hiragino Sans GB', 'STHeiti',
|
|
||||||
'Microsoft YaHei', 'Helvetica Neue', Helvetica, Arial, sans-serif;
|
|
||||||
font-size: 16px;
|
|
||||||
line-height: 1.6;
|
|
||||||
word-wrap: break-word;
|
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
min-width: 200px;
|
min-width: 200px;
|
||||||
/* max-width: 980px; */
|
max-width: 980px;
|
||||||
/* margin: 0 auto; */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.markdown-body a:not(.btn) {
|
@media (max-width: 767px) {
|
||||||
background-color: transparent;
|
.markdown-body {
|
||||||
-webkit-text-decoration-skip: objects;
|
padding: 15px;
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body a:active:not(.btn),
|
|
||||||
.markdown-body a:hover:not(.btn) {
|
|
||||||
outline-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body strong {
|
|
||||||
font-weight: inherit;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body strong {
|
|
||||||
font-weight: bolder;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1 {
|
|
||||||
font-size: 2em;
|
|
||||||
margin: 0.67em 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body img {
|
|
||||||
border-style: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body svg:not(:root) {
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body code,
|
|
||||||
.markdown-body kbd,
|
|
||||||
.markdown-body pre {
|
|
||||||
font-family: monospace, monospace;
|
|
||||||
font-size: 1em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body hr {
|
|
||||||
box-sizing: content-box;
|
|
||||||
height: 0;
|
|
||||||
overflow: visible;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body input {
|
|
||||||
font: inherit;
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body input {
|
|
||||||
overflow: visible;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body button:-moz-focusring,
|
|
||||||
.markdown-body [type='button']:-moz-focusring,
|
|
||||||
.markdown-body [type='reset']:-moz-focusring,
|
|
||||||
.markdown-body [type='submit']:-moz-focusring {
|
|
||||||
outline: 1px dotted ButtonText;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body [type='checkbox'] {
|
|
||||||
box-sizing: border-box;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table:not(.table) {
|
|
||||||
border-spacing: 0;
|
|
||||||
border-collapse: collapse;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table:not(.table) td,
|
|
||||||
.markdown-body table:not(.table) th {
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body * {
|
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body input {
|
|
||||||
font: 13px/1.4 Helvetica, arial, nimbussansl, liberationsans, freesans, clean,
|
|
||||||
sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 'Segoe UI Symbol';
|
|
||||||
}
|
|
||||||
|
|
||||||
/* .markdown-body a:not(.btn) {
|
|
||||||
color: #4078c0;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body a:hover:not(.btn),
|
|
||||||
.markdown-body a:active:not(.btn) {
|
|
||||||
color: #4078c0;
|
|
||||||
text-decoration: underline;
|
|
||||||
} */
|
|
||||||
|
|
||||||
.markdown-body hr {
|
|
||||||
height: 0;
|
|
||||||
margin: 15px 0;
|
|
||||||
overflow: hidden;
|
|
||||||
background: transparent;
|
|
||||||
border: 0;
|
|
||||||
border-bottom: 1px solid #ddd;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body hr::before {
|
|
||||||
display: table;
|
|
||||||
content: '';
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body hr::after {
|
|
||||||
display: table;
|
|
||||||
clear: both;
|
|
||||||
content: '';
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1,
|
|
||||||
.markdown-body > h2,
|
|
||||||
.markdown-body > h3,
|
|
||||||
.markdown-body > h4,
|
|
||||||
.markdown-body > h5,
|
|
||||||
.markdown-body > h6 {
|
|
||||||
margin-top: 0;
|
|
||||||
margin-bottom: 0;
|
|
||||||
line-height: 1.5;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1 {
|
|
||||||
font-size: 30px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h2 {
|
|
||||||
font-size: 21px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h3 {
|
|
||||||
font-size: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h4 {
|
|
||||||
font-size: 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h5 {
|
|
||||||
font-size: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h6 {
|
|
||||||
font-size: 11px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body p {
|
|
||||||
margin-top: 0;
|
|
||||||
margin-bottom: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body blockquote {
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body ul,
|
|
||||||
.markdown-body ol {
|
|
||||||
padding-left: 0;
|
|
||||||
margin-top: 0;
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body ol ol,
|
|
||||||
.markdown-body ul ol {
|
|
||||||
list-style-type: lower-roman;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body ul ul ol,
|
|
||||||
.markdown-body ul ol ol,
|
|
||||||
.markdown-body ol ul ol,
|
|
||||||
.markdown-body ol ol ol {
|
|
||||||
list-style-type: lower-alpha;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body dd {
|
|
||||||
margin-left: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body code {
|
|
||||||
font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace;
|
|
||||||
font-size: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body pre {
|
|
||||||
margin-top: 0;
|
|
||||||
margin-bottom: 0;
|
|
||||||
font: 12px Consolas, 'Liberation Mono', Menlo, Courier, monospace;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-0 {
|
|
||||||
padding-left: 0 !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-1 {
|
|
||||||
padding-left: 3px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-2 {
|
|
||||||
padding-left: 6px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-3 {
|
|
||||||
padding-left: 12px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-4 {
|
|
||||||
padding-left: 24px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-5 {
|
|
||||||
padding-left: 36px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-6 {
|
|
||||||
padding-left: 48px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .form-select::-ms-expand {
|
|
||||||
opacity: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body:before {
|
|
||||||
display: table;
|
|
||||||
content: '';
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body:after {
|
|
||||||
display: table;
|
|
||||||
clear: both;
|
|
||||||
content: '';
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > *:first-child {
|
|
||||||
margin-top: 0 !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > *:last-child {
|
|
||||||
margin-bottom: 0 !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body a:not([href]) {
|
|
||||||
color: inherit;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body a.anchor {
|
|
||||||
display: inline-block;
|
|
||||||
padding-right: 2px;
|
|
||||||
margin-left: -18px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body a.anchor:focus {
|
|
||||||
outline: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1,
|
|
||||||
.markdown-body > h2,
|
|
||||||
.markdown-body > h3,
|
|
||||||
.markdown-body > h4,
|
|
||||||
.markdown-body > h5,
|
|
||||||
.markdown-body > h6 {
|
|
||||||
margin-top: 1em;
|
|
||||||
margin-bottom: 16px;
|
|
||||||
font-weight: bold;
|
|
||||||
line-height: 1.4;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1 .octicon-link,
|
|
||||||
.markdown-body > h2 .octicon-link,
|
|
||||||
.markdown-body > h3 .octicon-link,
|
|
||||||
.markdown-body > h4 .octicon-link,
|
|
||||||
.markdown-body > h5 .octicon-link,
|
|
||||||
.markdown-body > h6 .octicon-link {
|
|
||||||
color: #000;
|
|
||||||
vertical-align: middle;
|
|
||||||
visibility: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1:hover .anchor,
|
|
||||||
.markdown-body > h2:hover .anchor,
|
|
||||||
.markdown-body > h3:hover .anchor,
|
|
||||||
.markdown-body > h4:hover .anchor,
|
|
||||||
.markdown-body > h5:hover .anchor,
|
|
||||||
.markdown-body > h6:hover .anchor {
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1:hover .anchor .octicon-link,
|
|
||||||
.markdown-body > h2:hover .anchor .octicon-link,
|
|
||||||
.markdown-body > h3:hover .anchor .octicon-link,
|
|
||||||
.markdown-body > h4:hover .anchor .octicon-link,
|
|
||||||
.markdown-body > h5:hover .anchor .octicon-link,
|
|
||||||
.markdown-body > h6:hover .anchor .octicon-link {
|
|
||||||
visibility: visible;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1 {
|
|
||||||
padding-bottom: 0.3em;
|
|
||||||
font-size: 2.25em;
|
|
||||||
line-height: 1.2;
|
|
||||||
border-bottom: 1px solid #dee5e7;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h1 .anchor {
|
|
||||||
line-height: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h2 {
|
|
||||||
padding-bottom: 0.3em;
|
|
||||||
font-size: 1.75em;
|
|
||||||
line-height: 1.225;
|
|
||||||
border-bottom: 1px solid #dee5e7;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h2 .anchor {
|
|
||||||
line-height: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h3 {
|
|
||||||
font-size: 1.5em;
|
|
||||||
line-height: 1.43;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h3 .anchor {
|
|
||||||
line-height: 1.2;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h4 {
|
|
||||||
font-size: 1.25em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h4 .anchor {
|
|
||||||
line-height: 1.2;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h5 {
|
|
||||||
font-size: 1em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h5 .anchor {
|
|
||||||
line-height: 1.1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > h6 {
|
|
||||||
font-size: 1em;
|
|
||||||
color: #777;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body h6 .anchor {
|
|
||||||
line-height: 1.1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body p,
|
|
||||||
.markdown-body blockquote,
|
|
||||||
.markdown-body ul,
|
|
||||||
.markdown-body ol,
|
|
||||||
.markdown-body dl,
|
|
||||||
.markdown-body table,
|
|
||||||
.markdown-body pre {
|
|
||||||
margin-top: 0;
|
|
||||||
margin-bottom: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body hr {
|
|
||||||
height: 4px;
|
|
||||||
padding: 0;
|
|
||||||
margin: 16px 0;
|
|
||||||
background-color: #e7e7e7;
|
|
||||||
border: 0 none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body ul:not(.dropdown-menu):not(.nav),
|
|
||||||
.markdown-body ol {
|
|
||||||
padding-left: 2em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body ul ul,
|
|
||||||
.markdown-body ul ol,
|
|
||||||
.markdown-body ol ol,
|
|
||||||
.markdown-body ol ul {
|
|
||||||
margin-top: 0;
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body li > p {
|
|
||||||
margin-top: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body dl {
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body dl dt {
|
|
||||||
padding: 0;
|
|
||||||
margin-top: 16px;
|
|
||||||
font-size: 1em;
|
|
||||||
font-style: italic;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body dl dd {
|
|
||||||
padding: 0 16px;
|
|
||||||
margin-bottom: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body blockquote {
|
|
||||||
padding: 0 15px;
|
|
||||||
color: #777;
|
|
||||||
border-left: 4px solid #ddd;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body blockquote > :first-child {
|
|
||||||
margin-top: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body blockquote > :last-child {
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table:not(.table) {
|
|
||||||
display: block;
|
|
||||||
width: 100%;
|
|
||||||
overflow: auto;
|
|
||||||
word-break: normal;
|
|
||||||
/* word-break: keep-all; */
|
|
||||||
border-left: 1px solid #ddd;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table:not(.table) th {
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table:not(.table) th,
|
|
||||||
.markdown-body table:not(.table) td {
|
|
||||||
padding: 6px 13px;
|
|
||||||
border: 1px solid #ddd;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table:not(.table) tr {
|
|
||||||
border-top: 1px solid #ccc;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table:not(.table) tr:nth-child(2n) {
|
|
||||||
background-color: var(--colors-neutral-fill-9);
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table td:first-child,
|
|
||||||
.markdown-body table th:first-child {
|
|
||||||
position: sticky;
|
|
||||||
left: 0;
|
|
||||||
z-index: 10;
|
|
||||||
background-color: var(--colors-neutral-fill-11);
|
|
||||||
border-left: 0px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table td:first-child::after,
|
|
||||||
.markdown-body table th:first-child::after {
|
|
||||||
box-shadow: inset 10px 0 8px -8px rgba(0, 0, 0, 0.1);
|
|
||||||
position: absolute;
|
|
||||||
top: 0;
|
|
||||||
right: 0;
|
|
||||||
bottom: 0px;
|
|
||||||
width: 30px;
|
|
||||||
transform: translate(100%);
|
|
||||||
transition: box-shadow 0.3s;
|
|
||||||
content: '';
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body table tr:nth-child(2n) td:first-child {
|
|
||||||
background-color: var(--colors-neutral-fill-9);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* modified by zhangjun08
|
|
||||||
* 更改文档中的图片展示样式
|
|
||||||
*/
|
|
||||||
.markdown-body img {
|
|
||||||
max-width: 90%;
|
|
||||||
margin-left: 5%;
|
|
||||||
margin-right: 5%;
|
|
||||||
box-sizing: content-box;
|
|
||||||
background-color: #fff;
|
|
||||||
border-radius: 5px;
|
|
||||||
border: 1px solid #ddd;
|
|
||||||
box-shadow: 0 8px 18px 0 rgba(0, 0, 0, 0.3);
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (min-width: 1200px) {
|
|
||||||
.markdown-body img {
|
|
||||||
max-width: 800px;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.markdown-body code {
|
.markdown {
|
||||||
padding: 0;
|
padding: 25px 45px 0;
|
||||||
padding-top: 0.2em;
|
|
||||||
padding-bottom: 0.2em;
|
|
||||||
margin: 0;
|
|
||||||
font-size: 85%;
|
|
||||||
background-color: rgba(0, 0, 0, 0.04);
|
|
||||||
border-radius: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body code:before,
|
|
||||||
.markdown-body code:after {
|
|
||||||
letter-spacing: -0.2em;
|
|
||||||
content: '\00a0';
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body pre > code {
|
|
||||||
padding: 0;
|
|
||||||
margin: 0;
|
|
||||||
font-size: 100%;
|
|
||||||
word-break: normal;
|
|
||||||
white-space: pre;
|
|
||||||
background: transparent;
|
|
||||||
border: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .highlight {
|
|
||||||
margin-bottom: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .highlight pre,
|
|
||||||
.markdown-body pre {
|
|
||||||
padding: 16px;
|
|
||||||
overflow: auto;
|
|
||||||
font-size: 85%;
|
|
||||||
line-height: 1.45;
|
|
||||||
background-color: #f7f7f7;
|
|
||||||
border-radius: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .highlight pre {
|
|
||||||
margin-bottom: 0;
|
|
||||||
word-break: normal;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body pre {
|
|
||||||
word-wrap: normal;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body pre code {
|
|
||||||
display: inline;
|
|
||||||
max-width: initial;
|
|
||||||
padding: 0;
|
|
||||||
margin: 0;
|
|
||||||
overflow: initial;
|
|
||||||
line-height: inherit;
|
|
||||||
word-wrap: normal;
|
|
||||||
background-color: transparent;
|
|
||||||
border: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body pre code:before,
|
|
||||||
.markdown-body pre code:after {
|
|
||||||
content: normal;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body kbd {
|
|
||||||
display: inline-block;
|
|
||||||
padding: 3px 5px;
|
|
||||||
font-size: 11px;
|
|
||||||
line-height: 10px;
|
|
||||||
color: #555;
|
|
||||||
vertical-align: middle;
|
|
||||||
background-color: #fcfcfc;
|
|
||||||
border: solid 1px #ccc;
|
|
||||||
border-bottom-color: #bbb;
|
|
||||||
border-radius: 3px;
|
|
||||||
box-shadow: inset 0 -1px 0 #bbb;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-c {
|
|
||||||
color: #969896;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-c1,
|
|
||||||
.markdown-body .pl-s .pl-v {
|
|
||||||
color: #0086b3;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-e,
|
|
||||||
.markdown-body .pl-en {
|
|
||||||
color: #795da3;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-s .pl-s1,
|
|
||||||
.markdown-body .pl-smi {
|
|
||||||
color: #333;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-ent {
|
|
||||||
color: #63a35c;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-k {
|
|
||||||
color: #a71d5d;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-pds,
|
|
||||||
.markdown-body .pl-s,
|
|
||||||
.markdown-body .pl-s .pl-pse .pl-s1,
|
|
||||||
.markdown-body .pl-sr,
|
|
||||||
.markdown-body .pl-sr .pl-cce,
|
|
||||||
.markdown-body .pl-sr .pl-sra,
|
|
||||||
.markdown-body .pl-sr .pl-sre {
|
|
||||||
color: #183691;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-v {
|
|
||||||
color: #ed6a43;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-id {
|
|
||||||
color: #b52a1d;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-ii {
|
|
||||||
background-color: #b52a1d;
|
|
||||||
color: #f8f8f8;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-sr .pl-cce {
|
|
||||||
color: #63a35c;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-ml {
|
|
||||||
color: #693a17;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-mh,
|
|
||||||
.markdown-body .pl-mh .pl-en,
|
|
||||||
.markdown-body .pl-ms {
|
|
||||||
color: #1d3e81;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-mq {
|
|
||||||
color: #008080;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-mi {
|
|
||||||
color: #333;
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-mb {
|
|
||||||
color: #333;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-md {
|
|
||||||
background-color: #ffecec;
|
|
||||||
color: #bd2c00;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-mi1 {
|
|
||||||
background-color: #eaffea;
|
|
||||||
color: #55a532;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-mdr {
|
|
||||||
color: #795da3;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .pl-mo {
|
|
||||||
color: #1d3e81;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body kbd {
|
|
||||||
display: inline-block;
|
|
||||||
padding: 3px 5px;
|
|
||||||
font: 11px Consolas, 'Liberation Mono', Menlo, Courier, monospace;
|
|
||||||
line-height: 10px;
|
|
||||||
color: #555;
|
|
||||||
vertical-align: middle;
|
|
||||||
background-color: #fcfcfc;
|
|
||||||
border: solid 1px #ccc;
|
|
||||||
border-bottom-color: #bbb;
|
|
||||||
border-radius: 3px;
|
|
||||||
box-shadow: inset 0 -1px 0 #bbb;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .full-commit .btn-outline:not(:disabled):hover {
|
|
||||||
color: #4078c0;
|
|
||||||
border: 1px solid #4078c0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body :checked + .radio-label {
|
|
||||||
position: relative;
|
|
||||||
z-index: 1;
|
|
||||||
border-color: #4078c0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .octicon {
|
|
||||||
display: inline-block;
|
|
||||||
vertical-align: text-top;
|
|
||||||
fill: currentColor;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .task-list-item {
|
|
||||||
list-style-type: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .task-list-item + .task-list-item {
|
|
||||||
margin-top: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .task-list-item input {
|
|
||||||
margin: 0 0.2em 0.25em -1.6em;
|
|
||||||
vertical-align: middle;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body hr {
|
|
||||||
border-bottom-color: #eee;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.doc-play-ground {
|
.doc-play-ground {
|
||||||
@ -765,25 +31,6 @@
|
|||||||
margin-bottom: 0;
|
margin-bottom: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.markdown-body .CodeMirror pre {
|
|
||||||
padding: 0 4px;
|
|
||||||
margin: 0;
|
|
||||||
background-color: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .CodeMirror,
|
|
||||||
.markdown-body .CodeMirror * {
|
|
||||||
box-sizing: content-box;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .CodeMirror-lines {
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body .tree-view ul {
|
|
||||||
padding-left: 0 !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markdown-body > .amis-preview {
|
.markdown-body > .amis-preview {
|
||||||
margin-bottom: 15px;
|
margin-bottom: 15px;
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,10 @@
|
|||||||
/>
|
/>
|
||||||
<link rel="stylesheet" href="prismjs/themes/prism.css" />
|
<link rel="stylesheet" href="prismjs/themes/prism.css" />
|
||||||
<!--DEPENDENCIES_INJECT_PLACEHOLDER-->
|
<!--DEPENDENCIES_INJECT_PLACEHOLDER-->
|
||||||
|
<link
|
||||||
|
rel="stylesheet"
|
||||||
|
href="../node_modules/github-markdown-css/github-markdown-light.css"
|
||||||
|
/>
|
||||||
<link rel="stylesheet" href="./doc.css" />
|
<link rel="stylesheet" href="./doc.css" />
|
||||||
|
|
||||||
<link rel="stylesheet" href="./style.scss" />
|
<link rel="stylesheet" href="./style.scss" />
|
||||||
|
@ -333,7 +333,7 @@ body {
|
|||||||
|
|
||||||
a.anchor {
|
a.anchor {
|
||||||
padding-top: 100px;
|
padding-top: 100px;
|
||||||
margin-top: -100px;
|
margin-top: -94px;
|
||||||
}
|
}
|
||||||
|
|
||||||
&-nav,
|
&-nav,
|
||||||
|
@ -26,6 +26,10 @@
|
|||||||
/>
|
/>
|
||||||
<link rel="stylesheet" href="/node_modules/katex/dist/katex.min.css" />
|
<link rel="stylesheet" href="/node_modules/katex/dist/katex.min.css" />
|
||||||
<link rel="stylesheet" href="/node_modules/prismjs/themes/prism.css" />
|
<link rel="stylesheet" href="/node_modules/prismjs/themes/prism.css" />
|
||||||
|
<link
|
||||||
|
rel="stylesheet"
|
||||||
|
href="/node_modules/github-markdown-css/github-markdown-light.css"
|
||||||
|
/>
|
||||||
<link rel="stylesheet" href="/examples/doc.css" />
|
<link rel="stylesheet" href="/examples/doc.css" />
|
||||||
|
|
||||||
<link rel="stylesheet" href="/examples/style.scss" />
|
<link rel="stylesheet" href="/examples/style.scss" />
|
||||||
|
@ -63,6 +63,7 @@
|
|||||||
"@types/prismjs": "^1.26.0",
|
"@types/prismjs": "^1.26.0",
|
||||||
"@types/react": "^18.0.24",
|
"@types/react": "^18.0.24",
|
||||||
"@types/react-dom": "^18.0.8",
|
"@types/react-dom": "^18.0.8",
|
||||||
|
"@types/react-syntax-highlighter": "^15.5.11",
|
||||||
"@vitejs/plugin-react": "^2.2.0",
|
"@vitejs/plugin-react": "^2.2.0",
|
||||||
"amis-publish": "^1.0.1",
|
"amis-publish": "^1.0.1",
|
||||||
"copy-to-clipboard": "3.3.1",
|
"copy-to-clipboard": "3.3.1",
|
||||||
@ -82,6 +83,7 @@
|
|||||||
"fis3-prepackager-stand-alone-pack": "^1.0.0",
|
"fis3-prepackager-stand-alone-pack": "^1.0.0",
|
||||||
"fis3-preprocessor-js-require-css": "^0.1.3",
|
"fis3-preprocessor-js-require-css": "^0.1.3",
|
||||||
"fis3-preprocessor-js-require-file": "^0.1.3",
|
"fis3-preprocessor-js-require-file": "^0.1.3",
|
||||||
|
"github-markdown-css": "^5.5.1",
|
||||||
"husky": "^8.0.0",
|
"husky": "^8.0.0",
|
||||||
"jest": "^29.0.3",
|
"jest": "^29.0.3",
|
||||||
"jest-environment-jsdom": "^29.0.3",
|
"jest-environment-jsdom": "^29.0.3",
|
||||||
@ -97,7 +99,10 @@
|
|||||||
"prismjs": "^1.29.0",
|
"prismjs": "^1.29.0",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.2.0",
|
||||||
|
"react-markdown": "^9.0.1",
|
||||||
"react-overlays": "5.1.1",
|
"react-overlays": "5.1.1",
|
||||||
|
"react-syntax-highlighter": "^15.5.0",
|
||||||
|
"remark-gfm": "^4.0.0",
|
||||||
"rollup": "^2.79.1",
|
"rollup": "^2.79.1",
|
||||||
"rollup-pluginutils": "^2.8.2",
|
"rollup-pluginutils": "^2.8.2",
|
||||||
"setprototypeof": "^1.2.0",
|
"setprototypeof": "^1.2.0",
|
||||||
|
@ -9,8 +9,11 @@
|
|||||||
.btn-configured {
|
.btn-configured {
|
||||||
position: relative;
|
position: relative;
|
||||||
justify-content: left;
|
justify-content: left;
|
||||||
padding-right: 32px;
|
padding: var(--button-size-sm-paddingTop) 32px
|
||||||
padding-left: 4px;
|
var(--button-size-sm-paddingBottom) 4px;
|
||||||
|
line-height: var(--button-size-sm-lineHeight);
|
||||||
|
height: var(--button-size-sm-height);
|
||||||
|
|
||||||
& > span {
|
& > span {
|
||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
@ -28,7 +31,8 @@
|
|||||||
bottom: 0;
|
bottom: 0;
|
||||||
right: var(--button-size-default-paddingRight);
|
right: var(--button-size-default-paddingRight);
|
||||||
margin: auto 0 !important;
|
margin: auto 0 !important;
|
||||||
font-size: 14px;
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
fill: var(--Form-input-clearBtn-color);
|
fill: var(--Form-input-clearBtn-color);
|
||||||
&:hover {
|
&:hover {
|
||||||
fill: var(--Form-input-clearBtn-color-onHover);
|
fill: var(--Form-input-clearBtn-color-onHover);
|
||||||
|
@ -2,6 +2,7 @@ import * as React from 'react';
|
|||||||
import {AlertComponent, ToastComponent, ContextMenu} from 'amis';
|
import {AlertComponent, ToastComponent, ContextMenu} from 'amis';
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
import AMisSchemaEditor from './Editor';
|
import AMisSchemaEditor from './Editor';
|
||||||
|
import {Link} from 'react-router-dom';
|
||||||
export default class App extends React.PureComponent {
|
export default class App extends React.PureComponent {
|
||||||
render() {
|
render() {
|
||||||
// 备注: 如果需要改用antd主题,还需要将index.html换成index-antd.html
|
// 备注: 如果需要改用antd主题,还需要将index.html换成index-antd.html
|
||||||
@ -9,7 +10,10 @@ export default class App extends React.PureComponent {
|
|||||||
return (
|
return (
|
||||||
<div className="Editor-Demo">
|
<div className="Editor-Demo">
|
||||||
<div id="headerBar" className="Editor-header">
|
<div id="headerBar" className="Editor-header">
|
||||||
<div className="Editor-title">amis 可视化编辑器</div>
|
<div className="Editor-title">
|
||||||
|
amis 可视化编辑器
|
||||||
|
<Link to="/basic">面板模版</Link>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<AMisSchemaEditor theme={curTheme} />
|
<AMisSchemaEditor theme={curTheme} />
|
||||||
<ToastComponent theme={curTheme} />
|
<ToastComponent theme={curTheme} />
|
||||||
|
83
packages/amis-editor/examples/component/Doc.tsx
Normal file
83
packages/amis-editor/examples/component/Doc.tsx
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import Markdown from 'react-markdown';
|
||||||
|
import remarkGfm from 'remark-gfm';
|
||||||
|
import {Prism as SyntaxHighlighter} from 'react-syntax-highlighter';
|
||||||
|
import {darcula} from 'react-syntax-highlighter/dist/esm/styles/prism';
|
||||||
|
import {getSchemaTpl, tipedLabel} from 'amis-editor-core';
|
||||||
|
import 'amis';
|
||||||
|
import '../../src/index';
|
||||||
|
import PanelPreview from './PanelPreview';
|
||||||
|
export interface DocProps {
|
||||||
|
children?: string | null | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function mdComment(fun: Function) {
|
||||||
|
const txt = fun.toString();
|
||||||
|
debugger;
|
||||||
|
return txt;
|
||||||
|
}
|
||||||
|
|
||||||
|
function str2schema(code: string) {
|
||||||
|
const fn = new Function('getSchemaTpl', 'tipedLabel', `return [${code}]`);
|
||||||
|
return fn.call(null, getSchemaTpl, tipedLabel);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Doc({children}: DocProps) {
|
||||||
|
return (
|
||||||
|
<div className="markdown">
|
||||||
|
<div className="markdown-body">
|
||||||
|
<Markdown
|
||||||
|
children={children}
|
||||||
|
remarkPlugins={[remarkGfm]}
|
||||||
|
components={{
|
||||||
|
pre(props) {
|
||||||
|
if ((props.children as any)?.props?.className) {
|
||||||
|
return <>{props.children}</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return <code>{props.children}</code>;
|
||||||
|
},
|
||||||
|
code(props) {
|
||||||
|
const {children, className, node, ref, ...rest} = props;
|
||||||
|
const match = /language-(\w+)/.exec(className || '');
|
||||||
|
|
||||||
|
if (match?.[1] === 'schema') {
|
||||||
|
const schema = str2schema(children as string);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="schema-tpl-preview">
|
||||||
|
<code>
|
||||||
|
<SyntaxHighlighter
|
||||||
|
{...rest}
|
||||||
|
PreTag="div"
|
||||||
|
children={String(children).replace(/\n$/, '')}
|
||||||
|
language={'jsx'}
|
||||||
|
style={darcula}
|
||||||
|
/>
|
||||||
|
</code>
|
||||||
|
|
||||||
|
<PanelPreview schema={schema} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return match ? (
|
||||||
|
<SyntaxHighlighter
|
||||||
|
{...rest}
|
||||||
|
PreTag="div"
|
||||||
|
children={String(children).replace(/\n$/, '')}
|
||||||
|
language={match[1]}
|
||||||
|
style={darcula}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<code {...rest} className={className}>
|
||||||
|
{children}
|
||||||
|
</code>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
60
packages/amis-editor/examples/component/PanelPreview.tsx
Normal file
60
packages/amis-editor/examples/component/PanelPreview.tsx
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
import {JsonView, render} from 'amis';
|
||||||
|
import cx from 'classnames';
|
||||||
|
import React from 'react';
|
||||||
|
export interface PanelPreviewProps {
|
||||||
|
schema: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function (props: PanelPreviewProps) {
|
||||||
|
const schema = React.useMemo(() => {
|
||||||
|
return {
|
||||||
|
type: 'form',
|
||||||
|
mode: 'normal',
|
||||||
|
wrapWithPanel: false,
|
||||||
|
className: cx('config-form-content', 'ae-Settings-content'),
|
||||||
|
wrapperComponent: 'div',
|
||||||
|
body: Array.isArray(props.schema) ? props.schema : [props.schema],
|
||||||
|
submitOnChange: true,
|
||||||
|
submitOnInit: true
|
||||||
|
};
|
||||||
|
}, [JSON.stringify(props.schema)]);
|
||||||
|
const [data, setData] = React.useState({});
|
||||||
|
const onFinished = React.useCallback((data: any) => {
|
||||||
|
setData(data);
|
||||||
|
return false;
|
||||||
|
}, []);
|
||||||
|
const onJsonEdit = React.useCallback((e: any) => {
|
||||||
|
setData(e.updated_src);
|
||||||
|
}, []);
|
||||||
|
const dom = React.useRef<HTMLDivElement | null>(null);
|
||||||
|
// const popOverContainer = React.useCallback(() => {
|
||||||
|
// return dom.current;
|
||||||
|
// }, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="PanelPreview" ref={dom}>
|
||||||
|
<div className="AMISCSSWrapper editor-right-panel">
|
||||||
|
{render(
|
||||||
|
schema,
|
||||||
|
{
|
||||||
|
data: data,
|
||||||
|
onFinished: onFinished
|
||||||
|
// popOverContainer
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// theme: 'cxd' // 右侧属性配置面板固定使用cxd主题展示
|
||||||
|
}
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<JsonView
|
||||||
|
name={false}
|
||||||
|
src={data}
|
||||||
|
theme={'rjv-default'}
|
||||||
|
enableClipboard={false}
|
||||||
|
onEdit={onJsonEdit}
|
||||||
|
onDelete={onJsonEdit}
|
||||||
|
onAdd={onJsonEdit}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
21
packages/amis-editor/examples/route/Basic.md
Normal file
21
packages/amis-editor/examples/route/Basic.md
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
# 基本模版
|
||||||
|
|
||||||
|
## 表单项名字
|
||||||
|
|
||||||
|
`formItemName`
|
||||||
|
|
||||||
|
```schema
|
||||||
|
getSchemaTpl('formItemName', {
|
||||||
|
value: 'name'
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
## 表单项展示模式
|
||||||
|
|
||||||
|
`formItemMode`
|
||||||
|
|
||||||
|
```schema
|
||||||
|
getSchemaTpl('formItemMode', {
|
||||||
|
|
||||||
|
})
|
||||||
|
```
|
117
packages/amis-editor/examples/route/Formula.md
Normal file
117
packages/amis-editor/examples/route/Formula.md
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
# 公式相关
|
||||||
|
|
||||||
|
公式输入相关组件规范说明。
|
||||||
|
|
||||||
|
## 显隐表达式
|
||||||
|
|
||||||
|
比如:`visibleOn`, `disabledOn` 因为存在两种表达式的写法,所以这里需要明确区分。
|
||||||
|
|
||||||
|
- 没有初始值时出现「点击编写表达式」按钮,点击会弹出一个表达式编辑器。
|
||||||
|
- 初始值存在时,且为 js 表达式(即不是 `${` 和 `}`包裹的)时,直接用文本输入框展示与输入。
|
||||||
|
- 初始值存在时,且为 `${` 和 `}`包裹的公式表达式时,用公式编辑器展示,同时点击后,会弹出一个表达式编辑器,能直接回显当前公式
|
||||||
|
- 公式弹窗内部始终是公式语法,不存在模版拼接语法。
|
||||||
|
- 当 js 表达式被删除后,只能用新版公式编辑器了。
|
||||||
|
|
||||||
|
```schema
|
||||||
|
getSchemaTpl('expressionFormulaControl', {
|
||||||
|
label: '无默认值',
|
||||||
|
name: 'var1'
|
||||||
|
}),
|
||||||
|
|
||||||
|
getSchemaTpl('expressionFormulaControl', {
|
||||||
|
label: '默认值为旧语法',
|
||||||
|
name: 'var2',
|
||||||
|
value: 'data.a == 1'
|
||||||
|
}),
|
||||||
|
|
||||||
|
getSchemaTpl('expressionFormulaControl', {
|
||||||
|
label: '默认值为新语法',
|
||||||
|
name: 'var3',
|
||||||
|
value: '\\${a == 1 ? "1" : a == 2 ? "二" : a == 3 ? "三" : "一个很长的表达式"}'
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
## 文本输入框默认值
|
||||||
|
|
||||||
|
默认展示为文本输入框,通过点击 `+fx` 来添加公式片段,公式部分高亮展示,是整体高亮,而不是表达式内部 token 高亮(变量名、操作符、字面量等会用不同的方式高亮)。整体高亮内部不细化高亮,展示 tooltip 时再细化高亮,并且点击后开始编辑点击部分公式,注意这里是编辑局部公式,而不是整个默认值输入框。
|
||||||
|
|
||||||
|
值格式中如果没有出现 `${` 和 `}` 包裹的公式,则认为是普通文本,如果存在,则会做公式处理。当时用户直接通过文本输入框输入公式时,会当成时输入普通文本,如果用户输入了 `$` 符号,要将其转成 `\\$`,这样就不会被误认为是公式了。要添加公式片段,用户必须通过点击 `+fx` 按钮添加。`+fx` 可以添加多个片段。
|
||||||
|
|
||||||
|
```schema
|
||||||
|
getSchemaTpl('tplFormulaControl', {
|
||||||
|
name: 'value',
|
||||||
|
label: '默认值'
|
||||||
|
}),
|
||||||
|
|
||||||
|
getSchemaTpl('tplFormulaControl', {
|
||||||
|
name: 'value2',
|
||||||
|
label: '默认值',
|
||||||
|
value: 'My name is \\${name}'
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
## 多行文本输入框默认值
|
||||||
|
|
||||||
|
```schema
|
||||||
|
getSchemaTpl('textareaDefaultValue', {
|
||||||
|
name: 'var1',
|
||||||
|
value: 'My name is \\${name}'
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
## 数字输入框默认值
|
||||||
|
|
||||||
|
默认展示为数字输入框,通过点击 `fx` 输入公式,因为不是文本,无法拼接,所以默认值要么是公式,要么是静态值。
|
||||||
|
|
||||||
|
```schema
|
||||||
|
getSchemaTpl('valueFormula', {
|
||||||
|
mode: 'vertical',
|
||||||
|
rendererSchema: (schema) => ({
|
||||||
|
type: 'input-number',
|
||||||
|
...schema,
|
||||||
|
displayMode: 'base'
|
||||||
|
}),
|
||||||
|
valueType: 'number' // 期望数值类型
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
## 日期输入框默认值
|
||||||
|
|
||||||
|
```schema
|
||||||
|
getSchemaTpl('valueFormula', {
|
||||||
|
mode: 'vertical',
|
||||||
|
rendererSchema: (schema) => ({
|
||||||
|
type: 'input-date',
|
||||||
|
...schema
|
||||||
|
}),
|
||||||
|
placeholder: '请选择静态值',
|
||||||
|
header: '表达式或相对值',
|
||||||
|
DateTimeType: 1,
|
||||||
|
label: tipedLabel('默认值', '支持例如: <code>now、+3days、-2weeks、+1hour、+2years</code> 等(minute|min|hour|day|week|month|year|weekday|second|millisecond)这种相对值用法')
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
## 选项类输入框默认值
|
||||||
|
|
||||||
|
```schema
|
||||||
|
getSchemaTpl('valueFormula', {
|
||||||
|
rendererSchema: (schema) => ({
|
||||||
|
...schema,
|
||||||
|
type: 'select',
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
label: '选项1',
|
||||||
|
value: '1'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '选项2',
|
||||||
|
value: '2'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
// 默认值组件设计有些问题,自动发起了请求,接口数据作为了默认值选项,接口形式应该是设置静态值或者FX
|
||||||
|
needDeleteProps: ['source'],
|
||||||
|
// 当数据源是自定义静态选项时,不额外配置默认值,在选项上直接勾选即可,放开会有个bug:当去掉勾选时,默认值配置组件不清空,只是schema清空了value
|
||||||
|
visibleOn: 'this.selectFirst !== true && this.source != null'
|
||||||
|
})
|
||||||
|
```
|
@ -286,3 +286,31 @@ $disabled-bg-color: #f7f7f9; // 禁用背景颜色
|
|||||||
#root > .a-Select-popover {
|
#root > .a-Select-popover {
|
||||||
z-index: 1500;
|
z-index: 1500;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.schema-tpl-preview {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
align-items: flex-start;
|
||||||
|
|
||||||
|
> *:first-child {
|
||||||
|
flex: 1;
|
||||||
|
min-width: 0;
|
||||||
|
background-color: rgb(43, 43, 43);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.PanelPreview {
|
||||||
|
background-color: #fff;
|
||||||
|
padding: 0 20px 20px 20px;
|
||||||
|
position: relative;
|
||||||
|
width: 400px;
|
||||||
|
flex-shrink: 0;
|
||||||
|
|
||||||
|
> .editor-right-panel {
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
> .react-json-view {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -27,6 +27,7 @@
|
|||||||
href="../../../node_modules/@fortawesome/fontawesome-free/css/v4-shims.css"
|
href="../../../node_modules/@fortawesome/fontawesome-free/css/v4-shims.css"
|
||||||
/>
|
/>
|
||||||
<link rel="stylesheet" title="cxd" href="../amis-ui/scss/themes/cxd.scss" />
|
<link rel="stylesheet" title="cxd" href="../amis-ui/scss/themes/cxd.scss" />
|
||||||
|
<link rel="stylesheet" title="cxd" href="../amis-ui/scss/helper.scss" />
|
||||||
<link rel="stylesheet" href="../amis-editor-core/scss/editor.scss" />
|
<link rel="stylesheet" href="../amis-editor-core/scss/editor.scss" />
|
||||||
<link rel="stylesheet" href="../../../examples/doc.css" />
|
<link rel="stylesheet" href="../../../examples/doc.css" />
|
||||||
<link rel="stylesheet" href="./examples/style.scss" />
|
<link rel="stylesheet" href="./examples/style.scss" />
|
||||||
|
@ -2,11 +2,207 @@
|
|||||||
* @file entry of this example.
|
* @file entry of this example.
|
||||||
* @author fex
|
* @author fex
|
||||||
*/
|
*/
|
||||||
import * as React from 'react';
|
import React from 'react';
|
||||||
import {createRoot} from 'react-dom/client';
|
import {createRoot} from 'react-dom/client';
|
||||||
import App from './examples/App';
|
|
||||||
|
import {Layout, AsideNav, Spinner, NotFound} from 'amis-ui';
|
||||||
|
import {eachTree, TreeArray, TreeItem} from 'amis-core';
|
||||||
|
import {
|
||||||
|
HashRouter as Router,
|
||||||
|
Route,
|
||||||
|
Redirect,
|
||||||
|
Switch,
|
||||||
|
Link,
|
||||||
|
NavLink
|
||||||
|
} from 'react-router-dom';
|
||||||
|
import Doc from './examples/component/Doc';
|
||||||
|
|
||||||
|
function MDComponent(fN: () => Promise<{default: {raw: string}}>) {
|
||||||
|
return React.lazy(() =>
|
||||||
|
fN().then(ret => ({default: () => <Doc children={ret.default.raw} />}))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const pages: TreeArray = [
|
||||||
|
{
|
||||||
|
label: '面板模版',
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
label: '基础',
|
||||||
|
path: '/basic',
|
||||||
|
component: MDComponent(() => import('./examples/route/Basic.md'))
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
label: '公式',
|
||||||
|
path: '/formula',
|
||||||
|
component: MDComponent(() => import('./examples/route/Formula.md'))
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
function getPath(path: string) {
|
||||||
|
return path ? (path[0] === '/' ? path : `/${path}`) : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function isActive(link: any, location: any) {
|
||||||
|
return !!(link.path && getPath(link.path) === location.pathname);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function navigations2route(
|
||||||
|
navigations: any,
|
||||||
|
additionalProperties?: any
|
||||||
|
) {
|
||||||
|
let routes: any = [];
|
||||||
|
|
||||||
|
navigations.forEach((root: any) => {
|
||||||
|
root.children &&
|
||||||
|
eachTree(root.children, (item: any) => {
|
||||||
|
if (item.path && item.component) {
|
||||||
|
routes.push(
|
||||||
|
additionalProperties ? (
|
||||||
|
<Route
|
||||||
|
key={routes.length + 1}
|
||||||
|
path={item.path[0] === '/' ? item.path : `/${item.path}`}
|
||||||
|
render={(props: any) => (
|
||||||
|
<item.component {...additionalProperties} {...props} />
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Route
|
||||||
|
key={routes.length + 1}
|
||||||
|
path={item.path[0] === '/' ? item.path : `/${item.path}`}
|
||||||
|
component={item.component}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return routes;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Main() {
|
||||||
|
function renderAside() {
|
||||||
|
return (
|
||||||
|
<AsideNav
|
||||||
|
navigations={pages.map((item: any) => ({
|
||||||
|
...item,
|
||||||
|
children: item.children
|
||||||
|
? item.children.map((item: any) => ({
|
||||||
|
...item,
|
||||||
|
className: 'is-top'
|
||||||
|
}))
|
||||||
|
: []
|
||||||
|
}))}
|
||||||
|
renderLink={({
|
||||||
|
link,
|
||||||
|
active,
|
||||||
|
toggleExpand,
|
||||||
|
classnames: cx,
|
||||||
|
depth
|
||||||
|
}: any) => {
|
||||||
|
let children: any[] = [];
|
||||||
|
|
||||||
|
if (link.children && link.children.length) {
|
||||||
|
children.push(
|
||||||
|
<span
|
||||||
|
key="expand-toggle"
|
||||||
|
className={cx('AsideNav-itemArrow')}
|
||||||
|
onClick={e => toggleExpand(link, e)}
|
||||||
|
></span>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
link.badge &&
|
||||||
|
children.push(
|
||||||
|
<b
|
||||||
|
key="badge"
|
||||||
|
className={cx(
|
||||||
|
`AsideNav-itemBadge`,
|
||||||
|
link.badgeClassName || 'bg-info'
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{link.badge}
|
||||||
|
</b>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (link.icon) {
|
||||||
|
children.push(
|
||||||
|
<i key="icon" className={cx(`AsideNav-itemIcon`, link.icon)} />
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
children.push(
|
||||||
|
<span className={cx('AsideNav-itemLabel')} key="label">
|
||||||
|
{link.label}
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
|
||||||
|
return link.path ? (
|
||||||
|
/^https?\:/.test(link.path) ? (
|
||||||
|
<a target="_blank" href={link.path} rel="noopener">
|
||||||
|
{children}
|
||||||
|
</a>
|
||||||
|
) : (
|
||||||
|
<Link
|
||||||
|
to={
|
||||||
|
getPath(link.path) ||
|
||||||
|
(link.children && getPath(link.children[0].path))
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Link>
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
<a onClick={link.children ? () => toggleExpand(link) : undefined}>
|
||||||
|
{children}
|
||||||
|
</a>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
isActive={(link: any) => isActive(link, location)}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Router>
|
||||||
|
<React.Suspense
|
||||||
|
fallback={<Spinner overlay spinnerClassName="m-t-lg" size="lg" />}
|
||||||
|
>
|
||||||
|
<Switch>
|
||||||
|
<Route
|
||||||
|
component={React.lazy(() => import('./examples/App'))}
|
||||||
|
exact
|
||||||
|
path="/"
|
||||||
|
/>
|
||||||
|
<Layout
|
||||||
|
header={
|
||||||
|
<div id="headerBar" className="box-shadow bg-dark">
|
||||||
|
<div className={`cxd-Layout-brand`}>编辑器面板示例</div>
|
||||||
|
<Link to="/">回到编辑器</Link>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
aside={renderAside()}
|
||||||
|
>
|
||||||
|
<React.Suspense
|
||||||
|
fallback={<Spinner overlay spinnerClassName="m-t-lg" size="lg" />}
|
||||||
|
>
|
||||||
|
<Switch>
|
||||||
|
{navigations2route(pages)}
|
||||||
|
<Route render={() => <NotFound description="Not found" />} />
|
||||||
|
</Switch>
|
||||||
|
</React.Suspense>
|
||||||
|
</Layout>
|
||||||
|
</Switch>
|
||||||
|
</React.Suspense>
|
||||||
|
</Router>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
export function bootstrap(mountTo: HTMLElement, initalState: any) {
|
export function bootstrap(mountTo: HTMLElement, initalState: any) {
|
||||||
const root = createRoot(mountTo);
|
const root = createRoot(mountTo);
|
||||||
root.render(<App />);
|
root.render(<Main />);
|
||||||
}
|
}
|
||||||
|
@ -3,10 +3,10 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import {autobind, FormControlProps} from 'amis-core';
|
import {autobind, FormControlProps, isExpression} from 'amis-core';
|
||||||
import cx from 'classnames';
|
import cx from 'classnames';
|
||||||
import {FormItem, Button, Icon, PickerContainer} from 'amis';
|
import {FormItem, Button, Icon, PickerContainer} from 'amis';
|
||||||
import {FormulaCodeEditor, FormulaEditor} from 'amis-ui';
|
import {FormulaCodeEditor, FormulaEditor, InputBox} from 'amis-ui';
|
||||||
import type {VariableItem} from 'amis-ui';
|
import type {VariableItem} from 'amis-ui';
|
||||||
import {reaction} from 'mobx';
|
import {reaction} from 'mobx';
|
||||||
import {getVariables} from 'amis-editor-core';
|
import {getVariables} from 'amis-editor-core';
|
||||||
@ -139,6 +139,7 @@ export default class ExpressionFormulaControl extends React.Component<
|
|||||||
render() {
|
render() {
|
||||||
const {value, className, variableMode, header, size, ...rest} = this.props;
|
const {value, className, variableMode, header, size, ...rest} = this.props;
|
||||||
const {formulaPickerValue, variables} = this.state;
|
const {formulaPickerValue, variables} = this.state;
|
||||||
|
const isNewExpression = isExpression(value);
|
||||||
|
|
||||||
// 自身字段
|
// 自身字段
|
||||||
const selfName = this.props?.data?.name;
|
const selfName = this.props?.data?.name;
|
||||||
@ -171,7 +172,9 @@ export default class ExpressionFormulaControl extends React.Component<
|
|||||||
size={size ?? 'lg'}
|
size={size ?? 'lg'}
|
||||||
>
|
>
|
||||||
{({onClick}: {onClick: (e: React.MouseEvent) => any}) =>
|
{({onClick}: {onClick: (e: React.MouseEvent) => any}) =>
|
||||||
formulaPickerValue ? (
|
value && !isNewExpression ? (
|
||||||
|
<InputBox value={value} onChange={rest.onChange} />
|
||||||
|
) : formulaPickerValue ? (
|
||||||
<Button
|
<Button
|
||||||
className="btn-configured"
|
className="btn-configured"
|
||||||
tooltip={{
|
tooltip={{
|
||||||
@ -202,6 +205,7 @@ export default class ExpressionFormulaControl extends React.Component<
|
|||||||
<FormulaCodeEditor
|
<FormulaCodeEditor
|
||||||
singleLine
|
singleLine
|
||||||
readOnly
|
readOnly
|
||||||
|
highlightMode="expression"
|
||||||
value={value}
|
value={value}
|
||||||
variables={variables}
|
variables={variables}
|
||||||
evalMode={false}
|
evalMode={false}
|
||||||
@ -215,6 +219,7 @@ export default class ExpressionFormulaControl extends React.Component<
|
|||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
<Button
|
<Button
|
||||||
|
size="sm"
|
||||||
className="btn-set-expression"
|
className="btn-set-expression"
|
||||||
onClick={e => this.handleOnClick(e, onClick)}
|
onClick={e => this.handleOnClick(e, onClick)}
|
||||||
>
|
>
|
||||||
|
@ -657,9 +657,9 @@ export default class FormulaControl extends React.Component<
|
|||||||
children: () => (
|
children: () => (
|
||||||
<FormulaCodeEditor
|
<FormulaCodeEditor
|
||||||
readOnly
|
readOnly
|
||||||
value={value}
|
value={exprValue}
|
||||||
variables={variables}
|
variables={variables}
|
||||||
evalMode={false}
|
evalMode={true}
|
||||||
editorTheme="dark"
|
editorTheme="dark"
|
||||||
editorOptions={{
|
editorOptions={{
|
||||||
lineNumbers: false
|
lineNumbers: false
|
||||||
@ -682,6 +682,7 @@ export default class FormulaControl extends React.Component<
|
|||||||
functions={[]}
|
functions={[]}
|
||||||
variables={variables}
|
variables={variables}
|
||||||
evalMode={false}
|
evalMode={false}
|
||||||
|
highlightMode="expression"
|
||||||
readOnly
|
readOnly
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
@ -10,6 +10,7 @@ import type {VariableItem, CodeMirror} from 'amis-ui';
|
|||||||
import {Icon, Button, FormItem, TooltipWrapper} from 'amis';
|
import {Icon, Button, FormItem, TooltipWrapper} from 'amis';
|
||||||
import {autobind, FormControlProps} from 'amis-core';
|
import {autobind, FormControlProps} from 'amis-core';
|
||||||
import {FormulaPlugin, editorFactory} from './textarea-formula/plugin';
|
import {FormulaPlugin, editorFactory} from './textarea-formula/plugin';
|
||||||
|
import {renderFormulaValue} from './FormulaControl';
|
||||||
import FormulaPicker, {
|
import FormulaPicker, {
|
||||||
CustomFormulaPickerProps
|
CustomFormulaPickerProps
|
||||||
} from './textarea-formula/FormulaPicker';
|
} from './textarea-formula/FormulaPicker';
|
||||||
@ -387,6 +388,12 @@ export class TplFormulaControl extends React.Component<
|
|||||||
} = this.state;
|
} = this.state;
|
||||||
|
|
||||||
const FormulaPickerCmp = customFormulaPicker ?? FormulaPicker;
|
const FormulaPickerCmp = customFormulaPicker ?? FormulaPicker;
|
||||||
|
const highlightValue = FormulaEditor.highlightValue(
|
||||||
|
formulaPickerValue,
|
||||||
|
variables
|
||||||
|
) || {
|
||||||
|
html: formulaPickerValue
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@ -439,23 +446,11 @@ export class TplFormulaControl extends React.Component<
|
|||||||
|
|
||||||
<TooltipWrapper
|
<TooltipWrapper
|
||||||
trigger="hover"
|
trigger="hover"
|
||||||
placement="left"
|
placement="top"
|
||||||
style={{fontSize: '12px'}}
|
style={{fontSize: '12px'}}
|
||||||
tooltip={{
|
tooltip={{
|
||||||
tooltipTheme: 'dark',
|
tooltipTheme: 'dark',
|
||||||
tooltipClassName: 'btn-configured-tooltip',
|
children: () => renderFormulaValue(highlightValue)
|
||||||
children: () => (
|
|
||||||
<FormulaCodeEditor
|
|
||||||
readOnly
|
|
||||||
value={formulaPickerValue}
|
|
||||||
variables={variables}
|
|
||||||
evalMode={true}
|
|
||||||
editorTheme="dark"
|
|
||||||
editorOptions={{
|
|
||||||
lineNumbers: false
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
)
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
|
@ -1325,11 +1325,12 @@ setSchemaTpl('pageSubTitle', {
|
|||||||
type: 'textarea'
|
type: 'textarea'
|
||||||
});
|
});
|
||||||
|
|
||||||
setSchemaTpl('textareaDefaultValue', () => {
|
setSchemaTpl('textareaDefaultValue', (options: any) => {
|
||||||
return getSchemaTpl('textareaFormulaControl', {
|
return getSchemaTpl('textareaFormulaControl', {
|
||||||
label: '默认值',
|
label: '默认值',
|
||||||
name: 'value',
|
name: 'value',
|
||||||
mode: 'normal'
|
mode: 'normal',
|
||||||
|
...options
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -633,13 +633,19 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.cm-field {
|
.cm-field {
|
||||||
background: #007bff;
|
background: #28a745;
|
||||||
}
|
}
|
||||||
.cm-func {
|
.cm-func {
|
||||||
color: #ae4597;
|
color: #ae4597;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
line-height: 14px;
|
line-height: 14px;
|
||||||
}
|
}
|
||||||
|
.cm-expression {
|
||||||
|
background-color: #007bff;
|
||||||
|
border-radius: 4px;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
.cm-error-token {
|
.cm-error-token {
|
||||||
background-position: left bottom;
|
background-position: left bottom;
|
||||||
background-repeat: repeat-x;
|
background-repeat: repeat-x;
|
||||||
@ -656,8 +662,9 @@
|
|||||||
|
|
||||||
&--singleLine {
|
&--singleLine {
|
||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
|
line-height: 20px;
|
||||||
> .CodeMirror {
|
> .CodeMirror {
|
||||||
height: 21px;
|
height: 20px;
|
||||||
|
|
||||||
.CodeMirror-hscrollbar,
|
.CodeMirror-hscrollbar,
|
||||||
.CodeMirror-vscrollbar {
|
.CodeMirror-vscrollbar {
|
||||||
@ -665,15 +672,20 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.CodeMirror-sizer {
|
.CodeMirror-sizer {
|
||||||
min-height: 21px !important;
|
min-height: 20px !important;
|
||||||
|
min-width: auto !important;
|
||||||
border-right-width: 0 !important;
|
border-right-width: 0 !important;
|
||||||
}
|
}
|
||||||
.CodeMirror-scroll {
|
.CodeMirror-scroll {
|
||||||
height: 21px;
|
height: 20px;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
overflow: hidden !important;
|
overflow: hidden !important;
|
||||||
}
|
}
|
||||||
|
.CodeMirror-sizer + div {
|
||||||
|
// 不设置可以用触摸板滚动,将代码滚不见了
|
||||||
|
height: 0 !important;
|
||||||
|
}
|
||||||
.CodeMirror-lines {
|
.CodeMirror-lines {
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
@ -51,6 +51,12 @@ export interface CodeEditorProps
|
|||||||
|
|
||||||
editorOptions?: any;
|
editorOptions?: any;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* expression 即高亮表达式整体
|
||||||
|
* formula 即高亮表达式内部
|
||||||
|
*/
|
||||||
|
highlightMode?: 'expression' | 'formula';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 用于提示的变量集合,默认为空
|
* 用于提示的变量集合,默认为空
|
||||||
*/
|
*/
|
||||||
@ -88,7 +94,8 @@ function CodeEditor(props: CodeEditorProps, ref: any) {
|
|||||||
editorTheme,
|
editorTheme,
|
||||||
theme: defaultTheme,
|
theme: defaultTheme,
|
||||||
editorOptions,
|
editorOptions,
|
||||||
placeholder
|
placeholder,
|
||||||
|
highlightMode
|
||||||
} = props;
|
} = props;
|
||||||
const pluginRef = React.useRef<FormulaPlugin>();
|
const pluginRef = React.useRef<FormulaPlugin>();
|
||||||
|
|
||||||
@ -165,6 +172,7 @@ function CodeEditor(props: CodeEditorProps, ref: any) {
|
|||||||
plugin.setEvalMode(!!evalMode);
|
plugin.setEvalMode(!!evalMode);
|
||||||
plugin.setFunctions(functions || []);
|
plugin.setFunctions(functions || []);
|
||||||
plugin.setVariables(variables || []);
|
plugin.setVariables(variables || []);
|
||||||
|
plugin.setHighlightMode(highlightMode || 'formula');
|
||||||
editorDidMount?.(cm, editor, plugin);
|
editorDidMount?.(cm, editor, plugin);
|
||||||
plugin.autoMarkText();
|
plugin.autoMarkText();
|
||||||
|
|
||||||
|
@ -151,6 +151,9 @@ export interface FormulaPickerProps
|
|||||||
onClick: (e: React.MouseEvent) => void;
|
onClick: (e: React.MouseEvent) => void;
|
||||||
setState: (state: any) => void;
|
setState: (state: any) => void;
|
||||||
isOpened: boolean;
|
isOpened: boolean;
|
||||||
|
value: any;
|
||||||
|
onChange: (value: any) => void;
|
||||||
|
disabled?: boolean;
|
||||||
}) => JSX.Element;
|
}) => JSX.Element;
|
||||||
|
|
||||||
onConfirm?: (value?: any) => void;
|
onConfirm?: (value?: any) => void;
|
||||||
@ -169,6 +172,7 @@ export interface FormulaPickerState {
|
|||||||
isOpened: boolean;
|
isOpened: boolean;
|
||||||
value: any;
|
value: any;
|
||||||
editorValue: string;
|
editorValue: string;
|
||||||
|
onConfirm?: (value?: any) => void;
|
||||||
isError: boolean | string;
|
isError: boolean | string;
|
||||||
variables?: Array<VariableItem>;
|
variables?: Array<VariableItem>;
|
||||||
functions?: Array<FuncGroup>;
|
functions?: Array<FuncGroup>;
|
||||||
@ -217,7 +221,7 @@ export class FormulaPicker extends React.Component<
|
|||||||
async componentDidUpdate(prevProps: FormulaPickerProps) {
|
async componentDidUpdate(prevProps: FormulaPickerProps) {
|
||||||
const {value} = this.props;
|
const {value} = this.props;
|
||||||
|
|
||||||
if (value !== prevProps.value) {
|
if (value !== prevProps.value && !this.state.isOpened) {
|
||||||
this.setState({
|
this.setState({
|
||||||
value: typeof value === 'string' || !this.isTextInput() ? value : '',
|
value: typeof value === 'string' || !this.isTextInput() ? value : '',
|
||||||
editorValue: this.value2EditorValue(this.props)
|
editorValue: this.value2EditorValue(this.props)
|
||||||
@ -373,6 +377,7 @@ export class FormulaPicker extends React.Component<
|
|||||||
|
|
||||||
confirm(value: any, ast?: any) {
|
confirm(value: any, ast?: any) {
|
||||||
const {mixedMode} = this.props;
|
const {mixedMode} = this.props;
|
||||||
|
const stateOnConfirm = this.state.onConfirm;
|
||||||
const validate = this.validate(value);
|
const validate = this.validate(value);
|
||||||
|
|
||||||
if (validate === true) {
|
if (validate === true) {
|
||||||
@ -389,8 +394,12 @@ export class FormulaPicker extends React.Component<
|
|||||||
: `\${${value}}`;
|
: `\${${value}}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.setState({value: result}, () => {
|
this.setState({value: result, onConfirm: undefined}, () => {
|
||||||
this.close(undefined, () => this.handleConfirm());
|
this.close(undefined, () => {
|
||||||
|
stateOnConfirm
|
||||||
|
? stateOnConfirm(this.state.value)
|
||||||
|
: this.handleConfirm();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
this.setState({isError: validate});
|
this.setState({isError: validate});
|
||||||
@ -399,12 +408,16 @@ export class FormulaPicker extends React.Component<
|
|||||||
|
|
||||||
@autobind
|
@autobind
|
||||||
async handleClick() {
|
async handleClick() {
|
||||||
const {variables, data} = this.props;
|
return this.openEditor(this.value2EditorValue(this.props));
|
||||||
|
}
|
||||||
|
|
||||||
|
@autobind
|
||||||
|
async openEditor(editorValue: string, onConfirm?: (value: any) => void) {
|
||||||
const state = {
|
const state = {
|
||||||
...(await this.props.onPickerOpen?.(this.props)),
|
...(await this.props.onPickerOpen?.(this.props)),
|
||||||
editorValue: this.value2EditorValue(this.props),
|
editorValue,
|
||||||
isOpened: true
|
isOpened: true,
|
||||||
|
onConfirm
|
||||||
};
|
};
|
||||||
|
|
||||||
if (state.functions) {
|
if (state.functions) {
|
||||||
@ -519,7 +532,10 @@ export class FormulaPicker extends React.Component<
|
|||||||
children({
|
children({
|
||||||
isOpened: this.state.isOpened,
|
isOpened: this.state.isOpened,
|
||||||
onClick: this.handleClick,
|
onClick: this.handleClick,
|
||||||
setState: this.updateState
|
setState: this.updateState,
|
||||||
|
value,
|
||||||
|
onChange: this.handleInputChange,
|
||||||
|
disabled
|
||||||
})
|
})
|
||||||
) : (
|
) : (
|
||||||
<div
|
<div
|
||||||
|
@ -63,6 +63,8 @@ export class FormulaPlugin {
|
|||||||
*/
|
*/
|
||||||
evalMode: boolean = true;
|
evalMode: boolean = true;
|
||||||
|
|
||||||
|
highlightMode: 'expression' | 'formula' = 'formula';
|
||||||
|
|
||||||
disableAutoMark = false;
|
disableAutoMark = false;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@ -90,6 +92,10 @@ export class FormulaPlugin {
|
|||||||
this.evalMode = evalMode;
|
this.evalMode = evalMode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setHighlightMode(highlightMode: 'expression' | 'formula') {
|
||||||
|
this.highlightMode = highlightMode;
|
||||||
|
}
|
||||||
|
|
||||||
setDisableAutoMark(disableAutoMark: boolean) {
|
setDisableAutoMark(disableAutoMark: boolean) {
|
||||||
this.disableAutoMark = disableAutoMark;
|
this.disableAutoMark = disableAutoMark;
|
||||||
this.autoMarkText(true);
|
this.autoMarkText(true);
|
||||||
@ -238,6 +244,7 @@ export class FormulaPlugin {
|
|||||||
const value = editor.getValue();
|
const value = editor.getValue();
|
||||||
const functions = this.functions;
|
const functions = this.functions;
|
||||||
const variables = this.variables;
|
const variables = this.variables;
|
||||||
|
const highlightMode = this.highlightMode;
|
||||||
|
|
||||||
// 把旧的清掉
|
// 把旧的清掉
|
||||||
this.widgets.forEach(widget => editor.removeLineWidget(widget));
|
this.widgets.forEach(widget => editor.removeLineWidget(widget));
|
||||||
@ -252,6 +259,25 @@ export class FormulaPlugin {
|
|||||||
variableMode: false
|
variableMode: false
|
||||||
});
|
});
|
||||||
traverseAst(ast, (ast: any): any => {
|
traverseAst(ast, (ast: any): any => {
|
||||||
|
if (highlightMode === 'expression') {
|
||||||
|
if (ast.type === 'script') {
|
||||||
|
this.markText(
|
||||||
|
{
|
||||||
|
line: ast.start.line - 1,
|
||||||
|
ch: ast.start.column - 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
line: ast.end.line - 1,
|
||||||
|
ch: ast.end.column - 1
|
||||||
|
},
|
||||||
|
value.substring(ast.start.index + 2, ast.end.index - 1),
|
||||||
|
'cm-expression',
|
||||||
|
value
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (ast.type === 'func_call') {
|
if (ast.type === 'func_call') {
|
||||||
const funName = ast.identifier;
|
const funName = ast.identifier;
|
||||||
const exists = functions.some(item =>
|
const exists = functions.some(item =>
|
||||||
|
6
packages/amis-ui/src/custom.d.ts
vendored
6
packages/amis-ui/src/custom.d.ts
vendored
@ -4,7 +4,11 @@ declare module '*.svg' {
|
|||||||
}
|
}
|
||||||
|
|
||||||
declare module '*.md' {
|
declare module '*.md' {
|
||||||
const content: any;
|
const content: {
|
||||||
|
toc: any;
|
||||||
|
html: string;
|
||||||
|
raw: string;
|
||||||
|
};
|
||||||
export default content;
|
export default content;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -81,6 +81,7 @@ renderer.link = function (href: string, title: string, text: string) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
function markdown2js(content: string, file: string) {
|
function markdown2js(content: string, file: string) {
|
||||||
|
var raw = content;
|
||||||
var m = rYml.exec(content);
|
var m = rYml.exec(content);
|
||||||
var info: any = {};
|
var info: any = {};
|
||||||
if (m && m[1]) {
|
if (m && m[1]) {
|
||||||
@ -214,6 +215,7 @@ function markdown2js(content: string, file: string) {
|
|||||||
}
|
}
|
||||||
) +
|
) +
|
||||||
'</div>';
|
'</div>';
|
||||||
|
info.raw = raw;
|
||||||
info.toc = toc;
|
info.toc = toc;
|
||||||
|
|
||||||
return 'export default ' + JSON.stringify(info, null, 2) + ';';
|
return 'export default ' + JSON.stringify(info, null, 2) + ';';
|
||||||
|
Loading…
Reference in New Issue
Block a user