2 more files converted

This commit is contained in:
Alex Janousek
2025-10-21 16:38:35 -04:00
parent 5e32691393
commit d119ac9c18
2 changed files with 102 additions and 89 deletions

View File

@@ -11,53 +11,62 @@
this.start = function(element) {
let textarea = $(element);
let textarea = element;
let visibleTextbox;
this.highlightPlaceholders = (
typeof textarea.data('highlightPlaceholders') === 'undefined' ||
!!textarea.data('highlightPlaceholders')
typeof textarea.dataset.highlightPlaceholders === 'undefined' ||
textarea.dataset.highlightPlaceholders !== 'false'
);
this.$textbox = textarea
.wrap(`
<div class='textbox-highlight-wrapper' />
`)
.after(this.$background = $(`
<div class="textbox-highlight-background" aria-hidden="true" />
`))
.on("input", this.update);
// Create wrapper div
const wrapper = document.createElement('div');
wrapper.className = 'textbox-highlight-wrapper';
$(window).on("resize", this.resize);
// Insert wrapper before textarea and move textarea into it
textarea.parentNode.insertBefore(wrapper, textarea);
wrapper.appendChild(textarea);
visibleTextbox = this.$textbox.clone().appendTo("body").css({
position: 'absolute',
visibility: 'hidden',
display: 'block'
});
this.initialHeight = visibleTextbox.height();
// Create background div
this.background = document.createElement('div');
this.background.className = 'textbox-highlight-background';
this.background.setAttribute('aria-hidden', 'true');
this.$background.css({
'border-width': this.$textbox.css('border-width')
});
// Insert background after textarea
textarea.parentNode.insertBefore(this.background, textarea.nextSibling);
this.textbox = textarea;
this.textbox.addEventListener("input", this.update);
window.addEventListener("resize", this.resize);
// Clone textbox to measure initial height
visibleTextbox = this.textbox.cloneNode(true);
visibleTextbox.style.position = 'absolute';
visibleTextbox.style.visibility = 'hidden';
visibleTextbox.style.display = 'block';
document.body.appendChild(visibleTextbox);
this.initialHeight = visibleTextbox.offsetHeight;
const borderWidth = window.getComputedStyle(this.textbox).borderWidth;
this.background.style.borderWidth = borderWidth;
visibleTextbox.remove();
this.$textbox
.trigger("input");
this.textbox.dispatchEvent(new Event("input"));
};
this.resize = () => {
this.$background.width(this.$textbox.width());
const computedStyle = window.getComputedStyle(this.textbox);
const width = parseFloat(computedStyle.width);
this.background.style.width = width + 'px';
this.$textbox.height(
Math.max(
this.initialHeight,
this.$background.outerHeight()
)
);
const backgroundHeight = this.background.offsetHeight;
this.textbox.style.height = Math.max(this.initialHeight, backgroundHeight) + 'px';
if ('stickAtBottomWhenScrolling' in window.NotifyModules) {
window.NotifyModules.stickAtBottomWhenScrolling.recalculate();
@@ -65,7 +74,11 @@
};
this.contentEscaped = () => $('<div/>').text(this.$textbox.val()).html();
this.contentEscaped = () => {
const div = document.createElement('div');
div.textContent = this.textbox.value;
return div.innerHTML;
};
this.contentReplaced = () => this.contentEscaped().replace(
tagPattern, (match, name, separator, value) => value && separator ?
@@ -75,9 +88,8 @@
this.update = () => {
this.$background.html(
this.highlightPlaceholders ? this.contentReplaced() : this.contentEscaped()
);
this.background.innerHTML =
this.highlightPlaceholders ? this.contentReplaced() : this.contentEscaped();
this.resize();

View File

@@ -70,12 +70,13 @@
};
let shiftFocus = function(elementToFocus, component) {
const radios = component.querySelectorAll('[type=radio]');
// The first option is always the default
if (elementToFocus === 'default') {
$('[type=radio]', component).eq(0).focus();
if (elementToFocus === 'default' && radios[0]) {
radios[0].focus();
}
if (elementToFocus === 'option') {
$('[type=radio]', component).eq(1).focus();
if (elementToFocus === 'option' && radios[1]) {
radios[1].focus();
}
};
@@ -83,24 +84,22 @@
this.start = function(component) {
let $component = $(component);
let render = (state, data) => {
$component.html(renderStates[state](data));
component.innerHTML = renderStates[state](data);
};
// store array of all options in component
let choices = $('label', $component).toArray().map(function(element) {
let $element = $(element);
let choices = Array.from(component.querySelectorAll('label')).map(function(element) {
return {
'id': $element.attr('for'),
'label': $.trim($element.text()),
'value': $element.prev('input').attr('value')
'id': element.htmlFor,
'label': element.textContent.trim(),
'value': element.previousElementSibling.value
};
});
let categories = $component.data('categories').split(',');
let name = $component.find('input').eq(0).attr('name');
let categories = component.dataset.categories.split(',');
let name = component.querySelector('input').name;
let mousedownOption = null;
let showNowAsDefault = (
$component.data('show-now-as-default').toString() === 'true' ?
component.dataset.showNowAsDefault === 'true' ?
{'name': name} : false
);
@@ -129,22 +128,23 @@
const parentNode = event.target.parentNode;
if (parentNode === mousedownOption) {
const value = $('input', parentNode).attr('value');
const input = parentNode.querySelector('input');
const value = input ? input.value : '';
selectOption(value);
// clear tracking
mousedownOption = null;
$(document).off('mouseup', trackMouseup);
document.removeEventListener('mouseup', trackMouseup);
}
};
// set events
$component
.on('click', '.radio-select__button--category', function(event) {
// set events using event delegation
component.addEventListener('click', function(event) {
// Handle category button clicks
if (event.target.classList.contains('radio-select__button--category')) {
event.preventDefault();
let wordsInDay = $(this).attr('value').split(' ');
let wordsInDay = event.target.value.split(' ');
let day = wordsInDay[wordsInDay.length - 1].toLowerCase();
render('choose', {
'choices': choices.filter(
@@ -154,57 +154,58 @@
'showNowAsDefault': showNowAsDefault
});
shiftFocus('option', component);
}
})
.on('mousedown', '.js-option', function(event) {
mousedownOption = this;
// mouseup on the same option completes the click action
$(document).on('mouseup', trackMouseup);
})
// space and enter, clicked on a radio confirm that option was selected
.on('keydown', 'input[type=radio]', function(event) {
// allow keypresses which arent enter or space through
if (event.which !== 13 && event.which !== 32) {
return true;
}
// Handle done button clicks
if (event.target.classList.contains('radio-select__button--done')) {
event.preventDefault();
let value = $(this).attr('value');
selectOption(value);
})
.on('click', '.radio-select__button--done', function(event) {
event.preventDefault();
let $selection = $('input[type=radio]:checked', this.parentNode);
if ($selection.length) {
let selection = event.target.parentNode.querySelector('input[type=radio]:checked');
if (selection) {
render('chosen', {
'choices': choices.filter(
element => element.value == $selection.eq(0).attr('value')
element => element.value == selection.value
),
'name': name,
'showNowAsDefault': showNowAsDefault
});
shiftFocus('option', component);
} else {
reset();
shiftFocus('default', component);
}
}
})
.on('click', '.radio-select__button--reset', function(event) {
// Handle reset button clicks
if (event.target.classList.contains('radio-select__button--reset')) {
event.preventDefault();
reset();
shiftFocus('default', component);
}
});
});
component.addEventListener('mousedown', function(event) {
// Handle option mousedown
const option = event.target.closest('.js-option');
if (option) {
mousedownOption = option;
// mouseup on the same option completes the click action
document.addEventListener('mouseup', trackMouseup);
}
});
component.addEventListener('keydown', function(event) {
// Handle radio keydown (space and enter)
if (event.target.type === 'radio') {
// allow keypresses which aren't enter or space through
if (event.which !== 13 && event.which !== 32) {
return true;
}
event.preventDefault();
let value = event.target.value;
selectOption(value);
}
});
// set HTML to initial state
render('initial', {
@@ -213,7 +214,7 @@
'showNowAsDefault': showNowAsDefault
});
$component.css({'height': 'auto'});
component.style.height = 'auto';
};