개발
Fabric.js에서 수직 정렬과 크기 조절이 가능한 TextboxVertical 구현하기
syaku
2024. 12. 19. 20:35
728x90
반응형
Fabric.js에서 수직 정렬과 크기 조절이 가능한 TextboxVertical 구현하기
Implementing TextboxVertical with Vertical Alignment and Resizing in Fabric.js
Fabric.js를 사용하다 보면 텍스트박스의 수직 정렬과 크기 조절에 제한이 있다는 것을 발견하게 됩니다. 이러한 제한을 해결하기 위해 Textbox 클래스를 확장한 TextboxVertical
클래스를 소개합니다.
When using Fabric.js, you might notice limitations in textbox vertical alignment and resizing. To address these limitations, we introduce the TextboxVertical
class that extends the Textbox class.
사용 목적 (Purpose)
- 텍스트박스 내 텍스트의 수직 정렬 (상단, 중앙, 하단)
Text vertical alignment within textbox (top, middle, bottom) - 텍스트박스의 높이와 너비를 자유롭게 조절
Freely adjustable height and width of textbox - 편집 모드에서도 정확한 커서 위치 유지
Maintain accurate cursor position even in edit mode
주요 기능 (Key Features)
수직 정렬 지원 (Vertical Alignment Support)
verticalAlign
속성을 통해 텍스트의 수직 위치를 지정
Specify vertical position of text throughverticalAlign
property- 'top', 'middle', 'bottom' 세 가지 옵션 제공
Provides three options: 'top', 'middle', 'bottom' - 텍스트박스의 높이가 변경되어도 정렬 유지
Maintains alignment even when textbox height changes
크기 조절 기능 (Resizing Capability)
- 텍스트박스의 높이와 너비를 자유롭게 조절 가능
Freely adjustable height and width of textbox - 스케일링 시 텍스트 왜곡 방지
Prevents text distortion during scaling - 크기 조절 후에도 텍스트 정렬 유지
Maintains text alignment after resizing
사용 방법 (Usage)
// TextboxVertical 인스턴스 생성
// Create TextboxVertical instance
const textbox = new TextboxVertical('Hello World', {
left: 100,
top: 100,
width: 200,
height: 150,
verticalAlign: 'middle', // 'top', 'middle', 'bottom'
textAlign: 'center'
});
// 캔버스에 추가
// Add to canvas
canvas.add(textbox);
장점 (Benefits)
- 유연한 레이아웃 (Flexible Layout)
- 텍스트의 수직 위치를 자유롭게 조정 가능
Freely adjustable vertical text position - 다양한 디자인 요구사항 충족
Meets various design requirements
- 텍스트의 수직 위치를 자유롭게 조정 가능
- 직관적인 사용성 (Intuitive Usability)
- 기존 Textbox와 동일한 방식으로 사용
Uses the same approach as the original Textbox - 추가된 verticalAlign 속성으로 쉽게 수직 정렬 제어
Easy vertical alignment control with added verticalAlign property
- 기존 Textbox와 동일한 방식으로 사용
- 안정적인 텍스트 편집 (Stable Text Editing)
- 편집 모드에서도 정확한 커서 위치 제공
Provides accurate cursor position in edit mode - 텍스트 입력 시 자연스러운 사용자 경험
Natural user experience during text input
- 편집 모드에서도 정확한 커서 위치 제공
활용 사례 (Use Cases)
- 배너 디자인 (Banner Design)
- 카드 레이아웃 (Card Layouts)
- 텍스트 중심 인포그래픽 (Text-focused Infographics)
- 동적 콘텐츠가 필요한 웹 애플리케이션 (Web Applications Requiring Dynamic Content)
- 텍스트 에디터 구현 (Text Editor Implementation)
이 커스텀 클래스를 통해 Fabric.js의 텍스트박스 기능을 한층 더 강화하고, 더 다양한 디자인과 레이아웃을 구현할 수 있습니다.
This custom class enhances Fabric.js textbox functionality and enables implementation of more diverse designs and layouts.
소스 코드
class TextboxVertical extends Textbox {
constructor(text, options) {
super(text, options);
this.verticalAlign = options.verticalAlign || 'top'; // Default to 'top'
}
initDimensions() {
if (!this.initialized) {
return;
}
this.isEditing && this.initDelayedCursor();
this._clearCache();
// clear dynamicMinWidth as it will be different after we re-wrap line
this.dynamicMinWidth = 0;
// wrap lines
this._styleMap = this._generateStyleMap(this._splitText());
// if after wrapping, the width is smaller than dynamicMinWidth, change the width and re-wrap
if (this.dynamicMinWidth > this.width) {
this._set('width', this.dynamicMinWidth);
}
if (this.textAlign.includes('justify')) {
// once text is measured we need to make space fatter to make justified text.
this.enlargeSpaces();
}
// clear cache and re-calculate height
this.height = !this.height ? this.calcTextHeight() : this.height;
}
_render(ctx) {
// 스케일 조정을 먼저 처리
if (this.scaleY !== 1) {
this.height = this.height * this.scaleY;
this.scaleY = 1;
}
if (this.scaleX !== 1) {
this.width = this.width * this.scaleX;
this.scaleX = 1;
}
// 텍스트 높이 계산
const totalTextHeight = this.calcTextHeight();
let verticalOffset = 0;
if (this.verticalAlign === 'middle') {
verticalOffset = (this.height - totalTextHeight) / 2;
} else if (this.verticalAlign === 'bottom') {
verticalOffset = this.height - totalTextHeight;
}
// 편집 모드일 때는 오프셋을 조정
if (this.isEditing) {
this.top += verticalOffset;
super._render(ctx);
this.top -= verticalOffset;
} else {
// 편집 모드가 아닐 때는 기존 방식 유지
ctx.translate(0, verticalOffset);
super._render(ctx);
ctx.translate(0, -verticalOffset);
}
}
}
728x90
반응형