initial: bootstrap from BukidBountyApp base

This commit is contained in:
Jonathan Sykes
2026-06-06 18:43:00 +08:00
commit eb4a5731fb
5674 changed files with 160857 additions and 0 deletions

View File

@@ -0,0 +1,493 @@
<script setup>
import { ref, onMounted } from 'vue';
import axios from 'axios';
import { usePageTitle } from '../composables/Core/usePageTitle';
const props = defineProps({ target: String });
usePageTitle('Join Cooperative');
const goToLogin = () => { window.location.href = '/login'; };
const cooperative = ref(null);
const prioritySectors = ref([]);
const loading = ref(true);
const errorMessage = ref('');
const fieldErrors = ref({});
const step = ref(1);
const submitting = ref(false);
const userHashkey = ref(null);
// ── Step 1 ──
const accountForm = ref({
name: '',
username: '',
mobile_number: '',
password: '',
});
const mobileError = ref('');
const validateMobile = (val) => {
if (!val) { mobileError.value = 'Mobile number is required.'; return false; }
if (!/^(09|\+639)\d{9}$/.test(val)) {
mobileError.value = 'Must be a valid Philippine mobile number (e.g. 09XXXXXXXXX).';
return false;
}
mobileError.value = '';
return true;
};
// ── Step 2 ──
const membershipTypes = ['REGULAR', 'ASSOCIATE', 'HONORARY', 'LABORATORY'];
const membershipLevels = ['PRIMARY', 'SECONDARY', 'TERTIARY'];
const commonBonds = ['Residential', 'Institutional', 'Occupational', 'Associational'];
const employmentStatuses = ['Employed', 'Underemployed', 'Unemployed', 'Self-employed'];
const slpTracks = [{ value: 'MD', label: 'Microenterprise Development (MD)' }, { value: 'EF', label: 'Employment Facilitation (EF)' }];
const tupadCategories = ['Underemployed', 'Displaced Worker', 'Senior Citizen (fit to work)', 'PWD', 'Solo Parent', 'Indigenous Person', 'Former Rebel'];
const vulnerabilityOptions = ['Indigenous People (IP)', 'Person with Disability (PWD)', 'Senior Citizen', 'Solo Parent', 'Out-of-School Youth (OSY)', 'Internally Displaced Person (IDP)', 'Distressed OFW', 'Former Rebel'];
const programOptions = ['SLP', 'TUPAD', 'OSEC/NSRP', '4Ps/Pantawid Pamilya', 'Listahanan'];
const memberForm = ref({
membership_type: 'REGULAR',
membership_level: 'PRIMARY',
year_beginning: new Date().getFullYear(),
officer_position: '',
officer_level: '',
concurrent_position: '',
concurrent_level: '',
cooperative_position: '',
cooperative_name_alt: '',
// Classification
priority_sector: [],
common_bond: '',
vulnerability_classifications: [],
// Gov IDs
philsys_id: '',
sss_number: '',
pagibig_number: '',
// SLP
slp_track: '',
slp_association_name: '',
listahanan_id: '',
fourtps_household_id: '',
// TUPAD
tupad_category: '',
tupad_insurance_beneficiary_name: '',
tupad_insurance_beneficiary_relation: '',
// OSEC/NSRP
preferred_occupation: '',
nsrp_skills: [],
employment_status: '',
// Programs
program_participation: [],
});
const newSkill = ref('');
const addSkill = () => {
const s = newSkill.value.trim();
if (s && !memberForm.value.nsrp_skills.includes(s)) {
memberForm.value.nsrp_skills.push(s);
}
newSkill.value = '';
};
const removeSkill = (i) => memberForm.value.nsrp_skills.splice(i, 1);
const fetchCooperative = async () => {
if (!props.target) {
errorMessage.value = 'No cooperative identifier provided.';
loading.value = false;
return;
}
try {
const [coopRes, settingsRes] = await Promise.all([
axios.get(`/api/public/cooperative/${props.target}`),
axios.get('/api/public/system-settings'),
]);
if (coopRes.data.success) cooperative.value = coopRes.data.data;
else errorMessage.value = coopRes.data.message || 'Cooperative not found.';
if (settingsRes.data?.priority_sectors) {
prioritySectors.value = settingsRes.data.priority_sectors;
}
} catch (err) {
errorMessage.value = err.response?.status === 404
? 'Cooperative not found. Please check the link.'
: 'Failed to load cooperative information.';
} finally {
loading.value = false;
}
};
const submitAccount = async () => {
if (submitting.value) return;
if (!validateMobile(accountForm.value.mobile_number)) return;
fieldErrors.value = {};
errorMessage.value = '';
submitting.value = true;
try {
const res = await axios.post('/api/public/cooperative/register', {
...accountForm.value,
cooperative_hash: props.target,
});
if (res.data.success) {
userHashkey.value = res.data.user_hashkey;
step.value = 2;
} else {
errorMessage.value = res.data.message || 'Registration failed.';
}
} catch (err) {
if (err.response?.data?.errors) fieldErrors.value = err.response.data.errors;
else errorMessage.value = err.response?.data?.message || 'An error occurred.';
} finally {
submitting.value = false;
}
};
const submitMembership = async () => {
if (submitting.value) return;
errorMessage.value = '';
submitting.value = true;
try {
const res = await axios.post('/api/public/cooperative/complete-membership', {
...memberForm.value,
user_hashkey: userHashkey.value,
cooperative_hash: props.target,
});
if (res.data.success) step.value = 3;
else errorMessage.value = res.data.message || 'Submission failed.';
} catch (err) {
errorMessage.value = err.response?.data?.message || 'An error occurred.';
} finally {
submitting.value = false;
}
};
onMounted(fetchCooperative);
</script>
<template>
<div class="container py-4" style="max-width:620px;">
<!-- Loading -->
<div v-if="loading" class="text-center py-5">
<div class="spinner-border text-primary" role="status"></div>
</div>
<div v-else-if="!cooperative && errorMessage" class="text-center py-5 animate-fade-in">
<i class="fas fa-exclamation-triangle fa-3x text-danger mb-3"></i>
<p class="text-danger fw-semibold">{{ errorMessage }}</p>
</div>
<template v-else-if="cooperative">
<!-- Coop Header -->
<div class="text-center mb-4 animate-fade-in">
<div class="bg-primary text-white rounded-circle d-inline-flex align-items-center justify-content-center mb-2" style="width:56px;height:56px;">
<i class="fas fa-users fa-lg"></i>
</div>
<h5 class="fw-bold mb-0">{{ cooperative.name }}</h5>
<span class="badge bg-primary-subtle text-primary rounded-pill px-3 mt-1 small">
{{ cooperative.cooperative_type || 'Cooperative' }}
</span>
<p v-if="cooperative.address" class="text-muted small mt-1 mb-0">
<i class="fas fa-map-marker-alt me-1"></i>{{ cooperative.address }}
</p>
</div>
<!-- Step Indicator -->
<div class="d-flex align-items-center justify-content-center gap-2 mb-4">
<template v-for="s in 2" :key="s">
<div class="rounded-circle d-inline-flex align-items-center justify-content-center fw-bold"
:class="step > s ? 'bg-success text-white' : step === s ? 'bg-primary text-white' : 'bg-light text-muted'"
style="width:32px;height:32px;font-size:13px;">
<i v-if="step > s" class="fas fa-check" style="font-size:11px;"></i>
<span v-else>{{ s }}</span>
</div>
<span class="small" :class="step >= s ? 'fw-semibold text-dark' : 'text-muted'">
{{ s === 1 ? 'Account' : 'Membership' }}
</span>
<i v-if="s < 2" class="fas fa-chevron-right text-muted small mx-1"></i>
</template>
</div>
<!-- STEP 1: Account -->
<div v-if="step === 1" class="card border-0 shadow-sm rounded-4 p-4 animate-fade-in">
<h6 class="fw-semibold mb-3">Create your account</h6>
<div v-if="errorMessage" class="alert alert-danger rounded-3 small py-2">{{ errorMessage }}</div>
<div class="mb-3">
<label class="form-label small fw-semibold">Full Name</label>
<input v-model="accountForm.name" type="text" class="form-control rounded-pill"
:class="{ 'is-invalid': fieldErrors.name }" placeholder="Juan Dela Cruz" />
<div v-if="fieldErrors.name" class="invalid-feedback">{{ fieldErrors.name[0] }}</div>
</div>
<div class="mb-3">
<label class="form-label small fw-semibold">Username</label>
<input v-model="accountForm.username" type="text" class="form-control rounded-pill"
:class="{ 'is-invalid': fieldErrors.username }" placeholder="juandelacruz" autocomplete="username" />
<div v-if="fieldErrors.username" class="invalid-feedback">{{ fieldErrors.username[0] }}</div>
</div>
<div class="mb-3">
<label class="form-label small fw-semibold">Mobile Number</label>
<input v-model="accountForm.mobile_number" type="tel"
class="form-control rounded-pill"
:class="{ 'is-invalid': mobileError || fieldErrors.mobile_number }"
placeholder="09XXXXXXXXX"
@blur="validateMobile(accountForm.mobile_number)" />
<div class="invalid-feedback">{{ mobileError || fieldErrors.mobile_number?.[0] }}</div>
<div class="form-text small text-muted">Philippine mobile number (09XXXXXXXXX or +639XXXXXXXXX)</div>
</div>
<div class="mb-4">
<label class="form-label small fw-semibold">Password</label>
<input v-model="accountForm.password" type="password" class="form-control rounded-pill"
:class="{ 'is-invalid': fieldErrors.password }" placeholder="Min. 6 characters" autocomplete="new-password" />
<div v-if="fieldErrors.password" class="invalid-feedback">{{ fieldErrors.password[0] }}</div>
</div>
<button @click="submitAccount" :disabled="submitting" class="btn btn-primary rounded-pill w-100 py-2 fw-semibold">
<span v-if="submitting" class="spinner-border spinner-border-sm me-2"></span>
<i v-else class="fas fa-arrow-right me-2"></i>
{{ submitting ? 'Creating account...' : 'Continue to Membership' }}
</button>
<div class="text-center mt-3">
<small class="text-muted">Already have an account?
<a href="#" @click.prevent="goToLogin" class="text-primary fw-semibold">Log in</a>
</small>
</div>
</div>
<!-- STEP 2: Membership -->
<div v-else-if="step === 2" class="animate-fade-in">
<div class="alert alert-success rounded-3 small py-2 mb-3">
<i class="fas fa-check-circle me-2"></i>Account created! Complete your membership application below.
</div>
<div v-if="errorMessage" class="alert alert-danger rounded-3 small py-2">{{ errorMessage }}</div>
<!-- Membership Info -->
<div class="card border-0 shadow-sm rounded-4 p-4 mb-3">
<h6 class="fw-semibold mb-3"><i class="fas fa-id-card text-primary me-2"></i>Membership Information</h6>
<div class="row g-3">
<div class="col-6">
<label class="form-label small fw-semibold">Type</label>
<select v-model="memberForm.membership_type" class="form-select rounded-pill">
<option v-for="t in membershipTypes" :key="t" :value="t">{{ t }}</option>
</select>
</div>
<div class="col-6">
<label class="form-label small fw-semibold">Level</label>
<select v-model="memberForm.membership_level" class="form-select rounded-pill">
<option v-for="l in membershipLevels" :key="l" :value="l">{{ l }}</option>
</select>
</div>
<div class="col-6">
<label class="form-label small fw-semibold">Year Joined</label>
<input v-model="memberForm.year_beginning" type="number" class="form-control rounded-pill" :min="1990" :max="new Date().getFullYear()" />
</div>
<div class="col-6">
<label class="form-label small fw-semibold">Common Bond</label>
<select v-model="memberForm.common_bond" class="form-select rounded-pill">
<option value=""> Select </option>
<option v-for="b in commonBonds" :key="b" :value="b">{{ b }}</option>
</select>
</div>
<div class="col-12" v-if="prioritySectors.length">
<label class="form-label small fw-semibold">Priority Sector <span class="text-muted fw-normal small">(select all that apply)</span></label>
<div class="row g-2 mt-1">
<div class="col-6 col-md-4" v-for="s in prioritySectors" :key="s">
<div class="form-check">
<input class="form-check-input" type="checkbox" :id="'ps-' + s" :value="s" v-model="memberForm.priority_sector">
<label class="form-check-label small" :for="'ps-' + s">{{ s }}</label>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- Vulnerability Classifications -->
<div class="card border-0 shadow-sm rounded-4 p-4 mb-3">
<h6 class="fw-semibold mb-3"><i class="fas fa-shield-alt text-warning me-2"></i>Vulnerability Classification <span class="text-muted fw-normal small">(check all that apply)</span></h6>
<div class="row g-2">
<div class="col-6" v-for="opt in vulnerabilityOptions" :key="opt">
<div class="form-check">
<input class="form-check-input" type="checkbox" :id="'vuln-'+opt"
:value="opt" v-model="memberForm.vulnerability_classifications" />
<label class="form-check-label small" :for="'vuln-'+opt">{{ opt }}</label>
</div>
</div>
</div>
</div>
<!-- Program Participation -->
<div class="card border-0 shadow-sm rounded-4 p-4 mb-3">
<h6 class="fw-semibold mb-3"><i class="fas fa-list-check text-info me-2"></i>Government Program Participation <span class="text-muted fw-normal small">(check all that apply)</span></h6>
<div class="row g-2">
<div class="col-6" v-for="prog in programOptions" :key="prog">
<div class="form-check">
<input class="form-check-input" type="checkbox" :id="'prog-'+prog"
:value="prog" v-model="memberForm.program_participation" />
<label class="form-check-label small" :for="'prog-'+prog">{{ prog }}</label>
</div>
</div>
</div>
</div>
<!-- Gov IDs -->
<div class="card border-0 shadow-sm rounded-4 p-4 mb-3">
<h6 class="fw-semibold mb-3"><i class="fas fa-id-badge text-secondary me-2"></i>Government ID Numbers <span class="text-muted fw-normal small">(Optional)</span></h6>
<div class="row g-3">
<div class="col-md-4">
<label class="form-label small fw-semibold">PhilSys ID</label>
<input v-model="memberForm.philsys_id" type="text" class="form-control rounded-pill" placeholder="National ID" />
</div>
<div class="col-md-4">
<label class="form-label small fw-semibold">SSS Number</label>
<input v-model="memberForm.sss_number" type="text" class="form-control rounded-pill" placeholder="00-0000000-0" />
</div>
<div class="col-md-4">
<label class="form-label small fw-semibold">Pag-IBIG Number</label>
<input v-model="memberForm.pagibig_number" type="text" class="form-control rounded-pill" placeholder="0000-0000-0000" />
</div>
</div>
</div>
<!-- SLP -->
<div class="card border-0 shadow-sm rounded-4 p-4 mb-3" v-if="memberForm.program_participation.includes('SLP')">
<h6 class="fw-semibold mb-3"><i class="fas fa-seedling text-success me-2"></i>SLP Sustainable Livelihood Program</h6>
<div class="row g-3">
<div class="col-md-6">
<label class="form-label small fw-semibold">SLP Track</label>
<select v-model="memberForm.slp_track" class="form-select rounded-pill">
<option value=""> Select </option>
<option v-for="t in slpTracks" :key="t.value" :value="t.value">{{ t.label }}</option>
</select>
</div>
<div class="col-md-6">
<label class="form-label small fw-semibold">SLPA / Association Name</label>
<input v-model="memberForm.slp_association_name" type="text" class="form-control rounded-pill" placeholder="Association name" />
</div>
<div class="col-md-6">
<label class="form-label small fw-semibold">Listahanan (NHTO) Household ID</label>
<input v-model="memberForm.listahanan_id" type="text" class="form-control rounded-pill" />
</div>
<div class="col-md-6">
<label class="form-label small fw-semibold">4Ps Household ID</label>
<input v-model="memberForm.fourtps_household_id" type="text" class="form-control rounded-pill" />
</div>
</div>
</div>
<!-- TUPAD -->
<div class="card border-0 shadow-sm rounded-4 p-4 mb-3" v-if="memberForm.program_participation.includes('TUPAD')">
<h6 class="fw-semibold mb-3"><i class="fas fa-hard-hat text-warning me-2"></i>TUPAD Tulong Panghanapbuhay sa Ating Disadvantaged Workers</h6>
<div class="row g-3">
<div class="col-12">
<label class="form-label small fw-semibold">Beneficiary Category</label>
<select v-model="memberForm.tupad_category" class="form-select rounded-pill">
<option value=""> Select </option>
<option v-for="c in tupadCategories" :key="c" :value="c">{{ c }}</option>
</select>
</div>
<div class="col-md-6">
<label class="form-label small fw-semibold">Micro-insurance Beneficiary Name</label>
<input v-model="memberForm.tupad_insurance_beneficiary_name" type="text" class="form-control rounded-pill" placeholder="Full name" />
</div>
<div class="col-md-6">
<label class="form-label small fw-semibold">Relationship to Beneficiary</label>
<input v-model="memberForm.tupad_insurance_beneficiary_relation" type="text" class="form-control rounded-pill" placeholder="e.g. Spouse, Child" />
</div>
</div>
</div>
<!-- OSEC / NSRP -->
<div class="card border-0 shadow-sm rounded-4 p-4 mb-3" v-if="memberForm.program_participation.includes('OSEC/NSRP')">
<h6 class="fw-semibold mb-3"><i class="fas fa-briefcase text-primary me-2"></i>OSEC / NSRP Employment Profile</h6>
<div class="row g-3">
<div class="col-md-6">
<label class="form-label small fw-semibold">Employment Status</label>
<select v-model="memberForm.employment_status" class="form-select rounded-pill">
<option value=""> Select </option>
<option v-for="s in employmentStatuses" :key="s" :value="s">{{ s }}</option>
</select>
</div>
<div class="col-md-6">
<label class="form-label small fw-semibold">Preferred Occupation</label>
<input v-model="memberForm.preferred_occupation" type="text" class="form-control rounded-pill" placeholder="e.g. Farmer, Welder" />
</div>
<div class="col-12">
<label class="form-label small fw-semibold">Technical Skills</label>
<div class="d-flex gap-2 mb-2">
<input v-model="newSkill" type="text" class="form-control rounded-pill"
placeholder="Add a skill and press +" @keyup.enter="addSkill" />
<button @click="addSkill" class="btn btn-outline-primary rounded-pill px-3">+</button>
</div>
<div class="d-flex flex-wrap gap-1">
<span v-for="(sk, i) in memberForm.nsrp_skills" :key="i"
class="badge bg-primary-subtle text-primary rounded-pill px-3 py-2">
{{ sk }} <i class="fas fa-times ms-1 cursor-pointer" @click="removeSkill(i)"></i>
</span>
</div>
</div>
</div>
</div>
<!-- Position Details -->
<div class="card border-0 shadow-sm rounded-4 p-4 mb-3">
<h6 class="fw-semibold mb-3"><i class="fas fa-user-tie text-primary me-2"></i>Position Details <span class="text-muted fw-normal small">(Optional)</span></h6>
<div class="row g-3">
<div class="col-8">
<label class="form-label small fw-semibold">Officer Position</label>
<input v-model="memberForm.officer_position" type="text" class="form-control rounded-pill" placeholder="e.g. Board Member" />
</div>
<div class="col-4">
<label class="form-label small fw-semibold">Level</label>
<select v-model="memberForm.officer_level" class="form-select rounded-pill">
<option value=""></option>
<option v-for="l in membershipLevels" :key="l" :value="l">{{ l }}</option>
</select>
</div>
<div class="col-8">
<label class="form-label small fw-semibold">Concurrent Position</label>
<input v-model="memberForm.concurrent_position" type="text" class="form-control rounded-pill" placeholder="e.g. Treasurer" />
</div>
<div class="col-4">
<label class="form-label small fw-semibold">Level</label>
<select v-model="memberForm.concurrent_level" class="form-select rounded-pill">
<option value=""></option>
<option v-for="l in membershipLevels" :key="l" :value="l">{{ l }}</option>
</select>
</div>
<div class="col-12">
<label class="form-label small fw-semibold">Cooperative Position</label>
<input v-model="memberForm.cooperative_position" type="text" class="form-control rounded-pill" placeholder="e.g. Chairperson" />
</div>
<div class="col-12">
<label class="form-label small fw-semibold">Alternative Cooperative Name</label>
<input v-model="memberForm.cooperative_name_alt" type="text" class="form-control rounded-pill" placeholder="Former or alternate name" />
</div>
</div>
</div>
<button @click="submitMembership" :disabled="submitting" class="btn btn-success rounded-pill w-100 py-2 fw-semibold mb-4">
<span v-if="submitting" class="spinner-border spinner-border-sm me-2"></span>
<i v-else class="fas fa-paper-plane me-2"></i>
{{ submitting ? 'Submitting...' : 'Submit Membership Application' }}
</button>
</div>
<!-- STEP 3: Done -->
<div v-else-if="step === 3" class="text-center py-5 animate-fade-in">
<i class="fas fa-check-circle fa-4x text-success mb-3"></i>
<h4 class="fw-bold">Application Submitted!</h4>
<p class="text-muted">You are now registered as a member of <strong>{{ cooperative?.name }}</strong>.<br>You may log in with your credentials.</p>
<button @click="goToLogin" class="btn btn-primary rounded-pill px-4 mt-2">
<i class="fas fa-sign-in-alt me-2"></i> Go to Login
</button>
</div>
</template>
</div>
</template>