- 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
42 lines
885 B
JavaScript
42 lines
885 B
JavaScript
'use strict';
|
|
|
|
const { PassThrough } = require('stream');
|
|
|
|
module.exports = function (/*streams...*/) {
|
|
var sources = []
|
|
var output = new PassThrough({objectMode: true})
|
|
|
|
output.setMaxListeners(0)
|
|
|
|
output.add = add
|
|
output.isEmpty = isEmpty
|
|
|
|
output.on('unpipe', remove)
|
|
|
|
Array.prototype.slice.call(arguments).forEach(add)
|
|
|
|
return output
|
|
|
|
function add (source) {
|
|
if (Array.isArray(source)) {
|
|
source.forEach(add)
|
|
return this
|
|
}
|
|
|
|
sources.push(source);
|
|
source.once('end', remove.bind(null, source))
|
|
source.once('error', output.emit.bind(output, 'error'))
|
|
source.pipe(output, {end: false})
|
|
return this
|
|
}
|
|
|
|
function isEmpty () {
|
|
return sources.length == 0;
|
|
}
|
|
|
|
function remove (source) {
|
|
sources = sources.filter(function (it) { return it !== source })
|
|
if (!sources.length && output.readable) { output.end() }
|
|
}
|
|
}
|