Skip to content
This repository has been archived by the owner on May 18, 2022. It is now read-only.
/ bluetooth-ANE Public archive

With Bluetooth air native extension, you'll have access to the Bluetooth hardware. It enable you to scan for other devices, connect to and pair with them and finally transfer data between them.

Notifications You must be signed in to change notification settings

myflashlab/bluetooth-ANE

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

6 Commits
 
 
 
 
 
 

Repository files navigation

Bluetooth ANE V2.0 (Android)

With Bluetooth ANE, you'll have access to the Bluetooth hardware. It enable you to scan for other devices, connect to and pair with them and finally transfer data between them. This ANE has been built with consideration of connecting to Android devices to each other. if you need it for other purposes, feel free to contact us here: http://www.myflashlabs.com/contact/ (This is NOT a Bluetooth LE API)

asdoc

find the latest asdoc for this ANE here.

Demo .apk

you may like to see the ANE in action? Download demo .apk

NOTICE: the demo ANE works only after you hit the "OK" button in the dialog which opens. in your tests make sure that you are NOT calling other ANE methods prior to hitting the "OK" button. Download the ANE

Air Usage:

package 
{
	import com.myflashlab.air.extensions.bluetooth.Bluetooth;
	import com.myflashlab.air.extensions.bluetooth.BluetoothEvent;
	import com.doitflash.consts.Direction;
	import com.doitflash.consts.Orientation;
	import com.doitflash.mobileProject.commonCpuSrc.DeviceInfo;
	import com.doitflash.starling.utils.list.List;
	import com.doitflash.text.modules.MySprite;
	import com.greensock.TweenMax;
	import flash.desktop.NativeApplication;
	import flash.desktop.SystemIdleMode;
	import flash.display.Sprite;
	import flash.display.StageAlign;
	import flash.display.StageScaleMode;
	import flash.events.Event;
	import flash.events.InvokeEvent;
	import flash.events.KeyboardEvent;
	import flash.events.MouseEvent;
	import flash.text.AntiAliasType;
	import flash.text.TextField;
	import flash.text.TextFieldAutoSize;
	import flash.text.TextFieldType;
	import flash.text.TextFormat;
	import flash.text.TextFormatAlign;
	import flash.ui.Keyboard;
	import flash.ui.Multitouch;
	import flash.ui.MultitouchInputMode;
	
	/**
	 * ...
	 * @author Hadi Tavakoli - 2/24/2014 5:30 PM
	 */
	public class Demo2 extends Sprite 
	{
		private var _ex:Bluetooth;
		private var _buttonsHolder:Sprite;
		private var _devicesHolder:Sprite;
		private var _list:List;
		
		private var _txt:TextField;
		
		private var _ball:MySprite;
		
		private const BTN_WIDTH:Number = 150;
		private const BTN_HEIGHT:Number = 100;
		
		public function Demo2():void 
		{
			Multitouch.inputMode = MultitouchInputMode.GESTURE;
			NativeApplication.nativeApplication.addEventListener(Event.ACTIVATE, handleActivate, false, 0, true);
			NativeApplication.nativeApplication.addEventListener(Event.DEACTIVATE, handleDeactivate, false, 0, true);
			NativeApplication.nativeApplication.addEventListener(InvokeEvent.INVOKE, onInvoke, false, 0, true);
			NativeApplication.nativeApplication.addEventListener(KeyboardEvent.KEY_DOWN, handleKeys, false, 0, true);
			
			stage.addEventListener(Event.RESIZE, onResize);
			stage.scaleMode = StageScaleMode.NO_SCALE;
			stage.align = StageAlign.TOP_LEFT;
			
			_buttonsHolder = new Sprite();
			_buttonsHolder.visible = false;
			this.addChild(_buttonsHolder);
			
			_devicesHolder = new Sprite();
			_devicesHolder.visible = false;
			this.addChild(_devicesHolder);
			
			_list = new List();
			_list.holder = _devicesHolder;
			_list.itemsHolder = new Sprite();
			_list.orientation = Orientation.VERTICAL;
			_list.hDirection = Direction.LEFT_TO_RIGHT;
			_list.vDirection = Direction.TOP_TO_BOTTOM;
			_list.space = 2;
			
			init();
			onResize();
		}
		
		private function onInvoke(e:InvokeEvent):void
		{
			NativeApplication.nativeApplication.removeEventListener(InvokeEvent.INVOKE, onInvoke);
		}
		
		private function handleActivate(e:Event):void
		{
			NativeApplication.nativeApplication.systemIdleMode = SystemIdleMode.KEEP_AWAKE;
		}
		
		private function handleDeactivate(e:Event):void
		{
			
		}
		
		private function handleKeys(e:KeyboardEvent):void
		{
			if(e.keyCode == Keyboard.BACK)
            {
				if (_ex) 
				{
					_ex.dispose(); // ALWAYS dispose the bluetooth extension before closing your app
				}
				e.preventDefault();
				NativeApplication.nativeApplication.exit();
            }
		}
		
		private function onResize(e:*=null):void
		{			
			if (_devicesHolder)
			{
				_devicesHolder.y = _buttonsHolder.y + _buttonsHolder.height + 5;
			}
			
			if (_txt)
			{
				_txt.width = stage.stageWidth;
				_txt.height = 100;
				_txt.y = stage.stageHeight - _txt.height;
				_txt.x = stage.stageWidth / 2 - _txt.width / 2;
			}
		}
		
		private function init():void
		{
			_txt = new TextField();
			_txt.mouseEnabled = false;
			_txt.selectable = false;
			//_txt.autoSize = TextFieldAutoSize.LEFT;
			_txt.defaultTextFormat = new TextFormat(null, 22, null, null, null, null, null, null, "center");
			_txt.text = "Scan to find a device";
			_txt.scaleX = _txt.scaleY = DeviceInfo.dpiScaleMultiplier;
			this.addChild(_txt);
			
			// initialize the extension
			_ex = new Bluetooth();
			_ex.addEventListener(BluetoothEvent.BLUETOOTH_STATE , bluetoothState);
			_ex.addEventListener(BluetoothEvent.COMMUNICATION_STATUS , communication);
			_ex.addEventListener(BluetoothEvent.CONNECTION , connection);
			_ex.addEventListener(BluetoothEvent.DIALOG_STATUS , dialog);
			_ex.addEventListener(BluetoothEvent.DISCOVERING_STATUS , discovering);
			_ex.addEventListener(BluetoothEvent.NEW_DISCOVERD , newDiscoverd);
			_ex.addEventListener(BluetoothEvent.READ_MESSAGE , readMessage);
			_ex.addEventListener(BluetoothEvent.SCAN_MODE , scanMode);
			_ex.addEventListener(BluetoothEvent.SEND_MESSAGE , sendMessage);
			
			var btn1:MySprite = createBtn("start scan");
			btn1.addEventListener(MouseEvent.CLICK, startScan);
			_buttonsHolder.addChild(btn1);
			
			function startScan(e:MouseEvent):void
			{
				_ex.startDiscovery();
			}
			
			// check if bluetooth is on
			if (_ex.isEnable)
			{
				_buttonsHolder.visible = true;
				_devicesHolder.visible = true;
				_ex.visible(0);
				_ex.initCommunicationService();
			}
			else
			{
				_ex.enable();
			}
		}
		
		private function initBall():void
		{
			if (_ball) removeBall();
			
			_list.removeAll();
			
			_ball = new MySprite();
			_ball.addEventListener(MouseEvent.MOUSE_DOWN, onDown);
			_ball.bgColor = 0xdf4c47;
			_ball.bgAlpha = 1;
			_ball.bgBottomLeftRadius = 20;
			_ball.bgBottomRightRadius = 20;
			_ball.bgTopLeftRadius = 20;
			_ball.bgTopRightRadius = 20;
			_ball.width = 200;
			_ball.height = 200;
			_ball.drawBg();
			_ball.x = stage.stageWidth / 2 - _ball.width / 2;
			_ball.y = stage.stageHeight / 2 - _ball.height / 2;
			this.addChild(_ball);
			TweenMax.from(_ball, 0.5, {alpha:0} );
			
			var txt:TextField = new TextField();
			txt.mouseEnabled = false;
			txt.selectable = false;
			txt.autoSize = TextFieldAutoSize.LEFT;
			txt.multiline = true;
			txt.defaultTextFormat = new TextFormat(null, 30, 0xFFFFFF, null, null, null, null, null, "center");
			txt.htmlText = "DRAG<br>ME";
			txt.scaleX = _txt.scaleY = DeviceInfo.dpiScaleMultiplier;
			txt.x = _ball.width / 2 - txt.width / 2;
			txt.y = _ball.height / 2 - txt.height / 2;
			_ball.addChild(txt);
		}
		
		private function removeBall():void
		{
			this.removeChild(_ball);
			_ball = null
		}
		
		private function onDown(e:MouseEvent):void
		{
			_ball.startDrag(false);
			_ball.addEventListener(MouseEvent.MOUSE_MOVE, onMove);
			_ball.addEventListener(MouseEvent.MOUSE_UP, onUp);
		}
		
		private function onMove(e:MouseEvent):void
		{
			//trace(_ball.x, _ball.y)
			_ex.sendMessage(String(Number(_ball.x / stage.stageWidth).toFixed(2)) + "|" + String(Number(_ball.y / stage.stageHeight).toFixed(2)));
		}
		
		private function onUp(e:MouseEvent):void
		{
			_ball.stopDrag();
			_ball.removeEventListener(MouseEvent.MOUSE_MOVE, onMove);
			_ball.removeEventListener(MouseEvent.MOUSE_UP, onUp);
		}
		
		private function bluetoothState(e:BluetoothEvent):void
		{
			trace("state >> " + e.param);
			
			if (e.param == "bluetoothOn")
			{
				_buttonsHolder.visible = true;
				_devicesHolder.visible = true;
				_ex.visible(0);
				_ex.initCommunicationService();
			}
		}
		
		private function communication(e:BluetoothEvent):void
		{
			trace("communication >> " + e.param);
			
			if (e.param == "connecting")
			{
				_txt.text = "connecting to the device...";
			}
			else if (e.param == "connected")
			{
				_txt.text = "CONNECTED";
				initBall();
			}
		}
		
		private function connection(e:BluetoothEvent):void
		{
			trace("connection >> " + e.param);
			
			if (e.param == "connectionLost")
			{
				_txt.text = "connection lost!";
				removeBall();
			}
		}
		
		private function dialog(e:BluetoothEvent):void
		{
			trace("dialogStatus >> " + e.param);
		}
		
		private function discovering(e:BluetoothEvent):void
		{
			trace("discovering >> " + e.param);
			
			if (e.param == "discoveringStarted")
			{
				_txt.text = "Scaning...";
				
				// remove everything inside _devicesHolder
				_list.removeAll()
				
				_devicesHolder.alpha = 0.5;
				_devicesHolder.mouseEnabled = false;
				_devicesHolder.mouseChildren = false;
			}
			else if (e.param == "discoveringFinished")
			{
				_txt.text = "Scan finished";
				
				_devicesHolder.alpha = 1;
				_devicesHolder.mouseEnabled = true;
				_devicesHolder.mouseChildren = true;
			}
		}
		
		private function newDiscoverd(e:BluetoothEvent):void
		{
			var obj:Object = e.param;
			
			trace("newDiscoverd >> " + "device = " + obj.device + " mac :" + obj.mac);
			
			var btn1:MySprite = createBtn("Device: " + obj.device);
			btn1.data = obj;
			btn1.addEventListener(MouseEvent.CLICK, toConnect);
			_list.add(btn1);
			_list.itemArrange();
			
			function toConnect(e:MouseEvent):void
			{
				var mac:String = e.target["data"].mac;
				
				// check if we are paired?
				var pairList:Array = _ex.pairedList;
				var currDevice:Object;
				for (var i:int = 0; i < pairList.length; i++) 
				{
					currDevice = pairList[i];
					/*for (var name:String in currDevice) 
					{
						trace(name + " = " + currDevice[name])
					}
					trace("-----------")*/
					if (currDevice.mac == mac)
					{
						_ex.connectTo(mac);
						return;
					}
				}
				
				// if we are here, it means that the devices are not paired. let's pair them now!
				_ex.pairWith(mac);
			}
		}
		
		private function readMessage(e:BluetoothEvent):void
		{
			trace("readMessage >> " + e.param);
			var arr:Array = e.param.split("|");
			
			// bluetooth speed is so high sometimes that it sends two pair of ball points!
			if (arr.length > 2) return; // validate the incoming value to make sure we have one X and one Y value
			
			TweenMax.to(_ball, 0.3, {x:stage.stageWidth * Number(arr[0]), y:stage.stageHeight * Number(arr[1])} );
			//_ball.x = stage.stageWidth * Number(arr[0]);
			//_ball.y = stage.stageHeight * Number(arr[1]);
		}
		
		private function scanMode(e:BluetoothEvent):void
		{
			trace("scanMode >> " + e.param);
		}
		
		private function sendMessage(e:BluetoothEvent):void
		{
			trace("sendMessage >> " + e.param);
		}
		
		private function createBtn($str:String, $txtColor:uint=0x666666, $editable:Boolean=false):MySprite
		{
			var sp:MySprite = new MySprite();
			sp.addEventListener(MouseEvent.MOUSE_OVER,  onOver);
			sp.addEventListener(MouseEvent.MOUSE_OUT,  onOut);
			sp.addEventListener(MouseEvent.CLICK,  onOut);
			sp.bgAlpha = 1;
			sp.bgColor = 0xDFE4FF;
			sp.drawBg();
			sp.width = BTN_WIDTH * DeviceInfo.dpiScaleMultiplier;
			sp.height = BTN_HEIGHT * DeviceInfo.dpiScaleMultiplier;
			
			function onOver(e:MouseEvent):void
			{
				sp.bgAlpha = 1;
				sp.bgColor = 0xFFDB48;
				sp.drawBg();
			}
			
			function onOut(e:MouseEvent):void
			{
				sp.bgAlpha = 1;
				sp.bgColor = 0xDFE4FF;
				sp.drawBg();
			}
			
			var format:TextFormat = new TextFormat("Arimo", 18, $txtColor, null, null, null, null, null, TextFormatAlign.CENTER);
			
			var txt:TextField = new TextField();
			txt.autoSize = TextFieldAutoSize.LEFT;
			txt.antiAliasType = AntiAliasType.ADVANCED;
			txt.mouseEnabled = $editable;
			txt.multiline = true;
			txt.wordWrap = true;
			txt.border = $editable;
			if ($editable) txt.type = TextFieldType.INPUT;
			txt.scaleX = txt.scaleY = DeviceInfo.dpiScaleMultiplier;
			txt.width = sp.width * (1 / (DeviceInfo.dpiScaleMultiplier));
			txt.defaultTextFormat = format;
			txt.htmlText = $str;
			
			txt.y = sp.height - txt.height >> 1;
			sp.addChild(txt);
			sp.data.txt = txt;
			
			return sp;
		}
	}
	
}

Air .xml manifest

<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />

<!-- 	
	Required for Bluetooth to work properly from Android 6.0 or higher. More info here: 
	http://developer.android.com/about/versions/marshmallow/android-6.0-changes.html#behavior-hardware-id 
-->
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>

Requirements:

  • Android SDK 10 or higher
  • Air SDK 19 or higher

Commercial Version

http://www.myflashlabs.com/product/bluetooth-ane-adobe-air-native-extension/

Bluetooth ANE

Tutorials

How to embed ANEs into FlashBuilder, FlashCC and FlashDevelop
Step by step tutorial showing you how to use this ANE to make a multiplayr-game

Changelog

Jan 01, 2016 - V2.0

  • upgraded to Android Studio

Nov 02, 2015 - V1.9

  • removed android-support-v4.jar dependency
  • doitflash devs merged into MyFLashLab Team

Feb 14, 2015 - V1.5

  • added Android 64 support

Feb 25, 2014 - V1.0

  • beginning of the journey!

About

With Bluetooth air native extension, you'll have access to the Bluetooth hardware. It enable you to scan for other devices, connect to and pair with them and finally transfer data between them.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published