- App.tsx: full navigation (Auth stack + Main tabs with 5 screens) - Auth: LoginScreen, RegisterScreen, ForgotPasswordScreen - HomeScreen: dashboard with IoT metrics, weather widget, alerts, quick actions, sensors - MapScreen: interactive map with layer toggles (6 layers) - MarketplaceScreen: categories (6), products (5), search - ChatScreen: AI chat with quick prompts (4), bot responses - ProfileScreen: user info, stats, menu (9 items), logout - AlertsScreen: alert list with severity, acknowledge - SensorsScreen: sensor list with type filters (6 types), search - ZonesScreen: zone cards with stats - SettingsScreen: language picker (FR/EN/ES/DE), privacy, about - Stores: iotStore (sensors, zones, alerts), notificationStore, uiStore + i18n - Hooks: useSensors, useAlerts, useNotifications, useLocation - Components: Card, Button, LoadingSpinner, ErrorBoundary, Header - Services: iotService, notificationService (with axios API client) - Utils: formatters (temp, AQI, noise, dates), validators (email, password, IBAN) - Theme: colors.ts with full design system (Blue Ocean palette) - Ditto: fixed MongoDB connection, new JWT secrets, official gateway image
67 lines
2.1 KiB
JavaScript
67 lines
2.1 KiB
JavaScript
"use strict";
|
|
|
|
// rawAsap provides everything we need except exception management.
|
|
var rawAsap = require("./raw");
|
|
// RawTasks are recycled to reduce GC churn.
|
|
var freeTasks = [];
|
|
// We queue errors to ensure they are thrown in right order (FIFO).
|
|
// Array-as-queue is good enough here, since we are just dealing with exceptions.
|
|
var pendingErrors = [];
|
|
var requestErrorThrow = rawAsap.makeRequestCallFromTimer(throwFirstError);
|
|
|
|
function throwFirstError() {
|
|
if (pendingErrors.length) {
|
|
throw pendingErrors.shift();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Calls a task as soon as possible after returning, in its own event, with priority
|
|
* over other events like animation, reflow, and repaint. An error thrown from an
|
|
* event will not interrupt, nor even substantially slow down the processing of
|
|
* other events, but will be rather postponed to a lower priority event.
|
|
* @param {{call}} task A callable object, typically a function that takes no
|
|
* arguments.
|
|
*/
|
|
module.exports = asap;
|
|
function asap(task) {
|
|
var rawTask;
|
|
if (freeTasks.length) {
|
|
rawTask = freeTasks.pop();
|
|
} else {
|
|
rawTask = new RawTask();
|
|
}
|
|
rawTask.task = task;
|
|
rawAsap(rawTask);
|
|
}
|
|
|
|
// We wrap tasks with recyclable task objects. A task object implements
|
|
// `call`, just like a function.
|
|
function RawTask() {
|
|
this.task = null;
|
|
}
|
|
|
|
// The sole purpose of wrapping the task is to catch the exception and recycle
|
|
// the task object after its single use.
|
|
RawTask.prototype.call = function () {
|
|
try {
|
|
this.task.call();
|
|
} catch (error) {
|
|
if (asap.onerror) {
|
|
// This hook exists purely for testing purposes.
|
|
// Its name will be periodically randomized to break any code that
|
|
// depends on its existence.
|
|
asap.onerror(error);
|
|
} else {
|
|
// In a web browser, exceptions are not fatal. However, to avoid
|
|
// slowing down the queue of pending tasks, we rethrow the error in a
|
|
// lower priority turn.
|
|
pendingErrors.push(error);
|
|
requestErrorThrow();
|
|
}
|
|
} finally {
|
|
this.task = null;
|
|
freeTasks[freeTasks.length] = this;
|
|
}
|
|
};
|