123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542 |
- /**
- * FeHelper Page Color Picker Tools
- */
- window.colorpickerContentScript = function () {
- let FeHelper = window.FeHelper || {};
- FeHelper.elemTool = {
- elm: function (nodeType, attributes, addchilds, appnedTo) {
- var ne = document.createElement(nodeType), i, l;
- if (attributes) {
- if (attributes.event || attributes.events) {
- var lev = attributes.event || attributes.events;
- if (typeof(lev[0]) == 'string') ne.addEventListener(lev[0], lev[1], lev[2]);
- else if (lev.length)
- for (i = 0, l = lev.length; i < l; i++)
- ne.addEventListener(lev[i][0], lev[i][1], lev[i][2]);
- }
- }
- for (i in attributes) {
- if (i.substring(0, 5) == 'event') {
- //handled earlier
- } else if (i == 'checked' || i == 'selected') {
- if (attributes[i]) ne.setAttribute(i, i);
- } else ne.setAttribute(i, attributes[i]);
- }
- if (addchilds) {
- for (i = 0, l = addchilds.length; i < l; i++) {
- if (addchilds[i]) ne.appendChild(addchilds[i]);//you probably forgot a comma when calling the function
- }
- }
- if (appnedTo) {
- this.insertNode(ne, appnedTo);
- }
- return ne;
- },
- /*elemTool.txt creates text nodes, does not support HTML entiteis */
- txt: function (textContent) {
- return document.createTextNode(textContent);
- },
- /*elemTool.ent creates text nodes that may or may not contain HTML entities. From a
- single entity to many entities interspersed with text are all supported by this */
- ent: function (textContent) {
- return document.createTextNode(this.unescapeHtml(textContent));
- },
- /*elemTool.paragraphs creates an array of nodes that may or may not contain HTML entities.*/
- paragraphs: function (textContent) {
- var textPieces = textContent.split("\n");
- var elmArray = [];
- for (var i = 0, l = textPieces.length; i < l; i++) {
- elmArray.push(elemTool.elm('p', {}, [elemTool.ent(textPieces[i])]));
- }
- return elmArray;
- },
- insertNode: function (newNode, parentElem, optionalInsertBefore) {
- if (!parentElem) parentElem = document.body;
- if (optionalInsertBefore && optionalInsertBefore.parentNode == parentElem) {
- parentElem.insertBefore(newNode, optionalInsertBefore);
- } else {
- parentElem.appendChild(newNode);
- }
- },
- insertNodes: function (newNodes, parentElem, optionalInsertBefore) {
- if (typeof(newNodes) != 'array')
- this.insertNode(newNodes, parentElem, optionalInsertBefore);
- else {
- for (var i = 0, l = newNodes.length; i < l; i++) {
- this.insertNode(newNodes[i], parentElem, optionalInsertBefore, true);
- }
- }
- },
- empty: function (node) {
- while (node.lastChild) node.removeChild(node.lastChild);
- },
- unescapeHtml: function (str) { //trick used to make HTMLentiites work inside textNodes
- if (str.length < 1) return str;
- var temp = document.createElement("div");
- str = str.replace(/<\/?\w(?:[^"'>]|"[^"]*"|'[^']*')*>/gmi, '');
- temp.innerHTML = str;
- var result = temp.childNodes[0].nodeValue;
- this.empty(temp);
- return result;
- }
- };
- /**
- * 页面取色器
- */
- FeHelper.ColorPicker = (function () {
- if (!(document.documentElement instanceof HTMLElement)) {
- return;
- }
- var elmid1 = 'fehelper-colorpicker-box', elmid2 = 'fehelper-colorpicker-result';
- function _ge(n) {
- return document.getElementById(n);
- }
- var n = false, c = false, hex = 'F00BAF', lasthex = '', rgb = null;
- var hsv = null;
- var ex = 0, ey = 0, isEnabled = false, isLocked = false, hexIsLowerCase = false, borderValue = '1px solid #666',
- blankgif = '';
- var isUpdating = false, lastTimeout = 0, lx = 0, ly = 0;
- var cvs = document.createElement('canvas');
- var ctx = cvs.getContext('2d'), x_cvs_scale = 1, y_cvs_scale = 1;
- function RGBtoHex(R, G, B) {
- return applyHexCase(toHex(R) + toHex(G) + toHex(B))
- }
- function applyHexCase(hex) {
- return hexIsLowerCase ? hex.toLowerCase() : hex;
- }
- function toHex(N) {//http://www.javascripter.net/faq/rgbtohex.htm
- if (N == null) return "00";
- N = parseInt(N);
- if (N == 0 || isNaN(N)) return "00";
- N = Math.max(0, N);
- N = Math.min(N, 255);
- N = Math.round(N);
- return "0123456789ABCDEF".charAt((N - N % 16) / 16) + "0123456789ABCDEF".charAt(N % 16);
- }
- function rgb2hsl(r, g, b) {//http://mjijackson.com/2008/02/rgb-to-hsl-and-rgb-to-hsv-color-model-conversion-algorithms-in-javascript
- r /= 255, g /= 255, b /= 255;
- var max = Math.max(r, g, b), min = Math.min(r, g, b);
- var h, s, l = (max + min) / 2;
- if (max == min) {
- h = s = 0; // achromatic
- } else {
- var d = max - min;
- s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
- switch (max) {
- case r:
- h = (g - b) / d + (g < b ? 6 : 0);
- break;
- case g:
- h = (b - r) / d + 2;
- break;
- case b:
- h = (r - g) / d + 4;
- break;
- }
- h /= 6;
- }
- return {
- h: Math.round(h * 360),
- s: Math.round(s * 100),
- v: Math.round(l * 100)
- };
- }
- function emptyNode(node) {
- while (node.lastChild) node.removeChild(node.lastChild);
- }
- function snapshotLoaded() {
- c.style.height = 'auto';
- c.style.width = (innerWidth) + 'px';
- x_cvs_scale = c.naturalWidth / innerWidth;
- y_cvs_scale = c.naturalHeight / innerHeight;
- cvs.width = c.naturalWidth;
- cvs.height = c.naturalHeight;
- ctx.drawImage(c, 0, 0);
- setTimeout(function () {
- isMakingNew = false;
- c.style.visibility = "visible";
- n.style.visibility = "visible";
- document.body.style.cursor = 'url() 16 16,crosshair';
- updateColorPreview();
- }, 255);
- }
- function setPixelPreview(pix, hxe, lhex) {
- if (isLocked) return;
- var wid = 75, padr = 32;
- wid = 150;
- hex = hxe ? hxe : hex;
- if (!_ge('fehelper-colorpicker-cpimprev') || (rgb && !_ge('cprgbvl'))) {
- emptyNode(n);
- FeHelper.elemTool.elm('div', {}, [
- FeHelper.elemTool.elm('img', {
- id: 'fehelper-colorpicker-cpimprev',
- height: wid,
- width: wid,
- src: pix,
- style: 'margin:0px;padding:0px;margin:0px;'
- }),
- FeHelper.elemTool.elm('br'),
- FeHelper.elemTool.elm('input', {
- type: 'text',
- size: 7,
- style: 'width:60px;height:20px;line-height:20px;font-size:10pt;border:' + borderValue,
- id: 'fehelper-colorpicker-cphexvl',
- value: '#' + hex,
- event: ['mouseover', selectTargElm]
- })
- ], n)
- keepOnScreen();
- } else {
- _ge('fehelper-colorpicker-cpimprev').src = pix;
- _ge('fehelper-colorpicker-cpimprev').width = wid;
- _ge('fehelper-colorpicker-cpimprev').height = wid;
- _ge('fehelper-colorpicker-cphexvl').value = hex;
- n.style.backgroundColor = '#' + hex;
- }
- }
- function setCurColor(r) {
- if (!n) return;
- hex = r.hex ? r.hex : hex;
- n.style.backgroundColor = '#' + hex;
- if (isLocked) setDisplay();
- }
- function selectTargElm(ev) {
- ev.target.select();
- }
- function setDisplay() {//FeHelper.elemTool.elm
- emptyNode(n);
- FeHelper.elemTool.elm('div', {}, [
- FeHelper.elemTool.elm('input', {
- type: 'text',
- size: 7,
- style: 'width:80px;height:20px;line-height:20px;font-size:10pt;border:' + borderValue,
- id: 'fehelper-colorpicker-cphexvl',
- value: '#' + hex,
- event: ['mouseover', selectTargElm]
- }),
- FeHelper.elemTool.elm('img', {
- style: 'width:20px;height:20px;position:absolute;top:-10px;right:-10px;cursor:pointer;',
- src: '',
- alt: 'Close',
- title: '[esc]键可直接关闭',
- id: 'fehelper-colorpicker-exitbtn',
- event: ['click', dissableColorPickerFromHere, true]
- })
- ], n);
- if (_ge('fehelper-colorpicker-cphexvl')) _ge('fehelper-colorpicker-cphexvl').select();
- keepOnScreen();
- }
- function picked() {
- if (isLocked) {
- lasthex = hex;
- isLocked = false;
- emptyNode(n);
- } else {
- isLocked = true;
- setDisplay();
- }
- }
- function dissableColorPickerFromHere() {
- setTimeout(disableColorPicker, 500)
- }
- function disableColorPicker() {
- isEnabled = false, isLocked = false;
- document.removeEventListener('mousemove', mmf);
- removeEventListener('scroll', ssf);
- removeEventListener('resize', ssf);
- removeEventListener('keyup', wk);
- removeExistingNodes();
- clearTimeout(lastNewTimeout);
- }
- function removeExistingNodes() {
- if (document.body) {
- c = _ge(elmid1), n = _ge(elmid2);
- if (c) document.body.removeChild(c);
- if (n) document.body.removeChild(n);
- c = false, n = false;
- document.body.style.cursor = '';
- }
- }
- function wk(ev) {
- if (!isEnabled) return;
- if (ev.keyCode == 27) {
- dissableColorPickerFromHere();
- } else if (ev.keyCode == 82 || ev.keyCode == 74) {//r or j refresh
- ssf();
- } else if (ev.keyCode == 13) {
- picked();
- }
- }
- function mmf(ev) {
- if (!isEnabled) return;
- if (!isLocked) {
- lx = (ev.pageX - pageXOffset), ly = (ev.pageY - pageYOffset);
- ex = Math.round(lx * x_cvs_scale),
- ey = Math.round(ly * y_cvs_scale);
- updateColorPreview();
- }
- }
- function ssf(ev) {
- if (!isEnabled) return;
- n.style.visibility = "hidden";
- c.style.visibility = "hidden";//redundent?
- clearTimeout(lastNewTimeout);
- lastNewTimeout = setTimeout(function () {
- newImage()//some delay required OR it won't update
- }, 250);
- }
- function initialInit() {
- removeExistingNodes();
- c = FeHelper.elemTool.elm('img', {
- id: elmid1,
- src: blankgif,
- style: 'position:fixed;max-width:none!important;max-height:none!important;top:0px;left:0px;margin:0px;padding:0px;overflow:hidden;z-index:2147483646;',
- events: [['click', picked, true], ['load', snapshotLoaded]]
- }, [], document.body);
- n = FeHelper.elemTool.elm('div', {
- id: elmid2,
- style: 'position:fixed;min-width:30px;max-width:300px;box-shadow:2px 2px 2px #666;border:' + borderValue + ';border-radius:5px;z-index:2147483646;cursor:default;padding:10px;text-align:center;'
- }, [], document.body);
- document.addEventListener('mousemove', mmf);
- addEventListener('keyup', wk);
- addEventListener('scroll', ssf);
- addEventListener('resize', ssf);
- initializeCanvas();
- remainingInit();
- }
- function enableColorPicker() {
- disableColorPicker();
- if (!n) {
- initialInit();
- return false;
- }
- return remainingInit();
- }
- function remainingInit() {
- if (!isEnabled) {
- n.style.visibility = "hidden";
- c.style.visibility = "hidden";
- if (isLocked) picked();//unlocks for next pick
- document.body.style.cursor = 'url() 16 16,crosshair';
- isEnabled = true;
- setTimeout(newImage, 1);
- return false;
- }
- return true;
- }
- function keepOnScreen() {
- if (!n) return;
- n.style.top = (ly + 8) + "px";
- n.style.left = (lx + 8) + "px";
- if (n.clientWidth + n.offsetLeft + 24 > innerWidth) {
- n.style.left = (lx - 8 - n.clientWidth) + "px";
- }
- if (n.clientHeight + n.offsetTop + 24 > innerHeight) {
- n.style.top = (ly - 8 - n.clientHeight) + "px";
- }
- }
- function updateColorPreview(ev) {
- if (!isEnabled) return;
- keepOnScreen();
- var data = ctx.getImageData(ex, ey, 1, 1).data;
- hsv = rgb2hsl(data[0], data[1], data[2]);
- rgb = {r: data[0], g: data[1], b: data[2]};
- setCurColor({hex: RGBtoHex(data[0], data[1], data[2])});
- handleRendering();
- }
- var isMakingNew = false, lastNewTimeout = 0;
- function newImage() {
- if (!isEnabled) return;
- if (isMakingNew) {
- clearTimeout(lastNewTimeout);
- lastNewTimeout = setTimeout(function () {
- newImage()
- }, 255);
- return;
- }
- document.body.style.cursor = 'wait';
- isMakingNew = true;
- n.style.visibility = "hidden";
- c.style.visibility = "hidden";
- c.src = blankgif;
- var x = innerWidth, y = innerHeight;
- c.style.width = x + 'px';
- c.style.height = y + 'px';
- setTimeout(function () {
- try {
- chrome.runtime.sendMessage({
- type: 'fh-dynamic-any-thing',
- thing:'color-picker-capture',
- params: {
- url: location.href
- }
- });
- } catch (e) {
- console.log('有错误发生,可提交此反馈到官网!', e);
- }
- }, 255);
- }
- var lastPreviewURI;
- var icvs = 0, totalWidth = 150;//750
- function handleRendering(quick) {
- var x = ex, y = ey;
- if (isMakingNew) {
- isUpdating = false;
- return;
- }
- var startPoint = Math.floor(totalWidth * 0.5);
- var ox = Math.round(x), oy = Math.round(y);
- if (quick) {
- var ictx = getMain2dContext();
- ictx.scale(2, 2);
- ictx.drawImage(cvs, -ox + (startPoint * 0.5), -oy + (startPoint * 0.5));
- ictx.scale(0.5, 0.5);
- ictx.fillStyle = "rgba(0,0,0,0.3)";//croshair
- ictx.fillRect(startPoint, 0, 1, totalWidth);
- ictx.fillRect(0, startPoint, totalWidth, 1);
- } else {
- var ictx = getMain2dContext();
- ictx.drawImage(cvs, -ox + (startPoint), -oy + (startPoint));
- var smi, spi, mp = 15 - 0;
- //xx,yy
- for (var i = 0; i < startPoint; i += 2) {
- smi = startPoint - i;
- spi = startPoint + i;
- //drawImage(image, sx, sy, sWidth, sHeight, dx, dy, dWidth, dHeight) //CANVAS
- ictx.drawImage(icvs, spi, 0, smi, totalWidth,//total width really??
- spi + 1, 0, smi, totalWidth);
- ictx.drawImage(icvs, 0, 0, smi + 1, totalWidth,
- -1, 0, smi + 1, totalWidth);
- ictx.drawImage(icvs, 0, spi, totalWidth, smi,
- 0, spi + 1, totalWidth, smi);
- ictx.drawImage(icvs, 0, 0, totalWidth, smi + 1,
- 0, -1, totalWidth, smi + 1);
- if (i == 0) {
- var dat = ictx.getImageData(startPoint, startPoint, 1, 1).data;//notarget
- var d = dat[0] + dat[1] + dat[2];
- if (d > 192) ictx.fillStyle = "rgba(30,30,30,0.8)";
- else ictx.fillStyle = "rgba(225,225,225,0.8)";
- } else ictx.fillStyle = "rgba(255,255,255,0.4)";
- for (var c = 0; c < mp; c++) {
- if (++i >= startPoint) break;
- smi = startPoint - i;
- spi = startPoint + i;
- ictx.drawImage(icvs, spi, 0, smi, totalWidth,
- spi + 1, 0, smi, totalWidth);
- ictx.drawImage(icvs, 0, 0, smi + 1, totalWidth,
- -1, 0, smi + 1, totalWidth);
- ictx.drawImage(icvs, 0, spi, totalWidth, smi,
- 0, spi + 1, totalWidth, smi);
- ictx.drawImage(icvs, 0, 0, totalWidth, smi + 1,
- 0, -1, totalWidth, smi + 1);
- }
- mp--;
- if (mp < 1) mp = 1;
- ictx.fillRect(spi + 1, 0, 1, totalWidth);
- ictx.fillRect(smi - 1, 0, 1, totalWidth);
- ictx.fillRect(0, spi + 1, totalWidth, 1);
- ictx.fillRect(0, smi - 1, totalWidth, 1);
- }
- }
- lastPreviewURI = icvs.toDataURL();//the last one, large size, is cached for revisiting the menu
- var browseIconWidth = (devicePixelRatio > 1 ? 38 : 19);
- var browseIconHalfWidth = Math.floor(browseIconWidth * 0.5);
- var tmpCvs = document.createElement('canvas');
- tmpCvs.width = browseIconWidth, tmpCvs.height = browseIconWidth;
- var tctx = tmpCvs.getContext("2d");
- tctx.drawImage(icvs, startPoint - browseIconHalfWidth, startPoint - browseIconHalfWidth, browseIconWidth, browseIconWidth, 0, 0, browseIconWidth, browseIconWidth);
- var pathData = {};
- pathData[browseIconWidth] = tmpCvs.toDataURL();
- setPixelPreview(lastPreviewURI, hex, lasthex);
- isUpdating = false;
- }
- function getMain2dContext() {
- var context = icvs.getContext("2d");
- if (context) return context;
- else {
- initializeCanvas();
- return icvs.getContext("2d");
- }
- }
- function initializeCanvas() {
- icvs = document.createElement('canvas');//icon canvas
- icvs.width = totalWidth;
- icvs.height = totalWidth;
- }
- return function (request) {
- if (request.setPickerImage) {
- c.src = request.pickerImage;
- } else {
- enableColorPicker();
- picked();
- }
- };
- })();
- // 给background page直接调用的
- window.colorpickerNoPage = function (request) {
- FeHelper.ColorPicker(request)
- };
- };
|