Boot: - Editor now picks local vs server mode based on URL flag, sign-in state, and a stale local-mode flag. Signed-in users are no longer bounced to IndexedDB if they had previously clicked "Use locally". Local mode: - New LocalWorkspaceClient (src/static/local-workspace.js) with pluggable IndexedDB and File System Access backends. Picked folder handles persist across reloads with a Reconnect button when the permission lapses. - Static-only host: scripts/serve_static_editor.py serves src/static/ with COOP/COEP so SharedArrayBuffer-backed sims keep working. - Bundled MicroPython stubs ship under src/static/bundled-lib/ for static hosting; FastAPI also exposes them at /api/public/lib-bundle. Workspace import / export: - Zero-dep ZIP encoder + reader (STORE + DEFLATE via DecompressionStream). Export/Import buttons in the workspace badge work in both local and server modes; imports are confined to code/. Pin / ADC / Serial simulation: - machine.py grows ADC, UART, expanded Pin, and PWM mocks, all driven by SharedArrayBuffer when cross-origin isolated and falling back to postMessage + [pin-out] stdout markers otherwise — pins, ADC slider, and serial input now keep working over plain HTTP / LAN-IP origins. - NeoPixel pins are claimed via a [pin-claim] marker and dropped from the Pins panel so the data line doesn't flicker per write(). - New demos: adc_slider_demo.py, pin_demo.py, serial_demo.py. Lib layout: - Single source of truth at repo lib/; workspace/lib/ caching layer removed and the directory deleted. Filesystem service reads stubs directly from PROJECT_ROOT/lib. UI: - Home page slimmed to "Sign in" + "Use locally" with optional editor / manage-users links. Admin user/invite UI moved to /users. - Workspace badge gains storage indicator, Folder…/Reconnect, Export, Import, and Exit controls. - Mobile-friendly tweaks: safer-area padding, larger touch targets, iOS-zoom-proof serial input, file-tree highlight fix. Tests: - test_auth.py patches PROJECT_ROOT for the lib-shared test so the repo-root lib refactor stays green. test_api.py asserts the new "LED Editor" branding. Co-authored-by: Cursor <cursoragent@cursor.com>
55 lines
1.2 KiB
Python
55 lines
1.2 KiB
Python
"""Pin features demo.
|
|
|
|
A "Pins" panel appears below the editor while this script runs:
|
|
|
|
* Pin 2 (OUT) — blinks every 200 ms; the indicator follows along.
|
|
* Pin 4 (OUT) — chases through .on() / .off() / .toggle().
|
|
* Pin 0 (IN) — click the toggle button in the panel to flip its value.
|
|
When it goes 0 -> 1 we register an IRQ that toggles pin 2.
|
|
* Pin 13 (PWM) — duty sweeps up and down; the bar shows the live duty cycle.
|
|
"""
|
|
|
|
import time
|
|
|
|
from machine import Pin, PWM
|
|
|
|
|
|
led_a = Pin(2, Pin.OUT)
|
|
led_b = Pin(4, Pin.OUT)
|
|
button = Pin(0, Pin.IN, Pin.PULL_UP)
|
|
fader = PWM(Pin(13), freq=1000, duty_u16=0)
|
|
|
|
|
|
def on_button(pin):
|
|
print("[irq] button rising edge -> toggling pin 2")
|
|
led_a.toggle()
|
|
|
|
|
|
button.irq(handler=on_button, trigger=Pin.IRQ_RISING)
|
|
|
|
|
|
tick = 0
|
|
duty = 0
|
|
direction = 1024
|
|
|
|
while True:
|
|
led_a.value(tick % 2)
|
|
if tick % 4 == 0:
|
|
led_b.on()
|
|
elif tick % 4 == 2:
|
|
led_b.off()
|
|
|
|
duty += direction
|
|
if duty >= 65535:
|
|
duty = 65535
|
|
direction = -1024
|
|
elif duty <= 0:
|
|
duty = 0
|
|
direction = 1024
|
|
fader.duty_u16(duty)
|
|
|
|
button.value()
|
|
|
|
tick += 1
|
|
time.sleep(0.1)
|