Hello, I have COGs with integer values, and I'd like to be able to assign colors to value ranges, e.g.: 0-200 - blue, 200-400 - green, etc., but the only renderer that seems to work is the automatically assigned stretch-renderer. I've tried uniqueValueRenderer, which the API reference suggests is my only option without statistics files, but the entire raster shows up as black. Could I possibly get a working example or something? Thanks.
Solved! Go to Solution.
ClassBreaksRenderer worked! I must've been doing something wrong when I was trying it before..
Thank you again for your time.
    const renderer = new ClassBreaksRenderer({
      field: "Value"
    })
    
    renderer.addClassBreakInfo({
      minValue: -250,
      maxValue: 0,
      symbol:{
        type:"simple-fill",
        color:"#00FFFF"
      }
    })
    const layer = new ImageryTileLayer({
      url: "http://localhost:6061/raster.tif",
      interpolation: "nearest",
      renderer: renderer
    })
Hi there, 
It looks like your COG file has more than one band (based on your code snippet). The UniqueValueRenderer cannot be applied to a service or COG that has more than one band. 
To assign UniqueValueRenderer to an ImageryTileLayer with COG the following two conditions must be met:
Hope this helps.
Thanks for the reply. My raster does have only 1 band. I have changed the UVR field property to "Value" and removed the bandIds property from the layer... Same result. Anything else that could be the issue?
Assuming you have a categorical data. Not sure why it is still black. I cant think of anything else without looking at the image itself. Can you share your COG?
Thank you so much for providing me with the cog file. So several things.
The imagery is black for two reasons. First, the pixel values for your COG file ranges between -55 to -227 and it has 153 unique values. However, you are looping through where i=0 to max number and applying the i value to uniqueValueInfo.value which does not exist in your image. Even if you fix this your image will be still black. See the following for why.
Second and the more important reason is that your cog tif file has pixel type of Int32 which translates to ImageryTileLayer.rasterInfo.pixelType of "S32". So I must amend my previous statement about assigning UniqueValueRenderer to an ImageryTileLayer... the following conditions must be met:
The image must have only one band.
Image must either
-- have a raster attribute table: you can use any numeric or string field from the table in UV.
--or u8 / s8 pixel type: you can only use "Value" field.
To apply UniqueValueRenderer to your COG you need to add attribute table to your image or change the pixel type to either u8 or s8. I have tested the image by changing the image pixel type and verified that it works. The u8 pixel values must range between 0 - 255, and s8 pixel values must range between -128 - 127. See pixelType doc for more info on pixel value ranges.
To convert your COG to have pixelType of u8 or s8, it is probably easiest to use ArcGIS PRO. First use, Plus function geoprocessing tool with a constant number (256 for example) + your tif. You know your data better so please adjust accordingly. Then use the Copy Raster Geoprocessing tool to convert the tif file to COG. For example, I did use the following params.
Then I use able to add and assign UVR to your COG successfully. I used only 10 color combos for the following image...
Thanks for all your help.
Sadly my data will not fit into 8 bits (nor am I keen to build attribute tables for each raster I generate). I did try building attribute tables via arcgis pro just to test though and my raster is still black.
Ideally I'd just set ranges with classBreaksRenderer but it seems there's no way to get this to work either?
ClassbreaksRenderer should work! Please feel free to share your code with me.
You can also use RasterStretchRenderer for your cog. The following an example that works your cog. Please refer to this doc to learn more about stretching: https://pro.arcgis.com/en/pro-app/latest/help/data/imagery/histogram-stretching.htm
 const layer = new ImageryTileLayer({
   renderer: new RasterStretchRenderer({
   stretchType: "min-max",
   statistics: [[-227, 50, -172, 19]],
   useGamma: false,
   computeGamma: false,
   dra: false,
   gamma: [1],
   colorRamp: {
     type: "multipart",
     colorRamps: [
       new AlgorithmicColorRamp({
          algorithm: "hsv",
          fromColor: [255, 45, 8, 255],
          toColor: [255, 57, 251, 255]
       }),
       new AlgorithmicColorRamp({
         algorithm: "hsv",
         fromColor: [255, 57, 251,255],
         toColor: [26,35,253,255]
       }),
       new AlgorithmicColorRamp({
         algorithm: "hsv",
         fromColor: [ 26, 35, 253, 255],
         toColor: [ 0, 181, 255, 255 ]
       }),
       new AlgorithmicColorRamp({
        algorithm: "hsv",
        fromColor: [ 0, 181, 255, 255 ],
        toColor: [ 0, 253, 255, 255 ]
       }),
       new AlgorithmicColorRamp({
         algorithm: "hsv",
         fromColor: [ 0, 253, 255, 255 ],
         toColor: [ 0, 251, 50, 255 ]
       }),
       new AlgorithmicColorRamp({
         algorithm: "hsv",
         fromColor: [ 0, 251, 50, 255 ],
         toColor: [ 255, 254, 52, 255 ]
       }),
       new AlgorithmicColorRamp({
         algorithm: "hsv",
         fromColor: [ 255, 254, 52, 255 ],
         toColor: [ 255, 181, 61, 255 ]
       }),
       new AlgorithmicColorRamp({
         algorithm: "hsv",
         fromColor: [ 255, 181, 61, 255 ],
         toColor: [ 255, 45, 8, 255 ]
       }),
       new AlgorithmicColorRamp({
         algorithm: "hsv",
         fromColor: [ 255, 45, 8, 255 ],
         toColor: [ 190, 190, 190, 255 ]
       })
     ]
    }
   }),
   url: "data/cog/raster.tif",
});
ClassBreaksRenderer worked! I must've been doing something wrong when I was trying it before..
Thank you again for your time.
    const renderer = new ClassBreaksRenderer({
      field: "Value"
    })
    
    renderer.addClassBreakInfo({
      minValue: -250,
      maxValue: 0,
      symbol:{
        type:"simple-fill",
        color:"#00FFFF"
      }
    })
    const layer = new ImageryTileLayer({
      url: "http://localhost:6061/raster.tif",
      interpolation: "nearest",
      renderer: renderer
    })
