I have an input with type=text which I want to show stars like an input with type=password using only CSS.
Basically I
I ended up in this thread a lot of times recently. My solution is utilizing JQuery input event (though it can also be written in raw JS or even in C# (Blazor) should you need it, the idea would be the same):
The core part is:
if (isPasswordVisible) { // if password is visible, then simply update value stored in the dictionary
value = newValue;
passwordInputsValues[$passwordInput.attr("my-guid")] = value;
} else { // else compute and update stored value
const newValueUntilCaret = newValue.take(caretPosition); // take chars before the caret
const unchangedCharsAtStart = newValueUntilCaret.takeWhile(c => c === "●").length; // count unchanged chars from the beginning
const unchangedCharsAtEnd = newValue.skip(caretPosition).length; // count unchanged chars after the caret
const insertedValue = newValueUntilCaret.skip(unchangedCharsAtStart); // get newly added string if any
value = oldValue.take(unchangedCharsAtStart) + insertedValue + oldValue.takeLast(unchangedCharsAtEnd); // create new value as concatenation of old value left part, new string and old value right part
passwordInputsValues[$passwordInput.attr("my-guid")] = value; // store newly created value in the dictionary
$passwordInput.prop("value", value.split("").map(_ => "●").join("")); // set value of the input to new masked value
$passwordInput[0].setSelectionRange(caretPosition, caretPosition); // set caret position to match the appropriate position
}
Below is the complete code of an example password control (I will try to update it if any problems arise):
// Utils
var guid = () => {
return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, function (c) {
const r = Math.random() * 16 | 0;
const v = c === "x" ? r : r & 0x3 | 0x8;
return v.toString(16);
});
}
// Array Extensions
Object.defineProperty(Array.prototype, "skip", {
value: function (n) {
if (typeof (n) !== "number") {
throw new Error("n is not a number");
}
return this.slice(n);
},
writable: true,
configurable: true
});
Object.defineProperty(Array.prototype, "take", {
value: function (n) {
if (typeof (n) !== "number") {
throw new Error("n is not a number");
}
return this.slice(0, n);
},
writable: true,
configurable: true
});
Object.defineProperty(Array.prototype, "takeLast", {
value: function (n) {
if (typeof (n) !== "number") {
throw new Error("n is not a number");
}
return this.slice(Math.max(this.length - n, 0));
},
writable: true,
configurable: true
});
Object.defineProperty(Array.prototype, "takeWhile", {
value: function (condition) {
if (typeof (condition) !== "function") {
throw new Error("condition is not a function");
}
const arr = [];
for (let el of this) {
if (condition(el))
arr.push(el);
else
break;
}
return arr;
},
writable: true,
configurable: true
});
// String Extensions
Object.defineProperty(String.prototype, "skip", {
value: function (n) {
return this.split("").skip(n).join("");
},
writable: true,
configurable: true
});
Object.defineProperty(String.prototype, "take", {
value: function (n) {
return this.split("").take(n).join("");
},
writable: true,
configurable: true
});
Object.defineProperty(String.prototype, "takeLast", {
value: function (n) {
return this.split("").takeLast(n).join("");
},
writable: true,
configurable: true
});
Object.defineProperty(String.prototype, "takeWhile", {
value: function (condition) {
return this.split("").takeWhile(condition).join("");
},
writable: true,
configurable: true
});
// JQuery Document Ready
$(document).ready(function() {
let isPasswordVisible = false;
const passwordInputsValues = {};
for (let $pi of $(".my-password-input").toArray().map(pi => $(pi))) {
const uid = guid();
$pi.attr("my-guid", uid);
passwordInputsValues[uid] = $pi.prop("value");
}
$(document).on("input", ".my-password-input", async function(e) {
const $passwordInput = $(this);
const newValue = $passwordInput.prop("value");
const oldValue = passwordInputsValues[$passwordInput.attr("my-guid")] || ""; // first time it will be undefined
const caretPosition = Math.max($passwordInput[0].selectionStart, $passwordInput[0].selectionEnd);
let value;
if (isPasswordVisible) {
value = newValue;
passwordInputsValues[$passwordInput.attr("my-guid")] = value;
} else {
const newValueUntilCaret = newValue.take(caretPosition);
const unchangedCharsAtStart = newValueUntilCaret.takeWhile(c => c === "●").length;
const unchangedCharsAtEnd = newValue.skip(caretPosition).length;
const insertedValue = newValueUntilCaret.skip(unchangedCharsAtStart);
value = oldValue.take(unchangedCharsAtStart) + insertedValue + oldValue.takeLast(unchangedCharsAtEnd);
passwordInputsValues[$passwordInput.attr("my-guid")] = value;
$passwordInput.prop("value", value.split("").map(_ => "●").join(""));
$passwordInput[0].setSelectionRange(caretPosition, caretPosition);
}
});
$(document).on("click", ".my-btn-toggle-password-visibility", function() {
const $btnTogglePassword = $(this);
const $iconPasswordShown = $btnTogglePassword.find(".my-icon-password-shown");
const $iconPasswordHidden = $btnTogglePassword.find(".my-icon-password-hidden");
const $passwordInput = $btnTogglePassword.parents(".my-input-group").first().children(".my-password-input").first();
const value = passwordInputsValues[$passwordInput.attr("my-guid")];
if (!isPasswordVisible) {
$iconPasswordHidden.removeClass("my-d-flex").addClass("my-d-none");
$iconPasswordShown.removeClass("my-d-none").addClass("my-d-flex");
$passwordInput.prop("value", value);
isPasswordVisible = true;
} else {
$iconPasswordShown.removeClass("my-d-flex").addClass("my-d-none");
$iconPasswordHidden.removeClass("my-d-none").addClass("my-d-flex");
$passwordInput.prop("value", value.split("").map(_ => "●").join(""));
isPasswordVisible = false;
}
});
});
body {
padding-top: 0;
color: white;
margin: 0;
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
font-size: 16px;
font-weight: 400;
line-height: 1.5;
text-align: left;
height: 100%;
max-height: 100%;
background-image: linear-gradient(rgba(0,0,0,0.2), rgba(0,0,0,0.2)), url();
background-clip: border-box;
background-origin: padding-box;
background-attachment: scroll;
background-repeat: repeat;
background-size: auto;
background-position: left top;
}
.snippet-container {
background: linear-gradient(135deg, #202020, black);
position: relative;
display: flex;
justify-content: center;
align-items: center;
width: 100%;
height: 200px;
}
.my-password-input {
background: linear-gradient(to bottom, #303030, #000000);
color: white;
display: block;
position: relative;
box-sizing: border-box;
padding: 5px 9px;
line-height: 24px;
height: 34px;
box-shadow: inset 0 0 0 1px #404040;
font-size: 16px;
font-weight: 400;
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
transition: all .15s ease-in-out;
width: 100%;
border: none;
}
.my-password-input:enabled:focus {
color: white;
box-shadow: inset 0 0 0 1px #404040, 0 0 6px 2px blue;
outline: none;
}
.my-input-group {
position: relative;
}
.my-input-group > .my-input-group-prepend {
display: flex;
position: absolute;
left: 0;
top: 0;
}
.my-input-group > .my-input-group-append {
display: flex;
position: absolute;
right: 0;
top: 0;
}
.my-icon {
display: flex;
align-items: center;
justify-content: center;
overflow: hidden;
}
.my-input-group > .my-input-group-prepend > .my-icon,
.my-input-group > .my-input-group-append > .my-icon {
width: auto;
height: 16px;
max-width: none;
max-height: 16px;
flex: 0 0 auto;
margin: 9px;
}
.my-input-group > .my-input-group-prepend > .my-icon > svg,
.my-input-group > .my-input-group-append > .my-icon > svg {
height: 100%;
width: auto;
margin: 0;
padding: 0;
overflow: hidden;
}
.my-input-group > .my-input-group-prepend > .my-btn,
.my-input-group > .my-input-group-append > .my-btn {
height: 100% !important;
width: auto;
}
button:enabled {
cursor: pointer;
}
.my-btn {
background: linear-gradient(to bottom, #303030, #000000);
color: white;
position: relative;
box-sizing: border-box;
padding: 5px;
line-height: 24px;
height: 34px;
font-size: 16px;
font-weight: 400;
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
transition: all .15s ease-in-out;
width: 100%;
display: flex;
justify-content: center;
align-items: center;
border: none;
box-shadow: 0 0 0 0 #FFFFFF, inset 0 0 0 1px #404040;
}
.my-btn-primary {
color: #fff;
background: linear-gradient(to bottom, #00008B, #000000);
box-shadow: 0 0 0 0 #FFFFFF, inset 0 0 0 1px #0000FF;
}
.my-btn-primary:hover:enabled {
box-shadow: 0 0 6px 2px #FFFFFF, inset 0 0 0 1px #FFFFFF;
background: linear-gradient(to top, #00008B, #000000);
}
.my-btn > .my-icon {
margin: 4px;
width: auto;
height: 16px;
max-width: none;
max-height: 16px;
flex: 0 0 auto;
}
.my-btn > .my-icon > svg {
height: 100%;
width: auto;
}
.my-d-none {
display: none !important;
}
.my-d-flex {
display: flex !important;
}
::-webkit-input-placeholder {
color: #404040;
font-style: italic;
}
:-moz-placeholder {
color: #404040;
font-style: italic;
}
::-moz-placeholder {
color: #404040;
font-style: italic;
}
:-ms-input-placeholder {
color: #404040;
font-style: italic;
}
::-moz-selection {
background-color: #f8b700;
color: #352011;
}
::selection {
background-color: #f8b700;
color: #352011;
}