forked from soapyigu/LeetCode-Swift
-
Notifications
You must be signed in to change notification settings - Fork 0
/
TextJustification.swift
79 lines (64 loc) · 2.31 KB
/
TextJustification.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/**
* Question Link: https://leetcode.com/problems/text-justification/
* Primary idea: Iterate the words, keep track of the index of first word and the length
* of the line. Insert spaces with fix spaces and extra spaces.
* Time Complexity: O(n), Space Complexity: O(n)
*/
class TextJustification {
func fullJustify(words: [String], _ maxWidth: Int) -> [String] {
var res = [String]()
var count = 0, last = 0
guard words.count > 0 && maxWidth >= 0 else {
return res
}
for i in 0 ..< words.count {
let wordLen = words[i].characters.count
if count + (i - last) + wordLen > maxWidth {
var spaceNum = 0
var extraNum = 0
if i - last - 1 > 0 {
spaceNum = (maxWidth - count) / (i - last - 1)
extraNum = (maxWidth - count) % (i - last - 1)
res.append(buildLine(words, last, i - 1, spaceNum, extraNum))
} else {
res.append(buildSpecialLine(words, last, last, maxWidth))
}
count = 0
last = i
}
count += wordLen
}
res.append(buildSpecialLine(words, last, words.count - 1, maxWidth))
return res
}
fileprivate func buildLine(words: [String], _ start: Int, _ end: Int, _ spaceNum: Int, _ extraNum: Int) -> String {
var res = ""
var extraNum = extraNum
for i in start ... end {
res += words[i]
if i != end {
for _ in 1 ... spaceNum {
res += " "
}
if extraNum != 0 {
res += " "
extraNum -= 1
}
}
}
return res
}
fileprivate func buildSpecialLine(words: [String], _ start: Int, _ end: Int, _ lineLength: Int) -> String {
var res = ""
for i in start ... end {
res += words[i]
if i < end {
res += " "
}
}
for _ in res.characters.count ..< lineLength {
res += " "
}
return res
}
}