记录--前端实现翻转图像

这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助

翻转图像是在视觉上比较两个不同图像的常用方法。单击其中一个将翻转它,并显示另一个图像。

布局

布局结构如下:

<div class="flipping-images">
    <div class="flipping-images__inner">
        <div class="flipping-images__side flipping-images__side--front">
            <!-- The image shown on the front -->
            <img class="flipping-images__img" src="..." />
        </div>

        <div class="flipping-images__side flipping-images__side--back">
            <!-- The image shown on the back -->
            <img class="flipping-images__img" src="..." />
        </div>
    </div>
</div>

它有正面和背面两面,放置在内部容器内。我们希望它们彼此重叠。最好的方法是使用容器 relative 的样式,并绝对定位它们。

.flipping-images__inner {
    /* Take full size of the root element */
    height: 100%;
    width: 100%;

    position: relative;
}

.flipping-images__side {
    /* Take full size of the inner container */
    height: 100%;
    width: 100%;

    /* Absolute position */
    position: absolute;
    top: 0;
    left: 0;
}

最初,正面显示在背面的上部。用户在单击正面之前不会看到背面。这就是 z-index 该属性派上用场的地方。

.flipping-images__side--back {
    z-index: 1;
}

.flipping-images__side--front {
    z-index: 2;
}

图像必须适合其侧面。在不显式设置 width 和 height 属性的情况下,我们可以使用 object-fit 属性将图像完美地放入每一侧:

.flipping-images__img {
    /* Take the full height */
    height: 100%;

    /* But don't exceed the side's width */
    max-width: 100%;

    /* Fit within each side */
    object-fit: cover;
}

内部居中

我们希望在根元素的中心显示内部容器。使用三个 CSS flexbox 属性的组合将使我们能够做到这一点:

.flipping-images {
    /* Center the content */
    align-items: center;
    display: flex;
    justify-content: center;
}

此外,我们还需要设置内容器的宽度。它必须与图像的宽度相同。我们可以处理其中一个图像 load 的事件,然后确定其宽度:

const handleLoad = (e) => {
    const imageEle = e.target;

    // Get the image's width
    const width = imageEle.getBoundingClientRect().width;

    // Assume `innerEle` represents the inner container
    innerEle.style.width = `${width}px`;
};


// Assume `containerEle` represents the root element
containerEle.querySelector('.flipping-images__img').addEventListener('load', handleLoad);

动画

为了获取您在本文开头看到的动画,我们需要在用户单击内部容器时旋转它。我们创建了一个翻转变体,将内部容器在垂直方向上旋转 180 度:

.flipping-images__inner {
    transition: transform 800ms;
}


.flipping-images__inner--flip {
    transform: rotateY(180deg);
}

当用户单击内部容器时,我们会切换翻转变体:

// Assume `innerEle` represents the inner container
innerEle.addEventListener('click', () => {
    innerEle.classList.toggle('flipping-images__inner--flip');
});

但是,结果是实际上只有第一个图像被旋转。背面的第二张图像仍处于隐藏状态。为了用背面替换正面,我们需要更多额外的样式:

.flipping-images__inner {
    transform-style: preserve-3d;
}

.flipping-images__side {
    backface-visibility: hidden;
}

这两个声明都是必需的。否则,当正面旋转时,无法看到背面。最后,由于我们旋转了内部容器,导致背面也旋转了。因此,我们必须反转旋转:

.flipping-images__side--back {
    transform: rotateY(-180deg);
}

3D动画

到目前为止,图像在内部容器内翻转。这种 perspective 将使我们有能力使翻转看起来像 3D 动画。

.flipping-images {
    perspective: 1000px;
}

如果 perspective 的值是内部容器宽度的两倍,则效果最佳。我们可以在上一节中提到的 load 事件处理程序中执行此操作:

const handleLoad = (e) => {
    const imageEle = e.target;

    // Get the image's width
    const width = imageEle.getBoundingClientRect().width;

    // Assume `containerEle` represents the root element
    containerEle.style.perspective = `${2 * width}px`;
};

水平翻转

我们使用该 rotateY 函数在垂直方向上翻转图像。如果要将翻转方向更改为水平,则可以使用该 rotateX 功能。

.flipping-images__inner--flip {
    transform: rotateX(180deg);
}

.flipping-images__side--back {
    transform: rotateX(-180deg);
}

本文转载于:

https://juejin.cn/post/7325389178926743587

如果对您有所帮助,欢迎您点个关注,我会定时更新技术文档,大家一起讨论学习,一起进步。

 

posted @ 2024-01-19 18:35  林恒  阅读(130)  评论(0编辑  收藏  举报