ক্যানভাস থেকে পিক্সেল রঙ পান, মাউসমেভে


85

মাউসের নীচে আরজিবি মান পিক্সেল পাওয়া কি সম্ভব? এর পূর্ণ উদাহরণ কি আছে? আমার এখন পর্যন্ত যা আছে তা এখানে:

function draw() {
      var ctx = document.getElementById('canvas').getContext('2d');
      var img = new Image();
      img.src = 'Your URL';

      img.onload = function(){
        ctx.drawImage(img,0,0);


      };

      canvas.onmousemove = function(e) {
            var mouseX, mouseY;

            if(e.offsetX) {
                mouseX = e.offsetX;
                mouseY = e.offsetY;
            }
            else if(e.layerX) {
                mouseX = e.layerX;
                mouseY = e.layerY;
            }
            var c = ctx.getImageData(mouseX, mouseY, 1, 1).data;
            
            $('#ttip').css({'left':mouseX+20, 'top':mouseY+20}).html(c[0]+'-'+c[1]+'-'+c[2]);
      };
    }

উত্তর:


150

এখানে একটি সম্পূর্ণ, স্বয়ংসম্পূর্ণ উদাহরণ's প্রথমে নিম্নলিখিত HTML ব্যবহার করুন:

<canvas id="example" width="200" height="60"></canvas>
<div id="status"></div>

তারপরে এলোমেলো ব্যাকগ্রাউন্ড রঙ সহ ক্যানভাসে কিছু স্কোয়ার রাখুন:

var example = document.getElementById('example');
var context = example.getContext('2d');
context.fillStyle = randomColor();
context.fillRect(0, 0, 50, 50);
context.fillStyle = randomColor();
context.fillRect(55, 0, 50, 50);
context.fillStyle = randomColor();
context.fillRect(110, 0, 50, 50);

এবং মাউসওভারে প্রতিটি রঙ মুদ্রণ করুন:

$('#example').mousemove(function(e) {
    var pos = findPos(this);
    var x = e.pageX - pos.x;
    var y = e.pageY - pos.y;
    var coord = "x=" + x + ", y=" + y;
    var c = this.getContext('2d');
    var p = c.getImageData(x, y, 1, 1).data; 
    var hex = "#" + ("000000" + rgbToHex(p[0], p[1], p[2])).slice(-6);
    $('#status').html(coord + "<br>" + hex);
});

উপরের কোডটি jQuery এবং নিম্নলিখিত ইউটিলিটি ফাংশনগুলির উপস্থিতি অনুমান করে:

function findPos(obj) {
    var curleft = 0, curtop = 0;
    if (obj.offsetParent) {
        do {
            curleft += obj.offsetLeft;
            curtop += obj.offsetTop;
        } while (obj = obj.offsetParent);
        return { x: curleft, y: curtop };
    }
    return undefined;
}

function rgbToHex(r, g, b) {
    if (r > 255 || g > 255 || b > 255)
        throw "Invalid color component";
    return ((r << 16) | (g << 8) | b).toString(16);
}

function randomInt(max) {
  return Math.floor(Math.random() * max);
}

function randomColor() {
    return `rgb(${randomInt(256)}, ${randomInt(256)}, ${randomInt(256)})`
}

এটি এখানে কর্মে দেখুন:


আপনি কি দয়া করে এই প্রশ্নটি সন্ধান করতে পারেন এবং দেখুন যে এর কোনও সমাধান আছে কিনা। আমি তার খুব প্রশংসা করব :)
খয়ের জেশান

আপনার নেস্টেড offsetParents এর ব্যবহার এটি সম্পর্কে দুর্দান্ত উপায় to আমি কখনই সে সম্পর্কে ভাবিনি। তবে আপনি কেন whileএকটি ifএবং তার পরে নিয়মিত লুপ ব্যবহার করবেন না do...while?
জোনাথন লাম

এই উত্তরটি আজ আমাকে সাহায্য করে (1-সেপ্টেম্বর-2017) .so +1 টি
এলাইভ টু ডাই

নিবন্ধন করুন ধন্যবাদ :)
ওয়েন

12

আমি জানি এটি একটি পুরানো প্রশ্ন, তবে এখানে একটি বিকল্প রয়েছে। আমি সেই চিত্রের ডেটাটি একটি অ্যারেতে সংরক্ষণ করব, তারপরে, ক্যানভাসের উপরে মাউস সরানোর ইভেন্টে:

var index = (Math.floor(y) * canvasWidth + Math.floor(x)) * 4
var r = data[index]
var g = data[index + 1]
var b = data[index + 2]
var a = data[index + 3]

প্রতিবার চিত্রের ডেটা পাওয়ার চেয়ে অনেক সহজ।


7

স্ট্যাকওভারফ্লোতে (উপরের নিবন্ধ সহ) এবং অন্যান্য সাইটগুলিতে এখানে পাওয়া যায় এমন বিভিন্ন রেফারেন্সগুলিকে একত্রিত করে আমি জাভাস্ক্রিপ্ট এবং জ্যাকুয়েরি ব্যবহার করে তা করেছি:

<html>
<body>
<canvas id="myCanvas" width="400" height="400" style="border:1px solid #c3c3c3;">
Your browser does not support the canvas element.
</canvas>
<script src="jquery.js"></script>
<script type="text/javascript">
    window.onload = function(){
        var canvas = document.getElementById('myCanvas');
        var context = canvas.getContext('2d');
        var img = new Image();
        img.src = 'photo_apple.jpg';
        context.drawImage(img, 0, 0);
    };

    function findPos(obj){
    var current_left = 0, current_top = 0;
    if (obj.offsetParent){
        do{
            current_left += obj.offsetLeft;
            current_top += obj.offsetTop;
        }while(obj = obj.offsetParent);
        return {x: current_left, y: current_top};
    }
    return undefined;
    }

    function rgbToHex(r, g, b){
    if (r > 255 || g > 255 || b > 255)
        throw "Invalid color component";
    return ((r << 16) | (g << 8) | b).toString(16);
    }

$('#myCanvas').click(function(e){
    var position = findPos(this);
    var x = e.pageX - position.x;
    var y = e.pageY - position.y;
    var coordinate = "x=" + x + ", y=" + y;
    var canvas = this.getContext('2d');
    var p = canvas.getImageData(x, y, 1, 1).data;
    var hex = "#" + ("000000" + rgbToHex(p[0], p[1], p[2])).slice(-6);
    alert("HEX: " + hex);
});
</script>
<img src="photo_apple.jpg"/>
</body>
</html>

এটি আমার সম্পূর্ণ সমাধান। এখানে আমি কেবল ক্যানভাস এবং একটি চিত্র ব্যবহার করেছি, তবে যদি আপনাকে <map>চিত্রটি ব্যবহার করতে হয় তবে এটিও সম্ভব।


5

প্রতিবার getImageData কল করা প্রক্রিয়াটি ধীর করবে ... জিনিসগুলি গতি বাড়ানোর জন্য আমি স্টোরের চিত্রের ডেটা প্রস্তাব করি এবং তারপরে আপনি সহজে এবং দ্রুত পিক্সের মান পেতে পারেন, তাই আরও ভাল পারফরম্যান্সের জন্য এই জাতীয় কিছু করুন

// keep it global
let imgData = false;  // initially no image data we have

// create some function block 
if(imgData === false){   
  // fetch once canvas data     
  var ctx = canvas.getContext("2d");
  imgData = ctx.getImageData(0, 0, canvas.width, canvas.height);
}
    // Prepare your X Y coordinates which you will be fetching from your mouse loc
    let x = 100;   // 
    let y = 100;
    // locate index of current pixel
    let index = (y * imgData.width + x) * 4;

        let red = imgData.data[index];
        let green = imgData.data[index+1];
        let blue = imgData.data[index+2];
        let alpha = imgData.data[index+3];
   // Output
   console.log('pix x ' + x +' y '+y+ ' index '+index +' COLOR '+red+','+green+','+blue+','+alpha);

খুব সহায়ক +1
ওয়েইন

4

আপনার যদি একক পিক্সেলের রঙের পরিবর্তে কোনও আয়তক্ষেত্রাকার গড় গড় রঙ প্রয়োজন, দয়া করে এই অন্যান্য প্রশ্নটি একবার দেখুন:

👉 জাভাস্ক্রিপ্ট - কোনও চিত্রের নির্দিষ্ট অঞ্চল থেকে গড় রঙ পান

যাইহোক, উভয়ই খুব একইভাবে সম্পন্ন হয়:

An একটি চিত্র বা ক্যানভাস থেকে একটি একক পিক্সেলের রঙ / মান পাওয়া

একক পিক্সেলের রঙ পেতে, আপনি প্রথমে সেই চিত্রটি কোনও ক্যানভাসে আঁকবেন, যা আপনি ইতিমধ্যে করেছেন:

const image = document.getElementById('image');
const canvas = document.createElement('canvas');
const context = canvas.getContext('2d');
const width = image.width;
const height = image.height;

canvas.width = width;
canvas.height = height;

context.drawImage(image, 0, 0, width, height);

এবং তারপরে একটি একক পিক্সেলের মান পান:

const data = context.getImageData(X, Y, 1, 1).data;

// RED   = data[0]
// GREEN = data[1]
// BLUE  = data[2]
// ALPHA = data[3]

Image সমস্ত ইমেজ ডেটা একবারে পেয়ে দ্রুত গতি বাড়ান

পুরো ইমেজের মান পেতে আপনাকে এই একই ক্যানভাস রেন্ডারিং কনটেক্সট 2 ডি.জেটআইমেজডেটা () ব্যবহার করতে হবে যা আপনি এর তৃতীয় এবং চতুর্থ প্যারাম পরিবর্তন করে করেন। এই ফাংশনের স্বাক্ষরটি হ'ল:

ImageData ctx.getImageData(sx, sy, sw, sh);
  • sx: আয়তক্ষেত্রের উপরের বাম কোণের x স্থানাঙ্ক যা থেকে চিত্রের ডেটা বের করা হবে।
  • sy: আয়তক্ষেত্রের উপরের বাম কোণের y স্থানাঙ্ক যা থেকে চিত্রের ডেটা বের করা হবে।
  • sw: আয়তক্ষেত্রটির প্রস্থ যা থেকে চিত্রের ডেটা বের করা হবে।
  • sh: আয়তক্ষেত্রের উচ্চতা যেখান থেকে ইমেজডাটা বের করা হবে।

আপনি দেখতে পান এটি কোনও ImageDataবস্তু, যা যা তা ফিরিয়ে দেয় । এখানে গুরুত্বপূর্ণ অংশটি হ'ল সেই অবজেক্টটির একটি .dataসম্পত্তি রয়েছে যা আমাদের সমস্ত পিক্সেল মান রাখে।

তবে নোট করুন যে .dataসম্পত্তিটি 1-মাত্রিক Uint8ClampedArray, যার অর্থ পিক্সেলের সমস্ত উপাদান সমতল হয়েছে, সুতরাং আপনি এমন কিছু পেয়ে যাচ্ছেন যা দেখায়:

ধরা যাক আপনার কাছে 2x2 চিত্র রয়েছে:

 RED PIXEL |       GREEN PIXEL
BLUE PIXEL | TRANSPARENT PIXEL

তারপরে, আপনি তাদের এইভাবে পাবেন:

[ 255, 0, 0, 255,    0, 255, 0, 255,    0, 0, 255, 255,    0, 0, 0, 0          ]
|   RED PIXEL   |    GREEN PIXEL   |     BLUE PIXEL   |    TRANSPAERENT  PIXEL |
|   1ST PIXEL   |      2ND PIXEL   |      3RD PIXEL   |             4TH  PIXEL | 

যেহেতু কলিং getImageDataএকটি ধীর অপারেশন, আপনি সমস্ত চিত্রের ডেটা ( sw= চিত্র প্রস্থ, sh= চিত্রের উচ্চতা) পেতে কেবল একবার কল করতে পারেন ।

এর পরে, উপরের উদাহরণে, তোমাদের মধ্যে উপাদান অ্যাক্সেস করতে চান তাহলে TRANSPARENT PIXEL, যে অবস্থানে এক x = 1, y = 1এই কাল্পনিক ইমেজ, আপনি তার প্রথম সূচক খুঁজতে হবে iতার মধ্যে ImageData'র dataহিসাবে সম্পত্তি:

const i = (y * imageData.width + x) * 4;

✨ আসুন এটি দেখতে দিন

const solidColor = document.getElementById('solidColor');
const alphaColor = document.getElementById('alphaColor');
const solidWeighted = document.getElementById('solidWeighted');

const solidColorCode = document.getElementById('solidColorCode');
const alphaColorCode = document.getElementById('alphaColorCode');
const solidWeightedCOde = document.getElementById('solidWeightedCode');

const brush = document.getElementById('brush');
const image = document.getElementById('image');
const canvas = document.createElement('canvas');
const context = canvas.getContext('2d');
const width = image.width;
const height = image.height;

const BRUSH_SIZE = brush.offsetWidth;
const BRUSH_CENTER = BRUSH_SIZE / 2;
const MIN_X = image.offsetLeft + 4;
const MAX_X = MIN_X + width - 1;
const MIN_Y = image.offsetTop + 4;
const MAX_Y = MIN_Y + height - 1;

canvas.width = width;
canvas.height = height;

context.drawImage(image, 0, 0, width, height);

const imageDataData = context.getImageData(0, 0, width, height).data;

function sampleColor(clientX, clientY) {
  if (clientX < MIN_X || clientX > MAX_X || clientY < MIN_Y || clientY > MAX_Y) {
    requestAnimationFrame(() => {
      brush.style.transform = `translate(${ clientX }px, ${ clientY }px)`;
      solidColorCode.innerText = solidColor.style.background = 'rgb(0, 0, 0)';
      alphaColorCode.innerText = alphaColor.style.background = 'rgba(0, 0, 0, 0.00)';
      solidWeightedCode.innerText = solidWeighted.style.background = 'rgb(0, 0, 0)';
    });
    
    return;
  }
  
  const imageX = clientX - MIN_X;
  const imageY = clientY - MIN_Y;
  
  const i = (imageY * width + imageX) * 4;

  // A single pixel (R, G, B, A) will take 4 positions in the array:
  const R = imageDataData[i];
  const G = imageDataData[i + 1];
  const B = imageDataData[i + 2];
  const A = imageDataData[i + 3] / 255;
  const iA = 1 - A;

  // Alpha-weighted color:
  const wR = (R * A + 255 * iA) | 0;
  const wG = (G * A + 255 * iA) | 0;
  const wB = (B * A + 255 * iA) | 0;

  // Update UI:
  
  requestAnimationFrame(() => {
    brush.style.transform = `translate(${ clientX }px, ${ clientY }px)`;

    solidColorCode.innerText = solidColor.style.background
      = `rgb(${ R }, ${ G }, ${ B })`;

    alphaColorCode.innerText = alphaColor.style.background
      = `rgba(${ R }, ${ G }, ${ B }, ${ A.toFixed(2) })`;

    solidWeightedCode.innerText = solidWeighted.style.background
      = `rgb(${ wR }, ${ wG }, ${ wB })`;
  });
}

document.onmousemove = (e) => sampleColor(e.clientX, e.clientY);
  
sampleColor(MIN_X, MIN_Y);
body {
  margin: 0;
  height: 100vh;
  display: flex;
  flex-direction: row;
  align-items: center;
  justify-content: center;
  cursor: none;
  font-family: monospace;
  overflow: hidden;
}

#image {
  border: 4px solid white;
  border-radius: 2px;
  box-shadow: 0 0 32px 0 rgba(0, 0, 0, .25);
  width: 150px;
  box-sizing: border-box;
}

#brush {
  position: absolute;
  top: 0;
  left: 0;
  pointer-events: none;
  width: 1px;
  height: 1px;
  mix-blend-mode: exclusion;
  border-radius: 100%;
}

#brush::before,
#brush::after {
  content: '';
  position: absolute;
  background: magenta;
}

#brush::before {
  top: -16px;
  left: 0;
  height: 33px;
  width: 100%;
}

#brush::after {
  left: -16px;
  top: 0;
  width: 33px;
  height: 100%;
}

#samples {
  position: relative;
  list-style: none;
  padding: 0;
  width: 250px;
}

#samples::before {
  content: '';
  position: absolute;
  top: 0;
  left: 27px;
  width: 2px;
  height: 100%;
  background: black;
  border-radius: 1px;
}

#samples > li {
  position: relative;
  display: flex;
  flex-direction: column;
  justify-content: center;
  padding-left: 56px;
}

#samples > li + li {
  margin-top: 8px;
}

.sample {
  position: absolute;
  top: 50%;
  left: 16px;
  transform: translate(0, -50%);
  display: block;
  width: 24px;
  height: 24px;
  border-radius: 100%;
  box-shadow: 0 0 16px 4px rgba(0, 0, 0, .25);  
  margin-right: 8px;
}

.sampleLabel {
  font-weight: bold;
  margin-bottom: 8px;
}

.sampleCode {
  
}
<img id="image" src="" >

<div id="brush"></div>

<ul id="samples">
  <li>
    <span class="sample" id="solidColor"></span>
    <div class="sampleLabel">solidColor</div>
    <div class="sampleCode" id="solidColorCode">rgb(0, 0, 0)</div>
  </li>
  <li>
    <span class="sample" id="alphaColor"></span>
    <div class="sampleLabel">alphaColor</div>
    <div class="sampleCode" id="alphaColorCode">rgba(0, 0, 0, 0.00)</div>
  </li>
  <li>
    <span class="sample" id="solidWeighted"></span>
    <div class="sampleLabel">solidWeighted (with white)</div>
    <div class="sampleCode" id="solidWeightedCode">rgb(0, 0, 0)</div>
  </li>
</ul>

⚠️ দ্রষ্টব্য আমি Cross-Originযদি কোনও বাহ্যিক চিত্র বা আমি দীর্ঘতর ডেটা ইউআরআই ব্যবহার করার চেষ্টা করি তবে অনুমোদিত থেকে বড় যে উত্তরটি অন্তর্ভুক্ত করা হয় তা সমস্যাগুলি এড়াতে আমি একটি ছোট ডেটা ইউআরআই ব্যবহার করছি ।

Colors এই রঙগুলি অদ্ভুত দেখাচ্ছে, তাই না?

আপনি যদি কর্টারটিকে অস্ট্রিক আকারের সীমানার চারপাশে সরিয়ে ফেলেন তবে আপনি দেখতে পাবেন যে কখনও কখনও avgSolidColorলাল হয়, তবে যে পিক্সেলটি আপনি নমুনা দিচ্ছেন তা সাদা দেখাচ্ছে। কারণ এটি Rযে পিক্সেলের উপাদানগুলি বেশি হতে পারে, তবুও আলফা চ্যানেলটি কম, তাই রঙটি আসলে লাল রঙের প্রায় স্বচ্ছ ছায়ায়, তবেavgSolidColor উপেক্ষা করে।

অন্যদিকে, avgAlphaColorগোলাপী দেখায়। ঠিক আছে, এটি সত্য নয়, এটি কেবল গোলাপী দেখাচ্ছে কারণ আমরা এখন আলফা চ্যানেলটি ব্যবহার করছি, এটি এটিকে স্বল্পস্বামী করে তোলে এবং আমাদের পৃষ্ঠার পটভূমি দেখতে দেয় যা এই ক্ষেত্রে সাদা।

🎨 আলফা-ভারিত রঙ

তারপরে, আমরা এটি ঠিক করতে কী করতে পারি? ঠিক আছে, দেখা যাচ্ছে যে আমাদের নতুন নমুনার উপাদানগুলি গণনা করার জন্য আমাদের কেবলমাত্র আলফা চ্যানেল এবং এর বিপরীতমুখী ব্যবহার করা দরকার, এক্ষেত্রে এটি সাদা দিয়ে মার্জ করা, কারণ এটি আমরা পটভূমি হিসাবে ব্যবহার করি use

এর অর্থ হ'ল যদি পিক্সেল থাকে তবে অন্তর R, G, B, Aকোথায় Aথাকে [0, 1], আমরা আলফা চ্যানেলের বিপরীত iAএবং ওজনিত নমুনার উপাদানগুলি এইভাবে গণনা করব :

const iA = 1 - A;
const wR = (R * A + 255 * iA) | 0;
const wG = (G * A + 255 * iA) | 0;
const wB = (B * A + 255 * iA) | 0;

পিক্সেলটি আরও স্বচ্ছ ( A0 এর কাছাকাছি), হালকা রঙটি নোট করুন ।


3

দ্রুত উত্তর

context.getImageData(x, y, 1, 1).data;একটি rgba অ্যারে প্রদান করে। যেমন[50, 50, 50, 255]


এখানে @ lwburk এর rgbToHex ফাংশনের একটি সংস্করণ যা rgba অ্যারেটিকে আর্গুমেন্ট হিসাবে গ্রহণ করে।

function rgbToHex(rgb){
  return '#' + ((rgb[0] << 16) | (rgb[1] << 8) | rgb[2]).toString(16);
};

এটি কেবল 16 বছরের উপরে রেড-মানগুলির সাথে কাজ করে, উদাহরণস্বরূপ [10, 42, 67, 255]উত্পাদন করে #a2a43যা বৈধ / ভাল ফর্ম্যাটেড হেক্স রঙের কোড নয়।
তি হউসমান


1

আমার কাছে ক্যানভাস থেকে পিক্সেল রঙ পাওয়ার খুব সহজ কাজ করার উদাহরণ রয়েছে।

প্রথমে কিছু প্রাথমিক HTML:

<canvas id="myCanvas" width="400" height="250" style="background:red;" onmouseover="echoColor(event)">
</canvas>

তারপরে জেএস ক্যানভাসে কিছু আঁকতে, এবং রঙ পেতে:

var c = document.getElementById("myCanvas");
var ctx = c.getContext("2d");
ctx.fillStyle = "black";
ctx.fillRect(10, 10, 50, 50);

function echoColor(e){
    var imgData = ctx.getImageData(e.pageX, e.pageX, 1, 1);
    red = imgData.data[0];
    green = imgData.data[1];
    blue = imgData.data[2];
    alpha = imgData.data[3];
    console.log(red + " " + green + " " + blue + " " + alpha);  
}

এখানে একটি কার্যকারী উদাহরণ , কেবল কনসোলটি দেখুন।


0

@ ওয়াইন বারকেটের উত্তর ভাল is আপনি যদি আরজিবা রঙ পেতে আলফা মানটিও বের করতে চান তবে আমরা এটি করতে পারি:

var r = p[0], g = p[1], b = p[2], a = p[3] / 255;
var rgba = "rgb(" + r + "," + g + "," + b + "," + a + ")";

আমি আলফা মানটি 255 দ্বারা বিভক্ত করেছি কারণ চিত্র-ডেটা অবজেক্ট এটিকে 0 - 255 এর মধ্যে একটি পূর্ণসংখ্যা হিসাবে সঞ্চয় করে, তবে বেশিরভাগ অ্যাপ্লিকেশনগুলিতে (উদাহরণস্বরূপ, CanvasRenderingContext2D.fillRect()) বৈধ সিএসএস ফর্ম্যাটে রঙ প্রয়োজন, যেখানে আলফা মান 0 এবং 1 এর মধ্যে থাকে।

(এও মনে রাখবেন যে আপনি যদি স্বচ্ছ রঙ বের করেন এবং এটি আবার ক্যানভাসে আঁকেন তবে পূর্বে যে কোনও রঙ রয়েছে তা ওভারলে করে দেবে So সুতরাং আপনি যদি rgba(0,0,0,0.1)একই জায়গায় 10 বার রঙ আঁকেন তবে এটি কালো হবে))

আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.