tfjs-evolution 0.0.4 → 0.0.5
Sign up to get free protection for your applications and to get access to all the features.
- package/README.md +6 -1
- package/esm2022/lib/components/display-panel/display-panel.component.mjs +32 -5
- package/esm2022/lib/models/custom-mobilenet.mjs +182 -0
- package/esm2022/lib/models/teachable-evolution.mjs +424 -0
- package/esm2022/lib/utils/class.mjs +2 -0
- package/esm2022/lib/utils/tf.mjs +29 -0
- package/esm2022/lib/utils/util.mjs +165 -0
- package/fesm2022/tfjs-evolution.mjs +824 -4
- package/fesm2022/tfjs-evolution.mjs.map +1 -1
- package/lib/components/display-panel/display-panel.component.d.ts +4 -0
- package/lib/models/custom-mobilenet.d.ts +63 -0
- package/lib/models/teachable-evolution.d.ts +101 -0
- package/lib/utils/class.d.ts +5 -0
- package/lib/utils/tf.d.ts +9 -0
- package/lib/utils/util.d.ts +43 -0
- package/package.json +1 -1
package/README.md
CHANGED
@@ -52,7 +52,7 @@ Make the call for loading images. You could most likely make the call at the con
|
|
52
52
|
```typescript
|
53
53
|
ngAfterViewInit(): void {
|
54
54
|
const options={
|
55
|
-
base: "./assets/dataset",//base where are
|
55
|
+
base: "./assets/dataset",//base where are your iamges
|
56
56
|
file_name:"image",//how your images are named
|
57
57
|
file_extension:"jpeg" //extension used
|
58
58
|
};
|
@@ -72,6 +72,11 @@ Finally, do not forget to add the HTML code
|
|
72
72
|
|
73
73
|
You should see in our HTML file, in Angular, the images
|
74
74
|
|
75
|
+
|
76
|
+
# Updates
|
77
|
+
|
78
|
+
I have finished! Hope to publish a paper soon!
|
79
|
+
|
75
80
|
# Further help
|
76
81
|
|
77
82
|
Feel free to get in touch: jorgeguerrabrazil@gmail.com
|
@@ -1,5 +1,6 @@
|
|
1
1
|
import { Component } from '@angular/core';
|
2
2
|
import { CommonModule } from '@angular/common';
|
3
|
+
import { TeachableMobileNet } from '../../models/teachable-evolution';
|
3
4
|
import * as i0 from "@angular/core";
|
4
5
|
import * as i1 from "@angular/common";
|
5
6
|
export class DisplayPanelComponent {
|
@@ -8,6 +9,7 @@ export class DisplayPanelComponent {
|
|
8
9
|
}
|
9
10
|
loadImages(number_of_species, classes_names, options) {
|
10
11
|
this.classes_names = classes_names;
|
12
|
+
this.number_of_samples_per_class = number_of_species;
|
11
13
|
this.add_species(number_of_species, options);
|
12
14
|
}
|
13
15
|
add_species(number_of_species, options) {
|
@@ -17,16 +19,41 @@ export class DisplayPanelComponent {
|
|
17
19
|
}
|
18
20
|
add_images(name, number_of_species, options) {
|
19
21
|
const class_add = [];
|
20
|
-
for (let i =
|
21
|
-
class_add.push(`${options.base}/${name}/${options.file_name} ${i}.${options.file_extension}`);
|
22
|
+
for (let i = 0; i < number_of_species; i++) {
|
23
|
+
class_add.push(`${options.base}/${name}/${options.file_name} ${i + 1}.${options.file_extension}`);
|
22
24
|
}
|
23
25
|
this.classes.push({ name: name, images: class_add });
|
24
26
|
}
|
27
|
+
async addexamples() {
|
28
|
+
//This is needed to make sure it gives time for the images to upload
|
29
|
+
//The images upload very fast, what makes this method execute before the images are on HTML
|
30
|
+
//It can be removed if somehow this method is just called after the images are available.
|
31
|
+
// console.log("Loading examples as tensors....")
|
32
|
+
await this.delay(0);
|
33
|
+
for (let i = 0; i < this.classes_names.length; i++) {
|
34
|
+
await this.add_example(this.classes_names[i], this.number_of_samples_per_class);
|
35
|
+
}
|
36
|
+
}
|
37
|
+
async add_example(name, number_of_species) {
|
38
|
+
const class_add = [];
|
39
|
+
// console.log(name)
|
40
|
+
for (let i = 0; i < number_of_species; i++) {
|
41
|
+
//Collecting the images from HTML
|
42
|
+
const aux = document.getElementById(`class-${name}-${i}`);
|
43
|
+
//Adding the example
|
44
|
+
const index = this.classes_names.findIndex((elem) => elem === name);
|
45
|
+
await TeachableMobileNet.addExample(index, name, aux);
|
46
|
+
}
|
47
|
+
// this.classes.push({name: name, images: class_add})
|
48
|
+
}
|
49
|
+
delay(ms) {
|
50
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
51
|
+
}
|
25
52
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: DisplayPanelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
26
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.4", type: DisplayPanelComponent, isStandalone: true, selector: "neuroevolution-display-panel", ngImport: i0, template: "<div *ngFor=\"let class of classes; index as i\">\r\n <h1>{{class.name}}</h1>\r\n <img *ngFor=\"let item of class.images; index as i\" [src]=\"item\" width=\"
|
53
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.4", type: DisplayPanelComponent, isStandalone: true, selector: "neuroevolution-display-panel", ngImport: i0, template: "<div *ngFor=\"let class of classes; index as i\">\r\n <h1>{{class.name}}</h1>\r\n <img *ngFor=\"let item of class.images; index as i\" [src]=\"item\" width=\"224\" height=\"224\" [id]=\"'class-' + class.name + '-' + i\" crossorigin=\"anonymous\" >\r\n</div>\r\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] }); }
|
27
54
|
}
|
28
55
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: DisplayPanelComponent, decorators: [{
|
29
56
|
type: Component,
|
30
|
-
args: [{ selector: 'neuroevolution-display-panel', standalone: true, imports: [CommonModule], template: "<div *ngFor=\"let class of classes; index as i\">\r\n <h1>{{class.name}}</h1>\r\n <img *ngFor=\"let item of class.images; index as i\" [src]=\"item\" width=\"
|
57
|
+
args: [{ selector: 'neuroevolution-display-panel', standalone: true, imports: [CommonModule], template: "<div *ngFor=\"let class of classes; index as i\">\r\n <h1>{{class.name}}</h1>\r\n <img *ngFor=\"let item of class.images; index as i\" [src]=\"item\" width=\"224\" height=\"224\" [id]=\"'class-' + class.name + '-' + i\" crossorigin=\"anonymous\" >\r\n</div>\r\n" }]
|
31
58
|
}] });
|
32
|
-
//# sourceMappingURL=data:application/json;base64,
|
59
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGlzcGxheS1wYW5lbC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90ZmpzLWV2b2x1dGlvbi9zcmMvbGliL2NvbXBvbmVudHMvZGlzcGxheS1wYW5lbC9kaXNwbGF5LXBhbmVsLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3RmanMtZXZvbHV0aW9uL3NyYy9saWIvY29tcG9uZW50cy9kaXNwbGF5LXBhbmVsL2Rpc3BsYXktcGFuZWwuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBQyxNQUFNLGVBQWUsQ0FBQztBQUV6QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sa0NBQWtDLENBQUM7OztBQVN0RSxNQUFNLE9BQU8scUJBQXFCO0lBUGxDO1FBU0UsWUFBTyxHQUFVLEVBQUUsQ0FBQztLQXlFckI7SUFuRUQsVUFBVSxDQUFDLGlCQUF5QixFQUFFLGFBQXVCLEVBQUUsT0FBZTtRQUUxRSxJQUFJLENBQUMsYUFBYSxHQUFDLGFBQWEsQ0FBQztRQUNqQyxJQUFJLENBQUMsMkJBQTJCLEdBQUMsaUJBQWlCLENBQUM7UUFFbkQsSUFBSSxDQUFDLFdBQVcsQ0FBQyxpQkFBaUIsRUFBRSxPQUFPLENBQUMsQ0FBQztJQUUvQyxDQUFDO0lBR0gsV0FBVyxDQUFDLGlCQUF5QixFQUFFLE9BQWU7UUFFbEQsS0FBSyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFO1lBQ2xELElBQUksQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxDQUFDLENBQUMsRUFBRSxpQkFBaUIsRUFBRSxPQUFPLENBQUMsQ0FBQztTQUVwRTtJQUVILENBQUM7SUFFRCxVQUFVLENBQUMsSUFBWSxFQUFFLGlCQUF5QixFQUFFLE9BQVk7UUFFOUQsTUFBTSxTQUFTLEdBQU8sRUFBRSxDQUFDO1FBRXpCLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxpQkFBaUIsRUFBRSxDQUFDLEVBQUUsRUFBRTtZQUMxQyxTQUFTLENBQUMsSUFBSSxDQUFDLEdBQUcsT0FBTyxDQUFDLElBQUksSUFBSSxJQUFJLElBQUksT0FBTyxDQUFDLFNBQVMsSUFBSSxDQUFDLEdBQUMsQ0FBQyxJQUFJLE9BQU8sQ0FBQyxjQUFjLEVBQUUsQ0FBQyxDQUFDO1NBQ25HO1FBRUQsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsRUFBQyxJQUFJLEVBQUUsSUFBSSxFQUFFLE1BQU0sRUFBRSxTQUFTLEVBQUMsQ0FBQyxDQUFBO0lBRWxELENBQUM7SUFFRixLQUFLLENBQUMsV0FBVztRQUVoQixvRUFBb0U7UUFDcEUsMkZBQTJGO1FBQzNGLHlGQUF5RjtRQUN6RixpREFBaUQ7UUFDakQsTUFBTSxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBRXBCLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtZQUNsRCxNQUFNLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxDQUFDLENBQUMsRUFBRSxJQUFJLENBQUMsMkJBQTJCLENBQUMsQ0FBQztTQUNqRjtJQUNELENBQUM7SUFFRCxLQUFLLENBQUMsV0FBVyxDQUFDLElBQVksRUFBRSxpQkFBeUI7UUFFdkQsTUFBTSxTQUFTLEdBQU8sRUFBRSxDQUFDO1FBQ3pCLG9CQUFvQjtRQUNwQixLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsaUJBQWlCLEVBQUUsQ0FBQyxFQUFFLEVBQUU7WUFFMUMsaUNBQWlDO1lBQ2pDLE1BQU0sR0FBRyxHQUFHLFFBQVEsQ0FBQyxjQUFjLENBQUMsU0FBUyxJQUFJLElBQUksQ0FBQyxFQUFFLENBQXFCLENBQUM7WUFFOUUsb0JBQW9CO1lBQ3BCLE1BQU0sS0FBSyxHQUFFLElBQUksQ0FBQyxhQUFhLENBQUMsU0FBUyxDQUFDLENBQUMsSUFBSSxFQUFDLEVBQUUsQ0FBQSxJQUFJLEtBQUcsSUFBSSxDQUFDLENBQUE7WUFFOUQsTUFBTSxrQkFBa0IsQ0FBQyxVQUFVLENBQUMsS0FBSyxFQUFFLElBQUksRUFBRSxHQUFHLENBQUMsQ0FBQztTQUV6RDtRQUVELHVEQUF1RDtJQUV2RCxDQUFDO0lBRUQsS0FBSyxDQUFDLEVBQVU7UUFDZCxPQUFPLElBQUksT0FBTyxDQUFPLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxVQUFVLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDbkUsQ0FBQzs4R0ExRVkscUJBQXFCO2tHQUFyQixxQkFBcUIsd0ZDWmxDLDBRQUlBLHlERElZLFlBQVk7OzJGQUlYLHFCQUFxQjtrQkFQakMsU0FBUzsrQkFDRSw4QkFBOEIsY0FDNUIsSUFBSSxXQUNQLENBQUMsWUFBWSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50fSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgR3JvdXAgfSBmcm9tICcuLi8uLi9pbnRlcmZhY2VzL2dyb3VwJztcclxuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgVGVhY2hhYmxlTW9iaWxlTmV0IH0gZnJvbSAnLi4vLi4vbW9kZWxzL3RlYWNoYWJsZS1ldm9sdXRpb24nO1xyXG5cclxuQENvbXBvbmVudCh7XHJcbiAgc2VsZWN0b3I6ICduZXVyb2V2b2x1dGlvbi1kaXNwbGF5LXBhbmVsJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdLFxyXG4gIHRlbXBsYXRlVXJsOiAnLi9kaXNwbGF5LXBhbmVsLmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybDogJy4vZGlzcGxheS1wYW5lbC5jb21wb25lbnQuY3NzJ1xyXG59KVxyXG5leHBvcnQgY2xhc3MgRGlzcGxheVBhbmVsQ29tcG9uZW50IHtcclxuXHJcbiAgY2xhc3NlczogR3JvdXBbXT1bXTtcclxuXHJcbiAgY2xhc3Nlc19uYW1lcyE6IHN0cmluZ1tdO1xyXG5cclxuICBudW1iZXJfb2Zfc2FtcGxlc19wZXJfY2xhc3MhOiBudW1iZXI7XHJcblxyXG5sb2FkSW1hZ2VzKG51bWJlcl9vZl9zcGVjaWVzOiBudW1iZXIsIGNsYXNzZXNfbmFtZXM6IHN0cmluZ1tdLCBvcHRpb25zOiBvYmplY3Qpe1xyXG4gICAgXHJcbiAgICB0aGlzLmNsYXNzZXNfbmFtZXM9Y2xhc3Nlc19uYW1lcztcclxuICAgIHRoaXMubnVtYmVyX29mX3NhbXBsZXNfcGVyX2NsYXNzPW51bWJlcl9vZl9zcGVjaWVzO1xyXG5cclxuICAgIHRoaXMuYWRkX3NwZWNpZXMobnVtYmVyX29mX3NwZWNpZXMsIG9wdGlvbnMpOyAgICBcclxuXHJcbiAgfVxyXG5cclxuXHJcbmFkZF9zcGVjaWVzKG51bWJlcl9vZl9zcGVjaWVzOiBudW1iZXIsIG9wdGlvbnM6IG9iamVjdCl7XHJcblxyXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCB0aGlzLmNsYXNzZXNfbmFtZXMubGVuZ3RoOyBpKyspIHtcclxuICAgICAgdGhpcy5hZGRfaW1hZ2VzKHRoaXMuY2xhc3Nlc19uYW1lc1tpXSwgbnVtYmVyX29mX3NwZWNpZXMsIG9wdGlvbnMpO1xyXG4gICAgICBcclxuICAgIH1cclxuXHJcbiAgfVxyXG5cclxuICBhZGRfaW1hZ2VzKG5hbWU6IHN0cmluZywgbnVtYmVyX29mX3NwZWNpZXM6IG51bWJlciwgb3B0aW9uczogYW55KXsgICBcclxuXHJcbiAgICBjb25zdCBjbGFzc19hZGQ6IGFueT0gW107XHJcblxyXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBudW1iZXJfb2Zfc3BlY2llczsgaSsrKSB7ICAgICAgXHJcbiAgICAgIGNsYXNzX2FkZC5wdXNoKGAke29wdGlvbnMuYmFzZX0vJHtuYW1lfS8ke29wdGlvbnMuZmlsZV9uYW1lfSAke2krMX0uJHtvcHRpb25zLmZpbGVfZXh0ZW5zaW9ufWApO1xyXG4gIH1cclxuXHJcbiAgdGhpcy5jbGFzc2VzLnB1c2goe25hbWU6IG5hbWUsIGltYWdlczogY2xhc3NfYWRkfSkgIFxyXG5cclxuICB9XHJcblxyXG4gYXN5bmMgYWRkZXhhbXBsZXMoKXtcclxuXHJcbiAgLy9UaGlzIGlzIG5lZWRlZCB0byBtYWtlIHN1cmUgaXQgZ2l2ZXMgdGltZSBmb3IgdGhlIGltYWdlcyB0byB1cGxvYWRcclxuICAvL1RoZSBpbWFnZXMgdXBsb2FkIHZlcnkgZmFzdCwgd2hhdCBtYWtlcyB0aGlzIG1ldGhvZCBleGVjdXRlIGJlZm9yZSB0aGUgaW1hZ2VzIGFyZSBvbiBIVE1MXHJcbiAgLy9JdCBjYW4gYmUgcmVtb3ZlZCBpZiBzb21laG93IHRoaXMgbWV0aG9kIGlzIGp1c3QgY2FsbGVkIGFmdGVyIHRoZSBpbWFnZXMgYXJlIGF2YWlsYWJsZS5cclxuICAvLyBjb25zb2xlLmxvZyhcIkxvYWRpbmcgZXhhbXBsZXMgYXMgdGVuc29ycy4uLi5cIilcclxuICBhd2FpdCB0aGlzLmRlbGF5KDApO1xyXG5cclxuICBmb3IgKGxldCBpID0gMDsgaSA8IHRoaXMuY2xhc3Nlc19uYW1lcy5sZW5ndGg7IGkrKykge1xyXG4gICAgYXdhaXQgdGhpcy5hZGRfZXhhbXBsZSh0aGlzLmNsYXNzZXNfbmFtZXNbaV0sIHRoaXMubnVtYmVyX29mX3NhbXBsZXNfcGVyX2NsYXNzKTtcclxuICB9XHJcbiAgfVxyXG5cclxuICBhc3luYyBhZGRfZXhhbXBsZShuYW1lOiBzdHJpbmcsIG51bWJlcl9vZl9zcGVjaWVzOiBudW1iZXIpeyAgIFxyXG5cclxuICAgIGNvbnN0IGNsYXNzX2FkZDogYW55PSBbXTtcclxuICAgIC8vIGNvbnNvbGUubG9nKG5hbWUpXHJcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IG51bWJlcl9vZl9zcGVjaWVzOyBpKyspIHsgXHJcbiAgICAgIFxyXG4gICAgICAvL0NvbGxlY3RpbmcgdGhlIGltYWdlcyBmcm9tIEhUTUxcclxuICAgICAgY29uc3QgYXV4ID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoYGNsYXNzLSR7bmFtZX0tJHtpfWApIGFzIEhUTUxJbWFnZUVsZW1lbnQ7ICAgICAgXHJcblxyXG4gICAgICAvL0FkZGluZyB0aGUgZXhhbXBsZVxyXG4gICAgICBjb25zdCBpbmRleD0gdGhpcy5jbGFzc2VzX25hbWVzLmZpbmRJbmRleCgoZWxlbSk9PmVsZW09PT1uYW1lKVxyXG5cclxuICAgICAgYXdhaXQgVGVhY2hhYmxlTW9iaWxlTmV0LmFkZEV4YW1wbGUoaW5kZXgsIG5hbWUsIGF1eCk7ICAgIFxyXG4gICAgICBcclxuICB9XHJcblxyXG4gIC8vIHRoaXMuY2xhc3Nlcy5wdXNoKHtuYW1lOiBuYW1lLCBpbWFnZXM6IGNsYXNzX2FkZH0pICBcclxuXHJcbiAgfVxyXG5cclxuICBkZWxheShtczogbnVtYmVyKSB7XHJcbiAgICByZXR1cm4gbmV3IFByb21pc2U8dm9pZD4oKHJlc29sdmUpID0+IHNldFRpbWVvdXQocmVzb2x2ZSwgbXMpKTtcclxufVxyXG59XHJcbiIsIjxkaXYgKm5nRm9yPVwibGV0IGNsYXNzIG9mIGNsYXNzZXM7IGluZGV4IGFzIGlcIj5cclxuICA8aDE+e3tjbGFzcy5uYW1lfX08L2gxPlxyXG4gIDxpbWcgICpuZ0Zvcj1cImxldCBpdGVtIG9mIGNsYXNzLmltYWdlczsgaW5kZXggYXMgaVwiIFtzcmNdPVwiaXRlbVwiIHdpZHRoPVwiMjI0XCIgaGVpZ2h0PVwiMjI0XCIgW2lkXT1cIidjbGFzcy0nICsgY2xhc3MubmFtZSArICctJyArIGlcIiBjcm9zc29yaWdpbj1cImFub255bW91c1wiID5cclxuPC9kaXY+XHJcbiJdfQ==
|
@@ -0,0 +1,182 @@
|
|
1
|
+
import * as tf from '@tensorflow/tfjs';
|
2
|
+
export const IMAGE_SIZE = 224;
|
3
|
+
const DEFAULT_MOBILENET_VERSION = 2;
|
4
|
+
const DEFAULT_TRAINING_LAYER_V1 = 'conv_pw_13_relu';
|
5
|
+
const DEFAULT_TRAINING_LAYER_V2 = "out_relu";
|
6
|
+
const DEFAULT_ALPHA_V1_v2 = 0.35;
|
7
|
+
const DEFAULT_ALPHA_V1 = 0.25; //256
|
8
|
+
const DEFAULT_ALPHA_V2 = 0.5; //512
|
9
|
+
const DEFAULT_ALPHA_V3 = 0.75; //768 features
|
10
|
+
const DEFAULT_ALPHA_V4 = 1; //1024 features
|
11
|
+
const DEFAULT_ALPHA = 1; //1024 features
|
12
|
+
// v2: 0.35, 0.50, 0.75 or 1.00.
|
13
|
+
const isAlphaValid = (version, alpha) => {
|
14
|
+
if (version === 1) {
|
15
|
+
if (alpha !== 0.25 && alpha !== 0.5 && alpha !== 0.75 && alpha !== 1) {
|
16
|
+
console.warn("Invalid alpha. Options are: 0.25, 0.50, 0.75 or 1.00.");
|
17
|
+
console.log("Loading model with alpha: ", DEFAULT_ALPHA_V1.toFixed(2));
|
18
|
+
return DEFAULT_ALPHA_V1;
|
19
|
+
}
|
20
|
+
}
|
21
|
+
else {
|
22
|
+
if (alpha !== 0.35 && alpha !== 0.5 && alpha !== 0.75 && alpha !== 1) {
|
23
|
+
console.warn("Invalid alpha. Options are: 0.35, 0.50, 0.75 or 1.00.");
|
24
|
+
console.log("Loading model with alpha: ", DEFAULT_ALPHA_V2.toFixed(2));
|
25
|
+
return DEFAULT_ALPHA_V2;
|
26
|
+
}
|
27
|
+
}
|
28
|
+
return alpha;
|
29
|
+
};
|
30
|
+
const parseModelOptions = (options) => {
|
31
|
+
options = options || {};
|
32
|
+
if (options.checkpointUrl && options.trainingLayer) {
|
33
|
+
if (options.alpha || options.version) {
|
34
|
+
console.warn("Checkpoint URL passed to modelOptions, alpha options are ignored");
|
35
|
+
}
|
36
|
+
return [options.checkpointUrl, options.trainingLayer];
|
37
|
+
}
|
38
|
+
else {
|
39
|
+
options.version = options.version || DEFAULT_MOBILENET_VERSION;
|
40
|
+
if (options.version === 1) {
|
41
|
+
options.alpha = options.alpha || DEFAULT_ALPHA_V4;
|
42
|
+
options.alpha = isAlphaValid(options.version, options.alpha);
|
43
|
+
console.log(`Loading mobilenet ${options.version} and alpha ${options.alpha}`);
|
44
|
+
// exception is alpha o f 1 can only be 1.0
|
45
|
+
let alphaString = options.alpha.toFixed(2);
|
46
|
+
if (alphaString === "1.00") {
|
47
|
+
alphaString = "1.0";
|
48
|
+
}
|
49
|
+
console.log("Using the model: ");
|
50
|
+
return [
|
51
|
+
// tslint:disable-next-line:max-line-length
|
52
|
+
//They are loading MobileNet_v1
|
53
|
+
`https://storage.googleapis.com/tfjs-models/tfjs/mobilenet_v1_${alphaString}_${IMAGE_SIZE}/model.json`,
|
54
|
+
DEFAULT_TRAINING_LAYER_V1
|
55
|
+
];
|
56
|
+
}
|
57
|
+
else if (options.version === 2) {
|
58
|
+
options.alpha = options.alpha || DEFAULT_ALPHA_V4;
|
59
|
+
options.alpha = isAlphaValid(options.version, options.alpha);
|
60
|
+
console.log(`Loading mobilenet ${options.version} and alpha ${options.alpha}`);
|
61
|
+
console.log(`Loading mobilenet ${options.version} and alpha ${options.alpha}`);
|
62
|
+
return [
|
63
|
+
// tslint:disable-next-line:max-line-length
|
64
|
+
`https://storage.googleapis.com/teachable-machine-models/mobilenet_v2_weights_tf_dim_ordering_tf_kernels_${options.alpha.toFixed(2)}_${IMAGE_SIZE}_no_top/model.json`,
|
65
|
+
DEFAULT_TRAINING_LAYER_V2
|
66
|
+
];
|
67
|
+
}
|
68
|
+
else {
|
69
|
+
throw new Error(`MobileNet V${options.version} doesn't exist`);
|
70
|
+
}
|
71
|
+
}
|
72
|
+
};
|
73
|
+
/**
|
74
|
+
* load the base mobilenet model
|
75
|
+
* @param modelOptions options determining what model to load
|
76
|
+
*/
|
77
|
+
export async function loadTruncatedMobileNet(modelOptions) {
|
78
|
+
const [checkpointUrl, trainingLayer] = parseModelOptions(modelOptions);
|
79
|
+
const mobilenet = await tf.loadLayersModel(checkpointUrl);
|
80
|
+
if (modelOptions && modelOptions.version === 1) {
|
81
|
+
const layer = mobilenet.getLayer(trainingLayer);
|
82
|
+
const truncatedModel = tf.model({ inputs: mobilenet.inputs, outputs: layer.output });
|
83
|
+
console.log("Feature model loaded, memory: ", tf.memory().numBytes);
|
84
|
+
const model = tf.sequential();
|
85
|
+
model.add(truncatedModel);
|
86
|
+
model.add(tf.layers.flatten());
|
87
|
+
return model;
|
88
|
+
}
|
89
|
+
else {
|
90
|
+
const layer = mobilenet.getLayer(trainingLayer);
|
91
|
+
const truncatedModel = tf.model({ inputs: mobilenet.inputs, outputs: layer.output });
|
92
|
+
console.log("Feature model loaded, memory: ", tf.memory().numBytes);
|
93
|
+
const model = tf.sequential();
|
94
|
+
model.add(truncatedModel);
|
95
|
+
model.add(tf.layers.globalAveragePooling2d({})); // go from shape [7, 7, 1280] to [1280]
|
96
|
+
return model;
|
97
|
+
}
|
98
|
+
}
|
99
|
+
export class CustomMobileNet {
|
100
|
+
static getinputShape() {
|
101
|
+
/**truncatedModel is the base model, the model used to apply transfer learning */
|
102
|
+
const inputShape = this.truncatedModel.outputs[0].shape.slice(1); // [ 7 x 7 x 1280] (not sure about those dimensions)
|
103
|
+
// console.log("Input Shape(complete): ", this.truncatedModel.outputs[0].shape);
|
104
|
+
// console.log("Input Shape: ", inputShape);
|
105
|
+
const inputSize = tf.util.sizeFromShape(inputShape);
|
106
|
+
// console.log("Input Size: ", inputSize);
|
107
|
+
return inputSize;
|
108
|
+
}
|
109
|
+
static get EXPECTED_IMAGE_SIZE() {
|
110
|
+
return IMAGE_SIZE;
|
111
|
+
}
|
112
|
+
getMetadata() {
|
113
|
+
return this._metadata;
|
114
|
+
}
|
115
|
+
constructor() {
|
116
|
+
// this._metadata = fillMetadata(metadata);
|
117
|
+
//Loading the truncated model
|
118
|
+
// loadTruncatedMobileNet();
|
119
|
+
// this.loadFeatureModel();
|
120
|
+
}
|
121
|
+
static async loadFeatureModel() {
|
122
|
+
this.truncatedModel = await loadTruncatedMobileNet();
|
123
|
+
}
|
124
|
+
/**
|
125
|
+
* get the total number of classes existing within model
|
126
|
+
*/
|
127
|
+
// getTotalClasses() {
|
128
|
+
// const output = this.model.output as SymbolicTensor;
|
129
|
+
// const totalClasses = output.shape[1];
|
130
|
+
// return totalClasses;
|
131
|
+
// }
|
132
|
+
/**
|
133
|
+
* get the model labels
|
134
|
+
*/
|
135
|
+
getClassLabels() {
|
136
|
+
return this._metadata.labels;
|
137
|
+
}
|
138
|
+
/**
|
139
|
+
* Given an image element, makes a prediction through mobilenet returning the
|
140
|
+
* probabilities of the top K classes.
|
141
|
+
* @param image the image to classify
|
142
|
+
* @param maxPredictions the maximum number of classification predictions
|
143
|
+
*/
|
144
|
+
// async predictTopK(image: ClassifierInputSource, maxPredictions = 10, flipped = false) {
|
145
|
+
// const croppedImage = cropTo(image, this._metadata.imageSize, flipped);
|
146
|
+
// const logits = tf.tidy(() => {
|
147
|
+
// const captured = capture(croppedImage, this._metadata.grayscale);
|
148
|
+
// return this.model.predict(captured);
|
149
|
+
// });
|
150
|
+
// // Convert logits to probabilities and class names.
|
151
|
+
// const classes = await getTopKClasses(this._metadata.labels, logits as tf.Tensor<tf.Rank>, maxPredictions);
|
152
|
+
// dispose(logits);
|
153
|
+
// return classes;
|
154
|
+
// }
|
155
|
+
/**
|
156
|
+
* Given an image element, makes a prediction through mobilenet returning the
|
157
|
+
* probabilities for ALL classes.
|
158
|
+
* @param image the image to classify
|
159
|
+
* @param flipped whether to flip the image on X
|
160
|
+
*/
|
161
|
+
// async predict(image: ClassifierInputSource, flipped = false) {
|
162
|
+
// const croppedImage = cropTo(image, this._metadata.imageSize, flipped);
|
163
|
+
// const logits = tf.tidy(() => {
|
164
|
+
// const captured = capture(croppedImage, this._metadata.grayscale);
|
165
|
+
// return this.model.predict(captured);
|
166
|
+
// });
|
167
|
+
// const values = await (logits as tf.Tensor<tf.Rank>).data();
|
168
|
+
// const classes = [];
|
169
|
+
// for (let i = 0; i < values.length; i++) {
|
170
|
+
// classes.push({
|
171
|
+
// className: this._metadata.labels[i],
|
172
|
+
// probability: values[i]
|
173
|
+
// });
|
174
|
+
// }
|
175
|
+
// dispose(logits);
|
176
|
+
// return classes;
|
177
|
+
// }
|
178
|
+
dispose() {
|
179
|
+
this.truncatedModel.dispose();
|
180
|
+
}
|
181
|
+
} // end of CustomMobileNet
|
182
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-mobilenet.js","sourceRoot":"","sources":["../../../../../projects/tfjs-evolution/src/lib/models/custom-mobilenet.ts"],"names":[],"mappings":"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","sourcesContent":["\r\nimport * as tf from '@tensorflow/tfjs';\r\n\r\nimport { SymbolicTensor } from '@tensorflow/tfjs';\r\n\r\n\r\nexport const IMAGE_SIZE = 224;\r\n\r\n\r\n/**\r\n * the metadata to describe the model's creation,\r\n * includes the labels associated with the classes\r\n * and versioning information from training.\r\n */\r\nexport interface Metadata {\r\n    tfjsVersion: string;\r\n    tmVersion?: string;\r\n    packageVersion: string;\r\n    packageName: string;\r\n    modelName?: string;\r\n    timeStamp?: string;\r\n    labels: string[];\r\n    userMetadata?: {};\r\n    grayscale?: boolean;\r\n    imageSize?: number;\r\n}\r\n\r\nexport interface ModelOptions {\r\n    version?: number;\r\n    checkpointUrl?: string;\r\n    alpha?: number;\r\n    trainingLayer?: string;\r\n}\r\n\r\nconst DEFAULT_MOBILENET_VERSION = 2;\r\nconst DEFAULT_TRAINING_LAYER_V1 = 'conv_pw_13_relu';\r\nconst DEFAULT_TRAINING_LAYER_V2 = \"out_relu\"; \r\nconst DEFAULT_ALPHA_V1_v2 = 0.35;\r\nconst DEFAULT_ALPHA_V1 = 0.25;//256\r\nconst DEFAULT_ALPHA_V2 = 0.5; //512\r\nconst DEFAULT_ALPHA_V3 = 0.75;//768 features\r\nconst DEFAULT_ALPHA_V4 = 1;//1024 features\r\nconst DEFAULT_ALPHA = 1;//1024 features\r\n\r\n// v2: 0.35, 0.50, 0.75 or 1.00.\r\n\r\nconst isAlphaValid = (version: number, alpha: number) => {\r\n    if (version === 1) {\r\n        if (alpha !== 0.25 && alpha !== 0.5 && alpha !== 0.75 && alpha !== 1) {\r\n            console.warn(\"Invalid alpha. Options are: 0.25, 0.50, 0.75 or 1.00.\");\r\n            console.log(\"Loading model with alpha: \", DEFAULT_ALPHA_V1.toFixed(2)); \r\n            return DEFAULT_ALPHA_V1;\r\n        }\r\n    }\r\n    else {\r\n        if (alpha !== 0.35 && alpha !== 0.5 && alpha !== 0.75 && alpha !== 1) {\r\n            console.warn(\"Invalid alpha. Options are: 0.35, 0.50, 0.75 or 1.00.\");\r\n            console.log(\"Loading model with alpha: \", DEFAULT_ALPHA_V2.toFixed(2)); \r\n            return DEFAULT_ALPHA_V2;\r\n        }\r\n    }\r\n\r\n    return alpha;\r\n};\r\n\r\n\r\nconst parseModelOptions = (options?: ModelOptions) => {\r\n    options = options || {}\r\n\r\n    if (options.checkpointUrl && options.trainingLayer) {\r\n        if (options.alpha || options.version){\r\n            console.warn(\"Checkpoint URL passed to modelOptions, alpha options are ignored\");\r\n        }        \r\n        return [options.checkpointUrl, options.trainingLayer];\r\n    } else {\r\n        options.version = options.version || DEFAULT_MOBILENET_VERSION;\r\n        \r\n        if(options.version === 1){\r\n            options.alpha = options.alpha || DEFAULT_ALPHA_V4;  \r\n            options.alpha = isAlphaValid(options.version, options.alpha);\r\n\r\n            console.log(`Loading mobilenet ${options.version} and alpha ${options.alpha}`);\r\n            // exception is alpha o f 1 can only be 1.0\r\n            let alphaString = options.alpha.toFixed(2);\r\n            if (alphaString === \"1.00\") { alphaString = \"1.0\"; }\r\n\r\n            console.log(\"Using the model: \",  )\r\n\r\n            return [\r\n                // tslint:disable-next-line:max-line-length\r\n                //They are loading MobileNet_v1        \r\n                `https://storage.googleapis.com/tfjs-models/tfjs/mobilenet_v1_${alphaString}_${IMAGE_SIZE}/model.json`,\r\n                DEFAULT_TRAINING_LAYER_V1\r\n            ];\r\n        }\r\n        else if (options.version === 2){\r\n            options.alpha = options.alpha || DEFAULT_ALPHA_V4;  \r\n            options.alpha = isAlphaValid(options.version, options.alpha);\r\n\r\n            console.log(`Loading mobilenet ${options.version} and alpha ${options.alpha}`);\r\n            console.log(`Loading mobilenet ${options.version} and alpha ${options.alpha}`);\r\n\r\n            return [\r\n                // tslint:disable-next-line:max-line-length        \r\n                `https://storage.googleapis.com/teachable-machine-models/mobilenet_v2_weights_tf_dim_ordering_tf_kernels_${options.alpha.toFixed(2)}_${IMAGE_SIZE}_no_top/model.json`,\r\n                DEFAULT_TRAINING_LAYER_V2\r\n            ];\r\n        } else {\r\n            throw new Error(`MobileNet V${options.version} doesn't exist`);\r\n        }   \r\n    }\r\n};\r\n\r\n\r\n/**\r\n * load the base mobilenet model\r\n * @param modelOptions options determining what model to load\r\n */\r\nexport async function loadTruncatedMobileNet(modelOptions?: ModelOptions) {\r\n    const [checkpointUrl, trainingLayer] = parseModelOptions(modelOptions);\r\n    \r\n    \r\n    const mobilenet = await tf.loadLayersModel(checkpointUrl);\r\n\r\n    if (modelOptions && modelOptions.version === 1){\r\n        const layer = mobilenet.getLayer(trainingLayer);\r\n        \r\n        const truncatedModel = tf.model({ inputs: mobilenet.inputs, outputs: layer.output });\r\n        console.log(\"Feature model loaded, memory: \", tf.memory().numBytes);\r\n\r\n        \r\n        const model = tf.sequential();\r\n        model.add(truncatedModel);\r\n        model.add(tf.layers.flatten());        \r\n        \r\n        \r\n        return model;\r\n    }\r\n    else {\r\n        const layer = mobilenet.getLayer(trainingLayer);\r\n        const truncatedModel = tf.model({ inputs: mobilenet.inputs, outputs: layer.output });\r\n        console.log(\"Feature model loaded, memory: \", tf.memory().numBytes);\r\n        const model = tf.sequential();\r\n        model.add(truncatedModel);\r\n        model.add(tf.layers.globalAveragePooling2d({})); // go from shape [7, 7, 1280] to [1280]\r\n        return model;\r\n    }\r\n}\r\n\r\n\r\nexport class CustomMobileNet {\r\n    /**\r\n     * the truncated mobilenet model we will train on top of\r\n     */\r\n    protected truncatedModel!: tf.LayersModel;\r\n    static truncatedModel: tf.Sequential;\r\n\r\nstatic getinputShape(){\r\n   /**truncatedModel is the base model, the model used to apply transfer learning */\r\n  const inputShape: any = this.truncatedModel.outputs[0].shape.slice(1); // [ 7 x 7 x 1280] (not sure about those dimensions)\r\n//   console.log(\"Input Shape(complete): \", this.truncatedModel.outputs[0].shape);\r\n//   console.log(\"Input Shape: \", inputShape);\r\n\r\n  const inputSize = tf.util.sizeFromShape(inputShape);\r\n\r\n//   console.log(\"Input Size: \", inputSize);\r\n\r\n  return inputSize;\r\n}\r\n\r\n    static get EXPECTED_IMAGE_SIZE() {\r\n        return IMAGE_SIZE;\r\n    }\r\n\r\n    protected _metadata!: Metadata;\r\n\r\n    public getMetadata() {\r\n        return this._metadata;\r\n    }\r\n\r\n    constructor() {\r\n        // this._metadata = fillMetadata(metadata);\r\n        //Loading the truncated model\r\n        // loadTruncatedMobileNet();\r\n        // this.loadFeatureModel();\r\n\r\n    }\r\n\r\n   static  async loadFeatureModel(){\r\n\r\n        this.truncatedModel = await loadTruncatedMobileNet();\r\n\r\n    }\r\n\r\n    /**\r\n     * get the total number of classes existing within model\r\n     */\r\n    // getTotalClasses() {\r\n    //     const output = this.model.output as SymbolicTensor;\r\n    //     const totalClasses = output.shape[1];\r\n    //     return totalClasses;\r\n    // }\r\n\r\n    /**\r\n     * get the model labels\r\n     */\r\n    getClassLabels() {\r\n        return this._metadata.labels;\r\n    }\r\n\r\n    /**\r\n     * Given an image element, makes a prediction through mobilenet returning the\r\n     * probabilities of the top K classes.\r\n     * @param image the image to classify\r\n     * @param maxPredictions the maximum number of classification predictions\r\n     */\r\n    // async predictTopK(image: ClassifierInputSource, maxPredictions = 10, flipped = false) {\r\n    //     const croppedImage = cropTo(image, this._metadata.imageSize, flipped);\r\n\r\n    //     const logits = tf.tidy(() => {\r\n    //         const captured = capture(croppedImage, this._metadata.grayscale);\r\n    //         return this.model.predict(captured);\r\n    //     });\r\n\r\n    //     // Convert logits to probabilities and class names.\r\n    //     const classes = await getTopKClasses(this._metadata.labels, logits as tf.Tensor<tf.Rank>, maxPredictions);\r\n    //     dispose(logits);\r\n\r\n    //     return classes;\r\n    // }\r\n\r\n    /**\r\n     * Given an image element, makes a prediction through mobilenet returning the\r\n     * probabilities for ALL classes.\r\n     * @param image the image to classify\r\n     * @param flipped whether to flip the image on X\r\n     */\r\n    // async predict(image: ClassifierInputSource, flipped = false) {\r\n    //     const croppedImage = cropTo(image, this._metadata.imageSize, flipped);\r\n\r\n    //     const logits = tf.tidy(() => {\r\n    //         const captured = capture(croppedImage, this._metadata.grayscale);\r\n    //         return this.model.predict(captured);\r\n    //     });\r\n\r\n    //     const values = await (logits as tf.Tensor<tf.Rank>).data();\r\n\r\n    //     const classes = [];\r\n    //     for (let i = 0; i < values.length; i++) {\r\n    //         classes.push({\r\n    //             className: this._metadata.labels[i],\r\n    //             probability: values[i]\r\n    //         });\r\n    //     }\r\n\r\n    //     dispose(logits);\r\n\r\n    //     return classes;\r\n    // }\r\n\r\n    public dispose() {\r\n        this.truncatedModel.dispose();\r\n    }\r\n}// end of CustomMobileNet\r\n"]}
|