443 lines
14 KiB
JavaScript
443 lines
14 KiB
JavaScript
|
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.raterJs = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()({1:[function(require,module,exports){
|
||
|
"use strict";
|
||
|
|
||
|
/*! rater-js. [c] 2018 Fredrik Olsson. MIT License */
|
||
|
var css = require('./style.css');
|
||
|
|
||
|
module.exports = function (options) {
|
||
|
//private fields
|
||
|
var showToolTip = true;
|
||
|
|
||
|
if (typeof options.element === "undefined" || options.element === null) {
|
||
|
throw new Error("element required");
|
||
|
}
|
||
|
|
||
|
if (typeof options.showToolTip !== "undefined") {
|
||
|
showToolTip = !!options.showToolTip;
|
||
|
}
|
||
|
|
||
|
if (typeof options.step !== "undefined") {
|
||
|
if (options.step <= 0 || options.step > 1) {
|
||
|
throw new Error("step must be a number between 0 and 1");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var elem = options.element;
|
||
|
var reverse = options.reverse;
|
||
|
var stars = options.max || 5;
|
||
|
var starSize = options.starSize || 16;
|
||
|
var step = options.step || 1;
|
||
|
var onHover = options.onHover;
|
||
|
var onLeave = options.onLeave;
|
||
|
var rating = null;
|
||
|
var myRating;
|
||
|
elem.classList.add("star-rating");
|
||
|
var div = document.createElement("div");
|
||
|
div.classList.add("star-value");
|
||
|
|
||
|
if (reverse) {
|
||
|
div.classList.add("rtl");
|
||
|
}
|
||
|
|
||
|
div.style.backgroundSize = starSize + "px";
|
||
|
elem.appendChild(div);
|
||
|
elem.style.width = starSize * stars + "px";
|
||
|
elem.style.height = starSize + "px";
|
||
|
elem.style.backgroundSize = starSize + "px";
|
||
|
var callback = options.rateCallback;
|
||
|
var disabled = !!options.readOnly;
|
||
|
var disableText;
|
||
|
var isRating = false;
|
||
|
var isBusyText = options.isBusyText;
|
||
|
var currentRating;
|
||
|
var ratingText;
|
||
|
|
||
|
if (typeof options.disableText !== "undefined") {
|
||
|
disableText = options.disableText;
|
||
|
} else {
|
||
|
disableText = "{rating}/{maxRating}";
|
||
|
}
|
||
|
|
||
|
if (typeof options.ratingText !== "undefined") {
|
||
|
ratingText = options.ratingText;
|
||
|
} else {
|
||
|
ratingText = "{rating}/{maxRating}";
|
||
|
}
|
||
|
|
||
|
if (options.rating) {
|
||
|
setRating(options.rating);
|
||
|
} else {
|
||
|
var dataRating = elem.dataset.rating;
|
||
|
|
||
|
if (dataRating) {
|
||
|
setRating(+dataRating);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!rating) {
|
||
|
elem.querySelector(".star-value").style.width = "0px";
|
||
|
}
|
||
|
|
||
|
if (disabled) {
|
||
|
disable();
|
||
|
} //private methods
|
||
|
|
||
|
|
||
|
function onMouseMove(e) {
|
||
|
onMove(e, false);
|
||
|
}
|
||
|
/**
|
||
|
* Called by eventhandlers when mouse or touch events are triggered
|
||
|
* @param {MouseEvent} e
|
||
|
*/
|
||
|
|
||
|
|
||
|
function onMove(e, isTouch) {
|
||
|
if (disabled === true || isRating === true) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
var xCoor = null;
|
||
|
var percent;
|
||
|
var width = elem.offsetWidth;
|
||
|
var parentOffset = elem.getBoundingClientRect();
|
||
|
|
||
|
if (reverse) {
|
||
|
if (isTouch) {
|
||
|
xCoor = e.changedTouches[0].pageX - parentOffset.left;
|
||
|
} else {
|
||
|
xCoor = e.pageX - window.scrollX - parentOffset.left;
|
||
|
}
|
||
|
|
||
|
var relXRtl = width - xCoor;
|
||
|
var valueForDivision = width / 100;
|
||
|
percent = relXRtl / valueForDivision;
|
||
|
} else {
|
||
|
if (isTouch) {
|
||
|
xCoor = e.changedTouches[0].pageX - parentOffset.left;
|
||
|
} else {
|
||
|
xCoor = e.offsetX;
|
||
|
}
|
||
|
|
||
|
percent = xCoor / width * 100;
|
||
|
}
|
||
|
|
||
|
if (percent < 101) {
|
||
|
if (step === 1) {
|
||
|
currentRating = Math.ceil(percent / 100 * stars);
|
||
|
} else {
|
||
|
var rat = percent / 100 * stars;
|
||
|
|
||
|
for (var i = 0;; i += step) {
|
||
|
if (i >= rat) {
|
||
|
currentRating = i;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
} //todo: check why this happens and fix
|
||
|
|
||
|
|
||
|
if (currentRating > stars) {
|
||
|
currentRating = stars;
|
||
|
}
|
||
|
|
||
|
elem.querySelector(".star-value").style.width = currentRating / stars * 100 + "%";
|
||
|
|
||
|
if (showToolTip) {
|
||
|
var toolTip = ratingText.replace("{rating}", currentRating);
|
||
|
toolTip = toolTip.replace("{maxRating}", stars);
|
||
|
elem.setAttribute("title", toolTip);
|
||
|
}
|
||
|
|
||
|
if (typeof onHover === "function") {
|
||
|
onHover(currentRating, rating);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
/**
|
||
|
* Called when mouse is released. This function will update the view with the rating.
|
||
|
* @param {MouseEvent} e
|
||
|
*/
|
||
|
|
||
|
|
||
|
function onStarOut(e) {
|
||
|
if (!rating) {
|
||
|
elem.querySelector(".star-value").style.width = "0%";
|
||
|
elem.removeAttribute("data-rating");
|
||
|
} else {
|
||
|
elem.querySelector(".star-value").style.width = rating / stars * 100 + "%";
|
||
|
elem.setAttribute("data-rating", rating);
|
||
|
}
|
||
|
|
||
|
if (typeof onLeave === "function") {
|
||
|
onLeave(currentRating, rating);
|
||
|
}
|
||
|
}
|
||
|
/**
|
||
|
* Called when star is clicked.
|
||
|
* @param {MouseEvent} e
|
||
|
*/
|
||
|
|
||
|
|
||
|
function onStarClick(e) {
|
||
|
if (disabled === true) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (isRating === true) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (typeof callback !== "undefined") {
|
||
|
isRating = true;
|
||
|
myRating = currentRating;
|
||
|
|
||
|
if (typeof isBusyText === "undefined") {
|
||
|
elem.removeAttribute("title");
|
||
|
} else {
|
||
|
elem.setAttribute("title", isBusyText);
|
||
|
}
|
||
|
|
||
|
elem.classList.add("is-busy");
|
||
|
callback.call(this, myRating, function () {
|
||
|
if (disabled === false) {
|
||
|
elem.removeAttribute("title");
|
||
|
}
|
||
|
|
||
|
isRating = false;
|
||
|
elem.classList.remove("is-busy");
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
/**
|
||
|
* Disables the rater so that it's not possible to click the stars.
|
||
|
*/
|
||
|
|
||
|
|
||
|
function disable() {
|
||
|
disabled = true;
|
||
|
elem.classList.add("disabled");
|
||
|
|
||
|
if (showToolTip && !!disableText) {
|
||
|
var toolTip = disableText.replace("{rating}", !!rating ? rating : 0);
|
||
|
toolTip = toolTip.replace("{maxRating}", stars);
|
||
|
elem.setAttribute("title", toolTip);
|
||
|
} else {
|
||
|
elem.removeAttribute("title");
|
||
|
}
|
||
|
}
|
||
|
/**
|
||
|
* Enabled the rater so that it's possible to click the stars.
|
||
|
*/
|
||
|
|
||
|
|
||
|
function enable() {
|
||
|
disabled = false;
|
||
|
elem.removeAttribute("title");
|
||
|
elem.classList.remove("disabled");
|
||
|
}
|
||
|
/**
|
||
|
* Sets the rating
|
||
|
*/
|
||
|
|
||
|
|
||
|
function setRating(value) {
|
||
|
if (typeof value === "undefined") {
|
||
|
throw new Error("Value not set.");
|
||
|
}
|
||
|
|
||
|
if (value === null) {
|
||
|
throw new Error("Value cannot be null.");
|
||
|
}
|
||
|
|
||
|
if (typeof value !== "number") {
|
||
|
throw new Error("Value must be a number.");
|
||
|
}
|
||
|
|
||
|
if (value < 0 || value > stars) {
|
||
|
throw new Error("Value too high. Please set a rating of " + stars + " or below.");
|
||
|
}
|
||
|
|
||
|
rating = value;
|
||
|
elem.querySelector(".star-value").style.width = value / stars * 100 + "%";
|
||
|
elem.setAttribute("data-rating", value);
|
||
|
}
|
||
|
/**
|
||
|
* Gets the rating
|
||
|
*/
|
||
|
|
||
|
|
||
|
function getRating() {
|
||
|
return rating;
|
||
|
}
|
||
|
/**
|
||
|
* Set the rating to a value to inducate it's not rated.
|
||
|
*/
|
||
|
|
||
|
|
||
|
function clear() {
|
||
|
rating = null;
|
||
|
elem.querySelector(".star-value").style.width = "0px";
|
||
|
elem.removeAttribute("title");
|
||
|
}
|
||
|
/**
|
||
|
* Remove event handlers.
|
||
|
*/
|
||
|
|
||
|
|
||
|
function dispose() {
|
||
|
elem.removeEventListener("mousemove", onMouseMove);
|
||
|
elem.removeEventListener("mouseleave", onStarOut);
|
||
|
elem.removeEventListener("click", onStarClick);
|
||
|
elem.removeEventListener("touchmove", handleMove, false);
|
||
|
elem.removeEventListener("touchstart", handleStart, false);
|
||
|
elem.removeEventListener("touchend", handleEnd, false);
|
||
|
elem.removeEventListener("touchcancel", handleCancel, false);
|
||
|
}
|
||
|
|
||
|
elem.addEventListener("mousemove", onMouseMove);
|
||
|
elem.addEventListener("mouseleave", onStarOut);
|
||
|
var module = {
|
||
|
setRating: setRating,
|
||
|
getRating: getRating,
|
||
|
disable: disable,
|
||
|
enable: enable,
|
||
|
clear: clear,
|
||
|
dispose: dispose,
|
||
|
|
||
|
get element() {
|
||
|
return elem;
|
||
|
}
|
||
|
|
||
|
};
|
||
|
/**
|
||
|
* Handles touchmove event.
|
||
|
* @param {TouchEvent} e
|
||
|
*/
|
||
|
|
||
|
function handleMove(e) {
|
||
|
e.preventDefault();
|
||
|
onMove(e, true);
|
||
|
}
|
||
|
/**
|
||
|
* Handles touchstart event.
|
||
|
* @param {TouchEvent} e
|
||
|
*/
|
||
|
|
||
|
|
||
|
function handleStart(e) {
|
||
|
e.preventDefault();
|
||
|
onMove(e, true);
|
||
|
}
|
||
|
/**
|
||
|
* Handles touchend event.
|
||
|
* @param {TouchEvent} e
|
||
|
*/
|
||
|
|
||
|
|
||
|
function handleEnd(evt) {
|
||
|
evt.preventDefault();
|
||
|
onMove(evt, true);
|
||
|
onStarClick.call(module);
|
||
|
}
|
||
|
/**
|
||
|
* Handles touchend event.
|
||
|
* @param {TouchEvent} e
|
||
|
*/
|
||
|
|
||
|
|
||
|
function handleCancel(e) {
|
||
|
e.preventDefault();
|
||
|
onStarOut(e);
|
||
|
}
|
||
|
|
||
|
elem.addEventListener("click", onStarClick.bind(module));
|
||
|
elem.addEventListener("touchmove", handleMove, false);
|
||
|
elem.addEventListener("touchstart", handleStart, false);
|
||
|
elem.addEventListener("touchend", handleEnd, false);
|
||
|
elem.addEventListener("touchcancel", handleCancel, false);
|
||
|
return module;
|
||
|
};
|
||
|
|
||
|
},{"./style.css":2}],2:[function(require,module,exports){
|
||
|
var css = ".star-rating {\n width: 0;\n position: relative;\n display: inline-block;\n background-image: url();\n background-position: 0 0;\n background-repeat: repeat-x;\n cursor: pointer;\n}\n.star-rating .star-value {\n position: absolute;\n height: 100%;\n width: 100%;\n background: url('');\n background-repeat: repeat-x;\n}\n.star-rating.disabled {\n cursor: default;\n}\n.star-rating.is-busy {\n cursor: wait;\n}\n.star-rating .star-value.rtl {\n -moz-transform: scaleX(-1);\n -o-transform: scaleX(-1);\n -webkit-transform: scaleX(-1);\n transform: scaleX(-1);\n filter: FlipH;\n -ms-filter: \"FlipH\";\n right: 0;\n left: auto;\n}\n"; (require("browserify-css").createStyle(css, { "href": "lib\\style.css" }, { "insertAt": "bottom" })); module.exports = css;
|
||
|
},{"browserify-css":3}],3:[function(require,module,exports){
|
||
|
'use strict';
|
||
|
// For more information about browser field, check out the browser field at https://github.com/substack/browserify-handbook#browser-field.
|
||
|
|
||
|
var styleElementsInsertedAtTop = [];
|
||
|
|
||
|
var insertStyleElement = function(styleElement, options) {
|
||
|
var head = document.head || document.getElementsByTagName('head')[0];
|
||
|
var lastStyleElementInsertedAtTop = styleElementsInsertedAtTop[styleElementsInsertedAtTop.length - 1];
|
||
|
|
||
|
options = options || {};
|
||
|
options.insertAt = options.insertAt || 'bottom';
|
||
|
|
||
|
if (options.insertAt === 'top') {
|
||
|
if (!lastStyleElementInsertedAtTop) {
|
||
|
head.insertBefore(styleElement, head.firstChild);
|
||
|
} else if (lastStyleElementInsertedAtTop.nextSibling) {
|
||
|
head.insertBefore(styleElement, lastStyleElementInsertedAtTop.nextSibling);
|
||
|
} else {
|
||
|
head.appendChild(styleElement);
|
||
|
}
|
||
|
styleElementsInsertedAtTop.push(styleElement);
|
||
|
} else if (options.insertAt === 'bottom') {
|
||
|
head.appendChild(styleElement);
|
||
|
} else {
|
||
|
throw new Error('Invalid value for parameter \'insertAt\'. Must be \'top\' or \'bottom\'.');
|
||
|
}
|
||
|
};
|
||
|
|
||
|
module.exports = {
|
||
|
// Create a <link> tag with optional data attributes
|
||
|
createLink: function(href, attributes) {
|
||
|
var head = document.head || document.getElementsByTagName('head')[0];
|
||
|
var link = document.createElement('link');
|
||
|
|
||
|
link.href = href;
|
||
|
link.rel = 'stylesheet';
|
||
|
|
||
|
for (var key in attributes) {
|
||
|
if ( ! attributes.hasOwnProperty(key)) {
|
||
|
continue;
|
||
|
}
|
||
|
var value = attributes[key];
|
||
|
link.setAttribute('data-' + key, value);
|
||
|
}
|
||
|
|
||
|
head.appendChild(link);
|
||
|
},
|
||
|
// Create a <style> tag with optional data attributes
|
||
|
createStyle: function(cssText, attributes, extraOptions) {
|
||
|
extraOptions = extraOptions || {};
|
||
|
|
||
|
var style = document.createElement('style');
|
||
|
style.type = 'text/css';
|
||
|
|
||
|
for (var key in attributes) {
|
||
|
if ( ! attributes.hasOwnProperty(key)) {
|
||
|
continue;
|
||
|
}
|
||
|
var value = attributes[key];
|
||
|
style.setAttribute('data-' + key, value);
|
||
|
}
|
||
|
|
||
|
if (style.sheet) { // for jsdom and IE9+
|
||
|
style.innerHTML = cssText;
|
||
|
style.sheet.cssText = cssText;
|
||
|
insertStyleElement(style, { insertAt: extraOptions.insertAt });
|
||
|
} else if (style.styleSheet) { // for IE8 and below
|
||
|
insertStyleElement(style, { insertAt: extraOptions.insertAt });
|
||
|
style.styleSheet.cssText = cssText;
|
||
|
} else { // for Chrome, Firefox, and Safari
|
||
|
style.appendChild(document.createTextNode(cssText));
|
||
|
insertStyleElement(style, { insertAt: extraOptions.insertAt });
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
},{}]},{},[1])(1)
|
||
|
});
|