零成本!无需服务器,搭建你的图床!

news2025/1/15 17:50:12

先给大家看看成品:
image.png

访问地址:http://cp64mba5g6h48vl4is50.app.memfiredb.cn/

这是我花十分钟做出来的零成本,不需要服务器的图床,不需要登录,任何人都可以在上面上传图片和拿到图片链接去使用,当然这只是一个简单的图床功能,后续我会慢慢优化更强大。接下来我将教大家如何做一个零成本,无需服务器的免费图床。

1.创建免费应用

先去前往MemFire Cloud登录后,在首页创建一个免费的应用

image.png

这个应用创建完成后,你将拥有

  • 免费云数据库
  • 自动生成api
  • 对象存储
  • 二十几种第三方认证服务
  • 实时数据库
  • 静态托管

然后我们这个图床主要用到的是自动生成api和对象存储

2.创建存储桶

应用创建完成后,我们先去存储的界面创建一个存储桶用来存放图片

image.png

image.png

3.初始化vue项目

大家可以用npx或者vue创建都可以,我是用vue创建的

vue create vue-tuchuang

创建完成后,在根目录下创建一个.env.local,来存放环境变量

4.安装依赖包

安装用于访问应用资源的依赖包

npm i @supabase/supabase-js

5.配置环境变量

在MemFire Cloud应用控制台的“应用设置”->“API”找到应用的URL和anon 公钥,分别填到.env.local里面

image.png

VITE_SUPABASE_URL=
VITE_SUPABASE_ANON_KEY=

6.初始化客户端

在src下面创建一个supabase.js,将下面的代码填入

import { createClient } from '@supabase/supabase-js'

const supabaseUrl = import.meta.env.VITE_SUPABASE_URL
const supabaseAnonKey = import.meta.env.VITE_SUPABASE_ANON_KEY

export const supabase = createClient(supabaseUrl, supabaseAnonKey)

7.编写页面

src\App.vue

<template>
  <div id="app">
    <header>
      <h1>图床</h1>
      <input type="file" @change="handleFileUpload" accept="image/*" />
    </header>
    <main>
      <section v-if="fileList.length > 0">
        <h2>已上传的图片</h2>
        <ul class="image-list">
          <li v-for="(image, index) in fileList" :key="index" class="image-item">
            <img :src="image.publicUrl" alt="uploaded image" @click="showImage(index)" />
            <div class="image-overlay">
              <span class="image-name">{{ truncateFileName(image.name) }}</span>
              <button @click="copyImageUrl(index)">复制URL</button>
            </div>
          </li>
        </ul>
      </section>
      <section v-else>
        <p style="color: black;">暂无上传的图片</p>
      </section>
    </main>
    <!-- 放大查看图片的弹窗 -->
    <div v-if="showModal" class="modal" @click="closeModal">
      <div class="modal-content">
        <img :src="selectedImageUrl" alt="enlarged image" />
      </div>
    </div>
  </div>
</template>

<script>
import { ref, onMounted } from 'vue';
import { supabase } from './supabase';

export default {
  setup() {
    const fileList = ref([]);
    const files = ref();
    const showModal = ref(false);
    const selectedImageUrl = ref('');

    onMounted(() => {
      getImageUrl();
    });

    const handleFileUpload = async (evt) => {
      files.value = evt.target.files;
      try {
        if (!files.value || files.value.length === 0) {
          throw new Error('请选择要上传的图片.');
        }
        const file = files.value[0];
        const fileExt = file.name.split('.').pop();
        const filePath = `${Math.random()}.${fileExt}`;
        let { error: uploadError } = await supabase.storage.from('drawingBoard').upload(filePath, file);
        if (uploadError) throw uploadError;
        getImageUrl();
      } catch (error) {
        alert(error.message);
      }
    };

    const copyImageUrl = (index) => {
      const publicUrl = fileList.value[index].publicUrl;
      navigator.clipboard
        .writeText(publicUrl)
        .then(() => {
          alert(`已复制图片URL:${publicUrl}`);
        })
        .catch((err) => {
          console.error('复制失败:', err);
        });
    };

    const getImageUrl = async () => {
      const { data, error } = await supabase
        .storage
        .from('drawingBoard')
        .list()
        const filteredData = data.filter(item => item.name !== ".emptyFolderPlaceholder");
        filteredData.forEach((item) => {
        const { data } = supabase
          .storage
          .from('drawingBoard')
          .getPublicUrl(item.name)
        if (data) {
          item.publicUrl = data.publicUrl
        }
      })
      if(filteredData.length >0){
        fileList.value = filteredData
      }
     

    }

    const showImage = (index) => {
      selectedImageUrl.value = fileList.value[index].publicUrl;
      showModal.value = true;
    };

    const closeModal = () => {
      showModal.value = false;
    };

    const truncateFileName = (fileName) => {
      const maxLength = 20; // 设置名称最大长度
      if (fileName.length > maxLength) {
        return fileName.substring(0, maxLength - 3) + '...'; // 超过长度则省略
      }
      return fileName;
    };

    return {
      fileList,
      handleFileUpload,
      copyImageUrl,
      showImage,
      showModal,
      selectedImageUrl,
      closeModal,
      truncateFileName
    };
  },
};
</script>

<style>
#app {
  font-family: Arial, sans-serif;
  margin: 0 auto;
  max-width: 800px;
  padding: 20px;
}

header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  margin-bottom: 20px;
}
h2{
  color: #000;
}
header h1{
  font-size: 24px;
  margin: 0;
}

input[type='file'] {
  display: inline-block;
  padding: 10px;
}

main {
  background-color: #f5f5f5;
  padding: 20px;
}
ul.image-list li img:hover {
 transform: scale(1.1); /* Enlarge on hover */
}
ul.image-list {
  list-style: none;
  padding: 0;
  display: flex;
  flex-wrap: wrap;
  gap: 20px;
}

.image-item {
  width: calc(33.33% - 20px);
  text-align: center;
  margin-bottom: 20px;
  position: relative;
}

.image-item img {
  /* max-width: 100%;
  border-radius: 5px;
  box-shadow: 0 0 5px rgba(0, 0, 0, 0.1);
  cursor: pointer; */

  max-width: 100%;
 height: auto; /* Maintain aspect ratio */
 border-radius: 5px;
 box-shadow: 0 0 5px rgba(0, 0, 0, 0.1);
 cursor: pointer; /* Show pointer on hover for clicking */
 transition: transform 0.2s ease; /* Add smooth transition for scaling */
}

.image-overlay {
  position: absolute;
  bottom: 10px;
  left: 0;
  right: 0;
  background-color: rgba(0, 0, 0, 0.5);
  padding: 5px;
  border-radius: 5px;
}

.image-overlay .image-name {
  display: block;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
  color: #000; /* 设置名称颜色为黑色 */
}

button {
  padding: 5px 10px;
  margin-left: 10px;
  background-color: #007bff;
  color: white;
  border: none;
  border-radius: 3px;
  cursor: pointer;
}

/* 弹窗样式 */
.modal {
  position: fixed;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  background-color: rgba(0, 0, 0, 0.7);
  display: flex;
  justify-content: center;
  align-items: center;
  z-index: 999;
}

.modal-content {
  max-width: 80%;
  max-height: 80%;
  overflow: auto;
}

.modal-content img {
  max-width: 100%;
  max-height: 100%;
  border-radius: 5px;
  box-shadow: 0 0 10px rgba(0, 0, 0, 0.5);
}
</style>

8.运行程序

npm run dev

效果:

image.png

9.静态托管

项目开发完了,大家肯定想把这个作品展示给自己朋友或者网友使用,刚好我们使用MemFire Cloud的静态托管来托管我们的项目,就相当于部署我们的项目,我们不需要在服务器上部署。
在根目录下执行命令

npm run build

打包完成后,我们进入文件资源管理器,将里面的所有文件选中打成一个zip压缩包

image.png

image.png

我们再次打开MemFire Cloud应用的控制台,我们来到静态托管的选项下,然后将我们刚刚打包好的压缩包上传到这里,上传完成后会发现上面出现一个访问地址,这个地址就可以给大家安全的访问啦,如果你觉得这个地址有点丑,你还可以去配置域名,前提是你已经有了一个域名并且已经备案了。

image.png

好啦,咱们这个简单的图床就做好了,全程没有需要服务器,零成本完成的!后续我会慢慢优化这个图床!大家也可以试试用MemFire Cloud来做一些应用。

具体的大家可以参考或者阅读一下文档

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1839044.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux系统安装ODBC驱动,统信服务器E版安装psqlodbc方法

应用场景 硬件/整机信息&#xff1a;AMD平台 OS版本信息&#xff1a;服务器e版 软件信息&#xff1a;psqlodbc 12.02版本 功能介绍 部分用户在使用etl工具连接数据库时&#xff0c;需要使用到odbc驱动&#xff0c;下面介绍下服务器e版系统中编译安装此工具的相关过程。 E…

深度学习算法面经(高频核心问题总结,持续更新)

学习的过程短期目标是丰富己身&#xff0c;长远来看有的人为了就业财富自由&#xff1b;有的则为了创造一些有意义的事物&#xff0c;更多的是为了前者。 此文章用于记录和总结深度学习相关算法岗的各种面试问题&#xff0c;搜集答案并加入博主一些浅显的理解,欢迎评论区纠正、…

Python+Selenium之断言

一、Assert 用于判断一个表达式&#xff0c;在表达式条件为 false 的时候触发异常。 #获取对应元素的文本值text02driver.find_element_by_xpath("//h1[text()用户登录]").text#判断text02是否包含“用户登录”字符串&#xff0c;包含即成功&#xff0c;不包含即失败…

同三维T80002JEHV H.265高清解码器

同三维T80002JEHV H.265高清解码器 1路HDMI1路VGA解码输出&#xff0c;1/2/4画面分割或16路轮询显示 产品简介&#xff1a; 同三维T80002JEHV解码器使用Linux系统&#xff0c;支持VGA/HDMI二种接口同时输出&#xff0c;支持多流输入多流解码及多屏显示&#xff0c;具有完善的…

【广度优先搜索 深度优先搜索 图论】854. 相似度为 K 的字符串

本文涉及知识点 广度优先搜索 深度优先搜索 图论 图论知识汇总 LeetCode 854. 相似度为 K 的字符串 对于某些非负整数 k &#xff0c;如果交换 s1 中两个字母的位置恰好 k 次&#xff0c;能够使结果字符串等于 s2 &#xff0c;则认为字符串 s1 和 s2 的 相似度为 k 。 给你…

功能测试 之 单模块测试----抢购模块

1.先测后台&#xff0c;再测前台 面试题1: 当你发现研发实现的结果与需求不一致时怎么办? 需求评审的时候&#xff1a;需要确认所有输入类型的校验是针对单独的输入框做的还是在最终提交时校验 抢购模块&#xff1a;需求跟实现的内容不一致 (跟产品和研发一起确认。研发为什…

JLINK调试妙用----读写flash

JLINK调试妙用----读写flash 前言 随着对jlink使用频率的增加&#xff0c;越发觉得它太强大了&#xff0c;可以满足好多调试功能&#xff0c;现在已经用的就是J-flash&#xff08;程序下载&#xff09;、RTT&#xff08;日志打印&#xff09;和J-flash SPI&#xff08;读写spi类…

dotnet new 命令详解

一、简介 dotnet new 命令用于基于指定的模板创建新项目、配置文件、解决方案。 二、常用选项 -o, --output <output>&#xff1a;指定创建项目后放置的目录名 示例&#xff1a; dotnet new console -o MyConsoleApp-n, --name <name>&#xff1a;指定项目的名…

【微服务网关——负载均衡】

1. 四大负载均衡策略 随机负载 随机挑选目标服务器IP 轮询负载 ABC三台服务器&#xff0c;ABCABC依次轮询 加权负载 给目标设置访问权重&#xff0c;按照权重轮询 一致性hash负载 请求固定URL访问指定IP 2.随机负载均衡 可以通过random函数来随机选择一个ip 2.1 代码实现 …

信号与系统概述

信号是消息的表现形式或传送载体&#xff0c;数学上用函数来表示。信号可以分为确定信号和随机信号、连续时间信号和离散时间信号、周期信号和非周期信号、能量信号和功率信号等。模拟信号是幅度连续的连续时间信号&#xff0c;而数字信号是幅度离散的离散时间信号。对于连续时…

一键解压,无限可能——BetterZip,您的Mac必备神器!

BetterZip for Mac 是一款高效、智能且安全的解压缩软件&#xff0c;专为Mac用户设计。它提供了直观易用的界面&#xff0c;使用户能够轻松应对各种压缩和解压缩需求。 这款软件不仅支持多种压缩格式&#xff0c;如ZIP、RAR、7Z等&#xff0c;还具备快速解压和压缩文件的能力。…

解锁TikTok数据潜力——高效获取TikTok视频评论回复数据接口

一、引言 在社交媒体蓬勃发展的今天&#xff0c;TikTok已成为全球范围内备受欢迎的短视频平台。为了帮助企业、个人和开发者更好地利用TikTok数据&#xff0c;我们推出了一款全新的接口服务&#xff0c;专注于高效获取TikTok视频评论回复数据。 二、核心功能介绍 高效获取评论…

储备教师和正式教师的区别是什么?

当谈论教育行业的未来&#xff0c;是否曾想过&#xff0c;那些被称为"储备教师"的群体&#xff0c;与我们熟知的"正式教师"之间&#xff0c;有何本质的区别&#xff1f; 储备教师&#xff0c;顾名思义&#xff0c;是学校为了应对未来可能的教学需求而提前招…

DuDuTalk:智能电子录音工牌在销售场景的应用价值

在快速变化的市场环境中&#xff0c;销售团队面临着日益激烈的竞争和不断变化的客户需求。为了提升销售效率、优化客户体验并加强团队协作&#xff0c;越来越多的企业开始采用智能电子录音工牌作为销售场景中的关键工具。本文将从多个方面探讨智能电子录音工牌在销售场景中的应…

渗透测试之存储型跨站脚本攻击(高危)

一、定义 跨站脚本攻击&#xff0c;指的是恶意用户往web页面里插入恶意HTML代码。当普通用户访问该web页面&#xff0c;嵌入其中的HTML代码会被执行&#xff0c;从而达到破坏的效果。 二、风险定级 高危 三、可输入的HTML标签示例 图片标签 <img src"#"> 超…

RIP与OSPF发布默认路由(华为)

#交换设备 RIP与OSPF发布默认路由 合理使用默认路由可以很大程度上减少本地路由表的大小&#xff0c;并可以较好的隐藏一个网络中的路由信息&#xff0c;保护自身网络的隐秘性 另外如果在同一个路由器两端使用了不同的路由协议&#xff0c;那么如果不做路由引入或者发布默认…

Excel 找出最大值及其相邻的 N 个成员

某列都是数值&#xff1a; A1132213464215496973482396101113712491342144015151631171718114719182030212222423252419251326272738283029163012312332333233419351436463723383739384028 请找出最大值及其相邻的 10 个成员&#xff0c;注意越界检查&#xff0c;实际符合条件…

华宽通成功中标“在星沙”平台智慧后勤板块项目

湖南华宽通科技有限公司凭借其在智慧园区领域的专业实力&#xff0c;成功中标“在星沙”平台智慧后勤板块项目。此次项目合作华宽通将充分发挥在智慧园区领域的优势&#xff0c;为“在星沙”平台提供全方位、一体化的智慧后勤解决方案&#xff0c;助力实现后勤管理平台的智能化…

k8s上尝试滚动更新和回滚

滚动更新和回滚 实验目标&#xff1a; 学习如何进行应用的滚动更新和回滚操作。 实验步骤&#xff1a; 创建一个 Deployment。更新 Deployment 的镜像版本&#xff0c;观察滚动更新过程。回滚到之前的版本&#xff0c;验证回滚操作。 今天呢&#xff0c;我们继续来进行我们k…

自学鸿蒙HarmonyOS的ArkTS语言<一>基本语法

一、一个ArkTs的目录结构 二、一个页面的结构 A、装饰器 Entry 装饰器 : 标记组件为入口组件&#xff0c;一个页面由多个自定义组件组成&#xff0c;但是只能有一个组件被标记 Component : 自定义组件, 仅能装饰struct关键字声明的数据结构 State&#xff1a;组件中的状态变量…