Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
xxadev
vscode
提交
0f04cff1
V
vscode
项目概览
xxadev
/
vscode
与 Fork 源项目一致
从无法访问的项目Fork
通知
2
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
V
vscode
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
体验新版 GitCode,发现更多精彩内容 >>
提交
0f04cff1
编写于
5月 10, 2018
作者:
R
Rachel Macfarlane
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
Move new comment button to content widget
上级
5cb18070
变更
4
展开全部
隐藏空白更改
内联
并排
Showing
4 changed file
with
545 addition
and
468 deletion
+545
-468
src/vs/workbench/parts/comments/electron-browser/commentGlyphWidget.ts
...nch/parts/comments/electron-browser/commentGlyphWidget.ts
+45
-0
src/vs/workbench/parts/comments/electron-browser/commentThreadWidget.ts
...ch/parts/comments/electron-browser/commentThreadWidget.ts
+403
-0
src/vs/workbench/parts/comments/electron-browser/commentsEditorContribution.ts
...s/comments/electron-browser/commentsEditorContribution.ts
+86
-463
src/vs/workbench/parts/comments/electron-browser/media/review.css
...orkbench/parts/comments/electron-browser/media/review.css
+11
-5
未找到文件。
src/vs/workbench/parts/comments/electron-browser/commentGlyphWidget.ts
0 → 100644
浏览文件 @
0f04cff1
/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
'
use strict
'
;
import
{
ICodeEditor
,
IContentWidget
,
IContentWidgetPosition
,
ContentWidgetPositionPreference
}
from
'
vs/editor/browser/editorBrowser
'
;
export
class
CommentGlyphWidget
implements
IContentWidget
{
private
_id
:
string
;
private
_lineNumber
:
number
;
private
_domNode
:
HTMLDivElement
;
private
_editor
:
ICodeEditor
;
constructor
(
id
:
string
,
editor
:
ICodeEditor
,
lineNumber
:
number
,
onClick
:
()
=>
void
)
{
this
.
_domNode
=
document
.
createElement
(
'
div
'
);
this
.
_domNode
.
className
=
'
new-comment-hint
'
;
this
.
_domNode
.
addEventListener
(
'
click
'
,
onClick
);
this
.
_lineNumber
=
lineNumber
;
this
.
_editor
=
editor
;
this
.
_editor
.
addContentWidget
(
this
);
}
getDomNode
():
HTMLElement
{
return
this
.
_domNode
;
}
getId
():
string
{
return
this
.
_id
;
}
getPosition
():
IContentWidgetPosition
{
return
{
position
:
{
lineNumber
:
this
.
_lineNumber
,
column
:
1
},
preference
:
[
ContentWidgetPositionPreference
.
EXACT
]
};
}
}
\ No newline at end of file
src/vs/workbench/parts/comments/electron-browser/commentThreadWidget.ts
0 → 100644
浏览文件 @
0f04cff1
/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
'
use strict
'
;
import
*
as
nls
from
'
vs/nls
'
;
import
{
$
}
from
'
vs/base/browser/builder
'
;
import
*
as
dom
from
'
vs/base/browser/dom
'
;
import
{
ActionBar
}
from
'
vs/base/browser/ui/actionbar/actionbar
'
;
import
{
Button
}
from
'
vs/base/browser/ui/button/button
'
;
import
{
Action
}
from
'
vs/base/common/actions
'
;
import
*
as
arrays
from
'
vs/base/common/arrays
'
;
import
{
Color
}
from
'
vs/base/common/color
'
;
import
{
Emitter
,
Event
}
from
'
vs/base/common/event
'
;
import
{
IDisposable
}
from
'
vs/base/common/lifecycle
'
;
import
{
ICodeEditor
,
IEditorMouseEvent
,
MouseTargetType
}
from
'
vs/editor/browser/editorBrowser
'
;
import
*
as
modes
from
'
vs/editor/common/modes
'
;
import
{
peekViewBorder
}
from
'
vs/editor/contrib/referenceSearch/referencesWidget
'
;
import
{
IOptions
,
ZoneWidget
}
from
'
vs/editor/contrib/zoneWidget/zoneWidget
'
;
import
{
ICommandService
}
from
'
vs/platform/commands/common/commands
'
;
import
{
attachButtonStyler
}
from
'
vs/platform/theme/common/styler
'
;
import
{
ITheme
,
IThemeService
}
from
'
vs/platform/theme/common/themeService
'
;
import
{
renderMarkdown
}
from
'
vs/base/browser/htmlContentRenderer
'
;
import
{
CommentGlyphWidget
}
from
'
vs/workbench/parts/comments/electron-browser/commentGlyphWidget
'
;
const
EXPAND_ACTION_CLASS
=
'
expand-review-action octicon octicon-chevron-down
'
;
const
COLLAPSE_ACTION_CLASS
=
'
expand-review-action octicon octicon-chevron-up
'
;
declare
var
ResizeObserver
:
any
;
export
class
CommentNode
{
private
_domNode
:
HTMLElement
;
private
_body
:
HTMLElement
;
private
_md
:
HTMLElement
;
public
get
domNode
():
HTMLElement
{
return
this
.
_domNode
;
}
constructor
(
public
comment
:
modes
.
Comment
,
)
{
this
.
_domNode
=
$
(
'
div.review-comment
'
).
getHTMLElement
();
let
avatar
=
$
(
'
div.avatar-container
'
).
appendTo
(
this
.
_domNode
).
getHTMLElement
();
let
img
=
<
HTMLImageElement
>
$
(
'
img.avatar
'
).
appendTo
(
avatar
).
getHTMLElement
();
img
.
src
=
comment
.
gravatar
;
let
commentDetailsContainer
=
$
(
'
.review-comment-contents
'
).
appendTo
(
this
.
_domNode
).
getHTMLElement
();
let
header
=
$
(
'
div
'
).
appendTo
(
commentDetailsContainer
).
getHTMLElement
();
let
author
=
$
(
'
strong.author
'
).
appendTo
(
header
).
getHTMLElement
();
author
.
innerText
=
comment
.
userName
;
this
.
_body
=
$
(
'
comment-body
'
).
appendTo
(
commentDetailsContainer
).
getHTMLElement
();
this
.
_md
=
renderMarkdown
(
comment
.
body
);
this
.
_body
.
appendChild
(
this
.
_md
);
}
update
(
newComment
:
modes
.
Comment
)
{
if
(
newComment
.
body
!==
this
.
comment
.
body
)
{
this
.
_body
.
removeChild
(
this
.
_md
);
this
.
_md
=
renderMarkdown
(
newComment
.
body
);
this
.
_body
.
appendChild
(
this
.
_md
);
}
this
.
comment
=
newComment
;
}
}
export
class
ReviewZoneWidget
extends
ZoneWidget
{
private
_headElement
:
HTMLElement
;
protected
_primaryHeading
:
HTMLElement
;
protected
_secondaryHeading
:
HTMLElement
;
protected
_metaHeading
:
HTMLElement
;
protected
_actionbarWidget
:
ActionBar
;
private
_bodyElement
:
HTMLElement
;
private
_commentsElement
:
HTMLElement
;
private
_commentElements
:
CommentNode
[];
private
_resizeObserver
:
any
;
private
_onDidClose
=
new
Emitter
<
ReviewZoneWidget
>
();
private
_isCollapsed
;
private
_toggleAction
:
Action
;
private
_commentThread
:
modes
.
CommentThread
;
public
get
commentThread
():
modes
.
CommentThread
{
return
this
.
_commentThread
;
}
private
_replyCommand
:
modes
.
Command
;
private
_owner
:
number
;
public
get
owner
():
number
{
return
this
.
_owner
;
}
private
_decorationIDs
:
string
[];
private
_localToDispose
:
IDisposable
[];
constructor
(
editor
:
ICodeEditor
,
owner
:
number
,
commentThread
:
modes
.
CommentThread
,
replyCommand
:
modes
.
Command
,
options
:
IOptions
=
{},
private
readonly
themeService
:
IThemeService
,
private
readonly
commandService
:
ICommandService
)
{
super
(
editor
,
options
);
this
.
_resizeObserver
=
null
;
this
.
_owner
=
owner
;
this
.
_commentThread
=
commentThread
;
this
.
_replyCommand
=
replyCommand
;
this
.
_isCollapsed
=
commentThread
.
collapsibleState
!==
modes
.
CommentThreadCollapsibleState
.
Expanded
;
this
.
_decorationIDs
=
[];
this
.
_localToDispose
=
[];
this
.
create
();
this
.
themeService
.
onThemeChange
(
this
.
_applyTheme
,
this
);
}
public
get
onDidClose
():
Event
<
ReviewZoneWidget
>
{
return
this
.
_onDidClose
.
event
;
}
public
reveal
()
{
if
(
this
.
_isCollapsed
)
{
this
.
_isCollapsed
=
false
;
if
(
this
.
_decorationIDs
&&
this
.
_decorationIDs
.
length
)
{
let
range
=
this
.
editor
.
getModel
().
getDecorationRange
(
this
.
_decorationIDs
[
0
]);
this
.
show
(
range
,
2
);
}
else
{
this
.
show
({
lineNumber
:
this
.
_commentThread
.
range
.
startLineNumber
,
column
:
1
},
2
);
}
}
}
protected
_fillContainer
(
container
:
HTMLElement
):
void
{
this
.
setCssClass
(
'
review-widget
'
);
this
.
_headElement
=
<
HTMLDivElement
>
$
(
'
.head
'
).
getHTMLElement
();
container
.
appendChild
(
this
.
_headElement
);
this
.
_fillHead
(
this
.
_headElement
);
this
.
_bodyElement
=
<
HTMLDivElement
>
$
(
'
.body
'
).
getHTMLElement
();
container
.
appendChild
(
this
.
_bodyElement
);
}
protected
_fillHead
(
container
:
HTMLElement
):
void
{
var
titleElement
=
$
(
'
.review-title
'
).
appendTo
(
this
.
_headElement
).
getHTMLElement
();
this
.
_primaryHeading
=
$
(
'
span.filename
'
).
appendTo
(
titleElement
).
getHTMLElement
();
this
.
_secondaryHeading
=
$
(
'
span.dirname
'
).
appendTo
(
titleElement
).
getHTMLElement
();
this
.
_metaHeading
=
$
(
'
span.meta
'
).
appendTo
(
titleElement
).
getHTMLElement
();
let
primaryHeading
=
'
Reviewers:
'
;
$
(
this
.
_primaryHeading
).
safeInnerHtml
(
primaryHeading
);
this
.
_primaryHeading
.
setAttribute
(
'
aria-label
'
,
primaryHeading
);
let
secondaryHeading
=
this
.
_commentThread
.
comments
.
filter
(
arrays
.
uniqueFilter
(
comment
=>
comment
.
userName
)).
map
(
comment
=>
`@
${
comment
.
userName
}
`
).
join
(
'
,
'
);
$
(
this
.
_secondaryHeading
).
safeInnerHtml
(
secondaryHeading
);
const
actionsContainer
=
$
(
'
.review-actions
'
).
appendTo
(
this
.
_headElement
);
this
.
_actionbarWidget
=
new
ActionBar
(
actionsContainer
.
getHTMLElement
(),
{});
this
.
_disposables
.
push
(
this
.
_actionbarWidget
);
this
.
_toggleAction
=
new
Action
(
'
review.expand
'
,
nls
.
localize
(
'
label.expand
'
,
"
Expand
"
),
this
.
_isCollapsed
?
EXPAND_ACTION_CLASS
:
COLLAPSE_ACTION_CLASS
,
true
,
()
=>
{
if
(
this
.
_isCollapsed
)
{
this
.
_isCollapsed
=
false
;
}
else
{
this
.
_isCollapsed
=
true
;
this
.
hide
();
}
return
null
;
});
this
.
_actionbarWidget
.
push
(
this
.
_toggleAction
,
{
label
:
false
,
icon
:
true
});
}
toggleExpand
()
{
this
.
_toggleAction
.
run
();
}
update
(
commentThread
:
modes
.
CommentThread
)
{
const
oldCommentsLen
=
this
.
_commentElements
.
length
;
const
newCommentsLen
=
commentThread
.
comments
.
length
;
let
commentElementsToDel
:
CommentNode
[]
=
[];
let
commentElementsToDelIndex
:
number
[]
=
[];
for
(
let
i
=
0
;
i
<
oldCommentsLen
;
i
++
)
{
let
comment
=
this
.
_commentElements
[
i
].
comment
;
let
newComment
=
commentThread
.
comments
.
filter
(
c
=>
c
.
commentId
===
comment
.
commentId
);
if
(
newComment
.
length
)
{
this
.
_commentElements
[
i
].
update
(
newComment
[
0
]);
}
else
{
commentElementsToDelIndex
.
push
(
i
);
commentElementsToDel
.
push
(
this
.
_commentElements
[
i
]);
}
}
// del removed elements
for
(
let
i
=
commentElementsToDel
.
length
-
1
;
i
>=
0
;
i
--
)
{
this
.
_commentElements
.
splice
(
commentElementsToDelIndex
[
i
]);
this
.
_commentsElement
.
removeChild
(
commentElementsToDel
[
i
].
domNode
);
}
if
(
this
.
_commentElements
.
length
===
0
)
{
this
.
_commentThread
=
commentThread
;
commentThread
.
comments
.
forEach
(
comment
=>
{
let
newElement
=
new
CommentNode
(
comment
);
this
.
_commentElements
.
push
(
newElement
);
this
.
_commentsElement
.
appendChild
(
newElement
.
domNode
);
});
return
;
}
let
lastCommentElement
:
HTMLElement
=
null
;
let
newCommentNodeList
:
CommentNode
[]
=
[];
for
(
let
i
=
newCommentsLen
-
1
;
i
>=
0
;
i
--
)
{
let
currentComment
=
commentThread
.
comments
[
i
];
let
oldCommentNode
=
this
.
_commentElements
.
filter
(
commentNode
=>
commentNode
.
comment
.
commentId
===
currentComment
.
commentId
);
if
(
oldCommentNode
.
length
)
{
lastCommentElement
=
oldCommentNode
[
0
].
domNode
;
newCommentNodeList
.
unshift
(
oldCommentNode
[
0
]);
}
else
{
let
newElement
=
new
CommentNode
(
currentComment
);
newCommentNodeList
.
unshift
(
newElement
);
if
(
lastCommentElement
)
{
this
.
_commentsElement
.
insertBefore
(
newElement
.
domNode
,
lastCommentElement
);
lastCommentElement
=
newElement
.
domNode
;
}
else
{
this
.
_commentsElement
.
appendChild
(
newElement
.
domNode
);
lastCommentElement
=
newElement
.
domNode
;
}
}
}
this
.
_commentThread
=
commentThread
;
this
.
_commentElements
=
newCommentNodeList
;
}
display
(
lineNumber
:
number
)
{
const
commentWidget
=
new
CommentGlyphWidget
(
`review_
${
lineNumber
}
`
,
this
.
editor
,
lineNumber
,
()
=>
{
// Does nothing when clicked
});
this
.
editor
.
layoutContentWidget
(
commentWidget
);
this
.
_localToDispose
.
push
(
this
.
editor
.
onMouseDown
(
e
=>
this
.
onEditorMouseDown
(
e
)));
this
.
_localToDispose
.
push
(
this
.
editor
.
onMouseUp
(
e
=>
this
.
onEditorMouseUp
(
e
)));
var
headHeight
=
Math
.
ceil
(
this
.
editor
.
getConfiguration
().
lineHeight
*
1.2
);
this
.
_headElement
.
style
.
height
=
`
${
headHeight
}
px`
;
this
.
_headElement
.
style
.
lineHeight
=
this
.
_headElement
.
style
.
height
;
this
.
_commentsElement
=
$
(
'
div.comments-container
'
).
appendTo
(
this
.
_bodyElement
).
getHTMLElement
();
this
.
_commentElements
=
[];
for
(
let
i
=
0
;
i
<
this
.
_commentThread
.
comments
.
length
;
i
++
)
{
let
newCommentNode
=
new
CommentNode
(
this
.
_commentThread
.
comments
[
i
]);
this
.
_commentElements
.
push
(
newCommentNode
);
this
.
_commentsElement
.
appendChild
(
newCommentNode
.
domNode
);
}
if
(
this
.
_commentThread
.
reply
)
{
const
commentForm
=
$
(
'
.comment-form
'
).
appendTo
(
this
.
_bodyElement
).
getHTMLElement
();
const
reviewThreadReplyButton
=
<
HTMLButtonElement
>
$
(
'
button.review-thread-reply-button
'
).
appendTo
(
commentForm
).
getHTMLElement
();
reviewThreadReplyButton
.
title
=
'
Reply...
'
;
reviewThreadReplyButton
.
textContent
=
'
Reply...
'
;
const
textArea
=
<
HTMLTextAreaElement
>
$
(
'
textarea
'
).
appendTo
(
commentForm
).
getHTMLElement
();
textArea
.
placeholder
=
'
Reply...
'
;
// bind click/escape actions for reviewThreadReplyButton and textArea
reviewThreadReplyButton
.
onclick
=
()
=>
{
if
(
!
dom
.
hasClass
(
commentForm
,
'
expand
'
))
{
dom
.
addClass
(
commentForm
,
'
expand
'
);
textArea
.
focus
();
}
};
dom
.
addDisposableListener
(
textArea
,
'
blur
'
,
()
=>
{
if
(
textArea
.
value
===
''
&&
dom
.
hasClass
(
commentForm
,
'
expand
'
))
{
dom
.
removeClass
(
commentForm
,
'
expand
'
);
}
});
dom
.
addDisposableListener
(
textArea
,
'
keydown
'
,
(
ev
:
KeyboardEvent
)
=>
{
if
(
textArea
.
value
===
''
&&
ev
.
keyCode
===
27
)
{
if
(
dom
.
hasClass
(
commentForm
,
'
expand
'
))
{
dom
.
removeClass
(
commentForm
,
'
expand
'
);
}
}
});
const
formActions
=
$
(
'
.form-actions
'
).
appendTo
(
commentForm
).
getHTMLElement
();
const
button
=
new
Button
(
formActions
);
attachButtonStyler
(
button
,
this
.
themeService
);
button
.
label
=
this
.
commentThread
.
reply
.
title
;
button
.
onDidClick
(
async
()
=>
{
let
newComment
=
await
this
.
commandService
.
executeCommand
(
this
.
_replyCommand
.
id
,
this
.
editor
.
getModel
().
uri
,
{
start
:
{
line
:
lineNumber
,
column
:
1
},
end
:
{
line
:
lineNumber
,
column
:
1
}
},
this
.
_commentThread
,
textArea
.
value
);
if
(
newComment
)
{
textArea
.
value
=
''
;
this
.
_commentThread
.
comments
.
push
(
newComment
);
let
newCommentNode
=
new
CommentNode
(
newComment
);
this
.
_commentElements
.
push
(
newCommentNode
);
this
.
_commentsElement
.
appendChild
(
newCommentNode
.
domNode
);
let
secondaryHeading
=
this
.
_commentThread
.
comments
.
filter
(
arrays
.
uniqueFilter
(
comment
=>
comment
.
userName
)).
map
(
comment
=>
`@
${
comment
.
userName
}
`
).
join
(
'
,
'
);
$
(
this
.
_secondaryHeading
).
safeInnerHtml
(
secondaryHeading
);
}
});
}
this
.
_resizeObserver
=
new
ResizeObserver
(
entries
=>
{
if
(
entries
[
0
].
target
===
this
.
_bodyElement
&&
!
this
.
_isCollapsed
)
{
const
lineHeight
=
this
.
editor
.
getConfiguration
().
lineHeight
;
const
arrowHeight
=
Math
.
round
(
lineHeight
/
3
);
const
computedLinesNumber
=
Math
.
ceil
((
headHeight
+
entries
[
0
].
contentRect
.
height
+
arrowHeight
)
/
lineHeight
);
this
.
_relayout
(
computedLinesNumber
);
}
});
this
.
_resizeObserver
.
observe
(
this
.
_bodyElement
);
if
(
this
.
_commentThread
.
collapsibleState
===
modes
.
CommentThreadCollapsibleState
.
Expanded
)
{
this
.
show
({
lineNumber
:
lineNumber
,
column
:
1
},
2
);
}
}
private
mouseDownInfo
:
{
lineNumber
:
number
,
iconClicked
:
boolean
};
private
onEditorMouseDown
(
e
:
IEditorMouseEvent
):
void
{
if
(
!
e
.
event
.
leftButton
)
{
return
;
}
let
range
=
e
.
target
.
range
;
if
(
!
range
)
{
return
;
}
let
iconClicked
=
false
;
switch
(
e
.
target
.
type
)
{
case
MouseTargetType
.
GUTTER_GLYPH_MARGIN
:
iconClicked
=
true
;
break
;
default
:
return
;
}
this
.
mouseDownInfo
=
{
lineNumber
:
range
.
startLineNumber
,
iconClicked
};
}
private
onEditorMouseUp
(
e
:
IEditorMouseEvent
):
void
{
if
(
!
this
.
mouseDownInfo
)
{
return
;
}
let
lineNumber
=
this
.
mouseDownInfo
.
lineNumber
;
let
iconClicked
=
this
.
mouseDownInfo
.
iconClicked
;
let
range
=
e
.
target
.
range
;
if
(
!
range
||
range
.
startLineNumber
!==
lineNumber
)
{
return
;
}
if
(
this
.
position
&&
this
.
position
.
lineNumber
!==
lineNumber
)
{
return
;
}
if
(
!
this
.
position
&&
lineNumber
!==
this
.
_commentThread
.
range
.
startLineNumber
)
{
return
;
}
if
(
iconClicked
)
{
if
(
e
.
target
.
type
!==
MouseTargetType
.
GUTTER_GLYPH_MARGIN
)
{
return
;
}
}
if
(
this
.
_isCollapsed
)
{
this
.
_isCollapsed
=
!
this
.
_isCollapsed
;
this
.
show
({
lineNumber
:
lineNumber
,
column
:
1
},
2
);
}
else
{
this
.
_isCollapsed
=
!
this
.
_isCollapsed
;
this
.
hide
();
}
}
private
_applyTheme
(
theme
:
ITheme
)
{
let
borderColor
=
theme
.
getColor
(
peekViewBorder
)
||
Color
.
transparent
;
this
.
style
({
arrowColor
:
borderColor
,
frameColor
:
borderColor
});
}
dispose
()
{
super
.
dispose
();
if
(
this
.
_resizeObserver
)
{
this
.
_resizeObserver
.
disconnect
();
this
.
_resizeObserver
=
null
;
}
this
.
editor
.
changeDecorations
(
accessor
=>
{
accessor
.
deltaDecorations
(
this
.
_decorationIDs
,
[]);
});
this
.
_localToDispose
.
forEach
(
local
=>
local
.
dispose
());
this
.
_onDidClose
.
fire
();
}
}
\ No newline at end of file
src/vs/workbench/parts/comments/electron-browser/commentsEditorContribution.ts
浏览文件 @
0f04cff1
此差异已折叠。
点击以展开。
src/vs/workbench/parts/comments/electron-browser/media/review.css
浏览文件 @
0f04cff1
...
...
@@ -10,12 +10,18 @@
background-position
:
center
center
;
}
.monaco-editor
.margin-view-overlays
.new-comment-hint
{
background-image
:
url('comment.svg')
;
opacity
:
0.5
;
.monaco-editor
.new-comment-hint
{
display
:
flex
;
align-items
:
center
;
justify-content
:
center
;
height
:
16px
;
width
:
20px
;
padding-left
:
2px
;
background
:
url('comment.svg')
center
center
no-repeat
;
}
.monaco-editor
.new-comment-hint
:hover
{
cursor
:
pointer
;
background-repeat
:
no-repeat
;
background-position
:
center
center
;
}
.monaco-editor
.review-widget
{
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录