/DDBluetooth

Primary LanguageObjective-C

#DDBluetooth Practical, block-based, and lightweight CoreBluetooth library for iOS.

##How to add to your project Simply add "DDBluetooth" directory to your project

##Sample Application

##Usage ###Initialize ・Initialize core bluetooth central
・callback CBCentralManagerState

[[DDCentralManager sharedManager] initialize:centralQueue options:options didInitialize:^(CBCentralManagerState state) {
        switch (state) {
            case CBCentralManagerStatePoweredOn:
				// Bluetooth is available on this device
                break;
            default:
                // Bluetooth is currently not available on this device"
                break;
        }
}];

Scan

・specify scanning duration
・callback when found peripheral
・callback when scan stops

[self.ddScanAgent startScanForServices:nil withPrefixes:nil interval:3.f
                                    found:^(CBPeripheral *peripheral) {
                                        // Do something with found peripheral
                                    }
                                     stop:^{
                                        // Do something on stop scanning
                                     }];

Connect

・Retry to connect for designated times with specified interval.
・callback when connected
・callback when timeout
・callback when disconnected unexpectedly

[self.ddConnectAgent startConnectionWithRetry:self.ddPeripheral
                                withMaxRetryCount:10
                                         interval:2.f
                                       didConnect:^(CBPeripheral * _Nullable peripheral, ConnectionStatus status, NSError * _Nullable error) {
                             
        if (error)
        {
        	// Do something when error happens due to;
        	// 1. timeout 
        	// 2. disconnected unexpectedly       
        }
        else
        {
			  if (status == ConnectionStatusConnected)
            {
                // Successfully connected
            }
            else
            {
                // Unexpected state
            }
        }
}];   

Discover Characteristic

・callback when finish discovering characteristics
・callback when timeout

[self.ddConnectAgent discover:self.ddPeripheral didDiscover:^(NSArray * _Nullable characteristics, NSError * _Nullable error) {

        if (error)
        {
			// Failed to discover characteristics
        }
        else
        {
			// Do something with discovered characteristics
        }
}];

Indication

・callback when success
・callback when fail

[self.ddTransactionAgent indicate:characteristic didIndicate:^(DDTransactionAgentCallbackType type, NSData *data, NSError *error) {

        if (error)
        {
			// Failed to indicate
        }
        else
        {
            if (type == DDTransactionAgentCallbackTypeIndication)
            {
                // Successfully indicate 
            }
            else if (type == DDTransactionAgentCallbackTypeNotifiedValue)
            {
				// Although a delegate mothod is prepared for notification,
				// you can also handle notification within block.
            }
            else
            {
                // Failed to indicate
            }
        }
}];

###Notification there are 2 ways to handle notified values from peripheral

  1. handle value in blocks
[self.ddTransactionAgent indicate:characteristic didIndicate:^(DDTransactionAgentCallbackType type, NSData *data, NSError *error) {

    if (type == DDTransactionAgentCallbackTypeNotifiedValue)
    {
		// "DDTransactionAgentCallbackTypeNotifiedValue" could be returned 
		// either read, write or idicate blocks.
    }
}];
  1. handle value with a delegate method
- (void)didNotifiedValue:(CBCharacteristic *)characteristic value:(NSData *)value error:(NSError *)error {
    if (error)
    {
        // Faild to receive notification
    }
    else
    {
		// Do something with notified value
    }
}

Read Chararacteristics value

・callback when success
・callback when fail
・callback when timeout

[self.ddTransactionAgent read:characteristic didRead:^(DDTransactionAgentCallbackType type, NSData *data, NSError *error) {
        
        if (error)
        {
            // Failed to read
        }
        else
        {
            if (type == DDTransactionAgentCallbackTypeReadValue)
            {
				// Do something with read value
            }
            else
            {
				// Unexpected state
            }
        }
    }];

Write to Characteristics

・When writing data size exceeds maximum write size at a time, data will be automatically split and sent one by one. ・callback when success
・callback when fail
・callback when timeout

[self.ddTransactionAgent writeSplit:characteristic
                                   data:data
                                   withResponse:isWriteWithResponse
                               didWrite:^(DDTransactionAgentCallbackType type, NSData *data, NSError *error) {
                                   
                                   if (error)
                                   {
												// Write failed
                                   }
                                   else
                                   {
                                       if (type == DDTransactionAgentCallbackTypeWriteValue)
                                       {
                                           // Successfully write data
                                       }
                                       else
                                       {
                                           // Unexpected state
                                       }
                                   }
                               }];

Disconnect

・callback when success
・callback when fail
・callback when timeout

[self.ddConnectAgent disconnect:^(CBPeripheral * _Nullable peripheral, ConnectionStatus status, NSError * _Nullable error) {
        if (error)
        {
        	// Failed to disconnect
        }
        else
        {
			// Successfuly disconnect
        }
}];

Retrieve

・retrieve peripherals discovered in the past.

##Practical Design for actual product development & iOS Cache Countermeasures

####Considering actual product development situation, the library comes with some useful features

  • Re-Scan when didModifyValue called (*The peripheral (server) needs to enable indication of Service Changed Characteristic. If paring is not required, "didModifyService" wouldn't be called)

  • Practical reconnect method to corp with iOS BTServer crash bug.

    • iPhone 5s and older devices seems to have a problem establishing a connection with peripherals that immediately disconnects after "didConnectPeripheral" called.
BTServer[57] <Error>: ATT Failed to set MTU to 158 with result BM3 STATUS 14  
BTServer[57] <Error>: Core Connection failed to device "XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX"
BTServer[57] <Error>: ATT Aborting command as device "XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX" is no longer connected 

##Blocks Callback Design

###Initialize
state - "CBCentralManagerState" will be returned to check bluetooth status on the central devices

Scan

found - called whenever new peripheral is found
stop - called when scan stops after specified interval

Read / Write / Indication

if(error) 
{
	// Handle error
}
else 
{
	// Do something
}

Connect / Disconnect

if(error) 
{
	// Handle error
}
else 
{
	// Do something
}

##Latest Changes

###2016. 5. 20 Added retrieve peripherals method

##TODO

retrieveConnectedPeripheralsWithServices

##License

Copyright 2015-2016 Daiki Ichikawa

Licensed under the Apache License, Version 2.0 (the "License");   
you may not use this file except in compliance with the License.   
You may obtain a copy of the License at   

http://www.apache.org/licenses/LICENSE-2.0   

Unless required by applicable law or agreed to in writing, software   
distributed under the License is distributed on an "AS IS" BASIS,   
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   
See the License for the specific language governing permissions and   
limitations under the License.