Club Delphi  
    FTP   CCD     Buscar   Trucos   Trabajo   Foros

Retroceder   Foros Club Delphi > Principal > Gráficos
Registrarse FAQ Miembros Calendario Guía de estilo Temas de Hoy

Respuesta
 
Herramientas Buscar en Tema Desplegado
  #1  
Antiguo 23-11-2008
JF Sebastian JF Sebastian is offline
Miembro
 
Registrado: oct 2006
Posts: 108
Poder: 18
JF Sebastian Va por buen camino
Interpolacion RGB

Desearia realizar un espectro de colores desde el rojo al azul.
Sabeis como variar los parametros RGB para recorrer el espectro de colores?

Saludos
Responder Con Cita
  #2  
Antiguo 23-11-2008
[coso] coso is offline
Miembro Premium
 
Registrado: may 2008
Ubicación: Girona
Posts: 1.678
Poder: 0
coso Va por buen camino
Hola,
TColor es un entero entre 0 y $02FFFFFF, si haces

Código Delphi [-]
color := r*$01 + g*$0100 + b*$010000;

tendras un color a medida.
Responder Con Cita
  #3  
Antiguo 23-11-2008
JF Sebastian JF Sebastian is offline
Miembro
 
Registrado: oct 2006
Posts: 108
Poder: 18
JF Sebastian Va por buen camino
lo que quiero es calcular que valores de r,g,b debo poner a un color para que quede entre el espectro de colores (llamalo arco iris...) siendo 1 el color rojo puro y 0 el azul puro siendo el color un numero entre 0 y 1
Responder Con Cita
  #4  
Antiguo 23-11-2008
[coso] coso is offline
Miembro Premium
 
Registrado: may 2008
Ubicación: Girona
Posts: 1.678
Poder: 0
coso Va por buen camino
Cita:
lo que quiero es calcular que valores de r,g,b debo poner a un color para que quede entre el espectro de colores (llamalo arco iris...)
un color siempre estara dentro del espectro de colores... ¿Que es realmente lo que buscas hacer? me da que lo que quieres hacer es un gradiente. Lo que tienes que hacer entonces es, desde unos r,g,b iniciales ir sumando unos incrementos (que dependeran del numero de pasos) hasta llegar a los r,g,b del color final. Busca la palabra gradiente o degradado en el foro (o en los trucos). Hay algunos ejemplos bastante elaborados. Saludos.
Responder Con Cita
  #5  
Antiguo 24-11-2008
JF Sebastian JF Sebastian is offline
Miembro
 
Registrado: oct 2006
Posts: 108
Poder: 18
JF Sebastian Va por buen camino
Gracias,
He encontrado este post http://www.clubdelphi.com/foros/showthread.php?t=29167
pero me gustaria implementar manualmente la funcion de la dll.
Basicamente quiero calcular el color intermedio que esta entre el rojo y el azul-violeta de modo que si asigno el numero 1 al rojo y el 0 al azul-violeta la funcion me devuelva la tonalidad correspondiente a un numero intermedio entre 0..1. El gradiente de colores ha de estar de acuerdo con el espectro estandar de colores. No necesito que me pinte el gradiente pues necesito como maximo 20 colores fijos intermedios.
No parece dificil, pero se me resite el problema...
Responder Con Cita
  #6  
Antiguo 24-11-2008
Avatar de Delphius
[Delphius] Delphius is offline
Miembro Premium
 
Registrado: jul 2004
Ubicación: Salta, Argentina
Posts: 5.582
Poder: 25
Delphius Va camino a la fama
Hola JF Sebastian,
La verdad es que no termino de comprender lo que buscas.
Por empezar el espacio formado entre rojo (255,0,0) y azul (0,0,255) es amplio. Por lo que existen muchos colores posibles.
Si entedemos a esto en el espacio, tendremos tres ejes: R, G, B. Los ejes van desde el 0 al 255. Si sólo debemos desplazarnos desde el eje R al B, quiere decir que en ningún momento intervendrá el eje G. Claramente se ve que se trata de un giro de 90º. Por tanto sólo obtendremos combinaciones de distintas tonalidades de rojo y azul.

Con esto en mente a mi se me ocurre rotar el vector (x,0,z) "a" grados uniformemente distribuídos. Es decir que cuando a es cero, tendremos:

x (rojo) = 255 * cos(0) = 255
z (azul) = 255 * sen(0) = 0
Por tanto el color que se obtiene es: (x,0,z) = (255,0,0)

Cuando a = 90, tenemos:

x (rojo) = 255 * cos(90) = 0
z (azul) = 255 * sen(90) = 255

Es decir el color (0,0,255)

¿Se entiende la idea?
Bueno ahora la cuestión es como obtener ese "a"... bueno si consideramos que 90º es el 100% o 1, y que 0º es 0% o 0... entonces el grado de desplazamiento estará dado por 90/cantidad, siendo "cantidad" la cantidad de colores que deseas obtener... por ejemplo tus 20. En este caso, los 20 colores estarán espaciados 90/20º (4,5º) uno de otro.

Bueno esa es mi idea... no se si es eso lo que buscas. Si no se me entiende y/o no es lo que buscas te agradecería que lo hicieras saber.

¿Cúal es el problema de éste método? que sólo obtendrás los colores ubicados sobre el contorno que describe el vector por lo que no comprende todo el especto posible con dos colores. Por ejemplo el color (255,0,255) no estará cubierto.

Si nos pudieras explicar con mejores detalles lo que buscas te podríamos ser de ayuda.

Saludos,
__________________
Delphius
[Guia de estilo][Buscar]
Responder Con Cita
  #7  
Antiguo 24-11-2008
[coso] coso is offline
Miembro Premium
 
Registrado: may 2008
Ubicación: Girona
Posts: 1.678
Poder: 0
coso Va por buen camino
Cita:
pero me gustaria implementar manualmente la funcion de la dll.
¿que dll?

Cita:
El gradiente de colores ha de estar de acuerdo con el espectro estandar de colores.
¿?

Lo que tienes que hacer es conseguir los r,g,b del color final, restar de ellos los r,g,b del color inicial. Estos valores, llamemosles dr,dg,db, los divides en un numero de pasos, pon 100, por ejemplo. Para encontrar el color medio en la posicion n tansolo tendrias que hacer Color nuevo = r + dr*n,g + dg*n, b + db*n. El color medio entre el final y el inicial seria con n = 50. Mirate bien los ejemplos que hay por el foro, pues hacen mas o menos esto mismo. saludos.
Responder Con Cita
  #8  
Antiguo 24-11-2008
Avatar de Lepe
[Lepe] Lepe is offline
Miembro Premium
 
Registrado: may 2003
Posts: 7.424
Poder: 28
Lepe Va por buen camino
Delphi ya cuenta con funciones para decodificar un color:
Código Delphi [-]

function RGB(r, g, b: Byte): COLORREF;

function GetRValue(rgb: DWORD): Byte;

function GetGValue(rgb: DWORD): Byte;

function GetBValue(rgb: DWORD): Byte;

function rgb
esas están en la unidad windows, pero también tienes otras en graphics, por ejemplo ColorToRGB

Saludos
__________________
Si usted entendió mi comentario, contácteme y gustosamente,
se lo volveré a explicar hasta que no lo entienda, Gracias.
Responder Con Cita
  #9  
Antiguo 24-11-2008
JF Sebastian JF Sebastian is offline
Miembro
 
Registrado: oct 2006
Posts: 108
Poder: 18
JF Sebastian Va por buen camino
Gracias por vuestra amabilidad.
Pero lo que necesito es una escala de todos los colores posibles y no de un degradado entre dos colores dados.
Aqui podeis ver el espectro del que hablo: http://www.fullspectrumrgb.com/why_do.htm
En principio seria indiferente el espectro de lo que el ojo ve o el de lo que ve una camara. Me bastaria con que recorriese de forma similar todos los colores posibles. Se como hacer un gradiente de dos colores dados.
Os cuento, si me permitis, con un poco mas de detalle:
Se trata de una aplicacion de analisis estructural en la que en un momento dado se precisa dibujar con colores los resultados del calculo de una placa. Esta placa esta sometida a fuerzas puntuales o distribuidas que producen unas deformaciones y unas tensiones. Una forma habitual de representar graficamente los resultados es con colores y normalmente se emplean todos los colores del espectro, rojo para el valor maximo y azul-violeta para el minimo. Al lado del espectro se anotan varios valores numericos (10 o 20)normalmente equiespaciados. Esta escala suele ser variable en cuanto al numero de valores a representar. La pregunta es dado un valor intermedio (de los resultados, no del color), que color (R,G,B) le corresponde?
Hasta ahora venia haciendo lo siguiente:
Tomaba R=1 y G=0 y decrementaba R y aumentaba B hasta R=0 y G=1 luego tomaba G=1 y B=0, decrementaba G y aumentaba B hasta G=0 y B=1 pero no sale nada parecido al espectro deseado (cualquiera parecido al del link que os dejo)
Si el numero de valores a representar fuese fijo la cosa parece facil de implementar pero puede ser variable a gusto del usuario y ahi esta la pega.

Imagino que la solucion buscada es una de las infinitas trayectorias que tendria un vector unitario que en tres instantes diferentes adoptase los valores [1,0,0],[0,1,0],[0,0,1]
Lo que se me ocurre es tomar los valores RGB de 7 colores basicos: rojo,naranja,amarillo,verde,azul,anil,violeta e interpolar linealmente entre cada dos de esos colores y creo que es lo que voy a hacer, pero si sabeis de algo mas elegante soy todo oidos
Os dejo otro link de mi aplicacion con lo que tengo actualmente implementado:

http://www.ustatic.net/Images/Demo10.jpg

Un saludo y muchas gracias.

Última edición por JF Sebastian fecha: 24-11-2008 a las 22:16:42.
Responder Con Cita
  #10  
Antiguo 24-11-2008
[coso] coso is offline
Miembro Premium
 
Registrado: may 2008
Ubicación: Girona
Posts: 1.678
Poder: 0
coso Va por buen camino
Hola de nuevo. Ahora si que te he entendido. Para hacer eso lo mejor es tratar el color por HSV (hue, saturation,value) y teniendo H fija, ir variando la saturacion y el valor del color. No se si en versiones posteriores a la 5 ya vienen funciones estandar para HSV. Te dejo este link a ver si te sirve. Saludos.

Última edición por coso fecha: 24-11-2008 a las 22:38:28.
Responder Con Cita
  #11  
Antiguo 24-11-2008
[coso] coso is offline
Miembro Premium
 
Registrado: may 2008
Ubicación: Girona
Posts: 1.678
Poder: 0
coso Va por buen camino
...y otra manera, aunque quiza sea mas dificil de implementar, seria usando el concepto de R³ como dice Delphius: crear un vector en R,G,B y ir desplazandolo gradualmente sin modificar su modulo. No estoy seguro si funcionaria pero podrias probar. Saludos de nuevo.
Responder Con Cita
  #12  
Antiguo 25-11-2008
Avatar de cHackAll
[cHackAll] cHackAll is offline
Baneado?
 
Registrado: oct 2006
Posts: 2.159
Poder: 20
cHackAll Va por buen camino
Código Delphi [-]
procedure TForm1.Button1Click(Sender: TObject); // matriz o paleta de colores
var Offset, Value, Index: Cardinal;
begin
 Offset := 0;
 Value := 255;
 for Index := 0 to 255 * 3 do
  begin
   Canvas.Pen.Color := Value;
   Canvas.MoveTo(Index, 0);
   Canvas.LineTo(Index + 16, 32);
   Dec(PByte(Cardinal(@Value) + (Offset mod 3))^);
   Inc(PByte(Cardinal(@Value) + ((Offset + 1) mod 3))^);
   if ((Index + 1) mod 255) = 0 then
    Inc(Offset);
  end;
end;
__________________
RTFM > STFW > Foro > Truco > Post > cHackAll > KeBugCheckEx
Responder Con Cita
  #13  
Antiguo 25-11-2008
Avatar de Lepe
[Lepe] Lepe is offline
Miembro Premium
 
Registrado: may 2003
Posts: 7.424
Poder: 28
Lepe Va por buen camino
Todavía me estoy preguntando cómo has puesto el código sin ninguna palabra, al menos el foro te obliga a escribir 10 letras.... Pero a lo que iba, tu código suele impresionar por su calidad, pero en este caso, es superado por la belleza del resultado.

Salud.
__________________
Si usted entendió mi comentario, contácteme y gustosamente,
se lo volveré a explicar hasta que no lo entienda, Gracias.
Responder Con Cita
  #14  
Antiguo 25-11-2008
Avatar de Delphius
[Delphius] Delphius is offline
Miembro Premium
 
Registrado: jul 2004
Ubicación: Salta, Argentina
Posts: 5.582
Poder: 25
Delphius Va camino a la fama
Viendo las nuevas aportaciones, me parece que lo que he dicho yo no es muy viable que digamos.
Por empezar yo mismo he aclarado que mi idea es rotar el vector desde el eje R hacia el B, un total de 90º. Por tanto todos los puntos que pertenezcan a dicho plano tendrán sólo dos componentes: rojo y azul.
Y es así como lo entendí puesto que por la forma en como describía el problema se trataba de una rotación del rojo al azul, pero no daba evidencia alguna del verde.

Por si no fuera poco, la segunda limitante a mi "diseño", no se cubre todo el espectro de la combinación de hasta dos colores. Sólo se obtienen aquellos cuyo valor de módulo sea 255.

Tal vez dos "trucos" sean posibles:
1. Dado un valor "y" (verde) llevar a cabo la rotación desde el rojo al azul. O tal vez, dado un rango de verde y_min, y_max obtener para cada valor perteneciente a dicho rango, su rotación. Por ejemplo y = [0,255]:

(255,0,0) -> ... -> (0,0,255)
...
(255,255,0) -> ... -> (0,255,255)

2. Alterar el valor del módulo para forzar el uso de los valores, y no necesariamente del valor correspondiente al módulo. Con esta "técnica" se podría llegar a valores "inaccesibles", como (255,y,255). Se puede aumentar o disminuir el módulo.

Si es necesario cubrir el especto visible pues lo veo un tanto difícil... pero no se como diablos hizo Javier para demostrarlo tan sencillo (1)

(1) Sencillo por las pocas líneas... yo la verdad que me pierdo en interpretarlo

Saludos,
__________________
Delphius
[Guia de estilo][Buscar]
Responder Con Cita
  #15  
Antiguo 25-11-2008
Avatar de Lepe
[Lepe] Lepe is offline
Miembro Premium
 
Registrado: may 2003
Posts: 7.424
Poder: 28
Lepe Va por buen camino
Cita:
Empezado por Delphius Ver Mensaje
(1) Sencillo por las pocas líneas... yo la verdad que me pierdo en interpretarlo
Pero eso es normal con c-Hack-All

Saludos
__________________
Si usted entendió mi comentario, contácteme y gustosamente,
se lo volveré a explicar hasta que no lo entienda, Gracias.
Responder Con Cita
  #16  
Antiguo 25-11-2008
[egostar] egostar is offline
Registrado
 
Registrado: feb 2006
Posts: 6.557
Poder: 25
egostar Va camino a la fama
Cita:
Empezado por Lepe Ver Mensaje
Todavía me estoy preguntando cómo has puesto el código sin ninguna palabra, al menos el foro te obliga a escribir 10 letras.... Pero a lo que iba, tu código suele impresionar por su calidad, pero en este caso, es superado por la belleza del resultado.

Salud.
Cita:
Empezado por Lepe Ver Mensaje
Pero eso es normal con c-Hack-All

Saludos
La respuesta a tu pregunta la has dado tu mismo

Salud OS
__________________
"La forma de empezar es dejar de hablar y empezar a hacerlo." - Walt Disney
Responder Con Cita
  #17  
Antiguo 26-11-2008
[coso] coso is offline
Miembro Premium
 
Registrado: may 2008
Ubicación: Girona
Posts: 1.678
Poder: 0
coso Va por buen camino
Hola de nuevo, y ya considerando el hilo resuelto : dejo aqui este codigo para ilustrar la idea de delphius de usar el 'cubo' RGB e ir rotando un vector de color en el.

Código Delphi [-]
procedure TForm1.FormPaint(Sender: TObject);
var
     phi,rho : double;
     m : integer;
     r,g,b : integer;
     x,y : integer;
begin
     m := TrackBar1.Position;
     phi := 0;
     while phi < PI/2 do
     begin
          rho := 0;
          while rho < PI/2 do
          begin
               r := Round(m*sin(phi));         // proyeccion del vector en r
               g := Round(m*cos(phi)*cos(rho));// proyeccion en g
               b := Round(m*cos(phi)*sin(rho));// proyeccion en b

               canvas.Pixels[20+Round(rho*100),20+Round(phi*100)] := r*$01 + g*$0100 + b*$010000;
               rho := rho + 0.01;
          end;
          phi := phi + 0.01;
     end;
end;

procedure TForm1.TrackBar1Change(Sender: TObject);
begin
     Paint;
end;

procedure TForm1.FormCreate(Sender: TObject);
begin
     doublebuffered := true;
     TrackBar1.Max := 255;
     TrackBar1.Min := 0;
end;

PD: tremendo el codigo, cHackAll
Responder Con Cita
  #18  
Antiguo 26-11-2008
JF Sebastian JF Sebastian is offline
Miembro
 
Registrado: oct 2006
Posts: 108
Poder: 18
JF Sebastian Va por buen camino
Muy buenas aportaciones, companeros.

Última edición por JF Sebastian fecha: 26-11-2008 a las 19:39:18.
Responder Con Cita
  #19  
Antiguo 26-11-2008
JF Sebastian JF Sebastian is offline
Miembro
 
Registrado: oct 2006
Posts: 108
Poder: 18
JF Sebastian Va por buen camino
Muy buenas aportaciones, companeros.

Os dejo otra solucion que encontre por ahi y que permite interpolar facilmente y obtener el color deseado...



implementation
{$R *.dfm}
uses math;
Código Delphi [-]
 
procedure TForm1.FormClick(Sender: TObject);
var i: integer; R,G,B: byte; cl:TColor;
begin
  for i := 400 to 750 do
  begin
    WavelengthToRGB(1.0*i,R,G,B);
    cl := rgb(r,g,b);
    Canvas.Pen.Color := cl;
    Canvas.Moveto(i-300,0);
    Canvas.LineTo(i-300,100);
  end;
end;
 
procedure TForm1.WavelengthToRGB(const Wavelength:  double; var R,G,B: byte);
const
  Gamma        =   0.80;
  IntensityMax = 255;
var
  Blue   :  double;
  factor :  double;
  Green  :  double;
  Red    :  double;
  function Adjust(const Color, Factor: double):integer;
  begin
    if   Color = 0.0
    then result := 0
    else result := round(IntensityMax * Power(Color * Factor, Gamma))
  end;
begin
  case trunc(Wavelength) OF
    380..439: begin
        Red   := -(Wavelength - 440) / (440 - 380);
        Green := 0.0;
        Blue  := 1.0
    end;
    440..489: begin
        Red   := 0.0;
        Green := (Wavelength - 440) / (490 - 440);
        Blue  := 1.0
    end;
    490..509: begin
        Red   := 0.0;
        Green := 1.0;
        Blue  := -(Wavelength - 510) / (510 - 490)
    end;
    510..579: begin
        Red   := (Wavelength - 510) / (580 - 510);
        Green := 1.0;
        Blue  := 0.0
    end;
    580..644: begin
        Red   := 1.0;
        Green := -(Wavelength - 645) / (645 - 580);
        Blue  := 0.0
    end;
    645..780: begin
        Red   := 1.0;
        Green := 0.0;
        Blue  := 0.0
    end;
    else
      Red   := 0.0;
      Green := 0.0;
      Blue  := 0.0
  end;
  case trunc(Wavelength) of
    380..419:  factor := 0.3 + 0.7*(Wavelength - 380) / (420 - 380);
    420..700:  factor := 1.0;
    701..780:  factor := 0.3 + 0.7*(780 - Wavelength) / (780 - 700)
    else       factor := 0.0
  end;
  R := Adjust(Red,   Factor);
  G := Adjust(Green, Factor);
  B := Adjust(Blue,  Factor)
end;
Responder Con Cita
  #20  
Antiguo 26-11-2008
JF Sebastian JF Sebastian is offline
Miembro
 
Registrado: oct 2006
Posts: 108
Poder: 18
JF Sebastian Va por buen camino
Que inutil soy
Responder Con Cita
Respuesta



Normas de Publicación
no Puedes crear nuevos temas
no Puedes responder a temas
no Puedes adjuntar archivos
no Puedes editar tus mensajes

El código vB está habilitado
Las caritas están habilitado
Código [IMG] está habilitado
Código HTML está deshabilitado
Saltar a Foro

Temas Similares
Tema Autor Foro Respuestas Último mensaje
Ampliar imagen o parte de ella - Interpolacion Lineal Delphius Gráficos 11 27-01-2007 04:00:45


La franja horaria es GMT +2. Ahora son las 10:08:36.


Powered by vBulletin® Version 3.6.8
Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
Traducción al castellano por el equipo de moderadores del Club Delphi
Copyright 1996-2007 Club Delphi