1
0
Fork 0
freesewing/src/pattern.js

263 lines
6.5 KiB
JavaScript
Raw Normal View History

import Attributes from "./attributes";
import { macroName } from "./utils";
import Part from "./part";
import Point from "./point";
import Path from "./path";
import Snippet from "./snippet";
import Svg from "./svg";
import Hooks from "./hooks";
2018-08-01 18:18:29 +02:00
import pack from "bin-pack";
import Store from "./store";
2018-08-10 15:36:39 +02:00
import * as hooklib from "hooks";
2018-07-23 11:12:06 +00:00
export default function Pattern(config = false) {
// width and height properties
2018-08-01 18:18:29 +02:00
this.width = false;
this.height = false;
2018-07-23 11:12:06 +00:00
// Hooks and Svg instance
2018-08-10 15:36:39 +02:00
for (let k in hooklib) this[k] = hooklib[k];
this.hooks = new Hooks();
2018-08-07 15:23:37 +02:00
Svg.prototype.hooks = this.hooks;
2018-07-23 11:12:06 +00:00
// Data containers
2018-08-09 15:10:15 +02:00
this.settings = {
mode: "draft"
};
this.options = {};
this.store = new Store();
2018-07-23 11:12:06 +00:00
this.parts = {};
2018-07-25 14:53:10 +00:00
// Merge config with defaults
let defaults = {
measurements: {},
options: {},
units: "metric"
};
this.config = { ...defaults, ...config };
for (let i in config.options) {
let option = config.options[i];
2018-08-09 15:10:15 +02:00
if (typeof option.type === "undefined") this.options[i] = option.val / 100;
else this.options[i] = option.val;
2018-07-23 11:12:06 +00:00
}
// Constructors
this.Part = Part;
this.Point = Point;
this.Path = Path;
this.Snippet = Snippet;
// Context object to inject in part prototype
2018-07-23 11:12:06 +00:00
this.context = {
parts: this.parts,
config: this.config,
2018-07-25 14:53:10 +00:00
settings: this.settings,
options: this.options,
2018-08-05 16:32:38 +02:00
store: this.store
};
this.Part.prototype.context = this.context;
this.Part.prototype.macros = {};
2018-08-07 13:46:38 +02:00
this.Part.prototype.hooks = this.hooks;
2018-07-23 20:14:32 +02:00
}
2018-07-23 11:12:06 +00:00
2018-07-23 20:14:32 +02:00
/**
* @throws Will throw an error when called
*/
Pattern.prototype.draft = function() {
2018-07-23 20:14:32 +02:00
throw Error(
"You have to implement the draft() method in your Pattern instance."
);
};
2018-07-23 11:12:06 +00:00
2018-08-09 15:10:15 +02:00
/**
* Handles pattern sampling
*/
Pattern.prototype.sample = function() {
if (this.settings.sample.type === "option") {
return this.sampleOption(this.settings.sample.option);
} else if (this.settings.sample.type === "measurement") {
return this.sampleMeasurement(this.settings.sample.measurement);
2018-08-10 14:25:26 +02:00
} else if (this.settings.sample.type === "models") {
return this.sampleModels(this.settings.sample.models);
2018-08-09 15:10:15 +02:00
}
this.draft();
};
2018-08-10 14:25:26 +02:00
Pattern.prototype.sampleParts = function() {
let parts = {};
this.settings.mode = "sample";
this.settings.paperless = false;
this.draft();
for (let i in this.parts) {
parts[i] = new Part();
parts[i].render = this.parts[i].render;
}
console.log("sample parts", parts);
return parts;
};
2018-08-09 15:10:15 +02:00
/**
* Handles option sampling
*/
Pattern.prototype.sampleOption = function(option) {
let factor, step, val;
2018-08-10 14:25:26 +02:00
let parts = this.sampleParts();
2018-08-09 15:10:15 +02:00
if (typeof this.config.options[option].type === "undefined") factor = 100;
else factor = 1;
val = this.config.options[option].min / factor;
step = (this.config.options[option].max / factor - val) / 9;
2018-08-10 14:25:26 +02:00
for (let l = 1; l < 11; l++) {
2018-08-09 15:10:15 +02:00
this.options[option] = val;
this.debug(`Sampling option ${option} with value ${val}`);
this.draft();
for (let i in this.parts) {
for (let j in this.parts[i].paths) {
parts[i].paths[j + "_" + l] = this.parts[i].paths[j]
.clone()
.attr("class", "sample-" + l, true);
}
}
2018-08-10 14:25:26 +02:00
val += step;
}
this.parts = parts;
return this;
};
/**
* Handles measurement sampling
*/
Pattern.prototype.sampleMeasurement = function(measurement) {
2018-08-10 14:25:26 +02:00
let parts = this.sampleParts();
let val = this.settings.measurements[measurement];
let step = val / 50;
val = val * 0.9;
2018-08-10 14:25:26 +02:00
for (let l = 1; l < 11; l++) {
this.settings.measurements[measurement] = val;
this.debug(`Sampling measurement ${measurement} with value ${val}`);
this.draft();
for (let i in this.parts) {
for (let j in this.parts[i].paths) {
parts[i].paths[j + "_" + l] = this.parts[i].paths[j]
.clone()
.attr("class", "sample-" + l, true);
2018-08-09 15:10:15 +02:00
}
}
2018-08-10 14:25:26 +02:00
val += step;
}
this.parts = parts;
return this;
};
/**
* Handles models sampling
*/
Pattern.prototype.sampleModels = function(models) {
let parts = this.sampleParts();
let count = 0;
for (let l in models) {
count++;
this.settings.measurements = models[l];
this.debug(`Sampling model ${l}`);
this.draft();
for (let i in this.parts) {
for (let j in this.parts[i].paths) {
parts[i].paths[j + "_" + count] = this.parts[i].paths[j]
.clone()
.attr("class", "sample-" + count, true);
}
}
2018-08-09 15:10:15 +02:00
}
this.parts = parts;
return this;
2018-08-09 15:10:15 +02:00
};
2018-08-07 15:23:37 +02:00
/** Debug method, exposes debug hook */
2018-08-10 15:36:39 +02:00
Pattern.prototype.debug = function(data) {};
2018-08-01 18:18:29 +02:00
2018-08-07 15:23:37 +02:00
Pattern.prototype.render = function() {
this.svg = new Svg(this);
2018-07-23 11:12:06 +00:00
2018-08-03 14:20:28 +02:00
return this.pack().svg.render(this);
2018-07-23 20:14:32 +02:00
};
2018-07-23 11:12:06 +00:00
Pattern.prototype.on = function(hook, method) {
2018-07-23 20:14:32 +02:00
if (typeof this.hooks._hooks[hook] === "undefined") {
this.hooks._hooks[hook] = [];
}
this.hooks._hooks[hook].push(method);
2018-08-10 15:36:39 +02:00
// Pattern object hooks need to be attached on load
let localHooks = [
"preDraft",
"postDraft",
"preSample",
"postSample",
"debug"
];
if (localHooks.includes(hook)) {
let self = this;
this.hooks.attach(hook, self);
}
2018-07-23 20:14:32 +02:00
};
2018-07-23 11:12:06 +00:00
Pattern.prototype.with = function(plugin) {
2018-08-07 16:32:16 +02:00
this.debug(`Plugin: ${plugin.name} v${plugin.version}`);
2018-07-23 20:14:32 +02:00
if (plugin.hooks) this.loadPluginHooks(plugin);
if (plugin.macros) this.loadPluginMacros(plugin);
2018-07-23 11:12:06 +00:00
2018-07-23 20:14:32 +02:00
return this;
};
2018-07-23 11:12:06 +00:00
Pattern.prototype.loadPluginHooks = function(plugin) {
2018-07-23 20:14:32 +02:00
for (let hook of this.hooks.all) {
if (typeof plugin.hooks[hook] === "function") {
this.on(hook, plugin.hooks[hook]);
2018-08-11 14:03:06 +02:00
} else if (typeof plugin.hooks[hook] === "array") {
for (let method of plugin.hooks[hook]) {
this.on(hook, method);
}
2018-07-23 11:12:06 +00:00
}
2018-07-23 20:14:32 +02:00
}
};
2018-07-23 11:12:06 +00:00
Pattern.prototype.loadPluginMacros = function(plugin) {
2018-07-23 20:14:32 +02:00
for (let macro in plugin.macros) {
if (typeof plugin.macros[macro] === "function") {
this.macro(macro, plugin.macros[macro]);
2018-07-23 11:12:06 +00:00
}
2018-07-23 20:14:32 +02:00
}
};
Pattern.prototype.macro = function(key, method) {
this.Part.prototype[macroName(key)] = method;
};
2018-08-01 18:18:29 +02:00
/** Packs parts in a 2D space and sets pattern size */
Pattern.prototype.pack = function() {
2018-08-01 18:18:29 +02:00
let bins = [];
for (let key in this.parts) {
let part = this.parts[key];
if (part.render) {
part.stack();
bins.push({
id: key,
2018-08-01 18:18:29 +02:00
width: part.bottomRight.x - part.topLeft.x,
height: part.bottomRight.y - part.topLeft.y
});
}
}
let size = pack(bins, { inPlace: true });
for (let bin of bins) {
let part = this.parts[bin.id];
if (bin.x !== 0 || bin.y !== 0)
part.attr("transform", `translate (${bin.x}, ${bin.y})`);
}
this.width = size.width;
this.height = size.height;
return this;
};