I need to get familiar with web programming, so I decided to write up a fairly simple PHP page that accepts GET requests and serves back an image of the Mandelbrot set... as a table of colored cells.
Examples:
and
I'm sending the response as a table because I didn't want to get into using image libraries yet, so I wanted to try doing it entirely using HTML, and I couldn't think of a simpler way.
Basically how it works is, you indicate where in the Set you want to view by specifying the "Bounds" fields, the size of the view and table, the aspect ratio (as a decimal), then nine colored fields that decide how the image will be colored.
The main logic is essentially a port of my C code from last year, and the coloring idea is pulled from a project I did several years ago in Clojure.
I'm looking for suggestions about anything. I haven't used HTML, PHP, or CSS is years, and I was never great with them to begin with. There's a lot here, so I don't expect super thorough reviews, but anything would be appreciated.
Specifically though:
- The function - new_color_f()is a disaster. Basically, I want to be able to produce a function that has some parameters already set, and I'm doing that by closing over the parameters of the enclosing function. It's long and ugly though, and is made worse by PHP's- usesyntax. Any suggestions there would be appreciated.
- The function - defaulting_get()seems like a smell as well. I need to account for the GET data being missing, and potentially empty. That lead to using a bizarre mix of- ??and- ?:operators though.
Anything except:
- Yes, this is stupid to do in PHP the way I am, and it's also stupid to do this using an HTML table. It causes issues with the input form being wiped every entry, and also causes incredible browser lag when receiving larger tables. I wanted to get some experience with PHP thought before I try something more complicated, and I've always loved this project. Really, I should be doing this in JavaScript, or using AJAX to request images instead of using a form.
The server is running PHP 7.4 for reference.
index.php
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Mandelbrot Generator</title>
    <style>            
        .display_table {
            font-size: 1px;
            border-spacing: 0;
        }
      
        .input_form {
            display: table;
        }
        
        .input_row {
            display: table-row;
        }
        
        .input_row label, .input_row input {
            display: table-cell;
        }
        
        .red_input input, .green_input input, .blue_input input {
            border-width: 5px;
        }
        
        .red_input input {
            border-color: red;
        }
        .green_input input {
            border-color: lightgreen;
        }
        .blue_input input {
            border-color: blue;
        }
        
    </style>
</head>
<body>
    <form method="get" class="input_form">
        <div class="input">
            <div class="location_input">
                <div class="input_row">
                    <label>Lower Real Bound</label>
                    <input name="lreal" type="number" min="-2" max="2" step="any">
                </div>
                <div class="input_row">
                    <label>Lower Imaginary Bound</label>
                    <input name="limag" type="number" min="-2" max="2" step="any">
                </div>
        
                <div class="input_row">
                    <label>View Width</label>
                    <input name="vwidth" type="number" min="0" max="4" step="any">
                </div>
                <div class="input_row">
                    <label>Pixels Wide</label>
                    <input name="pwidth" type="number" min="0" max="1000" step="any">
                </div>
        
                <div class="input_row">
                    <label>Aspect Ratio</label>
                    <input name="aratio" type="number" min="0" max="2" step="any">
                </div>
            </div>
            <div class="color_input">
                <div class="red_input">
                    <div class="input_row">
                        <label>Real</label>
                        <input name="rr" type="number" min="-1000" max="1000" step="any">
                    </div>
                    <div class="input_row">
                        <label>Imaginary</label>
                        <input name="rim" type="number" min="-1000" max="1000" step="any">
                    </div>
                    <div class="input_row">
                        <label>Iters</label>
                        <input name="rit" type="number" min="-1000" max="1000" step="any">
                    </div>
                </div>
                <div class="green_input">
                    <div class="input_row">
                        <label>Real</label>
                        <input name="gr" type="number" min="-1000" max="1000" step="any">
                    </div>
                    <div class="input_row">
                        <label>Imaginary</label>
                        <input name="gim" type="number" min="-1000" max="1000" step="any">
                    </div>
                    <div class="input_row">
                        <label>Iters</label>
                        <input name="git" type="number" min="-1000" max="1000" step="any">
                    </div>
                </div>
                <div class="blue_input">
                    <div class="input_row">
                        <label>Real</label>
                        <input name="br" type="number" min="-1000" max="1000" step="any">
                    </div>
                    <div class="input_row">
                        <label>Imaginary</label>
                        <input name="bim" type="number" min="-1000" max="1000" step="any">
                    </div>
                    <div class="input_row">
                        <label>Iters</label>
                        <input name="bit" type="number" min="-1000" max="1000" step="any">
                    </div>
                </div>
                
            </div>
        </div>
        <button type="submit">Submit</button>
    </form>
 
</body>
</html>
<?php
    include "display.php";
    
    const DEF_LOW_REAL = -2;
    const DEF_LOW_IMAG = -2;
    const DEF_VIEW_WIDTH = 4;
    const DEF_PIX_WIDTH = 100;
    const DEF_ASP_RATIO = 1;
    const DEF_COLOR_MULT = 2;
    
    function defaulting_get($key, $default) {
        return ($_GET[$key] ?? $default) ?: $default;
    }
    
    $low_real = defaulting_get("lreal", DEF_LOW_REAL);
    $low_imag = defaulting_get("limag", DEF_LOW_IMAG);
    $view_width = defaulting_get("vwidth", DEF_VIEW_WIDTH);
    $pixels_wide = defaulting_get("pwidth", DEF_PIX_WIDTH);
    $aspect_ratio = defaulting_get("aratio", DEF_ASP_RATIO);
    $view_height = $view_width / $aspect_ratio;
    $high_real = $low_real + $view_height;
    $high_imag = $low_imag + $view_width;
    $pixels_high = $pixels_wide / $aspect_ratio;
    
    $color_f = new_color_f(
        defaulting_get("rr", DEF_COLOR_MULT),
        defaulting_get("rim", DEF_COLOR_MULT),
        defaulting_get("rit", DEF_COLOR_MULT),
        defaulting_get("gr", DEF_COLOR_MULT),
        defaulting_get("gim", DEF_COLOR_MULT),
        defaulting_get("git", DEF_COLOR_MULT),
        defaulting_get("br", DEF_COLOR_MULT),
        defaulting_get("bim", DEF_COLOR_MULT),
        defaulting_get("bit", DEF_COLOR_MULT),
    );
    emit_mandelbrot_view(
        $color_f,
        $low_real,
        $high_real,
        $low_imag,
        $high_imag,
        $pixels_wide,
        $pixels_high);
display.php
<?php
    include "iteration.php";
    
    const COLOR_MAX = (2 << 7) - 1;
    
    function clamp_color($n) {
        return max(0, min(COLOR_MAX, $n));
    }
    
    function checked_produce_rgb($red, $green, $blue) {
        $c_red = clamp_color($red);
        $c_green = clamp_color($green);
        $c_blue = clamp_color(($blue));
        return "rgb($c_red, $c_green, $c_blue)";
    }
    
    function new_color_f($red_real, $red_imag, $red_iter,
                         $gre_real, $gre_imag, $gre_iter,
                         $blu_real, $blu_imag, $blu_iter) {
        
        $color_func = function($real, $imag, $iters) use
                              ($red_real, $red_imag, $red_iter,
                               $gre_real, $gre_imag, $gre_iter,
                               $blu_real, $blu_imag, $blu_iter) {
            
            return checked_produce_rgb(
                $real * $red_real + $imag * $red_imag + $iters * $red_iter,
                $real * $gre_real + $imag * $gre_imag + $iters * $gre_iter,
                $real * $blu_real + $imag * $blu_imag + $iters * $blu_iter
            );
        };
        
        return $color_func;
    }
    function produce_pixel($color) {
        return "<td style='background: $color; color: $color'>_</td>";
    }
    
    function emit_mandelbrot_view($color_f,
                                  $lower_real,
                                  $upper_real,
                                  $lower_imag,
                                  $upper_imag,
                                  $pixels_wide,
                                  $pixels_high) {
        
        $real_step =  ($upper_real - $lower_real) / $pixels_wide;
        $imag_step = ($upper_imag - $lower_imag) / $pixels_high;
        
        echo "<table class='display_table'>";
        for ($imag = $lower_imag; $imag <= $upper_imag; $imag += $imag_step) {
            echo "<tr>";
            for ($real = $lower_real; $real <= $upper_real; $real += $real_step) {
                $iters = test_point([$real, $imag]);
                $color = $color_f($real, $imag, $iters);
                
                echo produce_pixel($color);
            }
            echo "</tr>";
        }
        echo "</table>";
    }
iteration.php
<?php
    // Make mutative?
    
    const STD_MAX_ITERS = 200;
    const STD_INF_LIMIT = 2;
    function square_complex($complex) {
        [$real, $imag] = $complex;
        return [($real * $real) - ($imag * $imag),
                2 * $real * $imag];
    }
    
    function mandelbrot_iteration($init_complex, $curr_complex) {
        [$i_real, $i_imag] = $init_complex;
        $sqrd = square_complex($curr_complex);
        
        $sqrd[0] += $i_real;
        $sqrd[1] += $i_imag;
        
        return $sqrd;
    }
    
    function is_under_limit($complex, $inf_limit) {
        [$real, $imag] = $complex;
        return ($real * $real) + ($imag * $imag) <= ($inf_limit * $inf_limit);
    }
    
    function test_point($initial_complex,
                        $max_iters = STD_MAX_ITERS,
                        $inf_limit = STD_INF_LIMIT) {
        $current = $initial_complex;
    
        $i = 0;
        for (; $i < $max_iters && is_under_limit($current, $inf_limit); $i++) {
            $current = mandelbrot_iteration($initial_complex, $current);
        }
        
        return $i;
    }




