完整的项目,包括前端(Vue 3)和后端(Java Spring Boot),并确保所有功能都能正常工作。以下是详细的步骤:
1. 前端部分
1.1 创建 Vue 3 项目
首先,使用 Vue CLI 创建一个新的 Vue 3 项目:
bash
npm install -g @vue/cli
vue create vue-drag-and-drop-upload
cd vue-drag-and-drop-upload
在创建过程中,选择默认的 Vue 3 预设。
1.2 安装必要的依赖
安装 vuedraggable
和 axios
:
bash
npm install vuedraggable axios
1.3 创建组件 DragAndDropUpload.vue
在 src/components
目录下创建 DragAndDropUpload.vue
文件,并添加以下代码:
html
<template><div class="drag-and-drop-container"><div class="watermark"></div><divclass="drop-area"@dragenter.prevent@dragover.prevent@drop.prevent="onDrop"><p>Drag & Drop images here or click to upload</p><input type="file" multiple @change="onFileChange" accept="image/*" /></div><draggable:list="images"item-key="id"@end="onDragEnd"class="image-list"><template #item="{ element }"><div class="image-item"><img :src="element.url" alt="Uploaded Image" /><button @click="removeImage(element)">Remove</button><button v-if="!element.uploaded" @click="uploadImage(element.blob)">Upload</button><span v-else>Uploaded</span></div></template></draggable></div>
</template><script>
import { ref, onMounted } from 'vue';
import draggable from 'vuedraggable';
import axios from 'axios';export default {name: 'DragAndDropUpload',components: {draggable,},setup() {const images = ref([]);const onDrop = (event) => {const files = event.dataTransfer.files;handleFiles(files);};const onFileChange = (event) => {const files = event.target.files;handleFiles(files);};const handleFiles = (files) => {for (let i = 0; i < files.length; i++) {const file = files[i];if (file.type.startsWith('image/')) {compressImage(file).then(compressedBlob => {const reader = new FileReader();reader.onload = (e) => {images.value.push({id: Date.now() + i,url: e.target.result,blob: compressedBlob,uploaded: false,});};reader.readAsDataURL(compressedBlob);});}}};const removeImage = (image) => {images.value = images.value.filter(img => img.id !== image.id);};const onDragEnd = () => {console.log('Drag ended');};const compressImage = (file) => {return new Promise((resolve) => {const img = new Image();img.src = URL.createObjectURL(file);img.onload = () => {const canvas = document.createElement('canvas');let width = img.width;let height = img.height;// Maintain aspect ratio and set maximum dimensionsconst maxWidth = 800;const maxHeight = 600;if (width > height) {if (width > maxWidth) {height *= maxWidth / width;width = maxWidth;}} else {if (height > maxHeight) {width *= maxHeight / height;height = maxHeight;}}canvas.width = width;canvas.height = height;const ctx = canvas.getContext('2d');ctx.drawImage(img, 0, 0, width, height);canvas.toBlob(resolve, file.type, 0.8); // 0.8 is the quality of the compressed image};});};const uploadImage = (blob) => {const formData = new FormData();formData.append('file', blob);axios.post('http://172.168.0.1/upload', formData, {headers: {'Content-Type': 'multipart/form-data',},}).then(response => {console.log('Image processed successfully:', response.data);alert('Image processed successfully!');// Update the image preview with watermarked imageconst watermarkedUrl = `data:image/png;base64,${response.data.base64}`;const updatedImages = images.value.map(image => {if (image.blob === blob) {return { ...image, url: watermarkedUrl, uploaded: true };}return image;});images.value = updatedImages;}).catch(error => {console.error('Error processing image:', error);alert('Error processing image.');});};const addGlobalWatermark = () => {const watermarkDiv = document.createElement('div');watermarkDiv.className = 'global-watermark';watermarkDiv.textContent = 'Sample Watermark';document.body.appendChild(watermarkDiv);};onMounted(() => {addGlobalWatermark();});return {images,onDrop,onFileChange,handleFiles,removeImage,onDragEnd,compressImage,uploadImage,};},
};
</script><style scoped>
.drag-and-drop-container {display: flex;flex-direction: column;align-items: center;margin-top: 20px;
}.drop-area {border: 2px dashed #ccc;padding: 20px;text-align: center;width: 300px;position: relative;
}.drop-area input[type="file"] {position: absolute;top: 0;left: 0;width: 100%;height: 100%;opacity: 0;cursor: pointer;
}.image-list {display: flex;flex-wrap: wrap;justify-content: center;margin-top: 20px;
}.image-item {position: relative;margin: 10px;
}.image-item img {width: 100px;height: 100px;object-fit: cover;border-radius: 5px;
}.image-item button {position: absolute;bottom: 5px;right: 5px;background-color: red;color: white;border: none;padding: 5px 10px;border-radius: 5px;cursor: pointer;
}.image-item span {position: absolute;bottom: 5px;right: 5px;background-color: green;color: white;padding: 5px 10px;border-radius: 5px;
}
</style><style>
.global-watermark {position: fixed;top: 0;left: 0;width: 100%;height: 100%;z-index: 9999;pointer-events: none;display: flex;justify-content: center;align-items: center;font-size: 30px;font-weight: bold;color: rgba(255, 255, 255, 0.3);transform: rotate(-45deg);user-select: none;
}
</style>
1.4 更新 App.vue
在 src/App.vue
中引入并使用 DragAndDropUpload
组件:
html
<template><div id="app"><DragAndDropUpload /></div>
</template><script>
import DragAndDropUpload from './components/DragAndDropUpload.vue';export default {name: 'App',components: {DragAndDropUpload,},
};
</script><style>
#app {font-family: Avenir, Helvetica, Arial, sans-serif;-webkit-font-smoothing: antialiased;-moz-osx-font-smoothing: grayscale;text-align: center;color: #2c3e50;margin-top: 60px;
}
</style>
1.5 构建 Vue 项目
构建项目以生成静态文件:
bash
npm run build
这将在 dist
目录中生成构建后的文件。
2. 后端部分
2.1 创建 Spring Boot 项目
使用 Spring Initializr 创建一个新的 Spring Boot 项目。选择以下依赖:
- Spring Web
- Spring Boot DevTools
下载项目并解压。
2.2 添加依赖
在 pom.xml
中添加必要的依赖:
xml
<dependencies><!-- Spring Web --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><!-- Apache Commons IO --><dependency><groupId>commons-io</groupId><artifactId>commons-io</artifactId><version>2.11.0</version></dependency><!-- Lombok --><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><optional>true</optional></dependency><!-- Test dependencies --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency>
</dependencies>
2.3 创建控制器
创建一个 UploadController.java
文件来处理文件上传和处理请求:
java
package com.example.uploadservice.controller;import com.example.uploadservice.service.ImageService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.multipart.MultipartFile;@RestController
@RequestMapping("/api")
public class UploadController {@Autowiredprivate ImageService imageService;@PostMapping("/upload")public ResponseEntity<?> uploadFile(@RequestParam("file") MultipartFile file) {String base64Image = imageService.processImage(file);return ResponseEntity.ok().body(base64Image);}
}
2.4 创建服务
创建一个 ImageService.java
文件来处理图片压缩和水印添加:
java
package com.example.uploadservice.service;import org.apache.commons.io.IOUtils;
import org.springframework.stereotype.Service;import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;@Service
public class ImageService {public String processImage(MultipartFile file) {try {BufferedImage originalImage = ImageIO.read(new ByteArrayInputStream(file.getBytes()));BufferedImage resizedImage = resize(originalImage, 800, 600);BufferedImage watermarkedImage = addWatermark(resizedImage, "Sample Watermark");return encodeToBase64(watermarkedImage);} catch (IOException e) {e.printStackTrace();throw new RuntimeException("Failed to process image", e);}}private BufferedImage resize(BufferedImage img, int maxW, int maxH) {int w = img.getWidth();int h = img.getHeight();float ratio = (float) w / h;if (w > maxW || h > maxH) {if (ratio >= 1) {h = Math.round(maxW / ratio);w = maxW;} else {w = Math.round(maxH * ratio);h = maxH;}}BufferedImage resizedImg = new BufferedImage(w, h, BufferedImage.TYPE_INT_ARGB);Graphics2D g2d = resizedImg.createGraphics();g2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR);g2d.drawImage(img, 0, 0, w, h, null);g2d.dispose();return resizedImg;}private BufferedImage addWatermark(BufferedImage img, String watermarkText) {int width = img.getWidth();int height = img.getHeight();BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB);Graphics2D graphics = bufferedImage.createGraphics();graphics.drawImage(img, 0, 0, null);Font font = new Font("Arial", Font.BOLD, 30);Color color = new Color(255, 255, 255, 128);graphics.setFont(font);graphics.setColor(color);FontMetrics metrics = graphics.getFontMetrics(font);int x = (width - metrics.stringWidth(watermarkText)) / 2;int y = (height - metrics.getHeight()) / 2 + metrics.getAscent();graphics.drawString(watermarkText, x, y);graphics.dispose();return bufferedImage;}private String encodeToBase64(BufferedImage img) throws IOException {ByteArrayOutputStream baos = new ByteArrayOutputStream();ImageIO.write(img, "png", baos);byte[] bytes = baos.toByteArray();return IOUtils.encodeBase64String(bytes);}
}
2.5 运行 Spring Boot 应用
在项目根目录下运行 Spring Boot 应用:
bash
./mvnw spring-boot:run
确保防火墙允许 172.168.0.1
的 8080 端口流量。
3. Nginx 配置
3.1 创建一个新的站点配置文件
创建一个新的配置文件 /etc/nginx/sites-available/vue-drag-and-drop-upload
:
bash
sudo nano /etc/nginx/sites-available/vue-drag-and-drop-upload
3.2 编辑配置文件
在 vue-drag-and-drop-upload
文件中添加以下内容:
conf
server {listen 80;server_name example.com www.example.com;# Root directory of the built Vue approot /path/to/your/project/dist;index index.html;location / {try_files $uri $uri/ /index.html;}# Proxy API requests to backend serverlocation /upload {proxy_pass http://172.168.0.1:8080/api/upload;proxy_set_header Host $host;proxy_set_header X-Real-IP $remote_addr;proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;proxy_set_header X-Forwarded-Proto $scheme;}
}
请将 /path/to/your/project/dist
替换为你的实际项目路径。
3.3 启用站点配置
创建符号链接到 sites-enabled
目录以启用该站点配置:
bash
sudo ln -s /etc/nginx/sites-available/vue-drag-and-drop-upload /etc/nginx/sites-enabled/
3.4 测试配置并重新加载 Nginx
测试 Nginx 配置是否有语法错误:
bash
sudo nginx -t
如果没有错误,重新加载 Nginx 以应用新的配置:
bash
sudo systemctl reload nginx
总结
通过以上步骤,你已经成功实现了以下功能:
- Vue 3.0 项目:使用 Vue CLI 创建项目,并实现拖拽图片上传、等比压缩和前端预览。
- 图片压缩:使用
canvas
进行图片等比压缩。 - 图片预览:在前端预览压缩后的图片。
- 图片上传:使用
axios
将压缩后的图片上传到172.168.0.1
服务器。 - Java 服务端:使用 Spring Boot 处理图片上传、添加水印并返回 Base64 格式的图片。
- Nginx 配置:配置 Nginx 以提供静态文件服务,并设置反向代理以处理 API 请求,使用 HTTP 模式。
- 图片删除:支持从列表中删除图片。
- 全局水印:在页面上添加全局水印。
希望这个完整的示例对你有所帮助!