Question

J'essaie d'obtenir une valeur de couleur HSL de ma base de données. Actuellement, il n'y a qu'une valeur RVB stockée. Supposons que j'ai eu des colonnes séparées pour RGB: rouge vert rouge avec une valeur de numéro 0-255 dans chacune.

Résultat cible serait une légèreté de saturation de la teinte dans les Resultats, calculée à partir des valeurs RVB. J'ai vu beaucoup de calculs mais aucun d'entre eux ne semble assez facile à faire dans une requête? Ou je ne suis pas assez profondément en SQL en général pour savoir comment porter quelque chose comme une instruction de commutation sur SQL.

Le meilleur exemple pour la conversion que j'ai trouvé est ici: Comment obtenez-vous la teinte d'une couleur #xxxxxx?

function rgbToHsl(r, g, b){
    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, s, l];
}

Mais je suis complètement excentré pour une réponse sur la façon de faire quelque chose comme ce calcul dans SQL.

Une fois que cela ne fonctionne pas correctement (l'échantillon de code ci-dessus est incorrect de ce que je peux voir sur Wikipedia pour la conversion de la teinte, ainsi que de la teinte en pleine planche 0-360 plutôt que de 0 à 1) I J'ai commencé avec la solution d'Arth et je suis arrivé à cela, j'ai décidé de aller à R, G, B au préalable, car c'est simplement plus facile à suivre à partir de l'échantillon de code ci-dessus:

CASE 
  WHEN r>=g AND g>=b  THEN ((g-b)/(r-b))*60
  WHEN g>r AND r>=b THEN (2-(r-b)/(g-b))*60
  WHEN g>=b AND b>r THEN (2+(b-r)/(g-r))*60
  WHEN b>g AND g>r THEN (4-(g-r)/(b-r))*60
  WHEN b>r AND r>=g THEN (4+(r-g)/(b-g))*60
  WHEN r>=b AND b>g THEN (6-(b-g)/(r-g))*60
END h,
CASE 
  WHEN r=g  AND g=b  THEN 0
  WHEN r>=g AND g>=b AND  (r-b)>0.5 THEN (r-b)/(2-r-b)          
  WHEN r>=g AND g>=b THEN (r-b)/(r+b)
  WHEN r>=g AND b>g  AND  (r-g)>0.5 THEN (r-g)/(2-r-g)
  WHEN r>=g AND b>g  THEN (r-g)/(r+g) 
  WHEN g>=r AND r>=b AND  (g-b)>0.5 THEN (g-b)/(2-g-b) 
  WHEN g>=r AND r>=b THEN (g-b)/(g+b)
  WHEN g>=r AND b>r  AND  (g-r)>0.5 THEN (g-r)/(2-g-r) 
  WHEN g>=r AND b>r  THEN (g-r)/(g+r)
  WHEN b>=r AND r>=g AND  (b-g)>0.5 THEN (b-g)/(2-b-g) 
  WHEN b>=r AND r>=g THEN (b-g)/(b+g)
  WHEN b>=r AND g>r  AND  (b-r)>0.5 THEN (b-r)/(2-b-r) 
  WHEN b>=r AND g>r  THEN (b-r)/(b+r)
END s,
CASE 
  WHEN r=g  AND g=b  THEN r
  WHEN r>=g AND g>=b THEN (r+b)/2
  WHEN r>=g AND b>g  THEN (r+g)/2
  WHEN g>=r AND r>=b THEN (g+r)/2
  WHEN g>=r AND b>r  THEN (g+b)/2
  WHEN b>=r AND r>=g THEN (b+g)/2
  WHEN b>=r AND g>r  THEN (b+r)/2
END l

Était-ce utile?

La solution

C'est un cauchemar absolu et non testé, mais je suis allé:

SELECT 
    CASE 
      WHEN r=g  AND g=b  THEN 0
      WHEN r>=g AND g>b  THEN ((g-b)/(r-b))/6
      WHEN r>=g AND b>=g THEN ((g-b)/(r-g)+6)/6
      WHEN g>=r AND r>=b THEN ((b-r)/(g-b)+2)/6
      WHEN g>=r AND b>r  THEN ((b-r)/(g-r)+2)/6
      WHEN b>=r AND r>=g THEN ((r-g)/(b-g)+4)/6
      WHEN b>=r AND g>r  THEN ((r-g)/(b-r)+4)/6
    END h,
    CASE 
      WHEN r=g  AND g=b  THEN 0
      WHEN r>=g AND g>=b AND  (r-b)>0.5*255 THEN (r-b)/(510-r-b)          
      WHEN r>=g AND g>=b THEN (r-b)/(r+b)
      WHEN r>=g AND b>g  AND  (r-g)>0.5*255 THEN (r-g)/(510-r-g)
      WHEN r>=g AND b>g  THEN (r-g)/(r+g) 
      WHEN g>=r AND r>=b AND  (g-b)>0.5*255 THEN (g-b)/(510-g-b) 
      WHEN g>=r AND r>=b THEN (g-b)/(g+b)
      WHEN g>=r AND b>r  AND  (g-r)>0.5*255 THEN (g-r)/(510-g-r) 
      WHEN g>=r AND b>r  THEN (g-r)/(g+r)
      WHEN b>=r AND r>=g AND  (b-g)>0.5*255 THEN (b-g)/(510-b-g) 
      WHEN b>=r AND r>=g THEN (b-g)/(b+g)
      WHEN b>=r AND g>r  AND  (b-r)>0.5*255 THEN (b-r)/(510-b-r) 
      WHEN b>=r AND g>r  THEN (b-r)/(b+r)
    END s,
    CASE 
      WHEN r=g  AND g=b  THEN r/255
      WHEN r>=g AND g>=b THEN (r+b)/510
      WHEN r>=g AND b>g  THEN (r+g)/510
      WHEN g>=r AND r>=b THEN (g+r)/510
      WHEN g>=r AND b>r  THEN (g+b)/510
      WHEN b>=r AND r>=g THEN (b+g)/510
      WHEN b>=r AND g>r  THEN (b+r)/510
    END l
FROM table1

En conclusion, vous ferez probablement mieux la sélection des valeurs RVB et l'exécution de la conversion au niveau de l'application!

Licencié sous: CC-BY-SA avec attribution
Non affilié à StackOverflow
scroll top