On the right side of the page, there is a directory index, which can jump to the content you want to see according to the title |
---|
If not on the right, look for the left |
This article is suitable for students who have done full stack development. At least they need to be able to build the front and back-end environment of vue+spring boot and the basic front and back-end interaction logic. Otherwise, you can't understand it and at least you can't test it |
---|
If you are just a front-end engineer, you can mock the response yourself |
- I hope you can support this up and speak really well
1, Construction of helloworld environment
1. Front end environment construction
- Download the Vue simple upload source code https://github.com/simple-uploader/vue-uploader
data:image/s3,"s3://crabby-images/850d3/850d387d2d15cc8cf41dbc7e0360b76db6f0cd31" alt=""
- Import the project into the development tool, then enter the App.vue file in the example folder and specify the back-end upload file interface (this interface path is specified by ourselves. If you don't understand it, it will be written directly as I do)
data:image/s3,"s3://crabby-images/bee05/bee057cbbe69494040e10dadd8eda95ed3bcebdc" alt=""
- Solve cross domain problems
data:image/s3,"s3://crabby-images/0c6b7/0c6b7c563c6f759635a2f5a16a0864c749c2b1b4" alt=""
'/ffmpeg-video':{
target:'http://localhost:3000',
changeOrigin:true,
pathRewrite:{
'^/ffmpeg-video':'/ffmpeg-video'
}
}
- npm install installs all dependencies
data:image/s3,"s3://crabby-images/a95b0/a95b05e70c3f3ff3bf49d88c76531468fb6e2ce8" alt=""
- npm run dev startup project
data:image/s3,"s3://crabby-images/5d404/5d4046b8b0acc7a23737229dc5dcf201438fb3c5" alt=""
data:image/s3,"s3://crabby-images/dd5c1/dd5c17c121a0ccc395062d613868ab95518cc357" alt=""
2. Back end environment
- Create a basic spring boot project with spring boot starter web dependency
- Configuration file, configuration upload path and port number (the port number needs to be consistent with the one specified by your front end)
data:image/s3,"s3://crabby-images/af726/af72652136bb2e82a0e48ddc7dc5011146785d74" alt=""
- Entity class
data:image/s3,"s3://crabby-images/3508d/3508de15bc496d4cad3b9d2ad61e27299f9c50fa" alt=""
import lombok.Data;
import org.springframework.web.multipart.MultipartFile;
@Data
public class MultipartFileParams {
//Slice number
private int chunkNumber;
//Slice size
private long chunkSize;
//Current slice size
private long currentChunkSize;
//Total file size
private long totalSize;
//Slice id
private String identifier;
//file name
private String filename;
//Relative path
private String relativePath;
//Total number of slices
private int totalChunks;
//spring receives the file object transmitted from the front end
private MultipartFile file;
}
- service layer
data:image/s3,"s3://crabby-images/30058/300583b53dfdfa6730e92769b5dd7eff87756270" alt=""
data:image/s3,"s3://crabby-images/a8d4c/a8d4c31de5342ca3cf0562dc7b8ac05c7ab938c0" alt=""
import com.yzpnb.entity.MultipartFileParams;
import com.yzpnb.service.FileUploadService;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;
import java.io.File;
import java.io.IOException;
@Service
public class FileUploadServiceImpl implements FileUploadService {
@Value("${upload.file.path}")
private String uploadFilePath;
@Override
public ResponseEntity<String> upload(MultipartFileParams fileParams) {
String filePath = uploadFilePath + fileParams.getFilename();
File fileTemp = new File(filePath);
File parentFile = fileTemp.getParentFile();
if(!parentFile.exists()){
parentFile.mkdirs();
}
try {
MultipartFile file = fileParams.getFile();
//Use the transferTo(dest) method to write the uploaded file to the specified file on the server;
//It can only be used once because the file stream can only be received and read once. When the transmission is completed, the stream will be closed;
file.transferTo(fileTemp);
} catch (IOException e) {
e.printStackTrace();
}
return ResponseEntity.ok("SUCCESS");
}
}
- controller
data:image/s3,"s3://crabby-images/53c88/53c88f5fbfda015835fd2a09704ff16884e6ecdf" alt=""
import com.yzpnb.entity.MultipartFileParams;
import com.yzpnb.service.FileUploadService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping(value = "/ffmpeg-video",produces = "application/json; charset=UTF-8")
@Slf4j
public class UploadController {
@Autowired
private FileUploadService fileUploadService;
@PostMapping("/upload")
public ResponseEntity<String> upload(MultipartFileParams file){
log.info("file: ",file);//Print log
return fileUploadService.upload(file);
}
}
3. Upload file test
- debug start backend
- Remove the built-in debugger and start the front end
data:image/s3,"s3://crabby-images/4da6a/4da6a952ec05e0a6d10c456817365b59ccdfa556" alt=""
- Upload file
data:image/s3,"s3://crabby-images/cb6b1/cb6b1cf183d2573339c38ecf8f4bb0586eb12686" alt=""
2, Source code analysis
Before looking at the source code, let's learn a few knowledge |
---|
- directive allows you to create an instruction and encapsulate the corresponding logic
- mixins allows you to mix the encapsulated data, mounted, etc. into what you need, which is equivalent to copying one copy. For example, several components have the same data and mounted, so we don't need to write each file once, just mix it directly
- The function of extends is not to copy, but to inherit and extend
- Provide can be pseudo responsive, and a wide range of data and menthod can be shared. When we expose some things with provide, we can inject them into other components with inject, but there must be blood relationship
data:image/s3,"s3://crabby-images/7cf50/7cf509422200e536656bdcd43c57e04d0a6f8bfc" alt=""
- Parent components can use props to transfer data to child components.
- Child components can use $emit to let parent components listen to custom events.
- VM. $emit (event name, passed parameter) / / trigger the event on the current instance
- VM. $on (event name, function function)// Run fn after listening for event events
- For example, if the sub component defines vm.$emit("show", data), the parent component can reference events through @ show = "" when using the sub component. Every time the sub component executes vm.$emit("show", data), the parent component triggers @ show once
- Because Vue simple upload is an encapsulated simple uploader.js, you need to know the common properties and events
- Common properties
data:image/s3,"s3://crabby-images/3f307/3f307836fa3d3bb466b046b72d0769b245bb91f0" alt=""
- Common events
>8. Simple-uploader.js for more events, please refer to the official document https://github.com/simple-uploader/Uploader/blob/develop/README_zh-CN.md
I didn't see the above things. It's hard to look at the source code, or I don't know when looking at the source code. You can go back and check it |
---|
1. mixins.js file
- First, the uploader.vue file provides an uploader
data:image/s3,"s3://crabby-images/f9695/f96959be6be5147e5fd6da74fb607ee2967cfb0c" alt=""
mixins.js this file is specially provided for component mixing through the minxins instruction |
---|
data:image/s3,"s3://crabby-images/f65be/f65be8368442aa671742f055703fb32456123b3d" alt=""
- This file exposes the support variable and injects the uploader through inject
2. uploader-btn
data:image/s3,"s3://crabby-images/c6e95/c6e9500c5ea06a42157bc3deaa9a9a86bba77d21" alt=""
data:image/s3,"s3://crabby-images/3e460/3e460951deba88235fe51768259361c1f21fb03d" alt=""
- The above assignBrowse method is simple-uploader.js. The specific reason is explained in uploader.vue
- This method not only transfers three props variables, but also transfers itself in, and obtains the dom node through $refs (this. $refs. Name)
- This method is mainly to click Select File to pop up the select File window,
3. uploader-unsupport
This file is used to handle the prompt that the user does not support HTML5 |
---|
When your browser does not support Uploader.js, you will be prompted that this library needs to support HTML5 File API and file slicing.
data:image/s3,"s3://crabby-images/e7a97/e7a97902b3cb3745a029737bf299d390c169916f" alt=""
4. uploader-drop
This file is to drag the file to the specified location, that is, to select a file. You can drag the file here, but there is no upload logic, just select the file |
---|
data:image/s3,"s3://crabby-images/3e162/3e162d99eb72692be837f2e24e3983606980a26e" alt=""
data:image/s3,"s3://crabby-images/a7d8c/a7d8cee9e535efb1270a56f2262f3855032d0ab6" alt=""
5. uploader-list
This file is mainly responsible for the list display after uploading files (an array composed of Uploader.File file and folder objects, where files and folders coexist) |
---|
data:image/s3,"s3://crabby-images/bc915/bc9155defc84cb72a47a683daac865f72ea1e594" alt=""
- In the figure above, we can see that the fileList variable is defined in the calculated field instead of the data field. Its purpose is to display the results immediately when the value in uploader.fileList changes. For example, when we upload a file, the fileList will add a file object, and then it will be rendered by two-way binding immediately
- Computed is used to monitor the variables defined by itself. The variables are not declared in data, but directly defined in computed. Then, two-way data binding can be carried out on the page to display the results or used for other processing
- computed is more suitable for returning a result value after processing multiple variables or objects, that is, if a value in several variables changes, the value we monitor will also change. For example, the relationship between the commodity list in the shopping cart and the total amount, as long as the quantity of commodities in the commodity list changes, Or reduce or increase or delete goods, the total amount should change. The total amount here is calculated using the calculated attribute, which is the best choice
6. uploader-files
It is basically the same as the uploader list above. The only difference is that the referenced objects are different (an array of file objects, a pure file list, and no folders) |
---|
data:image/s3,"s3://crabby-images/00034/00034330e21b8d75777a7deccb44ff545f860fec" alt=""
7. uploader-file
A single component of file and folder is a single file displayed in the list. This component has more code. I move the contents of the document directly and introduce them one by one. It's a waste of time |
---|
data:image/s3,"s3://crabby-images/bc383/bc383e394fef999cba2ea0ebd593f6a7322d539a" alt=""
data:image/s3,"s3://crabby-images/44c07/44c07ce05a17d65191e9af71ded7427225387c70" alt=""
data:image/s3,"s3://crabby-images/760fb/760fbad9ab1ad0371ee49ebf15b400222c4456ff" alt=""
data:image/s3,"s3://crabby-images/328a3/328a3ae58adacfd61e0979bd8522e85ca92fe8a2" alt=""
data:image/s3,"s3://crabby-images/9728e/9728ecbc04da11661feeb20c96685593ddb7a6f2" alt=""
data:image/s3,"s3://crabby-images/cbc17/cbc177ab167135b2855494c352ad6868fe5faa04" alt=""
data:image/s3,"s3://crabby-images/89914/899141feef0e06e267c7ab62ce6466011d2beeae" alt=""
data:image/s3,"s3://crabby-images/fe07a/fe07a847de3a38529e84ab799ab7325ad28ca188" alt=""
data:image/s3,"s3://crabby-images/3189d/3189da206ecdc6198f8f9d91c32af4eab6807a45" alt=""
data:image/s3,"s3://crabby-images/5c0ce/5c0ced5b3732a26f1c66cfb543808cc3da4e289e" alt=""
data:image/s3,"s3://crabby-images/19993/19993fcf5cd5c164fe42d5613e580f622d63a775" alt=""
data:image/s3,"s3://crabby-images/fbd68/fbd6833a9b180dae1141e36ed7f870378eb91121" alt=""
data:image/s3,"s3://crabby-images/749f0/749f0d28524707de13196aec57bdda41256649dd" alt=""
8. uploader
Source location: Vue upload master folder - > SRC folder - > components folder - > uploader.vue file |
---|
data:image/s3,"s3://crabby-images/6cc97/6cc97abd78f21ec891a6cac51918224063c36c25" alt=""
data:image/s3,"s3://crabby-images/36b4c/36b4c8c16d06674f0ac498d8ba865ca3a672c026" alt=""
data:image/s3,"s3://crabby-images/5e9e9/5e9e9d90f1edf29c4217769c72260c2cd08c3d04" alt=""
data:image/s3,"s3://crabby-images/073b1/073b1bfdcae990c150f9eb53b35b6d995aad3291" alt=""
data:image/s3,"s3://crabby-images/9b4f9/9b4f95ea03555075be2fffd845d37259b09e60b9" alt=""
3, Fragment upload
1. Back end
- Entity class
data:image/s3,"s3://crabby-images/76b8a/76b8a1737f6b6fb26c7b7bf9e9f5002057b49632" alt=""
import lombok.Data;
@Data
public class FileInfo {
private String uniqueIdentifier;//File unique id
private String name;//file name
}
- controller
data:image/s3,"s3://crabby-images/0f3a0/0f3a00c4eed575244c8025e11835931984d0e25b" alt=""
import com.yzpnb.entity.FileInfo;
import com.yzpnb.entity.MultipartFileParams;
import com.yzpnb.service.FileUploadService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;
import java.util.Map;
/**
* @author luoliang
* @date 2018/6/19
*/
@RestController
@RequestMapping(value = "/ffmpeg-video",produces = "application/json; charset=UTF-8")
@Slf4j
public class UploadController {
@Autowired
private FileUploadService fileUploadService;
/**
* Call before uploading (only one time) to determine whether the file has been uploaded, if so, skip.
* If not, judge whether half of the fragments are transmitted. If yes, return the missing fragment number and let the front end transmit the missing fragment
* @param file File parameters
* @return
*/
@GetMapping("/upload")
public ResponseEntity<Map<String,Object>> uploadCheck(MultipartFileParams file){
log.info("file: "+file);//Print log
return fileUploadService.uploadCheck(file);
}
/**
* Upload call
* @param file
* @return
*/
@PostMapping("/upload")
public ResponseEntity<String> upload(MultipartFileParams file){
log.info("file: "+file);//Print log
return fileUploadService.upload(file);
}
/**
* After the upload is completed, call to merge the fragment files
*/
@PostMapping("/upload-success")
public ResponseEntity<String> uploadSuccess(@RequestBody FileInfo file){
return fileUploadService.uploadSuccess(file);
}
}
- service
data:image/s3,"s3://crabby-images/5d031/5d0315b1c4dbb135b2ed964f05fd14d59059edc9" alt=""
data:image/s3,"s3://crabby-images/9dbea/9dbea149bbeb9e0d75daabf7fbea021415f07223" alt=""
import com.yzpnb.entity.FileInfo;
import com.yzpnb.entity.MultipartFileParams;
import com.yzpnb.service.FileUploadService;
import com.yzpnb.utils.MergeFileUtil;
import lombok.extern.log4j.Log4j2;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;
import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@Service
@Log4j2
public class FileUploadServiceImpl implements FileUploadService {
@Value("${upload.file.path}")
private String uploadFilePath;
/**
* Judge whether the file has been uploaded. If yes, skip,
* If not, judge whether half of the fragments are transmitted. If yes, return the missing fragment number and let the front end transmit the missing fragment
* @param fileParams
* @return
*/
@Override
public ResponseEntity<Map<String,Object>> uploadCheck(MultipartFileParams fileParams) {
//Get file unique id
String fileDir = fileParams.getIdentifier();
String filename = fileParams.getFilename();
//Partition directory
String chunkPath = uploadFilePath + fileDir+"/chunk/";
//Fragment directory object
File file = new File(chunkPath);
//Get fragment collection
List<File> chunkFileList = MergeFileUtil.chunkFileList(file);
//Merged file path
//Merge file path, D:/develop/video / file unique id/merge/filename
String filePath = uploadFilePath + fileDir+"/merge/"+filename;
File fileMergeExist = new File(filePath);
String [] temp;//Save list of existing files
boolean isExists = fileMergeExist.exists();//Are files that have been merged
if(chunkFileList == null){
temp= new String[0];
}else{
temp = new String[chunkFileList.size()];
//If there is no merged file, the upload is not completed
//If the upload is not completed, save the existing slice list if there are slices, otherwise do not save
if(!isExists && chunkFileList.size()>0){
for(int i = 0;i<chunkFileList.size();i++){
temp[i] = chunkFileList.get(i).getName();//Save list of existing files
}
}
}
//Return result set
HashMap<String, Object> hashMap = new HashMap<>();
hashMap.put("code",1);
hashMap.put("message","Success");
hashMap.put("needSkiped",isExists);
hashMap.put("uploadList",temp);
return ResponseEntity.ok(hashMap);
}
/**D:/develop/video/File unique id/chunk / fragment number
* Fragment upload file
* @param fileParams File parameters
* @return
*/
@Override
public ResponseEntity<String> upload(MultipartFileParams fileParams) {
//Get file unique id
String fileDir = fileParams.getIdentifier();
//Slice number
int chunkNumber = fileParams.getChunkNumber();
//File path, file specific path, D: / development / video / file unique id/chunk/1
String filePath = uploadFilePath + fileDir+"/chunk/"+chunkNumber;
File fileTemp = new File(filePath);
File parentFile = fileTemp.getParentFile();
if(!parentFile.exists()){
parentFile.mkdirs();
}
try {
MultipartFile file = fileParams.getFile();
//Use the file.transferTo(dest) method to write the uploaded file to the specified dest file on the server;
//It can only be used once because the file stream can only be received and read once. When the transmission is completed, the stream will be closed;
file.transferTo(fileTemp);
} catch (IOException e) {
e.printStackTrace();
}
return ResponseEntity.ok("SUCCESS");
}
@Override
public ResponseEntity<String> uploadSuccess(FileInfo fileInfo) {
log.info("filename: "+fileInfo.getName());
log.info("UniqueIdentifier: "+fileInfo.getUniqueIdentifier());
//Partition directory path
String chunkPath = uploadFilePath + fileInfo.getUniqueIdentifier()+"/chunk/";
//Merge directory path
String mergePath = uploadFilePath + fileInfo.getUniqueIdentifier()+"/merge/";
//Merge file, D:/develop/video / file unique id/merge/filename
File file = MergeFileUtil.mergeFile(uploadFilePath,chunkPath, mergePath,fileInfo.getName());
if(file == null){
return ResponseEntity.ok("ERROR:File merge failed");
}
return ResponseEntity.ok("SUCCESS");
}
}
- Tool class
data:image/s3,"s3://crabby-images/6cc28/6cc28bdc5b3e14497cfe178c71f61331ad399c8e" alt=""
package com.yzpnb.utils;
import java.io.File;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.util.*;
public class MergeFileUtil {
/**
* Judge whether the directory path to upload the partition exists, and create it if it does not exist,
* @param filePath Fragment path
* @return File object
*/
public static File isUploadChunkParentPath(String filePath){
File fileTemp = new File(filePath);
File parentFile = fileTemp.getParentFile();
if(!parentFile.exists()){
parentFile.mkdirs();
}
return fileTemp;
}
/**
* Merge file, D:/develop/video / file unique id/merge/filename
* @param uploadPath Upload path D:/develop/video/
* @param chunkPath Partition file directory path
* @param mergePath Merge file directory D:/develop/video / file unique id/merge/
* @param fileName file name
* @return
*/
public static File mergeFile(String uploadPath,String chunkPath,String mergePath,String fileName){
//Get the directory where the block file is located
File file = new File(chunkPath);
List<File> chunkFileList = chunkFileList(file);
//Before merging files, first judge whether there is a merged directory, which is not created
File fileTemp = new File(mergePath);
if(!fileTemp.exists()){
fileTemp.mkdirs();
}
//Merge file path
File mergeFile = new File(mergePath + fileName);
// The merged file exists. Delete it before creating it
if(mergeFile.exists()){
mergeFile.delete();
}
boolean newFile = false;
try {
newFile = mergeFile.createNewFile();//When a file is created, false is returned if it already exists. If there is no created file, no direct exception is thrown in the directory
} catch (IOException e) {
e.printStackTrace();
}
if(!newFile){//If newFile=false, the file exists
return null;
}
try {
//Create write file object
RandomAccessFile raf_write = new RandomAccessFile(mergeFile,"rw");
//Traverse the block file and start merging
// Read file buffer
byte[] b = new byte[1024];
for(File chunkFile:chunkFileList){
RandomAccessFile raf_read = new RandomAccessFile(chunkFile,"r");
int len =-1;
//Read block file
while((len = raf_read.read(b))!=-1){
//Write data to merge file
raf_write.write(b,0,len);
}
raf_read.close();
}
raf_write.close();
} catch (Exception e) {
e.printStackTrace();
return null;
}
return mergeFile;
}
/**
* Gets all files in the specified block file directory
* @param file Block directory
* @return File list
*/
public static List<File> chunkFileList(File file){
//Get all files in the directory
File[] files = file.listFiles();
if(files == null){
return null;
}
//Convert to list to facilitate sorting
List<File> chunkFileList = new ArrayList<>();
chunkFileList.addAll(Arrays.asList(files));
//sort
Collections.sort(chunkFileList, new Comparator<File>() {
@Override public int compare(File o1, File o2) {
if(Integer.parseInt(o1.getName())>Integer.parseInt(o2.getName())){
return 1;
}
return -1;
}
});
return chunkFileList;
}
}
2. Front end
- Introducing axios
data:image/s3,"s3://crabby-images/7e75a/7e75a3a35f560f1f51573c8513c559e6f65f2cab" alt=""
data:image/s3,"s3://crabby-images/f258f/f258f6db493bfaa76ab5c78e002ead8b1a9e203e" alt=""
- Judgment before uploading: before uploading a file, first judge whether a file already exists (the code is placed at the end)
data:image/s3,"s3://crabby-images/d0ea4/d0ea4526124b3bcb3a850cddbd745344cc3887ed" alt=""
- The callback after successful upload requests the backend to upload the successful interface, and then merge the fragments
data:image/s3,"s3://crabby-images/fffd0/fffd058a76422edfeadf8ffbb68b2663c9876070" alt=""
data:image/s3,"s3://crabby-images/3d0d2/3d0d24df2f2f8bc892b504ce9a58706a6518a795" alt=""
<template>
<uploader
:options="options"
:file-status-text="statusText"
class="uploader-example"
ref="uploader"
@file-complete="fileComplete"
@complete="complete"
@file-success="fileSuccess"
></uploader>
</template>
<script>
export default {
data () {
return {
options: {
target: '/ffmpeg-video/upload', // '//jsonplaceholder.typicode.com/posts/',
testChunks: true,//Enable slice test. If it exists, it will not be uploaded. If it does not exist, it will be uploaded
//Before the whole file upload starts, a get request with the same name as' / ffmpeg video / upload 'will be sent, and the response body is message, which will be passed into the following function
//The backend is only requested once, but the following function is called automatically every time the fragment is sent
checkChunkUploadedByResponse:function(chunk,message){//Call the function every time you upload a fragment
let messageObj = JSON.parse(message)
if(messageObj.needSkiped){
return true//If the upload is completed, skip directly
}else{//Otherwise, according to
return (messageObj.uploadList || []).indexOf(chunk.offset+1+"")>=0
}
return true
}
},
attrs: {
accept: 'image/*'
},
statusText: {
success: 'succeed',
error: 'Error ',
uploading: 'Uploading',
paused: 'Suspended',
waiting: 'Waiting'
}
}
},
methods: {
complete () {
// debugger
console.log('complete', arguments)
},
fileComplete () {
console.log('file complete', arguments)
},
fileSuccess(){
this.$axios({
method:'post',
url:'/ffmpeg-video/upload-success',
data: arguments[1]//This is the callback value of fileSuccess, which can be used directly
}).then(response =>{
console.log("fileSuccess")
},error =>{
})
}
},
mounted () {
console.log( localStorage.getItem("Access-Token"))
this.$nextTick(() => {
window.uploader = this.$refs.uploader.uploader
})
}
}
</script>
<style>
.uploader-example {
width: 880px;
padding: 15px;
margin: 40px auto 0;
font-size: 12px;
box-shadow: 0 0 10px rgba(0, 0, 0, .4);
}
.uploader-example .uploader-btn {
margin-right: 4px;
}
.uploader-example .uploader-list {
max-height: 440px;
overflow: auto;
overflow-x: hidden;
overflow-y: auto;
}
</style>
3. Operation results
data:image/s3,"s3://crabby-images/31902/319025d61b1e83920bada2ed499a7070927e9bbe" alt=""
data:image/s3,"s3://crabby-images/08427/084276dea26a3d10a028c5af8d598c011bbbcc92" alt=""
data:image/s3,"s3://crabby-images/b73ee/b73ee9e2506842d1d41a4c41214301f6b223cead" alt=""
data:image/s3,"s3://crabby-images/5b4b6/5b4b6f7bef22eff4eee163c6e1a135f1186689bb" alt=""
data:image/s3,"s3://crabby-images/961da/961da6fc4047df21e705290df585bfb125e34eac" alt=""
data:image/s3,"s3://crabby-images/fb267/fb267238a01637dab537d5dafac1c273f4115e9f" alt=""
data:image/s3,"s3://crabby-images/56718/567189aedd5f360f7993c273575f9791c5682fa4" alt=""
data:image/s3,"s3://crabby-images/19a9e/19a9e4b2a6c79dc6e02f48c17f17f81c33e42793" alt=""
4, Actual development