Fellmonger is an example project that showcases a native desktop application developed using Swift 6, compatible with both macOS and Windows platforms. Itโs a native LLM client that supports Ollama, OpenAI, and Anthropic LLMs.
๐ This project is meant to be a learning resource for exploring cross-platform macOS/Windows Swift desktop application development. While primarily for my own learning, I hope it can benefit others too. ๐
The project is complex enough to represent a real application but is still at a prototype level. Many improvements can be made to make it production-ready, including UI/icon polish, feature enhancements, performance optimizations, better error handling and installer improvements.
Architectural decisions involve trade-offs. If you find these patterns useful, great! If you think this is overengineering or not the right tool for the job, thatโs okay too. Use whatever fits your development process and team. The most important thing is to ship cool stuff to your users sustainably.
Feel free to create PRs to improve this project so we can discover and document useful patterns!
You can select the active LLM provider and specific model name from the app Settings. At least one provider must be configured for the app to work properly.
Model | Notes | Default model |
---|---|---|
Ollama | Install Ollama and pull the desired model. By default, Fellmonger will try to use localhost to reach it but you can configure the URL from the Settings. |
llama3.1 |
OpenAI | Get an OpenAI service account key and configure it in Settings. | gpt-4o |
Anthropic | Get an Anthropic service account key and configure it in Settings. | claude-3-5-sonnet-20240620 |
The cross-platform application was implemented in Swift 6.
- Swift Package Manager
- Swift Concurrency architecture
- Swift Macros
- For Core:
- For macOS:
- Xcode 16.1 Beta targeting macOS 14
- SwiftUI
- For Windows:
From a macOS machine:
- Pull the code
- Open the macOS project in Xcode
- Fix the potential signing issues you might have (select your user)
- Build and run
You will need a Windows machine or a Windows VM. Only x64 is supported, so if you use a Windows VM from a Mac, it will need to be an x64 machine (not ARM).
The requirements are the same than The Browser Company sample Windows app:
- Install latest Swift SDK from thebrowsercompany/swift-build
- Visual Studio Community with C++ build tools
- Make sure to have the appropriate version of the Windows App Runtime installed as mentioned here
VSCode is the editor of choice for developing Windows apps on Swift. You can install it from https://code.visualstudio.com/download. Install the Swift extension from Swift Server Work Group.
โ ๏ธ In order for the extension to load the project, you will need to open the Windows Application Package.swift in VSCode first every time you restart VSCode.
-
Building: The build of the Windows applications is done through SPM. This can be done on the command line with
swift build
or in Visual Studio Code withCtrl+Shift+B
. -
Debugging: Debugging in VSCode is supported through LLDB. You can simply press
F5
or navigate to theRun and Debug
(Ctrl+Shift+D
) pane.
If you need more guidance about how to setup Windows development, check this blog post about it.
I remember I had to install Python (Swift for Windows dependency) and set up some environment variables after hitting some errors. If you really want to try to build it and these steps are not enough to have it working, please open an issue on this project and I will be happy to add more detailed steps to this guide.
This project explores various architectural patterns and practices for developing a cross-platform Swift application.
The project is organized into three main directories:
- Core: Contains the shared code between the Windows and macOS applications.
- MacOS: Contains macOS-specific code.
- Windows: Contains Windows-specific code.
Each directory is structured into several Swift packages:
- Core: The core is small and could fit in a single package well by just using directories instead of subpackages. However, I wanted to split it into packages to excercise that approach which might be useful for more complex applications.
- SwiftExtensions: Swift utility methods and types.
- Settings: Settings model and view models.
- Chat: LLM client implementations and view models.
- MacOS: It's an Xcode project for a macOS application. It contains everything that can't be stored in a Swift package. All of it'
s code was pushed to subpackages.
- MacOSSwiftExtensions: macOS Swift extensions.
- MacOSSettings: macOS views for settings.
- MacOSChat: macOS views for chat.
- MacOSRoot: Root package for the macOS application. This is the only package the Xcode project imports.
- Windows:
- WinUIExtensions: View model data binding utilities. Includes macros for that purpose.
- Application: Single package implementation for the entire app. Buildable with Swift for Windows and contains dependencies on Windows frameworks. Derived from The Browser Company sample Windows app. It could be split down in more packages as the macOS app if needed but I decided to keep it in a single package to constrast it with the multiple package approach. Simplicity is always good, especially if you are not ripping off the benefits of a more modular architecture, for example, by adding Dev Apps in a large team/project.
In the context of this project, Modules are a variation of the Composition Root pattern. Modules allow Swift packages to export living instances of types, which helps with encapsulation and lifecycle management.
Key points about the Module pattern:
- Swift packages may contain a @MainActor class with the same name as the package, representing the module.
- Modules handle dependency injection manually, making external frameworks unnecessary.
- Modules are retained by clients, ensuring that infrastructure objects are managed by the client while being hidden internally.
- Modules have an init method that may include other modules as dependencies.
Both SettingsModule
and ChatModule
are examples of this pattern. Both macOS and Windows applications link these modules to their respective views.
This project employs a variation of the MVVM with Observable pattern.
In this approach:
- Models are platform independent and might be one of two types:
- Data exchange models: Value types (
structs
andenums
) for data crossing the boundaries of the application - Application models: Value types and classes for internal application state that needs to be reused from multiple view models.
- Data exchange models: Value types (
- View Models are headless views that encapsulate UI state and behavior, using models for input or output. These are
@Observable @MainActor classes
and platform-independent. - Views only know about the view modelsโ surface, implemented with
SwiftUI
on macOS andWinUI3
on Windows.
This approach enables the reuse of models and view models across macOS and Windows applications. Models are implementation details of view models, so they are not necessarily exposed. The trade off here is consistency and reusability over flexibility and autonomy in the different platforms.
Windows Views are implemented using a hierarchy of WinUI FrameworkElements
. The View layer is implemented following a pattern very similar to AppKit development with code-only view controllers: there's a view
property on the view (controller) that exposes the root FrameworkElement
and there's one private(set) lazy var
per specific internal view. Finally, view model properties are bound to the view by using withObservationTracking
for detecting changes in one direction and FrameworkElement event handlers for the other. This project explored using generic methods and custom macros to make it easier to set up the glue code for bindings in a way that's similiar to how data binding works in Windows applications.
Rendering view models from SwiftUI is straightforward with a SwiftUI View. View/ViewModel binding is done implictly by using the @Observable
view model from the view body
.
When it's not straightforward to use the default view model implementation in a SwiftUI preview, a protocol can be used for the view model. In that case, there would be two concrete implementations for the view model in the project: a mock for the previews and the real one. A custom @BindableProtocol
implementation was implemented for that purpose. Using existential types as view dependencies might theoretically cause performance penalties but I haven't found evidence of this to actually become a real problem in this context.
MVVM with Observable on SwiftUI might feel a bit strange, specially coming from the traditional SwiftUI data flow which encourages the use of @State and value types instead of classes. In a way, this feels like opting out of the efficient SwiftUI diffing mechanism and doing it manually in a more hand crafted way with classes, i.e. more work for the developer. Also, many SwiftUI native tools (@Environment
, @AppStorage
, @Query
) don't work in this approach. However, this approach has some benefits that might make it worth for complex applications:
- View Models can be reused in macOS and Windows
- View and View Model concerns are better defined so there's a better separation of responsibilities and a more predictable codebase overall
- Views can be easily set up in the exact desired state for previews
- Scope of change when data changes is smaller so this should be more performant.
ChatView (Windows), ChatView (macOS), ChatViewModel and ChatMessage are examples of Views, View Models, and Models respectively.
This project leverages Dependency Injection extensively:
- Types focus on their primary responsibilities, making the codebase more maintainable and easier to test.
- Dependency Injection is used consistently throughout the stack:
- View Models are injected into Views. This allows Views to focus only on the platform specific rendering of the View Models.
- In SwiftUI previews, mock view models can be injected allowing the developer to render the previews in the exact specific desired state easily.
- Services are injected into View Models and other Services.
- Modules are injected into other modules.
- View Models are injected into Views. This allows Views to focus only on the platform specific rendering of the View Models.
Dependency injection ensures that types donโt create their dependencies, and instead assume that dependencies are created and passed to them by someone else, typically in Modules.
Note we are not using @Environment
from SwiftUI at all. We can't do it in the first place since we want reusable cross platform View Models.
Also, this flavor of dependency injection has some nice properties:
- Type safe
- No errors at runtime
- No requirements of default values (that might need their dependencies as well)
- Compiler will force you to create the right dependency graph... otherwise it won't compile.
- No external dependency injection framework is used.
- Lifecycle of objects is explicit by using the Modules pattern.
Service is an overloaded term in software developement but within this project it has specific semantics. Services are abstractions for interacting with remote systems or persistence. They provide a straightforward API that hides the details of these interactions.
- Services are usually implemented as
actors
.- This allows embracing the Swift Concurrency architecture which enables better multithreading usage in a safe way.
- Other dependencies or configuration might be injected into them.
- Services are injected into View Models
- Services are usually stateless gateways for data operations. The mutable state of the application is usually stored in View Models or Application Models.
- Models are usually implemented as value types that go through the API of Services.
OllamaService, OpenAIService, AnthropicService and ConfigurationPersistenceService are examples of services.
I wrote down a list of issues that I found while pushing this project forward. While they caused a bit of additional work, all of them can be worked around.
URLSession
is part ofFoundationNetworking
instead ofFoundation
. Conditional imports are required.URLSession
lacks thebytes
member inFoundationNetworking
. Custom implementations are necessary.URLSessionDelegate
behaves differently on macOS and Windows. Various approaches were explored to ensure consistent behavior across both platforms.UserDefaults
is not persistent in Swift Foundation on Windows. Custom settings persistence was implemented using UserDefaults on macOS and the Windows Registry on Windows.
The fixes become straightforward once you understand how to do them but it might be a bit intimidating at first.
I also found Alamofire comments about swift-corelibs-foundation reliablity and left me a bit concerned. I didn't hit blocking issues myself though.
CommandBar
is not available by default in swift-winui, but can be added to projections.json.- The community-toolkit, an Open Source framework for common components like Settings views, is not available to Swift because it is implemented in C#. I had to implement some of these components manually in Swift.
- swift-winui doesn't support ARM yet. I had to use an Intel-based Windows machine or Intel Mac. ARM support is expected soon.
- Swift Testing import isn't found on Swift for Windows.
- Macros tests donโt work due to multiple entry points error in Windows.
- VScode Swift extension doesn't support ARM yet. This would be useful to develop from Apple ARM machines in Parallels. This would make it way easier and faster to code in Swift for Windows and macOS at the same time.
- Sometimes auto complete doesn't work well. It might show options that don't apply for the current context.
- Code formatting adds unnecessary new lines between imports and reorders comments undesirably.
- I needed to restart VScode every time I deleted a file due to errors
- Having to open the Windows application Package.swift file every time VScode is reopened to be able to build is a bit annoying.
- When debugging the app, a command line window is opened. However, logs don't show up in there if the debugger is attached. The only way to see the logs is to manually launch the debug build.
These are small things that should get better over time. They might have been fixed by the time you read this.
I didnโt explore in detail creating high quality installers, how code signing works for Swift Windows applications and how to distribute updates. If you have insights into this, please let me know!
The Browser Company has shown the path to use Swift for UI development on Windows. While there are areas for improvement in the developer experience, itโs a viable option for teams with strong Swift focus and backgrounds.
The patterns explored in this project have proven useful in refining and refactoring the application. However, there are many valid approaches to development โthe key is delivering reliable and sustainable solutions to users.
This experiment has been incredibly rewarding! Iโd love to hear your thoughts. Thanks for reading! ๐