| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560 |
- #import "SCIndexView.h"
- #define kSCIndexViewSpace (self.configuration.indexItemHeight + self.configuration.indexItemsSpace)
- #define kSCIndexViewMargin ((self.bounds.size.height - kSCIndexViewSpace * self.dataSource.count) / 2)
- #define kSCIndexViewInsetTop (self.translucentForTableViewInNavigationBar ? UIApplication.sharedApplication.statusBarFrame.size.height + 44 : 0)
- static NSTimeInterval kAnimationDuration = 0.25;
- // 根据section值获取TextLayer的中心点y值
- static inline CGFloat SCGetTextLayerCenterY(NSUInteger position, CGFloat margin, CGFloat space)
- {
- return margin + (position + 1.0 / 2) * space;
- }
- // 根据y值获取TextLayer的section值
- static inline NSInteger SCPositionOfTextLayerInY(CGFloat y, CGFloat margin, CGFloat space)
- {
- CGFloat position = (y - margin) / space - 1.0 / 2;
- if (position <= 0) return 0;
- NSUInteger bigger = (NSUInteger)ceil(position);
- NSUInteger smaller = bigger - 1;
- CGFloat biggerCenterY = SCGetTextLayerCenterY(bigger, margin, space);
- CGFloat smallerCenterY = SCGetTextLayerCenterY(smaller, margin, space);
- return biggerCenterY + smallerCenterY > 2 * y ? smaller : bigger;
- }
- @interface SCTextLayer : CATextLayer
- @property (nonatomic, strong) UIFont *itemFont;
- @end
- @implementation SCTextLayer
- - (void)drawInContext:(CGContextRef)context {
- CGFloat height = self.bounds.size.height;
- CGFloat fontSize = self.itemFont.lineHeight;
- CGFloat yOffset = (height - fontSize) / 2;
-
- CGContextSaveGState(context);
- CGContextTranslateCTM(context, 0, yOffset);
- [super drawInContext:context];
- CGContextRestoreGState(context);
- }
- - (void)setItemFont:(UIFont *)itemFont {
- _itemFont = itemFont;
- self.font = (__bridge CFTypeRef _Nullable)(itemFont.fontName);
- self.fontSize = itemFont.pointSize;
- }
- @end
- @interface SCIndexView ()
- @property (nonatomic, strong, nullable) CAShapeLayer *searchLayer;
- @property (nonatomic, strong) NSMutableArray<SCTextLayer *> *subTextLayers;
- @property (nonatomic, strong) UILabel *indicator;
- @property (nonatomic, weak) UITableView *tableView;
- // 触摸索引视图
- @property (nonatomic, assign, getter=isTouchingIndexView) BOOL touchingIndexView;
- // 触感反馈
- @property (nonatomic, strong) UIImpactFeedbackGenerator *generator NS_AVAILABLE_IOS(10_0);
- @end
- @implementation SCIndexView
- #pragma mark - Life Cycle
- - (instancetype)initWithTableView:(UITableView *)tableView configuration:(SCIndexViewConfiguration *)configuration
- {
- if (self = [super initWithFrame:tableView.frame]) {
- _tableView = tableView;
- _currentSection = NSUIntegerMax;
- _configuration = configuration;
- _translucentForTableViewInNavigationBar = YES;
-
- [self addSubview:self.indicator];
- }
- return self;
- }
- - (void)layoutSubviews {
- [super layoutSubviews];
-
- CGFloat space = kSCIndexViewSpace;
- CGFloat margin = kSCIndexViewMargin;
-
- [CATransaction begin];
- [CATransaction setDisableActions:YES];
- if (self.searchLayer && !self.searchLayer.hidden) {
- self.searchLayer.frame = CGRectMake(self.bounds.size.width - self.configuration.indexItemRightMargin - self.configuration.indexItemHeight, SCGetTextLayerCenterY(0, margin, space) - self.configuration.indexItemHeight / 2, self.configuration.indexItemHeight, self.configuration.indexItemHeight);
- self.searchLayer.cornerRadius = self.configuration.indexItemHeight / 2;
- self.searchLayer.contentsScale = UIScreen.mainScreen.scale;
- self.searchLayer.backgroundColor = self.configuration.indexItemBackgroundColor.CGColor;
- }
-
- NSInteger deta = self.searchLayer ? 1 : 0;
- for (int i = 0; i < self.subTextLayers.count; i++) {
- SCTextLayer *textLayer = self.subTextLayers[i];
- NSUInteger section = i + deta;
- textLayer.frame = CGRectMake(self.bounds.size.width - self.configuration.indexItemRightMargin - self.configuration.indexItemHeight, SCGetTextLayerCenterY(section, margin, space) - self.configuration.indexItemHeight / 2, self.configuration.indexItemHeight, self.configuration.indexItemHeight);
- }
- [CATransaction commit];
- }
- #pragma mark - Public Methods
- - (void)refreshCurrentSection {
- [self onActionWithScroll];
- }
- #pragma mark -
- - (void)configSubLayersAndSubviews
- {
- BOOL hasSearchLayer = [self.dataSource.firstObject isEqualToString:UITableViewIndexSearch];
- NSUInteger deta = 0;
- if (hasSearchLayer) {
- if (!self.searchLayer) {
- self.searchLayer = [self createSearchLayer];
- [self.layer addSublayer:self.searchLayer];
- }
- self.searchLayer.hidden = NO;
- deta = 1;
- } else if (self.searchLayer) {
- self.searchLayer.hidden = YES;
- }
-
- NSInteger countDifference = self.dataSource.count - deta - self.subTextLayers.count;
- if (countDifference > 0) {
- for (int i = 0; i < countDifference; i++) {
- SCTextLayer *textLayer = [SCTextLayer layer];
- [self.layer addSublayer:textLayer];
- [self.subTextLayers addObject:textLayer];
- }
- } else {
- for (int i = 0; i < -countDifference; i++) {
- SCTextLayer *textLayer = self.subTextLayers.lastObject;
- [textLayer removeFromSuperlayer];
- [self.subTextLayers removeObject:textLayer];
- }
- }
-
- CGFloat space = kSCIndexViewSpace;
- CGFloat margin = kSCIndexViewMargin;
-
- [CATransaction begin];
- [CATransaction setDisableActions:YES];
-
- if (hasSearchLayer) {
- self.searchLayer.frame = CGRectMake(self.bounds.size.width - self.configuration.indexItemRightMargin - self.configuration.indexItemHeight, SCGetTextLayerCenterY(0, margin, space) - self.configuration.indexItemHeight / 2, self.configuration.indexItemHeight, self.configuration.indexItemHeight);
- self.searchLayer.cornerRadius = self.configuration.indexItemHeight / 2;
- self.searchLayer.contentsScale = UIScreen.mainScreen.scale;
- self.searchLayer.backgroundColor = self.configuration.indexItemBackgroundColor.CGColor;
- }
-
- for (int i = 0; i < self.subTextLayers.count; i++) {
- SCTextLayer *textLayer = self.subTextLayers[i];
- NSUInteger section = i + deta;
- textLayer.frame = CGRectMake(self.bounds.size.width - self.configuration.indexItemRightMargin - self.configuration.indexItemHeight, SCGetTextLayerCenterY(section, margin, space) - self.configuration.indexItemHeight / 2, self.configuration.indexItemHeight, self.configuration.indexItemHeight);
- textLayer.string = self.dataSource[section];
- textLayer.itemFont = self.configuration.indexItemTextFont;
- textLayer.cornerRadius = self.configuration.indexItemHeight / 2;
- textLayer.alignmentMode = kCAAlignmentCenter;
- textLayer.contentsScale = UIScreen.mainScreen.scale;
- textLayer.backgroundColor = self.configuration.indexItemBackgroundColor.CGColor;
- textLayer.foregroundColor = self.configuration.indexItemTextColor.CGColor;
- }
- [CATransaction commit];
-
- if (self.subTextLayers.count == 0) {
- self.currentSection = NSUIntegerMax;
- } else if (self.currentSection == NSUIntegerMax) {
- self.currentSection = self.searchLayer ? SCIndexViewSearchSection : 0;
- } else {
- self.currentSection = self.subTextLayers.count - 1;
- }
- }
- - (void)configCurrentSection
- {
- NSInteger currentSection = SCIndexViewInvalidSection;
- if (self.delegate && [self.delegate respondsToSelector:@selector(sectionOfIndexView:tableViewDidScroll:)]) {
- currentSection = [self.delegate sectionOfIndexView:self tableViewDidScroll:self.tableView];
- if ((currentSection >= 0 && currentSection != SCIndexViewInvalidSection)
- || currentSection == SCIndexViewSearchSection) {
- self.currentSection = currentSection;
- return;
- }
- }
-
- NSInteger firstVisibleSection = self.tableView.indexPathsForVisibleRows.firstObject.section;
- CGFloat insetTop = kSCIndexViewInsetTop;
- for (NSInteger section = firstVisibleSection; section < self.tableView.numberOfSections; section++) {
- CGRect sectionFrame = [self.tableView rectForSection:section];
- if (sectionFrame.origin.y + sectionFrame.size.height - self.tableView.contentOffset.y > insetTop) {
- currentSection = section;
- break;
- }
- }
-
- BOOL selectSearchLayer = NO;
- if (currentSection == 0 && self.searchLayer && currentSection < self.tableView.numberOfSections) {
- CGRect sectionFrame = [self.tableView rectForSection:currentSection];
- selectSearchLayer = (sectionFrame.origin.y - self.tableView.contentOffset.y - insetTop) > 0;
- }
-
- if (selectSearchLayer) {
- currentSection = SCIndexViewSearchSection;
- }
- else {
- currentSection = currentSection - self.startSection;
- }
-
- self.currentSection = currentSection;
- }
- #pragma mark - Event Response
- - (void)onActionWithDidSelect
- {
- if ((self.currentSection < 0 && self.currentSection != SCIndexViewSearchSection)
- || self.currentSection >= (NSInteger)self.subTextLayers.count) {
- return;
- }
-
- CGFloat insetTop = kSCIndexViewInsetTop;
- if (self.currentSection == SCIndexViewSearchSection) {
- [self.tableView setContentOffset:CGPointMake(0, -insetTop) animated:NO];
- } else {
- NSInteger currentSection = self.currentSection + self.startSection;
- if (currentSection >= 0 && currentSection < self.tableView.numberOfSections) {
- NSUInteger rowCountInSection = [self.tableView numberOfRowsInSection:currentSection];
- if (rowCountInSection > 0) {
- NSIndexPath *indexPath = [NSIndexPath indexPathForRow:0 inSection:currentSection];
- [self.tableView scrollToRowAtIndexPath:indexPath atScrollPosition:UITableViewScrollPositionTop animated:NO];
- }
- }
- }
-
- if (self.isTouchingIndexView) {
- if (@available(iOS 10.0, *)) {
- [self.generator prepare];
- [self.generator impactOccurred];
- }
- }
- }
- - (void)onActionWithScroll
- {
- if (self.isTouchingIndexView) {
- // 当滑动tableView视图时,另一手指滑动索引视图,让tableView滑动失效
- self.tableView.panGestureRecognizer.enabled = NO;
- self.tableView.panGestureRecognizer.enabled = YES;
-
- return; // 当滑动索引视图时,tableView滚动不能影响索引位置
- }
-
- [self configCurrentSection];
- }
- #pragma mark - Display
- - (UIBezierPath *)drawIndicatorPath
- {
- CGFloat indicatorRadius = self.configuration.indicatorHeight / 2;
- CGFloat sinPI_4_Radius = sin(M_PI_4) * indicatorRadius;
- CGFloat margin = (sinPI_4_Radius * 2 - indicatorRadius);
-
- CGPoint startPoint = CGPointMake(margin + indicatorRadius + sinPI_4_Radius, indicatorRadius - sinPI_4_Radius);
- CGPoint trianglePoint = CGPointMake(4 * sinPI_4_Radius, indicatorRadius);
- CGPoint centerPoint = CGPointMake(margin + indicatorRadius, indicatorRadius);
-
- UIBezierPath *bezierPath = [UIBezierPath bezierPath];
- [bezierPath moveToPoint:startPoint];
- [bezierPath addArcWithCenter:centerPoint radius:indicatorRadius startAngle:-M_PI_4 endAngle:M_PI_4 clockwise:NO];
- [bezierPath addLineToPoint:trianglePoint];
- [bezierPath addLineToPoint:startPoint];
- [bezierPath closePath];
- return bezierPath;
- }
- - (CAShapeLayer *)createSearchLayer
- {
- CGFloat radius = self.configuration.indexItemHeight / 4;
- CGFloat margin = self.configuration.indexItemHeight / 4;
- CGFloat start = radius * 2.5 + margin;
- CGFloat end = radius + sin(M_PI_4) * radius + margin;
- UIBezierPath *path = [UIBezierPath bezierPath];
- [path moveToPoint:CGPointMake(start, start)];
- [path addLineToPoint:CGPointMake(end, end)];
- [path addArcWithCenter:CGPointMake(radius + margin, radius + margin) radius:radius startAngle:M_PI_4 endAngle:2 * M_PI + M_PI_4 clockwise:YES];
- [path closePath];
-
- CAShapeLayer *layer = [CAShapeLayer layer];
- layer.fillColor = self.configuration.indexItemBackgroundColor.CGColor;
- layer.strokeColor = self.configuration.indexItemTextColor.CGColor;
- layer.contentsScale = [UIScreen mainScreen].scale;
- layer.lineWidth = self.configuration.indexItemHeight / 12;
- layer.path = path.CGPath;
- return layer;
- }
- - (void)showIndicator:(BOOL)animated
- {
- if (self.currentSection >= (NSInteger)self.subTextLayers.count) return;
-
- if (self.currentSection < 0) {
- if (self.currentSection == SCIndexViewSearchSection) {
- [self hideIndicator:animated];
- }
- return;
- }
-
- SCTextLayer *textLayer = self.subTextLayers[self.currentSection];
- if (self.configuration.indexViewStyle == SCIndexViewStyleDefault) {
- self.indicator.center = CGPointMake(self.bounds.size.width - self.indicator.bounds.size.width / 2 - self.configuration.indicatorRightMargin, textLayer.position.y);
- } else {
- self.indicator.center = CGPointMake(self.bounds.size.width / 2, self.bounds.size.height / 2);
- }
- self.indicator.text = textLayer.string;
-
- if (animated) {
- self.indicator.alpha = 0;
- self.indicator.hidden = NO;
- [UIView animateWithDuration:kAnimationDuration animations:^{
- self.indicator.alpha = 1;
- }];
- } else {
- self.indicator.alpha = 1;
- self.indicator.hidden = NO;
- }
- }
- - (void)hideIndicator:(BOOL)animated
- {
- if (self.indicator.hidden) return;
-
- if (animated) {
- self.indicator.alpha = 1;
- self.indicator.hidden = NO;
- [UIView animateWithDuration:kAnimationDuration animations:^{
- self.indicator.alpha = 0;
- } completion:^(BOOL finished) {
- self.indicator.alpha = 1;
- self.indicator.hidden = YES;
- }];
- } else {
- self.indicator.alpha = 1;
- self.indicator.hidden = YES;
- }
- }
- - (void)refreshTextLayer:(BOOL)selected
- {
- if (self.currentSection < 0 || self.currentSection >= (NSInteger)self.subTextLayers.count) return;
-
- SCTextLayer *textLayer = self.subTextLayers[self.currentSection];
- UIColor *backgroundColor, *foregroundColor;
- UIFont *font;
- if (selected) {
- backgroundColor = self.configuration.indexItemSelectedBackgroundColor;
- foregroundColor = self.configuration.indexItemSelectedTextColor;
- font = self.configuration.indexItemSelectedTextFont;
- } else {
- backgroundColor = self.configuration.indexItemBackgroundColor;
- foregroundColor = self.configuration.indexItemTextColor;
- font = self.configuration.indexItemTextFont;
- }
-
- [CATransaction begin];
- [CATransaction setDisableActions:YES];
- textLayer.backgroundColor = backgroundColor.CGColor;
- textLayer.foregroundColor = foregroundColor.CGColor;
- textLayer.itemFont = font;
- [CATransaction commit];
- }
- #pragma mark - UITouch and UIEvent
- - (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event
- {
- // 当滑动索引视图时,防止其他手指去触发事件
- if (self.touchingIndexView) return YES;
-
- CALayer *firstLayer = self.searchLayer ?: self.subTextLayers.firstObject;
- if (!firstLayer) return NO;
- CALayer *lastLayer = self.subTextLayers.lastObject ?: self.searchLayer;
- if (!lastLayer) return NO;
-
- CGFloat space = self.configuration.indexItemRightMargin * 2;
- if (point.x > self.bounds.size.width - space - self.configuration.indexItemHeight
- && point.x <= self.bounds.size.width
- && point.y > CGRectGetMinY(firstLayer.frame) - space
- && point.y < CGRectGetMaxY(lastLayer.frame) + space) {
- return YES;
- }
- return NO;
- }
- - (BOOL)beginTrackingWithTouch:(UITouch *)touch withEvent:(UIEvent *)event
- {
- self.touchingIndexView = YES;
- CGPoint location = [touch locationInView:self];
- NSInteger currentPosition = SCPositionOfTextLayerInY(location.y, kSCIndexViewMargin, kSCIndexViewSpace);
- if (currentPosition < 0 || currentPosition >= (NSInteger)self.dataSource.count) return YES;
-
- NSInteger deta = self.searchLayer ? 1 : 0;
- NSInteger currentSection = currentPosition - deta;
- self.currentSection = currentSection;
- [self showIndicator:YES];
- [self onActionWithDidSelect];
- if (self.delegate && [self.delegate respondsToSelector:@selector(indexView:didSelectAtSection:)]) {
- [self.delegate indexView:self didSelectAtSection:self.currentSection];
- }
- return YES;
- }
- - (BOOL)continueTrackingWithTouch:(UITouch *)touch withEvent:(UIEvent *)event
- {
- self.touchingIndexView = YES;
- CGPoint location = [touch locationInView:self];
- NSInteger currentPosition = SCPositionOfTextLayerInY(location.y, kSCIndexViewMargin, kSCIndexViewSpace);
-
- if (currentPosition < 0) {
- currentPosition = 0;
- } else if (currentPosition >= (NSInteger)self.dataSource.count) {
- currentPosition = self.dataSource.count - 1;
- }
-
- NSInteger deta = self.searchLayer ? 1 : 0;
- NSInteger currentSection = currentPosition - deta;
- if (currentSection == self.currentSection) return YES;
-
- self.currentSection = currentSection;
- [self showIndicator:NO];
- [self onActionWithDidSelect];
- if (self.delegate && [self.delegate respondsToSelector:@selector(indexView:didSelectAtSection:)]) {
- [self.delegate indexView:self didSelectAtSection:self.currentSection];
- }
- return YES;
- }
- - (void)endTrackingWithTouch:(UITouch *)touch withEvent:(UIEvent *)event
- {
- self.touchingIndexView = NO;
- NSInteger oldCurrentPosition = self.currentSection;
- [self refreshCurrentSection];
- if (oldCurrentPosition != self.currentSection) {
- [self showIndicator:NO];
- }
- [self hideIndicator:YES];
- }
- - (void)cancelTrackingWithEvent:(UIEvent *)event
- {
- self.touchingIndexView = NO;
- NSInteger oldCurrentPosition = self.currentSection;
- [self refreshCurrentSection];
- if (oldCurrentPosition != self.currentSection) {
- [self showIndicator:NO];
- }
- [self hideIndicator:YES];
- }
- #pragma mark - Getters and Setters
- - (void)setDataSource:(NSArray<NSString *> *)dataSource
- {
- if (_dataSource == dataSource) return;
-
- _dataSource = dataSource.copy;
-
- [self configSubLayersAndSubviews];
- [self configCurrentSection];
- }
- - (void)setCurrentSection:(NSInteger)currentSection
- {
- if (currentSection == _currentSection) return;
- if ((currentSection < 0 && currentSection != SCIndexViewSearchSection)
- || currentSection >= (NSInteger)self.subTextLayers.count) {
- [self refreshTextLayer:NO];
- return;
- }
-
- [self refreshTextLayer:NO];
- _currentSection = currentSection;
- [self refreshTextLayer:YES];
- }
- - (void)setStartSection:(NSUInteger)startSection {
- if (_startSection == startSection) return;
- _startSection = startSection;
- [self configCurrentSection];
- }
- - (NSMutableArray *)subTextLayers
- {
- if (!_subTextLayers) {
- _subTextLayers = [NSMutableArray array];
- }
- return _subTextLayers;
- }
- - (UILabel *)indicator
- {
- if (!_indicator) {
- _indicator = [UILabel new];
- _indicator.layer.backgroundColor = self.configuration.indicatorBackgroundColor.CGColor;
- _indicator.textColor = self.configuration.indicatorTextColor;
- _indicator.font = self.configuration.indicatorTextFont;
- _indicator.textAlignment = NSTextAlignmentCenter;
- _indicator.hidden = YES;
-
- switch (self.configuration.indexViewStyle) {
- case SCIndexViewStyleDefault:
- {
- CGFloat indicatorRadius = self.configuration.indicatorHeight / 2;
- CGFloat sinPI_4_Radius = sin(M_PI_4) * indicatorRadius;
- _indicator.bounds = CGRectMake(0, 0, (4 * sinPI_4_Radius), 2 * indicatorRadius);
-
- CAShapeLayer *maskLayer = [CAShapeLayer layer];
- maskLayer.path = [self drawIndicatorPath].CGPath;
- _indicator.layer.mask = maskLayer;
- }
- break;
-
- case SCIndexViewStyleCenterToast:
- {
- _indicator.bounds = CGRectMake(0, 0, self.configuration.indicatorHeight, self.configuration.indicatorHeight);
- _indicator.center = CGPointMake(self.bounds.size.width / 2, self.bounds.size.height / 2);
- _indicator.layer.cornerRadius = self.configuration.indicatorCornerRadius;
- }
- break;
-
- default:
- break;
- }
- }
- return _indicator;
- }
- - (UIImpactFeedbackGenerator *)generator {
- if (!_generator) {
- _generator = [[UIImpactFeedbackGenerator alloc] initWithStyle:UIImpactFeedbackStyleLight];
- }
- return _generator;
- }
- @end
|