sark 1 hónapja
szülő
commit
aa6cd94d2d

+ 147 - 0
resources/view/admin/slots/form.php

@@ -0,0 +1,147 @@
+<?php
+ob_start();
+?>
+
+<div class="min-h-screen bg-gray-50 py-6">
+    <div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8">
+        <!-- Header -->
+        <div class="mb-8">
+            <div class="md:flex md:items-center md:justify-between">
+                <div class="flex-1 min-w-0">
+                    <h2 class="text-2xl font-bold leading-7 text-gray-900 sm:text-3xl sm:truncate">
+                        <?= $action === 'create' ? 'Add New Slot' : 'Edit Slot' ?>
+                    </h2>
+                    <p class="mt-1 text-sm text-gray-500">
+                        <?= $action === 'create' ? 'Create a new casino slot' : 'Update slot information' ?>
+                    </p>
+                </div>
+            </div>
+        </div>
+
+        <!-- Success/Error Messages -->
+        <?php if (isset($success)): ?>
+            <div class="mb-6 bg-green-50 border border-green-200 rounded-md p-4">
+                <div class="flex">
+                    <div class="flex-shrink-0">
+                        <svg class="h-5 w-5 text-green-400" viewBox="0 0 20 20" fill="currentColor">
+                            <path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clip-rule="evenodd"/>
+                        </svg>
+                    </div>
+                    <div class="ml-3">
+                        <p class="text-sm font-medium text-green-800"><?= htmlspecialchars($success) ?></p>
+                    </div>
+                </div>
+            </div>
+        <?php endif; ?>
+
+        <?php if (isset($error)): ?>
+            <div class="mb-6 bg-red-50 border border-red-200 rounded-md p-4">
+                <div class="flex">
+                    <div class="flex-shrink-0">
+                        <svg class="h-5 w-5 text-red-400" viewBox="0 0 20 20" fill="currentColor">
+                            <path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z" clip-rule="evenodd"/>
+                        </svg>
+                    </div>
+                    <div class="ml-3">
+                        <p class="text-sm font-medium text-red-800"><?= htmlspecialchars($error) ?></p>
+                    </div>
+                </div>
+            </div>
+        <?php endif; ?>
+
+        <!-- Form -->
+        <form method="POST" enctype="multipart/form-data" class="space-y-6">
+            <input type="hidden" name="action" value="<?= $action ?>">
+            <?php if ($action === 'edit'): ?>
+                <input type="hidden" name="id" value="<?= $slot['id'] ?>">
+            <?php endif; ?>
+
+            <div class="bg-white shadow rounded-lg">
+                <div class="px-6 py-4 border-b border-gray-200">
+                    <h3 class="text-lg font-medium text-gray-900">Slot Information</h3>
+                    <p class="mt-1 text-sm text-gray-500">Basic information about the slot game.</p>
+                </div>
+
+                <div class="px-6 py-4 space-y-6">
+                    <!-- Slot Name -->
+                    <div>
+                        <label for="name" class="block text-sm font-medium text-gray-700 mb-2">Slot Name *</label>
+                        <input type="text" name="name" id="name" required
+                               value="<?= htmlspecialchars($slot['name'] ?? '') ?>" 
+                               class="w-full px-3 py-2 border border-gray-300 rounded-md shadow-sm focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
+                               placeholder="Enter slot name">
+                    </div>
+
+                    <!-- Provider -->
+                    <div>
+                        <label for="provider" class="block text-sm font-medium text-gray-700 mb-2">Provider</label>
+                        <input type="text" name="provider" id="provider"
+                               value="<?= htmlspecialchars($slot['provider'] ?? '') ?>" 
+                               class="w-full px-3 py-2 border border-gray-300 rounded-md shadow-sm focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
+                               placeholder="e.g. NetEnt, Microgaming, Pragmatic Play">
+                    </div>
+
+                    <!-- RTP -->
+                    <div>
+                        <label for="rtp" class="block text-sm font-medium text-gray-700 mb-2">RTP (%)</label>
+                        <input type="number" name="rtp" id="rtp" step="0.01" min="0" max="100"
+                               value="<?= htmlspecialchars($slot['rtp'] ?? '') ?>" 
+                               class="w-full px-3 py-2 border border-gray-300 rounded-md shadow-sm focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
+                               placeholder="e.g. 96.50">
+                        <p class="mt-1 text-xs text-gray-500">Return to Player percentage (e.g. 96.50 for 96.50%)</p>
+                    </div>
+
+                    <!-- Image Upload -->
+                    <div>
+                        <label for="image" class="block text-sm font-medium text-gray-700 mb-2">Slot Image</label>
+                        <div class="flex items-center space-x-4">
+                            <?php if (!empty($slot['image'])): ?>
+                                <div class="flex-shrink-0">
+                                    <img src="/<?= htmlspecialchars($slot['image']) ?>" alt="Current image" class="h-20 w-20 rounded-lg object-cover border border-gray-300">
+                                </div>
+                            <?php endif; ?>
+                            <div class="flex-1">
+                                <input type="file" name="image_upload" id="image_upload" accept="image/*"
+                                       class="block w-full text-sm text-gray-500 file:mr-4 file:py-2 file:px-4 file:rounded-md file:border-0 file:text-sm file:font-medium file:bg-blue-50 file:text-blue-700 hover:file:bg-blue-100">
+                                <input type="hidden" name="current_image" value="<?= htmlspecialchars($slot['image'] ?? '') ?>">
+                                <p class="mt-1 text-xs text-gray-500">Upload an image file for the slot</p>
+                            </div>
+                        </div>
+                    </div>
+
+                    <!-- Demo URL -->
+                    <div>
+                        <label for="demo_url" class="block text-sm font-medium text-gray-700 mb-2">Demo URL</label>
+                        <input type="url" name="demo_url" id="demo_url"
+                               value="<?= htmlspecialchars($slot['demo_url'] ?? '') ?>" 
+                               class="w-full px-3 py-2 border border-gray-300 rounded-md shadow-sm focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-blue-500"
+                               placeholder="https://example.com/slot-demo">
+                        <p class="mt-1 text-xs text-gray-500">URL to play the demo version of the slot</p>
+                    </div>
+                </div>
+            </div>
+
+            <!-- Form Actions -->
+            <div class="flex justify-between pt-6 border-t border-gray-200">
+                <a href="/admin/slots/" class="inline-flex items-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 transition-colors">
+                    <svg class="w-5 h-5 mr-2" fill="currentColor" viewBox="0 0 20 20">
+                        <path fill-rule="evenodd" d="M7.707 14.707a1 1 0 01-1.414 0L2.586 11H9a1 1 0 010 2H2.586l3.707 3.707a1 1 0 01-1.414 1.414l-5.414-5.414a1 1 0 010-1.414L4.879 6.879a1 1 0 011.414 1.414L2.586 11H9a1 1 0 010 2H2.586l3.707 3.707z" clip-rule="evenodd"/>
+                    </svg>
+                    Back to List
+                </a>
+
+                <button type="submit" class="inline-flex items-center px-6 py-2 border border-transparent text-sm font-medium rounded-md text-white bg-blue-600 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 transition-colors">
+                    <svg class="w-5 h-5 mr-2" fill="currentColor" viewBox="0 0 20 20">
+                        <path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd"/>
+                    </svg>
+                    <?= $action === 'create' ? 'Create Slot' : 'Update Slot' ?>
+                </button>
+            </div>
+        </form>
+    </div>
+</div>
+
+<?php
+$content = ob_get_clean();
+include __DIR__ . '/../partials/layout.php';
+?>

+ 163 - 0
resources/view/admin/slots/list.php

@@ -0,0 +1,163 @@
+<?php
+ob_start();
+?>
+
+<div class="min-h-screen bg-gray-50 py-6">
+    <div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
+        <!-- Header -->
+        <div class="md:flex md:items-center md:justify-between mb-8">
+            <div class="flex-1 min-w-0">
+                <h2 class="text-2xl font-bold leading-7 text-gray-900 sm:text-3xl sm:truncate">
+                    Slot Management
+                </h2>
+                <p class="mt-1 text-sm text-gray-500">
+                    Manage casino slots and games
+                </p>
+            </div>
+            <div class="mt-4 flex md:mt-0 md:ml-4">
+                <a href="/admin/slots/?action=create" 
+                   class="ml-3 inline-flex items-center px-4 py-2 border border-transparent rounded-md shadow-sm text-sm font-medium text-white bg-blue-600 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 transition-colors">
+                    <svg class="w-5 h-5 mr-2" fill="currentColor" viewBox="0 0 20 20">
+                        <path fill-rule="evenodd" d="M10 3a1 1 0 011 1v5h5a1 1 0 110 2h-5v5a1 1 0 11-2 0v-5H4a1 1 0 110-2h5V4a1 1 0 011-1z" clip-rule="evenodd"/>
+                    </svg>
+                    Add New Slot
+                </a>
+            </div>
+        </div>
+
+        <!-- Success/Error Messages -->
+        <?php if (isset($success)): ?>
+            <div class="mb-6 bg-green-50 border border-green-200 rounded-md p-4">
+                <div class="flex">
+                    <div class="flex-shrink-0">
+                        <svg class="h-5 w-5 text-green-400" viewBox="0 0 20 20" fill="currentColor">
+                            <path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clip-rule="evenodd"/>
+                        </svg>
+                    </div>
+                    <div class="ml-3">
+                        <p class="text-sm font-medium text-green-800"><?= htmlspecialchars($success) ?></p>
+                    </div>
+                </div>
+            </div>
+        <?php endif; ?>
+
+        <?php if (isset($error)): ?>
+            <div class="mb-6 bg-red-50 border border-red-200 rounded-md p-4">
+                <div class="flex">
+                    <div class="flex-shrink-0">
+                        <svg class="h-5 w-5 text-red-400" viewBox="0 0 20 20" fill="currentColor">
+                            <path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z" clip-rule="evenodd"/>
+                        </svg>
+                    </div>
+                    <div class="ml-3">
+                        <p class="text-sm font-medium text-red-800"><?= htmlspecialchars($error) ?></p>
+                    </div>
+                </div>
+            </div>
+        <?php endif; ?>
+
+        <!-- Slots Table -->
+        <div class="bg-white shadow overflow-hidden sm:rounded-md">
+            <div class="px-4 py-5 sm:px-6 border-b border-gray-200">
+                <h3 class="text-lg leading-6 font-medium text-gray-900">All Slots</h3>
+                <p class="mt-1 max-w-2xl text-sm text-gray-500">A list of all slots in the system.</p>
+            </div>
+            
+            <?php if (empty($slots)): ?>
+                <div class="text-center py-12">
+                    <svg class="mx-auto h-12 w-12 text-gray-400" stroke="currentColor" fill="none" viewBox="0 0 48 48">
+                        <path d="M34 40h10v-4a6 6 0 00-10.712-3.714M34 40H14m20 0v-4a9.971 9.971 0 00-.712-3.714M14 40H4v-4a6 6 0 0110.713-3.714M14 40v-4c0-1.313.253-2.566.713-3.714m0 0A9.971 9.971 0 0124 24c4.004 0 7.625 2.349 9.287 6m-9.287-6C16.348 24 12.723 26.349 11.061 30M4 16l4-4m0 0l4-4m-4 4L4 4m4 4v12"/>
+                    </svg>
+                    <h3 class="mt-2 text-sm font-medium text-gray-900">No slots</h3>
+                    <p class="mt-1 text-sm text-gray-500">Get started by creating a new slot.</p>
+                    <div class="mt-6">
+                        <a href="/admin/slots/?action=create" class="inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-blue-600 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500">
+                            <svg class="w-5 h-5 mr-2" fill="currentColor" viewBox="0 0 20 20">
+                                <path fill-rule="evenodd" d="M10 3a1 1 0 011 1v5h5a1 1 0 110 2h-5v5a1 1 0 11-2 0v-5H4a1 1 0 110-2h5V4a1 1 0 011-1z" clip-rule="evenodd"/>
+                            </svg>
+                            Add Slot
+                        </a>
+                    </div>
+                </div>
+            <?php else: ?>
+                <div class="overflow-x-auto">
+                    <table class="min-w-full divide-y divide-gray-200">
+                        <thead class="bg-gray-50">
+                            <tr>
+                                <th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
+                                    Slot
+                                </th>
+                                <th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
+                                    Provider
+                                </th>
+                                <th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
+                                    RTP
+                                </th>
+                                <th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
+                                    Created
+                                </th>
+                                <th scope="col" class="relative px-6 py-3">
+                                    <span class="sr-only">Actions</span>
+                                </th>
+                            </tr>
+                        </thead>
+                        <tbody class="bg-white divide-y divide-gray-200">
+                            <?php foreach ($slots as $slot): ?>
+                                <tr class="hover:bg-gray-50">
+                                    <td class="px-6 py-4 whitespace-nowrap">
+                                        <div class="flex items-center">
+                                            <?php if (!empty($slot['image'])): ?>
+                                                <div class="flex-shrink-0 h-10 w-10">
+                                                    <img class="h-10 w-10 rounded-lg object-cover" src="/<?= htmlspecialchars($slot['image']) ?>" alt="<?= htmlspecialchars($slot['name']) ?>">
+                                                </div>
+                                            <?php else: ?>
+                                                <div class="flex-shrink-0 h-10 w-10 bg-gray-300 rounded-lg flex items-center justify-center">
+                                                    <svg class="h-6 w-6 text-gray-500" fill="none" stroke="currentColor" viewBox="0 0 24 24">
+                                                        <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z"/>
+                                                    </svg>
+                                                </div>
+                                            <?php endif; ?>
+                                            <div class="ml-4">
+                                                <div class="text-sm font-medium text-gray-900"><?= htmlspecialchars($slot['name']) ?></div>
+                                                <?php if (!empty($slot['demo_url'])): ?>
+                                                    <div class="text-sm text-gray-500">
+                                                        <a href="<?= htmlspecialchars($slot['demo_url']) ?>" target="_blank" class="text-blue-600 hover:text-blue-800">Demo available</a>
+                                                    </div>
+                                                <?php endif; ?>
+                                            </div>
+                                        </div>
+                                    </td>
+                                    <td class="px-6 py-4 whitespace-nowrap">
+                                        <div class="text-sm text-gray-900"><?= htmlspecialchars($slot['provider'] ?? 'N/A') ?></div>
+                                    </td>
+                                    <td class="px-6 py-4 whitespace-nowrap">
+                                        <div class="text-sm text-gray-900"><?= $slot['rtp'] ? number_format($slot['rtp'], 2) . '%' : 'N/A' ?></div>
+                                    </td>
+                                    <td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
+                                        <?= date('M j, Y', strtotime($slot['created_at'])) ?>
+                                    </td>
+                                    <td class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
+                                        <div class="flex items-center justify-end space-x-2">
+                                            <a href="/admin/slots/?action=edit&id=<?= $slot['id'] ?>" 
+                                               class="text-blue-600 hover:text-blue-900 font-medium">Edit</a>
+                                            <form method="POST" class="inline" onsubmit="return confirm('Are you sure you want to delete this slot?');">
+                                                <input type="hidden" name="action" value="delete">
+                                                <input type="hidden" name="id" value="<?= $slot['id'] ?>">
+                                                <button type="submit" class="text-red-600 hover:text-red-900 font-medium ml-2">Delete</button>
+                                            </form>
+                                        </div>
+                                    </td>
+                                </tr>
+                            <?php endforeach; ?>
+                        </tbody>
+                    </table>
+                </div>
+            <?php endif; ?>
+        </div>
+    </div>
+</div>
+
+<?php
+$content = ob_get_clean();
+include __DIR__ . '/../partials/layout.php';
+?>

+ 95 - 0
src/Models/Slot.php

@@ -0,0 +1,95 @@
+<?php
+
+namespace App\Models;
+
+use App\Database;
+
+class Slot
+{
+    private $db;
+    
+    public function __construct()
+    {
+        $this->db = Database::getInstance();
+    }
+    
+    public function getAll()
+    {
+        return $this->db->fetchAll("SELECT * FROM slots ORDER BY created_at DESC");
+    }
+    
+    public function getById($id)
+    {
+        return $this->db->fetchOne("SELECT * FROM slots WHERE id = ?", [$id]);
+    }
+    
+    public function getByProvider($provider)
+    {
+        return $this->db->fetchAll("SELECT * FROM slots WHERE provider = ? ORDER BY name ASC", [$provider]);
+    }
+    
+    public function create($data)
+    {
+        $sql = "INSERT INTO slots (name, image, provider, rtp, demo_url) VALUES (?, ?, ?, ?, ?)";
+        $params = [
+            $data['name'],
+            $data['image'] ?? null,
+            $data['provider'] ?? null,
+            $data['rtp'] ?? null,
+            $data['demo_url'] ?? null
+        ];
+        
+        $this->db->execute($sql, $params);
+        return $this->db->lastInsertId();
+    }
+    
+    public function update($id, $data)
+    {
+        $sql = "UPDATE slots SET name = ?, image = ?, provider = ?, rtp = ?, demo_url = ?, updated_at = CURRENT_TIMESTAMP WHERE id = ?";
+        $params = [
+            $data['name'],
+            $data['image'] ?? null,
+            $data['provider'] ?? null,
+            $data['rtp'] ?? null,
+            $data['demo_url'] ?? null,
+            $id
+        ];
+        
+        return $this->db->execute($sql, $params);
+    }
+    
+    public function delete($id)
+    {
+        return $this->db->execute("DELETE FROM slots WHERE id = ?", [$id]);
+    }
+    
+    public function exists($name, $excludeId = null)
+    {
+        if ($excludeId) {
+            return $this->db->fetchOne("SELECT id FROM slots WHERE name = ? AND id != ?", [$name, $excludeId]);
+        }
+        
+        return $this->db->fetchOne("SELECT id FROM slots WHERE name = ?", [$name]);
+    }
+    
+    public function getProviders()
+    {
+        $result = $this->db->fetchAll("SELECT DISTINCT provider FROM slots WHERE provider IS NOT NULL ORDER BY provider ASC");
+        return array_column($result, 'provider');
+    }
+    
+    public function search($query, $provider = null)
+    {
+        $sql = "SELECT * FROM slots WHERE name LIKE ?";
+        $params = ["%{$query}%"];
+        
+        if ($provider) {
+            $sql .= " AND provider = ?";
+            $params[] = $provider;
+        }
+        
+        $sql .= " ORDER BY name ASC";
+        
+        return $this->db->fetchAll($sql, $params);
+    }
+}

+ 104 - 0
src/Pages/admin/slots.php

@@ -0,0 +1,104 @@
+<?php
+
+use App\Models\Slot;
+
+$slotModel = new Slot();
+$action = $_GET['action'] ?? 'list';
+$id = $_GET['id'] ?? null;
+
+// Handle form submissions
+if ($_SERVER['REQUEST_METHOD'] === 'POST') {
+    $data = $_POST;
+    
+    // Handle image upload
+    if (!empty($_FILES['image_upload']['name'])) {
+        $uploadedImage = uploadImage($_FILES['image_upload']);
+        if ($uploadedImage) {
+            $data['image'] = $uploadedImage;
+        }
+    } else {
+        // Keep current image if no new upload
+        $data['image'] = $data['current_image'] ?? '';
+    }
+    
+    // Get action from form data
+    $formAction = $data['action'] ?? $action;
+    
+    try {
+        switch ($formAction) {
+            case 'create':
+            case 'edit':
+                if ($formAction === 'create') {
+                    $slotId = $slotModel->create($data);
+                    $success = "Slot created successfully!";
+                } else {
+                    $slotModel->update($id, $data);
+                    $slotId = $id;
+                    $success = "Slot updated successfully!";
+                }
+                header("Location: /admin/slots/?action=edit&id=" . $slotId);
+                exit;
+                break;
+                
+            case 'delete':
+                if ($id) {
+                    $slotModel->delete($id);
+                    $success = "Slot deleted successfully!";
+                    $action = 'list';
+                }
+                break;
+        }
+    } catch (Exception $e) {
+        $error = "Error: " . $e->getMessage();
+    }
+}
+
+// Get data for different actions
+switch ($action) {
+    case 'create':
+    case 'new':
+        $slot = [
+            'name' => '',
+            'image' => '',
+            'provider' => '',
+            'rtp' => '',
+            'demo_url' => ''
+        ];
+        $action = 'create';
+        break;
+        
+    case 'edit':
+        if (!$id) {
+            $action = 'list';
+            break;
+        }
+        
+        $slot = $slotModel->getById($id);
+        if (!$slot) {
+            $error = "Slot not found!";
+            $action = 'list';
+            break;
+        }
+        break;
+        
+    case 'list':
+    default:
+        $slots = $slotModel->getAll();
+        $action = 'list';
+        break;
+}
+
+// Determine which template to use
+$template = match($action) {
+    'list' => 'admin/slots/list.php',
+    'create', 'edit' => 'admin/slots/form.php',
+    default => 'admin/slots/list.php'
+};
+
+return ViewRender($template, [
+    'action' => $action,
+    'slots' => $slots ?? [],
+    'slot' => $slot ?? null,
+    'success' => $success ?? null,
+    'error' => $error ?? null
+]);