forked from btclock/btclock_v3
Compare commits
1 commit
Author | SHA1 | Date | |
---|---|---|---|
|
9bbe672e3f |
77 changed files with 2027 additions and 9131 deletions
.forgejo/workflows
.github
.gitignore.gitmodulesCMakeLists.txtREADME.mdboards
ci
datadependencies.locklib/btclock
bitaxe_handler.cppbitaxe_handler.hppdata_handler.cppdata_handler.hppnostrdisplay_handler.cppnostrdisplay_handler.hpputils.cpputils.hpp
maintainers.yamlpartition.csvpartition_16mb.csvpartition_8mb.csvplatformio.inirequirements.txtscripts
sdkconfig.defaultssrc
fonts
icons
idf_component.ymllib
bitaxe_fetch.cppbitaxe_fetch.hppblock_notify.cppblock_notify.hppbutton_handler.cppbutton_handler.hppconfig.cppconfig.hppdefaults.hppepd.cppepd.hppimprov.cppimprov.hppled_handler.cppled_handler.hppnostr_notify.cppnostr_notify.hppnostr_subscribe.cppnostr_subscribe.hppota.cppota.hppprice_fetch.cppprice_fetch.hppprice_notify.cppprice_notify.hppscreen_handler.cppscreen_handler.hppshared.cppshared.hpptimers.cpptimers.hppv2_notify.cppv2_notify.hppwebserver.cppwebserver.hpp
main.cpptest/test_datahandler
x509_crt_bundle
|
@ -1,190 +0,0 @@
|
||||||
name: "BTClock CI"
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- "*"
|
|
||||||
workflow_dispatch:
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: docker
|
|
||||||
container:
|
|
||||||
image: ghcr.io/catthehacker/ubuntu:js-22.04
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
checks: write
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
submodules: recursive
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: lts/*
|
|
||||||
cache: yarn
|
|
||||||
cache-dependency-path: "**/yarn.lock"
|
|
||||||
- uses: actions/cache@v4
|
|
||||||
with:
|
|
||||||
path: |
|
|
||||||
~/.cache/pip
|
|
||||||
~/.platformio/.cache
|
|
||||||
~/data/node_modules
|
|
||||||
.pio
|
|
||||||
data/node_modules
|
|
||||||
key: ${{ runner.os }}-pio
|
|
||||||
- uses: actions/setup-python@v5
|
|
||||||
with:
|
|
||||||
python-version: "3.9"
|
|
||||||
cache: "pip"
|
|
||||||
- name: Get current date
|
|
||||||
id: dateAndTime
|
|
||||||
shell: bash
|
|
||||||
run: echo "dateAndTime=$(date +'%Y-%m-%d-%H:%M')" >> $GITHUB_OUTPUT
|
|
||||||
- name: Install PlatformIO Core
|
|
||||||
shell: bash
|
|
||||||
run: pip install --upgrade platformio
|
|
||||||
- name: Run unit tests
|
|
||||||
shell: bash
|
|
||||||
run: mkdir -p junit-reports && pio test -e native_test_only --junit-output-path junit-reports/
|
|
||||||
- name: Build BTClock firmware
|
|
||||||
shell: bash
|
|
||||||
run: pio run
|
|
||||||
- name: Build BTClock filesystem
|
|
||||||
shell: bash
|
|
||||||
run: pio run --target buildfs
|
|
||||||
- name: Copy bootloader to output folder
|
|
||||||
run: cp ~/.platformio/packages/framework-arduinoespressif32/tools/partitions/boot_app0.bin .pio
|
|
||||||
- name: Upload artifacts
|
|
||||||
uses: https://code.forgejo.org/forgejo/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
include-hidden-files: true
|
|
||||||
retention-days: 1
|
|
||||||
name: prepared-outputs
|
|
||||||
path: .pio/**/*.bin
|
|
||||||
merge:
|
|
||||||
runs-on: docker
|
|
||||||
container:
|
|
||||||
image: ghcr.io/catthehacker/ubuntu:js-22.04
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
checks: write
|
|
||||||
needs: build
|
|
||||||
continue-on-error: true
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
chip:
|
|
||||||
- name: lolin_s3_mini
|
|
||||||
version: esp32s3
|
|
||||||
- name: btclock_rev_b
|
|
||||||
version: esp32s3
|
|
||||||
- name: btclock_v8
|
|
||||||
version: esp32s3
|
|
||||||
epd_variant: [213epd, 29epd]
|
|
||||||
exclude:
|
|
||||||
- chip: { name: btclock_rev_b, version: esp32s3 }
|
|
||||||
epd_variant: 29epd
|
|
||||||
- chip: { name: btclock_v8, version: esp32s3 }
|
|
||||||
epd_variant: 29epd
|
|
||||||
steps:
|
|
||||||
- uses: https://code.forgejo.org/forgejo/download-artifact@v4
|
|
||||||
with:
|
|
||||||
name: prepared-outputs
|
|
||||||
path: .pio
|
|
||||||
- name: Install esptools.py
|
|
||||||
run: pip install --upgrade esptool
|
|
||||||
- name: Create merged firmware binary
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
mkdir -p ${{ matrix.chip.name }}_${{ matrix.epd_variant }}
|
|
||||||
if [ "${{ matrix.chip.name }}" == "btclock_v8" ]; then
|
|
||||||
esptool.py --chip ${{ matrix.chip.version }} merge_bin \
|
|
||||||
-o ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin \
|
|
||||||
--flash_mode dio \
|
|
||||||
--flash_freq 80m \
|
|
||||||
--flash_size 16MB \
|
|
||||||
0x0000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/bootloader.bin \
|
|
||||||
0x8000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/partitions.bin \
|
|
||||||
0xe000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/ota_data_initial.bin \
|
|
||||||
0x10000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin \
|
|
||||||
0xDF0000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/littlefs_16MB.bin
|
|
||||||
elif [ "${{ matrix.chip.name }}" == "btclock_rev_b" ]; then
|
|
||||||
esptool.py --chip ${{ matrix.chip.version }} merge_bin \
|
|
||||||
-o ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin \
|
|
||||||
--flash_mode dio \
|
|
||||||
--flash_freq 80m \
|
|
||||||
--flash_size 8MB \
|
|
||||||
0x0000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/bootloader.bin \
|
|
||||||
0x8000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/partitions.bin \
|
|
||||||
0xe000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/ota_data_initial.bin \
|
|
||||||
0x10000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin \
|
|
||||||
0x6F0000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/littlefs_8MB.bin;
|
|
||||||
else
|
|
||||||
esptool.py --chip ${{ matrix.chip.version }} merge_bin \
|
|
||||||
-o ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin \
|
|
||||||
--flash_mode dio \
|
|
||||||
0x0000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/bootloader.bin \
|
|
||||||
0x8000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/partitions.bin \
|
|
||||||
0xe000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/ota_data_initial.bin \
|
|
||||||
0x10000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin \
|
|
||||||
0x380000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/littlefs_4MB.bin
|
|
||||||
# Adjust the offset for littlefs or other files as needed for the original case
|
|
||||||
fi
|
|
||||||
|
|
||||||
- name: Create checksum for firmware
|
|
||||||
shell: bash
|
|
||||||
run: shasum -a 256 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin | awk '{print $1}' > ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}_firmware.bin.sha256
|
|
||||||
|
|
||||||
- name: Create checksum for merged binary
|
|
||||||
shell: bash
|
|
||||||
run: shasum -a 256 ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin | awk '{print $1}' > ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin.sha256
|
|
||||||
|
|
||||||
- name: Create checksum for littlefs partition
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
fs_file=$(find .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }} -name "littlefs*.bin")
|
|
||||||
echo $fs_file
|
|
||||||
fs_name=$(basename "$fs_file")
|
|
||||||
shasum -a 256 "$fs_file" | awk '{print $1}' > "${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${fs_name}.sha256"
|
|
||||||
cat "${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${fs_name}.sha256"
|
|
||||||
- name: Copy all artifacts to output folder
|
|
||||||
run: cp .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/*.bin .pio/boot_app0.bin ${{ matrix.chip.name }}_${{ matrix.epd_variant }}
|
|
||||||
|
|
||||||
- name: Create OTA binary file
|
|
||||||
run: mv ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}_firmware.bin
|
|
||||||
- name: Upload artifacts
|
|
||||||
uses: https://code.forgejo.org/forgejo/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: build-${{ matrix.chip.name }}-${{ matrix.epd_variant }}
|
|
||||||
path: |
|
|
||||||
${{ matrix.chip.name }}_${{ matrix.epd_variant }}/*.bin
|
|
||||||
${{ matrix.chip.name }}_${{ matrix.epd_variant }}/*.sha256
|
|
||||||
release:
|
|
||||||
runs-on: docker
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
checks: write
|
|
||||||
needs: merge
|
|
||||||
steps:
|
|
||||||
- name: Download matrix outputs
|
|
||||||
uses: https://code.forgejo.org/forgejo/download-artifact@v4
|
|
||||||
with:
|
|
||||||
pattern: build-*
|
|
||||||
merge-multiple: false
|
|
||||||
path: temp
|
|
||||||
- name: Copy files
|
|
||||||
run: |
|
|
||||||
mkdir -p release
|
|
||||||
find temp -type f \( -name "*.bin" -o -name "*.sha256" \) -exec cp -f {} release/ \;
|
|
||||||
- name: Create release
|
|
||||||
uses: https://code.forgejo.org/actions/forgejo-release@v2.4.0
|
|
||||||
with:
|
|
||||||
url: "https://git.btclock.dev"
|
|
||||||
repo: "${{ github.repository }}"
|
|
||||||
direction: upload
|
|
||||||
tag: "${{ github.ref_name }}"
|
|
||||||
sha: "${{ github.sha }}"
|
|
||||||
release-dir: release
|
|
||||||
token: ${{ secrets.TOKEN }}
|
|
||||||
override: ${{ github.ref_type != 'tag' && github.ref_name != 'main' }}
|
|
||||||
prerelease: ${{ github.ref_type != 'tag' && github.ref_name != 'main' }}
|
|
||||||
release-notes-assistant: false
|
|
8
.github/actions/install-build/action.yml
vendored
8
.github/actions/install-build/action.yml
vendored
|
@ -9,14 +9,14 @@ runs:
|
||||||
node-version: lts/*
|
node-version: lts/*
|
||||||
cache: yarn
|
cache: yarn
|
||||||
cache-dependency-path: '**/yarn.lock'
|
cache-dependency-path: '**/yarn.lock'
|
||||||
- uses: actions/cache@v4
|
- uses: actions/cache@v3
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
~/.cache/pip
|
~/.cache/pip
|
||||||
~/.platformio/.cache
|
~/.platformio/.cache
|
||||||
~/data/node_modules
|
~/data/node_modules
|
||||||
key: ${{ runner.os }}-pio
|
key: ${{ runner.os }}-pio
|
||||||
- uses: actions/setup-python@v5
|
- uses: actions/setup-python@v4
|
||||||
with:
|
with:
|
||||||
python-version: '3.9'
|
python-version: '3.9'
|
||||||
- name: Get current date
|
- name: Get current date
|
||||||
|
@ -37,7 +37,7 @@ runs:
|
||||||
detailed_summary: true
|
detailed_summary: true
|
||||||
- name: Build BTClock firmware
|
- name: Build BTClock firmware
|
||||||
shell: bash
|
shell: bash
|
||||||
run: pio run
|
run: pio run -e lolin_s3_mini_qr
|
||||||
- name: Build BTClock filesystem
|
- name: Build BTClock filesystem
|
||||||
shell: bash
|
shell: bash
|
||||||
run: pio run --target buildfs
|
run: pio run -e lolin_s3_mini_qr --target buildfs
|
133
.github/workflows/tagging.yml
vendored
133
.github/workflows/tagging.yml
vendored
|
@ -3,7 +3,7 @@ name: BTClock CI
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
tags:
|
tags:
|
||||||
- "*"
|
- '*'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
|
@ -17,138 +17,49 @@ jobs:
|
||||||
submodules: recursive
|
submodules: recursive
|
||||||
- name: "Install and build"
|
- name: "Install and build"
|
||||||
uses: ./.github/actions/install-build
|
uses: ./.github/actions/install-build
|
||||||
- name: Copy bootloader to output folder
|
|
||||||
run: cp ~/.platformio/packages/framework-arduinoespressif32/tools/partitions/boot_app0.bin .pio
|
|
||||||
- name: Upload artifacts
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
include-hidden-files: true
|
|
||||||
retention-days: 1
|
|
||||||
name: prepared-outputs
|
|
||||||
path: .pio/**/*.bin
|
|
||||||
merge:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
checks: write
|
|
||||||
needs: build
|
|
||||||
continue-on-error: true
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
chip:
|
|
||||||
- name: lolin_s3_mini
|
|
||||||
version: esp32s3
|
|
||||||
- name: btclock_rev_b
|
|
||||||
version: esp32s3
|
|
||||||
- name: btclock_v8
|
|
||||||
version: esp32s3
|
|
||||||
epd_variant: [213epd, 29epd]
|
|
||||||
exclude:
|
|
||||||
- chip: {name: btclock_rev_b, version: esp32s3}
|
|
||||||
epd_variant: 29epd
|
|
||||||
- chip: {name: btclock_v8, version: esp32s3}
|
|
||||||
epd_variant: 29epd
|
|
||||||
steps:
|
|
||||||
- uses: actions/download-artifact@v4
|
|
||||||
with:
|
|
||||||
name: prepared-outputs
|
|
||||||
path: .pio
|
|
||||||
- name: Install esptools.py
|
- name: Install esptools.py
|
||||||
run: pip install --upgrade esptool
|
run: pip install --upgrade esptool
|
||||||
# - name: Create merged firmware binary
|
|
||||||
# run: mkdir -p ${{ matrix.chip.name }}_${{ matrix.epd_variant }} && esptool.py --chip ${{ matrix.chip.version }} merge_bin -o ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin --flash_mode dio 0x0000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/bootloader.bin 0x8000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/partitions.bin 0xe000 .pio/boot_app0.bin 0x10000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin 0x369000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/littlefs.bin
|
|
||||||
- name: Create merged firmware binary
|
|
||||||
run: |
|
|
||||||
if [ "${{ matrix.chip.name }}" == "btclock_v8" ]; then
|
|
||||||
mkdir -p ${{ matrix.chip.name }}_${{ matrix.epd_variant }} && \
|
|
||||||
esptool.py --chip ${{ matrix.chip.version }} merge_bin \
|
|
||||||
-o ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin \
|
|
||||||
--flash_mode dio \
|
|
||||||
--flash_freq 80m \
|
|
||||||
--flash_size 16MB \
|
|
||||||
0x0000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/bootloader.bin \
|
|
||||||
0x8000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/partitions.bin \
|
|
||||||
0xe000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/ota_data_initial.bin \
|
|
||||||
0x10000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin \
|
|
||||||
0x810000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/littlefs.bin;
|
|
||||||
else
|
|
||||||
# Original command for other cases
|
|
||||||
mkdir -p ${{ matrix.chip.name }}_${{ matrix.epd_variant }} && \
|
|
||||||
esptool.py --chip ${{ matrix.chip.version }} merge_bin \
|
|
||||||
-o ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin \
|
|
||||||
--flash_mode dio \
|
|
||||||
0x0000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/bootloader.bin \
|
|
||||||
0x8000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/partitions.bin \
|
|
||||||
0xe000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/ota_data_initial.bin \
|
|
||||||
0x10000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin \
|
|
||||||
0x369000 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/littlefs.bin
|
|
||||||
# Adjust the offset for littlefs or other files as needed for the original case
|
|
||||||
fi
|
|
||||||
|
|
||||||
- name: Create checksum for firmware
|
- name: Create merged firmware binary
|
||||||
run: shasum -a 256 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin | awk '{print $1}' > ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}_firmware.bin.sha256
|
run: mkdir -p output && esptool.py --chip esp32s3 merge_bin -o output/full-firmware.bin --flash_mode dio 0x0000 .pio/build/lolin_s3_mini_qr/bootloader.bin 0x8000 .pio/build/lolin_s3_mini_qr/partitions.bin 0xe000 ~/.platformio/packages/framework-arduinoespressif32/tools/partitions/boot_app0.bin 0x10000 .pio/build/lolin_s3_mini_qr/firmware.bin 0x369000 .pio/build/lolin_s3_mini_qr/littlefs.bin
|
||||||
|
|
||||||
- name: Create checksum for merged binary
|
- name: Create checksum for merged binary
|
||||||
run: shasum -a 256 ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin | awk '{print $1}' > ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}.bin.sha256
|
run: shasum -a 256 output/full-firmware.bin | awk '{print $1}' > output/full-firmware.sha256
|
||||||
|
|
||||||
- name: Create checksum for littlefs partition
|
|
||||||
run: shasum -a 256 .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/littlefs.bin | awk '{print $1}' > ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/littlefs.bin.sha256
|
|
||||||
|
|
||||||
- name: Copy all artifacts to output folder
|
|
||||||
run: cp .pio/build/${{ matrix.chip.name }}_${{ matrix.epd_variant }}/*.bin .pio/boot_app0.bin ${{ matrix.chip.name }}_${{ matrix.epd_variant }}
|
|
||||||
|
|
||||||
- name: Create OTA binary file
|
|
||||||
run: mv ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/firmware.bin ${{ matrix.chip.name }}_${{ matrix.epd_variant }}/${{ matrix.chip.name }}_${{ matrix.epd_variant }}_firmware.bin
|
|
||||||
|
|
||||||
- name: Upload artifacts
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: build-${{ matrix.chip.name }}-${{ matrix.epd_variant }}
|
|
||||||
path: |
|
|
||||||
${{ matrix.chip.name }}_${{ matrix.epd_variant }}/*.bin
|
|
||||||
${{ matrix.chip.name }}_${{ matrix.epd_variant }}/*.sha256
|
|
||||||
release:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
checks: write
|
|
||||||
needs: merge
|
|
||||||
steps:
|
|
||||||
- name: Download matrix outputs
|
|
||||||
uses: actions/download-artifact@v4
|
|
||||||
with:
|
|
||||||
pattern: build-*
|
|
||||||
merge-multiple: false
|
|
||||||
- name: Write commit hash to file
|
- name: Write commit hash to file
|
||||||
run: echo $GITHUB_SHA > commit.txt
|
run: echo $GITHUB_SHA > output/commit.txt
|
||||||
|
|
||||||
- name: Write build date to file
|
- name: Write build date to file
|
||||||
run: echo "$(date -u +'%Y-%m-%dT%H:%M:%SZ')" > date.txt
|
run: echo "$(date -u +'%Y-%m-%dT%H:%M:%SZ')" > output/date.txt
|
||||||
|
|
||||||
|
- name: Copy all artifacts to output folder
|
||||||
|
run: cp .pio/build/lolin_s3_mini_qr/*.bin ~/.platformio/packages/framework-arduinoespressif32/tools/partitions/boot_app0.bin output
|
||||||
|
|
||||||
|
- name: Upload artifacts
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
.pio/build/lolin_s3_mini_qr/*.bin
|
||||||
|
output/full-firmware.bin
|
||||||
|
output/full-firmware.sha256
|
||||||
- name: Create release
|
- name: Create release
|
||||||
uses: ncipollo/release-action@v1
|
uses: ncipollo/release-action@v1
|
||||||
with:
|
with:
|
||||||
artifacts: "**/*.bin,**/*.sha256"
|
artifacts: "output/full-firmware.bin,output/full-firmware.sha256,.pio/build/lolin_s3_mini_qr/*.bin"
|
||||||
allowUpdates: true
|
allowUpdates: true
|
||||||
removeArtifacts: true
|
removeArtifacts: true
|
||||||
makeLatest: true
|
makeLatest: true
|
||||||
# - name: Create release
|
|
||||||
# uses: ncipollo/release-action@v1
|
|
||||||
# with:
|
|
||||||
# artifacts: "output/full-firmware.bin,output/full-firmware.sha256,.pio/build/lolin_s3_mini_qr/*.bin"
|
|
||||||
# allowUpdates: true
|
|
||||||
# removeArtifacts: true
|
|
||||||
# makeLatest: true
|
|
||||||
- name: Pushes full-firmware.bin to web flasher
|
- name: Pushes full-firmware.bin to web flasher
|
||||||
id: push_directory
|
id: push_directory
|
||||||
uses: cpina/github-action-push-to-another-repository@main
|
uses: cpina/github-action-push-to-another-repository@main
|
||||||
env:
|
env:
|
||||||
SSH_DEPLOY_KEY: ${{ secrets.SSH_DEPLOY_KEY }}
|
SSH_DEPLOY_KEY: ${{ secrets.SSH_DEPLOY_KEY }}
|
||||||
with:
|
with:
|
||||||
source-directory: .
|
source-directory: output/
|
||||||
target-directory: firmware_v3/
|
target-directory: firmware_v3/
|
||||||
destination-github-username: "btclock"
|
destination-github-username: 'btclock'
|
||||||
destination-repository-name: "web-flasher"
|
destination-repository-name: 'web-flasher'
|
||||||
target-branch: main
|
target-branch: btclock
|
||||||
user-name: ${{github.actor}}
|
user-name: ${{github.actor}}
|
||||||
user-email: ${{github.actor}}@users.noreply.github.com
|
user-email: ${{github.actor}}@users.noreply.github.com
|
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -11,4 +11,3 @@ data/node_modules
|
||||||
node_modules
|
node_modules
|
||||||
.DS_Store
|
.DS_Store
|
||||||
*.bin
|
*.bin
|
||||||
ci/cache
|
|
2
.gitmodules
vendored
2
.gitmodules
vendored
|
@ -1,3 +1,3 @@
|
||||||
[submodule "data"]
|
[submodule "data"]
|
||||||
path = data
|
path = data
|
||||||
url = https://git.btclock.dev/btclock/webui.git
|
url = https://github.com/btclock/webui.git
|
||||||
|
|
|
@ -1,7 +1,4 @@
|
||||||
cmake_minimum_required(VERSION 3.16.0)
|
cmake_minimum_required(VERSION 3.16.0)
|
||||||
include($ENV{IDF_PATH}/tools/cmake/project.cmake)
|
include($ENV{IDF_PATH}/tools/cmake/project.cmake)
|
||||||
|
list(APPEND EXTRA_COMPONENT_DIRS managed_components)
|
||||||
get_filename_component(configName "${CMAKE_BINARY_DIR}" NAME)
|
|
||||||
list(APPEND EXTRA_COMPONENT_DIRS "${CMAKE_SOURCE_DIR}/.pio/libdeps/${configName}/esp_littlefs")
|
|
||||||
|
|
||||||
project(btclock_espidf)
|
project(btclock_espidf)
|
||||||
|
|
11
README.md
11
README.md
|
@ -1,8 +1,6 @@
|
||||||
# BTClock v3
|
# BTClock v3
|
||||||
|
|
||||||
[![Latest release](https://git.btclock.dev/btclock/btclock_v3/badges/release.svg)](https://git.btclock.dev/btclock/btclock_v3/releases/latest)
|
[![BTClock CI](https://github.com/btclock/btclock_v3/actions/workflows/tagging.yml/badge.svg)](https://github.com/btclock/btclock_v3/actions/workflows/tagging.yml)
|
||||||
|
|
||||||
[![BTClock CI](https://git.btclock.dev/btclock/btclock_v3/badges/workflows/push.yaml/badge.svg)](https://git.btclock.dev/btclock/btclock_v3/actions?workflow=push.yaml&actor=0&status=0)
|
|
||||||
|
|
||||||
Software for the BTClock project.
|
Software for the BTClock project.
|
||||||
|
|
||||||
|
@ -14,16 +12,11 @@ Biggest differences with v2 are:
|
||||||
- Added market capitalization screen
|
- Added market capitalization screen
|
||||||
- LED flash on new block (and focus to block height screen on new block)
|
- LED flash on new block (and focus to block height screen on new block)
|
||||||
|
|
||||||
New features:
|
|
||||||
- BitAxe integration
|
|
||||||
- Zap notifier
|
|
||||||
-
|
|
||||||
|
|
||||||
"Steal focus on new block" means that when a new block is mined, the display will switch to the block height screen if it's not on it already.
|
"Steal focus on new block" means that when a new block is mined, the display will switch to the block height screen if it's not on it already.
|
||||||
|
|
||||||
Most [information](https://github.com/btclock/btclock_v2/wiki) about BTClock v2 is still valid for this version.
|
Most [information](https://github.com/btclock/btclock_v2/wiki) about BTClock v2 is still valid for this version.
|
||||||
|
|
||||||
**NOTE**: The software assumes that the hardware is run in a controlled private network. ~~The Web UI and the OTA update mechanism are not password protected and accessible to anyone in the network. Also, since the device only fetches numbers through WebSockets it will skip server certificate verification to save resources.~~ Since 3.2.0 the WebUI is password protectable and all certificates are verified. OTA update mechanism is not password-protected.
|
**NOTE**: The software assumes that the hardware is run in a controlled private network. The Web UI and the OTA update mechanism are not password protected and accessible to anyone in the network. Also, since the device only fetches numbers through WebSockets it will skip server certificate verification to save resources.
|
||||||
|
|
||||||
## Building
|
## Building
|
||||||
|
|
||||||
|
|
|
@ -10,7 +10,7 @@
|
||||||
"-DBOARD_HAS_PSRAM",
|
"-DBOARD_HAS_PSRAM",
|
||||||
"-DARDUINO_BTCLOCK",
|
"-DARDUINO_BTCLOCK",
|
||||||
"-DARDUINO_ESP32S3_DEV",
|
"-DARDUINO_ESP32S3_DEV",
|
||||||
"-DIS_BTCLOCK_V8",
|
"-DIS_BTCLOCK_S3",
|
||||||
"-DARDUINO_USB_MODE=1",
|
"-DARDUINO_USB_MODE=1",
|
||||||
"-DARDUINO_RUNNING_CORE=1",
|
"-DARDUINO_RUNNING_CORE=1",
|
||||||
"-DARDUINO_EVENT_RUNNING_CORE=1",
|
"-DARDUINO_EVENT_RUNNING_CORE=1",
|
||||||
|
@ -20,8 +20,8 @@
|
||||||
"f_flash": "80000000L",
|
"f_flash": "80000000L",
|
||||||
"flash_mode": "qio",
|
"flash_mode": "qio",
|
||||||
"psram_type": "opi",
|
"psram_type": "opi",
|
||||||
"esp-idf": {
|
"espidf": {
|
||||||
"sdkconfig_path": "boards/sdkconfig.btclock_v8"
|
"sdkconfig_path": "boards"
|
||||||
},
|
},
|
||||||
"hwids": [
|
"hwids": [
|
||||||
[
|
[
|
|
@ -1,60 +0,0 @@
|
||||||
{
|
|
||||||
"build": {
|
|
||||||
"arduino":{
|
|
||||||
"ldscript": "esp32s3_out.ld",
|
|
||||||
"partitions": "default_8MB.csv"
|
|
||||||
},
|
|
||||||
"core": "esp32",
|
|
||||||
"extra_flags": [
|
|
||||||
"-DBOARD_HAS_PSRAM",
|
|
||||||
"-DARDUINO_BTCLOCK_REV_B",
|
|
||||||
"-DARDUINO_ESP32S3_DEV",
|
|
||||||
"-DIS_BTCLOCK_REV_B",
|
|
||||||
"-DARDUINO_USB_MODE=1",
|
|
||||||
"-DARDUINO_RUNNING_CORE=1",
|
|
||||||
"-DARDUINO_EVENT_RUNNING_CORE=1",
|
|
||||||
"-DARDUINO_USB_CDC_ON_BOOT=1"
|
|
||||||
],
|
|
||||||
"f_cpu": "240000000L",
|
|
||||||
"f_flash": "80000000L",
|
|
||||||
"flash_mode": "qio",
|
|
||||||
"espidf": {
|
|
||||||
"sdkconfig_path": "boards"
|
|
||||||
},
|
|
||||||
"hwids": [
|
|
||||||
[
|
|
||||||
"0x303A",
|
|
||||||
"0x1001"
|
|
||||||
]
|
|
||||||
],
|
|
||||||
"mcu": "esp32s3",
|
|
||||||
"variant": "esp32s3"
|
|
||||||
},
|
|
||||||
"connectivity": [
|
|
||||||
"bluetooth",
|
|
||||||
"wifi"
|
|
||||||
],
|
|
||||||
"debug": {
|
|
||||||
"default_tool": "esp-builtin",
|
|
||||||
"onboard_tools": [
|
|
||||||
"esp-builtin"
|
|
||||||
],
|
|
||||||
"openocd_target": "esp32s3.cfg"
|
|
||||||
},
|
|
||||||
"frameworks": [
|
|
||||||
"arduino",
|
|
||||||
"espidf"
|
|
||||||
],
|
|
||||||
"name": "BTClock (rev. B)",
|
|
||||||
"upload": {
|
|
||||||
"flash_size": "8MB",
|
|
||||||
"maximum_ram_size": 327680,
|
|
||||||
"maximum_size": 8388608,
|
|
||||||
"use_1200bps_touch": true,
|
|
||||||
"wait_for_upload_port": true,
|
|
||||||
"require_upload_port": true,
|
|
||||||
"speed": 460800
|
|
||||||
},
|
|
||||||
"url": "http://github.com/btclock",
|
|
||||||
"vendor": "BTClock"
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,15 +0,0 @@
|
||||||
# Use the official Python 3.9 image as the base
|
|
||||||
FROM python:3.9-slim
|
|
||||||
|
|
||||||
# Set the working directory
|
|
||||||
WORKDIR /workspace
|
|
||||||
|
|
||||||
RUN apt-get update && apt-get install -y git
|
|
||||||
|
|
||||||
# Install PlatformIO
|
|
||||||
RUN pip install platformio
|
|
||||||
|
|
||||||
WORKDIR /usr/src
|
|
||||||
|
|
||||||
CMD ["platformio", "run"]
|
|
||||||
|
|
2
data
2
data
|
@ -1 +1 @@
|
||||||
Subproject commit 85b9b17506f89696b89ab6f6e6ed231b7a8f6e91
|
Subproject commit 3f20d67f1abc10b20ddecfb5aa0ff4eb78c4c149
|
|
@ -1,9 +1,15 @@
|
||||||
dependencies:
|
dependencies:
|
||||||
|
esp_littlefs:
|
||||||
|
component_hash: null
|
||||||
|
source:
|
||||||
|
path: /Users/padjuri/src/btclock_espidf/btclock_espidf/managed_components/esp_littlefs
|
||||||
|
type: local
|
||||||
|
version: 1.10.2
|
||||||
idf:
|
idf:
|
||||||
component_hash: null
|
component_hash: null
|
||||||
source:
|
source:
|
||||||
type: idf
|
type: idf
|
||||||
version: 4.4.7
|
version: 4.4.5
|
||||||
manifest_hash: cd2f3ee15e776d949eb4ea4eddc8f39b30c2a7905050850eed01ab4928143cff
|
manifest_hash: 4796491ac0ef21bc9e7da581f1db6c59f92d6096be0ffd2d5fa4f2645943c54a
|
||||||
target: esp32s3
|
target: esp32s3
|
||||||
version: 1.0.0
|
version: 1.0.0
|
||||||
|
|
|
@ -1,52 +0,0 @@
|
||||||
#include "bitaxe_handler.hpp"
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parseBitaxeHashRate(std::string text)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> ret;
|
|
||||||
ret.fill(""); // Initialize all elements to empty strings
|
|
||||||
|
|
||||||
std::size_t textLength = text.length();
|
|
||||||
|
|
||||||
// Calculate the position where the digits should start
|
|
||||||
// Account for the position of the "mdi:pickaxe" and the "GH/S" label
|
|
||||||
std::size_t startIndex = NUM_SCREENS - 1 - textLength;
|
|
||||||
|
|
||||||
// Insert the "mdi:pickaxe" icon just before the digits
|
|
||||||
if (startIndex > 0)
|
|
||||||
{
|
|
||||||
ret[startIndex - 1] = "mdi:pickaxe";
|
|
||||||
}
|
|
||||||
|
|
||||||
// Place the digits
|
|
||||||
for (std::size_t i = 0; i < textLength; ++i)
|
|
||||||
{
|
|
||||||
ret[startIndex + i] = text.substr(i, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
ret[NUM_SCREENS - 1] = "GH/S";
|
|
||||||
ret[0] = "mdi:bitaxe";
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parseBitaxeBestDiff(std::string text)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> ret;
|
|
||||||
std::uint32_t firstIndex = 0;
|
|
||||||
|
|
||||||
if (text.length() < NUM_SCREENS)
|
|
||||||
{
|
|
||||||
text.insert(text.begin(), NUM_SCREENS - text.length(), ' ');
|
|
||||||
ret[0] = "mdi:bitaxe";
|
|
||||||
ret[1] = "mdi:rocket";
|
|
||||||
firstIndex = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (std::uint8_t i = firstIndex; i < NUM_SCREENS; i++)
|
|
||||||
{
|
|
||||||
ret[i] = text[i];
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,5 +0,0 @@
|
||||||
#include <array>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parseBitaxeHashRate(std::string text);
|
|
||||||
std::array<std::string, NUM_SCREENS> parseBitaxeBestDiff(std::string text);
|
|
|
@ -1,186 +1,60 @@
|
||||||
#include "data_handler.hpp"
|
#include "data_handler.hpp"
|
||||||
#ifdef __EMSCRIPTEN__
|
|
||||||
#include <emscripten.h>
|
|
||||||
#include <emscripten/bind.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
char getCurrencySymbol(char input)
|
std::array<std::string, NUM_SCREENS> parsePriceData(std::uint32_t price, char currencySymbol)
|
||||||
{
|
|
||||||
switch (input)
|
|
||||||
{
|
|
||||||
case CURRENCY_EUR:
|
|
||||||
return '[';
|
|
||||||
break;
|
|
||||||
case CURRENCY_GBP:
|
|
||||||
return ']';
|
|
||||||
break;
|
|
||||||
case CURRENCY_JPY:
|
|
||||||
return '^';
|
|
||||||
break;
|
|
||||||
case CURRENCY_AUD:
|
|
||||||
case CURRENCY_CAD:
|
|
||||||
case CURRENCY_USD:
|
|
||||||
return '$';
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return input;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string getCurrencyCode(char input)
|
|
||||||
{
|
|
||||||
switch (input)
|
|
||||||
{
|
|
||||||
case CURRENCY_EUR:
|
|
||||||
return CURRENCY_CODE_EUR;
|
|
||||||
break;
|
|
||||||
case CURRENCY_GBP:
|
|
||||||
return CURRENCY_CODE_GBP;
|
|
||||||
break;
|
|
||||||
case CURRENCY_JPY:
|
|
||||||
return CURRENCY_CODE_JPY;
|
|
||||||
break;
|
|
||||||
case CURRENCY_AUD:
|
|
||||||
return CURRENCY_CODE_AUD;
|
|
||||||
break;
|
|
||||||
case CURRENCY_CAD:
|
|
||||||
return CURRENCY_CODE_CAD;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return CURRENCY_CODE_USD;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
char getCurrencyChar(const std::string& input)
|
|
||||||
{
|
|
||||||
if (input == "EUR")
|
|
||||||
return CURRENCY_EUR;
|
|
||||||
else if (input == "GBP")
|
|
||||||
return CURRENCY_GBP;
|
|
||||||
else if (input == "JPY")
|
|
||||||
return CURRENCY_JPY;
|
|
||||||
else if (input == "AUD")
|
|
||||||
return CURRENCY_AUD;
|
|
||||||
else if (input == "CAD")
|
|
||||||
return CURRENCY_CAD;
|
|
||||||
else
|
|
||||||
return CURRENCY_USD; // Assuming USD is the default for unknown inputs
|
|
||||||
}
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parsePriceData(std::uint32_t price, char currencySymbol, bool useSuffixFormat, bool mowMode, bool shareDot)
|
|
||||||
{
|
{
|
||||||
std::array<std::string, NUM_SCREENS> ret;
|
std::array<std::string, NUM_SCREENS> ret;
|
||||||
std::string priceString;
|
std::string priceString;
|
||||||
if (std::to_string(price).length() >= NUM_SCREENS || useSuffixFormat)
|
if (std::to_string(price).length() >= NUM_SCREENS) {
|
||||||
{
|
priceString = formatNumberWithSuffix(price);
|
||||||
int numScreens = shareDot || mowMode ? NUM_SCREENS - 1 : NUM_SCREENS - 2;
|
} else {
|
||||||
priceString = getCurrencySymbol(currencySymbol) + formatNumberWithSuffix(price, numScreens, mowMode);
|
priceString = currencySymbol + std::to_string(price);
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
priceString = getCurrencySymbol(currencySymbol) + std::to_string(price);
|
|
||||||
}
|
}
|
||||||
std::uint32_t firstIndex = 0;
|
std::uint32_t firstIndex = 0;
|
||||||
if ((shareDot && priceString.length() <= (NUM_SCREENS)) || priceString.length() < (NUM_SCREENS))
|
if (priceString.length() < (NUM_SCREENS))
|
||||||
{
|
{
|
||||||
priceString.insert(priceString.begin(), NUM_SCREENS - priceString.length(), ' ');
|
priceString.insert(priceString.begin(), NUM_SCREENS - priceString.length(), ' ');
|
||||||
|
if (currencySymbol == '[')
|
||||||
if (mowMode)
|
|
||||||
{
|
{
|
||||||
ret[0] = "MOW/UNITS";
|
ret[0] = "BTC/EUR";
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ret[0] = "BTC/" + getCurrencyCode(currencySymbol);
|
ret[0] = "BTC/USD";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
firstIndex = 1;
|
firstIndex = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t dotPosition = priceString.find('.');
|
for (std::uint32_t i = firstIndex; i < NUM_SCREENS; i++)
|
||||||
|
|
||||||
if (shareDot && dotPosition != std::string::npos && dotPosition > 0)
|
|
||||||
{
|
{
|
||||||
std::vector<std::string> tempArray;
|
ret[i] = priceString[i];
|
||||||
if (dotPosition != std::string::npos && dotPosition > 0)
|
|
||||||
{
|
|
||||||
for (size_t i = 0; i < priceString.length(); ++i)
|
|
||||||
{
|
|
||||||
if (i == dotPosition - 1)
|
|
||||||
{
|
|
||||||
tempArray.push_back(std::string(1, priceString[i]) + ".");
|
|
||||||
++i; // Skip the dot in the next iteration
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
tempArray.push_back(std::string(1, priceString[i]));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Copy from tempArray to ret
|
|
||||||
for (std::uint32_t i = firstIndex; i < NUM_SCREENS && i - firstIndex < tempArray.size(); ++i)
|
|
||||||
{
|
|
||||||
ret[i] = tempArray[i - firstIndex];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
for (std::uint32_t i = firstIndex; i < NUM_SCREENS; i++)
|
|
||||||
{
|
|
||||||
ret[i] = std::string(1, priceString[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parseSatsPerCurrency(std::uint32_t price,char currencySymbol, bool withSatsSymbol)
|
std::array<std::string, NUM_SCREENS> parseSatsPerCurrency(std::uint32_t price, char currencySymbol)
|
||||||
{
|
{
|
||||||
std::array<std::string, NUM_SCREENS> ret;
|
std::array<std::string, NUM_SCREENS> ret;
|
||||||
std::string priceString = std::to_string(int(round(1 / float(price) * 10e7)));
|
std::string priceString = std::to_string(int(round(1 / float(price) * 10e7)));
|
||||||
std::uint32_t firstIndex = 0;
|
std::uint32_t firstIndex = 0;
|
||||||
std::uint8_t insertSatSymbol = NUM_SCREENS - priceString.length() - 1;
|
|
||||||
|
|
||||||
if (priceString.length() < (NUM_SCREENS))
|
if (priceString.length() < (NUM_SCREENS))
|
||||||
{
|
{
|
||||||
// Check if price is greater than 1 billion
|
priceString.insert(priceString.begin(), NUM_SCREENS - priceString.length(), ' ');
|
||||||
if (price >= 100000000)
|
if (currencySymbol == '[')
|
||||||
{
|
{
|
||||||
double satsPerCurrency = (1.0 / static_cast<double>(price)) * 1e8; // Calculate satoshis
|
ret[0] = "SATS/EUR";
|
||||||
std::ostringstream oss;
|
|
||||||
oss << std::fixed << std::setprecision(3) << satsPerCurrency; // Format with 3 decimal places
|
|
||||||
priceString = oss.str();
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
priceString = std::to_string(static_cast<int>(round(1.0 / static_cast<double>(price) * 1e8))); // Default formatting
|
|
||||||
}
|
|
||||||
|
|
||||||
// Pad the string with spaces if necessary
|
|
||||||
if (priceString.length() < NUM_SCREENS)
|
|
||||||
{
|
|
||||||
priceString.insert(priceString.begin(), NUM_SCREENS - priceString.length(), ' ');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (currencySymbol != CURRENCY_USD || price >= 100000000) // no time anymore when earlier than 1
|
|
||||||
ret[0] = "SATS/" + getCurrencyCode(currencySymbol);
|
|
||||||
else
|
|
||||||
ret[0] = "MSCW/TIME";
|
ret[0] = "MSCW/TIME";
|
||||||
|
}
|
||||||
firstIndex = 1;
|
firstIndex = 1;
|
||||||
|
|
||||||
for (std::uint32_t i = firstIndex; i < NUM_SCREENS; i++)
|
for (std::uint32_t i = firstIndex; i < NUM_SCREENS; i++)
|
||||||
{
|
{
|
||||||
ret[i] = priceString[i];
|
ret[i] = priceString[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (withSatsSymbol)
|
|
||||||
{
|
|
||||||
ret[insertSatSymbol] = "STS";
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -206,67 +80,24 @@ std::array<std::string, NUM_SCREENS> parseBlockHeight(std::uint32_t blockHeight)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parseBlockFees(std::uint16_t blockFees)
|
std::array<std::string, NUM_SCREENS> parseHalvingCountdown(std::uint32_t blockHeight)
|
||||||
{
|
{
|
||||||
std::array<std::string, NUM_SCREENS> ret;
|
std::array<std::string, NUM_SCREENS> ret;
|
||||||
std::string blockFeesString = std::to_string(blockFees);
|
|
||||||
std::uint32_t firstIndex = 0;
|
|
||||||
|
|
||||||
if (blockFeesString.length() < NUM_SCREENS)
|
|
||||||
{
|
|
||||||
blockFeesString.insert(blockFeesString.begin(), NUM_SCREENS - blockFeesString.length() - 1, ' ');
|
|
||||||
ret[0] = "FEE/RATE";
|
|
||||||
firstIndex = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (std::uint8_t i = firstIndex; i < NUM_SCREENS - 1; i++)
|
|
||||||
{
|
|
||||||
ret[i] = blockFeesString[i];
|
|
||||||
}
|
|
||||||
|
|
||||||
ret[NUM_SCREENS - 1] = "sat/vB";
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parseHalvingCountdown(std::uint32_t blockHeight, bool asBlocks)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> ret;
|
|
||||||
const std::uint32_t nextHalvingBlock = 210000 - (blockHeight % 210000);
|
const std::uint32_t nextHalvingBlock = 210000 - (blockHeight % 210000);
|
||||||
const std::uint32_t minutesToHalving = nextHalvingBlock * 10;
|
const std::uint32_t minutesToHalving = nextHalvingBlock * 10;
|
||||||
|
|
||||||
if (asBlocks)
|
const int years = floor(minutesToHalving / 525600);
|
||||||
{
|
const int days = floor((minutesToHalving - (years * 525600)) / (24 * 60));
|
||||||
std::string blockNrString = std::to_string(nextHalvingBlock);
|
const int hours = floor((minutesToHalving - (years * 525600) - (days * (24 * 60))) / 60);
|
||||||
std::uint32_t firstIndex = 0;
|
const int mins = floor(minutesToHalving - (years * 525600) - (days * (24 * 60)) - (hours * 60));
|
||||||
|
ret[0] = "BIT/COIN";
|
||||||
if (blockNrString.length() < NUM_SCREENS)
|
ret[1] = "HALV/ING";
|
||||||
{
|
ret[(NUM_SCREENS - 5)] = std::to_string(years) + "/YRS";
|
||||||
blockNrString.insert(blockNrString.begin(), NUM_SCREENS - blockNrString.length(), ' ');
|
ret[(NUM_SCREENS - 4)] = std::to_string(days) + "/DAYS";
|
||||||
ret[0] = "HAL/VING";
|
ret[(NUM_SCREENS - 3)] = std::to_string(hours) + "/HRS";
|
||||||
firstIndex = 1;
|
ret[(NUM_SCREENS - 2)] = std::to_string(mins) + "/MINS";
|
||||||
}
|
ret[(NUM_SCREENS - 1)] = "TO/GO";
|
||||||
|
|
||||||
for (std::uint32_t i = firstIndex; i < NUM_SCREENS; i++)
|
|
||||||
{
|
|
||||||
ret[i] = blockNrString[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
|
|
||||||
const int years = floor(minutesToHalving / 525600);
|
|
||||||
const int days = floor((minutesToHalving - (years * 525600)) / (24 * 60));
|
|
||||||
const int hours = floor((minutesToHalving - (years * 525600) - (days * (24 * 60))) / 60);
|
|
||||||
const int mins = floor(minutesToHalving - (years * 525600) - (days * (24 * 60)) - (hours * 60));
|
|
||||||
ret[0] = "BIT/COIN";
|
|
||||||
ret[1] = "HAL/VING";
|
|
||||||
ret[(NUM_SCREENS - 5)] = std::to_string(years) + "/YRS";
|
|
||||||
ret[(NUM_SCREENS - 4)] = std::to_string(days) + "/DAYS";
|
|
||||||
ret[(NUM_SCREENS - 3)] = std::to_string(hours) + "/HRS";
|
|
||||||
ret[(NUM_SCREENS - 2)] = std::to_string(mins) + "/MINS";
|
|
||||||
ret[(NUM_SCREENS - 1)] = "TO/GO";
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -276,16 +107,21 @@ std::array<std::string, NUM_SCREENS> parseMarketCap(std::uint32_t blockHeight, s
|
||||||
std::array<std::string, NUM_SCREENS> ret;
|
std::array<std::string, NUM_SCREENS> ret;
|
||||||
std::uint32_t firstIndex = 0;
|
std::uint32_t firstIndex = 0;
|
||||||
double supply = getSupplyAtBlock(blockHeight);
|
double supply = getSupplyAtBlock(blockHeight);
|
||||||
uint64_t marketCap = static_cast<std::uint64_t>(supply * double(price));
|
int64_t marketCap = static_cast<std::int64_t>(supply * double(price));
|
||||||
|
if (currencySymbol == '[')
|
||||||
ret[0] = getCurrencyCode(currencySymbol) + "/MCAP";
|
{
|
||||||
|
ret[0] = "EUR/MCAP";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ret[0] = "USD/MCAP";
|
||||||
|
}
|
||||||
|
|
||||||
if (bigChars)
|
if (bigChars)
|
||||||
{
|
{
|
||||||
firstIndex = 1;
|
firstIndex = 1;
|
||||||
// Serial.print("Market cap: ");
|
|
||||||
// Serial.println(marketCap);
|
std::string priceString = currencySymbol + formatNumberWithSuffix(marketCap);
|
||||||
std::string priceString = currencySymbol + formatNumberWithSuffix(marketCap, (NUM_SCREENS - 2));
|
|
||||||
priceString.insert(priceString.begin(), NUM_SCREENS - priceString.length(), ' ');
|
priceString.insert(priceString.begin(), NUM_SCREENS - priceString.length(), ' ');
|
||||||
|
|
||||||
for (std::uint32_t i = firstIndex; i < NUM_SCREENS; i++)
|
for (std::uint32_t i = firstIndex; i < NUM_SCREENS; i++)
|
||||||
|
@ -312,7 +148,7 @@ std::array<std::string, NUM_SCREENS> parseMarketCap(std::uint32_t blockHeight, s
|
||||||
ret[i] = "";
|
ret[i] = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
ret[NUM_SCREENS - groups - 1] = std::string(" ") + currencySymbol + " ";
|
ret[NUM_SCREENS - groups - 1] = " $ ";
|
||||||
for (std::uint32_t i = 0; i < groups; i++)
|
for (std::uint32_t i = 0; i < groups; i++)
|
||||||
{
|
{
|
||||||
ret[(NUM_SCREENS - groups + i)] = stringValue.substr(i * 3, 3).c_str();
|
ret[(NUM_SCREENS - groups + i)] = stringValue.substr(i * 3, 3).c_str();
|
||||||
|
@ -321,70 +157,3 @@ std::array<std::string, NUM_SCREENS> parseMarketCap(std::uint32_t blockHeight, s
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef __EMSCRIPTEN__
|
|
||||||
emscripten::val arrayToStringArray(const std::array<std::string, NUM_SCREENS> &arr)
|
|
||||||
{
|
|
||||||
emscripten::val jsArray = emscripten::val::array();
|
|
||||||
for (const auto &str : arr)
|
|
||||||
{
|
|
||||||
jsArray.call<void>("push", str);
|
|
||||||
}
|
|
||||||
return jsArray;
|
|
||||||
}
|
|
||||||
|
|
||||||
emscripten::val vectorToStringArray(const std::vector<std::string> &vec)
|
|
||||||
{
|
|
||||||
emscripten::val jsArray = emscripten::val::array();
|
|
||||||
for (size_t i = 0; i < vec.size(); ++i)
|
|
||||||
{
|
|
||||||
jsArray.set(i, vec[i]);
|
|
||||||
}
|
|
||||||
return jsArray;
|
|
||||||
}
|
|
||||||
|
|
||||||
emscripten::val parseBlockHeightArray(std::uint32_t blockHeight)
|
|
||||||
{
|
|
||||||
return arrayToStringArray(parseBlockHeight(blockHeight));
|
|
||||||
}
|
|
||||||
|
|
||||||
emscripten::val parsePriceDataArray(std::uint32_t price, const std::string ¤cySymbol, bool useSuffixFormat = false, bool mowMode = false, bool shareDot = false)
|
|
||||||
{
|
|
||||||
return arrayToStringArray(parsePriceData(price, currencySymbol[0], useSuffixFormat, mowMode, shareDot));
|
|
||||||
}
|
|
||||||
|
|
||||||
emscripten::val parseHalvingCountdownArray(std::uint32_t blockHeight, bool asBlocks)
|
|
||||||
{
|
|
||||||
return arrayToStringArray(parseHalvingCountdown(blockHeight, asBlocks));
|
|
||||||
}
|
|
||||||
|
|
||||||
emscripten::val parseMarketCapArray(std::uint32_t blockHeight, std::uint32_t price, const std::string ¤cySymbol, bool bigChars)
|
|
||||||
{
|
|
||||||
return arrayToStringArray(parseMarketCap(blockHeight, price, currencySymbol[0], bigChars));
|
|
||||||
}
|
|
||||||
|
|
||||||
emscripten::val parseBlockFeesArray(std::uint16_t blockFees)
|
|
||||||
{
|
|
||||||
return arrayToStringArray(parseBlockFees(blockFees));
|
|
||||||
}
|
|
||||||
|
|
||||||
emscripten::val parseSatsPerCurrencyArray(std::uint32_t price, const std::string ¤cySymbol, bool withSatsSymbol)
|
|
||||||
{
|
|
||||||
return arrayToStringArray(parseSatsPerCurrency(price, currencySymbol[0], withSatsSymbol));
|
|
||||||
}
|
|
||||||
|
|
||||||
EMSCRIPTEN_BINDINGS(my_module)
|
|
||||||
{
|
|
||||||
// emscripten::register_vector<std::string>("StringList");
|
|
||||||
|
|
||||||
emscripten::function("parseBlockHeight", &parseBlockHeightArray);
|
|
||||||
emscripten::function("parseHalvingCountdown", &parseHalvingCountdownArray);
|
|
||||||
emscripten::function("parseMarketCap", &parseMarketCapArray);
|
|
||||||
emscripten::function("parseBlockFees", &parseBlockFeesArray);
|
|
||||||
emscripten::function("parseSatsPerCurrency", &parseSatsPerCurrencyArray);
|
|
||||||
emscripten::function("parsePriceData", &parsePriceDataArray);
|
|
||||||
|
|
||||||
emscripten::function("arrayToStringArray", &arrayToStringArray);
|
|
||||||
emscripten::function("vectorToStringArray", &vectorToStringArray);
|
|
||||||
}
|
|
||||||
#endif
|
|
|
@ -2,31 +2,11 @@
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#include "utils.hpp"
|
#include "utils.hpp"
|
||||||
|
|
||||||
const char CURRENCY_USD = '$';
|
std::array<std::string, NUM_SCREENS> parsePriceData(std::uint32_t price, char currencySymbol);
|
||||||
const char CURRENCY_EUR = '[';
|
std::array<std::string, NUM_SCREENS> parseSatsPerCurrency(std::uint32_t price, char currencySymbol);
|
||||||
const char CURRENCY_GBP = ']';
|
|
||||||
const char CURRENCY_JPY = '^';
|
|
||||||
const char CURRENCY_AUD = '_';
|
|
||||||
const char CURRENCY_CAD = '`';
|
|
||||||
|
|
||||||
const std::string CURRENCY_CODE_USD = "USD";
|
|
||||||
const std::string CURRENCY_CODE_EUR = "EUR";
|
|
||||||
const std::string CURRENCY_CODE_GBP = "GBP";
|
|
||||||
const std::string CURRENCY_CODE_JPY = "JPY";
|
|
||||||
const std::string CURRENCY_CODE_AUD = "AUD";
|
|
||||||
const std::string CURRENCY_CODE_CAD = "CAD";
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parsePriceData(std::uint32_t price, char currency, bool useSuffixFormat = false, bool mowMode = false, bool shareDot = false);
|
|
||||||
std::array<std::string, NUM_SCREENS> parseSatsPerCurrency(std::uint32_t price, char currencySymbol, bool withSatsSymbol);
|
|
||||||
std::array<std::string, NUM_SCREENS> parseBlockHeight(std::uint32_t blockHeight);
|
std::array<std::string, NUM_SCREENS> parseBlockHeight(std::uint32_t blockHeight);
|
||||||
std::array<std::string, NUM_SCREENS> parseHalvingCountdown(std::uint32_t blockHeight, bool asBlocks);
|
std::array<std::string, NUM_SCREENS> parseHalvingCountdown(std::uint32_t blockHeight);
|
||||||
std::array<std::string, NUM_SCREENS> parseMarketCap(std::uint32_t blockHeight, std::uint32_t price, char currencySymbol, bool bigChars);
|
std::array<std::string, NUM_SCREENS> parseMarketCap(std::uint32_t blockHeight, std::uint32_t price, char currencySymbol, bool bigChars);
|
||||||
std::array<std::string, NUM_SCREENS> parseBlockFees(std::uint16_t blockFees);
|
|
||||||
|
|
||||||
char getCurrencySymbol(char input);
|
|
||||||
std::string getCurrencyCode(char input);
|
|
||||||
char getCurrencyChar(const std::string& input);
|
|
|
@ -1,24 +0,0 @@
|
||||||
#include "nostrdisplay_handler.hpp"
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parseZapNotify(std::uint16_t amount, bool withSatsSymbol)
|
|
||||||
{
|
|
||||||
std::string text = std::to_string(amount);
|
|
||||||
std::size_t textLength = text.length();
|
|
||||||
std::size_t startIndex = NUM_SCREENS - textLength;
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> textEpdContent = {"ZAP", "mdi-lnbolt", "", "", "", "", ""};
|
|
||||||
|
|
||||||
// Insert the sats symbol just before the digits
|
|
||||||
if (startIndex > 0 && withSatsSymbol)
|
|
||||||
{
|
|
||||||
textEpdContent[startIndex - 1] = "STS";
|
|
||||||
}
|
|
||||||
|
|
||||||
// Place the digits
|
|
||||||
for (std::size_t i = 0; i < textLength; i++)
|
|
||||||
{
|
|
||||||
textEpdContent[startIndex + i] = text.substr(i, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
return textEpdContent;
|
|
||||||
}
|
|
|
@ -1,5 +0,0 @@
|
||||||
#include <array>
|
|
||||||
#include <string>
|
|
||||||
#include "utils.hpp"
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> parseZapNotify(std::uint16_t amount, bool withSatsSymbol);
|
|
|
@ -5,21 +5,18 @@ int modulo(int x, int N)
|
||||||
return (x % N + N) % N;
|
return (x % N + N) % N;
|
||||||
}
|
}
|
||||||
|
|
||||||
double getSupplyAtBlock(std::uint32_t blockNr)
|
double getSupplyAtBlock(std::uint32_t blockNr) {
|
||||||
{
|
if (blockNr >= 33 * 210000) {
|
||||||
if (blockNr >= 33 * 210000)
|
|
||||||
{
|
|
||||||
return 20999999.9769;
|
return 20999999.9769;
|
||||||
}
|
}
|
||||||
|
|
||||||
const int initialBlockReward = 50; // Initial block reward
|
const int initialBlockReward = 50; // Initial block reward
|
||||||
const int halvingInterval = 210000; // Number of blocks before halving
|
const int halvingInterval = 210000; // Number of blocks before halving
|
||||||
|
|
||||||
int halvingCount = blockNr / halvingInterval;
|
int halvingCount = blockNr / halvingInterval;
|
||||||
double totalBitcoinInCirculation = 0;
|
double totalBitcoinInCirculation = 0;
|
||||||
|
|
||||||
for (int i = 0; i < halvingCount; ++i)
|
for (int i = 0; i < halvingCount; ++i) {
|
||||||
{
|
|
||||||
totalBitcoinInCirculation += halvingInterval * initialBlockReward * std::pow(0.5, i);
|
totalBitcoinInCirculation += halvingInterval * initialBlockReward * std::pow(0.5, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -28,139 +25,24 @@ double getSupplyAtBlock(std::uint32_t blockNr)
|
||||||
return totalBitcoinInCirculation;
|
return totalBitcoinInCirculation;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string formatNumberWithSuffix(std::uint64_t num, int numCharacters)
|
std::string formatNumberWithSuffix(std::uint64_t num) {
|
||||||
{
|
|
||||||
return formatNumberWithSuffix(num, numCharacters, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string formatNumberWithSuffix(std::uint64_t num, int numCharacters, bool mowMode)
|
|
||||||
{
|
|
||||||
static char result[20]; // Adjust size as needed
|
|
||||||
const long long quadrillion = 1000000000000000LL;
|
const long long quadrillion = 1000000000000000LL;
|
||||||
const long long trillion = 1000000000000LL;
|
const long long trillion = 1000000000000LL;
|
||||||
const long long billion = 1000000000;
|
const long long billion = 1000000000;
|
||||||
const long long million = 1000000;
|
const long long million = 1000000;
|
||||||
const long long thousand = 1000;
|
const long long thousand = 1000;
|
||||||
|
|
||||||
double numDouble = (double)num;
|
if (num >= quadrillion) {
|
||||||
int numDigits = (int)log10(num) + 1;
|
return std::to_string(num / quadrillion) + "Q";
|
||||||
char suffix;
|
} else if (num >= trillion) {
|
||||||
|
return std::to_string(num / trillion) + "T";
|
||||||
if (num >= quadrillion || numDigits > 15)
|
} else if (num >= billion) {
|
||||||
{
|
return std::to_string(num / billion) + "B";
|
||||||
numDouble /= quadrillion;
|
} else if (num >= million) {
|
||||||
suffix = 'Q';
|
return std::to_string(num / million) + "M";
|
||||||
|
} else if (num >= thousand) {
|
||||||
|
return std::to_string(num / thousand) + "K";
|
||||||
|
} else {
|
||||||
|
return std::to_string(num);
|
||||||
}
|
}
|
||||||
else if (num >= trillion || numDigits > 12)
|
|
||||||
{
|
|
||||||
numDouble /= trillion;
|
|
||||||
suffix = 'T';
|
|
||||||
}
|
|
||||||
else if (num >= billion || numDigits > 9)
|
|
||||||
{
|
|
||||||
numDouble /= billion;
|
|
||||||
suffix = 'B';
|
|
||||||
}
|
|
||||||
else if (num >= million || numDigits > 6 || (mowMode && num >= thousand))
|
|
||||||
{
|
|
||||||
numDouble /= million;
|
|
||||||
suffix = 'M';
|
|
||||||
}
|
|
||||||
else if (!mowMode && (num >= thousand || numDigits > 3))
|
|
||||||
{
|
|
||||||
numDouble /= thousand;
|
|
||||||
suffix = 'K';
|
|
||||||
}
|
|
||||||
else if (!mowMode)
|
|
||||||
{
|
|
||||||
snprintf(result, sizeof(result), "%llu", (unsigned long long)num);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
else // mowMode is true and num < 1000
|
|
||||||
{
|
|
||||||
numDouble /= million;
|
|
||||||
suffix = 'M';
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add suffix
|
|
||||||
int len;
|
|
||||||
|
|
||||||
// Mow Mode always uses string truncation to avoid rounding
|
|
||||||
std::string mowAsString = std::to_string(numDouble);
|
|
||||||
if (mowMode) {
|
|
||||||
// Default to one decimal place
|
|
||||||
len = snprintf(result, sizeof(result), "%s%c", mowAsString.substr(0, mowAsString.find(".") + 2).c_str(), suffix);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
len = snprintf(result, sizeof(result), "%.0f%c", numDouble, suffix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If there's room, add more decimal places
|
|
||||||
if (len < numCharacters)
|
|
||||||
{
|
|
||||||
int restLen = mowMode ? numCharacters - len : numCharacters - len - 1;
|
|
||||||
|
|
||||||
if (mowMode) {
|
|
||||||
snprintf(result, sizeof(result), "%s%c", mowAsString.substr(0, mowAsString.find(".") + 2 + restLen).c_str(), suffix);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
snprintf(result, sizeof(result), "%.*f%c", restLen, numDouble, suffix);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get sat amount from a bolt11 invoice
|
|
||||||
*
|
|
||||||
* Based on https://github.com/lnbits/nostr-zap-lamp/blob/main/nostrZapLamp/nostrZapLamp.ino
|
|
||||||
*/
|
|
||||||
int64_t getAmountInSatoshis(std::string bolt11) {
|
|
||||||
int64_t number = -1;
|
|
||||||
char multiplier = ' ';
|
|
||||||
|
|
||||||
for (unsigned int i = 0; i < bolt11.length(); ++i) {
|
|
||||||
if (isdigit(bolt11[i])) {
|
|
||||||
number = 0;
|
|
||||||
while (isdigit(bolt11[i])) {
|
|
||||||
number = number * 10 + (bolt11[i] - '0');
|
|
||||||
++i;
|
|
||||||
}
|
|
||||||
for (unsigned int j = i; j < bolt11.length(); ++j) {
|
|
||||||
if (isalpha(bolt11[j])) {
|
|
||||||
multiplier = bolt11[j];
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (number == -1 || multiplier == ' ') {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
int64_t satoshis = number;
|
|
||||||
|
|
||||||
switch (multiplier) {
|
|
||||||
case 'm':
|
|
||||||
satoshis *= 100000; // 0.001 * 100,000,000
|
|
||||||
break;
|
|
||||||
case 'u':
|
|
||||||
satoshis *= 100; // 0.000001 * 100,000,000
|
|
||||||
break;
|
|
||||||
case 'n':
|
|
||||||
satoshis /= 10; // 0.000000001 * 100,000,000
|
|
||||||
break;
|
|
||||||
case 'p':
|
|
||||||
satoshis /= 10000; // 0.000000000001 * 100,000,000
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return satoshis;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,13 +3,9 @@
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <sstream>
|
|
||||||
#include <iomanip>
|
|
||||||
|
|
||||||
int modulo(int x,int N);
|
int modulo(int x,int N);
|
||||||
|
|
||||||
double getSupplyAtBlock(std::uint32_t blockNr);
|
double getSupplyAtBlock(std::uint32_t blockNr);
|
||||||
|
|
||||||
std::string formatNumberWithSuffix(std::uint64_t num, int numCharacters = 4);
|
std::string formatNumberWithSuffix(std::uint64_t num);
|
||||||
std::string formatNumberWithSuffix(std::uint64_t num, int numCharacters, bool mowMode);
|
|
||||||
int64_t getAmountInSatoshis(std::string bolt11);
|
|
|
@ -1,20 +0,0 @@
|
||||||
identifier: BTClock
|
|
||||||
maintainers:
|
|
||||||
- npub1k5f85zx0xdskyayqpfpc0zq6n7vwqjuuxugkayk72fgynp34cs3qfcvqg2
|
|
||||||
relays:
|
|
||||||
- wss://relay.noderunners.network/
|
|
||||||
- wss://nostr.sathoarder.com/
|
|
||||||
- wss://offchain.pub/
|
|
||||||
- wss://nostr3.daedaluslabs.io/
|
|
||||||
- wss://nostr4.daedaluslabs.io/
|
|
||||||
- wss://nostr.dbtc.link/
|
|
||||||
- wss://purplepag.es/
|
|
||||||
- wss://nos.lol/
|
|
||||||
- wss://nostr1.daedaluslabs.io/
|
|
||||||
- wss://nostr.noderunners.network/
|
|
||||||
- wss://nostr.lnbitcoin.cz/
|
|
||||||
- wss://relay.primal.net/
|
|
||||||
- wss://relay.damus.io
|
|
||||||
- wss://nostr-relay.derekross.me/
|
|
||||||
- wss://nostr2.azzamo.net/
|
|
||||||
- wss://nostr2.daedaluslabs.io/
|
|
|
@ -1,7 +1,7 @@
|
||||||
# Name, Type, SubType, Offset, Size, Flags
|
# Name, Type, SubType, Offset, Size, Flags
|
||||||
nvs, data, nvs, 0x9000, 0x5000,
|
nvs, data, nvs, 36K, 20K,
|
||||||
otadata, data, ota, 0xe000, 0x2000,
|
otadata, data, ota, 56K, 8K,
|
||||||
app0, app, ota_0, 0x10000, 0x1b8000,
|
app0, app, ota_0, 64K, 1700K,
|
||||||
app1, app, ota_1, , 0x1b8000,
|
app1, app, ota_1, , 1700K,
|
||||||
spiffs, data, spiffs, , 0x66C00,
|
spiffs, data, spiffs, , 400K,
|
||||||
coredump, data, coredump,, 0x10000,
|
coredump, data, coredump,, 64K,
|
||||||
|
|
|
|
@ -1,7 +1,7 @@
|
||||||
# Name, Type, SubType, Offset, Size, Flags
|
# Name, Type, SubType, Offset, Size, Flags
|
||||||
nvs, data, nvs, 0x9000, 0x5000,
|
nvs, data, nvs, 36K, 20K,
|
||||||
otadata, data, ota, 0xe000, 0x2000,
|
otadata, data, ota, 56K, 8K,
|
||||||
app0, app, ota_0, 0x10000, 0x6F0000,
|
app0, app, ota_0, 64K, 4096K,
|
||||||
app1, app, ota_1, , 0x6F0000,
|
app1, app, ota_1, , 4096K,
|
||||||
spiffs, data, spiffs, , 0x200000,
|
spiffs, data, spiffs, , 3072K,
|
||||||
coredump, data, coredump,, 0x10000,
|
coredump, data, coredump,, 64K,
|
||||||
|
|
|
|
@ -1,7 +0,0 @@
|
||||||
# Name, Type, SubType, Offset, Size, Flags
|
|
||||||
nvs, data, nvs, 0x9000, 0x5000,
|
|
||||||
otadata, data, ota, 0xe000, 0x2000,
|
|
||||||
app0, app, ota_0, 0x10000, 0x370000,
|
|
||||||
app1, app, ota_1, , 0x370000,
|
|
||||||
spiffs, data, spiffs, , 0xCD000,
|
|
||||||
coredump, data, coredump,, 0x10000,
|
|
|
|
@ -9,20 +9,17 @@
|
||||||
; https://docs.platformio.org/page/projectconf.html
|
; https://docs.platformio.org/page/projectconf.html
|
||||||
[platformio]
|
[platformio]
|
||||||
data_dir = data/build_gz
|
data_dir = data/build_gz
|
||||||
default_envs = lolin_s3_mini_213epd, lolin_s3_mini_29epd, btclock_rev_b_213epd, btclock_v8_213epd
|
|
||||||
|
|
||||||
[env]
|
[env]
|
||||||
|
|
||||||
|
|
||||||
[btclock_base]
|
[btclock_base]
|
||||||
platform = espressif32 @ ^6.9.0
|
platform = https://github.com/platformio/platform-espressif32.git
|
||||||
framework = arduino, espidf
|
framework = arduino, espidf
|
||||||
monitor_speed = 115200
|
monitor_speed = 115200
|
||||||
monitor_filters = esp32_exception_decoder, colorize
|
monitor_filters = esp32_exception_decoder, colorize
|
||||||
board_build.filesystem = littlefs
|
board_build.filesystem = littlefs
|
||||||
extra_scripts = pre:scripts/pre_script.py, post:scripts/extra_script.py
|
extra_scripts = post:scripts/extra_script.py
|
||||||
board_build.embed_files =
|
|
||||||
x509_crt_bundle
|
|
||||||
build_flags =
|
build_flags =
|
||||||
!python scripts/git_rev.py
|
!python scripts/git_rev.py
|
||||||
-DLAST_BUILD_TIME=$UNIX_TIME
|
-DLAST_BUILD_TIME=$UNIX_TIME
|
||||||
|
@ -33,15 +30,16 @@ build_unflags =
|
||||||
-Werror=all
|
-Werror=all
|
||||||
-fno-exceptions
|
-fno-exceptions
|
||||||
lib_deps =
|
lib_deps =
|
||||||
https://github.com/joltwallet/esp_littlefs.git
|
bblanchon/ArduinoJson@^6.21.3
|
||||||
bblanchon/ArduinoJson@^7.2.1
|
esphome/Improv@^1.2.3
|
||||||
mathieucarbou/ESPAsyncWebServer @ 3.3.23
|
esphome/ESPAsyncWebServer-esphome@^3.1.0
|
||||||
robtillaart/MCP23017@^0.8.0
|
adafruit/Adafruit BusIO@^1.14.5
|
||||||
adafruit/Adafruit NeoPixel@^1.12.3
|
adafruit/Adafruit MCP23017 Arduino Library@^2.3.2
|
||||||
https://github.com/dsbaars/universal_pin#feature/mcp23017_rt
|
adafruit/Adafruit NeoPixel@^1.11.0
|
||||||
|
https://github.com/dsbaars/universal_pin
|
||||||
https://github.com/dsbaars/GxEPD2#universal_pin
|
https://github.com/dsbaars/GxEPD2#universal_pin
|
||||||
https://github.com/tzapu/WiFiManager.git#v2.0.17
|
https://github.com/tzapu/WiFiManager.git#v2.0.16-rc.2
|
||||||
rblb/Nostrduino@1.2.8
|
https://github.com/dsbaars/arduino-nostr#less_verbose
|
||||||
|
|
||||||
[env:lolin_s3_mini]
|
[env:lolin_s3_mini]
|
||||||
extends = btclock_base
|
extends = btclock_base
|
||||||
|
@ -55,77 +53,23 @@ build_flags =
|
||||||
-D NUM_SCREENS=7
|
-D NUM_SCREENS=7
|
||||||
-D I2C_SDA_PIN=35
|
-D I2C_SDA_PIN=35
|
||||||
-D I2C_SCK_PIN=36
|
-D I2C_SCK_PIN=36
|
||||||
-DARDUINO_USB_CDC_ON_BOOT=1
|
|
||||||
-D IS_HW_REV_A
|
|
||||||
build_unflags =
|
build_unflags =
|
||||||
${btclock_base.build_unflags}
|
${btclock_base.build_unflags}
|
||||||
|
|
||||||
|
[env:lolin_s3_mini_qr]
|
||||||
[env:btclock_rev_b]
|
|
||||||
extends = btclock_base
|
|
||||||
board = btclock_rev_b
|
|
||||||
board_build.partitions = partition_8mb.csv
|
|
||||||
build_flags =
|
|
||||||
${btclock_base.build_flags}
|
|
||||||
-D MCP_INT_PIN=8
|
|
||||||
-D NEOPIXEL_PIN=15
|
|
||||||
-D NEOPIXEL_COUNT=4
|
|
||||||
-D NUM_SCREENS=7
|
|
||||||
-D I2C_SDA_PIN=35
|
|
||||||
-D I2C_SCK_PIN=36
|
|
||||||
-D HAS_FRONTLIGHT
|
|
||||||
-D PCA_OE_PIN=45
|
|
||||||
-D PCA_I2C_ADDR=0x42
|
|
||||||
-D IS_HW_REV_B
|
|
||||||
lib_deps =
|
|
||||||
${btclock_base.lib_deps}
|
|
||||||
robtillaart/PCA9685@^0.7.1
|
|
||||||
claws/BH1750@^1.3.0
|
|
||||||
build_unflags =
|
|
||||||
${btclock_base.build_unflags}
|
|
||||||
|
|
||||||
[env:lolin_s3_mini_213epd]
|
|
||||||
extends = env:lolin_s3_mini
|
extends = env:lolin_s3_mini
|
||||||
test_framework = unity
|
test_framework = unity
|
||||||
build_flags =
|
build_flags =
|
||||||
${env:lolin_s3_mini.build_flags}
|
${env:lolin_s3_mini.build_flags}
|
||||||
-D USE_QR
|
-D USE_QR
|
||||||
-D VERSION_EPD_2_13
|
|
||||||
-D HW_REV=\"REV_A_EPD_2_13\"
|
|
||||||
|
|
||||||
|
[env:btclock_s3]
|
||||||
[env:btclock_rev_b_213epd]
|
|
||||||
extends = env:btclock_rev_b
|
|
||||||
test_framework = unity
|
|
||||||
build_flags =
|
|
||||||
${env:btclock_rev_b.build_flags}
|
|
||||||
-D USE_QR
|
|
||||||
-D VERSION_EPD_2_13
|
|
||||||
-D HW_REV=\"REV_B_EPD_2_13\"
|
|
||||||
|
|
||||||
[env:lolin_s3_mini_29epd]
|
|
||||||
extends = env:lolin_s3_mini
|
|
||||||
test_framework = unity
|
|
||||||
build_flags =
|
|
||||||
${env:lolin_s3_mini.build_flags}
|
|
||||||
-D USE_QR
|
|
||||||
-D VERSION_EPD_2_9
|
|
||||||
-D HW_REV=\"REV_A_EPD_2_9\"
|
|
||||||
|
|
||||||
[env:btclock_rev_b_29epd]
|
|
||||||
extends = env:btclock_rev_b
|
|
||||||
test_framework = unity
|
|
||||||
build_flags =
|
|
||||||
${env:btclock_rev_b.build_flags}
|
|
||||||
-D USE_QR
|
|
||||||
-D VERSION_EPD_2_9
|
|
||||||
-D HW_REV=\"REV_B_EPD_2_9\"
|
|
||||||
|
|
||||||
[env:btclock_v8]
|
|
||||||
extends = btclock_base
|
extends = btclock_base
|
||||||
board = btclock_v8
|
board = btclock
|
||||||
board_build.partitions = partition_16mb.csv
|
board_build.partitions = partition_16mb.csv
|
||||||
board_build.flash_mode = qio
|
upload_protocol = cmsis-dap
|
||||||
|
debug_tool = cmsis-dap
|
||||||
|
|
||||||
test_framework = unity
|
test_framework = unity
|
||||||
build_flags =
|
build_flags =
|
||||||
${btclock_base.build_flags}
|
${btclock_base.build_flags}
|
||||||
|
@ -147,15 +91,6 @@ build_flags =
|
||||||
build_unflags =
|
build_unflags =
|
||||||
${btclock_base.build_unflags}
|
${btclock_base.build_unflags}
|
||||||
|
|
||||||
[env:btclock_v8_213epd]
|
|
||||||
extends = env:btclock_v8
|
|
||||||
test_framework = unity
|
|
||||||
build_flags =
|
|
||||||
${env:btclock_v8.build_flags}
|
|
||||||
-D USE_QR
|
|
||||||
-D VERSION_EPD_2_13
|
|
||||||
-D HW_REV=\"REV_V8_EPD_2_13\"
|
|
||||||
|
|
||||||
[env:native_test_only]
|
[env:native_test_only]
|
||||||
platform = native
|
platform = native
|
||||||
test_framework = unity
|
test_framework = unity
|
||||||
|
|
|
@ -1 +0,0 @@
|
||||||
platformio
|
|
|
@ -3,16 +3,6 @@ import os
|
||||||
import gzip
|
import gzip
|
||||||
from shutil import copyfileobj, rmtree
|
from shutil import copyfileobj, rmtree
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
import subprocess
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
revision = (
|
|
||||||
subprocess.check_output(["git", "rev-parse", "HEAD"])
|
|
||||||
.strip()
|
|
||||||
.decode("utf-8")
|
|
||||||
)
|
|
||||||
|
|
||||||
def gzip_file(input_file, output_file):
|
def gzip_file(input_file, output_file):
|
||||||
with open(input_file, 'rb') as f_in:
|
with open(input_file, 'rb') as f_in:
|
||||||
|
@ -34,9 +24,7 @@ def process_directory(input_dir, output_dir):
|
||||||
output_file_path = os.path.join(output_root, file + '.gz')
|
output_file_path = os.path.join(output_root, file + '.gz')
|
||||||
gzip_file(input_file_path, output_file_path)
|
gzip_file(input_file_path, output_file_path)
|
||||||
print(f'Compressed: {input_file_path} -> {output_file_path}')
|
print(f'Compressed: {input_file_path} -> {output_file_path}')
|
||||||
file_path = os.path.join(output_dir, "fs_hash.txt")
|
|
||||||
with open(file_path, "w") as file:
|
|
||||||
file.write(revision)
|
|
||||||
|
|
||||||
|
|
||||||
# Build web interface before building FS
|
# Build web interface before building FS
|
||||||
|
@ -46,15 +34,5 @@ def before_buildfs(source, target, env):
|
||||||
output_directory = 'data/build_gz'
|
output_directory = 'data/build_gz'
|
||||||
process_directory(input_directory, output_directory)
|
process_directory(input_directory, output_directory)
|
||||||
|
|
||||||
flash_size = env.BoardConfig().get("upload.flash_size", "4MB")
|
|
||||||
fs_image_name = f"littlefs_{flash_size}"
|
|
||||||
env.Replace(ESP32_FS_IMAGE_NAME=fs_image_name)
|
|
||||||
env.Replace(ESP8266_FS_IMAGE_NAME=fs_image_name)
|
|
||||||
|
|
||||||
os.environ["PUBLIC_BASE_URL"] = ""
|
os.environ["PUBLIC_BASE_URL"] = ""
|
||||||
fs_name = env.get("ESP32_FS_IMAGE_NAME", "littlefs.bin")
|
env.AddPreAction("$BUILD_DIR/littlefs.bin", before_buildfs)
|
||||||
# Or alternatively:
|
|
||||||
# fs_name = env.get("FSTOOLNAME", "littlefs.bin")
|
|
||||||
|
|
||||||
# Use the variable in the pre-action
|
|
||||||
env.AddPreAction(f"$BUILD_DIR/{fs_name}.bin", before_buildfs)
|
|
||||||
|
|
|
@ -48,8 +48,8 @@ class Listener(ServiceListener):
|
||||||
#arguments = [f"-i {str()} -f -r"]
|
#arguments = [f"-i {str()} -f -r"]
|
||||||
namespace = argparse.Namespace(
|
namespace = argparse.Namespace(
|
||||||
esp_ip=info.parsed_addresses()[0],
|
esp_ip=info.parsed_addresses()[0],
|
||||||
image=f"{os.getcwd()}/.pio/build/lolin_s3_mini_213epd/firmware.bin",
|
image=f"{os.getcwd()}/.pio/build/lolin_s3_mini_qr/firmware.bin",
|
||||||
littlefs=f"{os.getcwd()}/.pio/build/lolin_s3_mini_213epd/littlefs.bin",
|
littlefs=f"{os.getcwd()}/.pio/build/lolin_s3_mini_qr/littlefs.bin",
|
||||||
progress=True
|
progress=True
|
||||||
)
|
)
|
||||||
if (str(info.properties.get(b"version").decode())) != "3.0":
|
if (str(info.properties.get(b"version").decode())) != "3.0":
|
||||||
|
@ -64,7 +64,7 @@ class Listener(ServiceListener):
|
||||||
print("Different version, going to update")
|
print("Different version, going to update")
|
||||||
#espota.serve(namespace.esp_ip, "0.0.0.0", 3232, random.randint(10000,60000), "", namespace.littlefs, SPIFFS)
|
#espota.serve(namespace.esp_ip, "0.0.0.0", 3232, random.randint(10000,60000), "", namespace.littlefs, SPIFFS)
|
||||||
|
|
||||||
espota.serve(namespace.esp_ip, "0.0.0.0", 3232, random.randint(10000,60000), "", namespace.image, FLASH)
|
#espota.serve(namespace.esp_ip, "0.0.0.0", 3232, random.randint(10000,60000), "", namespace.image, FLASH)
|
||||||
#print(arguments)
|
#print(arguments)
|
||||||
|
|
||||||
#logging.basicConfig(level = logging.DEBUG, format = '%(asctime)-8s [%(levelname)s]: %(message)s', datefmt = '%H:%M:%S')
|
#logging.basicConfig(level = logging.DEBUG, format = '%(asctime)-8s [%(levelname)s]: %(message)s', datefmt = '%H:%M:%S')
|
||||||
|
|
|
@ -5,17 +5,4 @@ revision = (
|
||||||
.strip()
|
.strip()
|
||||||
.decode("utf-8")
|
.decode("utf-8")
|
||||||
)
|
)
|
||||||
|
|
||||||
try:
|
|
||||||
tag = (
|
|
||||||
subprocess.check_output(["git", "describe", "--tags", "--exact-match"])
|
|
||||||
.strip()
|
|
||||||
.decode("utf-8")
|
|
||||||
)
|
|
||||||
git_tag_define = '\'-DGIT_TAG=\"%s\"\'' % tag
|
|
||||||
except subprocess.CalledProcessError:
|
|
||||||
git_tag_define = ''
|
|
||||||
|
|
||||||
print("'-DGIT_REV=\"%s\"'" % revision)
|
print("'-DGIT_REV=\"%s\"'" % revision)
|
||||||
if git_tag_define:
|
|
||||||
print(git_tag_define)
|
|
|
@ -1,7 +0,0 @@
|
||||||
Import("env")
|
|
||||||
|
|
||||||
flash_size = env.BoardConfig().get("upload.flash_size", "4MB")
|
|
||||||
fs_image_name = f"littlefs_{flash_size}"
|
|
||||||
env.Replace(ESP32_FS_IMAGE_NAME=fs_image_name)
|
|
||||||
env.Replace(ESP8266_FS_IMAGE_NAME=fs_image_name)
|
|
||||||
|
|
|
@ -7,8 +7,8 @@ CONFIG_MBEDTLS_KEY_EXCHANGE_PSK=y
|
||||||
#CONFIG_FREERTOS_USE_TRACE_FACILITY=y
|
#CONFIG_FREERTOS_USE_TRACE_FACILITY=y
|
||||||
#CONFIG_FREERTOS_USE_STATS_FORMATTING_FUNCTIONS=y
|
#CONFIG_FREERTOS_USE_STATS_FORMATTING_FUNCTIONS=y
|
||||||
#CONFIG_FREERTOS_GENERATE_RUN_TIME_STATS=n
|
#CONFIG_FREERTOS_GENERATE_RUN_TIME_STATS=n
|
||||||
#CONFIG_ESP_TLS_INSECURE=y
|
CONFIG_ESP_TLS_INSECURE=y
|
||||||
#CONFIG_ESP_TLS_SKIP_SERVER_CERT_VERIFY=y
|
CONFIG_ESP_TLS_SKIP_SERVER_CERT_VERIFY=y
|
||||||
|
|
||||||
CONFIG_HEAP_CORRUPTION_DETECTION=CONFIG_HEAP_POISONING_LIGHT
|
CONFIG_HEAP_CORRUPTION_DETECTION=CONFIG_HEAP_POISONING_LIGHT
|
||||||
CONFIG_HEAP_POISONING_LIGHT=y
|
CONFIG_HEAP_POISONING_LIGHT=y
|
||||||
|
@ -16,14 +16,18 @@ CONFIG_HEAP_POISONING_LIGHT=y
|
||||||
CONFIG_ESP32S3_SPIRAM_SUPPORT=y
|
CONFIG_ESP32S3_SPIRAM_SUPPORT=y
|
||||||
CONFIG_SPIRAM_TRY_ALLOCATE_WIFI_LWIP=y
|
CONFIG_SPIRAM_TRY_ALLOCATE_WIFI_LWIP=y
|
||||||
CONFIG_BOOTLOADER_LOG_LEVEL=0
|
CONFIG_BOOTLOADER_LOG_LEVEL=0
|
||||||
|
CONFIG_LOG_BOOTLOADER_LEVEL_NONE=y
|
||||||
CONFIG_BOOTLOADER_LOG_LEVEL_NONE=y
|
CONFIG_BOOTLOADER_LOG_LEVEL_NONE=y
|
||||||
CONFIG_ESP32S3_SPIRAM_SUPPORT=y
|
CONFIG_ESP32S3_SPIRAM_SUPPORT=y
|
||||||
CONFIG_LOG_DEFAULT_LEVEL_NONE=y
|
CONFIG_LOG_DEFAULT_LEVEL_NONE=y
|
||||||
CONFIG_LOG_DEFAULT_LEVEL=0
|
CONFIG_LOG_DEFAULT_LEVEL=0
|
||||||
CONFIG_LOG_MAXIMUM_LEVEL=0
|
CONFIG_LOG_MAXIMUM_LEVEL=0
|
||||||
|
CONFIG_LOG_BOOTLOADER_LEVEL_NONE=y
|
||||||
|
CONFIG_LOG_BOOTLOADER_LEVEL=0
|
||||||
|
CONFIG_CXX_EXCEPTIONS=y
|
||||||
CONFIG_COMPILER_CXX_EXCEPTIONS=y
|
CONFIG_COMPILER_CXX_EXCEPTIONS=y
|
||||||
#CONFIG_BOOTLOADER_WDT_ENABLE=n
|
#CONFIG_BOOTLOADER_WDT_ENABLE=n
|
||||||
#CONFIG_ESP_TASK_WDT=n
|
#CONFIG_TASK_WDT=n
|
||||||
|
|
||||||
#Required for BTClock
|
#Required for BTClock
|
||||||
#CONFIG_SPIRAM_MODE_OCT=y
|
#CONFIG_SPIRAM_MODE_OCT=y
|
||||||
|
@ -38,11 +42,12 @@ CONFIG_ESP_WIFI_DYNAMIC_TX_BUFFER_NUM=12
|
||||||
CONFIG_ESP32_WIFI_RX_BA_WIN=6
|
CONFIG_ESP32_WIFI_RX_BA_WIN=6
|
||||||
|
|
||||||
CONFIG_FREERTOS_TIMER_TASK_STACK_DEPTH=3120
|
CONFIG_FREERTOS_TIMER_TASK_STACK_DEPTH=3120
|
||||||
|
CONFIG_ESP_DEFAULT_CPU_FREQ_MHZ_240
|
||||||
CONFIG_ESP_DEFAULT_CPU_FREQ_MHZ_240=y
|
CONFIG_ESP_DEFAULT_CPU_FREQ_MHZ_240=y
|
||||||
CONFIG_RTC_CLK_CAL_CYCLES=576
|
CONFIG_RTC_CLK_CAL_CYCLES=576
|
||||||
CONFIG_FREERTOS_WATCHPOINT_END_OF_STACK=y
|
CONFIG_FREERTOS_WATCHPOINT_END_OF_STACK=y
|
||||||
CONFIG_FREERTOS_TIMER_TASK_STACK_DEPTH=3120
|
CONFIG_FREERTOS_TIMER_TASK_STACK_DEPTH=3120
|
||||||
CONFIG_ESP_SYSTEM_MEMPROT_FEATURE=n
|
CONFIG_ESP_SYSTEM_MEMPROT_FEATURE=n
|
||||||
CONFIG_SPIRAM_CACHE_WORKAROUND=y
|
CONFIG_SPIRAM_CACHE_WORKAROUND=y
|
||||||
CONFIG_COMPILER_OPTIMIZATION_SIZE=y
|
CONFIG_COMPILER_OPTIMIZATION_PERF=y
|
||||||
#CONFIG_NEWLIB_NANO_FORMAT=y
|
CONFIG_COMPILER_OPTIMIZATION_LEVEL_RELEASE=y
|
|
@ -1526,78 +1526,7 @@ const uint8_t Antonio_SemiBold40pt7bBitmaps[] PROGMEM = {
|
||||||
0x3F, 0xE0, 0x3F, 0xE0, 0xFF, 0xC0, 0x7F, 0xFF, 0xFF, 0x80, 0xFF, 0xFF,
|
0x3F, 0xE0, 0x3F, 0xE0, 0xFF, 0xC0, 0x7F, 0xFF, 0xFF, 0x80, 0xFF, 0xFF,
|
||||||
0xFE, 0x00, 0xFF, 0xFF, 0xFC, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0xFF, 0xFF,
|
0xFE, 0x00, 0xFF, 0xFF, 0xFC, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0xFF, 0xFF,
|
||||||
0xC0, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0xFF, 0xF8, 0x00, 0x00, 0x1F, 0x80,
|
0xC0, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0xFF, 0xF8, 0x00, 0x00, 0x1F, 0x80,
|
||||||
0x00,
|
0x00};
|
||||||
// euro
|
|
||||||
0x00, 0x07, 0xFC, 0x00, 0x00, 0x3F, 0xFF, 0x00, 0x01, 0xFF, 0xFF, 0x00,
|
|
||||||
0x07, 0xFF, 0xFF, 0x00, 0x1F, 0xFF, 0xFF, 0x00, 0x3F, 0xFF, 0xFF, 0x00,
|
|
||||||
0xFF, 0xFF, 0xFE, 0x01, 0xFF, 0xFF, 0xFE, 0x07, 0xFE, 0x07, 0xFC, 0x0F,
|
|
||||||
0xF8, 0x0F, 0xF8, 0x1F, 0xF0, 0x0F, 0xF8, 0x3F, 0xE0, 0x1F, 0xF0, 0xFF,
|
|
||||||
0x80, 0x3F, 0xE1, 0xFF, 0x00, 0x7F, 0xC3, 0xFE, 0x00, 0xFF, 0x87, 0xFC,
|
|
||||||
0x00, 0xFF, 0x0F, 0xF8, 0x01, 0xFE, 0x1F, 0xF0, 0x03, 0xFC, 0x3F, 0xE0,
|
|
||||||
0x07, 0xF8, 0x7F, 0xC0, 0x0F, 0xF0, 0xFF, 0x80, 0x1F, 0xE1, 0xFF, 0x00,
|
|
||||||
0x3F, 0xC3, 0xFE, 0x00, 0x7F, 0x87, 0xFC, 0x00, 0xFF, 0x0F, 0xF8, 0x01,
|
|
||||||
0xFE, 0x1F, 0xF0, 0x03, 0xFF, 0xFF, 0xFF, 0xC7, 0xFF, 0xFF, 0xFF, 0x80,
|
|
||||||
0x0F, 0xFF, 0xFF, 0x00, 0x1F, 0xFF, 0xFE, 0x00, 0x3F, 0xFF, 0xFC, 0x00,
|
|
||||||
0x07, 0xFC, 0x00, 0x00, 0x0F, 0xF8, 0x00, 0x00, 0x1F, 0xF0, 0x00, 0x00,
|
|
||||||
0x3F, 0xE0, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x0F, 0xFF, 0xFF, 0x00, 0x1F,
|
|
||||||
0xFF, 0xFE, 0x00, 0x3F, 0xFF, 0xFC, 0x00, 0x7F, 0xFF, 0xF8, 0x00, 0x0F,
|
|
||||||
0xF8, 0x00, 0x00, 0x1F, 0xF0, 0x03, 0xFC, 0x3F, 0xE0, 0x07, 0xF8, 0x7F,
|
|
||||||
0xC0, 0x0F, 0xF0, 0xFF, 0x80, 0x1F, 0xE1, 0xFF, 0x00, 0x3F, 0xC3, 0xFE,
|
|
||||||
0x00, 0x7F, 0x87, 0xFC, 0x00, 0xFF, 0x0F, 0xF8, 0x01, 0xFE, 0x1F, 0xF0,
|
|
||||||
0x03, 0xFC, 0x3F, 0xE0, 0x07, 0xF8, 0x7F, 0xC0, 0x0F, 0xF0, 0xFF, 0x80,
|
|
||||||
0x1F, 0xE1, 0xFF, 0x00, 0x7F, 0xC3, 0xFE, 0x00, 0xFF, 0x87, 0xFC, 0x01,
|
|
||||||
0xFF, 0x0F, 0xF8, 0x03, 0xFE, 0x0F, 0xF8, 0x07, 0xFC, 0x1F, 0xF0, 0x0F,
|
|
||||||
0xF0, 0x3F, 0xE0, 0x3F, 0xE0, 0x3F, 0xE0, 0xFF, 0xC0, 0x7F, 0xFF, 0xFF,
|
|
||||||
0x80, 0xFF, 0xFF, 0xFE, 0x00, 0xFF, 0xFF, 0xFC, 0x00, 0xFF, 0xFF, 0xF0,
|
|
||||||
0x00, 0xFF, 0xFF, 0xC0, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0xFF, 0xF8, 0x00,
|
|
||||||
0x00, 0x3F, 0x80, 0x00,
|
|
||||||
// pound
|
|
||||||
0x00, 0x0F, 0xFC, 0x00, 0x01, 0xFF, 0xFC, 0x00, 0x0F, 0xFF, 0xFC, 0x00,
|
|
||||||
0x7F, 0xFF, 0xF8, 0x03, 0xFF, 0xFF, 0xE0, 0x1F, 0xFF, 0xFF, 0xC0, 0x7F,
|
|
||||||
0xFF, 0xFF, 0x83, 0xFF, 0x03, 0xFE, 0x0F, 0xF8, 0x07, 0xF8, 0x3F, 0xC0,
|
|
||||||
0x1F, 0xE1, 0xFF, 0x00, 0x3F, 0xC7, 0xFC, 0x00, 0xFF, 0x1F, 0xE0, 0x03,
|
|
||||||
0xFC, 0x7F, 0x80, 0x0F, 0xF1, 0xFE, 0x00, 0x3F, 0xC7, 0xF8, 0x00, 0xFF,
|
|
||||||
0x1F, 0xF0, 0x03, 0xFC, 0x7F, 0xC0, 0x0F, 0xF1, 0xFF, 0x00, 0x3F, 0xC3,
|
|
||||||
0xFC, 0x00, 0xFF, 0x0F, 0xF0, 0x00, 0x00, 0x3F, 0xE0, 0x00, 0x00, 0xFF,
|
|
||||||
0x80, 0x00, 0x01, 0xFE, 0x00, 0x00, 0x07, 0xF8, 0x00, 0x00, 0x1F, 0xF0,
|
|
||||||
0x00, 0x00, 0x7F, 0xC0, 0x00, 0x00, 0xFF, 0x00, 0x00, 0x03, 0xFC, 0x00,
|
|
||||||
0x00, 0x0F, 0xF8, 0x00, 0x00, 0x3F, 0xE0, 0x00, 0x3F, 0xFF, 0xFF, 0xF8,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xE3, 0xFF, 0xFF, 0xFF, 0x8F, 0xFF, 0xFF, 0xFE, 0x3F,
|
|
||||||
0xFF, 0xFF, 0xF8, 0xFF, 0xFF, 0xFF, 0xE0, 0x03, 0xFE, 0x00, 0x00, 0x0F,
|
|
||||||
0xF8, 0x00, 0x00, 0x3F, 0xE0, 0x00, 0x00, 0x7F, 0x80, 0x00, 0x01, 0xFE,
|
|
||||||
0x00, 0x00, 0x07, 0xFC, 0x00, 0x00, 0x1F, 0xF0, 0x00, 0x00, 0x7F, 0xC0,
|
|
||||||
0x00, 0x01, 0xFF, 0x00, 0x00, 0x07, 0xFC, 0x00, 0x00, 0x1F, 0xF0, 0x00,
|
|
||||||
0x00, 0x7F, 0xC0, 0x00, 0x01, 0xFF, 0x00, 0x00, 0x07, 0xFC, 0x00, 0x00,
|
|
||||||
0x1F, 0xF0, 0x00, 0x00, 0x7F, 0xC0, 0x00, 0x03, 0xFE, 0x00, 0x00, 0x0F,
|
|
||||||
0xF8, 0x00, 0x00, 0x3F, 0xE0, 0x00, 0x01, 0xFF, 0x00, 0x00, 0x0F, 0xF8,
|
|
||||||
0x00, 0x00, 0x7F, 0xC0, 0x00, 0x03, 0xFE, 0x00, 0x00, 0x1F, 0xF0, 0x00,
|
|
||||||
0x01, 0xFF, 0xFF, 0xFF, 0xF7, 0xFF, 0xFF, 0xFF, 0xDF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0x7F, 0xFF, 0xFF, 0xFD, 0xFF, 0xFF, 0xFF, 0xF7, 0xFF, 0xFF, 0xFF, 0xDF,
|
|
||||||
0xFF, 0xFF, 0xFF,
|
|
||||||
// yen
|
|
||||||
0xFF, 0x00, 0x03, 0xFF, 0xFF, 0x00, 0x07, 0xFB, 0xFE, 0x00, 0x1F, 0xF3,
|
|
||||||
0xFC, 0x00, 0x3F, 0xE7, 0xFC, 0x00, 0x7F, 0xCF, 0xF8, 0x00, 0xFF, 0x0F,
|
|
||||||
0xF0, 0x03, 0xFE, 0x1F, 0xE0, 0x07, 0xFC, 0x3F, 0xE0, 0x0F, 0xF0, 0x7F,
|
|
||||||
0xC0, 0x1F, 0xE0, 0x7F, 0x80, 0x7F, 0xC0, 0xFF, 0x80, 0xFF, 0x81, 0xFF,
|
|
||||||
0x01, 0xFE, 0x01, 0xFE, 0x03, 0xFC, 0x03, 0xFC, 0x07, 0xF8, 0x07, 0xFC,
|
|
||||||
0x1F, 0xF0, 0x0F, 0xF8, 0x3F, 0xC0, 0x0F, 0xF0, 0x7F, 0x80, 0x1F, 0xE0,
|
|
||||||
0xFF, 0x00, 0x3F, 0xE3, 0xFC, 0x00, 0x3F, 0xC7, 0xF8, 0x00, 0x7F, 0x8F,
|
|
||||||
0xF0, 0x00, 0xFF, 0x9F, 0xE0, 0x00, 0xFF, 0x7F, 0x80, 0x01, 0xFE, 0xFF,
|
|
||||||
0x00, 0x03, 0xFD, 0xFE, 0x00, 0x07, 0xFF, 0xFC, 0x00, 0x07, 0xFF, 0xF0,
|
|
||||||
0x00, 0x0F, 0xFF, 0xE0, 0x00, 0x1F, 0xFF, 0xC0, 0x00, 0x1F, 0xFF, 0x00,
|
|
||||||
0x00, 0x3F, 0xFE, 0x00, 0x00, 0x7F, 0xFC, 0x00, 0x00, 0xFF, 0xF8, 0x00,
|
|
||||||
0x00, 0xFF, 0xE0, 0x00, 0x01, 0xFF, 0xC0, 0x00, 0x03, 0xFF, 0x80, 0x07,
|
|
||||||
0xFF, 0xFF, 0xFE, 0x0F, 0xFF, 0xFF, 0xFC, 0x1F, 0xFF, 0xFF, 0xF8, 0x3F,
|
|
||||||
0xFF, 0xFF, 0xF0, 0x7F, 0xFF, 0xFF, 0xE0, 0x00, 0x7F, 0xC0, 0x00, 0x00,
|
|
||||||
0xFF, 0x80, 0x00, 0x01, 0xFF, 0x00, 0x00, 0x03, 0xFE, 0x00, 0x00, 0x07,
|
|
||||||
0xFC, 0x00, 0x00, 0x0F, 0xF8, 0x00, 0x3F, 0xFF, 0xFF, 0xF0, 0x7F, 0xFF,
|
|
||||||
0xFF, 0xE0, 0xFF, 0xFF, 0xFF, 0xC1, 0xFF, 0xFF, 0xFF, 0x83, 0xFF, 0xFF,
|
|
||||||
0xFF, 0x00, 0x03, 0xFE, 0x00, 0x00, 0x07, 0xFC, 0x00, 0x00, 0x0F, 0xF8,
|
|
||||||
0x00, 0x00, 0x1F, 0xF0, 0x00, 0x00, 0x3F, 0xE0, 0x00, 0x00, 0x7F, 0xC0,
|
|
||||||
0x00, 0x00, 0xFF, 0x80, 0x00, 0x01, 0xFF, 0x00, 0x00, 0x03, 0xFE, 0x00,
|
|
||||||
0x00, 0x07, 0xFC, 0x00, 0x00, 0x0F, 0xF8, 0x00, 0x00, 0x1F, 0xF0, 0x00,
|
|
||||||
0x00, 0x3F, 0xE0, 0x00, 0x00, 0x7F, 0xC0, 0x00
|
|
||||||
};
|
|
||||||
|
|
||||||
const GFXglyph Antonio_SemiBold40pt7bGlyphs[] PROGMEM = {
|
const GFXglyph Antonio_SemiBold40pt7bGlyphs[] PROGMEM = {
|
||||||
{0, 1, 1, 17, 0, 0}, // 0x20 ' '
|
{0, 1, 1, 17, 0, 0}, // 0x20 ' '
|
||||||
|
@ -1659,10 +1588,10 @@ const GFXglyph Antonio_SemiBold40pt7bGlyphs[] PROGMEM = {
|
||||||
{10963, 28, 67, 32, 2, -66}, // 0x58 'X'
|
{10963, 28, 67, 32, 2, -66}, // 0x58 'X'
|
||||||
{11198, 31, 67, 32, 1, -66}, // 0x59 'Y'
|
{11198, 31, 67, 32, 1, -66}, // 0x59 'Y'
|
||||||
{11458, 23, 67, 27, 3, -66}, // 0x5A 'Z'
|
{11458, 23, 67, 27, 3, -66}, // 0x5A 'Z'
|
||||||
{18021, 31, 69, 37, 2, -67}, // 0x5B '[' --> euro { 18290, 31, 69, 37, 2, -67 } was {11651, 17, 70, 26, 6, -66}
|
{11651, 17, 70, 26, 6, -66}, // 0x5B '['
|
||||||
{11800, 24, 67, 30, 3, -66}, // 0x5C '\'
|
{11800, 24, 67, 30, 3, -66}, // 0x5C '\'
|
||||||
{18557, 30, 68, 36, 3, -67 }, // 0x5D ']' --> pound { 0, 30, 68, 36, 3, -67 } was {12001, 16, 70, 26, 4, -66}
|
{12001, 16, 70, 26, 4, -66}, // 0x5D ']'
|
||||||
{18812, 31, 67, 32, 1, -66 }, // 0x5E '^' --> yen { 0, 31, 67, 32, 1, -66 } was {12141, 29, 35, 37, 4, -66
|
{12141, 29, 35, 37, 4, -66}, // 0x5E '^'
|
||||||
{12268, 25, 7, 29, 2, 2}, // 0x5F '_'
|
{12268, 25, 7, 29, 2, 2}, // 0x5F '_'
|
||||||
{12290, 12, 15, 16, 2, -77}, // 0x60 '`'
|
{12290, 12, 15, 16, 2, -77}, // 0x60 '`'
|
||||||
{12313, 27, 59, 36, 4, -57}, // 0x61 'a'
|
{12313, 27, 59, 36, 4, -57}, // 0x61 'a'
|
||||||
|
@ -1694,13 +1623,10 @@ const GFXglyph Antonio_SemiBold40pt7bGlyphs[] PROGMEM = {
|
||||||
{17509, 20, 75, 27, 4, -67}, // 0x7B '{'
|
{17509, 20, 75, 27, 4, -67}, // 0x7B '{'
|
||||||
{17697, 9, 75, 21, 6, -70}, // 0x7C '|'
|
{17697, 9, 75, 21, 6, -70}, // 0x7C '|'
|
||||||
{17782, 19, 75, 27, 4, -67}, // 0x7D '}'
|
{17782, 19, 75, 27, 4, -67}, // 0x7D '}'
|
||||||
{17961, 34, 14, 43, 4, -44}}; // 0x7E '~'
|
{17961, 34, 14, 43, 4, -44}, {18021, 31, 69, 37, 2, -67}}; // 0x7E '~'
|
||||||
|
|
||||||
|
|
||||||
//, {18021, 31, 69, 37, 2, -67}
|
|
||||||
|
|
||||||
const GFXfont Antonio_SemiBold40pt7b PROGMEM = {
|
const GFXfont Antonio_SemiBold40pt7b PROGMEM = {
|
||||||
(uint8_t *)Antonio_SemiBold40pt7bBitmaps,
|
(uint8_t *)Antonio_SemiBold40pt7bBitmaps,
|
||||||
(GFXglyph *)Antonio_SemiBold40pt7bGlyphs, 0x20, 0x7E, 100};
|
(GFXglyph *)Antonio_SemiBold40pt7bGlyphs, 0x20, 0x7E, 101};
|
||||||
|
|
||||||
// Approx. 18961 bytes
|
// Approx. 18961 bytes
|
||||||
|
|
|
@ -5029,231 +5029,7 @@ const uint8_t Antonio_SemiBold90pt7bBitmaps[] PROGMEM = {
|
||||||
0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00,
|
0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00,
|
||||||
0x00, 0x1F, 0xFF, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF,
|
0x00, 0x1F, 0xFF, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF,
|
||||||
0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0x80, 0x00, 0x00,
|
0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0x80, 0x00, 0x00,
|
||||||
0x00,
|
0x00};
|
||||||
|
|
||||||
// Pound sign (start 60325)
|
|
||||||
0x00, 0x00, 0x00, 0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F,
|
|
||||||
0xFF, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF, 0xFF, 0xE0,
|
|
||||||
0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x03,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x00, 0x3F, 0xFF, 0xFF, 0xFF, 0xFF, 0xF8, 0x00, 0x00,
|
|
||||||
0x0F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x80, 0x00, 0x03, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xF0, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00,
|
|
||||||
0x00, 0x3F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF0, 0x00, 0x07, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFE, 0x00, 0x01, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xE0, 0x00, 0x7F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFC, 0x00, 0x0F, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xC0, 0x03, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xF8, 0x00, 0x7F, 0xFF, 0xFF, 0x00, 0x1F, 0xFF, 0xFF, 0x80, 0x0F,
|
|
||||||
0xFF, 0xFF, 0x80, 0x00, 0xFF, 0xFF, 0xF0, 0x03, 0xFF, 0xFF, 0xE0, 0x00,
|
|
||||||
0x0F, 0xFF, 0xFE, 0x00, 0x7F, 0xFF, 0xF8, 0x00, 0x00, 0xFF, 0xFF, 0xE0,
|
|
||||||
0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x01, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x00, 0x01, 0xFF, 0xFF, 0x80, 0x7F, 0xFF, 0xF8, 0x00, 0x00, 0x3F, 0xFF,
|
|
||||||
0xF0, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x07, 0xFF, 0xFE, 0x01, 0xFF, 0xFF,
|
|
||||||
0xC0, 0x00, 0x00, 0x7F, 0xFF, 0xC0, 0x3F, 0xFF, 0xF8, 0x00, 0x00, 0x0F,
|
|
||||||
0xFF, 0xFC, 0x07, 0xFF, 0xFF, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x80, 0xFF,
|
|
||||||
0xFF, 0xE0, 0x00, 0x00, 0x3F, 0xFF, 0xF0, 0x1F, 0xFF, 0xF8, 0x00, 0x00,
|
|
||||||
0x07, 0xFF, 0xFE, 0x03, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x1F, 0xFF, 0xF8, 0x0F, 0xFF, 0xFC, 0x00,
|
|
||||||
0x00, 0x01, 0xFF, 0xFF, 0x01, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x3F, 0xFF,
|
|
||||||
0xE0, 0x3F, 0xFF, 0xF0, 0x00, 0x00, 0x07, 0xFF, 0xFC, 0x07, 0xFF, 0xFE,
|
|
||||||
0x00, 0x00, 0x00, 0xFF, 0xFF, 0x80, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x1F,
|
|
||||||
0xFF, 0xF0, 0x1F, 0xFF, 0xFC, 0x00, 0x00, 0x03, 0xFF, 0xFE, 0x03, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x00, 0x7F, 0xFF, 0xC0, 0x7F, 0xFF, 0xF0, 0x00, 0x00,
|
|
||||||
0x0F, 0xFF, 0xF8, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x01,
|
|
||||||
0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x3F, 0xFF, 0xE0, 0x3F, 0xFF, 0xF8, 0x00,
|
|
||||||
0x00, 0x07, 0xFF, 0xFC, 0x03, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0xFF, 0xFF,
|
|
||||||
0x80, 0x7F, 0xFF, 0xF0, 0x00, 0x00, 0x1F, 0xFF, 0xF0, 0x0F, 0xFF, 0xFE,
|
|
||||||
0x00, 0x00, 0x03, 0xFF, 0xFE, 0x01, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x7F,
|
|
||||||
0xFF, 0xC0, 0x3F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF8, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x3F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF,
|
|
||||||
0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF8, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFE, 0x07, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xC0, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF8, 0x1F,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x03, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xE0, 0x7F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFC,
|
|
||||||
0x0F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x81, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xF0, 0x3F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFE, 0x07, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xC0, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF8, 0x1F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0x03, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xE0, 0x7F,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFC, 0x00, 0x00, 0x03, 0xFF, 0xFF,
|
|
||||||
0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x0F, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF,
|
|
||||||
0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x7F, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x1F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF,
|
|
||||||
0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF,
|
|
||||||
0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xFC, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x7F, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x3F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF8, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x01, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xFC,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF,
|
|
||||||
0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F,
|
|
||||||
0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x3F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x3F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF,
|
|
||||||
0x80, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0xFF, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF,
|
|
||||||
0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x01, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF,
|
|
||||||
0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xFE, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x07,
|
|
||||||
0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFC, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0x9F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF3, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFE, 0x7F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xCF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF9, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x3F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xE7, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFC, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x9F, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xF3, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFE,
|
|
||||||
0x7F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xCF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xF9, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0x3F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xE0,
|
|
||||||
|
|
||||||
// yen-sign (start 61607)
|
|
||||||
0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFE, 0xFF, 0xFF, 0xE0,
|
|
||||||
0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF9, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00,
|
|
||||||
0xFF, 0xFF, 0xF1, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0xE3,
|
|
||||||
0xFF, 0xFF, 0x80, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x87, 0xFF, 0xFF, 0x00,
|
|
||||||
0x00, 0x00, 0x07, 0xFF, 0xFF, 0x0F, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x1F,
|
|
||||||
0xFF, 0xFE, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF8, 0x1F,
|
|
||||||
0xFF, 0xFC, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x3F, 0xFF, 0xFC, 0x00,
|
|
||||||
0x00, 0x00, 0xFF, 0xFF, 0xE0, 0x3F, 0xFF, 0xF8, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0xFF, 0xC0, 0x7F, 0xFF, 0xF0, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x00, 0xFF,
|
|
||||||
0xFF, 0xE0, 0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x01, 0xFF, 0xFF, 0xE0, 0x00,
|
|
||||||
0x00, 0x1F, 0xFF, 0xFC, 0x01, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x7F, 0xFF,
|
|
||||||
0xF8, 0x03, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0xFF, 0xFF, 0xE0, 0x07, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x01, 0xFF, 0xFF, 0xC0, 0x07, 0xFF, 0xFF, 0x00, 0x00,
|
|
||||||
0x03, 0xFF, 0xFF, 0x80, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x0F, 0xFF, 0xFE,
|
|
||||||
0x00, 0x1F, 0xFF, 0xFC, 0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x00, 0x3F, 0xFF,
|
|
||||||
0xFC, 0x00, 0x00, 0x3F, 0xFF, 0xF8, 0x00, 0x3F, 0xFF, 0xF8, 0x00, 0x00,
|
|
||||||
0x7F, 0xFF, 0xF0, 0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x01, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x00, 0xFF, 0xFF, 0xE0, 0x00, 0x03, 0xFF, 0xFF, 0x80, 0x00, 0xFF, 0xFF,
|
|
||||||
0xE0, 0x00, 0x07, 0xFF, 0xFF, 0x00, 0x01, 0xFF, 0xFF, 0xC0, 0x00, 0x0F,
|
|
||||||
0xFF, 0xFE, 0x00, 0x03, 0xFF, 0xFF, 0x80, 0x00, 0x3F, 0xFF, 0xF8, 0x00,
|
|
||||||
0x07, 0xFF, 0xFF, 0x80, 0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x07, 0xFF, 0xFF,
|
|
||||||
0x00, 0x00, 0xFF, 0xFF, 0xE0, 0x00, 0x0F, 0xFF, 0xFE, 0x00, 0x01, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x1F, 0xFF, 0xFC, 0x00, 0x03, 0xFF, 0xFF, 0x00, 0x00,
|
|
||||||
0x1F, 0xFF, 0xFC, 0x00, 0x0F, 0xFF, 0xFE, 0x00, 0x00, 0x3F, 0xFF, 0xF8,
|
|
||||||
0x00, 0x1F, 0xFF, 0xFC, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x3F, 0xFF,
|
|
||||||
0xF0, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x00, 0x7F, 0xFF, 0xE0, 0x00, 0x00,
|
|
||||||
0xFF, 0xFF, 0xE0, 0x01, 0xFF, 0xFF, 0xC0, 0x00, 0x01, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x03, 0xFF, 0xFF, 0x80, 0x00, 0x03, 0xFF, 0xFF, 0x80, 0x07, 0xFF, 0xFE,
|
|
||||||
0x00, 0x00, 0x03, 0xFF, 0xFF, 0x80, 0x0F, 0xFF, 0xFC, 0x00, 0x00, 0x07,
|
|
||||||
0xFF, 0xFF, 0x00, 0x3F, 0xFF, 0xF8, 0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x00,
|
|
||||||
0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x0F, 0xFF, 0xFC, 0x00, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x01, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x3F,
|
|
||||||
0xFF, 0xF8, 0x07, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x0F,
|
|
||||||
0xFF, 0xFC, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF0, 0x1F, 0xFF, 0xF8, 0x00,
|
|
||||||
0x00, 0x00, 0xFF, 0xFF, 0xE0, 0x3F, 0xFF, 0xF0, 0x00, 0x00, 0x01, 0xFF,
|
|
||||||
0xFF, 0xC0, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x81, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x83, 0xFF, 0xFF, 0x00, 0x00,
|
|
||||||
0x00, 0x07, 0xFF, 0xFF, 0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x0F, 0xFF,
|
|
||||||
0xFE, 0x1F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFE, 0x3F, 0xFF,
|
|
||||||
0xF0, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFC, 0x7F, 0xFF, 0xE0, 0x00, 0x00,
|
|
||||||
0x00, 0x3F, 0xFF, 0xF8, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x3F, 0xFF,
|
|
||||||
0xF3, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF7, 0xFF, 0xFE,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xEF, 0xFF, 0xFC, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0xFF, 0xFF, 0xDF, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00,
|
|
||||||
0x07, 0xFF, 0xFF, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFF, 0xFF, 0xF8, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x3F, 0xFF, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xF0, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0xFF, 0xE0, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x07, 0xFF, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x0F,
|
|
||||||
0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFF, 0xFE,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFF, 0xFC, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x3F, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F,
|
|
||||||
0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x01, 0xFF, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0xFF, 0xFC, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFC, 0x00, 0x3F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF8, 0x00, 0x7F,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF0, 0x00, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xE0, 0x01, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xC0, 0x03, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x80, 0x07, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x0F, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFE, 0x00, 0x1F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFC,
|
|
||||||
0x00, 0x3F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF8, 0x00, 0x7F, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF0, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFE, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x3F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x7F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xF0,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x03, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07,
|
|
||||||
0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x3F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F,
|
|
||||||
0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xF0, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x03, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xF0, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xE0, 0x01, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xC0, 0x03, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x80, 0x07, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0x00, 0x0F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFE,
|
|
||||||
0x00, 0x1F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFC, 0x00, 0x3F, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xF8, 0x00, 0x7F, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xF0, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xE0,
|
|
||||||
0x01, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00,
|
|
||||||
0x3F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF8,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x01, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03,
|
|
||||||
0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x1F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F,
|
|
||||||
0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF8, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x01, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x1F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF,
|
|
||||||
0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF8, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x01, 0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF,
|
|
||||||
0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x1F, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xFC,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xF8, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xFF, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x0F, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00
|
|
||||||
};
|
|
||||||
|
|
||||||
const GFXglyph Antonio_SemiBold90pt7bGlyphs[] PROGMEM = {
|
const GFXglyph Antonio_SemiBold90pt7bGlyphs[] PROGMEM = {
|
||||||
{0, 1, 1, 38, 0, 0}, // 0x20 ' '
|
{0, 1, 1, 38, 0, 0}, // 0x20 ' '
|
||||||
|
@ -5270,7 +5046,7 @@ const GFXglyph Antonio_SemiBold90pt7bGlyphs[] PROGMEM = {
|
||||||
{9988, 53, 55, 63, 5, -94}, // 0x2B '+'
|
{9988, 53, 55, 63, 5, -94}, // 0x2B '+'
|
||||||
{10353, 23, 50, 41, 10, -20}, // 0x2C ','
|
{10353, 23, 50, 41, 10, -20}, // 0x2C ','
|
||||||
{10497, 39, 14, 60, 8, -75}, // 0x2D '-'
|
{10497, 39, 14, 60, 8, -75}, // 0x2D '-'
|
||||||
{10566, 17, 19, 46, 14, 106}, // 0x2E '.'
|
{10566, 17, 19, 46, 14, -18}, // 0x2E '.'
|
||||||
{10607, 55, 152, 66, 6, 105}, // 0x2F '/'
|
{10607, 55, 152, 66, 6, 105}, // 0x2F '/'
|
||||||
{11652, 59, 155, 79, 10, 104}, // 0x30 '0'
|
{11652, 59, 155, 79, 10, 104}, // 0x30 '0'
|
||||||
{12796, 41, 151, 67, 8, 106}, // 0x31 '1'
|
{12796, 41, 151, 67, 8, 106}, // 0x31 '1'
|
||||||
|
@ -5314,16 +5090,11 @@ const GFXglyph Antonio_SemiBold90pt7bGlyphs[] PROGMEM = {
|
||||||
{53398, 108, 151, 116, 4, 106}, // 0x57 'W'
|
{53398, 108, 151, 116, 4, 106}, // 0x57 'W'
|
||||||
{55437, 64, 151, 72, 4, 106}, // 0x58 'X'
|
{55437, 64, 151, 72, 4, 106}, // 0x58 'X'
|
||||||
{56645, 71, 151, 73, 1, 106}, // 0x59 'Y'
|
{56645, 71, 151, 73, 1, 106}, // 0x59 'Y'
|
||||||
{57986, 52, 151, 61, 6, 106}, // 0x5A 'Z'
|
{57986, 52, 151, 61, 6, 106},
|
||||||
{58968, 70, 155, 84, 5, 104}, // Euro-sign as [
|
{58968, 70, 155, 84, 5, 104}}; // 0x5A 'Z'
|
||||||
{0, 0, 0, 0, 0, 0}, // Skip backslash
|
|
||||||
{60325, 67, 153, 82, 7, 104 }, // Pound-sign as ]
|
|
||||||
{61607, 71, 151, 73, 1, 106 }, // Yen-sign as ^
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
const GFXfont Antonio_SemiBold90pt7b PROGMEM = {
|
const GFXfont Antonio_SemiBold90pt7b PROGMEM = {
|
||||||
(uint8_t *)Antonio_SemiBold90pt7bBitmaps,
|
(uint8_t *)Antonio_SemiBold90pt7bBitmaps,
|
||||||
(GFXglyph *)Antonio_SemiBold90pt7bGlyphs, 0x20, 0x5E, 231};
|
(GFXglyph *)Antonio_SemiBold90pt7bGlyphs, 0x20, 0x5B, 228};
|
||||||
|
|
||||||
// Approx. 60745 bytes
|
// Approx. 60745 bytes
|
||||||
|
|
|
@ -1,12 +1,9 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "antonio-semibold20.h"
|
#include "antonio-semibold20.h"
|
||||||
//#include "antonio-semibold30.h"
|
#include "antonio-semibold30.h"
|
||||||
#include "antonio-semibold40.h"
|
#include "antonio-semibold40.h"
|
||||||
#include "antonio-semibold90.h"
|
#include "antonio-semibold90.h"
|
||||||
#include "sats-symbol.h"
|
|
||||||
//#include "icons.h"
|
|
||||||
|
|
||||||
// #include "oswald-20.h"
|
// #include "oswald-20.h"
|
||||||
// #include "oswald-30.h"
|
// #include "oswald-30.h"
|
||||||
// #include "oswald-90.h"
|
// #include "oswald-90.h"
|
||||||
|
|
|
@ -1,201 +0,0 @@
|
||||||
const uint8_t Satoshi_Symbol90pt7bBitmaps[] PROGMEM = {
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x0F, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x03, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F,
|
|
||||||
0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFE, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0x80, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x3F, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x0F, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x03, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF,
|
|
||||||
0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xF8, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFE, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x3F, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x0F, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x3F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x3F, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x0F, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xE0, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xF8, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x03, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x3F, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFE,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0x80, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xE0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x0F, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x03, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFE,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFC, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x1F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x3F, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xE0,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x01, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03,
|
|
||||||
0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFE, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x1F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F,
|
|
||||||
0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xE0, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x01, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF,
|
|
||||||
0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFE, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x1F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF,
|
|
||||||
0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xE0, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x01, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x1F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF0,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
|
|
||||||
0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x0F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F,
|
|
||||||
0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF0, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF,
|
|
||||||
0xFF, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x0F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF,
|
|
||||||
0xF8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF0, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF,
|
|
||||||
0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x0F, 0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xF8,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF0, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0xFF, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x80,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F,
|
|
||||||
0xFF, 0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xF8, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF,
|
|
||||||
0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x80, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF,
|
|
||||||
0xFC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xF8, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF,
|
|
||||||
0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x80, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x03, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x07, 0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFC,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xF8, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x3F, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x7F, 0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xC0,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x03, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07,
|
|
||||||
0xFF, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFC, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xF8, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x3F, 0xFF, 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F,
|
|
||||||
0xFF, 0xE0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xC0, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x80, 0x00, 0x00, 0x00, 0x00,
|
|
||||||
0x00, 0x03, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x07, 0xFF,
|
|
||||||
0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0F, 0xFF, 0xFC, 0x00, 0x00,
|
|
||||||
0x00, 0x00, 0x00, 0x00, 0x1F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
||||||
0xFF, 0xE0 };
|
|
||||||
|
|
||||||
const GFXglyph Satoshi_Symbol90pt7bGlyphs[] PROGMEM = {
|
|
||||||
{ 0, 82, 127, 99, 8, -126 }, { 1302, 71, 109, 93, 0, -117 } }; // 0x53 'S'
|
|
||||||
|
|
||||||
const GFXfont Satoshi_Symbol90pt7b PROGMEM = {
|
|
||||||
(uint8_t *)Satoshi_Symbol90pt7bBitmaps,
|
|
||||||
(GFXglyph *)Satoshi_Symbol90pt7bGlyphs,
|
|
||||||
0x53, 0x53, 192 };
|
|
||||||
|
|
||||||
// Approx. 2284 bytes
|
|
1281
src/icons/icons.cpp
1281
src/icons/icons.cpp
File diff suppressed because it is too large
Load diff
|
@ -1,10 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef ICONS_H
|
|
||||||
#define ICONS_H
|
|
||||||
|
|
||||||
#include <Arduino.h>
|
|
||||||
|
|
||||||
extern const unsigned char* epd_icons_allArray[];
|
|
||||||
|
|
||||||
#endif // ICONS_H
|
|
|
@ -1,3 +1,6 @@
|
||||||
dependencies:
|
dependencies:
|
||||||
# Required IDF version
|
# Required IDF version
|
||||||
idf: ">=4.4"
|
idf: ">=4.4"
|
||||||
|
|
||||||
|
esp_littlefs:
|
||||||
|
git: https://github.com/joltwallet/esp_littlefs.git
|
||||||
|
|
|
@ -1,61 +0,0 @@
|
||||||
#include "bitaxe_fetch.hpp"
|
|
||||||
|
|
||||||
TaskHandle_t bitaxeFetchTaskHandle;
|
|
||||||
|
|
||||||
std::string bitaxeHashrate;
|
|
||||||
std::string bitaxeBestDiff;
|
|
||||||
|
|
||||||
std::string getBitAxeHashRate()
|
|
||||||
{
|
|
||||||
return bitaxeHashrate;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string getBitaxeBestDiff()
|
|
||||||
{
|
|
||||||
return bitaxeBestDiff;
|
|
||||||
}
|
|
||||||
|
|
||||||
void taskBitaxeFetch(void *pvParameters)
|
|
||||||
{
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
|
||||||
|
|
||||||
HTTPClient http;
|
|
||||||
http.setUserAgent(USER_AGENT);
|
|
||||||
String bitaxeApiUrl = "http://" + preferences.getString("bitaxeHostname", DEFAULT_BITAXE_HOSTNAME) + "/api/system/info";
|
|
||||||
http.begin(bitaxeApiUrl.c_str());
|
|
||||||
|
|
||||||
int httpCode = http.GET();
|
|
||||||
|
|
||||||
if (httpCode == 200)
|
|
||||||
{
|
|
||||||
String payload = http.getString();
|
|
||||||
JsonDocument doc;
|
|
||||||
deserializeJson(doc, payload);
|
|
||||||
bitaxeHashrate = std::to_string(static_cast<int>(std::round(doc["hashRate"].as<float>())));
|
|
||||||
bitaxeBestDiff = doc["bestDiff"].as<std::string>();
|
|
||||||
|
|
||||||
if (workQueue != nullptr && (getCurrentScreen() == SCREEN_BITAXE_HASHRATE || getCurrentScreen() == SCREEN_BITAXE_BESTDIFF))
|
|
||||||
{
|
|
||||||
WorkItem priceUpdate = {TASK_BITAXE_UPDATE, 0};
|
|
||||||
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.print(
|
|
||||||
F("Error retrieving BitAxe data. HTTP status code: "));
|
|
||||||
Serial.println(httpCode);
|
|
||||||
Serial.println(bitaxeApiUrl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setupBitaxeFetchTask()
|
|
||||||
{
|
|
||||||
xTaskCreate(taskBitaxeFetch, "bitaxeFetch", (6 * 1024), NULL, tskIDLE_PRIORITY,
|
|
||||||
&bitaxeFetchTaskHandle);
|
|
||||||
|
|
||||||
xTaskNotifyGive(bitaxeFetchTaskHandle);
|
|
||||||
}
|
|
|
@ -1,15 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <Arduino.h>
|
|
||||||
#include <HTTPClient.h>
|
|
||||||
|
|
||||||
#include "lib/config.hpp"
|
|
||||||
#include "lib/shared.hpp"
|
|
||||||
|
|
||||||
extern TaskHandle_t bitaxeFetchTaskHandle;
|
|
||||||
|
|
||||||
void setupBitaxeFetchTask();
|
|
||||||
void taskBitaxeFetch(void *pvParameters);
|
|
||||||
|
|
||||||
std::string getBitAxeHashRate();
|
|
||||||
std::string getBitaxeBestDiff();
|
|
|
@ -2,62 +2,65 @@
|
||||||
|
|
||||||
char *wsServer;
|
char *wsServer;
|
||||||
esp_websocket_client_handle_t blockNotifyClient = NULL;
|
esp_websocket_client_handle_t blockNotifyClient = NULL;
|
||||||
uint currentBlockHeight = 873400;
|
uint currentBlockHeight = 816000;
|
||||||
uint blockMedianFee = 1;
|
|
||||||
bool blockNotifyInit = false;
|
|
||||||
unsigned long int lastBlockUpdate;
|
|
||||||
|
|
||||||
const char *mempoolWsCert = R"EOF(
|
// const char *mempoolWsCert = R"(-----BEGIN CERTIFICATE-----
|
||||||
-----BEGIN CERTIFICATE-----
|
// MIIHfTCCBmWgAwIBAgIRANFX3mhqRYDt1NFuENoSyaAwDQYJKoZIhvcNAQELBQAw
|
||||||
MIIF3jCCA8agAwIBAgIQAf1tMPyjylGoG7xkDjUDLTANBgkqhkiG9w0BAQwFADCB
|
// gZUxCzAJBgNVBAYTAkdCMRswGQYDVQQIExJHcmVhdGVyIE1hbmNoZXN0ZXIxEDAO
|
||||||
iDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCk5ldyBKZXJzZXkxFDASBgNVBAcTC0pl
|
// BgNVBAcTB1NhbGZvcmQxGDAWBgNVBAoTD1NlY3RpZ28gTGltaXRlZDE9MDsGA1UE
|
||||||
cnNleSBDaXR5MR4wHAYDVQQKExVUaGUgVVNFUlRSVVNUIE5ldHdvcmsxLjAsBgNV
|
// AxM0U2VjdGlnbyBSU0EgT3JnYW5pemF0aW9uIFZhbGlkYXRpb24gU2VjdXJlIFNl
|
||||||
BAMTJVVTRVJUcnVzdCBSU0EgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkwHhcNMTAw
|
// cnZlciBDQTAeFw0yMzA3MjQwMDAwMDBaFw0yNDA4MjIyMzU5NTlaMFcxCzAJBgNV
|
||||||
MjAxMDAwMDAwWhcNMzgwMTE4MjM1OTU5WjCBiDELMAkGA1UEBhMCVVMxEzARBgNV
|
// BAYTAkpQMQ4wDAYDVQQIEwVUb2t5bzEgMB4GA1UEChMXTUVNUE9PTCBTUEFDRSBD
|
||||||
BAgTCk5ldyBKZXJzZXkxFDASBgNVBAcTC0plcnNleSBDaXR5MR4wHAYDVQQKExVU
|
// Ty4sIExURC4xFjAUBgNVBAMTDW1lbXBvb2wuc3BhY2UwggEiMA0GCSqGSIb3DQEB
|
||||||
aGUgVVNFUlRSVVNUIE5ldHdvcmsxLjAsBgNVBAMTJVVTRVJUcnVzdCBSU0EgQ2Vy
|
// AQUAA4IBDwAwggEKAoIBAQCqmiPRWgo58d25R0biQjAksXMq5ciH7z7ZQo2w2AbB
|
||||||
dGlmaWNhdGlvbiBBdXRob3JpdHkwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIK
|
// rHxpnlIry74b9S4wRY5UJeYmd6ZwA76NdSioDvxTJc29bLplY+Ftmfc4ET0zYb2k
|
||||||
AoICAQCAEmUXNg7D2wiz0KxXDXbtzSfTTK1Qg2HiqiBNCS1kCdzOiZ/MPans9s/B
|
// Fi86z7GOWb6Ezor/qez9uMM9cxd021Bvcs0/2OrL6Sgp66u9keDZv9NyvFPpXfuR
|
||||||
3PHTsdZ7NygRK0faOca8Ohm0X6a9fZ2jY0K2dvKpOyuR+OJv0OwWIJAJPuLodMkY
|
// tdV2r4HF57VJqZn105PN4k80kNWgDbae8aw+BuUNvQYKEe71yfB7Bh6zSh9pCSfM
|
||||||
tJHUYmTbf6MG8YgYapAiPLz+E/CHFHv25B+O1ORRxhFnRghRy4YUVD+8M/5+bJz/
|
// I6pIJdQzoada2uY1dQMoJeIq8qKNKqAPKGsH5McemUT5ZIKU/tjk3nfX0pz/sQa4
|
||||||
Fp0YvVGONaanZshyZ9shZrHUm3gDwFA66Mzw3LyeTP6vBZY1H1dat//O+T23LLb2
|
// CN7tLH6UeUlctei92GFd6Xtn7RbKLhDUbc4Sq02Cc9iXAgMBAAGjggQDMIID/zAf
|
||||||
VN3I5xI6Ta5MirdcmrS3ID3KfyI0rn47aGYBROcBTkZTmzNg95S+UzeQc0PzMsNT
|
// BgNVHSMEGDAWgBQX2dYlJ2f5McJJQ9kwNkSMbKlP6zAdBgNVHQ4EFgQUXkxoddJ6
|
||||||
79uq/nROacdrjGCT3sTHDN/hMq7MkztReJVni+49Vv4M0GkPGw/zJSZrM233bkf6
|
// rKobsbmDdtuCK1ywXuIwDgYDVR0PAQH/BAQDAgWgMAwGA1UdEwEB/wQCMAAwHQYD
|
||||||
c0Plfg6lZrEpfDKEY1WJxA3Bk1QwGROs0303p+tdOmw1XNtB1xLaqUkL39iAigmT
|
// VR0lBBYwFAYIKwYBBQUHAwEGCCsGAQUFBwMCMEoGA1UdIARDMEEwNQYMKwYBBAGy
|
||||||
Yo61Zs8liM2EuLE/pDkP2QKe6xJMlXzzawWpXhaDzLhn4ugTncxbgtNMs+1b/97l
|
// MQECAQMEMCUwIwYIKwYBBQUHAgEWF2h0dHBzOi8vc2VjdGlnby5jb20vQ1BTMAgG
|
||||||
c6wjOy0AvzVVdAlJ2ElYGn+SNuZRkg7zJn0cTRe8yexDJtC/QV9AqURE9JnnV4ee
|
// BmeBDAECAjBaBgNVHR8EUzBRME+gTaBLhklodHRwOi8vY3JsLnNlY3RpZ28uY29t
|
||||||
UB9XVKg+/XRjL7FQZQnmWEIuQxpMtPAlR1n6BB6T1CZGSlCBst6+eLf8ZxXhyVeE
|
// L1NlY3RpZ29SU0FPcmdhbml6YXRpb25WYWxpZGF0aW9uU2VjdXJlU2VydmVyQ0Eu
|
||||||
Hg9j1uliutZfVS7qXMYoCAQlObgOK6nyTJccBz8NUvXt7y+CDwIDAQABo0IwQDAd
|
// Y3JsMIGKBggrBgEFBQcBAQR+MHwwVQYIKwYBBQUHMAKGSWh0dHA6Ly9jcnQuc2Vj
|
||||||
BgNVHQ4EFgQUU3m/WqorSs9UgOHYm8Cd8rIDZsswDgYDVR0PAQH/BAQDAgEGMA8G
|
// dGlnby5jb20vU2VjdGlnb1JTQU9yZ2FuaXphdGlvblZhbGlkYXRpb25TZWN1cmVT
|
||||||
A1UdEwEB/wQFMAMBAf8wDQYJKoZIhvcNAQEMBQADggIBAFzUfA3P9wF9QZllDHPF
|
// ZXJ2ZXJDQS5jcnQwIwYIKwYBBQUHMAGGF2h0dHA6Ly9vY3NwLnNlY3RpZ28uY29t
|
||||||
Up/L+M+ZBn8b2kMVn54CVVeWFPFSPCeHlCjtHzoBN6J2/FNQwISbxmtOuowhT6KO
|
// MIIBgAYKKwYBBAHWeQIEAgSCAXAEggFsAWoAdwB2/4g/Crb7lVHCYcz1h7o0tKTN
|
||||||
VWKR82kV2LyI48SqC/3vqOlLVSoGIG1VeCkZ7l8wXEskEVX/JJpuXior7gtNn3/3
|
// uyncaEIKn+ZnTFo6dAAAAYmc9m/gAAAEAwBIMEYCIQD8XOozx411S/bnZambGjTB
|
||||||
ATiUFJVDBwn7YKnuHKsSjKCaXqeYalltiz8I+8jRRa8YFWSQEg9zKC7F4iRO/Fjs
|
// yTcr2fCmggUfQLSmqksD5gIhAIjiEMg0o1VSuQW31gWzfzL6idCkIZeSKN104cdp
|
||||||
8PRF/iKz6y+O0tlFYQXBl2+odnKPi4w2r78NBc5xjeambx9spnFixdjQg3IM8WcR
|
// xa4SAHcA2ra/az+1tiKfm8K7XGvocJFxbLtRhIU0vaQ9MEjX+6sAAAGJnPZwPwAA
|
||||||
iQycE0xyNN+81XHfqnHd4blsjDwSXWXavVcStkNr/+XeTWYRUc+ZruwXtuhxkYze
|
// BAMASDBGAiEA2sPTZTzvxewzQ8vk36+BWAKuJS7AvJ5W3clvfwCa8OUCIQC74ekT
|
||||||
Sf7dNXGiFSeUHM9h4ya7b6NnJSFd5t0dCy5oGzuCr+yDZ4XUmFF0sbmZgIn/f3gZ
|
// Ged2fqQE4sVy74aS6HRA2ihC9VLtNrASJx1YjQB2AO7N0GTV2xrOxVy3nbTNE6Iy
|
||||||
XHlKYC6SQK5MNyosycdiyA5d9zZbyuAlJQG03RoHnHcAP9Dc1ew91Pq7P8yF1m9/
|
// h0Z8vOzew1FIWUZxH7WbAAABiZz2cA8AAAQDAEcwRQIgEklH7wYCFuuJIFUHX5PY
|
||||||
qS3fuQL39ZeatTXaw2ewh0qpKJ4jjv9cJ2vhsE/zB+4ALtRZh8tSQZXq9EfX7mRB
|
// /vZ3bDoxOp+061PT3caa+rICIQC0abgfGlBKiHxp47JZxnW3wcVqWdiYX4ViLm9H
|
||||||
VXyNWQKV3WKdwrnuWih0hKWbt5DHDAff9Yk2dDLWKMGwsAvgnEzDHNb842m1R0aB
|
// xfx4ljCBxgYDVR0RBIG+MIG7gg1tZW1wb29sLnNwYWNlghMqLmZtdC5tZW1wb29s
|
||||||
L6KCq9NjRHDEjf8tM7qtj3u1cIiuPhnPQCjY/MiQu12ZIvVS5ljFH4gxQ+6IHdfG
|
// LnNwYWNlghMqLmZyYS5tZW1wb29sLnNwYWNlgg8qLm1lbXBvb2wuc3BhY2WCEyou
|
||||||
jjxDah2nGN59PRbxYvnKkKj9
|
// dGs3Lm1lbXBvb2wuc3BhY2WCEyoudmExLm1lbXBvb2wuc3BhY2WCDGJpc3EubWFy
|
||||||
-----END CERTIFICATE-----
|
// a2V0c4IKYmlzcS5uaW5qYYIObGlxdWlkLm5ldHdvcmuCDGxpcXVpZC5wbGFjZYIN
|
||||||
)EOF";
|
// bWVtcG9vbC5uaW5qYTANBgkqhkiG9w0BAQsFAAOCAQEAFvOSRnlHDfq9C8acjZEG
|
||||||
|
// 5XIqjNYigyWyjOvx83of6Z3PBKkAZB5D/UHBPp+jBDJiEb/QXC7Z7Y7kpuvnoVib
|
||||||
|
// b4jDc0RjGEsxL+3F7cSw26m3wILJhhHooGZRmFY4GOAeCZtYCOTzJsiZvFpDoQjU
|
||||||
|
// hTBxtaps05z0Ly9/eYvkXnjnBNROZJVR+KYHlq4TIoGNc4q4KvpfHv2I/vhS2M1e
|
||||||
|
// bECNNPEyRxHGKdXXO3huocE7aVKpy+JDR6cWwDu6hpdc1j/SCDqdTDFQ7McHOrqA
|
||||||
|
// fpPh4FcfePMh7Mqxtg2pSs5pXPtiP0ZjLgxd7HbAXct8Y+/jGk+k3sx3SeYXVimr
|
||||||
|
// ew==
|
||||||
|
// -----END CERTIFICATE-----)";
|
||||||
|
|
||||||
|
void setupBlockNotify() {
|
||||||
|
// currentBlockHeight = preferences.getUInt("blockHeight", 816000);
|
||||||
|
|
||||||
void setupBlockNotify()
|
|
||||||
{
|
|
||||||
IPAddress result;
|
IPAddress result;
|
||||||
|
|
||||||
int dnsErr = -1;
|
int dnsErr = -1;
|
||||||
String mempoolInstance =
|
String mempoolInstance =
|
||||||
preferences.getString("mempoolInstance", DEFAULT_MEMPOOL_INSTANCE);
|
preferences.getString("mempoolInstance", DEFAULT_MEMPOOL_INSTANCE);
|
||||||
|
|
||||||
while (dnsErr != 1 && !strchr(mempoolInstance.c_str(), ':'))
|
while (dnsErr != 1) {
|
||||||
{
|
|
||||||
dnsErr = WiFi.hostByName(mempoolInstance.c_str(), result);
|
dnsErr = WiFi.hostByName(mempoolInstance.c_str(), result);
|
||||||
|
|
||||||
if (dnsErr != 1)
|
if (dnsErr != 1) {
|
||||||
{
|
|
||||||
Serial.print(mempoolInstance);
|
Serial.print(mempoolInstance);
|
||||||
Serial.println(F("mempool DNS could not be resolved"));
|
Serial.println(F("mempool DNS could not be resolved"));
|
||||||
WiFi.reconnect();
|
WiFi.reconnect();
|
||||||
|
@ -66,283 +69,119 @@ void setupBlockNotify()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get current block height through regular API
|
// Get current block height through regular API
|
||||||
int blockFetch = getBlockFetch();
|
HTTPClient *http = new HTTPClient();
|
||||||
|
http->begin("https://" + mempoolInstance + "/api/blocks/tip/height");
|
||||||
|
int httpCode = http->GET();
|
||||||
|
|
||||||
if (blockFetch > currentBlockHeight)
|
if (httpCode > 0 && httpCode == HTTP_CODE_OK) {
|
||||||
currentBlockHeight = blockFetch;
|
String blockHeightStr = http->getString();
|
||||||
|
currentBlockHeight = blockHeightStr.toInt();
|
||||||
if (currentBlockHeight != -1)
|
// xTaskNotifyGive(blockUpdateTaskHandle);
|
||||||
{
|
if (workQueue != nullptr) {
|
||||||
lastBlockUpdate = esp_timer_get_time() / 1000000;
|
WorkItem blockUpdate = {TASK_BLOCK_UPDATE, 0};
|
||||||
}
|
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
||||||
|
}
|
||||||
if (workQueue != nullptr)
|
|
||||||
{
|
|
||||||
WorkItem blockUpdate = {TASK_BLOCK_UPDATE, 0};
|
|
||||||
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!preferences.getBool("fetchEurPrice", DEFAULT_FETCH_EUR_PRICE) && preferences.getBool("ownDataSource", DEFAULT_OWN_DATA_SOURCE))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// std::strcpy(wsServer, String("wss://" + mempoolInstance +
|
// std::strcpy(wsServer, String("wss://" + mempoolInstance +
|
||||||
// "/api/v1/ws").c_str());
|
// "/api/v1/ws").c_str());
|
||||||
|
|
||||||
const String protocol = preferences.getBool("mempoolSecure", DEFAULT_MEMPOOL_SECURE) ? "wss" : "ws";
|
|
||||||
|
|
||||||
String mempoolUri = protocol + "://" + preferences.getString("mempoolInstance", DEFAULT_MEMPOOL_INSTANCE) + "/api/v1/ws";
|
|
||||||
|
|
||||||
esp_websocket_client_config_t config = {
|
esp_websocket_client_config_t config = {
|
||||||
// .uri = "wss://mempool.space/api/v1/ws",
|
.uri = "wss://mempool.space/api/v1/ws",
|
||||||
.task_stack = (6*1024),
|
// .task_stack = (6*1024),
|
||||||
.user_agent = USER_AGENT
|
// .cert_pem = mempoolWsCert,
|
||||||
|
.user_agent = USER_AGENT,
|
||||||
};
|
};
|
||||||
|
|
||||||
if (preferences.getBool("mempoolSecure", DEFAULT_MEMPOOL_SECURE)) {
|
|
||||||
config.cert_pem = mempoolWsCert;
|
|
||||||
}
|
|
||||||
|
|
||||||
config.uri = mempoolUri.c_str();
|
|
||||||
|
|
||||||
Serial.printf("Connecting to %s\r\n", preferences.getString("mempoolInstance", DEFAULT_MEMPOOL_INSTANCE));
|
|
||||||
|
|
||||||
blockNotifyClient = esp_websocket_client_init(&config);
|
blockNotifyClient = esp_websocket_client_init(&config);
|
||||||
esp_websocket_register_events(blockNotifyClient, WEBSOCKET_EVENT_ANY,
|
esp_websocket_register_events(blockNotifyClient, WEBSOCKET_EVENT_ANY,
|
||||||
onWebsocketBlockEvent, blockNotifyClient);
|
onWebsocketEvent, blockNotifyClient);
|
||||||
esp_websocket_client_start(blockNotifyClient);
|
esp_websocket_client_start(blockNotifyClient);
|
||||||
}
|
}
|
||||||
|
|
||||||
void onWebsocketBlockEvent(void *handler_args, esp_event_base_t base,
|
void onWebsocketEvent(void *handler_args, esp_event_base_t base,
|
||||||
int32_t event_id, void *event_data)
|
int32_t event_id, void *event_data) {
|
||||||
{
|
|
||||||
esp_websocket_event_data_t *data = (esp_websocket_event_data_t *)event_data;
|
esp_websocket_event_data_t *data = (esp_websocket_event_data_t *)event_data;
|
||||||
const String sub = "{\"action\": \"want\", \"data\":[\"blocks\", \"mempool-blocks\"]}";
|
const String sub = "{\"action\": \"want\", \"data\":[\"blocks\"]}";
|
||||||
switch (event_id)
|
switch (event_id) {
|
||||||
{
|
case WEBSOCKET_EVENT_CONNECTED:
|
||||||
case WEBSOCKET_EVENT_CONNECTED:
|
Serial.println(F("Connected to Mempool.space WebSocket"));
|
||||||
blockNotifyInit = true;
|
|
||||||
|
|
||||||
Serial.println(F("Connected to Mempool.space WebSocket"));
|
Serial.println(sub);
|
||||||
|
if (esp_websocket_client_send_text(blockNotifyClient, sub.c_str(),
|
||||||
|
sub.length(), portMAX_DELAY) == -1) {
|
||||||
|
Serial.println(F("Mempool.space WS Block Subscribe Error"));
|
||||||
|
}
|
||||||
|
|
||||||
Serial.println(sub);
|
break;
|
||||||
if (esp_websocket_client_send_text(blockNotifyClient, sub.c_str(),
|
case WEBSOCKET_EVENT_DATA:
|
||||||
sub.length(), portMAX_DELAY) == -1)
|
onWebsocketMessage(data);
|
||||||
{
|
break;
|
||||||
Serial.println(F("Mempool.space WS Block Subscribe Error"));
|
case WEBSOCKET_EVENT_ERROR:
|
||||||
}
|
Serial.println(F("Mempool.space WS Connnection error"));
|
||||||
|
break;
|
||||||
break;
|
case WEBSOCKET_EVENT_DISCONNECTED:
|
||||||
case WEBSOCKET_EVENT_DATA:
|
Serial.println(F("Mempool.space WS Connnection Closed"));
|
||||||
onWebsocketBlockMessage(data);
|
break;
|
||||||
break;
|
|
||||||
case WEBSOCKET_EVENT_ERROR:
|
|
||||||
Serial.println(F("Mempool.space WS Connnection error"));
|
|
||||||
break;
|
|
||||||
case WEBSOCKET_EVENT_DISCONNECTED:
|
|
||||||
Serial.println(F("Mempool.space WS Connnection Closed"));
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void onWebsocketBlockMessage(esp_websocket_event_data_t *event_data)
|
void onWebsocketMessage(esp_websocket_event_data_t *event_data) {
|
||||||
{
|
SpiRamJsonDocument doc(event_data->data_len);
|
||||||
JsonDocument doc;
|
|
||||||
|
|
||||||
JsonDocument filter;
|
deserializeJson(doc, (char *)event_data->data_ptr);
|
||||||
filter["block"]["height"] = true;
|
|
||||||
filter["mempool-blocks"][0]["medianFee"] = true;
|
|
||||||
|
|
||||||
deserializeJson(doc, (char *)event_data->data_ptr, DeserializationOption::Filter(filter));
|
if (doc.containsKey("block")) {
|
||||||
|
|
||||||
// if (error) {
|
|
||||||
// Serial.print("deserializeJson() failed: ");
|
|
||||||
// Serial.println(error.c_str());
|
|
||||||
// return;
|
|
||||||
// }
|
|
||||||
|
|
||||||
if (doc.containsKey("block"))
|
|
||||||
{
|
|
||||||
JsonObject block = doc["block"];
|
JsonObject block = doc["block"];
|
||||||
|
|
||||||
if (block["height"].as<uint>() == currentBlockHeight) {
|
currentBlockHeight = block["height"].as<uint>();
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
processNewBlock(block["height"].as<uint>());
|
Serial.printf("New block found: %d\r\n", block["height"].as<uint>());
|
||||||
}
|
|
||||||
else if (doc.containsKey("mempool-blocks"))
|
|
||||||
{
|
|
||||||
JsonArray blockInfo = doc["mempool-blocks"].as<JsonArray>();
|
|
||||||
|
|
||||||
uint medianFee = (uint)round(blockInfo[0]["medianFee"].as<double>());
|
|
||||||
|
|
||||||
processNewBlockFee(medianFee);
|
|
||||||
}
|
|
||||||
|
|
||||||
doc.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
void processNewBlock(uint newBlockHeight) {
|
|
||||||
if (newBlockHeight < currentBlockHeight)
|
|
||||||
return;
|
|
||||||
|
|
||||||
currentBlockHeight = newBlockHeight;
|
|
||||||
|
|
||||||
// Serial.printf("New block found: %d\r\n", block["height"].as<uint>());
|
|
||||||
preferences.putUInt("blockHeight", currentBlockHeight);
|
preferences.putUInt("blockHeight", currentBlockHeight);
|
||||||
lastBlockUpdate = esp_timer_get_time() / 1000000;
|
|
||||||
|
|
||||||
if (workQueue != nullptr)
|
if (workQueue != nullptr) {
|
||||||
{
|
|
||||||
WorkItem blockUpdate = {TASK_BLOCK_UPDATE, 0};
|
WorkItem blockUpdate = {TASK_BLOCK_UPDATE, 0};
|
||||||
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
||||||
// xTaskNotifyGive(blockUpdateTaskHandle);
|
// xTaskNotifyGive(blockUpdateTaskHandle);
|
||||||
|
|
||||||
if (getCurrentScreen() != SCREEN_BLOCK_HEIGHT &&
|
if (getCurrentScreen() != SCREEN_BLOCK_HEIGHT &&
|
||||||
preferences.getBool("stealFocus", DEFAULT_STEAL_FOCUS))
|
preferences.getBool("stealFocus", true)) {
|
||||||
{
|
|
||||||
uint64_t timerPeriod = 0;
|
uint64_t timerPeriod = 0;
|
||||||
if (isTimerActive())
|
if (isTimerActive()) {
|
||||||
{
|
|
||||||
// store timer periode before making inactive to prevent artifacts
|
// store timer periode before making inactive to prevent artifacts
|
||||||
timerPeriod = getTimerSeconds();
|
timerPeriod = getTimerSeconds();
|
||||||
esp_timer_stop(screenRotateTimer);
|
esp_timer_stop(screenRotateTimer);
|
||||||
}
|
}
|
||||||
setCurrentScreen(SCREEN_BLOCK_HEIGHT);
|
setCurrentScreen(SCREEN_BLOCK_HEIGHT);
|
||||||
if (timerPeriod > 0)
|
if (timerPeriod > 0) {
|
||||||
{
|
|
||||||
esp_timer_start_periodic(screenRotateTimer,
|
esp_timer_start_periodic(screenRotateTimer,
|
||||||
timerPeriod * usPerSecond);
|
timerPeriod * usPerSecond);
|
||||||
}
|
}
|
||||||
vTaskDelay(pdMS_TO_TICKS(315*NUM_SCREENS)); // Extra delay because of screen switching
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (preferences.getBool("ledFlashOnUpd", DEFAULT_LED_FLASH_ON_UPD))
|
if (getCurrentScreen() == SCREEN_BLOCK_HEIGHT &&
|
||||||
{
|
preferences.getBool("ledFlashOnUpd", false)) {
|
||||||
vTaskDelay(pdMS_TO_TICKS(250)); // Wait until screens are updated
|
vTaskDelay(pdMS_TO_TICKS(250)); // Wait until screens are updated
|
||||||
queueLedEffect(LED_FLASH_BLOCK_NOTIFY);
|
queueLedEffect(LED_FLASH_BLOCK_NOTIFY);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void processNewBlockFee(uint newBlockFee) {
|
doc.clear();
|
||||||
if (blockMedianFee == newBlockFee)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Serial.printf("New median fee: %d\r\n", medianFee);
|
|
||||||
blockMedianFee = newBlockFee;
|
|
||||||
|
|
||||||
if (workQueue != nullptr)
|
|
||||||
{
|
|
||||||
WorkItem blockUpdate = {TASK_FEE_UPDATE, 0};
|
|
||||||
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
uint getBlockHeight() { return currentBlockHeight; }
|
uint getBlockHeight() { return currentBlockHeight; }
|
||||||
|
|
||||||
void setBlockHeight(uint newBlockHeight)
|
void setBlockHeight(uint newBlockHeight) {
|
||||||
{
|
|
||||||
currentBlockHeight = newBlockHeight;
|
currentBlockHeight = newBlockHeight;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint getBlockMedianFee() { return blockMedianFee; }
|
bool isBlockNotifyConnected() {
|
||||||
|
if (blockNotifyClient == NULL) return false;
|
||||||
void setBlockMedianFee(uint newBlockMedianFee)
|
|
||||||
{
|
|
||||||
blockMedianFee = newBlockMedianFee;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isBlockNotifyConnected()
|
|
||||||
{
|
|
||||||
if (blockNotifyClient == NULL)
|
|
||||||
return false;
|
|
||||||
return esp_websocket_client_is_connected(blockNotifyClient);
|
return esp_websocket_client_is_connected(blockNotifyClient);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool getBlockNotifyInit()
|
void stopBlockNotify() {
|
||||||
{
|
|
||||||
return blockNotifyInit;
|
|
||||||
}
|
|
||||||
|
|
||||||
void stopBlockNotify()
|
|
||||||
{
|
|
||||||
if (blockNotifyClient == NULL)
|
|
||||||
return;
|
|
||||||
|
|
||||||
esp_websocket_client_close(blockNotifyClient, pdMS_TO_TICKS(5000));
|
|
||||||
esp_websocket_client_stop(blockNotifyClient);
|
esp_websocket_client_stop(blockNotifyClient);
|
||||||
esp_websocket_client_destroy(blockNotifyClient);
|
esp_websocket_client_destroy(blockNotifyClient);
|
||||||
|
|
||||||
blockNotifyClient = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
void restartBlockNotify()
|
|
||||||
{
|
|
||||||
stopBlockNotify();
|
|
||||||
|
|
||||||
if (blockNotifyClient == NULL) {
|
|
||||||
setupBlockNotify();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// esp_websocket_client_close(blockNotifyClient, pdMS_TO_TICKS(5000));
|
|
||||||
// esp_websocket_client_stop(blockNotifyClient);
|
|
||||||
// esp_websocket_client_start(blockNotifyClient);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int getBlockFetch()
|
|
||||||
{
|
|
||||||
try {
|
|
||||||
WiFiClientSecure client;
|
|
||||||
|
|
||||||
if (preferences.getBool("mempoolSecure", DEFAULT_MEMPOOL_SECURE)) {
|
|
||||||
client.setCACertBundle(rootca_crt_bundle_start);
|
|
||||||
}
|
|
||||||
|
|
||||||
String mempoolInstance =
|
|
||||||
preferences.getString("mempoolInstance", DEFAULT_MEMPOOL_INSTANCE);
|
|
||||||
|
|
||||||
// Get current block height through regular API
|
|
||||||
HTTPClient http;
|
|
||||||
|
|
||||||
const String protocol = preferences.getBool("mempoolSecure", DEFAULT_MEMPOOL_SECURE) ? "https" : "http";
|
|
||||||
|
|
||||||
if (preferences.getBool("mempoolSecure", DEFAULT_MEMPOOL_SECURE))
|
|
||||||
http.begin(client, protocol + "://" + mempoolInstance + "/api/blocks/tip/height");
|
|
||||||
else
|
|
||||||
http.begin(protocol + "://" + mempoolInstance + "/api/blocks/tip/height");
|
|
||||||
|
|
||||||
Serial.println("Fetching block height from " + protocol + "://" + mempoolInstance + "/api/blocks/tip/height");
|
|
||||||
int httpCode = http.GET();
|
|
||||||
|
|
||||||
if (httpCode > 0 && httpCode == HTTP_CODE_OK)
|
|
||||||
{
|
|
||||||
String blockHeightStr = http.getString();
|
|
||||||
return blockHeightStr.toInt();
|
|
||||||
} else {
|
|
||||||
Serial.println("HTTP code" + String(httpCode));
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (...) {
|
|
||||||
Serial.println(F("An exception occured while trying to get the latest block"));
|
|
||||||
}
|
|
||||||
|
|
||||||
return 2203; // B-T-C
|
|
||||||
}
|
|
||||||
|
|
||||||
uint getLastBlockUpdate()
|
|
||||||
{
|
|
||||||
return lastBlockUpdate;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setLastBlockUpdate(uint lastUpdate)
|
|
||||||
{
|
|
||||||
lastBlockUpdate = lastUpdate;
|
|
||||||
}
|
}
|
|
@ -11,31 +11,17 @@
|
||||||
|
|
||||||
#include "lib/led_handler.hpp"
|
#include "lib/led_handler.hpp"
|
||||||
#include "lib/screen_handler.hpp"
|
#include "lib/screen_handler.hpp"
|
||||||
#include "lib/timers.hpp"
|
|
||||||
#include "lib/shared.hpp"
|
#include "lib/shared.hpp"
|
||||||
|
|
||||||
// using namespace websockets;
|
// using namespace websockets;
|
||||||
|
|
||||||
void setupBlockNotify();
|
void setupBlockNotify();
|
||||||
|
|
||||||
void onWebsocketBlockEvent(void *handler_args, esp_event_base_t base,
|
void onWebsocketEvent(void *handler_args, esp_event_base_t base,
|
||||||
int32_t event_id, void *event_data);
|
int32_t event_id, void *event_data);
|
||||||
void onWebsocketBlockMessage(esp_websocket_event_data_t *event_data);
|
void onWebsocketMessage(esp_websocket_event_data_t *event_data);
|
||||||
|
|
||||||
void setBlockHeight(uint newBlockHeight);
|
void setBlockHeight(uint newBlockHeight);
|
||||||
uint getBlockHeight();
|
uint getBlockHeight();
|
||||||
|
|
||||||
void setBlockMedianFee(uint blockMedianFee);
|
|
||||||
uint getBlockMedianFee();
|
|
||||||
|
|
||||||
bool isBlockNotifyConnected();
|
bool isBlockNotifyConnected();
|
||||||
void stopBlockNotify();
|
void stopBlockNotify();
|
||||||
void restartBlockNotify();
|
|
||||||
|
|
||||||
void processNewBlock(uint newBlockHeight);
|
|
||||||
void processNewBlockFee(uint newBlockFee);
|
|
||||||
|
|
||||||
bool getBlockNotifyInit();
|
|
||||||
uint getLastBlockUpdate();
|
|
||||||
int getBlockFetch();
|
|
||||||
void setLastBlockUpdate(uint lastUpdate);
|
|
|
@ -4,52 +4,39 @@ TaskHandle_t buttonTaskHandle = NULL;
|
||||||
const TickType_t debounceDelay = pdMS_TO_TICKS(50);
|
const TickType_t debounceDelay = pdMS_TO_TICKS(50);
|
||||||
TickType_t lastDebounceTime = 0;
|
TickType_t lastDebounceTime = 0;
|
||||||
|
|
||||||
#ifdef IS_BTCLOCK_V8
|
|
||||||
#define BTN_1 256
|
|
||||||
#define BTN_2 512
|
|
||||||
#define BTN_3 1024
|
|
||||||
#define BTN_4 2048
|
|
||||||
#else
|
|
||||||
#define BTN_1 2048
|
|
||||||
#define BTN_2 1024
|
|
||||||
#define BTN_3 512
|
|
||||||
#define BTN_4 256
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void buttonTask(void *parameter) {
|
void buttonTask(void *parameter) {
|
||||||
while (1) {
|
while (1) {
|
||||||
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
||||||
std::lock_guard<std::mutex> lock(mcpMutex);
|
std::lock_guard<std::mutex> lock(mcpMutex);
|
||||||
|
|
||||||
TickType_t currentTime = xTaskGetTickCount();
|
TickType_t currentTime = xTaskGetTickCount();
|
||||||
|
|
||||||
if ((currentTime - lastDebounceTime) >= debounceDelay) {
|
if ((currentTime - lastDebounceTime) >= debounceDelay) {
|
||||||
lastDebounceTime = currentTime;
|
lastDebounceTime = currentTime;
|
||||||
|
|
||||||
if (!digitalRead(MCP_INT_PIN)) {
|
if (!digitalRead(MCP_INT_PIN)) {
|
||||||
uint pin = mcp1.getInterruptFlagRegister();
|
uint pin = mcp1.getLastInterruptPin();
|
||||||
|
|
||||||
switch (pin) {
|
switch (pin) {
|
||||||
case BTN_1:
|
case 3:
|
||||||
toggleTimerActive();
|
toggleTimerActive();
|
||||||
break;
|
break;
|
||||||
case BTN_2:
|
case 2:
|
||||||
nextScreen();
|
nextScreen();
|
||||||
break;
|
break;
|
||||||
case BTN_3:
|
case 1:
|
||||||
previousScreen();
|
previousScreen();
|
||||||
break;
|
break;
|
||||||
case BTN_4:
|
case 0:
|
||||||
showSystemStatusScreen();
|
showSystemStatusScreen();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
mcp1.getInterruptCaptureRegister();
|
mcp1.clearInterrupts();
|
||||||
} else {
|
} else {
|
||||||
}
|
}
|
||||||
// Very ugly, but for some reason this is necessary
|
// Very ugly, but for some reason this is necessary
|
||||||
while (!digitalRead(MCP_INT_PIN)) {
|
while (!digitalRead(MCP_INT_PIN)) {
|
||||||
mcp1.getInterruptCaptureRegister();
|
mcp1.clearInterrupts();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -63,7 +50,7 @@ void IRAM_ATTR handleButtonInterrupt() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void setupButtonTask() {
|
void setupButtonTask() {
|
||||||
xTaskCreate(buttonTask, "ButtonTask", 3072, NULL, tskIDLE_PRIORITY,
|
xTaskCreate(buttonTask, "ButtonTask", 4096, NULL, tskIDLE_PRIORITY,
|
||||||
&buttonTaskHandle); // Create the FreeRTOS task
|
&buttonTaskHandle); // Create the FreeRTOS task
|
||||||
// Use interrupt instead of task
|
// Use interrupt instead of task
|
||||||
attachInterrupt(MCP_INT_PIN, handleButtonInterrupt, CHANGE);
|
attachInterrupt(MCP_INT_PIN, handleButtonInterrupt, CHANGE);
|
||||||
|
|
|
@ -4,7 +4,6 @@
|
||||||
|
|
||||||
#include "lib/screen_handler.hpp"
|
#include "lib/screen_handler.hpp"
|
||||||
#include "lib/shared.hpp"
|
#include "lib/shared.hpp"
|
||||||
#include "lib/timers.hpp"
|
|
||||||
|
|
||||||
extern TaskHandle_t buttonTaskHandle;
|
extern TaskHandle_t buttonTaskHandle;
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,88 +1,60 @@
|
||||||
#pragma once
|
#pragma once;
|
||||||
|
#include <Adafruit_MCP23X17.h>
|
||||||
#include <MCP23017.h>
|
|
||||||
#include <Arduino.h>
|
#include <Arduino.h>
|
||||||
#include <Preferences.h>
|
#include <Preferences.h>
|
||||||
#include <WiFiClientSecure.h>
|
#include <WiFiClientSecure.h>
|
||||||
#include <WiFiManager.h>
|
#include <WiFiManager.h>
|
||||||
#include <base64.h>
|
#include <base64.h>
|
||||||
#include <esp_task_wdt.h>
|
#include <esp_task_wdt.h>
|
||||||
#include <nvs_flash.h>
|
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
#include "lib/block_notify.hpp"
|
#include "lib/block_notify.hpp"
|
||||||
#include "lib/button_handler.hpp"
|
#include "lib/button_handler.hpp"
|
||||||
#include "lib/epd.hpp"
|
#include "lib/epd.hpp"
|
||||||
// #include "lib/improv.hpp"
|
#include "lib/improv.hpp"
|
||||||
#include "lib/led_handler.hpp"
|
#include "lib/led_handler.hpp"
|
||||||
#include "lib/ota.hpp"
|
#include "lib/ota.hpp"
|
||||||
#include "lib/nostr_notify.hpp"
|
|
||||||
#include "lib/bitaxe_fetch.hpp"
|
|
||||||
|
|
||||||
#include "lib/v2_notify.hpp"
|
|
||||||
|
|
||||||
#include "lib/price_notify.hpp"
|
#include "lib/price_notify.hpp"
|
||||||
#include "lib/screen_handler.hpp"
|
#include "lib/screen_handler.hpp"
|
||||||
|
#include "lib/nostr_subscribe.hpp"
|
||||||
|
|
||||||
#include "lib/shared.hpp"
|
#include "lib/shared.hpp"
|
||||||
#include "lib/webserver.hpp"
|
#include "lib/webserver.hpp"
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
#include "PCA9685.h"
|
|
||||||
#include "BH1750.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define NTP_SERVER "pool.ntp.org"
|
#define NTP_SERVER "pool.ntp.org"
|
||||||
#define DEFAULT_TIME_OFFSET_SECONDS 3600
|
#define DEFAULT_MEMPOOL_INSTANCE "mempool.space"
|
||||||
#define USER_AGENT "BTClock/3.0"
|
#define TIME_OFFSET_SECONDS 3600
|
||||||
#ifndef MCP_DEV_ADDR
|
#define USER_AGENT "BTClock/2.0"
|
||||||
#define MCP_DEV_ADDR 0x20
|
#define MCP_DEV_ADDR 0x20
|
||||||
#endif
|
#define DEFAULT_SECONDS_BETWEEN_PRICE_UPDATE 30
|
||||||
|
#define DEFAULT_MINUTES_FULL_REFRESH 60
|
||||||
|
#define DEFAULT_NOSTR_RELAYS "nostr.dbtc.link,relay.damus.io,nostr.onsats.org"
|
||||||
|
#define DEFAULT_NOSTR_BLOCKS_AUTHOR "55cf67598c37737af6d0278cbdcb400bf1a90b7654e964bda08f451bd3dffe5a"
|
||||||
|
#define DEFAULT_NOSTR_PRICE_AUTHOR "df66a305cca14265c558a6fe61ed6aa5e1e1b80a7ef5d89d2bbe82f62698c1ca"
|
||||||
|
|
||||||
|
#define DEFAULT_FG_COLOR GxEPD_WHITE
|
||||||
|
#define DEFAULT_BG_COLOR GxEPD_BLACK
|
||||||
|
|
||||||
void setup();
|
void setup();
|
||||||
void syncTime();
|
void setupTime();
|
||||||
uint getLastTimeSync();
|
|
||||||
void setupPreferences();
|
void setupPreferences();
|
||||||
void setupWebsocketClients(void *pvParameters);
|
void setupWebsocketClients(void *pvParameters);
|
||||||
void setupHardware();
|
void setupHardware();
|
||||||
void setupWifi();
|
void tryImprovSetup();
|
||||||
void setupTimers();
|
void setupTimers();
|
||||||
void finishSetup();
|
void finishSetup();
|
||||||
void setupMcp();
|
void setupMcp();
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
void setupFrontlight();
|
|
||||||
float getLightLevel();
|
|
||||||
bool hasLightLevel();
|
|
||||||
extern PCA9685 flArray;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
String getMyHostname();
|
String getMyHostname();
|
||||||
std::vector<ScreenMapping> getScreenNameMap();
|
std::vector<std::string> getScreenNameMap();
|
||||||
|
|
||||||
std::vector<std::string> getLocalUrl();
|
std::vector<std::string> getLocalUrl();
|
||||||
// bool improv_connectWifi(std::string ssid, std::string password);
|
bool improv_connectWifi(std::string ssid, std::string password);
|
||||||
// void improvGetAvailableWifiNetworks();
|
void improvGetAvailableWifiNetworks();
|
||||||
// bool onImprovCommandCallback(improv::ImprovCommand cmd);
|
bool onImprovCommandCallback(improv::ImprovCommand cmd);
|
||||||
// void onImprovErrorCallback(improv::Error err);
|
void onImprovErrorCallback(improv::Error err);
|
||||||
// void improv_set_state(improv::State state);
|
void improv_set_state(improv::State state);
|
||||||
// void improv_send_response(std::vector<uint8_t> &response);
|
void improv_send_response(std::vector<uint8_t> &response);
|
||||||
// void improv_set_error(improv::Error error);
|
void improv_set_error(improv::Error error);
|
||||||
//void addCurrencyMappings(const std::vector<std::string>& currencies);
|
|
||||||
std::vector<std::string> getActiveCurrencies();
|
|
||||||
std::vector<std::string> getAvailableCurrencies();
|
|
||||||
|
|
||||||
bool isActiveCurrency(std::string ¤cy);
|
|
||||||
|
|
||||||
void WiFiEvent(WiFiEvent_t event, WiFiEventInfo_t info);
|
void WiFiEvent(WiFiEvent_t event, WiFiEventInfo_t info);
|
||||||
String getHwRev();
|
|
||||||
bool isWhiteVersion();
|
|
||||||
String getFsRev();
|
|
||||||
|
|
||||||
void addScreenMapping(int value, const char* name);
|
|
||||||
// void addScreenMapping(int value, const String& name);
|
|
||||||
// void addScreenMapping(int value, const std::string& name);
|
|
||||||
|
|
||||||
int findScreenIndexByValue(int value);
|
|
||||||
String replaceAmbiguousChars(String input);
|
|
||||||
const char* getFirmwareFilename();
|
|
||||||
const char* getWebUiFilename();
|
|
||||||
// void loadIcons();
|
|
|
@ -1,73 +0,0 @@
|
||||||
#define INITIAL_BLOCK_HEIGHT 851500
|
|
||||||
#define INITIAL_LAST_PRICE 50000
|
|
||||||
#define DEFAULT_TX_POWER 0
|
|
||||||
|
|
||||||
#define DEFAULT_MEMPOOL_SECURE true
|
|
||||||
#define DEFAULT_LED_TEST_ON_POWER true
|
|
||||||
#define DEFAULT_LED_FLASH_ON_UPD false
|
|
||||||
#define DEFAULT_LED_BRIGHTNESS 128
|
|
||||||
#define DEFAULT_STEAL_FOCUS false
|
|
||||||
#define DEFAULT_MCAP_BIG_CHAR true
|
|
||||||
#define DEFAULT_MDNS_ENABLED true
|
|
||||||
#define DEFAULT_OTA_ENABLED true
|
|
||||||
#define DEFAULT_FETCH_EUR_PRICE false
|
|
||||||
#define DEFAULT_USE_SATS_SYMBOL false
|
|
||||||
#define DEFAULT_USE_BLOCK_COUNTDOWN true
|
|
||||||
#define DEFAULT_SUFFIX_PRICE false
|
|
||||||
#define DEFAULT_DISABLE_LEDS false
|
|
||||||
#define DEFAULT_DISABLE_FL false
|
|
||||||
#define DEFAULT_OWN_DATA_SOURCE true
|
|
||||||
#define DEFAULT_STAGING_SOURCE false
|
|
||||||
#define DEFAULT_MOW_MODE false
|
|
||||||
#define DEFAULT_SUFFIX_SHARE_DOT false
|
|
||||||
|
|
||||||
#define DEFAULT_V2_SOURCE_CURRENCY CURRENCY_USD
|
|
||||||
|
|
||||||
|
|
||||||
#define DEFAULT_TIME_OFFSET_SECONDS 3600
|
|
||||||
|
|
||||||
#define DEFAULT_HOSTNAME_PREFIX "btclock"
|
|
||||||
#define DEFAULT_MEMPOOL_INSTANCE "mempool.space"
|
|
||||||
|
|
||||||
#define DEFAULT_USE_NOSTR false
|
|
||||||
#define DEFAULT_NOSTR_NPUB "642317135fd4c4205323b9dea8af3270657e62d51dc31a657c0ec8aab31c6288"
|
|
||||||
#define DEFAULT_NOSTR_RELAY "wss://relay.primal.net"
|
|
||||||
|
|
||||||
#define DEFAULT_SECONDS_BETWEEN_PRICE_UPDATE 30
|
|
||||||
#define DEFAULT_MINUTES_FULL_REFRESH 60
|
|
||||||
|
|
||||||
#define DEFAULT_FG_COLOR GxEPD_WHITE
|
|
||||||
#define DEFAULT_BG_COLOR GxEPD_BLACK
|
|
||||||
|
|
||||||
#define DEFAULT_WP_TIMEOUT 15*60
|
|
||||||
|
|
||||||
#define DEFAULT_FL_MAX_BRIGHTNESS 2048
|
|
||||||
#define DEFAULT_FL_EFFECT_DELAY 15
|
|
||||||
|
|
||||||
#define DEFAULT_LUX_LIGHT_TOGGLE 128
|
|
||||||
#define DEFAULT_FL_OFF_WHEN_DARK true
|
|
||||||
|
|
||||||
#define DEFAULT_FL_ALWAYS_ON false
|
|
||||||
#define DEFAULT_FL_FLASH_ON_UPDATE false
|
|
||||||
|
|
||||||
#define DEFAULT_LED_STATUS false
|
|
||||||
#define DEFAULT_TIMER_ACTIVE true
|
|
||||||
#define DEFAULT_TIMER_SECONDS 1800
|
|
||||||
#define DEFAULT_CURRENT_SCREEN 0
|
|
||||||
|
|
||||||
#define DEFAULT_BITAXE_ENABLED false
|
|
||||||
#define DEFAULT_BITAXE_HOSTNAME "bitaxe1"
|
|
||||||
|
|
||||||
#define DEFAULT_ZAP_NOTIFY_ENABLED false
|
|
||||||
#define DEFAULT_ZAP_NOTIFY_PUBKEY "b5127a08cf33616274800a4387881a9f98e04b9c37116e92de5250498635c422"
|
|
||||||
#define DEFAULT_LED_FLASH_ON_ZAP true
|
|
||||||
#define DEFAULT_FL_FLASH_ON_ZAP true
|
|
||||||
|
|
||||||
#define DEFAULT_HTTP_AUTH_ENABLED false
|
|
||||||
#define DEFAULT_HTTP_AUTH_USERNAME "btclock"
|
|
||||||
#define DEFAULT_HTTP_AUTH_PASSWORD "satoshi"
|
|
||||||
|
|
||||||
#define DEFAULT_ACTIVE_CURRENCIES "USD,EUR,JPY"
|
|
||||||
|
|
||||||
#define DEFAULT_GIT_RELEASE_URL "https://git.btclock.dev/api/v1/repos/btclock/btclock_v3/releases/latest"
|
|
||||||
#define DEFAULT_VERTICAL_DESC true
|
|
457
src/lib/epd.cpp
457
src/lib/epd.cpp
|
@ -1,65 +1,6 @@
|
||||||
#include "epd.hpp"
|
#include "epd.hpp"
|
||||||
|
|
||||||
#ifdef IS_BTCLOCK_REV_B
|
#ifndef IS_BTCLOCK_S3
|
||||||
Native_Pin EPD_CS[NUM_SCREENS] = {
|
|
||||||
Native_Pin(2),
|
|
||||||
Native_Pin(4),
|
|
||||||
Native_Pin(6),
|
|
||||||
Native_Pin(10),
|
|
||||||
Native_Pin(38),
|
|
||||||
Native_Pin(21),
|
|
||||||
Native_Pin(17),
|
|
||||||
};
|
|
||||||
Native_Pin EPD_BUSY[NUM_SCREENS] = {
|
|
||||||
Native_Pin(3),
|
|
||||||
Native_Pin(5),
|
|
||||||
Native_Pin(7),
|
|
||||||
Native_Pin(9),
|
|
||||||
Native_Pin(37),
|
|
||||||
Native_Pin(18),
|
|
||||||
Native_Pin(16),
|
|
||||||
};
|
|
||||||
MCP23X17_Pin EPD_RESET_MPD[NUM_SCREENS] = {
|
|
||||||
MCP23X17_Pin(mcp1, 8),
|
|
||||||
MCP23X17_Pin(mcp1, 9),
|
|
||||||
MCP23X17_Pin(mcp1, 10),
|
|
||||||
MCP23X17_Pin(mcp1, 11),
|
|
||||||
MCP23X17_Pin(mcp1, 12),
|
|
||||||
MCP23X17_Pin(mcp1, 13),
|
|
||||||
MCP23X17_Pin(mcp1, 14),
|
|
||||||
};
|
|
||||||
|
|
||||||
Native_Pin EPD_DC = Native_Pin(14);
|
|
||||||
#elif IS_BTCLOCK_V8
|
|
||||||
Native_Pin EPD_DC = Native_Pin(38);
|
|
||||||
|
|
||||||
MCP23X17_Pin EPD_BUSY[NUM_SCREENS] = {
|
|
||||||
MCP23X17_Pin(mcp1, 8),
|
|
||||||
MCP23X17_Pin(mcp1, 9),
|
|
||||||
MCP23X17_Pin(mcp1, 10),
|
|
||||||
MCP23X17_Pin(mcp1, 11),
|
|
||||||
MCP23X17_Pin(mcp1, 12),
|
|
||||||
MCP23X17_Pin(mcp1, 13),
|
|
||||||
MCP23X17_Pin(mcp1, 14),
|
|
||||||
MCP23X17_Pin(mcp1, 4),
|
|
||||||
};
|
|
||||||
|
|
||||||
MCP23X17_Pin EPD_CS[NUM_SCREENS] = {
|
|
||||||
MCP23X17_Pin(mcp2, 8), MCP23X17_Pin(mcp2, 10), MCP23X17_Pin(mcp2, 12),
|
|
||||||
MCP23X17_Pin(mcp2, 14), MCP23X17_Pin(mcp2, 0), MCP23X17_Pin(mcp2, 2),
|
|
||||||
MCP23X17_Pin(mcp2, 4), MCP23X17_Pin(mcp2, 6)};
|
|
||||||
|
|
||||||
MCP23X17_Pin EPD_RESET_MPD[NUM_SCREENS] = {
|
|
||||||
MCP23X17_Pin(mcp2, 9),
|
|
||||||
MCP23X17_Pin(mcp2, 11),
|
|
||||||
MCP23X17_Pin(mcp2, 13),
|
|
||||||
MCP23X17_Pin(mcp2, 15),
|
|
||||||
MCP23X17_Pin(mcp2, 1),
|
|
||||||
MCP23X17_Pin(mcp2, 3),
|
|
||||||
MCP23X17_Pin(mcp2, 5),
|
|
||||||
MCP23X17_Pin(mcp2, 7),
|
|
||||||
};
|
|
||||||
#else
|
|
||||||
Native_Pin EPD_CS[NUM_SCREENS] = {
|
Native_Pin EPD_CS[NUM_SCREENS] = {
|
||||||
Native_Pin(2),
|
Native_Pin(2),
|
||||||
Native_Pin(4),
|
Native_Pin(4),
|
||||||
|
@ -75,37 +16,48 @@ Native_Pin EPD_CS[NUM_SCREENS] = {
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
Native_Pin EPD_BUSY[NUM_SCREENS] = {
|
Native_Pin EPD_BUSY[NUM_SCREENS] = {
|
||||||
Native_Pin(3),
|
Native_Pin(3), Native_Pin(5), Native_Pin(7), Native_Pin(9),
|
||||||
Native_Pin(5),
|
Native_Pin(37), Native_Pin(18), Native_Pin(16),
|
||||||
Native_Pin(7),
|
|
||||||
Native_Pin(9),
|
|
||||||
Native_Pin(37),
|
|
||||||
Native_Pin(18),
|
|
||||||
Native_Pin(16),
|
|
||||||
};
|
};
|
||||||
MCP23X17_Pin EPD_RESET_MPD[NUM_SCREENS] = {
|
MCP23X17_Pin EPD_RESET_MPD[NUM_SCREENS] = {
|
||||||
MCP23X17_Pin(mcp1, 8),
|
MCP23X17_Pin(mcp1, 8), MCP23X17_Pin(mcp1, 9), MCP23X17_Pin(mcp1, 10),
|
||||||
MCP23X17_Pin(mcp1, 9),
|
MCP23X17_Pin(mcp1, 11), MCP23X17_Pin(mcp1, 12), MCP23X17_Pin(mcp1, 13),
|
||||||
MCP23X17_Pin(mcp1, 10),
|
|
||||||
MCP23X17_Pin(mcp1, 11),
|
|
||||||
MCP23X17_Pin(mcp1, 12),
|
|
||||||
MCP23X17_Pin(mcp1, 13),
|
|
||||||
MCP23X17_Pin(mcp1, 14),
|
MCP23X17_Pin(mcp1, 14),
|
||||||
};
|
};
|
||||||
|
|
||||||
Native_Pin EPD_DC = Native_Pin(14);
|
Native_Pin EPD_DC = Native_Pin(14);
|
||||||
|
#else
|
||||||
|
Native_Pin EPD_DC = Native_Pin(38);
|
||||||
|
|
||||||
|
MCP23X17_Pin EPD_BUSY[NUM_SCREENS] = {
|
||||||
|
MCP23X17_Pin(mcp1, 8), MCP23X17_Pin(mcp1, 9), MCP23X17_Pin(mcp1, 10),
|
||||||
|
MCP23X17_Pin(mcp1, 11), MCP23X17_Pin(mcp1, 12), MCP23X17_Pin(mcp1, 13),
|
||||||
|
MCP23X17_Pin(mcp1, 14), MCP23X17_Pin(mcp1, 4),
|
||||||
|
};
|
||||||
|
|
||||||
|
MCP23X17_Pin EPD_CS[NUM_SCREENS] = {
|
||||||
|
MCP23X17_Pin(mcp2, 8), MCP23X17_Pin(mcp2, 10), MCP23X17_Pin(mcp2, 12),
|
||||||
|
MCP23X17_Pin(mcp2, 14), MCP23X17_Pin(mcp2, 0), MCP23X17_Pin(mcp2, 2),
|
||||||
|
MCP23X17_Pin(mcp2, 4), MCP23X17_Pin(mcp2, 6)};
|
||||||
|
|
||||||
|
MCP23X17_Pin EPD_RESET_MPD[NUM_SCREENS] = {
|
||||||
|
MCP23X17_Pin(mcp2, 9), MCP23X17_Pin(mcp2, 11), MCP23X17_Pin(mcp2, 13),
|
||||||
|
MCP23X17_Pin(mcp2, 15), MCP23X17_Pin(mcp2, 1), MCP23X17_Pin(mcp2, 3),
|
||||||
|
MCP23X17_Pin(mcp2, 5), MCP23X17_Pin(mcp2, 7),
|
||||||
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
GxEPD2_BW<EPD_CLASS, EPD_CLASS::HEIGHT> displays[NUM_SCREENS] = {
|
GxEPD2_BW<GxEPD2_213_B74, GxEPD2_213_B74::HEIGHT> displays[NUM_SCREENS] = {
|
||||||
EPD_CLASS(&EPD_CS[0], &EPD_DC, &EPD_RESET_MPD[0], &EPD_BUSY[0]),
|
GxEPD2_213_B74(&EPD_CS[0], &EPD_DC, &EPD_RESET_MPD[0], &EPD_BUSY[0]),
|
||||||
EPD_CLASS(&EPD_CS[1], &EPD_DC, &EPD_RESET_MPD[1], &EPD_BUSY[1]),
|
GxEPD2_213_B74(&EPD_CS[1], &EPD_DC, &EPD_RESET_MPD[1], &EPD_BUSY[1]),
|
||||||
EPD_CLASS(&EPD_CS[2], &EPD_DC, &EPD_RESET_MPD[2], &EPD_BUSY[2]),
|
GxEPD2_213_B74(&EPD_CS[2], &EPD_DC, &EPD_RESET_MPD[2], &EPD_BUSY[2]),
|
||||||
EPD_CLASS(&EPD_CS[3], &EPD_DC, &EPD_RESET_MPD[3], &EPD_BUSY[3]),
|
GxEPD2_213_B74(&EPD_CS[3], &EPD_DC, &EPD_RESET_MPD[3], &EPD_BUSY[3]),
|
||||||
EPD_CLASS(&EPD_CS[4], &EPD_DC, &EPD_RESET_MPD[4], &EPD_BUSY[4]),
|
GxEPD2_213_B74(&EPD_CS[4], &EPD_DC, &EPD_RESET_MPD[4], &EPD_BUSY[4]),
|
||||||
EPD_CLASS(&EPD_CS[5], &EPD_DC, &EPD_RESET_MPD[5], &EPD_BUSY[5]),
|
GxEPD2_213_B74(&EPD_CS[5], &EPD_DC, &EPD_RESET_MPD[5], &EPD_BUSY[5]),
|
||||||
EPD_CLASS(&EPD_CS[6], &EPD_DC, &EPD_RESET_MPD[6], &EPD_BUSY[6]),
|
GxEPD2_213_B74(&EPD_CS[6], &EPD_DC, &EPD_RESET_MPD[6], &EPD_BUSY[6]),
|
||||||
#ifdef IS_BTCLOCK_V8
|
#ifdef IS_BTCLOCK_S3
|
||||||
EPD_CLASS(&EPD_CS[7], &EPD_DC, &EPD_RESET_MPD[7], &EPD_BUSY[7]),
|
GxEPD2_213_B74(&EPD_CS[7], &EPD_DC, &EPD_RESET_MPD[6], &EPD_BUSY[7]),
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -126,36 +78,24 @@ int bgColor = GxEPD_BLACK;
|
||||||
#define FONT_SMALL Antonio_SemiBold20pt7b
|
#define FONT_SMALL Antonio_SemiBold20pt7b
|
||||||
#define FONT_BIG Antonio_SemiBold90pt7b
|
#define FONT_BIG Antonio_SemiBold90pt7b
|
||||||
#define FONT_MEDIUM Antonio_SemiBold40pt7b
|
#define FONT_MEDIUM Antonio_SemiBold40pt7b
|
||||||
#define FONT_SATSYMBOL Satoshi_Symbol90pt7b
|
|
||||||
std::mutex epdUpdateMutex;
|
std::mutex epdUpdateMutex;
|
||||||
std::mutex epdMutex[NUM_SCREENS];
|
std::mutex epdMutex[NUM_SCREENS];
|
||||||
|
|
||||||
uint8_t qrcode[800];
|
uint8_t qrcode[800];
|
||||||
|
|
||||||
#ifdef IS_BTCLOCK_V8
|
void forceFullRefresh() {
|
||||||
#define EPD_TASK_STACK_SIZE 4096
|
for (uint i = 0; i < NUM_SCREENS; i++) {
|
||||||
#else
|
|
||||||
#define EPD_TASK_STACK_SIZE 2048
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void forceFullRefresh()
|
|
||||||
{
|
|
||||||
for (uint i = 0; i < NUM_SCREENS; i++)
|
|
||||||
{
|
|
||||||
lastFullRefresh[i] = NULL;
|
lastFullRefresh[i] = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void refreshFromMemory()
|
void refreshFromMemory() {
|
||||||
{
|
for (uint i = 0; i < NUM_SCREENS; i++) {
|
||||||
for (uint i = 0; i < NUM_SCREENS; i++)
|
|
||||||
{
|
|
||||||
int *taskParam = new int;
|
int *taskParam = new int;
|
||||||
*taskParam = i;
|
*taskParam = i;
|
||||||
|
|
||||||
xTaskCreate(
|
xTaskCreate(
|
||||||
[](void *pvParameters)
|
[](void *pvParameters) {
|
||||||
{
|
|
||||||
const int epdIndex = *(int *)pvParameters;
|
const int epdIndex = *(int *)pvParameters;
|
||||||
delete (int *)pvParameters;
|
delete (int *)pvParameters;
|
||||||
displays[epdIndex].refresh(false);
|
displays[epdIndex].refresh(false);
|
||||||
|
@ -165,62 +105,41 @@ void refreshFromMemory()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void setupDisplays()
|
void setupDisplays() {
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lockMcp(mcpMutex);
|
std::lock_guard<std::mutex> lockMcp(mcpMutex);
|
||||||
|
|
||||||
for (uint i = 0; i < NUM_SCREENS; i++)
|
for (uint i = 0; i < NUM_SCREENS; i++) {
|
||||||
{
|
|
||||||
displays[i].init(0, true, 30);
|
displays[i].init(0, true, 30);
|
||||||
}
|
}
|
||||||
|
|
||||||
updateQueue = xQueueCreate(UPDATE_QUEUE_SIZE, sizeof(UpdateDisplayTaskItem));
|
updateQueue = xQueueCreate(UPDATE_QUEUE_SIZE, sizeof(UpdateDisplayTaskItem));
|
||||||
|
|
||||||
xTaskCreate(prepareDisplayUpdateTask, "PrepareUpd", EPD_TASK_STACK_SIZE, NULL, 11, NULL);
|
xTaskCreate(prepareDisplayUpdateTask, "PrepareUpd", 4096, NULL, 11, NULL);
|
||||||
|
|
||||||
for (uint i = 0; i < NUM_SCREENS; i++)
|
for (uint i = 0; i < NUM_SCREENS; i++) {
|
||||||
{
|
|
||||||
// epdUpdateSemaphore[i] = xSemaphoreCreateBinary();
|
// epdUpdateSemaphore[i] = xSemaphoreCreateBinary();
|
||||||
// xSemaphoreGive(epdUpdateSemaphore[i]);
|
// xSemaphoreGive(epdUpdateSemaphore[i]);
|
||||||
|
|
||||||
int *taskParam = new int;
|
int *taskParam = new int;
|
||||||
*taskParam = i;
|
*taskParam = i;
|
||||||
|
|
||||||
xTaskCreate(updateDisplay, ("EpdUpd" + String(i)).c_str(), EPD_TASK_STACK_SIZE, taskParam,
|
xTaskCreate(updateDisplay, ("EpdUpd" + String(i)).c_str(), 2048, taskParam,
|
||||||
11, &tasks[i]); // create task
|
11, &tasks[i]); // create task
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hold lower button to enable "storage mode" (prevents burn-in of ePaper displays)
|
epdContent = {"B", "T", "C", "L", "O", "C", "K"};
|
||||||
if (mcp1.read1(0) == LOW)
|
|
||||||
{
|
|
||||||
setFgColor(GxEPD_BLACK);
|
|
||||||
setBgColor(GxEPD_WHITE);
|
|
||||||
|
|
||||||
epdContent.fill("");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
#ifdef IS_BTCLOCK_V8
|
|
||||||
epdContent = {"B", "T", "C", "L", "O", "C", "K", "v8"};
|
|
||||||
#else
|
|
||||||
epdContent = {"B", "T", "C", "L", "O", "C", "K"};
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
setEpdContent(epdContent);
|
setEpdContent(epdContent);
|
||||||
}
|
}
|
||||||
|
|
||||||
void setEpdContent(std::array<String, NUM_SCREENS> newEpdContent)
|
void setEpdContent(std::array<String, NUM_SCREENS> newEpdContent) {
|
||||||
{
|
|
||||||
setEpdContent(newEpdContent, false);
|
setEpdContent(newEpdContent, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void setEpdContent(std::array<std::string, NUM_SCREENS> newEpdContent)
|
void setEpdContent(std::array<std::string, NUM_SCREENS> newEpdContent) {
|
||||||
{
|
|
||||||
std::array<String, NUM_SCREENS> conv;
|
std::array<String, NUM_SCREENS> conv;
|
||||||
|
|
||||||
for (size_t i = 0; i < newEpdContent.size(); ++i)
|
for (size_t i = 0; i < newEpdContent.size(); ++i) {
|
||||||
{
|
|
||||||
conv[i] = String(newEpdContent[i].c_str());
|
conv[i] = String(newEpdContent[i].c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -228,16 +147,13 @@ void setEpdContent(std::array<std::string, NUM_SCREENS> newEpdContent)
|
||||||
}
|
}
|
||||||
|
|
||||||
void setEpdContent(std::array<String, NUM_SCREENS> newEpdContent,
|
void setEpdContent(std::array<String, NUM_SCREENS> newEpdContent,
|
||||||
bool forceUpdate)
|
bool forceUpdate) {
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(epdUpdateMutex);
|
std::lock_guard<std::mutex> lock(epdUpdateMutex);
|
||||||
|
|
||||||
waitUntilNoneBusy();
|
waitUntilNoneBusy();
|
||||||
|
|
||||||
for (uint i = 0; i < NUM_SCREENS; i++)
|
for (uint i = 0; i < NUM_SCREENS; i++) {
|
||||||
{
|
if (newEpdContent[i].compareTo(currentEpdContent[i]) != 0 || forceUpdate) {
|
||||||
if (newEpdContent[i].compareTo(currentEpdContent[i]) != 0 || forceUpdate)
|
|
||||||
{
|
|
||||||
epdContent[i] = newEpdContent[i];
|
epdContent[i] = newEpdContent[i];
|
||||||
UpdateDisplayTaskItem dispUpdate = {i};
|
UpdateDisplayTaskItem dispUpdate = {i};
|
||||||
xQueueSend(updateQueue, &dispUpdate, portMAX_DELAY);
|
xQueueSend(updateQueue, &dispUpdate, portMAX_DELAY);
|
||||||
|
@ -245,15 +161,12 @@ void setEpdContent(std::array<String, NUM_SCREENS> newEpdContent,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void prepareDisplayUpdateTask(void *pvParameters)
|
void prepareDisplayUpdateTask(void *pvParameters) {
|
||||||
{
|
|
||||||
UpdateDisplayTaskItem receivedItem;
|
UpdateDisplayTaskItem receivedItem;
|
||||||
|
|
||||||
while (1)
|
while (1) {
|
||||||
{
|
|
||||||
// Wait for a work item to be available in the queue
|
// Wait for a work item to be available in the queue
|
||||||
if (xQueueReceive(updateQueue, &receivedItem, portMAX_DELAY))
|
if (xQueueReceive(updateQueue, &receivedItem, portMAX_DELAY)) {
|
||||||
{
|
|
||||||
uint epdIndex = receivedItem.dispNum;
|
uint epdIndex = receivedItem.dispNum;
|
||||||
std::lock_guard<std::mutex> lock(epdMutex[epdIndex]);
|
std::lock_guard<std::mutex> lock(epdMutex[epdIndex]);
|
||||||
// displays[epdIndex].init(0, false); // Little longer reset duration
|
// displays[epdIndex].init(0, false); // Little longer reset duration
|
||||||
|
@ -261,47 +174,21 @@ void prepareDisplayUpdateTask(void *pvParameters)
|
||||||
|
|
||||||
bool updatePartial = true;
|
bool updatePartial = true;
|
||||||
|
|
||||||
if (epdContent[epdIndex].length() > 1 && strstr(epdContent[epdIndex].c_str(), "/") != NULL)
|
if (strstr(epdContent[epdIndex].c_str(), "/") != NULL) {
|
||||||
{
|
|
||||||
String top = epdContent[epdIndex].substring(
|
String top = epdContent[epdIndex].substring(
|
||||||
0, epdContent[epdIndex].indexOf("/"));
|
0, epdContent[epdIndex].indexOf("/"));
|
||||||
String bottom = epdContent[epdIndex].substring(
|
String bottom = epdContent[epdIndex].substring(
|
||||||
epdContent[epdIndex].indexOf("/") + 1);
|
epdContent[epdIndex].indexOf("/") + 1);
|
||||||
splitText(epdIndex, top, bottom, updatePartial);
|
splitText(epdIndex, top, bottom, updatePartial);
|
||||||
}
|
} else if (epdContent[epdIndex].startsWith(F("qr"))) {
|
||||||
else if (epdContent[epdIndex].startsWith(F("qr")))
|
|
||||||
{
|
|
||||||
renderQr(epdIndex, epdContent[epdIndex], updatePartial);
|
renderQr(epdIndex, epdContent[epdIndex], updatePartial);
|
||||||
}
|
} else if (epdContent[epdIndex].length() > 5) {
|
||||||
else if (epdContent[epdIndex].startsWith(F("mdi")))
|
|
||||||
{
|
|
||||||
renderIcon(epdIndex, epdContent[epdIndex], updatePartial);
|
|
||||||
}
|
|
||||||
else if (epdContent[epdIndex].length() > 5)
|
|
||||||
{
|
|
||||||
renderText(epdIndex, epdContent[epdIndex], updatePartial);
|
renderText(epdIndex, epdContent[epdIndex], updatePartial);
|
||||||
}
|
} else {
|
||||||
else
|
if (epdContent[epdIndex].length() > 1) {
|
||||||
{
|
showChars(epdIndex, epdContent[epdIndex], updatePartial,
|
||||||
if (epdContent[epdIndex].length() == 2) {
|
&FONT_MEDIUM);
|
||||||
showChars(epdIndex, epdContent[epdIndex], updatePartial, &FONT_BIG);
|
} else {
|
||||||
}
|
|
||||||
else if (epdContent[epdIndex].length() > 1 && epdContent[epdIndex].indexOf(".") == -1)
|
|
||||||
{
|
|
||||||
if (epdContent[epdIndex].equals("STS"))
|
|
||||||
{
|
|
||||||
showDigit(epdIndex, 'S', updatePartial,
|
|
||||||
&FONT_SATSYMBOL);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
showChars(epdIndex, epdContent[epdIndex], updatePartial,
|
|
||||||
&FONT_MEDIUM);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
|
|
||||||
showDigit(epdIndex, epdContent[epdIndex].c_str()[0], updatePartial,
|
showDigit(epdIndex, epdContent[epdIndex].c_str()[0], updatePartial,
|
||||||
&FONT_BIG);
|
&FONT_BIG);
|
||||||
}
|
}
|
||||||
|
@ -312,13 +199,11 @@ void prepareDisplayUpdateTask(void *pvParameters)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void updateDisplay(void *pvParameters) noexcept
|
extern "C" void updateDisplay(void *pvParameters) noexcept {
|
||||||
{
|
|
||||||
const int epdIndex = *(int *)pvParameters;
|
const int epdIndex = *(int *)pvParameters;
|
||||||
delete (int *)pvParameters;
|
delete (int *)pvParameters;
|
||||||
|
|
||||||
for (;;)
|
for (;;) {
|
||||||
{
|
|
||||||
// Wait for the task notification
|
// Wait for the task notification
|
||||||
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
||||||
|
|
||||||
|
@ -330,8 +215,7 @@ extern "C" void updateDisplay(void *pvParameters) noexcept
|
||||||
displays[epdIndex].init(0, false, 40);
|
displays[epdIndex].init(0, false, 40);
|
||||||
}
|
}
|
||||||
uint count = 0;
|
uint count = 0;
|
||||||
while (EPD_BUSY[epdIndex].digitalRead() == HIGH || count < 10)
|
while (EPD_BUSY[epdIndex].digitalRead() == HIGH || count < 10) {
|
||||||
{
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(100));
|
vTaskDelay(pdMS_TO_TICKS(100));
|
||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
|
@ -343,20 +227,16 @@ extern "C" void updateDisplay(void *pvParameters) noexcept
|
||||||
(millis() - lastFullRefresh[epdIndex]) >
|
(millis() - lastFullRefresh[epdIndex]) >
|
||||||
(preferences.getUInt("fullRefreshMin",
|
(preferences.getUInt("fullRefreshMin",
|
||||||
DEFAULT_MINUTES_FULL_REFRESH) *
|
DEFAULT_MINUTES_FULL_REFRESH) *
|
||||||
60 * 1000))
|
60 * 1000)) {
|
||||||
{
|
|
||||||
updatePartial = false;
|
updatePartial = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
char tries = 0;
|
char tries = 0;
|
||||||
while (tries < 3)
|
while (tries < 3) {
|
||||||
{
|
if (displays[epdIndex].displayWithReturn(updatePartial)) {
|
||||||
if (displays[epdIndex].displayWithReturn(updatePartial))
|
|
||||||
{
|
|
||||||
displays[epdIndex].powerOff();
|
displays[epdIndex].powerOff();
|
||||||
currentEpdContent[epdIndex] = epdContent[epdIndex];
|
currentEpdContent[epdIndex] = epdContent[epdIndex];
|
||||||
if (!updatePartial)
|
if (!updatePartial) lastFullRefresh[epdIndex] = millis();
|
||||||
lastFullRefresh[epdIndex] = millis();
|
|
||||||
|
|
||||||
if (eventSourceTaskHandle != NULL)
|
if (eventSourceTaskHandle != NULL)
|
||||||
xTaskNotifyGive(eventSourceTaskHandle);
|
xTaskNotifyGive(eventSourceTaskHandle);
|
||||||
|
@ -371,13 +251,8 @@ extern "C" void updateDisplay(void *pvParameters) noexcept
|
||||||
}
|
}
|
||||||
|
|
||||||
void splitText(const uint dispNum, const String &top, const String &bottom,
|
void splitText(const uint dispNum, const String &top, const String &bottom,
|
||||||
bool partial)
|
bool partial) {
|
||||||
{
|
displays[dispNum].setRotation(2);
|
||||||
if(preferences.getBool("verticalDesc", DEFAULT_VERTICAL_DESC) && dispNum == 0) {
|
|
||||||
displays[dispNum].setRotation(1);
|
|
||||||
} else {
|
|
||||||
displays[dispNum].setRotation(2);
|
|
||||||
}
|
|
||||||
displays[dispNum].setFont(&FONT_SMALL);
|
displays[dispNum].setFont(&FONT_SMALL);
|
||||||
displays[dispNum].setTextColor(getFgColor());
|
displays[dispNum].setTextColor(getFgColor());
|
||||||
|
|
||||||
|
@ -414,141 +289,37 @@ void splitText(const uint dispNum, const String &top, const String &bottom,
|
||||||
displays[dispNum].print(bottom);
|
displays[dispNum].print(bottom);
|
||||||
}
|
}
|
||||||
|
|
||||||
// void showChars(const uint dispNum, const String &chars, bool partial,
|
|
||||||
// const GFXfont *font)
|
|
||||||
// {
|
|
||||||
// displays[dispNum].setRotation(2);
|
|
||||||
// displays[dispNum].setFont(font);
|
|
||||||
// displays[dispNum].setTextColor(getFgColor());
|
|
||||||
// int16_t tbx, tby;
|
|
||||||
// uint16_t tbw, tbh;
|
|
||||||
|
|
||||||
// displays[dispNum].getTextBounds(chars, 0, 0, &tbx, &tby, &tbw, &tbh);
|
|
||||||
|
|
||||||
// // center the bounding box by transposition of the origin:
|
|
||||||
// uint16_t x = ((displays[dispNum].width() - tbw) / 2) - tbx;
|
|
||||||
// uint16_t y = ((displays[dispNum].height() - tbh) / 2) - tby;
|
|
||||||
|
|
||||||
// displays[dispNum].fillScreen(getBgColor());
|
|
||||||
|
|
||||||
// displays[dispNum].setCursor(x, y);
|
|
||||||
// displays[dispNum].print(chars);
|
|
||||||
|
|
||||||
// // displays[dispNum].setCursor(10, 3);
|
|
||||||
// // displays[dispNum].setFont(&FONT_SMALL);
|
|
||||||
// // displays[dispNum].setTextColor(getFgColor());
|
|
||||||
// // displays[dispNum].println("Y = " + y);
|
|
||||||
// }
|
|
||||||
|
|
||||||
void showDigit(const uint dispNum, char chr, bool partial,
|
void showDigit(const uint dispNum, char chr, bool partial,
|
||||||
const GFXfont *font)
|
const GFXfont *font) {
|
||||||
{
|
|
||||||
String str(chr);
|
String str(chr);
|
||||||
|
|
||||||
if (chr == '.')
|
|
||||||
{
|
|
||||||
str = "!";
|
|
||||||
}
|
|
||||||
displays[dispNum].setRotation(2);
|
displays[dispNum].setRotation(2);
|
||||||
displays[dispNum].setFont(font);
|
displays[dispNum].setFont(font);
|
||||||
displays[dispNum].setTextColor(getFgColor());
|
displays[dispNum].setTextColor(getFgColor());
|
||||||
int16_t tbx, tby;
|
int16_t tbx, tby;
|
||||||
uint16_t tbw, tbh;
|
uint16_t tbw, tbh;
|
||||||
|
|
||||||
displays[dispNum].getTextBounds(str, 0, 0, &tbx, &tby, &tbw, &tbh);
|
displays[dispNum].getTextBounds(str, 0, 0, &tbx, &tby, &tbw, &tbh);
|
||||||
|
|
||||||
// center the bounding box by transposition of the origin:
|
// center the bounding box by transposition of the origin:
|
||||||
uint16_t x = ((displays[dispNum].width() - tbw) / 2) - tbx;
|
uint16_t x = ((displays[dispNum].width() - tbw) / 2) - tbx;
|
||||||
uint16_t y = ((displays[dispNum].height() - tbh) / 2) - tby;
|
uint16_t y = ((displays[dispNum].height() - tbh) / 2) - tby;
|
||||||
|
|
||||||
// if (str.equals("."))
|
|
||||||
// {
|
|
||||||
// // int16_t yAdvance = font->yAdvance;
|
|
||||||
// // uint8_t charIndex = 46 - font->first;
|
|
||||||
// // GFXglyph *glyph = (&font->glyph)[charIndex];
|
|
||||||
// int16_t tbx2, tby2;
|
|
||||||
// uint16_t tbw2, tbh2;
|
|
||||||
// displays[dispNum].getTextBounds(".!", 0, 0, &tbx2, &tby2, &tbw2, &tbh2);
|
|
||||||
|
|
||||||
// y = ((displays[dispNum].height() - tbh2) / 2) - tby2;
|
|
||||||
// // Serial.print("yAdvance");
|
|
||||||
// // Serial.println(yAdvance);
|
|
||||||
// // if (glyph != nullptr) {
|
|
||||||
// // Serial.print("height");
|
|
||||||
// // Serial.println(glyph->height);
|
|
||||||
// // Serial.print("yOffset");
|
|
||||||
// // Serial.println(glyph->yOffset);
|
|
||||||
// // }
|
|
||||||
|
|
||||||
// // y = 250-99+18+19;
|
|
||||||
// }
|
|
||||||
|
|
||||||
displays[dispNum].fillScreen(getBgColor());
|
displays[dispNum].fillScreen(getBgColor());
|
||||||
|
|
||||||
displays[dispNum].setCursor(x, y);
|
displays[dispNum].setCursor(x, y);
|
||||||
displays[dispNum].print(str);
|
displays[dispNum].print(str);
|
||||||
|
|
||||||
if (chr == '.')
|
|
||||||
{
|
|
||||||
displays[dispNum].fillRect(x, y, displays[dispNum].width(), round(displays[dispNum].height() * 0.9), getBgColor());
|
|
||||||
}
|
|
||||||
|
|
||||||
// displays[dispNum].setCursor(10, 3);
|
|
||||||
// displays[dispNum].setFont(&FONT_SMALL);
|
|
||||||
// displays[dispNum].setTextColor(getFgColor());
|
|
||||||
// displays[dispNum].println("Y = " + y);
|
|
||||||
}
|
|
||||||
|
|
||||||
int16_t calculateDescent(const GFXfont *font) {
|
|
||||||
int16_t maxDescent = 0;
|
|
||||||
for (uint16_t i = font->first; i <= font->last; i++) {
|
|
||||||
GFXglyph *glyph = &font->glyph[i - font->first];
|
|
||||||
int16_t descent = glyph->yOffset;
|
|
||||||
if (descent > maxDescent) {
|
|
||||||
maxDescent = descent;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return maxDescent;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void showChars(const uint dispNum, const String &chars, bool partial,
|
void showChars(const uint dispNum, const String &chars, bool partial,
|
||||||
const GFXfont *font)
|
const GFXfont *font) {
|
||||||
{
|
|
||||||
displays[dispNum].setRotation(2);
|
displays[dispNum].setRotation(2);
|
||||||
displays[dispNum].setFont(font);
|
displays[dispNum].setFont(font);
|
||||||
displays[dispNum].setTextColor(getFgColor());
|
displays[dispNum].setTextColor(getFgColor());
|
||||||
int16_t tbx, tby;
|
int16_t tbx, tby;
|
||||||
uint16_t tbw, tbh;
|
uint16_t tbw, tbh;
|
||||||
displays[dispNum].getTextBounds(chars, 0, 0, &tbx, &tby, &tbw, &tbh);
|
displays[dispNum].getTextBounds(chars, 0, 0, &tbx, &tby, &tbw, &tbh);
|
||||||
|
|
||||||
int16_t descent = calculateDescent(font);
|
|
||||||
|
|
||||||
// center the bounding box by transposition of the origin:
|
// center the bounding box by transposition of the origin:
|
||||||
uint16_t x = ((displays[dispNum].width() - tbw) / 2) - tbx;
|
uint16_t x = ((displays[dispNum].width() - tbw) / 2) - tbx;
|
||||||
uint16_t y = ((displays[dispNum].height() - tbh) / 2) - tby;
|
uint16_t y = ((displays[dispNum].height() - tbh) / 2) - tby;
|
||||||
displays[dispNum].fillScreen(getBgColor());
|
displays[dispNum].fillScreen(getBgColor());
|
||||||
// displays[dispNum].setCursor(x, y);
|
displays[dispNum].setCursor(x, y);
|
||||||
// displays[dispNum].print(chars);
|
displays[dispNum].print(chars);
|
||||||
|
|
||||||
for (int i = 0; i < chars.length(); i++) {
|
|
||||||
char c = chars[i];
|
|
||||||
if (c == '.' || c == ',') {
|
|
||||||
// For the dot, calculate its specific descent
|
|
||||||
GFXglyph *dotGlyph = &font->glyph[c -font->first];
|
|
||||||
int16_t dotDescent = dotGlyph->yOffset;
|
|
||||||
|
|
||||||
// Draw the dot with adjusted y-position
|
|
||||||
displays[dispNum].setCursor(x, y + dotDescent + dotGlyph->height + 8);
|
|
||||||
displays[dispNum].print(c);
|
|
||||||
} else {
|
|
||||||
// For other characters, use the original y-position
|
|
||||||
displays[dispNum].setCursor(x, y);
|
|
||||||
displays[dispNum].print(c);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Move x-position for the next character
|
|
||||||
x += font->glyph[c - font->first].xAdvance;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int getBgColor() { return bgColor; }
|
int getBgColor() { return bgColor; }
|
||||||
|
@ -559,12 +330,10 @@ void setBgColor(int color) { bgColor = color; }
|
||||||
|
|
||||||
void setFgColor(int color) { fgColor = color; }
|
void setFgColor(int color) { fgColor = color; }
|
||||||
|
|
||||||
std::array<String, NUM_SCREENS> getCurrentEpdContent()
|
std::array<String, NUM_SCREENS> getCurrentEpdContent() {
|
||||||
{
|
|
||||||
return currentEpdContent;
|
return currentEpdContent;
|
||||||
}
|
}
|
||||||
void renderText(const uint dispNum, const String &text, bool partial)
|
void renderText(const uint dispNum, const String &text, bool partial) {
|
||||||
{
|
|
||||||
displays[dispNum].setRotation(2);
|
displays[dispNum].setRotation(2);
|
||||||
displays[dispNum].setPartialWindow(0, 0, displays[dispNum].width(),
|
displays[dispNum].setPartialWindow(0, 0, displays[dispNum].width(),
|
||||||
displays[dispNum].height());
|
displays[dispNum].height());
|
||||||
|
@ -577,58 +346,20 @@ void renderText(const uint dispNum, const String &text, bool partial)
|
||||||
|
|
||||||
std::string line;
|
std::string line;
|
||||||
|
|
||||||
while (std::getline(ss, line, '\n'))
|
while (std::getline(ss, line, '\n')) {
|
||||||
{
|
if (line.rfind("*", 0) == 0) {
|
||||||
if (line.rfind("*", 0) == 0)
|
|
||||||
{
|
|
||||||
line.erase(std::remove(line.begin(), line.end(), '*'), line.end());
|
line.erase(std::remove(line.begin(), line.end(), '*'), line.end());
|
||||||
|
|
||||||
displays[dispNum].setFont(&FreeSansBold9pt7b);
|
displays[dispNum].setFont(&FreeSansBold9pt7b);
|
||||||
displays[dispNum].println(line.c_str());
|
displays[dispNum].println(line.c_str());
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
displays[dispNum].setFont(&FreeSans9pt7b);
|
displays[dispNum].setFont(&FreeSans9pt7b);
|
||||||
displays[dispNum].println(line.c_str());
|
displays[dispNum].println(line.c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void renderIcon(const uint dispNum, const String &text, bool partial)
|
void renderQr(const uint dispNum, const String &text, bool partial) {
|
||||||
{
|
|
||||||
displays[dispNum].setRotation(2);
|
|
||||||
|
|
||||||
displays[dispNum].setPartialWindow(0, 0, displays[dispNum].width(),
|
|
||||||
displays[dispNum].height());
|
|
||||||
displays[dispNum].fillScreen(getBgColor());
|
|
||||||
displays[dispNum].setTextColor(getFgColor());
|
|
||||||
|
|
||||||
uint iconIndex = 0;
|
|
||||||
if (text.endsWith("rocket")) {
|
|
||||||
iconIndex = 1;
|
|
||||||
}
|
|
||||||
else if (text.endsWith("lnbolt")) {
|
|
||||||
iconIndex = 3;
|
|
||||||
}
|
|
||||||
else if (text.endsWith("bitaxe")) {
|
|
||||||
iconIndex = 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
displays[dispNum].drawInvertedBitmap(0,0, epd_icons_allArray[iconIndex], 122, 250, getFgColor());
|
|
||||||
|
|
||||||
|
|
||||||
// displays[dispNum].drawInvertedBitmap(0,0, getOceanIcon(), 122, 250, getFgColor());
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void renderQr(const uint dispNum, const String &text, bool partial)
|
|
||||||
{
|
|
||||||
#ifdef USE_QR
|
#ifdef USE_QR
|
||||||
|
|
||||||
uint8_t tempBuffer[800];
|
uint8_t tempBuffer[800];
|
||||||
|
@ -648,10 +379,8 @@ void renderQr(const uint dispNum, const String &text, bool partial)
|
||||||
displays[dispNum].fillScreen(GxEPD_WHITE);
|
displays[dispNum].fillScreen(GxEPD_WHITE);
|
||||||
const int border = 0;
|
const int border = 0;
|
||||||
|
|
||||||
for (int y = -border; y < size * 4 + border; y++)
|
for (int y = -border; y < size * 4 + border; y++) {
|
||||||
{
|
for (int x = -border; x < size * 4 + border; x++) {
|
||||||
for (int x = -border; x < size * 4 + border; x++)
|
|
||||||
{
|
|
||||||
displays[dispNum].drawPixel(
|
displays[dispNum].drawPixel(
|
||||||
padding + x, paddingY + y,
|
padding + x, paddingY + y,
|
||||||
qrcodegen_getModule(qrcode, floor(float(x) / 4), floor(float(y) / 4))
|
qrcodegen_getModule(qrcode, floor(float(x) / 4), floor(float(y) / 4))
|
||||||
|
@ -662,22 +391,16 @@ void renderQr(const uint dispNum, const String &text, bool partial)
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void waitUntilNoneBusy()
|
void waitUntilNoneBusy() {
|
||||||
{
|
for (int i = 0; i < NUM_SCREENS; i++) {
|
||||||
for (int i = 0; i < NUM_SCREENS; i++)
|
|
||||||
{
|
|
||||||
uint count = 0;
|
uint count = 0;
|
||||||
while (EPD_BUSY[i].digitalRead())
|
while (EPD_BUSY[i].digitalRead()) {
|
||||||
{
|
|
||||||
count++;
|
count++;
|
||||||
vTaskDelay(10);
|
vTaskDelay(10);
|
||||||
if (count == 200)
|
if (count == 200) {
|
||||||
{
|
|
||||||
// displays[i].init(0, false);
|
// displays[i].init(0, false);
|
||||||
vTaskDelay(100);
|
vTaskDelay(100);
|
||||||
}
|
} else if (count > 205) {
|
||||||
else if (count > 205)
|
|
||||||
{
|
|
||||||
Serial.printf("Busy timeout %d", i);
|
Serial.printf("Busy timeout %d", i);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,6 @@
|
||||||
#include <Fonts/FreeSansBold9pt7b.h>
|
#include <Fonts/FreeSansBold9pt7b.h>
|
||||||
#include <GxEPD2_BW.h>
|
#include <GxEPD2_BW.h>
|
||||||
|
|
||||||
|
|
||||||
#include <mcp23x17_pin.hpp>
|
#include <mcp23x17_pin.hpp>
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
#include <native_pin.hpp>
|
#include <native_pin.hpp>
|
||||||
|
@ -13,7 +12,6 @@
|
||||||
#include "fonts/fonts.hpp"
|
#include "fonts/fonts.hpp"
|
||||||
#include "lib/config.hpp"
|
#include "lib/config.hpp"
|
||||||
#include "lib/shared.hpp"
|
#include "lib/shared.hpp"
|
||||||
#include "icons/icons.h"
|
|
||||||
|
|
||||||
#ifdef USE_QR
|
#ifdef USE_QR
|
||||||
#include "qrcodegen.h"
|
#include "qrcodegen.h"
|
||||||
|
@ -44,7 +42,6 @@ int getFgColor();
|
||||||
void setBgColor(int color);
|
void setBgColor(int color);
|
||||||
void setFgColor(int color);
|
void setFgColor(int color);
|
||||||
|
|
||||||
void renderIcon(const uint dispNum, const String &text, bool partial);
|
|
||||||
void renderText(const uint dispNum, const String &text, bool partial);
|
void renderText(const uint dispNum, const String &text, bool partial);
|
||||||
void renderQr(const uint dispNum, const String &text, bool partial);
|
void renderQr(const uint dispNum, const String &text, bool partial);
|
||||||
|
|
||||||
|
|
143
src/lib/improv.cpp
Normal file
143
src/lib/improv.cpp
Normal file
|
@ -0,0 +1,143 @@
|
||||||
|
#include "improv.h"
|
||||||
|
|
||||||
|
namespace improv {
|
||||||
|
|
||||||
|
ImprovCommand parse_improv_data(const std::vector<uint8_t> &data,
|
||||||
|
bool check_checksum) {
|
||||||
|
return parse_improv_data(data.data(), data.size(), check_checksum);
|
||||||
|
}
|
||||||
|
|
||||||
|
ImprovCommand parse_improv_data(const uint8_t *data, size_t length,
|
||||||
|
bool check_checksum) {
|
||||||
|
ImprovCommand improv_command;
|
||||||
|
Command command = (Command)data[0];
|
||||||
|
uint8_t data_length = data[1];
|
||||||
|
|
||||||
|
if (data_length != length - 2 - check_checksum) {
|
||||||
|
improv_command.command = UNKNOWN;
|
||||||
|
return improv_command;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (check_checksum) {
|
||||||
|
uint8_t checksum = data[length - 1];
|
||||||
|
|
||||||
|
uint32_t calculated_checksum = 0;
|
||||||
|
for (uint8_t i = 0; i < length - 1; i++) {
|
||||||
|
calculated_checksum += data[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((uint8_t)calculated_checksum != checksum) {
|
||||||
|
improv_command.command = BAD_CHECKSUM;
|
||||||
|
return improv_command;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (command == WIFI_SETTINGS) {
|
||||||
|
uint8_t ssid_length = data[2];
|
||||||
|
uint8_t ssid_start = 3;
|
||||||
|
size_t ssid_end = ssid_start + ssid_length;
|
||||||
|
|
||||||
|
uint8_t pass_length = data[ssid_end];
|
||||||
|
size_t pass_start = ssid_end + 1;
|
||||||
|
size_t pass_end = pass_start + pass_length;
|
||||||
|
|
||||||
|
std::string ssid(data + ssid_start, data + ssid_end);
|
||||||
|
std::string password(data + pass_start, data + pass_end);
|
||||||
|
return {.command = command, .ssid = ssid, .password = password};
|
||||||
|
}
|
||||||
|
|
||||||
|
improv_command.command = command;
|
||||||
|
return improv_command;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool parse_improv_serial_byte(size_t position, uint8_t byte,
|
||||||
|
const uint8_t *buffer,
|
||||||
|
std::function<bool(ImprovCommand)> &&callback,
|
||||||
|
std::function<void(Error)> &&on_error) {
|
||||||
|
if (position == 0) return byte == 'I';
|
||||||
|
if (position == 1) return byte == 'M';
|
||||||
|
if (position == 2) return byte == 'P';
|
||||||
|
if (position == 3) return byte == 'R';
|
||||||
|
if (position == 4) return byte == 'O';
|
||||||
|
if (position == 5) return byte == 'V';
|
||||||
|
|
||||||
|
if (position == 6) return byte == IMPROV_SERIAL_VERSION;
|
||||||
|
|
||||||
|
if (position <= 8) return true;
|
||||||
|
|
||||||
|
uint8_t type = buffer[7];
|
||||||
|
uint8_t data_len = buffer[8];
|
||||||
|
|
||||||
|
if (position <= 8 + data_len) return true;
|
||||||
|
|
||||||
|
if (position == 8 + data_len + 1) {
|
||||||
|
uint8_t checksum = 0x00;
|
||||||
|
for (size_t i = 0; i < position; i++) checksum += buffer[i];
|
||||||
|
|
||||||
|
if (checksum != byte) {
|
||||||
|
on_error(ERROR_INVALID_RPC);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type == TYPE_RPC) {
|
||||||
|
auto command = parse_improv_data(&buffer[9], data_len, false);
|
||||||
|
return callback(command);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<uint8_t> build_rpc_response(Command command,
|
||||||
|
const std::vector<std::string> &datum,
|
||||||
|
bool add_checksum) {
|
||||||
|
std::vector<uint8_t> out;
|
||||||
|
uint32_t length = 0;
|
||||||
|
out.push_back(command);
|
||||||
|
for (const auto &str : datum) {
|
||||||
|
uint8_t len = str.length();
|
||||||
|
length += len + 1;
|
||||||
|
out.push_back(len);
|
||||||
|
out.insert(out.end(), str.begin(), str.end());
|
||||||
|
}
|
||||||
|
out.insert(out.begin() + 1, length);
|
||||||
|
|
||||||
|
if (add_checksum) {
|
||||||
|
uint32_t calculated_checksum = 0;
|
||||||
|
|
||||||
|
for (uint8_t byte : out) {
|
||||||
|
calculated_checksum += byte;
|
||||||
|
}
|
||||||
|
out.push_back(calculated_checksum);
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef ARDUINO
|
||||||
|
std::vector<uint8_t> build_rpc_response(Command command,
|
||||||
|
const std::vector<String> &datum,
|
||||||
|
bool add_checksum) {
|
||||||
|
std::vector<uint8_t> out;
|
||||||
|
uint32_t length = 0;
|
||||||
|
out.push_back(command);
|
||||||
|
for (const auto &str : datum) {
|
||||||
|
uint8_t len = str.length();
|
||||||
|
length += len;
|
||||||
|
out.push_back(len);
|
||||||
|
out.insert(out.end(), str.begin(), str.end());
|
||||||
|
}
|
||||||
|
out.insert(out.begin() + 1, length);
|
||||||
|
|
||||||
|
if (add_checksum) {
|
||||||
|
uint32_t calculated_checksum = 0;
|
||||||
|
|
||||||
|
for (uint8_t byte : out) {
|
||||||
|
calculated_checksum += byte;
|
||||||
|
}
|
||||||
|
out.push_back(calculated_checksum);
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
#endif // ARDUINO
|
||||||
|
|
||||||
|
} // namespace improv
|
86
src/lib/improv.hpp
Normal file
86
src/lib/improv.hpp
Normal file
|
@ -0,0 +1,86 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifdef ARDUINO
|
||||||
|
#include <Arduino.h>
|
||||||
|
#endif // ARDUINO
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
#include <functional>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace improv {
|
||||||
|
|
||||||
|
static const char *const SERVICE_UUID = "00467768-6228-2272-4663-277478268000";
|
||||||
|
static const char *const STATUS_UUID = "00467768-6228-2272-4663-277478268001";
|
||||||
|
static const char *const ERROR_UUID = "00467768-6228-2272-4663-277478268002";
|
||||||
|
static const char *const RPC_COMMAND_UUID =
|
||||||
|
"00467768-6228-2272-4663-277478268003";
|
||||||
|
static const char *const RPC_RESULT_UUID =
|
||||||
|
"00467768-6228-2272-4663-277478268004";
|
||||||
|
static const char *const CAPABILITIES_UUID =
|
||||||
|
"00467768-6228-2272-4663-277478268005";
|
||||||
|
|
||||||
|
enum Error : uint8_t {
|
||||||
|
ERROR_NONE = 0x00,
|
||||||
|
ERROR_INVALID_RPC = 0x01,
|
||||||
|
ERROR_UNKNOWN_RPC = 0x02,
|
||||||
|
ERROR_UNABLE_TO_CONNECT = 0x03,
|
||||||
|
ERROR_NOT_AUTHORIZED = 0x04,
|
||||||
|
ERROR_UNKNOWN = 0xFF,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum State : uint8_t {
|
||||||
|
STATE_STOPPED = 0x00,
|
||||||
|
STATE_AWAITING_AUTHORIZATION = 0x01,
|
||||||
|
STATE_AUTHORIZED = 0x02,
|
||||||
|
STATE_PROVISIONING = 0x03,
|
||||||
|
STATE_PROVISIONED = 0x04,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum Command : uint8_t {
|
||||||
|
UNKNOWN = 0x00,
|
||||||
|
WIFI_SETTINGS = 0x01,
|
||||||
|
IDENTIFY = 0x02,
|
||||||
|
GET_CURRENT_STATE = 0x02,
|
||||||
|
GET_DEVICE_INFO = 0x03,
|
||||||
|
GET_WIFI_NETWORKS = 0x04,
|
||||||
|
BAD_CHECKSUM = 0xFF,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const uint8_t CAPABILITY_IDENTIFY = 0x01;
|
||||||
|
static const uint8_t IMPROV_SERIAL_VERSION = 1;
|
||||||
|
|
||||||
|
enum ImprovSerialType : uint8_t {
|
||||||
|
TYPE_CURRENT_STATE = 0x01,
|
||||||
|
TYPE_ERROR_STATE = 0x02,
|
||||||
|
TYPE_RPC = 0x03,
|
||||||
|
TYPE_RPC_RESPONSE = 0x04
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ImprovCommand {
|
||||||
|
Command command;
|
||||||
|
std::string ssid;
|
||||||
|
std::string password;
|
||||||
|
};
|
||||||
|
|
||||||
|
ImprovCommand parse_improv_data(const std::vector<uint8_t> &data,
|
||||||
|
bool check_checksum = true);
|
||||||
|
ImprovCommand parse_improv_data(const uint8_t *data, size_t length,
|
||||||
|
bool check_checksum = true);
|
||||||
|
|
||||||
|
bool parse_improv_serial_byte(size_t position, uint8_t byte,
|
||||||
|
const uint8_t *buffer,
|
||||||
|
std::function<bool(ImprovCommand)> &&callback,
|
||||||
|
std::function<void(Error)> &&on_error);
|
||||||
|
|
||||||
|
std::vector<uint8_t> build_rpc_response(Command command,
|
||||||
|
const std::vector<std::string> &datum,
|
||||||
|
bool add_checksum = true);
|
||||||
|
#ifdef ARDUINO
|
||||||
|
std::vector<uint8_t> build_rpc_response(Command command,
|
||||||
|
const std::vector<String> &datum,
|
||||||
|
bool add_checksum = true);
|
||||||
|
#endif // ARDUINO
|
||||||
|
|
||||||
|
} // namespace improv
|
|
@ -5,443 +5,121 @@ QueueHandle_t ledTaskQueue = NULL;
|
||||||
Adafruit_NeoPixel pixels(NEOPIXEL_COUNT, NEOPIXEL_PIN, NEO_GRB + NEO_KHZ800);
|
Adafruit_NeoPixel pixels(NEOPIXEL_COUNT, NEOPIXEL_PIN, NEO_GRB + NEO_KHZ800);
|
||||||
uint ledTaskParams;
|
uint ledTaskParams;
|
||||||
|
|
||||||
#ifdef HAS_FRONTLIGHT
|
void ledTask(void *parameter) {
|
||||||
#define FL_FADE_STEP 25
|
while (1) {
|
||||||
|
if (ledTaskQueue != NULL) {
|
||||||
bool frontlightOn = false;
|
|
||||||
bool flInTransition = false;
|
|
||||||
|
|
||||||
void frontlightFlash(int flDelayTime)
|
|
||||||
{
|
|
||||||
if (preferences.getBool("flDisable"))
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (frontlightOn)
|
|
||||||
{
|
|
||||||
frontlightFadeOutAll(flDelayTime, true);
|
|
||||||
frontlightFadeInAll(flDelayTime, true);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
frontlightFadeInAll(flDelayTime, true);
|
|
||||||
frontlightFadeOutAll(flDelayTime, true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeInAll()
|
|
||||||
{
|
|
||||||
frontlightFadeInAll(preferences.getUInt("flEffectDelay"));
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeOutAll()
|
|
||||||
{
|
|
||||||
frontlightFadeOutAll(preferences.getUInt("flEffectDelay"));
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeIn(uint num)
|
|
||||||
{
|
|
||||||
frontlightFadeIn(num, preferences.getUInt("flEffectDelay"));
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeOut(uint num)
|
|
||||||
{
|
|
||||||
frontlightFadeOut(num, preferences.getUInt("flEffectDelay"));
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightSetBrightness(uint brightness)
|
|
||||||
{
|
|
||||||
if (brightness > 4096)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int ledPin = 0; ledPin <= NUM_SCREENS; ledPin++)
|
|
||||||
{
|
|
||||||
flArray.setPWM(ledPin, 0, brightness);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeInAll(int flDelayTime)
|
|
||||||
{
|
|
||||||
frontlightFadeInAll(flDelayTime, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeInAll(int flDelayTime, bool staggered)
|
|
||||||
{
|
|
||||||
if (preferences.getBool("flDisable"))
|
|
||||||
return;
|
|
||||||
if (frontlightIsOn())
|
|
||||||
return;
|
|
||||||
if (flInTransition)
|
|
||||||
return;
|
|
||||||
|
|
||||||
flInTransition = true;
|
|
||||||
|
|
||||||
if (staggered)
|
|
||||||
{
|
|
||||||
int maxBrightness = preferences.getUInt("flMaxBrightness");
|
|
||||||
int step = FL_FADE_STEP;
|
|
||||||
int staggerDelay = flDelayTime / NUM_SCREENS;
|
|
||||||
|
|
||||||
for (int dutyCycle = 0; dutyCycle <= maxBrightness + (NUM_SCREENS - 1) * maxBrightness / NUM_SCREENS; dutyCycle += step)
|
|
||||||
{
|
|
||||||
for (int ledPin = 0; ledPin < NUM_SCREENS; ledPin++)
|
|
||||||
{
|
|
||||||
int ledBrightness = dutyCycle - ledPin * maxBrightness / NUM_SCREENS;
|
|
||||||
if (ledBrightness < 0)
|
|
||||||
ledBrightness = 0;
|
|
||||||
else if (ledBrightness > maxBrightness)
|
|
||||||
ledBrightness = maxBrightness;
|
|
||||||
|
|
||||||
flArray.setPWM(ledPin + 1, 0, ledBrightness);
|
|
||||||
}
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(staggerDelay));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
for (int dutyCycle = 0; dutyCycle <= preferences.getUInt("flMaxBrightness"); dutyCycle += FL_FADE_STEP)
|
|
||||||
{
|
|
||||||
for (int ledPin = 0; ledPin <= NUM_SCREENS; ledPin++)
|
|
||||||
{
|
|
||||||
flArray.setPWM(ledPin, 0, dutyCycle);
|
|
||||||
}
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(flDelayTime));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
frontlightOn = true;
|
|
||||||
flInTransition = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeOutAll(int flDelayTime)
|
|
||||||
{
|
|
||||||
frontlightFadeOutAll(flDelayTime, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeOutAll(int flDelayTime, bool staggered)
|
|
||||||
{
|
|
||||||
if (preferences.getBool("flDisable"))
|
|
||||||
return;
|
|
||||||
if (!frontlightIsOn())
|
|
||||||
return;
|
|
||||||
if (flInTransition)
|
|
||||||
return;
|
|
||||||
flInTransition = true;
|
|
||||||
|
|
||||||
if (staggered)
|
|
||||||
{
|
|
||||||
int maxBrightness = preferences.getUInt("flMaxBrightness");
|
|
||||||
int step = FL_FADE_STEP;
|
|
||||||
int staggerDelay = flDelayTime / NUM_SCREENS;
|
|
||||||
|
|
||||||
for (int dutyCycle = maxBrightness; dutyCycle >= 0; dutyCycle -= step)
|
|
||||||
{
|
|
||||||
for (int ledPin = 0; ledPin < NUM_SCREENS; ledPin++)
|
|
||||||
{
|
|
||||||
int ledBrightness = dutyCycle - (NUM_SCREENS - 1 - ledPin) * maxBrightness / NUM_SCREENS;
|
|
||||||
if (ledBrightness < 0)
|
|
||||||
ledBrightness = 0;
|
|
||||||
else if (ledBrightness > maxBrightness)
|
|
||||||
ledBrightness = maxBrightness;
|
|
||||||
|
|
||||||
flArray.setPWM(ledPin + 1, 0, ledBrightness);
|
|
||||||
}
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(staggerDelay));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
for (int dutyCycle = preferences.getUInt("flMaxBrightness"); dutyCycle >= 0; dutyCycle -= FL_FADE_STEP)
|
|
||||||
{
|
|
||||||
for (int ledPin = 0; ledPin <= NUM_SCREENS; ledPin++)
|
|
||||||
{
|
|
||||||
flArray.setPWM(ledPin, 0, dutyCycle);
|
|
||||||
}
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(flDelayTime));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
flArray.allOFF();
|
|
||||||
frontlightOn = false;
|
|
||||||
flInTransition = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<uint16_t> frontlightGetStatus()
|
|
||||||
{
|
|
||||||
std::vector<uint16_t> statuses;
|
|
||||||
for (int ledPin = 1; ledPin <= NUM_SCREENS; ledPin++)
|
|
||||||
{
|
|
||||||
uint16_t a = 0, b = 0;
|
|
||||||
flArray.getPWM(ledPin, &a, &b);
|
|
||||||
statuses.push_back(round(b - a / 4096));
|
|
||||||
}
|
|
||||||
|
|
||||||
return statuses;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool frontlightIsOn()
|
|
||||||
{
|
|
||||||
return frontlightOn;
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeIn(uint num, int flDelayTime)
|
|
||||||
{
|
|
||||||
if (preferences.getBool("flDisable"))
|
|
||||||
return;
|
|
||||||
for (int dutyCycle = 0; dutyCycle <= preferences.getUInt("flMaxBrightness"); dutyCycle += 5)
|
|
||||||
{
|
|
||||||
flArray.setPWM(num, 0, dutyCycle);
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(flDelayTime));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void frontlightFadeOut(uint num, int flDelayTime)
|
|
||||||
{
|
|
||||||
if (preferences.getBool("flDisable"))
|
|
||||||
return;
|
|
||||||
if (!frontlightIsOn())
|
|
||||||
return;
|
|
||||||
|
|
||||||
for (int dutyCycle = preferences.getUInt("flMaxBrightness"); dutyCycle >= 0; dutyCycle -= 5)
|
|
||||||
{
|
|
||||||
flArray.setPWM(num, 0, dutyCycle);
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(flDelayTime));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void ledTask(void *parameter)
|
|
||||||
{
|
|
||||||
while (1)
|
|
||||||
{
|
|
||||||
if (ledTaskQueue != NULL)
|
|
||||||
{
|
|
||||||
if (xQueueReceive(ledTaskQueue, &ledTaskParams, portMAX_DELAY) ==
|
if (xQueueReceive(ledTaskQueue, &ledTaskParams, portMAX_DELAY) ==
|
||||||
pdPASS)
|
pdPASS) {
|
||||||
{
|
|
||||||
|
|
||||||
if (preferences.getBool("disableLeds", DEFAULT_DISABLE_LEDS))
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t oldLights[NEOPIXEL_COUNT];
|
uint32_t oldLights[NEOPIXEL_COUNT];
|
||||||
|
|
||||||
// get current state
|
// get current state
|
||||||
for (int i = 0; i < NEOPIXEL_COUNT; i++)
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
||||||
{
|
|
||||||
oldLights[i] = pixels.getPixelColor(i);
|
oldLights[i] = pixels.getPixelColor(i);
|
||||||
}
|
}
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
uint flDelayTime = preferences.getUInt("flEffectDelay");
|
|
||||||
#endif
|
|
||||||
switch (ledTaskParams)
|
|
||||||
{
|
|
||||||
case LED_POWER_TEST:
|
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
frontlightFadeInAll(preferences.getUInt("flEffectDelay"), true);
|
|
||||||
#endif
|
|
||||||
ledRainbow(20);
|
|
||||||
pixels.clear();
|
|
||||||
break;
|
|
||||||
case LED_EFFECT_WIFI_CONNECT_ERROR:
|
|
||||||
blinkDelayTwoColor(100, 3, pixels.Color(8, 161, 236),
|
|
||||||
pixels.Color(255, 0, 0));
|
|
||||||
break;
|
|
||||||
case LED_FLASH_ERROR:
|
|
||||||
blinkDelayColor(250, 3, 255, 0, 0);
|
|
||||||
break;
|
|
||||||
case LED_EFFECT_HEARTBEAT:
|
|
||||||
blinkDelayColor(150, 2, 0, 0, 255);
|
|
||||||
break;
|
|
||||||
case LED_DATA_BLOCK_ERROR:
|
|
||||||
blinkDelayColor(150, 2, 128, 0, 128);
|
|
||||||
break;
|
|
||||||
case LED_DATA_PRICE_ERROR:
|
|
||||||
blinkDelayColor(150, 2, 177, 90, 31);
|
|
||||||
break;
|
|
||||||
case LED_FLASH_IDENTIFY:
|
|
||||||
blinkDelayTwoColor(100, 2, pixels.Color(255, 0, 0),
|
|
||||||
pixels.Color(0, 255, 255));
|
|
||||||
blinkDelayTwoColor(100, 2, pixels.Color(0, 255, 0),
|
|
||||||
pixels.Color(0, 0, 255));
|
|
||||||
break;
|
|
||||||
case LED_EFFECT_WIFI_CONNECT_SUCCESS:
|
|
||||||
case LED_FLASH_SUCCESS:
|
|
||||||
blinkDelayColor(150, 3, 0, 255, 0);
|
|
||||||
break;
|
|
||||||
case LED_PROGRESS_100:
|
|
||||||
pixels.setPixelColor(0, pixels.Color(0, 255, 0));
|
|
||||||
case LED_PROGRESS_75:
|
|
||||||
pixels.setPixelColor(1, pixels.Color(0, 255, 0));
|
|
||||||
case LED_PROGRESS_50:
|
|
||||||
pixels.setPixelColor(2, pixels.Color(0, 255, 0));
|
|
||||||
case LED_PROGRESS_25:
|
|
||||||
pixels.setPixelColor(3, pixels.Color(0, 255, 0));
|
|
||||||
pixels.show();
|
|
||||||
break;
|
|
||||||
case LED_EFFECT_NOSTR_ZAP:
|
|
||||||
{
|
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
bool frontlightWasOn = false;
|
|
||||||
|
|
||||||
if (preferences.getBool("flFlashOnZap", DEFAULT_FL_FLASH_ON_ZAP))
|
switch (ledTaskParams) {
|
||||||
{
|
case LED_POWER_TEST:
|
||||||
if (frontlightOn)
|
ledRainbow(20);
|
||||||
{
|
pixels.clear();
|
||||||
frontlightWasOn = true;
|
break;
|
||||||
frontlightFadeOutAll(flDelayTime, true);
|
case LED_EFFECT_WIFI_CONNECT_ERROR:
|
||||||
}
|
blinkDelayTwoColor(100, 3, pixels.Color(8, 161, 236),
|
||||||
else
|
pixels.Color(255, 0, 0));
|
||||||
{
|
break;
|
||||||
frontlightFadeInAll(flDelayTime, true);
|
case LED_FLASH_ERROR:
|
||||||
}
|
blinkDelayColor(250, 3, 255, 0, 0);
|
||||||
}
|
break;
|
||||||
#endif
|
case LED_EFFECT_HEARTBEAT:
|
||||||
for (int flash = 0; flash < random(7, 10); flash++)
|
blinkDelayColor(150, 2, 0, 0, 255);
|
||||||
{
|
break;
|
||||||
lightningStrike();
|
case LED_EFFECT_WIFI_CONNECT_SUCCESS:
|
||||||
delay(random(50, 150));
|
case LED_FLASH_SUCCESS:
|
||||||
}
|
blinkDelayColor(150, 3, 0, 255, 0);
|
||||||
// blinkDelayColor(250, 3, 142, 48, 235);
|
break;
|
||||||
// blinkDelayTwoColor(250, 3, pixels.Color(142, 48, 235),
|
case LED_PROGRESS_100:
|
||||||
// pixels.Color(169, 21, 255));
|
pixels.setPixelColor(0, pixels.Color(0, 255, 0));
|
||||||
#ifdef HAS_FRONTLIGHT
|
case LED_PROGRESS_75:
|
||||||
if (preferences.getBool("flFlashOnZap", DEFAULT_FL_FLASH_ON_ZAP))
|
pixels.setPixelColor(1, pixels.Color(0, 255, 0));
|
||||||
{
|
case LED_PROGRESS_50:
|
||||||
vTaskDelay(pdMS_TO_TICKS(10));
|
pixels.setPixelColor(2, pixels.Color(0, 255, 0));
|
||||||
if (frontlightWasOn)
|
case LED_PROGRESS_25:
|
||||||
{
|
pixels.setPixelColor(3, pixels.Color(0, 255, 0));
|
||||||
frontlightFadeInAll(flDelayTime, true);
|
pixels.show();
|
||||||
}
|
break;
|
||||||
else
|
case LED_FLASH_UPDATE:
|
||||||
{
|
break;
|
||||||
frontlightFadeOutAll(flDelayTime, true);
|
case LED_FLASH_BLOCK_NOTIFY:
|
||||||
}
|
blinkDelayTwoColor(250, 3, pixels.Color(224, 67, 0),
|
||||||
}
|
pixels.Color(8, 2, 0));
|
||||||
#endif
|
break;
|
||||||
break;
|
case LED_EFFECT_WIFI_WAIT_FOR_CONFIG:
|
||||||
}
|
blinkDelayTwoColor(100, 1, pixels.Color(8, 161, 236),
|
||||||
case LED_FLASH_UPDATE:
|
pixels.Color(156, 225, 240));
|
||||||
blinkDelayTwoColor(250, 3, pixels.Color(0, 230, 0),
|
break;
|
||||||
pixels.Color(230, 230, 0));
|
case LED_EFFECT_WIFI_ERASE_SETTINGS:
|
||||||
break;
|
blinkDelay(100, 3);
|
||||||
case LED_FLASH_BLOCK_NOTIFY:
|
break;
|
||||||
{
|
case LED_EFFECT_WIFI_CONNECTING:
|
||||||
#ifdef HAS_FRONTLIGHT
|
for (int i = NEOPIXEL_COUNT; i >= 0; i--) {
|
||||||
bool frontlightWasOn = false;
|
for (int j = NEOPIXEL_COUNT; j >= 0; j--) {
|
||||||
|
if (j == i) {
|
||||||
if (preferences.getBool("flFlashOnUpd", DEFAULT_FL_FLASH_ON_UPDATE))
|
pixels.setPixelColor(i, pixels.Color(16, 197, 236));
|
||||||
{
|
} else {
|
||||||
if (frontlightOn)
|
pixels.setPixelColor(j, pixels.Color(0, 0, 0));
|
||||||
{
|
}
|
||||||
frontlightWasOn = true;
|
|
||||||
frontlightFadeOutAll(flDelayTime, true);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
frontlightFadeInAll(flDelayTime, true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
blinkDelayTwoColor(250, 3, pixels.Color(224, 67, 0),
|
|
||||||
pixels.Color(8, 2, 0));
|
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
if (preferences.getBool("flFlashOnUpd", DEFAULT_FL_FLASH_ON_UPDATE))
|
|
||||||
{
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(10));
|
|
||||||
if (frontlightWasOn)
|
|
||||||
{
|
|
||||||
frontlightFadeInAll(flDelayTime, true);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
frontlightFadeOutAll(flDelayTime, true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case LED_EFFECT_WIFI_WAIT_FOR_CONFIG:
|
|
||||||
blinkDelayTwoColor(100, 1, pixels.Color(8, 161, 236),
|
|
||||||
pixels.Color(156, 225, 240));
|
|
||||||
break;
|
|
||||||
case LED_EFFECT_WIFI_ERASE_SETTINGS:
|
|
||||||
blinkDelay(100, 3);
|
|
||||||
break;
|
|
||||||
case LED_EFFECT_WIFI_CONNECTING:
|
|
||||||
for (int i = NEOPIXEL_COUNT; i >= 0; i--)
|
|
||||||
{
|
|
||||||
for (int j = NEOPIXEL_COUNT; j >= 0; j--)
|
|
||||||
{
|
|
||||||
if (j == i)
|
|
||||||
{
|
|
||||||
pixels.setPixelColor(i, pixels.Color(16, 197, 236));
|
|
||||||
}
|
}
|
||||||
else
|
pixels.show();
|
||||||
{
|
vTaskDelay(pdMS_TO_TICKS(100));
|
||||||
pixels.setPixelColor(j, pixels.Color(0, 0, 0));
|
}
|
||||||
|
break;
|
||||||
|
case LED_EFFECT_PAUSE_TIMER:
|
||||||
|
for (int i = NEOPIXEL_COUNT; i >= 0; i--) {
|
||||||
|
for (int j = NEOPIXEL_COUNT; j >= 0; j--) {
|
||||||
|
uint32_t c = pixels.Color(0, 0, 0);
|
||||||
|
if (i == j) c = pixels.Color(0, 255, 0);
|
||||||
|
pixels.setPixelColor(j, c);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
pixels.show();
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(100));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case LED_EFFECT_PAUSE_TIMER:
|
|
||||||
for (int i = NEOPIXEL_COUNT; i >= 0; i--)
|
|
||||||
{
|
|
||||||
for (int j = NEOPIXEL_COUNT; j >= 0; j--)
|
|
||||||
{
|
|
||||||
uint32_t c = pixels.Color(0, 0, 0);
|
|
||||||
if (i == j)
|
|
||||||
c = pixels.Color(0, 255, 0);
|
|
||||||
pixels.setPixelColor(j, c);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
pixels.show();
|
||||||
|
|
||||||
|
delay(100);
|
||||||
|
}
|
||||||
|
pixels.setPixelColor(0, pixels.Color(255, 0, 0));
|
||||||
pixels.show();
|
pixels.show();
|
||||||
|
|
||||||
delay(100);
|
delay(900);
|
||||||
}
|
|
||||||
pixels.setPixelColor(0, pixels.Color(255, 0, 0));
|
|
||||||
pixels.show();
|
|
||||||
|
|
||||||
delay(900);
|
|
||||||
|
|
||||||
pixels.clear();
|
|
||||||
pixels.show();
|
|
||||||
break;
|
|
||||||
case LED_EFFECT_START_TIMER:
|
|
||||||
pixels.clear();
|
|
||||||
pixels.setPixelColor((NEOPIXEL_COUNT - 1), pixels.Color(255, 0, 0));
|
|
||||||
pixels.show();
|
|
||||||
|
|
||||||
delay(900);
|
|
||||||
|
|
||||||
for (int i = NEOPIXEL_COUNT; i--; i > 0)
|
|
||||||
{
|
|
||||||
for (int j = NEOPIXEL_COUNT; j--; j > 0)
|
|
||||||
{
|
|
||||||
uint32_t c = pixels.Color(0, 0, 0);
|
|
||||||
if (i == j)
|
|
||||||
c = pixels.Color(0, 255, 0);
|
|
||||||
|
|
||||||
pixels.setPixelColor(j, c);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
pixels.clear();
|
||||||
|
pixels.show();
|
||||||
|
break;
|
||||||
|
case LED_EFFECT_START_TIMER:
|
||||||
|
pixels.clear();
|
||||||
|
pixels.setPixelColor((NEOPIXEL_COUNT - 1), pixels.Color(255, 0, 0));
|
||||||
pixels.show();
|
pixels.show();
|
||||||
|
|
||||||
delay(100);
|
delay(900);
|
||||||
}
|
|
||||||
|
|
||||||
pixels.clear();
|
for (int i = NEOPIXEL_COUNT; i--; i > 0) {
|
||||||
pixels.show();
|
for (int j = NEOPIXEL_COUNT; j--; j > 0) {
|
||||||
break;
|
uint32_t c = pixels.Color(0, 0, 0);
|
||||||
|
if (i == j) c = pixels.Color(0, 255, 0);
|
||||||
|
|
||||||
|
pixels.setPixelColor(j, c);
|
||||||
|
}
|
||||||
|
|
||||||
|
pixels.show();
|
||||||
|
|
||||||
|
delay(100);
|
||||||
|
}
|
||||||
|
|
||||||
|
pixels.clear();
|
||||||
|
pixels.show();
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// revert to previous state unless power test
|
// revert to previous state unless power test
|
||||||
|
|
||||||
for (int i = 0; i < NEOPIXEL_COUNT; i++)
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
||||||
{
|
|
||||||
pixels.setPixelColor(i, oldLights[i]);
|
pixels.setPixelColor(i, oldLights[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -451,17 +129,14 @@ void ledTask(void *parameter)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void setupLeds()
|
void setupLeds() {
|
||||||
{
|
|
||||||
pixels.begin();
|
pixels.begin();
|
||||||
pixels.setBrightness(preferences.getUInt("ledBrightness", DEFAULT_LED_BRIGHTNESS));
|
pixels.setBrightness(preferences.getUInt("ledBrightness", 128));
|
||||||
pixels.clear();
|
pixels.clear();
|
||||||
pixels.show();
|
pixels.show();
|
||||||
setupLedTask();
|
setupLedTask();
|
||||||
if (preferences.getBool("ledTestOnPower", DEFAULT_LED_TEST_ON_POWER))
|
if (preferences.getBool("ledTestOnPower", true)) {
|
||||||
{
|
while (!ledTaskQueue) {
|
||||||
while (!ledTaskQueue)
|
|
||||||
{
|
|
||||||
delay(1);
|
delay(1);
|
||||||
// wait until queue is available
|
// wait until queue is available
|
||||||
}
|
}
|
||||||
|
@ -469,17 +144,14 @@ void setupLeds()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void setupLedTask()
|
void setupLedTask() {
|
||||||
{
|
|
||||||
ledTaskQueue = xQueueCreate(5, sizeof(uint));
|
ledTaskQueue = xQueueCreate(5, sizeof(uint));
|
||||||
|
|
||||||
xTaskCreate(ledTask, "LedTask", 2048, NULL, 10, &ledTaskHandle);
|
xTaskCreate(ledTask, "LedTask", 2048, NULL, tskIDLE_PRIORITY, &ledTaskHandle);
|
||||||
}
|
}
|
||||||
|
|
||||||
void blinkDelay(int d, int times)
|
void blinkDelay(int d, int times) {
|
||||||
{
|
for (int j = 0; j < times; j++) {
|
||||||
for (int j = 0; j < times; j++)
|
|
||||||
{
|
|
||||||
pixels.setPixelColor(0, pixels.Color(255, 0, 0));
|
pixels.setPixelColor(0, pixels.Color(255, 0, 0));
|
||||||
pixels.setPixelColor(1, pixels.Color(0, 255, 0));
|
pixels.setPixelColor(1, pixels.Color(0, 255, 0));
|
||||||
pixels.setPixelColor(2, pixels.Color(255, 0, 0));
|
pixels.setPixelColor(2, pixels.Color(255, 0, 0));
|
||||||
|
@ -498,12 +170,9 @@ void blinkDelay(int d, int times)
|
||||||
pixels.show();
|
pixels.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
void blinkDelayColor(int d, int times, uint r, uint g, uint b)
|
void blinkDelayColor(int d, int times, uint r, uint g, uint b) {
|
||||||
{
|
for (int j = 0; j < times; j++) {
|
||||||
for (int j = 0; j < times; j++)
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
||||||
{
|
|
||||||
for (int i = 0; i < NEOPIXEL_COUNT; i++)
|
|
||||||
{
|
|
||||||
pixels.setPixelColor(i, pixels.Color(r, g, b));
|
pixels.setPixelColor(i, pixels.Color(r, g, b));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -518,19 +187,15 @@ void blinkDelayColor(int d, int times, uint r, uint g, uint b)
|
||||||
pixels.show();
|
pixels.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
void blinkDelayTwoColor(int d, int times, uint32_t c1, uint32_t c2)
|
void blinkDelayTwoColor(int d, int times, uint32_t c1, uint32_t c2) {
|
||||||
{
|
for (int j = 0; j < times; j++) {
|
||||||
for (int j = 0; j < times; j++)
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
||||||
{
|
|
||||||
for (int i = 0; i < NEOPIXEL_COUNT; i++)
|
|
||||||
{
|
|
||||||
pixels.setPixelColor(i, c1);
|
pixels.setPixelColor(i, c1);
|
||||||
}
|
}
|
||||||
pixels.show();
|
pixels.show();
|
||||||
vTaskDelay(pdMS_TO_TICKS(d));
|
vTaskDelay(pdMS_TO_TICKS(d));
|
||||||
|
|
||||||
for (int i = 0; i < NEOPIXEL_COUNT; i++)
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
||||||
{
|
|
||||||
pixels.setPixelColor(i, c2);
|
pixels.setPixelColor(i, c2);
|
||||||
}
|
}
|
||||||
pixels.show();
|
pixels.show();
|
||||||
|
@ -540,8 +205,7 @@ void blinkDelayTwoColor(int d, int times, uint32_t c1, uint32_t c2)
|
||||||
pixels.show();
|
pixels.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
void clearLeds()
|
void clearLeds() {
|
||||||
{
|
|
||||||
preferences.putBool("ledStatus", false);
|
preferences.putBool("ledStatus", false);
|
||||||
pixels.clear();
|
pixels.clear();
|
||||||
pixels.show();
|
pixels.show();
|
||||||
|
@ -549,31 +213,24 @@ void clearLeds()
|
||||||
|
|
||||||
void setLights(int r, int g, int b) { setLights(pixels.Color(r, g, b)); }
|
void setLights(int r, int g, int b) { setLights(pixels.Color(r, g, b)); }
|
||||||
|
|
||||||
void setLights(uint32_t color)
|
void setLights(uint32_t color) {
|
||||||
{
|
|
||||||
bool ledStatus = true;
|
bool ledStatus = true;
|
||||||
|
|
||||||
for (int i = 0; i < NEOPIXEL_COUNT; i++)
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
||||||
{
|
|
||||||
pixels.setPixelColor(i, color);
|
pixels.setPixelColor(i, color);
|
||||||
}
|
}
|
||||||
pixels.show();
|
pixels.show();
|
||||||
|
|
||||||
if (color == pixels.Color(0, 0, 0))
|
if (color == pixels.Color(0, 0, 0)) {
|
||||||
{
|
|
||||||
ledStatus = false;
|
ledStatus = false;
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
saveLedState();
|
saveLedState();
|
||||||
}
|
}
|
||||||
preferences.putBool("ledStatus", ledStatus);
|
preferences.putBool("ledStatus", ledStatus);
|
||||||
}
|
}
|
||||||
|
|
||||||
void saveLedState()
|
void saveLedState() {
|
||||||
{
|
for (int i = 0; i < pixels.numPixels(); i++) {
|
||||||
for (int i = 0; i < pixels.numPixels(); i++)
|
|
||||||
{
|
|
||||||
int pixelColor = pixels.getPixelColor(i);
|
int pixelColor = pixels.getPixelColor(i);
|
||||||
char key[12];
|
char key[12];
|
||||||
snprintf(key, 12, "%s%d", "ledColor_", i);
|
snprintf(key, 12, "%s%d", "ledColor_", i);
|
||||||
|
@ -583,10 +240,8 @@ void saveLedState()
|
||||||
xTaskNotifyGive(eventSourceTaskHandle);
|
xTaskNotifyGive(eventSourceTaskHandle);
|
||||||
}
|
}
|
||||||
|
|
||||||
void restoreLedState()
|
void restoreLedState() {
|
||||||
{
|
for (int i = 0; i < pixels.numPixels(); i++) {
|
||||||
for (int i = 0; i < pixels.numPixels(); i++)
|
|
||||||
{
|
|
||||||
char key[12];
|
char key[12];
|
||||||
snprintf(key, 12, "%s%d", "ledColor_", i);
|
snprintf(key, 12, "%s%d", "ledColor_", i);
|
||||||
uint pixelColor = preferences.getUInt(key, pixels.Color(0, 0, 0));
|
uint pixelColor = preferences.getUInt(key, pixels.Color(0, 0, 0));
|
||||||
|
@ -598,10 +253,8 @@ void restoreLedState()
|
||||||
|
|
||||||
QueueHandle_t getLedTaskQueue() { return ledTaskQueue; }
|
QueueHandle_t getLedTaskQueue() { return ledTaskQueue; }
|
||||||
|
|
||||||
bool queueLedEffect(uint effect)
|
bool queueLedEffect(uint effect) {
|
||||||
{
|
if (ledTaskQueue == NULL) {
|
||||||
if (ledTaskQueue == NULL)
|
|
||||||
{
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -609,15 +262,13 @@ bool queueLedEffect(uint effect)
|
||||||
xQueueSend(ledTaskQueue, &flashType, portMAX_DELAY);
|
xQueueSend(ledTaskQueue, &flashType, portMAX_DELAY);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ledRainbow(int wait)
|
void ledRainbow(int wait) {
|
||||||
{
|
|
||||||
// Hue of first pixel runs 5 complete loops through the color wheel.
|
// Hue of first pixel runs 5 complete loops through the color wheel.
|
||||||
// Color wheel has a range of 65536 but it's OK if we roll over, so
|
// Color wheel has a range of 65536 but it's OK if we roll over, so
|
||||||
// just count from 0 to 5*65536. Adding 256 to firstPixelHue each time
|
// just count from 0 to 5*65536. Adding 256 to firstPixelHue each time
|
||||||
// means we'll make 5*65536/256 = 1280 passes through this loop:
|
// means we'll make 5*65536/256 = 1280 passes through this loop:
|
||||||
for (long firstPixelHue = 0; firstPixelHue < 5 * 65536;
|
for (long firstPixelHue = 0; firstPixelHue < 5 * 65536;
|
||||||
firstPixelHue += 256)
|
firstPixelHue += 256) {
|
||||||
{
|
|
||||||
// strip.rainbow() can take a single argument (first pixel hue) or
|
// strip.rainbow() can take a single argument (first pixel hue) or
|
||||||
// optionally a few extras: number of rainbow repetitions (default 1),
|
// optionally a few extras: number of rainbow repetitions (default 1),
|
||||||
// saturation and value (brightness) (both 0-255, similar to the
|
// saturation and value (brightness) (both 0-255, similar to the
|
||||||
|
@ -626,78 +277,45 @@ void ledRainbow(int wait)
|
||||||
pixels.rainbow(firstPixelHue);
|
pixels.rainbow(firstPixelHue);
|
||||||
// Above line is equivalent to:
|
// Above line is equivalent to:
|
||||||
// strip.rainbow(firstPixelHue, 1, 255, 255, true);
|
// strip.rainbow(firstPixelHue, 1, 255, 255, true);
|
||||||
pixels.show(); // Update strip with new contents
|
pixels.show(); // Update strip with new contents
|
||||||
delayMicroseconds(wait);
|
delayMicroseconds(wait);
|
||||||
// vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
// vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ledTheaterChase(uint32_t color, int wait)
|
void ledTheaterChase(uint32_t color, int wait) {
|
||||||
{
|
for (int a = 0; a < 10; a++) { // Repeat 10 times...
|
||||||
for (int a = 0; a < 10; a++)
|
for (int b = 0; b < 3; b++) { // 'b' counts from 0 to 2...
|
||||||
{ // Repeat 10 times...
|
pixels.clear(); // Set all pixels in RAM to 0 (off)
|
||||||
for (int b = 0; b < 3; b++)
|
|
||||||
{ // 'b' counts from 0 to 2...
|
|
||||||
pixels.clear(); // Set all pixels in RAM to 0 (off)
|
|
||||||
// 'c' counts up from 'b' to end of strip in steps of 3...
|
// 'c' counts up from 'b' to end of strip in steps of 3...
|
||||||
for (int c = b; c < pixels.numPixels(); c += 3)
|
for (int c = b; c < pixels.numPixels(); c += 3) {
|
||||||
{
|
pixels.setPixelColor(c, color); // Set pixel 'c' to value 'color'
|
||||||
pixels.setPixelColor(c, color); // Set pixel 'c' to value 'color'
|
|
||||||
}
|
}
|
||||||
pixels.show(); // Update strip with new contents
|
pixels.show(); // Update strip with new contents
|
||||||
vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ledTheaterChaseRainbow(int wait)
|
void ledTheaterChaseRainbow(int wait) {
|
||||||
{
|
int firstPixelHue = 0; // First pixel starts at red (hue 0)
|
||||||
int firstPixelHue = 0; // First pixel starts at red (hue 0)
|
for (int a = 0; a < 30; a++) { // Repeat 30 times...
|
||||||
for (int a = 0; a < 30; a++)
|
for (int b = 0; b < 3; b++) { // 'b' counts from 0 to 2...
|
||||||
{ // Repeat 30 times...
|
pixels.clear(); // Set all pixels in RAM to 0 (off)
|
||||||
for (int b = 0; b < 3; b++)
|
|
||||||
{ // 'b' counts from 0 to 2...
|
|
||||||
pixels.clear(); // Set all pixels in RAM to 0 (off)
|
|
||||||
// 'c' counts up from 'b' to end of strip in increments of 3...
|
// 'c' counts up from 'b' to end of strip in increments of 3...
|
||||||
for (int c = b; c < pixels.numPixels(); c += 3)
|
for (int c = b; c < pixels.numPixels(); c += 3) {
|
||||||
{
|
|
||||||
// hue of pixel 'c' is offset by an amount to make one full
|
// hue of pixel 'c' is offset by an amount to make one full
|
||||||
// revolution of the color wheel (range 65536) along the length
|
// revolution of the color wheel (range 65536) along the length
|
||||||
// of the strip (strip.numPixels() steps):
|
// of the strip (strip.numPixels() steps):
|
||||||
int hue = firstPixelHue + c * 65536L / pixels.numPixels();
|
int hue = firstPixelHue + c * 65536L / pixels.numPixels();
|
||||||
uint32_t color = pixels.gamma32(pixels.ColorHSV(hue)); // hue -> RGB
|
uint32_t color = pixels.gamma32(pixels.ColorHSV(hue)); // hue -> RGB
|
||||||
pixels.setPixelColor(c, color); // Set pixel 'c' to value 'color'
|
pixels.setPixelColor(c, color); // Set pixel 'c' to value 'color'
|
||||||
}
|
}
|
||||||
pixels.show(); // Update strip with new contents
|
pixels.show(); // Update strip with new contents
|
||||||
vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
||||||
firstPixelHue += 65536 / 90; // One cycle of color wheel over 90 frames
|
firstPixelHue += 65536 / 90; // One cycle of color wheel over 90 frames
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void lightningStrike()
|
|
||||||
{
|
|
||||||
uint32_t PURPLE = pixels.Color(128, 0, 128);
|
|
||||||
uint32_t YELLOW = pixels.Color(255, 226, 41);
|
|
||||||
|
|
||||||
// Randomly choose which LEDs to light up
|
|
||||||
for (int i = 0; i < pixels.numPixels(); i++)
|
|
||||||
{
|
|
||||||
if (random(2) == 0)
|
|
||||||
{ // 50% chance for each LED
|
|
||||||
pixels.setPixelColor(i, YELLOW);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
pixels.setPixelColor(i, PURPLE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pixels.show();
|
|
||||||
|
|
||||||
delay(random(10, 50)); // Flash duration
|
|
||||||
|
|
||||||
// Return to purple background
|
|
||||||
// setAllPixels(PURPLE);
|
|
||||||
}
|
|
||||||
|
|
||||||
Adafruit_NeoPixel getPixels() { return pixels; }
|
Adafruit_NeoPixel getPixels() { return pixels; }
|
|
@ -27,19 +27,10 @@ const int LED_EFFECT_WIFI_CONNECTING = 101;
|
||||||
const int LED_EFFECT_WIFI_CONNECT_ERROR = 102;
|
const int LED_EFFECT_WIFI_CONNECT_ERROR = 102;
|
||||||
const int LED_EFFECT_WIFI_CONNECT_SUCCESS = 103;
|
const int LED_EFFECT_WIFI_CONNECT_SUCCESS = 103;
|
||||||
const int LED_EFFECT_WIFI_ERASE_SETTINGS = 104;
|
const int LED_EFFECT_WIFI_ERASE_SETTINGS = 104;
|
||||||
|
|
||||||
|
|
||||||
const int LED_PROGRESS_25 = 200;
|
const int LED_PROGRESS_25 = 200;
|
||||||
const int LED_PROGRESS_50 = 201;
|
const int LED_PROGRESS_50 = 201;
|
||||||
const int LED_PROGRESS_75 = 202;
|
const int LED_PROGRESS_75 = 202;
|
||||||
const int LED_PROGRESS_100 = 203;
|
const int LED_PROGRESS_100 = 203;
|
||||||
|
|
||||||
const int LED_DATA_PRICE_ERROR = 300;
|
|
||||||
const int LED_DATA_BLOCK_ERROR = 301;
|
|
||||||
|
|
||||||
const int LED_EFFECT_NOSTR_ZAP = 400;
|
|
||||||
|
|
||||||
const int LED_FLASH_IDENTIFY = 990;
|
|
||||||
const int LED_POWER_TEST = 999;
|
const int LED_POWER_TEST = 999;
|
||||||
extern TaskHandle_t ledTaskHandle;
|
extern TaskHandle_t ledTaskHandle;
|
||||||
extern Adafruit_NeoPixel pixels;
|
extern Adafruit_NeoPixel pixels;
|
||||||
|
@ -61,25 +52,3 @@ void ledRainbow(int wait);
|
||||||
void ledTheaterChaseRainbow(int wait);
|
void ledTheaterChaseRainbow(int wait);
|
||||||
void ledTheaterChase(uint32_t color, int wait);
|
void ledTheaterChase(uint32_t color, int wait);
|
||||||
Adafruit_NeoPixel getPixels();
|
Adafruit_NeoPixel getPixels();
|
||||||
void lightningStrike();
|
|
||||||
|
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
void frontlightFlash(int flDelayTime);
|
|
||||||
void frontlightFadeInAll();
|
|
||||||
void frontlightFadeOutAll();
|
|
||||||
void frontlightFadeIn(uint num);
|
|
||||||
void frontlightFadeOut(uint num);
|
|
||||||
|
|
||||||
std::vector<uint16_t> frontlightGetStatus();
|
|
||||||
|
|
||||||
void frontlightSetBrightness(uint brightness);
|
|
||||||
bool frontlightIsOn();
|
|
||||||
|
|
||||||
void frontlightFadeInAll(int flDelayTime);
|
|
||||||
void frontlightFadeInAll(int flDelayTime, bool staggered);
|
|
||||||
void frontlightFadeOutAll(int flDelayTime);
|
|
||||||
void frontlightFadeOutAll(int flDelayTime, bool staggered);
|
|
||||||
|
|
||||||
void frontlightFadeIn(uint num, int flDelayTime);
|
|
||||||
void frontlightFadeOut(uint num, int flDelayTime);
|
|
||||||
#endif
|
|
|
@ -1,267 +0,0 @@
|
||||||
#include "nostr_notify.hpp"
|
|
||||||
|
|
||||||
std::vector<nostr::NostrPool *> pools;
|
|
||||||
nostr::Transport *transport;
|
|
||||||
TaskHandle_t nostrTaskHandle = NULL;
|
|
||||||
boolean nostrIsConnected = false;
|
|
||||||
boolean nostrIsSubscribed = false;
|
|
||||||
boolean nostrIsSubscribing = true;
|
|
||||||
|
|
||||||
String subIdZap;
|
|
||||||
|
|
||||||
void setupNostrNotify(bool asDatasource, bool zapNotify)
|
|
||||||
{
|
|
||||||
nostr::esp32::ESP32Platform::initNostr(false);
|
|
||||||
// time_t now;
|
|
||||||
// time(&now);
|
|
||||||
// struct tm *utcTimeInfo;
|
|
||||||
// utcTimeInfo = gmtime(&now);
|
|
||||||
// time_t utcNow = mktime(utcTimeInfo);
|
|
||||||
// time_t timestamp60MinutesAgo = utcNow - 3600;
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
transport = nostr::esp32::ESP32Platform::getTransport();
|
|
||||||
nostr::NostrPool *pool = new nostr::NostrPool(transport);
|
|
||||||
String relay = preferences.getString("nostrRelay");
|
|
||||||
String pubKey = preferences.getString("nostrPubKey");
|
|
||||||
pools.push_back(pool);
|
|
||||||
|
|
||||||
std::vector<std::map<NostrString, std::initializer_list<NostrString>>> filters;
|
|
||||||
|
|
||||||
if (zapNotify)
|
|
||||||
{
|
|
||||||
subscribeZaps(pool, relay, 60);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (asDatasource)
|
|
||||||
{
|
|
||||||
String subId = pool->subscribeMany(
|
|
||||||
{relay},
|
|
||||||
{// First filter
|
|
||||||
{
|
|
||||||
{"kinds", {"1"}},
|
|
||||||
{"since", {String(getMinutesAgo(60))}},
|
|
||||||
{"authors", {pubKey}},
|
|
||||||
}},
|
|
||||||
handleNostrEventCallback,
|
|
||||||
onNostrSubscriptionClosed,
|
|
||||||
onNostrSubscriptionEose);
|
|
||||||
|
|
||||||
Serial.println("[ Nostr ] Subscribing to Nostr Data Feed");
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<nostr::NostrRelay *> *relays = pool->getConnectedRelays();
|
|
||||||
for (nostr::NostrRelay *relay : *relays)
|
|
||||||
{
|
|
||||||
Serial.println("[ Nostr ] Registering to connection events of: " + relay->getUrl());
|
|
||||||
relay->getConnection()->addConnectionStatusListener([&](const nostr::ConnectionStatus &status)
|
|
||||||
{
|
|
||||||
String sstatus="UNKNOWN";
|
|
||||||
if(status==nostr::ConnectionStatus::CONNECTED){
|
|
||||||
nostrIsConnected = true;
|
|
||||||
sstatus="CONNECTED";
|
|
||||||
}else if(status==nostr::ConnectionStatus::DISCONNECTED){
|
|
||||||
nostrIsConnected = false;
|
|
||||||
nostrIsSubscribed = false;
|
|
||||||
sstatus="DISCONNECTED";
|
|
||||||
}else if(status==nostr::ConnectionStatus::ERROR){
|
|
||||||
sstatus = "ERROR";
|
|
||||||
}
|
|
||||||
Serial.println("[ Nostr ] Connection status changed: " + sstatus);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (const std::exception &e)
|
|
||||||
{
|
|
||||||
Serial.println("[ Nostr ] Error: " + String(e.what()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void nostrTask(void *pvParameters)
|
|
||||||
{
|
|
||||||
if(preferences.getBool("useNostr", DEFAULT_USE_NOSTR)) {
|
|
||||||
int blockFetch = getBlockFetch();
|
|
||||||
processNewBlock(blockFetch);
|
|
||||||
}
|
|
||||||
|
|
||||||
while (1)
|
|
||||||
{
|
|
||||||
for (nostr::NostrPool *pool : pools)
|
|
||||||
{
|
|
||||||
// Run internal loop: refresh relays, complete pending connections, send
|
|
||||||
// pending messages
|
|
||||||
pool->loop();
|
|
||||||
if (!nostrIsSubscribed && !nostrIsSubscribing) {
|
|
||||||
Serial.println(F("Not subscribed"));
|
|
||||||
subscribeZaps(pool, preferences.getString("nostrRelay"), 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(100));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setupNostrTask()
|
|
||||||
{
|
|
||||||
xTaskCreate(nostrTask, "nostrTask", 16384, NULL, 10, &nostrTaskHandle);
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean nostrConnected()
|
|
||||||
{
|
|
||||||
return nostrIsConnected;
|
|
||||||
}
|
|
||||||
|
|
||||||
void onNostrSubscriptionClosed(const String &subId, const String &reason)
|
|
||||||
{
|
|
||||||
// This is the callback that will be called when the subscription is
|
|
||||||
// closed
|
|
||||||
Serial.println("[ Nostr ] Subscription closed: " + reason);
|
|
||||||
}
|
|
||||||
|
|
||||||
void onNostrSubscriptionEose(const String &subId)
|
|
||||||
{
|
|
||||||
// This is the callback that will be called when the subscription is
|
|
||||||
// EOSE
|
|
||||||
Serial.println("[ Nostr ] Subscription EOSE: " + subId);
|
|
||||||
nostrIsSubscribing = false;
|
|
||||||
nostrIsSubscribed = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void handleNostrEventCallback(const String &subId, nostr::SignedNostrEvent *event)
|
|
||||||
{
|
|
||||||
// Received events callback, we can access the event content with
|
|
||||||
// event->getContent() Here you should handle the event, for this
|
|
||||||
// test we will just serialize it and print to console
|
|
||||||
JsonDocument doc;
|
|
||||||
JsonArray arr = doc["data"].to<JsonArray>();
|
|
||||||
event->toSendableEvent(arr);
|
|
||||||
// Access the second element which is the object
|
|
||||||
JsonObject obj = arr[1].as<JsonObject>();
|
|
||||||
JsonArray tags = obj["tags"].as<JsonArray>();
|
|
||||||
|
|
||||||
// Flag to check if the tag was found
|
|
||||||
bool tagFound = false;
|
|
||||||
uint medianFee = 0;
|
|
||||||
String typeValue;
|
|
||||||
|
|
||||||
// Iterate over the tags array
|
|
||||||
for (JsonArray tag : tags)
|
|
||||||
{
|
|
||||||
// Check if the tag is an array with two elements
|
|
||||||
if (tag.size() == 2)
|
|
||||||
{
|
|
||||||
const char *key = tag[0];
|
|
||||||
const char *value = tag[1];
|
|
||||||
|
|
||||||
// Check if the key is "type" and the value is "priceUsd"
|
|
||||||
if (strcmp(key, "type") == 0 && (strcmp(value, "priceUsd") == 0 || strcmp(value, "blockHeight") == 0))
|
|
||||||
{
|
|
||||||
typeValue = value;
|
|
||||||
tagFound = true;
|
|
||||||
}
|
|
||||||
else if (strcmp(key, "medianFee") == 0)
|
|
||||||
{
|
|
||||||
medianFee = tag[1].as<uint>();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (tagFound)
|
|
||||||
{
|
|
||||||
if (typeValue.equals("priceUsd"))
|
|
||||||
{
|
|
||||||
processNewPrice(obj["content"].as<uint>(), CURRENCY_USD);
|
|
||||||
}
|
|
||||||
else if (typeValue.equals("blockHeight"))
|
|
||||||
{
|
|
||||||
processNewBlock(obj["content"].as<uint>());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (medianFee != 0)
|
|
||||||
{
|
|
||||||
processNewBlockFee(medianFee);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
time_t getMinutesAgo(int min) {
|
|
||||||
time_t now;
|
|
||||||
time(&now);
|
|
||||||
return now - (min * 60);
|
|
||||||
}
|
|
||||||
|
|
||||||
void subscribeZaps(nostr::NostrPool *pool, const String &relay, int minutesAgo) {
|
|
||||||
if (subIdZap) {
|
|
||||||
pool->closeSubscription(subIdZap);
|
|
||||||
}
|
|
||||||
nostrIsSubscribing = true;
|
|
||||||
|
|
||||||
subIdZap = pool->subscribeMany(
|
|
||||||
{relay},
|
|
||||||
{
|
|
||||||
{
|
|
||||||
{"kinds", {"9735"}},
|
|
||||||
{"limit", {"1"}},
|
|
||||||
{"since", {String(getMinutesAgo(minutesAgo))}},
|
|
||||||
{"#p", {preferences.getString("nostrZapPubkey", DEFAULT_ZAP_NOTIFY_PUBKEY)}},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
handleNostrZapCallback,
|
|
||||||
onNostrSubscriptionClosed,
|
|
||||||
onNostrSubscriptionEose);
|
|
||||||
Serial.println("[ Nostr ] Subscribing to Zap Notifications since " + String(getMinutesAgo(minutesAgo)));
|
|
||||||
}
|
|
||||||
|
|
||||||
void handleNostrZapCallback(const String &subId, nostr::SignedNostrEvent *event) {
|
|
||||||
// Received events callback, we can access the event content with
|
|
||||||
// event->getContent() Here you should handle the event, for this
|
|
||||||
// test we will just serialize it and print to console
|
|
||||||
JsonDocument doc;
|
|
||||||
JsonArray arr = doc["data"].to<JsonArray>();
|
|
||||||
event->toSendableEvent(arr);
|
|
||||||
// Access the second element which is the object
|
|
||||||
JsonObject obj = arr[1].as<JsonObject>();
|
|
||||||
JsonArray tags = obj["tags"].as<JsonArray>();
|
|
||||||
|
|
||||||
// Iterate over the tags array
|
|
||||||
for (JsonArray tag : tags)
|
|
||||||
{
|
|
||||||
// Check if the tag is an array with two elements
|
|
||||||
if (tag.size() == 2)
|
|
||||||
{
|
|
||||||
const char *key = tag[0];
|
|
||||||
const char *value = tag[1];
|
|
||||||
|
|
||||||
if (strcmp(key, "bolt11") == 0)
|
|
||||||
{
|
|
||||||
Serial.print(F("Got a zap of "));
|
|
||||||
|
|
||||||
int64_t satsAmount = getAmountInSatoshis(std::string(value));
|
|
||||||
Serial.print(satsAmount);
|
|
||||||
Serial.println(F(" sats"));
|
|
||||||
|
|
||||||
std::array<std::string, NUM_SCREENS> textEpdContent = parseZapNotify(satsAmount, preferences.getBool("useSatsSymbol", DEFAULT_USE_SATS_SYMBOL));
|
|
||||||
|
|
||||||
uint64_t timerPeriod = 0;
|
|
||||||
if (isTimerActive())
|
|
||||||
{
|
|
||||||
// store timer periode before making inactive to prevent artifacts
|
|
||||||
timerPeriod = getTimerSeconds();
|
|
||||||
esp_timer_stop(screenRotateTimer);
|
|
||||||
}
|
|
||||||
setCurrentScreen(SCREEN_CUSTOM);
|
|
||||||
|
|
||||||
setEpdContent(textEpdContent);
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(315 * NUM_SCREENS) + pdMS_TO_TICKS(250));
|
|
||||||
if (preferences.getBool("ledFlashOnZap", DEFAULT_LED_FLASH_ON_ZAP))
|
|
||||||
{
|
|
||||||
queueLedEffect(LED_EFFECT_NOSTR_ZAP);
|
|
||||||
}
|
|
||||||
if (timerPeriod > 0)
|
|
||||||
{
|
|
||||||
esp_timer_start_periodic(screenRotateTimer,
|
|
||||||
timerPeriod * usPerSecond);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,30 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "shared.hpp"
|
|
||||||
|
|
||||||
#include <ArduinoJson.h>
|
|
||||||
#include <nostrdisplay_handler.hpp>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
#include "esp32/ESP32Platform.h"
|
|
||||||
#include "NostrEvent.h"
|
|
||||||
#include "NostrPool.h"
|
|
||||||
|
|
||||||
#include "price_notify.hpp"
|
|
||||||
#include "block_notify.hpp"
|
|
||||||
#include "lib/timers.hpp"
|
|
||||||
|
|
||||||
void setupNostrNotify(bool asDatasource, bool zapNotify);
|
|
||||||
|
|
||||||
void nostrTask(void *pvParameters);
|
|
||||||
void setupNostrTask();
|
|
||||||
|
|
||||||
boolean nostrConnected();
|
|
||||||
void handleNostrEventCallback(const String &subId, nostr::SignedNostrEvent *event);
|
|
||||||
void handleNostrZapCallback(const String &subId, nostr::SignedNostrEvent *event);
|
|
||||||
|
|
||||||
void onNostrSubscriptionClosed(const String &subId, const String &reason);
|
|
||||||
void onNostrSubscriptionEose(const String &subId);
|
|
||||||
|
|
||||||
time_t getMinutesAgo(int min);
|
|
||||||
void subscribeZaps(nostr::NostrPool *pool, const String &relay, int minutesAgo);
|
|
163
src/lib/nostr_subscribe.cpp
Normal file
163
src/lib/nostr_subscribe.cpp
Normal file
|
@ -0,0 +1,163 @@
|
||||||
|
#include "nostr_subscribe.hpp"
|
||||||
|
|
||||||
|
TaskHandle_t nostrSubscribeTaskHandle;
|
||||||
|
NostrEvent nostr;
|
||||||
|
NostrRelayManager nostrRelayManager;
|
||||||
|
NostrQueueProcessor nostrQueue;
|
||||||
|
|
||||||
|
bool hasSentEvent = false;
|
||||||
|
unsigned long int lastNostrPriceUpdate;
|
||||||
|
unsigned long int lastNostrBlockUpdate;
|
||||||
|
|
||||||
|
void okEvent(const std::string& key, const char* payload) {
|
||||||
|
Serial.println("OK event");
|
||||||
|
Serial.println("payload is: ");
|
||||||
|
Serial.println(payload);
|
||||||
|
}
|
||||||
|
|
||||||
|
void nip01Event(const std::string& key, const char* payload) {
|
||||||
|
// Serial.println("NIP01 event");
|
||||||
|
// Serial.println("payload is: ");
|
||||||
|
// Serial.println(payload);
|
||||||
|
|
||||||
|
uint minSecPriceUpd = preferences.getUInt(
|
||||||
|
"minSecPriceUpd", DEFAULT_SECONDS_BETWEEN_PRICE_UPDATE);
|
||||||
|
uint currentTime = esp_timer_get_time() / 1000000;
|
||||||
|
|
||||||
|
SpiRamJsonDocument doc(1024);
|
||||||
|
DeserializationError error = deserializeJson(doc, payload);
|
||||||
|
|
||||||
|
if (doc[2].containsKey("tags")) {
|
||||||
|
for (int i = 0; i < doc[2]["tags"].size(); i++) {
|
||||||
|
if (doc[2]["tags"][i][0].as<String>().equals("type") &&
|
||||||
|
doc[2]["tags"][i][1].as<String>().equals("blockHeight")) {
|
||||||
|
uint currentBlockHeight = doc[2]["content"].as<uint>();
|
||||||
|
|
||||||
|
if (currentBlockHeight <= getBlockHeight())
|
||||||
|
return;
|
||||||
|
|
||||||
|
Serial.printf("Block Height: %d\n", currentBlockHeight);
|
||||||
|
setBlockHeight(currentBlockHeight);
|
||||||
|
if (workQueue != nullptr) {
|
||||||
|
WorkItem blockUpdate = {TASK_BLOCK_UPDATE, 0};
|
||||||
|
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (getCurrentScreen() != SCREEN_BLOCK_HEIGHT &&
|
||||||
|
preferences.getBool("stealFocus", true)) {
|
||||||
|
uint64_t timerPeriod = 0;
|
||||||
|
if (isTimerActive()) {
|
||||||
|
// store timer periode before making inactive to prevent artifacts
|
||||||
|
timerPeriod = getTimerSeconds();
|
||||||
|
esp_timer_stop(screenRotateTimer);
|
||||||
|
}
|
||||||
|
setCurrentScreen(SCREEN_BLOCK_HEIGHT);
|
||||||
|
if (timerPeriod > 0) {
|
||||||
|
esp_timer_start_periodic(screenRotateTimer,
|
||||||
|
timerPeriod * usPerSecond);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (getCurrentScreen() == SCREEN_BLOCK_HEIGHT &&
|
||||||
|
preferences.getBool("ledFlashOnUpd", false)) {
|
||||||
|
vTaskDelay(pdMS_TO_TICKS(250)); // Wait until screens are updated
|
||||||
|
queueLedEffect(LED_FLASH_BLOCK_NOTIFY);
|
||||||
|
}
|
||||||
|
|
||||||
|
preferences.putUInt("blockHeight", currentBlockHeight);
|
||||||
|
} else if (doc[2]["tags"][i][0].as<String>().equals("type") &&
|
||||||
|
doc[2]["tags"][i][1].as<String>().equals("priceUsd")) {
|
||||||
|
uint usdPrice = doc[2]["content"].as<uint>();
|
||||||
|
|
||||||
|
if (lastNostrPriceUpdate == 0 ||
|
||||||
|
(currentTime - lastNostrPriceUpdate) > minSecPriceUpd) {
|
||||||
|
setPrice(usdPrice);
|
||||||
|
|
||||||
|
if (workQueue != nullptr &&
|
||||||
|
(getCurrentScreen() == SCREEN_BTC_TICKER ||
|
||||||
|
getCurrentScreen() == SCREEN_MSCW_TIME ||
|
||||||
|
getCurrentScreen() == SCREEN_MARKET_CAP)) {
|
||||||
|
WorkItem priceUpdate = {TASK_PRICE_UPDATE, 0};
|
||||||
|
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
||||||
|
}
|
||||||
|
lastNostrPriceUpdate = currentTime;
|
||||||
|
|
||||||
|
preferences.putUInt("lastPrice", usdPrice);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void taskNostrSubscribe(void* pvParameters) {
|
||||||
|
std::vector<String> relays = retrieveStringVector("nostrRelays");
|
||||||
|
|
||||||
|
nostr.setLogging(false);
|
||||||
|
nostrRelayManager.setRelays(relays);
|
||||||
|
nostrRelayManager.setMinRelaysAndTimeout(2, 10000);
|
||||||
|
|
||||||
|
nostrRelayManager.setEventCallback("ok", okEvent);
|
||||||
|
nostrRelayManager.setEventCallback(1, nip01Event);
|
||||||
|
nostrRelayManager.connect();
|
||||||
|
|
||||||
|
NostrRequestOptions* eventRequestOptions = new NostrRequestOptions();
|
||||||
|
String authors[] = {
|
||||||
|
preferences.getString("nostrBlocksAuth", DEFAULT_NOSTR_BLOCKS_AUTHOR),
|
||||||
|
preferences.getString("nostrPriceAuth", DEFAULT_NOSTR_PRICE_AUTHOR)};
|
||||||
|
eventRequestOptions->authors = authors;
|
||||||
|
eventRequestOptions->authors_count = sizeof(authors) / sizeof(authors[0]);
|
||||||
|
|
||||||
|
int kinds[] = {1};
|
||||||
|
eventRequestOptions->kinds = kinds;
|
||||||
|
eventRequestOptions->kinds_count = sizeof(kinds) / sizeof(kinds[0]);
|
||||||
|
|
||||||
|
eventRequestOptions->limit = 5;
|
||||||
|
nostrRelayManager.requestEvents(eventRequestOptions);
|
||||||
|
while (1) {
|
||||||
|
nostrRelayManager.loop();
|
||||||
|
nostrRelayManager.broadcastEvents();
|
||||||
|
|
||||||
|
vTaskDelay(100);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void setupNostrSubscribeTask() {
|
||||||
|
xTaskCreate(taskNostrSubscribe, "nostrSubscribe", (10 * 1024), NULL, 12,
|
||||||
|
&nostrSubscribeTaskHandle);
|
||||||
|
|
||||||
|
// taskNostrSubscribe(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
void storeStringVector(const char* key, const std::vector<String>& vec) {
|
||||||
|
String serializedVector = "";
|
||||||
|
|
||||||
|
// Serialize the vector of strings
|
||||||
|
for (size_t i = 0; i < vec.size(); i++) {
|
||||||
|
serializedVector += vec[i];
|
||||||
|
if (i < vec.size() - 1) {
|
||||||
|
serializedVector += ",";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Store the serialized vector as a string
|
||||||
|
preferences.putString(key, serializedVector);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<String> retrieveStringVector(const char* key) {
|
||||||
|
String serializedVector = preferences.getString(key, DEFAULT_NOSTR_RELAYS);
|
||||||
|
std::vector<String> result;
|
||||||
|
|
||||||
|
// Deserialize the string into a vector of strings
|
||||||
|
size_t startPos = 0;
|
||||||
|
size_t commaPos = serializedVector.indexOf(",");
|
||||||
|
while (commaPos != -1) {
|
||||||
|
result.push_back(serializedVector.substring(startPos, commaPos));
|
||||||
|
startPos = commaPos + 1;
|
||||||
|
commaPos = serializedVector.indexOf(",", startPos);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add the last element
|
||||||
|
result.push_back(serializedVector.substring(startPos));
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
16
src/lib/nostr_subscribe.hpp
Normal file
16
src/lib/nostr_subscribe.hpp
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
#include <Arduino.h>
|
||||||
|
#include <NostrEvent.h>
|
||||||
|
#include <NostrRelayManager.h>
|
||||||
|
|
||||||
|
#include "lib/config.hpp"
|
||||||
|
#include "lib/shared.hpp"
|
||||||
|
#include "lib/block_notify.hpp"
|
||||||
|
extern TaskHandle_t nostrSubscribeTaskHandle;
|
||||||
|
|
||||||
|
void okEvent(const std::string& key, const char* payload);
|
||||||
|
void nip01Event(const std::string& key, const char* payload);
|
||||||
|
void setupNostrSubscribeTask();
|
||||||
|
void taskNostrSubscribe(void *pvParameters);
|
||||||
|
|
||||||
|
void storeStringVector(const char* key, const std::vector<String>& vec);
|
||||||
|
std::vector<String> retrieveStringVector(const char* key);
|
425
src/lib/ota.cpp
425
src/lib/ota.cpp
|
@ -1,15 +1,9 @@
|
||||||
#include "ota.hpp"
|
#include "ota.hpp"
|
||||||
|
|
||||||
TaskHandle_t taskOtaHandle = NULL;
|
TaskHandle_t taskOtaHandle = NULL;
|
||||||
bool isOtaUpdating = false;
|
|
||||||
QueueHandle_t otaQueue;
|
|
||||||
|
|
||||||
|
void setupOTA() {
|
||||||
|
if (preferences.getBool("otaEnabled", true)) {
|
||||||
void setupOTA()
|
|
||||||
{
|
|
||||||
if (preferences.getBool("otaEnabled", DEFAULT_OTA_ENABLED))
|
|
||||||
{
|
|
||||||
ArduinoOTA.onStart(onOTAStart);
|
ArduinoOTA.onStart(onOTAStart);
|
||||||
|
|
||||||
ArduinoOTA.onProgress(onOTAProgress);
|
ArduinoOTA.onProgress(onOTAProgress);
|
||||||
|
@ -21,38 +15,31 @@ void setupOTA()
|
||||||
ArduinoOTA.setRebootOnSuccess(false);
|
ArduinoOTA.setRebootOnSuccess(false);
|
||||||
ArduinoOTA.begin();
|
ArduinoOTA.begin();
|
||||||
// downloadUpdate();
|
// downloadUpdate();
|
||||||
otaQueue = xQueueCreate(1, sizeof(UpdateMessage));
|
|
||||||
|
|
||||||
xTaskCreate(handleOTATask, "handleOTA", 8192, NULL, 20,
|
xTaskCreate(handleOTATask, "handleOTA", 4096, NULL, tskIDLE_PRIORITY,
|
||||||
&taskOtaHandle);
|
&taskOtaHandle);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void onOTAProgress(unsigned int progress, unsigned int total)
|
void onOTAProgress(unsigned int progress, unsigned int total) {
|
||||||
{
|
|
||||||
uint percentage = progress / (total / 100);
|
uint percentage = progress / (total / 100);
|
||||||
pixels.fill(pixels.Color(0, 255, 0));
|
pixels.fill(pixels.Color(0, 255, 0));
|
||||||
if (percentage < 100)
|
if (percentage < 100) {
|
||||||
{
|
|
||||||
pixels.setPixelColor(0, pixels.Color(0, 0, 0));
|
pixels.setPixelColor(0, pixels.Color(0, 0, 0));
|
||||||
}
|
}
|
||||||
if (percentage < 75)
|
if (percentage < 75) {
|
||||||
{
|
|
||||||
pixels.setPixelColor(1, pixels.Color(0, 0, 0));
|
pixels.setPixelColor(1, pixels.Color(0, 0, 0));
|
||||||
}
|
}
|
||||||
if (percentage < 50)
|
if (percentage < 50) {
|
||||||
{
|
|
||||||
pixels.setPixelColor(2, pixels.Color(0, 0, 0));
|
pixels.setPixelColor(2, pixels.Color(0, 0, 0));
|
||||||
}
|
}
|
||||||
if (percentage < 25)
|
if (percentage < 25) {
|
||||||
{
|
|
||||||
pixels.setPixelColor(3, pixels.Color(0, 0, 0));
|
pixels.setPixelColor(3, pixels.Color(0, 0, 0));
|
||||||
}
|
}
|
||||||
pixels.show();
|
pixels.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
void onOTAStart()
|
void onOTAStart() {
|
||||||
{
|
|
||||||
forceFullRefresh();
|
forceFullRefresh();
|
||||||
std::array<String, NUM_SCREENS> epdContent = {"U", "P", "D", "A",
|
std::array<String, NUM_SCREENS> epdContent = {"U", "P", "D", "A",
|
||||||
"T", "E", "!"};
|
"T", "E", "!"};
|
||||||
|
@ -60,385 +47,97 @@ void onOTAStart()
|
||||||
// Stop all timers
|
// Stop all timers
|
||||||
esp_timer_stop(screenRotateTimer);
|
esp_timer_stop(screenRotateTimer);
|
||||||
esp_timer_stop(minuteTimer);
|
esp_timer_stop(minuteTimer);
|
||||||
isOtaUpdating = true;
|
|
||||||
// Stop or suspend all tasks
|
// Stop or suspend all tasks
|
||||||
// vTaskSuspend(priceUpdateTaskHandle);
|
// vTaskSuspend(priceUpdateTaskHandle);
|
||||||
// vTaskSuspend(blockUpdateTaskHandle);
|
// vTaskSuspend(blockUpdateTaskHandle);
|
||||||
vTaskSuspend(workerTaskHandle);
|
vTaskSuspend(workerTaskHandle);
|
||||||
vTaskSuspend(taskScreenRotateTaskHandle);
|
vTaskSuspend(taskScreenRotateTaskHandle);
|
||||||
|
|
||||||
// vTaskSuspend(ledTaskHandle);
|
vTaskSuspend(ledTaskHandle);
|
||||||
vTaskSuspend(buttonTaskHandle);
|
vTaskSuspend(buttonTaskHandle);
|
||||||
|
|
||||||
// stopWebServer();
|
stopWebServer();
|
||||||
stopBlockNotify();
|
stopBlockNotify();
|
||||||
stopPriceNotify();
|
stopPriceNotify();
|
||||||
}
|
}
|
||||||
|
|
||||||
void handleOTATask(void *parameter)
|
void handleOTATask(void *parameter) {
|
||||||
{
|
for (;;) {
|
||||||
UpdateMessage msg;
|
ArduinoOTA.handle(); // Allow OTA updates to occur
|
||||||
|
vTaskDelay(pdMS_TO_TICKS(2500));
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
if (xQueueReceive(otaQueue, &msg, 0) == pdTRUE)
|
|
||||||
{
|
|
||||||
if (msg.updateType == UPDATE_ALL) {
|
|
||||||
queueLedEffect(LED_FLASH_UPDATE);
|
|
||||||
int resultWebUi = downloadUpdateHandler(UPDATE_WEBUI);
|
|
||||||
queueLedEffect(LED_FLASH_UPDATE);
|
|
||||||
int resultFw = downloadUpdateHandler(UPDATE_FIRMWARE);
|
|
||||||
|
|
||||||
if (resultWebUi == 0 && resultFw == 0) {
|
|
||||||
ESP.restart();
|
|
||||||
} else {
|
|
||||||
queueLedEffect(LED_FLASH_ERROR);
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(3000));
|
|
||||||
ESP.restart();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ArduinoOTA.handle(); // Allow OTA updates to occur
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(2000));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ReleaseInfo getLatestRelease(const String &fileToDownload)
|
void downloadUpdate() {
|
||||||
{
|
|
||||||
String releaseUrl = preferences.getString("gitReleaseUrl");
|
|
||||||
WiFiClientSecure client;
|
WiFiClientSecure client;
|
||||||
// client.setCACert(isrg_root_x1cert);
|
client.setInsecure();
|
||||||
client.setCACertBundle(rootca_crt_bundle_start);
|
|
||||||
|
|
||||||
|
|
||||||
HTTPClient http;
|
HTTPClient http;
|
||||||
http.begin(client, releaseUrl);
|
|
||||||
http.setUserAgent(USER_AGENT);
|
http.setUserAgent(USER_AGENT);
|
||||||
|
|
||||||
|
// Send HTTP request to CoinGecko API
|
||||||
|
http.useHTTP10(true);
|
||||||
|
|
||||||
|
http.begin(client,
|
||||||
|
"https://api.github.com/repos/btclock/btclock_v3/releases/latest");
|
||||||
int httpCode = http.GET();
|
int httpCode = http.GET();
|
||||||
|
|
||||||
ReleaseInfo info = {"", ""};
|
if (httpCode == 200) {
|
||||||
|
// WiFiClient * stream = http->getStreamPtr();
|
||||||
|
|
||||||
if (httpCode > 0)
|
StaticJsonDocument<64> filter;
|
||||||
{
|
|
||||||
String payload = http.getString();
|
|
||||||
|
|
||||||
JsonDocument doc;
|
JsonObject filter_assets_0 = filter["assets"].createNestedObject();
|
||||||
deserializeJson(doc, payload);
|
filter_assets_0["name"] = true;
|
||||||
|
filter_assets_0["browser_download_url"] = true;
|
||||||
|
|
||||||
JsonArray assets = doc["assets"];
|
SpiRamJsonDocument doc(1536);
|
||||||
|
|
||||||
for (JsonObject asset : assets)
|
DeserializationError error = deserializeJson(
|
||||||
{
|
doc, http.getStream(), DeserializationOption::Filter(filter));
|
||||||
String assetName = asset["name"].as<String>();
|
|
||||||
if (assetName == fileToDownload)
|
|
||||||
{
|
|
||||||
info.fileUrl = asset["browser_download_url"].as<String>();
|
|
||||||
}
|
|
||||||
else if (assetName == fileToDownload + ".sha256")
|
|
||||||
{
|
|
||||||
info.checksumUrl = asset["browser_download_url"].as<String>();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!info.fileUrl.isEmpty() && !info.checksumUrl.isEmpty())
|
if (error) {
|
||||||
{
|
Serial.print("deserializeJson() failed: ");
|
||||||
|
Serial.println(error.c_str());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
String downloadUrl;
|
||||||
|
for (JsonObject asset : doc["assets"].as<JsonArray>()) {
|
||||||
|
if (asset["name"].as<String>().compareTo("firmware.bin") == 0) {
|
||||||
|
downloadUrl = asset["browser_download_url"].as<String>();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Serial.printf("Latest release URL: %s\r\n", info.fileUrl.c_str());
|
|
||||||
Serial.printf("Checksum URL: %s\r\n", info.checksumUrl.c_str());
|
|
||||||
}
|
|
||||||
http.end();
|
|
||||||
return info;
|
|
||||||
}
|
|
||||||
|
|
||||||
int downloadUpdateHandler(char updateType)
|
Serial.printf("Download update from %s", downloadUrl);
|
||||||
{
|
|
||||||
WiFiClientSecure client;
|
|
||||||
client.setCACertBundle(rootca_crt_bundle_start);
|
|
||||||
HTTPClient http;
|
|
||||||
http.setFollowRedirects(HTTPC_STRICT_FOLLOW_REDIRECTS);
|
|
||||||
|
|
||||||
ReleaseInfo latestRelease;
|
// esp_http_client_config_t config = {
|
||||||
|
// .url = CONFIG_FIRMWARE_UPGRADE_URL,
|
||||||
switch (updateType)
|
// };
|
||||||
{
|
// esp_https_ota_config_t ota_config = {
|
||||||
case UPDATE_FIRMWARE:
|
// .http_config = &config,
|
||||||
{
|
// };
|
||||||
latestRelease = getLatestRelease(getFirmwareFilename());
|
// esp_err_t ret = esp_https_ota(&ota_config);
|
||||||
}
|
// if (ret == ESP_OK)
|
||||||
break;
|
// {
|
||||||
case UPDATE_WEBUI:
|
// esp_restart();
|
||||||
{
|
// }
|
||||||
latestRelease = getLatestRelease(getWebUiFilename());
|
|
||||||
// updateWebUi(latestRelease.fileUrl, U_SPIFFS);
|
|
||||||
// return 0;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// First, download the expected SHA256
|
|
||||||
String expectedSHA256 = downloadSHA256(latestRelease.checksumUrl);
|
|
||||||
if (expectedSHA256.isEmpty())
|
|
||||||
{
|
|
||||||
Serial.println("Failed to get SHA256 checksum. Aborting update.");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
http.begin(client, latestRelease.fileUrl);
|
|
||||||
http.setUserAgent(USER_AGENT);
|
|
||||||
|
|
||||||
int httpCode = http.GET();
|
|
||||||
if (httpCode == HTTP_CODE_OK)
|
|
||||||
{
|
|
||||||
int contentLength = http.getSize();
|
|
||||||
if (contentLength > 0)
|
|
||||||
{
|
|
||||||
// Allocate memory to store the firmware
|
|
||||||
uint8_t *firmware = (uint8_t *)malloc(contentLength);
|
|
||||||
if (!firmware)
|
|
||||||
{
|
|
||||||
Serial.println(F("Not enough memory to store firmware"));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
WiFiClient *stream = http.getStreamPtr();
|
|
||||||
size_t bytesRead = 0;
|
|
||||||
while (bytesRead < contentLength)
|
|
||||||
{
|
|
||||||
size_t available = stream->available();
|
|
||||||
if (available)
|
|
||||||
{
|
|
||||||
size_t readBytes = stream->readBytes(firmware + bytesRead, available);
|
|
||||||
bytesRead += readBytes;
|
|
||||||
}
|
|
||||||
yield(); // Allow background tasks to run
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bytesRead != contentLength)
|
|
||||||
{
|
|
||||||
Serial.println("Failed to read entire firmware");
|
|
||||||
free(firmware);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calculate SHA256
|
|
||||||
String calculated_sha256 = calculateSHA256(firmware, contentLength);
|
|
||||||
|
|
||||||
Serial.print("Calculated checksum: ");
|
|
||||||
Serial.println(calculated_sha256);
|
|
||||||
Serial.print("Expected checksum: ");
|
|
||||||
Serial.println(expectedSHA256);
|
|
||||||
|
|
||||||
if (calculated_sha256 != expectedSHA256)
|
|
||||||
{
|
|
||||||
Serial.println("Checksum mismatch. Aborting update.");
|
|
||||||
free(firmware);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
Update.onProgress(onOTAProgress);
|
|
||||||
|
|
||||||
if (Update.begin(contentLength, updateType))
|
|
||||||
{
|
|
||||||
onOTAStart();
|
|
||||||
size_t written = Update.write(firmware, contentLength);
|
|
||||||
|
|
||||||
if (written == contentLength)
|
|
||||||
{
|
|
||||||
Serial.println("Written : " + String(written) + " successfully");
|
|
||||||
free(firmware);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Written only : " + String(written) + "/" + String(contentLength) + ". Retry?");
|
|
||||||
free(firmware);
|
|
||||||
return 503;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Update.end())
|
|
||||||
{
|
|
||||||
Serial.println("OTA done!");
|
|
||||||
if (Update.isFinished())
|
|
||||||
{
|
|
||||||
Serial.println("Update successfully completed. Rebooting.");
|
|
||||||
// ESP.restart();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Update not finished? Something went wrong!");
|
|
||||||
free(firmware);
|
|
||||||
return 503;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Error Occurred. Error #: " + String(Update.getError()));
|
|
||||||
free(firmware);
|
|
||||||
return 503;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Not enough space to begin OTA");
|
|
||||||
free(firmware);
|
|
||||||
return 503;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Invalid content length");
|
|
||||||
return 503;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.printf("HTTP error: %d\n", httpCode);
|
|
||||||
return 503;
|
|
||||||
}
|
|
||||||
http.end();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void updateWebUi(String latestRelease, int command)
|
|
||||||
{
|
|
||||||
WiFiClientSecure client;
|
|
||||||
client.setCACertBundle(rootca_crt_bundle_start);
|
|
||||||
HTTPClient http;
|
|
||||||
http.setFollowRedirects(HTTPC_STRICT_FOLLOW_REDIRECTS);
|
|
||||||
http.begin(client, latestRelease);
|
|
||||||
http.setUserAgent(USER_AGENT);
|
|
||||||
|
|
||||||
int httpCode = http.GET();
|
|
||||||
if (httpCode == HTTP_CODE_OK)
|
|
||||||
{
|
|
||||||
int contentLength = http.getSize();
|
|
||||||
if (contentLength > 0)
|
|
||||||
{
|
|
||||||
uint8_t *buffer = (uint8_t *)malloc(contentLength);
|
|
||||||
if (buffer)
|
|
||||||
{
|
|
||||||
WiFiClient *stream = http.getStreamPtr();
|
|
||||||
size_t written = stream->readBytes(buffer, contentLength);
|
|
||||||
|
|
||||||
if (written == contentLength)
|
|
||||||
{
|
|
||||||
String expectedSHA256 = "";
|
|
||||||
if (command == U_FLASH)
|
|
||||||
{
|
|
||||||
expectedSHA256 = downloadSHA256(getFirmwareFilename());
|
|
||||||
Serial.print("Expected checksum: ");
|
|
||||||
Serial.println(expectedSHA256);
|
|
||||||
}
|
|
||||||
|
|
||||||
String calculated_sha256 = calculateSHA256(buffer, contentLength);
|
|
||||||
Serial.print("Checksum is ");
|
|
||||||
Serial.println(calculated_sha256);
|
|
||||||
if ((command == U_FLASH && expectedSHA256.equals(calculated_sha256)) || command == U_SPIFFS)
|
|
||||||
{
|
|
||||||
Serial.println("Checksum verified. Proceeding with update.");
|
|
||||||
|
|
||||||
Update.onProgress(onOTAProgress);
|
|
||||||
|
|
||||||
if (Update.begin(contentLength, command))
|
|
||||||
{
|
|
||||||
onOTAStart();
|
|
||||||
|
|
||||||
Update.write(buffer, contentLength);
|
|
||||||
if (Update.end())
|
|
||||||
{
|
|
||||||
Serial.println("Update complete. Rebooting.");
|
|
||||||
ESP.restart();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Error in update process.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Not enough space to begin OTA");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Checksum mismatch. Aborting update.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Error downloading firmware");
|
|
||||||
}
|
|
||||||
free(buffer);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Not enough memory to allocate buffer");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println("Invalid content length");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.print(httpCode);
|
|
||||||
Serial.println("Error on HTTP request");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void onOTAError(ota_error_t error)
|
void onOTAError(ota_error_t error) {
|
||||||
{
|
Serial.println("\nOTA update error, restarting");
|
||||||
Serial.println(F("\nOTA update error, restarting"));
|
|
||||||
Wire.end();
|
|
||||||
SPI.end();
|
|
||||||
isOtaUpdating = false;
|
|
||||||
delay(1000);
|
|
||||||
ESP.restart();
|
|
||||||
}
|
|
||||||
|
|
||||||
void onOTAComplete()
|
|
||||||
{
|
|
||||||
Serial.println(F("\nOTA update finished"));
|
|
||||||
Wire.end();
|
Wire.end();
|
||||||
SPI.end();
|
SPI.end();
|
||||||
delay(1000);
|
delay(1000);
|
||||||
ESP.restart();
|
ESP.restart();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool getIsOTAUpdating()
|
void onOTAComplete() {
|
||||||
{
|
Serial.println("\nOTA update finished");
|
||||||
return isOtaUpdating;
|
Wire.end();
|
||||||
}
|
SPI.end();
|
||||||
|
delay(1000);
|
||||||
String downloadSHA256(const String &sha256Url)
|
ESP.restart();
|
||||||
{
|
|
||||||
if (sha256Url.isEmpty())
|
|
||||||
{
|
|
||||||
Serial.println("Failed to get SHA256 file URL");
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
WiFiClientSecure client;
|
|
||||||
client.setCACertBundle(rootca_crt_bundle_start);
|
|
||||||
HTTPClient http;
|
|
||||||
http.setFollowRedirects(HTTPC_STRICT_FOLLOW_REDIRECTS);
|
|
||||||
http.begin(client, sha256Url);
|
|
||||||
http.setUserAgent(USER_AGENT);
|
|
||||||
|
|
||||||
int httpCode = http.GET();
|
|
||||||
if (httpCode == HTTP_CODE_OK)
|
|
||||||
{
|
|
||||||
String sha256 = http.getString();
|
|
||||||
sha256.trim(); // Remove any whitespace or newline characters
|
|
||||||
return sha256;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.printf("Failed to download SHA256 file. HTTP error: %d\n", httpCode);
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
}
|
}
|
|
@ -1,38 +1,13 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <Arduino.h>
|
#include <Arduino.h>
|
||||||
#include <ArduinoOTA.h>
|
#include <ArduinoOTA.h>
|
||||||
|
|
||||||
#include "lib/config.hpp"
|
#include "lib/config.hpp"
|
||||||
#include "lib/shared.hpp"
|
#include "lib/shared.hpp"
|
||||||
#include "lib/timers.hpp"
|
|
||||||
|
|
||||||
#ifndef UPDATE_MESSAGE_HPP
|
|
||||||
#define UPDATE_MESSAGE_HPP
|
|
||||||
typedef struct {
|
|
||||||
char updateType;
|
|
||||||
} UpdateMessage;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
extern QueueHandle_t otaQueue;
|
|
||||||
|
|
||||||
struct ReleaseInfo {
|
|
||||||
String fileUrl;
|
|
||||||
String checksumUrl;
|
|
||||||
};
|
|
||||||
|
|
||||||
void setupOTA();
|
void setupOTA();
|
||||||
void onOTAStart();
|
void onOTAStart();
|
||||||
void handleOTATask(void *parameter);
|
void handleOTATask(void *parameter);
|
||||||
void onOTAProgress(unsigned int progress, unsigned int total);
|
void onOTAProgress(unsigned int progress, unsigned int total);
|
||||||
// void downloadUpdate();
|
void downloadUpdate();
|
||||||
void onOTAError(ota_error_t error);
|
void onOTAError(ota_error_t error);
|
||||||
void onOTAComplete();
|
void onOTAComplete();
|
||||||
int downloadUpdateHandler(char updateType);
|
|
||||||
ReleaseInfo getLatestRelease(const String& fileToDownload);
|
|
||||||
|
|
||||||
bool getIsOTAUpdating();
|
|
||||||
|
|
||||||
void updateWebUi(String latestRelease, int command);
|
|
||||||
String downloadSHA256(const String& filename);
|
|
||||||
|
|
||||||
|
|
57
src/lib/price_fetch.cpp
Normal file
57
src/lib/price_fetch.cpp
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
#include "price_fetch.hpp"
|
||||||
|
|
||||||
|
const PROGMEM char *cgApiUrl =
|
||||||
|
"https://api.coingecko.com/api/v3/simple/"
|
||||||
|
"price?ids=bitcoin&vs_currencies=usd%2Ceur";
|
||||||
|
|
||||||
|
TaskHandle_t priceFetchTaskHandle;
|
||||||
|
|
||||||
|
void taskPriceFetch(void *pvParameters) {
|
||||||
|
WiFiClientSecure *client = new WiFiClientSecure;
|
||||||
|
client->setInsecure();
|
||||||
|
for (;;) {
|
||||||
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
||||||
|
|
||||||
|
HTTPClient *http = new HTTPClient();
|
||||||
|
http->setUserAgent(USER_AGENT);
|
||||||
|
|
||||||
|
// Send HTTP request to CoinGecko API
|
||||||
|
http->begin(*client, cgApiUrl);
|
||||||
|
|
||||||
|
int httpCode = http->GET();
|
||||||
|
|
||||||
|
// Parse JSON response and extract average price
|
||||||
|
uint usdPrice, eurPrice;
|
||||||
|
if (httpCode == 200) {
|
||||||
|
String payload = http->getString();
|
||||||
|
StaticJsonDocument<96> doc;
|
||||||
|
deserializeJson(doc, payload);
|
||||||
|
// usdPrice = doc["bitcoin"]["usd"];
|
||||||
|
eurPrice = doc["bitcoin"]["eur"].as<uint>();
|
||||||
|
|
||||||
|
setPrice(eurPrice);
|
||||||
|
if (workQueue != nullptr && (getCurrentScreen() == SCREEN_BTC_TICKER ||
|
||||||
|
getCurrentScreen() == SCREEN_MSCW_TIME ||
|
||||||
|
getCurrentScreen() == SCREEN_MARKET_CAP)) {
|
||||||
|
WorkItem priceUpdate = {TASK_PRICE_UPDATE, 0};
|
||||||
|
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
||||||
|
}
|
||||||
|
|
||||||
|
preferences.putUInt("lastPrice", eurPrice);
|
||||||
|
} else {
|
||||||
|
Serial.print(
|
||||||
|
F("Error retrieving BTC/USD price (CoinGecko). HTTP status code: "));
|
||||||
|
Serial.println(httpCode);
|
||||||
|
if (httpCode == -1) {
|
||||||
|
WiFi.reconnect();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void setupPriceFetchTask() {
|
||||||
|
xTaskCreate(taskPriceFetch, "priceFetch", (6 * 1024), NULL, tskIDLE_PRIORITY,
|
||||||
|
&priceFetchTaskHandle);
|
||||||
|
|
||||||
|
xTaskNotifyGive(priceFetchTaskHandle);
|
||||||
|
}
|
10
src/lib/price_fetch.hpp
Normal file
10
src/lib/price_fetch.hpp
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
#include <Arduino.h>
|
||||||
|
#include <HTTPClient.h>
|
||||||
|
|
||||||
|
#include "lib/config.hpp"
|
||||||
|
#include "lib/shared.hpp"
|
||||||
|
|
||||||
|
extern TaskHandle_t priceFetchTaskHandle;
|
||||||
|
|
||||||
|
void setupPriceFetchTask();
|
||||||
|
void taskPriceFetch(void *pvParameters);
|
|
@ -1,205 +1,117 @@
|
||||||
#include "price_notify.hpp"
|
#include "price_notify.hpp"
|
||||||
|
|
||||||
const char *wsOwnServerPrice = "wss://ws.btclock.dev/ws?assets=bitcoin";
|
|
||||||
const char *wsOwnServerV2 = "wss://ws-staging.btclock.dev/api/v2/ws";
|
|
||||||
|
|
||||||
const char *wsServerPrice = "wss://ws.coincap.io/prices?assets=bitcoin";
|
const char *wsServerPrice = "wss://ws.coincap.io/prices?assets=bitcoin";
|
||||||
|
|
||||||
|
// const char* coinCapWsCert = R"(-----BEGIN CERTIFICATE-----
|
||||||
|
// MIIFMjCCBNmgAwIBAgIQBtgXvFyc28MsvQ1HjCnXJTAKBggqhkjOPQQDAjBKMQsw
|
||||||
|
// CQYDVQQGEwJVUzEZMBcGA1UEChMQQ2xvdWRmbGFyZSwgSW5jLjEgMB4GA1UEAxMX
|
||||||
|
// Q2xvdWRmbGFyZSBJbmMgRUNDIENBLTMwHhcNMjMwNTEwMDAwMDAwWhcNMjQwNTA5
|
||||||
|
// MjM1OTU5WjB1MQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEWMBQG
|
||||||
|
// A1UEBxMNU2FuIEZyYW5jaXNjbzEZMBcGA1UEChMQQ2xvdWRmbGFyZSwgSW5jLjEe
|
||||||
|
// MBwGA1UEAxMVc25pLmNsb3VkZmxhcmVzc2wuY29tMFkwEwYHKoZIzj0CAQYIKoZI
|
||||||
|
// zj0DAQcDQgAEpvFIXzQKHuqTo+IE6c6sB4p0PMXK1KsseEGf2UN/CNRhG5hO7lr8
|
||||||
|
// JtXrPZkawWBysZxOsEoetkPrDHMugCLfXKOCA3QwggNwMB8GA1UdIwQYMBaAFKXO
|
||||||
|
// N+rrsHUOlGeItEX62SQQh5YfMB0GA1UdDgQWBBShsZDJohaR1a5E0Qj7yblZjKDC
|
||||||
|
// gDA6BgNVHREEMzAxggwqLmNvaW5jYXAuaW+CCmNvaW5jYXAuaW+CFXNuaS5jbG91
|
||||||
|
// ZGZsYXJlc3NsLmNvbTAOBgNVHQ8BAf8EBAMCB4AwHQYDVR0lBBYwFAYIKwYBBQUH
|
||||||
|
// AwEGCCsGAQUFBwMCMHsGA1UdHwR0MHIwN6A1oDOGMWh0dHA6Ly9jcmwzLmRpZ2lj
|
||||||
|
// ZXJ0LmNvbS9DbG91ZGZsYXJlSW5jRUNDQ0EtMy5jcmwwN6A1oDOGMWh0dHA6Ly9j
|
||||||
|
// cmw0LmRpZ2ljZXJ0LmNvbS9DbG91ZGZsYXJlSW5jRUNDQ0EtMy5jcmwwPgYDVR0g
|
||||||
|
// BDcwNTAzBgZngQwBAgIwKTAnBggrBgEFBQcCARYbaHR0cDovL3d3dy5kaWdpY2Vy
|
||||||
|
// dC5jb20vQ1BTMHYGCCsGAQUFBwEBBGowaDAkBggrBgEFBQcwAYYYaHR0cDovL29j
|
||||||
|
// c3AuZGlnaWNlcnQuY29tMEAGCCsGAQUFBzAChjRodHRwOi8vY2FjZXJ0cy5kaWdp
|
||||||
|
// Y2VydC5jb20vQ2xvdWRmbGFyZUluY0VDQ0NBLTMuY3J0MAwGA1UdEwEB/wQCMAAw
|
||||||
|
// ggF+BgorBgEEAdZ5AgQCBIIBbgSCAWoBaAB1AO7N0GTV2xrOxVy3nbTNE6Iyh0Z8
|
||||||
|
// vOzew1FIWUZxH7WbAAABiAPnoRAAAAQDAEYwRAIgAP2W09OozuhmKeKKMsaVBcae
|
||||||
|
// o+nPHF1WUWk0i387YYYCIDIM1Wll7/4O3GNx2/Fx9bC6pi69Uya4pLxsCfW3fZMe
|
||||||
|
// AHYASLDja9qmRzQP5WoC+p0w6xxSActW3SyB2bu/qznYhHMAAAGIA+eg+QAABAMA
|
||||||
|
// RzBFAiEAuNpSqrbx47gYBgBMz5M6q0CnV/WMJqWQOxYFKrwfwVACIH3nCs4bKToT
|
||||||
|
// e+MiBrqSDaekixk4kPFEQESO9qHCkWY5AHcA2ra/az+1tiKfm8K7XGvocJFxbLtR
|
||||||
|
// hIU0vaQ9MEjX+6sAAAGIA+eg1gAABAMASDBGAiEAolCFl2IfbOHUPAOxoi4BLclS
|
||||||
|
// v9FVXb7LwIvTuCfyrEQCIQDcvehwhV9XGopKGl17F2LYYKI7hvlO3RmpPZQJt1da
|
||||||
|
// MDAKBggqhkjOPQQDAgNHADBEAiAXRWZ/JVMsfpSFFTHQHUSqRnQ/7cCOWx+9svIy
|
||||||
|
// mYnFZQIgHMEG0Cm7O4cn5KUzKOsTwwK+2U15s/jPUQi2n2IDTEM=
|
||||||
|
// -----END CERTIFICATE-----)";
|
||||||
|
|
||||||
// WebsocketsClient client;
|
// WebsocketsClient client;
|
||||||
esp_websocket_client_handle_t clientPrice = NULL;
|
esp_websocket_client_handle_t clientPrice = NULL;
|
||||||
esp_websocket_client_config_t config;
|
uint currentPrice = 30000;
|
||||||
uint currentPrice = 90000;
|
|
||||||
unsigned long int lastPriceUpdate;
|
unsigned long int lastPriceUpdate;
|
||||||
bool priceNotifyInit = false;
|
|
||||||
std::map<char, std::uint64_t> currencyMap;
|
|
||||||
std::map<char, unsigned long int> lastUpdateMap;
|
|
||||||
WebSocketsClient priceNotifyWs;
|
|
||||||
|
|
||||||
void setupPriceNotify()
|
void setupPriceNotify() {
|
||||||
{
|
// currentPrice = preferences.get("lastPrice", 30000);
|
||||||
if (preferences.getBool("ownDataSource", DEFAULT_OWN_DATA_SOURCE))
|
|
||||||
{
|
|
||||||
config = {.uri = wsOwnServerPrice,
|
|
||||||
.user_agent = USER_AGENT};
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
config = {.uri = wsServerPrice,
|
|
||||||
.user_agent = USER_AGENT};
|
|
||||||
config.cert_pem = isrg_root_x1cert;
|
|
||||||
|
|
||||||
config.task_stack = (6*1024);
|
esp_websocket_client_config_t config = {.uri = wsServerPrice,
|
||||||
}
|
// .task_stack = (7*1024),
|
||||||
|
// .cert_pem = coinCapWsCert,
|
||||||
|
.user_agent = USER_AGENT};
|
||||||
|
|
||||||
clientPrice = esp_websocket_client_init(&config);
|
clientPrice = esp_websocket_client_init(&config);
|
||||||
esp_websocket_register_events(clientPrice, WEBSOCKET_EVENT_ANY,
|
esp_websocket_register_events(clientPrice, WEBSOCKET_EVENT_ANY,
|
||||||
onWebsocketPriceEvent, clientPrice);
|
onWebsocketPriceEvent, clientPrice);
|
||||||
esp_websocket_client_start(clientPrice);
|
esp_websocket_client_start(clientPrice);
|
||||||
|
|
||||||
// priceNotifyWs.beginSSL("ws.coincap.io", 443, "/prices?assets=bitcoin");
|
|
||||||
// priceNotifyWs.onEvent(onWebsocketPriceEvent);
|
|
||||||
// priceNotifyWs.setReconnectInterval(5000);
|
|
||||||
// priceNotifyWs.enableHeartbeat(15000, 3000, 2);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// void onWebsocketPriceEvent(WStype_t type, uint8_t * payload, size_t length) {
|
|
||||||
// switch(type) {
|
|
||||||
// case WStype_DISCONNECTED:
|
|
||||||
// Serial.printf("[WSc] Disconnected!\n");
|
|
||||||
// break;
|
|
||||||
// case WStype_CONNECTED:
|
|
||||||
// {
|
|
||||||
// Serial.printf("[WSc] Connected to url: %s\n", payload);
|
|
||||||
|
|
||||||
|
|
||||||
// break;
|
|
||||||
// }
|
|
||||||
// case WStype_TEXT:
|
|
||||||
// String message = String((char*)payload);
|
|
||||||
// onWebsocketPriceMessage(message);
|
|
||||||
// break;
|
|
||||||
// case WStype_BIN:
|
|
||||||
// break;
|
|
||||||
// case WStype_ERROR:
|
|
||||||
// case WStype_FRAGMENT_TEXT_START:
|
|
||||||
// case WStype_FRAGMENT_BIN_START:
|
|
||||||
// case WStype_FRAGMENT:
|
|
||||||
// case WStype_PING:
|
|
||||||
// case WStype_PONG:
|
|
||||||
// case WStype_FRAGMENT_FIN:
|
|
||||||
// break;
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
void onWebsocketPriceEvent(void *handler_args, esp_event_base_t base,
|
void onWebsocketPriceEvent(void *handler_args, esp_event_base_t base,
|
||||||
int32_t event_id, void *event_data)
|
int32_t event_id, void *event_data) {
|
||||||
{
|
|
||||||
esp_websocket_event_data_t *data = (esp_websocket_event_data_t *)event_data;
|
esp_websocket_event_data_t *data = (esp_websocket_event_data_t *)event_data;
|
||||||
|
|
||||||
switch (event_id)
|
switch (event_id) {
|
||||||
{
|
case WEBSOCKET_EVENT_CONNECTED:
|
||||||
case WEBSOCKET_EVENT_CONNECTED:
|
Serial.println(F("Connected to CoinCap.io WebSocket"));
|
||||||
Serial.println("Connected to " + String(config.uri) + " WebSocket");
|
break;
|
||||||
priceNotifyInit = true;
|
case WEBSOCKET_EVENT_DATA:
|
||||||
|
onWebsocketPriceMessage(data);
|
||||||
break;
|
break;
|
||||||
case WEBSOCKET_EVENT_DATA:
|
case WEBSOCKET_EVENT_ERROR:
|
||||||
onWebsocketPriceMessage(data);
|
Serial.println(F("Price WS Connnection error"));
|
||||||
if (preferences.getBool("ownDataSource", DEFAULT_OWN_DATA_SOURCE))
|
break;
|
||||||
{
|
case WEBSOCKET_EVENT_DISCONNECTED:
|
||||||
onWebsocketBlockMessage(data);
|
Serial.println(F("Price WS Connnection Closed"));
|
||||||
}
|
break;
|
||||||
break;
|
|
||||||
case WEBSOCKET_EVENT_ERROR:
|
|
||||||
Serial.println(F("Price WS Connnection error"));
|
|
||||||
break;
|
|
||||||
case WEBSOCKET_EVENT_DISCONNECTED:
|
|
||||||
Serial.println(F("Price WS Connnection Closed"));
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void onWebsocketPriceMessage(esp_websocket_event_data_t *event_data)
|
void onWebsocketPriceMessage(esp_websocket_event_data_t *event_data) {
|
||||||
{
|
SpiRamJsonDocument doc(event_data->data_len);
|
||||||
JsonDocument doc;
|
|
||||||
|
|
||||||
deserializeJson(doc, (char *)event_data->data_ptr);
|
deserializeJson(doc, (char *)event_data->data_ptr);
|
||||||
|
|
||||||
if (doc.containsKey("bitcoin"))
|
if (doc.containsKey("bitcoin")) {
|
||||||
{
|
if (currentPrice != doc["bitcoin"].as<long>()) {
|
||||||
if (currentPrice != doc["bitcoin"].as<long>())
|
uint minSecPriceUpd = preferences.getUInt(
|
||||||
{
|
"minSecPriceUpd", DEFAULT_SECONDS_BETWEEN_PRICE_UPDATE);
|
||||||
processNewPrice(doc["bitcoin"].as<long>(), CURRENCY_USD);
|
uint currentTime = esp_timer_get_time() / 1000000;
|
||||||
|
|
||||||
|
if (lastPriceUpdate == 0 ||
|
||||||
|
(currentTime - lastPriceUpdate) > minSecPriceUpd) {
|
||||||
|
// const unsigned long oldPrice = currentPrice;
|
||||||
|
currentPrice = doc["bitcoin"].as<uint>();
|
||||||
|
preferences.putUInt("lastPrice", currentPrice);
|
||||||
|
lastPriceUpdate = currentTime;
|
||||||
|
// if (abs((int)(oldPrice-currentPrice)) > round(0.0015*oldPrice)) {
|
||||||
|
if (workQueue != nullptr && (getCurrentScreen() == SCREEN_BTC_TICKER ||
|
||||||
|
getCurrentScreen() == SCREEN_MSCW_TIME ||
|
||||||
|
getCurrentScreen() == SCREEN_MARKET_CAP)) {
|
||||||
|
WorkItem priceUpdate = {TASK_PRICE_UPDATE, 0};
|
||||||
|
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
||||||
|
}
|
||||||
|
//}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void processNewPrice(uint newPrice, char currency)
|
uint getPrice() { return currentPrice; }
|
||||||
{
|
|
||||||
uint minSecPriceUpd = preferences.getUInt(
|
|
||||||
"minSecPriceUpd", DEFAULT_SECONDS_BETWEEN_PRICE_UPDATE);
|
|
||||||
uint currentTime = esp_timer_get_time() / 1000000;
|
|
||||||
|
|
||||||
if (lastUpdateMap.find(currency) == lastUpdateMap.end() ||
|
void setPrice(uint newPrice) { currentPrice = newPrice; }
|
||||||
(currentTime - lastUpdateMap[currency]) > minSecPriceUpd)
|
|
||||||
{
|
|
||||||
// const unsigned long oldPrice = currentPrice;
|
|
||||||
currencyMap[currency] = newPrice;
|
|
||||||
if (currency == CURRENCY_USD && ( lastUpdateMap[currency] == 0 ||
|
|
||||||
(currentTime - lastUpdateMap[currency]) > 120))
|
|
||||||
{
|
|
||||||
preferences.putUInt("lastPrice", currentPrice);
|
|
||||||
}
|
|
||||||
lastUpdateMap[currency] = currentTime;
|
|
||||||
// if (abs((int)(oldPrice-currentPrice)) > round(0.0015*oldPrice)) {
|
|
||||||
if (workQueue != nullptr && (getCurrentScreen() == SCREEN_BTC_TICKER ||
|
|
||||||
getCurrentScreen() == SCREEN_SATS_PER_CURRENCY ||
|
|
||||||
getCurrentScreen() == SCREEN_MARKET_CAP))
|
|
||||||
{
|
|
||||||
WorkItem priceUpdate = {TASK_PRICE_UPDATE, currency};
|
|
||||||
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
|
||||||
}
|
|
||||||
//}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
uint getLastPriceUpdate(char currency)
|
bool isPriceNotifyConnected() {
|
||||||
{
|
if (clientPrice == NULL) return false;
|
||||||
if (lastUpdateMap.find(currency) == lastUpdateMap.end())
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return lastUpdateMap[currency];
|
|
||||||
}
|
|
||||||
|
|
||||||
uint getPrice(char currency)
|
|
||||||
{
|
|
||||||
if (currencyMap.find(currency) == currencyMap.end())
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
return currencyMap[currency];
|
|
||||||
}
|
|
||||||
|
|
||||||
void setPrice(uint newPrice, char currency)
|
|
||||||
{
|
|
||||||
currencyMap[currency] = newPrice;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isPriceNotifyConnected()
|
|
||||||
{
|
|
||||||
if (clientPrice == NULL)
|
|
||||||
return false;
|
|
||||||
return esp_websocket_client_is_connected(clientPrice);
|
return esp_websocket_client_is_connected(clientPrice);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool getPriceNotifyInit()
|
void stopPriceNotify() {
|
||||||
{
|
|
||||||
return priceNotifyInit;
|
|
||||||
}
|
|
||||||
|
|
||||||
void stopPriceNotify()
|
|
||||||
{
|
|
||||||
if (clientPrice == NULL)
|
|
||||||
return;
|
|
||||||
esp_websocket_client_close(clientPrice, pdMS_TO_TICKS(5000));
|
|
||||||
esp_websocket_client_stop(clientPrice);
|
esp_websocket_client_stop(clientPrice);
|
||||||
esp_websocket_client_destroy(clientPrice);
|
esp_websocket_client_destroy(clientPrice);
|
||||||
|
|
||||||
clientPrice = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
void restartPriceNotify()
|
|
||||||
{
|
|
||||||
stopPriceNotify();
|
|
||||||
if (clientPrice == NULL)
|
|
||||||
{
|
|
||||||
setupPriceNotify();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// esp_websocket_client_close(clientPrice, pdMS_TO_TICKS(5000));
|
|
||||||
// esp_websocket_client_stop(clientPrice);
|
|
||||||
// esp_websocket_client_start(clientPrice);
|
|
||||||
}
|
}
|
|
@ -3,7 +3,7 @@
|
||||||
#include <Arduino.h>
|
#include <Arduino.h>
|
||||||
#include <ArduinoJson.h>
|
#include <ArduinoJson.h>
|
||||||
#include <esp_websocket_client.h>
|
#include <esp_websocket_client.h>
|
||||||
#include "block_notify.hpp"
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
#include "lib/screen_handler.hpp"
|
#include "lib/screen_handler.hpp"
|
||||||
|
@ -12,19 +12,10 @@ void setupPriceNotify();
|
||||||
|
|
||||||
void onWebsocketPriceEvent(void *handler_args, esp_event_base_t base,
|
void onWebsocketPriceEvent(void *handler_args, esp_event_base_t base,
|
||||||
int32_t event_id, void *event_data);
|
int32_t event_id, void *event_data);
|
||||||
//void onWebsocketPriceEvent(WStype_t type, uint8_t * payload, size_t length);
|
|
||||||
|
|
||||||
void onWebsocketPriceMessage(esp_websocket_event_data_t *event_data);
|
void onWebsocketPriceMessage(esp_websocket_event_data_t *event_data);
|
||||||
|
|
||||||
uint getPrice(char currency);
|
uint getPrice();
|
||||||
void setPrice(uint newPrice, char currency);
|
void setPrice(uint newPrice);
|
||||||
|
|
||||||
//void processNewPrice(uint newPrice);
|
|
||||||
void processNewPrice(uint newPrice, char currency);
|
|
||||||
|
|
||||||
bool isPriceNotifyConnected();
|
bool isPriceNotifyConnected();
|
||||||
void stopPriceNotify();
|
void stopPriceNotify();
|
||||||
void restartPriceNotify();
|
|
||||||
|
|
||||||
bool getPriceNotifyInit();
|
|
||||||
uint getLastPriceUpdate(char currency);
|
|
|
@ -5,16 +5,17 @@
|
||||||
// TaskHandle_t timeUpdateTaskHandle;
|
// TaskHandle_t timeUpdateTaskHandle;
|
||||||
TaskHandle_t taskScreenRotateTaskHandle;
|
TaskHandle_t taskScreenRotateTaskHandle;
|
||||||
TaskHandle_t workerTaskHandle;
|
TaskHandle_t workerTaskHandle;
|
||||||
|
esp_timer_handle_t screenRotateTimer;
|
||||||
|
esp_timer_handle_t minuteTimer;
|
||||||
|
|
||||||
|
std::array<std::string, NUM_SCREENS> taskEpdContent = {"", "", "", "",
|
||||||
std::array<std::string, NUM_SCREENS> taskEpdContent = {};
|
"", "", ""};
|
||||||
std::string priceString;
|
std::string priceString;
|
||||||
|
|
||||||
#define WORK_QUEUE_SIZE 10
|
#define WORK_QUEUE_SIZE 10
|
||||||
QueueHandle_t workQueue = NULL;
|
QueueHandle_t workQueue = NULL;
|
||||||
|
|
||||||
uint currentScreen = SCREEN_BLOCK_HEIGHT;
|
uint currentScreen;
|
||||||
uint currentCurrency = CURRENCY_USD;
|
|
||||||
|
|
||||||
void workerTask(void *pvParameters) {
|
void workerTask(void *pvParameters) {
|
||||||
WorkItem receivedItem;
|
WorkItem receivedItem;
|
||||||
|
@ -22,52 +23,34 @@ void workerTask(void *pvParameters) {
|
||||||
while (1) {
|
while (1) {
|
||||||
// Wait for a work item to be available in the queue
|
// Wait for a work item to be available in the queue
|
||||||
if (xQueueReceive(workQueue, &receivedItem, portMAX_DELAY)) {
|
if (xQueueReceive(workQueue, &receivedItem, portMAX_DELAY)) {
|
||||||
|
uint firstIndex = 0;
|
||||||
|
|
||||||
// Process the work item based on its type
|
// Process the work item based on its type
|
||||||
switch (receivedItem.type) {
|
switch (receivedItem.type) {
|
||||||
case TASK_BITAXE_UPDATE: {
|
|
||||||
if (getCurrentScreen() == SCREEN_BITAXE_HASHRATE) {
|
|
||||||
taskEpdContent =
|
|
||||||
parseBitaxeHashRate(getBitAxeHashRate());
|
|
||||||
} else if (getCurrentScreen() == SCREEN_BITAXE_BESTDIFF) {
|
|
||||||
taskEpdContent =
|
|
||||||
parseBitaxeBestDiff(getBitaxeBestDiff());
|
|
||||||
}
|
|
||||||
setEpdContent(taskEpdContent);
|
|
||||||
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case TASK_PRICE_UPDATE: {
|
case TASK_PRICE_UPDATE: {
|
||||||
uint currency = getCurrentCurrency();
|
uint price = getPrice();
|
||||||
uint price = getPrice(currency);
|
char priceSymbol = '$';
|
||||||
|
if (preferences.getBool("fetchEurPrice", false)) {
|
||||||
|
priceSymbol = '[';
|
||||||
|
}
|
||||||
if (getCurrentScreen() == SCREEN_BTC_TICKER) {
|
if (getCurrentScreen() == SCREEN_BTC_TICKER) {
|
||||||
taskEpdContent = parsePriceData(price, currency, preferences.getBool("suffixPrice", DEFAULT_SUFFIX_PRICE),
|
taskEpdContent = parsePriceData(price, priceSymbol);
|
||||||
preferences.getBool("mowMode", DEFAULT_MOW_MODE),
|
} else if (getCurrentScreen() == SCREEN_MSCW_TIME) {
|
||||||
preferences.getBool("suffixShareDot", DEFAULT_SUFFIX_SHARE_DOT)
|
taskEpdContent = parseSatsPerCurrency(price, priceSymbol);
|
||||||
);
|
|
||||||
} else if (getCurrentScreen() == SCREEN_SATS_PER_CURRENCY) {
|
|
||||||
taskEpdContent = parseSatsPerCurrency(price, currency, preferences.getBool("useSatsSymbol", DEFAULT_USE_SATS_SYMBOL));
|
|
||||||
} else {
|
} else {
|
||||||
taskEpdContent =
|
taskEpdContent =
|
||||||
parseMarketCap(getBlockHeight(), price, currency,
|
parseMarketCap(getBlockHeight(), price, priceSymbol,
|
||||||
preferences.getBool("mcapBigChar", DEFAULT_MCAP_BIG_CHAR));
|
preferences.getBool("mcapBigChar", true));
|
||||||
}
|
}
|
||||||
|
|
||||||
setEpdContent(taskEpdContent);
|
setEpdContent(taskEpdContent);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case TASK_FEE_UPDATE: {
|
|
||||||
if (getCurrentScreen() == SCREEN_BLOCK_FEE_RATE) {
|
|
||||||
taskEpdContent = parseBlockFees(static_cast<std::uint16_t>(getBlockMedianFee()));
|
|
||||||
setEpdContent(taskEpdContent);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case TASK_BLOCK_UPDATE: {
|
case TASK_BLOCK_UPDATE: {
|
||||||
if (getCurrentScreen() != SCREEN_HALVING_COUNTDOWN) {
|
if (getCurrentScreen() != SCREEN_HALVING_COUNTDOWN) {
|
||||||
taskEpdContent = parseBlockHeight(getBlockHeight());
|
taskEpdContent = parseBlockHeight(getBlockHeight());
|
||||||
} else {
|
} else {
|
||||||
taskEpdContent = parseHalvingCountdown(getBlockHeight(), preferences.getBool("useBlkCountdown", DEFAULT_USE_BLOCK_COUNTDOWN));
|
taskEpdContent = parseHalvingCountdown(getBlockHeight());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (getCurrentScreen() == SCREEN_HALVING_COUNTDOWN ||
|
if (getCurrentScreen() == SCREEN_HALVING_COUNTDOWN ||
|
||||||
|
@ -114,7 +97,36 @@ void taskScreenRotate(void *pvParameters) {
|
||||||
for (;;) {
|
for (;;) {
|
||||||
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
||||||
|
|
||||||
nextScreen();
|
int nextScreen = (currentScreen + 1) % SCREEN_COUNT;
|
||||||
|
String key = "screen" + String(nextScreen) + "Visible";
|
||||||
|
|
||||||
|
while (!preferences.getBool(key.c_str(), true)) {
|
||||||
|
nextScreen = (nextScreen + 1) % SCREEN_COUNT;
|
||||||
|
key = "screen" + String(nextScreen) + "Visible";
|
||||||
|
}
|
||||||
|
|
||||||
|
setCurrentScreen(nextScreen);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void IRAM_ATTR minuteTimerISR(void *arg) {
|
||||||
|
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
||||||
|
// vTaskNotifyGiveFromISR(timeUpdateTaskHandle, &xHigherPriorityTaskWoken);
|
||||||
|
WorkItem timeUpdate = {TASK_TIME_UPDATE, 0};
|
||||||
|
xQueueSendFromISR(workQueue, &timeUpdate, &xHigherPriorityTaskWoken);
|
||||||
|
if (priceFetchTaskHandle != NULL) {
|
||||||
|
vTaskNotifyGiveFromISR(priceFetchTaskHandle, &xHigherPriorityTaskWoken);
|
||||||
|
}
|
||||||
|
if (xHigherPriorityTaskWoken == pdTRUE) {
|
||||||
|
portYIELD_FROM_ISR();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void IRAM_ATTR screenRotateTimerISR(void *arg) {
|
||||||
|
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
||||||
|
vTaskNotifyGiveFromISR(taskScreenRotateTaskHandle, &xHigherPriorityTaskWoken);
|
||||||
|
if (xHigherPriorityTaskWoken == pdTRUE) {
|
||||||
|
portYIELD_FROM_ISR();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -124,15 +136,72 @@ void setupTasks() {
|
||||||
xTaskCreate(workerTask, "workerTask", 4096, NULL, tskIDLE_PRIORITY,
|
xTaskCreate(workerTask, "workerTask", 4096, NULL, tskIDLE_PRIORITY,
|
||||||
&workerTaskHandle);
|
&workerTaskHandle);
|
||||||
|
|
||||||
xTaskCreate(taskScreenRotate, "rotateScreen", 4096, NULL, tskIDLE_PRIORITY,
|
xTaskCreate(taskScreenRotate, "rotateScreen", 2048, NULL, tskIDLE_PRIORITY,
|
||||||
&taskScreenRotateTaskHandle);
|
&taskScreenRotateTaskHandle);
|
||||||
|
|
||||||
waitUntilNoneBusy();
|
waitUntilNoneBusy();
|
||||||
|
setCurrentScreen(preferences.getUInt("currentScreen", 0));
|
||||||
if (findScreenIndexByValue(preferences.getUInt("currentScreen", DEFAULT_CURRENT_SCREEN)) != -1)
|
|
||||||
setCurrentScreen(preferences.getUInt("currentScreen", DEFAULT_CURRENT_SCREEN));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setupTimeUpdateTimer(void *pvParameters) {
|
||||||
|
const esp_timer_create_args_t minuteTimerConfig = {
|
||||||
|
.callback = &minuteTimerISR, .name = "minute_timer"};
|
||||||
|
|
||||||
|
esp_timer_create(&minuteTimerConfig, &minuteTimer);
|
||||||
|
|
||||||
|
time_t currentTime;
|
||||||
|
struct tm timeinfo;
|
||||||
|
time(¤tTime);
|
||||||
|
localtime_r(¤tTime, &timeinfo);
|
||||||
|
uint32_t secondsUntilNextMinute = 60 - timeinfo.tm_sec;
|
||||||
|
|
||||||
|
if (secondsUntilNextMinute > 0)
|
||||||
|
vTaskDelay(pdMS_TO_TICKS((secondsUntilNextMinute * 1000)));
|
||||||
|
|
||||||
|
esp_timer_start_periodic(minuteTimer, usPerMinute);
|
||||||
|
|
||||||
|
WorkItem timeUpdate = {TASK_TIME_UPDATE, 0};
|
||||||
|
xQueueSend(workQueue, &timeUpdate, portMAX_DELAY);
|
||||||
|
// xTaskNotifyGive(timeUpdateTaskHandle);
|
||||||
|
|
||||||
|
vTaskDelete(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setupScreenRotateTimer(void *pvParameters) {
|
||||||
|
const esp_timer_create_args_t screenRotateTimerConfig = {
|
||||||
|
.callback = &screenRotateTimerISR, .name = "screen_rotate_timer"};
|
||||||
|
|
||||||
|
esp_timer_create(&screenRotateTimerConfig, &screenRotateTimer);
|
||||||
|
|
||||||
|
if (preferences.getBool("timerActive", true)) {
|
||||||
|
esp_timer_start_periodic(screenRotateTimer,
|
||||||
|
getTimerSeconds() * usPerSecond);
|
||||||
|
}
|
||||||
|
|
||||||
|
vTaskDelete(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint getTimerSeconds() { return preferences.getUInt("timerSeconds", 1800); }
|
||||||
|
|
||||||
|
bool isTimerActive() { return esp_timer_is_active(screenRotateTimer); }
|
||||||
|
|
||||||
|
void setTimerActive(bool status) {
|
||||||
|
if (status) {
|
||||||
|
esp_timer_start_periodic(screenRotateTimer,
|
||||||
|
getTimerSeconds() * usPerSecond);
|
||||||
|
queueLedEffect(LED_EFFECT_START_TIMER);
|
||||||
|
preferences.putBool("timerActive", true);
|
||||||
|
} else {
|
||||||
|
esp_timer_stop(screenRotateTimer);
|
||||||
|
queueLedEffect(LED_EFFECT_PAUSE_TIMER);
|
||||||
|
preferences.putBool("timerActive", false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (eventSourceTaskHandle != NULL) xTaskNotifyGive(eventSourceTaskHandle);
|
||||||
|
}
|
||||||
|
|
||||||
|
void toggleTimerActive() { setTimerActive(!isTimerActive()); }
|
||||||
|
|
||||||
uint getCurrentScreen() { return currentScreen; }
|
uint getCurrentScreen() { return currentScreen; }
|
||||||
|
|
||||||
void setCurrentScreen(uint newScreen) {
|
void setCurrentScreen(uint newScreen) {
|
||||||
|
@ -157,136 +226,43 @@ void setCurrentScreen(uint newScreen) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case SCREEN_MARKET_CAP:
|
case SCREEN_MARKET_CAP:
|
||||||
case SCREEN_SATS_PER_CURRENCY:
|
case SCREEN_MSCW_TIME:
|
||||||
case SCREEN_BTC_TICKER: {
|
case SCREEN_BTC_TICKER: {
|
||||||
WorkItem priceUpdate = {TASK_PRICE_UPDATE, 0};
|
WorkItem priceUpdate = {TASK_PRICE_UPDATE, 0};
|
||||||
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
||||||
// xTaskNotifyGive(priceUpdateTaskHandle);
|
// xTaskNotifyGive(priceUpdateTaskHandle);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case SCREEN_BLOCK_FEE_RATE: {
|
|
||||||
WorkItem blockUpdate = {TASK_FEE_UPDATE, 0};
|
|
||||||
xQueueSend(workQueue, &blockUpdate, portMAX_DELAY);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case SCREEN_BITAXE_BESTDIFF:
|
|
||||||
case SCREEN_BITAXE_HASHRATE: {
|
|
||||||
if (preferences.getBool("bitaxeEnabled", DEFAULT_BITAXE_ENABLED)) {
|
|
||||||
WorkItem bitaxeUpdate = {TASK_BITAXE_UPDATE, 0};
|
|
||||||
xQueueSend(workQueue, &bitaxeUpdate, portMAX_DELAY);
|
|
||||||
} else {
|
|
||||||
setCurrentScreen(SCREEN_BLOCK_HEIGHT);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (eventSourceTaskHandle != NULL) xTaskNotifyGive(eventSourceTaskHandle);
|
if (eventSourceTaskHandle != NULL) xTaskNotifyGive(eventSourceTaskHandle);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool isCurrencySpecific(uint screen) {
|
|
||||||
switch (screen) {
|
|
||||||
case SCREEN_BTC_TICKER:
|
|
||||||
case SCREEN_SATS_PER_CURRENCY:
|
|
||||||
case SCREEN_MARKET_CAP:
|
|
||||||
return true;
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void nextScreen() {
|
void nextScreen() {
|
||||||
int currentIndex = findScreenIndexByValue(getCurrentScreen());
|
int newCurrentScreen = (getCurrentScreen() + 1) % SCREEN_COUNT;
|
||||||
std::vector<ScreenMapping> screenMappings = getScreenNameMap();
|
|
||||||
|
|
||||||
if (preferences.getBool("ownDataSource", DEFAULT_OWN_DATA_SOURCE) && isCurrencySpecific(getCurrentScreen())) {
|
|
||||||
std::vector<std::string> ac = getActiveCurrencies();
|
|
||||||
std::string curCode = getCurrencyCode(getCurrentCurrency());
|
|
||||||
if (getCurrencyCode(getCurrentCurrency()) != ac.back()) {
|
|
||||||
auto it = std::find(ac.begin(), ac.end(), curCode);
|
|
||||||
if (it != ac.end()) {
|
|
||||||
size_t index = std::distance(ac.begin(), it);
|
|
||||||
setCurrentCurrency(getCurrencyChar(ac.at(index+1)));
|
|
||||||
setCurrentScreen(getCurrentScreen());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
setCurrentCurrency(getCurrencyChar(ac.front()));
|
|
||||||
}
|
|
||||||
|
|
||||||
int newCurrentScreen;
|
|
||||||
|
|
||||||
if (currentIndex < screenMappings.size() - 1) {
|
|
||||||
newCurrentScreen = (screenMappings[currentIndex + 1].value);
|
|
||||||
} else {
|
|
||||||
newCurrentScreen = screenMappings.front().value;
|
|
||||||
}
|
|
||||||
|
|
||||||
String key = "screen" + String(newCurrentScreen) + "Visible";
|
String key = "screen" + String(newCurrentScreen) + "Visible";
|
||||||
|
|
||||||
while (!preferences.getBool(key.c_str(), true)) {
|
while (!preferences.getBool(key.c_str(), true)) {
|
||||||
currentIndex = findScreenIndexByValue(newCurrentScreen);
|
newCurrentScreen = (newCurrentScreen + 1) % SCREEN_COUNT;
|
||||||
if (currentIndex < screenMappings.size() - 1) {
|
|
||||||
newCurrentScreen = (screenMappings[currentIndex + 1].value);
|
|
||||||
} else {
|
|
||||||
newCurrentScreen = screenMappings.front().value;
|
|
||||||
}
|
|
||||||
|
|
||||||
key = "screen" + String(newCurrentScreen) + "Visible";
|
key = "screen" + String(newCurrentScreen) + "Visible";
|
||||||
}
|
}
|
||||||
|
|
||||||
setCurrentScreen(newCurrentScreen);
|
setCurrentScreen(newCurrentScreen);
|
||||||
}
|
}
|
||||||
|
|
||||||
void previousScreen() {
|
void previousScreen() {
|
||||||
int currentIndex = findScreenIndexByValue(getCurrentScreen());
|
int newCurrentScreen = modulo(getCurrentScreen() - 1, SCREEN_COUNT);
|
||||||
std::vector<ScreenMapping> screenMappings = getScreenNameMap();
|
|
||||||
|
|
||||||
if (preferences.getBool("ownDataSource", DEFAULT_OWN_DATA_SOURCE) && isCurrencySpecific(getCurrentScreen())) {
|
|
||||||
std::vector<std::string> ac = getActiveCurrencies();
|
|
||||||
std::string curCode = getCurrencyCode(getCurrentCurrency());
|
|
||||||
if (getCurrencyCode(getCurrentCurrency()) != ac.front()) {
|
|
||||||
auto it = std::find(ac.begin(), ac.end(), curCode);
|
|
||||||
if (it != ac.end()) {
|
|
||||||
size_t index = std::distance(ac.begin(), it);
|
|
||||||
setCurrentCurrency(getCurrencyChar(ac.at(index-1)));
|
|
||||||
setCurrentScreen(getCurrentScreen());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
setCurrentCurrency(getCurrencyChar(ac.back()));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int newCurrentScreen;
|
|
||||||
|
|
||||||
if (currentIndex > 0) {
|
|
||||||
newCurrentScreen = screenMappings[currentIndex - 1].value;
|
|
||||||
} else {
|
|
||||||
newCurrentScreen = screenMappings.back().value;
|
|
||||||
}
|
|
||||||
|
|
||||||
String key = "screen" + String(newCurrentScreen) + "Visible";
|
String key = "screen" + String(newCurrentScreen) + "Visible";
|
||||||
|
|
||||||
while (!preferences.getBool(key.c_str(), true)) {
|
while (!preferences.getBool(key.c_str(), true)) {
|
||||||
int currentIndex = findScreenIndexByValue(newCurrentScreen);
|
newCurrentScreen = modulo(newCurrentScreen - 1, SCREEN_COUNT);
|
||||||
if (currentIndex > 0) {
|
|
||||||
newCurrentScreen = screenMappings[currentIndex - 1].value;
|
|
||||||
} else {
|
|
||||||
newCurrentScreen = screenMappings.back().value;
|
|
||||||
}
|
|
||||||
|
|
||||||
key = "screen" + String(newCurrentScreen) + "Visible";
|
key = "screen" + String(newCurrentScreen) + "Visible";
|
||||||
}
|
}
|
||||||
setCurrentScreen(newCurrentScreen);
|
setCurrentScreen(newCurrentScreen);
|
||||||
}
|
}
|
||||||
|
|
||||||
void showSystemStatusScreen() {
|
void showSystemStatusScreen() {
|
||||||
std::array<String, NUM_SCREENS> sysStatusEpdContent;
|
std::array<String, NUM_SCREENS> sysStatusEpdContent = {"", "", "", "",
|
||||||
std::fill(sysStatusEpdContent.begin(), sysStatusEpdContent.end(), "");
|
"", "", ""};
|
||||||
|
|
||||||
|
|
||||||
String ipAddr = WiFi.localIP().toString();
|
String ipAddr = WiFi.localIP().toString();
|
||||||
String subNet = WiFi.subnetMask().toString();
|
String subNet = WiFi.subnetMask().toString();
|
||||||
|
@ -311,12 +287,3 @@ void showSystemStatusScreen() {
|
||||||
setCurrentScreen(SCREEN_CUSTOM);
|
setCurrentScreen(SCREEN_CUSTOM);
|
||||||
setEpdContent(sysStatusEpdContent);
|
setEpdContent(sysStatusEpdContent);
|
||||||
}
|
}
|
||||||
|
|
||||||
void setCurrentCurrency(char currency) {
|
|
||||||
currentCurrency = currency;
|
|
||||||
preferences.putUChar("lastCurrency", currency);
|
|
||||||
}
|
|
||||||
|
|
||||||
uint getCurrentCurrency() {
|
|
||||||
return currentCurrency;
|
|
||||||
}
|
|
|
@ -5,9 +5,9 @@
|
||||||
#include <freertos/task.h>
|
#include <freertos/task.h>
|
||||||
|
|
||||||
#include <data_handler.hpp>
|
#include <data_handler.hpp>
|
||||||
#include <bitaxe_handler.hpp>
|
|
||||||
|
|
||||||
#include "lib/epd.hpp"
|
#include "lib/epd.hpp"
|
||||||
|
#include "lib/price_fetch.hpp"
|
||||||
#include "lib/shared.hpp"
|
#include "lib/shared.hpp"
|
||||||
|
|
||||||
// extern TaskHandle_t priceUpdateTaskHandle;
|
// extern TaskHandle_t priceUpdateTaskHandle;
|
||||||
|
@ -16,14 +16,15 @@
|
||||||
extern TaskHandle_t workerTaskHandle;
|
extern TaskHandle_t workerTaskHandle;
|
||||||
extern TaskHandle_t taskScreenRotateTaskHandle;
|
extern TaskHandle_t taskScreenRotateTaskHandle;
|
||||||
|
|
||||||
|
extern esp_timer_handle_t screenRotateTimer;
|
||||||
|
extern esp_timer_handle_t minuteTimer;
|
||||||
|
|
||||||
extern QueueHandle_t workQueue;
|
extern QueueHandle_t workQueue;
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
TASK_PRICE_UPDATE,
|
TASK_PRICE_UPDATE,
|
||||||
TASK_BLOCK_UPDATE,
|
TASK_BLOCK_UPDATE,
|
||||||
TASK_FEE_UPDATE,
|
TASK_TIME_UPDATE
|
||||||
TASK_TIME_UPDATE,
|
|
||||||
TASK_BITAXE_UPDATE
|
|
||||||
} TaskType;
|
} TaskType;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
|
@ -39,16 +40,20 @@ void previousScreen();
|
||||||
|
|
||||||
void showSystemStatusScreen();
|
void showSystemStatusScreen();
|
||||||
|
|
||||||
|
void setupTimeUpdateTimer(void *pvParameters);
|
||||||
|
void setupScreenRotateTimer(void *pvParameters);
|
||||||
|
|
||||||
|
void IRAM_ATTR minuteTimerISR(void *arg);
|
||||||
|
void IRAM_ATTR screenRotateTimerISR(void *arg);
|
||||||
|
|
||||||
// void taskPriceUpdate(void *pvParameters);
|
// void taskPriceUpdate(void *pvParameters);
|
||||||
// void taskBlockUpdate(void *pvParameters);
|
// void taskBlockUpdate(void *pvParameters);
|
||||||
// void taskTimeUpdate(void *pvParameters);
|
// void taskTimeUpdate(void *pvParameters);
|
||||||
void taskScreenRotate(void *pvParameters);
|
void taskScreenRotate(void *pvParameters);
|
||||||
|
|
||||||
|
uint getTimerSeconds();
|
||||||
|
bool isTimerActive();
|
||||||
|
void setTimerActive(bool status);
|
||||||
|
void toggleTimerActive();
|
||||||
|
|
||||||
void setupTasks();
|
void setupTasks();
|
||||||
void setCurrentCurrency(char currency);
|
|
||||||
|
|
||||||
uint getCurrentCurrency();
|
|
|
@ -1,154 +0,0 @@
|
||||||
#include "shared.hpp"
|
|
||||||
|
|
||||||
// const char *github_root_ca =
|
|
||||||
// "-----BEGIN CERTIFICATE-----\n"
|
|
||||||
// "MIICjzCCAhWgAwIBAgIQXIuZxVqUxdJxVt7NiYDMJjAKBggqhkjOPQQDAzCBiDEL\n"
|
|
||||||
// "MAkGA1UEBhMCVVMxEzARBgNVBAgTCk5ldyBKZXJzZXkxFDASBgNVBAcTC0plcnNl\n"
|
|
||||||
// "eSBDaXR5MR4wHAYDVQQKExVUaGUgVVNFUlRSVVNUIE5ldHdvcmsxLjAsBgNVBAMT\n"
|
|
||||||
// "JVVTRVJUcnVzdCBFQ0MgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkwHhcNMTAwMjAx\n"
|
|
||||||
// "MDAwMDAwWhcNMzgwMTE4MjM1OTU5WjCBiDELMAkGA1UEBhMCVVMxEzARBgNVBAgT\n"
|
|
||||||
// "Ck5ldyBKZXJzZXkxFDASBgNVBAcTC0plcnNleSBDaXR5MR4wHAYDVQQKExVUaGUg\n"
|
|
||||||
// "VVNFUlRSVVNUIE5ldHdvcmsxLjAsBgNVBAMTJVVTRVJUcnVzdCBFQ0MgQ2VydGlm\n"
|
|
||||||
// "aWNhdGlvbiBBdXRob3JpdHkwdjAQBgcqhkjOPQIBBgUrgQQAIgNiAAQarFRaqflo\n"
|
|
||||||
// "I+d61SRvU8Za2EurxtW20eZzca7dnNYMYf3boIkDuAUU7FfO7l0/4iGzzvfUinng\n"
|
|
||||||
// "o4N+LZfQYcTxmdwlkWOrfzCjtHDix6EznPO/LlxTsV+zfTJ/ijTjeXmjQjBAMB0G\n"
|
|
||||||
// "A1UdDgQWBBQ64QmG1M8ZwpZ2dEl23OA1xmNjmjAOBgNVHQ8BAf8EBAMCAQYwDwYD\n"
|
|
||||||
// "VR0TAQH/BAUwAwEB/zAKBggqhkjOPQQDAwNoADBlAjA2Z6EWCNzklwBBHU6+4WMB\n"
|
|
||||||
// "zzuqQhFkoJ2UOQIReVx7Hfpkue4WQrO/isIJxOzksU0CMQDpKmFHjFJKS04YcPbW\n"
|
|
||||||
// "RNZu9YO6bVi9JNlWSOrvxKJGgYhqOkbRqZtNyWHa0V1Xahg=\n"
|
|
||||||
// "-----END CERTIFICATE-----\n"
|
|
||||||
// "-----BEGIN CERTIFICATE-----\n"
|
|
||||||
// "MIIDjjCCAnagAwIBAgIQAzrx5qcRqaC7KGSxHQn65TANBgkqhkiG9w0BAQsFADBh\n"
|
|
||||||
// "MQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3\n"
|
|
||||||
// "d3cuZGlnaWNlcnQuY29tMSAwHgYDVQQDExdEaWdpQ2VydCBHbG9iYWwgUm9vdCBH\n"
|
|
||||||
// "MjAeFw0xMzA4MDExMjAwMDBaFw0zODAxMTUxMjAwMDBaMGExCzAJBgNVBAYTAlVT\n"
|
|
||||||
// "MRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5j\n"
|
|
||||||
// "b20xIDAeBgNVBAMTF0RpZ2lDZXJ0IEdsb2JhbCBSb290IEcyMIIBIjANBgkqhkiG\n"
|
|
||||||
// "9w0BAQEFAAOCAQ8AMIIBCgKCAQEAuzfNNNx7a8myaJCtSnX/RrohCgiN9RlUyfuI\n"
|
|
||||||
// "2/Ou8jqJkTx65qsGGmvPrC3oXgkkRLpimn7Wo6h+4FR1IAWsULecYxpsMNzaHxmx\n"
|
|
||||||
// "1x7e/dfgy5SDN67sH0NO3Xss0r0upS/kqbitOtSZpLYl6ZtrAGCSYP9PIUkY92eQ\n"
|
|
||||||
// "q2EGnI/yuum06ZIya7XzV+hdG82MHauVBJVJ8zUtluNJbd134/tJS7SsVQepj5Wz\n"
|
|
||||||
// "tCO7TG1F8PapspUwtP1MVYwnSlcUfIKdzXOS0xZKBgyMUNGPHgm+F6HmIcr9g+UQ\n"
|
|
||||||
// "vIOlCsRnKPZzFBQ9RnbDhxSJITRNrw9FDKZJobq7nMWxM4MphQIDAQABo0IwQDAP\n"
|
|
||||||
// "BgNVHRMBAf8EBTADAQH/MA4GA1UdDwEB/wQEAwIBhjAdBgNVHQ4EFgQUTiJUIBiV\n"
|
|
||||||
// "5uNu5g/6+rkS7QYXjzkwDQYJKoZIhvcNAQELBQADggEBAGBnKJRvDkhj6zHd6mcY\n"
|
|
||||||
// "1Yl9PMWLSn/pvtsrF9+wX3N3KjITOYFnQoQj8kVnNeyIv/iPsGEMNKSuIEyExtv4\n"
|
|
||||||
// "NeF22d+mQrvHRAiGfzZ0JFrabA0UWTW98kndth/Jsw1HKj2ZL7tcu7XUIOGZX1NG\n"
|
|
||||||
// "Fdtom/DzMNU+MeKNhJ7jitralj41E6Vf8PlwUHBHQRFXGU7Aj64GxJUTFy8bJZ91\n"
|
|
||||||
// "8rGOmaFvE7FBcf6IKshPECBV1/MUReXgRPTqh5Uykw7+U0b6LJ3/iyK5S9kJRaTe\n"
|
|
||||||
// "pLiaWN0bfVKfjllDiIGknibVb63dDcY3fe0Dkhvld1927jyNxF1WW6LZZm6zNTfl\n"
|
|
||||||
// "MrY=\n"
|
|
||||||
// "-----END CERTIFICATE-----\n";
|
|
||||||
|
|
||||||
const char* isrg_root_x1cert = R"EOF(
|
|
||||||
-----BEGIN CERTIFICATE-----
|
|
||||||
MIIFazCCA1OgAwIBAgIRAIIQz7DSQONZRGPgu2OCiwAwDQYJKoZIhvcNAQELBQAw
|
|
||||||
TzELMAkGA1UEBhMCVVMxKTAnBgNVBAoTIEludGVybmV0IFNlY3VyaXR5IFJlc2Vh
|
|
||||||
cmNoIEdyb3VwMRUwEwYDVQQDEwxJU1JHIFJvb3QgWDEwHhcNMTUwNjA0MTEwNDM4
|
|
||||||
WhcNMzUwNjA0MTEwNDM4WjBPMQswCQYDVQQGEwJVUzEpMCcGA1UEChMgSW50ZXJu
|
|
||||||
ZXQgU2VjdXJpdHkgUmVzZWFyY2ggR3JvdXAxFTATBgNVBAMTDElTUkcgUm9vdCBY
|
|
||||||
MTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAK3oJHP0FDfzm54rVygc
|
|
||||||
h77ct984kIxuPOZXoHj3dcKi/vVqbvYATyjb3miGbESTtrFj/RQSa78f0uoxmyF+
|
|
||||||
0TM8ukj13Xnfs7j/EvEhmkvBioZxaUpmZmyPfjxwv60pIgbz5MDmgK7iS4+3mX6U
|
|
||||||
A5/TR5d8mUgjU+g4rk8Kb4Mu0UlXjIB0ttov0DiNewNwIRt18jA8+o+u3dpjq+sW
|
|
||||||
T8KOEUt+zwvo/7V3LvSye0rgTBIlDHCNAymg4VMk7BPZ7hm/ELNKjD+Jo2FR3qyH
|
|
||||||
B5T0Y3HsLuJvW5iB4YlcNHlsdu87kGJ55tukmi8mxdAQ4Q7e2RCOFvu396j3x+UC
|
|
||||||
B5iPNgiV5+I3lg02dZ77DnKxHZu8A/lJBdiB3QW0KtZB6awBdpUKD9jf1b0SHzUv
|
|
||||||
KBds0pjBqAlkd25HN7rOrFleaJ1/ctaJxQZBKT5ZPt0m9STJEadao0xAH0ahmbWn
|
|
||||||
OlFuhjuefXKnEgV4We0+UXgVCwOPjdAvBbI+e0ocS3MFEvzG6uBQE3xDk3SzynTn
|
|
||||||
jh8BCNAw1FtxNrQHusEwMFxIt4I7mKZ9YIqioymCzLq9gwQbooMDQaHWBfEbwrbw
|
|
||||||
qHyGO0aoSCqI3Haadr8faqU9GY/rOPNk3sgrDQoo//fb4hVC1CLQJ13hef4Y53CI
|
|
||||||
rU7m2Ys6xt0nUW7/vGT1M0NPAgMBAAGjQjBAMA4GA1UdDwEB/wQEAwIBBjAPBgNV
|
|
||||||
HRMBAf8EBTADAQH/MB0GA1UdDgQWBBR5tFnme7bl5AFzgAiIyBpY9umbbjANBgkq
|
|
||||||
hkiG9w0BAQsFAAOCAgEAVR9YqbyyqFDQDLHYGmkgJykIrGF1XIpu+ILlaS/V9lZL
|
|
||||||
ubhzEFnTIZd+50xx+7LSYK05qAvqFyFWhfFQDlnrzuBZ6brJFe+GnY+EgPbk6ZGQ
|
|
||||||
3BebYhtF8GaV0nxvwuo77x/Py9auJ/GpsMiu/X1+mvoiBOv/2X/qkSsisRcOj/KK
|
|
||||||
NFtY2PwByVS5uCbMiogziUwthDyC3+6WVwW6LLv3xLfHTjuCvjHIInNzktHCgKQ5
|
|
||||||
ORAzI4JMPJ+GslWYHb4phowim57iaztXOoJwTdwJx4nLCgdNbOhdjsnvzqvHu7Ur
|
|
||||||
TkXWStAmzOVyyghqpZXjFaH3pO3JLF+l+/+sKAIuvtd7u+Nxe5AW0wdeRlN8NwdC
|
|
||||||
jNPElpzVmbUq4JUagEiuTDkHzsxHpFKVK7q4+63SM1N95R1NbdWhscdCb+ZAJzVc
|
|
||||||
oyi3B43njTOQ5yOf+1CceWxG1bQVs5ZufpsMljq4Ui0/1lvh+wjChP4kqKOJ2qxq
|
|
||||||
4RgqsahDYVvTH9w7jXbyLeiNdd8XM2w9U/t7y0Ff/9yi0GE44Za4rF2LN9d11TPA
|
|
||||||
mRGunUHBcnWEvgJBQl9nJEiU0Zsnvgc/ubhPgXRR4Xq37Z0j4r7g1SgEEzwxA57d
|
|
||||||
emyPxgcYxn/eR44/KJ4EBs+lVDR3veyJm+kXQ99b21/+jh5Xos1AnX5iItreGCc=
|
|
||||||
-----END CERTIFICATE-----
|
|
||||||
)EOF";
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef TEST_SCREENS
|
|
||||||
uint8_t input_buffer[3 * input_buffer_pixels]; // up to depth 24
|
|
||||||
uint8_t output_row_mono_buffer[max_row_width / 8]; // buffer for at least one row of b/w bits
|
|
||||||
uint8_t output_row_color_buffer[max_row_width / 8]; // buffer for at least one row of color bits
|
|
||||||
uint8_t mono_palette_buffer[max_palette_pixels / 8]; // palette buffer for depth <= 8 b/w
|
|
||||||
uint8_t color_palette_buffer[max_palette_pixels / 8]; // palette buffer for depth <= 8 c/w
|
|
||||||
uint16_t rgb_palette_buffer[max_palette_pixels]; // palette buffer for depth <= 8 for buffered graphics, needed for 7-color display
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Function to calculate SHA-256 hash
|
|
||||||
String calculateSHA256(uint8_t *data, size_t len)
|
|
||||||
{
|
|
||||||
byte shaResult[32];
|
|
||||||
mbedtls_md_context_t ctx;
|
|
||||||
mbedtls_md_type_t md_type = MBEDTLS_MD_SHA256;
|
|
||||||
|
|
||||||
mbedtls_md_init(&ctx);
|
|
||||||
mbedtls_md_setup(&ctx, mbedtls_md_info_from_type(md_type), 0);
|
|
||||||
mbedtls_md_starts(&ctx);
|
|
||||||
mbedtls_md_update(&ctx, data, len);
|
|
||||||
mbedtls_md_finish(&ctx, shaResult);
|
|
||||||
mbedtls_md_free(&ctx);
|
|
||||||
|
|
||||||
char sha256_str[65];
|
|
||||||
for (int i = 0; i < 32; i++)
|
|
||||||
{
|
|
||||||
sprintf(sha256_str + (i * 2), "%02x", shaResult[i]);
|
|
||||||
}
|
|
||||||
sha256_str[64] = 0;
|
|
||||||
|
|
||||||
return String(sha256_str);
|
|
||||||
}
|
|
||||||
|
|
||||||
String calculateSHA256(WiFiClient *stream, size_t contentLength) {
|
|
||||||
mbedtls_md_context_t ctx;
|
|
||||||
mbedtls_md_type_t md_type = MBEDTLS_MD_SHA256;
|
|
||||||
|
|
||||||
mbedtls_md_init(&ctx);
|
|
||||||
mbedtls_md_setup(&ctx, mbedtls_md_info_from_type(md_type), 0);
|
|
||||||
mbedtls_md_starts(&ctx);
|
|
||||||
|
|
||||||
uint8_t buff[1024];
|
|
||||||
size_t bytesRead = 0;
|
|
||||||
|
|
||||||
while (bytesRead < contentLength) {
|
|
||||||
size_t toRead = min((size_t)(contentLength - bytesRead), sizeof(buff));
|
|
||||||
size_t readBytes = stream->readBytes(buff, toRead);
|
|
||||||
|
|
||||||
if (readBytes == 0) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
mbedtls_md_update(&ctx, buff, readBytes);
|
|
||||||
bytesRead += readBytes;
|
|
||||||
}
|
|
||||||
|
|
||||||
byte shaResult[32];
|
|
||||||
mbedtls_md_finish(&ctx, shaResult);
|
|
||||||
mbedtls_md_free(&ctx);
|
|
||||||
|
|
||||||
String result = "";
|
|
||||||
for (int i = 0; i < sizeof(shaResult); i++) {
|
|
||||||
char str[3];
|
|
||||||
sprintf(str, "%02x", (int)shaResult[i]);
|
|
||||||
result += str;
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
// uint8_t* getOceanIcon() {
|
|
||||||
// zlib_turbo zt;
|
|
||||||
// int iUncompSize = zt.gzip_info((uint8_t *)ocean_logo_comp, ocean_logo_size);
|
|
||||||
// uint8_t *pUncompressed;
|
|
||||||
// pUncompressed = (uint8_t *)malloc(iUncompSize+4);
|
|
||||||
// zt.gunzip((uint8_t *)ocean_logo_comp, ocean_logo_size, pUncompressed);
|
|
||||||
// }
|
|
|
@ -1,94 +1,47 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "MCP23017.h"
|
#include <Adafruit_MCP23X17.h>
|
||||||
// #include <zlib_turbo.h>
|
|
||||||
#include <ArduinoJson.h>
|
#include <ArduinoJson.h>
|
||||||
#include <WiFiClientSecure.h>
|
|
||||||
#include <Preferences.h>
|
#include <Preferences.h>
|
||||||
#include <freertos/FreeRTOS.h>
|
#include <freertos/FreeRTOS.h>
|
||||||
#include <freertos/task.h>
|
#include <freertos/task.h>
|
||||||
#include <GxEPD2.h>
|
|
||||||
#include <GxEPD2_BW.h>
|
|
||||||
#include <mbedtls/md.h>
|
|
||||||
#include "esp_crt_bundle.h"
|
|
||||||
#include <Update.h>
|
|
||||||
|
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
#include <utils.hpp>
|
#include <utils.hpp>
|
||||||
|
|
||||||
#include "defaults.hpp"
|
extern Adafruit_MCP23X17 mcp1;
|
||||||
|
#ifdef IS_BTCLOCK_S3
|
||||||
extern MCP23017 mcp1;
|
extern Adafruit_MCP23X17 mcp2;
|
||||||
#ifdef IS_BTCLOCK_V8
|
|
||||||
extern MCP23017 mcp2;
|
|
||||||
#endif
|
#endif
|
||||||
extern Preferences preferences;
|
extern Preferences preferences;
|
||||||
extern std::mutex mcpMutex;
|
extern std::mutex mcpMutex;
|
||||||
|
|
||||||
#ifdef VERSION_EPD_2_13
|
|
||||||
#define EPD_CLASS GxEPD2_213_B74
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef VERSION_EPD_2_9
|
|
||||||
#define EPD_CLASS GxEPD2_290_T94
|
|
||||||
#endif
|
|
||||||
|
|
||||||
const PROGMEM int SCREEN_BLOCK_HEIGHT = 0;
|
const PROGMEM int SCREEN_BLOCK_HEIGHT = 0;
|
||||||
|
const PROGMEM int SCREEN_MSCW_TIME = 1;
|
||||||
|
const PROGMEM int SCREEN_BTC_TICKER = 2;
|
||||||
const PROGMEM int SCREEN_TIME = 3;
|
const PROGMEM int SCREEN_TIME = 3;
|
||||||
const PROGMEM int SCREEN_HALVING_COUNTDOWN = 4;
|
const PROGMEM int SCREEN_HALVING_COUNTDOWN = 4;
|
||||||
const PROGMEM int SCREEN_BLOCK_FEE_RATE = 6;
|
const PROGMEM int SCREEN_MARKET_CAP = 5;
|
||||||
|
|
||||||
const PROGMEM int SCREEN_SATS_PER_CURRENCY = 10;
|
|
||||||
|
|
||||||
const PROGMEM int SCREEN_BTC_TICKER = 20;
|
|
||||||
// const PROGMEM int SCREEN_BTC_TICKER_USD = 20;
|
|
||||||
// const PROGMEM int SCREEN_BTC_TICKER_EUR = 21;
|
|
||||||
// const PROGMEM int SCREEN_BTC_TICKER_GBP = 22;
|
|
||||||
// const PROGMEM int SCREEN_BTC_TICKER_JPY = 23;
|
|
||||||
// const PROGMEM int SCREEN_BTC_TICKER_AUD = 24;
|
|
||||||
// const PROGMEM int SCREEN_BTC_TICKER_CAD = 25;
|
|
||||||
|
|
||||||
const PROGMEM int SCREEN_MARKET_CAP = 30;
|
|
||||||
// const PROGMEM int SCREEN_MARKET_CAP_USD = 30;
|
|
||||||
// const PROGMEM int SCREEN_MARKET_CAP_EUR = 31;
|
|
||||||
// const PROGMEM int SCREEN_MARKET_CAP_GBP = 32;
|
|
||||||
// const PROGMEM int SCREEN_MARKET_CAP_JPY = 33;
|
|
||||||
// const PROGMEM int SCREEN_MARKET_CAP_AUD = 34;
|
|
||||||
// const PROGMEM int SCREEN_MARKET_CAP_CAD = 35;
|
|
||||||
|
|
||||||
const PROGMEM int SCREEN_BITAXE_HASHRATE = 80;
|
|
||||||
const PROGMEM int SCREEN_BITAXE_BESTDIFF = 81;
|
|
||||||
|
|
||||||
const PROGMEM int SCREEN_COUNTDOWN = 98;
|
const PROGMEM int SCREEN_COUNTDOWN = 98;
|
||||||
const PROGMEM int SCREEN_CUSTOM = 99;
|
const PROGMEM int SCREEN_CUSTOM = 99;
|
||||||
const int SCREEN_COUNT = 7;
|
const int SCREEN_COUNT = 6;
|
||||||
const PROGMEM int screens[SCREEN_COUNT] = {
|
const PROGMEM int screens[SCREEN_COUNT] = {
|
||||||
SCREEN_BLOCK_HEIGHT, SCREEN_SATS_PER_CURRENCY, SCREEN_BTC_TICKER,
|
SCREEN_BLOCK_HEIGHT, SCREEN_MSCW_TIME, SCREEN_BTC_TICKER,
|
||||||
SCREEN_TIME, SCREEN_HALVING_COUNTDOWN, SCREEN_MARKET_CAP,
|
SCREEN_TIME, SCREEN_HALVING_COUNTDOWN, SCREEN_MARKET_CAP};
|
||||||
SCREEN_BLOCK_FEE_RATE};
|
|
||||||
const int usPerSecond = 1000000;
|
const int usPerSecond = 1000000;
|
||||||
const int usPerMinute = 60 * usPerSecond;
|
const int usPerMinute = 60 * usPerSecond;
|
||||||
|
|
||||||
// extern const char *github_root_ca;
|
struct SpiRamAllocator {
|
||||||
extern const char *isrg_root_x1cert;
|
void *allocate(size_t size) {
|
||||||
|
return heap_caps_malloc(size, MALLOC_CAP_SPIRAM);
|
||||||
|
}
|
||||||
|
|
||||||
extern const uint8_t rootca_crt_bundle_start[] asm("_binary_x509_crt_bundle_start");
|
void deallocate(void *pointer) { heap_caps_free(pointer); }
|
||||||
// extern const uint8_t ocean_logo_comp[] asm("_binary_ocean_gz_start");
|
|
||||||
// extern const uint8_t ocean_logo_comp_end[] asm("_binary_ocean_gz_end");
|
|
||||||
|
|
||||||
// uint8_t* getOceanIcon();
|
void *reallocate(void *ptr, size_t new_size) {
|
||||||
|
return heap_caps_realloc(ptr, new_size, MALLOC_CAP_SPIRAM);
|
||||||
// const size_t ocean_logo_size = ocean_logo_comp_end - ocean_logo_comp;
|
}
|
||||||
|
|
||||||
const PROGMEM char UPDATE_FIRMWARE = U_FLASH;
|
|
||||||
const PROGMEM char UPDATE_WEBUI = U_SPIFFS;
|
|
||||||
const PROGMEM char UPDATE_ALL = 99;
|
|
||||||
|
|
||||||
struct ScreenMapping {
|
|
||||||
int value;
|
|
||||||
const char* name;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
String calculateSHA256(uint8_t* data, size_t len);
|
using SpiRamJsonDocument = BasicJsonDocument<SpiRamAllocator>;
|
||||||
String calculateSHA256(WiFiClient *stream, size_t contentLength);
|
|
||||||
|
|
|
@ -1,86 +0,0 @@
|
||||||
#include "timers.hpp"
|
|
||||||
|
|
||||||
esp_timer_handle_t screenRotateTimer;
|
|
||||||
esp_timer_handle_t minuteTimer;
|
|
||||||
|
|
||||||
void setupTimeUpdateTimer(void *pvParameters) {
|
|
||||||
const esp_timer_create_args_t minuteTimerConfig = {
|
|
||||||
.callback = &minuteTimerISR, .name = "minute_timer"};
|
|
||||||
|
|
||||||
esp_timer_create(&minuteTimerConfig, &minuteTimer);
|
|
||||||
|
|
||||||
time_t currentTime;
|
|
||||||
struct tm timeinfo;
|
|
||||||
time(¤tTime);
|
|
||||||
localtime_r(¤tTime, &timeinfo);
|
|
||||||
uint32_t secondsUntilNextMinute = 60 - timeinfo.tm_sec;
|
|
||||||
|
|
||||||
if (secondsUntilNextMinute > 0)
|
|
||||||
vTaskDelay(pdMS_TO_TICKS((secondsUntilNextMinute * 1000)));
|
|
||||||
|
|
||||||
esp_timer_start_periodic(minuteTimer, usPerMinute);
|
|
||||||
|
|
||||||
WorkItem timeUpdate = {TASK_TIME_UPDATE, 0};
|
|
||||||
xQueueSend(workQueue, &timeUpdate, portMAX_DELAY);
|
|
||||||
// xTaskNotifyGive(timeUpdateTaskHandle);
|
|
||||||
|
|
||||||
vTaskDelete(NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setupScreenRotateTimer(void *pvParameters) {
|
|
||||||
const esp_timer_create_args_t screenRotateTimerConfig = {
|
|
||||||
.callback = &screenRotateTimerISR, .name = "screen_rotate_timer"};
|
|
||||||
|
|
||||||
esp_timer_create(&screenRotateTimerConfig, &screenRotateTimer);
|
|
||||||
|
|
||||||
if (preferences.getBool("timerActive", DEFAULT_TIMER_ACTIVE)) {
|
|
||||||
esp_timer_start_periodic(screenRotateTimer,
|
|
||||||
getTimerSeconds() * usPerSecond);
|
|
||||||
}
|
|
||||||
|
|
||||||
vTaskDelete(NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
uint getTimerSeconds() { return preferences.getUInt("timerSeconds", DEFAULT_TIMER_SECONDS); }
|
|
||||||
|
|
||||||
bool isTimerActive() { return esp_timer_is_active(screenRotateTimer); }
|
|
||||||
|
|
||||||
void setTimerActive(bool status) {
|
|
||||||
if (status) {
|
|
||||||
esp_timer_start_periodic(screenRotateTimer,
|
|
||||||
getTimerSeconds() * usPerSecond);
|
|
||||||
queueLedEffect(LED_EFFECT_START_TIMER);
|
|
||||||
preferences.putBool("timerActive", true);
|
|
||||||
} else {
|
|
||||||
esp_timer_stop(screenRotateTimer);
|
|
||||||
queueLedEffect(LED_EFFECT_PAUSE_TIMER);
|
|
||||||
preferences.putBool("timerActive", false);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (eventSourceTaskHandle != NULL) xTaskNotifyGive(eventSourceTaskHandle);
|
|
||||||
}
|
|
||||||
|
|
||||||
void toggleTimerActive() { setTimerActive(!isTimerActive()); }
|
|
||||||
|
|
||||||
void IRAM_ATTR minuteTimerISR(void *arg) {
|
|
||||||
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
|
||||||
// vTaskNotifyGiveFromISR(timeUpdateTaskHandle, &xHigherPriorityTaskWoken);
|
|
||||||
WorkItem timeUpdate = {TASK_TIME_UPDATE, 0};
|
|
||||||
xQueueSendFromISR(workQueue, &timeUpdate, &xHigherPriorityTaskWoken);
|
|
||||||
|
|
||||||
if (bitaxeFetchTaskHandle != NULL) {
|
|
||||||
vTaskNotifyGiveFromISR(bitaxeFetchTaskHandle, &xHigherPriorityTaskWoken);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (xHigherPriorityTaskWoken == pdTRUE) {
|
|
||||||
portYIELD_FROM_ISR();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void IRAM_ATTR screenRotateTimerISR(void *arg) {
|
|
||||||
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
|
||||||
vTaskNotifyGiveFromISR(taskScreenRotateTaskHandle, &xHigherPriorityTaskWoken);
|
|
||||||
if (xHigherPriorityTaskWoken == pdTRUE) {
|
|
||||||
portYIELD_FROM_ISR();
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,22 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <esp_timer.h>
|
|
||||||
#include <freertos/FreeRTOS.h>
|
|
||||||
#include <freertos/task.h>
|
|
||||||
|
|
||||||
#include "lib/shared.hpp"
|
|
||||||
#include "lib/screen_handler.hpp"
|
|
||||||
|
|
||||||
extern esp_timer_handle_t screenRotateTimer;
|
|
||||||
extern esp_timer_handle_t minuteTimer;
|
|
||||||
|
|
||||||
void setupTimeUpdateTimer(void *pvParameters);
|
|
||||||
void setupScreenRotateTimer(void *pvParameters);
|
|
||||||
|
|
||||||
void IRAM_ATTR minuteTimerISR(void *arg);
|
|
||||||
void IRAM_ATTR screenRotateTimerISR(void *arg);
|
|
||||||
|
|
||||||
uint getTimerSeconds();
|
|
||||||
bool isTimerActive();
|
|
||||||
void setTimerActive(bool status);
|
|
||||||
void toggleTimerActive();
|
|
|
@ -1,160 +0,0 @@
|
||||||
#include "v2_notify.hpp"
|
|
||||||
|
|
||||||
using namespace V2Notify;
|
|
||||||
|
|
||||||
namespace V2Notify
|
|
||||||
{
|
|
||||||
WebSocketsClient webSocket;
|
|
||||||
|
|
||||||
TaskHandle_t v2NotifyTaskHandle;
|
|
||||||
|
|
||||||
void setupV2Notify()
|
|
||||||
{
|
|
||||||
String hostname = "ws.btclock.dev";
|
|
||||||
if (preferences.getBool("stagingSource", DEFAULT_STAGING_SOURCE))
|
|
||||||
{
|
|
||||||
Serial.println(F("Connecting to V2 staging source"));
|
|
||||||
hostname = "ws-staging.btclock.dev";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Serial.println(F("Connecting to V2 source"));
|
|
||||||
}
|
|
||||||
|
|
||||||
webSocket.beginSSL(hostname, 443, "/api/v2/ws");
|
|
||||||
webSocket.onEvent(V2Notify::onWebsocketV2Event);
|
|
||||||
webSocket.setReconnectInterval(5000);
|
|
||||||
webSocket.enableHeartbeat(15000, 3000, 2);
|
|
||||||
|
|
||||||
V2Notify::setupV2NotifyTask();
|
|
||||||
}
|
|
||||||
|
|
||||||
void onWebsocketV2Event(WStype_t type, uint8_t *payload, size_t length)
|
|
||||||
{
|
|
||||||
switch (type)
|
|
||||||
{
|
|
||||||
case WStype_DISCONNECTED:
|
|
||||||
Serial.printf("[WSc] Disconnected!\n");
|
|
||||||
break;
|
|
||||||
case WStype_CONNECTED:
|
|
||||||
{
|
|
||||||
Serial.printf("[WSc] Connected to url: %s\n", payload);
|
|
||||||
|
|
||||||
JsonDocument response;
|
|
||||||
|
|
||||||
response["type"] = "subscribe";
|
|
||||||
response["eventType"] = "blockfee";
|
|
||||||
size_t responseLength = measureMsgPack(response);
|
|
||||||
uint8_t *buffer = new uint8_t[responseLength];
|
|
||||||
serializeMsgPack(response, buffer, responseLength);
|
|
||||||
webSocket.sendBIN(buffer, responseLength);
|
|
||||||
delete[] buffer;
|
|
||||||
|
|
||||||
buffer = new uint8_t[responseLength];
|
|
||||||
|
|
||||||
response["type"] = "subscribe";
|
|
||||||
response["eventType"] = "blockheight";
|
|
||||||
responseLength = measureMsgPack(response);
|
|
||||||
buffer = new uint8_t[responseLength];
|
|
||||||
serializeMsgPack(response, buffer, responseLength);
|
|
||||||
webSocket.sendBIN(buffer, responseLength);
|
|
||||||
|
|
||||||
delete[] buffer;
|
|
||||||
|
|
||||||
buffer = new uint8_t[responseLength];
|
|
||||||
|
|
||||||
response["type"] = "subscribe";
|
|
||||||
response["eventType"] = "price";
|
|
||||||
|
|
||||||
JsonArray currenciesArray = response["currencies"].to<JsonArray>();
|
|
||||||
|
|
||||||
for (const auto &str : getActiveCurrencies())
|
|
||||||
{
|
|
||||||
currenciesArray.add(str);
|
|
||||||
}
|
|
||||||
|
|
||||||
// response["currencies"] = currenciesArray;
|
|
||||||
responseLength = measureMsgPack(response);
|
|
||||||
buffer = new uint8_t[responseLength];
|
|
||||||
serializeMsgPack(response, buffer, responseLength);
|
|
||||||
webSocket.sendBIN(buffer, responseLength);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case WStype_TEXT:
|
|
||||||
Serial.printf("[WSc] get text: %s\n", payload);
|
|
||||||
|
|
||||||
// send message to server
|
|
||||||
// webSocket.sendTXT("message here");
|
|
||||||
break;
|
|
||||||
case WStype_BIN:
|
|
||||||
{
|
|
||||||
JsonDocument doc;
|
|
||||||
DeserializationError error = deserializeMsgPack(doc, payload, length);
|
|
||||||
|
|
||||||
V2Notify::handleV2Message(doc);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case WStype_ERROR:
|
|
||||||
case WStype_FRAGMENT_TEXT_START:
|
|
||||||
case WStype_FRAGMENT_BIN_START:
|
|
||||||
case WStype_FRAGMENT:
|
|
||||||
case WStype_PING:
|
|
||||||
case WStype_PONG:
|
|
||||||
case WStype_FRAGMENT_FIN:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void handleV2Message(JsonDocument doc)
|
|
||||||
{
|
|
||||||
if (doc.containsKey("blockheight"))
|
|
||||||
{
|
|
||||||
uint newBlockHeight = doc["blockheight"].as<uint>();
|
|
||||||
|
|
||||||
if (newBlockHeight == getBlockHeight())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
processNewBlock(newBlockHeight);
|
|
||||||
}
|
|
||||||
else if (doc.containsKey("blockfee"))
|
|
||||||
{
|
|
||||||
uint medianFee = doc["blockfee"].as<uint>();
|
|
||||||
|
|
||||||
processNewBlockFee(medianFee);
|
|
||||||
}
|
|
||||||
else if (doc.containsKey("price"))
|
|
||||||
{
|
|
||||||
|
|
||||||
// Iterate through the key-value pairs of the "price" object
|
|
||||||
for (JsonPair kv : doc["price"].as<JsonObject>())
|
|
||||||
{
|
|
||||||
const char *currency = kv.key().c_str();
|
|
||||||
uint newPrice = kv.value().as<uint>();
|
|
||||||
|
|
||||||
processNewPrice(newPrice, getCurrencyChar(currency));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void taskV2Notify(void *pvParameters)
|
|
||||||
{
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
webSocket.loop();
|
|
||||||
vTaskDelay(10 / portTICK_PERIOD_MS);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setupV2NotifyTask()
|
|
||||||
{
|
|
||||||
xTaskCreate(V2Notify::taskV2Notify, "v2Notify", (6 * 1024), NULL, tskIDLE_PRIORITY,
|
|
||||||
&V2Notify::v2NotifyTaskHandle);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isV2NotifyConnected()
|
|
||||||
{
|
|
||||||
return webSocket.isConnected();
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,26 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <Arduino.h>
|
|
||||||
#include <ArduinoJson.h>
|
|
||||||
#include <esp_websocket_client.h>
|
|
||||||
#include "block_notify.hpp"
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
#include "lib/screen_handler.hpp"
|
|
||||||
|
|
||||||
namespace V2Notify {
|
|
||||||
extern TaskHandle_t v2NotifyTaskHandle;
|
|
||||||
|
|
||||||
void setupV2NotifyTask();
|
|
||||||
void taskV2Notify(void *pvParameters);
|
|
||||||
|
|
||||||
void setupV2Notify();
|
|
||||||
void onWebsocketV2Event(WStype_t type, uint8_t * payload, size_t length);
|
|
||||||
void handleV2Message(JsonDocument doc);
|
|
||||||
|
|
||||||
bool isV2NotifyConnected();
|
|
||||||
}
|
|
||||||
// void stopV2Notify();
|
|
||||||
// void restartV2Notify();
|
|
||||||
// bool getPriceNotifyInit();
|
|
||||||
// uint getLastPriceUpdate();
|
|
File diff suppressed because it is too large
Load diff
|
@ -21,26 +21,18 @@ void stopWebServer();
|
||||||
void setupWebserver();
|
void setupWebserver();
|
||||||
bool processEpdColorSettings(AsyncWebServerRequest *request);
|
bool processEpdColorSettings(AsyncWebServerRequest *request);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void onApiStatus(AsyncWebServerRequest *request);
|
void onApiStatus(AsyncWebServerRequest *request);
|
||||||
void onApiSystemStatus(AsyncWebServerRequest *request);
|
void onApiSystemStatus(AsyncWebServerRequest *request);
|
||||||
void onApiSetWifiTxPower(AsyncWebServerRequest *request);
|
void onApiSetWifiTxPower(AsyncWebServerRequest *request);
|
||||||
|
|
||||||
void onApiScreenNext(AsyncWebServerRequest *request);
|
|
||||||
void onApiScreenPrevious(AsyncWebServerRequest *request);
|
|
||||||
|
|
||||||
void onApiShowScreen(AsyncWebServerRequest *request);
|
void onApiShowScreen(AsyncWebServerRequest *request);
|
||||||
void onApiShowCurrency(AsyncWebServerRequest *request);
|
|
||||||
|
|
||||||
void onApiShowText(AsyncWebServerRequest *request);
|
void onApiShowText(AsyncWebServerRequest *request);
|
||||||
void onApiIdentify(AsyncWebServerRequest *request);
|
|
||||||
|
|
||||||
void onApiShowTextAdvanced(AsyncWebServerRequest *request, JsonVariant &json);
|
void onApiShowTextAdvanced(AsyncWebServerRequest *request, JsonVariant &json);
|
||||||
|
|
||||||
void onApiActionPause(AsyncWebServerRequest *request);
|
void onApiActionPause(AsyncWebServerRequest *request);
|
||||||
void onApiActionTimerRestart(AsyncWebServerRequest *request);
|
void onApiActionTimerRestart(AsyncWebServerRequest *request);
|
||||||
void onApiSettingsGet(AsyncWebServerRequest *request);
|
void onApiSettingsGet(AsyncWebServerRequest *request);
|
||||||
|
void onApiSettingsPost(AsyncWebServerRequest *request);
|
||||||
void onApiSettingsPatch(AsyncWebServerRequest *request, JsonVariant &json);
|
void onApiSettingsPatch(AsyncWebServerRequest *request, JsonVariant &json);
|
||||||
void onApiFullRefresh(AsyncWebServerRequest *request);
|
void onApiFullRefresh(AsyncWebServerRequest *request);
|
||||||
|
|
||||||
|
@ -50,28 +42,11 @@ void onApiLightsSetColor(AsyncWebServerRequest *request);
|
||||||
void onApiLightsSetJson(AsyncWebServerRequest *request, JsonVariant &json);
|
void onApiLightsSetJson(AsyncWebServerRequest *request, JsonVariant &json);
|
||||||
|
|
||||||
void onApiRestart(AsyncWebServerRequest *request);
|
void onApiRestart(AsyncWebServerRequest *request);
|
||||||
void onFirmwareUpdate(AsyncWebServerRequest *request);
|
|
||||||
void asyncFirmwareUpdateHandler(AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final);
|
|
||||||
void asyncFileUpdateHandler(AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final, int command);
|
|
||||||
void asyncWebuiUpdateHandler(AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final);
|
|
||||||
void onAutoUpdateFirmware(AsyncWebServerRequest *request);
|
|
||||||
|
|
||||||
void onIndex(AsyncWebServerRequest *request);
|
void onIndex(AsyncWebServerRequest *request);
|
||||||
void onNotFound(AsyncWebServerRequest *request);
|
void onNotFound(AsyncWebServerRequest *request);
|
||||||
|
|
||||||
JsonDocument getLedStatusObject();
|
StaticJsonDocument<512> getLedStatusObject();
|
||||||
JsonDocument getStatusObject();
|
StaticJsonDocument<768> getStatusObject();
|
||||||
void eventSourceUpdate();
|
void eventSourceUpdate();
|
||||||
void eventSourceTask(void *pvParameters);
|
void eventSourceTask(void *pvParameters);
|
||||||
|
|
||||||
void onApiStopDataSources(AsyncWebServerRequest *request);
|
|
||||||
void onApiRestartDataSources(AsyncWebServerRequest *request);
|
|
||||||
|
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
void onApiFrontlightOn(AsyncWebServerRequest *request);
|
|
||||||
void onApiFrontlightFlash(AsyncWebServerRequest *request);
|
|
||||||
void onApiFrontlightSetBrightness(AsyncWebServerRequest *request);
|
|
||||||
|
|
||||||
void onApiFrontlightStatus(AsyncWebServerRequest *request);
|
|
||||||
void onApiFrontlightOff(AsyncWebServerRequest *request);
|
|
||||||
#endif
|
|
145
src/main.cpp
145
src/main.cpp
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2023-2024 Djuri Baars
|
* Copyright 2023 Djuri Baars
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -20,19 +20,14 @@
|
||||||
#include "lib/config.hpp"
|
#include "lib/config.hpp"
|
||||||
|
|
||||||
uint wifiLostConnection;
|
uint wifiLostConnection;
|
||||||
uint priceNotifyLostConnection = 0;
|
|
||||||
uint blockNotifyLostConnection = 0;
|
|
||||||
// char ptrTaskList[1500];
|
|
||||||
|
|
||||||
extern "C" void app_main()
|
extern "C" void app_main() {
|
||||||
{
|
|
||||||
initArduino();
|
initArduino();
|
||||||
|
|
||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
setup();
|
setup();
|
||||||
|
|
||||||
while (true)
|
while (true) {
|
||||||
{
|
|
||||||
// vTaskList(ptrTaskList);
|
// vTaskList(ptrTaskList);
|
||||||
// Serial.println(F("**********************************"));
|
// Serial.println(F("**********************************"));
|
||||||
// Serial.println(F("Task State Prio Stack Num"));
|
// Serial.println(F("Task State Prio Stack Num"));
|
||||||
|
@ -42,132 +37,22 @@ extern "C" void app_main()
|
||||||
if (eventSourceTaskHandle != NULL)
|
if (eventSourceTaskHandle != NULL)
|
||||||
xTaskNotifyGive(eventSourceTaskHandle);
|
xTaskNotifyGive(eventSourceTaskHandle);
|
||||||
|
|
||||||
int64_t currentUptime = esp_timer_get_time() / 1000000;
|
if (!WiFi.isConnected()) {
|
||||||
;
|
if (!wifiLostConnection) {
|
||||||
|
wifiLostConnection = esp_timer_get_time() / 1000000;
|
||||||
if (!getIsOTAUpdating())
|
Serial.println("Lost WiFi connection, trying to reconnect...");
|
||||||
{
|
|
||||||
#ifdef HAS_FRONTLIGHT
|
|
||||||
if (hasLightLevel()) {
|
|
||||||
if (preferences.getUInt("luxLightToggle", DEFAULT_LUX_LIGHT_TOGGLE) != 0)
|
|
||||||
{
|
|
||||||
if (hasLightLevel() && getLightLevel() <= 1 && preferences.getBool("flOffWhenDark", DEFAULT_FL_OFF_WHEN_DARK))
|
|
||||||
{
|
|
||||||
if (frontlightIsOn()) {
|
|
||||||
frontlightFadeOutAll();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (hasLightLevel() && getLightLevel() < preferences.getUInt("luxLightToggle", DEFAULT_LUX_LIGHT_TOGGLE) && !frontlightIsOn())
|
|
||||||
{
|
|
||||||
frontlightFadeInAll();
|
|
||||||
}
|
|
||||||
else if (frontlightIsOn() && getLightLevel() > preferences.getUInt("luxLightToggle", DEFAULT_LUX_LIGHT_TOGGLE))
|
|
||||||
{
|
|
||||||
frontlightFadeOutAll();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (!WiFi.isConnected())
|
|
||||||
{
|
|
||||||
if (!wifiLostConnection)
|
|
||||||
{
|
|
||||||
wifiLostConnection = currentUptime;
|
|
||||||
Serial.println(F("Lost WiFi connection, trying to reconnect..."));
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((currentUptime - wifiLostConnection) > 600)
|
|
||||||
{
|
|
||||||
Serial.println(F("Still no connection after 10 minutes, restarting..."));
|
|
||||||
delay(2000);
|
|
||||||
ESP.restart();
|
|
||||||
}
|
|
||||||
|
|
||||||
WiFi.begin();
|
|
||||||
}
|
|
||||||
else if (wifiLostConnection)
|
|
||||||
{
|
|
||||||
wifiLostConnection = 0;
|
|
||||||
Serial.println(F("Connection restored, reset timer."));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (getPriceNotifyInit() && !preferences.getBool("fetchEurPrice", DEFAULT_FETCH_EUR_PRICE) && !isPriceNotifyConnected())
|
if (((esp_timer_get_time() / 1000000) - wifiLostConnection) > 600) {
|
||||||
{
|
Serial.println("Still no connection after 10 minutes, restarting...");
|
||||||
priceNotifyLostConnection++;
|
delay(2000);
|
||||||
Serial.println(F("Lost price data connection..."));
|
ESP.restart();
|
||||||
queueLedEffect(LED_DATA_PRICE_ERROR);
|
|
||||||
|
|
||||||
// if price WS connection does not come back after 6*5 seconds, destroy and recreate
|
|
||||||
if (priceNotifyLostConnection > 6)
|
|
||||||
{
|
|
||||||
Serial.println(F("Restarting price handler..."));
|
|
||||||
|
|
||||||
restartPriceNotify();
|
|
||||||
// setupPriceNotify();
|
|
||||||
priceNotifyLostConnection = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (priceNotifyLostConnection > 0 && isPriceNotifyConnected())
|
|
||||||
{
|
|
||||||
priceNotifyLostConnection = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (getBlockNotifyInit() && !isBlockNotifyConnected())
|
WiFi.begin();
|
||||||
{
|
} else if (wifiLostConnection) {
|
||||||
blockNotifyLostConnection++;
|
wifiLostConnection = 0;
|
||||||
Serial.println(F("Lost block data connection..."));
|
Serial.println("Connection restored, reset timer.");
|
||||||
queueLedEffect(LED_DATA_BLOCK_ERROR);
|
|
||||||
// if mempool WS connection does not come back after 6*5 seconds, destroy and recreate
|
|
||||||
if (blockNotifyLostConnection > 6)
|
|
||||||
{
|
|
||||||
Serial.println(F("Restarting block handler..."));
|
|
||||||
|
|
||||||
restartBlockNotify();
|
|
||||||
// setupBlockNotify();
|
|
||||||
blockNotifyLostConnection = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (blockNotifyLostConnection > 0 && isBlockNotifyConnected())
|
|
||||||
{
|
|
||||||
blockNotifyLostConnection = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// if more than 5 price updates are missed, there is probably something wrong, reconnect
|
|
||||||
if ((getLastPriceUpdate(CURRENCY_USD) - currentUptime) > (preferences.getUInt("minSecPriceUpd", DEFAULT_SECONDS_BETWEEN_PRICE_UPDATE) * 5))
|
|
||||||
{
|
|
||||||
Serial.println(F("Detected 5 missed price updates... restarting price handler."));
|
|
||||||
|
|
||||||
restartPriceNotify();
|
|
||||||
// setupPriceNotify();
|
|
||||||
|
|
||||||
priceNotifyLostConnection = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If after 45 minutes no mempool blocks, check the rest API
|
|
||||||
if ((getLastBlockUpdate() - currentUptime) > 45 * 60)
|
|
||||||
{
|
|
||||||
Serial.println(F("Long time (45 min) since last block, checking if I missed anything..."));
|
|
||||||
int currentBlock = getBlockFetch();
|
|
||||||
if (currentBlock != -1)
|
|
||||||
{
|
|
||||||
if (currentBlock != getBlockHeight())
|
|
||||||
{
|
|
||||||
Serial.println(F("Detected stuck block height... restarting block handler."));
|
|
||||||
// Mempool source stuck, restart
|
|
||||||
restartBlockNotify();
|
|
||||||
// setupBlockNotify();
|
|
||||||
}
|
|
||||||
// set last block update so it doesn't fetch for 45 minutes
|
|
||||||
setLastBlockUpdate(currentUptime);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (currentUptime - getLastTimeSync() > 24 * 60 * 60)
|
|
||||||
{
|
|
||||||
Serial.println(F("Last time update is longer than 24 hours ago, sync again"));
|
|
||||||
syncTime();
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
vTaskDelay(pdMS_TO_TICKS(5000));
|
vTaskDelay(pdMS_TO_TICKS(5000));
|
||||||
|
|
|
@ -1,310 +1,68 @@
|
||||||
#include <data_handler.hpp>
|
#include <data_handler.hpp>
|
||||||
#include <unity.h>
|
#include <unity.h>
|
||||||
|
|
||||||
template<size_t N>
|
void setUp(void) {
|
||||||
std::string joinArrayWithBrackets(const std::array<std::string, N>& arr, const std::string& separator = " ") {
|
|
||||||
std::ostringstream result;
|
|
||||||
for (size_t i = 0; i < N; ++i) {
|
|
||||||
if (i > 0) {
|
|
||||||
result << separator;
|
|
||||||
}
|
|
||||||
result << '[' << arr[i] << ']';
|
|
||||||
}
|
|
||||||
return result.str();
|
|
||||||
}
|
|
||||||
|
|
||||||
void setUp(void)
|
|
||||||
{
|
|
||||||
// set stuff up here
|
// set stuff up here
|
||||||
}
|
}
|
||||||
|
|
||||||
void tearDown(void)
|
void tearDown(void) {
|
||||||
{
|
|
||||||
// clean stuff up here
|
// clean stuff up here
|
||||||
}
|
}
|
||||||
|
|
||||||
void test_CorrectSatsPerDollarConversion(void)
|
void test_CorrectSatsPerDollarConversion(void) {
|
||||||
{
|
std::array<std::string, NUM_SCREENS> output = parseSatsPerCurrency(37253, '$');
|
||||||
std::array<std::string, NUM_SCREENS> output = parseSatsPerCurrency(37253, CURRENCY_USD, false);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("MSCW/TIME", output[0].c_str());
|
TEST_ASSERT_EQUAL_STRING("MSCW/TIME", output[0].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("2", output[NUM_SCREENS - 4].c_str());
|
TEST_ASSERT_EQUAL_STRING("2", output[NUM_SCREENS-4].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("6", output[NUM_SCREENS - 3].c_str());
|
TEST_ASSERT_EQUAL_STRING("6", output[NUM_SCREENS-3].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("8", output[NUM_SCREENS - 2].c_str());
|
TEST_ASSERT_EQUAL_STRING("8", output[NUM_SCREENS-2].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("4", output[NUM_SCREENS - 1].c_str());
|
TEST_ASSERT_EQUAL_STRING("4", output[NUM_SCREENS-1].c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
void test_SatsPerDollarAfter1B(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseSatsPerCurrency(120000000, CURRENCY_USD, false);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("SATS/USD", output[0].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[NUM_SCREENS - 5].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING(".", output[NUM_SCREENS - 4].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("8", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("3", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("3", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_CorrectSatsPerPoundConversion(void)
|
void test_SixCharacterBlockHeight(void) {
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseSatsPerCurrency(37253, CURRENCY_GBP, false);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("SATS/GBP", output[0].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("2", output[NUM_SCREENS - 4].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("6", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("8", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("4", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_SixCharacterBlockHeight(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseBlockHeight(999999);
|
std::array<std::string, NUM_SCREENS> output = parseBlockHeight(999999);
|
||||||
TEST_ASSERT_EQUAL_STRING("BLOCK/HEIGHT", output[0].c_str());
|
TEST_ASSERT_EQUAL_STRING("BLOCK/HEIGHT", output[0].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("9", output[1].c_str());
|
TEST_ASSERT_EQUAL_STRING("9", output[1].c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
void test_SevenCharacterBlockHeight(void)
|
void test_SevenCharacterBlockHeight(void) {
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseBlockHeight(1000000);
|
std::array<std::string, NUM_SCREENS> output = parseBlockHeight(1000000);
|
||||||
TEST_ASSERT_EQUAL_STRING("1", output[0].c_str());
|
TEST_ASSERT_EQUAL_STRING("1", output[0].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[1].c_str());
|
TEST_ASSERT_EQUAL_STRING("0", output[1].c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
void test_FeeRateDisplay(void)
|
void test_PriceOf100kusd(void) {
|
||||||
{
|
|
||||||
uint testValue = 21;
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseBlockFees(static_cast<std::uint16_t>(testValue));
|
|
||||||
TEST_ASSERT_EQUAL_STRING("FEE/RATE", output[0].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("2", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("1", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("sat/vB", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_PriceOf100kusd(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parsePriceData(100000, '$');
|
std::array<std::string, NUM_SCREENS> output = parsePriceData(100000, '$');
|
||||||
TEST_ASSERT_EQUAL_STRING("$", output[0].c_str());
|
TEST_ASSERT_EQUAL_STRING("$", output[0].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("1", output[1].c_str());
|
TEST_ASSERT_EQUAL_STRING("1", output[1].c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
void test_PriceOf1MillionUsd(void)
|
void test_PriceOf1MillionUsd(void) {
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parsePriceData(1000000, '$');
|
std::array<std::string, NUM_SCREENS> output = parsePriceData(1000000, '$');
|
||||||
TEST_ASSERT_EQUAL_STRING("BTC/USD", output[0].c_str());
|
TEST_ASSERT_EQUAL_STRING("BTC/USD", output[0].c_str());
|
||||||
|
for (int i = 1; i <= NUM_SCREENS-3; i++) {
|
||||||
TEST_ASSERT_EQUAL_STRING("1", output[NUM_SCREENS - 5].c_str());
|
TEST_ASSERT_EQUAL_STRING(" ", output[i].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING(".", output[NUM_SCREENS - 4].c_str());
|
}
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[NUM_SCREENS - 3].c_str());
|
TEST_ASSERT_EQUAL_STRING("1", output[NUM_SCREENS-2].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[NUM_SCREENS - 2].c_str());
|
TEST_ASSERT_EQUAL_STRING("M", output[NUM_SCREENS-1].c_str());
|
||||||
TEST_ASSERT_EQUAL_STRING("M", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_PriceSuffixMode(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parsePriceData(93000, '$', true, false);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("BTC/USD", output[0].c_str());
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING("9", output[NUM_SCREENS - 5].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("3", output[NUM_SCREENS - 4].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING(".", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("K", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_PriceSuffixModeCompact1(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parsePriceData(100000, '$', true, false, true);
|
|
||||||
|
|
||||||
std::string joined = joinArrayWithBrackets(output);
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("BTC/USD", output[0].c_str(), joined.c_str());
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("$", output[NUM_SCREENS - 6].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("1", output[NUM_SCREENS - 5].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0", output[NUM_SCREENS - 4].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0.", output[NUM_SCREENS - 3].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0", output[NUM_SCREENS - 2].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("K", output[NUM_SCREENS - 1].c_str(), joined.c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_PriceSuffixModeCompact2(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parsePriceData(1000000, '$', true, false, true);
|
|
||||||
|
|
||||||
std::string joined = joinArrayWithBrackets(output);
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("BTC/USD", output[0].c_str(), joined.c_str());
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("$", output[NUM_SCREENS - 6].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("1.", output[NUM_SCREENS - 5].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0", output[NUM_SCREENS - 4].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0", output[NUM_SCREENS - 3].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0", output[NUM_SCREENS - 2].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("M", output[NUM_SCREENS - 1].c_str(), joined.c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_PriceSuffixModeMow(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parsePriceData(93600, '$', true, true);
|
|
||||||
|
|
||||||
std::string joined = joinArrayWithBrackets(output);
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("$", output[0].c_str(), joined.c_str());
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE(".", output[NUM_SCREENS - 5].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0", output[NUM_SCREENS - 4].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("9", output[NUM_SCREENS - 3].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("3", output[NUM_SCREENS - 2].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("M", output[NUM_SCREENS - 1].c_str(), joined.c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_PriceSuffixModeMowCompact(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parsePriceData(93600, '$', true, true, true);
|
|
||||||
|
|
||||||
std::string joined = joinArrayWithBrackets(output);
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("MOW/UNITS", output[0].c_str(), joined.c_str());
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("$", output[NUM_SCREENS - 6].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0.", output[NUM_SCREENS - 5].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("0", output[NUM_SCREENS - 4].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("9", output[NUM_SCREENS - 3].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("3", output[NUM_SCREENS - 2].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("M", output[NUM_SCREENS - 1].c_str(), joined.c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_McapLowerUsd(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseMarketCap(810000, 26000, '$', true);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("USD/MCAP", output[0].c_str());
|
|
||||||
|
|
||||||
// TEST_ASSERT_EQUAL_STRING("$", output[NUM_SCREENS-6].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("$", output[NUM_SCREENS - 5].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("5", output[NUM_SCREENS - 4].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("7", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("B", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_Mcap1TrillionUsd(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseMarketCap(831000, 52000, '$', true);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("USD/MCAP", output[0].c_str());
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING("$", output[NUM_SCREENS - 6].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("1", output[NUM_SCREENS - 5].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING(".", output[NUM_SCREENS - 4].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("2", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("T", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_Mcap1TrillionUsdSmallChars(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseMarketCap(831000, 52000, '$', false);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("USD/MCAP", output[0].c_str());
|
|
||||||
|
|
||||||
std::string joined = joinArrayWithBrackets(output);
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE(" $ ", output[NUM_SCREENS - 6].c_str(), joined.c_str());
|
|
||||||
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE(" 1", output[NUM_SCREENS - 5].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("020", output[NUM_SCREENS - 4].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("825", output[NUM_SCREENS - 3].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("000", output[NUM_SCREENS - 2].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("000", output[NUM_SCREENS - 1].c_str(), joined.c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_Mcap1TrillionEur(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseMarketCap(831000, 52000, CURRENCY_EUR, true);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("EUR/MCAP", output[0].c_str());
|
|
||||||
TEST_ASSERT_TRUE(CURRENCY_EUR == output[NUM_SCREENS - 6].c_str()[0]);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("1", output[NUM_SCREENS - 5].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING(".", output[NUM_SCREENS - 4].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("2", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("T", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_Mcap1TrillionEurSmallChars(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseMarketCap(831000, 52000, CURRENCY_EUR, false);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("EUR/MCAP", output[0].c_str());
|
|
||||||
|
|
||||||
std::string joined = joinArrayWithBrackets(output);
|
|
||||||
|
|
||||||
char result[4];
|
|
||||||
snprintf(result, sizeof(result), " %c ", CURRENCY_EUR);
|
|
||||||
TEST_ASSERT_EQUAL_STRING(result, output[NUM_SCREENS - 6].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE(" 1", output[NUM_SCREENS - 5].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("020", output[NUM_SCREENS - 4].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("825", output[NUM_SCREENS - 3].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("000", output[NUM_SCREENS - 2].c_str(), joined.c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING_MESSAGE("000", output[NUM_SCREENS - 1].c_str(), joined.c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_Mcap1TrillionJpy(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseMarketCap(831000, 52000, CURRENCY_JPY, true);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("JPY/MCAP", output[0].c_str());
|
|
||||||
TEST_ASSERT_TRUE(CURRENCY_JPY == output[NUM_SCREENS - 6].c_str()[0]);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("1", output[NUM_SCREENS - 5].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING(".", output[NUM_SCREENS - 4].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("0", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("2", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("T", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void test_Mcap1TrillionJpySmallChars(void)
|
|
||||||
{
|
|
||||||
std::array<std::string, NUM_SCREENS> output = parseMarketCap(831000, 52000, CURRENCY_JPY, false);
|
|
||||||
TEST_ASSERT_EQUAL_STRING("JPY/MCAP", output[0].c_str());
|
|
||||||
|
|
||||||
char result[4];
|
|
||||||
snprintf(result, sizeof(result), " %c ", CURRENCY_JPY);
|
|
||||||
TEST_ASSERT_EQUAL_STRING(result, output[NUM_SCREENS - 6].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING(" 1", output[NUM_SCREENS - 5].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("020", output[NUM_SCREENS - 4].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("825", output[NUM_SCREENS - 3].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("000", output[NUM_SCREENS - 2].c_str());
|
|
||||||
TEST_ASSERT_EQUAL_STRING("000", output[NUM_SCREENS - 1].c_str());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// not needed when using generate_test_runner.rb
|
// not needed when using generate_test_runner.rb
|
||||||
int runUnityTests(void)
|
int runUnityTests(void) {
|
||||||
{
|
|
||||||
UNITY_BEGIN();
|
UNITY_BEGIN();
|
||||||
RUN_TEST(test_CorrectSatsPerDollarConversion);
|
RUN_TEST(test_CorrectSatsPerDollarConversion);
|
||||||
RUN_TEST(test_CorrectSatsPerPoundConversion);
|
|
||||||
RUN_TEST(test_SatsPerDollarAfter1B);
|
|
||||||
RUN_TEST(test_SixCharacterBlockHeight);
|
RUN_TEST(test_SixCharacterBlockHeight);
|
||||||
RUN_TEST(test_SevenCharacterBlockHeight);
|
RUN_TEST(test_SevenCharacterBlockHeight);
|
||||||
RUN_TEST(test_FeeRateDisplay);
|
|
||||||
RUN_TEST(test_PriceOf100kusd);
|
RUN_TEST(test_PriceOf100kusd);
|
||||||
RUN_TEST(test_McapLowerUsd);
|
RUN_TEST(test_PriceOf1MillionUsd);
|
||||||
RUN_TEST(test_Mcap1TrillionUsd);
|
|
||||||
RUN_TEST(test_Mcap1TrillionUsdSmallChars);
|
|
||||||
RUN_TEST(test_Mcap1TrillionEur);
|
|
||||||
RUN_TEST(test_Mcap1TrillionEurSmallChars);
|
|
||||||
RUN_TEST(test_Mcap1TrillionJpy);
|
|
||||||
RUN_TEST(test_Mcap1TrillionJpySmallChars);
|
|
||||||
RUN_TEST(test_PriceSuffixMode);
|
|
||||||
RUN_TEST(test_PriceSuffixModeCompact1);
|
|
||||||
RUN_TEST(test_PriceSuffixModeCompact2);
|
|
||||||
RUN_TEST(test_PriceSuffixModeMow);
|
|
||||||
RUN_TEST(test_PriceSuffixModeMowCompact);
|
|
||||||
|
|
||||||
return UNITY_END();
|
return UNITY_END();
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(void)
|
int main(void) {
|
||||||
{
|
return runUnityTests();
|
||||||
return runUnityTests();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void app_main()
|
extern "C" void app_main() {
|
||||||
{
|
runUnityTests();
|
||||||
runUnityTests();
|
|
||||||
}
|
}
|
||||||
|
|
BIN
x509_crt_bundle
BIN
x509_crt_bundle
Binary file not shown.
Loading…
Reference in a new issue