feat(smart-app): implement complete mobile app MVP

- 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
This commit is contained in:
Eric FELIXINE
2026-06-01 18:00:35 -04:00
parent 08ca495bde
commit e30ae8ed09
35578 changed files with 3703534 additions and 43 deletions

View File

@@ -0,0 +1,6 @@
module.exports = {
DEFAULT_INITIAL_SIZE: (8 * 1024),
DEFAULT_INCREMENT_AMOUNT: (8 * 1024),
DEFAULT_FREQUENCY: 1,
DEFAULT_CHUNK_SIZE: 1024
};

View File

@@ -0,0 +1,136 @@
var stream = require("stream"),
constants = require("./constants"),
util = require("util");
var ReadableStreamBuffer = module.exports = function(opts) {
var that = this;
stream.Stream.call(this);
opts = opts || {};
var frequency = opts.hasOwnProperty("frequency") ? opts.frequency : constants.DEFAULT_FREQUENCY;
var chunkSize = opts.chunkSize || constants.DEFAULT_CHUNK_SIZE;
var initialSize = opts.initialSize || constants.DEFAULT_INITIAL_SIZE;
var incrementAmount = opts.incrementAmount || constants.DEFAULT_INCREMENT_AMOUNT;
var size = 0;
var buffer = new Buffer(initialSize);
var encoding = null;
this.readable = true;
this.writable = false;
var sendData = function() {
var amount = Math.min(chunkSize, size);
if (amount > 0) {
var chunk = null;
if(encoding) {
chunk = buffer.toString(encoding, 0, amount);
}
else {
chunk = new Buffer(amount);
buffer.copy(chunk, 0, 0, amount);
}
that.emit("data", chunk);
if(amount < buffer.length)
buffer.copy(buffer, 0, amount, size);
size -= amount;
}
if(size === 0 && !that.readable) {
that.emit("end");
that.emit("close");
if (sendData && sendData.interval) {
clearInterval(sendData.interval);
sendData.interval = null;
}
}
};
this.size = function() {
return size;
};
this.maxSize = function() {
return buffer.length;
};
var increaseBufferIfNecessary = function(incomingDataSize) {
if((buffer.length - size) < incomingDataSize) {
var factor = Math.ceil((incomingDataSize - (buffer.length - size)) / incrementAmount);
var newBuffer = new Buffer(buffer.length + (incrementAmount * factor));
buffer.copy(newBuffer, 0, 0, size);
buffer = newBuffer;
}
};
this.put = function(data, encoding) {
if(!that.readable) return;
var wasEmpty = size === 0;
if(Buffer.isBuffer(data)) {
increaseBufferIfNecessary(data.length);
data.copy(buffer, size, 0);
size += data.length;
}
else {
data = data + "";
var dataSizeInBytes = Buffer.byteLength(data);
increaseBufferIfNecessary(dataSizeInBytes);
buffer.write(data, size, encoding || "utf8");
size += dataSizeInBytes;
}
if (wasEmpty && size > 0) {
this.emit('readable')
}
if (!this.isPaused && !frequency) {
while (size > 0) {
sendData();
}
}
};
this.pause = function() {
this.isPaused = true;
if(sendData && sendData.interval) {
clearInterval(sendData.interval);
delete sendData.interval;
}
};
this.resume = function() {
this.isPaused = false;
if(sendData && !sendData.interval && frequency > 0) {
sendData.interval = setInterval(sendData, frequency);
}
};
this.destroy = function() {
that.emit("end");
if(sendData.interval) clearInterval(sendData.interval);
sendData = null;
that.readable = false;
that.emit("close");
};
this.destroySoon = function() {
that.readable = false;
if (!sendData.interval) {
that.emit("end");
that.emit("close");
}
};
this.setEncoding = function(_encoding) {
encoding = _encoding;
};
this.resume();
};
util.inherits(ReadableStreamBuffer, stream.Stream);

View File

@@ -0,0 +1,3 @@
module.exports = require("./constants");
module.exports.ReadableStreamBuffer = require("./readable_streambuffer");
module.exports.WritableStreamBuffer = require("./writable_streambuffer");

View File

@@ -0,0 +1,97 @@
var util = require("util"),
stream = require("stream"),
constants = require("./constants");
// TODO: clear up specs on returning false from a write and emitting a drain event.
// Does this mean if I return false from a write, I should ignore any write requests between that false return and the drain event?
var WritableStreamBuffer = module.exports = function(opts) {
var that = this;
stream.Stream.call(this);
opts = opts || {};
var initialSize = opts.initialSize || constants.DEFAULT_INITIAL_SIZE;
var incrementAmount = opts.incrementAmount || constants.DEFAULT_INCREMENT_AMOUNT;
var buffer = new Buffer(initialSize);
var size = 0;
this.writable = true;
this.readable = false;
this.size = function() {
return size;
};
this.maxSize = function() {
return buffer.length;
};
this.getContents = function(length) {
if(!size) return false;
var data = new Buffer(Math.min(length || size, size));
buffer.copy(data, 0, 0, data.length);
if(data.length < size)
buffer.copy(buffer, 0, data.length);
size -= data.length;
return data;
};
this.getContentsAsString = function(encoding, length) {
if(!size) return false;
var data = buffer.toString(encoding || "utf8", 0, Math.min(length || size, size));
var dataLength = Buffer.byteLength(data);
if(dataLength < size)
buffer.copy(buffer, 0, dataLength);
size -= dataLength;
return data;
};
var increaseBufferIfNecessary = function(incomingDataSize) {
if((buffer.length - size) < incomingDataSize) {
var factor = Math.ceil((incomingDataSize - (buffer.length - size)) / incrementAmount);
var newBuffer = new Buffer(buffer.length + (incrementAmount * factor));
buffer.copy(newBuffer, 0, 0, size);
buffer = newBuffer;
}
};
this.write = function(data, encoding, callback) {
if(!that.writable) return;
if(Buffer.isBuffer(data)) {
increaseBufferIfNecessary(data.length);
data.copy(buffer, size, 0);
size += data.length;
}
else {
data = data + "";
increaseBufferIfNecessary(Buffer.byteLength(data));
buffer.write(data, size, encoding || "utf8");
size += Buffer.byteLength(data);
}
if(typeof callback === "function") { callback() ;}
};
this.end = function() {
var args = Array.prototype.slice.apply(arguments);
if(args.length) that.write.apply(that, args);
that.emit('finish');
that.destroy();
};
this.destroySoon = this.destroy = function() {
that.writable = false;
that.emit("close");
};
};
util.inherits(WritableStreamBuffer, stream.Stream);