1 // Copyright 2014 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
7 #include "../../include/pdfwindow/PDFWindow.h"
8 #include "../../include/pdfwindow/PWL_Wnd.h"
9 #include "../../include/pdfwindow/PWL_EditCtrl.h"
10 #include "../../include/pdfwindow/PWL_ScrollBar.h"
11 #include "../../include/pdfwindow/PWL_Utils.h"
12 #include "../../include/pdfwindow/PWL_Caret.h"
13 #include "../../include/pdfwindow/PWL_FontMap.h"
15 #define IsFloatZero(f) ((f) < 0.0001 && (f) > -0.0001)
16 #define IsFloatBigger(fa,fb) ((fa) > (fb) && !IsFloatZero((fa) - (fb)))
17 #define IsFloatSmaller(fa,fb) ((fa) < (fb) && !IsFloatZero((fa) - (fb)))
18 #define IsFloatEqual(fa,fb) IsFloatZero((fa)-(fb))
20 /* ---------------------------- CPWL_EditCtrl ------------------------------ */
22 CPWL_EditCtrl::CPWL_EditCtrl() :
27 m_nCharSet(DEFAULT_CHARSET),
30 m_pEdit = IFX_Edit::NewEdit();
31 ASSERT(m_pEdit != NULL);
34 CPWL_EditCtrl::~CPWL_EditCtrl()
36 IFX_Edit::DelEdit(m_pEdit);
39 void CPWL_EditCtrl::OnCreate(PWL_CREATEPARAM& cp)
41 cp.eCursorType = FXCT_VBEAM;
44 void CPWL_EditCtrl::OnCreated()
46 SetFontSize(GetCreationParam().fFontSize);
48 m_pEdit->SetFontMap(GetFontMap());
49 m_pEdit->SetNotify(this);
50 m_pEdit->Initialize();
53 FX_BOOL CPWL_EditCtrl::IsWndHorV()
55 CPDF_Matrix mt = GetWindowMatrix();
56 CPDF_Point point1(0,1);
57 CPDF_Point point2(1,1);
59 mt.Transform(point1.x, point1.y);
60 mt.Transform(point2.x, point2.y);
62 return point2.y == point1.y;
65 void CPWL_EditCtrl::SetCursor()
69 if (IFX_SystemHandler* pSH = GetSystemHandler())
72 pSH->SetCursor(FXCT_VBEAM);
74 pSH->SetCursor(FXCT_HBEAM);
79 void CPWL_EditCtrl::RePosChildWnd()
81 m_pEdit->SetPlateRect(GetClientRect());
84 void CPWL_EditCtrl::OnNotify(CPWL_Wnd* pWnd, FX_DWORD msg, intptr_t wParam, intptr_t lParam)
86 CPWL_Wnd::OnNotify(pWnd,msg,wParam,lParam);
90 case PNM_SETSCROLLINFO:
94 if (CPWL_Wnd * pChild = GetVScrollBar())
96 pChild->OnNotify(pWnd,PNM_SETSCROLLINFO,wParam,lParam);
101 case PNM_SETSCROLLPOS:
105 if (CPWL_Wnd * pChild = GetVScrollBar())
107 pChild->OnNotify(pWnd,PNM_SETSCROLLPOS,wParam,lParam);
112 case PNM_SCROLLWINDOW:
114 FX_FLOAT fPos = *(FX_FLOAT*)lParam;
118 m_pEdit->SetScrollPos(CPDF_Point(m_pEdit->GetScrollPos().x,fPos));
123 case PNM_SETCARETINFO:
125 if (PWL_CARET_INFO * pCaretInfo = (PWL_CARET_INFO *)wParam)
127 SetCaret(pCaretInfo->bVisible,
136 void CPWL_EditCtrl::CreateChildWnd(const PWL_CREATEPARAM & cp)
142 void CPWL_EditCtrl::CreateEditCaret(const PWL_CREATEPARAM & cp)
146 m_pEditCaret = new CPWL_Caret;
147 m_pEditCaret->SetInvalidRect(GetClientRect());
149 PWL_CREATEPARAM ecp = cp;
150 ecp.pParentWnd = this;
151 ecp.dwFlags = PWS_CHILD | PWS_NOREFRESHCLIP;
152 ecp.dwBorderWidth = 0;
153 ecp.nBorderStyle = PBS_SOLID;
154 ecp.rcRectWnd = CPDF_Rect(0,0,0,0);
156 m_pEditCaret->Create(ecp);
160 void CPWL_EditCtrl::SetFontSize(FX_FLOAT fFontSize)
162 m_pEdit->SetFontSize(fFontSize);
165 FX_FLOAT CPWL_EditCtrl::GetFontSize() const
167 return m_pEdit->GetFontSize();
170 FX_BOOL CPWL_EditCtrl::OnKeyDown(FX_WORD nChar, FX_DWORD nFlag)
172 if (m_bMouseDown) return TRUE;
174 FX_BOOL bRet = CPWL_Wnd::OnKeyDown(nChar,nFlag);
181 case FWL_VKEY_Delete:
188 case FWL_VKEY_Insert:
202 if (nChar == FWL_VKEY_Delete)
204 if (m_pEdit->IsSelected())
205 nChar = FWL_VKEY_Unknown;
210 case FWL_VKEY_Delete:
213 case FWL_VKEY_Insert:
214 if (IsSHIFTpressed(nFlag))
218 m_pEdit->OnVK_UP(IsSHIFTpressed(nFlag),FALSE);
221 m_pEdit->OnVK_DOWN(IsSHIFTpressed(nFlag),FALSE);
224 m_pEdit->OnVK_LEFT(IsSHIFTpressed(nFlag),FALSE);
227 m_pEdit->OnVK_RIGHT(IsSHIFTpressed(nFlag),FALSE);
230 m_pEdit->OnVK_HOME(IsSHIFTpressed(nFlag),IsCTRLpressed(nFlag));
233 m_pEdit->OnVK_END(IsSHIFTpressed(nFlag),IsCTRLpressed(nFlag));
235 case FWL_VKEY_Unknown:
236 if (!IsSHIFTpressed(nFlag))
248 FX_BOOL CPWL_EditCtrl::OnChar(FX_WORD nChar, FX_DWORD nFlag)
250 if (m_bMouseDown) return TRUE;
252 CPWL_Wnd::OnChar(nChar,nFlag);
264 FX_BOOL bCtrl = IsCTRLpressed(nFlag);
265 FX_BOOL bAlt = IsALTpressed(nFlag);
266 FX_BOOL bShift = IsSHIFTpressed(nFlag);
268 FX_WORD word = nChar;
298 if (IsReadOnly()) return TRUE;
300 if (m_pEdit->IsSelected() && word == FWL_VKEY_Back)
301 word = FWL_VKEY_Unknown;
310 case FWL_VKEY_Return:
313 case FWL_VKEY_Unknown:
316 if (IsINSERTpressed(nFlag))
318 InsertWord(word, GetCharSet());
325 FX_BOOL CPWL_EditCtrl::OnLButtonDown(const CPDF_Point & point, FX_DWORD nFlag)
327 CPWL_Wnd::OnLButtonDown(point,nFlag);
329 if (ClientHitTest(point))
337 m_pEdit->OnMouseDown(point,IsSHIFTpressed(nFlag),IsCTRLpressed(nFlag));
343 FX_BOOL CPWL_EditCtrl::OnLButtonUp(const CPDF_Point & point, FX_DWORD nFlag)
345 CPWL_Wnd::OnLButtonUp(point,nFlag);
349 //can receive keybord message
350 if (ClientHitTest(point) && !IsFocused())
354 m_bMouseDown = FALSE;
360 FX_BOOL CPWL_EditCtrl::OnMouseMove(const CPDF_Point & point, FX_DWORD nFlag)
362 CPWL_Wnd::OnMouseMove(point,nFlag);
365 m_pEdit->OnMouseMove(point,FALSE,FALSE);
370 CPDF_Rect CPWL_EditCtrl::GetContentRect() const
372 return m_pEdit->GetContentRect();
375 void CPWL_EditCtrl::SetEditCaret(FX_BOOL bVisible)
377 CPDF_Point ptHead(0,0),ptFoot(0,0);
381 GetCaretInfo(ptHead,ptFoot);
384 CPVT_WordPlace wpTemp = m_pEdit->GetCaretWordPlace();
385 IOnSetCaret(bVisible,ptHead,ptFoot,wpTemp);
388 void CPWL_EditCtrl::GetCaretInfo(CPDF_Point & ptHead, CPDF_Point & ptFoot) const
390 if (IFX_Edit_Iterator * pIterator = m_pEdit->GetIterator())
392 pIterator->SetAt(m_pEdit->GetCaret());
395 if (pIterator->GetWord(word))
397 ptHead.x = word.ptWord.x + word.fWidth;
398 ptHead.y = word.ptWord.y + word.fAscent;
399 ptFoot.x = word.ptWord.x + word.fWidth;
400 ptFoot.y = word.ptWord.y + word.fDescent;
402 else if (pIterator->GetLine(line))
404 ptHead.x = line.ptLine.x;
405 ptHead.y = line.ptLine.y + line.fLineAscent;
406 ptFoot.x = line.ptLine.x;
407 ptFoot.y = line.ptLine.y + line.fLineDescent;
412 void CPWL_EditCtrl::GetCaretPos(int32_t& x, int32_t& y) const
414 CPDF_Point ptHead(0,0), ptFoot(0,0);
416 GetCaretInfo(ptHead,ptFoot);
418 PWLtoWnd(ptHead, x, y);
421 void CPWL_EditCtrl::SetCaret(FX_BOOL bVisible, const CPDF_Point & ptHead, const CPDF_Point & ptFoot)
425 if (!IsFocused() || m_pEdit->IsSelected())
428 m_pEditCaret->SetCaret(bVisible, ptHead, ptFoot);
432 FX_BOOL CPWL_EditCtrl::IsModified() const
434 return m_pEdit->IsModified();
437 CFX_WideString CPWL_EditCtrl::GetText() const
439 return m_pEdit->GetText();
442 void CPWL_EditCtrl::SetSel(int32_t nStartChar,int32_t nEndChar)
444 m_pEdit->SetSel(nStartChar, nEndChar);
447 void CPWL_EditCtrl::GetSel(int32_t & nStartChar, int32_t & nEndChar ) const
449 m_pEdit->GetSel(nStartChar, nEndChar);
452 void CPWL_EditCtrl::Clear()
458 void CPWL_EditCtrl::SelectAll()
460 m_pEdit->SelectAll();
463 void CPWL_EditCtrl::Paint()
469 void CPWL_EditCtrl::EnableRefresh(FX_BOOL bRefresh)
472 m_pEdit->EnableRefresh(bRefresh);
475 int32_t CPWL_EditCtrl::GetCaret() const
478 return m_pEdit->GetCaret();
483 void CPWL_EditCtrl::SetCaret(int32_t nPos)
486 m_pEdit->SetCaret(nPos);
489 int32_t CPWL_EditCtrl::GetTotalWords() const
492 return m_pEdit->GetTotalWords();
497 void CPWL_EditCtrl::SetScrollPos(const CPDF_Point& point)
500 m_pEdit->SetScrollPos(point);
503 CPDF_Point CPWL_EditCtrl::GetScrollPos() const
506 return m_pEdit->GetScrollPos();
508 return CPDF_Point(0.0f, 0.0f);
511 CPDF_Font * CPWL_EditCtrl::GetCaretFont() const
513 int32_t nFontIndex = 0;
515 if (IFX_Edit_Iterator * pIterator = m_pEdit->GetIterator())
517 pIterator->SetAt(m_pEdit->GetCaret());
519 CPVT_Section section;
520 if (pIterator->GetWord(word))
522 nFontIndex = word.nFontIndex;
524 else if (HasFlag(PES_RICH))
526 if (pIterator->GetSection(section))
528 nFontIndex = section.WordProps.nFontIndex;
533 if (IFX_Edit_FontMap* pFontMap = GetFontMap())
534 return pFontMap->GetPDFFont(nFontIndex);
539 FX_FLOAT CPWL_EditCtrl::GetCaretFontSize() const
541 FX_FLOAT fFontSize = GetFontSize();
543 if (IFX_Edit_Iterator * pIterator = m_pEdit->GetIterator())
545 pIterator->SetAt(m_pEdit->GetCaret());
547 CPVT_Section section;
548 if (pIterator->GetWord(word))
550 fFontSize = word.fFontSize;
552 else if (HasFlag(PES_RICH))
554 if (pIterator->GetSection(section))
556 fFontSize = section.WordProps.fFontSize;
564 void CPWL_EditCtrl::SetText(const FX_WCHAR* csText)
566 m_pEdit->SetText(csText);
569 void CPWL_EditCtrl::CopyText()
573 void CPWL_EditCtrl::PasteText()
577 void CPWL_EditCtrl::CutText()
581 void CPWL_EditCtrl::ShowVScrollBar(FX_BOOL bShow)
585 void CPWL_EditCtrl::InsertText(const FX_WCHAR* csText)
588 m_pEdit->InsertText(csText);
591 void CPWL_EditCtrl::InsertWord(FX_WORD word, int32_t nCharset)
594 m_pEdit->InsertWord(word, nCharset);
597 void CPWL_EditCtrl::InsertReturn()
600 m_pEdit->InsertReturn();
603 void CPWL_EditCtrl::Delete()
609 void CPWL_EditCtrl::Backspace()
612 m_pEdit->Backspace();
615 FX_BOOL CPWL_EditCtrl::CanUndo() const
617 return !IsReadOnly() && m_pEdit->CanUndo();
620 FX_BOOL CPWL_EditCtrl::CanRedo() const
622 return !IsReadOnly() && m_pEdit->CanRedo();
625 void CPWL_EditCtrl::Redo()
631 void CPWL_EditCtrl::Undo()
637 void CPWL_EditCtrl::IOnSetScrollInfoY(FX_FLOAT fPlateMin, FX_FLOAT fPlateMax,
638 FX_FLOAT fContentMin, FX_FLOAT fContentMax,
639 FX_FLOAT fSmallStep, FX_FLOAT fBigStep)
641 PWL_SCROLL_INFO Info;
643 Info.fPlateWidth = fPlateMax - fPlateMin;
644 Info.fContentMin = fContentMin;
645 Info.fContentMax = fContentMax;
646 Info.fSmallStep = fSmallStep;
647 Info.fBigStep = fBigStep;
649 OnNotify(this,PNM_SETSCROLLINFO,SBT_VSCROLL,(intptr_t)&Info);
651 if (IsFloatBigger(Info.fPlateWidth,Info.fContentMax-Info.fContentMin)
652 || IsFloatEqual(Info.fPlateWidth,Info.fContentMax-Info.fContentMin))
654 ShowVScrollBar(FALSE);
658 ShowVScrollBar(TRUE);
662 void CPWL_EditCtrl::IOnSetScrollPosY(FX_FLOAT fy)
664 OnNotify(this, PNM_SETSCROLLPOS,SBT_VSCROLL, (intptr_t)&fy);
667 void CPWL_EditCtrl::IOnSetCaret(FX_BOOL bVisible, const CPDF_Point & ptHead, const CPDF_Point & ptFoot, const CPVT_WordPlace& place)
669 PWL_CARET_INFO cInfo;
670 cInfo.bVisible = bVisible;
671 cInfo.ptHead = ptHead;
672 cInfo.ptFoot = ptFoot;
674 OnNotify(this, PNM_SETCARETINFO, (intptr_t)&cInfo, (intptr_t)NULL);
677 void CPWL_EditCtrl::IOnCaretChange(const CPVT_SecProps & secProps, const CPVT_WordProps & wordProps)
681 void CPWL_EditCtrl::IOnContentChange(const CPDF_Rect& rcContent)
687 m_pEditNotify->OnContentChange(rcContent);
692 void CPWL_EditCtrl::IOnInvalidateRect(CPDF_Rect * pRect)
694 InvalidateRect(pRect);
697 int32_t CPWL_EditCtrl::GetCharSet() const
699 return m_nCharSet < 0 ? DEFAULT_CHARSET : m_nCharSet;
702 void CPWL_EditCtrl::GetTextRange(const CPDF_Rect& rect, int32_t & nStartChar, int32_t & nEndChar) const
704 nStartChar = m_pEdit->WordPlaceToWordIndex(m_pEdit->SearchWordPlace(CPDF_Point(rect.left, rect.top)));
705 nEndChar = m_pEdit->WordPlaceToWordIndex(m_pEdit->SearchWordPlace(CPDF_Point(rect.right, rect.bottom)));
708 CFX_WideString CPWL_EditCtrl::GetText(int32_t & nStartChar, int32_t & nEndChar) const
710 CPVT_WordPlace wpStart = m_pEdit->WordIndexToWordPlace(nStartChar);
711 CPVT_WordPlace wpEnd = m_pEdit->WordIndexToWordPlace(nEndChar);
712 return m_pEdit->GetRangeText(CPVT_WordRange(wpStart, wpEnd));
715 void CPWL_EditCtrl::SetReadyToInput()
720 m_bMouseDown = FALSE;