213 lines
No EOL
5.4 KiB
JavaScript
213 lines
No EOL
5.4 KiB
JavaScript
import "../../../src/components/VImg/VImg.sass"; // Components
|
|
|
|
import VResponsive from '../VResponsive'; // Utils
|
|
|
|
import { consoleError, consoleWarn } from '../../util/console';
|
|
/* @vue/component */
|
|
|
|
export default VResponsive.extend({
|
|
name: 'v-img',
|
|
props: {
|
|
alt: String,
|
|
contain: Boolean,
|
|
gradient: String,
|
|
lazySrc: String,
|
|
position: {
|
|
type: String,
|
|
default: 'center center'
|
|
},
|
|
sizes: String,
|
|
src: {
|
|
type: [String, Object],
|
|
default: ''
|
|
},
|
|
srcset: String,
|
|
transition: {
|
|
type: [Boolean, String],
|
|
default: 'fade-transition'
|
|
}
|
|
},
|
|
|
|
data() {
|
|
return {
|
|
currentSrc: '',
|
|
image: null,
|
|
isLoading: true,
|
|
calculatedAspectRatio: undefined,
|
|
naturalWidth: undefined
|
|
};
|
|
},
|
|
|
|
computed: {
|
|
computedAspectRatio() {
|
|
return Number(this.normalisedSrc.aspect || this.calculatedAspectRatio);
|
|
},
|
|
|
|
normalisedSrc() {
|
|
return typeof this.src === 'string' ? {
|
|
src: this.src,
|
|
srcset: this.srcset,
|
|
lazySrc: this.lazySrc,
|
|
aspect: Number(this.aspectRatio)
|
|
} : {
|
|
src: this.src.src,
|
|
srcset: this.srcset || this.src.srcset,
|
|
lazySrc: this.lazySrc || this.src.lazySrc,
|
|
aspect: Number(this.aspectRatio || this.src.aspect)
|
|
};
|
|
},
|
|
|
|
__cachedImage() {
|
|
if (!(this.normalisedSrc.src || this.normalisedSrc.lazySrc)) return [];
|
|
const backgroundImage = [];
|
|
const src = this.isLoading ? this.normalisedSrc.lazySrc : this.currentSrc;
|
|
if (this.gradient) backgroundImage.push(`linear-gradient(${this.gradient})`);
|
|
if (src) backgroundImage.push(`url("${src}")`);
|
|
const image = this.$createElement('div', {
|
|
staticClass: 'v-image__image',
|
|
class: {
|
|
'v-image__image--preload': this.isLoading,
|
|
'v-image__image--contain': this.contain,
|
|
'v-image__image--cover': !this.contain
|
|
},
|
|
style: {
|
|
backgroundImage: backgroundImage.join(', '),
|
|
backgroundPosition: this.position
|
|
},
|
|
key: +this.isLoading
|
|
});
|
|
if (!this.transition) return image;
|
|
return this.$createElement('transition', {
|
|
attrs: {
|
|
name: this.transition,
|
|
mode: 'in-out'
|
|
}
|
|
}, [image]);
|
|
}
|
|
|
|
},
|
|
watch: {
|
|
src() {
|
|
if (!this.isLoading) this.init();else this.loadImage();
|
|
},
|
|
|
|
'$vuetify.breakpoint.width': 'getSrc'
|
|
},
|
|
|
|
mounted() {
|
|
this.init();
|
|
},
|
|
|
|
methods: {
|
|
init() {
|
|
if (this.normalisedSrc.lazySrc) {
|
|
const lazyImg = new Image();
|
|
lazyImg.src = this.normalisedSrc.lazySrc;
|
|
this.pollForSize(lazyImg, null);
|
|
}
|
|
/* istanbul ignore else */
|
|
|
|
|
|
if (this.normalisedSrc.src) this.loadImage();
|
|
},
|
|
|
|
onLoad() {
|
|
this.getSrc();
|
|
this.isLoading = false;
|
|
this.$emit('load', this.src);
|
|
},
|
|
|
|
onError() {
|
|
consoleError(`Image load failed\n\n` + `src: ${this.normalisedSrc.src}`, this);
|
|
this.$emit('error', this.src);
|
|
},
|
|
|
|
getSrc() {
|
|
/* istanbul ignore else */
|
|
if (this.image) this.currentSrc = this.image.currentSrc || this.image.src;
|
|
},
|
|
|
|
loadImage() {
|
|
const image = new Image();
|
|
this.image = image;
|
|
|
|
image.onload = () => {
|
|
/* istanbul ignore if */
|
|
if (image.decode) {
|
|
image.decode().catch(err => {
|
|
consoleWarn(`Failed to decode image, trying to render anyway\n\n` + `src: ${this.normalisedSrc.src}` + (err.message ? `\nOriginal error: ${err.message}` : ''), this);
|
|
}).then(this.onLoad);
|
|
} else {
|
|
this.onLoad();
|
|
}
|
|
};
|
|
|
|
image.onerror = this.onError;
|
|
image.src = this.normalisedSrc.src;
|
|
this.sizes && (image.sizes = this.sizes);
|
|
this.normalisedSrc.srcset && (image.srcset = this.normalisedSrc.srcset);
|
|
this.aspectRatio || this.pollForSize(image);
|
|
this.getSrc();
|
|
},
|
|
|
|
pollForSize(img, timeout = 100) {
|
|
const poll = () => {
|
|
const {
|
|
naturalHeight,
|
|
naturalWidth
|
|
} = img;
|
|
|
|
if (naturalHeight || naturalWidth) {
|
|
this.naturalWidth = naturalWidth;
|
|
this.calculatedAspectRatio = naturalWidth / naturalHeight;
|
|
} else {
|
|
timeout != null && setTimeout(poll, timeout);
|
|
}
|
|
};
|
|
|
|
poll();
|
|
},
|
|
|
|
genContent() {
|
|
const content = VResponsive.options.methods.genContent.call(this);
|
|
|
|
if (this.naturalWidth) {
|
|
this._b(content.data, 'div', {
|
|
style: {
|
|
width: `${this.naturalWidth}px`
|
|
}
|
|
});
|
|
}
|
|
|
|
return content;
|
|
},
|
|
|
|
__genPlaceholder() {
|
|
if (this.$slots.placeholder) {
|
|
const placeholder = this.isLoading ? [this.$createElement('div', {
|
|
staticClass: 'v-image__placeholder'
|
|
}, this.$slots.placeholder)] : [];
|
|
if (!this.transition) return placeholder[0];
|
|
return this.$createElement('transition', {
|
|
attrs: {
|
|
name: this.transition
|
|
}
|
|
}, placeholder);
|
|
}
|
|
}
|
|
|
|
},
|
|
|
|
render(h) {
|
|
const node = VResponsive.options.render.call(this, h);
|
|
node.data.staticClass += ' v-image';
|
|
node.data.attrs = {
|
|
role: this.alt ? 'img' : undefined,
|
|
'aria-label': this.alt
|
|
};
|
|
node.children = [this.__cachedSizer, this.__cachedImage, this.__genPlaceholder(), this.genContent()];
|
|
return h(node.tag, node.data, node.children);
|
|
}
|
|
|
|
});
|
|
//# sourceMappingURL=VImg.js.map
|