为了账号安全,请及时绑定邮箱和手机立即绑定

函数处理的不是变量的最后一条语句?

函数处理的不是变量的最后一条语句?

繁星coding 2023-06-29 21:03:33
我正在编写这段代码,目标是:检查是否是PNG,检查是否是256X256,并将img转换为base64以上传到服务器。因此,如果我选择 256 x 256 img,它会加载,但是,如果我在选择 256 x 256 后加载另一个不是 256 x 256 的 img,它会加载,我不知道为什么!我该如何解决?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><form id="count"></form>
查看完整描述

1 回答

?
叮当猫咪

TA贡献1776条经验 获得超12个赞

您需要在预览事件的处理程序内对图像的自然尺寸进行检查load。


你所面临的问题是由于时机不当而不是逻辑错误造成的。更具体地说,它在于您设置新的预览,然后同步src检查加载图像的自然尺寸。


这种方法是错误的,因为图像是异步加载的,因此您的代码不会等待它加载后再继续检查。因此,当您在选择有效图像后选择无效图像时,您仍然会检查前者的尺寸,因为后者尚未加载。


要更正此错误,您必须在预览事件的处理程序内执行检查load,如下所示。


工作片段:


(我重写了该函数以使其更易于阅读)


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>

笔记:

  1. 最好使用临时预览来检查所选图像是否满足您的条件,因为如果您将其加载到可见预览中然后提醒用户,则将显示加载的图像。

  2. delete preview: 不要这样做。JavaScript 中的变量不应该像这样被删除。

  3. 传递给的第三个参数addEventListener确定事件是在冒泡阶段还是捕获阶段捕获。默认情况下,它是false,因此您可以省略它。


查看完整回答
反对 回复 2023-06-29
  • 1 回答
  • 0 关注
  • 66 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信