├── public └── slemons.png ├── src ├── assets │ ├── bg.png │ ├── ing.png │ ├── steve.png │ ├── arrow-del.png │ ├── arrow-left.png │ ├── arrow-right.png │ ├── model-slim.png │ ├── model-default.png │ └── vue.svg ├── main.js ├── components │ ├── HelloWorld.vue │ ├── 3d.vue │ ├── up.vue │ └── skin.vue ├── App.vue └── style.css ├── .vscode └── extensions.json ├── vite.config.js ├── .gitignore ├── index.html ├── README.md ├── package.json ├── .github └── workflows │ └── npm-publish-github-packages.yml └── app.js /public/slemons.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/public/slemons.png -------------------------------------------------------------------------------- /src/assets/bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/src/assets/bg.png -------------------------------------------------------------------------------- /src/assets/ing.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/src/assets/ing.png -------------------------------------------------------------------------------- /src/assets/steve.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/src/assets/steve.png -------------------------------------------------------------------------------- /src/assets/arrow-del.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/src/assets/arrow-del.png -------------------------------------------------------------------------------- /src/assets/arrow-left.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/src/assets/arrow-left.png -------------------------------------------------------------------------------- /src/assets/arrow-right.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/src/assets/arrow-right.png -------------------------------------------------------------------------------- /src/assets/model-slim.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/src/assets/model-slim.png -------------------------------------------------------------------------------- /.vscode/extensions.json: -------------------------------------------------------------------------------- 1 | { 2 | "recommendations": ["Vue.volar", "Vue.vscode-typescript-vue-plugin"] 3 | } 4 | -------------------------------------------------------------------------------- /src/assets/model-default.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SOURsLEMONS/skinpack/HEAD/src/assets/model-default.png -------------------------------------------------------------------------------- /src/main.js: -------------------------------------------------------------------------------- 1 | import { createApp } from 'vue' 2 | import './style.css' 3 | import App from './App.vue' 4 | 5 | createApp(App).mount('#app') 6 | -------------------------------------------------------------------------------- /vite.config.js: -------------------------------------------------------------------------------- 1 | import { defineConfig } from 'vite' 2 | import vue from '@vitejs/plugin-vue' 3 | 4 | // https://vitejs.dev/config/ 5 | export default defineConfig({ 6 | plugins: [vue()], 7 | }) 8 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | pnpm-debug.log* 8 | lerna-debug.log* 9 | 10 | node_modules 11 | dist 12 | dist-ssr 13 | *.local 14 | 15 | # Editor directories and files 16 | .vscode/* 17 | !.vscode/extensions.json 18 | .idea 19 | .DS_Store 20 | *.suo 21 | *.ntvs* 22 | *.njsproj 23 | *.sln 24 | *.sw? 25 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 皮肤生成器 9 | 10 | 11 | 12 | 13 |
14 | 15 | 16 | 17 | 18 | 21 | 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # 基于 Vue 3 的纯前端我的世界基岩版皮肤包生成器 2 | 3 | 已部署在 [https://skin.suanlemon.cc](https://skin.suanlemon.cc) 4 | 5 | ## 介绍 6 | 7 | 这是一个基于 Vue 3 的我的世界基岩版皮肤包生成器,可以生成一个符合我的世界基岩版皮肤包格式的 zip 文件,可以直接在我的世界基岩版中使用。 8 | 9 | ## 使用 10 | 11 | 1. 在 [https://skin.suanlemon.cc](https://skin.suanlemon.cc) 中输入上传你的皮肤图片 12 | 2. 设置你的皮肤名称(你也可以不设置,不设置的话会自动生成名字) 13 | 3. 设置你的皮肤皮肤包的名字(你也可以不设置,不设置的话会自动生成名字) 14 | 4. 点击生成按钮,好了大功告成 15 | 16 | ## 嘀咕 17 | 18 | 这个项目是我在学习 Vue 3 的时候做的,所以代码写的很烂,有很多地方写的很不好,如果你有什么好的建议,欢迎提出来,我会很感谢的。 19 | -------------------------------------------------------------------------------- /src/assets/vue.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "name", 3 | "private": true, 4 | "version": "0.0.0", 5 | "type": "module", 6 | "scripts": { 7 | "dev": "vite", 8 | "build": "vite build", 9 | "preview": "vite preview" 10 | }, 11 | "dependencies": { 12 | "file-saver": "^2.0.5", 13 | "jszip": "^3.10.1", 14 | "skinview3d": "^3.0.0-alpha.1", 15 | "vue": "^3.2.47" 16 | }, 17 | "devDependencies": { 18 | "@vitejs/plugin-vue": "^4.1.0", 19 | "three": "^0.151.3", 20 | "vite": "^4.2.0" 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /src/components/HelloWorld.vue: -------------------------------------------------------------------------------- 1 | 6 | 7 | 21 | 22 | 27 | -------------------------------------------------------------------------------- /src/components/3d.vue: -------------------------------------------------------------------------------- 1 | 6 | 7 | 33 | -------------------------------------------------------------------------------- /.github/workflows/npm-publish-github-packages.yml: -------------------------------------------------------------------------------- 1 | # This workflow will run tests using node and then publish a package to GitHub Packages when a release is created 2 | # For more information see: https://docs.github.com/en/actions/publishing-packages/publishing-nodejs-packages 3 | 4 | name: Node.js Package 5 | 6 | on: 7 | release: 8 | types: [created] 9 | 10 | jobs: 11 | build: 12 | runs-on: ubuntu-latest 13 | steps: 14 | - uses: actions/checkout@v4 15 | - uses: actions/setup-node@v4 16 | with: 17 | node-version: 20 18 | - run: npm ci 19 | - run: npm test 20 | 21 | publish-gpr: 22 | needs: build 23 | runs-on: ubuntu-latest 24 | permissions: 25 | contents: read 26 | packages: write 27 | steps: 28 | - uses: actions/checkout@v4 29 | - uses: actions/setup-node@v4 30 | with: 31 | node-version: 20 32 | registry-url: https://npm.pkg.github.com/ 33 | - run: npm ci 34 | - run: npm publish 35 | env: 36 | NODE_AUTH_TOKEN: ${{secrets.GITHUB_TOKEN}} 37 | -------------------------------------------------------------------------------- /src/App.vue: -------------------------------------------------------------------------------- 1 | 5 | 6 | 16 | 17 | 18 | 62 | -------------------------------------------------------------------------------- /src/style.css: -------------------------------------------------------------------------------- 1 | :root { 2 | font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif; 3 | line-height: 1.5; 4 | font-weight: 400; 5 | 6 | color-scheme: light dark; 7 | color: rgba(255, 255, 255, 0.87); 8 | background-color: #242424; 9 | 10 | font-synthesis: none; 11 | text-rendering: optimizeLegibility; 12 | -webkit-font-smoothing: antialiased; 13 | -moz-osx-font-smoothing: grayscale; 14 | -webkit-text-size-adjust: 100%; 15 | } 16 | 17 | a { 18 | font-weight: 500; 19 | color: #646cff; 20 | text-decoration: inherit; 21 | } 22 | a:hover { 23 | color: #535bf2; 24 | } 25 | 26 | a { 27 | font-weight: 500; 28 | color: #646cff; 29 | text-decoration: inherit; 30 | } 31 | a:hover { 32 | color: #535bf2; 33 | } 34 | 35 | body { 36 | margin: 0; 37 | display: flex; 38 | place-items: center; 39 | min-width: 320px; 40 | min-height: 100vh; 41 | } 42 | 43 | h1 { 44 | font-size: 3.2em; 45 | line-height: 1.1; 46 | } 47 | 48 | button { 49 | border-radius: 8px; 50 | border: 1px solid transparent; 51 | padding: 0.6em 1.2em; 52 | font-size: 1em; 53 | font-weight: 500; 54 | font-family: inherit; 55 | background-color: #1a1a1a; 56 | cursor: pointer; 57 | transition: border-color 0.25s; 58 | } 59 | button:hover { 60 | border-color: #646cff; 61 | } 62 | button:focus, 63 | button:focus-visible { 64 | outline: 4px auto -webkit-focus-ring-color; 65 | } 66 | 67 | .card { 68 | padding: 2em; 69 | } 70 | 71 | #app { 72 | max-width: 1280px; 73 | margin: 0 auto; 74 | padding: 2rem; 75 | text-align: center; 76 | } 77 | 78 | @media (prefers-color-scheme: light) { 79 | :root { 80 | color: #213547; 81 | background-color: #ffffff; 82 | } 83 | a:hover { 84 | color: #747bff; 85 | } 86 | button { 87 | background-color: #f9f9f9; 88 | } 89 | } 90 | -------------------------------------------------------------------------------- /src/components/up.vue: -------------------------------------------------------------------------------- 1 | 106 | 107 | 108 | 109 | -------------------------------------------------------------------------------- /app.js: -------------------------------------------------------------------------------- 1 | !function(t){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=t() 2 | else if("function"==typeof define&&define.amd)define([],t) 3 | else{var e 4 | e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this,e.McHead=t()}}(function(){return function t(e,i,s){function r(n,a){if(!i[n]){if(!e[n]){var h="function"==typeof require&&require 5 | if(!a&&h)return h(n,!0) 6 | if(o)return o(n,!0) 7 | var u=new Error("Cannot find module '"+n+"'") 8 | throw u.code="MODULE_NOT_FOUND",u}var l=i[n]={exports:{}} 9 | e[n][0].call(l.exports,function(t){var i=e[n][1][t] 10 | return r(i?i:t)},l,l.exports,t,e,i,s)}return i[n].exports}for(var o="function"==typeof require&&require,n=0;n 2 | import { VueElement, ref } from "vue"; 3 | import { SkinViewer } from "skinview3d"; 4 | import JSZip from "jszip"; 5 | import { saveAs } from "file-saver"; 6 | 7 | var loaders = true 8 | 9 | var steve = 10 | ""; 11 | var slemons = 12 | ""; 13 | var skinlist = [ 14 | { 15 | name: "皮肤1", 16 | src: steve, 17 | slim: false, 18 | }, 19 | // { 20 | // name: "皮肤2", 21 | // src: slemons, 22 | // slim: false, 23 | // }, 24 | ]; 25 | 26 | //生成uuid 27 | function uuid() { 28 | var s = []; 29 | var hexDigits = "0123456789abcdef"; 30 | for (var i = 0; i < 36; i++) { 31 | s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1); 32 | } 33 | s[14] = "4"; 34 | s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1); 35 | s[8] = s[13] = s[18] = s[23] = "-"; 36 | 37 | var uuid = s.join(""); 38 | return uuid; 39 | } 40 | 41 | 42 | 43 | 44 | var skins = { 45 | serialize_name: "生成的皮肤包",//包名 46 | localization_name: "SLSkinPack",//标识符 47 | skins: [ 48 | ], 49 | } 50 | 51 | 52 | var zh_CN = { 53 | skinpack: "SL皮肤包", 54 | skinlist: [ 55 | "皮肤1", 56 | "皮肤2", 57 | ], 58 | } 59 | 60 | function lang(lang) { 61 | var str = "skinpack."; 62 | str = str + skins.localization_name + "=" + lang.skinpack + "\n"; 63 | for (var i = 0; i < skins.skins.length; i++) { 64 | str += 'skin.' + skins.localization_name + "." + skins.skins[i].localization_name + "=" + lang.skinlist[i].name + "\n"; 65 | // //console.log(lang.skinlist[i]); 66 | } 67 | return str; 68 | } 69 | 70 | 71 | // var skinname = skinlist[skinlistindex].name; 72 | 73 | 74 | // var skinmodel = false; 75 | 76 | var skinViewer; 77 | 78 | 79 | export default { 80 | name: "sk", 81 | data() { 82 | return { 83 | skinpackname: "SL皮肤包", 84 | ulr: steve, 85 | skinlist: skinlist, 86 | skinlistindex: 0, 87 | skinname: '', 88 | skinmodel: false, 89 | loaders: true 90 | }; 91 | }, 92 | methods: { 93 | //这是上传图片的方法 94 | upload: function (e) { 95 | var that = this; 96 | //console.log(e.target.files[0]); 97 | //获取图片大小 98 | let file = e.target.files[0]; 99 | let reader = new FileReader(); 100 | reader.readAsDataURL(file); 101 | let img = new Image(); 102 | reader.onload = function (e) { 103 | //console.log(e.target.result); 104 | img.src = e.target.result; 105 | img.onload = function () { 106 | //console.log(img.width); 107 | //console.log(img.height); 108 | if ( 109 | img.width != img.height && (img.width != 128 ||img.width != 64 || img.width != 32) 110 | ) { 111 | alert("图片尺寸不正确,请上传128 64 32的图片"); 112 | return; 113 | } 114 | //这里ulr 115 | this.ulr = e.target.result; 116 | //console.log(this.ulr); 117 | 118 | skinViewer.loadSkin(this.ulr); 119 | //console.log(skinViewer.loadSkin(this.ulr)); 120 | skinViewer.autoRotate = true; 121 | setTimeout(() => { 122 | skinViewer.autoRotate = false; 123 | }, 6280); 124 | 125 | // //console.log(skinViewer.playerObject.skin.modelType); 126 | 127 | that.skinlistindex = that.skinlist.push({ 128 | name: "皮肤" + (skinlist.length + 1), 129 | src: this.ulr, 130 | slim: skinViewer.playerObject.skin.modelType, 131 | }) - 1; 132 | 133 | //调用skinmodelslim()方法 134 | that.skinmodelslim(this.ulr); 135 | }; 136 | }; 137 | }, 138 | //上一张 下一张 139 | skinlistbrowse: function (n) { 140 | if (this.skinlistindex + n >= 0 && this.skinlistindex + n < skinlist.length) { 141 | this.skinlist[this.skinlistindex].name = this.skinname; 142 | this.skinlistindex += n; 143 | this.skinname = this.skinlist[this.skinlistindex].name; 144 | //console.log(this.skinname); 145 | } 146 | 147 | 148 | if (skinlist[this.skinlistindex].slim) { 149 | //console.log(skinlist[this.skinlistindex].slim); 150 | skinViewer.loadSkin(skinlist[this.skinlistindex].src, { model: "slim" }); 151 | this.skinmodel = true; 152 | } else { 153 | skinViewer.loadSkin(skinlist[this.skinlistindex].src, { model: "default" }); 154 | this.skinmodel = false; 155 | //console.log(skinlist[this.skinlistindex].slim); 156 | } 157 | //console.log(skinlist[this.skinlistindex].slim); 158 | //console.log(this.skinlistindex); 159 | //console.log(this.skinmodel); 160 | 161 | //model 162 | }, 163 | //切换模型 皮肤 164 | skinmodelslim(url = this.skinlist[this.skinlistindex].src) { 165 | if (this.skinmodel) { 166 | skinViewer.loadSkin(url, { model: "default" }); 167 | this.skinlist[this.skinlistindex].slim = false; 168 | this.skinmodel = false; 169 | } else { 170 | skinViewer.loadSkin(url, { model: "slim" }); 171 | //console.log(this.skinlistindex); 172 | //console.log(this.skinlist[this.skinlistindex]); 173 | this.skinlist[this.skinlistindex].slim = true; 174 | this.skinmodel = true; 175 | } 176 | }, 177 | //删除皮肤 178 | skinlistdelete: function () { 179 | if (this.skinlist.length > 1) { 180 | this.skinlist.splice(this.skinlistindex, 1); 181 | this.skinlistindex = 0; 182 | this.skinname = this.skinlist[this.skinlistindex].name; 183 | if (this.skinlist[this.skinlistindex].slim) { 184 | skinViewer.loadSkin(this.skinlist[this.skinlistindex].src, { 185 | model: "slim", 186 | }); 187 | this.skinmodel = true; 188 | } else { 189 | skinViewer.loadSkin(this.skinlist[this.skinlistindex].src, { 190 | model: "default", 191 | }); 192 | this.skinmodel = false; 193 | } 194 | } else { 195 | alert("至少保留一张皮肤"); 196 | } 197 | }, 198 | //生成皮肤包 199 | spawnpack: function () { 200 | this.loaders = true; 201 | var that = this; 202 | this.skinlist[this.skinlistindex].name = this.skinname; 203 | // //console.log(this.skinlist); 204 | 205 | var manifest = { 206 | header: { 207 | name: "包名", 208 | version: [1, 0, 0], 209 | uuid: uuid(), 210 | }, 211 | modules: [ 212 | { 213 | version: [1, 0, 0], 214 | type: "skin_pack", 215 | uuid: uuid(), 216 | }, 217 | ], 218 | format_version: 1, 219 | } 220 | 221 | 222 | var time = new Date(); 223 | time = time.getTime().toString() 224 | time = time.substring(time.length, time.length - 6); 225 | 226 | skins.serialize_name = this.skinpackname; 227 | skins.localization_name = "SLSkinPack" + time; 228 | skins.skins = function () { 229 | var arr = []; 230 | for (var i = 0; i < that.skinlist.length; i++) { 231 | arr.push({ 232 | localization_name: 'Skin' + i, 233 | geometry: skinlist[i].slim ? "geometry.humanoid.customSlim" : "geometry.humanoid.custom", 234 | texture: 'skin' + i + '.png', 235 | type: 'free', 236 | }); 237 | } 238 | return arr; 239 | }(); 240 | //console.log(JSON.stringify(skins, null, 4)); 241 | 242 | manifest.header.name = this.skinpackname; 243 | //console.log(JSON.stringify(manifest, null, 4)); 244 | 245 | zh_CN.skinpack = this.skinpackname; 246 | zh_CN.skinlist = function () { 247 | var arr = []; 248 | for (var i = 0; i < that.skinlist.length; i++) { 249 | arr.push({ 250 | name: that.skinlist[i].name, 251 | }); 252 | } 253 | return arr; 254 | }(); 255 | 256 | //console.log(lang(zh_CN)); 257 | 258 | 259 | var zip = new JSZip(); 260 | 261 | zip.file("manifest.json", JSON.stringify(manifest, null, 4)); 262 | zip.file("skins.json", JSON.stringify(skins, null, 4)); 263 | var texts = zip.folder("texts"); 264 | texts.file("zh_CN.lang", lang(zh_CN)); 265 | for (var i = 0; i < this.skinlist.length; i++) { 266 | zip.file("skin" + i + ".png", this.skinlist[i].src.split(",")[1], { 267 | base64: true, 268 | }); 269 | } 270 | zip.generateAsync({ type: "blob" }) 271 | .then(function (content) { 272 | saveAs(content, "SLSkinPack.zip.mcpack"); 273 | that.loaders = false; 274 | }); 275 | }, 276 | }, 277 | mounted() { 278 | skinViewer = new SkinViewer({ 279 | canvas: document.getElementById("skin_container"), 280 | width: 150, 281 | height: 200, 282 | skin: steve, 283 | }); 284 | this.loaders = false; 285 | // //console.log(this); 286 | // this.skinmodel = false; 287 | }, 288 | }; 289 | // //console.log(Vue); 290 | 291 | 346 | 347 | 679 | --------------------------------------------------------------------------------