Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added methods to box Swift collections into WinRT-compatible collections #130

Merged
merged 1 commit into from
Apr 8, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 3 additions & 1 deletion Generator/Sources/ProjectionModel/SupportModules.swift
Original file line number Diff line number Diff line change
Expand Up @@ -128,7 +128,9 @@ extension SupportModules.WinRT {
]
internal static let windowsFoundationCollections = [
"IIterable`1",
"IIterator`1"
"IIterator`1",
"IVector`1",
"IVectorView`1"
]
}

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
extension Array {
public func toWinRTIVector(elementEquals: @escaping (Element, Element) -> Bool) -> ArrayVector<Element> {
ArrayVector(self, elementEquals: elementEquals)
}
}

extension Array where Element: Equatable {
public func toWinRTIVector() -> ArrayVector<Element> {
ArrayVector(self, elementEquals: ==)
}
}

/// Wraps a Swift array into a type implementing WinRT's Windows.Foundation.Collections.IVector<T>.
public class ArrayVector<T>: WinRTExport<IInspectableProjection>,
WindowsFoundationCollections_IVectorProtocol, WindowsFoundationCollections_IVectorViewProtocol {
public var array: [T]
public var elementEquals: (T, T) -> Bool

init(_ array: [T], elementEquals: @escaping (T, T) -> Bool) {
self.array = array
self.elementEquals = elementEquals
}

public func _size() throws -> UInt32 {
UInt32(array.count)
}

public func first() throws -> WindowsFoundationCollections_IIterator<T> {
SequenceIterator(array.makeIterator())
}

public func getAt(_ index: UInt32) throws -> T {
array[Int(index)]
}

public func getView() throws -> WindowsFoundationCollections_IVectorView<T> {
array.toWinRTIVectorView(elementEquals: elementEquals)
}

public func indexOf(_ value: T, _ index: inout UInt32) throws -> Bool {
if let foundIndex = array.firstIndex(where: { elementEquals($0, value) }) {
index = UInt32(foundIndex)
return true
} else {
index = 0
return false
}
}

public func setAt(_ index: UInt32, _ value: T) throws {
array[Int(index)] = value
}

public func insertAt(_ index: UInt32, _ value: T) throws {
array.insert(value, at: Int(index))
}

public func removeAt(_ index: UInt32) throws {
array.remove(at: Int(index))
}

public func append(_ value: T) throws {
array.append(value)
}

public func removeAtEnd() throws {
array.removeLast()
}

public func clear() throws {
array.removeAll()
}

public func getMany(_ startIndex: UInt32, _ items: [T]) throws -> UInt32 {
throw HResult.Error.notImpl // TODO(#31): Implement out arrays
}

public func replaceAll(_ items: [T]) throws {
array = items
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
extension Collection where Index == Int {
public func toWinRTIVectorView(elementEquals: @escaping (Element, Element) -> Bool) -> WindowsFoundationCollections_IVectorView<Element> {
CollectionVectorView(self, elementEquals: elementEquals)
}
}

extension Collection where Index == Int, Element: Equatable {
public func toWinRTIVectorView() -> WindowsFoundationCollections_IVectorView<Element> {
CollectionVectorView(self, elementEquals: ==)
}
}

fileprivate class CollectionVectorView<C: Collection>: WinRTExport<IInspectableProjection>,
WindowsFoundationCollections_IVectorViewProtocol
where C.Index == Int {
public typealias T = C.Element

public var collection: C
public var elementEquals: (T, T) -> Bool

init(_ collection: C, elementEquals: @escaping (T, T) -> Bool) {
self.collection = collection
self.elementEquals = elementEquals
}

public func _size() throws -> UInt32 {
UInt32(collection.count)
}

public func first() throws -> WindowsFoundationCollections_IIterator<T> {
SequenceIterator(collection.makeIterator())
}

public func getAt(_ index: UInt32) throws -> C.Element {
collection[Int(index)]
}

public func indexOf(_ value: C.Element, _ index: inout UInt32) throws -> Bool {
if let foundIndex = collection.firstIndex(where: { elementEquals($0, value) }) {
index = UInt32(foundIndex)
return true
} else {
index = 0
return false
}
}

public func getMany(_ startIndex: UInt32, _ items: [C.Element]) throws -> UInt32 {
throw HResult.Error.notImpl // TODO(#31): Implement out arrays
}
}
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
// Public types and protocols in here must be compatible with what the code generator would emit.

/// Exposes an iterator that supports simple iteration over a collection of a specified type.
public typealias WindowsFoundationCollections_IIterable<T> = any WindowsFoundationCollections_IIterableProtocol<T>

Expand Down
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
// Public types and protocols in here must be compatible with what the code generator would emit.

/// Supports simple iteration over a collection.
public typealias WindowsFoundationCollections_IIterator<T> = any WindowsFoundationCollections_IIteratorProtocol<T>

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
// Public types and protocols in here must be compatible with what the code generator would emit.

/// Represents a random-access collection of elements.
public typealias WindowsFoundationCollections_IVector<T> = any WindowsFoundationCollections_IVectorProtocol<T>

/// Represents a random-access collection of elements.
public protocol WindowsFoundationCollections_IVectorProtocol<T>: WindowsFoundationCollections_IIterableProtocol {
associatedtype T

/// Returns the item at the specified index in the vector.
/// - Parameter index: The zero-based index of the item.
/// - Returns: The item at the specified index.
func getAt(_ index: Swift.UInt32) throws -> T

/// Returns an immutable view of the vector.
/// - Returns: The view of the vector.
func getView() throws -> WindowsFoundationCollections_IVectorView<T>

/// Retrieves the index of a specified item in the vector.
/// - Parameter value: The item to find in the vector.
/// - Parameter index: If the item is found, this is the zero-based index of the item; otherwise, this parameter is 0.
/// - Returns: **true** if the item is found; otherwise, **false**.
func indexOf(_ value: T, _ index: inout Swift.UInt32) throws -> Swift.Bool

/// Sets the value at the specified index in the vector.
/// - Parameter index: The zero-based index at which to set the value.
/// - Parameter value: The item to set.
func setAt(_ index: Swift.UInt32, _ value: T) throws

/// Inserts an item at a specified index in the vector.
/// - Parameter index: The zero-based index.
/// - Parameter value: The item to insert.
func insertAt(_ index: Swift.UInt32, _ value: T) throws

/// Removes the item at the specified index in the vector.
/// - Parameter index: The zero-based index of the vector item to remove.
func removeAt(_ index: Swift.UInt32) throws

/// Appends an item to the end of the vector.
/// - Parameter value: The item to append to the vector.
func append(_ value: T) throws

/// Removes the last item from the vector.
func removeAtEnd() throws

/// Removes all items from the vector.
func clear() throws

/// Gets a collection of items from the vector beginning at the given index.
/// - Parameter startIndex: The zero-based index to start at.
/// - Parameter items: An array to copy the items into.
/// - Returns: A status code indicating the result of the operation.
func getMany(_ startIndex: Swift.UInt32, _ items: [T]) throws -> Swift.UInt32

/// Replaces all the items in the vector with the specified items.
/// - Parameter items: The collection of items to add to the vector.
func replaceAll(_ items: [T]) throws

/// Gets the number of items in the vector.
/// - Returns: The number of items in the vector.
func _size() throws -> Swift.UInt32
}

extension WindowsFoundationCollections_IVectorProtocol {
/// Gets the number of items in the vector.
/// - Returns: The number of items in the vector.
public var size: Swift.UInt32 {
try! self._size()
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
// Public types and protocols in here must be compatible with what the code generator would emit.

/// Represents an immutable view into a vector.
public typealias WindowsFoundationCollections_IVectorView<T> = any WindowsFoundationCollections_IVectorViewProtocol<T>

/// Represents an immutable view into a vector.
public protocol WindowsFoundationCollections_IVectorViewProtocol<T>: WindowsFoundationCollections_IIterableProtocol {
associatedtype T

/// Returns the item at the specified index in the vector view.
/// - Parameter index: The zero-based index of the item.
/// - Returns: The item at the specified index.
func getAt(_ index: Swift.UInt32) throws -> T

/// Retrieves the index of a specified item in the vector view.
/// - Parameter value: The item to find in the vector view.
/// - Parameter index: If the item is found, this is the zero-based index of the item; otherwise, this parameter is 0.
/// - Returns: **true** if the item is found; otherwise, **false**.
func indexOf(_ value: T, _ index: inout Swift.UInt32) throws -> Swift.Bool

/// Gets a collection of items from the vector view beginning at the given index.
/// - Parameter startIndex: The zero-based index to start at.
/// - Parameter items: An array to copy the items into.
/// - Returns: A status code indicating the result of the operation.
func getMany(_ startIndex: Swift.UInt32, _ items: [T]) throws -> Swift.UInt32

/// Gets the number of items in the vector view.
/// - Returns: The number of items in the vector view.
func _size() throws -> Swift.UInt32
}

extension WindowsFoundationCollections_IVectorViewProtocol {
/// Gets the number of items in the vector view.
/// - Returns: The number of items in the vector view.
public var size: Swift.UInt32 {
try! self._size()
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
extension Sequence {
/// Converts a Swift sequence to a WinRT IIterable.
/// - Parameter sequence: The sequence to convert.
/// - Returns: The converted IIterable.
public func toWinRTIIterable() -> WindowsFoundationCollections_IIterable<Element> {
SequenceIterable(self)
}
}

fileprivate class SequenceIterable<S: Sequence>: WinRTExport<IInspectableProjection>, WindowsFoundationCollections_IIterableProtocol {
typealias T = S.Element

private let sequence: S

init(_ sequence: S) {
self.sequence = sequence
}

func first() throws -> WindowsFoundationCollections_IIterator<S.Element> {
SequenceIterator(sequence.makeIterator())
}
}

internal class SequenceIterator<I: IteratorProtocol>: WinRTExport<IInspectableProjection>, WindowsFoundationCollections_IIteratorProtocol {
typealias T = I.Element

private var iterator: I
private var current: T?

init(_ iterator: I) {
self.iterator = iterator
self.current = self.iterator.next()
}

func _hasCurrent() throws -> Bool { current != nil }

func _current() throws -> T {
guard let current else { throw HResult.Error.illegalMethodCall }
return current
}

func moveNext() throws -> Bool {
current = iterator.next()
return current != nil
}

func getMany(_ items: [I.Element]) throws -> UInt32 {
throw HResult.Error.notImpl // TODO(#31): Implement out arrays
}
}
Loading