DeclarativeTextKit

0.3.0

Swift DSL to perform mutations on e.g. NSTextViews in a declarative style.
CleanCocoa/DeclarativeTextKit

What's New

Fully Undoable command stacks

2024-06-01T15:14:48Z

What's Changed

  • Buffer.evaluate { ... } is a new Result Builder block-based entry point
  • undo() and redo() restore selection state in both directions if required: direct modifications don't alter the selection state, and direct user-based selection also is not undoable, but if you run a command group through Buffer.evaluate { ... } that's undoable, you'll get complete context restoration including selection for free
  • CI: Run tests via GitHub Actions by @ikelax in #5

New Contributors

  • @ikelax made their first contribution in #5

Full Changelog: 0.1.2...0.3.0

Declarative Text Kit

Build Status Platforms Documentation

A Swift DSL to make modifications to text buffers.

Goals

Express your intent of multiple changes in one swoop.

We replace procedural code with a declarative DSL for this.

For changes to a text view on the user's behalf with proper undo support, you often need to

  1. modify the selected range, e.g. to word boundaries,
  2. check whether the selected range is modifiable (abort if it isn't),
  3. perform changes to the text -- starting from back to front to not invalidate indices of upcoming changes,
  4. notify the text view that changes have been completed in the range.

If you want to modify a piece of a text in multiple places at once, the index reversal alone makes code harder to understand.

What This Looks Like

Expand the user's selection to full lines, then wrap the text in GitHub-flavored Markdown fenced code blocks and put the insertion point after the opening triple backticks:

buffer.evaluate {
    // Expand selection to the whole block (full lines).
    Select(LineRange(selectedRange)) { lineRange in
        // In that range, attempt to wrap the selected text
        // in two lines to make it a code block.
        // (Abort if the text view doesn't permit changes.)
        Modifying(lineRange) { rangeToWrap in
            Insert(rangeToWrap.location) { Line("```") }
            Insert(rangeToWrap.endLocation) { Line("```") }
        }

        // Move insertion point to the
        // position after the opening backticks
        Select(lineRange.location + length(of: "```"))
    }
}

Documentation

Approach

We operate on the abstraction of a Buffer to perform changes.

This enables usage of the declarative API on multiple buffers at once without having to put the text into a UI component to render.

A NSTextView is a buffer. You can use this declarative API to make changes to text views on screen.

You can also use purely in-memory buffers for text mutations of things you don't want to render. This allows you to read multiple files into buffers in your app and use the declarative API to change their contents, while only rendering a single selected file in a text view.

License

Copyright © 2024 Christian Tietze. All rights reserved. Distributed under the MIT License.

See the LICENSE file.

Description

  • Swift Tools 5.9.0
View More Packages from this Author

Dependencies

  • None
Last updated: Thu Jan 23 2025 06:46:10 GMT-1000 (Hawaii-Aleutian Standard Time)