mirror of
https://gitlab.ub.uni-bielefeld.de/sfb1288inf/nopaque.git
synced 2025-01-15 04:20:34 +00:00
Compare commits
No commits in common. "c046fbfb1eba0c7af359b829928c26322dec6da9" and "bf249193af9195cd2ffbaec5bf5f334575c57e60" have entirely different histories.
c046fbfb1e
...
bf249193af
@ -34,6 +34,7 @@ class ConcordanceQueryBuilder {
|
|||||||
{
|
{
|
||||||
onOpenStart: () => {
|
onOpenStart: () => {
|
||||||
this.tokenAttributeBuilderFunctions.preparePositionalAttrModal();
|
this.tokenAttributeBuilderFunctions.preparePositionalAttrModal();
|
||||||
|
this.tokenAttributeBuilderFunctions.optionToggleHandler();
|
||||||
},
|
},
|
||||||
onCloseStart: () => {
|
onCloseStart: () => {
|
||||||
this.tokenAttributeBuilderFunctions.resetPositionalAttrModal();
|
this.tokenAttributeBuilderFunctions.resetPositionalAttrModal();
|
||||||
|
@ -19,13 +19,17 @@ class ElementReferencesQueryBuilder {
|
|||||||
// Token Attribute Builder Elements
|
// Token Attribute Builder Elements
|
||||||
this.positionalAttrModal = M.Modal.getInstance(document.querySelector('#corpus-analysis-concordance-positional-attr-modal'));
|
this.positionalAttrModal = M.Modal.getInstance(document.querySelector('#corpus-analysis-concordance-positional-attr-modal'));
|
||||||
this.positionalAttrSelection = document.querySelector('#corpus-analysis-concordance-positional-attr-selection');
|
this.positionalAttrSelection = document.querySelector('#corpus-analysis-concordance-positional-attr-selection');
|
||||||
this.tokenBuilderContent = document.querySelector('#corpus-analysis-concordance-token-builder-content');
|
|
||||||
this.tokenQuery = document.querySelector('#corpus-analysis-concordance-token-query');
|
this.tokenQuery = document.querySelector('#corpus-analysis-concordance-token-query');
|
||||||
this.tokenQueryTemplate = document.querySelector('#corpus-analysis-concordance-token-query-template');
|
|
||||||
this.tokenSubmitButton = document.querySelector('#corpus-analysis-concordance-token-submit');
|
this.tokenSubmitButton = document.querySelector('#corpus-analysis-concordance-token-submit');
|
||||||
this.noValueMessage = document.querySelector('#corpus-analysis-concordance-no-value-message');
|
this.noValueMessage = document.querySelector('#corpus-analysis-concordance-no-value-message');
|
||||||
this.isTokenQueryInvalid = false;
|
this.isTokenQueryInvalid = false;
|
||||||
|
|
||||||
|
this.wordInput = document.querySelector('#corpus-analysis-concordance-word-input');
|
||||||
|
this.lemmaInput = document.querySelector('#corpus-analysis-concordance-lemma-input');
|
||||||
|
this.englishPosSelection = document.querySelector('#corpus-analysis-concordance-english-pos-selection');
|
||||||
|
this.germanPosSelection = document.querySelector('#corpus-analysis-concordance-german-pos-selection');
|
||||||
|
this.simplePosSelection = document.querySelector('#corpus-analysis-concordance-simple-pos-selection');
|
||||||
|
|
||||||
this.ignoreCaseCheckbox = document.querySelector('#corpus-analysis-concordance-ignore-case-checkbox');
|
this.ignoreCaseCheckbox = document.querySelector('#corpus-analysis-concordance-ignore-case-checkbox');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -34,9 +34,7 @@ class GeneralFunctionsQueryBuilder {
|
|||||||
|
|
||||||
resetMaterializeSelection(selectionElements, value = "default") {
|
resetMaterializeSelection(selectionElements, value = "default") {
|
||||||
selectionElements.forEach(selectionElement => {
|
selectionElements.forEach(selectionElement => {
|
||||||
if (selectionElement.querySelector(`option[value=${value}]`) !== null) {
|
|
||||||
selectionElement.querySelector(`option[value=${value}]`).selected = true;
|
selectionElement.querySelector(`option[value=${value}]`).selected = true;
|
||||||
}
|
|
||||||
let instance = M.FormSelect.getInstance(selectionElement);
|
let instance = M.FormSelect.getInstance(selectionElement);
|
||||||
instance.destroy();
|
instance.destroy();
|
||||||
M.FormSelect.init(selectionElement);
|
M.FormSelect.init(selectionElement);
|
||||||
|
@ -12,7 +12,13 @@ class TokenAttributeBuilderFunctionsQueryBuilder extends GeneralFunctionsQueryBu
|
|||||||
button.addEventListener('click', () => {this.actionButtonInOptionSectionHandler(button.dataset.optionsAction);});
|
button.addEventListener('click', () => {this.actionButtonInOptionSectionHandler(button.dataset.optionsAction);});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Eventlistener for kind of token
|
||||||
this.elements.tokenSubmitButton.addEventListener('click', () => {this.addTokenToQuery();});
|
this.elements.tokenSubmitButton.addEventListener('click', () => {this.addTokenToQuery();});
|
||||||
|
this.elements.wordInput.addEventListener('input', () => {this.optionToggleHandler();});
|
||||||
|
this.elements.lemmaInput.addEventListener('input', () => {this.optionToggleHandler();});
|
||||||
|
this.elements.englishPosSelection.addEventListener('change', () => {this.optionToggleHandler();});
|
||||||
|
this.elements.germanPosSelection.addEventListener('change', () => {this.optionToggleHandler();});
|
||||||
|
this.elements.simplePosSelection.addEventListener('change', () => {this.optionToggleHandler();});
|
||||||
}
|
}
|
||||||
|
|
||||||
resetPositionalAttrModal() {
|
resetPositionalAttrModal() {
|
||||||
@ -22,38 +28,32 @@ class TokenAttributeBuilderFunctionsQueryBuilder extends GeneralFunctionsQueryBu
|
|||||||
<option value="lemma" >lemma</option>
|
<option value="lemma" >lemma</option>
|
||||||
<option value="english-pos">english pos</option>
|
<option value="english-pos">english pos</option>
|
||||||
<option value="german-pos">german pos</option>
|
<option value="german-pos">german pos</option>
|
||||||
<option value="simple_pos">simple_pos</option>
|
<option value="simple-pos">simple_pos</option>
|
||||||
<option value="empty-token">empty token</option>
|
<option value="empty-token">empty token</option>
|
||||||
`;
|
`;
|
||||||
this.elements.positionalAttrSelection.innerHTML = originalSelectionList;
|
this.elements.positionalAttrSelection.innerHTML = originalSelectionList;
|
||||||
this.elements.tokenQuery.innerHTML = '';
|
this.elements.tokenQuery.innerHTML = '';
|
||||||
this.elements.tokenBuilderContent.innerHTML = '';
|
this.toggleClass(['word', 'lemma', 'english-pos', 'german-pos', 'simple-pos'], 'hide', 'add');
|
||||||
this.toggleClass(['input-field-options'], 'hide', 'remove');
|
this.toggleClass(['word', 'input-field-options'], 'hide', 'remove');
|
||||||
this.toggleClass(['incidence-modifiers', 'or', 'and'], 'disabled', 'add');
|
this.toggleClass(['incidence-modifiers', 'or', 'and'], 'disabled', 'add');
|
||||||
|
|
||||||
|
document.querySelector('#corpus-analysis-concordance-positional-attr-selection option[value="word"]').selected = true;
|
||||||
|
this.elements.wordInput.value = '';
|
||||||
|
this.elements.lemmaInput.value = '';
|
||||||
|
this.resetMaterializeSelection([this.elements.englishPosSelection, this.elements.germanPosSelection, this.elements.simplePosSelection]);
|
||||||
this.resetMaterializeSelection([this.elements.positionalAttrSelection], "word");
|
this.resetMaterializeSelection([this.elements.positionalAttrSelection], "word");
|
||||||
this.elements.ignoreCaseCheckbox.checked = false;
|
|
||||||
this.elements.editingModusOn = false;
|
this.elements.editingModusOn = false;
|
||||||
this.elements.editedQueryChipElementIndex = undefined;
|
this.elements.editedQueryChipElementIndex = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
preparePositionalAttrModal() {
|
preparePositionalAttrModal() {
|
||||||
let selection = this.elements.positionalAttrSelection.value;
|
let selection = this.elements.positionalAttrSelection.value;
|
||||||
|
this.toggleClass(['word', 'lemma', 'english-pos', 'german-pos', 'simple-pos'], 'hide', 'add');
|
||||||
if (selection !== 'empty-token') {
|
if (selection !== 'empty-token') {
|
||||||
let selectionTemplate = document.querySelector(`.token-builder-section[data-token-builder-section="${selection}"]`);
|
this.toggleClass([selection], 'hide', 'remove');
|
||||||
let selectionTemplateClone = selectionTemplate.content.cloneNode(true);
|
this.toggleClass(['incidence-modifiers', 'or', 'and'], 'disabled', 'add');
|
||||||
|
this.resetMaterializeSelection([this.elements.englishPosSelection, this.elements.germanPosSelection, this.elements.simplePosSelection]);
|
||||||
this.elements.tokenBuilderContent.innerHTML = '';
|
|
||||||
this.elements.tokenBuilderContent.appendChild(selectionTemplateClone);
|
|
||||||
if (this.elements.tokenBuilderContent.querySelector('select') !== null) {
|
|
||||||
let selectElement = this.elements.tokenBuilderContent.querySelector('select');
|
|
||||||
M.FormSelect.init(selectElement);
|
|
||||||
selectElement.addEventListener('change', () => {this.optionToggleHandler();});
|
|
||||||
} else {
|
|
||||||
this.elements.tokenBuilderContent.querySelector('input').addEventListener('input', () => {this.optionToggleHandler();});
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
this.optionToggleHandler();
|
|
||||||
|
|
||||||
if (selection === 'word' || selection === 'lemma') {
|
if (selection === 'word' || selection === 'lemma') {
|
||||||
this.toggleClass(['input-field-options'], 'hide', 'remove');
|
this.toggleClass(['input-field-options'], 'hide', 'remove');
|
||||||
} else if (selection === 'empty-token'){
|
} else if (selection === 'empty-token'){
|
||||||
@ -63,22 +63,35 @@ class TokenAttributeBuilderFunctionsQueryBuilder extends GeneralFunctionsQueryBu
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
tokenInputCheck(elem) {
|
tokenInputCheck() {
|
||||||
return elem.querySelector('select') !== null ? elem.querySelector('select') : elem.querySelector('input');
|
let input;
|
||||||
|
|
||||||
|
if (!document.querySelector('[data-toggle-area="word"]').classList.contains('hide')) {
|
||||||
|
input = this.elements.wordInput;
|
||||||
|
} else if (!document.querySelector('[data-toggle-area="lemma"]').classList.contains('hide')){
|
||||||
|
input = this.elements.lemmaInput;
|
||||||
|
} else if (!document.querySelector('[data-toggle-area="english-pos"]').classList.contains('hide')){
|
||||||
|
input = this.elements.englishPosSelection;
|
||||||
|
} else if (!document.querySelector('[data-toggle-area="german-pos"]').classList.contains('hide')){
|
||||||
|
input = this.elements.germanPosSelection;
|
||||||
|
} else if (!document.querySelector('[data-toggle-area="simple-pos"]').classList.contains('hide')){
|
||||||
|
input = this.elements.simplePosSelection;
|
||||||
|
}
|
||||||
|
|
||||||
|
return input;
|
||||||
}
|
}
|
||||||
|
|
||||||
optionToggleHandler() {
|
optionToggleHandler() {
|
||||||
let input = this.tokenInputCheck(this.elements.tokenBuilderContent);
|
let input = this.tokenInputCheck();
|
||||||
if ((input.value === '' || input.value === 'default') && this.elements.editingModusOn === false) {
|
if ((input.value === '' || input.value === 'default') && this.elements.editingModusOn === false) {
|
||||||
this.toggleClass(['incidence-modifiers', 'or', 'and'], 'disabled', 'add');
|
this.toggleClass(['incidence-modifiers', 'or', 'and'], 'disabled', 'add');
|
||||||
} else if (this.elements.positionalAttrSelection.querySelectorAll('option').length === 1) {
|
|
||||||
this.toggleClass(['and'], 'disabled', 'add');
|
|
||||||
} else {
|
} else {
|
||||||
this.toggleClass(['incidence-modifiers', 'or', 'and'], 'disabled', 'remove');
|
this.toggleClass(['incidence-modifiers', 'or', 'and'], 'disabled', 'remove');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
disableTokenSubmit() {
|
disableTokenSubmit() {
|
||||||
|
this.elements.isTokenQueryInvalid = true;
|
||||||
this.elements.tokenSubmitButton.classList.add('red');
|
this.elements.tokenSubmitButton.classList.add('red');
|
||||||
this.elements.noValueMessage.classList.remove('hide');
|
this.elements.noValueMessage.classList.remove('hide');
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
@ -89,52 +102,93 @@ class TokenAttributeBuilderFunctionsQueryBuilder extends GeneralFunctionsQueryBu
|
|||||||
}, 3000);
|
}, 3000);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tokenChipFactory(prettyQueryText, tokenText) {
|
||||||
|
tokenText = encodeURI(tokenText);
|
||||||
|
let builderElement;
|
||||||
|
let queryChipElement;
|
||||||
|
builderElement = document.createElement('div');
|
||||||
|
builderElement.innerHTML = `
|
||||||
|
<div class='chip col s2 l2' style='margin-top:20px;' data-tokentext='${tokenText}'>
|
||||||
|
${prettyQueryText}
|
||||||
|
<i class='material-icons close'>close</i>
|
||||||
|
</div>`;
|
||||||
|
queryChipElement = builderElement.firstElementChild;
|
||||||
|
this.elements.tokenQuery.appendChild(queryChipElement);
|
||||||
|
}
|
||||||
|
|
||||||
addTokenToQuery() {
|
addTokenToQuery() {
|
||||||
|
let c = this.elements.ignoreCaseCheckbox.checked ? ' %c' : '';
|
||||||
let tokenQueryPrettyText = '';
|
let tokenQueryPrettyText = '';
|
||||||
let tokenQueryCQLText = '';
|
let tokenQueryCQLText = '';
|
||||||
let input;
|
this.elements.isTokenQueryInvalid = false;
|
||||||
let kindOfToken = this.kindOfTokenCheck(this.elements.positionalAttrSelection.value);
|
|
||||||
|
|
||||||
// Takes all rows of the token query (if there is a query concatenation).
|
this.elements.tokenQuery.childNodes.forEach(element => {
|
||||||
// Adds their contents to tokenQueryPrettyText and tokenQueryCQLText, which will later be expanded with the current input field.
|
tokenQueryPrettyText += ' ' + element.firstChild.data + ' ';
|
||||||
let tokenQueryRows = this.elements.tokenQuery.querySelectorAll('.row');
|
tokenQueryCQLText += decodeURI(element.dataset.tokentext);
|
||||||
tokenQueryRows.forEach(row => {
|
|
||||||
let ignoreCaseCheckbox = row.querySelector('input[type="checkbox"]');
|
|
||||||
let c = ignoreCaseCheckbox !== null && ignoreCaseCheckbox.checked ? ' %c' : '';
|
|
||||||
let tokenQueryRowInput = this.tokenInputCheck(row.querySelector('.token-query-template-content'));
|
|
||||||
let tokenQueryKindOfToken = this.kindOfTokenCheck(tokenQueryRowInput.closest('.input-field').dataset.kindOfToken);
|
|
||||||
let tokenConditionPrettyText = row.querySelector('[data-condition-pretty-text]').dataset.conditionPrettyText;
|
|
||||||
let tokenConditionCQLText = row.querySelector('[data-condition-cql-text]').dataset.conditionCqlText;
|
|
||||||
tokenQueryPrettyText += `${tokenQueryKindOfToken}=${tokenQueryRowInput.value}${c} ${tokenConditionPrettyText} `;
|
|
||||||
tokenQueryCQLText += `${tokenQueryKindOfToken}="${tokenQueryRowInput.value}"${c} ${tokenConditionCQLText}`;
|
|
||||||
});
|
});
|
||||||
|
|
||||||
if (kindOfToken === 'empty-token') {
|
switch (this.elements.positionalAttrSelection.value) {
|
||||||
tokenQueryPrettyText += 'empty token';
|
case 'word':
|
||||||
|
if (this.elements.wordInput.value === '') {
|
||||||
|
this.disableTokenSubmit();
|
||||||
} else {
|
} else {
|
||||||
let c = this.elements.ignoreCaseCheckbox.checked ? ' %c' : '';
|
tokenQueryPrettyText += `word=${this.elements.wordInput.value}${c}`;
|
||||||
input = this.tokenInputCheck(this.elements.tokenBuilderContent);
|
tokenQueryCQLText += `word="${this.elements.wordInput.value}"${c}`;
|
||||||
tokenQueryPrettyText += `${kindOfToken}=${input.value}${c}`;
|
this.elements.wordInput.value = '';
|
||||||
tokenQueryCQLText += `${kindOfToken}="${input.value}"${c}`;
|
}
|
||||||
|
break;
|
||||||
|
case 'lemma':
|
||||||
|
if (this.elements.lemmaInput.value === '') {
|
||||||
|
this.disableTokenSubmit();
|
||||||
|
} else {
|
||||||
|
tokenQueryPrettyText += `lemma=${this.elements.lemmaInput.value}${c}`;
|
||||||
|
tokenQueryCQLText += `lemma="${this.elements.lemmaInput.value}"${c}`;
|
||||||
|
this.elements.lemmaInput.value = '';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'english-pos':
|
||||||
|
if (this.elements.englishPosSelection.value === 'default') {
|
||||||
|
this.disableTokenSubmit();
|
||||||
|
} else {
|
||||||
|
tokenQueryPrettyText += `pos=${this.elements.englishPosSelection.value}`;
|
||||||
|
tokenQueryCQLText += `pos="${this.elements.englishPosSelection.value}"`;
|
||||||
|
this.elements.englishPosSelection.value = '';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'german-pos':
|
||||||
|
if (this.elements.germanPosSelection.value === 'default') {
|
||||||
|
this.disableTokenSubmit();
|
||||||
|
} else {
|
||||||
|
tokenQueryPrettyText += `pos=${this.elements.germanPosSelection.value}`;
|
||||||
|
tokenQueryCQLText += `pos="${this.elements.germanPosSelection.value}"`;
|
||||||
|
this.elements.germanPosSelection.value = '';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'simple-pos':
|
||||||
|
if (this.elements.simplePosSelection.value === 'default') {
|
||||||
|
this.disableTokenSubmit();
|
||||||
|
} else {
|
||||||
|
tokenQueryPrettyText += `simple_pos=${this.elements.simplePosSelection.value}`;
|
||||||
|
tokenQueryCQLText += `simple_pos="${this.elements.simplePosSelection.value}"`;
|
||||||
|
this.elements.simplePosSelection.value = '';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'empty-token':
|
||||||
|
tokenQueryPrettyText += 'empty token';
|
||||||
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// isTokenQueryInvalid looks if a valid value is passed. If the input fields/dropdowns are empty (isTokenQueryInvalid === true), no token is added.
|
// isTokenQueryInvalid looks if a valid value is passed. If the input fields/dropdowns are empty (isTokenQueryInvalid === true), no token is added.
|
||||||
if ((input.value === '' || input.value === 'default') && this.elements.positionalAttrSelection.value !== 'empty-token') {
|
if (this.elements.isTokenQueryInvalid === false) {
|
||||||
this.disableTokenSubmit();
|
tokenQueryCQLText = '[' + tokenQueryCQLText + ']';
|
||||||
} else {
|
|
||||||
tokenQueryCQLText = `[${tokenQueryCQLText}]`;
|
|
||||||
this.submitQueryChipElement('token', tokenQueryPrettyText, tokenQueryCQLText, null, false, true);
|
this.submitQueryChipElement('token', tokenQueryPrettyText, tokenQueryCQLText, null, false, true);
|
||||||
this.elements.positionalAttrModal.close();
|
this.elements.positionalAttrModal.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
kindOfTokenCheck(kindOfToken) {
|
|
||||||
return kindOfToken === 'english-pos' || kindOfToken === 'german-pos' ? 'pos' : kindOfToken;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
actionButtonInOptionSectionHandler(elem) {
|
actionButtonInOptionSectionHandler(elem) {
|
||||||
let input = this.tokenInputCheck(this.elements.tokenBuilderContent);
|
let input = this.tokenInputCheck();
|
||||||
switch (elem) {
|
switch (elem) {
|
||||||
case 'option-group':
|
case 'option-group':
|
||||||
input.value += '(option1|option2)';
|
input.value += '(option1|option2)';
|
||||||
@ -147,10 +201,10 @@ class TokenAttributeBuilderFunctionsQueryBuilder extends GeneralFunctionsQueryBu
|
|||||||
input.value += '.';
|
input.value += '.';
|
||||||
break;
|
break;
|
||||||
case 'and':
|
case 'and':
|
||||||
this.conditionHandler('and');
|
this.conditionHandler('and', " & ");
|
||||||
break;
|
break;
|
||||||
case 'or':
|
case 'or':
|
||||||
this.conditionHandler('or');
|
this.conditionHandler('or', " | ");
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
@ -159,8 +213,32 @@ class TokenAttributeBuilderFunctionsQueryBuilder extends GeneralFunctionsQueryBu
|
|||||||
}
|
}
|
||||||
|
|
||||||
characterIncidenceModifierHandler(elem) {
|
characterIncidenceModifierHandler(elem) {
|
||||||
let input = this.tokenInputCheck(this.elements.tokenBuilderContent);
|
// For word and lemma, the incidence modifiers are inserted in the input field. For the others, one or two chips are created which contain the respective value of the token and the incidence modifier.
|
||||||
|
switch (this.elements.positionalAttrSelection.value) {
|
||||||
|
case 'empty-token':
|
||||||
|
this.tokenChipFactory(elem.innerText, elem.dataset.token);
|
||||||
|
break;
|
||||||
|
case 'english-pos':
|
||||||
|
this.tokenChipFactory(`pos=${this.elements.englishPosSelection.value}`, `pos="${this.elements.englishPosSelection.value}"`);
|
||||||
|
this.tokenChipFactory(elem.innerText, elem.dataset.token);
|
||||||
|
break;
|
||||||
|
case 'german-pos':
|
||||||
|
this.tokenChipFactory(`pos=${this.elements.germanPosSelection.value}`, `pos="${this.elements.germanPosSelection.value}"`);
|
||||||
|
this.tokenChipFactory(elem.innerText, elem.dataset.token);
|
||||||
|
break;
|
||||||
|
case 'simple-pos':
|
||||||
|
this.tokenChipFactory(`simple_pos=${this.elements.simplePosSelection.value}`, `simple_pos="${this.elements.simplePosSelection.value}"`);
|
||||||
|
this.tokenChipFactory(elem.innerText, elem.dataset.token);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
let input = this.tokenInputCheck();
|
||||||
input.value += elem.dataset.token;
|
input.value += elem.dataset.token;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.elements.positionalAttrSelection.value !== "word" && this.elements.positionalAttrSelection.value !== "lemma") {
|
||||||
|
this.toggleClass([this.elements.positionalAttrSelection.value], "hide", "add");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
characterNMSubmitHandler(modalId) {
|
characterNMSubmitHandler(modalId) {
|
||||||
@ -172,96 +250,89 @@ class TokenAttributeBuilderFunctionsQueryBuilder extends GeneralFunctionsQueryBu
|
|||||||
|
|
||||||
let instance = M.Modal.getInstance(modal);
|
let instance = M.Modal.getInstance(modal);
|
||||||
instance.close();
|
instance.close();
|
||||||
let tokenInput = this.tokenInputCheck(this.elements.tokenBuilderContent);
|
|
||||||
tokenInput.value += '{' + input + '}';
|
switch (this.elements.positionalAttrSelection.value) {
|
||||||
|
case 'word':
|
||||||
|
this.elements.wordInput.value += '{' + input + '}';
|
||||||
|
break;
|
||||||
|
case 'lemma':
|
||||||
|
this.elements.lemmaInput.value += '{' + input + '}';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
conditionHandler(conditionText) {
|
conditionHandler(conditionText, conditionQueryContent) {
|
||||||
let tokenQueryTemplateClone = this.elements.tokenQueryTemplate.content.cloneNode(true);
|
let tokenQueryPrettyText;
|
||||||
tokenQueryTemplateClone.querySelector('.token-query-template-content').appendChild(this.elements.tokenBuilderContent.firstElementChild);
|
let tokenQueryCQLText;
|
||||||
let notSelectedButton = tokenQueryTemplateClone.querySelector(`[data-condition-pretty-text]:not([data-condition-pretty-text="${conditionText}"])`);
|
let c = this.elements.ignoreCaseCheckbox.checked ? ' %c' : '';
|
||||||
let deleteButton = tokenQueryTemplateClone.querySelector(`[data-token-query-content-action="delete"]`);
|
|
||||||
deleteButton.addEventListener('click', (event) => {
|
switch (this.elements.positionalAttrSelection.value) {
|
||||||
this.deleteTokenQueryRow(event.target);
|
case 'word':
|
||||||
});
|
tokenQueryPrettyText = `word=${this.elements.wordInput.value}${c}`;
|
||||||
notSelectedButton.parentNode.removeChild(notSelectedButton);
|
tokenQueryCQLText = `word="${this.elements.wordInput.value}"${c}`;
|
||||||
this.elements.tokenQuery.appendChild(tokenQueryTemplateClone);
|
this.elements.wordInput.value = '';
|
||||||
|
break;
|
||||||
|
case 'lemma':
|
||||||
|
tokenQueryPrettyText = `lemma=${this.elements.lemmaInput.value}${c}`;
|
||||||
|
tokenQueryCQLText = `lemma="${this.elements.lemmaInput.value}"${c}`;
|
||||||
|
this.elements.lemmaInput.value = '';
|
||||||
|
break;
|
||||||
|
case 'english-pos':
|
||||||
|
tokenQueryPrettyText = `pos=${this.elements.englishPosSelection.value}`;
|
||||||
|
tokenQueryCQLText = `pos="${this.elements.englishPosSelection.value}"`;
|
||||||
|
this.elements.englishPosSelection.value = '';
|
||||||
|
break;
|
||||||
|
case 'german-pos':
|
||||||
|
tokenQueryPrettyText = `pos=${this.elements.germanPosSelection.value}`;
|
||||||
|
tokenQueryCQLText = `pos="${this.elements.germanPosSelection.value}"`;
|
||||||
|
this.elements.germanPosSelection.value = '';
|
||||||
|
break;
|
||||||
|
case 'simple-pos':
|
||||||
|
tokenQueryPrettyText = `simple_pos=${this.elements.simplePosSelection.value}`;
|
||||||
|
tokenQueryCQLText = `simple_pos="${this.elements.simplePosSelection.value}"`;
|
||||||
|
this.elements.simplePosSelection.value = '';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
// Deleting the options which do not make sense in the context of the condition like "word" AND "word". Also sets selection default.
|
// Deleting the options which do not make sense in the context of the condition like "word" AND "word". Also sets selection default.
|
||||||
let selectionDefault = "word";
|
let selectionDefault = "word";
|
||||||
let optionDeleteList = ['empty-token'];
|
let optionDeleteList = ['empty-token'];
|
||||||
if (conditionText === 'and') {
|
if (conditionText === 'and') {
|
||||||
switch (this.elements.positionalAttrSelection.value) {
|
if (this.elements.positionalAttrSelection.value === 'word' || this.elements.positionalAttrSelection.value === 'lemma') {
|
||||||
case 'english-pos' || 'german-pos':
|
selectionDefault = "english-pos";
|
||||||
|
optionDeleteList.push('word', 'lemma');
|
||||||
|
} else if (this.elements.positionalAttrSelection.value === 'english-pos' || this.elements.positionalAttrSelection.value === 'german-pos') {
|
||||||
optionDeleteList.push('english-pos', 'german-pos');
|
optionDeleteList.push('english-pos', 'german-pos');
|
||||||
break;
|
|
||||||
default:
|
|
||||||
optionDeleteList.push(this.elements.positionalAttrSelection.value);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
let originalSelectionList =
|
optionDeleteList.push('simple-pos');
|
||||||
`
|
|
||||||
<option value="word" selected>word</option>
|
|
||||||
<option value="lemma" >lemma</option>
|
|
||||||
<option value="english-pos">english pos</option>
|
|
||||||
<option value="german-pos">german pos</option>
|
|
||||||
<option value="simple_pos">simple_pos</option>
|
|
||||||
`;
|
|
||||||
this.elements.positionalAttrSelection.innerHTML = originalSelectionList;
|
|
||||||
M.FormSelect.init(this.elements.positionalAttrSelection);
|
|
||||||
}
|
}
|
||||||
let lastTokenQueryRow = this.elements.tokenQuery.lastElementChild;
|
|
||||||
if(lastTokenQueryRow.querySelector('[data-kind-of-token="word"]') || lastTokenQueryRow.querySelector('[data-kind-of-token="lemma"]')) {
|
|
||||||
this.appendIgnoreCaseCheckbox(lastTokenQueryRow.querySelector('.token-query-template-content'), this.elements.ignoreCaseCheckbox.checked);
|
|
||||||
}
|
}
|
||||||
this.elements.ignoreCaseCheckbox.checked = false;
|
|
||||||
|
this.resetMaterializeSelection([this.elements.englishPosSelection, this.elements.germanPosSelection, this.elements.simplePosSelection]);
|
||||||
|
|
||||||
|
this.tokenChipFactory(tokenQueryPrettyText, tokenQueryCQLText);
|
||||||
|
this.tokenChipFactory(conditionText, conditionQueryContent);
|
||||||
this.setTokenSelection(selectionDefault, optionDeleteList);
|
this.setTokenSelection(selectionDefault, optionDeleteList);
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteTokenQueryRow(deleteButton) {
|
|
||||||
let deletedRow = deleteButton.closest('.row');
|
|
||||||
let condition = deletedRow.querySelector('[data-condition-pretty-text]').dataset.conditionPrettyText;
|
|
||||||
if (condition === 'and') {
|
|
||||||
let kindOfToken = deletedRow.querySelector('[data-kind-of-token]').dataset.kindOfToken;
|
|
||||||
switch (kindOfToken) {
|
|
||||||
case 'english-pos' || 'german-pos':
|
|
||||||
this.createOptionElementForPosAttrSelection('english-pos');
|
|
||||||
this.createOptionElementForPosAttrSelection('german-pos');
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
this.createOptionElementForPosAttrSelection(kindOfToken);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
M.FormSelect.init(this.elements.positionalAttrSelection);
|
|
||||||
}
|
|
||||||
deletedRow.remove();
|
|
||||||
}
|
|
||||||
|
|
||||||
createOptionElementForPosAttrSelection(kindOfToken) {
|
|
||||||
let option = document.createElement('option');
|
|
||||||
option.value = kindOfToken;
|
|
||||||
option.text = kindOfToken;
|
|
||||||
this.elements.positionalAttrSelection.appendChild(option);
|
|
||||||
}
|
|
||||||
|
|
||||||
appendIgnoreCaseCheckbox(parentElement, checked = false) {
|
|
||||||
let ignoreCaseCheckboxClone = document.querySelector('#ignore-case-checkbox-template').content.cloneNode(true);
|
|
||||||
parentElement.appendChild(ignoreCaseCheckboxClone);
|
|
||||||
M.Tooltip.init(parentElement.querySelectorAll('.tooltipped'));
|
|
||||||
if (checked) {
|
|
||||||
parentElement.querySelector('input[type="checkbox"]').checked = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
setTokenSelection(selection, optionDeleteList) {
|
setTokenSelection(selection, optionDeleteList) {
|
||||||
optionDeleteList.forEach(option => {
|
optionDeleteList.forEach(option => {
|
||||||
if (this.elements.positionalAttrSelection.querySelector(`option[value=${option}]`) !== null) {
|
|
||||||
this.elements.positionalAttrSelection.querySelector(`option[value=${option}]`).remove();
|
this.elements.positionalAttrSelection.querySelector(`option[value=${option}]`).remove();
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
this.resetMaterializeSelection([this.elements.positionalAttrSelection], selection);
|
this.resetMaterializeSelection([this.elements.positionalAttrSelection], selection);
|
||||||
this.preparePositionalAttrModal();
|
|
||||||
|
this.toggleClass(['word', 'lemma', 'english-pos', 'german-pos', 'simple-pos'], 'hide', 'add');
|
||||||
|
this.toggleClass([selection], 'hide', 'remove');
|
||||||
|
this.toggleClass(['incidence-modifiers', 'or', 'and'], 'disabled', 'add');
|
||||||
|
if (selection === "word" || selection === "lemma") {
|
||||||
|
this.toggleClass(['input-field-options'], 'hide', 'remove');
|
||||||
|
} else {
|
||||||
|
this.toggleClass(['input-field-options'], 'hide', 'add');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -17,10 +17,10 @@
|
|||||||
<div class="card">
|
<div class="card">
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col s2">
|
<div class="col s1">
|
||||||
<span class="card-title">Query <i class="material-icons left" style="font-size: inherit;">search</i></span>
|
<span class="card-title">Query <i class="material-icons left" style="font-size: inherit;">search</i></span>
|
||||||
</div>
|
</div>
|
||||||
<div class="col s1">
|
<div class="col s2">
|
||||||
<div class="switch" style="margin-top:8px; margin-left:0px;">
|
<div class="switch" style="margin-top:8px; margin-left:0px;">
|
||||||
<label>
|
<label>
|
||||||
<input type="checkbox" id="corpus-analysis-concordance-expert-mode-switch">
|
<input type="checkbox" id="corpus-analysis-concordance-expert-mode-switch">
|
||||||
|
@ -162,52 +162,35 @@
|
|||||||
<option value="lemma" >lemma</option>
|
<option value="lemma" >lemma</option>
|
||||||
<option value="english-pos">english pos</option>
|
<option value="english-pos">english pos</option>
|
||||||
<option value="german-pos">german pos</option>
|
<option value="german-pos">german pos</option>
|
||||||
<option value="simple_pos">simple_pos</option>
|
<option value="simple-pos">simple_pos</option>
|
||||||
<option value="empty-token">empty token</option>
|
<option value="empty-token">empty token</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<p></p>
|
<p></p>
|
||||||
|
<div id="corpus-analysis-concordance-token-builder-content">
|
||||||
|
<div class="row" >
|
||||||
<div id="corpus-analysis-concordance-token-query"></div>
|
<div id="corpus-analysis-concordance-token-query"></div>
|
||||||
<template id="corpus-analysis-concordance-token-query-template">
|
|
||||||
<div class="row">
|
|
||||||
<div class="token-query-template-content"></div>
|
|
||||||
<div class="col s4" style="margin-top:15px;">
|
|
||||||
<a class="btn-small waves-effect waves-light disabled" data-condition-pretty-text="or" data-condition-cql-text=" | ">or</a>
|
|
||||||
<a class="btn-small waves-effect waves-light disabled" data-condition-pretty-text="and" data-condition-cql-text=" & ">and</a>
|
|
||||||
<a class="btn-floating waves-effect waves-light red" data-token-query-content-action="delete" style="margin-left:8px;"><i class="material-icons right">delete</i></a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
<div class="row">
|
|
||||||
<div id="corpus-analysis-concordance-token-builder-content"></div>
|
|
||||||
|
|
||||||
<template id="ignore-case-checkbox-template">
|
<div id="corpus-analysis-concordance-word-builder" data-toggle-area="word">
|
||||||
<span class="col s1 center-align tooltipped" style="margin-top: 22px;" data-position="bottom" data-tooltip="Ignore Case">
|
<div class= "input-field col s3 l4">
|
||||||
<label>
|
|
||||||
<input type="checkbox" class="filled-in"/>
|
|
||||||
<span>%c</span>
|
|
||||||
</label>
|
|
||||||
</span>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<template class="token-builder-section" data-token-builder-section="word">
|
|
||||||
<div class= "input-field col s4" data-kind-of-token="word">
|
|
||||||
<i class="material-icons prefix">mode_edit</i>
|
<i class="material-icons prefix">mode_edit</i>
|
||||||
<input placeholder="Type in your word" type="text">
|
<input placeholder="Type in your word" type="text" id="corpus-analysis-concordance-word-input">
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
|
||||||
|
|
||||||
<template class="token-builder-section" data-token-builder-section="lemma">
|
<div id="corpus-analysis-concordance-lemma-builder" class="hide" data-toggle-area="lemma">
|
||||||
<div class= "input-field col s4" data-kind-of-token="lemma">
|
<div class= "input-field col s3 l4">
|
||||||
<i class="material-icons prefix">mode_edit</i>
|
<i class="material-icons prefix">mode_edit</i>
|
||||||
<input placeholder="Type in your lemma" type="text">
|
<input placeholder="Type in your lemma" type="text" id="corpus-analysis-concordance-lemma-input">
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
|
||||||
|
|
||||||
<template class="token-builder-section" data-token-builder-section="english-pos">
|
<div id="corpus-analysis-concordance-english-pos-builder" class="hide" data-toggle-area="english-pos">
|
||||||
<div class= "input-field col s4" data-kind-of-token="english-pos">
|
<div class="col s6 m4 l4">
|
||||||
<select name="englishpos">
|
<div class="row">
|
||||||
|
<div class= "input-field col s12">
|
||||||
|
<select name="englishpos" id="corpus-analysis-concordance-english-pos-selection">
|
||||||
<option value="default" disabled selected>English pos tagset</option>
|
<option value="default" disabled selected>English pos tagset</option>
|
||||||
<option value="ADD">email</option>
|
<option value="ADD">email</option>
|
||||||
<option value="AFX">affix</option>
|
<option value="AFX">affix</option>
|
||||||
@ -259,11 +242,15 @@
|
|||||||
</select>
|
</select>
|
||||||
<label>Part-of-speech tags</label>
|
<label>Part-of-speech tags</label>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<template class="token-builder-section" data-token-builder-section="german-pos">
|
<div id="corpus-analysis-concordance-german-pos-builder" class="hide" data-toggle-area="german-pos">
|
||||||
<div class= "input-field col s4" data-kind-of-token="german-pos">
|
<div class="col s6 m4 l4">
|
||||||
<select name="germanpos">
|
<div class="row">
|
||||||
|
<div class= "input-field col s12">
|
||||||
|
<select name="germanpos" id="corpus-analysis-concordance-german-pos-selection">
|
||||||
<option value="default" disabled selected>German pos tagset</option>
|
<option value="default" disabled selected>German pos tagset</option>
|
||||||
<option value="ADJA">adjective, attributive</option>
|
<option value="ADJA">adjective, attributive</option>
|
||||||
<option value="ADJD">adjective, adverbial or predicative</option>
|
<option value="ADJD">adjective, adverbial or predicative</option>
|
||||||
@ -322,11 +309,15 @@
|
|||||||
</select>
|
</select>
|
||||||
<label>Part-of-speech tags</label>
|
<label>Part-of-speech tags</label>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<template class="token-builder-section" data-token-builder-section="simple_pos">
|
<div id="corpus-analysis-concordance-simple-pos-builder" class="hide" data-toggle-area="simple-pos">
|
||||||
<div class= "input-field col s4" data-kind-of-token="simple_pos">
|
<div class="col s6 m4 l4">
|
||||||
<select name="simplepos">
|
<div class="row">
|
||||||
|
<div class= "input-field col s12">
|
||||||
|
<select name="simplepos" id="corpus-analysis-concordance-simple-pos-selection">
|
||||||
<option value="default" disabled selected>simple_pos tagset</option>
|
<option value="default" disabled selected>simple_pos tagset</option>
|
||||||
<option value="ADJ">adjective</option>
|
<option value="ADJ">adjective</option>
|
||||||
<option value="ADP">adposition</option>
|
<option value="ADP">adposition</option>
|
||||||
@ -348,36 +339,40 @@
|
|||||||
</select>
|
</select>
|
||||||
<label>Simple part-of-speech tags</label>
|
<label>Simple part-of-speech tags</label>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="col s4" data-toggle-area="condition-option-container">
|
<div class="col s1 l1 center-align">
|
||||||
<a class="btn-small tooltipped waves-effect waves-light disabled positional-attr-options-action-button" data-options-action="or" data-toggle-area="or" data-position="bottom" data-tooltip="You can add another condition to your token. <br>At least one must be fulfilled">or</a>
|
<p class="btn-floating waves-effect waves-light" id="corpus-analysis-concordance-token-submit">
|
||||||
<a class="btn-small tooltipped waves-effect waves-light disabled positional-attr-options-action-button" data-options-action="and" data-toggle-area="and" data-position="bottom" data-tooltip="You can add another condition to your token. <br>Both must be fulfilled">and</a>
|
|
||||||
<p class="btn-floating waves-effect waves-light" id="corpus-analysis-concordance-token-submit" style="margin-left:10px;">
|
|
||||||
<i class="material-icons right">send</i>
|
<i class="material-icons right">send</i>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="hide" id="corpus-analysis-concordance-no-value-message"><i>No value entered!</i></div>
|
<div class="hide" id="corpus-analysis-concordance-no-value-message"><i>No value entered!</i></div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div id="corpus-analysis-concordance-token-edit-options" data-toggle-area="input-field-options">
|
|
||||||
|
<div id="corpus-analysis-concordance-token-edit-options">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<h6>Options to edit your token: <a class="modal-trigger" data-manual-modal-chapter="manual-modal-query-builder" href="#manual-modal"><i class="material-icons left" id="corpus-analysis-concordance-edit-options-tutorial-info-icon">help_outline</i></a></h6>
|
<h6>Options to edit your token: <a class="modal-trigger" data-manual-modal-chapter="manual-modal-query-builder" href="#manual-modal"><i class="material-icons left" id="corpus-analysis-concordance-edit-options-tutorial-info-icon">help_outline</i></a></h6>
|
||||||
</div>
|
</div>
|
||||||
<p></p>
|
<p></p>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col s8" >
|
<div class="col s8" data-toggle-area="input-field-options">
|
||||||
<a class="btn-small waves-effect waves-light tooltipped positional-attr-options-action-button" data-options-action="wildcard-char" data-position="top" data-tooltip="Look for a variable character (also called wildcard character)">Wildcard character</a>
|
<a class="btn-small waves-effect waves-light tooltipped positional-attr-options-action-button" data-options-action="wildcard-char" data-position="top" data-tooltip="Look for a variable character (also called wildcard character)">Wildcard character</a>
|
||||||
<a class="btn-small waves-effect waves-light tooltipped positional-attr-options-action-button" data-options-action="option-group" data-position="top" data-tooltip="Find character sequences from a list of options">Option Group</a>
|
<a class="btn-small waves-effect waves-light tooltipped positional-attr-options-action-button" data-options-action="option-group" data-position="top" data-tooltip="Find character sequences from a list of options">Option Group</a>
|
||||||
<a class="dropdown-trigger btn-small waves-effect waves-light disabled" href="#" data-target="corpus-analysis-concordance-character-incidence-modifiers-dropdown" data-toggle-area="incidence-modifiers" data-position="top" data-tooltip="Incidence Modifiers are special characters or patterns, <br>which determine how often a character represented previously should occur.">incidence modifiers</a>
|
<a class="dropdown-trigger btn-small waves-effect waves-light disabled" href="#" data-target="corpus-analysis-concordance-character-incidence-modifiers-dropdown" data-toggle-area="incidence-modifiers" data-position="top" data-tooltip="Incidence Modifiers are special characters or patterns, <br>which determine how often a character represented previously should occur.">incidence modifiers</a>
|
||||||
<span>
|
<span data-toggle-area="ignore-case-checkbox">
|
||||||
<label>
|
<label>
|
||||||
<input type="checkbox" class="filled-in" id="corpus-analysis-concordance-ignore-case-checkbox"/>
|
<input type="checkbox" class="filled-in" id="corpus-analysis-concordance-ignore-case-checkbox"/>
|
||||||
<span>Ignore Case</span>
|
<span>Ignore Case</span>
|
||||||
</label>
|
</label>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="col s2" data-toggle-area="condition-option-container">
|
||||||
|
<a class="btn-small tooltipped waves-effect waves-light disabled positional-attr-options-action-button" data-options-action="or" data-toggle-area="or" data-position="bottom" data-tooltip="You can add another condition to your token. <br>At least one must be fulfilled">or</a>
|
||||||
|
<a class="btn-small tooltipped waves-effect waves-light disabled positional-attr-options-action-button" data-options-action="and" data-toggle-area="and" data-position="bottom" data-tooltip="You can add another condition to your token. <br>Both must be fulfilled">and</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
<ul id="corpus-analysis-concordance-character-incidence-modifiers-dropdown" class="dropdown-content">
|
<ul id="corpus-analysis-concordance-character-incidence-modifiers-dropdown" class="dropdown-content">
|
||||||
{{ incidence_modifiers_dropdown_content("character") }}
|
{{ incidence_modifiers_dropdown_content("character") }}
|
||||||
@ -385,6 +380,7 @@
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
{{ exactly_n_modal_content("character") }}
|
{{ exactly_n_modal_content("character") }}
|
||||||
{{ exactly_nm_modal_content("character") }}
|
{{ exactly_nm_modal_content("character") }}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user