React-inspired framework for building component-based user interfaces in Swift.
Features | |
---|---|
π€ | Completely native - write your app in Swift |
π | Declarative - define your UI using markup |
π¦ | Components - encapsulate functionality into reusable chunks |
π | Layout - Flexbox for layout, just like on the web |
π | State - automatically flush state changes to UI |
β | Diffing - the minimum set of updates are flushed to UI |
π | Performance - diffing and layout are done on background threads |
β² | Animation - built-in support for animating any property |
You define your UI using a simple markup language, which is inspired by HTML and CSS. This UI definition is rendered into a tree of native elements.
<template>
<style>
#container > .button {
color: #000;
}
#container > .button-selected {
color: #f00;
}
</style>
<box id="container">
<text text="$properties.title" />
<text text="Click me!" onTap="handleClick" classNames="$textClasses" />
</box>
</template>
Functionality and state is encapsulated into components, which do things like handle user events and flush state changes to the UI. Components have strongly typed State
and Properties
values, that are used to figure out what ends up getting pushed out to UIKit.
struct ComponentState: State {
var selected: Bool?
}
struct ComponentProperties: Properties {
var core = CoreProperties()
var title: String? = "This is a default title"
}
class MyComponent: CompositeComponent<ComponentState, ComponentProperties, UIView> {
// Stored properties on the component are made available to template.
var textClasses: String?
// As are functions, referenced by their selector name.
@objc func handleClick() {
updateComponentState { state in
state.selected = !state.selected
}
}
override func render() -> Element {
textClasses = state.selected ? "button" : "button-selected"
return render("http://localhost:8000/Component.xml")
}
}
Rendering components is as easy as calling a render
function, which asynchronously computes and flushes a component to the supplied container view.
override func viewDidLoad() {
super.viewDidLoad()
UIKitRenderer.render(component(MyComponent.self), container: self.view, context: self) { component in
self.component = component
}
}
See the included Example project for more examples of how to use TemplateKit.
Because you like writing your apps completely in Swift. TemplateKit is fully native and compiled.
Writing user interfaces in a declarative style makes it easier to reason about how model data and user actions affect what gets rendered. Out-of-the-box support for XML. Extensible if you want to add your own template format (e.g., protocol buffers).
Components make it easy to encapsulate application functionality into re-usable building blocks. These blocks can then be composed to create more complex interfaces.
Flexbox-based layout primitives allow developers to use the the same expressive layout system available in modern browsers.
All layout computation, text sizing, tree diffing, image decoding is performed in the background. This keeps the main thread available for responding to user actions. Only the absolute minimum set of changes needed to update the view hierarchy are actually flushed to the rendered views.
Use stylesheets to style components, just like you do on the web.
Animate layout, style and arbitrary properties using an intuitive API.
Automatically reload changes to user interfaces without having to re-build binaries or restart your application. Usable in both development and production environments.
Add custom components, custom native views, custom template loading schemes and more.
Plug it in anywhere you want to render a view in your application. Plays nicely with the rest of your app.
You can install Carthage with Homebrew using the following command:
$ brew update
$ brew install carthage
Add the following line to your Cartfile:
github "mcudich/TemplateKit"
Run carthage update
, then make sure to add TemplateKit.framework
, CSSLayout.framework
, and CSSParser.framework
to "Linked Frameworks and Libraries" and "copy-frameworks" Build Phases.
CocoaPods is a dependency manager for Cocoa projects. You can install it with the following command:
$ gem install cocoapods
To integrate TemplateKit into your Xcode project using CocoaPods, specify it in your Podfile
:
source 'https://github.com/CocoaPods/Specs.git'
platform :ios, '10.0'
use_frameworks!
target '<Your Target Name>' do
pod 'TemplateKit', '~> 0.1.0'
end
Then, run the following command:
$ pod install
- iOS 9.3+
- Xcode 8.0+
- Swift 3.0+
- If you found a bug, open an issue.
- If you have a feature request, open an issue.
- If you want to contribute, submit a pull request.
At its core, TemplateKit is comprised of Element
and Node
instances. Elements are used to describe trees of nodes, which can be anything that implements the Node
interface. Nodes are used to vend out and manage view hierarchies.
Out of the box, there are several Node
implementations that make it easy to set up UI hierarchies: Component
, ViewNode
, and a set of native controls like buttons, text labels, text fields and so on.
Building a component is as simple as subclassing Component
, overriding its render()
function, and deciding the set of properties it might accept and use as part of rendering. render()
simply needs to return a Template
, which can be constructed programmatically, or via an XML document (or other custom payload). When it comes time to render your component into a view, you simply call UIKitRenderer.render
, and pass in the view that should contain your component's rendered output. This will in turn call render()
on your component instance, compute the layout and styles for the view tree, build this tree and then apply the layout and styles to it as appropriate.
When it comes time to update your component's state, you can call updateState
from within your component implementation. This function receives a function that is passed the current state value (each Component
can declare a State
type, in the same way it declares a Properties
type). This function in turn enqueues an update to the component, which will cause it to re-render, taking into account whatever changes were made to the state. This update is intelligent, and compares the current incarnation of the rendered view tree against the proposed element tree. Only the deltas between these two are flushed out to the view layer.
If there are parts of your UI that are easier to deal with as plain UIViews
, TemplateKit provides a simple abstraction Node
called ViewNode
that allows you to include these "opaque" views as part of any TemplateKit-managed tree. TemplateKit stays out of the way, and simply sets the frame
of these views for you, so they sit nicely within in whatever UI tree you've composed.
TemplateKit provides UITableView
and UICollectionView
subclasses which are able to load, and asynchronously size and render Components
into cells with just a little bit of configuration. Tables and collections can be used via Table
and Collection
components, or simply wrapped as ViewNode
instances. The Table
and Collection
components have built-in support for diffing, so that data-source updates result in the minimum set of operations required to have the respective UIKit views reflect data changes. See Diff.swift for more information.
TemplateKit is implemented in Swift (and a bit of C). If you like writing entirely in Swift, then this framework might be for you.
React Native relies on a very well-tested library (React), and has been shipping in popular apps for some time now. This means it probably has way fewer rough edges, has sorted out many performance issues TemplateKit has yet to face, and so on.
A lot.
There's no AppKit support yet (though it would be straightforward to add). Lots of tests have yet to be written. Performance testing has yet to be done. The entirety of the applicable CSS spec is not supported. Animation features are rudimentary at best. Many gesture types need to be added. And much more.
If you'd like something added, please file a feature request or send a pull request!
If TemplateKit isn't exactly what you're looking for, check out these other great projects!