A simple, highly customisable, data-driven 3D carousel for iOS and Mac OS

Related tags

UI iCarousel
Overview

Purpose

iCarousel is a class designed to simplify the implementation of various types of carousel (paged, scrolling views) on iPhone, iPad and Mac OS. iCarousel implements a number of common effects such as cylindrical, flat and "CoverFlow" style carousels, as well as providing hooks to implement your own bespoke effects. Unlike many other "CoverFlow" libraries, iCarousel can work with any kind of view, not just images, so it is ideal for presenting paged data in a fluid and impressive way in your app. It also makes it extremely easy to swap between different carousel effects with minimal code changes.

Supported OS & SDK Versions

  • Supported build target - iOS 10.0 / Mac OS 10.12 (Xcode 8.0, Apple LLVM compiler 8.0)
  • Earliest supported deployment target - iOS 5.0 / Mac OS 10.7
  • Earliest compatible deployment target - iOS 4.3 / Mac OS 10.6

NOTE: 'Supported' means that the library has been tested with this version. 'Compatible' means that the library should work on this OS version (i.e. it doesn't rely on any unavailable SDK features) but is no longer being tested for compatibility and may require tweaking or bug fixes to run correctly.

ARC Compatibility

As of version 1.8, iCarousel requires ARC. If you wish to use iCarousel in a non-ARC project, just add the -fobjc-arc compiler flag to the iCarousel.m class. To do this, go to the Build Phases tab in your target settings, open the Compile Sources group, double-click iCarousel.m in the list and type -fobjc-arc into the popover.

If you wish to convert your whole project to ARC, comment out the #error line in iCarousel.m, then run the Edit > Refactor > Convert to Objective-C ARC... tool in Xcode and make sure all files that you wish to use ARC for (including iCarousel.m) are checked.

Thread Safety

iCarousel is derived from UIView and - as with all UIKit components - it should only be accessed from the main thread. You may wish to use threads for loading or updating carousel contents or items, but always ensure that once your content has loaded, you switch back to the main thread before updating the carousel.

Installation

To use the iCarousel class in an app, just drag the iCarousel class files (demo files and assets are not needed) into your project and add the QuartzCore framework. You can also install it using Cocoapods in the normal way.

Chameleon Support

iCarousel is now compatible with the Chameleon iOS-to-Mac conversion library (https://github.com/BigZaphod/Chameleon). To use iCarousel with Chameleon, add USING_CHAMELEON to your project's preprocessor macros. Check out the Chameleon Demo example project for how to port your iOS iCarousel app to Mac OS using Chameleon - the example demonstrates how to run the No Nib iPhone example on Mac OS using Chameleon. Note that tap-to-center doesn't currently work, and scrolling must be done using a two-fingered scroll gesture, not click-and-drag (both of these are due to features/limitations of the Chameleon UIGestureRecognizer implementation).

Carousel Types

iCarousel supports the following built-in display types:

  • iCarouselTypeLinear
  • iCarouselTypeRotary
  • iCarouselTypeInvertedRotary
  • iCarouselTypeCylinder
  • iCarouselTypeInvertedCylinder
  • iCarouselTypeWheel
  • iCarouselTypeInvertedWheel
  • iCarouselTypeCoverFlow
  • iCarouselTypeCoverFlow2
  • iCarouselTypeTimeMachine
  • iCarouselTypeInvertedTimeMachine

You can also implement your own bespoke carousel styles using iCarouselTypeCustom and the carousel:itemTransformForOffset:baseTransform: delegate method.

NOTE: The difference between iCarouselTypeCoverFlow and iCarouselTypeCoverFlow2 types is quite subtle, however the logic for iCarouselTypeCoverFlow2 is substantially more complex. If you flick the carousel they are basically identical, but if you drag the carousel slowly with your finger the difference should be apparent. iCarouselTypeCoverFlow2 is designed to simulate the standard Apple CoverFlow effect as closely as possible and may change subtly in future in the interests of that goal.

Display Type Visual Examples


Linear

Linear

Rotary

Rotary

Inverted Rotary

Inverted Rotary

Cylinder

Cylinder

Inverted Cylinder

Inverted Cylinder

Wheel

Wheel

Inverted Wheel

Inverted Wheel

Cover Flow

Cover Flow

Cover Flow2

Cover Flow2

Time Machine

Time Machine

Inverted Time Machine

Inverted Time Machine

Properties

The iCarousel has the following properties (note: for Mac OS, substitute NSView for UIView when using properties):

@property (nonatomic, weak) IBOutlet id<iCarouselDataSource> dataSource;

An object that supports the iCarouselDataSource protocol and can provide views to populate the carousel.

@property (nonatomic, weak) IBOutlet id<iCarouselDelegate> delegate;

An object that supports the iCarouselDelegate protocol and can respond to carousel events and layout requests.

@property (nonatomic, assign) iCarouselType type;

Used to switch the carousel display type (see above for details).

@property (nonatomic, assign) CGFloat perspective;

Used to tweak the perspective foreshortening effect for the various 3D carousel views. Should be a negative value, less than 0 and greater than -0.01. Values outside of this range will yield very strange results. The default is -1/500, or -0.005;

@property (nonatomic, assign) CGSize contentOffset;

This property is used to adjust the offset of the carousel item views relative to the center of the carousel. It defaults to CGSizeZero, meaning that the carousel items are centered. Changing this value moves the carousel items without changing their perspective, i.e. the vanishing point moves with the carousel items, so if you move the carousel items down, it does not appear as if you are looking down on the carousel.

@property (nonatomic, assign) CGSize viewpointOffset;

This property is used to adjust the user viewpoint relative to the carousel items. It has the opposite effect to adjusting the contentOffset, i.e. if you move the viewpoint up then the carousel appears to move down. Unlike the contentOffset, moving the viewpoint also changes the perspective vanishing point relative to the carousel items, so if you move the viewpoint up, it will appear as if you are looking down on the carousel.

@property (nonatomic, assign) CGFloat decelerationRate;

The rate at which the carousel decelerates when flicked. Higher values mean slower deceleration. The default value is 0.95. Values should be in the range 0.0 (carousel stops immediately when released) to 1.0 (carousel continues indefinitely without slowing down, unless it reaches the end).

@property (nonatomic, assign) BOOL bounces;

Sets whether the carousel should bounce past the end and return, or stop dead. Note that this has no effect on carousel types that are designed to wrap, or where the carouselShouldWrap delegate method returns YES.

@property (nonatomic, assign) CGFloat bounceDistance;

The maximum distance that a non-wrapped carousel will bounce when it overshoots the end. This is measured in multiples of the itemWidth, so a value of 1.0 would means the carousel will bounce by one whole item width, a value of 0.5 would be half an item's width, and so on. The default value is 1.0;

@property (nonatomic, assign, getter = isScrollEnabled) BOOL scrollEnabled;

Enables and disables user scrolling of the carousel. The carousel can still be scrolled programmatically if this property is set to NO.

@property (nonatomic, readonly, getter = isWrapEnabled) BOOL wrapEnabled;

Returns YES if wrapping is enabled and NO if it isn't. This property is read only. If you wish to override the default value, implement the carousel:valueForOption:withDefault: delegate method and return a value for iCarouselOptionWrap.

@property (nonatomic, assign, getter = isPagingEnabled) BOOL pagingEnabled;

Enables and disables paging. When paging is enabled, the carousel will stop at each item view as the user scrolls, much like the pagingEnabled property of a UIScrollView.

@property (nonatomic, readonly) NSInteger numberOfItems;

The number of items in the carousel (read only). To set this, implement the numberOfItemsInCarousel: dataSource method. Note that not all of these item views will be loaded or visible at a given point in time - the carousel loads item views on demand as it scrolls.

@property (nonatomic, readonly) NSInteger numberOfPlaceholders;

The number of placeholder views to display in the carousel (read only). To set this, implement the numberOfPlaceholdersInCarousel: dataSource method.

@property (nonatomic, readonly) NSInteger numberOfVisibleItems;

The maximum number of carousel item views to be displayed concurrently on screen (read only). This property is important for performance optimisation, and is calculated automatically based on the carousel type and view frame. If you wish to override the default value, implement the carousel:valueForOption:withDefault: delegate method and return a value for iCarouselOptionVisibleItems.

@property (nonatomic, strong, readonly) NSArray *indexesForVisibleItems;

An array containing the indexes of all item views currently loaded and visible in the carousel, including placeholder views. The array contains NSNumber objects whose integer values match the indexes of the views. The indexes for item views start at zero and match the indexes passed to the dataSource to load the view, however the indexes for any visible placeholder views will either be negative (less than zero) or greater than or equal to numberOfItems. Indexes for placeholder views in this array do not equate to the placeholder view index used with the dataSource.

@property (nonatomic, strong, readonly) NSArray *visibleItemViews;

An array of all the item views currently displayed in the carousel (read only). This includes any visible placeholder views. The indexes of views in this array do not match the item indexes, however the order of these views matches the order of the visibleItemIndexes array property, i.e. you can get the item index of a given view in this array by retrieving the equivalent object from the visibleItemIndexes array (or, you can just use the indexOfItemView: method, which is much easier).

@property (nonatomic, strong, readonly) UIView *contentView;

The view containing the carousel item views. You can add subviews to this view if you want to intersperse them with the carousel items. If you want a view to appear in front or behind all of the carousel items, you should add it directly to the iCarousel view itself instead. Note that the order of views inside the contentView is subject to frequent and undocumented change whilst the app is running. Any views added to the contentView should have their userInteractionEnabled property set to NO to prevent conflicts with iCarousel's touch event handling.

@property (nonatomic, assign) CGFloat scrollOffset;

This is the current scroll offset of the carousel in multiples of the itemWidth. This value, rounded to the nearest integer, is the currentItemIndex value. You can use this value to position other screen elements while the carousel is in motion. The value can also be set if you wish to scroll the carousel to a particular offset programmatically. This may be useful if you wish to disable the built-in gesture handling and provide your own implementation.

@property (nonatomic, readonly) CGFloat offsetMultiplier;

This is the offset multiplier used when the user drags the carousel with their finger. It does not affect programmatic scrolling or deceleration speed. This defaults to 1.0 for most carousel types, but defaults to 2.0 for the CoverFlow-style carousels to compensate for the fact that their items are more closely spaced and so must be dragged further to move the same distance. You cannot set this property directly, but you can override the default value by implementing the carouselOffsetMultiplier: delegate method.

@property (nonatomic, assign) NSInteger currentItemIndex;

The index of the currently centered item in the carousel. Setting this property is equivalent to calling scrollToItemAtIndex:animated: with the animated argument set to NO.

@property (nonatomic, strong, readonly) UIView *currentItemView;

The currently centered item view in the carousel. The index of this view matches currentItemIndex.

@property (nonatomic, readonly) CGFloat itemWidth;

The display width of items in the carousel (read only). This is derived automatically from the first view passed in to the carousel using the carousel:viewForItemAtIndex:reusingView: dataSource method. You can also override this value using the carouselItemWidth: delegate method, which will alter the space allocated for carousel items (but won't resize or scale the item views).

@property (nonatomic, assign) BOOL centerItemWhenSelected;

When set to YES, tapping any item in the carousel other than the one matching the currentItemIndex will cause it to smoothly animate to the center. Tapping the currently selected item will have no effect. Defaults to YES.

@property (nonatomic, assign) CGFloat scrollSpeed;

This is the scroll speed multiplier when the user flicks the carousel with their finger. Defaults to 1.0.

@property (nonatomic, readonly) CGFloat toggle;

This property is used for the iCarouselTypeCoverFlow2 carousel transform. It is exposed so that you can implement your own variants of the CoverFlow2 style using the carousel:itemTransformForOffset:baseTransform: delegate method.

@property (nonatomic, assign) BOOL stopAtItemBoundary;

By default, the carousel will come to rest at an exact item boundary when it is flicked. If you set this property to NO, it will stop naturally and then - if scrollToItemBoundary is set to YES - scroll back or forwards to the nearest boundary.

@property (nonatomic, assign) BOOL scrollToItemBoundary;

By default whenever the carousel stops moving it will automatically scroll to the nearest item boundary. If you set this property to NO, the carousel will not scroll after stopping and will stay wherever it is, even if it's not perfectly aligned on the current index. The exception to this is that if wrapping is disabled and bounces is set to YES then regardless of this setting, the carousel will automatically scroll back to the first or last item index if it comes to rest beyond the end of the carousel.

@property (nonatomic, assign, getter = isVertical) BOOL vertical;

This property toggles whether the carousel is displayed horizontally or vertically on screen. All the built-in carousel types work in both orientations. Switching to vertical changes both the layout of the carousel and also the direction of swipe detection on screen. Note that custom carousel transforms are not affected by this property, however the swipe gesture direction will still be affected.

@property (nonatomic, readonly, getter = isDragging) BOOL dragging;

Returns YES if user has started scrolling the carousel and has not yet released it.

@property (nonatomic, readonly, getter = isDecelerating) BOOL decelerating;

Returns YES if the user isn't dragging the carousel any more, but it is still moving.

@property (nonatomic, readonly, getter = isScrolling) BOOL scrolling;

Returns YES if the carousel is currently being scrolled programatically.

@property (nonatomic, assign) BOOL ignorePerpendicularSwipes;

If YES, the carousel will ignore swipe gestures that are perpendicular to the orientation of the carousel. So for a horizontal carousel, vertical swipes will not be intercepted. This means that you can have a vertically scrolling scrollView inside a carousel item view and it will still function correctly. Defaults to YES.

@property (nonatomic, assign) BOOL clipsToBounds;

This is actually not a property of iCarousel but is inherited from UIView. It's included here because it's a frequently missed feature. Set this to YES to prevent the carousel item views overflowing their bounds. You can set this property in Interface Builder by ticking the 'Clip Subviews' option. Defaults to NO.

@property (nonatomic, assign) CGFloat autoscroll;

This property can be used to set the carousel scrolling at a constant speed. A value of 1.0 would scroll the carousel forwards at a rate of one item per second. The autoscroll value can be positive or negative and defaults to 0.0 (stationary). Autoscrolling will stop if the user interacts with the carousel, and will resume when they stop.

Methods

The iCarousel class has the following methods (note: for Mac OS, substitute NSView for UIView in method arguments):

- (void)scrollToItemAtIndex:(NSInteger)index animated:(BOOL)animated;

This will center the carousel on the specified item, either immediately or with a smooth animation. For wrapped carousels, the carousel will automatically determine the shortest (direct or wraparound) distance to scroll. If you need to control the scroll direction, or want to scroll by more than one revolution, use the scrollByNumberOfItems method instead.

- (void)scrollToItemAtIndex:(NSInteger)index duration:(NSTimeInterval)scrollDuration;

This method allows you to control how long the carousel takes to scroll to the specified index.

- (void)scrollByNumberOfItems:(NSInteger)itemCount duration:(NSTimeInterval)duration;

This method allows you to scroll the carousel by a fixed distance, measured in carousel item widths. Positive or negative values may be specified for itemCount, depending on the direction you wish to scroll. iCarousel gracefully handles bounds issues, so if you specify a distance greater than the number of items in the carousel, scrolling will either be clamped when it reaches the end of the carousel (if wrapping is disabled) or wrap around seamlessly.

- (void)scrollToOffset:(CGFloat)offset duration:(NSTimeInterval)duration;

This works the same way as scrollToItemAtIndex:, but allows you to scroll to a fractional offset. This may be useful if you wish to achieve a very precise animation effect. Note that if the scrollToItemBoundary property is set to YES, the carousel will automatically scroll to the nearest item index after you call this method. anyway.

- (void)scrollByOffset:(CGFloat)offset duration:(NSTimeInterval)duration;

This works the same way as scrollByNumberOfItems:, but allows you to scroll by a fractional number of items. This may be useful if you wish to achieve a very precise animation effect. Note that if the scrollToItemBoundary property is set to YES, the carousel will automatically scroll to the nearest item index after you call this method anyway.

- (void)reloadData;

This reloads all carousel views from the dataSource and refreshes the carousel display.

- (UIView *)itemViewAtIndex:(NSInteger)index;

Returns the visible item view with the specified index. Note that the index relates to the position in the carousel, and not the position in the visibleItemViews array, which may be different. Pass a negative index or one greater than or equal to numberOfItems to retrieve placeholder views. The method only works for visible item views and will return nil if the view at the specified index has not been loaded, or if the index is out of bounds.

- (NSInteger)indexOfItemView:(UIView *)view;

The index for a given item view in the carousel. Works for item views and placeholder views, however placeholder view indexes do not match the ones used by the dataSource and may be negative (see indexesForVisibleItems property above for more details). This method only works for visible item views and will return NSNotFound for views that are not currently loaded. For a list of all currently loaded views, use the visibleItemViews property.

- (NSInteger)indexOfItemViewOrSubview:(UIView *)view

This method gives you the item index of either the view passed or the view containing the view passed as a parameter. It works by walking up the view hierarchy starting with the view passed until it finds an item view and returns its index within the carousel. If no currently-loaded item view is found, it returns NSNotFound. This method is extremely useful for handling events on controls embedded within an item view. This allows you to bind all your item controls to a single action method on your view controller, and then work out which item the control that triggered the action was related to. You can see an example of this technique in the Controls Demo example project.

- (CGFloat)offsetForItemAtIndex:(NSInteger)index;

Returns the offset for the specified item index in multiples of itemWidth from the center position. This is the same value used for calculating the view transform and alpha, and can be used to customise item views based on their position in the carousel. This value can be expected to change for each view whenever the carouselDidScroll: delegate method is called.

- (UIView *)itemViewAtPoint:(CGPoint)point;

Returns the frontmost item view at the specified point within the bounds of the carousel. Useful for implementing your own tap detection.

- (void)removeItemAtIndex:(NSInteger)index animated:(BOOL)animated;

This removes an item from the carousel. The remaining items will slide across to fill the gap. Note that the data source is not automatically updated when this method is called, so a subsequent call to reloadData will restore the removed item.

- (void)insertItemAtIndex:(NSInteger)index animated:(BOOL)animated;

This inserts an item into the carousel. The new item will be requested from the dataSource, so make sure that the new item has been added to the data source data before calling this method, or you will get duplicate items in the carousel, or other weirdness.

- (void)reloadItemAtIndex:(NSInteger)index animated:(BOOL)animated;

This method will reload the specified item view. The new item will be requested from the dataSource. If the animated argument is YES, it will cross-fade from the old to the new item view, otherwise it will swap instantly.

Protocols

The iCarousel follows the Apple convention for data-driven views by providing two protocol interfaces, iCarouselDataSource and iCarouselDelegate. The iCarouselDataSource protocol has the following required methods (note: for Mac OS, substitute NSView for UIView in method arguments):

- (NSUInteger)numberOfItemsInCarousel:(iCarousel *)carousel;

Return the number of items (views) in the carousel.

- (UIView *)carousel:(iCarousel *)carousel viewForItemAtIndex:(NSUInteger)index reusingView:(UIView *)view;

Return a view to be displayed at the specified index in the carousel. The reusingView argument works like a UIPickerView, where views that have previously been displayed in the carousel are passed back to the method to be recycled. If this argument is not nil, you can set its properties and return it instead of creating a new view instance, which will slightly improve performance. Unlike UITableView, there is no reuseIdentifier for distinguishing between different carousel view types, so if your carousel contains multiple different view types then you should just ignore this parameter and return a new view each time the method is called. You should ensure that each time the carousel:viewForItemAtIndex:reusingView: method is called, it either returns the reusingView or a brand new view instance rather than maintaining your own pool of recyclable views, as returning multiple copies of the same view for different carousel item indexes may cause display issues with the carousel.

The iCarouselDataSource protocol has the following optional methods:

- (NSUInteger)numberOfPlaceholdersInCarousel:(iCarousel *)carousel;

Returns the number of placeholder views to display in the carousel. Placeholder views are intended to be used when the number of items in the carousel is too few to fill the carousel width, and you wish to display something in the empty space. They move with the carousel and behave just like any other carousel item, but they do not count towards the numberOfItems value, and cannot be set as the currently selected item. Placeholders are hidden when wrapping is enabled. Placeholders appear on either side of the carousel items. For n placeholder views, the first n/2 items will appear to the left of the item views and the next n/2 will appear to the right. You can have an odd number of placeholders, in which case the carousel will be asymmetrical.

- (UIView *)carousel:(iCarousel *)carousel placeholderViewAtIndex:(NSUInteger)index reusingView:(UIView *)view;

Return a view to be displayed as the placeholder view. Works the same way as carousel:viewForItemAtIndex:reusingView:. Placeholder reusingViews are stored in a separate pool to the reusingViews used for regular carousel, so it's not a problem if your placeholder views are different to the item views.

The iCarouselDelegate protocol has the following optional methods:

- (void)carouselWillBeginScrollingAnimation:(iCarousel *)carousel;

This method is called whenever the carousel will begin an animated scroll. This can be triggered programatically or automatically after the user finishes scrolling the carousel, as the carousel re-aligns itself.

- (void)carouselDidEndScrollingAnimation:(iCarousel *)carousel;

This method is called when the carousel ends an animated scroll.

- (void)carouselDidScroll:(iCarousel *)carousel;

This method is called whenever the carousel is scrolled. It is called regardless of whether the carousel was scrolled programatically or through user interaction.

- (void)carouselCurrentItemIndexDidChange:(iCarousel *)carousel;

This method is called whenever the carousel scrolls far enough for the currentItemIndex property to change. It is called regardless of whether the item index was updated programatically or through user interaction.

- (void)carouselWillBeginDragging:(iCarousel *)carousel;

This method is called when the user begins dragging the carousel. It will not fire if the user taps/clicks the carousel, or if the carousel is scrolled programmatically.

- (void)carouselDidEndDragging:(iCarousel *)carousel willDecelerate:(BOOL)decelerate;

This method is called when the user stops dragging the carousel. The willDecelerate parameter indicates whether the carousel is travelling fast enough that it needs to decelerate before it stops (i.e. the current index is not necessarily the one it will stop at) or if it will stop where it is. Note that even if willDecelerate is NO, the carousel will still scroll automatically until it aligns exactly on the current index. If you need to know when it has stopped moving completely, use the carouselDidEndScrollingAnimation delegate method.

- (void)carouselWillBeginDecelerating:(iCarousel *)carousel;

This method is called when the carousel starts decelerating. it will typically be called immediately after the carouselDidEndDragging:willDecelerate: method, assuming willDecelerate was YES.

- (void)carouselDidEndDecelerating:(iCarousel *)carousel;

This method is called when the carousel finishes decelerating and you can assume that the currentItemIndex at this point is the final stopping value. Unlike previous versions, the carousel will now stop exactly on the final index position in most cases. The only exception is on non-wrapped carousels with bounce enabled, where, if the final stopping position is beyond the end of the carousel, the carousel will then scroll automatically until it aligns exactly on the end index. For backwards compatibility, the carousel will always call scrollToItemAtIndex:animated: after it finishes decelerating. If you need to know for certain when the carousel has stopped moving completely, use the carouselDidEndScrollingAnimation delegate method.

- (CGFloat)carouselItemWidth:(iCarousel *)carousel;

Returns the width of each item in the carousel - i.e. the spacing for each item view. If the method is not implemented, this defaults to the width of the first item view that is returned by the carousel:viewForItemAtIndex:reusingView: dataSource method. This method should only be used to crop or pad item views if the views returned from carousel:viewForItemAtIndex:reusingView: are not correct (e.g. if the views are differing sizes, or include a drop shadow or outer glow in their background image that affects their size) - if you just want to space out the views a bit then it's better to use the iCarouselOptionSpacing value instead.

- (CATransform3D)carousel:(iCarousel *)carousel itemTransformForOffset:(CGFloat)offset baseTransform:(CATransform3D)transform;

This method can be used to provide a custom transform for each carousel view. The offset argument is the distance of the view from the middle of the carousel. The currently centred item view would have an offset of 0.0, the one to the right would have an offset value of 1.0, the one to the left an offset value of -1.0, and so on. To implement the linear carousel style, you would therefore simply multiply the offset value by the item width and use it as the x value of the transform. This method is only called if the carousel type is iCarouselTypeCustom.

- (CGFloat)carousel:(iCarousel *)carousel valueForOption:(iCarouselOption)option withDefault:(CGFloat)value;

This method is used to customise the parameters of the standard carousel types. By implementing this method, you can tweak options such as the number of items displayed in a circular carousel, or the amount of tilt in a coverflow carousel, as well as whether the carousel should wrap and if it should fade out at the ends, etc. For any option you are not interested in tweaking, just return the default value. The meaning of these options is listed below under iCarouselOption values. Check the Options Demo for an advanced example of using this method.

- (void)carousel:(iCarousel *)carousel didSelectItemAtIndex:(NSInteger)index;

This method will fire if the user taps any carousel item view (not including placeholder views), including the currently selected view. This method will not fire if the user taps a control within the currently selected view (i.e. any view that is a subclass of UIControl).

- (BOOL)carousel:(iCarousel *)carousel shouldSelectItemAtIndex:(NSInteger)index;

This method will fire if the user taps any carousel item view (not including placeholder views), including the currently selected view. The purpose of a method is to give you the opportunity to ignore a tap on the carousel. If you return YES from the method, or don't implement it, the tap will be processed as normal and the carousel:didSelectItemAtIndex: method will be called. If you return NO, the carousel will ignore the tap and it will continue to propagate up the view hierarchy. This is a good way to prevent the carousel intercepting tap events intended for processing by another view.

iCarouselOption values

These are the tweakable options for standard carousels. Check the Options Demo for an example of the effect that these parameters have.

iCarouselOptionWrap

A boolean indicating whether the carousel should wrap when it scrolls to the end. Return YES if you want the carousel to wrap around when it reaches the end, and NO if you want it to stop. Generally, circular carousel types will wrap by default and linear ones won't. Don't worry that the return type is a floating point value - any value other than 0.0 will be treated as YES.

iCarouselOptionShowBackfaces

For some carousel types, e.g. iCarouselTypeCylinder, the rear side of some views can be seen (iCarouselTypeInvertedCylinder now hides the back faces by default). If you wish to hide the backward-facing views you can return NO for this option. To override the default back-face hiding for the iCarouselTypeInvertedCylinder, you can return YES. This option may also be useful for custom carousel transforms that cause the back face of views to be displayed.

iCarouselOptionOffsetMultiplier

The offset multiplier to use when the user drags the carousel with their finger. It does not affect programmatic scrolling or deceleration speed. This defaults to 1.0 for most carousel types, but defaults to 2.0 for the CoverFlow-style carousels to compensate for the fact that their items are more closely spaced and so must be dragged further to move the same distance.

iCarouselOptionVisibleItems

This is the maximum number of item views (including placeholders) that should be visible in the carousel at once. Half of this number of views will be displayed to either side of the currently selected item index. Views beyond that will not be loaded until they are scrolled into view. This allows for the carousel to contain a very large number of items without adversely affecting performance. iCarousel chooses a suitable default value based on the carousel type, however you may wish to override that value using this property (e.g. if you have implemented a custom carousel type).

iCarouselOptionCount

The number of items to be displayed in the Rotary, Cylinder and Wheel transforms. Normally this is calculated automatically based on the view size and number of items in the carousel, but you can override this if you want more precise control of the carousel appearance. This property is used to calculate the carousel radius, so another option is to manipulate the radius directly.

iCarouselOptionArc

The arc of the Rotary, Cylinder and Wheel transforms (in radians). Normally this defaults to 2*M_PI (a complete circle) but you can specify a smaller value, so for example a value of M_PI will create a half-circle or cylinder. This property is used to calculate the carousel radius and angle step, so another option is to manipulate those values directly.

iCarouselOptionRadius

The radius of the Rotary, Cylinder and Wheel transforms in pixels/points. This is usually calculated so that the number of visible items exactly fits into the specified arc. You can manipulate this value to increase or reduce the item spacing (and the radius of the circle).

iCarouselOptionAngle

The angular step between each item in the Rotary, Cylinder and Wheel transforms (in radians). Manipulating this value without changing the radius will cause a gap at the end of the carousel or cause the items to overlap.

iCarouselOptionTilt

The tilt applied to the non-centered items in the CoverFlow, CoverFlow2 and TimeMachine carousel types. This value should be in the range 0.0 to 1.0.

iCarouselOptionSpacing

The spacing between item views. This value is multiplied by the item width (or height, if the carousel is vertical) to get the total space between each item, so a value of 1.0 (the default) means no space between views (unless the views already include padding, as they do in many of the example projects).

iCarouselOptionFadeMin
iCarouselOptionFadeMax
iCarouselOptionFadeRange
iCarouselOptionFadeMinAlpha

These four options control the fading out of carousel item views based on their offset from the currently centered item. FadeMin is the minimum negative offset an item view can reach before it begins to fade. FadeMax is the maximum positive offset a view can reach before if begins to fade. FadeRange is the distance over which the fadeout occurs, measured in multiples of an item width (defaults to 1.0), and FadeMinAlpha is the minimum alpha value to which the views will fade (defaults to 0.0 - fully transparent).

Detecting Taps on Item Views

There are two basic approaches to detecting taps on views in iCarousel on iOS. The first approach is to simply use the carousel:didSelectItemAtIndex: delegate method, which fires every time an item is tapped. If you are only interested in taps on the currently centered item, you can compare the currentItemIndex property against the index parameter of this method.

Alternatively, if you want a little more control you can supply a UIButton or UIControl as the item view and handle the touch interactions yourself. See the Buttons Demo example project for an example of how this is done (doesn't work on Mac OS; see below).

You can also nest UIControls within your item views and these will receive touches as expected (see the Controls Demo example project for an example).

If you wish to detect other types of interaction such as swipes, double taps or long presses, the simplest way is to attach a UIGestureRecognizer to your item view or its subviews before passing it to the carousel.

Note that taps and gestures will be ignored on any item view except the currently selected one, unless you set the centerItemWhenSelected property to NO.

On Mac OS there is no easy way to embed controls within iCarousel item views currently. You cannot just supply an NSButton as or inside your item view because the transforms applied to the item views mean that hit detection doesn't work properly. I'm investigating possible solutions to this (if you know a good way to fix this, please get in touch, or fork the project on github).

Example projects

iCarousel includes a number of example projects to help you get started. Here is a lift and brief description for each:

Basic iOS Example

This is a very simple example for iOS that demonstrates setting up a carousel with the iCarouselCoverflow2 type.

iOS Demo

This is a more complex iOS demo app that shows off all the different carousel types and additional features such as dynamic insertion/deletion of items.

Mac Demo

This is a Mac OS port of the iOS Demo example, which replicates all the same functionality.

Buttons Demo

This example demonstrates how to use UIButtons as your item views on iOS and correctly handle the events.

Controls Demo

This example demonstrates how to nest controls within your item views on iOS and correctly handle the events, as well as how to load complex item views from a nib file instead of generating them in code.

Multiple Carousels

This example demonstrates how to use multiple carousels within a single view controller.

No Nib Demo

This example demonstrates how to set up iCarousel without using a nib file on iOS.

Storyboard Demo

This example demonstrates how to set up iCarousel using Storyboards on iOS 5 and above.

Offsets Demo

This example demonstrates how to use the contentOffset and viewpointOffset properties, and the effect they have.

Options Demo

This example demonstrates how to customise the appearance of each carousel type using the iCarouselOption API.

Fading Demo

This example demonstrates how to use the iCarouselOption API to implement a nice looking fade out effect at the edges of the carousel.

Dynamic View Reflections

This example demonstrates how to use the ReflectionView class (https://github.com/nicklockwood/ReflectionView) to dynamically generate reflections for your item views. This is applicable to item views that contain subviews or controls. For item views that are just images, it's better to use the approach shown in the Dynamic Image Effects example.

Dynamic Image Effects

This example demonstrates how to use the FXImageView class (https://github.com/nicklockwood/FXImageView) to dynamically generate reflections and drop shadows for your carousel images.

Dynamic Downloads

This example demonstrates how to use the AsyncImageView class (https://github.com/nicklockwood/AsyncImageView) to dynamically download remote images and display them in a carousel without blocking the main thread or negatively affecting performance.

Downloads & Effects

This example demonstrates how to use the FXImageView class (https://github.com/nicklockwood/FXImageView) to download images on the fly and apply reflections and drop shadows to them in real time.

Swift Example

This example demonstrates how to use iCarousel with Swift instead of Objective-C.

Tables Demo

This example demonstrates how to use UITableViews inside your iCarousel item views and connect up the datasources without needing to use container controllers.

FAQ

Q. Does iCarousel support Swift?
A. Yes, check out Swift Example and Swift3 Example projects.

Q. I upgraded to the new version of iCarousel and it broke my project, how do I get the old one back?
A. Every previous release of iCarousel is tagged as a separate download on github - look in the tags tab.

Q. Can I use iCarousel without a nib file?
A. Yes, check out the *No Nib Demo* for how to set up iCarousel without nibs

Q. Can I use iCarousel with a Storyboard?
A. Yes, this is pretty much the same as using it with a nib file. Check out the *Storyboard Demo* to see how it's done.

Q. How do I prevent iCarousel item views from overflowing their bounds?
A. Set the `clipsToBounds` property to YES on your iCarousel view. You can set this property in Interface Builder by ticking the 'Clip Subviews' option.

Q. I'm getting weird issues where views turn up at the wrong points in the carousel. What's going on?
A. You're probably recycling views in your `carousel:viewForItemAtIndex:reusingView:` using the `reusingView` parameter without setting the view contents each time. Study the demo app more closely and make sure you aren't doing all your item view setup in the wrong place.

Q. I'm loading 50 images in my carousel and I keep running out of memory. How can I fix it?
A. The trick is to load the views on a background thread as the carousel is scrolling instead of loading them all in advance. Check out the *Dynamic Downloads* example for how to do this using the AsyncImageView library. The example is using remote image URLs, but the exact same approach will work just as well for locally hosted images in your app - just create local file URLs using `[NSURL fileUrlWithPath:...]`.

Q. Can I use multiple carousels in the same view controller?
A. Yes, check out the *Multiple Carousels* example for how to do this.

Q. I can't figure out how to use iCarousel in my project, is there a simple example?
A. Yes, check out the *Basic iOS Example* project for a bare-bones implementation. If you're still not clear what's going on, read up about how UITableView works, and once you understand that, iCarousel will make more sense.

Q. In the iCarouselTypeCylinder carousel, the back-side of the item views is visible. How can I hide these views?
A. You can either return NO as the value for the `iCarouselOptionShowBackfaces` option, or set the `view.layer.doubleSided` property of your item views to `NO` to hide them when they are facing backwards.

Q. What is the `reusingView` property for in the `carousel:viewForItemAtIndex:reusingView:` dataSource method?
A. You can improve iCarousel performance by recycling item views when they move offscreen instead of creating a new one each time it's needed. Check if this value is nil, and if not you can re-use this view instead of creating a new one. Note however that the view will still have any subviews or properties you added when it was first created, so be careful not to introduce leaks by re-adding those views each time. You may find it's easier and safer to ignore this paramater and create a fresh view each time if you're not sure what you are doing.

Q. If the views in my carousel all have completely different layouts, should I still use the `reusingView` parameter?
A. Probably not, and unless you have hundreds of views in your carousel, it's unlikely to be worth the trouble.

Q. How can I make iCarousel behave like a UIScrollView with paging enabled?
A. As of version 1.8, iCarousel has a pagingEnabled property that emulates the behaviour of a UIScrollView (see the *Paging Example* project). The bounce physics are not quite the same though, and you may want to consider using the SwipeView library instead (https://github.com/nicklockwood/SwipeView) which is very similar to iCarousel, but based on a UIScrollView under the hood.

Q. I want my carousel items to have a real reflection, but the reflection in the examples is just drawn on. How can I render reflections dynamically?
A. iCarousel doesn't have built-in reflection support, but you can use some additional libraries to do this. Check out the *Dynamic View Reflections* and  *Dynamic Image Effects* examples.

Q. I want to download a bunch of images on the fly and display them in my carousel. How can I do that?
A. Downloading images asynchronously and displaying them is quite complex. You can use my AsyncImageView library to simplify the process. Check out the *Dynamic Downloads* example.

Q. What if I want to download images on the fly *and* add a reflection? Can I combine the ReflectionView and AsyncImageView classes?
A. Technically yes, but if you are downloading images you'd be better off using the FXImageView class instead of ReflectionView. Check out the *Downloads & Reflections* example.

Q. The edges of my item views look jaggy. Is there any way to smooth/antialias them?
A. If you include (at least) a single pixel of transparent space around the edge of your item view images then iOS will smooth them automatically. This is because iOS automatically antialiases the pixels inside images, but doesn't antialias the edges of views. Even if your item views are a flat color, it's worth adding a background image of the same color to the views in order to get the smoothing effect.

Release Notes

Version 1.8.3

  • Fixed warnings and updated examples for Xcode 8
  • Added Swift 3 example

Version 1.8.2

  • Fixed some warnings and updated examples for Xcode 7 beta
  • Scrolling now goes the right way when using Chameleon
  • Added nullability qualifiers to improve Swift interop

Version 1.8.1

  • Fixed bug where item views could be left invible after reloading

Version 1.8

  • iCarousel now requires ARC
  • iCarousel now requires 64-bit processors on Mac OS 10.6
  • Added autoscroll property to set carousel rotating at a constant speed
  • Added pagingEnabled property to force carousel to only move a single index per swipe
  • Added itemViewAtPoint: method
  • Fixed bug which occasionally caused carousel item views to pop-in from right
  • Fixed bug where views with userInteractionEnabled = NO would get re-enabled
  • Removed all deprecated methods and APIs
  • Added iCarouselOptionFadeMinAlpha option for setting minimum fade
  • Setting scrollOffset now lets you exactly mirror behaviour from one carousel to another
  • Fixed bug where item indexes were selected incorrectly when tapping views in vertical mode
  • Perspective and viewpointOffset properties can now be animated
  • Added some minimal accessibility support
  • Single items now bounce by default (set scrollEnabled = NO to disable)
  • Now conforms to the -Weverything warning level

Version 1.7.6

  • Fixed animation timer bug when using ARC

Version 1.7.5

  • Fixed an issue with latest llvm compiler
  • Fixed conflict between iCarousel animation and UIScrollView scrolling

Version 1.7.4

  • Fixed deprecation warnings when using Xcode 4.6
  • Fixed wide items in a vertical carousel getting incorrect hitboxes
  • Added podspec file

Version 1.7.3

  • Moved ARCHelper macros into .m file to avoid affecting other classes

Version 1.7.2

  • Core animation is no longer disabled when calling delegate methods.
  • Fixed a potential divide-by-zero error when scrolling
  • Removed useDisplayLink option as the performance benefits are unclear
  • Added dragging, scrolling and decelerating properties (readonly)
  • iCarouselWrap property is now refreshed after inserting/deleting items
  • Added Autoscrolling Example
  • Added Fading Demo example

Version 1.7.1

  • Fixed issue where reloading carousel with fewer items could sometimes crash
  • Fixed issue where scrollToItemAtIndex:duration: method would sometimes scroll to the wrong index

Version 1.7

  • Renamed iCarouselTransformOption... values to "iCarouselOption", and extended the list of available options
  • Simplified carousel interface by deprecating a number of delegate methods in favour of the simpler iCarouselOption API
  • numberOfVisibleItems is now calculated automatically
  • Added Dynamic Downloads example using AsyncImageView
  • Added Dynamic Effects examples using FXImageView
  • Item view interaction area is now more precisely set on iOS
  • Fixed backface interaction issue for iCarouselTypeInvertedCylinder
  • It is now much simpler to implement alpha fading logic based on item offset
  • Now supports item click events and centering on Mac OS (buttons and controls within item views still won't work correctly receive clicks however).
  • Added scrollToOffset: and scrollByOffset: methods
  • Disabled broken scrollwheel support for carousel on Mac OS
  • currentItemIndex property is now writable
  • scrollOffset property is now writable
  • Fixed bug with scrollByNumberOfItems where duration is 0
  • Switched CADisplayLink to use NSDefaultRunLoopMode, which is less likely to interfere with UIScrollViews, etc
  • Added Storyboard example

Version 1.6.3

  • Added offsetForItemAtIndex: method
  • Fixed bug in reloadItemViewAtIndex:animated: method
  • Fixed Mac OS scrolling glitch when using numberOfVisibleViews
  • iCarousel now respects the layer.doubleSided property of item views
  • Added Basic and Multiple Carousels example projects

Version 1.6.2

  • Fixed long-standing viewpointOffset bug
  • Fixed potential bug around not setting default carousel type
  • LLVM GCC compiler is no longer supported, Now requires Apple LLVM compiler
  • Upgraded to latest ARC Helper code

Version 1.6.1

  • Added automatic support for ARC compile targets
  • Now compiles correctly again under LLVM GCC 4.2
  • Vertical Time Machine carousel is now right-way-up on Mac OS
  • Added Inverted Time Machine carousel type
  • Added dynamic reflections example
  • Fixed crashing bug in examples on iPad
  • Carousels now behave better with 1 or 2 items

Version 1.6

  • Added support for item view recycling
  • Carousels can now be either horizontal or vertical
  • Added Wheel and Time Machine carousel types
  • Added new iCarouselTransformOption system for tweaking the standard carousel transforms without having to provide a completely bespoke implementation
  • Added ignorePerpendicularSwipes property
  • Fixed issue with scrolling immediately after reloading
  • Added useDisplayLink toggle to manually force use of NSTimer
  • Added support for the Chameleon iOS-to-Mac porting library
  • Removed the deprecated visibleViews property
  • Added carousel:alphaForViewAtItemWithOffset: delegate method for controlling view opacity
  • Added indexOfItemViewOrSubview: method to simplify handling of controls within carousel item views
  • Deprecated/renamed some dataSource and delegate methods - check your projects for compatibility
  • Expanded examples and tests

Version 1.5.8

  • Fixed bug in previous UITableCell fix
  • No longer gets stuck at a negative offset when inserting items into an empty carousel
  • Example app no longer crashes when inserting item into empty carousel
  • Better behaviour when reloading carousel

Version 1.5.7

  • Fixed ARC compatibility issues with sorting logic
  • UISwitches, UISliders and UITableCells now work correctly with item views
  • Fixed bug in carousel:shouldSelectItemAtIndex: delegate logic

Version 1.5.6

  • Added reloadItemAtIndex:animated: method.
  • Fixed some issues when setting offset or carousel bounds on the fly.
  • Less aggressive use of [CATransaction setDisableActions:YES] means more properties of carousel are now animatable (including setting the type).
  • Fixed NaN bug when displaying an empty carousel
  • Fixed glitch when programatically scrolling CoverFlow2-type carousel.
  • Cylinder and rotary carousel are now sized to fit visible number of items.

Version 1.5.5

  • Deprecated visibleViews property
  • Added visibleItemViews property, which is an array
  • Added indexesForVisibleItems property
  • Added itemViewAtIndex: method
  • Added currentItemView property
  • Added indexOfItemView: method
  • Fixed glitch when unwrapped CoverFlow2-type carousels reach the far right
  • Scroll animation events are now called immediately after scrolling ends if the carousel does not need to scroll a significant distance.
  • Fixed jerky bounce animation on CoverFlow2-type carousels when stopAtItemBoundary is set to NO.

Version 1.5.4

  • Fixed a bug where insertItemAtIndex method would not allow items to be inserted at the rightmost end of the carousel.

Version 1.5.3

  • Fixed a bug on wrapped carousels when the total number of carousel items exceeds the number of visible items.
  • Changed numberOfVisibleItems property to be a dataSource method, removing the arbitrary default limit of 21.
  • Fixed a flickering issue on CoverFlow2 carousel type.
  • Fixed bug on Mac where clicking would spin the carousel a random distance.
  • scrollSpeed is now a read/write property, and only affects speed when carousel is flicked.
  • Removed carouselScrollSpeed delegate method and replaced it with new offsetMultiplier property and carouselOffsetMultiplier delegate method to control the offset when dragging.
  • scrollOffset property is now public (readonly).
  • Floating point arguments and properties are now CGFloats instead of floats.

Version 1.5.2

  • Added bounceDistance property for finer control over bounce behaviour.
  • Added carousel:shouldSelectItemAtIndex: delegate method to allow carousel to selectively ignore taps.
  • Added stopAtItemBoundary and scrollToItemBoundary properties.
  • Fixed issue with carousel wrapping unexpectedly when wrapping and bouncing are disabled.
  • Fixed issue with rightmost visible view sometimes not loading.
  • Improved the iCarouselTypeCoverFlow2 implementation.
  • Tweaked acceleration parameters for smoother scrolling behaviour.
  • Improved loading sequence to reduce repeated calls to dataSource methods during startup.
  • Added No-nib example for iPhone to demonstrate setting up the carousel in code.

Version 1.5.1

  • Fixed issue with item button events being blocked by gesture recogniser.
  • Tweaked bounce and acceleration parameters for smoother scrolling.
  • Made toggle property public (used to implement CoverFlow2 carousel style).

Version 1.5

  • Added a new carousel type, iCarouselTypeCoverflow2, which more closely matches the appearance of the standard Apple CoverFlow implementation.
  • iCarousel now always stops exactly on an item boundary when decelerating, instead of over or under-shooting and scrolling to the correct position.
  • Added carouselScrollSpeed delegate method and increased default scrolling speed for CoverFlow carousel types.
  • Animation now uses CADisplayLink on iOS for better performance.
  • Animation is now paused when carousel is not moving.
  • Reduced default deceleration rate so carousel can be 'flicked' further.
  • Added insert/remove carousel item support for Mac OS.

Version 1.4

  • Added a new dynamic loading system to load and release views as needed. The carousel can now contain hundreds of thousands of items but only display a subset of them.
  • Added numberOfVisibleItems property to control the number of views to be loaded concurrently.
  • Added visibleViews property, containing a set of all visible item views.
  • Removed the itemViews and placeholderViews properties, as these no longer work with the new loading system.

Version 1.3.4

  • Changed order of execution for when carouselDidEndScrollingAnimation is called to prevent animation glitches when making changes to carousel in the callback.
  • Fixed glitch where didSelectItemAtIndex method would never fire for the currently centered view.

Version 1.3.3

  • Added several additional delegate methods for tracking when the carousel scrolls and decelerates.
  • Fixed some glitches when using the scrollwheel/trackpad on a Mac.

Version 1.3.2

  • Fixed additional scrolling bug introduced by 1.3.1 fix.
  • Improved deceleration logic on non-wrapped carousels.

Version 1.3.1

  • Fixed scrolling bug on wrapped carousels when scrolling from item at index zero back to the last item.

Version 1.3

  • Added Mac OS support. Most - but not all - features from the iOS version are supported (see readme.md for details. Thanks go to Sushant Prakash for proving the concept).
  • Added centerItemWhenSelected property to disable the auto-centering behaviour when tapping on item views.
  • Added carousel:didSelectItemAtIndex: delegate method which is called when a user taps any item view in the carousel.
  • Changed placeholder view behaviour slightly to be more flexible and intuitive.
  • Bug and performance fixes to the depth sorting and positioning logic.
  • Extended the example project to demonstrate the use of placeholder views and the new delegate method.
  • Converted documentation and license files to markdown format.

Version 1.2.4

  • Added ability to specify scroll animation duration.
  • Added scrollByNumberOfItems method to programmatically scroll the carousel by a specified distance in either direction.
  • Fixed retain cycle that prevented carousel from being deallocated properly.

Version 1.2.3

  • Added contentOffset and viewpointOffset properties for adjusting layout and perspective.
  • Made contentView property public.
  • Fixed issue when inserting items into an empty carousel.
  • Fixed issue when scrolling more than one step in a wrapped carousel.

Version 1.2.2

  • Removed an iOS 4-specific API so that iCarousel will work on iOS 3.2.
  • Fixed crash when removing last item in a carousel or inserting items into an empty carousel.

Version 1.2.1

  • Fixed scrolling issue with carousel when last item is removed.
  • Added insert/delete buttons to example project.

Version 1.2

  • Smooth scrolling and acceleration, no longer based on UIScrollView.
  • Added decelerationRate and bounces properties.
  • Tap any visible item in the carousel to center it.
  • Fixed crash on memory warning.
  • Added wrap on/off selector to example.
  • Example project is now a universal app.

Version 1.1.2

  • Smoother animation when inserting items into carousel.
  • iCarousel no longer modifies item view alpha when adding/removing items.

Version 1.1.1

  • Fixed bug when inserting items into the carousel without animation.
  • Fixed page scrolling bug on iOS 4.2 and earlier.

Version 1.1

  • Added itemViews property.
  • Added perspective property.
  • Added scrollEnabled property.
  • Added placeholder views feature.
  • Fixed vanishing views bug in example project.

Version 1.0

  • Initial release.
Comments
  • Loading Multiple Nib's into the one carousel

    Loading Multiple Nib's into the one carousel

    Hi there,

    I'm loving iCarousel, it's an unbelievable powerful too, but there's one thing I'm unable to get my head around and that's loading 2 or more different nibs into the one carousel.

    Is this possible at all to do? I'm imagining that I have to insert code in

    - (UIView *)carousel:(iCarousel *)carousel viewForItemAtIndex:(NSUInteger)index reusingView:(UIView *)view
    

    but i've no idea how to accomplish my task. Would you be able help please?

    Also, is it possible to change the item that the app opens on, eg: currently the app (the controls demo), opens on item 1, is it possible to open it on say item 7? Is that just a reference to the index path?

    opened by Mark117 36
  • Retain View

    Retain View

    Hi sir Nick, when deleting iCarousel view, like this: [carousel removeItemAtIndex:index animated:YES];

    How to maintain/Retain the deleted views/index even going to different viewController?. Because in my experience of it, it returns its view back, Or putting them in the array? is this possible? thanks for the help.

    opened by KobeCena 26
  • iCarousel Index on Sender starts back at 0 after 3

    iCarousel Index on Sender starts back at 0 after 3

    Hi,

    Im working with your library and i have to say its awesome! I have been running into some hurdles recently and i think you will be bale to help me.

    I am trying to play one instance of the icarousel at a time and i figure i would do that with the index being passed through my sender. Though i get this outcome

    0 1 2 3 0 1 2 0 1 2

    I have 10 objects,what could be causing this?

    opened by keeano92 25
  • GestureRecognizer disabled after resizing a custom UIView in iCarousel

    GestureRecognizer disabled after resizing a custom UIView in iCarousel

    Hello,

    I have an UIViewController with an UIView. The Views are in the carousel an work fine with all functionality! But after a resize with the setFrame method the GestureRecognizer in the UIView dont react...

        UITapGestureRecognizer *TapGesture = [[UITapGestureRecognizer alloc]
                                              initWithTarget:self action:@selector(handleTapforNewImage:)];
        TapGesture.numberOfTapsRequired = 1;
        [TapGesture setDelegate:self];
        ImageView.userInteractionEnabled = YES;
        [ImageView addGestureRecognizer:TapGesture];
    
    opened by Philsen 23
  • Repeated views using FXImageView as viewForItemAtIndex

    Repeated views using FXImageView as viewForItemAtIndex

    After installing the latest version of FXImageView (which is great by the way), I have noticed that occasionally I am seeing repeated images displayed in my carousel. This can happen while scrolling and when I come to a stop, an image appears to not update from it's previous version. I have also seen it when loading a brand new carousel, with the first item having the wrong image.

    I am curious if this is indicative of a cache problem or a background processing issue? Any thoughts on how best to determine the root cause? I haven't been able to obtain repeatability yet unfortunately. Cheers,

    opened by MobileVet 23
  • Index 0 always called in viewForItemAtIndex: reusingView:

    Index 0 always called in viewForItemAtIndex: reusingView:

    I'm using my iCarousel a little different where I have a tableview that when a cell is selected, that index of the tableviewcell sets the currentIndex for the carousel. This all works fine, until it goes to display the carousel and calls the viewForItemAtIndex: reusingView:.

    Everytime this method is called, the first index it loads is always 0, which I believe is causing an issue when displaying my feed items. For example, lets say I select the 12th item in my tableview, it will load the following:

    0, 10, 11, 12, 13, 14

    When it does this, it makes one of the views index 0's object and messes everything up.

    opened by jnpetersen 22
  • Calling reloadData causes views to overlap when scrolled

    Calling reloadData causes views to overlap when scrolled

    When I call reloadData on the carousel, everything is refreshed. When I scroll though, the old views are still there and overlap with the new views. Here are two pictures (before and after):

    www.chillysky.com/Before.png www.chillysky.com/After.png

    Any idea why this is happening?

    opened by jackhumphries 22
  • Make Next and Previous image blur

    Make Next and Previous image blur

    Hi Nik,

    I am using this library in my app. I am using iCarouselTypeLinear type. What I want is, make previous and upcoming next image blur. I know how to make blur. But I want to know how I can get previous and next view while scrolling.

    Also, when should I process the image after scroll did end or while scrolling ?

    Any suggestions ?

    Thanks.

    opened by maulikpat 18
  • Can't apply to UITableView Cell

    Can't apply to UITableView Cell

    I really like iCarousel and I must admit @nicklockwood you sir have done a great job!

    However, I have one problem, I can't get iCarousel to apply to the UITableView cells INSTEAD of the UITableView itself. Is it possible to apply iCarousel to UiTableView cells instead??

    Here is my code:

    -(void)createIcarousel {
        categorySubView = [[iCarousel alloc]initWithFrame:CGRectMake(160, 274, 480, 125)];
    
        // CategorySubView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
        categorySubView.delegate = self;
        categorySubView.dataSource = self;
        categorySubView.type = iCarouselTypeCoverFlow2;
        [self.view addSubview:categorySubView];
    }
    
    -(NSUInteger)numberOfItemsInCarousel:(iCarousel *)carousel {
        //return [episodes count];
        return 10;
    }
    
    -(UIView *) carousel:(iCarousel *)carousel viewForItemAtIndex:(NSUInteger)index reusingView:(UIView *)view {
        //UITableView *sampleTable = [[UITableView alloc] initWithFrame:CGRectMake(0, 0, 150, 150) style:UITableViewStylePlain];
        //sampleTable.dataSource = self;
        //sampleTable.delegate = self;
        //[sampleTable setBackgroundColor:[UIColor blackColor]];
    
        return jsonTable;
    }
    
    -(BOOL)carousel:(iCarousel *)carousel shouldSelectItemAtIndex:(NSInteger)index{
        return YES;
    }
    
    -(CGFloat)carouselItemWidth:(iCarousel *)carousel {
        // Usually this should be slightly wider than the item views
        return 180;
    }
    
    -(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
        // Delegate call back for cell at index path.
    
        static NSString *CellIdentifier = @"Cell";
        CustomCell *cell = (CustomCell *)[tableView dequeueReusableCellWithIdentifier:CellIdentifier];
    
        if (cell == nil) {
            NSArray *nib = [[NSBundle mainBundle] loadNibNamed:@"CustomCell" owner:self options:nil];
            cell = [nib objectAtIndex: 0];
        }
    
        cell.username.text = [[[episodes objectAtIndex:indexPath.row] valueForKey:@"user"] valueForKey:@"username"];
    
        cell.username.clipsToBounds = YES;
        cell.profilepic.clipsToBounds = YES;
        cell.geo.clipsToBounds = YES;
        cell.picture.clipsToBounds = YES;
        cell.contentView.clipsToBounds = NO;
        cell.clipsToBounds = YES;
    
        return cell;
    }
    
    opened by Supertecnoboff 18
  • iCarouselTypeLinear spacing between images

    iCarouselTypeLinear spacing between images

    Is there any way to maintain distance between images when using iCarouselTypeLinear, currently all images looks attached each other. And also adjust the size of the middle / center image a bit larger than the rest with a border wrapped around indicating the current selection. Please let me know how to do this. Thanks, Mohan.

    opened by mohan-agadkar 18
  • EXC_CRASH at -[iCarousel stopAnimation] (iCarousel.m:1696)

    EXC_CRASH at -[iCarousel stopAnimation] (iCarousel.m:1696)

    I've come across this a few times, and I'm still trying to see if I can make the crash consistent or figure out what might cause the problem, but the behavior in the UI is a tell-tale that this crash is imminent.

    What happens is that the Carousel (I'm using a custom type) stops doing its timed rotation animation — i.e. swiping allows the carousel to move, but it only moves under my finger. It doesn't continue moving with momentum. So, in order to make something move, I have to swipe it continuously: as soon as I stop dragging my finger, the carousel will stop dead.

    I can keep doing this through my data set no problem (although it's not the desired behavior, obviously.) And, as soon as I remove the view containing the carousel (swiping a target view elsewhere on the screen), the app crashes. The backtrace I captured is below, and I'm guessing that _timer is hinky — perhaps nil.

    I wonder if anyone else has experienced this behavior. Like I said, it's not consistently repeatable, so I'm not at all sure what might be causing it, but I suspect something happens to the animation timer at some point.

    The line in my source is the [_timer invalidate] one (my fork, but I'm pretty sure it's operationally identical to the current master)

    - (void)stopAnimation
    {
        [_timer invalidate];
        _timer = nil;
    }
    

    Julian

    Last Exception Backtrace:
    0   CoreFoundation                  0x3a8683e2 __exceptionPreprocess + 158
    1   libobjc.A.dylib                 0x38f7395e objc_exception_throw + 26
    2   CoreFoundation                  0x3a86bf2c -[NSObject(NSObject) doesNotRecognizeSelector:] + 180
    3   CoreFoundation                  0x3a86a648 ___forwarding___ + 388
    4   CoreFoundation                  0x3a7c2204 _CF_forwarding_prep_0 + 20
    5   Humans                          0x0015b67c -[iCarousel stopAnimation] (iCarousel.m:1696)
    6   Humans                          0x0015c672 -[iCarousel didMoveToSuperview] (iCarousel.m:1858)
    7   UIKit                           0x3474ddba __UIViewWasRemovedFromSuperview + 214
    8   UIKit                           0x34596906 -[UIView(Hierarchy) removeFromSuperview] + 154
    9   UIKit                           0x345970fc -[UIView dealloc] + 292
    10  libobjc.A.dylib                 0x38f70490 (anonymous namespace)::AutoreleasePoolPage::pop(void*) + 164
    11  CoreFoundation                  0x3a7ab82c _CFAutoreleasePoolPop + 12
    12  CoreFoundation                  0x3a83c1b2 __CFRunLoopRun + 1290
    13  CoreFoundation                  0x3a7af238 CFRunLoopRunSpecific + 352
    14  CoreFoundation                  0x3a7af0c4 CFRunLoopRunInMode + 100
    15  GraphicsServices                0x36ab5336 GSEventRunModal + 70
    16  UIKit                           0x345bc28c UIApplicationMain + 1116
    17  Humans                          0x0006a850 main (main.m:16)
    18  Humans                          0x0006a7ec start + 36
    
    opened by bleeckerj 17
  • Cycling custom type animation

    Cycling custom type animation

    Does anyone know how I can cycle custom type?

    Layout my view looks like this:

    image

    I use a custom type from here: Link

    
    func carousel(_ carousel: iCarousel, itemTransformForOffset offset: CGFloat, baseTransform transform: CATransform3D) -> CATransform3D {
                let centerItemZoom: CGFloat = 1.329
                let centerItemSpacing: CGFloat = 1.41
            
                let spacing: CGFloat = 0.87
                let absClampedOffset = min(1.0, abs(offset))
                let clampedOffset = min(1.0, max(-1.0, offset))
                let scaleFactor = 1.0 + absClampedOffset * (1.0/centerItemZoom - 1.0)
                let offset = (scaleFactor * offset + scaleFactor * (centerItemSpacing - 1.0) * clampedOffset) * carousel.itemWidth * spacing
                var transform = CATransform3DTranslate(transform, offset, 0.0, -absClampedOffset)
                transform = CATransform3DScale(transform, scaleFactor, scaleFactor, 1.0)
                transform = CATransform3DTranslate(transform, 0, 11, 0)
                return transform
            }
    
    opened by maukur 0
  • tvos icarousel

    tvos icarousel

    hello, please Is there any news or support regarding icarousel for tvos ? i test it, it showing but can't be focused if some one can help please. thank you

    opened by mirco1000 0
  • iOS 13 deprecate in iCarousel.m new warnings due to use of old animation method

    iOS 13 deprecate in iCarousel.m new warnings due to use of old animation method

    'beginAnimations:context:' is deprecated: first deprecated in iOS 13.0 - Use the block-based animation API instead 'setAnimationDelegate:' is deprecated: first deprecated in iOS 13.0 - Use the block-based animation API instead 'beginAnimations:context:' is deprecated: first deprecated in iOS 13.0 - Use the block-based animation API instead

    opened by KaranVakharia789 0
  • Support Swift Package Manager

    Support Swift Package Manager

    Proposed changes

    Support framework distribution with Swift Package Manager

    Types of changes

    [ ] Bugfix (a non-breaking change that fixes an issue) [x] New feature (non-breaking change which adds functionality) [x] Breaking change (fix or feature that would cause existing functionality to not work as expected) [x] Documentation Update (if none of the other choices apply)

    Checklist

    [x] I have read the CONTRIBUTING doc [x] Lint and unit tests pass locally with my changes [x] I have added tests that prove my fix is effective or that my feature works [x] I have added necessary documentation (if appropriate) [x] Any dependent changes have been merged and published in downstream modules

    How Was This Tested?

    • Initiate an iCarousel example project
    • Put some code that demonstrates how iCarousel works
    • Install iCarousel via Swift Package Manager associated with my forked repository
    • Build and run the example project successfully
    • Uninstall iCarousel via Swift Package Manager
    • Install iCarousel via Cocoapods
    • Build and run the example project successfully

    The example project locale can be found here

    opened by tphduy 2
  • Carousel inside reusable UITableViewCell, carousel views shuffled

    Carousel inside reusable UITableViewCell, carousel views shuffled

    Hi,

    I'm using iCarousel inside reusable UITableViewCell. When I change current item of carousel inside one UITableView cell, and scroll down the tableview, upcoming carousel's current item index is already changed to previous carousel's index.

    I've tried the UITableViewCell's prepareForReuse method,

    override func prepareForReuse() { super.prepareForReuse() postViewHeightConstraint.constant = 0 carousel.invalidateIntrinsicContentSize() }

    setting datasource and delegate nil in prepareForReuse renders empty carousel.

    opened by rajdhakate 0
📖 A simple, highly informative page view controller

TL;DR UIPageViewController done properly. ⭐️ Features Simplified data source management & enhanced delegation. Dynamically insert & remove pages. Infi

UI At Six 1.8k Dec 24, 2022
A highly configurable and out-of-the-box-pretty UI library

We absolutely love beautiful interfaces! As an organization named Unicorn, we are obligated to be unique and majestic.

Clayton (McIlrath) Unicorn 225 Feb 11, 2022
SwiftUI components and extensions that seem to be highly reusable

SwiftUI components and extensions that seem to be highly reusable

Yusuke Hosonuma 56 Dec 15, 2022
Declarative, configurable & highly reusable UI development as making Lego bricks.

LeeGo is a lightweight Swift framework that helps you decouple & modularise your UI component into small pieces of LEGO style's bricks, to make UI dev

WANG Shengjia 969 Dec 29, 2022
Highly customizable Action Sheet Controller with Assets Preview written in Swift

PPAssetsActionController Play with me ▶️ ?? If you want to play with me, just tap here and enjoy! ?? ?? Show me ?? Try me ?? The easiest way to try me

Pavel Pantus 72 Feb 4, 2022
Easy to use, highly customizable gauge view

GDGauge - Customizable Gauge View Requirements Xcode 11+ Swift 5 iOS 9+ Installation Swift Package Manager .package(url: "https://github.com/saeid/GDG

Saeid 74 Dec 5, 2022
DivKit is an open source Server-Driven UI (SDUI) framework.

DivKit ?? is an open source Server-Driven UI (SDUI) framework. It allows you to roll out server-sourced updates to different app versions. Also, it can be used for fast UI prototyping, allowing you to write a layout once and then ship it to iOS, Android, and Web platforms.

DivKit 1.6k Jan 9, 2023
List tree data souce to display hierachical data structures in lists-like way. It's UI agnostic, just like view-model and doesn't depend on UI framework

SwiftListTreeDataSource List tree data souce to display hierachical data structures in lists-like way. It's UI agnostic, just like view-model, so can

Dzmitry Antonenka 26 Nov 26, 2022
Work in progress gallery of controls available to Catalyst apps using Optimized for Mac

Catalyst Controls Gallery Very simple work-in-progress demonstration of many common controls available to Mac Catalyst as of macOS 11. Provided moreso

Steven Troughton-Smith 163 Sep 18, 2022
Creating a blurred window background in Mac Catalyst

TransparentChrome In response to a developer question, I looked for the easiest way to provide a translucent full-window chrome with a blurred backgro

Steven Troughton-Smith 37 Dec 2, 2022
Super awesome Swift minion for Core Data (iOS, macOS, tvOS)

⚠️ Since this repository is going to be archived soon, I suggest migrating to NSPersistentContainer instead (available since iOS 10). For other conven

Marko Tadić 306 Sep 23, 2022
Basic iOS app to track stocks (data from Finnhub, Tiingo, or IEX Cloud)

Basic iOS app to track stocks (data from Finnhub, Tiingo, or IEX Cloud)

null 33 Dec 21, 2022
A fancy hexagonal layout for displaying data like your Apple Watch

Hexacon is a new way to display content in your app like the Apple Watch SpringBoard Highly inspired by the work of lmmenge. Special thanks to zenly f

Gautier Gédoux 340 Dec 4, 2022
Beautiful animated placeholders for showing loading of data

KALoader Create breautiful animated placeholders for showing loading of data. You can change colors like you want. Swift 4 compatible. Usage To add an

Kirill Avery 105 May 2, 2022
Create SwiftUI Views with any data

Create SwiftUI Views with any data

Zach Eriksen 20 Jun 27, 2022
A multi-platform SwiftUI component for tabular data

SwiftTabler A multi-platform SwiftUI component for tabular data. NOTE this component is BRAND NEW and under active development. If you need stability,

OpenAlloc 45 Jan 3, 2023
Creating a simple selectable tag view in SwiftUI is quite a challenge. here is a simple & elegant example of it.

SwiftUI TagView Creating a simple selectable tag view in SwiftUI is quite a challenge. here is a simple & elegant example of it. Usage: Just copy the

Ahmadreza 16 Dec 28, 2022
🏞 A simple iOS photo and video browser with optional grid view, captions and selections written in Swift5.0

Introduction ?? MediaBrowser can display one or more images or videos by providing either UIImage objects, PHAsset objects, or URLs to library assets,

Kyle Yi 631 Dec 29, 2022
A simple and elegant UIKit for iOS.

HamsterUIKit A simple and elegant UIKit(Chart) for iOS, written in Swift. ?? Curve and bar Charts. ?? Protocols are designed based on UIKit(UITableVie

Howard Wang 30 Oct 2, 2022