feat: Initial commit
This commit is contained in:
7
.editorconfig
Normal file
7
.editorconfig
Normal file
@@ -0,0 +1,7 @@
|
||||
root = true
|
||||
|
||||
[*.swift]
|
||||
indent_style = space
|
||||
indent_size = 2
|
||||
tab_width = 2
|
||||
trim_trailing_whitespace = true
|
||||
8
.gitignore
vendored
Normal file
8
.gitignore
vendored
Normal file
@@ -0,0 +1,8 @@
|
||||
.DS_Store
|
||||
/.build
|
||||
/Packages
|
||||
xcuserdata/
|
||||
DerivedData/
|
||||
.swiftpm/configuration/registries.json
|
||||
.swiftpm/xcode/package.xcworkspace/contents.xcworkspacedata
|
||||
.netrc
|
||||
11
.swiftformat
Normal file
11
.swiftformat
Normal file
@@ -0,0 +1,11 @@
|
||||
--self init-only
|
||||
--indent 2
|
||||
--ifdef indent
|
||||
--trimwhitespace always
|
||||
--wraparguments before-first
|
||||
--wrapparameters before-first
|
||||
--wrapcollections preserve
|
||||
--wrapconditions after-first
|
||||
--typeblanklines preserve
|
||||
--commas inline
|
||||
--stripunusedargs closure-only
|
||||
10
.swiftlint.yml
Normal file
10
.swiftlint.yml
Normal file
@@ -0,0 +1,10 @@
|
||||
disabled_rules:
|
||||
- closing_brace
|
||||
- fuction_body_length
|
||||
- opening_brace
|
||||
|
||||
included:
|
||||
- Sources
|
||||
- Tests
|
||||
|
||||
ignore_multiline_statement_conditions: true
|
||||
15
Package.resolved
Normal file
15
Package.resolved
Normal file
@@ -0,0 +1,15 @@
|
||||
{
|
||||
"originHash" : "ce5e40ef3be5875abe47bbfe26413215dee9b937de7df5293343757c7476d755",
|
||||
"pins" : [
|
||||
{
|
||||
"identity" : "rainbow",
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/onevcat/Rainbow",
|
||||
"state" : {
|
||||
"revision" : "e0dada9cd44e3fa7ec3b867e49a8ddbf543e3df3",
|
||||
"version" : "4.0.1"
|
||||
}
|
||||
}
|
||||
],
|
||||
"version" : 3
|
||||
}
|
||||
26
Package.swift
Normal file
26
Package.swift
Normal file
@@ -0,0 +1,26 @@
|
||||
// swift-tools-version: 6.0
|
||||
|
||||
import PackageDescription
|
||||
|
||||
let package = Package(
|
||||
name: "swift-cli-doc",
|
||||
products: [
|
||||
.library(name: "CliDoc", targets: ["CliDoc"])
|
||||
],
|
||||
dependencies: [
|
||||
.package(url: "https://github.com/onevcat/Rainbow", from: "4.0.0")
|
||||
],
|
||||
targets: [
|
||||
.target(
|
||||
name: "CliDoc",
|
||||
dependencies: [
|
||||
.product(name: "Rainbow", package: "Rainbow")
|
||||
]
|
||||
|
||||
),
|
||||
.testTarget(
|
||||
name: "CliDocTests",
|
||||
dependencies: ["CliDoc"]
|
||||
)
|
||||
]
|
||||
)
|
||||
59
Sources/CliDoc/Modifiers/LabelColor.swift
Normal file
59
Sources/CliDoc/Modifiers/LabelColor.swift
Normal file
@@ -0,0 +1,59 @@
|
||||
import Rainbow
|
||||
|
||||
public extension Group {
|
||||
func labelColor(_ color: NamedColor) -> some Node {
|
||||
modifier(GroupLabelModifier(color: color))
|
||||
}
|
||||
}
|
||||
|
||||
public extension Node where Self.Body == Group {
|
||||
func labelColor(_ color: NamedColor) -> some Node {
|
||||
body.modifier(GroupLabelModifier(color: color))
|
||||
}
|
||||
}
|
||||
|
||||
public extension Node where Self == Label {
|
||||
func labelColor(_ color: NamedColor) -> some Node {
|
||||
modifier(LabelColorModifier(color: color))
|
||||
}
|
||||
}
|
||||
|
||||
public extension Note where Label == CliDoc.Label {
|
||||
func labelColor(_ color: NamedColor) -> some Node {
|
||||
var node = self
|
||||
node.label.color = color
|
||||
return node
|
||||
}
|
||||
}
|
||||
|
||||
struct LabelColorModifier: NodeModifier {
|
||||
let color: NamedColor
|
||||
|
||||
func render(content: Label) -> some Node {
|
||||
var label = content
|
||||
label.color = color
|
||||
return label
|
||||
}
|
||||
}
|
||||
|
||||
struct GroupLabelModifier: NodeModifier {
|
||||
let color: NamedColor
|
||||
|
||||
func render(content: Group) -> some Node {
|
||||
var group = content
|
||||
applyLabelColor(&group)
|
||||
return group
|
||||
}
|
||||
|
||||
private func applyLabelColor(_ group: inout Group) {
|
||||
for (idx, node) in group.nodes.enumerated() {
|
||||
if var label = node as? Label {
|
||||
label.color = color
|
||||
group.nodes[idx] = label
|
||||
} else if var nestedGroup = node as? Group {
|
||||
applyLabelColor(&nestedGroup)
|
||||
group.nodes[idx] = nestedGroup
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
21
Sources/CliDoc/Modifiers/LabelStyle.swift
Normal file
21
Sources/CliDoc/Modifiers/LabelStyle.swift
Normal file
@@ -0,0 +1,21 @@
|
||||
import Rainbow
|
||||
|
||||
// public extension Node {
|
||||
// func labelStyel(_ styles: Style...) -> any Node {}
|
||||
// }
|
||||
|
||||
public extension Node where Self == Label {
|
||||
func labelStyle(_ styles: Style...) -> some Node {
|
||||
modifier(LabelStyleModifier(styles: styles))
|
||||
}
|
||||
}
|
||||
|
||||
struct LabelStyleModifier: NodeModifier {
|
||||
let styles: [Style]
|
||||
|
||||
func render(content: Label) -> some Node {
|
||||
var label = content
|
||||
label.styles = styles
|
||||
return label
|
||||
}
|
||||
}
|
||||
27
Sources/CliDoc/Node.swift
Normal file
27
Sources/CliDoc/Node.swift
Normal file
@@ -0,0 +1,27 @@
|
||||
public protocol NodeRepresentable {
|
||||
func render() -> String
|
||||
}
|
||||
|
||||
public protocol Node: NodeRepresentable {
|
||||
// swiftlint:disable type_name
|
||||
associatedtype _Body: Node
|
||||
typealias Body = _Body
|
||||
// swiftlint:enable type_name
|
||||
|
||||
@NodeBuilder
|
||||
var body: Body { get }
|
||||
}
|
||||
|
||||
public extension Node {
|
||||
func render() -> String {
|
||||
body.render()
|
||||
}
|
||||
}
|
||||
|
||||
extension String: NodeRepresentable {
|
||||
public func render() -> String { self }
|
||||
}
|
||||
|
||||
extension String: Node {
|
||||
public var body: some Node { self }
|
||||
}
|
||||
36
Sources/CliDoc/NodeBuilder.swift
Normal file
36
Sources/CliDoc/NodeBuilder.swift
Normal file
@@ -0,0 +1,36 @@
|
||||
@resultBuilder
|
||||
public enum NodeBuilder {
|
||||
|
||||
public static func buildPartialBlock<N: Node>(first: N) -> N {
|
||||
first
|
||||
}
|
||||
|
||||
public static func buildArray<N: Node>(_ components: [N]) -> Group {
|
||||
.init(components)
|
||||
}
|
||||
|
||||
public static func buildOptional<N: Node>(_ component: N?) -> OptionalNode<N> {
|
||||
.init(component: component)
|
||||
}
|
||||
|
||||
public static func buildEither<N: Node>(first component: N) -> N {
|
||||
component
|
||||
}
|
||||
|
||||
public static func buildEither<N: Node>(second component: N) -> N {
|
||||
component
|
||||
}
|
||||
|
||||
public static func buildPartialBlock<N0: Node, N1: Node>(accumulated: N0, next: N1) -> Group {
|
||||
.init([accumulated, next])
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public struct OptionalNode<N: Node>: Node {
|
||||
let component: N?
|
||||
|
||||
public var body: some Node {
|
||||
component?.render() ?? ""
|
||||
}
|
||||
}
|
||||
59
Sources/CliDoc/NodeModifier.swift
Normal file
59
Sources/CliDoc/NodeModifier.swift
Normal file
@@ -0,0 +1,59 @@
|
||||
public protocol NodeModifier {
|
||||
associatedtype Body: Node
|
||||
// swiftlint:disable type_name
|
||||
associatedtype _Content: Node
|
||||
typealias Content = _Content
|
||||
// swiftlint:enable type_name
|
||||
|
||||
func render(content: Content) -> Body
|
||||
}
|
||||
|
||||
public extension NodeModifier {
|
||||
func modifier<T: NodeModifier>(
|
||||
_ modifier: T
|
||||
) -> ModifiedNode<Self, T> where T.Content == Body {
|
||||
.init(content: self, modifier: modifier)
|
||||
}
|
||||
}
|
||||
|
||||
public extension Node {
|
||||
|
||||
func modifier<T: NodeModifier>(_ modifier: T) -> ModifiedNode<Self, T> {
|
||||
.init(content: self, modifier: modifier)
|
||||
}
|
||||
}
|
||||
|
||||
public struct ModifiedNode<Content, Modifier> {
|
||||
var content: Content
|
||||
var modifier: Modifier
|
||||
}
|
||||
|
||||
extension ModifiedNode: NodeRepresentable where Content: NodeRepresentable,
|
||||
Modifier: NodeModifier,
|
||||
Modifier.Content == Content
|
||||
{
|
||||
public func render() -> String {
|
||||
modifier.render(content: content).render()
|
||||
}
|
||||
}
|
||||
|
||||
extension ModifiedNode: Node where Content: Node,
|
||||
Modifier: NodeModifier,
|
||||
Modifier.Content == Content
|
||||
{
|
||||
public var body: some Node {
|
||||
content
|
||||
}
|
||||
}
|
||||
|
||||
extension ModifiedNode: NodeModifier where
|
||||
Content: NodeModifier,
|
||||
Modifier: NodeModifier,
|
||||
Content.Body == Modifier.Content,
|
||||
Content: Node
|
||||
{
|
||||
public func render(content: Content) -> some Node {
|
||||
let body = content.body
|
||||
return modifier.render(content: body).render()
|
||||
}
|
||||
}
|
||||
18
Sources/CliDoc/Nodes/AnyNode.swift
Normal file
18
Sources/CliDoc/Nodes/AnyNode.swift
Normal file
@@ -0,0 +1,18 @@
|
||||
/// A type erased node.
|
||||
public struct AnyNode: Node {
|
||||
private var renderNode: () -> String
|
||||
|
||||
public init<N: NodeRepresentable>(_ node: N) {
|
||||
self.renderNode = node.render
|
||||
}
|
||||
|
||||
public var body: some Node {
|
||||
renderNode()
|
||||
}
|
||||
}
|
||||
|
||||
public extension Node {
|
||||
func eraseToAnyNode() -> AnyNode {
|
||||
.init(self)
|
||||
}
|
||||
}
|
||||
18
Sources/CliDoc/Nodes/Colored.swift
Normal file
18
Sources/CliDoc/Nodes/Colored.swift
Normal file
@@ -0,0 +1,18 @@
|
||||
import Rainbow
|
||||
|
||||
public struct Colored: Node {
|
||||
var color: NamedColor
|
||||
var node: any Node
|
||||
|
||||
public init(
|
||||
color: NamedColor,
|
||||
@NodeBuilder build: () -> any Node
|
||||
) {
|
||||
self.color = color
|
||||
self.node = build()
|
||||
}
|
||||
|
||||
public var body: some Node {
|
||||
node.render().applyingColor(color)
|
||||
}
|
||||
}
|
||||
28
Sources/CliDoc/Nodes/Group.swift
Normal file
28
Sources/CliDoc/Nodes/Group.swift
Normal file
@@ -0,0 +1,28 @@
|
||||
/// A group container holding one or more nodes.
|
||||
public struct Group: Node {
|
||||
var nodes: [any Node]
|
||||
var separator: String
|
||||
|
||||
init(_ nodes: [any Node], separator: String = "\n") {
|
||||
self.nodes = nodes
|
||||
self.separator = separator
|
||||
}
|
||||
|
||||
public init(
|
||||
separator: String = " ",
|
||||
@NodeBuilder build: () -> any Node
|
||||
) {
|
||||
let node = build()
|
||||
if var group = node as? Self {
|
||||
group.separator = separator
|
||||
self = group
|
||||
} else {
|
||||
self.init([node], separator: separator)
|
||||
}
|
||||
}
|
||||
|
||||
public var body: some Node {
|
||||
nodes.map { $0.render() }.joined(separator: separator)
|
||||
}
|
||||
|
||||
}
|
||||
45
Sources/CliDoc/Nodes/Label.swift
Normal file
45
Sources/CliDoc/Nodes/Label.swift
Normal file
@@ -0,0 +1,45 @@
|
||||
import Rainbow
|
||||
|
||||
public struct Label: Node {
|
||||
|
||||
var color: NamedColor?
|
||||
var styles: [Style]
|
||||
let node: any Node
|
||||
|
||||
public init(
|
||||
_ label: String,
|
||||
color: NamedColor? = nil,
|
||||
style styles: Style...
|
||||
) {
|
||||
self.color = color
|
||||
self.node = label
|
||||
self.styles = styles
|
||||
}
|
||||
|
||||
public init(
|
||||
_ label: String,
|
||||
color: NamedColor? = nil,
|
||||
style styles: [Style] = []
|
||||
) {
|
||||
self.color = color
|
||||
self.node = label
|
||||
self.styles = styles
|
||||
}
|
||||
|
||||
public init(
|
||||
color: NamedColor? = nil,
|
||||
styles: [Style] = [],
|
||||
@NodeBuilder _ build: () -> any Node
|
||||
) {
|
||||
self.color = color
|
||||
self.styles = styles
|
||||
self.node = build()
|
||||
}
|
||||
|
||||
public var body: some Node {
|
||||
let output = styles.reduce(node.render()) { $0.applyingStyle($1) }
|
||||
guard let color else { return output }
|
||||
return output.applyingColor(color)
|
||||
}
|
||||
|
||||
}
|
||||
31
Sources/CliDoc/Nodes/Note.swift
Normal file
31
Sources/CliDoc/Nodes/Note.swift
Normal file
@@ -0,0 +1,31 @@
|
||||
public struct Note<Label: Node, Content: Node>: Node {
|
||||
|
||||
var separator: String
|
||||
var label: Label
|
||||
var content: Content
|
||||
|
||||
public init(
|
||||
separator: String = " ",
|
||||
@NodeBuilder label: () -> Label,
|
||||
@NodeBuilder content: () -> Content
|
||||
) {
|
||||
self.separator = separator
|
||||
self.label = label()
|
||||
self.content = content()
|
||||
}
|
||||
|
||||
public var body: some Node {
|
||||
Group([label, content], separator: separator)
|
||||
}
|
||||
}
|
||||
|
||||
public extension Note where Label == CliDoc.Label {
|
||||
|
||||
init(
|
||||
separator: String = " ",
|
||||
label: String = "NOTE:",
|
||||
@NodeBuilder content: () -> Content
|
||||
) {
|
||||
self.init(separator: separator, label: { CliDoc.Label(label) }, content: content)
|
||||
}
|
||||
}
|
||||
20
Sources/CliDoc/Nodes/ShellCommand.swift
Normal file
20
Sources/CliDoc/Nodes/ShellCommand.swift
Normal file
@@ -0,0 +1,20 @@
|
||||
public struct ShellCommand<Content: Node>: Node {
|
||||
|
||||
public var symbol: String
|
||||
public var content: Content
|
||||
|
||||
public init(
|
||||
symbol: String = "$",
|
||||
@NodeBuilder content: () -> Content
|
||||
) {
|
||||
self.symbol = symbol
|
||||
self.content = content()
|
||||
}
|
||||
|
||||
public var body: some Node {
|
||||
Group(separator: " ") {
|
||||
symbol
|
||||
content
|
||||
}
|
||||
}
|
||||
}
|
||||
66
Tests/CliDocTests/CliDocTests.swift
Normal file
66
Tests/CliDocTests/CliDocTests.swift
Normal file
@@ -0,0 +1,66 @@
|
||||
@testable import CliDoc
|
||||
@preconcurrency import Rainbow
|
||||
import Testing
|
||||
|
||||
let setupRainbow: Bool = {
|
||||
Rainbow.enabled = true
|
||||
Rainbow.outputTarget = .console
|
||||
return true
|
||||
}()
|
||||
|
||||
@Test
|
||||
func checkStringBuilder() {
|
||||
let group = Group {
|
||||
Label("foo:")
|
||||
"bar"
|
||||
}
|
||||
|
||||
#expect(group.render() == "foo: bar")
|
||||
|
||||
#expect(setupRainbow)
|
||||
let coloredLabel = group.labelColor(.green)
|
||||
#expect(
|
||||
coloredLabel.render() == """
|
||||
\("foo:".green) bar
|
||||
""")
|
||||
}
|
||||
|
||||
@Test
|
||||
func checkLabelColorModifier() {
|
||||
#expect(setupRainbow)
|
||||
let group = Group(separator: "\n") {
|
||||
Label("Foo:")
|
||||
Group(separator: "\n") {
|
||||
"Bar"
|
||||
Label("baz:")
|
||||
Group {
|
||||
Label("Bang")
|
||||
"boom"
|
||||
}
|
||||
.labelColor(.green)
|
||||
}
|
||||
}
|
||||
.labelColor(.blue)
|
||||
// .labelColor(.green)
|
||||
print(type(of: group))
|
||||
print(type(of: group.body))
|
||||
|
||||
let expected = """
|
||||
\("Foo:".blue)
|
||||
Bar
|
||||
\("baz:".blue)
|
||||
\("Bang".green) boom
|
||||
"""
|
||||
#expect(group.render() == expected)
|
||||
}
|
||||
|
||||
@Test
|
||||
func checkNote() {
|
||||
#expect(setupRainbow)
|
||||
let note = Note {
|
||||
"My note..."
|
||||
}
|
||||
.labelColor(.yellow)
|
||||
|
||||
#expect(note.render() == "\("NOTE:".yellow) My note...")
|
||||
}
|
||||
Reference in New Issue
Block a user