Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
xxadev
vscode
提交
c89cc7e5
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,发现更多精彩内容 >>
提交
c89cc7e5
编写于
9月 25, 2018
作者:
J
Joao Moreno
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
wip: propagate filter result to tree nodes
上级
a4daeb87
变更
2
隐藏空白更改
内联
并排
Showing
2 changed file
with
77 addition
and
55 deletion
+77
-55
src/vs/base/browser/ui/tree/tree.ts
src/vs/base/browser/ui/tree/tree.ts
+2
-2
src/vs/base/browser/ui/tree/treeModel.ts
src/vs/base/browser/ui/tree/treeModel.ts
+75
-53
未找到文件。
src/vs/base/browser/ui/tree/tree.ts
浏览文件 @
c89cc7e5
...
...
@@ -101,7 +101,7 @@ class TreeRenderer<T, TFilterData, TTemplateData> implements IRenderer<ITreeNode
templateData
.
twistie
.
style
.
width
=
`
${
10
+
node
.
depth
*
10
}
px`
;
renderTwistie
(
node
,
templateData
.
twistie
);
templateData
.
count
.
textContent
=
`
${
node
.
visible
Count
}
`
;
templateData
.
count
.
textContent
=
`
${
node
.
revealed
Count
}
`
;
this
.
renderer
.
renderElement
(
node
.
element
,
index
,
templateData
.
templateData
);
}
...
...
@@ -122,7 +122,7 @@ class TreeRenderer<T, TFilterData, TTemplateData> implements IRenderer<ITreeNode
}
renderTwistie
(
node
,
templateData
.
twistie
);
templateData
.
count
.
textContent
=
`
${
node
.
visible
Count
}
`
;
templateData
.
count
.
textContent
=
`
${
node
.
revealed
Count
}
`
;
}
dispose
():
void
{
...
...
src/vs/base/browser/ui/tree/treeModel.ts
浏览文件 @
c89cc7e5
...
...
@@ -23,7 +23,7 @@ export interface ITreeNode<T, TFilterData = void> {
readonly
depth
:
number
;
readonly
collapsible
:
boolean
;
readonly
collapsed
:
boolean
;
readonly
visible
Count
:
number
;
readonly
revealed
Count
:
number
;
readonly
visible
:
boolean
;
readonly
filterData
:
TFilterData
|
undefined
;
}
...
...
@@ -32,7 +32,7 @@ interface IMutableTreeNode<T, TFilterData> extends ITreeNode<T, TFilterData> {
readonly
parent
:
IMutableTreeNode
<
T
,
TFilterData
>
|
undefined
;
readonly
children
:
IMutableTreeNode
<
T
,
TFilterData
>
[];
collapsed
:
boolean
;
visible
Count
:
number
;
revealed
Count
:
number
;
visible
:
boolean
;
filterData
:
TFilterData
|
undefined
;
}
...
...
@@ -40,7 +40,7 @@ interface IMutableTreeNode<T, TFilterData> extends ITreeNode<T, TFilterData> {
export
const
enum
Visibility
{
Hidden
,
Visible
,
Recurse
// TODO@joao come up with a better name
//
Recurse // TODO@joao come up with a better name
}
export
interface
IFilterResult
<
TFilterData
>
{
...
...
@@ -48,12 +48,16 @@ export interface IFilterResult<TFilterData> {
data
:
TFilterData
;
}
function
isFilterResult
<
T
>
(
obj
:
any
):
obj
is
IFilterResult
<
T
>
{
return
typeof
obj
===
'
object
'
&&
'
visibility
'
in
obj
&&
'
data
'
in
obj
;
}
export
interface
IFilter
<
T
,
TFilterData
>
{
filter
(
element
:
T
):
Visibility
|
IFilterResult
<
TFilterData
>
;
getVisibility
(
element
:
T
):
Visibility
|
IFilterResult
<
TFilterData
>
;
}
function
visibleCountReducer
<
T
>
(
result
:
number
,
node
:
IMutableTreeNode
<
T
,
any
>
):
number
{
return
result
+
(
node
.
collapsed
?
1
:
node
.
visible
Count
);
return
result
+
(
node
.
collapsed
?
1
:
node
.
revealed
Count
);
}
function
getVisibleCount
<
T
>
(
nodes
:
IMutableTreeNode
<
T
,
any
>
[]):
number
{
...
...
@@ -65,20 +69,20 @@ function getVisibleCount<T>(nodes: IMutableTreeNode<T, any>[]): number {
* all the visible nodes in an array.
*/
function
updateVisibleCount
<
T
,
TFilterData
>
(
node
:
IMutableTreeNode
<
T
,
TFilterData
>
):
ITreeNode
<
T
,
TFilterData
>
[]
{
const
previousVisibleCount
=
node
.
visible
Count
;
const
previousVisibleCount
=
node
.
revealed
Count
;
const
result
:
ITreeNode
<
T
,
TFilterData
>
[]
=
[];
function
_updateVisibleCount
(
node
:
IMutableTreeNode
<
T
,
TFilterData
>
):
number
{
result
.
push
(
node
);
node
.
visible
Count
=
1
;
node
.
revealed
Count
=
1
;
if
(
!
node
.
collapsed
)
{
for
(
const
child
of
node
.
children
)
{
node
.
visible
Count
+=
_updateVisibleCount
(
child
);
node
.
revealed
Count
+=
_updateVisibleCount
(
child
);
}
}
return
node
.
visible
Count
;
return
node
.
revealed
Count
;
}
_updateVisibleCount
(
node
);
...
...
@@ -87,7 +91,7 @@ function updateVisibleCount<T, TFilterData>(node: IMutableTreeNode<T, TFilterDat
node
=
node
.
parent
;
while
(
node
)
{
node
.
visible
Count
+=
visibleCountDiff
;
node
.
revealed
Count
+=
visibleCountDiff
;
node
=
node
.
parent
;
}
...
...
@@ -104,26 +108,6 @@ function getTreeElementIterator<T>(elements: Iterator<ITreeElement<T>> | ITreeEl
}
}
function
treeElementToNode
<
T
,
TFilterData
>
(
treeElement
:
ITreeElement
<
T
>
,
parent
:
IMutableTreeNode
<
T
,
TFilterData
>
,
visible
:
boolean
,
treeListElements
:
ITreeNode
<
T
,
TFilterData
>
[]):
IMutableTreeNode
<
T
,
TFilterData
>
{
const
depth
=
parent
.
depth
+
1
;
const
{
element
,
collapsible
,
collapsed
}
=
treeElement
;
const
node
=
{
parent
,
element
,
children
:
[],
depth
,
collapsible
:
!!
collapsible
,
collapsed
:
!!
collapsed
,
visibleCount
:
1
,
visible
:
true
,
filterData
:
undefined
};
if
(
visible
)
{
treeListElements
.
push
(
node
);
}
const
children
=
getTreeElementIterator
(
treeElement
.
children
);
node
.
children
=
Iterator
.
collect
(
Iterator
.
map
(
children
,
el
=>
treeElementToNode
(
el
,
node
,
visible
&&
!
treeElement
.
collapsed
,
treeListElements
)));
node
.
collapsible
=
node
.
collapsible
||
node
.
children
.
length
>
0
;
if
(
!
collapsed
)
{
node
.
visibleCount
+=
getVisibleCount
(
node
.
children
);
}
return
node
;
}
function
treeNodeToElement
<
T
>
(
node
:
IMutableTreeNode
<
T
,
any
>
):
ITreeElement
<
T
>
{
const
{
element
,
collapsed
}
=
node
;
const
children
=
Iterator
.
map
(
Iterator
.
fromArray
(
node
.
children
),
treeNodeToElement
);
...
...
@@ -142,6 +126,10 @@ export function getNodeLocation<T>(node: ITreeNode<T, any>): number[] {
return
location
.
reverse
();
}
export
interface
ITreeOptions
<
T
,
TFilterData
=
void
>
{
filter
?:
IFilter
<
T
,
TFilterData
>
;
}
export
class
TreeModel
<
T
,
TFilterData
=
void
>
{
private
root
:
IMutableTreeNode
<
T
,
TFilterData
>
=
{
...
...
@@ -151,7 +139,7 @@ export class TreeModel<T, TFilterData = void> {
depth
:
0
,
collapsible
:
false
,
collapsed
:
false
,
visible
Count
:
1
,
revealed
Count
:
1
,
visible
:
true
,
filterData
:
undefined
};
...
...
@@ -159,45 +147,79 @@ export class TreeModel<T, TFilterData = void> {
private
_onDidChangeCollapseState
=
new
Emitter
<
ITreeNode
<
T
,
TFilterData
>>
();
readonly
onDidChangeCollapseState
:
Event
<
ITreeNode
<
T
,
TFilterData
>>
=
this
.
_onDidChangeCollapseState
.
event
;
constructor
(
private
list
:
ISpliceable
<
ITreeNode
<
T
,
TFilterData
>>
)
{
}
private
filter
?:
IFilter
<
T
,
TFilterData
>
;
constructor
(
private
list
:
ISpliceable
<
ITreeNode
<
T
,
TFilterData
>>
,
options
:
ITreeOptions
<
T
,
TFilterData
>
=
{})
{
this
.
filter
=
options
.
filter
;
}
splice
(
location
:
number
[],
deleteCount
:
number
,
toInsert
?:
ISequence
<
ITreeElement
<
T
>>
):
Iterator
<
ITreeElement
<
T
>>
{
if
(
location
.
length
===
0
)
{
throw
new
Error
(
'
Invalid tree location
'
);
}
const
{
parentNode
,
listIndex
,
visible
}
=
this
.
findParentNode
(
location
);
const
{
parentNode
,
listIndex
,
revealed
}
=
this
.
findParentNode
(
location
);
const
treeListElementsToInsert
:
ITreeNode
<
T
,
TFilterData
>
[]
=
[];
const
elementsToInsert
=
getTreeElementIterator
(
toInsert
);
const
nodesToInsert
=
Iterator
.
collect
(
Iterator
.
map
(
elementsToInsert
,
el
=>
t
reeElementToNode
(
el
,
parentNode
,
visible
,
treeListElementsToInsert
)));
const
nodesToInsert
=
Iterator
.
collect
(
Iterator
.
map
(
elementsToInsert
,
el
=>
t
his
.
createTreeNode
(
el
,
parentNode
,
revealed
,
treeListElementsToInsert
)));
const
lastIndex
=
location
[
location
.
length
-
1
];
const
deletedNodes
=
parentNode
.
children
.
splice
(
lastIndex
,
deleteCount
,
...
nodesToInsert
);
const
visibleDeleteCount
=
getVisibleCount
(
deletedNodes
);
parentNode
.
visible
Count
+=
getVisibleCount
(
nodesToInsert
)
-
visibleDeleteCount
;
parentNode
.
revealed
Count
+=
getVisibleCount
(
nodesToInsert
)
-
visibleDeleteCount
;
if
(
visible
)
{
if
(
revealed
)
{
this
.
list
.
splice
(
listIndex
,
visibleDeleteCount
,
treeListElementsToInsert
);
}
return
Iterator
.
map
(
Iterator
.
fromArray
(
deletedNodes
),
treeNodeToElement
);
}
private
createTreeNode
(
treeElement
:
ITreeElement
<
T
>
,
parent
:
IMutableTreeNode
<
T
,
TFilterData
>
,
revealed
:
boolean
,
treeListElements
:
ITreeNode
<
T
,
TFilterData
>
[]):
IMutableTreeNode
<
T
,
TFilterData
>
{
const
depth
=
parent
.
depth
+
1
;
const
{
element
,
collapsible
,
collapsed
}
=
treeElement
;
const
visibility
=
this
.
filter
?
this
.
filter
.
getVisibility
(
element
)
:
Visibility
.
Visible
;
let
visible
=
true
;
let
filterData
:
TFilterData
|
undefined
=
undefined
;
if
(
isFilterResult
(
visibility
))
{
visible
=
visibility
.
visibility
===
Visibility
.
Visible
;
filterData
=
visibility
.
data
;
}
const
node
=
{
parent
,
element
,
children
:
[],
depth
,
collapsible
:
!!
collapsible
,
collapsed
:
!!
collapsed
,
revealedCount
:
1
,
visible
,
filterData
};
if
(
revealed
)
{
treeListElements
.
push
(
node
);
}
const
children
=
getTreeElementIterator
(
treeElement
.
children
);
node
.
children
=
Iterator
.
collect
(
Iterator
.
map
(
children
,
el
=>
this
.
createTreeNode
(
el
,
node
,
revealed
&&
!
treeElement
.
collapsed
,
treeListElements
)));
node
.
collapsible
=
node
.
collapsible
||
node
.
children
.
length
>
0
;
if
(
!
collapsed
)
{
node
.
revealedCount
+=
getVisibleCount
(
node
.
children
);
}
return
node
;
}
getListIndex
(
location
:
number
[]):
number
{
return
this
.
findNode
(
location
).
listIndex
;
}
setCollapsed
(
location
:
number
[],
collapsed
:
boolean
):
boolean
{
const
{
node
,
listIndex
,
visible
}
=
this
.
findNode
(
location
);
return
this
.
_setCollapsed
(
node
,
listIndex
,
visible
,
collapsed
);
const
{
node
,
listIndex
,
revealed
}
=
this
.
findNode
(
location
);
return
this
.
_setCollapsed
(
node
,
listIndex
,
revealed
,
collapsed
);
}
toggleCollapsed
(
location
:
number
[]):
void
{
const
{
node
,
listIndex
,
visible
}
=
this
.
findNode
(
location
);
this
.
_setCollapsed
(
node
,
listIndex
,
visible
);
const
{
node
,
listIndex
,
revealed
}
=
this
.
findNode
(
location
);
this
.
_setCollapsed
(
node
,
listIndex
,
revealed
);
}
private
_setCollapsed
(
node
:
IMutableTreeNode
<
T
,
TFilterData
>
,
listIndex
:
number
,
visible
:
boolean
,
collapsed
?:
boolean
|
undefined
):
boolean
{
private
_setCollapsed
(
node
:
IMutableTreeNode
<
T
,
TFilterData
>
,
listIndex
:
number
,
revealed
:
boolean
,
collapsed
?:
boolean
|
undefined
):
boolean
{
if
(
!
node
.
collapsible
)
{
return
false
;
}
...
...
@@ -212,8 +234,8 @@ export class TreeModel<T, TFilterData = void> {
node
.
collapsed
=
collapsed
;
if
(
visible
)
{
const
previousVisibleCount
=
node
.
visible
Count
;
if
(
revealed
)
{
const
previousVisibleCount
=
node
.
revealed
Count
;
const
toInsert
=
updateVisibleCount
(
node
);
this
.
list
.
splice
(
listIndex
+
1
,
previousVisibleCount
-
1
,
toInsert
.
slice
(
1
));
...
...
@@ -231,8 +253,8 @@ export class TreeModel<T, TFilterData = void> {
while
(
queue
.
length
>
0
)
{
const
node
=
queue
.
shift
();
const
visible
=
listIndex
<
this
.
root
.
children
.
length
;
this
.
_setCollapsed
(
node
,
listIndex
,
visible
,
collapsed
);
const
revealed
=
listIndex
<
this
.
root
.
children
.
length
;
this
.
_setCollapsed
(
node
,
listIndex
,
revealed
,
collapsed
);
queue
.
push
(...
node
.
children
);
listIndex
++
;
...
...
@@ -244,8 +266,8 @@ export class TreeModel<T, TFilterData = void> {
return
this
.
findNode
(
location
).
node
.
collapsed
;
}
private
findNode
(
location
:
number
[]):
{
node
:
IMutableTreeNode
<
T
,
TFilterData
>
,
listIndex
:
number
,
visible
:
boolean
}
{
const
{
parentNode
,
listIndex
,
visible
}
=
this
.
findParentNode
(
location
);
private
findNode
(
location
:
number
[]):
{
node
:
IMutableTreeNode
<
T
,
TFilterData
>
,
listIndex
:
number
,
revealed
:
boolean
}
{
const
{
parentNode
,
listIndex
,
revealed
}
=
this
.
findParentNode
(
location
);
const
index
=
location
[
location
.
length
-
1
];
if
(
index
<
0
||
index
>
parentNode
.
children
.
length
)
{
...
...
@@ -254,10 +276,10 @@ export class TreeModel<T, TFilterData = void> {
const
node
=
parentNode
.
children
[
index
];
return
{
node
,
listIndex
,
visible
};
return
{
node
,
listIndex
,
revealed
};
}
private
findParentNode
(
location
:
number
[],
node
:
IMutableTreeNode
<
T
,
TFilterData
>
=
this
.
root
,
listIndex
:
number
=
0
,
visible
=
true
):
{
parentNode
:
IMutableTreeNode
<
T
,
TFilterData
>
;
listIndex
:
number
;
visible
:
boolean
;
}
{
private
findParentNode
(
location
:
number
[],
node
:
IMutableTreeNode
<
T
,
TFilterData
>
=
this
.
root
,
listIndex
:
number
=
0
,
revealed
=
true
):
{
parentNode
:
IMutableTreeNode
<
T
,
TFilterData
>
;
listIndex
:
number
;
revealed
:
boolean
;
}
{
const
[
index
,
...
rest
]
=
location
;
if
(
index
<
0
||
index
>
node
.
children
.
length
)
{
...
...
@@ -266,15 +288,15 @@ export class TreeModel<T, TFilterData = void> {
// TODO@joao perf!
for
(
let
i
=
0
;
i
<
index
;
i
++
)
{
listIndex
+=
node
.
children
[
i
].
visible
Count
;
listIndex
+=
node
.
children
[
i
].
revealed
Count
;
}
visible
=
visible
&&
!
node
.
collapsed
;
revealed
=
revealed
&&
!
node
.
collapsed
;
if
(
rest
.
length
===
0
)
{
return
{
parentNode
:
node
,
listIndex
,
visible
};
return
{
parentNode
:
node
,
listIndex
,
revealed
};
}
return
this
.
findParentNode
(
rest
,
node
.
children
[
index
],
listIndex
+
1
,
visible
);
return
this
.
findParentNode
(
rest
,
node
.
children
[
index
],
listIndex
+
1
,
revealed
);
}
}
\ No newline at end of file
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录