initial: bootstrap from BukidBountyApp base
This commit is contained in:
274
resources/js/Pages/Auth/Login.vue
Normal file
274
resources/js/Pages/Auth/Login.vue
Normal file
@@ -0,0 +1,274 @@
|
||||
<script setup>
|
||||
import { usePageTitle } from '../../composables/Core/usePageTitle';
|
||||
usePageTitle('Login');
|
||||
|
||||
import { ref, onMounted } from 'vue';
|
||||
import axios from 'axios';
|
||||
import { useNavigate } from '../../composables/Core/useNavigate.js';
|
||||
import { resetRole } from '../../composables/Core/useAuth.js';
|
||||
import { useUserStore } from '../../stores/user.js';
|
||||
|
||||
const { navigate } = useNavigate();
|
||||
import { useUIStore } from '../../stores/ui';
|
||||
const uiStore = useUIStore();
|
||||
|
||||
onMounted(() => {
|
||||
// Clear any stale session artifacts on login landing
|
||||
sessionStorage.clear();
|
||||
localStorage.clear();
|
||||
resetRole();
|
||||
useUserStore().resetCurrentUser();
|
||||
|
||||
// Unregister service workers as per the legacy login.blade.php
|
||||
if ('serviceWorker' in navigator) {
|
||||
navigator.serviceWorker.getRegistrations().then(function (registrations) {
|
||||
for (let registration of registrations) {
|
||||
registration.unregister();
|
||||
}
|
||||
}).catch(function (error) {
|
||||
console.error('Error while unregistering service workers:', error);
|
||||
});
|
||||
}
|
||||
|
||||
// Clear all Cache Storage to prevent stale service worker pages (stuck homepage issue)
|
||||
if (window.caches) {
|
||||
caches.keys().then(function(names) {
|
||||
for (let name of names) {
|
||||
caches.delete(name);
|
||||
}
|
||||
}).catch(function(error) {
|
||||
console.error('Error while clearing caches:', error);
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
const usernumber = ref('');
|
||||
const userpassword = ref('');
|
||||
const keepAlive = ref(false);
|
||||
const loading = ref(false);
|
||||
const errorMessage = ref('');
|
||||
|
||||
const handleLogin = async () => {
|
||||
if (!usernumber.value || !userpassword.value) {
|
||||
errorMessage.value = 'Please fill in all fields.';
|
||||
return;
|
||||
}
|
||||
|
||||
loading.value = true;
|
||||
errorMessage.value = '';
|
||||
|
||||
try {
|
||||
const response = await axios.post('/post/loginnow', {
|
||||
mobile_number: usernumber.value,
|
||||
password: userpassword.value,
|
||||
keepalive: keepAlive.value
|
||||
}, {
|
||||
withCredentials: true
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
// Redirect to home or reload to refresh auth state
|
||||
window.location.href = '/';
|
||||
} else {
|
||||
errorMessage.value = response.data.message || 'Invalid credentials.';
|
||||
if (window.toastr) window.toastr.error(errorMessage.value);
|
||||
userpassword.value = ''; // Clear password on failure
|
||||
}
|
||||
} catch (error) {
|
||||
errorMessage.value = error.response?.data?.message || 'An error occurred during login.';
|
||||
if (window.toastr) window.toastr.error(errorMessage.value);
|
||||
userpassword.value = '';
|
||||
} finally {
|
||||
loading.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const handleKeypress = (event) => {
|
||||
if (event.key === 'Enter') {
|
||||
handleLogin();
|
||||
}
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="login-page-container">
|
||||
<div class="login-card shadow-lg p-4">
|
||||
<div class="login-logo text-center mb-4">
|
||||
<img :src="uiStore.appLogo" :alt="uiStore.appName" class="login-logo-img mb-2">
|
||||
<h2 class="fw_7">{{ uiStore.appName }}</h2>
|
||||
</div>
|
||||
|
||||
<div v-if="errorMessage" class="alert alert-danger mb-4" role="alert">
|
||||
{{ errorMessage }}
|
||||
</div>
|
||||
|
||||
<div class="form-group mb-3">
|
||||
<label for="usernumber" class="form-label fw_6">Mobile Number</label>
|
||||
<div class="input-group custom-input-group">
|
||||
<span class="input-group-text custom-input-addon">
|
||||
<i class="fas fa-phone text-muted"></i>
|
||||
</span>
|
||||
<input type="text" id="usernumber" v-model="usernumber" class="form-control custom-input-field"
|
||||
placeholder="e.g. 09123456789" @keypress="handleKeypress">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="form-group mb-4">
|
||||
<label for="userpassword" class="form-label fw_6">Password</label>
|
||||
<div class="input-group custom-input-group">
|
||||
<span class="input-group-text custom-input-addon">
|
||||
<i class="fas fa-lock text-muted"></i>
|
||||
</span>
|
||||
<input type="password" id="userpassword" v-model="userpassword"
|
||||
class="form-control custom-input-field" placeholder="Your password" @keypress="handleKeypress">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
<div class="form-check mb-4">
|
||||
<input class="form-check-input" type="checkbox" v-model="keepAlive" id="keepAlive">
|
||||
<label class="form-check-label text-muted" for="keepAlive">
|
||||
Keep me signed in
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<button @click="handleLogin" class="btn btn-primary w-100 py-3 rounded-pill fw_7 mb-3" :disabled="loading">
|
||||
<template v-if="loading">
|
||||
<span class="spinner-border spinner-border-sm me-2" role="status" aria-hidden="true"></span>
|
||||
Signing in...
|
||||
</template>
|
||||
<template v-else>
|
||||
Sign In
|
||||
</template>
|
||||
</button>
|
||||
|
||||
<div class="text-center">
|
||||
<p class="text-muted small mb-0">Don't have an account?</p>
|
||||
<a href="#" class="text-primary fw_6 undecorated">Contact your administrator</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.login-page-container {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
min-height: calc(100vh - 120px);
|
||||
padding: 20px;
|
||||
background-color: var(--bg-primary);
|
||||
transition: background-color 0.3s ease;
|
||||
}
|
||||
|
||||
.login-card {
|
||||
max-width: 400px;
|
||||
width: 100%;
|
||||
background: var(--bg-card);
|
||||
border-radius: 20px;
|
||||
border: 1px solid var(--border-color);
|
||||
box-shadow: 0 10px 30px rgba(0, 0, 0, 0.05) !important;
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.login-logo h2 {
|
||||
margin-bottom: 0;
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.login-logo-img {
|
||||
max-width: 96px;
|
||||
height: auto;
|
||||
object-fit: contain;
|
||||
}
|
||||
|
||||
.form-label {
|
||||
font-size: 0.85rem;
|
||||
color: var(--text-secondary);
|
||||
margin-left: 5px;
|
||||
}
|
||||
|
||||
.custom-input-group {
|
||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.02);
|
||||
}
|
||||
|
||||
.custom-input-addon {
|
||||
background-color: var(--bg-secondary) !important;
|
||||
border: 1px solid var(--border-color) !important;
|
||||
border-right: none !important;
|
||||
border-radius: 12px 0 0 12px !important;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.custom-input-field {
|
||||
background-color: var(--bg-secondary) !important;
|
||||
border: 1px solid var(--border-color) !important;
|
||||
border-left: none !important;
|
||||
border-radius: 0 12px 12px 0 !important;
|
||||
padding: 12px;
|
||||
font-size: 1rem;
|
||||
color: var(--text-primary) !important;
|
||||
}
|
||||
|
||||
.custom-input-field:focus {
|
||||
box-shadow: none;
|
||||
border-color: var(--accent-color) !important;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: #42b983;
|
||||
border: none;
|
||||
font-size: 1.1rem;
|
||||
box-shadow: 0 4px 10px rgba(66, 185, 131, 0.3);
|
||||
transition: all 0.2s ease;
|
||||
color: white !important;
|
||||
}
|
||||
|
||||
.btn-primary:hover {
|
||||
background: #38a171;
|
||||
transform: translateY(-1px);
|
||||
box-shadow: 0 6px 15px rgba(66, 185, 131, 0.4);
|
||||
}
|
||||
|
||||
.btn-primary:active {
|
||||
transform: scale(0.98);
|
||||
}
|
||||
|
||||
.btn-primary:disabled {
|
||||
background: #a8dcc3;
|
||||
opacity: 0.7;
|
||||
}
|
||||
|
||||
.text-primary {
|
||||
color: #42b983 !important;
|
||||
}
|
||||
|
||||
.undecorated {
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.undecorated:hover {
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
/* Dark mode specific enhancements */
|
||||
:global(.dark-mode) .login-card {
|
||||
box-shadow: 0 10px 40px rgba(0, 0, 0, 0.4) !important;
|
||||
border-color: rgba(255, 255, 255, 0.05);
|
||||
}
|
||||
|
||||
:global(.dark-mode) .custom-input-addon,
|
||||
:global(.dark-mode) .custom-input-field {
|
||||
background-color: rgba(255, 255, 255, 0.03) !important;
|
||||
}
|
||||
|
||||
:global(.dark-mode) .form-check-input {
|
||||
background-color: var(--bg-tertiary);
|
||||
border-color: var(--border-color);
|
||||
}
|
||||
|
||||
:global(.dark-mode) .form-check-input:checked {
|
||||
background-color: #42b983;
|
||||
border-color: #42b983;
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user