iOS-正六邊形堆疊

一些app會有此類效果驰唬,我按照自己的理解仿寫了一個

如圖

1.如何繪制單個正六邊形

使用繼承于CAShapeLayerYYHexagonsLayer設(shè)置路徑來繪制單個六邊形
//YYHexagonsLayer的屬性與方法
@property (nonatomic, strong) UIColor *normalColor;
@property (nonatomic, strong) UIColor *highlightColor;
@property (nonatomic, assign, readonly) CGFloat sideLength;
@property (nonatomic, assign, getter=isSelected) BOOL selected;
+ (instancetype)layerWithSideLength:(CGFloat)sideLength;
//構(gòu)造方法
+ (instancetype)layerWithSideLength:(CGFloat)sideLength {
    YYHexagonsLayer *layer = [YYHexagonsLayer layer];
    
    CGFloat utilAngle = M_PI / 3;
    
    UIBezierPath *path = [UIBezierPath bezierPath];
    [path moveToPoint:CGPointMake(cos(utilAngle * 0.5) * sideLength, sin(utilAngle * 0.5) * sideLength)];
    
    [path addLineToPoint:CGPointMake(cos(utilAngle * 1.5) * sideLength, sin(utilAngle * 1.5) * sideLength)];
    [path addLineToPoint:CGPointMake(cos(utilAngle * 2.5) * sideLength, sin(utilAngle * 2.5) * sideLength)];
    [path addLineToPoint:CGPointMake(cos(utilAngle * 3.5) * sideLength, sin(utilAngle * 3.5) * sideLength)];
    [path addLineToPoint:CGPointMake(cos(utilAngle * 4.5) * sideLength, sin(utilAngle * 4.5) * sideLength)];
    [path addLineToPoint:CGPointMake(cos(utilAngle * 5.5) * sideLength, sin(utilAngle * 5.5) * sideLength)];
    
    layer.path = path.CGPath;
    layer.fillColor = UIColor.orangeColor.CGColor;
    layer.bounds = CGRectMake(0, 0, sideLength * 2, sin(utilAngle * 1) * sideLength * 2);
    
    layer->_sideLength = sideLength;
    
    return layer;
}
有了YYHexagonsLayer之后叙甸,使用layerWithSideLength:方法創(chuàng)建layer缝呕,設(shè)置position并添加到superLayer上常挚,一個正六邊形就出現(xiàn)了彼宠。
YYHexagonsLayer *layer = [YYHexagonsLayer layerWithSideLength:30];
layer.normalColor = UIColor.orangeColor;
layer.position = CGPointMake(100, 100);
[self.view.layer addSublayer:layer];
一個正六邊形
到此第一步就算完成了频丘。

2.如何將多個正六邊形堆疊在一起

每個正六邊形的position的計算都要用到三角函數(shù)變換郁稍,并不難辜梳,只是麻煩了一些粱甫。
//首先,我仿照UITableView作瞄,為YYHexagonsGroupView添加了數(shù)據(jù)源茶宵、代理方法
@protocol YYHexagonsGroupViewDelegate <NSObject>

@required
- (NSInteger)numberOfHexagonsInGroupView:(YYHexagonsGroupView *)hexagonsGroupView;
- (YYHexagonsLayer *)hexagonsGroupView:(YYHexagonsGroupView *)hexagonsGroupView hexagonsForRowAtIndex:(NSInteger)index;

@optional
- (void)hexagonsGroupView:(YYHexagonsGroupView *)hexagonsGroupView didSelectRowAtIndex:(NSInteger)index;

@end

//下面是屬性與方法
@property (nonatomic, weak) id<YYHexagonsGroupViewDelegate> delegate;
@property (nonatomic, assign) CGFloat utilWidth;
@property (nonatomic, assign) CGFloat margin;
//添加了刷新所有和刷新某幾個視圖的方法
- (void)reloadData;
- (void)reloadIndexs:(NSArray<NSNumber *> *)indexs;

- (YYHexagonsLayer *)hexagonsLayerWithIndex:(NSInteger)index;
YYHexagonsGroupView.m中的實現(xiàn)
使用一個字典來存放所有的六邊形,key值為@(index)
//屬性
@property (nonatomic, strong) UIScrollView *scrollView;
@property (nonatomic, strong) NSMutableDictionary<NSNumber *,YYHexagonsLayer *> *hexagonsLayers;
創(chuàng)建子視圖
//創(chuàng)建子視圖
- (void)createSubviews {
    _scrollView = [[UIScrollView alloc] init];
    [_scrollView addGestureRecognizer:[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(tap:)]];
    [self addSubview:_scrollView];
}
刷新方法
//刷新方法
- (void)reloadData {
    [_hexagonsLayers enumerateKeysAndObjectsUsingBlock:^(NSNumber * _Nonnull key, YYHexagonsLayer * _Nonnull obj, BOOL * _Nonnull stop) {
        [obj removeFromSuperlayer];
    }];
    [_hexagonsLayers removeAllObjects];
    
    [self createSublayers];
}

- (void)reloadIndexs:(NSArray<NSNumber *> *)indexs {
    [indexs enumerateObjectsUsingBlock:^(NSNumber * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        [_hexagonsLayers removeObjectForKey:obj];
        [_hexagonsLayers[obj] removeFromSuperlayer];
        _hexagonsLayers[obj] = [_delegate hexagonsGroupView:self hexagonsForRowAtIndex:obj.integerValue];
    }];
    
}
創(chuàng)建正六邊形, 并計算position_scrollViewcontentSize
//創(chuàng)建正六邊形方法
- (void)createSublayers {
    if (!_hexagonsLayers) {
        _hexagonsLayers = [NSMutableDictionary dictionary];
    }
    
    NSInteger MaxCount = 0;
    if ([_delegate respondsToSelector:@selector(numberOfHexagonsInGroupView:)]) {
        MaxCount = [_delegate numberOfHexagonsInGroupView:self];
    } else {
        //此處為測試時的代碼宗挥,可刪掉
        CGFloat width = self.frame.size.width;
        CGFloat height = self.frame.size.height;
        
        CGFloat utilWidth = _utilWidth;
        CGFloat margin = _margin;
        
        NSInteger MaxI = floor((width - (0.5 * utilWidth * 2 * sin(M_PI / 3))) / (utilWidth * 2 * sin(M_PI / 3) + margin));
        
        MaxCount = floor((height - utilWidth) / (utilWidth + cos(M_PI/3) * (utilWidth + margin))) * MaxI;
    }
    CGFloat maxY = 0;
    for (NSInteger i = 0; i < MaxCount; i ++) {
        maxY = [self addSublayerWithIndex:i];
    }
    
    _scrollView.contentSize = CGSizeMake(self.bounds.size.width, maxY + _utilWidth);
    
}

- (CGFloat)addSublayerWithIndex:(NSInteger)index {
    
    NSInteger row = 0, i = 0;
    
    CGFloat width = self.frame.size.width;
    CGFloat height = self.frame.size.height;
    
    CGFloat utilWidth = _utilWidth;
    CGFloat margin = _margin;
    
    NSInteger MaxI = floor((width - (0.5 * utilWidth * 2 * sin(M_PI / 3))) / (utilWidth * 2 * sin(M_PI / 3) + margin));
    
    row = index / MaxI;
    i = index % MaxI;
    
    if (row * MaxI + i > [_delegate numberOfHexagonsInGroupView:self]) {
        return _scrollView.contentSize.height;
    }
    
    NSInteger MaxRow = 0;
    if ([_delegate respondsToSelector:@selector(numberOfHexagonsInGroupView:)]) {
        MaxRow = ceil([_delegate numberOfHexagonsInGroupView:self] / (double)MaxI);
    } else {
        //此處為測試時的代碼乌庶,可刪掉
        MaxRow = floor((height - utilWidth) / (utilWidth + cos(M_PI/3) * (utilWidth + margin)));
    }
    
    
    CGFloat positionY = utilWidth * 2 + (utilWidth + cos(M_PI/3) * (utilWidth + margin)) * row;
    YYHexagonsLayer *layer = nil;
    
    if ([_delegate respondsToSelector:@selector(hexagonsGroupView:hexagonsForRowAtIndex:)]) {
        layer = [_delegate hexagonsGroupView:self hexagonsForRowAtIndex:row * MaxI + i];
    } else {
        //此處為測試時的代碼沼溜,可刪掉
        layer = [YYHexagonsLayer layerWithSideLength:utilWidth];
        layer.normalColor = UIColor.orangeColor;
        layer.highlightColor = UIColor.cyanColor;
    }
    
    CGFloat x_offset = 0;
    if (row % 2 == 0) {
        x_offset = utilWidth * 2;
    } else {
        x_offset = utilWidth + margin * 0.5;
    }
    
    CGFloat positionX = (i + 0.5) * utilWidth * 2 * sin(M_PI / 3) + i * margin + x_offset;
    layer.position = CGPointMake(positionX, positionY);
    
    [_scrollView.layer addSublayer:layer];
    return layer.position.y;
}

- (YYHexagonsLayer *)hexagonsLayerWithIndex:(NSInteger)index {
    YYHexagonsLayer *layer = _hexagonsLayers[@(index)];
    if (layer == nil) {
        layer = [YYHexagonsLayer layerWithSideLength:_utilWidth];
        _hexagonsLayers[@(index)] = layer;
        [self addSublayerWithIndex:index];
        NSLog(@"%ld--%ld", (long)index, _hexagonsLayers.count);
    }
    
    return layer;
}
點擊事件丹喻,獲取點擊位置point,轉(zhuǎn)換坐標(biāo)系獲得convertPoint抖锥,遍歷_hexagonsLayers字典宵喂,看convertPoint在哪一個六邊形的路徑path內(nèi)糠赦,使用到的函數(shù)是CG_EXTERN bool CGPathContainsPoint(CGPathRef cg_nullable path, const CGAffineTransform * __nullable m, CGPoint point, bool eoFill)
//點擊事件
- (void)tap:(UITapGestureRecognizer *)tap {
    CGPoint point = [tap locationInView:_scrollView];
    [_hexagonsLayers enumerateKeysAndObjectsUsingBlock:^(NSNumber * _Nonnull key, YYHexagonsLayer * _Nonnull layer, BOOL * _Nonnull stop) {
        CGPoint convertPoint = [layer convertPoint:point fromLayer:_scrollView.layer];
        if (CGPathContainsPoint(layer.path, NULL, convertPoint, NO)) {
            if ([_delegate respondsToSelector:@selector(hexagonsGroupView:didSelectRowAtIndex:)]) {
                [_delegate hexagonsGroupView:self didSelectRowAtIndex:key.integerValue];
            } else {
                layer.selected = !layer.isSelected;
            }
            *stop = YES;
        }
    }];
}
//layoutSubviews
- (void)layoutSubviews {
    [super layoutSubviews];
    _scrollView.frame = self.bounds;
    [self createSublayers];
}
到此第2步就完成了

3.使用YYHexagonsGroupView

//創(chuàng)建
_groupView = [[YYHexagonsGroupView alloc] init];
_groupView.translatesAutoresizingMaskIntoConstraints = NO;
_groupView.utilWidth = 15;
_groupView.margin = 2;
_groupView.delegate = self;
...設(shè)置約束代碼略去
//實現(xiàn)代理方法
- (NSInteger)numberOfHexagonsInGroupView:(YYHexagonsGroupView *)hexagonsGroupView {
    return _count;
}
- (YYHexagonsLayer *)hexagonsGroupView:(YYHexagonsGroupView *)hexagonsGroupView hexagonsForRowAtIndex:(NSInteger)index {
    YYHexagonsLayer *layer = [hexagonsGroupView hexagonsLayerWithIndex:index];
    layer.highlightColor = UIColor.cyanColor;
    layer.normalColor = UIColor.orangeColor;
    layer.selected = _selected[index];
    
    return layer;
}

- (void)hexagonsGroupView:(YYHexagonsGroupView *)hexagonsGroupView didSelectRowAtIndex:(NSInteger)index {
    _selected[index] = !_selected[index];
    [hexagonsGroupView reloadIndexs:@[@(index)]];
}

- (void)addHexagonsCount:(UIButton *)button {
    _count += 20;
    [_groupView reloadData];
}
到此第三步就結(jié)束了
最后放上效果圖和demo地址
效果圖
效果圖
demo鏈接
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市锅棕,隨后出現(xiàn)的幾起案子拙泽,更是在濱河造成了極大的恐慌,老刑警劉巖裸燎,帶你破解...
    沈念sama閱讀 217,406評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件顾瞻,死亡現(xiàn)場離奇詭異,居然都是意外死亡德绿,警方通過查閱死者的電腦和手機荷荤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評論 3 393
  • 文/潘曉璐 我一進店門退渗,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人蕴纳,你說我怎么就攤上這事会油。” “怎么了古毛?”我有些...
    開封第一講書人閱讀 163,711評論 0 353
  • 文/不壞的土叔 我叫張陵翻翩,是天一觀的道長。 經(jīng)常有香客問我稻薇,道長嫂冻,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,380評論 1 293
  • 正文 為了忘掉前任塞椎,我火速辦了婚禮桨仿,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘案狠。我一直安慰自己服傍,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,432評論 6 392
  • 文/花漫 我一把揭開白布莺戒。 她就那樣靜靜地躺著伴嗡,像睡著了一般急波。 火紅的嫁衣襯著肌膚如雪从铲。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,301評論 1 301
  • 那天澄暮,我揣著相機與錄音名段,去河邊找鬼。 笑死泣懊,一個胖子當(dāng)著我的面吹牛伸辟,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播馍刮,決...
    沈念sama閱讀 40,145評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼信夫,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了卡啰?” 一聲冷哼從身側(cè)響起静稻,我...
    開封第一講書人閱讀 39,008評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎匈辱,沒想到半個月后振湾,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,443評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡亡脸,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,649評論 3 334
  • 正文 我和宋清朗相戀三年押搪,在試婚紗的時候發(fā)現(xiàn)自己被綠了树酪。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,795評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡大州,死狀恐怖续语,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情厦画,我是刑警寧澤绵载,帶...
    沈念sama閱讀 35,501評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站苛白,受9級特大地震影響娃豹,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜购裙,卻給世界環(huán)境...
    茶點故事閱讀 41,119評論 3 328
  • 文/蒙蒙 一懂版、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧躏率,春花似錦躯畴、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至夯到,卻和暖如春嚷缭,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背耍贾。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評論 1 269
  • 我被黑心中介騙來泰國打工阅爽, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人荐开。 一個月前我還...
    沈念sama閱讀 47,899評論 2 370
  • 正文 我出身青樓付翁,卻偏偏與公主長得像,于是被迫代替她去往敵國和親晃听。 傳聞我的和親對象是個殘疾皇子百侧,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,724評論 2 354

推薦閱讀更多精彩內(nèi)容