Check-Server-Status/gauge/jquery-asPieProgress.es.js
2023-11-17 12:07:46 +01:00

519 lines
12 KiB
JavaScript

/**
* jQuery asPieProgress v0.4.6
* https://github.com/amazingSurge/jquery-asPieProgress
*
* Copyright (c) amazingSurge
* Released under the LGPL-3.0 license
*/
import $ from 'jquery';
const SvgElement = (tag, attrs) => {
'use strict';
const elem = document.createElementNS('http://www.w3.org/2000/svg', tag);
if (!attrs) {
return elem;
}
for (let key in attrs) {
if (!Object.hasOwnProperty.call(attrs, key)) {
continue;
}
elem.setAttribute(key, attrs[key]);
}
return elem;
};
if (!Date.now) {
Date.now = () => {
'use strict';
return new Date().getTime();
};
}
const vendors = ['webkit', 'moz'];
for (let i = 0; i < vendors.length && !window.requestAnimationFrame; ++i) {
const vp = vendors[i];
window.requestAnimationFrame = window[`${vp}RequestAnimationFrame`];
window.cancelAnimationFrame = (window[`${vp}CancelAnimationFrame`] || window[`${vp}CancelRequestAnimationFrame`]);
}
if (/iP(ad|hone|od).*OS (6|7|8)/.test(window.navigator.userAgent) // iOS6 is buggy
||
!window.requestAnimationFrame || !window.cancelAnimationFrame) {
let lastTime = 0;
window.requestAnimationFrame = callback => {
'use strict';
const now = getTime();
const nextTime = Math.max(lastTime + 16, now);
return setTimeout(() => {
callback(lastTime = nextTime);
},
nextTime - now);
};
window.cancelAnimationFrame = clearTimeout;
}
const getTime = () => {
if (typeof window.performance !== 'undefined' && window.performance.now) {
return window.performance.now();
}
return Date.now();
};
const isPercentage = (n) => {
'use strict';
return typeof n === 'string' && n.indexOf('%') !== -1;
};
const svgSupported = 'createElementNS' in document && new SvgElement('svg', {}).createSVGRect;
const easingBezier = (mX1, mY1, mX2, mY2) => {
'use strict';
const a = (aA1, aA2) => {
return 1.0 - 3.0 * aA2 + 3.0 * aA1;
};
const b = (aA1, aA2) => {
return 3.0 * aA2 - 6.0 * aA1;
};
const c = (aA1) => {
return 3.0 * aA1;
};
// Returns x(t) given t, x1, and x2, or y(t) given t, y1, and y2.
const calcBezier = (aT, aA1, aA2) => {
return ((a(aA1, aA2) * aT + b(aA1, aA2)) * aT + c(aA1)) * aT;
};
// Returns dx/dt given t, x1, and x2, or dy/dt given t, y1, and y2.
const getSlope = (aT, aA1, aA2) => {
return 3.0 * a(aA1, aA2) * aT * aT + 2.0 * b(aA1, aA2) * aT + c(aA1);
};
const getTForX = (aX) => {
// Newton raphson iteration
let aGuessT = aX;
for (let i = 0; i < 4; ++i) {
let currentSlope = getSlope(aGuessT, mX1, mX2);
if (currentSlope === 0.0) {
return aGuessT;
}
let currentX = calcBezier(aGuessT, mX1, mX2) - aX;
aGuessT -= currentX / currentSlope;
}
return aGuessT;
};
if (mX1 === mY1 && mX2 === mY2) {
return {
css: 'linear',
fn(aX) {
return aX;
}
};
}
return {
css: `cubic-bezier(${mX1},${mY1},${mX2},${mY2})`,
fn(aX) {
return calcBezier(getTForX(aX), mY1, mY2);
}
};
};
var EASING = {
ease: easingBezier(0.25, 0.1, 0.25, 1.0),
linear: easingBezier(0.00, 0.0, 1.00, 1.0),
'ease-in': easingBezier(0.42, 0.0, 1.00, 1.0),
'ease-out': easingBezier(0.00, 0.0, 0.58, 1.0),
'ease-in-out': easingBezier(0.42, 0.0, 0.58, 1.0)
};
var DEFAULTS = {
namespace: 'asPieProgress',
classes: {
svg: 'pie_progress__svg',
element: 'pie_progress',
number: 'pie_progress__number',
content: 'pie_progress__content'
},
min: 0,
max: 100,
goal: 100,
size: 160,
speed: 15, // speed of 1/100
barcolor: '#ef1e25',
barsize: '4',
trackcolor: '#f2f2f2',
fillcolor: 'none',
easing: 'ease',
numberCallback(n) {
'use strict';
const percentage = Math.round(this.getPercentage(n));
return `${percentage}%`;
},
contentCallback: null
};
const NAMESPACE$1 = 'asPieProgress';
class asPieProgress {
constructor(element, options) {
this.element = element;
this.$element = $(element);
this.options = $.extend(true, {}, DEFAULTS, options, this.$element.data());
this.namespace = this.options.namespace;
this.classes = this.options.classes;
this.easing = EASING[this.options.easing] || EASING.ease;
this.$element.addClass(this.classes.element);
this.min = this.$element.attr('aria-valuemin');
this.max = this.$element.attr('aria-valuemax');
this.min = this.min ? parseInt(this.min, 10) : this.options.min;
this.max = this.max ? parseInt(this.max, 10) : this.options.max;
this.first = this.$element.attr('aria-valuenow');
this.first = this.first ? parseInt(this.first, 10) : (this.options.first ? this.options.first : this.min);
this.now = this.first;
this.goal = this.options.goal;
this._frameId = null;
this.initialized = false;
this._trigger('init');
this.init();
}
init() {
this.$number = this.$element.find(`.${this.classes.number}`);
this.$content = this.$element.find(`.${this.classes.content}`);
this.size = this.options.size;
this.width = this.size;
this.height = this.size;
this.prepare();
this.initialized = true;
this._trigger('ready');
}
prepare() {
if (!svgSupported) {
return;
}
this.svg = new SvgElement('svg', {
version: '1.1',
preserveAspectRatio: 'xMinYMin meet',
viewBox: `0 0 ${this.width} ${this.height}`
});
this.buildTrack();
this.buildBar();
$(`<div class="${this.classes.svg}"></div>`).append(this.svg).appendTo(this.$element);
}
buildTrack() {
const height = this.size,
width = this.size;
const cx = width / 2,
cy = height / 2;
const barsize = this.options.barsize;
const ellipse = new SvgElement('ellipse', {
rx: cx - barsize / 2,
ry: cy - barsize / 2,
cx,
cy,
stroke: this.options.trackcolor,
fill: this.options.fillcolor,
'stroke-width': barsize
});
this.svg.appendChild(ellipse);
}
buildBar() {
if (!svgSupported) {
return;
}
const path = new SvgElement('path', {
fill: 'none',
'stroke-width': this.options.barsize,
stroke: this.options.barcolor
});
this.bar = path;
this.svg.appendChild(path);
this._drawBar(this.first);
this._updateBar();
}
_drawBar(n) {
if (!svgSupported) {
return;
}
this.barGoal = n;
const height = this.size,
width = this.size;
const cx = width / 2,
cy = height / 2,
startAngle = 0;
const barsize = this.options.barsize;
const r = Math.min(cx, cy) - barsize / 2;
this.r = r;
let percentage = this.getPercentage(n);
if (percentage === 100) {
percentage -= 0.0001;
}
const endAngle = startAngle + percentage * Math.PI * 2 / 100;
const x1 = cx + r * Math.sin(startAngle),
x2 = cx + r * Math.sin(endAngle),
y1 = cy - r * Math.cos(startAngle),
y2 = cy - r * Math.cos(endAngle);
// This is a flag for angles larger than than a half circle
// It is required by the SVG arc drawing component
let big = 0;
if (endAngle - startAngle > Math.PI) {
big = 1;
}
// This string holds the path details
const d = `M${x1},${y1} A${r},${r} 0 ${big} 1 ${x2},${y2}`;
this.bar.setAttribute('d', d);
}
_updateBar() {
if (!svgSupported) {
return;
}
const percenage = this.getPercentage(this.now);
const length = this.bar.getTotalLength();
const offset = length * (1 - percenage / this.getPercentage(this.barGoal));
this.bar.style.strokeDasharray = `${length} ${length}`;
this.bar.style.strokeDashoffset = offset;
}
_trigger(eventType, ...params) {
const data = [this].concat(params);
// event
this.$element.trigger(`${NAMESPACE$1}::${eventType}`, data);
// callback
eventType = eventType.replace(/\b\w+\b/g, (word) => {
return word.substring(0, 1).toUpperCase() + word.substring(1);
});
const onFunction = `on${eventType}`;
if (typeof this.options[onFunction] === 'function') {
this.options[onFunction].apply(this, params);
}
}
// Return the percentage based on the current step
getPercentage(n) {
return 100 * (n - this.min) / (this.max - this.min);
}
go(goal) {
const that = this;
this._clear();
if (isPercentage(goal)) {
goal = parseInt(goal.replace('%', ''), 10);
goal = Math.round(this.min + (goal / 100) * (this.max - this.min));
}
if (typeof goal === 'undefined') {
goal = this.goal;
}
if (goal > this.max) {
goal = this.max;
} else if (goal < this.min) {
goal = this.min;
}
if (this.barGoal < goal) {
this._drawBar(goal);
}
const start = that.now;
const startTime = getTime();
const endTime = startTime + Math.abs(start - goal) * 100 * that.options.speed / (that.max - that.min);
const animation = time => {
let next;
if (time > endTime) {
next = goal;
} else {
const distance = (time - startTime) / that.options.speed;
next = Math.round(that.easing.fn(distance / 100) * (that.max - that.min));
if (goal > start) {
next = start + next;
if (next > goal) {
next = goal;
}
} else {
next = start - next;
if (next < goal) {
next = goal;
}
}
}
that._update(next);
if (next === goal) {
window.cancelAnimationFrame(that._frameId);
that._frameId = null;
if (that.now === that.goal) {
that._trigger('finish');
}
} else {
that._frameId = window.requestAnimationFrame(animation);
}
};
that._frameId = window.requestAnimationFrame(animation);
}
_update(n) {
this.now = n;
this._updateBar();
this.$element.attr('aria-valuenow', this.now);
if (this.$number.length > 0 && typeof this.options.numberCallback === 'function') {
this.$number.html(this.options.numberCallback.call(this, [this.now]));
}
if (this.$content.length > 0 && typeof this.options.contentCallback === 'function') {
this.$content.html(this.options.contentCallback.call(this, [this.now]));
}
this._trigger('update', n);
}
_clear() {
if (this._frameId) {
window.cancelAnimationFrame(this._frameId);
this._frameId = null;
}
}
get() {
return this.now;
}
start() {
this._clear();
this._trigger('start');
this.go(this.goal);
}
reset() {
this._clear();
this._drawBar(this.first);
this._update(this.first);
this._trigger('reset');
}
stop() {
this._clear();
this._trigger('stop');
}
finish() {
this._clear();
this._update(this.goal);
this._trigger('finish');
}
destroy() {
this.$element.data(NAMESPACE$1, null);
this._trigger('destroy');
}
static registerEasing(name, ...args) {
EASING[name] = easingBezier(...args);
}
static getEasing(name) {
return EASING[name];
}
static setDefaults(options) {
$.extend(true, DEFAULTS, $.isPlainObject(options) && options);
}
}
var info = {
version:'0.4.6'
};
const NAMESPACE = 'asPieProgress';
const OtherAsPieProgress = $.fn.asPieProgress;
const jQueryAsPieProgress = function(options, ...args) {
if (typeof options === 'string') {
const method = options;
if (/^_/.test(method)) {
return false;
} else if ((/^(get)/.test(method))) {
const instance = this.first().data(NAMESPACE);
if (instance && typeof instance[method] === 'function') {
return instance[method](...args);
}
} else {
return this.each(function() {
const instance = $.data(this, NAMESPACE);
if (instance && typeof instance[method] === 'function') {
instance[method](...args);
}
});
}
}
return this.each(function() {
if (!$(this).data(NAMESPACE)) {
$(this).data(NAMESPACE, new asPieProgress(this, options));
}
});
};
$.fn.asPieProgress = jQueryAsPieProgress;
$.asPieProgress = $.extend({
setDefaults: asPieProgress.setDefaults,
registerEasing: asPieProgress.registerEasing,
getEasing: asPieProgress.getEasing,
noConflict: function() {
$.fn.asPieProgress = OtherAsPieProgress;
return jQueryAsPieProgress;
}
}, info);