Minefun FPS & CPS Counter, Key Display,CPS Graph,Reload & FPS Alert & Crosshair

minefun.ioでFPSとCPS、過去5秒のCPS線グラフ、押されているキー表示、リロード、FPS低下アラート、照準を追加

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==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';

    });



})();