Angular 21 SPA. Multi-tenant dashboard with a lazy-loaded src.palm.vision sub-app for MPOB-standard palm oil fruit bunch (FFB) ripeness detection. Three inference engines: browser ONNX WASM, browser TFLite WASM, and NestJS remote (WebSocket).
npm start # Dev server at 0.0.0.0:4200
npm run build # Production build → dist/fisapp-ui
npm run build:dev # Dev build → dist/dev
npm run build:prod # Prod build → dist/rc
npm run build:leave:prod # Leave-module variant
npm run build:quot:prod # Quotation variant
npm run build:maf:quot:prod # MAF quotation variant
npm run test # Karma + Jasmine
npm run clean # Clear Angular + npm caches
src/app/)app.config.ts — ApplicationConfig: bootstraps NGXS root store (withNgxsStoragePlugin({keys:'*'}), withNgxsReduxDevtoolsPlugin), Angular Router (hash location, onSameUrlNavigation:'reload'), HTTP client, service worker (registered immediately).app.routes.ts — Root routes: dashboard, auth, leave, tender, src.palm.vision (all lazy-loaded except dashboard).app.component.ts — Root component: 30-minute session timeout, PWA install prompt, theme management (light/dark + blue/pink), multi-language (en_US, ms_MY, zh_Hans), maintenance mode alert.src/src.palm.vision/)Lazy-loaded at /src.palm.vision. Provides its own NGXS VisionState via provideStates([VisionState]) in the feature module.
Sub-routes:
| Path | Component | Purpose |
|---|---|---|
analyzer (default) |
AnalyzerComponent |
Inference UI — file/camera input, bounding box canvas |
vault |
HistoryComponent |
Batch history vault with canvas thumbnails |
chat |
ChatbotComponent |
Industrial Intelligence Portal (n8n chat proxy) |
src/src.palm.vision/services/)InferenceService — Local WASM inference:
analyze(file, batchId?, mode) → Observable<InferenceFrame> — preprocesses image → posts tensor to Web Worker → emits resultpreprocessImage(dataUrl) — resizes to 640×640 on canvas, converts RGBA→CHW, normalizes [0.0, 1.0] → Float32Array [1, 3, 640, 640]results$: Subject<InferenceFrame> — hot stream of all completed framesqueueDepth$: BehaviorSubject<number> — pending worker frame countpendingMap: Map<string, fn> — in-memory frameId → observer resolverRemoteInferenceService — WebSocket bridge to NestJS backend:
analyze(file, sourceLabel?, batchId?) → Observable<InferenceFrame> — sends PalmVision:analyze via FIS envelopegetHistory() → Observable<any[]> — History:getAlldeleteRecord(archiveId) → Observable<{ deleted: boolean }> — History:deleteclearHistory() → Observable<{ deleted: number }> — History:clearAllgetImage(archiveId) → Observable<{ archiveId, image_data }> — PalmHistory:GetImagesaveExternalResult(payload) → Observable<any> — PalmHistory:SaveExternalResultAll calls are wrapped in the FIS envelope and routed via DpService.stream().
Connection config loaded from src/src.palm.vision/config/config.json:
{ "connection": { "uacp": "http://localhost:3000", "uacp_ws": "ws://localhost:3000/socket.io", "uacpEmulation": "on" } }
src/src.palm.vision/store/)VisionStateModel:
{
items: any[]; // History records from server
loading: boolean;
expandedBatchIds: string[]; // Expanded accordion groups
currentInference: InferenceFrame | null;
batchFrames: InferenceFrame[]; // Current batch results
selectedFrameIndex: number | null;
}
Selectors: VisionState.items, VisionState.loading, VisionState.expandedBatchIds, VisionState.currentInference
Actions (vision.actions.ts):
| Action | Payload | Effect |
|---|---|---|
SubmitBatchAnalysis |
{ files: File[]; mode: 'local-onnx' \| 'local-tflite' \| 'remote' } |
Run batch inference; local results persisted via saveExternalResult before vault display |
ToggleBatchGroup |
{ batchId: string } |
Expand/collapse history accordion |
LoadGroupImages |
{ batchId: string } |
Lazy-load archived images for a batch |
LoadHistory |
— | Fetch last 50 records from SQLite |
DeleteHistoryRecord |
{ id: string } |
Delete record + disk image |
ClearAllHistory |
— | Wipe all records |
Key submitBatchAnalysis behaviour: generates a shared batchId = UUID, fans out per-file streams via merge(), collects via toArray(), then dispatches LoadHistory. In local-onnx/local-tflite mode each frame is first persisted to the backend via saveExternalResult() (with timeout(5000) + catchError fallback) before being committed to the batch.
AnalyzerComponent — Three-mode inference UI:
local-onnx (ONNX WASM), local-tflite (TFLite WASM), remote (NestJS server)getUserMedia, environment-facing, 640×640)batchFramesrenderPredictionsWithBoxes(frame) — uses norm_box normalized coords preferentially, falls back to box pixel coordsRipe #4caf50, Unripe #ff9800, Underripe #ffeb3b, Overripe #9c27b0, Abnormal #f44336, Empty_Bunch #607d8bHistoryComponent — Batch vault:
items by batch_id into BatchGroup[] via combineLatest([items$, expandedBatchIds$])@ViewChildren('thumbCanvas') — canvas thumbnails rendered via renderThumbnailWithBoxes() using norm_box coordinatespaintAllVisibleCanvases() with setTimeout(0) deferral; data-archive-id attribute maps canvas elements to data itemsONNX, TFLite, ServerChatbotComponent — Thin wrapper around angularlib/chat/chat.component with title "Industrial Intelligence Portal".
src/src.palm.vision/workers/inference.worker.ts)Receives { frameId, batchId, imageDataUrl, tensor: ArrayBuffer (transferred), processingStart, mode }.
local-onnx): Loads ONNX session once from /assets/models/onnx/best.onnx; runs via onnxruntime-web; output shape [1, N, 6] (x, y, x, y, conf, classIdx); filters at confidence ≥ 0.25.local-tflite): Dynamically imports /assets/tflite-wasm/tflite_web_api_client.js; initializes TFLiteWebModelRunner for /assets/models/tflite/best_float16.tflite; converts CHW→HWC before upload; model includes internal NMS; output [1, 300, 6]; filters at confidence ≥ 0.20; maps [ymin, xmin, ymax, xmax] → [nx1, ny1, nx2, ny2].Returns InferenceFrame-shaped postMessage correlated by frameId.
interface DetectionResult {
bunch_id: number;
class: string; // MPOB class name
confidence: number; // [0, 1]
is_health_alert: boolean; // true if Abnormal | Empty_Bunch
box: [x1, y1, x2, y2]; // absolute pixel coords
norm_box?: [nx1, ny1, nx2, ny2]; // normalized [0,1] coords
}
interface InferenceFrame {
frameId: string;
batchId?: string;
imageDataUrl: string;
detections: DetectionResult[];
inference_ms: number;
processing_ms: number;
total_count: number;
industrial_summary: Record<string, number>;
source: 'wasm-local' | 'remote' | 'n8n';
}
src/dependencies/)| Alias | Path | Contents |
|---|---|---|
angularlib/* |
src/dependencies/angularlib/ |
UI components (forms, dialogs, chat), auth, UIState, ChatState |
dp-ui/* |
src/dependencies/dp-ui/ |
DpService, NgxSocketService, DPState, FIS message models |
fis/* |
src/dependencies/fis/ |
Domain modules (leave, tender, approval), MetadataState |
fis-commons/* |
src/dependencies/fis-commons/ |
CDN-hosted shared utilities |
src/assets/models/onnx/best.onnx — YOLOv8 ONNX model for browser ONNX WASM inferencesrc/assets/models/tflite/best_float16.tflite — TFLite model (loaded by worker in local-tflite mode)src/assets/wasm/ — ONNX Runtime Web JS + WASM bundlessrc/assets/tflite-wasm/ — TensorFlow Lite WASM runtime (tflite_web_api_client.js + WASM)angular.json defines four production configurations that swap menu and assets via fileReplacements:
| Config | Output | Swaps |
|---|---|---|
production |
dist/rc |
— |
leave-prod |
dist/leave |
menu.ts → menu.leave.ts, leave assets |
quotation-prod |
dist/quotation |
menu.ts → menu.quotation.ts, quotation assets |
maf-quot-prod |
dist/maf/quotation |
MAF quotation assets |
Development build uses a mock socket service (dp.service.t.ts) for offline development.
Service worker registered immediately (registerImmediately) via ngsw-config.json. PWA install prompt handled in AppComponent (mobile platforms only). Three language packs: en_US, ms_MY, zh_Hans.
Frontend connects to NestJS backend at https://localhost:3000 (configurable via src/config/config.json). Backend CORS whitelist is hardcoded in server-desktop/src/main.ts and server-android/src/main.ts — add new device IPs there.