Skip to content

Commit 63eb2b7

Browse files
authored
Do another general cleanup (#114)
- Updates the documentation - Refactor utility functions to be more Kotlin-idiomatic - Refactor `DiffRowGenerator` to allow passing a custom algorithm, instead of a custom equalizer only
1 parent b90ea21 commit 63eb2b7

File tree

12 files changed

+100
-130
lines changed

12 files changed

+100
-130
lines changed

src/commonMain/kotlin/io/github/petertrr/diffutils/DiffUtils.kt

+20-39
Original file line numberDiff line numberDiff line change
@@ -22,11 +22,9 @@ package io.github.petertrr.diffutils
2222

2323
import io.github.petertrr.diffutils.algorithm.DiffAlgorithm
2424
import io.github.petertrr.diffutils.algorithm.DiffAlgorithmListener
25-
import io.github.petertrr.diffutils.algorithm.DiffEqualizer
2625
import io.github.petertrr.diffutils.algorithm.NoopAlgorithmListener
2726
import io.github.petertrr.diffutils.algorithm.myers.MyersDiff
2827
import io.github.petertrr.diffutils.patch.Patch
29-
import io.github.petertrr.diffutils.patch.PatchFailedException
3028
import io.github.petertrr.diffutils.text.DiffRowGenerator
3129
import kotlin.jvm.JvmName
3230
import kotlin.jvm.JvmOverloads
@@ -35,16 +33,16 @@ import kotlin.jvm.JvmOverloads
3533
private val lineBreak = Regex("\r\n|\r|\n")
3634

3735
/**
38-
* Computes the difference between the source and target text.
36+
* Computes the difference between two strings.
3937
*
4038
* By default, uses the Myers algorithm.
4139
*
42-
* @param sourceText The original text
43-
* @param targetText The target text
40+
* @param sourceText A string representing the original text
41+
* @param targetText A string representing the revised text
4442
* @param algorithm The diff algorithm to use
4543
* @param progress The diff algorithm progress listener
4644
* @param includeEqualParts Whether to include equal data parts into the patch. `false` by default.
47-
* @return The patch describing the difference between the original and target text
45+
* @return The patch describing the difference between the original and revised strings
4846
*/
4947
@JvmOverloads
5048
public fun diff(
@@ -62,37 +60,17 @@ public fun diff(
6260
includeEqualParts = includeEqualParts,
6361
)
6462

65-
/**
66-
* Computes the difference between the source and target list of elements using the Myers algorithm.
67-
*
68-
* @param source The original elements
69-
* @param target The target elements
70-
* @param equalizer The equalizer to replace the default compare algorithm [Any.equals].
71-
* If `null`, the default equalizer of the default algorithm is used.
72-
* @return The patch describing the difference between the source and target sequences
73-
*/
74-
public fun <T> diff(
75-
source: List<T>,
76-
target: List<T>,
77-
equalizer: DiffEqualizer<T>,
78-
): Patch<T> =
79-
diff(
80-
source = source,
81-
target = target,
82-
algorithm = MyersDiff(equalizer),
83-
)
84-
8563
/**
8664
* Computes the difference between the original and target list of elements.
8765
*
88-
* By default, uses the Meyers algorithm.
66+
* By default, uses the Myers algorithm.
8967
*
90-
* @param source The original elements
91-
* @param target The target elements
68+
* @param source A list representing the original sequence of elements
69+
* @param target A list representing the revised sequence of elements
9270
* @param algorithm The diff algorithm to use
9371
* @param progress The diff algorithm progress listener
94-
* @param includeEqualParts Whether to include equal data parts into the patch. `false` by default.
95-
* @return The patch describing the difference between the original and target sequences
72+
* @param includeEqualParts Whether to include equal parts in the resulting patch. `false` by default.
73+
* @return The patch describing the difference between the original and revised sequences
9674
*/
9775
@JvmOverloads
9876
public fun <T> diff(
@@ -114,6 +92,10 @@ public fun <T> diff(
11492
*
11593
* This one uses the "trick" to make out of texts lists of characters,
11694
* like [DiffRowGenerator] does and merges those changes at the end together again.
95+
*
96+
* @param original A string representing the original text
97+
* @param revised A string representing the revised text
98+
* @return The patch describing the difference between the original and revised text
11799
*/
118100
public fun diffInline(original: String, revised: String): Patch<String> {
119101
val origChars = original.toCharArray()
@@ -142,22 +124,21 @@ public fun diffInline(original: String, revised: String): Patch<String> {
142124
}
143125

144126
/**
145-
* Patch the original text with the given patch.
127+
* Applies the given patch to the original list and returns the revised list.
146128
*
147-
* @param original The original text
129+
* @param original A list representing the original sequence of elements
148130
* @param patch The patch to apply
149-
* @return The revised text
150-
* @throws PatchFailedException If the patch cannot be applied
131+
* @return A list representing the revised sequence of elements
151132
*/
152133
public fun <T> patch(original: List<T>, patch: Patch<T>): List<T> =
153134
patch.applyTo(original)
154135

155136
/**
156-
* Unpatch the revised text for a given patch
137+
* Applies the given patch to the revised list and returns the original list.
157138
*
158-
* @param revised The revised text
159-
* @param patch The given patch
160-
* @return The original text
139+
* @param revised A list representing the revised sequence of elements
140+
* @param patch The patch to apply
141+
* @return A list representing the original sequence of elements
161142
*/
162143
public fun <T> unpatch(revised: List<T>, patch: Patch<T>): List<T> =
163144
patch.restore(revised)

src/commonMain/kotlin/io/github/petertrr/diffutils/text/StringUtils.kt src/commonMain/kotlin/io/github/petertrr/diffutils/StringUtils.kt

+16-16
Original file line numberDiff line numberDiff line change
@@ -18,57 +18,57 @@
1818
*/
1919
@file:JvmName("StringUtils")
2020

21-
package io.github.petertrr.diffutils.text
21+
package io.github.petertrr.diffutils
2222

2323
import kotlin.jvm.JvmName
2424

2525
/**
2626
* Replaces all opening and closing tags (`<` and `>`)
2727
* with their escaped sequences (`&lt;` and `&gt;`).
2828
*/
29-
internal fun htmlEntities(str: String): String =
30-
str.replace("<", "&lt;").replace(">", "&gt;")
29+
internal fun String.htmlEntities(): String =
30+
replace("<", "&lt;").replace(">", "&gt;")
3131

3232
/**
3333
* Normalizes a string by escaping some HTML meta characters
3434
* and replacing tabs with 4 spaces each.
3535
*/
36-
internal fun normalize(str: String): String =
37-
htmlEntities(str).replace("\t", " ")
36+
internal fun String.normalize(): String =
37+
htmlEntities().replace("\t", " ")
3838

3939
/**
40-
* Wrap the text with the given column width
40+
* Wrap the text with the given column width.
4141
*/
42-
internal fun wrapText(line: String, columnWidth: Int): String {
42+
internal fun String.wrapText(columnWidth: Int): String {
4343
require(columnWidth >= 0) { "Column width must be greater than or equal to 0" }
4444

4545
if (columnWidth == 0) {
46-
return line
46+
return this
4747
}
4848

49-
val length = line.length
50-
val delimiter = "<br/>".length
49+
val length = length
50+
val delimiterLength = "<br/>".length
5151
var widthIndex = columnWidth
52-
val b = StringBuilder(line)
52+
val sb = StringBuilder(this)
5353
var count = 0
5454

5555
while (length > widthIndex) {
56-
var breakPoint = widthIndex + delimiter * count
56+
var breakPoint = widthIndex + delimiterLength * count
5757

58-
if (b[breakPoint - 1].isHighSurrogate() && b[breakPoint].isLowSurrogate()) {
58+
if (sb[breakPoint - 1].isHighSurrogate() && sb[breakPoint].isLowSurrogate()) {
5959
// Shift a breakpoint that would split a supplemental code-point.
6060
breakPoint += 1
6161

62-
if (breakPoint == b.length) {
62+
if (breakPoint == sb.length) {
6363
// Break before instead of after if this is the last code-point.
6464
breakPoint -= 2
6565
}
6666
}
6767

68-
b.insert(breakPoint, "<br/>")
68+
sb.insert(breakPoint, "<br/>")
6969
widthIndex += columnWidth
7070
count++
7171
}
7272

73-
return b.toString()
73+
return sb.toString()
7474
}

src/commonMain/kotlin/io/github/petertrr/diffutils/algorithm/myers/MyersDiff.kt

+1-1
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,7 @@ public class MyersDiff<T>(private val equalizer: DiffEqualizer<T> = EqualsDiffEq
5858
val max = origSize + revSize + 1
5959
val size = 1 + 2 * max
6060
val middle = size / 2
61-
val diagonal: Array<PathNode?> = arrayOfNulls(size)
61+
val diagonal = arrayOfNulls<PathNode>(size)
6262
diagonal[middle + 1] = PathNode(0, -1, snake = true, bootstrap = true, prev = null)
6363

6464
for (d in 0..<max) {

src/commonMain/kotlin/io/github/petertrr/diffutils/algorithm/myers/MyersDiffWithLinearSpace.kt

+11-11
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,6 @@ public class MyersDiffWithLinearSpace<T>(
5050
progress: DiffAlgorithmListener,
5151
) {
5252
progress.diffStep((end1 - start1) / 2 + (end2 - start2) / 2, -1)
53-
5453
val middle = getMiddleSnake(data, start1, end1, start2, end2)
5554

5655
if (middle == null ||
@@ -65,28 +64,29 @@ public class MyersDiffWithLinearSpace<T>(
6564
++i
6665
++j
6766
} else {
67+
// index is less than 0 here if data.script is empty
68+
val index = data.script.size - 1
69+
6870
// TODO: compress these commands
6971
if (end1 - start1 > end2 - start2) {
70-
if (data.script.isEmpty() ||
71-
data.script[data.script.size - 1].endOriginal != i ||
72-
data.script[data.script.size - 1].deltaType != DeltaType.DELETE
72+
if (index < 0 ||
73+
data.script[index].endOriginal != i ||
74+
data.script[index].deltaType != DeltaType.DELETE
7375
) {
7476
data.script.add(Change(DeltaType.DELETE, i, i + 1, j, j))
7577
} else {
76-
data.script[data.script.size - 1] =
77-
data.script[data.script.size - 1].copy(endOriginal = i + 1)
78+
data.script[index] = data.script[index].copy(endOriginal = i + 1)
7879
}
7980

8081
++i
8182
} else {
82-
if (data.script.isEmpty() ||
83-
data.script[data.script.size - 1].endRevised != j ||
84-
data.script[data.script.size - 1].deltaType != DeltaType.INSERT
83+
if (index < 0 ||
84+
data.script[index].endRevised != j ||
85+
data.script[index].deltaType != DeltaType.INSERT
8586
) {
8687
data.script.add(Change(DeltaType.INSERT, i, i, j, j + 1))
8788
} else {
88-
data.script[data.script.size - 1] =
89-
data.script[data.script.size - 1].copy(endRevised = j + 1)
89+
data.script[index] = data.script[index].copy(endRevised = j + 1)
9090
}
9191

9292
++j

src/commonMain/kotlin/io/github/petertrr/diffutils/algorithm/myers/PathNode.kt

+1-5
Original file line numberDiff line numberDiff line change
@@ -55,11 +55,7 @@ internal class PathNode(
5555
return null
5656
}
5757

58-
return if (!snake && prev != null) {
59-
prev.previousSnake()
60-
} else {
61-
this
62-
}
58+
return if (!snake && prev != null) prev.previousSnake() else this
6359
}
6460

6561
override fun toString(): String {

0 commit comments

Comments
 (0)