/* ═══════════════════════════════════════════════════════════════
   VALIDATORS · estils compartits
   Icona ✓/✗ a la dreta del input + missatge sota
   ═══════════════════════════════════════════════════════════════ */

/* Wrapper que envolta NOMÉS l'input (creat dinàmicament pel JS).
   Ocupa exactament l'alçada de l'input perquè la icona es pugui
   centrar verticalment de manera fiable a tots els layouts (flex
   column .alta-field, .field amb label sobre, etc.). */
.v-input-wrap {
  position: relative;
  display: block;
  width: 100%;
}
/* L'input ha d'omplir el wrapper · així els radius/padding queden
   intactes encara que ara sigui dins d'un span */
.v-input-wrap > input,
.v-input-wrap > textarea {
  width: 100%;
  display: block;
  box-sizing: border-box;
}

/* Contenidor extern (label.alta-field, .field, .ck-field) també
   relatiu per compatibilitat amb estils legacy on la icona encara
   no estigui dins del .v-input-wrap. */
.v-wrap,
.field:has(input[data-validate]),
label:has(input[data-validate]),
.alta-field:has(input[data-validate]),
.ck-field:has(input[data-validate]) {
  position: relative;
}

/* Icona ✓ / ✗ · a la dreta de l'input, sempre centrada verticalment
   perquè ara s'ancora al .v-input-wrap que té exactament l'alçada
   de l'input. */
.v-icon {
  position: absolute;
  right: 12px;
  top: 50%;
  transform: translateY(-50%);
  font-weight: 800;
  font-size: 14px;
  line-height: 1;
  pointer-events: none;
  z-index: 2;
  transition: opacity .15s ease;
}
.v-icon:empty { opacity: 0; }
.v-icon.v-ok  { color: #0F7A3D; }
.v-icon.v-err { color: #A61E1E; }

/* Compensa padding dret de l'input perquè el text no quedi sota la icona */
input.v-ok,
input.v-err {
  padding-right: 36px !important;
}

/* Estats del input */
input.v-ok {
  border-color: #0F7A3D !important;
  box-shadow: 0 0 0 3px rgba(15,122,61,.08) !important;
}
input.v-err {
  border-color: #A61E1E !important;
  box-shadow: 0 0 0 3px rgba(166,30,30,.08) !important;
}

/* Missatge de feedback sota l'input */
.v-msg {
  font-size: 11px;
  line-height: 1.35;
  margin-top: 4px;
  min-height: 1em;
  transition: opacity .15s ease;
}
.v-msg:empty { opacity: 0; }
.v-msg.v-err  { color: #A61E1E; font-weight: 600; }
.v-msg.v-warn { color: #8A5A0E; }

/* Mobile: assegura que la icona no es solapi amb el text */
@media (max-width: 480px) {
  .v-icon { right: 10px; font-size: 13px; }
  input.v-ok, input.v-err { padding-right: 32px !important; }
}
