從零開始設(shè)計(jì)搭建ios App框架(十)

支付模塊


相信做過電商App的同胞們有個(gè)功能模塊是必需的,那就是線上支付。app用戶通過線上支付完成商品和服務(wù)的購買。

現(xiàn)如今線上支付的方式有多種,ios自家的支付就有蘋果內(nèi)購,applePay,其他大的支付平臺(tái)有支付寶、微信、銀聯(lián),以及跟銀聯(lián)合作而衍生出來小支付平臺(tái)。在這里就不一一介紹各各支付平臺(tái)。

總之大部分支付平臺(tái)支付流程跟下圖相似

支付.jpg

當(dāng)然此文我重點(diǎn)是介紹客戶端的支付模塊設(shè)計(jì)。

支付場(chǎng)景:
  • 當(dāng)app用戶購買商品或者服務(wù)到結(jié)算頁面時(shí),就會(huì)有下單與支付需求;
  • 當(dāng)用戶進(jìn)入自己的訂單列表時(shí),可以對(duì)沒有支付的訂單進(jìn)行支付;
  • 當(dāng)用戶進(jìn)入未支付的訂單詳情頁面時(shí),也可以進(jìn)行訂單支付;
app線上支付模塊設(shè)計(jì):

1、不需要對(duì)原有App框架作大的改動(dòng),也就是說低偶合;
2、支付API簡(jiǎn)單明了,方便集成;
3、支付結(jié)果能正確告之支付發(fā)起者;
4、在原有模塊中可以方便擴(kuò)展添加新的支付平臺(tái);

經(jīng)過分析會(huì)得出下圖的左邊部分

客戶端支付模塊.jpg

發(fā)起支付的場(chǎng)景一般都會(huì)在ViewController中進(jìn)行,為了方便也為了代碼簡(jiǎn)潔,我們一般不會(huì)在每個(gè)支付發(fā)起的Controller中都去寫創(chuàng)建訂單、獲取支付參數(shù)等這些相似的代碼。有沒有辦法解決一下呢?

這個(gè)時(shí)候想到最快的方式就是構(gòu)建個(gè)BaseViewController繼承于ViewController,將上述相關(guān)代碼封裝好。然后支付場(chǎng)景的ViewController都繼承這個(gè)BaseViewController。

繼承的確可以做到,但不知道你有沒有這樣的經(jīng)歷,當(dāng)有新功能需新建一個(gè)Demo來預(yù)演時(shí),通過我們會(huì)拷貝現(xiàn)有工程中的部分文件,不想重新寫,ViewController一般是包括在其中的,這個(gè)時(shí)候就你要把支付相關(guān)的模塊邏輯也拷貝出來,不然編譯不過。有一種拔出蘿卜帶出泥的感覺。

其實(shí)還有做法就是用catagory,為Controller新建一支付相關(guān)的分類,目前我采用的就是這種方法。

typedef void(^PGPayFinishBlock)(PGPayResult *resultObj);
typedef void(^PGPayParamBlock)(NSString *orderId, PGPayType payType);

@interface PGBaseController (pay)<PGPayManagerDelegate>
/*
 支付完成時(shí)的block回調(diào)
 */
@property(nonatomic, copy)PGPayFinishBlock payFinishBlock;
/*
 獲取支付所需的支付參數(shù)
 */
@property(nonatomic, copy)PGPayParamBlock payParamBlock;
/*
 訂單號(hào)
 */
@property(nonatomic, copy)NSString *payOrderId;

/*
 進(jìn)行支付
 */
- (void)pay:(NSString *)szOrderId;
/*
 用對(duì)應(yīng)的支付方式進(jìn)行支付
 */
- (void)startPayWithParam:(NSDictionary *)dicParam
                  payType:(PGPayType)payType
                  orderId:(NSString *)szOrderId;

@end

支付方式這么多種,我是不是可以有一個(gè)統(tǒng)一的管理類對(duì)這些支付邏輯進(jìn)行模塊化封裝。用這個(gè)類統(tǒng)一處理第三方支付平臺(tái)(支付,結(jié)果 回調(diào)處理)邏輯,處理完成后回調(diào)給支付場(chǎng)景。
這樣分析后就會(huì)得出上圖中右邊部分。
定義支付方式:

/*
 支付方式
 */
typedef NS_ENUM(NSInteger, PGPayType) {
    PGPayType_None      = 0,//不支付
    PGPayType_WxPay     = 1,//微信支付
    PGPayType_AliPay    = 2,//支付寶支付
};

支付定義結(jié)果:

/*
 支付結(jié)果狀態(tài)
 */
typedef NS_ENUM(NSInteger, PayErrCode) {
    PaySuccess           = 0,    /**< 成功    */
    PayErrCodeCommon     = -1,   /**< 失敗    */
    PayErrCodeUserCancel = -2,   /**< 用戶點(diǎn)擊取消并返回    */
};

#pragma mark -
/**
 支付結(jié)果
 */
@interface PGPayResult : NSObject
@property(nonatomic, strong)NSString *szOrder;
@property(nonatomic, assign)PayErrCode errorCode;
@property(nonatomic, strong)NSString *szDesc;
@end

支付管理類PGPayManager:

#pragma mark -
@protocol PGPayManagerDelegate <NSObject>
@required
/*
 payResult: 支付結(jié)果
 */
- (void)payFinished:(PGPayResult *)payResult;
/*
 平臺(tái)App沒有安裝
 */
- (void)noInstallPlatformApp:(PGPayType)type orderId:(NSString *)orderId;
@end

#pragma mark -
@interface PGPayManager : NSObject

+ (PGPayManager *)shareInstance;
/*
 支付完成后,應(yīng)用間的信息傳遞
 */
- (void)handleOpenURL:(NSURL *)url;
/*
 初始化各支付平臺(tái)
 */
- (void)platformInit;
/*
 開始支付
 dicParam: 平臺(tái)SDK支付所需的參數(shù)
 type:支付方式
 delegate: 支付完成后的回調(diào)
 */
- (void)startPayWithParam:(NSDictionary *)dicParam
                  orderId:(NSString *)orderId
                  payType:(PGPayType)type
                 delegate:(id<PGPayManagerDelegate>)delegate;
/*
 獲取微信支付下載地址
 */
+ (NSString *)getWXAppInstallUrl;


@end

支付場(chǎng)景發(fā)起支付調(diào)用示例

#pragma mark -
- (void)payOrder
{
    //訂單支付
    [self pay:self.payOrderId];
}

#pragma mark - PGApiDelegate
- (void)dataRequestFinish:(PGResultObject *)resultObj apiType:(PGApiType)apiType
{
    [self hideWaitingView];
    if(apiType == API_TYPE_ORDER_PAY)
    {
        if(resultObj.nCode == 0)
        {
            //
            //獲取支付所需的參數(shù)進(jìn)行支付
            // ...
            //
            NSDictionary *dicParam = [[NSDictionary alloc] init];
            
            
            NSInteger payType = [(NSNumber *)resultObj.extendParam integerValue];
            [self startPayWithParam:dicParam payType:payType orderId:self.payOrderId];
        }
        else
        {
            [self showMsg:resultObj.szErrorDes];
        }
    }
}
- (void)createInitData
{
    [super createInitData];
    
    self.payOrderId = @"訂單號(hào)xxxxxxx";
    
    WEAKSELF
    self.payParamBlock = ^(NSString *orderId, PGPayType payType) {
        dispatch_async(dispatch_get_main_queue(), ^{
            [weakSelf showWaitingView:nil];
            //去服務(wù)器端通過訂單號(hào)與支付方式獲取支付所需的參數(shù)
            [PGRequestManager startPostClient:API_TYPE_ORDER_PAY param:@{@"orderId":orderId} target:weakSelf extendParam:[NSNumber numberWithInteger:payType]];
        });
    };
    
    //支付完成后回調(diào)
    self.payFinishBlock = ^(PGPayResult *result) {
        if(result.errorCode == PaySuccess)
        {
            //支付成功
        }
        else if(result.errorCode == PayErrCodeCommon)
        {
            //支付失敗
        }
        else
        {
            //支付取消
        }
        
        //作相應(yīng)的頁面跳轉(zhuǎn)
        
    };
}

調(diào)用效果


C5EF7091-E5B9-4D77-B6BB-353753DF5ADA.png

感謝您的閱讀!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容