Je travaille sur ce code dont l'objectif est de : vérifier si c'est un PNG, vérifier si c'est 256X256, et transformer img en base64 pour le télécharger sur le serveur.

Donc si je sélectionne un img 256 x 256 c'est des charges mais, si je charge un autre img qui n'est pas un 256 x 256 après avoir sélectionné un 256 x 256 c'est des charges et je ne sais pas pourquoi ! Vidéo diffusée Comment puis-je le résoudre?

function isFileImage(file) {
    const acceptedImageTypes = ['image/png'];

    return file && acceptedImageTypes.includes(file['type'])
}



function importFileandPreview() {
  var preview = document.querySelector('img');
  var file    = document.querySelector('input[type=file]').files[0];
  var reader  = new FileReader();

  if (file) {
    //read img
    reader.readAsDataURL(file);
    //verify type
    var text = file.type;
    if (text === "image/png") {
      //load img
      reader.addEventListener("load", function () {
        //put img on <img> src
        //console logs
        //console.log(preview);
        //console.log("pre_nat_height:"+preview.naturalHeight);
      // console.log("pre_nat_width:"+preview.naturalWidth);
       //technically check if img is 256 x 256
       var old = preview.src;
       preview.src = reader.result;
        if(preview.naturalHeight === 256 && preview.naturalWidth === 256){
          //create a hidden input (works but value is not from last img)
          var element = document.getElementById("imga");
            if(typeof(element) != 'undefined' && element != null){
            document.getElementById("imga").remove();
          }
            input = document.createElement('input');
            input.setAttribute("type","hidden");
            input.setAttribute("id","imga");
            input.setAttribute("name","img")
            input.setAttribute("value",preview.src);
            document.getElementById('count').appendChild(input);

            delete preview;
        }
        //else if img is not 256 x 256
        else {
          preview.src = old;
          alert("Must be a PNG and have 256px X 256px!");
          delete preview;
        }
      }, false);// i don't know why false but ye
    }
  }
}
<input type="file" id="file" accept="image/PNG" onchange="importFileandPreview()"><br>
<img src=" " height="256" width="256">
<form id="count">
</form>
0
Victor 30 oct. 2020 à 01:01

1 réponse

Meilleure réponse

tl;dr : Vous devez effectuer la vérification par rapport aux dimensions naturelles de l'image à l'intérieur du gestionnaire pour l'événement load de l'aperçu.


Le problème auquel vous êtes confronté est une conséquence d'un mauvais timing plutôt que d'une mauvaise logique. Plus précisément, cela réside dans le fait que vous définissez le nouveau src de votre aperçu, puis vérifiez les dimensions naturelles de l'image chargée de manière synchrone.

Cette approche est erronée car une image se charge de manière asynchrone et, par conséquent, votre code n'attend pas qu'elle se charge avant de procéder à votre vérification. Par conséquent, lorsque, après avoir sélectionné une image valide, vous en sélectionnez une non valide, vous vérifiez toujours les dimensions de la première car la dernière n'a pas encore été chargée.

Pour corriger cette erreur, vous devez effectuer votre vérification dans le gestionnaire pour l'événement load de l'aperçu, comme indiqué ci-dessous.

Extrait de travail:

(J'ai réécrit la fonction pour la rendre plus facile à lire)

function isFileImage (file) {
    const acceptedImageTypes = ["image/png"];
    return file && acceptedImageTypes.includes(file["type"])
}

function importFileandPreview () {
    var preview = document.querySelector("img");
    var file    = document.querySelector("input[type=file]").files[0];
    var reader  = new FileReader();

    /* Terminate the function prematurely if the file isn't an image. */
    if (!isFileImage(file)) return;

    /* Load the file into the reader. */
    reader.readAsDataURL(file);

    /* Set the 'load' event of the reader. */
    reader.addEventListener("load", function () {
        /* Create a temporary image to use for the check to avoid flashing. */
        var tempPreview = new Image();

        /* Set the 'load' event of the temporary preview. */
        tempPreview.addEventListener("load", function () {
            /* Check whether the selected image isn't 256x256 pixels. */
            if (this.naturalHeight != 256 || this.naturalWidth != 256) {
                /* Alert the user. */
                alert("Must be a PNG and have 256px X 256px!");

                /* Restore the default src to the preview. */
                preview.src = preview.dataset.defaultSrc;

                /* Terminate the function prematurely. */
                return;
            }

            /* Set the result of the reader as the source of the preview. */
            preview.src = reader.result;

            /* Save the image to the form. */
            saveImageToInput(reader.result);
        });

        /* Set the result of the reader as the source of the image. */
        tempPreview.src = reader.result;
    });

    /* Save the default image in a data-* attribute to use again if needed. */
    preview.dataset.defaultSrc = preview.dataset.defaultSrc || preview.src;
}


function saveImageToInput (image) {
    /* Find the hidden input. */
    var input = document.getElementById("imga");

    /* Check whether the input doesn't exist. */
    if (!input) {
        /* Create a new hidden input. */
        input = document.createElement("input");
        input.setAttribute("id", "imga");
        input.setAttribute("name", "img");
        input.setAttribute("type", "hidden");

        /* Place the input in the form. */
        document.getElementById("count").appendChild(input);
    }

    /* Save the image as the input's value. */
    input.setAttribute("value", image);
}
<input type="file" id="file" accept="image/PNG" onchange="importFileandPreview()"><br>
<img src=" " height="256" width="256">
<form id="count">
</form>

Remarques:

  1. C'est une bonne idée d'utiliser un aperçu temporaire pour vérifier si l'image sélectionnée répond à vos critères car, si vous la chargez dans l'aperçu visible puis alertez l'utilisateur, l'image chargée s'affichera.
  2. delete preview : Ne faites pas ça. Les variables en JavaScript ne sont pas censées être supprimées comme ça.
  3. Le troisième argument passé à addEventListener détermine si l'événement est capturé dans la phase de bullage ou de capture. Par défaut, c'est false, vous pouvez donc l'omettre.
0
Angel Politis 31 oct. 2020 à 00:44