@tsparticles/plugin-emitters-shape-circle 4.0.0-alpha.2 → 4.0.0-alpha.20

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.
@@ -4,7 +4,7 @@
4
4
  * Demo / Generator : https://particles.js.org/
5
5
  * GitHub : https://www.github.com/matteobruni/tsparticles
6
6
  * How to use? : Check the GitHub README
7
- * v4.0.0-alpha.2
7
+ * v4.0.0-alpha.20
8
8
  */
9
9
  /*
10
10
  * ATTENTION: The "eval" devtool has been used (maybe by default in mode: "development").
@@ -28,16 +28,6 @@ return /******/ (() => { // webpackBootstrap
28
28
  /******/ "use strict";
29
29
  /******/ var __webpack_modules__ = ({
30
30
 
31
- /***/ "./dist/browser/index.js"
32
- /*!*******************************!*\
33
- !*** ./dist/browser/index.js ***!
34
- \*******************************/
35
- (__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
36
-
37
- eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ loadEmittersShapeCircle: () => (/* binding */ loadEmittersShapeCircle)\n/* harmony export */ });\nfunction loadEmittersShapeCircle(engine) {\n engine.checkVersion(\"4.0.0-alpha.2\");\n engine.register(async emittersEngine => {\n const {\n EmittersCircleShapeGenerator\n } = await __webpack_require__.e(/*! import() */ \"dist_browser_EmittersCircleShapeGenerator_js\").then(__webpack_require__.bind(__webpack_require__, /*! ./EmittersCircleShapeGenerator.js */ \"./dist/browser/EmittersCircleShapeGenerator.js\"));\n emittersEngine.addEmitterShapeGenerator?.(\"circle\", new EmittersCircleShapeGenerator());\n });\n}\n\n//# sourceURL=webpack://@tsparticles/plugin-emitters-shape-circle/./dist/browser/index.js?\n}");
38
-
39
- /***/ },
40
-
41
31
  /***/ "@tsparticles/engine"
42
32
  /*!*********************************************************************************************************************************!*\
43
33
  !*** external {"commonjs":"@tsparticles/engine","commonjs2":"@tsparticles/engine","amd":"@tsparticles/engine","root":"window"} ***!
@@ -56,6 +46,16 @@ module.exports = __WEBPACK_EXTERNAL_MODULE__tsparticles_engine__;
56
46
 
57
47
  module.exports = __WEBPACK_EXTERNAL_MODULE__tsparticles_plugin_emitters__;
58
48
 
49
+ /***/ },
50
+
51
+ /***/ "./dist/browser/index.js"
52
+ /*!*******************************!*\
53
+ !*** ./dist/browser/index.js ***!
54
+ \*******************************/
55
+ (__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
56
+
57
+ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ loadEmittersShapeCircle: () => (/* binding */ loadEmittersShapeCircle)\n/* harmony export */ });\nasync function loadEmittersShapeCircle(engine) {\n engine.checkVersion(\"4.0.0-alpha.20\");\n await engine.register(async (e)=>{\n const { ensureEmittersPluginLoaded } = await Promise.resolve(/*! import() */).then(__webpack_require__.t.bind(__webpack_require__, /*! @tsparticles/plugin-emitters */ \"@tsparticles/plugin-emitters\", 19));\n ensureEmittersPluginLoaded(e);\n const { EmittersCircleShapeGenerator } = await __webpack_require__.e(/*! import() */ \"dist_browser_EmittersCircleShapeGenerator_js\").then(__webpack_require__.bind(__webpack_require__, /*! ./EmittersCircleShapeGenerator.js */ \"./dist/browser/EmittersCircleShapeGenerator.js\"));\n e.addEmitterShapeGenerator?.(\"circle\", new EmittersCircleShapeGenerator());\n });\n}\n\n\n//# sourceURL=webpack://@tsparticles/plugin-emitters-shape-circle/./dist/browser/index.js?\n}");
58
+
59
59
  /***/ }
60
60
 
61
61
  /******/ });
@@ -94,6 +94,36 @@ module.exports = __WEBPACK_EXTERNAL_MODULE__tsparticles_plugin_emitters__;
94
94
  /******/ __webpack_require__.m = __webpack_modules__;
95
95
  /******/
96
96
  /************************************************************************/
97
+ /******/ /* webpack/runtime/create fake namespace object */
98
+ /******/ (() => {
99
+ /******/ var getProto = Object.getPrototypeOf ? (obj) => (Object.getPrototypeOf(obj)) : (obj) => (obj.__proto__);
100
+ /******/ var leafPrototypes;
101
+ /******/ // create a fake namespace object
102
+ /******/ // mode & 1: value is a module id, require it
103
+ /******/ // mode & 2: merge all properties of value into the ns
104
+ /******/ // mode & 4: return value when already ns object
105
+ /******/ // mode & 16: return value when it's Promise-like
106
+ /******/ // mode & 8|1: behave like require
107
+ /******/ __webpack_require__.t = function(value, mode) {
108
+ /******/ if(mode & 1) value = this(value);
109
+ /******/ if(mode & 8) return value;
110
+ /******/ if(typeof value === 'object' && value) {
111
+ /******/ if((mode & 4) && value.__esModule) return value;
112
+ /******/ if((mode & 16) && typeof value.then === 'function') return value;
113
+ /******/ }
114
+ /******/ var ns = Object.create(null);
115
+ /******/ __webpack_require__.r(ns);
116
+ /******/ var def = {};
117
+ /******/ leafPrototypes = leafPrototypes || [null, getProto({}), getProto([]), getProto(getProto)];
118
+ /******/ for(var current = mode & 2 && value; (typeof current == 'object' || typeof current == 'function') && !~leafPrototypes.indexOf(current); current = getProto(current)) {
119
+ /******/ Object.getOwnPropertyNames(current).forEach((key) => (def[key] = () => (value[key])));
120
+ /******/ }
121
+ /******/ def['default'] = () => (value);
122
+ /******/ __webpack_require__.d(ns, def);
123
+ /******/ return ns;
124
+ /******/ };
125
+ /******/ })();
126
+ /******/
97
127
  /******/ /* webpack/runtime/define property getters */
98
128
  /******/ (() => {
99
129
  /******/ // define getter functions for harmony exports
@@ -1,2 +1,2 @@
1
- /*! For license information please see tsparticles.plugin.emitters.shape.circle.min.js.LICENSE.txt */
2
- !function(e,t){if("object"==typeof exports&&"object"==typeof module)module.exports=t(require("@tsparticles/plugin-emitters"),require("@tsparticles/engine"));else if("function"==typeof define&&define.amd)define(["@tsparticles/plugin-emitters","@tsparticles/engine"],t);else{var r="object"==typeof exports?t(require("@tsparticles/plugin-emitters"),require("@tsparticles/engine")):t(e.window,e.window);for(var i in r)("object"==typeof exports?exports:e)[i]=r[i]}}(this,((e,t)=>(()=>{var r,i,o={303(e){e.exports=t},526(t){t.exports=e}},n={};function a(e){var t=n[e];if(void 0!==t)return t.exports;var r=n[e]={exports:{}};return o[e](r,r.exports,a),r.exports}a.m=o,a.d=(e,t)=>{for(var r in t)a.o(t,r)&&!a.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},a.f={},a.e=e=>Promise.all(Object.keys(a.f).reduce(((t,r)=>(a.f[r](e,t),t)),[])),a.u=e=>e+".min.js",a.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),a.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),r={},i="@tsparticles/plugin-emitters-shape-circle:",a.l=(e,t,o,n)=>{if(r[e])r[e].push(t);else{var s,c;if(void 0!==o)for(var p=document.getElementsByTagName("script"),l=0;l<p.length;l++){var u=p[l];if(u.getAttribute("src")==e||u.getAttribute("data-webpack")==i+o){s=u;break}}s||(c=!0,(s=document.createElement("script")).charset="utf-8",a.nc&&s.setAttribute("nonce",a.nc),s.setAttribute("data-webpack",i+o),s.src=e),r[e]=[t];var d=(t,i)=>{s.onerror=s.onload=null,clearTimeout(f);var o=r[e];if(delete r[e],s.parentNode&&s.parentNode.removeChild(s),o&&o.forEach((e=>e(i))),t)return t(i)},f=setTimeout(d.bind(null,void 0,{type:"timeout",target:s}),12e4);s.onerror=d.bind(null,s.onerror),s.onload=d.bind(null,s.onload),c&&document.head.appendChild(s)}},a.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},(()=>{var e;a.g.importScripts&&(e=a.g.location+"");var t=a.g.document;if(!e&&t&&(t.currentScript&&"SCRIPT"===t.currentScript.tagName.toUpperCase()&&(e=t.currentScript.src),!e)){var r=t.getElementsByTagName("script");if(r.length)for(var i=r.length-1;i>-1&&(!e||!/^http(s?):/.test(e));)e=r[i--].src}if(!e)throw new Error("Automatic publicPath is not supported in this browser");e=e.replace(/^blob:/,"").replace(/#.*$/,"").replace(/\?.*$/,"").replace(/\/[^\/]+$/,"/"),a.p=e})(),(()=>{var e={938:0};a.f.j=(t,r)=>{var i=a.o(e,t)?e[t]:void 0;if(0!==i)if(i)r.push(i[2]);else{var o=new Promise(((r,o)=>i=e[t]=[r,o]));r.push(i[2]=o);var n=a.p+a.u(t),s=new Error;a.l(n,(r=>{if(a.o(e,t)&&(0!==(i=e[t])&&(e[t]=void 0),i)){var o=r&&("load"===r.type?"missing":r.type),n=r&&r.target&&r.target.src;s.message="Loading chunk "+t+" failed.\n("+o+": "+n+")",s.name="ChunkLoadError",s.type=o,s.request=n,i[1](s)}}),"chunk-"+t,t)}};var t=(t,r)=>{var i,o,[n,s,c]=r,p=0;if(n.some((t=>0!==e[t]))){for(i in s)a.o(s,i)&&(a.m[i]=s[i]);if(c)c(a)}for(t&&t(r);p<n.length;p++)o=n[p],a.o(e,o)&&e[o]&&e[o][0](),e[o]=0},r=this.webpackChunk_tsparticles_plugin_emitters_shape_circle=this.webpackChunk_tsparticles_plugin_emitters_shape_circle||[];r.forEach(t.bind(null,0)),r.push=t.bind(null,r.push.bind(r))})();var s={};function c(e){e.checkVersion("4.0.0-alpha.2"),e.register((async e=>{const{EmittersCircleShapeGenerator:t}=await a.e(430).then(a.bind(a,430));e.addEmitterShapeGenerator?.("circle",new t)}))}return a.r(s),a.d(s,{loadEmittersShapeCircle:()=>c}),s})()));
1
+ !function(e,t){if("object"==typeof exports&&"object"==typeof module)module.exports=t(require("@tsparticles/plugin-emitters"),require("@tsparticles/engine"));else if("function"==typeof define&&define.amd)define(["@tsparticles/plugin-emitters","@tsparticles/engine"],t);else{var r="object"==typeof exports?t(require("@tsparticles/plugin-emitters"),require("@tsparticles/engine")):t(e.window,e.window);for(var i in r)("object"==typeof exports?exports:e)[i]=r[i]}}(this,(e,t)=>(()=>{"use strict";var r,i,o,n={303(e){e.exports=t},526(t){t.exports=e}},a={};function s(e){var t=a[e];if(void 0!==t)return t.exports;var r=a[e]={exports:{}};return n[e](r,r.exports,s),r.exports}s.m=n,p=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,s.t=function(e,t){if(1&t&&(e=this(e)),8&t||"object"==typeof e&&e&&(4&t&&e.__esModule||16&t&&"function"==typeof e.then))return e;var r=Object.create(null);s.r(r);var i={};c=c||[null,p({}),p([]),p(p)];for(var o=2&t&&e;("object"==typeof o||"function"==typeof o)&&!~c.indexOf(o);o=p(o))Object.getOwnPropertyNames(o).forEach(t=>i[t]=()=>e[t]);return i.default=()=>e,s.d(r,i),r},s.d=(e,t)=>{for(var r in t)s.o(t,r)&&!s.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},s.f={},s.e=e=>Promise.all(Object.keys(s.f).reduce((t,r)=>(s.f[r](e,t),t),[])),s.u=e=>""+e+".min.js",s.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||Function("return this")()}catch(e){if("object"==typeof window)return window}}(),s.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),l={},s.l=(e,t,r,i)=>{if(l[e])return void l[e].push(t);if(void 0!==r)for(var o,n,a=document.getElementsByTagName("script"),c=0;c<a.length;c++){var p=a[c];if(p.getAttribute("src")==e||p.getAttribute("data-webpack")=="@tsparticles/plugin-emitters-shape-circle:"+r){o=p;break}}o||(n=!0,(o=document.createElement("script")).charset="utf-8",s.nc&&o.setAttribute("nonce",s.nc),o.setAttribute("data-webpack","@tsparticles/plugin-emitters-shape-circle:"+r),o.src=e),l[e]=[t];var u=(t,r)=>{o.onerror=o.onload=null,clearTimeout(f);var i=l[e];if(delete l[e],o.parentNode&&o.parentNode.removeChild(o),i&&i.forEach(e=>e(r)),t)return t(r)},f=setTimeout(u.bind(null,void 0,{type:"timeout",target:o}),12e4);o.onerror=u.bind(null,o.onerror),o.onload=u.bind(null,o.onload),n&&document.head.appendChild(o)},s.r=e=>{"u">typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},s.g.importScripts&&(u=s.g.location+"");var c,p,l,u,f=s.g.document;if(!u&&f&&(f.currentScript&&"SCRIPT"===f.currentScript.tagName.toUpperCase()&&(u=f.currentScript.src),!u)){var d=f.getElementsByTagName("script");if(d.length)for(var h=d.length-1;h>-1&&(!u||!/^http(s?):/.test(u));)u=d[h--].src}if(!u)throw Error("Automatic publicPath is not supported in this browser");s.p=u=u.replace(/^blob:/,"").replace(/#.*$/,"").replace(/\?.*$/,"").replace(/\/[^\/]+$/,"/"),r={938:0},s.f.j=(e,t)=>{var i=s.o(r,e)?r[e]:void 0;if(0!==i)if(i)t.push(i[2]);else{var o=new Promise((t,o)=>i=r[e]=[t,o]);t.push(i[2]=o);var n=s.p+s.u(e),a=Error();s.l(n,t=>{if(s.o(r,e)&&(0!==(i=r[e])&&(r[e]=void 0),i)){var o=t&&("load"===t.type?"missing":t.type),n=t&&t.target&&t.target.src;a.message="Loading chunk "+e+` failed.
2
+ (`+o+": "+n+")",a.name="ChunkLoadError",a.type=o,a.request=n,i[1](a)}},"chunk-"+e,e)}},i=(e,t)=>{var i,o,[n,a,c]=t,p=0;if(n.some(e=>0!==r[e])){for(i in a)s.o(a,i)&&(s.m[i]=a[i]);c&&c(s)}for(e&&e(t);p<n.length;p++)o=n[p],s.o(r,o)&&r[o]&&r[o][0](),r[o]=0},(o=this.webpackChunk_tsparticles_plugin_emitters_shape_circle=this.webpackChunk_tsparticles_plugin_emitters_shape_circle||[]).forEach(i.bind(null,0)),o.push=i.bind(null,o.push.bind(o));var b={};async function m(e){e.checkVersion("4.0.0-alpha.20"),await e.register(async e=>{let{ensureEmittersPluginLoaded:t}=await Promise.resolve().then(s.t.bind(s,526,19));t(e);let{EmittersCircleShapeGenerator:r}=await s.e(154).then(s.bind(s,154));e.addEmitterShapeGenerator?.("circle",new r)})}return s.r(b),s.d(b,{loadEmittersShapeCircle:()=>m}),b})());
@@ -1,5 +1,5 @@
1
- import type { ICoordinates, IDimension } from "@tsparticles/engine";
1
+ import type { Container, ICoordinates, IDimension } from "@tsparticles/engine";
2
2
  import type { IEmitterShape, IEmitterShapeGenerator } from "@tsparticles/plugin-emitters";
3
3
  export declare class EmittersCircleShapeGenerator implements IEmitterShapeGenerator {
4
- generate(position: ICoordinates, size: IDimension, fill: boolean, options: unknown): IEmitterShape;
4
+ generate(_container: Container, position: ICoordinates, size: IDimension, fill: boolean, options: unknown): IEmitterShape;
5
5
  }
package/types/index.d.ts CHANGED
@@ -1,2 +1,2 @@
1
1
  import type { Engine } from "@tsparticles/engine";
2
- export declare function loadEmittersShapeCircle(engine: Engine): void;
2
+ export declare function loadEmittersShapeCircle(engine: Engine): Promise<void>;
@@ -12,7 +12,7 @@
12
12
  exports.EmittersCircleShape = void 0;
13
13
  const plugin_emitters_1 = require("@tsparticles/plugin-emitters");
14
14
  const engine_1 = require("@tsparticles/engine");
15
- const quarter = 0.25, double = 2, doublePI = Math.PI * double, squareExp = 2, half = 0.5;
15
+ const squareExp = 2;
16
16
  class EmittersCircleShape extends plugin_emitters_1.EmitterShapeBase {
17
17
  constructor(position, size, fill, options) {
18
18
  super(position, size, fill, options);
@@ -21,20 +21,20 @@
21
21
  }
22
22
  randomPosition() {
23
23
  const size = this.size, fill = this.fill, position = this.position, generateTheta = (x, y) => {
24
- const u = (0, engine_1.getRandom)() * quarter, theta = Math.atan((y / x) * Math.tan(doublePI * u)), v = (0, engine_1.getRandom)();
25
- if (v < quarter) {
24
+ const u = (0, engine_1.getRandom)() * engine_1.quarter, theta = Math.atan((y / x) * Math.tan(engine_1.doublePI * u)), v = (0, engine_1.getRandom)();
25
+ if (v < engine_1.quarter) {
26
26
  return theta;
27
27
  }
28
- else if (v < double * quarter) {
28
+ else if (v < engine_1.double * engine_1.quarter) {
29
29
  return Math.PI - theta;
30
30
  }
31
- else if (v < double * quarter + quarter) {
31
+ else if (v < engine_1.double * engine_1.quarter + engine_1.quarter) {
32
32
  return Math.PI + theta;
33
33
  }
34
34
  else {
35
35
  return -theta;
36
36
  }
37
- }, radius = (x, y, theta) => (x * y) / Math.sqrt((y * Math.cos(theta)) ** squareExp + (x * Math.sin(theta)) ** squareExp), [a, b] = [size.width * half, size.height * half], randomTheta = generateTheta(a, b), maxRadius = radius(a, b, randomTheta), randomRadius = fill ? maxRadius * Math.sqrt((0, engine_1.getRandom)()) : maxRadius;
37
+ }, radius = (x, y, theta) => (x * y) / Math.sqrt((y * Math.cos(theta)) ** squareExp + (x * Math.sin(theta)) ** squareExp), [a, b] = [size.width * engine_1.half, size.height * engine_1.half], randomTheta = generateTheta(a, b), maxRadius = radius(a, b, randomTheta), randomRadius = fill ? maxRadius * Math.sqrt((0, engine_1.getRandom)()) : maxRadius;
38
38
  return {
39
39
  position: {
40
40
  x: position.x + randomRadius * Math.cos(randomTheta),
@@ -12,7 +12,7 @@
12
12
  exports.EmittersCircleShapeGenerator = void 0;
13
13
  const EmittersCircleShape_js_1 = require("./EmittersCircleShape.js");
14
14
  class EmittersCircleShapeGenerator {
15
- generate(position, size, fill, options) {
15
+ generate(_container, position, size, fill, options) {
16
16
  return new EmittersCircleShape_js_1.EmittersCircleShape(position, size, fill, options);
17
17
  }
18
18
  }
package/umd/index.js CHANGED
@@ -44,11 +44,13 @@ var __importStar = (this && this.__importStar) || (function () {
44
44
  var __syncRequire = typeof module === "object" && typeof module.exports === "object";
45
45
  Object.defineProperty(exports, "__esModule", { value: true });
46
46
  exports.loadEmittersShapeCircle = loadEmittersShapeCircle;
47
- function loadEmittersShapeCircle(engine) {
48
- engine.checkVersion("4.0.0-alpha.2");
49
- engine.register(async (emittersEngine) => {
50
- const { EmittersCircleShapeGenerator } = await (__syncRequire ? Promise.resolve().then(() => __importStar(require("./EmittersCircleShapeGenerator.js"))) : new Promise((resolve_1, reject_1) => { require(["./EmittersCircleShapeGenerator.js"], resolve_1, reject_1); }).then(__importStar));
51
- emittersEngine.addEmitterShapeGenerator?.("circle", new EmittersCircleShapeGenerator());
47
+ async function loadEmittersShapeCircle(engine) {
48
+ engine.checkVersion("4.0.0-alpha.20");
49
+ await engine.register(async (e) => {
50
+ const { ensureEmittersPluginLoaded } = await (__syncRequire ? Promise.resolve().then(() => __importStar(require("@tsparticles/plugin-emitters"))) : new Promise((resolve_1, reject_1) => { require(["@tsparticles/plugin-emitters"], resolve_1, reject_1); }).then(__importStar));
51
+ ensureEmittersPluginLoaded(e);
52
+ const { EmittersCircleShapeGenerator } = await (__syncRequire ? Promise.resolve().then(() => __importStar(require("./EmittersCircleShapeGenerator.js"))) : new Promise((resolve_2, reject_2) => { require(["./EmittersCircleShapeGenerator.js"], resolve_2, reject_2); }).then(__importStar));
53
+ e.addEmitterShapeGenerator?.("circle", new EmittersCircleShapeGenerator());
52
54
  });
53
55
  }
54
56
  });
package/430.min.js DELETED
@@ -1,2 +0,0 @@
1
- /*! For license information please see 430.min.js.LICENSE.txt */
2
- (this.webpackChunk_tsparticles_plugin_emitters_shape_circle=this.webpackChunk_tsparticles_plugin_emitters_shape_circle||[]).push([[430],{430(t,s,e){e.d(s,{EmittersCircleShapeGenerator:()=>c});var a=e(526),i=e(303);const n=.25,h=2*Math.PI;class r extends a.EmitterShapeBase{constructor(t,s,e,a){super(t,s,e,a)}async init(){}randomPosition(){const t=this.size,s=this.fill,e=this.position,[a,r]=[.5*t.width,.5*t.height],c=((t,s)=>{const e=(0,i.getRandom)()*n,a=Math.atan(s/t*Math.tan(h*e)),r=(0,i.getRandom)();return r<n?a:r<.5?Math.PI-a:r<.75?Math.PI+a:-a})(a,r),o=(u=c,(l=a)*(M=r)/Math.sqrt((M*Math.cos(u))**2+(l*Math.sin(u))**2)),p=s?o*Math.sqrt((0,i.getRandom)()):o;var l,M,u;return{position:{x:e.x+p*Math.cos(c),y:e.y+p*Math.sin(c)}}}}class c{generate(t,s,e,a){return new r(t,s,e,a)}}}}]);
@@ -1 +0,0 @@
1
- /*! tsParticles Emitters Shape Circle Plugin v4.0.0-alpha.2 by Matteo Bruni */
@@ -1 +0,0 @@
1
- /*! tsParticles Emitters Shape Circle Plugin v4.0.0-alpha.2 by Matteo Bruni */