// ==UserScript==
// @name Stove Flake Automation Script
// @namespace https://github.com/TellurideX/Stove-Flake-Automation-Script-Tampermonkey
// @version 1.2.3
// @description 스토브 플레이크 샵 뽑기 자동화 스크립트
// @author TellurideX
// @match https://reward.onstove.com/ko/event*
// @icon 
// @homepageURL https://github.com/TellurideX/Stove-Flake-Automation-Script-Tampermonkey
// @updateURL https://raw.githubusercontent.com/TellurideX/Stove-Flake-Automation-Script-Tampermonkey/main/stove-flake-automation.user.js
// @downloadURL https://raw.githubusercontent.com/TellurideX/Stove-Flake-Automation-Script-Tampermonkey/main/stove-flake-automation.user.js
// @grant GM_setValue
// @grant GM_getValue
// ==/UserScript==
(function() {
'use strict';
// ================================
// 스크립트 버전 & 패치노트 관리
// ================================
// ⚠ @version 메타데이터와 반드시 동일하게 유지하세요.
const SCRIPT_VERSION = '1.2.3';
// 패치내역은 여기만 수정하면 됩니다.
// 새 버전 배포 시, 맨 위에 항목을 하나 더 추가하세요.
const PATCH_HISTORY = [
{
version: '1.2.3',
title: 'v1.2.3',
lines: [
'- 버그 픽스'
]
},
{
version: '1.2.2',
title: 'v1.2.2',
lines: [
'- 버그 픽스'
]
},
{
version: '1.2.1',
title: 'v1.2.1',
lines: [
'- 100 뽑기 / 1,000 뽑기 횟수 통계 표시 기능 추가',
'- 플레이크 순이익(실제 획득) 추가',
]
},
{
version: '1.0.0',
title: 'v1.0.0',
lines: [
'- Stove Flake Automation Script 최초 공개 버전'
]
}
];
const VERSION_STORAGE_KEY = 'stove_flake_last_version_seen';
function gmGet(key, defaultValue) {
try {
if (typeof GM_getValue === 'function') {
return GM_getValue(key, defaultValue);
}
} catch (e) {
console.log('[GM] GM_getValue 오류', e);
}
return defaultValue;
}
function gmSet(key, value) {
try {
if (typeof GM_setValue === 'function') {
GM_setValue(key, value);
}
} catch (e) {
console.log('[GM] GM_setValue 오류', e);
}
}
function getCurrentPatchEntry() {
for (let i = 0; i < PATCH_HISTORY.length; i += 1) {
if (PATCH_HISTORY[i].version === SCRIPT_VERSION) {
return PATCH_HISTORY[i];
}
}
return null;
}
// ================================
// 자동화 상태
// ================================
let isAuto100Running = false;
let isAuto1000Running = false;
let auto100Timer = null;
let auto1000Timer = null;
let remainingDraws = null; // 오늘 남은 뽑기 횟수 (자동 실행용)
// ================================
// 계정별 보상 기록 상태 (하루 단위)
// ================================
const BASE_STORAGE_KEY_REWARD = 'stove_flake_reward_v1';
let rewardState = null;
let currentAccountId = null;
function getRewardStorageKey() {
if (!currentAccountId) {
currentAccountId = 'default';
}
return BASE_STORAGE_KEY_REWARD + '::' + currentAccountId;
}
// ================================
// 계정 ID 해석
// ================================
function extractNicknameFromUserInfoLayer() {
const layer = document.getElementById('gnb-userinfo-layer');
if (!layer) return null;
const nickSpan = layer.querySelector(
'span.stds-text.text-2xl.leading-xl.font-bold'
);
if (!nickSpan) return null;
const nick = (nickSpan.textContent || '').trim();
return nick || null;
}
function closeUserInfoLayer(triggerBtn) {
const layer = document.getElementById('gnb-userinfo-layer');
if (!layer) {
try {
triggerBtn.click();
} catch (e) {
// ignore
}
return;
}
const closeBtn = layer.querySelector('button.stds-button-ghost');
if (closeBtn) {
try {
closeBtn.click();
console.log('[계정] 팝업 닫기 버튼 클릭');
return;
} catch (e) {
console.log('[계정] 팝업 닫기 버튼 클릭 실패', e);
}
}
try {
triggerBtn.click();
console.log('[계정] gnb-user-menu-button 재클릭으로 팝업 닫기 시도');
} catch (e) {
console.log('[계정] gnb-user-menu-button 재클릭 실패', e);
}
}
// 렌더링을 기다리면서 계정 ID를 결정
function resolveAccountId(callback) {
if (currentAccountId) {
if (callback) callback();
return;
}
let tries = 0;
const maxTries = 40; // 40 * 250ms ≒ 10초
const interval = 250;
let menuClicked = false;
const timer = window.setInterval(function() {
tries += 1;
if (currentAccountId) {
window.clearInterval(timer);
if (callback) callback();
return;
}
const nick = extractNicknameFromUserInfoLayer();
if (nick) {
currentAccountId = nick;
console.log('[계정] 메뉴에서 닉네임 읽음 →', currentAccountId);
const menuRootNow = document.getElementById('gnb-user-menu-button');
const triggerBtnNow = menuRootNow ? menuRootNow.querySelector('button') : null;
if (triggerBtnNow) {
closeUserInfoLayer(triggerBtnNow);
}
window.clearInterval(timer);
if (callback) callback();
return;
}
const menuRoot = document.getElementById('gnb-user-menu-button');
if (!menuRoot) {
if (tries >= maxTries) {
window.clearInterval(timer);
console.log('[계정] gnb-user-menu-button 미등장 → default 사용');
currentAccountId = 'default';
if (callback) callback();
}
return;
}
const triggerBtn = menuRoot.querySelector('button');
if (!triggerBtn) {
if (tries >= maxTries) {
window.clearInterval(timer);
console.log('[계정] gnb-user-menu-button 안에 button 없음 → default 사용');
currentAccountId = 'default';
if (callback) callback();
}
return;
}
if (!menuClicked) {
try {
triggerBtn.click();
menuClicked = true;
console.log('[계정] 계정 메뉴 자동 오픈 시도');
} catch (e) {
console.log('[계정] 계정 메뉴 자동 오픈 실패 → default 사용', e);
window.clearInterval(timer);
currentAccountId = 'default';
if (callback) callback();
}
return;
}
if (tries >= maxTries) {
window.clearInterval(timer);
console.log('[계정] 닉네임을 찾지 못함 → default 사용');
currentAccountId = 'default';
closeUserInfoLayer(triggerBtn);
if (callback) callback();
}
}, interval);
}
// ================================
// 날짜/보상 상태 관리
// ================================
function getTodayDateString() {
const d = new Date();
const y = d.getFullYear();
const m = String(d.getMonth() + 1).padStart(2, '0');
const day = String(d.getDate()).padStart(2, '0');
return y + '-' + m + '-' + day;
}
function initRewardState() {
const today = getTodayDateString();
const key = getRewardStorageKey();
const stored = gmGet(key, null);
if (stored && stored.date === today && Array.isArray(stored.logs)) {
rewardState = {
date: stored.date,
logs: stored.logs || [],
flakeTotal: stored.flakeTotal || 0,
otherCounts: stored.otherCounts || {},
lastIndex: stored.lastIndex || 0,
lastLabel: stored.lastLabel || null,
// 뽑기 종류별 횟수
drawCount100: stored.drawCount100 || 0,
drawCount1000: stored.drawCount1000 || 0
};
} else {
rewardState = {
date: today,
logs: [],
flakeTotal: 0,
otherCounts: {},
lastIndex: 0,
lastLabel: null,
drawCount100: 0,
drawCount1000: 0
};
gmSet(key, rewardState);
}
console.log('[보상 상태] init, account =', currentAccountId, ', logs =', rewardState.logs.length);
}
function ensureRewardStateForToday() {
const today = getTodayDateString();
const key = getRewardStorageKey();
if (!rewardState || rewardState.date !== today) {
rewardState = {
date: today,
logs: [],
flakeTotal: 0,
otherCounts: {},
lastIndex: 0,
lastLabel: null,
drawCount100: 0,
drawCount1000: 0
};
gmSet(key, rewardState);
}
}
function saveRewardState() {
if (!rewardState) return;
const key = getRewardStorageKey();
gmSet(key, rewardState);
}
// ================================
// UI: 보상 기록 패널
// ================================
function escapeHtml(str) {
if (!str) return '';
return str.replace(/[&<>"']/g, function(ch) {
switch (ch) {
case '&': return '&';
case '<': return '<';
case '>': return '>';
case '"': return '"';
case '\'': return ''';
default: return ch;
}
});
}
function renderRewardPanel() {
if (!rewardState) return;
let panel = document.getElementById('stove-reward-panel');
if (!panel) {
panel = document.createElement('div');
panel.id = 'stove-reward-panel';
panel.style.position = 'fixed';
panel.style.right = '20px';
panel.style.bottom = '180px';
panel.style.width = '260px';
panel.style.background = 'rgba(0, 0, 0, 0.7)';
panel.style.color = '#fff';
panel.style.borderRadius = '8px';
panel.style.padding = '8px 10px';
panel.style.fontSize = '12px';
panel.style.zIndex = '999999';
panel.style.boxShadow = '0 2px 8px rgba(0, 0, 0, 0.4)';
panel.style.backdropFilter = 'blur(4px)';
document.body.appendChild(panel);
}
const accountLabel = currentAccountId || 'default';
let html = '';
// 제목 + 날짜
html += '
보상 기록 (계정: ' +
escapeHtml(accountLabel) + ')
';
html += '날짜: ' +
escapeHtml(rewardState.date) + '
';
// 로그 영역 (스크롤 자동 하단 이동용 id)
html += '';
if (!rewardState.logs || rewardState.logs.length === 0) {
html += '
기록된 보상이 없습니다.
';
} else {
rewardState.logs.forEach(function(log) {
const idx = String(log.index).padStart(2, '0');
html += '
[' + idx + '] "' + escapeHtml(log.label) + '" 보상 획득
';
});
}
html += '
'; // 로그 박스 닫기
// 종합 보상 계산 (플레이크 순이익 포함)
const totalRewardFlake = rewardState.flakeTotal || 0;
const drawCount100 = rewardState.drawCount100 || 0;
const drawCount1000 = rewardState.drawCount1000 || 0;
const consumedFlake = drawCount100 * 100 + drawCount1000 * 1000;
const netFlake = totalRewardFlake - consumedFlake;
// 종합 보상 제목에 뽑기 횟수 표시
html += '' +
'종합 보상 (100 뽑기 : ' + drawCount100 + '회 // 1,000 뽑기 : ' + drawCount1000 + '회)' +
'
';
html += '';
let hasSummary = false;
// 플레이크 : 총 보상 + 실제 획득
if (totalRewardFlake > 0 || consumedFlake !== 0) {
const totalFormatted = totalRewardFlake.toLocaleString('ko-KR');
const netFormatted = netFlake.toLocaleString('ko-KR');
html += '
플레이크 : ' + totalFormatted + ' (실제 획득 : ' + netFormatted + ')
';
hasSummary = true;
}
// 기타 보상
if (rewardState.otherCounts) {
const labels = Object.keys(rewardState.otherCounts);
labels.forEach(function(label) {
const count = rewardState.otherCounts[label];
html += '
' + escapeHtml(label) + ' : ' + count + '
';
hasSummary = true;
});
}
if (!hasSummary) {
html += '
집계된 보상이 없습니다.
';
}
html += '
'; // 종합 보상 박스 닫기
// 실제 DOM에 반영
panel.innerHTML = html;
// 로그 스크롤을 항상 맨 아래로 이동
const logBox = document.getElementById('stove-reward-log');
if (logBox) {
logBox.scrollTop = logBox.scrollHeight;
}
}
// ================================
// 오늘 뽑기 횟수 파싱
// ================================
function getTodayDrawInfo() {
const boxes = document.querySelectorAll('.stds-box');
for (let i = 0; i < boxes.length; i += 1) {
const box = boxes[i];
if (!box) continue;
const raw = (box.textContent || '').replace(/\s+/g, '');
if (raw.indexOf('오늘뽑기') === -1) continue;
const match = raw.match(/(\d+)\s*\/\s*(\d+)회?/);
if (!match) continue;
const used = parseInt(match[1], 10);
const total = parseInt(match[2], 10);
if (isNaN(used) || isNaN(total)) continue;
return { used: used, total: total };
}
return null;
}
function getRemainingDrawsFromPage() {
const info = getTodayDrawInfo();
if (!info) return null;
let remaining = info.total - info.used;
if (remaining < 0) remaining = 0;
if (remaining > info.total) remaining = info.total;
return remaining;
}
// ================================
// 버튼 찾기 헬퍼
// ================================
function findButtonByText(keyword) {
const buttons = document.querySelectorAll('button');
for (let i = 0; i < buttons.length; i += 1) {
const btn = buttons[i];
if (!btn) continue;
const text = (btn.innerText || btn.textContent || '').trim();
if (text.indexOf(keyword) !== -1) {
return btn;
}
}
return null;
}
function findButtonByTextCommaInsensitive(keyword) {
const normalizedKeyword = keyword.replace(/,/g, '');
const buttons = document.querySelectorAll('button');
for (let i = 0; i < buttons.length; i += 1) {
const btn = buttons[i];
if (!btn) continue;
const rawText = (btn.innerText || btn.textContent || '').trim();
const normalizedText = rawText.replace(/,/g, '');
if (normalizedText.indexOf(normalizedKeyword) !== -1) {
return btn;
}
}
return null;
}
// ================================
// 메인/팝업 버튼 찾기
// ================================
function findMain100Button() {
const buttons = document.querySelectorAll('button');
for (let i = 0; i < buttons.length; i += 1) {
const btn = buttons[i];
if (!btn) continue;
const text = (btn.innerText || btn.textContent || '').trim();
if (text.indexOf('100 뽑기') !== -1 && text.indexOf('100 뽑기 한번 더!') === -1) {
return btn;
}
}
return null;
}
function findPopup100MoreButton() {
return findButtonByText('100 뽑기 한번 더!');
}
function findMain1000Button() {
return findButtonByTextCommaInsensitive('1000 뽑기');
}
function findPopup1000MoreButton() {
return findButtonByTextCommaInsensitive('1000 뽑기 한번 더!');
}
function findPopupCloseButton() {
return findButtonByText('닫기');
}
// ================================
// 남은 횟수 사용 / 체크
// ================================
function ensureRemainingDraws() {
if (remainingDraws === null) {
console.log('[공통] remainingDraws가 null 입니다.');
return false;
}
if (remainingDraws <= 0) {
console.log('[공통] 남은 뽑기 횟수가 0회입니다. 자동 종료.');
stopAutomation();
return false;
}
return true;
}
function consumeOneDraw(tag) {
if (remainingDraws === null) return;
remainingDraws -= 1;
console.log('[' + tag + '] 1회 사용 → 남은 횟수: ' + remainingDraws);
if (remainingDraws <= 0) {
console.log('[' + tag + '] 설정된 남은 횟수를 모두 사용했습니다. 자동 종료.');
stopAutomation();
}
}
// ================================
// 보상 기록 로직
// ================================
// 팝업에서 100/1000 뽑기 종류 추출
function detectDrawTypeFromPanel(panelEl) {
if (!panelEl) return null;
const buttons = panelEl.querySelectorAll('button');
for (let i = 0; i < buttons.length; i += 1) {
const btn = buttons[i];
const text = (btn.innerText || btn.textContent || '').replace(/\s+/g, '');
if (text.includes('100뽑기한번더!')) {
return '100';
}
if (text.includes('1000뽑기한번더!')) {
return '1000';
}
}
return null;
}
function recordReward(rewardLabel, drawType) {
// drawType: '100' | '1000' | null
ensureRewardStateForToday();
const info = getTodayDrawInfo();
let index;
if (info && typeof info.used === 'number') {
index = info.used;
} else if (rewardState.logs.length > 0) {
index = rewardState.logs[rewardState.logs.length - 1].index + 1;
} else {
index = 1;
}
if (rewardState.lastIndex === index && rewardState.lastLabel === rewardLabel) {
console.log('[보상 기록] 중복 감지 (index=' + index + ', label=' + rewardLabel + '), 기록 생략');
return;
}
rewardState.logs.push({
index: index,
label: rewardLabel
});
// 플레이크 보상 합계
if (rewardLabel.indexOf('플레이크') !== -1) {
const match = rewardLabel.replace(/,/g, '').match(/(\d+)/);
if (match) {
const amount = parseInt(match[1], 10);
if (!isNaN(amount)) {
rewardState.flakeTotal = (rewardState.flakeTotal || 0) + amount;
}
}
} else {
// 기타 보상 개수
if (!rewardState.otherCounts[rewardLabel]) {
rewardState.otherCounts[rewardLabel] = 0;
}
rewardState.otherCounts[rewardLabel] += 1;
}
// 뽑기 종류별 사용 횟수 기록
if (drawType === '100') {
rewardState.drawCount100 = (rewardState.drawCount100 || 0) + 1;
} else if (drawType === '1000') {
rewardState.drawCount1000 = (rewardState.drawCount1000 || 0) + 1;
}
rewardState.lastIndex = index;
rewardState.lastLabel = rewardLabel;
saveRewardState();
renderRewardPanel();
}
function handleRewardPanel(panelEl) {
if (!panelEl) return;
const rewardSpan = panelEl.querySelector('.l1l2-flakehub-popup-common-received_reward');
if (!rewardSpan) return;
const rewardText = (rewardSpan.textContent || '').trim();
if (!rewardText) return;
const drawType = detectDrawTypeFromPanel(panelEl); // '100' | '1000' | null
console.log('[보상 기록] 팝업 감지: ' + rewardText + ', drawType=' + drawType);
recordReward(rewardText, drawType);
}
function scanAllRewardPanels() {
const panels = document.querySelectorAll('.stds-dialog-panel');
panels.forEach(function(panel) {
handleRewardPanel(panel);
});
}
function setupPopupObserver() {
if (!window.MutationObserver) {
console.log('[보상 기록] MutationObserver 미지원');
return;
}
const observer = new MutationObserver(function() {
scanAllRewardPanels();
});
observer.observe(document.body, { childList: true, subtree: true });
}
// ================================
// 100 뽑기 자동 루프
// ================================
function loop100() {
if (!isAuto100Running) return;
if (!ensureRemainingDraws()) {
return;
}
const popupBtn = findPopup100MoreButton();
if (popupBtn) {
if (popupBtn.disabled) {
console.log('[100] 팝업 버튼 비활성화 → 0.9초 후 재시도');
auto100Timer = window.setTimeout(loop100, 900);
return;
}
try {
popupBtn.click();
console.log('[100] 팝업 "100 뽑기 한번 더!" 클릭');
consumeOneDraw('100');
if (!isAuto100Running) return;
} catch (e) {
console.log('[100] "100 뽑기 한번 더!" 클릭 중 오류, 자동 종료', e);
stopAutomation();
return;
}
auto100Timer = window.setTimeout(loop100, 900);
return;
}
const mainBtn = findMain100Button();
if (mainBtn) {
if (mainBtn.disabled) {
console.log('[100] 메인 "100 뽑기" 버튼 비활성화 → 자동 종료');
stopAutomation();
return;
}
try {
mainBtn.click();
console.log('[100] 메인 "100 뽑기" 클릭');
consumeOneDraw('100');
if (!isAuto100Running) return;
} catch (e) {
console.log('[100] 메인 "100 뽑기" 클릭 중 오류, 자동 종료', e);
stopAutomation();
return;
}
auto100Timer = window.setTimeout(loop100, 4000);
return;
}
console.log('[100] 관련 버튼을 찾지 못했습니다. 자동 종료');
stopAutomation();
}
// ================================
// 1,000 뽑기 자동 루프
// ================================
function loop1000() {
if (!isAuto1000Running) return;
if (!ensureRemainingDraws()) {
return;
}
const popupBtn = findPopup1000MoreButton();
if (popupBtn) {
if (popupBtn.disabled) {
console.log('[1000] 팝업 버튼 비활성화 → 0.9초 후 재시도');
auto1000Timer = window.setTimeout(loop1000, 900);
return;
}
try {
popupBtn.click();
console.log('[1000] 팝업 "1000 뽑기 한번 더!" 클릭');
consumeOneDraw('1000');
if (!isAuto1000Running) return;
} catch (e) {
console.log('[1000] "1000 뽑기 한번 더!" 클릭 중 오류, 자동 종료', e);
stopAutomation();
return;
}
auto1000Timer = window.setTimeout(loop1000, 900);
return;
}
const mainBtn = findMain1000Button();
if (mainBtn) {
if (mainBtn.disabled) {
console.log('[1000] 메인 "1,000 뽑기" 버튼 비활성화 → 자동 종료');
stopAutomation();
return;
}
try {
mainBtn.click();
console.log('[1000] 메인 "1,000 뽑기" 클릭');
consumeOneDraw('1000');
if (!isAuto1000Running) return;
} catch (e) {
console.log('[1000] 메인 "1,000 뽑기" 클릭 중 오류, 자동 종료', e);
stopAutomation();
return;
}
auto1000Timer = window.setTimeout(loop1000, 4000);
return;
}
console.log('[1000] 관련 버튼을 찾지 못했습니다. 자동 종료');
stopAutomation();
}
// ================================
// 패치노트 모달 UI
// ================================
function openPatchNotesModal() {
if (document.getElementById('stove-flake-patch-overlay')) {
return;
}
const overlay = document.createElement('div');
overlay.id = 'stove-flake-patch-overlay';
overlay.style.position = 'fixed';
overlay.style.left = '0';
overlay.style.top = '0';
overlay.style.right = '0';
overlay.style.bottom = '0';
overlay.style.background = 'rgba(0, 0, 0, 0.55)';
overlay.style.backdropFilter = 'blur(2px)';
overlay.style.zIndex = '1000000';
overlay.style.display = 'flex';
overlay.style.alignItems = 'center';
overlay.style.justifyContent = 'center';
const modal = document.createElement('div');
modal.style.background = 'rgba(15, 15, 15, 0.95)';
modal.style.color = '#ffffff';
modal.style.borderRadius = '10px';
modal.style.padding = '14px 16px 12px 16px';
modal.style.width = '480px';
modal.style.maxHeight = '70vh';
modal.style.boxShadow = '0 4px 16px rgba(0, 0, 0, 0.6)';
modal.style.display = 'flex';
modal.style.flexDirection = 'column';
modal.style.fontSize = '13px';
const header = document.createElement('div');
header.textContent = 'Stove Flake Automation 패치 노트 (v' + SCRIPT_VERSION + ')';
header.style.fontWeight = 'bold';
header.style.marginBottom = '8px';
header.style.fontSize = '14px';
const content = document.createElement('div');
content.style.flex = '1';
content.style.overflowY = 'auto';
content.style.paddingRight = '4px';
// 패치내역 전체 렌더링
PATCH_HISTORY.forEach(function(entry) {
const section = document.createElement('div');
section.style.marginBottom = '10px';
const title = document.createElement('div');
title.textContent = entry.title || ('v' + entry.version);
title.style.fontWeight = 'bold';
title.style.marginBottom = '4px';
title.style.fontSize = '13px';
section.appendChild(title);
if (entry.lines && entry.lines.length > 0) {
const ul = document.createElement('ul');
ul.style.paddingLeft = '18px';
ul.style.margin = '0';
entry.lines.forEach(function(line) {
const li = document.createElement('li');
li.textContent = line;
li.style.marginBottom = '2px';
ul.appendChild(li);
});
section.appendChild(ul);
}
content.appendChild(section);
});
const footer = document.createElement('div');
footer.style.display = 'flex';
footer.style.justifyContent = 'flex-end';
footer.style.marginTop = '8px';
const closeBtn = document.createElement('button');
closeBtn.textContent = '닫기';
closeBtn.style.padding = '6px 12px';
closeBtn.style.borderRadius = '6px';
closeBtn.style.border = 'none';
closeBtn.style.cursor = 'pointer';
closeBtn.style.fontSize = '12px';
closeBtn.style.background = '#555';
closeBtn.style.color = '#fff';
closeBtn.addEventListener('mouseenter', function() {
closeBtn.style.background = '#666';
});
closeBtn.addEventListener('mouseleave', function() {
closeBtn.style.background = '#555';
});
closeBtn.addEventListener('click', function(e) {
e.stopPropagation();
closePatchNotesModal();
});
footer.appendChild(closeBtn);
modal.appendChild(header);
modal.appendChild(content);
modal.appendChild(footer);
// 오버레이 클릭 시 모달 닫기 (모달 내부 클릭은 전파 차단)
modal.addEventListener('click', function(e) {
e.stopPropagation();
});
overlay.addEventListener('click', function() {
closePatchNotesModal();
});
overlay.appendChild(modal);
document.body.appendChild(overlay);
}
function closePatchNotesModal() {
const overlay = document.getElementById('stove-flake-patch-overlay');
if (overlay && overlay.parentNode) {
overlay.parentNode.removeChild(overlay);
}
}
// ================================
// 버전 최초 1회만 보여주는 작은 팝업
// ================================
function showVersionBannerIfNeeded() {
const lastSeen = gmGet(VERSION_STORAGE_KEY, null);
if (lastSeen === SCRIPT_VERSION) {
return;
}
// 현재 버전으로 갱신 (이후 재접속 시에는 다시 뜨지 않음)
gmSet(VERSION_STORAGE_KEY, SCRIPT_VERSION);
const entry = getCurrentPatchEntry();
const banner = document.createElement('div');
banner.id = 'stove-flake-version-banner';
banner.style.position = 'fixed';
banner.style.right = '20px';
banner.style.bottom = '420px';
banner.style.width = '260px';
banner.style.background = 'rgba(0, 0, 0, 0.82)';
banner.style.color = '#fff';
banner.style.borderRadius = '8px';
banner.style.padding = '8px 10px';
banner.style.fontSize = '12px';
banner.style.zIndex = '999999';
banner.style.boxShadow = '0 2px 8px rgba(0, 0, 0, 0.5)';
banner.style.cursor = 'pointer';
banner.style.backdropFilter = 'blur(4px)';
let html = '';
html += 'Stove Flake 패치 안내 (v' +
SCRIPT_VERSION + ')
';
if (entry && entry.lines && entry.lines.length > 0) {
html += '';
entry.lines.forEach(function(line) {
html += escapeHtml(line) + '
';
});
html += '
';
} else {
html += '' +
'새 버전이 적용되었습니다.' +
'
';
}
html += '(창을 클릭하면 닫힙니다.)
';
banner.innerHTML = html;
// 아무 곳이나 클릭하면 종료
banner.addEventListener('click', function(e) {
e.stopPropagation();
const el = document.getElementById('stove-flake-version-banner');
if (el && el.parentNode) {
el.parentNode.removeChild(el);
}
});
document.body.appendChild(banner);
}
// ================================
// 자동화 시작 / 중단
// ================================
function startAuto100() {
if (isAuto100Running) {
alert('이미 100 뽑기 자동화가 진행중입니다.');
return;
}
if (isAuto1000Running) {
alert('이미 1,000 뽑기 자동화가 진행중입니다. 100 뽑기 자동화를 실행하려면 먼저 중단하기 버튼을 눌러주세요.');
return;
}
const remain = getRemainingDrawsFromPage();
if (remain === null) {
alert('오늘 뽑기 횟수를 확인하지 못했습니다. 페이지 구조가 변경되었을 수 있습니다.');
return;
}
if (remain <= 0) {
alert('오늘은 이미 모두 참여했습니다.\n00시 이후에 다시 시도해주시길 바랍니다.');
return;
}
remainingDraws = remain;
console.log('[100] 자동화 시작, 남은 횟수: ' + remainingDraws);
isAuto100Running = true;
loop100();
}
function startAuto1000() {
if (isAuto1000Running) {
alert('이미 1,000 뽑기 자동화가 진행중입니다.');
return;
}
if (isAuto100Running) {
alert('이미 100 뽑기 자동화가 진행중입니다. 1,000 뽑기로 변경하려면 중단하기 버튼을 누르고 다시 시도해주세요.');
return;
}
const remain = getRemainingDrawsFromPage();
if (remain === null) {
alert('오늘 뽑기 횟수를 확인하지 못했습니다. 페이지 구조가 변경되었을 수 있습니다.');
return;
}
if (remain <= 0) {
alert('오늘은 이미 모두 참여했습니다.\n00시 이후에 다시 시도해주시길 바랍니다.');
return;
}
remainingDraws = remain;
console.log('[1000] 자동화 시작, 남은 횟수: ' + remainingDraws);
isAuto1000Running = true;
loop1000();
}
function stopAutomation() {
if (!isAuto100Running && !isAuto1000Running) {
alert('현재 스크립트가 작동중이지 않습니다.');
}
console.log('[중단] 자동화 종료 요청');
isAuto100Running = false;
isAuto1000Running = false;
if (auto100Timer !== null) {
window.clearTimeout(auto100Timer);
auto100Timer = null;
}
if (auto1000Timer !== null) {
window.clearTimeout(auto1000Timer);
auto1000Timer = null;
}
remainingDraws = null;
// 팝업 닫기 딜레이 1초
window.setTimeout(function() {
const closeBtn = findPopupCloseButton();
if (!closeBtn) {
console.log('[중단] 팝업 닫기 버튼을 찾지 못했습니다.');
return;
}
try {
closeBtn.click();
console.log('[중단] 팝업 닫기 버튼 클릭');
} catch (e) {
console.log('[중단] 팝업 닫기 버튼 클릭 중 오류', e);
}
}, 1000);
}
// ================================
// UI: 자동화 버튼 + 툴팁
// ================================
function createButtonWithTooltip(options) {
const id = options.id;
const label = options.label;
const bottom = options.bottom;
const onClick = options.onClick;
const background = options.background;
const tooltip = options.tooltip;
if (document.getElementById(id)) {
return;
}
const container = document.createElement('div');
container.style.position = 'fixed';
container.style.right = '20px';
container.style.bottom = bottom + 'px';
container.style.zIndex = '999999';
container.style.display = 'flex';
container.style.alignItems = 'center';
container.style.gap = '6px';
const icon = document.createElement('div');
icon.textContent = '?';
icon.style.width = '18px';
icon.style.height = '18px';
icon.style.borderRadius = '50%';
icon.style.background = '#555';
icon.style.color = '#fff';
icon.style.display = 'flex';
icon.style.alignItems = 'center';
icon.style.justifyContent = 'center';
icon.style.fontSize = '12px';
icon.style.cursor = 'default';
icon.style.opacity = '0.85';
const tooltipBox = document.createElement('div');
tooltipBox.textContent = tooltip;
tooltipBox.style.position = 'absolute';
tooltipBox.style.right = '110%';
tooltipBox.style.top = '50%';
tooltipBox.style.transform = 'translateY(-50%)';
tooltipBox.style.background = '#333';
tooltipBox.style.color = '#fff';
tooltipBox.style.padding = '6px 10px';
tooltipBox.style.borderRadius = '6px';
tooltipBox.style.whiteSpace = 'nowrap';
tooltipBox.style.fontSize = '12px';
tooltipBox.style.opacity = '0';
tooltipBox.style.pointerEvents = 'none';
tooltipBox.style.transition = 'opacity 0.1s ease';
icon.addEventListener('mouseenter', function() {
tooltipBox.style.opacity = '1';
});
icon.addEventListener('mouseleave', function() {
tooltipBox.style.opacity = '0';
});
const btn = document.createElement('button');
btn.id = id;
btn.textContent = label;
btn.style.padding = '10px 12px';
btn.style.borderRadius = '6px';
btn.style.border = 'none';
btn.style.cursor = 'pointer';
btn.style.fontSize = '12px';
btn.style.fontFamily = 'inherit';
btn.style.color = '#ffffff';
btn.style.background = background || '#ff6b00';
btn.style.boxShadow = '0 2px 6px rgba(0, 0, 0, 0.3)';
btn.style.opacity = '0.9';
btn.style.width = '190px';
btn.style.textAlign = 'center';
btn.addEventListener('mouseenter', function() {
btn.style.opacity = '1';
});
btn.addEventListener('mouseleave', function() {
btn.style.opacity = '0.9';
});
btn.addEventListener('click', function(e) {
e.preventDefault();
e.stopPropagation();
onClick();
});
container.appendChild(icon);
container.appendChild(btn);
container.appendChild(tooltipBox);
document.body.appendChild(container);
}
function initUIButtons() {
// 버튼 위치를 조금 위로 올려서 패치노트 버튼을 맨 아래에 배치
createButtonWithTooltip({
id: 'stove-auto-100',
label: '100 뽑기 자동화 시작하기',
bottom: 140,
background: '#ff6b00',
onClick: startAuto100,
tooltip: '오늘 뽑기 남은 횟수만큼 100 뽑기를 자동으로 진행합니다.'
});
createButtonWithTooltip({
id: 'stove-auto-1000',
label: '1,000 뽑기 자동화 시작하기',
bottom: 100,
background: '#ff6b00',
onClick: startAuto1000,
tooltip: '오늘 뽑기 남은 횟수만큼 1,000 뽑기를 자동으로 진행합니다.'
});
createButtonWithTooltip({
id: 'stove-auto-stop',
label: '중단하기',
bottom: 60,
background: '#d62828',
onClick: stopAutomation,
tooltip: '진행 중인 자동화를 중단하고, 팝업이 있다면 닫기 버튼을 누릅니다.'
});
// ✅ 패치노트 버튼 (초록색)
createButtonWithTooltip({
id: 'stove-patch-notes',
label: '패치노트',
bottom: 20,
background: '#2f9e44',
onClick: openPatchNotesModal,
tooltip: '버전별 패치 내역을 확인합니다.'
});
}
// ================================
// 초기화
// ================================
function onReady() {
resolveAccountId(function() {
initRewardState();
initUIButtons();
renderRewardPanel();
setupPopupObserver();
scanAllRewardPanels();
// 버전 최초 1회 팝업
showVersionBannerIfNeeded();
});
}
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', onReady);
} else {
onReady();
}
})();