// ==UserScript==
// @name Minefun FPS & CPS Counter, Key Display,CPS Graph,Reload & FPS Alert & Crosshair
// @namespace https://minefun.io/
// @license MIT
// @icon 
// @version 1.9.9.1
// @description minefun.ioでFPSとCPS、過去5秒のCPS線グラフ、押されているキー表示、リロード、FPS低下アラート、照準を追加
// @match *://minefun.io/*
// @grant none
// ==/UserScript==
(function() {
'use strict';
// スタイル定義
const styles = `
#mfCounterContainer {
position: fixed;
top: 10px;
left: 10px;
background-color: rgba(30, 30, 30, 0.8);
color: #eee;
padding: 10px;
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
font-size: 14px;
z-index: 1000;
cursor: grab;
border-radius: 5px;
box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.3);
display: flex;
flex-direction: column;
align-items: flex-start;
gap: 8px;
}
#mfCounterText {
white-space: pre-line;
}
#mfKeyDisplay {
display: flex;
gap: 5px;
margin-top: 5px;
}
.key-icon {
background-color: #444;
color: #eee;
padding: 5px;
border-radius: 3px;
font-size: 12px;
min-width: 20px;
text-align: center;
}
#mfImageContainer {
position: relative;
width: 40px;
height: 40px;
border-radius: 50%;
overflow: hidden;
border: 1px solid #555;
display: flex;
justify-content: center;
align-items: center;
}
#mfImage {
width: 100%;
height: 100%;
object-fit: cover;
}
#mfReloadButton {
background-color: #555;
color: #eee;
border: none;
padding: 6px 10px;
border-radius: 5px;
font-size: 12px;
cursor: pointer;
transition: background-color 0.3s ease;
margin-top: 5px;
}
#mfReloadButton:hover {
background-color: #777;
}
#mfFpsAlert {
background-color: rgba(255, 0, 0, 0.7);
color: white;
padding: 5px;
border-radius: 5px;
font-size: 12px;
margin-top: 5px;
}
.switch-container {
display: flex;
align-items: center;
gap: 8px;
margin-top: 5px;
}
.switch {
position: relative;
display: inline-block;
width: 40px;
height: 20px;
}
.switch input {
opacity: 0;
width: 0;
height: 0;
}
.slider {
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: #ccc;
transition: .4s;
border-radius: 20px;
}
.slider:before {
position: absolute;
content: "";
height: 16px;
width: 16px;
left: 2px;
bottom: 2px;
background-color: white;
transition: .4s;
border-radius: 50%;
}
input:checked + .slider {
background-color: #2196F3;
}
input:focus + .slider {
box-shadow: 0 0 1px #2196F3;
}
input:checked + .slider:before {
transform: translateX(20px);
}
.switch-label {
font-size: 12px;
color: #ccc;
}
#mfCrosshair {
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
width: 20px;
height: 20px;
border: 1px solid rgba(255, 255, 255, 0.5);
border-radius: 50%;
z-index: 9999;
display: none; /* 初期状態は非表示 */
pointer-events: none; /* クリックを透過 */
}
#mfCrosshair:before, #mfCrosshair:after {
content: '';
position: absolute;
background-color: rgba(255, 255, 255, 0.5);
}
#mfCrosshair:before {
left: 50%;
top: 2px;
bottom: 2px;
width: 1px;
transform: translateX(-50%);
}
#mfCrosshair:after {
top: 50%;
left: 2px;
right: 2px;
height: 1px;
transform: translateY(-50%);
}
#mfCpsGraph {
width: 100px;
height: 30px;
margin-top: 5px;
border: 1px solid #555;
border-radius: 3px;
background-color: #333;
}
`;
const styleSheet = document.createElement("style");
styleSheet.type = "text/css";
styleSheet.innerText = styles;
document.head.appendChild(styleSheet);
// コンテナ要素の作成
const counterContainer = document.createElement('div');
counterContainer.id = 'mfCounterContainer';
document.body.appendChild(counterContainer);
// テキスト要素の作成
const textElement = document.createElement('span');
textElement.id = 'mfCounterText';
counterContainer.appendChild(textElement);
// キー表示コンテナの作成
const keyDisplayContainer = document.createElement('div');
keyDisplayContainer.id = 'mfKeyDisplay';
counterContainer.appendChild(keyDisplayContainer);
// CPSグラフ表示用のキャンバス要素を作成
const cpsGraphCanvas = document.createElement('canvas');
cpsGraphCanvas.id = 'mfCpsGraph';
counterContainer.appendChild(cpsGraphCanvas);
const cpsGraphCtx = cpsGraphCanvas.getContext('2d');
const cpsHistory = [];
const maxCpsHistoryLength = 5; // 過去5秒
// 画像要素の作成
const imageContainer = document.createElement('div');
imageContainer.id = 'mfImageContainer';
counterContainer.appendChild(imageContainer);
const imageElement = document.createElement('img');
imageElement.id = 'mfImage';
imageElement.src = 'https://cdn.minefun.io/users/67629a12aa9a815752742647/27EY08XPCCCXH.webp';
imageContainer.appendChild(imageElement);
// スーパーリロードボタンの作成
const reloadButton = document.createElement('button');
reloadButton.id = 'mfReloadButton';
reloadButton.textContent = 'Reload';
reloadButton.addEventListener('click', () => {
window.location.reload(true); // 強制リロード
});
counterContainer.appendChild(reloadButton);
// FPS低下アラート表示要素
const fpsAlertElement = document.createElement('div');
fpsAlertElement.id = 'mfFpsAlert';
fpsAlertElement.style.display = 'none'; // 初期状態は非表示
counterContainer.appendChild(fpsAlertElement);
// 照準表示のコンテナ
const crosshairSwitchContainer = document.createElement('div');
crosshairSwitchContainer.classList.add('switch-container');
counterContainer.appendChild(crosshairSwitchContainer);
// 照準表示のトグルスイッチ
const crosshairLabel = document.createElement('label');
crosshairLabel.classList.add('switch');
crosshairSwitchContainer.appendChild(crosshairLabel);
const crosshairToggle = document.createElement('input');
crosshairToggle.type = 'checkbox';
crosshairLabel.appendChild(crosshairToggle);
const crosshairSlider = document.createElement('span');
crosshairSlider.classList.add('slider');
crosshairLabel.appendChild(crosshairSlider);
const crosshairSwitchLabel = document.createElement('span');
crosshairSwitchLabel.classList.add('switch-label');
crosshairSwitchLabel.textContent = 'Crosshair';
crosshairSwitchContainer.appendChild(crosshairSwitchLabel);
// 照準要素の作成
const crosshairElement = document.createElement('div');
crosshairElement.id = 'mfCrosshair';
document.body.appendChild(crosshairElement);
// FPS計測
let fps = 0;
let lastFPSTime = performance.now();
let frameCount = 0;
function updateFPS(currentTime) {
frameCount++;
const deltaTime = currentTime - lastFPSTime;
if (deltaTime >= 100) {
fps = Math.round((frameCount * 1000) / deltaTime * 10) / 10;
frameCount = 0;
lastFPSTime = currentTime;
// FPS低下アラートの表示/非表示
if (fps < 30) {
fpsAlertElement.textContent = `FPSが${fps}です`;
fpsAlertElement.style.display = 'block';
} else {
fpsAlertElement.style.display = 'none';
}
}
requestAnimationFrame(updateFPS);
}
requestAnimationFrame(updateFPS);
// CPS計測と履歴の更新
let cps = 0;
let clickCount = 0;
let lastCPSTime = performance.now();
document.addEventListener('mousedown', () => {
clickCount++;
});
function updateCPS() {
const currentTime = performance.now();
const deltaTime = currentTime - lastCPSTime;
if (deltaTime >= 1000) {
cps = clickCount;
clickCount = 0;
lastCPSTime = currentTime;
cpsHistory.push(cps);
if (cpsHistory.length > maxCpsHistoryLength) {
cpsHistory.shift(); // 古いデータを削除
}
}
requestAnimationFrame(updateCPS);
}
requestAnimationFrame(updateCPS);
// CPSグラフの描画 (線グラフ)
function drawCpsGraph() {
cpsGraphCtx.clearRect(0, 0, cpsGraphCanvas.width, cpsGraphCanvas.height);
if (cpsHistory.length > 1) {
cpsGraphCtx.strokeStyle = '#00ff00'; // 線の色
cpsGraphCtx.lineWidth = 2;
cpsGraphCtx.beginPath();
const maxCps = Math.max(...cpsHistory, 1); // 最小値を1として0除算を防ぐ
const xIncrement = cpsGraphCanvas.width / (cpsHistory.length - 1);
cpsHistory.forEach((cpsValue, index) => {
const x = index * xIncrement;
const y = cpsGraphCanvas.height - (cpsValue / maxCps) * cpsGraphCanvas.height;
if (index === 0) {
cpsGraphCtx.moveTo(x, y);
} else {
cpsGraphCtx.lineTo(x, y);
}
});
cpsGraphCtx.stroke();
}
requestAnimationFrame(drawCpsGraph);
}
requestAnimationFrame(drawCpsGraph);
// 押されているキーの表示 (アイコン)
const pressedKeys = new Set();
const keyIconMap = {
'w': 'W', 'a': 'A', 's': 'S', 'd': 'D',
' ': 'Space', 'shift': 'Shift', 'ctrl': 'Ctrl', 'alt': 'Alt',
'tab': 'Tab', 'capslock': 'Caps', 'enter': 'Enter',
'arrowup': '↑', 'arrowdown': '↓', 'arrowleft': '←', 'arrowright': '→',
'escape': 'Esc',
'1': '1', '2': '2', '3': '3', '4': '4', '5': '5', '6': '6', '7': '7', '8': '8', '9': '9', '0': '0',
'`': '`', '-': '-', '=': '=', '[': '[', ']': ']', '\\': '\\',
';': ';', '\'': '\'', ',': ',', '.': '.', '/': '/',
'f1': 'F1', 'f2': 'F2', 'f3': 'F3', 'f4': 'F4', 'f5': 'F5', 'f6': 'F6',
'f7': 'F7', 'f8': 'F8', 'f9': 'F9', 'f10': 'F10', 'f11': 'F11', 'f12': 'F12'
// 必要に応じて他のキーも追加
};
document.addEventListener('keydown', (event) => {
pressedKeys.add(event.key.toLowerCase());
updateKeyDisplay();
});
document.addEventListener('keyup', (event) => {
pressedKeys.delete(event.key.toLowerCase());
updateKeyDisplay();
});
function updateKeyDisplay() {
keyDisplayContainer.innerHTML = ''; // Clear previous icons
pressedKeys.forEach(key => {
const iconText = keyIconMap[key] || key.toUpperCase(); // マップにない場合は大文字で表示
const keyIcon = document.createElement('div');
keyIcon.classList.add('key-icon');
keyIcon.textContent = iconText;
keyDisplayContainer.appendChild(keyIcon);
});
}
// 表示の更新
function updateDisplay() {
textElement.textContent = `FPS: ${fps}\nCPS: ${cps}\nBy kutu524_`;
requestAnimationFrame(updateDisplay);
}
requestAnimationFrame(updateDisplay);
// ドラッグ機能
let isDragging = false;
let offsetX, offsetY;
counterContainer.addEventListener('mousedown', (e) => {
isDragging = true;
counterContainer.style.cursor = 'grabbing';
offsetX = e.clientX - counterContainer.getBoundingClientRect().left;
offsetY = e.clientY - counterContainer.getBoundingClientRect().top;
});
document.addEventListener('mouseup', () => {
isDragging = false;
counterContainer.style.cursor = 'grab';
});
document.addEventListener('mousemove', (e) => {
if (!isDragging) return;
counterContainer.style.left = e.clientX - offsetX + 'px';
counterContainer.style.top = e.clientY - offsetY + 'px';
});
// 照準の表示/非表示を切り替え
crosshairToggle.addEventListener('change', function() {
crosshairElement.style.display = this.checked ? 'block' : 'none';
});
})();