vue动态定义图片路径

最近在使用vue动态设置图片路径的时候遇到了一些问题,特此整理。

我想实现的效果:点击图片,弹出系统图片选择框,选择图片后替换原图片。

App.vue中:

<input type="file" id="avatar" @change="chooseFile($event)" accept="image/*" style="display: none" >
<label for="avatar">
    <img :src="../assets/user_1.png" alt="图片">
</label>

结果:

默认图片加载成功,点击该图片可以弹出图片选择框。

本着组件抽象的原则,我想把上述代码抽象成组件,并且希望默认的图片url可以在父组件指定,于是,我这样写:
App.vue:

<template>
    <ItemShow-imageChoose id="userPicture" imageSrc="../assets/user.png"></ItemShow-imageChoose>
</template>
<script>
    import ItemShow_imageChoose from "./components/ItemShow-imageChoose"
    export default {
      name:'App',
      components: {
          "ItemShow-imageChoose":ItemShow_imageChoose
      }
    }

./components/ItemShow-imageChoose.vue:

<template>
    <div>
      <input type="file" :id="id" @change="chooseFile($event)" accept="image/*" style="display: none" >
        <label :for="id">
          <img :src="imageSrc" alt="图片">
        </label>
    </div>
  </template>
  <script>
      export default{
          name:'ItemShow_imageChoose',
          props:{
              "id":[String],
              "imageSrc":[String]
          },
          methods:{
              chooseFile(e){
//                  this.$emit('choose',e.target.files)
              }
          }
      }
</script>

结果:

默认图片没有加载出来。

开始一步步找错....

首先,把图片的src移至子组件的data中,看图片是否能加载出来:

./components/ItemShow-imageChoose.vue:

<template>
    <div>
        <input type="file" :id="id" @change="chooseFile($event)" accept="image/*" style="display: none" >
        <label :for="id">
            <img :src="src" alt="图片" />
        </label>
    </div>
</template>
<script>
    export default{
        name:'ItemShow_imageChoose',
        props:{
            "id":[String],
            "imageSrc":[String]
        },
        data(){
          return {
              src:'../assets/user.png'    //重点看这里
          }
        },
        methods:{
            chooseFile(e){
//                this.$emit('choose',e.target.files)
            }
        }
    }
</script>

结果:

本来以为这样写肯定是对的,结果竟然不对,但是这个结果也说明:问题不是出在父子组件的参数传递上。

于是开始查资料....

看了一些文章和问答,也大致理解了出错的原因(参考文章见附录),是图片路径的问题。

如果图片地址是直接写死在html或者css里的,webpack会帮你处理这个图片最终的地址(要用到url-loader),例如:
初始:

<template>
    <img src="./相对地址.jpg" />
</template>

webpack编译后会变成:

    <img src="/绝对地址.jpg" />
    <!-- 或者 -->
    <img src="data:image/jpg;base64......." />

实测:

<template>
    <img src="../assets/user.png" />
</template>

打包成绝对路径:

打包成base64字符串:

项目目录:

以上就是webpack打包的两种方式,我们重点看第一种。第一种方式,图片是以绝对路径(以/开头的路径就是绝对路径,/指根目录,根目录在本地就是指磁盘,在github上就是指仓库的根目录,在网站上就是指服务器的根目录)进行查找,图片的目录为/static/img,但是我们查看上图项目目录,发现static目录下并没有img目录,也没有图片,那么这里的路径是怎么来的呢?

运行npm run build,再看一下项目目录:

我们在dist目录下面找到了该图片。由此知道,webpack打包后,会将静态资源文件放在dist/static/img下,我们的网站实际上以dist目录作为根目录,并由此加载该目录下的index.html所需的css、js、img等

现在我们使用Vue.js来动态定义图片路径:

<template>
    <div>
        <img :src="src" alt="图片" />
    </div>
</template>
<script>
    export default{
        data(){
          return {
              src:`../assets/user.png`
          }
        }
    }
</script>

结果:

dist目录为根目录,index.html的路径为“./index.html”,那“../assets/user.png”即为与dist目录平级的assets目录下面的user.png,我们发现该目录不存在,图片自然加载失败。

当Vue.js来动态定义图片路径的时候,url-loader是无法探测到图片路径的。我们build后发现,图片根本不会打包输出到dist目录(webpack是按需打包的):

由上图可知,dist目录下无图片文件或文件夹。

如何解决?

我整理并测试了以下三种解决办法:

1、绝对路径访问

把图片放到静态资源目录static目录下(build 会将static目录中的文件或者文件夹按照原本的结构放在dist目录下),并用/static绝对路径访问:

<template>
    <div>
        <img :src="src" alt="图片" />
    </div>
</template>
<script>
    export default{
        data(){
          return {
              src:`/static/img/user.png`
          }
        }
    }
</script>

结果:
2、使用require

如果想在不调整目录结构的情况下读取图片,还可以使用require:

data(){
     return {
         src:require('../assets/user.png')    //重点看这里
    }
}

该结果与上面说的webpack打包的第一种方式结果相同。

缺点:由于CommonJS只允许使用字符串字面量,所以不利于组件化,灵活性较差。

3、使用import

也可以用import引入图片路径:

<template>
    <div>
        <img :src="src" alt="图片" />
    </div>
</template>
<script>
    import userPath from '../assets/user.png'
    export default{
        data(){
          return {
              src:userPath 
          }
        }
    }
</script>

结果:

该结果与上面说的webpack打包的第一种方式结果相同。

这篇博客就先整理到这里。多花些时间踩踩坑也是很好的,学到了就是赚到了。由于个人水平有限,博客错误之处,烦请指正!

附完整代码

./components/ItemShow-imageChoose.vue:

<template>
    <div>
        <input type="file" :id="id" @change="chooseFile($event)" accept="image/*" style="display: none" >
        <label :for="id">
            <img :src="'/static/img/'+imageSrc" alt="图片" />
        </label>
    </div>
</template>
<script>
   export default{
        name:'ItemShow_imageChoose',
        props:{
            "id":[String],
            "imageSrc":[String]
        },
        methods:{
            chooseFile(e){
                var currentImg=e.target.nextElementSibling.childNodes[0]
                currentImg.setAttribute('src',window.URL.createObjectURL(e.target.files[0]))
                currentImg.onload=function () {
                    window.URL.revokeObjectURL(this.src)
                }
            }
        }
    }

</script>

参考:
1、Vue中img的src属性绑定与static文件夹
2、问:vue+webpack动态设置图片src导致404错误

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容