226 lines
7.7 KiB
TypeScript
226 lines
7.7 KiB
TypeScript
/*
|
|
* Copyright (C) 2026 Fluxer Contributors
|
|
*
|
|
* This file is part of Fluxer.
|
|
*
|
|
* Fluxer is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Fluxer is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with Fluxer. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import {beforeEach, describe, expect, it} from 'vitest';
|
|
import {TextareaSegmentManager} from './TextareaSegmentManager';
|
|
|
|
describe('Textarea Autocomplete Flow Integration', () => {
|
|
let manager: TextareaSegmentManager;
|
|
|
|
beforeEach(() => {
|
|
manager = new TextareaSegmentManager();
|
|
});
|
|
|
|
function simulateAutocompleteSelect(
|
|
currentValue: string,
|
|
matchStart: number,
|
|
matchEnd: number,
|
|
displayText: string,
|
|
actualText: string,
|
|
segmentType: 'user' | 'role' | 'channel' | 'emoji' | 'special',
|
|
segmentId: string,
|
|
capturedWhitespace = '',
|
|
): string {
|
|
const hasLeadingSpace = capturedWhitespace.length > 0;
|
|
const beforeMatch = currentValue.slice(0, matchStart + capturedWhitespace.length);
|
|
const afterMatch = currentValue.slice(matchEnd);
|
|
|
|
const guildBeforeMatch = hasLeadingSpace || beforeMatch.endsWith(' ') || beforeMatch.length === 0 ? '' : ' ';
|
|
const insertPosition = beforeMatch.length + guildBeforeMatch.length;
|
|
|
|
const changeStart = beforeMatch.length;
|
|
const changeEnd = matchEnd;
|
|
manager.updateSegmentsForTextChange(changeStart, changeEnd, guildBeforeMatch.length);
|
|
|
|
const tempText = `${beforeMatch}${guildBeforeMatch}`;
|
|
const {newText: updatedText} = manager.insertSegment(
|
|
tempText,
|
|
insertPosition,
|
|
`${displayText} `,
|
|
`${actualText} `,
|
|
segmentType,
|
|
segmentId,
|
|
);
|
|
|
|
const finalText = updatedText + afterMatch;
|
|
const trimmed = finalText.trimStart();
|
|
|
|
return trimmed;
|
|
}
|
|
|
|
it('should handle first mention autocomplete', () => {
|
|
const value = '@Hampus';
|
|
const matchStart = 0;
|
|
const matchEnd = 7;
|
|
|
|
const result = simulateAutocompleteSelect(value, matchStart, matchEnd, '@Hampus#0001', '<@123>', 'user', '123');
|
|
|
|
expect(result).toBe('@Hampus#0001 ');
|
|
expect(manager.getSegments()).toHaveLength(1);
|
|
expect(manager.displayToActual(result)).toBe('<@123> ');
|
|
});
|
|
|
|
it('should handle second consecutive mention - the failing case', () => {
|
|
let value = '@Hampus';
|
|
let result = simulateAutocompleteSelect(value, 0, 7, '@Hampus#0001', '<@123>', 'user', '123');
|
|
expect(result).toBe('@Hampus#0001 ');
|
|
expect(manager.getSegments()).toHaveLength(1);
|
|
|
|
value = `${result}@Hampus`;
|
|
const matchStart = result.length;
|
|
const matchEnd = value.length;
|
|
|
|
result = simulateAutocompleteSelect(value, matchStart, matchEnd, '@Hampus#0001', '<@123>', 'user', '123');
|
|
|
|
expect(result).toBe('@Hampus#0001 @Hampus#0001 ');
|
|
const segments = manager.getSegments();
|
|
expect(segments).toHaveLength(2);
|
|
expect(manager.displayToActual(result)).toBe('<@123> <@123> ');
|
|
});
|
|
|
|
it('should handle three consecutive mentions', () => {
|
|
let value = '@Hampus';
|
|
let prevLength = 0;
|
|
let result = simulateAutocompleteSelect(value, 0, 7, '@Hampus#0001', '<@123>', 'user', '123');
|
|
|
|
prevLength = result.length;
|
|
value = `${result}@Hampus`;
|
|
result = simulateAutocompleteSelect(value, prevLength, value.length, '@Hampus#0001', '<@123>', 'user', '123');
|
|
|
|
prevLength = result.length;
|
|
value = `${result}@Hampus`;
|
|
result = simulateAutocompleteSelect(value, prevLength, value.length, '@Hampus#0001', '<@123>', 'user', '123');
|
|
|
|
expect(result).toBe('@Hampus#0001 @Hampus#0001 @Hampus#0001 ');
|
|
const segments = manager.getSegments();
|
|
expect(segments).toHaveLength(3);
|
|
expect(manager.displayToActual(result)).toBe('<@123> <@123> <@123> ');
|
|
});
|
|
|
|
it('should handle mention with text before it', () => {
|
|
const value = 'Hey @User';
|
|
const matchStart = 4;
|
|
const matchEnd = 9;
|
|
|
|
const result = simulateAutocompleteSelect(value, matchStart, matchEnd, '@User#0001', '<@123>', 'user', '123');
|
|
|
|
expect(result).toBe('Hey @User#0001 ');
|
|
expect(manager.getSegments()).toHaveLength(1);
|
|
expect(manager.displayToActual(result)).toBe('Hey <@123> ');
|
|
});
|
|
|
|
it('should handle multiple mentions with text between them', () => {
|
|
let value = 'Hey @User1';
|
|
let result = simulateAutocompleteSelect(value, 4, 10, '@User1#0001', '<@123>', 'user', '123');
|
|
|
|
value = `${result}and @User2`;
|
|
result = simulateAutocompleteSelect(value, result.length + 4, value.length, '@User2#0002', '<@456>', 'user', '456');
|
|
|
|
expect(result).toBe('Hey @User1#0001 and @User2#0002 ');
|
|
const segments = manager.getSegments();
|
|
expect(segments).toHaveLength(2);
|
|
expect(manager.displayToActual(result)).toBe('Hey <@123> and <@456> ');
|
|
});
|
|
|
|
it('should handle consecutive mentions', () => {
|
|
let value = '@H';
|
|
let result = simulateAutocompleteSelect(value, 0, 2, '@Hampus#0001', '<@123>', 'user', '123');
|
|
|
|
const prevLength = result.length;
|
|
value = `${result}@H`;
|
|
|
|
result = simulateAutocompleteSelect(value, prevLength, value.length, '@Hampus#0001', '<@123>', 'user', '123');
|
|
|
|
expect(result).toBe('@Hampus#0001 @Hampus#0001 ');
|
|
const segments = manager.getSegments();
|
|
expect(segments).toHaveLength(2);
|
|
expect(segments[0]).toMatchObject({id: '123', start: 0, end: 13});
|
|
expect(segments[1]).toMatchObject({id: '123', start: 13, end: 26});
|
|
expect(manager.displayToActual(result)).toBe('<@123> <@123> ');
|
|
});
|
|
|
|
it('should handle typing between autocompletes with handleTextChange', () => {
|
|
let previousValue = '';
|
|
let value = '@H';
|
|
|
|
value = simulateAutocompleteSelect(value, 0, 2, '@Hampus#0001', '<@123>', 'user', '123');
|
|
previousValue = value;
|
|
|
|
expect(value).toBe('@Hampus#0001 ');
|
|
expect(manager.getSegments()).toHaveLength(1);
|
|
|
|
const newValue = `${value}@`;
|
|
const change = TextareaSegmentManager.detectChange(previousValue, newValue);
|
|
manager.updateSegmentsForTextChange(change.changeStart, change.changeEnd, change.replacementLength);
|
|
value = newValue;
|
|
previousValue = value;
|
|
|
|
expect(manager.getSegments()).toHaveLength(1);
|
|
expect(manager.getSegments()[0]).toMatchObject({id: '123', start: 0, end: 13});
|
|
|
|
const newValue2 = `${value}H`;
|
|
const change2 = TextareaSegmentManager.detectChange(previousValue, newValue2);
|
|
manager.updateSegmentsForTextChange(change2.changeStart, change2.changeEnd, change2.replacementLength);
|
|
value = newValue2;
|
|
previousValue = value;
|
|
|
|
expect(manager.getSegments()).toHaveLength(1);
|
|
|
|
const matchStart = '@Hampus#0001 '.length;
|
|
const matchEnd = value.length;
|
|
value = simulateAutocompleteSelect(value, matchStart, matchEnd, '@Hampus#0001', '<@123>', 'user', '123');
|
|
|
|
expect(value).toBe('@Hampus#0001 @Hampus#0001 ');
|
|
expect(manager.getSegments()).toHaveLength(2);
|
|
expect(manager.displayToActual(value)).toBe('<@123> <@123> ');
|
|
});
|
|
|
|
it('should reproduce and fix the bug - regex match includes space', () => {
|
|
const MENTION_REGEX = /(^|\s)@(\S*)$/;
|
|
|
|
let value = '@Hampus#0001 ';
|
|
manager.insertSegment('', 0, '@Hampus#0001 ', '<@123> ', 'user', '123');
|
|
|
|
value += '@H';
|
|
|
|
const valueUpToCursor = value;
|
|
const match = valueUpToCursor.match(MENTION_REGEX);
|
|
|
|
if (match) {
|
|
const matchStart = match.index ?? 0;
|
|
const matchEnd = matchStart + match[0].length;
|
|
const capturedWhitespace = match[1] || '';
|
|
|
|
const result = simulateAutocompleteSelect(
|
|
value,
|
|
matchStart,
|
|
matchEnd,
|
|
'@Hampus#0001',
|
|
'<@123>',
|
|
'user',
|
|
'123',
|
|
capturedWhitespace,
|
|
);
|
|
|
|
expect(manager.getSegments()).toHaveLength(2);
|
|
expect(manager.displayToActual(result)).toBe('<@123> <@123> ');
|
|
}
|
|
});
|
|
});
|