/* * (c)2014-2017 Forward Computing and Control Pty. Ltd. * NSW Australia, www.forward.com.au * This code is not warranted to be fit for any purpose. You may only use it at your own risk. * This generated code may be freely used for both private and commercial use * provide this copyright is maintained. */ // download the libraries from http://www.forward.com.au/pfod/pfodParserLibraries/index.html // pfodParser V3.16+, pfodDwgControls V1.2+ #include #include #include const char* deviceName = "pfod bluefruit52"; // 30 chars or less in name // BLE Service BLEDis bledis; BLEUart bleuart; pfodParser parser("S1"); // create a parser to handle the pfod messages pfodDwgs dwgs(&parser); // drawing support Gauge gauge(&dwgs); Slider slider('a', &dwgs); const int maxTextChars = 11; byte currentText[maxTextChars + 1] = "Hello"; // allow max 11 chars + null == 12, note msg {'x`11~Example Text Input screen.\n" enforces max 11 bytes in return value const int imagesize = 50; // this set the image size bool sendCleanImage = false; // this is set when next image load/update should send blank image int currentSingleSelection = 3; const int MAX_MULTI_SELECTIONS = 5; //max 5 possible selections out of all possible, long multiSelections[MAX_MULTI_SELECTIONS]; // -1 means not selected int redValue = 127; int greenValue = 127; int blueValue = 127; int fanPosition = 0; const unsigned long RAW_DATA_INTERVAL = 2000; // 2 sec unsigned long rawDataTimer; int counter = 0; // give the board pins names, if you change the pin number here you will change the pin controlled const int LED_pin = LED_RED; void setup() { parser.connect(&bleuart); Bluefruit.autoConnLed(true); // use Bluefruit.autoConnLed(false); to save power or to control BLUE Led manually via Pin 19 Bluefruit.begin(); // Set max power. Accepted values are: -40, -30, -20, -16, -12, -8, -4, 0, 4 Bluefruit.setTxPower(4); Bluefruit.setName(deviceName); // 30 chars or less in name Bluefruit.setConnectCallback(connect_callback); Bluefruit.setDisconnectCallback(disconnect_callback); // Configure and Start Device Information Service bledis.setManufacturer("Adafruit Industries"); bledis.setModel("Bluefruit Feather52"); bledis.begin(); // Configure and Start BLE Uart Service bleuart.begin(); // Set up and start advertising startAdv(); // <<<<<<<<< Your extra setup code goes here slider.setLabel(F("Red Led\n")); gauge.setLabel(F("Red Led\n")); } void loop() { uint8_t cmd = parser.parse(); // parse incoming data from connection // parser returns non-zero when a pfod command is fully parsed if (cmd != 0) { // have parsed a complete msg { to } uint8_t* pfodFirstArg = parser.getFirstArg(); // may point to \0 if no arguments in this msg. pfod_MAYBE_UNUSED(pfodFirstArg); // may not be used, just suppress warning long pfodLongRtn; // used for parsing long return arguments, if any pfod_MAYBE_UNUSED(pfodLongRtn); // may not be used, just suppress warning if ('.' == cmd) { // pfodApp has connected and sent {.} , it is asking for the main menu // send back the menu designed if (!parser.isRefresh()) { sendMainMenu(); } else { sendMainMenuUpdate(); } } else if ('p' == cmd) { sendPlottingScreen(); // plots the raw data being logged } else if ('l' == cmd) { if (!parser.isRefresh()) { sendListMenu(); } else { sendListMenuUpdate(); } } else if ('s' == cmd) { if (!parser.isRefresh()) { sendSliderMenu(); } else { sendSliderMenuUpdate(); } } else if ('x' == cmd) { // return from text input, pickup the first arg which is the number strncpy((char*)currentText, (const char*)pfodFirstArg, sizeof(currentText) - 1); // keep null at end parser.print(F("{}")); // nothing to update pfodApp will request previous menu } else if ('y' == cmd) { sendSingleSelectionScreen(); } else if ('S' == cmd) { // return from single selection input, pickup the first arg which is the number parser.parseLong(pfodFirstArg, &pfodLongRtn); // stores result in a long currentSingleSelection = (int)pfodLongRtn; parser.print(F("{}")); } else if ('F' == cmd) { if (!parser.isRefresh()) { sendFanControl(); } else { sendFanControlUpdate(); } } else if ('G' == cmd) { if (!parser.isRefresh()) { dwgMenu(); } else { dwgMenuUpdate(); } } else if ('A' == cmd) { // user pressed menu item that loaded drawing with load cmd 'z' byte dwgCmd = parser.parseDwgCmd(); // parse rest of dwgCmd, return first char of active cmd if (slider.getCmd() == dwgCmd) { // user touched LED control int col = parser.getTouchedCol(); slider.setValue(col); // set new value gauge.setValue(slider.getValue()); // set gauge from slider analogWrite(LED_pin, slider.getValue()); // set LED PWM 0 == OFF } sendDrawingUpdates_z(); // update the drawing // always send back a response or pfodApp will timeout } else if ('z' == cmd) { // pfodApp is asking to load dwg 'z' if (!parser.isRefresh()) { // not refresh send whole dwg parser.parseLong(pfodFirstArg,&pfodLongRtn); // parse first arg as a long sendDrawing_z(pfodLongRtn); } else { // refresh just update drawing state sendDrawingUpdates_z(); } } else if ('o' == cmd) { // return from fan control, pickup the first arg which is the number parser.parseLong(pfodFirstArg, &pfodLongRtn); fanPosition = (int)pfodLongRtn; sendFanControlUpdate(); } else if ('r' == cmd) { // return from red colour selector, pickup the first arg which is the number parser.parseLong(pfodFirstArg, &pfodLongRtn); redValue = (int)pfodLongRtn; sendColourSelectorUpdate(); } else if ('g' == cmd) { // return from green colour selector, pickup the first arg which is the number parser.parseLong(pfodFirstArg, &pfodLongRtn); greenValue = (int)pfodLongRtn; sendColourSelectorUpdate(); } else if ('b' == cmd) { // return from blue colour selector, pickup the first arg which is the number parser.parseLong(pfodFirstArg, &pfodLongRtn); blueValue = (int)pfodLongRtn; sendColourSelectorUpdate(); } else if ('m' == cmd) { sendMultiSelectionScreen(); } else if ('M' == cmd) { // return from multi selection input byte* argIdx = pfodFirstArg; // pickup the first arg which is the number // will be null if no arges int i = 0; for (; (i < MAX_MULTI_SELECTIONS) && (*argIdx != 0); i++) { argIdx = parser.parseLong(argIdx, &multiSelections[i]); //argIdx will be null after last arg is parsed } if (i < MAX_MULTI_SELECTIONS) { // add a -1 to terminate the array of indices multiSelections[i] = -1; } parser.print(F("{}")); } else if ('L' == cmd) { if (!parser.isRefresh()) { sendColourSelector(); } else { sendColourSelectorUpdate(); } } else if ('i' == cmd) { sendTextInputScreen(); } else if ('u' == cmd) { sendRawDataScreen(); } else if ('!' == cmd) { // CloseConnection command closeConnection(parser.getPfodAppStream()); } else { // unknown command parser.print(F("{}")); // always send back a pfod msg otherwise pfodApp will disconnect. } } sendData(); // <<<<<<<<<<< Your other loop() code goes here } void closeConnection(Stream *io) { // nothing special here } void startAdv(void) { // Advertising packet Bluefruit.Advertising.addFlags(BLE_GAP_ADV_FLAGS_LE_ONLY_GENERAL_DISC_MODE); Bluefruit.Advertising.addTxPower(); // Include bleuart 128-bit uuid Bluefruit.Advertising.addService(bleuart); Bluefruit.ScanResponse.addName(); Bluefruit.Advertising.restartOnDisconnect(true); Bluefruit.Advertising.setInterval(32, 244); // in unit of 0.625 ms Bluefruit.Advertising.setFastTimeout(30); // number of seconds in fast mode Bluefruit.Advertising.start(0); // 0 = Don't stop advertising after n seconds } void connect_callback(uint16_t conn_handle) { } void disconnect_callback(uint16_t conn_handle, uint8_t reason) { // clear parser with -1 and insert {!} incase connection just lost parser.parse((byte) - 1); // eof parser.parse((byte)'{'); // parser.parse((byte)'!'); // close connection parser.parse((byte)'}'); // } void sendData() { if (millis() - rawDataTimer > RAW_DATA_INTERVAL) { // send raw data // note DO NOT call Serial print directly as it will interfer with the msg hash on secure connections parser.print(rawDataTimer / 1000); // the time in seconds, first field parser.print(','); // field separator parser.print(counter++); // second field parser.print(','); // field separator parser.print(analogRead(A4)); // third field parser.println(); // terminate data record rawDataTimer = millis(); // allow it to drift due to delays in sending data } } void sendMainMenu() { // put main msg in input array parser.print(F("{,~Redbear BLE Nano V2\nSample Screens")); parser.sendVersion(); // send pfod menu version parser.print(F("|^l~Lists|^s~Sliders and Dwgs|^i~Text Input|^u~Raw Data|^p~Plots}")); } void sendMainMenuUpdate() { parser.print(F("{;}")); // nothing to update here } void sendSliderMenu() { parser.print(F("{,~Slider and Dwg Examples")); parser.sendVersion(); // send pfod menu version parser.print(F("|L~<+1>Colour Selector Example" "|F~<+1>Fan Control" "|G~<+1>Drawings\nSlider/Gauge Example" "}")); } void sendSliderMenuUpdate() { parser.print(F("{;}")); // nothing to update here } void sendListMenu() { parser.print(F("{,~Lists")); parser.sendVersion(); // send pfod menu version parser.print(F("|y~Single Selection|m~Multi-selection}")); } void sendListMenuUpdate() { parser.print(F("{;}")); // nothing to update here } void sendTextInputScreen() { parser.print(F("{'x`")); parser.print(maxTextChars); parser.print(F("~Text Input\n(Limited to 11 chars by Arduino code to match allocated storage)" "|")); parser.print((char*)currentText); parser.print(F("}")); } void sendSingleSelectionScreen() { parser.print(F("{?S`")); // rest of msg handle in main loop parser.print(currentSingleSelection); parser.print(F("~Single Selection" "|High|Medium|Low|Off}")); } void sendMultiSelectionScreen() { parser.print(F("{*M")); // add the current selections for (int i = 0; i < MAX_MULTI_SELECTIONS; i++) { long idx = multiSelections[i]; if (idx < 0) { break; } // else parser.print(F("`")); parser.print(idx); } parser.print(F("~Multi-selection" "|Fade on/off|3 Levels|Feature A|Feature B|Feature C}")); } void sendColourSelector() { parser.print(F("{,")); sendColourScreen(); parser.print(F("}")); } void sendColourSelectorUpdate() { parser.print(F("{;")); parser.print(F("")); parser.print(F("|r")); parser.print('`'); parser.print(redValue); parser.print(F("|g")); parser.print('`'); parser.print(greenValue); parser.print(F("|b")); parser.print('`'); parser.print(blueValue); parser.print(F("}")); } void sendColourScreen() { parser.print(F("~")); parser.sendVersion(); // send pfod menu version parser.print(F("|r")); parser.print('`'); parser.print(redValue); parser.print(F("~Red ")); parser.print(F("`255`0~")); // parser.print(F("|g")); parser.print('`'); parser.print(greenValue); parser.print(F("~Green ")); parser.print(F("`255`0~")); // parser.print(F("|b")); parser.print('`'); parser.print(blueValue); parser.print(F("~Blue ")); parser.print(F("`255`0~")); // } void sendFanControl() { parser.print(F("{,")); sendFanControlScreen(); parser.print(F("}")); } void sendFanControlUpdate() { parser.print(F("{;")); parser.print('`'); parser.print(fanPosition); parser.print(F("}")); } void sendFanControlScreen() { parser.print(F("~<+5>Fan Control")); parser.sendVersion(); // send pfod menu version parser.print(F("|o")); parser.print('`'); parser.print(fanPosition); parser.print(F("~<+4>Fan is ")); parser.print(F("~~Off\\Low\\High")); } void dwgMenu() { parser.print(F("{,")); // start a Menu screen pfod message // send menu background, format, prompt, refresh and version parser.print(F("~`0")); parser.sendVersion(); // send the menu version // send menu items parser.print(F("|+A~z")); parser.print(F("}")); // close pfod message } void dwgMenuUpdate() { parser.print(F("{;")); // start an Update Menu pfod message // send menu items parser.print(F("|A")); parser.print(F("}")); // close pfod message } void sendDrawing_z(long dwgIdx) { if (dwgIdx == 0) { sendDrawing_z0(); } else if (dwgIdx == 1) { sendDrawing_z1(); // add more parts of the dwg here if needed } else { parser.print(F("{}")); // always return somthing } } void sendDrawing_z0() { dwgs.start(50, 35, dwgs.WHITE,true); // background defaults to WHITE if omitted i.e. dwgs.start(50,35); parser.sendVersion(); // send the parser version to cache this image dwgs.pushZero(25, 12, 1.5); // move zero to centre of dwg to 25,12 and scale by 1.5 times gauge.draw(); // draw the control dwgs.popZero(); dwgs.pushZero(9, 29); // move zero to centre of dwg to 9,30 and scale is 1 (default) slider.draw(); // draw the control dwgs.popZero(); dwgs.end(); } void sendDrawing_z1() { dwgs.startUpdate(true); //true means more to come, // empty date so pfodApp will not ask for more dwgs.end(); } void sendDrawingUpdates_z() { dwgs.startUpdate(); gauge.update(); // update with current state slider.update(); // update with current state dwgs.end(); } void sendRawDataScreen() { // this illustrates how you can send much more then 255 bytes as raw data. // The {=Raw Data Screen} just tells the pfodApp to open the rawData screen and give it a title // all the rest of the text (outside the { } ) is just raw data text and can be a much as you like // Note the raw data includes { } using { is OK as long as the next char is not a pfodApp msg identifer // that is the following cannot appear in rawData {@ {. {: {, {; {^ {= {' {# {? or {* parser.print(F("{=Raw Data Screen}" "This is the Raw Data Screen\n\n" "The pfodDevice can write more the 255 chars to this screen.\n" "Any bytes sent outside the pfod message { } start/end characters are" " displayed here.\n" " The raw data used for plotting and data logging is also displayed here.\n" "\nUse the Back Button on the mobile to go back to the previous menu." "\n \n")); } void sendPlottingScreen() { parser.print(F("{=" // streaming raw data screen "counter and ADC A4 versus time" // plot title "|Time (sec)|counter|A4}")); // show data in a plot // these match the 3 fields in the raw data record }