add snapshot tool, masks, and build for mac

This commit is contained in:
2026-03-23 21:52:17 -07:00
parent 080eefbef6
commit a34b1c980d
29 changed files with 2016 additions and 170 deletions

65
scripts/build-linux.sh Executable file
View File

@@ -0,0 +1,65 @@
#!/usr/bin/env bash
set -euo pipefail
ONE_FILE=false
CREATE_TAR=true
while [[ $# -gt 0 ]]; do
case "$1" in
--onefile) ONE_FILE=true; shift ;;
--no-tar) CREATE_TAR=false; shift ;;
*) echo "Unknown option: $1"; exit 1 ;;
esac
done
REPO_ROOT="$(cd "$(dirname "$0")/.." && pwd)"
cd "$REPO_ROOT"
if [ -d ".venv/bin" ]; then
PYTHON=".venv/bin/python"
else
PYTHON="python3"
fi
FRONTEND_DIST="$REPO_ROOT/frontend/dist"
DEMO_DIR="$REPO_ROOT/demo"
echo "Building frontend bundle..."
npm run build
echo "Installing desktop build dependencies..."
uv pip install -e ".[desktop]"
if $ONE_FILE; then
MODE="--onefile"
else
MODE="--onedir"
fi
echo "Packaging desktop app with PyInstaller..."
$PYTHON -m PyInstaller \
desktop.py \
--noconfirm \
--clean \
--name argonode \
--windowed \
$MODE \
--distpath desktop-dist \
--workpath desktop-build \
--specpath desktop-build \
--add-data "${FRONTEND_DIST}:frontend/dist" \
--add-data "${DEMO_DIR}:demo" \
--collect-all matplotlib \
--collect-all scipy \
--collect-all skimage \
--collect-all webview
if $CREATE_TAR; then
TAR_PATH="desktop-dist/argonode-linux.tar.gz"
echo "Creating tarball..."
tar -czf "$TAR_PATH" -C desktop-dist argonode
echo "Tarball created: $TAR_PATH"
fi
echo "Desktop build complete."
echo "Output: $REPO_ROOT/desktop-dist/"

110
scripts/build-mac.sh Executable file
View File

@@ -0,0 +1,110 @@
#!/usr/bin/env bash
set -euo pipefail
ONE_FILE=false
CREATE_DMG=true
while [[ $# -gt 0 ]]; do
case "$1" in
--onefile) ONE_FILE=true; shift ;;
--no-dmg) CREATE_DMG=false; shift ;;
*) echo "Unknown option: $1"; exit 1 ;;
esac
done
REPO_ROOT="$(cd "$(dirname "$0")/.." && pwd)"
cd "$REPO_ROOT"
if [ -d ".venv/bin" ]; then
PYTHON=".venv/bin/python"
else
PYTHON="python3"
fi
FRONTEND_DIST="$REPO_ROOT/frontend/dist"
DEMO_DIR="$REPO_ROOT/demo"
echo "Building frontend bundle..."
npm run build
echo "Installing desktop build dependencies..."
uv pip install -e ".[desktop]"
if $ONE_FILE; then
MODE="--onefile"
else
MODE="--onedir"
fi
echo "Packaging desktop app with PyInstaller..."
$PYTHON -m PyInstaller \
desktop.py \
--noconfirm \
--clean \
--name argonode \
--windowed \
$MODE \
--distpath desktop-dist \
--workpath desktop-build \
--specpath desktop-build \
--add-data "${FRONTEND_DIST}:frontend/dist" \
--add-data "${DEMO_DIR}:demo" \
--collect-all matplotlib \
--collect-all scipy \
--collect-all skimage \
--collect-all webview \
--icon resources/icon.icns 2>/dev/null || \
$PYTHON -m PyInstaller \
desktop.py \
--noconfirm \
--clean \
--name argonode \
--windowed \
$MODE \
--distpath desktop-dist \
--workpath desktop-build \
--specpath desktop-build \
--add-data "${FRONTEND_DIST}:frontend/dist" \
--add-data "${DEMO_DIR}:demo" \
--collect-all matplotlib \
--collect-all scipy \
--collect-all skimage \
--collect-all webview
APP_BUNDLE="desktop-dist/argonode.app"
if [ ! -d "$APP_BUNDLE" ]; then
# --onedir puts it inside a folder
if [ -d "desktop-dist/argonode/argonode.app" ]; then
APP_BUNDLE="desktop-dist/argonode/argonode.app"
else
echo "Warning: .app bundle not found; skipping DMG creation."
CREATE_DMG=false
fi
fi
if $CREATE_DMG; then
DMG_PATH="desktop-dist/argonode.dmg"
echo "Creating DMG installer..."
rm -f "$DMG_PATH"
# Create a temporary directory for DMG contents
DMG_STAGING="desktop-build/dmg-staging"
rm -rf "$DMG_STAGING"
mkdir -p "$DMG_STAGING"
cp -R "$APP_BUNDLE" "$DMG_STAGING/"
ln -s /Applications "$DMG_STAGING/Applications"
hdiutil create \
-volname "argonode" \
-srcfolder "$DMG_STAGING" \
-ov \
-format UDZO \
"$DMG_PATH"
rm -rf "$DMG_STAGING"
echo "DMG created: $DMG_PATH"
fi
echo "Desktop build complete."
echo "Output: $REPO_ROOT/desktop-dist/"

View File

@@ -14,6 +14,7 @@ $pythonExe = if (Test-Path ".\.venv\Scripts\python.exe") {
"python"
}
$frontendDist = Join-Path $repoRoot "frontend\dist"
$demoDir = Join-Path $repoRoot "demo"
Write-Host "Building frontend bundle..."
npm run build
@@ -28,13 +29,14 @@ $pyInstallerArgs = @(
"desktop.py",
"--noconfirm",
"--clean",
"--name", "Argonode",
"--name", "argonode",
"--windowed",
$mode,
"--distpath", "desktop-dist",
"--workpath", "desktop-build",
"--specpath", "desktop-build",
"--add-data", "${frontendDist};frontend/dist",
"--add-data", "${demoDir};demo",
"--collect-all", "matplotlib",
"--collect-all", "scipy",
"--collect-all", "skimage",
@@ -45,4 +47,4 @@ Write-Host "Packaging desktop app..."
& $pythonExe @pyInstallerArgs
Write-Host "Desktop build complete."
Write-Host "Output folder: $repoRoot\desktop-dist\Argonode"
Write-Host "Output folder: $repoRoot\desktop-dist\argonode"

View File

@@ -0,0 +1,79 @@
#!/usr/bin/env python3
"""Generate a synthetic nanoparticle image for the demo/ folder.
The image simulates an AFM scan of particles on a flat substrate:
- Slightly noisy background
- ~20 hemisphere-shaped particles with varying radii and heights
- Saved as both .npy (calibrated float64) and .png (visual preview)
Run from project root:
python scripts/generate_demo_particles.py
"""
import numpy as np
from pathlib import Path
DEMO_DIR = Path(__file__).resolve().parent.parent / "demo"
DEMO_DIR.mkdir(exist_ok=True)
RNG = np.random.default_rng(2024)
# --- Image parameters ---
N = 256 # pixels
SCAN_SIZE = 5e-6 # 5 µm scan
PIXEL_SIZE = SCAN_SIZE / N # metres per pixel
BG_NOISE_RMS = 0.3e-9 # 0.3 nm background noise
# --- Generate particles ---
particles = []
# Hand-placed cluster + random scatter to give a realistic spread
fixed = [
# (cx_frac, cy_frac, radius_nm, height_nm)
(0.25, 0.30, 120, 30),
(0.28, 0.34, 80, 20),
(0.70, 0.25, 150, 45),
(0.50, 0.55, 100, 25),
(0.55, 0.60, 60, 15),
(0.15, 0.75, 200, 55),
(0.80, 0.80, 90, 22),
]
for cx_f, cy_f, r_nm, h_nm in fixed:
particles.append((cx_f * N, cy_f * N, r_nm * 1e-9, h_nm * 1e-9))
# Random particles
for _ in range(15):
cx = RNG.uniform(20, N - 20)
cy = RNG.uniform(20, N - 20)
radius = RNG.uniform(30, 180) * 1e-9 # 30180 nm
height = RNG.uniform(8, 60) * 1e-9 # 860 nm
particles.append((cx, cy, radius, height))
# --- Render height map ---
image = RNG.normal(0, BG_NOISE_RMS, (N, N))
yy, xx = np.mgrid[0:N, 0:N]
for cx, cy, radius_m, height_m in particles:
radius_px = radius_m / PIXEL_SIZE
dist2 = (xx - cx) ** 2 + (yy - cy) ** 2
inside = dist2 < radius_px ** 2
# Hemisphere profile: z = h * sqrt(1 - (r/R)^2)
z = np.zeros_like(image)
z[inside] = height_m * np.sqrt(1.0 - dist2[inside] / radius_px ** 2)
image = np.maximum(image, z) # particles don't subtract from each other
# --- Save .npy (float64 metres) ---
npy_path = DEMO_DIR / "nanoparticles.npy"
np.save(str(npy_path), image)
print(f"Saved {npy_path} shape={image.shape} range=[{image.min():.2e}, {image.max():.2e}] m")
# --- Save .png (8-bit grayscale for quick visual) ---
from PIL import Image
normed = (image - image.min()) / (image.max() - image.min())
uint8 = (normed * 255).astype(np.uint8)
png_path = DEMO_DIR / "nanoparticles.png"
Image.fromarray(uint8, mode="L").save(str(png_path))
print(f"Saved {png_path}")
print(f"\n{len(particles)} particles generated on a {SCAN_SIZE*1e6:.0f} µm × {SCAN_SIZE*1e6:.0f} µm scan")