2 Commits

Author SHA1 Message Date
e019238c2c onlyVisibleItems repaired 2020-01-21 20:58:35 +01:00
4380ddf9a8 Improved function for finding not collapsed nodes. 2020-01-20 15:06:39 +01:00
4 changed files with 99 additions and 20 deletions

View File

@ -15,7 +15,7 @@ android {
minSdkVersion 23 minSdkVersion 23
targetSdkVersion 29 targetSdkVersion 29
versionCode 1 versionCode 1
versionName "0.0.1-alpha02" versionName "0.0.1-beta02"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
consumerProguardFiles 'consumer-rules.pro' consumerProguardFiles 'consumer-rules.pro'
@ -35,7 +35,7 @@ dependencies {
implementation 'androidx.appcompat:appcompat:1.1.0' implementation 'androidx.appcompat:appcompat:1.1.0'
implementation 'androidx.core:core-ktx:1.1.0' implementation 'androidx.core:core-ktx:1.1.0'
implementation "androidx.recyclerview:recyclerview:1.1.0" implementation "androidx.recyclerview:recyclerview:1.1.0"
implementation "com.github.adriankuta:tree-structure:1.2.0" implementation "com.github.adriankuta:tree-structure:1.2.3"
testImplementation 'junit:junit:4.12' testImplementation 'junit:junit:4.12'
androidTestImplementation 'androidx.test.ext:junit:1.1.1' androidTestImplementation 'androidx.test.ext:junit:1.1.1'

View File

@ -6,33 +6,78 @@ import com.github.adriankuta.datastructure.tree.TreeNode
class ExpandableTreeNode<T>(value: T) : TreeNode<T>(value) { class ExpandableTreeNode<T>(value: T) : TreeNode<T>(value) {
var expanded: Boolean = true var expanded: Boolean = true
set(value) {
field = value
children.forEach {
(it as ExpandableTreeNode).expanded = value
}
}
override fun child(value: T, childDeclaration: ChildDeclaration<T>?) { override fun child(value: T, childDeclaration: ChildDeclaration<T>?) : ExpandableTreeNode<T> {
val newChild = ExpandableTreeNode(value) val newChild = ExpandableTreeNode(value)
if (childDeclaration != null) if (childDeclaration != null)
newChild.childDeclaration() newChild.childDeclaration()
addChild(newChild) addChild(newChild)
return newChild
} }
fun getVisibleNodeCount(): Int { /**
return onlyVisibleItems() * @param skipRootNode If `True`, then root node won't be counted.
*/
internal fun getVisibleNodeCount(skipRootNode: Boolean): Int {
var size = onlyVisibleItems()
.size .size
if (skipRootNode && size > 0)
size--
return size
} }
fun getVisibleNode(position: Int): ExpandableTreeNode<T> { /**
return onlyVisibleItems()[position] * This function use Pre-order iteration to go through tree:
* ```
* e.g.
* 1
* / | \
* / | \
* 2 3 4
* / \ / | \
* 5 6 7 8 9
* / / | \
* 10 11 12 13
*
* Output (skipRootNode = false): 1 2 5 10 6 11 12 13 3 4 7 8 9
* Output (skipRootNode = true): 2 5 10 6 11 12 13 3 4 7 8 9
* ```
* @param skipRootNode If `True` root element will be omitted, and position 0 will be for first left child.
*/
internal fun getVisibleNode(position: Int, skipRootNode: Boolean): ExpandableTreeNode<T> {
val nodePosition = if (skipRootNode) position + 1 else position
return onlyVisibleItems()[nodePosition]
} }
private fun onlyVisibleItems(): List<ExpandableTreeNode<T>> { /**
* @return List of nodes which parent or higher ancestor aren't collapsed.
*/
fun onlyVisibleItems(): List<ExpandableTreeNode<T>> {
//Visible if parent of node is expanded. //Visible if parent of node is expanded.
return map { it as ExpandableTreeNode } return map { it as ExpandableTreeNode }
.filter { (it.parent as? ExpandableTreeNode)?.expanded ?: true } .filter { allAncestorsAreExpanded(it) }
}
/**
* `Ancestor` is a node reachable by repeated proceeding from child to parent.
* @return `True` if parent, and all parent's ancestors are expanded.
*/
private fun allAncestorsAreExpanded(node: ExpandableTreeNode<T>): Boolean {
var ancestor = node.parent as? ExpandableTreeNode
var ancestorsAreExpanded = isAncestorExpanded(ancestor)
while (ancestorsAreExpanded && ancestor != null) {
ancestor = ancestor.parent as? ExpandableTreeNode<T>
ancestorsAreExpanded = ancestorsAreExpanded.and(isAncestorExpanded(ancestor))
}
return ancestorsAreExpanded
}
private fun isAncestorExpanded(ancestor: ExpandableTreeNode<T>?): Boolean {
return ancestor?.expanded ?: true
} }
} }

View File

@ -12,25 +12,35 @@ abstract class MultilevelRecyclerViewAdapter<T, VH : RecyclerView.ViewHolder> :
abstract override fun onCreateViewHolder(parent: ViewGroup, nestLevel: Int): VH abstract override fun onCreateViewHolder(parent: ViewGroup, nestLevel: Int): VH
final override fun onBindViewHolder(holder: VH, position: Int) { final override fun onBindViewHolder(holder: VH, position: Int) {
onBindViewHolder(holder, treeNodes.getVisibleNode(position)) val visibleNode = treeNodes.getVisibleNode(position, true)
onBindViewHolder(holder, visibleNode, visibleNode.depth())
} }
abstract fun onBindViewHolder(holder: VH, treeNode: ExpandableTreeNode<T>) abstract fun onBindViewHolder(holder: VH, treeNode: ExpandableTreeNode<T>, nestLevel: Int)
abstract fun getTreeNodes(): ExpandableTreeNode<T> abstract fun getTreeNodes(): ExpandableTreeNode<T>
final override fun getItemCount(): Int { final override fun getItemCount(): Int {
treeNodes = getTreeNodes() treeNodes = getTreeNodes()
return treeNodes.getVisibleNodeCount() return treeNodes.getVisibleNodeCount(true) //We don't want to show root element.
} }
final override fun getItemViewType(position: Int): Int { final override fun getItemViewType(position: Int): Int {
return treeNodes.getVisibleNode(position, true).depth()
return treeNodes.getVisibleNode(position).depth()
} }
fun toggleGroup(expandableTreeNode: ExpandableTreeNode<T>) { fun toggleGroup(expandableTreeNode: ExpandableTreeNode<T>) {
expandableTreeNode.expanded = !expandableTreeNode.expanded expandableTreeNode.expanded = !expandableTreeNode.expanded
notifyDataSetChanged() notifyDataSetChanged()
} }
fun expand(expandableTreeNode: ExpandableTreeNode<T>) {
expandableTreeNode.expanded = true
notifyDataSetChanged()
}
fun collapse(expandableTreeNode: ExpandableTreeNode<T>) {
expandableTreeNode.expanded = false
notifyDataSetChanged()
}
} }

View File

@ -0,0 +1,24 @@
package com.github.adriankuta.expandable_recyclerview
import org.junit.Assert
import org.junit.Test
class ExpandableTreeNodeTest {
@Test
fun getVisibleNode() {
//given
val root = expandableTree("Root") {
child("Level 1") {
child("Level 2") {
child("Level 3") {
child("Level 4")
}
}
}
}
root.expanded = false
Assert.assertEquals("Root", root.onlyVisibleItems().first().value)
}
}