Compare commits
35 Commits
@ -1,22 +1,39 @@
|
|||||||
language: node_js
|
language: android
|
||||||
node_js:
|
|
||||||
- "8"
|
|
||||||
|
|
||||||
branches:
|
android:
|
||||||
only:
|
components:
|
||||||
- master
|
- tools
|
||||||
|
- platform-tools
|
||||||
|
- build-tools-27.0.3
|
||||||
|
- android-27
|
||||||
|
|
||||||
before_script:
|
before_install:
|
||||||
|
- yes | sdkmanager "platforms;android-27" # accept android licenses
|
||||||
|
|
||||||
|
install:
|
||||||
|
- nvm install 12
|
||||||
|
- npm install -g ionic cordova
|
||||||
- npm install
|
- npm install
|
||||||
|
|
||||||
script:
|
script:
|
||||||
- npm run build
|
- npm run android:build # implies web build in www/ dir
|
||||||
|
|
||||||
|
before_deploy: "cp platforms/android/app/build/outputs/apk/release/app-release-unsigned.apk sensebox_blockly_${TRAVIS_TAG}.apk"
|
||||||
|
|
||||||
deploy:
|
deploy:
|
||||||
provider: pages
|
# deploy web build to gh-pages branch
|
||||||
skip-cleanup: true
|
- provider: pages
|
||||||
github-token: $GITHUB_TOKEN
|
skip-cleanup: true
|
||||||
keep-history: true
|
github-token: $GITHUB_TOKEN
|
||||||
on:
|
keep-history: true
|
||||||
tags: true
|
on:
|
||||||
local_dir: www
|
tags: true
|
||||||
|
local_dir: www
|
||||||
|
|
||||||
|
- provider: releases
|
||||||
|
api_key: $GITHUB_TOKEN
|
||||||
|
file:
|
||||||
|
- "sensebox_blockly_${TRAVIS_TAG}.apk"
|
||||||
|
skip_cleanup: true
|
||||||
|
on:
|
||||||
|
tags: true
|
||||||
|
Before Width: | Height: | Size: 5.0 KiB After Width: | Height: | Size: 2.8 KiB |
Before Width: | Height: | Size: 2.3 KiB After Width: | Height: | Size: 1.5 KiB |
Before Width: | Height: | Size: 3.2 KiB After Width: | Height: | Size: 2.0 KiB |
Before Width: | Height: | Size: 6.9 KiB After Width: | Height: | Size: 3.8 KiB |
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 5.5 KiB |
Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 7.4 KiB |
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 17 KiB |
Before Width: | Height: | Size: 6.9 KiB After Width: | Height: | Size: 6.1 KiB |
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 9.6 KiB |
Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 30 KiB |
Before Width: | Height: | Size: 73 KiB After Width: | Height: | Size: 44 KiB |
Before Width: | Height: | Size: 89 KiB After Width: | Height: | Size: 58 KiB |
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 16 KiB |
Before Width: | Height: | Size: 6.8 KiB After Width: | Height: | Size: 6.1 KiB |
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 9.5 KiB |
Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 30 KiB |
Before Width: | Height: | Size: 72 KiB After Width: | Height: | Size: 42 KiB |
Before Width: | Height: | Size: 88 KiB After Width: | Height: | Size: 57 KiB |
Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 40 KiB |
@ -1 +0,0 @@
|
|||||||
41e1c536c5dfb47db1311f7335b357b3
|
|
Before Width: | Height: | Size: 59 KiB |
Before Width: | Height: | Size: 59 KiB After Width: | Height: | Size: 44 KiB |
Before Width: | Height: | Size: 1.2 KiB After Width: | Height: | Size: 1.7 KiB |
Before Width: | Height: | Size: 3.5 KiB After Width: | Height: | Size: 3.2 KiB |
Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 4.7 KiB |
Before Width: | Height: | Size: 1.9 KiB After Width: | Height: | Size: 2.1 KiB |
Before Width: | Height: | Size: 4.6 KiB After Width: | Height: | Size: 4.0 KiB |
Before Width: | Height: | Size: 2.4 KiB After Width: | Height: | Size: 2.5 KiB |
Before Width: | Height: | Size: 5.9 KiB After Width: | Height: | Size: 4.7 KiB |
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 7.2 KiB |
Before Width: | Height: | Size: 2.8 KiB After Width: | Height: | Size: 2.8 KiB |
Before Width: | Height: | Size: 7.6 KiB After Width: | Height: | Size: 5.5 KiB |
Before Width: | Height: | Size: 3.2 KiB After Width: | Height: | Size: 3.1 KiB |
Before Width: | Height: | Size: 8.1 KiB After Width: | Height: | Size: 5.9 KiB |
Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 6.6 KiB |
Before Width: | Height: | Size: 818 B After Width: | Height: | Size: 1.6 KiB |
Before Width: | Height: | Size: 2.2 KiB After Width: | Height: | Size: 2.3 KiB |
Before Width: | Height: | Size: 3.8 KiB After Width: | Height: | Size: 3.4 KiB |
Before Width: | Height: | Size: 2.1 KiB After Width: | Height: | Size: 2.4 KiB |
Before Width: | Height: | Size: 5.6 KiB After Width: | Height: | Size: 4.4 KiB |
Before Width: | Height: | Size: 31 KiB After Width: | Height: | Size: 26 KiB |
Before Width: | Height: | Size: 40 KiB After Width: | Height: | Size: 32 KiB |
Before Width: | Height: | Size: 44 KiB After Width: | Height: | Size: 65 KiB |
Before Width: | Height: | Size: 44 KiB After Width: | Height: | Size: 70 KiB |
Before Width: | Height: | Size: 100 KiB After Width: | Height: | Size: 58 KiB |
Before Width: | Height: | Size: 158 KiB After Width: | Height: | Size: 71 KiB |
Before Width: | Height: | Size: 22 KiB After Width: | Height: | Size: 23 KiB |
Before Width: | Height: | Size: 97 KiB After Width: | Height: | Size: 57 KiB |
Before Width: | Height: | Size: 153 KiB After Width: | Height: | Size: 70 KiB |
Before Width: | Height: | Size: 22 KiB After Width: | Height: | Size: 23 KiB |
Before Width: | Height: | Size: 18 KiB After Width: | Height: | Size: 21 KiB |
Before Width: | Height: | Size: 77 KiB After Width: | Height: | Size: 78 KiB |
Before Width: | Height: | Size: 7.0 KiB After Width: | Height: | Size: 9.5 KiB |
Before Width: | Height: | Size: 92 KiB After Width: | Height: | Size: 76 KiB |
@ -1 +0,0 @@
|
|||||||
287a089b0cd0ff44dfda2fd7bd7ad812
|
|
Before Width: | Height: | Size: 77 KiB |
@ -1 +1 @@
|
|||||||
Subproject commit c43a8244e72996457ce832317bb463787645caae
|
Subproject commit cc464a6466c2e6b2648e93e386d574d526cd3f40
|
@ -0,0 +1,96 @@
|
|||||||
|
import { ElementRef } from '@angular/core';
|
||||||
|
import { LoggingProvider } from '../../providers/logging/logging';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* this file defines & implements the message passing protocol to communicate
|
||||||
|
* with an iframe that has src/assets/blockly.html loaded.
|
||||||
|
* The underlying postMessage protocol is wrapped into a promise API.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export class BlocklyMessageProtocol {
|
||||||
|
// resolve ready promise once the blocklyFrame is ready
|
||||||
|
ready = new Promise(resolve => {
|
||||||
|
window.addEventListener('message', (ev: IframePostMessageEvent) => {
|
||||||
|
// @HACK @FIXME: timeout is required, as blockly resolves some async functions
|
||||||
|
// after firing the `ready` event..
|
||||||
|
if (ev.data.type === 'ready') setTimeout(resolve, 300)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
static reqResPatterns: BlocklyReqPatterns = {
|
||||||
|
'getSketch': 'sketch',
|
||||||
|
'getXml': 'xml',
|
||||||
|
'setXml': null,
|
||||||
|
'toggleView': null,
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor (private blocklyFrame: ElementRef, private log: LoggingProvider) {
|
||||||
|
// set up event listeners for non-request log messages
|
||||||
|
window.addEventListener('message', (ev: IframePostMessageEvent) => {
|
||||||
|
if (ev.data.type === 'log')
|
||||||
|
this.log.warn('log entry from blockly:', ev.data)
|
||||||
|
else
|
||||||
|
this.log.debug(`received ${ev.data.type} message from blockly`, { message: ev.data })
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
toggleView() { this.sendRequest({ type: 'toggleView' }) }
|
||||||
|
setXml(data: string) { this.sendRequest({ type: 'setXml', data }) }
|
||||||
|
getXml() { return this.sendRequest({ type: 'getXml' }) }
|
||||||
|
getSketch() { return this.sendRequest({ type: 'getSketch' }) }
|
||||||
|
|
||||||
|
private async sendRequest(req: BlocklyRequest): Promise<any> {
|
||||||
|
await this.ready
|
||||||
|
|
||||||
|
if (
|
||||||
|
!this.blocklyFrame ||
|
||||||
|
!this.blocklyFrame.nativeElement ||
|
||||||
|
!this.blocklyFrame.nativeElement.contentWindow
|
||||||
|
) {
|
||||||
|
throw new Error('cannot access blockly frame')
|
||||||
|
}
|
||||||
|
|
||||||
|
const expectResponse = BlocklyMessageProtocol.reqResPatterns[req.type]
|
||||||
|
this.log.debug(`sending ${req.type} message to blockly, expecting response: ${expectResponse}`, { message: req })
|
||||||
|
|
||||||
|
if (!expectResponse)
|
||||||
|
return this.blocklyFrame.nativeElement.contentWindow.postMessage(req, '*')
|
||||||
|
|
||||||
|
// create promise waiting for the response event
|
||||||
|
const resPromise = new Promise<any>((resolve, reject) => {
|
||||||
|
const resHandler = ({ data: res }: IframePostMessageEvent) => {
|
||||||
|
if (expectResponse !== res.type) return
|
||||||
|
window.removeEventListener('message', resHandler)
|
||||||
|
if (res.type === 'error') reject(res.data)
|
||||||
|
else resolve(res.data)
|
||||||
|
}
|
||||||
|
// TODO: promise reject after timeout?
|
||||||
|
window.addEventListener('message', resHandler)
|
||||||
|
})
|
||||||
|
|
||||||
|
// send message *after* registering the response handler!
|
||||||
|
this.blocklyFrame.nativeElement.contentWindow.postMessage(req, '*')
|
||||||
|
return resPromise
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
interface IframePostMessageEvent extends MessageEvent {
|
||||||
|
data: BlocklyResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
type BlocklyReqPatterns = {
|
||||||
|
[k in BlocklyRequest['type']]: BlocklyResponse['type']
|
||||||
|
}
|
||||||
|
|
||||||
|
type BlocklyRequest =
|
||||||
|
{ type: 'getSketch' } |
|
||||||
|
{ type: 'getXml' } |
|
||||||
|
{ type: 'setXml', data: string } |
|
||||||
|
{ type: 'toggleView' }
|
||||||
|
|
||||||
|
type BlocklyResponse =
|
||||||
|
{ type: 'log', data: any } |
|
||||||
|
{ type: 'error', data: any } |
|
||||||
|
{ type: 'ready', data: undefined } |
|
||||||
|
{ type: 'sketch', data: string } |
|
||||||
|
{ type: 'xml', data: string }
|
@ -1,3 +1,3 @@
|
|||||||
page-settings {
|
.error {
|
||||||
|
color: red !important;
|
||||||
}
|
}
|
||||||
|
@ -1,33 +1,41 @@
|
|||||||
import { Injectable } from '@angular/core';
|
import { Injectable } from '@angular/core';
|
||||||
|
|
||||||
export const SETTINGS = 'appsettings'
|
export const SETTINGS = 'appsettings'
|
||||||
|
export const LASTSKETCH = 'lastsketch'
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class StorageProvider {
|
export class StorageProvider {
|
||||||
private cache: Map<string, any> = new Map()
|
private cache: Map<string, any> = new Map()
|
||||||
|
|
||||||
constructor () {
|
constructor () {
|
||||||
this.registerKey(SETTINGS, {
|
this.init()
|
||||||
logOptin: false,
|
}
|
||||||
})
|
|
||||||
|
init () {
|
||||||
|
// set up default values
|
||||||
|
this.registerKey(SETTINGS, { logOptin: false })
|
||||||
|
this.registerKey(LASTSKETCH, '')
|
||||||
}
|
}
|
||||||
|
|
||||||
registerKey (key, defaultValue) {
|
registerKey (key, defaultValue) {
|
||||||
const stored = localStorage.getItem(key)
|
const stored = localStorage.getItem(key)
|
||||||
if (!stored) {
|
if (stored === null) {
|
||||||
localStorage.setItem(key, JSON.stringify(defaultValue))
|
this.set(key, defaultValue)
|
||||||
this.cache[key] = defaultValue
|
|
||||||
} else {
|
} else {
|
||||||
this.cache[key] = JSON.parse(stored)
|
this.cache[key] = JSON.parse(stored)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
get (key) {
|
get (key) { return this.cache[key] }
|
||||||
return this.cache[key]
|
|
||||||
}
|
|
||||||
|
|
||||||
set (key, value) {
|
set (key, value) {
|
||||||
localStorage.setItem(key, JSON.stringify(value))
|
localStorage.setItem(key, JSON.stringify(value))
|
||||||
this.cache[key] = value
|
this.cache[key] = value
|
||||||
}
|
}
|
||||||
|
|
||||||
|
reset () {
|
||||||
|
localStorage.clear()
|
||||||
|
this.cache = new Map()
|
||||||
|
this.init()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|