BNO ILR Continuous Residence Checker
Check whether you meet the 5-year continuous residence requirement to apply for Indefinite Leave to Remain under the Hong Kong BN(O) route.
1
Your BNO Leave Details
📅
e.g. 31/01/2021
📅
e.g. 31/01/2026
Your qualifying period is 5 years from the date your BNO leave was first granted. You can apply up to 28 days before completing 5 years.
2
Travel History (Absences from the UK)
Enter every trip outside the UK during your qualifying period. Include the date you left and the date you returned. Only whole days absent are counted (departure and return days are not counted).
3
Other ILR Requirements
✓
Eligibility Checklist
Residence Timeline
In the UK Absent from the UK
Rolling 12-Month Absence Analysis
The table below shows the maximum number of whole days absent within any rolling 12-month window during your qualifying period. You must not exceed 180 days in any such period.
| 12-Month Window Start | 12-Month Window End | Days Absent | Status |
|---|
Important Disclaimer:
This tool provides a preliminary assessment only and does not constitute legal advice. Immigration Rules are subject to change. For a full eligibility assessment tailored to your individual circumstances, please contact Dual HK UK Legal Services.
(function() {

// ==================== LANGUAGE ====================
let currentLang = new URLSearchParams(window.location.search).get('lang') === 'zh' ? 'zh' : 'en';

const MONTHS_EN = ['January','February','March','April','May','June','July','August','September','October','November','December'];
const MONTHS_EN_SHORT = ['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec'];
const MONTHS_ZH = ['1\u6708','2\u6708','3\u6708','4\u6708','5\u6708','6\u6708','7\u6708','8\u6708','9\u6708','10\u6708','11\u6708','12\u6708'];

function setLang(lang) {
  currentLang = lang;
  document.querySelectorAll('.lang-btn').forEach(btn => btn.classList.remove('active'));
  document.querySelectorAll('.lang-btn').forEach(btn => {
    if ((lang === 'en' && btn.textContent === 'English') || (lang === 'zh' && btn.textContent === '\u7e41\u9ad4\u4e2d\u6587'))
      btn.classList.add('active');
  });
  document.documentElement.lang = lang === 'zh' ? 'zh-Hant' : 'en';
  document.querySelectorAll('[data-' + lang + ']').forEach(el => {
    if (el.tagName === 'OPTION') return;
    // Don't overwrite date hint if it has success/error state
    if (el.classList && (el.classList.contains('success') || el.classList.contains('error'))) return;
    el.textContent = el.getAttribute('data-' + lang);
  });
  document.querySelectorAll('select option[data-' + lang + ']').forEach(opt => {
    opt.textContent = opt.getAttribute('data-' + lang);
  });

  // Re-validate date inputs to update hint text in new language
  ['leaveStartDate', 'applicationDate'].forEach(id => {
    const val = document.getElementById(id).value;
    if (val) {
      const hintEl = document.getElementById(id + 'Hint');
      if (hintEl) { hintEl.className = 'date-hint success'; hintEl.textContent = formatDateDisplay(val); }
    }
  });

  renderTrips();
}

function t(en, zh) { return currentLang === 'zh' ? zh : en; }

function formatDateDisplay(isoStr) {
  if (!isoStr) return '';
  const d = new Date(isoStr + 'T00:00:00');
  const months = currentLang === 'zh' ? MONTHS_ZH : MONTHS_EN;
  if (currentLang === 'zh') {
    return `${d.getFullYear()}\u5e74 ${months[d.getMonth()]} ${d.getDate()}\u65e5`;
  }
  return `${d.getDate()} ${months[d.getMonth()]} ${d.getFullYear()}`;
}

function formatDate(date) {
  const d = new Date(date);
  const months = currentLang === 'zh' ? MONTHS_ZH : MONTHS_EN_SHORT;
  if (currentLang === 'zh') {
    return `${d.getFullYear()}\u5e74 ${months[d.getMonth()]} ${d.getDate()}\u65e5`;
  }
  return `${d.getDate()} ${months[d.getMonth()]} ${d.getFullYear()}`;
}

// ==================== AUTO-FORMAT DATE INPUT ====================
function getDaysInMonth(month, year) {
  return new Date(year, month + 1, 0).getDate();
}

/**
 * initDateInput \u2014 attach auto-format behaviour to a text input.
 * As the user types digits the field auto-inserts " / " separators
 * so the display reads  DD / MM / YYYY.  It writes the ISO value
 * (YYYY-MM-DD) into the paired hidden input and shows a hint.
 *
 * @param {string} inputId  \u2014 visible text input id
 * @param {string} hiddenId \u2014 hidden input id that receives ISO value
 * @param {string} hintId   \u2014 hint div id
 */
function initDateInput(inputId, hiddenId, hintId) {
  const input = document.getElementById(inputId);
  const hidden = document.getElementById(hiddenId);
  const hint   = document.getElementById(hintId);

  // Strip non-digits and keep only up to 8 digits
  function digitsOnly(str) { return str.replace(/\D/g, '').slice(0, 8); }

  // Format 8 raw digits into "DD / MM / YYYY"
  function formatDisplay(digits) {
    let out = '';
    for (let i = 0; i < digits.length; i++) {
      if (i === 2 || i === 4) out += ' / ';
      out += digits[i];
    }
    return out;
  }

  // Given raw digits, figure caret position in formatted string
  function caretForDigitIndex(idx) {
    if (idx <= 2) return idx;
    if (idx <= 4) return idx + 3; // after " / "
    return idx + 6; // after two " / "
  }

  function validate(digits) {
    hidden.value = '';
    input.classList.remove('valid', 'invalid');
    if (hint) { hint.className = 'date-hint'; hint.textContent = hint.getAttribute('data-' + currentLang) || hint.getAttribute('data-en') || ''; }

    if (digits.length < 8) return;

    const dd = parseInt(digits.slice(0, 2), 10);
    const mm = parseInt(digits.slice(2, 4), 10);
    const yyyy = parseInt(digits.slice(4, 8), 10);

    if (mm < 1 || mm > 12) {
      input.classList.add('invalid');
      if (hint) { hint.className = 'date-hint error'; hint.textContent = t('Invalid month', '\u6708\u4efd\u7121\u6548'); }
      return;
    }
    const maxDay = getDaysInMonth(mm - 1, yyyy);
    if (dd < 1 || dd > maxDay) {
      input.classList.add('invalid');
      if (hint) { hint.className = 'date-hint error'; hint.textContent = t('Invalid day for this month', '\u8a72\u6708\u4efd\u7684\u65e5\u671f\u7121\u6548'); }
      return;
    }
    if (yyyy < 2019 || yyyy > 2036) {
      input.classList.add('invalid');
      if (hint) { hint.className = 'date-hint error'; hint.textContent = t('Year must be between 2019 and 2036', '\u5e74\u4efd\u9808\u4ecb\u4e4e 2019 \u81f3 2036'); }
      return;
    }

    const iso = `${yyyy}-${String(mm).padStart(2,'0')}-${String(dd).padStart(2,'0')}`;
    hidden.value = iso;
    input.classList.add('valid');
    if (hint) { hint.className = 'date-hint success'; hint.textContent = formatDateDisplay(iso); }
  }

  input.addEventListener('input', function (e) {
    const raw = digitsOnly(this.value);
    const formatted = formatDisplay(raw);
    this.value = formatted;
    // Place caret after the last typed digit
    const pos = caretForDigitIndex(raw.length);
    this.setSelectionRange(pos, pos);
    validate(raw);
  });

  // Allow arrow keys and backspace to work naturally, and handle paste
  input.addEventListener('keydown', function (e) {
    // Allow: backspace, delete, tab, escape, arrows, home, end, select-all
    if ([8,9,27,46,35,36,37,38,39,40].includes(e.keyCode) || (e.ctrlKey && e.keyCode === 65) || (e.metaKey && e.keyCode === 65)) return;
    // Allow paste
    if ((e.ctrlKey || e.metaKey) && (e.keyCode === 86 || e.key === 'v')) return;
    // Block non-digit keys
    if (!/^\d$/.test(e.key)) { e.preventDefault(); return; }
    // Block if already 8 digits
    const raw = digitsOnly(this.value);
    if (raw.length >= 8) { e.preventDefault(); }
  });

  // Handle paste
  input.addEventListener('paste', function (e) {
    e.preventDefault();
    const paste = (e.clipboardData || window.clipboardData).getData('text');
    const raw = digitsOnly(paste);
    const formatted = formatDisplay(raw);
    this.value = formatted;
    const pos = caretForDigitIndex(raw.length);
    this.setSelectionRange(pos, pos);
    validate(raw);
  });

  // Pre-populate if hidden input already has a value
  if (hidden.value) {
    const d = new Date(hidden.value + 'T00:00:00');
    const raw = String(d.getDate()).padStart(2,'0') + String(d.getMonth()+1).padStart(2,'0') + String(d.getFullYear());
    input.value = formatDisplay(raw);
    validate(raw);
  }
}

// ==================== TRIP MANAGEMENT ====================
let trips = [];
let tripIdCounter = 0;

function addTrip() {
  trips.push({ id: tripIdCounter++, departure: '', return: '' });
  renderTrips();
}

function removeTrip(id) {
  trips = trips.filter(t => t.id !== id);
  renderTrips();
}

// Trip date inputs write back into the trips array via a change watcher
function initTripDateInput(inputId, hiddenId, tripId, field) {
  const input = document.getElementById(inputId);
  const hidden = document.getElementById(hiddenId);

  function digitsOnly(str) { return str.replace(/\D/g, '').slice(0, 8); }

  function formatDisplay(digits) {
    let out = '';
    for (let i = 0; i < digits.length; i++) {
      if (i === 2 || i === 4) out += ' / ';
      out += digits[i];
    }
    return out;
  }

  function caretForDigitIndex(idx) {
    if (idx <= 2) return idx;
    if (idx <= 4) return idx + 3;
    return idx + 6;
  }

  function validate(digits) {
    hidden.value = '';
    input.classList.remove('valid', 'invalid');
    if (digits.length < 8) { updateTripField(tripId, field, ''); return; }
    const dd = parseInt(digits.slice(0, 2), 10);
    const mm = parseInt(digits.slice(2, 4), 10);
    const yyyy = parseInt(digits.slice(4, 8), 10);
    if (mm < 1 || mm > 12 || yyyy < 2019 || yyyy > 2036) { input.classList.add('invalid'); updateTripField(tripId, field, ''); return; }
    const maxDay = getDaysInMonth(mm - 1, yyyy);
    if (dd < 1 || dd > maxDay) { input.classList.add('invalid'); updateTripField(tripId, field, ''); return; }
    const iso = `${yyyy}-${String(mm).padStart(2,'0')}-${String(dd).padStart(2,'0')}`;
    hidden.value = iso;
    input.classList.add('valid');
    updateTripField(tripId, field, iso);
  }

  input.addEventListener('input', function () {
    const raw = digitsOnly(this.value);
    this.value = formatDisplay(raw);
    const pos = caretForDigitIndex(raw.length);
    this.setSelectionRange(pos, pos);
    validate(raw);
  });

  input.addEventListener('keydown', function (e) {
    if ([8,9,27,46,35,36,37,38,39,40].includes(e.keyCode) || (e.ctrlKey && e.keyCode === 65) || (e.metaKey && e.keyCode === 65)) return;
    if ((e.ctrlKey || e.metaKey) && (e.keyCode === 86 || e.key === 'v')) return;
    if (!/^\d$/.test(e.key)) { e.preventDefault(); return; }
    if (digitsOnly(this.value).length >= 8) { e.preventDefault(); }
  });

  input.addEventListener('paste', function (e) {
    e.preventDefault();
    const paste = (e.clipboardData || window.clipboardData).getData('text');
    const raw = digitsOnly(paste);
    this.value = formatDisplay(raw);
    const pos = caretForDigitIndex(raw.length);
    this.setSelectionRange(pos, pos);
    validate(raw);
  });

  // Pre-populate from trip data
  const trip = trips.find(t => t.id === tripId);
  if (trip && trip[field]) {
    const d = new Date(trip[field] + 'T00:00:00');
    const raw = String(d.getDate()).padStart(2,'0') + String(d.getMonth()+1).padStart(2,'0') + String(d.getFullYear());
    input.value = formatDisplay(raw);
    validate(raw);
  }
}

function updateTripField(tripId, field, value) {
  const trip = trips.find(t => t.id === tripId);
  if (trip) {
    trip[field] = value;
    // Update the days display for this trip
    const daysEl = document.getElementById(`tripDays_${tripId}`);
    if (daysEl) {
      const days = calcWholeDays(trip.departure, trip.return);
      daysEl.className = 'trip-days' + (days > 180 ? ' danger' : days > 150 ? ' warning' : '');
      daysEl.textContent = days > 0 ? days + ' ' + t('days', '\u5929') : '-';
    }
  }
}

function calcWholeDays(departure, returnDate) {
  if (!departure || !returnDate) return 0;
  const dep = new Date(departure);
  const ret = new Date(returnDate);
  if (ret <= dep) return 0;
  const msPerDay = 86400000;
  return Math.max(0, Math.floor((ret - dep) / msPerDay) - 1);
}

function renderTrips() {
  const list = document.getElementById('tripList');
  if (trips.length === 0) {
    list.innerHTML = '<p style="text-align:center;color:#718096;padding:20px;font-size:14px;">' +
      t('No trips added yet. Click below to add your travel history.', '\u5c1a\u672a\u6dfb\u52a0\u884c\u7a0b\u3002\u9ede\u64ca\u4e0b\u65b9\u65b0\u589e\u60a8\u7684\u65c5\u884c\u8a18\u9304\u3002') + '</p>';
    return;
  }

  list.innerHTML = trips.map(trip => {
    const days = calcWholeDays(trip.departure, trip.return);
    let daysClass = '';
    if (days > 180) daysClass = 'danger';
    else if (days > 150) daysClass = 'warning';

    const depInputId = `tripDep_${trip.id}`;
    const depHiddenId = `tripDepH_${trip.id}`;
    const retInputId = `tripRet_${trip.id}`;
    const retHiddenId = `tripRetH_${trip.id}`;

    return `
      <div class="trip-item">
        <div class="form-group">
          <label>${t('Left UK', '\u96e2\u958b\u82f1\u570b')}</label>
          <div class="date-input-wrap">
            <span class="date-input-icon">&#128197;</span>
            <input type="text" class="date-auto" id="${depInputId}" inputmode="numeric" placeholder="DD / MM / YYYY" maxlength="14" autocomplete="off" style="font-size:14px;padding:8px 10px 8px 34px;min-height:38px;">
          </div>
          <input type="hidden" id="${depHiddenId}">
        </div>
        <div class="form-group">
          <label>${t('Returned to UK', '\u8fd4\u56de\u82f1\u570b')}</label>
          <div class="date-input-wrap">
            <span class="date-input-icon">&#128197;</span>
            <input type="text" class="date-auto" id="${retInputId}" inputmode="numeric" placeholder="DD / MM / YYYY" maxlength="14" autocomplete="off" style="font-size:14px;padding:8px 10px 8px 34px;min-height:38px;">
          </div>
          <input type="hidden" id="${retHiddenId}">
        </div>
        <div class="trip-days ${daysClass}" id="tripDays_${trip.id}">${days > 0 ? days + ' ' + t('days', '\u5929') : '-'}</div>
        <button class="btn-remove" onclick="removeTrip(${trip.id})" title="${t('Remove', '\u522a\u9664')}">&#10005;</button>
      </div>
    `;
  }).join('');

  // Initialise auto-format handlers for each trip row
  trips.forEach(trip => {
    initTripDateInput(`tripDep_${trip.id}`, `tripDepH_${trip.id}`, trip.id, 'departure');
    initTripDateInput(`tripRet_${trip.id}`, `tripRetH_${trip.id}`, trip.id, 'return');
  });
}

// ==================== CALCULATION ====================
function calculate() {
  const leaveStart = document.getElementById('leaveStartDate').value;
  const appDate = document.getElementById('applicationDate').value;
  const lifeTest = document.getElementById('lifeInUkTest').value;
  const engLang = document.getElementById('englishLang').value;
  const criminal = document.getElementById('criminalRecord').value;


  if (!leaveStart || !appDate) {
    alert(t('Please select both the BNO leave start date and intended application date.', '\u8acb\u9078\u64c7 BNO \u7c3d\u8b49\u958b\u59cb\u65e5\u671f\u548c\u8a08\u5283\u7533\u8acb\u65e5\u671f\u3002'));
    return;
  }

  const leaveStartDt = new Date(leaveStart);
  const appDateDt = new Date(appDate);
  const qualifyingEndDt = new Date(leaveStartDt);
  qualifyingEndDt.setFullYear(qualifyingEndDt.getFullYear() + 5);

  const msPerDay = 86400000;
  const totalDays = Math.floor((appDateDt - leaveStartDt) / msPerDay);
  const daysUntilQualified = Math.floor((qualifyingEndDt - appDateDt) / msPerDay);
  const isOverstayer = daysUntilQualified < 0; // application date is after leave expires
  const hasCompleted5Years = daysUntilQualified <= 28 && !isOverstayer;
  const earliestAppDate = new Date(qualifyingEndDt.getTime() - 28 * msPerDay);
  const isTooEarly = daysUntilQualified > 28;

  let totalAbsenceDays = 0;
  const validTrips = trips.filter(trip => trip.departure && trip.return).map(trip => {
    const dep = new Date(trip.departure);
    const ret = new Date(trip.return);
    const days = calcWholeDays(trip.departure, trip.return);
    // Absent days: day after departure to day before return
    const absentStart = new Date(dep.getTime() + msPerDay);
    const absentEnd = new Date(ret.getTime() - msPerDay);
    const effStart = absentStart > leaveStartDt ? absentStart : leaveStartDt;
    const effEnd = absentEnd < appDateDt ? absentEnd : appDateDt;
    const effectiveDays = effEnd >= effStart ? Math.floor((effEnd - effStart) / msPerDay) + 1 : 0;
    return { departure: trip.departure, return: trip.return, totalDays: days, effectiveDays, depDate: dep, retDate: ret };
  }).sort((a, b) => a.depDate - b.depDate);

  validTrips.forEach(t => totalAbsenceDays += t.effectiveDays);

  let maxAbsenceIn12Months = 0;
  let worstWindowStart = null;
  let worstWindowEnd = null;

  const keyDates = new Set();
  validTrips.forEach(trip => {
    keyDates.add(trip.departure); keyDates.add(trip.return);
    const d1 = new Date(trip.departure); d1.setFullYear(d1.getFullYear() - 1); keyDates.add(d1.toISOString().split('T')[0]);
    const d2 = new Date(trip.return); d2.setFullYear(d2.getFullYear() - 1); keyDates.add(d2.toISOString().split('T')[0]);
    const d3 = new Date(trip.departure); d3.setFullYear(d3.getFullYear() + 1); keyDates.add(d3.toISOString().split('T')[0]);
  });

  const checkDate = new Date(leaveStartDt);
  while (checkDate <= appDateDt) { keyDates.add(checkDate.toISOString().split('T')[0]); checkDate.setMonth(checkDate.getMonth() + 1); }

  function countAbsenceDaysInWindow(windowStart, windowEnd) {
    let days = 0;
    validTrips.forEach(trip => {
      const depDt = new Date(trip.departure), retDt = new Date(trip.return);
      // Person is absent from day after departure to day before return
      const absentStart = new Date(depDt.getTime() + msPerDay);
      const absentEnd = new Date(retDt.getTime() - msPerDay);
      if (absentEnd < absentStart) return; // next-day return = 0 absent days
      const oS = absentStart > windowStart ? absentStart : windowStart;
      const oE = absentEnd < windowEnd ? absentEnd : windowEnd;
      if (oE >= oS) days += Math.floor((oE - oS) / msPerDay) + 1;
    });
    return days;
  }

  const checkedWindows = new Map();
  Array.from(keyDates).sort().forEach(dateStr => {
    const wS = new Date(dateStr);
    if (wS < leaveStartDt || wS > appDateDt) return;
    const wE = new Date(wS); wE.setFullYear(wE.getFullYear() + 1);
    if (wE > appDateDt) wE.setTime(appDateDt.getTime());
    if (checkedWindows.has(dateStr)) return;
    const abs = countAbsenceDaysInWindow(wS, wE);
    checkedWindows.set(dateStr, abs);
    if (abs > maxAbsenceIn12Months) { maxAbsenceIn12Months = abs; worstWindowStart = new Date(wS); worstWindowEnd = new Date(wE); }
  });

  validTrips.forEach(trip => {
    [trip.departure, trip.return].forEach(dateStr => {
      const dt = new Date(dateStr);
      const wS = new Date(dt); wS.setFullYear(wS.getFullYear() - 1);
      if (wS < leaveStartDt) wS.setTime(leaveStartDt.getTime());
      if (dt <= appDateDt) {
        const abs = countAbsenceDaysInWindow(wS, dt);
        if (abs > maxAbsenceIn12Months) { maxAbsenceIn12Months = abs; worstWindowStart = new Date(wS); worstWindowEnd = new Date(dt); }
      }
    });
  });

  const absenceBreached = maxAbsenceIn12Months > 180;

  const tableRows = [];
  if (validTrips.length > 0) {
    const seenStarts = new Set();
    const candidateWindows = [];
    validTrips.forEach(trip => {
      const wS1 = new Date(trip.return); wS1.setFullYear(wS1.getFullYear() - 1);
      if (wS1 < leaveStartDt) wS1.setTime(leaveStartDt.getTime());
      const wE1 = new Date(trip.return);
      if (wE1 > appDateDt) wE1.setTime(appDateDt.getTime());
      const key1 = wS1.toISOString().split('T')[0];
      if (!seenStarts.has(key1)) {
        seenStarts.add(key1);
        candidateWindows.push({ start: wS1, end: wE1, days: countAbsenceDaysInWindow(wS1, wE1), isWorst: false });
      }
      const wS2 = new Date(trip.departure);
      if (wS2 < leaveStartDt) wS2.setTime(leaveStartDt.getTime());
      const wE2 = new Date(wS2); wE2.setFullYear(wE2.getFullYear() + 1);
      if (wE2 > appDateDt) wE2.setTime(appDateDt.getTime());
      const key2 = wS2.toISOString().split('T')[0];
      if (!seenStarts.has(key2)) {
        seenStarts.add(key2);
        candidateWindows.push({ start: wS2, end: wE2, days: countAbsenceDaysInWindow(wS2, wE2), isWorst: false });
      }
    });
    candidateWindows.sort((a, b) => b.days - a.days);
    const topWindows = candidateWindows.slice(0, 6);
    if (worstWindowStart && worstWindowEnd) {
      const worstKey = worstWindowStart.toISOString().split('T')[0];
      const existing = topWindows.find(r => r.start.toISOString().split('T')[0] === worstKey);
      if (existing) { existing.isWorst = true; }
      else { topWindows.unshift({ start: worstWindowStart, end: worstWindowEnd, days: maxAbsenceIn12Months, isWorst: true }); }
    }
    topWindows.sort((a, b) => a.start - b.start);
    topWindows.forEach(w => tableRows.push(w));
  }

  const checks = {
    fiveYears: hasCompleted5Years, absence: !absenceBreached,
    lifeTest: lifeTest === 'passed' || lifeTest === 'exempt',
    english: engLang === 'met', character: criminal === 'none'
  };

  const allRequiredMet = checks.fiveYears && checks.absence;
  const allOptionalMet = checks.lifeTest && checks.english && checks.character;
  const someUnanswered = !lifeTest || !engLang || !criminal;
  let overallStatus = 'pass';
  if (isOverstayer) overallStatus = 'fail';
  else if (!allRequiredMet) overallStatus = 'fail';
  else if (!allOptionalMet && !someUnanswered) overallStatus = 'fail';
  else if (someUnanswered) overallStatus = 'warn';

  const resultsEl = document.getElementById('results');
  resultsEl.classList.add('show');
  const expBtn = document.getElementById('exportBtnContainer');
  if (expBtn) expBtn.style.display = 'block';

  const banner = document.getElementById('resultBanner');
  const icon = document.getElementById('resultIcon');
  const title = document.getElementById('resultTitle');
  const subtitle = document.getElementById('resultSubtitle');
  banner.className = 'result-banner ' + overallStatus;

  if (isOverstayer) {
    icon.innerHTML = '&#9888;'; title.textContent = t('OVERSTAYER - Your BNO leave has expired', '\u903e\u671f\u5c45\u7559 - \u60a8\u7684 BNO \u7c3d\u8b49\u5df2\u904e\u671f');
    subtitle.textContent = t('Your intended application date is after your BNO leave expires. You cannot apply for ILR without valid leave. Please seek legal advice immediately.', '\u60a8\u7684\u8a08\u5283\u7533\u8acb\u65e5\u671f\u5728 BNO \u7c3d\u8b49\u5230\u671f\u4e4b\u5f8c\u3002\u6c92\u6709\u6709\u6548\u7c3d\u8b49\ufffd \ufffd\u6cd5\u7533\u8acb\u6c38\u5c45\u3002\u8acb\u7acb\u5373\u5c0b\u6c42\u6cd5\u5f8b\u5efa\u8b70\u3002');
  } else if (overallStatus === 'pass') {
    icon.innerHTML = '&#10003;'; title.textContent = t('You appear to meet the continuous residence requirement', '\u60a8\u4f3c\u4e4e\u7b26\u5408\u9023\u7e8c\u5c45\u4f4f\u8981\u6c42');
    subtitle.textContent = t('Based on the information provided, you may be eligible to apply for ILR.', '\u6839\u64da\u6240\u63d0\u4f9b\u7684\u8cc7\u6599\uff0c\u60a8\u53ef\u80fd\u6709\u8cc7\u683c\u7533\u8acb\u6c38\u4e45\u5c45\u7559\u3002');
  } else if (overallStatus === 'warn') {
    icon.innerHTML = '&#9888;'; title.textContent = t('Residence requirement appears to be met, but some items are incomplete', '\u5c45\u4f4f\u8981\u6c42\u4f3c\u4e4e\u5df2\u6eff\u8db3\uff0c\u4f46\u90e8\u5206\u9805\u76ee\u5c1a\u672a\u586b\u5beb');
    subtitle.textContent = t('Please complete all fields above for a full assessment.', '\u8acb\u586b\u5beb\u4ee5\u4e0a\u6240\u6709\u6b04\u4f4d\u4ee5\u7372\u5f97\u5b8c\u6574\u8a55\u4f30\u3002');
  } else {
    icon.innerHTML = '&#10007;'; title.textContent = t('You may not currently meet the requirements for ILR', '\u60a8\u76ee\u524d\u53ef\u80fd\u4e0d\u7b26\u5408\u6c38\u5c45\u8981\u6c42');
    subtitle.textContent = t('Please review the details below and consider seeking professional advice.', '\u8acb\u67e5\u770b\u4ee5\u4e0b\u8a73\u7d30\u8cc7\u6599\u4e26\u8003\u616e\u5c0b\u6c42\u5c08\u696d\u6cd5\u5f8b\u5efa\u8b70\u3002');
  }

  const daysInUk = totalDays - totalAbsenceDays;
  const percentInUk = totalDays > 0 ? Math.round((daysInUk / totalDays) * 100) : 0;

  document.getElementById('statsGrid').innerHTML = `
    <div class="stat-card"><div class="stat-value">${totalDays}</div><div class="stat-label">${t('Total days in qualifying period', '\u5408\u8cc7\u683c\u671f\u9593\u7e3d\u5929\u6578')}</div></div>
    <div class="stat-card"><div class="stat-value" style="color:${totalAbsenceDays > 0 ? '#c05621' : '#276749'}">${totalAbsenceDays}</div><div class="stat-label">${t('Total whole days absent', '\u7e3d\u7f3a\u5e2d\u6574\u5929\u6578')}</div></div>
    <div class="stat-card"><div class="stat-value" style="color:${maxAbsenceIn12Months > 180 ? '#c53030' : maxAbsenceIn12Months > 150 ? '#c05621' : '#276749'}">${maxAbsenceIn12Months}</div><div class="stat-label">${t('Max days absent in any 12-month window', '\u4efb\u4f55 12 \u500b\u6708\u7a97\u53e3\u6700\u5927\u7f3a\u5e2d\u5929\u6578')}</div></div>
    <div class="stat-card"><div class="stat-value">${percentInUk}%</div><div class="stat-label">${t('Time spent in the UK', '\u5728\u82f1\u570b\u7684\u6642\u9593')}</div></div>`;

  const checkItems = [
    { status: isOverstayer ? 'fail' : checks.fiveYears ? 'pass' : 'fail', text: isOverstayer
      ? t(`OVERSTAYER WARNING: Your BNO leave expired on ${formatDate(qualifyingEndDt)}. Your application date is ${Math.abs(daysUntilQualified)} days after your leave expired. You must apply before your leave expires or obtain an extension. Seek legal advice immediately.`, `\u903e\u671f\u5c45\u7559\u8b66\u544a\uff1a\u60a8\u7684 BNO \u7c3d\u8b49\u5df2\u65bc ${formatDate(qualifyingEndDt)} \u5230\u671f\u3002\u60a8\u7684\u7533\u8acb\u65e5\u671f\u6bd4\u7c3d\u8b49\u5230\u671f\u65e5\u9072\u4e86 ${Math.abs(daysUntilQualified)} \u5929\u3002\u60a8\u5fc5\u9808\u5728\u7c3d\u8b49\u5230\u671f\u524d\u7533\u8acb\u6216\u7372\u5f97\u5ef6\u671f\u3002\u8acb\u7acb\u5373\u5c0b\u6c42\u6cd5\u5f8b\u5efa\u8b70\u3002`)
      : checks.fiveYears
        ? t(`5-year qualifying period: ${daysUntilQualified <= 0 ? 'Completed' : 'Completable within 28 days of application date'}`, `5 \u5e74\u5408\u8cc7\u683c\u671f\u9650\uff1a${daysUntilQualified <= 0 ? '\u5df2\u5b8c\u6210' : '\u53ef\u5728\u7533\u8acb\u65e5\u671f 28 \u5929\u5167\u5b8c\u6210'}`)
        : t(`5-year qualifying period: ${Math.abs(daysUntilQualified)} days remaining (earliest application: ${formatDate(earliestAppDate)})`, `5 \u5e74\u5408\u8cc7\u683c\u671f\u9650\uff1a\u5c1a\u9918 ${Math.abs(daysUntilQualified)} \u5929\uff08\u6700\u65e9\u7533\u8acb\u65e5\u671f\uff1a${formatDate(earliestAppDate)}\uff09`) },
    { status: checks.absence ? 'pass' : 'fail', text: checks.absence
      ? t(`Absences: Maximum ${maxAbsenceIn12Months} days in any rolling 12-month period (limit: 180 days)`, `\u7f3a\u5e2d\uff1a\u4efb\u4f55\u6efe\u52d5 12 \u500b\u6708\u671f\u9593\u6700\u591a ${maxAbsenceIn12Months} \u5929\uff08\u9650\u5236\uff1a180 \u5929\uff09`)
      : t(`Absences: ${maxAbsenceIn12Months} days in a 12-month period EXCEEDS the 180-day limit`, `\u7f3a\u5e2d\uff1a12 \u500b\u6708\u671f\u9593 ${maxAbsenceIn12Months} \u5929\uff0c\u8d85\u51fa 180 \u5929\u9650\u5236`) },
    { status: !lifeTest ? 'neutral' : checks.lifeTest ? 'pass' : 'fail', text: !lifeTest ? t('Life in the UK test: Not answered', '\u82f1\u570b\u751f\u6d3b\u5e38\u8b58\u8003\u8a66\uff1a\u672a\u56de\u7b54') : checks.lifeTest ? t('Life in the UK test: ' + (lifeTest === 'exempt' ? 'Exempt' : 'Passed'), '\u82f1\u570b\u751f\u6d3b\u5e38\u8b58\u8003\u8a66\uff1a' + (lifeTest === 'exempt' ? '\u8c41\u514d' : '\u5df2\u901a\u904e')) : t('Life in the UK test: Not yet passed', '\u82f1\u570b\u751f\u6d3b\u5e38\u8b58\u8003\u8a66\uff1a\u5c1a\u672a\u901a\u904e') },
    { status: !engLang ? 'neutral' : checks.english ? 'pass' : 'fail', text: !engLang ? t('English language: Not answered', '\u82f1\u8a9e\u8a9e\u8a00\uff1a\u672a\u56de\u7b54') : checks.english ? t('English language requirement: Satisfied', '\u82f1\u8a9e\u8a9e\u8a00\u8981\u6c42\uff1a\u5df2\u6eff\u8db3') : t('English language requirement: Not yet satisfied', '\u82f1\u8a9e\u8a9e\u8a00\u8981\u6c42\uff1a\u5c1a\u672a\u6eff\u8db3') },
    { status: !criminal ? 'neutral' : checks.character ? 'pass' : 'fail', text: !criminal ? t('Good character: Not answered', '\u826f\u597d\u54c1\u683c\uff1a\u672a\u56de\u7b54') : checks.character ? t('Good character: No criminal convictions', '\u826f\u597d\u54c1\u683c\uff1a\u7121\u5211\u4e8b\u5b9a\u7f6a') : t('Good character: Criminal record declared - seek legal advice', '\u826f\u597d\u54c1\u683c\uff1a\u5df2\u7533\u5831\u5211\u4e8b\u8a18\u9304 - \u8acb\u5c0b\u6c42\u6cd5\u5f8b\u5efa\u8b70') }
  ];
  document.getElementById('checklist').innerHTML = checkItems.map(item => `<li><div class="check-icon ${item.status}">${item.status === 'pass' ? '&#10003;' : item.status === 'fail' ? '&#10007;' : '?'}</div><div>${item.text}</div></li>`).join('');

  // Timeline
  renderTimeline(leaveStartDt, appDateDt, validTrips);

  // Rolling table
  const tbody = document.getElementById('rollingTableBody');
  if (tableRows.length === 0) {
    tbody.innerHTML = `<tr><td colspan="4" style="text-align:center;color:#718096;">${t('No absences recorded', '\u7121\u7f3a\u5e2d\u8a18\u9304')}</td></tr>`;
  } else {
    tbody.innerHTML = tableRows.map(row => {
      let sc = 'safe', st = t('Within limit', '\u5728\u9650\u5236\u4e4b\u5167');
      if (row.days > 180) { sc = 'danger'; st = t('EXCEEDS LIMIT', '\u8d85\u51fa\u9650\u5236'); }
      else if (row.days > 150) { sc = 'warning'; st = t('Approaching limit', '\u63a5\u8fd1\u9650\u5236'); }
      const label = row.isWorst ? ` <strong>(${t('worst window', '\u6700\u5dee\u7a97\u53e3')})</strong>` : '';
      return `<tr><td>${formatDate(row.start)}${label}</td><td>${formatDate(row.end)}</td><td class="${sc}">${row.days}</td><td class="${sc}">${st}</td></tr>`;
    }).join('');
  }

  resultsEl.scrollIntoView({ behavior: 'smooth', block: 'start' });
}

function renderTimeline(startDate, endDate, trips) {
  const container = document.getElementById('timeline');
  const msPerDay = 86400000;
  const totalDays = Math.floor((endDate - startDate) / msPerDay);
  if (totalDays <= 0) { container.innerHTML = '<p style="color:#718096;text-align:center;">Invalid date range</p>'; return; }

  let absenceBlocks = '';
  trips.forEach(trip => {
    const effectiveDep = trip.depDate < startDate ? startDate : trip.depDate;
    const effectiveRet = trip.retDate > endDate ? endDate : trip.retDate;
    if (effectiveRet <= effectiveDep) return;
    const left = ((effectiveDep - startDate) / msPerDay) / totalDays * 100;
    const width = ((effectiveRet - effectiveDep) / msPerDay) / totalDays * 100;
    absenceBlocks += `<div class="timeline-absence" style="left:${left}%;width:${Math.max(width, 0.5)}%;" title="${trip.effectiveDays} ${t('days', '\u5929')}">${trip.effectiveDays > 0 && width > 4 ? trip.effectiveDays + 'd' : ''}</div>`;
  });

  let yearMarkers = '';
  for (let y = startDate.getFullYear() + 1; y <= endDate.getFullYear(); y++) {
    const yd = new Date(y, 0, 1);
    if (yd > startDate && yd < endDate) {
      const pos = ((yd - startDate) / msPerDay) / totalDays * 100;
      yearMarkers += `<div style="position:absolute;left:${pos}%;top:0;height:100%;width:1px;background:rgba(0,0,0,0.1);"></div>`;
    }
  }

  container.innerHTML = `
    <div class="timeline-labels"><span>${formatDate(startDate)} (${t('Leave granted', '\u7c3d\u8b49\u7372\u6279')})</span><span>${formatDate(endDate)} (${t('Application', '\u7533\u8acb')})</span></div>
    <div class="timeline-bar">${yearMarkers}${absenceBlocks}</div>`;
}

// ==================== INIT ====================
initDateInput('leaveStartDateInput', 'leaveStartDate', 'leaveStartDateHint');
initDateInput('applicationDateInput', 'applicationDate', 'applicationDateHint');
addTrip();

// Expose functions globally for onclick handlers

function loadScript(src) {
  return new Promise((resolve, reject) => {
    if (document.querySelector('script[src="' + src + '"]')) { resolve(); return; }
    const s = document.createElement('script');
    s.src = src;
    s.onload = resolve;
    s.onerror = reject;
    document.head.appendChild(s);
  });
}

async function exportPDF() {
  const btn = document.querySelector('.btn-export');
  const origText = btn.querySelector('span').textContent;
  btn.disabled = true;
  btn.querySelector('span').textContent = currentLang === 'zh' ? '\u6b63\u5728\u751f\u6210 PDF...' : 'Generating PDF...';

  try {
    await loadScript('https://cdnjs.cloudflare.com/ajax/libs/html2canvas/1.4.1/html2canvas.min.js');
    await loadScript('https://cdnjs.cloudflare.com/ajax/libs/jspdf/2.5.1/jspdf.umd.min.js');

    const { jsPDF } = window.jspdf;
    const pdf = new jsPDF('p', 'mm', 'a4');
    const pageW = 210;
    const pageH = 297;
    const margin = 15;
    const contentW = pageW - margin * 2;

    // Build everything as HTML for html2canvas (fixes CJK font issue)
    const container = document.createElement('div');
    container.style.cssText = 'position:absolute;left:-9999px;top:0;width:860px;background:white;padding:0;';

    // Header as HTML
    const headerTitle = currentLang === 'zh' ? 'BNO \u6c38\u4e45\u5c45\u7559\u9023\u7e8c\u5c45\u4f4f\u6aa2\u67e5\u5668' : 'BNO ILR Continuous Residence Checker';
    const betaText = currentLang === 'zh' ? '\u6e2c\u8a66\u7248' : 'Beta';
    const reportLabel = currentLang === 'zh' ? '\u5831\u544a\u65e5\u671f' : 'Report date';
    const today = new Date().toLocaleDateString(currentLang === 'zh' ? 'zh-HK' : 'en-GB');
    const disclaimerText = currentLang === 'zh' ? '\u6b64\u5831\u544a\u50c5\u4f9b\u53c3\u8003\uff0c\u4e0d\u69cb\u6210\u6cd5\u5f8b\u610f\u898b\u3002' : 'This report is for reference only and does not constitute legal advice.';

    const headerEl = document.createElement('div');
    headerEl.style.cssText = 'background:#1a365d;color:white;padding:20px 24px 16px;margin-bottom:20px;font-family:Inter,system-ui,sans-serif;';
    headerEl.innerHTML = '<div style="display:flex;justify-content:space-between;align-items:flex-start;">' +
      '<div><div style="color:#e8c872;font-size:20px;font-weight:700;margin-bottom:4px;">' + headerTitle +
      ' <span style="font-size:11px;background:#e8c872;color:#1a365d;padding:2px 6px;border-radius:3px;margin-left:6px;vertical-align:middle;">' + betaText + '</span></div>' +
      '<div style="font-size:12px;opacity:0.85;">Dual HK UK Legal Services | dualhkuklegal.co.uk</div></div>' +
      '<div style="text-align:right;font-size:11px;opacity:0.8;">' + reportLabel + '<br><strong>' + today + '</strong></div></div>';
    container.appendChild(headerEl);

    // Clone input cards (first 3)
    const cards = document.querySelectorAll('#bno-ilr-checker .main > .card');
    for (let i = 0; i < Math.min(3, cards.length); i++) {
      const clone = cards[i].cloneNode(true);
      clone.style.marginBottom = '12px';
      container.appendChild(clone);
    }

    // Clone results section
    const results = document.getElementById('results');
    const resultsClone = results.cloneNode(true);
    const exportBtnClone = resultsClone.querySelector('#exportBtnContainer');
    if (exportBtnClone) exportBtnClone.remove();
    resultsClone.style.display = 'block';
    resultsClone.classList.add('show');
    container.appendChild(resultsClone);

    // Footer as HTML
    const footerEl = document.createElement('div');
    footerEl.style.cssText = 'margin-top:20px;padding:12px 24px;border-top:1px solid #ddd;font-size:10px;color:#999;font-family:Inter,system-ui,sans-serif;display:flex;justify-content:space-between;';
    footerEl.innerHTML = '<span>' + disclaimerText + '</span><span>Dual HK UK Legal Services</span>';
    container.appendChild(footerEl);

    document.body.appendChild(container);

    const canvas = await html2canvas(container, {
      scale: 2,
      useCORS: true,
      backgroundColor: '#ffffff',
      logging: false,
      onclone: function(clonedDoc) {
        var langKey = 'data-' + currentLang;
        clonedDoc.querySelectorAll('[' + langKey + ']').forEach(function(el) {
          if (el.tagName === 'OPTION') return;
          if (el.classList && (el.classList.contains('success') || el.classList.contains('error'))) return;
          el.textContent = el.getAttribute(langKey);
        });
        clonedDoc.querySelectorAll('select option[' + langKey + ']').forEach(function(opt) {
          opt.textContent = opt.getAttribute(langKey);
        });
      }
    });

    document.body.removeChild(container);

    const imgW = contentW;
    const pxPerMm = canvas.width / contentW;
    let srcY = 0;
    const srcTotalH = canvas.height;
    let firstPage = true;

    while (srcY < srcTotalH) {
      if (!firstPage) pdf.addPage();
      const availH = pageH - margin * 2;
      const slicePx = Math.min(availH * pxPerMm, srcTotalH - srcY);
      const sliceH = slicePx / pxPerMm;

      const sliceCanvas = document.createElement('canvas');
      sliceCanvas.width = canvas.width;
      sliceCanvas.height = Math.ceil(slicePx);
      const ctx = sliceCanvas.getContext('2d');
      ctx.drawImage(canvas, 0, srcY, canvas.width, slicePx, 0, 0, canvas.width, slicePx);

      const sliceData = sliceCanvas.toDataURL('image/jpeg', 0.95);
      pdf.addImage(sliceData, 'JPEG', margin, margin, contentW, sliceH);

      srcY += slicePx;
      firstPage = false;
    }

    const filename = 'BNO-ILR-Residence-Check-' + new Date().toISOString().slice(0,10) + '.pdf';
    pdf.save(filename);

  } catch (err) {
    console.error('PDF export error:', err);
    alert(currentLang === 'zh' ? 'PDF \u532f\u51fa\u5931\u6557\uff0c\u8acb\u518d\u8a66\u4e00\u6b21\u3002' : 'PDF export failed. Please try again.');
  } finally {
    btn.disabled = false;
    btn.querySelector('span').textContent = origText;
  }
}

window.setLang = setLang;
window.addTrip = addTrip;
window.removeTrip = removeTrip;
window.calculate = calculate;
window.exportPDF = exportPDF;

setLang(currentLang);
})();

