UNPKG

@iotize/ionic

Version:

Iotize specific building blocks on top of @ionic/angular.

181 lines 21.9 kB
import { ChangeDetectorRef, Component, EventEmitter, Inject, Input, Output, TemplateRef, } from '@angular/core'; import { LoadingController } from '@ionic/angular'; import { Subject } from 'rxjs'; import { takeUntil } from 'rxjs/operators'; import { TAP_BLE_SCANNER } from '../../../injection-types'; import * as i0 from "@angular/core"; import * as i1 from "@ionic/angular"; import * as i2 from "@angular/common"; import * as i3 from "./tap-scan-results/tap-scan-results.component"; /** * Scanner for bluetooth */ export class TapScannerBleComponent { tapScanner; changeDetector; loadingCtrl; /** * Scan result template to use to display scan results. * * If not provided, it will use a default component to display devices. * * @example * <!-- Display devices with tap-ble-scan-result component --> * <tap-scanner-ble [scanResultsTemplate]="scanResultsTemplate"> * <ng-template #scanResultsTemplate let-devices> * <tap-ble-scan-result [devices]="devices" [showSignalStrenght]="false"></tap-ble-scan-result> * </ng-template> * </tap-scanner-ble> * */ scanResultsTemplate; /** * @deprecated Not used anymore. start/stop buttons must be define by user */ showScanButton = true; /** * RSSI update threshold. * If device list is the same, we will only emit a device update event * if the Math.abs(previousRSSI - newRSSI) >= rssiUpdateThreshold */ rssiUpdateThreshold = 10; showMacAddress = true; showSignalStrength = true; /** * Event emitted when a scan result has been selected * @deprecated see protocolSelected */ onProtocolSelected = new EventEmitter(); /** * Event emitted when a scan result has been selected */ protocolSelected = new EventEmitter(); /** * Event emitted when an error occurs */ onError = new EventEmitter(); results = []; destroyed = new Subject(); isScanning = false; constructor(tapScanner, changeDetector, loadingCtrl) { this.tapScanner = tapScanner; this.changeDetector = changeDetector; this.loadingCtrl = loadingCtrl; } ngOnInit() { this.tapScanner.results .pipe(takeUntil(this.destroyed)) .subscribe((values) => { this.mergeScanResults(values); }); this.tapScanner.scanning.pipe(takeUntil(this.destroyed)).subscribe((v) => { if (v !== this.isScanning) { this.isScanning = v; } }); } notifyError(error) { console.error('BLEScannerError', error); this.onError.emit(error); } ngOnDestroy() { this.stopScan(); this.destroyed.next(); this.destroyed.complete(); } onItemClicked(device) { const protocolMeta = { type: 'ble', info: { id: device.address, mac: device.address, name: device.name, }, }; const event = { meta: protocolMeta, }; this.protocolSelected.emit(event); this.onProtocolSelected.emit(event); } startScan() { // this.devices = this.devices.filter(device => device.address == this.tapScanner.selectedDevice); // this.devices.splice(0, this.devices.length); this.tapScanner.start().catch((err) => { this.notifyError(err); }); // this.onScanStateChanged.emit(true); } stopScan() { this.tapScanner.stop().catch((err) => { this.notifyError(err); }); // this.onScanStateChanged.emit(false); } mergeScanResults(newResults) { /* newResults.forEach((newResult) => { this.addOrRefreshScanResultList(newResult); }); const indexToRemove = this.results.map((oldDevice) => { return newResults.findIndex( (entry) => entry.address === oldDevice.address ); }, []); indexToRemove.forEach((resultIndex) => { if (resultIndex < 0) { this.results.splice(resultIndex, 1); } }); */ this.results = newResults; this.changeDetector.detectChanges(); } addOrRefreshScanResultList(newDevice) { const devices = this.results; let hasChanged = false; const storedDeviceIndex = devices.findIndex((entry) => entry.address === newDevice.address); if (storedDeviceIndex >= 0) { const storedDevice = devices[storedDeviceIndex]; if (storedDevice.name !== newDevice.name) { storedDevice.name = newDevice.name; hasChanged = true; } if (Math.abs(storedDevice.rssi - newDevice.rssi) >= this.rssiUpdateThreshold) { storedDevice.rssi = newDevice.rssi; hasChanged = true; } } else { devices.push(newDevice); hasChanged = true; } return hasChanged; } /** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TapScannerBleComponent, deps: [{ token: TAP_BLE_SCANNER }, { token: i0.ChangeDetectorRef }, { token: i1.LoadingController }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: TapScannerBleComponent, selector: "tap-scanner-ble", inputs: { scanResultsTemplate: "scanResultsTemplate", showScanButton: "showScanButton", rssiUpdateThreshold: "rssiUpdateThreshold", showMacAddress: "showMacAddress", showSignalStrength: "showSignalStrength" }, outputs: { onProtocolSelected: "onProtocolSelected", protocolSelected: "protocolSelected", onError: "onError" }, ngImport: i0, template: "<ng-container\n *ngTemplateOutlet=\"scanResultsContainer; context: { $implicit: results }\"\n></ng-container>\n\n<ng-template #scanResultsContainer let-devices>\n <!-- <ng-content #scanResultsInner select=\"[scan-results]\"></ng-content> -->\n <ng-template\n *ngTemplateOutlet=\"scanResultsTemplate; context: { $implicit: results }\"\n ></ng-template>\n <tap-ble-scan-result\n *ngIf=\"!scanResultsTemplate\"\n [devices]=\"devices\"\n [showMacAddress]=\"showMacAddress\"\n [showSignalStrength]=\"showSignalStrength\"\n (onItemClicked)=\"onItemClicked($event)\"\n ></tap-ble-scan-result>\n</ng-template>\n", styles: [""], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i3.TapScanResultsComponent, selector: "tap-ble-scan-result", inputs: ["lines", "showMacAddress", "showSignalStrenght", "showSignalStrength", "devices"], outputs: ["onItemClicked"] }] }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TapScannerBleComponent, decorators: [{ type: Component, args: [{ selector: 'tap-scanner-ble', template: "<ng-container\n *ngTemplateOutlet=\"scanResultsContainer; context: { $implicit: results }\"\n></ng-container>\n\n<ng-template #scanResultsContainer let-devices>\n <!-- <ng-content #scanResultsInner select=\"[scan-results]\"></ng-content> -->\n <ng-template\n *ngTemplateOutlet=\"scanResultsTemplate; context: { $implicit: results }\"\n ></ng-template>\n <tap-ble-scan-result\n *ngIf=\"!scanResultsTemplate\"\n [devices]=\"devices\"\n [showMacAddress]=\"showMacAddress\"\n [showSignalStrength]=\"showSignalStrength\"\n (onItemClicked)=\"onItemClicked($event)\"\n ></tap-ble-scan-result>\n</ng-template>\n" }] }], ctorParameters: () => [{ type: undefined, decorators: [{ type: Inject, args: [TAP_BLE_SCANNER] }] }, { type: i0.ChangeDetectorRef }, { type: i1.LoadingController }], propDecorators: { scanResultsTemplate: [{ type: Input }], showScanButton: [{ type: Input }], rssiUpdateThreshold: [{ type: Input }], showMacAddress: [{ type: Input }], showSignalStrength: [{ type: Input }], onProtocolSelected: [{ type: Output }], protocolSelected: [{ type: Output }], onError: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tap-scanner-ble.component.js","sourceRoot":"","sources":["../../../../../../../projects/iotize-ionic/src/lib/tap-scanner/tap-scanner-ble/tap-scanner-ble.component.ts","../../../../../../../projects/iotize-ionic/src/lib/tap-scanner/tap-scanner-ble/tap-scanner-ble.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  TemplateRef,\n} from '@angular/core';\nimport { LoadingController } from '@ionic/angular';\nimport { DeviceScanner } from '@iotize/tap/scanner/api';\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\nimport { TAP_BLE_SCANNER } from '../../../injection-types';\nimport { ProtocolMetaBle } from '../../extensions/protocol-info';\nimport { ProtocolSelectedEvent } from '../api';\nimport { DiscoveredDeviceType } from './definitions';\n\n/**\n * Scanner for bluetooth\n */\n@Component({\n  selector: 'tap-scanner-ble',\n  templateUrl: './tap-scanner-ble.component.html',\n  styleUrls: ['./tap-scanner-ble.component.scss'],\n})\nexport class TapScannerBleComponent implements OnInit, OnDestroy {\n  /**\n   * Scan result template to use to display scan results.\n   *\n   * If not provided, it will use a default component to display devices.\n   *\n   * @example\n   * <!-- Display devices with tap-ble-scan-result component -->\n   * <tap-scanner-ble [scanResultsTemplate]=\"scanResultsTemplate\">\n   *    <ng-template #scanResultsTemplate let-devices>\n   *        <tap-ble-scan-result [devices]=\"devices\" [showSignalStrenght]=\"false\"></tap-ble-scan-result>\n   *    </ng-template>\n   * </tap-scanner-ble>\n   *\n   */\n  @Input() scanResultsTemplate?: TemplateRef<any>;\n\n  /**\n   * @deprecated Not used anymore. start/stop buttons must be define by user\n   */\n  @Input() showScanButton = true;\n\n  /**\n   * RSSI update threshold.\n   * If device list is the same, we will only emit a device update event\n   * if the Math.abs(previousRSSI - newRSSI) >= rssiUpdateThreshold\n   */\n  @Input() rssiUpdateThreshold = 10;\n\n  @Input() showMacAddress = true;\n\n  @Input() showSignalStrength = true;\n\n  /**\n   * Event emitted when a scan result has been selected\n   * @deprecated see protocolSelected\n   */\n  @Output() onProtocolSelected = new EventEmitter<ProtocolSelectedEvent>();\n  /**\n   * Event emitted when a scan result has been selected\n   */\n  @Output() protocolSelected = new EventEmitter<ProtocolSelectedEvent>();\n\n  /**\n   * Event emitted when an error occurs\n   */\n  @Output() onError = new EventEmitter<Error>();\n\n  results: DiscoveredDeviceType[] = [];\n\n  private readonly destroyed = new Subject<void>();\n  isScanning = false;\n\n  constructor(\n    @Inject(TAP_BLE_SCANNER)\n    public tapScanner: DeviceScanner<DiscoveredDeviceType>,\n    public changeDetector: ChangeDetectorRef,\n    public loadingCtrl: LoadingController\n  ) {}\n\n  ngOnInit() {\n    this.tapScanner.results\n      .pipe(takeUntil(this.destroyed))\n      .subscribe((values) => {\n        this.mergeScanResults(values);\n      });\n\n    this.tapScanner.scanning.pipe(takeUntil(this.destroyed)).subscribe((v) => {\n      if (v !== this.isScanning) {\n        this.isScanning = v;\n      }\n    });\n  }\n\n  notifyError(error: any) {\n    console.error('BLEScannerError', error);\n    this.onError.emit(error);\n  }\n\n  ngOnDestroy() {\n    this.stopScan();\n    this.destroyed.next();\n    this.destroyed.complete();\n  }\n\n  onItemClicked(device: DiscoveredDeviceType) {\n    const protocolMeta: ProtocolMetaBle = {\n      type: 'ble',\n      info: {\n        id: device.address,\n        mac: device.address,\n        name: device.name,\n      },\n    };\n    const event = {\n      meta: protocolMeta,\n    };\n    this.protocolSelected.emit(event);\n    this.onProtocolSelected.emit(event);\n  }\n\n  startScan() {\n    // this.devices = this.devices.filter(device => device.address == this.tapScanner.selectedDevice);\n    // this.devices.splice(0, this.devices.length);\n    this.tapScanner.start().catch((err) => {\n      this.notifyError(err);\n    });\n    // this.onScanStateChanged.emit(true);\n  }\n\n  stopScan() {\n    this.tapScanner.stop().catch((err) => {\n      this.notifyError(err);\n    });\n    // this.onScanStateChanged.emit(false);\n  }\n\n  private mergeScanResults(newResults: DiscoveredDeviceType[]) {\n    /*\n    newResults.forEach((newResult) => {\n      this.addOrRefreshScanResultList(newResult);\n    });\n    const indexToRemove = this.results.map((oldDevice) => {\n      return newResults.findIndex(\n        (entry) => entry.address === oldDevice.address\n      );\n    }, []);\n    indexToRemove.forEach((resultIndex) => {\n      if (resultIndex < 0) {\n        this.results.splice(resultIndex, 1);\n      }\n    });\n    */\n    this.results = newResults;\n    this.changeDetector.detectChanges();\n  }\n\n  private addOrRefreshScanResultList(newDevice: DiscoveredDeviceType) {\n    const devices = this.results;\n    let hasChanged = false;\n    const storedDeviceIndex = devices.findIndex(\n      (entry) => entry.address === newDevice.address\n    );\n    if (storedDeviceIndex >= 0) {\n      const storedDevice = devices[storedDeviceIndex];\n      if (storedDevice.name !== newDevice.name) {\n        storedDevice.name = newDevice.name;\n        hasChanged = true;\n      }\n      if (\n        Math.abs(storedDevice.rssi - newDevice.rssi) >= this.rssiUpdateThreshold\n      ) {\n        storedDevice.rssi = newDevice.rssi;\n        hasChanged = true;\n      }\n    } else {\n      devices.push(newDevice);\n      hasChanged = true;\n    }\n    return hasChanged;\n  }\n}\n","<ng-container\n  *ngTemplateOutlet=\"scanResultsContainer; context: { $implicit: results }\"\n></ng-container>\n\n<ng-template #scanResultsContainer let-devices>\n  <!-- <ng-content #scanResultsInner select=\"[scan-results]\"></ng-content> -->\n  <ng-template\n    *ngTemplateOutlet=\"scanResultsTemplate; context: { $implicit: results }\"\n  ></ng-template>\n  <tap-ble-scan-result\n    *ngIf=\"!scanResultsTemplate\"\n    [devices]=\"devices\"\n    [showMacAddress]=\"showMacAddress\"\n    [showSignalStrength]=\"showSignalStrength\"\n    (onItemClicked)=\"onItemClicked($event)\"\n  ></tap-ble-scan-result>\n</ng-template>\n"]}