Compare commits

...

3 commits

Author SHA1 Message Date
Abdelilah El Aissaoui
a7d40638ab
Cleanup and changed current branch priority if there are uncommited changes 2024-02-10 01:31:07 +01:00
Abdelilah El Aissaoui
1476295415
Restored refs on each commit 2024-02-01 01:32:56 +01:00
Abdelilah El Aissaoui
2c8f8da9d8
[WIP] Created new implementation of how commits graph is generated 2024-01-31 14:46:23 +01:00
19 changed files with 459 additions and 840 deletions

View file

@ -101,7 +101,7 @@ kotlin {
} }
tasks.withType<KotlinCompile> { tasks.withType<KotlinCompile> {
kotlinOptions.allWarningsAsErrors = true kotlinOptions.allWarningsAsErrors = false
kotlinOptions.freeCompilerArgs += "-opt-in=kotlin.RequiresOptIn" kotlinOptions.freeCompilerArgs += "-opt-in=kotlin.RequiresOptIn"
} }

View file

@ -61,7 +61,7 @@ val Ref.remoteName: String
val Ref.isBranch: Boolean val Ref.isBranch: Boolean
get() { get() {
return this is ObjectIdRef.PeeledNonTag return this.name.startsWith(Constants.R_HEADS) || this.name.startsWith(Constants.R_REMOTES)
} }
val Ref.isHead: Boolean val Ref.isHead: Boolean

View file

@ -0,0 +1,222 @@
package com.jetpackduba.gitnuro.git.graph
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.withContext
import org.eclipse.jgit.api.Git
import org.eclipse.jgit.lib.Ref
import org.eclipse.jgit.revwalk.RevCommit
import javax.inject.Inject
class GenerateLogWalkUseCase @Inject constructor() {
suspend operator fun invoke(
git: Git,
firstCommit: RevCommit,
allRefs: List<Ref>,
stashes: List<RevCommit>,
hasUncommittedChanges: Boolean,
commitsLimit: Int?,
): GraphCommitList2 = withContext(Dispatchers.IO) {
val reservedLanes = mutableMapOf<Int, String>()
val graphNodes = mutableListOf<GraphNode2>()
var maxLane = 0
val availableCommitsToAdd = mutableMapOf<String, RevCommit>()
val refsWithCommits = allRefs.map {
val commit = git.repository.parseCommit(it.objectId)
commit to it
}
val commitsOfRefs = refsWithCommits.map {
it.first.name to it.first
}
val commitsOfStashes = stashes.map { it.name to it }
availableCommitsToAdd[firstCommit.name] = firstCommit
availableCommitsToAdd.putAll(commitsOfRefs)
availableCommitsToAdd.putAll(commitsOfStashes)
var currentCommit = getNextCommit(availableCommitsToAdd.values.toList())
if (hasUncommittedChanges) {
reservedLanes[0] = firstCommit.name
}
availableCommitsToAdd.remove(currentCommit?.name)
while (currentCommit != null && (commitsLimit == null || graphNodes.count() <= commitsLimit)) {
val lanes = getReservedLanes(reservedLanes, currentCommit.name)
val lane = lanes.first()
val forkingLanes = lanes - lane
val isStash = stashes.any { it == currentCommit }
val parents = sortParentsByPriority(git, currentCommit)
.filterStashParentsIfRequired(isStash)
val parentsCount = parents.count()
val mergingLanes = mutableListOf<Int>()
if (parentsCount == 1) {
reservedLanes[lane] = parents.first().name
} else if (parentsCount > 1) {
reservedLanes[lane] = parents.first().name
for (i in 1 until parentsCount) {
val availableLane = firstAvailableLane(reservedLanes)
reservedLanes[availableLane] = parents[i].name
mergingLanes.add(availableLane)
}
}
val refs = refsByCommit(refsWithCommits, currentCommit)
val graphNode = createGraphNode(
currentCommit = currentCommit,
isStash = isStash,
lane = lane,
forkingLanes = forkingLanes,
reservedLanes = reservedLanes,
mergingLanes = mergingLanes,
refs = refs
)
if (lane > maxLane) {
maxLane = lane
}
graphNodes.add(graphNode)
removeFromAllLanes(reservedLanes, graphNode.name)
availableCommitsToAdd.putAll(parents.map { it.name to it })
currentCommit = getNextCommit(availableCommitsToAdd.values.toList())
availableCommitsToAdd.remove(currentCommit?.name)
}
GraphCommitList2(graphNodes, maxLane)
}
private fun createGraphNode(
currentCommit: RevCommit,
isStash: Boolean,
lane: Int,
forkingLanes: List<Int>,
reservedLanes: MutableMap<Int, String>,
mergingLanes: MutableList<Int>,
refs: List<Ref>
) = GraphNode2(
currentCommit.name,
currentCommit.shortMessage,
currentCommit.fullMessage,
currentCommit.authorIdent,
currentCommit.committerIdent,
currentCommit.parentCount,
isStash = isStash,
lane = lane,
forkingLanes = forkingLanes,
passingLanes = reservedLanes.keys.toList() - mergingLanes.toSet() - forkingLanes.toSet(),
mergingLanes = mergingLanes,
refs = refs,
)
private fun refsByCommit(
refsWithCommits: List<Pair<RevCommit, Ref>>,
commit: RevCommit
): List<Ref> = refsWithCommits
.filter { it.first == commit }
.map { it.second }
private fun sortParentsByPriority(git: Git, currentCommit: RevCommit): List<RevCommit> {
val parents = currentCommit
.parents
.map { git.repository.parseCommit(it) }
.toMutableList()
return if (parents.count() <= 1) {
parents
} else if (parents.count() == 2) {
if (parents[0].parents.any { it.name == parents[1].name }) {
listOf(parents[1], parents[0])
} else if (parents[1].parents.any { it.name == parents[0].name }) {
parents
} else {
parents // TODO Sort by longer tree or detect the origin branch
}
} else {
parents.sortedBy { it.committerIdent.`when` }
}
}
fun List<RevCommit>.filterStashParentsIfRequired(isStash: Boolean): List<RevCommit> {
return if (isStash) {
filterNot {
it.shortMessage.startsWith("index on") ||
it.shortMessage.startsWith("untracked files on")
}
} else {
this
}
}
fun getNextCommit(availableCommits: List<RevCommit>): RevCommit? {
return availableCommits.sortedByDescending { it.committerIdent.`when` }.firstOrNull()
}
fun getReservedLanes(reservedLanes: Map<Int, String>, hash: String): List<Int> {
val reservedLanesFiltered = reservedLanes.entries
.asSequence()
.map { it.key to it.value }
.filter { (key, value) -> value == hash }
.sortedBy { it.first }
.toList()
return if (reservedLanesFiltered.isEmpty()) {
listOf(firstAvailableLane(reservedLanes))
} else {
reservedLanesFiltered.map { it.first }
}
}
fun removeFromAllLanes(reservedLanes: MutableMap<Int, String>, hash: String) {
val lanes = reservedLanes.entries.filter { it.value == hash }
for (lane in lanes) {
reservedLanes.remove(lane.key)
}
}
fun firstAvailableLane(reservedLanes: Map<Int, String>): Int {
val sortedKeys = reservedLanes.keys.sorted()
return if (sortedKeys.isEmpty() || sortedKeys.first() > 0) {
0
} else if (sortedKeys.count() == 1) {
val first = sortedKeys.first()
if (first == 0) {
1
} else {
0
}
} else {
sortedKeys.asSequence()
.zipWithNext { a, b ->
if (b - a > 1)
a + 1
else
null
}
.filterNotNull()
.firstOrNull() ?: (sortedKeys.max() + 1)
}
}
}
sealed interface ReservationType {
val hash: String
class ParentInSameLane(override val hash: String): ReservationType
class ParentInVariableLane(override val hash: String): ReservationType
}

View file

@ -1,366 +0,0 @@
package com.jetpackduba.gitnuro.git.graph
import org.eclipse.jgit.internal.JGitText
import org.eclipse.jgit.lib.AnyObjectId
import org.eclipse.jgit.revwalk.RevCommit
import org.eclipse.jgit.revwalk.RevCommitList
import org.eclipse.jgit.revwalk.RevWalk
import java.text.MessageFormat
import java.util.*
/**
* An ordered list of [GraphNode] subclasses.
*
*
* Commits are allocated into lanes as they enter the list, based upon their
* connections between descendant (child) commits and ancestor (parent) commits.
*
*
* The source of the list must be a [GraphWalk]
* and [.fillTo] must be used to populate the list.
*
* type of lane used by the application.
</L> */
class GraphCommitList : RevCommitList<GraphNode>() {
private var positionsAllocated = 0
private val freePositions = TreeSet<Int>()
private val activeLanes = HashSet<GraphLane>(32)
var maxLine = 0
private set
/** number of (child) commits on a lane */
private val laneLength = HashMap<GraphLane, Int?>(
32
)
override fun clear() {
super.clear()
positionsAllocated = 0
freePositions.clear()
activeLanes.clear()
laneLength.clear()
}
override fun source(revWalk: RevWalk) {
if (revWalk !is GraphWalk) throw ClassCastException(
MessageFormat.format(
JGitText.get().classCastNotA,
GraphWalk::class.java.name
)
)
super.source(revWalk)
}
private var parentId: AnyObjectId? = null
private val graphCommit = UncommittedChangesGraphNode()
fun addUncommittedChangesGraphCommit(parent: RevCommit) {
parentId = parent.id
graphCommit.lane = nextFreeLane()
}
override fun enter(index: Int, currCommit: GraphNode) {
var isUncommittedChangesNodeParent = false
if (currCommit.id == parentId) {
graphCommit.graphParent = currCommit
currCommit.addChild(graphCommit, addFirst = true)
isUncommittedChangesNodeParent = true
}
setupChildren(currCommit)
val nChildren = currCommit.childCount
if (nChildren == 0) {
currCommit.lane = nextFreeLane()
} else if (nChildren == 1
&& currCommit.children[0].graphParentCount < 2
) {
// Only one child, child has only us as their parent.
// Stay in the same lane as the child.
val graphNode: GraphNode = currCommit.children[0]
currCommit.lane = graphNode.lane
var len = laneLength[currCommit.lane]
len = if (len != null) Integer.valueOf(len.toInt() + 1) else Integer.valueOf(0)
if (currCommit.lane.position != INVALID_LANE_POSITION)
laneLength[currCommit.lane] = len
} else {
// We look for the child lane the current commit should continue.
// Candidate lanes for this are those with children, that have the
// current commit as their first parent.
// There can be multiple candidate lanes. In that case the longest
// lane is chosen, as this is usually the lane representing the
// branch the commit actually was made on.
// When there are no candidate lanes (i.e. the current commit has
// only children whose non-first parent it is) we place the current
// commit on a new lane.
// The lane the current commit will be placed on:
var reservedLane: GraphLane? = null
var childOnReservedLane: GraphNode? = null
var lengthOfReservedLane = -1
if (isUncommittedChangesNodeParent) {
val length = laneLength[graphCommit.lane]
if (length != null) {
reservedLane = graphCommit.lane
childOnReservedLane = graphCommit
lengthOfReservedLane = length
}
} else {
val children = currCommit.children.sortedBy { it.lane.position }
for (i in 0 until nChildren) {
val c: GraphNode = children[i]
if (c.getGraphParent(0) === currCommit) {
if (c.lane.position < 0)
println("c.lane.position is invalid (${c.lane.position})")
val length = laneLength[c.lane]
// we may be the first parent for multiple lines of
// development, try to continue the longest one
if (length != null && length > lengthOfReservedLane) {
reservedLane = c.lane
childOnReservedLane = c
lengthOfReservedLane = length
break
}
}
}
}
if (reservedLane != null) {
currCommit.lane = reservedLane
laneLength[reservedLane] = Integer.valueOf(lengthOfReservedLane + 1)
handleBlockedLanes(index, currCommit, childOnReservedLane)
} else {
currCommit.lane = nextFreeLane()
handleBlockedLanes(index, currCommit, null)
}
// close lanes of children, if there are no first parents that might
// want to continue the child lanes
for (i in 0 until nChildren) {
val graphNode = currCommit.children[i]
val firstParent = graphNode.getGraphParent(0)
if (firstParent.lane.position != INVALID_LANE_POSITION && firstParent.lane !== graphNode.lane)
closeLane(graphNode.lane)
}
}
continueActiveLanes(currCommit)
if (currCommit.parentCount == 0 && currCommit.lane.position == INVALID_LANE_POSITION)
closeLane(currCommit.lane)
}
private fun continueActiveLanes(currCommit: GraphNode) {
for (lane in activeLanes) {
if (lane !== currCommit.lane) {
currCommit.addPassingLane(lane)
}
}
}
/**
* Sets up fork and merge information in the involved PlotCommits.
* Recognizes and handles blockades that involve forking or merging arcs.
*
* @param index
* the index of `currCommit` in the list
* @param currentNode
* @param childOnLane
* the direct child on the same lane as `currCommit`,
* may be null if `currCommit` is the first commit on
* the lane
*/
private fun handleBlockedLanes(
index: Int, currentNode: GraphNode,
childOnLane: GraphNode?
) {
for (child in currentNode.children) {
if (child === childOnLane) continue // simple continuations of lanes are handled by
// continueActiveLanes() calls in enter()
// Is the child a merge or is it forking off?
val childIsMerge = child.getGraphParent(0) !== currentNode
if (childIsMerge) {
var laneToUse = currentNode.lane
laneToUse = handleMerge(
index, currentNode, childOnLane, child,
laneToUse
)
child.addMergingLane(laneToUse)
} else {
// We want to draw a forking arc in the child's lane.
// As an active lane, the child lane already continues
// (unblocked) up to this commit, we only need to mark it as
// forking off from the current commit.
val laneToUse = child.lane
currentNode.addForkingOffLane(laneToUse)
}
}
}
// Handles the case where currCommit is a non-first parent of the child
private fun handleMerge(
index: Int, currCommit: GraphNode,
childOnLane: GraphNode?, child: GraphNode, laneToUse: GraphLane
): GraphLane {
// find all blocked positions between currCommit and this child
var newLaneToUse = laneToUse
var childIndex = index // useless initialization, should
// always be set in the loop below
val blockedPositions = BitSet()
for (r in index - 1 downTo 0) {
val graphNode: GraphNode? = get(r)
if (graphNode === child) {
childIndex = r
break
}
addBlockedPosition(blockedPositions, graphNode)
}
// handle blockades
if (blockedPositions[newLaneToUse.position]) {
// We want to draw a merging arc in our lane to the child,
// which is on another lane, but our lane is blocked.
// Check if childOnLane is beetween commit and the child we
// are currently processing
var needDetour = false
if (childOnLane != null) {
for (r in index - 1 downTo childIndex + 1) {
val graphNode: GraphNode? = get(r)
if (graphNode === childOnLane) {
needDetour = true
break
}
}
}
if (needDetour) {
// It is childOnLane which is blocking us. Repositioning
// our lane would not help, because this repositions the
// child too, keeping the blockade.
// Instead, we create a "detour lane" which gets us
// around the blockade. That lane has no commits on it.
newLaneToUse = nextFreeLane(blockedPositions)
currCommit.addForkingOffLane(newLaneToUse)
closeLane(newLaneToUse)
} else {
// The blockade is (only) due to other (already closed)
// lanes at the current lane's position. In this case we
// reposition the current lane.
// We are the first commit on this lane, because
// otherwise the child commit on this lane would have
// kept other lanes from blocking us. Since we are the
// first commit, we can freely reposition.
val newPos = getFreePosition(blockedPositions)
freePositions.add(newLaneToUse.position)
newLaneToUse.position = newPos
}
}
// Actually connect currCommit to the merge child
drawLaneToChild(index, child, newLaneToUse)
return newLaneToUse
}
/**
* Connects the commit at commitIndex to the child, using the given lane.
* All blockades on the lane must be resolved before calling this method.
*
* @param commitIndex
* @param child
* @param laneToContinue
*/
private fun drawLaneToChild(
commitIndex: Int, child: GraphNode,
laneToContinue: GraphLane
) {
for (index in commitIndex - 1 downTo 0) {
val graphNode: GraphNode? = get(index)
if (graphNode === child) break
graphNode?.addPassingLane(laneToContinue)
}
}
private fun closeLane(lane: GraphLane) {
if (activeLanes.remove(lane)) {
laneLength.remove(lane)
freePositions.add(Integer.valueOf(lane.position))
}
}
private fun setupChildren(currCommit: GraphNode) {
val nParents = currCommit.parentCount
for (i in 0 until nParents) (currCommit.getParent(i) as GraphNode).addChild(currCommit)
}
private fun nextFreeLane(blockedPositions: BitSet? = null): GraphLane {
val newPlotLane = GraphLane(position = getFreePosition(blockedPositions))
activeLanes.add(newPlotLane)
laneLength[newPlotLane] = Integer.valueOf(1)
return newPlotLane
}
/**
* @param blockedPositions
* may be null
* @return a free lane position
*/
private fun getFreePosition(blockedPositions: BitSet?): Int {
if (freePositions.isEmpty()) return positionsAllocated++
if (blockedPositions != null) {
for (pos in freePositions) if (!blockedPositions[pos]) {
freePositions.remove(pos)
return pos
}
return positionsAllocated++
}
val min = freePositions.first()
freePositions.remove(min)
return min.toInt()
}
private fun addBlockedPosition(
blockedPositions: BitSet,
graphNode: GraphNode?
) {
if (graphNode != null) {
val lane = graphNode.lane
// Positions may be blocked by a commit on a lane.
if (lane.position != INVALID_LANE_POSITION) {
blockedPositions.set(lane.position)
}
// Positions may also be blocked by forking off and merging lanes.
// We don't consider passing lanes, because every passing lane forks
// off and merges at it ends.
for (graphLane in graphNode.forkingOffLanes) {
blockedPositions.set(graphLane.position)
}
for (graphLane in graphNode.mergingLanes) {
blockedPositions.set(graphLane.position)
}
}
}
fun calcMaxLine() {
if (this.isNotEmpty()) {
maxLine = this.maxOf { it.lane.position }
}
}
}

View file

@ -0,0 +1,6 @@
package com.jetpackduba.gitnuro.git.graph
data class GraphCommitList2(
val nodes: List<GraphNode2>,
val maxLane: Int
) : List<GraphNode2> by nodes

View file

@ -1,5 +0,0 @@
package com.jetpackduba.gitnuro.git.graph
const val INVALID_LANE_POSITION = -1
class GraphLane(var position: Int = 0)

View file

@ -1,105 +0,0 @@
package com.jetpackduba.gitnuro.git.graph
import org.eclipse.jgit.lib.AnyObjectId
import org.eclipse.jgit.lib.Ref
import org.eclipse.jgit.revwalk.RevCommit
private val NO_CHILDREN = arrayOf<GraphNode>()
private val NO_LANES = arrayOf<GraphLane>()
private val NO_LANE = GraphLane(INVALID_LANE_POSITION)
val NO_REFS = listOf<Ref>()
open class GraphNode(id: AnyObjectId?) : RevCommit(id), IGraphNode {
var forkingOffLanes: Array<GraphLane> = NO_LANES
var passingLanes: Array<GraphLane> = NO_LANES
var mergingLanes: Array<GraphLane> = NO_LANES
var lane: GraphLane = NO_LANE
var children: Array<GraphNode> = NO_CHILDREN
var refs: List<Ref> = NO_REFS
var isStash: Boolean = false
fun addForkingOffLane(graphLane: GraphLane) {
forkingOffLanes = addLane(graphLane, forkingOffLanes)
}
fun addPassingLane(graphLane: GraphLane) {
passingLanes = addLane(graphLane, passingLanes)
}
fun addMergingLane(graphLane: GraphLane) {
mergingLanes = addLane(graphLane, mergingLanes)
}
fun addChild(c: GraphNode, addFirst: Boolean = false) {
when (val childrenCount = children.count()) {
0 -> children = arrayOf(c)
1 -> {
if (!c.id.equals(children[0].id)) {
children = if (addFirst) {
arrayOf(c, children[0])
} else
arrayOf(children[0], c)
}
}
else -> {
for (pc in children)
if (c.id.equals(pc.id))
return
val resultArray = if (addFirst) {
val childList = mutableListOf(c)
childList.addAll(children)
childList.toTypedArray()
} else {
children.copyOf(childrenCount + 1).run {
this[childrenCount] = c
requireNoNulls()
}
}
children = resultArray
}
}
}
val childCount: Int
get() {
return children.size
}
override fun reset() {
forkingOffLanes = NO_LANES
passingLanes = NO_LANES
mergingLanes = NO_LANES
children = NO_CHILDREN
lane = NO_LANE
super.reset()
}
private fun addLane(graphLane: GraphLane, lanes: Array<GraphLane>): Array<GraphLane> {
var newLines = lanes
when (val linesCount = newLines.count()) {
0 -> newLines = arrayOf(graphLane)
1 -> newLines = arrayOf(newLines[0], graphLane)
else -> {
val n = newLines.copyOf(linesCount + 1).run {
this[linesCount] = graphLane
requireNoNulls()
}
newLines = n
}
}
return newLines
}
override val graphParentCount: Int
get() = parentCount
override fun getGraphParent(nth: Int): GraphNode {
return getParent(nth) as GraphNode
}
}

View file

@ -0,0 +1,22 @@
package com.jetpackduba.gitnuro.git.graph
import org.eclipse.jgit.lib.AnyObjectId
import org.eclipse.jgit.lib.PersonIdent
import org.eclipse.jgit.lib.Ref
import org.eclipse.jgit.revwalk.RevCommit
data class GraphNode2(
val name: String,
val message: String,
val fullMessage: String,
val authorIdent: PersonIdent,
val committerIdent: PersonIdent,
val parentCount: Int,
val isStash: Boolean,
val lane: Int,
val passingLanes: List<Int>,
val forkingLanes: List<Int>,
val mergingLanes: List<Int>,
val refs: List<Ref>,
)

View file

@ -1,172 +0,0 @@
package com.jetpackduba.gitnuro.git.graph
import org.eclipse.jgit.errors.MissingObjectException
import org.eclipse.jgit.internal.JGitText
import org.eclipse.jgit.lib.AnyObjectId
import org.eclipse.jgit.lib.Constants
import org.eclipse.jgit.lib.Ref
import org.eclipse.jgit.lib.Repository
import org.eclipse.jgit.revwalk.*
import java.io.IOException
/**
* Specialized RevWalk to be used for load data in a structured way to be displayed in a graph of commits.
*/
class GraphWalk(private var repository: Repository?) : RevWalk(repository) {
private var additionalRefMap: MutableMap<AnyObjectId, Set<Ref>>? = HashMap()
private var reverseRefMap: MutableMap<AnyObjectId, Set<Ref>>? = null
init {
super.sort(RevSort.TOPO, true)
}
override fun dispose() {
super.dispose()
if (reverseRefMap != null) {
reverseRefMap?.clear()
reverseRefMap = null
}
if (additionalRefMap != null) {
additionalRefMap?.clear()
additionalRefMap = null
}
repository = null
}
override fun sort(revSort: RevSort, use: Boolean) {
require(!(revSort == RevSort.TOPO && !use)) {
JGitText.get().topologicalSortRequired
}
super.sort(revSort, use)
}
override fun createCommit(id: AnyObjectId): RevCommit {
return GraphNode(id)
}
override fun next(): RevCommit? {
val graphNode = super.next() as GraphNode?
if (graphNode != null) {
val refs = getRefs(graphNode)
graphNode.isStash = refs.count() == 1 && refs.firstOrNull()?.name == "refs/stash"
graphNode.refs = refs
}
return graphNode
}
private fun getRefs(commitId: AnyObjectId): List<Ref> {
val repository = this.repository
var reverseRefMap = this.reverseRefMap
var additionalRefMap = this.additionalRefMap
if (reverseRefMap == null && repository != null && additionalRefMap != null) {
reverseRefMap = repository.allRefsByPeeledObjectId
this.reverseRefMap = reverseRefMap
for (entry in additionalRefMap.entries) {
val refsSet = reverseRefMap[entry.key]
var additional = entry.value.toMutableSet()
if (refsSet != null) {
if (additional.size == 1) {
// It's an unmodifiable singleton set...
additional = HashSet(additional)
}
additional.addAll(refsSet)
}
reverseRefMap[entry.key] = additional
}
additionalRefMap.clear()
additionalRefMap = null
this.additionalRefMap = additionalRefMap
}
requireNotNull(reverseRefMap) // This should never be null
val refsSet = reverseRefMap[commitId]
?: return NO_REFS
val tags = refsSet.toList()
tags.sortedWith(GraphRefComparator())
return tags
}
fun markStartAllRefs(prefix: String) {
repository?.let { repo ->
for (ref in repo.refDatabase.getRefsByPrefix(prefix)) {
if (ref.isSymbolic) continue
markStartRef(ref)
}
}
}
private fun markStartRef(ref: Ref) {
try {
val refTarget = parseAny(ref.leaf.objectId)
when (refTarget) {
is RevCommit -> markStart(refTarget)
// RevTag case handles commits without branches but only tags.
is RevTag -> {
if (refTarget.`object` is RevCommit) {
val commit = lookupCommit(refTarget.`object`)
markStart(commit)
} else {
println("Tag ${refTarget.tagName} is pointing to ${refTarget.`object`::class.simpleName}")
}
}
}
} catch (e: MissingObjectException) {
// Ignore missing Refs
}
}
internal inner class GraphRefComparator : Comparator<Ref> {
override fun compare(o1: Ref, o2: Ref): Int {
try {
val obj1 = parseAny(o1.objectId)
val obj2 = parseAny(o2.objectId)
val t1 = timeOf(obj1)
val t2 = timeOf(obj2)
if (t1 > t2) return -1
if (t1 < t2) return 1
} catch (e: IOException) {
// ignore
}
var cmp = kind(o1) - kind(o2)
if (cmp == 0)
cmp = o1.name.compareTo(o2.name)
return cmp
}
private fun timeOf(revObject: RevObject): Long {
if (revObject is RevCommit) return revObject.commitTime.toLong()
if (revObject is RevTag) {
try {
parseBody(revObject)
} catch (e: IOException) {
return 0
}
val who = revObject.taggerIdent
return who?.getWhen()?.time ?: 0
}
return 0
}
private fun kind(r: Ref): Int {
if (r.name.startsWith(Constants.R_TAGS)) return 0
if (r.name.startsWith(Constants.R_HEADS)) return 1
return if (r.name.startsWith(Constants.R_REMOTES)) 2 else 3
}
}
}

View file

@ -1,6 +0,0 @@
package com.jetpackduba.gitnuro.git.graph
interface IGraphNode {
val graphParentCount: Int
fun getGraphParent(nth: Int): GraphNode
}

View file

@ -1,15 +0,0 @@
package com.jetpackduba.gitnuro.git.graph
import org.eclipse.jgit.lib.ObjectId
class UncommittedChangesGraphNode : GraphNode(ObjectId(0, 0, 0, 0, 0)) {
var graphParent: GraphNode? = null
override val graphParentCount: Int
get() = 1 // Uncommitted changes can have a max of 1 parent commit
override fun getGraphParent(nth: Int): GraphNode {
return requireNotNull(graphParent)
}
}

View file

@ -1,66 +1,38 @@
package com.jetpackduba.gitnuro.git.log package com.jetpackduba.gitnuro.git.log
import com.jetpackduba.gitnuro.git.graph.GraphCommitList import com.jetpackduba.gitnuro.git.graph.GenerateLogWalkUseCase
import com.jetpackduba.gitnuro.git.graph.GraphWalk import com.jetpackduba.gitnuro.git.graph.GraphCommitList2
import com.jetpackduba.gitnuro.git.stash.GetStashListUseCase
import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.ensureActive
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
import org.eclipse.jgit.api.Git import org.eclipse.jgit.api.Git
import org.eclipse.jgit.lib.Constants import org.eclipse.jgit.lib.Constants
import org.eclipse.jgit.lib.Ref
import org.eclipse.jgit.lib.Repository
import javax.inject.Inject import javax.inject.Inject
class GetLogUseCase @Inject constructor() { class GetLogUseCase @Inject constructor(
private var graphWalkCached: GraphWalk? = null private val getStashListUseCase: GetStashListUseCase,
private val generateLogWalkUseCase: GenerateLogWalkUseCase,
suspend operator fun invoke(git: Git, currentBranch: Ref?, hasUncommittedChanges: Boolean, commitsLimit: Int) = ) {
suspend operator fun invoke(git: Git, hasUncommittedChanges: Boolean, commitsLimit: Int?) =
withContext(Dispatchers.IO) { withContext(Dispatchers.IO) {
val commitList = GraphCommitList() val logList = git.log().setMaxCount(1).call().toList()
val repositoryState = git.repository.repositoryState val firstCommit = logList.firstOrNull()
val allRefs =
if (currentBranch != null || repositoryState.isRebasing) { // Current branch is null when there is no log (new repo) or rebasing git.repository.refDatabase.refs.filterNot { it.name.startsWith(Constants.R_STASH) } // remove stash as it only returns the latest, we get all afterward
val logList = git.log().setMaxCount(1).call().toList() val stashes = getStashListUseCase(git)
val walk = GraphWalk(git.repository)
walk.use {
// Without this, during rebase conflicts the graph won't show the HEAD commits (new commits created
// by the rebase)
walk.markStart(walk.lookupCommit(logList.first()))
walk.markStartAllRefs(Constants.R_HEADS)
walk.markStartAllRefs(Constants.R_REMOTES)
walk.markStartAllRefs(Constants.R_TAGS)
walk.markStartAllRefs(Constants.R_STASH)
if (hasUncommittedChanges)
commitList.addUncommittedChangesGraphCommit(logList.first())
commitList.source(walk)
commitList.fillTo(commitsLimit)
}
ensureActive()
return@withContext if (firstCommit == null) {
GraphCommitList2(emptyList(), 0)
} else {
generateLogWalkUseCase.invoke(
git,
firstCommit,
allRefs,
stashes,
hasUncommittedChanges,
commitsLimit
)
} }
commitList.calcMaxLine()
return@withContext commitList
} }
private fun cachedGraphWalk(repository: Repository): GraphWalk {
val graphWalkCached = this.graphWalkCached
return if (graphWalkCached != null) {
graphWalkCached
} else {
val newGraphWalk = GraphWalk(repository)
this.graphWalkCached = newGraphWalk
newGraphWalk
}
}
} }

View file

@ -453,6 +453,7 @@ private fun CommitTreeItemEntry(
is TreeItem.Dir -> DirectoryEntry( is TreeItem.Dir -> DirectoryEntry(
dirName = entry.displayName, dirName = entry.displayName,
isExpanded = entry.isExpanded,
onClick = { onDirectoryClick(entry) }, onClick = { onDirectoryClick(entry) },
depth = entry.depth, depth = entry.depth,
onGenerateContextMenu = { onGenerateDirectoryContextMenu(entry) }, onGenerateContextMenu = { onGenerateDirectoryContextMenu(entry) },

View file

@ -1124,58 +1124,15 @@ private fun UncommittedTreeItemEntry(
) )
is TreeItem.Dir -> DirectoryEntry( is TreeItem.Dir -> DirectoryEntry(
entry.displayName, dirName = entry.displayName,
onClick, isExpanded = entry.isExpanded,
onClick = onClick,
depth = entry.depth, depth = entry.depth,
onGenerateContextMenu = { onGenerateDirectoryContextMenu(entry) }, onGenerateContextMenu = { onGenerateDirectoryContextMenu(entry) },
) )
} }
} }
//@Composable
//private fun TreeItemEntry(
// entry: TreeItem<StatusEntry>,
// isSelected: Boolean,
// actionTitle: String,
// actionColor: Color,
// onClick: () -> Unit,
// onButtonClick: () -> Unit,
// onGenerateContextMenu: (StatusEntry) -> List<ContextMenuElement>,
// onGenerateDirectoryContextMenu: (TreeItem.Dir) -> List<ContextMenuElement>,
//) {
// when (entry) {
// is TreeItem.File -> TreeFileEntry(
// entry,
// isSelected,
// actionTitle,
// actionColor,
// onClick,
// onButtonClick,
// onGenerateContextMenu,
// )
//
// is TreeItem.Dir -> TreeDirEntry(
// entry,
// onClick,
// onGenerateDirectoryContextMenu,
// )
// }
//}
internal fun placeRightOrBottom(
totalSize: Int,
size: IntArray,
outPosition: IntArray,
reverseInput: Boolean
) {
val consumedSize = size.fold(0) { a, b -> a + b }
var current = totalSize - consumedSize
size.forEachIndexed(reverseInput) { index, it ->
outPosition[index] = current
current += it
}
}
private inline fun IntArray.forEachIndexed(reversed: Boolean, action: (Int, Int) -> Unit) { private inline fun IntArray.forEachIndexed(reversed: Boolean, action: (Int, Int) -> Unit) {
if (!reversed) { if (!reversed) {
forEachIndexed(action) forEachIndexed(action)

View file

@ -141,13 +141,14 @@ fun FileEntry(
@Composable @Composable
fun DirectoryEntry( fun DirectoryEntry(
dirName: String, dirName: String,
isExpanded: Boolean,
onClick: () -> Unit, onClick: () -> Unit,
depth: Int = 0, depth: Int = 0,
onGenerateContextMenu: () -> List<ContextMenuElement>, onGenerateContextMenu: () -> List<ContextMenuElement>,
) { ) {
FileEntry( FileEntry(
icon = painterResource(AppIcons.FOLDER), icon = painterResource(if (isExpanded) AppIcons.FOLDER_OPEN else AppIcons.FOLDER),
iconColor = MaterialTheme.colors.onBackground, iconColor = MaterialTheme.colors.onBackground,
isSelected = false, isSelected = false,
onClick = onClick, onClick = onClick,

View file

@ -38,8 +38,8 @@ import androidx.compose.ui.unit.Dp
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import com.jetpackduba.gitnuro.AppIcons import com.jetpackduba.gitnuro.AppIcons
import com.jetpackduba.gitnuro.extensions.* import com.jetpackduba.gitnuro.extensions.*
import com.jetpackduba.gitnuro.git.graph.GraphCommitList import com.jetpackduba.gitnuro.git.graph.GraphCommitList2
import com.jetpackduba.gitnuro.git.graph.GraphNode import com.jetpackduba.gitnuro.git.graph.GraphNode2
import com.jetpackduba.gitnuro.git.workspace.StatusSummary import com.jetpackduba.gitnuro.git.workspace.StatusSummary
import com.jetpackduba.gitnuro.keybindings.KeybindingOption import com.jetpackduba.gitnuro.keybindings.KeybindingOption
import com.jetpackduba.gitnuro.keybindings.matchesBinding import com.jetpackduba.gitnuro.keybindings.matchesBinding
@ -175,7 +175,7 @@ private fun LogLoaded(
if (graphWidth.value < CANVAS_MIN_WIDTH) graphWidth = CANVAS_MIN_WIDTH.dp if (graphWidth.value < CANVAS_MIN_WIDTH) graphWidth = CANVAS_MIN_WIDTH.dp
val maxLinePosition = if (commitList.isNotEmpty()) val maxLinePosition = if (commitList.isNotEmpty())
commitList.maxLine commitList.maxLane
else else
MIN_GRAPH_LANES MIN_GRAPH_LANES
@ -307,7 +307,7 @@ private fun LogLoaded(
suspend fun scrollToCommit( suspend fun scrollToCommit(
verticalScrollState: LazyListState, verticalScrollState: LazyListState,
commitList: GraphCommitList, commitList: GraphCommitList2,
commit: RevCommit?, commit: RevCommit?,
) { ) {
val index = commitList.indexOfFirst { it.name == commit?.name } val index = commitList.indexOfFirst { it.name == commit?.name }
@ -319,7 +319,7 @@ suspend fun scrollToCommit(
suspend fun scrollToUncommittedChanges( suspend fun scrollToUncommittedChanges(
verticalScrollState: LazyListState, verticalScrollState: LazyListState,
commitList: GraphCommitList, commitList: GraphCommitList2,
) { ) {
if (commitList.isNotEmpty()) if (commitList.isNotEmpty())
verticalScrollState.scrollToItem(0) verticalScrollState.scrollToItem(0)
@ -429,12 +429,12 @@ fun SearchFilter(
fun CommitsList( fun CommitsList(
scrollState: LazyListState, scrollState: LazyListState,
hasUncommittedChanges: Boolean, hasUncommittedChanges: Boolean,
searchFilter: List<GraphNode>?, searchFilter: List<GraphNode2>?,
selectedCommit: RevCommit?, selectedCommit: RevCommit?,
logStatus: LogStatus.Loaded, logStatus: LogStatus.Loaded,
repositoryState: RepositoryState, repositoryState: RepositoryState,
selectedItem: SelectedItem, selectedItem: SelectedItem,
commitList: GraphCommitList, commitList: GraphCommitList2,
logViewModel: LogViewModel, logViewModel: LogViewModel,
commitsLimit: Int, commitsLimit: Int,
onMerge: (Ref) -> Unit, onMerge: (Ref) -> Unit,
@ -485,19 +485,19 @@ fun CommitsList(
graphNode = graphNode, graphNode = graphNode,
isSelected = selectedCommit?.name == graphNode.name, isSelected = selectedCommit?.name == graphNode.name,
currentBranch = logStatus.currentBranch, currentBranch = logStatus.currentBranch,
matchesSearchFilter = searchFilter?.contains(graphNode), matchesSearchFilter = false, //searchFilter?.contains(graphNode),
horizontalScrollState = horizontalScrollState, horizontalScrollState = horizontalScrollState,
showCreateNewBranch = { onShowLogDialog(LogDialog.NewBranch(graphNode)) }, showCreateNewBranch = { onShowLogDialog(LogDialog.NewBranch(graphNode)) },
showCreateNewTag = { onShowLogDialog(LogDialog.NewTag(graphNode)) }, showCreateNewTag = { onShowLogDialog(LogDialog.NewTag(graphNode)) },
resetBranch = { onShowLogDialog(LogDialog.ResetBranch(graphNode)) }, resetBranch = { onShowLogDialog(LogDialog.ResetBranch(graphNode)) },
onMergeBranch = onMerge, onMergeBranch = onMerge,
onRebaseBranch = onRebase, onRebaseBranch = onRebase,
onRebaseInteractive = { logViewModel.rebaseInteractive(graphNode) }, onRebaseInteractive = { /*logViewModel.rebaseInteractive(graphNode)*/ },
onRevCommitSelected = { logViewModel.selectLogLine(graphNode) }, onRevCommitSelected = { logViewModel.selectLogLine(graphNode) },
onChangeDefaultUpstreamBranch = { onShowLogDialog(LogDialog.ChangeDefaultBranch(it)) }, onChangeDefaultUpstreamBranch = { onShowLogDialog(LogDialog.ChangeDefaultBranch(it)) },
onDeleteStash = { logViewModel.deleteStash(graphNode) }, onDeleteStash = { /*logViewModel.deleteStash(graphNode)*/ },
onApplyStash = { logViewModel.applyStash(graphNode) }, onApplyStash = { /*logViewModel.applyStash(graphNode)*/ },
onPopStash = { logViewModel.popStash(graphNode) }, onPopStash = { /*logViewModel.popStash(graphNode)*/ },
) )
} }
@ -732,7 +732,7 @@ fun SummaryEntry(
private fun CommitLine( private fun CommitLine(
graphWidth: Dp, graphWidth: Dp,
logViewModel: LogViewModel, logViewModel: LogViewModel,
graphNode: GraphNode, graphNode: GraphNode2,
isSelected: Boolean, isSelected: Boolean,
currentBranch: Ref?, currentBranch: Ref?,
matchesSearchFilter: Boolean?, matchesSearchFilter: Boolean?,
@ -749,7 +749,7 @@ private fun CommitLine(
onChangeDefaultUpstreamBranch: (Ref) -> Unit, onChangeDefaultUpstreamBranch: (Ref) -> Unit,
horizontalScrollState: ScrollState, horizontalScrollState: ScrollState,
) { ) {
val isLastCommitOfCurrentBranch = currentBranch?.objectId?.name == graphNode.id.name val isLastCommitOfCurrentBranch = currentBranch?.objectId?.name == graphNode.name
ContextMenu( ContextMenu(
items = { items = {
@ -777,7 +777,7 @@ private fun CommitLine(
modifier = Modifier modifier = Modifier
.clickable { onRevCommitSelected() } .clickable { onRevCommitSelected() }
) { ) {
val nodeColor = colors[graphNode.lane.position % colors.size] val nodeColor = colors[graphNode.lane/*.position*/ % colors.size]
Box { Box {
Row( Row(
@ -843,7 +843,7 @@ private fun CommitLine(
@Composable @Composable
fun CommitMessage( fun CommitMessage(
commit: GraphNode, commit: GraphNode2,
currentBranch: Ref?, currentBranch: Ref?,
nodeColor: Color, nodeColor: Color,
matchesSearchFilter: Boolean?, matchesSearchFilter: Boolean?,
@ -903,9 +903,9 @@ fun CommitMessage(
} }
} }
val message = remember(commit.id.name) { val message = commit.message /*remember(commit.id.name) {
commit.getShortMessageTrimmed() commit.getShortMessageTrimmed()
} }*/
Text( Text(
text = message, text = message,
@ -957,12 +957,12 @@ fun SimpleDividerLog(modifier: Modifier) {
@Composable @Composable
fun CommitsGraph( fun CommitsGraph(
modifier: Modifier = Modifier, modifier: Modifier = Modifier,
plotCommit: GraphNode, plotCommit: GraphNode2,
nodeColor: Color, nodeColor: Color,
isSelected: Boolean, isSelected: Boolean,
) { ) {
val passingLanes = plotCommit.passingLanes val passingLanes = plotCommit.passingLanes
val forkingOffLanes = plotCommit.forkingOffLanes val forkingOffLanes = plotCommit.forkingLanes
val mergingLanes = plotCommit.mergingLanes val mergingLanes = plotCommit.mergingLanes
val density = LocalDensity.current.density val density = LocalDensity.current.density
val laneWidthWithDensity = remember(density) { val laneWidthWithDensity = remember(density) {
@ -976,34 +976,34 @@ fun CommitsGraph(
contentAlignment = Alignment.CenterStart, contentAlignment = Alignment.CenterStart,
) { ) {
val itemPosition = plotCommit.lane.position val itemPosition = plotCommit.lane
Canvas( Canvas(
modifier = Modifier.fillMaxSize() modifier = Modifier.fillMaxSize()
) { ) {
clipRect { clipRect {
if (plotCommit.childCount > 0) { // if (plotCommit.childCount > 0) {
drawLine( // drawLine(
color = colors[itemPosition % colors.size], // color = colors[itemPosition % colors.size],
start = Offset(laneWidthWithDensity * (itemPosition + 1), this.center.y), // start = Offset(laneWidthWithDensity * (itemPosition + 1), this.center.y),
end = Offset(laneWidthWithDensity * (itemPosition + 1), 0f), // end = Offset(laneWidthWithDensity * (itemPosition + 1), 0f),
strokeWidth = 2f, // strokeWidth = 2f,
) // )
} // }
forkingOffLanes.forEach { plotLane -> forkingOffLanes.forEach { plotLane ->
drawLine( drawLine(
color = colors[plotLane.position % colors.size], color = colors[plotLane % colors.size],
start = Offset(laneWidthWithDensity * (itemPosition + 1), this.center.y), start = Offset(laneWidthWithDensity * (itemPosition + 1), this.center.y),
end = Offset(laneWidthWithDensity * (plotLane.position + 1), 0f), end = Offset(laneWidthWithDensity * (plotLane + 1), 0f),
strokeWidth = 2f, strokeWidth = 2f,
) )
} }
mergingLanes.forEach { plotLane -> mergingLanes.forEach { plotLane ->
drawLine( drawLine(
color = colors[plotLane.position % colors.size], color = colors[plotLane % colors.size],
start = Offset(laneWidthWithDensity * (plotLane.position + 1), this.size.height), start = Offset(laneWidthWithDensity * (plotLane + 1), this.size.height),
end = Offset(laneWidthWithDensity * (itemPosition + 1), this.center.y), end = Offset(laneWidthWithDensity * (itemPosition + 1), this.center.y),
strokeWidth = 2f, strokeWidth = 2f,
) )
@ -1020,9 +1020,9 @@ fun CommitsGraph(
passingLanes.forEach { plotLane -> passingLanes.forEach { plotLane ->
drawLine( drawLine(
color = colors[plotLane.position % colors.size], color = colors[plotLane % colors.size],
start = Offset(laneWidthWithDensity * (plotLane.position + 1), 0f), start = Offset(laneWidthWithDensity * (plotLane + 1), 0f),
end = Offset(laneWidthWithDensity * (plotLane.position + 1), this.size.height), end = Offset(laneWidthWithDensity * (plotLane + 1), this.size.height),
strokeWidth = 2f, strokeWidth = 2f,
) )
} }
@ -1042,7 +1042,7 @@ fun CommitsGraph(
@Composable @Composable
fun CommitNode( fun CommitNode(
modifier: Modifier = Modifier, modifier: Modifier = Modifier,
plotCommit: GraphNode, plotCommit: GraphNode2,
color: Color, color: Color,
) { ) {
val author = plotCommit.authorIdent val author = plotCommit.authorIdent

View file

@ -1,12 +1,12 @@
package com.jetpackduba.gitnuro.ui.log package com.jetpackduba.gitnuro.ui.log
import com.jetpackduba.gitnuro.git.graph.GraphNode import com.jetpackduba.gitnuro.git.graph.GraphNode2
import org.eclipse.jgit.lib.Ref import org.eclipse.jgit.lib.Ref
sealed interface LogDialog { sealed interface LogDialog {
data object None : LogDialog data object None : LogDialog
data class NewBranch(val graphNode: GraphNode) : LogDialog data class NewBranch(val graphNode: GraphNode2) : LogDialog
data class NewTag(val graphNode: GraphNode) : LogDialog data class NewTag(val graphNode: GraphNode2) : LogDialog
data class ResetBranch(val graphNode: GraphNode) : LogDialog data class ResetBranch(val graphNode: GraphNode2) : LogDialog
data class ChangeDefaultBranch(val ref: Ref) : LogDialog data class ChangeDefaultBranch(val ref: Ref) : LogDialog
} }

View file

@ -3,14 +3,13 @@ package com.jetpackduba.gitnuro.viewmodels
import androidx.compose.foundation.ScrollState import androidx.compose.foundation.ScrollState
import androidx.compose.foundation.lazy.LazyListState import androidx.compose.foundation.lazy.LazyListState
import com.jetpackduba.gitnuro.extensions.delayedStateChange import com.jetpackduba.gitnuro.extensions.delayedStateChange
import com.jetpackduba.gitnuro.extensions.shortName
import com.jetpackduba.gitnuro.extensions.simpleName import com.jetpackduba.gitnuro.extensions.simpleName
import com.jetpackduba.gitnuro.git.RefreshType import com.jetpackduba.gitnuro.git.RefreshType
import com.jetpackduba.gitnuro.git.TabState import com.jetpackduba.gitnuro.git.TabState
import com.jetpackduba.gitnuro.git.TaskEvent import com.jetpackduba.gitnuro.git.TaskEvent
import com.jetpackduba.gitnuro.git.branches.* import com.jetpackduba.gitnuro.git.branches.*
import com.jetpackduba.gitnuro.git.graph.GraphCommitList import com.jetpackduba.gitnuro.git.graph.GraphCommitList2
import com.jetpackduba.gitnuro.git.graph.GraphNode import com.jetpackduba.gitnuro.git.graph.GraphNode2
import com.jetpackduba.gitnuro.git.log.* import com.jetpackduba.gitnuro.git.log.*
import com.jetpackduba.gitnuro.git.rebase.RebaseBranchUseCase import com.jetpackduba.gitnuro.git.rebase.RebaseBranchUseCase
import com.jetpackduba.gitnuro.git.rebase.StartRebaseInteractiveUseCase import com.jetpackduba.gitnuro.git.rebase.StartRebaseInteractiveUseCase
@ -30,6 +29,7 @@ import kotlinx.coroutines.flow.*
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import org.eclipse.jgit.api.Git import org.eclipse.jgit.api.Git
import org.eclipse.jgit.api.errors.CheckoutConflictException import org.eclipse.jgit.api.errors.CheckoutConflictException
import org.eclipse.jgit.lib.ObjectId
import org.eclipse.jgit.lib.Ref import org.eclipse.jgit.lib.Ref
import org.eclipse.jgit.revwalk.RevCommit import org.eclipse.jgit.revwalk.RevCommit
import javax.inject.Inject import javax.inject.Inject
@ -147,14 +147,14 @@ class LogViewModel @Inject constructor(
val commitsLimit = if (appSettings.commitsLimitEnabled) { val commitsLimit = if (appSettings.commitsLimitEnabled) {
appSettings.commitsLimit appSettings.commitsLimit
} else } else
Int.MAX_VALUE null
val commitsLimitDisplayed = if (appSettings.commitsLimitEnabled) { val commitsLimitDisplayed = if (appSettings.commitsLimitEnabled) {
appSettings.commitsLimit appSettings.commitsLimit
} else } else
-1 -1
val log = getLogUseCase(git, currentBranch, hasUncommittedChanges, commitsLimit) val log = getLogUseCase(git, hasUncommittedChanges, commitsLimit)
_logStatus.value = _logStatus.value =
LogStatus.Loaded(hasUncommittedChanges, log, currentBranch, statusSummary, commitsLimitDisplayed) LogStatus.Loaded(hasUncommittedChanges, log, currentBranch, statusSummary, commitsLimitDisplayed)
@ -189,29 +189,29 @@ class LogViewModel @Inject constructor(
) )
} }
fun checkoutCommit(revCommit: RevCommit) = tabState.safeProcessing( fun checkoutCommit(revCommit: GraphNode2) = tabState.safeProcessing(
refreshType = RefreshType.ALL_DATA, refreshType = RefreshType.ALL_DATA,
title = "Commit checkout", title = "Commit checkout",
subtitle = "Checking out commit ${revCommit.name}", subtitle = "Checking out commit ${revCommit.name}",
) { git -> ) { git ->
checkoutCommitUseCase(git, revCommit) // checkoutCommitUseCase(git, revCommit)
} }
fun revertCommit(revCommit: RevCommit) = tabState.safeProcessing( fun revertCommit(revCommit: GraphNode2) = tabState.safeProcessing(
refreshType = RefreshType.ALL_DATA, refreshType = RefreshType.ALL_DATA,
title = "Commit revert", title = "Commit revert",
subtitle = "Reverting commit ${revCommit.name}", subtitle = "Reverting commit ${revCommit.name}",
refreshEvenIfCrashes = true, refreshEvenIfCrashes = true,
) { git -> ) { git ->
revertCommitUseCase(git, revCommit) // revertCommitUseCase(git, revCommit)
} }
fun resetToCommit(revCommit: RevCommit, resetType: ResetType) = tabState.safeProcessing( fun resetToCommit(revCommit: GraphNode2, resetType: ResetType) = tabState.safeProcessing(
refreshType = RefreshType.ALL_DATA, refreshType = RefreshType.ALL_DATA,
title = "Branch reset", title = "Branch reset",
subtitle = "Reseting branch to commit ${revCommit.shortName}", // subtitle = "Reseting branch to commit ${revCommit.shortName}",
) { git -> ) { git ->
resetToCommitUseCase(git, revCommit, resetType = resetType) // resetToCommitUseCase(git, revCommit, resetType = resetType)
} }
fun checkoutRef(ref: Ref) = tabState.safeProcessing( fun checkoutRef(ref: Ref) = tabState.safeProcessing(
@ -222,29 +222,29 @@ class LogViewModel @Inject constructor(
checkoutRefUseCase(git, ref) checkoutRefUseCase(git, ref)
} }
fun cherrypickCommit(revCommit: RevCommit) = tabState.safeProcessing( fun cherrypickCommit(revCommit: GraphNode2) = tabState.safeProcessing(
refreshType = RefreshType.UNCOMMITTED_CHANGES_AND_LOG, refreshType = RefreshType.UNCOMMITTED_CHANGES_AND_LOG,
title = "Cherry-pick", title = "Cherry-pick",
subtitle = "Cherry-picking commit ${revCommit.shortName}", // subtitle = "Cherry-picking commit ${revCommit.shortName}",
) { git -> ) { git ->
cherryPickCommitUseCase(git, revCommit) // cherryPickCommitUseCase(git, revCommit)
} }
fun createBranchOnCommit(branch: String, revCommit: RevCommit) = tabState.safeProcessing( fun createBranchOnCommit(branch: String, revCommit: GraphNode2) = tabState.safeProcessing(
refreshType = RefreshType.ALL_DATA, refreshType = RefreshType.ALL_DATA,
title = "New branch", title = "New branch",
subtitle = "Creating new branch \"$branch\" on commit ${revCommit.shortName}", // subtitle = "Creating new branch \"$branch\" on commit ${revCommit.shortName}",
refreshEvenIfCrashesInteractive = { it is CheckoutConflictException }, refreshEvenIfCrashesInteractive = { it is CheckoutConflictException },
) { git -> ) { git ->
createBranchOnCommitUseCase(git, branch, revCommit) // createBranchOnCommitUseCase(git, branch, revCommit)
} }
fun createTagOnCommit(tag: String, revCommit: RevCommit) = tabState.safeProcessing( fun createTagOnCommit(tag: String, revCommit: GraphNode2) = tabState.safeProcessing(
refreshType = RefreshType.ALL_DATA, refreshType = RefreshType.ALL_DATA,
title = "New tag", title = "New tag",
subtitle = "Creating new tag \"$tag\" on commit ${revCommit.shortName}", // subtitle = "Creating new tag \"$tag\" on commit ${revCommit.shortName}",
) { git -> ) { git ->
createTagOnCommitUseCase(git, tag, revCommit) // createTagOnCommitUseCase(git, tag, revCommit)
} }
fun mergeBranch(ref: Ref) = tabState.safeProcessing( fun mergeBranch(ref: Ref) = tabState.safeProcessing(
@ -331,10 +331,12 @@ class LogViewModel @Inject constructor(
NONE_MATCHING_INDEX NONE_MATCHING_INDEX
} }
fun selectLogLine(commit: GraphNode) = tabState.runOperation( fun selectLogLine(commit: GraphNode2) = tabState.runOperation(
refreshType = RefreshType.NONE, refreshType = RefreshType.NONE,
) { ) { git ->
tabState.newSelectedCommit(commit) val oid = ObjectId.fromString(commit.name)
tabState.newSelectedCommit(git.repository.parseCommit(oid))
println("Commit SHA: ${commit.name}")
val searchValue = _logSearchFilterResults.value val searchValue = _logSearchFilterResults.value
if (searchValue is LogSearch.SearchResults) { if (searchValue is LogSearch.SearchResults) {
@ -371,7 +373,7 @@ class LogViewModel @Inject constructor(
var startingUiIndex = NONE_MATCHING_INDEX var startingUiIndex = NONE_MATCHING_INDEX
if (matchingCommits.isNotEmpty()) { if (matchingCommits.isNotEmpty()) {
_focusCommit.emit(matchingCommits.first()) // TODO _focusCommit.emit(matchingCommits.first())
startingUiIndex = FIRST_INDEX startingUiIndex = FIRST_INDEX
} }
@ -397,7 +399,7 @@ class LogViewModel @Inject constructor(
val newCommitToSelect = commits[newIndex - 1] val newCommitToSelect = commits[newIndex - 1]
_logSearchFilterResults.value = logSearchFilterResultsValue.copy(index = newIndex) _logSearchFilterResults.value = logSearchFilterResultsValue.copy(index = newIndex)
_focusCommit.emit(newCommitToSelect) //TODO _focusCommit.emit(newCommitToSelect)
} }
suspend fun selectNextFilterCommit() { suspend fun selectNextFilterCommit() {
@ -419,7 +421,7 @@ class LogViewModel @Inject constructor(
val newCommitToSelect = commits[index] val newCommitToSelect = commits[index]
_logSearchFilterResults.value = logSearchFilterResultsValue.copy(index = newIndex) _logSearchFilterResults.value = logSearchFilterResultsValue.copy(index = newIndex)
_focusCommit.emit(newCommitToSelect) // TODO _focusCommit.emit(newCommitToSelect)
} }
fun showDialog(dialog: LogDialog) { fun showDialog(dialog: LogDialog) {
@ -449,7 +451,7 @@ sealed interface LogStatus {
data object Loading : LogStatus data object Loading : LogStatus
class Loaded( class Loaded(
val hasUncommittedChanges: Boolean, val hasUncommittedChanges: Boolean,
val plotCommitList: GraphCommitList, val plotCommitList: GraphCommitList2,
val currentBranch: Ref?, val currentBranch: Ref?,
val statusSummary: StatusSummary, val statusSummary: StatusSummary,
val commitsLimit: Int, val commitsLimit: Int,
@ -459,7 +461,7 @@ sealed interface LogStatus {
sealed interface LogSearch { sealed interface LogSearch {
data object NotSearching : LogSearch data object NotSearching : LogSearch
data class SearchResults( data class SearchResults(
val commits: List<GraphNode>, val commits: List<GraphNode2>,
val index: Int, val index: Int,
val totalCount: Int = commits.count(), val totalCount: Int = commits.count(),
) : LogSearch ) : LogSearch

View file

@ -0,0 +1,105 @@
package com.jetpackduba.gitnuro.git.graph
import org.junit.jupiter.api.Test
import org.junit.jupiter.api.Assertions.*
class GenerateLogWalkUseCaseTest {
@Test
fun getReservedLane() {
val generateLogWalkUseCase = GenerateLogWalkUseCase()
val reservedLanes = mapOf(
0 to "*",
1 to "A",
2 to "B",
3 to "C",
4 to "D",
5 to "E",
6 to "F",
)
val reservedLane = generateLogWalkUseCase.getReservedLanes(reservedLanes, "A")
assertEquals(listOf(1), reservedLane)
}
@Test
fun getReservedLane_when_value_not_present() {
val generateLogWalkUseCase = GenerateLogWalkUseCase()
val reservedLanes = mapOf(
// 0 to "*",
1 to "A",
2 to "B",
3 to "C",
4 to "D",
5 to "E",
6 to "F",
)
val reservedLane = generateLogWalkUseCase.getReservedLanes(reservedLanes, "P")
assertEquals(listOf(0), reservedLane)
}
@Test
fun firstAvailableLane_without_first_item() {
val generateLogWalkUseCase = GenerateLogWalkUseCase()
val reservedLanes = mapOf(
// 0 to "*",
1 to "A",
2 to "B",
3 to "C",
// 4 to "D",
5 to "E",
6 to "F",
)
val firstAvailableLane = generateLogWalkUseCase.firstAvailableLane(reservedLanes)
assertEquals(0, firstAvailableLane)
}
@Test
fun firstAvailableLane_without_middle_item() {
val generateLogWalkUseCase = GenerateLogWalkUseCase()
val reservedLanes = mapOf(
0 to "*",
1 to "A",
2 to "B",
3 to "C",
// 4 to "D",
5 to "E",
6 to "F",
)
val firstAvailableLane = generateLogWalkUseCase.firstAvailableLane(reservedLanes)
assertEquals(4, firstAvailableLane)
}
@Test
fun firstAvailableLane_with_empty_reserved_lanes() {
val generateLogWalkUseCase = GenerateLogWalkUseCase()
val reservedLanes = mapOf<Int, String>()
val firstAvailableLane = generateLogWalkUseCase.firstAvailableLane(reservedLanes)
assertEquals(0, firstAvailableLane)
}
@Test
fun firstAvailableLane_without_single_non_zero() {
val generateLogWalkUseCase = GenerateLogWalkUseCase()
val reservedLanes = mapOf(
1 to "A",
)
val firstAvailableLane = generateLogWalkUseCase.firstAvailableLane(reservedLanes)
assertEquals(0, firstAvailableLane)
}
@Test
fun firstAvailableLane_without_2_keys_non_zero() {
val generateLogWalkUseCase = GenerateLogWalkUseCase()
val reservedLanes = mapOf(
1 to "A",
2 to "B",
)
val firstAvailableLane = generateLogWalkUseCase.firstAvailableLane(reservedLanes)
assertEquals(0, firstAvailableLane)
}
}