XP - Change Fog Color Tone with Javascript?

Started by Heretic86, March 07, 2022, 02:33:03 am

Previous topic - Next topic

Heretic86

In the XP Editor in Change Fog Color Tone, its obvious to me that it uses an RGBA value.

If I wanted to simulate the effect in CSS / Javascript using a filter, what would I need need to do to use to achieve the same effects on regular images or a Canvas image?

*cough cough* Blizz?
Current Scripts:
Heretic's Moving Platforms

Current Demos:
Collection of Art and 100% Compatible Scripts

(Script Demos are all still available in the Collection link above.  I lost some individual demos due to a server crash.)

KK20

Tone != Color

Tone has Gray instead of Alpha. The idea is you're changing the graphic's luminance. It's easier to think about changing a graphic to grayscale--how do you determine what shade of gray a colored pixel should have?

When I was working on an editor myself, I ended up writing a DLL to do this calculation.
Code: c++
/*
The four parameters tone_r, tone_g, tone_b, and tone_gray represent the integer values entered into "Change Fog Color Tone".
pixels is just a pointer to the graphic itself
length is how many pixels the graphic has (i.e. width * height)
*/

void DLL_EXPORT ToneCalc(long * pixels, long length, int tone_r, int tone_g, int tone_b, int tone_gray)
{
    // Iterate through every single pixel in the graphic
    for (int i = 0; i < length; ++i)
    {
        long pixel = pixels[i];
        // Break up the hex color code to individual variables between 0-255
        int alpha = (pixel & 0xFF000000) >> 24;
        int red = (pixel & 0xFF0000) >> 16;
        int green = (pixel & 0xFF00) >> 8;
        int blue = (pixel & 0xFF);

        // The algorithm...
        int gray = (red * 299 + green * 587 + blue * 114) / 1000;
        int factor = (255 - tone_gray) * 1000 / 255;
        red = std::max(std::min((red - gray) * factor / 1000 + gray + tone_r, 255), 0);
        green = std::max(std::min((green - gray) * factor / 1000 + gray + tone_g, 255), 0);
        blue = std::max(std::min((blue - gray) * factor / 1000 + gray + tone_b, 255), 0);

        // Replace the pixel color
        pixels[i] = blue + (green << 8) + (red << 16) + (alpha << 24);
    }
}
I can't remember how I eventually came up with this algorithm, want to say some Stackoverflow answer.

Other Projects
RPG Maker XP Ace  Upgrade RMXP to RMVXA performance!
XPA Tilemap  Tilemap rewrite with many features, including custom resolution!

Nintendo Switch Friend Code: 8310-1917-5318
Discord: KK20 Tyler#8901

Join the CP Discord Server!

Heretic86

March 07, 2022, 04:21:05 pm #2 Last Edit: March 13, 2022, 12:56:38 am by Heretic86 Reason: Edited with solution to my question
Thanks!

I was figuring this would be more complicated than one of the existing CSS image filters.  The results almost look like an HSL value as the S for Saturation might be applicable?  I wasnt really planning on doing this on a pixel by pixel redraw, rather apply to the whole image.

Do you think an HSL filter might be better?  If anything I would rather have MORE functionality rather than less...

---

Edit: Got it, problem solved.

So yes, it IS possible to use an SVG with a Raster Image, apply an SVG filter (such as feColorMatrix), then draw that filtered image to an HTML5 Canvas element.

NOTE: I suck at SVG.

For anyone else that ends up here, the trick is to create a new Image, set an Event Listener for onload on the new Image() to handle drawing to your canvas, then set the matrix values on the feMatrix element, then set the image source  (img.src = 'data:image/svg+xml;base64,' + btoa(svg.outerHTML).replace(/\=+$/, ''); ) to a base64 encoded string of your full SVG element (svg.outerHTML).

It appears to me so far that the onload event listener is needed as even though the data src may be base64 encoded, it appears to be considered as "not fully loaded" until the render with filters is complete.

It may also be necessary to wrap ctx.clearRect and ctx.drawImage in a setTimeout call.  In my situation, I did not need it.

Spoiler: ShowHide

INLINE SVG HTML (contains a base64 data source for a Color Chip)
            <svg id="colorChipSVG" width="73" height="118" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
              <filter id="colormatrix" x="0" y="0" width="1" height="1">
                <feColorMatrix id="feMatrix" type="matrix" values="1 0 0 0 0  0 1 0 0 0   0 0 1 0 0   0 0 0 1 0" />
                <feColorMatrix id="fs" type="saturate" values="1"/>
              </filter>
              <image id="svgImg" width="73" height="118" filter="url(#colormatrix)" xlink:href="" />
            </svg>

Javascript
    function drawColorChip(){
      let tempImg = new Image();
     
      tempImg.onload = function(){
        ctx.clearRect(0, 0, ctx.width, ctx.height);
        ctx.drawImage(tempImg, 0, 0, 73, 118);
      };
     
      tempImg.src = 'data:image/svg+xml;base64,' + btoa(svg.outerHTML).replace(/\=+$/, '');
    }
Current Scripts:
Heretic's Moving Platforms

Current Demos:
Collection of Art and 100% Compatible Scripts

(Script Demos are all still available in the Collection link above.  I lost some individual demos due to a server crash.)

KK20

That is exactly what it is; it's actually the YIQ format as Google points out.

I don't really know what constitutes as an "HSL filter". If that's just another way of saying a pixel shader, then sure use one (though I'm doubtful one exists). That is basically what the code I shared is after all.

Other Projects
RPG Maker XP Ace  Upgrade RMXP to RMVXA performance!
XPA Tilemap  Tilemap rewrite with many features, including custom resolution!

Nintendo Switch Friend Code: 8310-1917-5318
Discord: KK20 Tyler#8901

Join the CP Discord Server!

Heretic86

March 09, 2022, 02:56:48 pm #4 Last Edit: March 09, 2022, 03:03:10 pm by Heretic86
Well, I am half way there, without a dll, or going pixel by pixel.

Possible solution is SVG.  SVG can be drawn to a canvas!

What I did was to use a feColorMatrix filter to the SVG.

I am having some trouble loading the image source graphic into Blob.  I can load the SVG by itself with the external image just fine and the effects appear to be the same as Color Tone in XP.  When I try to make a Blob out of the SVG, it doesnt use the Image Source Data.

Any suggestions to convert the SVG to an Image so I can draw that Image to the canvas?

Spoiler: ShowHide

          <svg xmlns="http://www.w3.org/2000/svg" id="changeMapFogToneSVG" class="changeMapFogTone">
            <filter id="svgCssFilter">
              <feColorMatrix id="svgMatrix" type="matrix"
                values="1 0 0 0 1
                        0 1 0 0 0
                        0 0 1 0 0
                        0 0 0 1 0">
              </feColorMatrix>
            </filter>
            <img id="svgImg" class="changeMapFogToneSvgFilter" src="" />
          </svg>

      img = new Image();
      let domurl = window.URL || window.webkitURL || window;
      let svgTemp = new Blob([svg], {type: 'image/svg+xml'});
      let url = domurl.createObjectURL(svgTemp);
      img.src = url;

Current Scripts:
Heretic's Moving Platforms

Current Demos:
Collection of Art and 100% Compatible Scripts

(Script Demos are all still available in the Collection link above.  I lost some individual demos due to a server crash.)