feat: Initial commit

This commit is contained in:
2024-12-02 17:04:28 -05:00
commit 22dfc6ce51
18 changed files with 505 additions and 0 deletions

7
.editorconfig Normal file
View 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
View 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
View 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
View 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
View 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
View 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"]
)
]
)

View 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
}
}
}
}

View 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
View 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 }
}

View 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() ?? ""
}
}

View 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()
}
}

View 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)
}
}

View 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)
}
}

View 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)
}
}

View 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)
}
}

View 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)
}
}

View 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
}
}
}

View 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...")
}