添加pay

This commit is contained in:
2025-11-13 21:22:10 +08:00
parent ae79d1b1ba
commit d164514fcf
12 changed files with 709 additions and 56 deletions

View File

@@ -0,0 +1,359 @@
//
// KBJfPay.m
// keyBoard
#import "KBJfPay.h"
#import "KBJfPayCell.h"
static NSString * const kKBJfPayCellId = @"kKBJfPayCellId";
@interface KBJfPay () <UICollectionViewDataSource, UICollectionViewDelegateFlowLayout>
@property (nonatomic, strong) UIImageView *bgImageView; //
//
@property (nonatomic, strong) UILabel *myPointsTitleLabel; // My Points
@property (nonatomic, strong) UILabel *pointsLabel; //
@property (nonatomic, strong) UIImageView *bigCoinImageView; // pay_big_icon
// Recharge Now
@property (nonatomic, strong) UIImageView *smallLeftIcon; // shop_jb_icon
@property (nonatomic, strong) UILabel *rechargeLabel; // Recharge Now
// /
@property (nonatomic, strong) UIView *listContainerView; // collectionView
@property (nonatomic, strong) UICollectionView *collectionView;
// /
@property (nonatomic, strong) UIButton *payButton; //
@property (nonatomic, strong) UILabel *agreementLabel; //
@property (nonatomic, strong) UIButton *agreementButton;
//
@property (nonatomic, strong) NSArray<NSDictionary *> *data; // @{coins, price}
@property (nonatomic, assign) NSInteger selectedIndex; //
@end
@implementation KBJfPay
- (void)viewDidLoad {
[super viewDidLoad];
self.bgImageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"my_bg_icon"]];
self.bgImageView.contentMode = UIViewContentModeScaleAspectFill;
self.kb_navView.backgroundColor = [UIColor clearColor];
[self.view insertSubview:self.bgImageView belowSubview:self.kb_navView];
[self.bgImageView mas_makeConstraints:^(MASConstraintMaker *make) {
make.edges.equalTo(self.view);
}];
self.kb_titleLabel.text = @"Recharge";
//
self.data = @[
@{ @"coins": @690, @"price": @"$6.90" },
@{ @"coins": @1280, @"price": @"$12.90" },
@{ @"coins": @3290, @"price": @"$32.90" },
@{ @"coins": @4990, @"price": @"$49.90" },
@{ @"coins": @9990, @"price": @"$99.90" },
@{ @"coins": @19990,@"price": @"$199.90" },
];
self.selectedIndex = 1; //
//
[self.view addSubview:self.myPointsTitleLabel];
[self.view addSubview:self.pointsLabel];
[self.view addSubview:self.listContainerView];
[self.view addSubview:self.bigCoinImageView];
[self.listContainerView addSubview:self.smallLeftIcon];
[self.listContainerView addSubview:self.rechargeLabel];
[self.listContainerView addSubview:self.collectionView];
[self.view addSubview:self.payButton];
[self.view addSubview:self.agreementLabel];
[self.view addSubview:self.agreementButton];
// mas
[self.myPointsTitleLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.equalTo(self.view).offset(16);
make.top.equalTo(self.view).offset(KB_NAV_TOTAL_HEIGHT + 40);
}];
[self.pointsLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.equalTo(self.myPointsTitleLabel);
make.top.equalTo(self.myPointsTitleLabel.mas_bottom).offset(4);
}];
[self.bigCoinImageView mas_makeConstraints:^(MASConstraintMaker *make) {
make.right.equalTo(self.view).offset(-12);
make.top.equalTo(self.view).offset(KB_NAV_TOTAL_HEIGHT + 10);
make.width.mas_equalTo(131);
make.height.mas_equalTo(144);
}];
// + /
[self.listContainerView mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.right.equalTo(self.view);
make.top.mas_equalTo(KB_NAV_TOTAL_HEIGHT + 123);
make.bottom.equalTo(self.payButton.mas_top).offset(-16);
make.height.greaterThanOrEqualTo(@220);
}];
[self.smallLeftIcon mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.equalTo(self.listContainerView).offset(16);
make.top.equalTo(self.listContainerView).offset(16);
make.width.height.mas_equalTo(20);
}];
[self.rechargeLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.centerY.equalTo(self.smallLeftIcon);
make.left.equalTo(self.smallLeftIcon.mas_right).offset(8);
}];
[self.collectionView mas_makeConstraints:^(MASConstraintMaker *make) {
make.bottom.left.right.equalTo(self.listContainerView).inset(16);
make.top.equalTo(self.smallLeftIcon.mas_bottom).offset(19);
}];
[self.agreementButton mas_makeConstraints:^(MASConstraintMaker *make) {
make.centerX.equalTo(self.view);
make.bottom.equalTo(self.view).offset(-KB_SAFE_BOTTOM - 15);
}];
[self.agreementLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.centerX.equalTo(self.view);
make.bottom.equalTo(self.agreementButton.mas_top).offset(-8);
}];
//
[self.payButton mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.equalTo(self.view).offset(24);
make.right.equalTo(self.view).offset(-24);
make.bottom.equalTo(self.agreementLabel.mas_top).offset(-14);
make.height.mas_equalTo(58);
}];
//
[self.collectionView reloadData];
//
dispatch_async(dispatch_get_main_queue(), ^{
NSIndexPath *ip = [NSIndexPath indexPathForItem:self.selectedIndex inSection:0];
// 便 setSelected UI
if (ip) {
[self.collectionView selectItemAtIndexPath:ip animated:NO scrollPosition:UICollectionViewScrollPositionNone];
}
KBJfPayCell *cell = (KBJfPayCell *)[self.collectionView cellForItemAtIndexPath:ip];
if (cell) { [cell applySelected:YES animated:NO]; }
});
}
- (void)viewDidAppear:(BOOL)animated {
[super viewDidAppear:animated];
// reload cell
NSIndexPath *ip = [NSIndexPath indexPathForItem:self.selectedIndex inSection:0];
if (ip) {
[self.collectionView selectItemAtIndexPath:ip animated:NO scrollPosition:UICollectionViewScrollPositionNone];
}
KBJfPayCell *cell = (KBJfPayCell *)[self.collectionView cellForItemAtIndexPath:ip];
if (cell) { [cell applySelected:YES animated:NO]; }
}
#pragma mark - UICollectionView Delegate (ensure first show)
- (void)collectionView:(UICollectionView *)collectionView willDisplayCell:(UICollectionViewCell *)cell forItemAtIndexPath:(NSIndexPath *)indexPath {
if (![cell isKindOfClass:KBJfPayCell.class]) { return; }
KBJfPayCell *c = (KBJfPayCell *)cell;
BOOL sel = (indexPath.item == self.selectedIndex);
if (sel) {
[collectionView selectItemAtIndexPath:indexPath animated:NO scrollPosition:UICollectionViewScrollPositionNone];
}
[c applySelected:sel animated:NO];
}
#pragma mark -
- (void)viewDidLayoutSubviews {
[super viewDidLayoutSubviews];
//
UIRectCorner corners = UIRectCornerTopLeft | UIRectCornerTopRight;
UIBezierPath *path = [UIBezierPath bezierPathWithRoundedRect:self.listContainerView.bounds
byRoundingCorners:corners
cornerRadii:CGSizeMake(20, 20)];
CAShapeLayer *mask = [CAShapeLayer layer];
mask.frame = self.listContainerView.bounds;
mask.path = path.CGPath;
self.listContainerView.layer.mask = mask;
}
#pragma mark - UICollectionView
- (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section {
return self.data.count;
}
- (__kindof UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath {
KBJfPayCell *cell = [collectionView dequeueReusableCellWithReuseIdentifier:kKBJfPayCellId forIndexPath:indexPath];
NSDictionary *item = self.data[indexPath.item];
NSString *coins = [NSString stringWithFormat:@"%@", item[@"coins"]];
NSString *price = item[@"price"]; // "$6.90"
[cell configCoins:coins price:price];
[cell applySelected:(indexPath.item == self.selectedIndex) animated:NO];
return cell;
}
- (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath {
if (self.selectedIndex == indexPath.item) { return; }
NSInteger old = self.selectedIndex;
self.selectedIndex = indexPath.item;
KBJfPayCell *newCell = (KBJfPayCell *)[collectionView cellForItemAtIndexPath:indexPath];
[newCell applySelected:YES animated:YES];
if (old >= 0 && old < self.data.count) {
NSIndexPath *oldIP = [NSIndexPath indexPathForItem:old inSection:0];
KBJfPayCell *oldCell = (KBJfPayCell *)[collectionView cellForItemAtIndexPath:oldIP];
[oldCell applySelected:NO animated:YES];
}
}
//
- (CGSize)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout sizeForItemAtIndexPath:(NSIndexPath *)indexPath {
CGFloat totalW = collectionView.bounds.size.width;
CGFloat spacing = 10.0; //
CGFloat columns = 3.0;
CGFloat insets = 0; // mas 16 cell inset
CGFloat w = floor((totalW - insets - spacing * (columns - 1)) / columns);
CGFloat h = KBFit(116);
return CGSizeMake(MAX(0, w), h);
}
- (CGFloat)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout minimumInteritemSpacingForSectionAtIndex:(NSInteger)section {
return 10.0;
}
- (CGFloat)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout minimumLineSpacingForSectionAtIndex:(NSInteger)section {
return 30;
}
#pragma mark - Actions
- (void)onTapPayButton {
// UI
if (self.selectedIndex >= 0 && self.selectedIndex < self.data.count) {
NSDictionary *item = self.data[self.selectedIndex];
NSString *msg = [NSString stringWithFormat:@"购买:%@ Coins %@", item[@"coins"], item[@"price"]];
[KBHUD showInfo:msg];
}
}
- (void)agreementButtonAction{
[KBHUD showInfo:@"跳转协议"];
}
#pragma mark - Lazy UI
- (UILabel *)myPointsTitleLabel {
if (!_myPointsTitleLabel) {
_myPointsTitleLabel = [UILabel new];
_myPointsTitleLabel.text = @"My Points";
_myPointsTitleLabel.font = [UIFont systemFontOfSize:16 weight:UIFontWeightSemibold];
_myPointsTitleLabel.textColor = [UIColor colorWithHex:KBBlackValue];
}
return _myPointsTitleLabel;
}
- (UILabel *)pointsLabel {
if (!_pointsLabel) {
_pointsLabel = [UILabel new];
_pointsLabel.text = @"4230"; //
_pointsLabel.font = [UIFont systemFontOfSize:36 weight:UIFontWeightBold];
_pointsLabel.textColor = [UIColor colorWithHex:KBColorValue];
}
return _pointsLabel;
}
- (UIImageView *)bigCoinImageView {
if (!_bigCoinImageView) {
_bigCoinImageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"pay_big_icon"]];
_bigCoinImageView.contentMode = UIViewContentModeScaleAspectFit;
}
return _bigCoinImageView;
}
- (UIImageView *)smallLeftIcon {
if (!_smallLeftIcon) {
_smallLeftIcon = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"shop_jb_icon"]];
_smallLeftIcon.contentMode = UIViewContentModeScaleAspectFit;
}
return _smallLeftIcon;
}
- (UILabel *)rechargeLabel {
if (!_rechargeLabel) {
_rechargeLabel = [UILabel new];
_rechargeLabel.text = @"Recharge Now";
_rechargeLabel.font = [UIFont systemFontOfSize:16 weight:UIFontWeightSemibold];
_rechargeLabel.textColor = [UIColor colorWithHex:KBBlackValue];
}
return _rechargeLabel;
}
- (UIView *)listContainerView {
if (!_listContainerView) {
_listContainerView = [UIView new];
//
_listContainerView.backgroundColor = [UIColor colorWithWhite:1 alpha:0.43];
}
return _listContainerView;
}
- (UICollectionView *)collectionView {
if (!_collectionView) {
UICollectionViewFlowLayout *layout = [UICollectionViewFlowLayout new];
layout.scrollDirection = UICollectionViewScrollDirectionVertical;
_collectionView = [[UICollectionView alloc] initWithFrame:CGRectZero collectionViewLayout:layout];
_collectionView.backgroundColor = UIColor.clearColor;
_collectionView.dataSource = self;
_collectionView.delegate = self;
_collectionView.alwaysBounceVertical = YES;
[_collectionView registerClass:KBJfPayCell.class forCellWithReuseIdentifier:kKBJfPayCellId];
_collectionView.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever;
}
return _collectionView;
}
- (UIButton *)payButton {
if (!_payButton) {
_payButton = [UIButton buttonWithType:UIButtonTypeCustom];
[_payButton setTitle:@"Recharge Now" forState:UIControlStateNormal];
[_payButton setTitleColor:[UIColor colorWithHex:KBBlackValue] forState:UIControlStateNormal];
_payButton.titleLabel.font = [UIFont systemFontOfSize:16 weight:UIFontWeightSemibold];
// 使退
UIImage *bg = [UIImage imageNamed:@"recharge_now_icon"];
if (bg) {
[_payButton setBackgroundImage:bg forState:UIControlStateNormal];
} else {
UIImage *fallback = [UIImage kb_gradientImageWithColors:@[[UIColor colorWithHex:0xC7F8F0], [UIColor colorWithHex:0xE8FFF6]] size:CGSizeMake(10, 58) direction:KBGradientDirectionLeftToRight];
[_payButton setBackgroundImage:[fallback resizableImageWithCapInsets:UIEdgeInsetsMake(29, 29, 29, 29) resizingMode:UIImageResizingModeStretch] forState:UIControlStateNormal];
}
[_payButton addTarget:self action:@selector(onTapPayButton) forControlEvents:UIControlEventTouchUpInside];
}
return _payButton;
}
- (UILabel *)agreementLabel {
if (!_agreementLabel) {
_agreementLabel = [UILabel new];
_agreementLabel.text = @"By clicking Pay, you indicate your agreement to the"; //
_agreementLabel.font = [UIFont systemFontOfSize:11 weight:UIFontWeightRegular];
_agreementLabel.textColor = [UIColor colorWithWhite:0.45 alpha:1.0];
}
return _agreementLabel;
}
- (UIButton *)agreementButton {
if (!_agreementButton) {
_agreementButton = [UIButton buttonWithType:UIButtonTypeCustom];
[_agreementButton setTitle:@"《Embership Agreement》" forState:UIControlStateNormal];
[_agreementButton setTitleColor:[UIColor colorWithHex:KBColorValue] forState:UIControlStateNormal];
_agreementButton.titleLabel.font = [UIFont systemFontOfSize:10 weight:UIFontWeightSemibold];
[_agreementButton addTarget:self action:@selector(agreementButtonAction) forControlEvents:UIControlEventTouchUpInside];
}
return _agreementButton;
}
@end