- PackDefinition: replace mods[]/resourcepacks[] with modsFolder (string) + resourcepackPath (string); drop PackAsset - Editor: replace dynamic add/remove lists with two single inputs; remove the now-dead JS for adding/removing rows - Server: expose GET /file/mods/<folder>/index.json that returns the list of .jar names; folder name restricted to [a-zA-Z0-9_-]+ - Installer: fetch the listing JSON and download each jar from /file/mods/<folder>/<file>.jar; download the single resourcepack from /file/resourcepacks/<file>.zip directly into resourcepacks/ Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
130 lines
5.3 KiB
Plaintext
130 lines
5.3 KiB
Plaintext
<!doctype html>
|
|
<html lang="ko">
|
|
<head>
|
|
<meta charset="utf-8" />
|
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
<title><%= pack.name %> 편집</title>
|
|
<link rel="stylesheet" href="/static/styles.css" />
|
|
</head>
|
|
<body class="siteBody">
|
|
<%- include('../partials/navbar', { userId }) %>
|
|
|
|
<main class="pageWrap">
|
|
<section class="editorHeader">
|
|
<div>
|
|
<p class="eyebrow">PACK EDITOR</p>
|
|
<h1><%= pack.name %></h1>
|
|
</div>
|
|
<a class="ghostLink" href="/op/dashboard">목록으로</a>
|
|
</section>
|
|
|
|
<form method="post" class="editorForm" id="editorForm">
|
|
<div class="gridTwo">
|
|
<label>
|
|
<span>음악퀴즈 이름</span>
|
|
<input name="displayName" value="<%= pack.name %>" required />
|
|
</label>
|
|
<label>
|
|
<span>JSON 파일 이름 (확장자 제외)</span>
|
|
<input name="fileName" value="<%= packKey %>" required pattern="[a-zA-Z0-9_\-]+" />
|
|
</label>
|
|
</div>
|
|
|
|
<div class="gridTwo">
|
|
<label>
|
|
<span>마인크래프트 버전</span>
|
|
<select name="mcVersion" required>
|
|
<% releases.forEach(function (release) { %>
|
|
<option value="<%= release %>" <%= release === pack.mcVersion ? 'selected' : '' %>><%= release %></option>
|
|
<% }) %>
|
|
</select>
|
|
</label>
|
|
<label>
|
|
<span>모드 플랫폼</span>
|
|
<select name="platformType" id="platformType">
|
|
<% ['vanilla','forge','fabric','neoforge'].forEach(function (loader) { %>
|
|
<option value="<%= loader %>" <%= pack.platform.type === loader ? 'selected' : '' %>><%= loader %></option>
|
|
<% }) %>
|
|
</select>
|
|
</label>
|
|
<label class="fullSpan" id="platformDownloadField">
|
|
<span>플랫폼 설치파일 URL</span>
|
|
<input name="platformDownloadUrl" value="<%= pack.platform.downloadUrl || '' %>" placeholder="/forge-installer.jar 또는 https://example.com/forge-installer.jar" />
|
|
<small class="muted">도메인 없이 입력하면 manifest.json 도메인의 <code>/file/platforms/<파일명></code>으로 해석됩니다.</small>
|
|
</label>
|
|
<label>
|
|
<span>서버 최소 램 (MB)</span>
|
|
<input type="number" name="serverMinRam" value="<%= pack.serverMinRam %>" min="512" required />
|
|
</label>
|
|
<label>
|
|
<span>서버 최대 램 (MB)</span>
|
|
<input type="number" name="serverMaxRam" value="<%= pack.serverMaxRam %>" min="512" required />
|
|
</label>
|
|
<label>
|
|
<span>클라이언트 최소 램 (MB)</span>
|
|
<input type="number" name="clientMinRam" value="<%= pack.clientMinRam %>" min="512" required />
|
|
</label>
|
|
<label>
|
|
<span>클라이언트 권장 램 (MB)</span>
|
|
<input type="number" name="clientRecommendedRam" value="<%= pack.clientRecommendedRam %>" min="512" required />
|
|
</label>
|
|
<label>
|
|
<span>맵 파일 (.zip)</span>
|
|
<input name="mapPath" value="<%= pack.mapPath %>" placeholder="my-map.zip" pattern=".+\.zip" />
|
|
<small class="muted">/file/maps/ 아래 zip 파일 이름.</small>
|
|
</label>
|
|
<label>
|
|
<span>서버 파일 (.zip)</span>
|
|
<input name="serverPath" value="<%= pack.serverPath %>" placeholder="my-server.zip" pattern=".+\.zip" />
|
|
<small class="muted">/file/servers/ 아래 zip 파일 이름. 멀티 모드 전용.</small>
|
|
</label>
|
|
</div>
|
|
|
|
<div class="gridTwo">
|
|
<label>
|
|
<span>모드 폴더 이름</span>
|
|
<input name="modsFolder" value="<%= pack.modsFolder %>" placeholder="my-pack" pattern="[a-zA-Z0-9_\-]*" />
|
|
<small class="muted">/file/mods/<폴더이름>/ 안의 모든 .jar을 자동으로 받습니다. 비워두면 모드를 받지 않습니다.</small>
|
|
</label>
|
|
<label>
|
|
<span>리소스팩 (.zip)</span>
|
|
<input name="resourcepackPath" value="<%= pack.resourcepackPath %>" placeholder="my-pack.zip" pattern=".*\.zip|" />
|
|
<small class="muted">/file/resourcepacks/ 아래 .zip 파일 이름. 비워두면 리소스팩을 받지 않습니다.</small>
|
|
</label>
|
|
</div>
|
|
|
|
<button class="primaryButton" type="submit">저장</button>
|
|
</form>
|
|
</main>
|
|
|
|
<script>
|
|
(function () {
|
|
var platformSelect = document.getElementById('platformType')
|
|
var downloadField = document.getElementById('platformDownloadField')
|
|
|
|
function syncPlatformVisibility() {
|
|
if (platformSelect.value === 'vanilla') {
|
|
downloadField.setAttribute('hidden', '')
|
|
downloadField.querySelector('input').value = ''
|
|
} else {
|
|
downloadField.removeAttribute('hidden')
|
|
}
|
|
}
|
|
|
|
platformSelect.addEventListener('change', syncPlatformVisibility)
|
|
syncPlatformVisibility()
|
|
|
|
var form = document.getElementById('editorForm')
|
|
form.addEventListener('submit', function (event) {
|
|
var clientMin = Number(form.clientMinRam.value)
|
|
var clientReco = Number(form.clientRecommendedRam.value)
|
|
if (clientMin > clientReco) {
|
|
event.preventDefault()
|
|
alert('클라이언트 최소 램은 권장 램보다 클 수 없습니다.')
|
|
}
|
|
})
|
|
})()
|
|
</script>
|
|
</body>
|
|
</html>
|