1

It looks to me like labels are meant to be anchored to objects in the map. In this case, I want to display some text on top of the map all of time.

I want it to be in the webgl itself, not an html element on top of it.

Keith Carter
  • 422
  • 1
  • 6
  • 18
  • Relevant, but I am not sure if they found a definitive answer: https://stackoverflow.com/questions/49807311/how-to-get-usable-canvas-from-mapbox-gl-js – chriswhong Oct 17 '22 at 12:44

2 Answers2

1

There are probably a million different ways to better accomplish your REAL objective, if you'd only state what that is, but anyway, if you're really determined to do what your question states...

  1. Make an image with your text.
  2. Base64 Encode that image.
  3. Initialize the map with preserveDrawingBuffer: true (note, performance will suffer)
  4. Use the following code. Replace the long base64 string with YOUR image. Replace the token with YOUR token.


    // TO MAKE THE MAP APPEAR YOU MUST
    // ADD YOUR ACCESS TOKEN FROM
    // https://account.mapbox.com
    mapboxgl.accessToken = 'REPLACEME';
    const map = new mapboxgl.Map({
        container: 'map', // container ID
        // Choose from Mapbox's core styles, or make your own style with Mapbox Studio
        style: 'mapbox://styles/mapbox/streets-v11', // style URL
        center: [-74.5, 40], // starting position [lng, lat]
        zoom: 9, // starting zoom
        projection: 'globe', // display the map as a 3D globe,
        preserveDrawingBuffer: true
    });

    map.on('render', () => {
      var img, tex, vloc, tloc, vertexBuff, texBuff;
      var cvs3d = map.getCanvas();
       var ctx3d  = cvs3d.getContext("experimental-webgl");
       
       var uLoc;
        ctx3d.pixelStorei(ctx3d.UNPACK_FLIP_Y_WEBGL, true); 
    // create shaders
    var vertexShaderSrc = `
        attribute vec2 aVertex;
        attribute vec2 aUV;
        varying vec2 vTex;
        uniform vec2 pos;
        void main(void) {
            gl_Position = vec4(aVertex + pos, 0.0, 1.0);
            vTex = aUV;
        }`;

    var fragmentShaderSrc = `
        precision highp float;
        varying vec2 vTex;
        uniform sampler2D sampler0;
        void main(void){
            gl_FragColor = texture2D(sampler0, vTex);
        }`;

    var vertShaderObj = ctx3d.createShader(ctx3d.VERTEX_SHADER);
    var fragShaderObj = ctx3d.createShader(ctx3d.FRAGMENT_SHADER);
    ctx3d.shaderSource(vertShaderObj, vertexShaderSrc);
    ctx3d.shaderSource(fragShaderObj, fragmentShaderSrc);
    ctx3d.compileShader(vertShaderObj);
    ctx3d.compileShader(fragShaderObj);

    var progObj = ctx3d.createProgram();
    ctx3d.attachShader(progObj, vertShaderObj);
    ctx3d.attachShader(progObj, fragShaderObj);

    ctx3d.linkProgram(progObj);
    ctx3d.useProgram(progObj);

    ctx3d.viewport(0, 0, 64, 64);
  
    vertexBuff = ctx3d.createBuffer();
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
    ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([-1, 1, -1, -1, 1, -1, 1, 1]), ctx3d.STATIC_DRAW);

    texBuff = ctx3d.createBuffer();
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
    ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([0, 1, 0, 0, 1, 0, 1, 1]), ctx3d.STATIC_DRAW);

    vloc = ctx3d.getAttribLocation(progObj, 'aVertex');
    tloc = ctx3d.getAttribLocation(progObj, 'aUV');
    uLoc = ctx3d.getUniformLocation(progObj, 'pos');

    var drawImage = function(imgobj, x, y, w, h) {
        tex = ctx3d.createTexture();
        ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
        ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MIN_FILTER, ctx3d.NEAREST);
        ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MAG_FILTER, ctx3d.NEAREST);
        ctx3d.texImage2D(ctx3d.TEXTURE_2D, 0, ctx3d.RGBA, ctx3d.RGBA, ctx3d.UNSIGNED_BYTE, imgobj);

        ctx3d.enableVertexAttribArray(vloc);
        ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
        ctx3d.vertexAttribPointer(vloc, 2, ctx3d.FLOAT, false, 0, 0);

        ctx3d.enableVertexAttribArray(tloc);
        ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
        ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
        ctx3d.vertexAttribPointer(tloc, 2, ctx3d.FLOAT, false, 0, 0);
    ctx3d.drawArrays(ctx3d.TRIANGLE_FAN, 0, 4);
    
    };

    img = new Image();
    img.src = '';

    img.onload = function() {
        drawImage(this, 0, 0, 64, 64);

        var img2 = new Image();
        img2.src = '';
        img2.onload = function() {
            drawImage(img2, 64-16, 64-16, 16, 16); // draw in bottom right corner
        }
    };
       
       
    });
      
       

Fiddle (you have to change map token in order to see it in action): https://jsfiddle.net/rygj51ca/1/

The above code will render an image of a smiley on the mapbox canvas, bottom left corner, every time the map renders. enter image description here

pkExec
  • 1,752
  • 1
  • 20
  • 39
  • THANK YOU for the response. The bounty expired before I could award it to you, is there some other way to transfer credit to you? – Keith Carter Nov 11 '22 at 23:12
  • I have asked this question with more context here. I apologize for not asking with proper context initially, and would really appreciate it if you could help out! : https://stackoverflow.com/questions/74409133/how-can-i-show-the-distance-a-polyline-has-covered-while-animating-it-in-mapbox – Keith Carter Nov 11 '22 at 23:22
  • @KeithCarter, if it answers your question, you should mark it as accepted. – pkExec Nov 13 '22 at 17:35
-1

If you want text in "the top right corner", then you don't want it in the map at all, you want it over the map.

In that case, the best way is just to use a different HTML element, styled however you like. Along these lines:

<div id="map" />
<div id="text-overlay" style="position: absolute; right: 0; top: 0;>Here's my text</div>
Steve Bennett
  • 114,604
  • 39
  • 168
  • 219