Compare commits

...

30 Commits

Author SHA1 Message Date
Arnaud Dorgans a576a83530 Merge pull request #30 from lm2343635/master
Support RxSwift 6+
2021-07-10 19:20:17 +02:00
Meng Li 4949a88839 Fix for build error 2021-07-09 23:53:25 +09:00
Meng Li 11e0c80586 Update RxSwift to 6.0 2021-07-09 23:51:29 +09:00
Arnaud Dorgans 0037147caf Merge pull request #16 from cdann-leafly/master
Update Package.swift manifest to define RxInfiniteLayout product
2020-02-18 00:13:59 +01:00
Chris D'Annunzio 35543f6b6c Update Package.swift manifest to define RxInfiniteLayout product and target, allowing RxSwift-related classes and extensions to be imported via Swift Package Manager. 2020-02-03 11:46:55 -08:00
Arnaud Dorgans 227269190f Merge pull request #13 from arnauddorgans/release/0.4.2
0.4.2
2019-09-05 18:14:46 +02:00
Arnaud Dorgans 38f6b056f1 Update README.md 2019-09-05 18:13:40 +02:00
Arnaud Dorgans 90a0ce56da release(0.4.2) @dlbuckley pr, Swift Package Manager 2019-09-05 18:02:11 +02:00
Arnaud Dorgans 0e9d75b59d Update README.md 2019-09-05 16:42:16 +02:00
Arnaud Dorgans da9c8b1372 Merge pull request #12 from dlbuckley/feature/add_willChangeCenteredIndexPath
Added willChangeCenteredIndexPath method to InfiniteCollectionViewDelegate protocol
2019-09-05 16:40:21 +02:00
Dale Buckley 83960eb567 rx fix 2019-09-05 12:15:27 +01:00
Dale Buckley cae11d5404 fixed a typo 2019-09-05 11:57:12 +01:00
Dale Buckley ab7324dc5b refactored based on code review comments by @arnauddorgans 2019-09-05 11:53:15 +01:00
Dale Buckley 8cd7e843af Added willChangeCenteredIndexPath method to InfiniteCollectionViewDelegate protocol 2019-08-29 15:08:01 +01:00
Arnaud Dorgans 519d724e17 Merge pull request #9 from lm2343635/master
RxSwift 5 compatible.
2019-05-22 09:29:53 +02:00
Meng Li d80114e454 Fix the errors. 2019-05-22 12:15:21 +09:00
Meng Li 1d4c698ab0 Update to RxSwift 5. 2019-05-22 12:11:52 +09:00
Arnaud Dorgans 0ebf480310 Merge pull request #8 from lm2343635/master
Swift 5.0 and Xcode 10.2.
2019-04-13 10:43:51 +02:00
Meng Li b178f69898 Update build script. 2019-04-09 15:20:18 +09:00
Meng Li 62782d83f9 Update .travis.yml 2019-04-09 14:49:44 +09:00
Meng Li b3038a90ec Update podspec. 2019-04-09 14:21:00 +09:00
Meng Li 03ba583c49 Swift 5.0 and Xcode 10.2. 2019-04-09 14:19:36 +09:00
Arnaud Dorgans f29e4b0716 travis fix 2019-03-20 14:15:49 +01:00
Arnaud Dorgans 41ce466ae3 bump swift version to 4.2, fix delegate proxy 2019-03-20 11:24:38 +01:00
Arnaud Dorgans c3688a9e1d fix isEnabled 2018-10-24 22:27:38 +02:00
Arnaud Dorgans 512885a166 bump & add public isEnabled property 2018-10-24 22:07:54 +02:00
Arnaud Dorgans 8fad95711e Update README.md 2018-08-29 18:17:52 +02:00
Arnaud Dorgans d6e792e3ec Update README.md 2018-08-29 18:15:39 +02:00
Arnaud Dorgans fcea3c29e7 bump 2018-08-12 14:58:45 +02:00
Arnaud Dorgans 7eb29bd82a fix scroll to index 2018-08-12 14:58:27 +02:00
27 changed files with 355 additions and 120 deletions
-1
View File
@@ -1 +0,0 @@
4.0
@@ -0,0 +1,7 @@
<?xml version="1.0" encoding="UTF-8"?>
<Workspace
version = "1.0">
<FileRef
location = "self:">
</FileRef>
</Workspace>
+7 -8
View File
@@ -2,13 +2,12 @@
# * http://www.objc.io/issue-6/travis-ci.html
# * https://github.com/supermarin/xcpretty#usage
osx_image: xcode7.3
language: objective-c
# cache: cocoapods
# podfile: Example/Podfile
# before_install:
# - gem install cocoapods # Since Travis is not always on latest version
# - pod install --project-directory=Example
osx_image: xcode10.2
language: swift
before_install:
- gem install cocoapods # Since Travis is not always on latest version
- pod repo update
- pod install --project-directory=Example
script:
- set -o pipefail && xcodebuild test -enableCodeCoverage YES -workspace Example/InfiniteLayout.xcworkspace -scheme InfiniteLayout-Example -sdk iphonesimulator9.3 ONLY_ACTIVE_ARCH=NO | xcpretty
- set -o pipefail && xcodebuild -workspace Example/InfiniteLayout.xcworkspace -scheme InfiniteLayout-Example -sdk iphonesimulator build CODE_SIGNING_REQUIRED=NO | xcpretty -c
- pod lib lint
@@ -262,18 +262,18 @@
isa = PBXProject;
attributes = {
LastSwiftUpdateCheck = 0830;
LastUpgradeCheck = 0830;
LastUpgradeCheck = 1020;
ORGANIZATIONNAME = CocoaPods;
TargetAttributes = {
607FACCF1AFB9204008FA782 = {
CreatedOnToolsVersion = 6.3.1;
DevelopmentTeam = CPE39ZT2AY;
LastSwiftMigration = 0900;
LastSwiftMigration = 1020;
};
607FACE41AFB9204008FA782 = {
CreatedOnToolsVersion = 6.3.1;
DevelopmentTeam = CPE39ZT2AY;
LastSwiftMigration = 0900;
LastSwiftMigration = 1020;
TestTargetID = 607FACCF1AFB9204008FA782;
};
};
@@ -283,6 +283,7 @@
developmentRegion = English;
hasScannedForEncodings = 0;
knownRegions = (
English,
en,
Base,
);
@@ -324,26 +325,26 @@
files = (
);
inputPaths = (
"${SRCROOT}/Pods/Target Support Files/Pods-InfiniteLayout_Example/Pods-InfiniteLayout_Example-frameworks.sh",
"${BUILT_PRODUCTS_DIR}/CocoaProxy/CocoaProxy.framework",
"${PODS_ROOT}/Target Support Files/Pods-InfiniteLayout_Example/Pods-InfiniteLayout_Example-frameworks.sh",
"${BUILT_PRODUCTS_DIR}/Differentiator/Differentiator.framework",
"${BUILT_PRODUCTS_DIR}/InfiniteLayout/InfiniteLayout.framework",
"${BUILT_PRODUCTS_DIR}/RxCocoa/RxCocoa.framework",
"${BUILT_PRODUCTS_DIR}/RxDataSources/RxDataSources.framework",
"${BUILT_PRODUCTS_DIR}/RxRelay/RxRelay.framework",
"${BUILT_PRODUCTS_DIR}/RxSwift/RxSwift.framework",
);
name = "[CP] Embed Pods Frameworks";
outputPaths = (
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/CocoaProxy.framework",
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/Differentiator.framework",
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/InfiniteLayout.framework",
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/RxCocoa.framework",
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/RxDataSources.framework",
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/RxRelay.framework",
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/RxSwift.framework",
);
runOnlyForDeploymentPostprocessing = 0;
shellPath = /bin/sh;
shellScript = "\"${SRCROOT}/Pods/Target Support Files/Pods-InfiniteLayout_Example/Pods-InfiniteLayout_Example-frameworks.sh\"\n";
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-InfiniteLayout_Example/Pods-InfiniteLayout_Example-frameworks.sh\"\n";
showEnvVarsInLog = 0;
};
BB5E5FB53E7C451064DBC9A2 /* [CP] Check Pods Manifest.lock */ = {
@@ -441,6 +442,7 @@
isa = XCBuildConfiguration;
buildSettings = {
ALWAYS_SEARCH_USER_PATHS = NO;
CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES;
CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
CLANG_CXX_LIBRARY = "libc++";
CLANG_ENABLE_MODULES = YES;
@@ -449,12 +451,14 @@
CLANG_WARN_BOOL_CONVERSION = YES;
CLANG_WARN_COMMA = YES;
CLANG_WARN_CONSTANT_CONVERSION = YES;
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
CLANG_WARN_EMPTY_BODY = YES;
CLANG_WARN_ENUM_CONVERSION = YES;
CLANG_WARN_INFINITE_RECURSION = YES;
CLANG_WARN_INT_CONVERSION = YES;
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
@@ -487,6 +491,7 @@
ONLY_ACTIVE_ARCH = YES;
SDKROOT = iphoneos;
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
SWIFT_VERSION = 5.0;
};
name = Debug;
};
@@ -494,6 +499,7 @@
isa = XCBuildConfiguration;
buildSettings = {
ALWAYS_SEARCH_USER_PATHS = NO;
CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES;
CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
CLANG_CXX_LIBRARY = "libc++";
CLANG_ENABLE_MODULES = YES;
@@ -502,12 +508,14 @@
CLANG_WARN_BOOL_CONVERSION = YES;
CLANG_WARN_COMMA = YES;
CLANG_WARN_CONSTANT_CONVERSION = YES;
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
CLANG_WARN_EMPTY_BODY = YES;
CLANG_WARN_ENUM_CONVERSION = YES;
CLANG_WARN_INFINITE_RECURSION = YES;
CLANG_WARN_INT_CONVERSION = YES;
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
@@ -532,6 +540,7 @@
MTL_ENABLE_DEBUG_INFO = NO;
SDKROOT = iphoneos;
SWIFT_OPTIMIZATION_LEVEL = "-Owholemodule";
SWIFT_VERSION = 5.0;
VALIDATE_PRODUCT = YES;
};
name = Release;
@@ -547,8 +556,7 @@
MODULE_NAME = ExampleApp;
PRODUCT_BUNDLE_IDENTIFIER = "org.cocoapods.demo.$(PRODUCT_NAME:rfc1034identifier)";
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_SWIFT3_OBJC_INFERENCE = Default;
SWIFT_VERSION = 4.0;
SWIFT_VERSION = 5.0;
TARGETED_DEVICE_FAMILY = 1;
};
name = Debug;
@@ -564,8 +572,7 @@
MODULE_NAME = ExampleApp;
PRODUCT_BUNDLE_IDENTIFIER = "org.cocoapods.demo.$(PRODUCT_NAME:rfc1034identifier)";
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_SWIFT3_OBJC_INFERENCE = Default;
SWIFT_VERSION = 4.0;
SWIFT_VERSION = 5.0;
TARGETED_DEVICE_FAMILY = 1;
};
name = Release;
@@ -587,8 +594,7 @@
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks";
PRODUCT_BUNDLE_IDENTIFIER = "org.cocoapods.$(PRODUCT_NAME:rfc1034identifier)";
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_SWIFT3_OBJC_INFERENCE = Default;
SWIFT_VERSION = 4.0;
SWIFT_VERSION = 5.0;
};
name = Debug;
};
@@ -605,8 +611,7 @@
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks";
PRODUCT_BUNDLE_IDENTIFIER = "org.cocoapods.$(PRODUCT_NAME:rfc1034identifier)";
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_SWIFT3_OBJC_INFERENCE = Default;
SWIFT_VERSION = 4.0;
SWIFT_VERSION = 5.0;
};
name = Release;
};
@@ -1,6 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<Scheme
LastUpgradeVersion = "0900"
LastUpgradeVersion = "1020"
version = "1.3">
<BuildAction
parallelizeBuildables = "YES"
@@ -40,7 +40,6 @@
buildConfiguration = "Debug"
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
language = ""
shouldUseLaunchSchemeArgsEnv = "YES">
<Testables>
<TestableReference
@@ -70,7 +69,6 @@
buildConfiguration = "Debug"
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
language = ""
launchStyle = "0"
useCustomWorkingDirectory = "NO"
ignoresPersistentStateOnLaunch = "NO"
+1 -1
View File
@@ -14,7 +14,7 @@ class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
// Override point for customization after application launch.
return true
}
+1 -1
View File
@@ -31,7 +31,7 @@ enum CellStyle {
let colors = [#colorLiteral(red: 0.5254901961, green: 0.6901960784, blue: 0.9137254902, alpha: 1), #colorLiteral(red: 0.5254901961, green: 0.6196078431, blue: 0.9137254902, alpha: 1), #colorLiteral(red: 0.6078431373, green: 0.5254901961, blue: 0.9137254902, alpha: 1), #colorLiteral(red: 0.9137254902, green: 0.5254901961, blue: 0.8392156863, alpha: 1), #colorLiteral(red: 0.9137254902, green: 0.5254901961, blue: 0.6, alpha: 1), #colorLiteral(red: 0.9137254902, green: 0.6784313725, blue: 0.5254901961, alpha: 1), #colorLiteral(red: 0.9137254902, green: 0.9058823529, blue: 0.5254901961, alpha: 1), #colorLiteral(red: 0.5254901961, green: 0.9137254902, blue: 0.5921568627, alpha: 1), #colorLiteral(red: 0.5254901961, green: 0.8, blue: 0.9137254902, alpha: 1)]
@IBInspectable var styleIndex: Int {
get { return CellStyle.all.index(of: style)! }
get { return CellStyle.all.firstIndex(of: style)! }
set { style = CellStyle.all[newValue % CellStyle.all.count] }
}
@@ -12,6 +12,7 @@ import InfiniteLayout
class CustomViewController: UIViewController {
@IBOutlet weak var infiniteCollectionView: InfiniteCollectionView!
}
extension CustomViewController: UICollectionViewDataSource {
@@ -29,10 +29,6 @@ class RxBaseCollectionViewController: UIViewController {
infiniteCollectionView.rx.modelCentered(Int.self)
.asDriver()
.drive(onNext: { current in
guard let current = current else {
print("centered: none")
return
}
print("centered: \(current + 1)")
}).disposed(by: disposeBag)
+1 -2
View File
@@ -1,4 +1,5 @@
use_frameworks!
platform :ios, '10.0'
target 'InfiniteLayout_Example' do
pod 'InfiniteLayout', :path => '../'
@@ -6,7 +7,5 @@ target 'InfiniteLayout_Example' do
target 'InfiniteLayout_Tests' do
inherit! :search_paths
end
end
+31 -30
View File
@@ -1,35 +1,36 @@
PODS:
- CocoaProxy (0.1.1)
- Differentiator (3.0.2)
- InfiniteLayout (0.2.3.1):
- CocoaProxy (~> 0)
- InfiniteLayout/Core (= 0.2.3.1)
- InfiniteLayout/Core (0.2.3.1):
- CocoaProxy (~> 0)
- InfiniteLayout/Rx (0.2.3.1):
- CocoaProxy (~> 0)
- InfiniteLayout/Core (~> 0)
- RxCocoa (~> 4)
- RxDataSources (~> 3)
- RxSwift (~> 4)
- RxCocoa (4.1.0):
- RxSwift (~> 4.0)
- RxDataSources (3.0.2):
- Differentiator (~> 3.0)
- RxCocoa (~> 4.0)
- RxSwift (~> 4.0)
- RxSwift (4.1.0)
- Differentiator (5.0.0)
- InfiniteLayout (0.5):
- InfiniteLayout/Core (= 0.5)
- InfiniteLayout/CocoaProxy (0.5)
- InfiniteLayout/Core (0.5):
- InfiniteLayout/CocoaProxy
- InfiniteLayout/Rx (0.5):
- InfiniteLayout/Core
- RxCocoa (~> 6)
- RxDataSources (~> 5)
- RxSwift (~> 6)
- RxCocoa (6.2.0):
- RxRelay (= 6.2.0)
- RxSwift (= 6.2.0)
- RxDataSources (5.0.0):
- Differentiator (~> 5.0)
- RxCocoa (~> 6.0)
- RxSwift (~> 6.0)
- RxRelay (6.2.0):
- RxSwift (= 6.2.0)
- RxSwift (6.2.0)
DEPENDENCIES:
- InfiniteLayout (from `../`)
- InfiniteLayout/Rx (from `../`)
SPEC REPOS:
https://github.com/cocoapods/specs.git:
- CocoaProxy
trunk:
- Differentiator
- RxCocoa
- RxDataSources
- RxRelay
- RxSwift
EXTERNAL SOURCES:
@@ -37,13 +38,13 @@ EXTERNAL SOURCES:
:path: "../"
SPEC CHECKSUMS:
CocoaProxy: 35ab81e24325b33834cffe45a3d1fd48ca67ef3a
Differentiator: a87be69eba49ec4ab460c7671143ee3a9eececfd
InfiniteLayout: 17bbab705aa4d941fa82a7e9b658c96d13ce91a3
RxCocoa: cc1fec49cdc8fabe645964de7c51c099a36c2aa8
RxDataSources: cb7c31e652a87ebb919da45f716bbb87b3765f6b
RxSwift: 4219941c1244c88002901bd87a69d3aea9ae71f0
Differentiator: e8497ceab83c1b10ca233716d547b9af21b9344d
InfiniteLayout: 2c55b0fc14b6a9d924fbe13c0e2d88b1cbf5423d
RxCocoa: 4baf94bb35f2c0ab31bc0cb9f1900155f646ba42
RxDataSources: aa47cc1ed6c500fa0dfecac5c979b723542d79cf
RxRelay: e72dbfd157807478401ef1982e1c61c945c94b2f
RxSwift: d356ab7bee873611322f134c5f9ef379fa183d8f
PODFILE CHECKSUM: 3501aeed91e62112dd38178a52f26b204b97c8c0
PODFILE CHECKSUM: 0dc7a9b37f9b7c5d5266beaab9e94861b80aa80f
COCOAPODS: 1.5.3
COCOAPODS: 1.10.1
+15 -11
View File
@@ -8,7 +8,7 @@
Pod::Spec.new do |s|
s.name = 'InfiniteLayout'
s.version = '0.2.3.2'
s.version = '0.5'
s.summary = 'Horizontal and Vertical infinite scrolling feature for UICollectionView with Paging, NSProxy delegate, Reactive extension'
# This description is used to generate tags and improve search results.
@@ -28,7 +28,7 @@ Horizontal and Vertical infinite scrolling feature for UICollectionView with Pag
s.source = { :git => 'https://github.com/arnauddorgans/InfiniteLayout.git', :tag => s.version.to_s }
s.social_media_url = 'https://twitter.com/arnauddorgans'
s.ios.deployment_target = '8.0'
s.ios.deployment_target = '9.0'
s.tvos.deployment_target = '9.0'
#s.xcconfig = { 'SWIFT_OBJC_BRIDGING_HEADER' => '${POD_ROOT}/InfiniteLayout/BridgeHeader.h' }
@@ -39,19 +39,23 @@ Horizontal and Vertical infinite scrolling feature for UICollectionView with Pag
# s.public_header_files = 'Pod/Classes/**/*.h'
# s.frameworks = 'UIKit', 'MapKit'
s.dependency 'CocoaProxy', '~> 0'
s.swift_versions = ['5.0', '5.1']
s.default_subspec = 'Core'
s.subspec 'Core' do |core|
core.source_files = 'InfiniteLayout/Classes/**/*'
core.source_files = 'Sources/InfiniteLayout/**/*'
core.dependency 'InfiniteLayout/CocoaProxy'
core.exclude_files = '**/*/SPMBridge.swift'
end
s.subspec 'CocoaProxy' do |core|
core.source_files = 'Sources/CocoaProxy/**/*'
end
s.subspec 'Rx' do |rx|
rx.dependency 'InfiniteLayout/Core', '~> 0'
rx.dependency 'RxSwift', '~> 4'
rx.dependency 'RxCocoa', '~> 4'
rx.dependency 'RxDataSources', '~> 3'
rx.source_files = 'InfiniteLayout/Rx/**/*'
rx.dependency 'InfiniteLayout/Core'
rx.dependency 'RxSwift', '~> 6'
rx.dependency 'RxCocoa', '~> 6'
rx.dependency 'RxDataSources', '~> 5'
rx.source_files = 'Sources/Rx/**/*'
end
end
View File
View File
+37
View File
@@ -0,0 +1,37 @@
// swift-tools-version:5.0
// The swift-tools-version declares the minimum version of Swift required to build this package.
import PackageDescription
let package = Package(
name: "InfiniteLayout",
platforms: [.iOS(.v9), .tvOS(.v9)],
products: [
.library(
name: "InfiniteLayout",
targets: ["InfiniteLayout"]),
.library(
name: "RxInfiniteLayout",
targets: ["RxInfiniteLayout"]),
],
dependencies: [
.package(url: "https://github.com/ReactiveX/RxSwift.git", .upToNextMajor(from: "6.0.0")),
.package(url: "https://github.com/RxSwiftCommunity/RxDataSources.git", .upToNextMajor(from: "5.0.0")),
],
targets: [
.target(
name: "CocoaProxy",
dependencies: [],
publicHeadersPath: "./"
),
.target(
name: "InfiniteLayout",
dependencies: ["CocoaProxy"]
),
.target(
name: "RxInfiniteLayout",
dependencies: ["InfiniteLayout", "RxSwift", "RxCocoa", "RxDataSources"],
path: "Sources/Rx"
),
]
)
+26 -4
View File
@@ -1,9 +1,10 @@
# InfiniteLayout
[![CI Status](http://img.shields.io/travis/arnauddorgans/InfiniteLayout.svg?style=flat)](https://travis-ci.org/arnauddorgans/InfiniteLayout)
[![Version](https://img.shields.io/cocoapods/v/InfiniteLayout.svg?style=flat)](http://cocoapods.org/pods/InfiniteLayout)
[![License](https://img.shields.io/cocoapods/l/InfiniteLayout.svg?style=flat)](http://cocoapods.org/pods/InfiniteLayout)
[![Platform](https://img.shields.io/cocoapods/p/InfiniteLayout.svg?style=flat)](http://cocoapods.org/pods/InfiniteLayout)
[![Version](https://img.shields.io/cocoapods/v/InfiniteLayout.svg?style=flat)](http://cocoapods.org/pods/InfiniteLayout)
[![Swift Package Manager compatible](https://img.shields.io/badge/Swift%20Package%20Manager-compatible-brightgreen.svg)](https://github.com/apple/swift-package-manager)
<img src="https://github.com/arnauddorgans/InfiniteLayout/raw/master/horizontal.gif" width="250" height="540"><img src="https://github.com/arnauddorgans/InfiniteLayout/raw/master/vertical.gif" width="250" height="540"><img src="https://github.com/arnauddorgans/InfiniteLayout/raw/master/custom.gif" width="250" height="540">
@@ -12,10 +13,10 @@
To run the example project, clone the repo, and run `pod install` from the Example directory first.
## Requirements
## Installation
### [CocoaPods](https://guides.cocoapods.org/using/using-cocoapods.html)
InfiniteLayout is available through [CocoaPods](http://cocoapods.org). To install
it, simply add the following line to your Podfile:
@@ -23,6 +24,27 @@ it, simply add the following line to your Podfile:
pod 'InfiniteLayout'
```
### [Swift Package Manager](https://github.com/apple/swift-package-manager)
Create a `Package.swift` file.
```swift
// swift-tools-version:5.0
import PackageDescription
let package = Package(
name: "InfiniteLayoutTestProject",
dependencies: [
.package(url: "https://github.com/arnauddorgans/InfiniteLayout.git", from: "0.4.2")
],
targets: [
.target(name: "InfiniteLayoutTestProject", dependencies: ["InfiniteLayout"])
]
)
```
## Usage
```swift
@@ -85,7 +107,7 @@ self.infiniteCollectionView.preferredCenteredIndexPath = nil // center the close
InfiniteCollectionView provide an **infiniteDelegate** protocol used to get the centered IndexPath, usefull if you want to use an InfiniteCollectionView like a Picker.
```swift
func infiniteCollectionView(_ infiniteCollectionView: InfiniteCollectionView, didChangeCenteredIndexPath centeredIndexPath: IndexPath?)
func infiniteCollectionView(_ infiniteCollectionView: InfiniteCollectionView, didChangeCenteredIndexPath from: IndexPath?, to: IndexPath?)
```
### Rx
+20
View File
@@ -0,0 +1,20 @@
//
// CocoaProxy.h
// CocoaProxy
//
// Created by Arnaud Dorgans on 27/12/2017.
//
#import <Foundation/Foundation.h>
@interface CocoaProxy : NSProxy
- (instancetype _Nonnull)init;
- (instancetype _Nonnull)initWithProxies:(nonnull NSArray<id<NSObject>>*)proxies;
- (NSArray<id<NSObject>> *_Nonnull)proxiesForSelector:(SEL _Nonnull )aSelector;
@property (nonatomic, strong) NSArray<id<NSObject>>* _Nonnull proxies;
@property (nonatomic, copy) BOOL (^ _Nullable proxyFilter)(id<NSObject> _Nonnull proxy, SEL _Nonnull selector);
@end
+80
View File
@@ -0,0 +1,80 @@
//
// CocoaProxy.m
// CocoaProxy
//
// Created by Arnaud Dorgans on 27/12/2017.
//
#import "CocoaProxy.h"
@interface CocoaProxy () { }
@property (nonatomic, strong) NSPointerArray *pointerArray;
@end
@implementation CocoaProxy
- (instancetype _Nonnull)initWithProxies:(nonnull NSArray<NSObject*>*)proxies {
[self setProxies: proxies];
return self;
}
- (instancetype _Nonnull)init {
return [self initWithProxies: @[]];
}
- (BOOL)respondsToSelector:(SEL)aSelector
{
return ([self methodSignatureForSelector: aSelector] != nil);
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
for (NSObject* proxy in [self proxiesForSelector: aSelector]) {
if ([proxy respondsToSelector: aSelector]) {
return [proxy methodSignatureForSelector: aSelector];
}
}
return nil;
}
- (NSArray<id<NSObject>> *_Nonnull)proxiesForSelector:(SEL _Nonnull )aSelector
{
if (self.proxyFilter) {
NSPredicate *predicate = [NSPredicate predicateWithBlock:^BOOL(id _Nullable evaluatedObject, NSDictionary<NSString *,id> * _Nullable bindings) {
return self.proxyFilter(evaluatedObject, aSelector);
}];
return [self.proxies filteredArrayUsingPredicate: predicate];
}
return self.proxies;
}
- (void)forwardInvocation:(NSInvocation *)invocation
{
for (NSObject* proxy in [self proxiesForSelector: invocation.selector]) {
[self invokeInvocation: invocation onProxy: proxy];
}
}
- (BOOL)invokeInvocation:(NSInvocation *)invocation onProxy:(id<NSObject>)proxy
{
if ([proxy respondsToSelector: invocation.selector]) {
[invocation invokeWithTarget: proxy];
return YES;
}
return NO;
}
- (void)setProxies:(NSArray<id<NSObject>> *)proxies {
self.pointerArray = [NSPointerArray weakObjectsPointerArray];
for (NSObject* proxy in proxies) {
[self.pointerArray addPointer: (void *)proxy];
}
}
- (NSArray<id<NSObject>> *)proxies {
return self.pointerArray.allObjects;
}
@end
@@ -8,8 +8,8 @@
import UIKit
@objc public protocol InfiniteCollectionViewDelegate {
@objc optional func infiniteCollectionView(_ infiniteCollectionView: InfiniteCollectionView, didChangeCenteredIndexPath centeredIndexPath: IndexPath?)
@objc optional func infiniteCollectionView(_ infiniteCollectionView: InfiniteCollectionView, didChangeCenteredIndexPath from: IndexPath?, to: IndexPath?)
}
open class InfiniteCollectionView: UICollectionView {
@@ -22,7 +22,7 @@ open class InfiniteCollectionView: UICollectionView {
open private(set) var centeredIndexPath: IndexPath?
open var preferredCenteredIndexPath: IndexPath? = IndexPath(item: 0, section: 0)
var forwardDelegate: Bool { return true }
open var forwardDelegate: Bool { return true }
var _contentSize: CGSize?
override open weak var delegate: UICollectionViewDelegate? {
@@ -134,7 +134,7 @@ extension InfiniteCollectionView: UICollectionViewDataSource {
}
private var multiplier: Int {
return InfiniteDataSources.multiplier(estimatedItemSize: self.infiniteLayout.itemSize)
return InfiniteDataSources.multiplier(estimatedItemSize: self.infiniteLayout.itemSize, enabled: self.infiniteLayout.isEnabled)
}
public func section(from infiniteSection: Int) -> Int {
@@ -170,8 +170,9 @@ extension InfiniteCollectionView: UICollectionViewDelegate {
let preferredVisibleIndexPath = infiniteLayout.preferredVisibleLayoutAttributes()?.indexPath
if self.centeredIndexPath != preferredVisibleIndexPath {
let previousCenteredIndexPath = self.centeredIndexPath
self.centeredIndexPath = preferredVisibleIndexPath
self.infiniteDelegate?.infiniteCollectionView?(self, didChangeCenteredIndexPath: preferredVisibleIndexPath)
self.infiniteDelegate?.infiniteCollectionView?(self, didChangeCenteredIndexPath: previousCenteredIndexPath, to: self.centeredIndexPath)
}
}
@@ -202,5 +203,6 @@ extension InfiniteCollectionView: UICollectionViewDelegate {
if isItemPagingEnabled {
self.infiniteLayout.centerCollectionView(withVelocity: velocity, targetContentOffset: targetContentOffset)
}
self.delegateProxy.delegate?.scrollViewWillEndDragging?(scrollView, withVelocity: velocity, targetContentOffset: targetContentOffset)
}
}
@@ -6,7 +6,6 @@
//
import UIKit
import CocoaProxy
class InfiniteCollectionViewProxy<T: NSObjectProtocol>: CocoaProxy {
@@ -7,27 +7,31 @@
import UIKit
class InfiniteDataSources {
static func section(from infiniteSection: Int, numberOfSections: Int) -> Int {
open class InfiniteDataSources {
public static var originCount: Int = 0
public static func section(from infiniteSection: Int, numberOfSections: Int) -> Int {
return infiniteSection % numberOfSections
}
static func indexPath(from infiniteIndexPath: IndexPath, numberOfSections: Int, numberOfItems: Int) -> IndexPath {
public static func indexPath(from infiniteIndexPath: IndexPath, numberOfSections: Int, numberOfItems: Int) -> IndexPath {
return IndexPath(item: infiniteIndexPath.item % numberOfItems, section: self.section(from: infiniteIndexPath.section, numberOfSections: numberOfSections))
}
static func multiplier(estimatedItemSize: CGSize) -> Int {
public static func multiplier(estimatedItemSize: CGSize, enabled: Bool) -> Int {
guard enabled else {
return 1
}
let min = Swift.min(estimatedItemSize.width, estimatedItemSize.height)
let count = ceil(InfiniteLayout.minimumContentSize / min)
return Int(count)
}
static func numberOfSections(numberOfSections: Int, multiplier: Int) -> Int {
public static func numberOfSections(numberOfSections: Int, multiplier: Int) -> Int {
return numberOfSections > 1 ? numberOfSections * multiplier : numberOfSections
}
static func numberOfItemsInSection(numberOfItemsInSection: Int, numberOfSections: Int, multiplier: Int) -> Int {
public static func numberOfItemsInSection(numberOfItemsInSection: Int, numberOfSections: Int, multiplier: Int) -> Int {
originCount = numberOfItemsInSection
return numberOfSections > 1 ? numberOfItemsInSection : numberOfItemsInSection * multiplier
}
}
@@ -15,7 +15,20 @@ open class InfiniteLayout: UICollectionViewFlowLayout {
private var contentSize: CGSize = .zero
private (set) var isEnabled: Bool = false
private var hasValidLayout: Bool = false
@IBInspectable public var isEnabled: Bool = true {
didSet {
self.invalidateLayout()
}
}
public var currentPage: CGPoint {
guard let collectionView = self.collectionView else {
return .zero
}
return self.page(for: collectionView.contentOffset)
}
open override func shouldInvalidateLayout(forBoundsChange newBounds: CGRect) -> Bool {
return true
@@ -42,8 +55,8 @@ open class InfiniteLayout: UICollectionViewFlowLayout {
override open func prepare() {
let collectionViewContentSize = super.collectionViewContentSize
self.contentSize = CGSize(width: collectionViewContentSize.width, height: collectionViewContentSize.height)
self.isEnabled = {
guard let collectionView = self.collectionView, collectionView.bounds != .zero else {
self.hasValidLayout = {
guard let collectionView = self.collectionView, collectionView.bounds != .zero, self.isEnabled else {
return false
}
return (scrollDirection == .horizontal ? self.contentSize.width : self.contentSize.height) >=
@@ -53,15 +66,22 @@ open class InfiniteLayout: UICollectionViewFlowLayout {
}
override open var collectionViewContentSize: CGSize {
guard isEnabled else {
return super.collectionViewContentSize
guard hasValidLayout else {
return self.contentSize
}
return CGSize(width: scrollDirection == .horizontal ? self.contentSize.width * multiplier : self.contentSize.width,
height: scrollDirection == .vertical ? self.contentSize.height * multiplier : self.contentSize.height)
}
open override func layoutAttributesForItem(at indexPath: IndexPath) -> UICollectionViewLayoutAttributes? {
guard let attributes = super.layoutAttributesForItem(at: indexPath) else {
return nil
}
return self.layoutAttributes(from: attributes, page: currentPage)
}
override open func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
guard isEnabled else {
guard hasValidLayout else {
return super.layoutAttributesForElements(in: rect)
}
let page = self.page(for: rect.origin)
@@ -143,7 +163,7 @@ open class InfiniteLayout: UICollectionViewFlowLayout {
}
public func loopCollectionViewIfNeeded() {
guard let collectionView = self.collectionView, self.isEnabled else {
guard let collectionView = self.collectionView, self.hasValidLayout else {
return
}
let page = self.pageIndex(from: self.page(for: collectionView.contentOffset))
@@ -233,7 +253,7 @@ open class InfiniteLayout: UICollectionViewFlowLayout {
}
public func centerCollectionView(withVelocity velocity: CGPoint, targetContentOffset: UnsafeMutablePointer<CGPoint>) {
guard let collectionView = self.collectionView, self.isEnabled else {
guard let collectionView = self.collectionView, self.hasValidLayout else {
return
}
let newTarget = CGPoint(x: self.scrollDirection == .horizontal ? collectionView.contentOffset.x + velocity.x * velocityMultiplier : targetContentOffset.pointee.x,
@@ -247,7 +267,7 @@ open class InfiniteLayout: UICollectionViewFlowLayout {
}
public func centerCollectionViewIfNeeded(indexPath: IndexPath? = nil) {
guard let collectionView = self.collectionView, self.isEnabled else {
guard let collectionView = self.collectionView, self.hasValidLayout else {
return
}
guard let preferredAttributes = self.preferredVisibleLayoutAttributes(indexPath: indexPath),
+8
View File
@@ -0,0 +1,8 @@
//
// InfiniteCollectionView.swift
// InfiniteLayout
//
// Created by Arnaud Dorgans on 05/09/2019.
//
@_exported import CocoaProxy
@@ -5,6 +5,10 @@
// Created by Arnaud Dorgans on 03/01/2018.
//
#if canImport(InfiniteLayout)
import InfiniteLayout
#endif
import UIKit
import RxSwift
import RxCocoa
@@ -31,34 +35,52 @@ class RxInfiniteCollectionViewDelegate: DelegateProxy<InfiniteCollectionView, In
}
}
public extension Reactive where Base: InfiniteCollectionView {
extension Reactive where Base: InfiniteCollectionView {
private var infiniteDelegate: RxInfiniteCollectionViewDelegate {
return RxInfiniteCollectionViewDelegate.proxy(for: self.base)
}
public var itemCentered: ControlEvent<IndexPath?> {
let source = infiniteDelegate.sentMessage(#selector(InfiniteCollectionViewDelegate.infiniteCollectionView(_:didChangeCenteredIndexPath:)))
let source = infiniteDelegate.sentMessage(#selector(InfiniteCollectionViewDelegate.infiniteCollectionView(_:didChangeCenteredIndexPath:to:)))
.map { $0.last as? IndexPath }
return ControlEvent(events: source)
}
public func modelCentered<T>(_ type: T.Type) -> ControlEvent<T?> {
let source = itemCentered
.map { indexPath -> T? in
return try indexPath.flatMap { try self.model(at: $0) }
public func modelCentered<T>(_ type: T.Type) -> ControlEvent<T> {
let source: Observable<T> = itemCentered.flatMap { [weak view = self.base as InfiniteCollectionView] indexPath -> Observable<T> in
guard let view = view, var indexPath = indexPath else {
return Observable.empty()
}
indexPath.row %= InfiniteDataSources.originCount
return Observable.just(try view.rx.model(at: indexPath))
}
return ControlEvent(events: source)
}
public func modelSelected<T>(_ modelType: T.Type) -> ControlEvent<T> {
let source: Observable<T> = itemSelected.flatMap { [weak view = self.base as InfiniteCollectionView] indexPath -> Observable<T> in
guard let view = view else {
return Observable.empty()
}
var indexPath = indexPath
indexPath.row %= InfiniteDataSources.originCount
return Observable.just(try view.rx.model(at: indexPath))
}
return ControlEvent(events: source)
}
}
public extension Reactive where Base: RxInfiniteCollectionView {
extension Reactive where Base: RxInfiniteCollectionView {
public func items<S: Sequence, O: ObservableType>
(infinite: Bool)
-> (_ source: O)
-> (_ cellFactory: @escaping (UICollectionView, Int, S.Iterator.Element) -> UICollectionViewCell)
-> Disposable where O.E == S {
-> Disposable where O.Element == S {
return { source in
guard infinite else {
return self.items(source)
@@ -76,7 +98,7 @@ public extension Reactive where Base: RxInfiniteCollectionView {
(cellIdentifier: String, cellType: Cell.Type = Cell.self, infinite: Bool)
-> (_ source: O)
-> (_ configureCell: @escaping (Int, S.Iterator.Element, Cell) -> Void)
-> Disposable where O.E == S {
-> Disposable where O.Element == S {
guard infinite else {
return self.items(cellIdentifier: cellIdentifier, cellType: cellType)
}
@@ -5,6 +5,10 @@
// Created by Arnaud Dorgans on 03/01/2018.
//
#if canImport(InfiniteLayout)
import InfiniteLayout
#endif
import UIKit
import RxSwift
import RxCocoa
@@ -13,7 +17,7 @@ open class RxInfiniteCollectionView: InfiniteCollectionView {
let disposeBag = DisposeBag()
override var forwardDelegate: Bool {
override open var forwardDelegate: Bool {
return false
}
@@ -5,17 +5,23 @@
// Created by Arnaud Dorgans on 03/01/2018.
//
#if canImport(InfiniteLayout)
import InfiniteLayout
#endif
import UIKit
import RxDataSources
open class RxInfiniteCollectionViewSectionedReloadDataSource<S: SectionModelType>: RxCollectionViewSectionedReloadDataSource<S> {
public var isEnabled: Bool = true
open override subscript(section: Int) -> S {
let section = InfiniteDataSources.section(from: section, numberOfSections: sectionModels.count)
return self.sectionModels[section]
}
open override subscript(indexPath: IndexPath) -> I {
open override subscript(indexPath: IndexPath) -> Item {
get {
let indexPath = InfiniteDataSources.indexPath(from: indexPath,
numberOfSections: sectionModels.count,
@@ -33,7 +39,7 @@ open class RxInfiniteCollectionViewSectionedReloadDataSource<S: SectionModelType
guard let layout = collectionView.collectionViewLayout as? UICollectionViewFlowLayout else {
fatalError()
}
return InfiniteDataSources.multiplier(estimatedItemSize: layout.itemSize)
return InfiniteDataSources.multiplier(estimatedItemSize: layout.itemSize, enabled: isEnabled)
}
open override func numberOfSections(in collectionView: UICollectionView) -> Int {
@@ -54,12 +60,14 @@ open class RxInfiniteCollectionViewSectionedReloadDataSource<S: SectionModelType
open class RxInfiniteCollectionViewSectionedAnimatedDataSource<S: AnimatableSectionModelType>: RxCollectionViewSectionedAnimatedDataSource<S> {
public var isEnabled: Bool = true
open override subscript(section: Int) -> S {
let section = InfiniteDataSources.section(from: section, numberOfSections: sectionModels.count)
return self.sectionModels[section]
}
open override subscript(indexPath: IndexPath) -> I {
open override subscript(indexPath: IndexPath) -> Item {
get {
let indexPath = InfiniteDataSources.indexPath(from: indexPath,
numberOfSections: sectionModels.count,
@@ -77,7 +85,7 @@ open class RxInfiniteCollectionViewSectionedAnimatedDataSource<S: AnimatableSect
guard let layout = collectionView.collectionViewLayout as? UICollectionViewFlowLayout else {
fatalError()
}
return InfiniteDataSources.multiplier(estimatedItemSize: layout.itemSize)
return InfiniteDataSources.multiplier(estimatedItemSize: layout.itemSize, enabled: isEnabled)
}
open override func numberOfSections(in collectionView: UICollectionView) -> Int {