@mostajs/face 1.3.0 → 1.3.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -71,15 +71,18 @@ function FaceDetector({ photo, onCapture, onClear, verifyDescriptor, onVerifyRes
71
71
  const stream = await navigator.mediaDevices.getUserMedia({
72
72
  video: { width: 320, height: 240, facingMode: 'user' },
73
73
  });
74
- if (videoRef.current) {
75
- videoRef.current.srcObject = stream;
76
- videoRef.current.play();
77
- setStreaming(true);
78
- if (enabled && modelsReady) {
79
- videoRef.current.onloadeddata = () => {
80
- animFrameRef.current = requestAnimationFrame(detectLoop);
81
- };
82
- }
74
+ // If component unmounted while awaiting getUserMedia, stop tracks immediately
75
+ if (!videoRef.current) {
76
+ stream.getTracks().forEach((track) => track.stop());
77
+ return;
78
+ }
79
+ videoRef.current.srcObject = stream;
80
+ await videoRef.current.play().catch(() => { });
81
+ setStreaming(true);
82
+ if (enabled && modelsReady) {
83
+ videoRef.current.onloadeddata = () => {
84
+ animFrameRef.current = requestAnimationFrame(detectLoop);
85
+ };
83
86
  }
84
87
  }
85
88
  catch {
@@ -96,13 +99,20 @@ function FaceDetector({ photo, onCapture, onClear, verifyDescriptor, onVerifyRes
96
99
  setStreaming(false);
97
100
  setFaceDetected(false);
98
101
  }, []);
99
- // Cleanup on unmount
102
+ // Cleanup on unmount — stop all tracks and detach stream
100
103
  (0, react_1.useEffect)(() => {
101
104
  return () => {
102
105
  cancelAnimationFrame(animFrameRef.current);
103
- if (videoRef.current?.srcObject) {
104
- const tracks = videoRef.current.srcObject.getTracks();
105
- tracks.forEach((track) => track.stop());
106
+ const video = videoRef.current;
107
+ if (video) {
108
+ video.pause();
109
+ const stream = video.srcObject;
110
+ if (stream) {
111
+ stream.getTracks().forEach((track) => track.stop());
112
+ }
113
+ video.srcObject = null;
114
+ video.removeAttribute('src');
115
+ video.load();
106
116
  }
107
117
  };
108
118
  }, []);
package/dist/index.d.ts CHANGED
@@ -7,6 +7,7 @@ export { createRecognizeHandler } from './api/recognize.route';
7
7
  export { createDetectHandler } from './api/detect.route';
8
8
  export { default as FaceDetector } from './components/FaceDetector';
9
9
  export type { FaceDetectorProps } from './components/FaceDetector';
10
+ export { faceMenuContribution } from './lib/menu';
10
11
  export type { MostaFaceConfig, FaceDetectionResult, FaceMatchResult, FaceDescriptor, FaceSettings } from './types';
11
12
  export { DEFAULT_FACE_SETTINGS } from './types';
12
13
  export type { FaceCandidate, RecognizeHandlerConfig } from './api/recognize.route';
package/dist/index.js CHANGED
@@ -5,7 +5,7 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
5
5
  return (mod && mod.__esModule) ? mod : { "default": mod };
6
6
  };
7
7
  Object.defineProperty(exports, "__esModule", { value: true });
8
- exports.DEFAULT_FACE_SETTINGS = exports.FaceDetector = exports.createDetectHandler = exports.createRecognizeHandler = exports.useFaceDetection = exports.useCamera = exports.drawDetection = exports.isValidDescriptor = exports.arrayToDescriptor = exports.descriptorToArray = exports.findAllMatches = exports.findMatch = exports.compareFaces = exports.extractDescriptor = exports.detectAllFaces = exports.detectFace = exports.isLoaded = exports.loadModels = void 0;
8
+ exports.DEFAULT_FACE_SETTINGS = exports.faceMenuContribution = exports.FaceDetector = exports.createDetectHandler = exports.createRecognizeHandler = exports.useFaceDetection = exports.useCamera = exports.drawDetection = exports.isValidDescriptor = exports.arrayToDescriptor = exports.descriptorToArray = exports.findAllMatches = exports.findMatch = exports.compareFaces = exports.extractDescriptor = exports.detectAllFaces = exports.detectFace = exports.isLoaded = exports.loadModels = void 0;
9
9
  // Core face-api service
10
10
  var face_api_1 = require("./lib/face-api");
11
11
  Object.defineProperty(exports, "loadModels", { enumerable: true, get: function () { return face_api_1.loadModels; } });
@@ -37,5 +37,8 @@ Object.defineProperty(exports, "createDetectHandler", { enumerable: true, get: f
37
37
  // Components
38
38
  var FaceDetector_1 = require("./components/FaceDetector");
39
39
  Object.defineProperty(exports, "FaceDetector", { enumerable: true, get: function () { return __importDefault(FaceDetector_1).default; } });
40
+ // Menu contribution
41
+ var menu_1 = require("./lib/menu");
42
+ Object.defineProperty(exports, "faceMenuContribution", { enumerable: true, get: function () { return menu_1.faceMenuContribution; } });
40
43
  var types_1 = require("./types");
41
44
  Object.defineProperty(exports, "DEFAULT_FACE_SETTINGS", { enumerable: true, get: function () { return types_1.DEFAULT_FACE_SETTINGS; } });
@@ -0,0 +1,2 @@
1
+ import type { ModuleMenuContribution } from '@mostajs/menu';
2
+ export declare const faceMenuContribution: ModuleMenuContribution;
@@ -0,0 +1,13 @@
1
+ "use strict";
2
+ // @mostajs/face — Menu contribution
3
+ // Author: Dr Hamid MADANI drmdh@msn.com
4
+ //
5
+ // @mostajs/face ne fournit PAS de pages dashboard propres.
6
+ // Il s'integre dans d'autres pages (reception, clients...).
7
+ // Ce fichier est present par convention avec une contribution vide.
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.faceMenuContribution = void 0;
10
+ exports.faceMenuContribution = {
11
+ moduleKey: 'face',
12
+ // Pas d'items ni de groups : ce module n'a pas de pages propres
13
+ };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@mostajs/face",
3
- "version": "1.3.0",
3
+ "version": "1.3.2",
4
4
  "description": "Reusable face recognition module — detection, descriptor extraction, 1:N matching",
5
5
  "author": "Dr Hamid MADANI <drmdh@msn.com>",
6
6
  "license": "MIT",
@@ -61,11 +61,27 @@
61
61
  "default": "./dist/types/index.js"
62
62
  }
63
63
  },
64
- "files": ["dist", "LICENSE", "README.md"],
65
- "keywords": ["face-recognition", "face-detection", "face-api", "biometrics", "react", "mosta"],
66
- "repository": { "type": "git", "url": "https://github.com/apolocine/mosta-face" },
64
+ "files": [
65
+ "dist",
66
+ "LICENSE",
67
+ "README.md"
68
+ ],
69
+ "keywords": [
70
+ "face-recognition",
71
+ "face-detection",
72
+ "face-api",
73
+ "biometrics",
74
+ "react",
75
+ "mosta"
76
+ ],
77
+ "repository": {
78
+ "type": "git",
79
+ "url": "https://github.com/apolocine/mosta-face"
80
+ },
67
81
  "homepage": "https://mostajs.dev/packages/face",
68
- "engines": { "node": ">=18.0.0" },
82
+ "engines": {
83
+ "node": ">=18.0.0"
84
+ },
69
85
  "scripts": {
70
86
  "build": "tsc",
71
87
  "prepublishOnly": "npm run build"
@@ -74,11 +90,18 @@
74
90
  "@vladmandic/face-api": "^1.7.0"
75
91
  },
76
92
  "peerDependencies": {
93
+ "@mostajs/menu": ">=1.0.2",
77
94
  "react": ">=18"
78
95
  },
79
96
  "devDependencies": {
97
+ "@mostajs/menu": "^1.0.2",
80
98
  "@types/react": "^19.0.0",
81
99
  "react": "^19.0.0",
82
100
  "typescript": "^5.6.0"
101
+ },
102
+ "peerDependenciesMeta": {
103
+ "@mostajs/menu": {
104
+ "optional": true
105
+ }
83
106
  }
84
107
  }