@zag-js/password-input
Version:
Core logic for the password-input widget implemented as a state machine
130 lines (119 loc) • 3.74 kB
text/typescript
import { ariaAttr, dataAttr, isLeftClick } from "@zag-js/dom-query"
import type { NormalizeProps, PropTypes } from "@zag-js/types"
import { parts } from "./password-input.anatomy"
import * as dom from "./password-input.dom"
import type { PasswordInputApi, PasswordInputService } from "./password-input.types"
export function connect<T extends PropTypes>(
service: PasswordInputService,
normalize: NormalizeProps<T>,
): PasswordInputApi<T> {
const { scope, prop, context } = service
const visible = context.get("visible")
const disabled = !!prop("disabled")
const invalid = !!prop("invalid")
const readOnly = !!prop("readOnly")
const interactive = !(readOnly || disabled)
const translations = prop("translations")
return {
visible,
disabled,
invalid,
focus() {
dom.getInputEl(scope)?.focus()
},
setVisible(value) {
service.send({ type: "VISIBILITY.SET", value })
},
toggleVisible() {
service.send({ type: "VISIBILITY.SET", value: !visible })
},
getRootProps() {
return normalize.element({
...parts.root.attrs,
dir: prop("dir"),
"data-disabled": dataAttr(disabled),
"data-invalid": dataAttr(invalid),
"data-readonly": dataAttr(readOnly),
})
},
getLabelProps() {
return normalize.label({
...parts.label.attrs,
htmlFor: dom.getInputId(scope),
"data-disabled": dataAttr(disabled),
"data-invalid": dataAttr(invalid),
"data-readonly": dataAttr(readOnly),
})
},
getInputProps() {
return normalize.input({
...parts.input.attrs,
id: dom.getInputId(scope),
autoCapitalize: "off",
name: prop("name"),
required: prop("required"),
autoComplete: prop("autoComplete"),
spellCheck: false,
readOnly,
disabled,
type: visible ? "text" : "password",
"data-state": visible ? "visible" : "hidden",
"aria-invalid": ariaAttr(invalid),
"data-disabled": dataAttr(disabled),
"data-invalid": dataAttr(invalid),
"data-readonly": dataAttr(readOnly),
...(prop("ignorePasswordManagers") ? passwordManagerProps : {}),
})
},
getVisibilityTriggerProps() {
return normalize.button({
...parts.visibilityTrigger.attrs,
type: "button",
tabIndex: -1,
"aria-controls": dom.getInputId(scope),
"aria-expanded": visible,
"data-readonly": dataAttr(readOnly),
disabled,
"data-disabled": dataAttr(disabled),
"data-state": visible ? "visible" : "hidden",
"aria-label": translations?.visibilityTrigger?.(visible),
onPointerDown(event) {
if (!isLeftClick(event)) return
if (!interactive) return
event.preventDefault()
service.send({ type: "TRIGGER.CLICK" })
},
})
},
getIndicatorProps() {
return normalize.element({
...parts.indicator.attrs,
"aria-hidden": true,
"data-state": visible ? "visible" : "hidden",
"data-disabled": dataAttr(disabled),
"data-invalid": dataAttr(invalid),
"data-readonly": dataAttr(readOnly),
})
},
getControlProps() {
return normalize.element({
...parts.control.attrs,
"data-disabled": dataAttr(disabled),
"data-invalid": dataAttr(invalid),
"data-readonly": dataAttr(readOnly),
})
},
}
}
const passwordManagerProps = {
// 1Password
"data-1p-ignore": "",
// LastPass
"data-lpignore": "true",
// Bitwarden
"data-bwignore": "true",
// Dashlane
"data-form-type": "other",
// Proton Pass
"data-protonpass-ignore": "true",
}