///////////////////////////////////////////////////////////////////////////////
-// Name: tests/textctrl/textctrltest.cpp
+// Name: tests/controls/textctrltest.cpp
// Purpose: wxTextCtrl unit test
// Author: Vadim Zeitlin
// Created: 2007-09-25
#include "testprec.h"
+#if wxUSE_TEXTCTRL
+
#ifdef __BORLANDC__
#pragma hdrstop
#endif
#include "wx/textctrl.h"
#endif // WX_PRECOMP
+#include "wx/scopeguard.h"
+
+#include "textentrytest.h"
+#include "testableframe.h"
+#include "asserthelper.h"
+#include "wx/uiaction.h"
+
// ----------------------------------------------------------------------------
// test class
// ----------------------------------------------------------------------------
-class TextCtrlTestCase : public CppUnit::TestCase
+class TextCtrlTestCase : public TextEntryTestCase, public CppUnit::TestCase
{
public:
TextCtrlTestCase() { }
virtual void tearDown();
private:
+ virtual wxTextEntry *GetTestEntry() const { return m_text; }
+ virtual wxWindow *GetTestWindow() const { return m_text; }
+
CPPUNIT_TEST_SUITE( TextCtrlTestCase );
- CPPUNIT_TEST( SetValue );
- CPPUNIT_TEST( TextChangeEvents );
- CPPUNIT_TEST( Selection );
- CPPUNIT_TEST( InsertionPoint );
+ wxTEXT_ENTRY_TESTS();
+ CPPUNIT_TEST( MultiLineReplace );
+ WXUISIM_TEST( ReadOnly );
+ WXUISIM_TEST( MaxLength );
+ CPPUNIT_TEST( StreamInput );
+ CPPUNIT_TEST( Redirector );
+ //WXUISIM_TEST( ProcessEnter );
+ WXUISIM_TEST( Url );
+ CPPUNIT_TEST( Style );
+ CPPUNIT_TEST( Lines );
+ CPPUNIT_TEST( LogTextCtrl );
CPPUNIT_TEST_SUITE_END();
- void SetValue();
- void TextChangeEvents();
- void Selection();
- void InsertionPoint();
-
- // Selection() test helper: verify that selection is as described by the
- // function parameters
- void AssertSelection(int from, int to, const char *sel);
+ void MultiLineReplace();
+ void ReadOnly();
+ void MaxLength();
+ void StreamInput();
+ void Redirector();
+ //void ProcessEnter();
+ void Url();
+ void Style();
+ void Lines();
+ void LogTextCtrl();
wxTextCtrl *m_text;
- wxFrame *m_frame;
DECLARE_NO_COPY_CLASS(TextCtrlTestCase)
};
void TextCtrlTestCase::tearDown()
{
- delete m_text;
- m_text = NULL;
+ wxDELETE(m_text);
}
// ----------------------------------------------------------------------------
// tests themselves
// ----------------------------------------------------------------------------
-void TextCtrlTestCase::SetValue()
+void TextCtrlTestCase::MultiLineReplace()
{
- CPPUNIT_ASSERT( m_text->IsEmpty() );
+ // we need a multiline control for this test so recreate it
+ delete m_text;
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY, "",
+ wxDefaultPosition, wxDefaultSize,
+ wxTE_MULTILINE);
- m_text->SetValue("foo");
- CPPUNIT_ASSERT_EQUAL( "foo", m_text->GetValue() );
+ m_text->SetValue("Hello replace\n"
+ "0123456789012");
+ m_text->SetInsertionPoint(0);
- m_text->SetValue("");
- CPPUNIT_ASSERT( m_text->IsEmpty() );
+ m_text->Replace(6, 13, "changed");
- m_text->SetValue("hi");
- CPPUNIT_ASSERT_EQUAL( "hi", m_text->GetValue() );
+ CPPUNIT_ASSERT_EQUAL("Hello changed\n"
+ "0123456789012",
+ m_text->GetValue());
+ CPPUNIT_ASSERT_EQUAL(13, m_text->GetInsertionPoint());
- m_text->SetValue("bye");
- CPPUNIT_ASSERT_EQUAL( "bye", m_text->GetValue() );
+ m_text->Replace(13, -1, "");
+ CPPUNIT_ASSERT_EQUAL("Hello changed", m_text->GetValue());
+ CPPUNIT_ASSERT_EQUAL(13, m_text->GetInsertionPoint());
+
+ delete m_text;
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY);
}
-void TextCtrlTestCase::TextChangeEvents()
+void TextCtrlTestCase::ReadOnly()
{
- class TextTestEventHandler : public wxEvtHandler
- {
- public:
- TextTestEventHandler() { m_events = 0; }
+#if wxUSE_UIACTIONSIMULATOR
+ // we need a read only control for this test so recreate it
+ delete m_text;
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY, "",
+ wxDefaultPosition, wxDefaultSize,
+ wxTE_READONLY);
+
+ wxTestableFrame* frame = wxStaticCast(wxTheApp->GetTopWindow(),
+ wxTestableFrame);
+
+ EventCounter count(m_text, wxEVT_COMMAND_TEXT_UPDATED);
+
+ m_text->SetFocus();
+
+ wxUIActionSimulator sim;
+ sim.Text("abcdef");
+ wxYield();
- // calling this automatically resets the events counter
- int GetEvents()
- {
- const int events = m_events;
- m_events = 0;
- return events;
- }
+ CPPUNIT_ASSERT_EQUAL("", m_text->GetValue());
+ CPPUNIT_ASSERT_EQUAL(0, frame->GetEventCount());
- void OnText(wxCommandEvent& WXUNUSED(event)) { m_events++; }
+ // SetEditable() is supposed to override wxTE_READONLY
+ m_text->SetEditable(true);
- private:
- int m_events;
- } handler;
+ sim.Text("abcdef");
+ wxYield();
- m_text->Connect(wxEVT_COMMAND_TEXT_UPDATED,
- wxCommandEventHandler(TextTestEventHandler::OnText),
- NULL,
- &handler);
+ CPPUNIT_ASSERT_EQUAL("abcdef", m_text->GetValue());
+ CPPUNIT_ASSERT_EQUAL(6, frame->GetEventCount());
+
+ delete m_text;
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY);
+#endif
+}
+
+void TextCtrlTestCase::MaxLength()
+{
+#if wxUSE_UIACTIONSIMULATOR
+ wxTestableFrame* frame = wxStaticCast(wxTheApp->GetTopWindow(),
+ wxTestableFrame);
+
+ EventCounter count(m_text, wxEVT_COMMAND_TEXT_UPDATED);
+ EventCounter count1(m_text, wxEVT_COMMAND_TEXT_MAXLEN);
+
+ m_text->SetFocus();
+ m_text->SetMaxLength(10);
+
+ wxUIActionSimulator sim;
+ sim.Text("abcdef");
+ wxYield();
+
+ CPPUNIT_ASSERT_EQUAL(0, frame->GetEventCount(wxEVT_COMMAND_TEXT_MAXLEN));
+
+ sim.Text("ghij");
+ wxYield();
+
+ CPPUNIT_ASSERT_EQUAL(0, frame->GetEventCount(wxEVT_COMMAND_TEXT_MAXLEN));
+ CPPUNIT_ASSERT_EQUAL(10, frame->GetEventCount(wxEVT_COMMAND_TEXT_UPDATED));
+
+ sim.Text("k");
+ wxYield();
+
+ CPPUNIT_ASSERT_EQUAL(1, frame->GetEventCount(wxEVT_COMMAND_TEXT_MAXLEN));
+ CPPUNIT_ASSERT_EQUAL(0, frame->GetEventCount(wxEVT_COMMAND_TEXT_UPDATED));
+
+ m_text->SetMaxLength(0);
+
+ sim.Text("k");
+ wxYield();
+
+ CPPUNIT_ASSERT_EQUAL(0, frame->GetEventCount(wxEVT_COMMAND_TEXT_MAXLEN));
+ CPPUNIT_ASSERT_EQUAL(1, frame->GetEventCount(wxEVT_COMMAND_TEXT_UPDATED));
+#endif
+}
+
+void TextCtrlTestCase::StreamInput()
+{
+#ifndef __WXOSX__
+ {
+ // Ensure we use decimal point and not a comma.
+ char * const locOld = setlocale(LC_NUMERIC, "C");
+ wxON_BLOCK_EXIT2( setlocale, (int)LC_NUMERIC, locOld );
+
+ *m_text << "stringinput"
+ << 10
+ << 1000L
+ << 3.14f
+ << 2.71
+ << 'a'
+ << L'b';
+ }
+
+ CPPUNIT_ASSERT_EQUAL("stringinput1010003.142.71ab", m_text->GetValue());
- // notice that SetValue() generates an event even if the text didn't change
m_text->SetValue("");
- CPPUNIT_ASSERT_EQUAL( 1, handler.GetEvents() );
- m_text->SetValue("foo");
- CPPUNIT_ASSERT_EQUAL( 1, handler.GetEvents() );
+#if wxHAS_TEXT_WINDOW_STREAM
+
+ std::ostream stream(m_text);
+
+ // We don't test a wide character as this is not a wide stream
+ stream << "stringinput"
+ << 10
+ << 1000L
+ << 3.14f
+ << 2.71
+ << 'a';
+
+ stream.flush();
+
+ CPPUNIT_ASSERT_EQUAL("stringinput1010003.142.71a", m_text->GetValue());
- m_text->SetValue("foo");
- CPPUNIT_ASSERT_EQUAL( 1, handler.GetEvents() );
+#endif // wxHAS_TEXT_WINDOW_STREAM
+#endif // !__WXOSX__
+}
+
+void TextCtrlTestCase::Redirector()
+{
+#if wxHAS_TEXT_WINDOW_STREAM && wxUSE_STD_IOSTREAM
+
+ wxStreamToTextRedirector redirect(m_text);
+
+ std::cout << "stringinput"
+ << 10
+ << 1000L
+ << 3.14f
+ << 2.71
+ << 'a';
+
+ CPPUNIT_ASSERT_EQUAL("stringinput1010003.142.71a", m_text->GetValue());
+
+#endif
+}
+
+#if 0
+void TextCtrlTestCase::ProcessEnter()
+{
+#if wxUSE_UIACTIONSIMULATOR
+ wxTestableFrame* frame = wxStaticCast(wxTheApp->GetTopWindow(),
+ wxTestableFrame);
+
+ EventCounter count(m_text, wxEVT_COMMAND_TEXT_ENTER);
- m_text->ChangeValue("bar");
- CPPUNIT_ASSERT_EQUAL( 0, handler.GetEvents() );
+ m_text->SetFocus();
- m_text->AppendText("bar");
- CPPUNIT_ASSERT_EQUAL( 1, handler.GetEvents() );
+ wxUIActionSimulator sim;
+ sim.Char(WXK_RETURN);
+ wxYield();
- m_text->Replace(3, 6, "baz");
- CPPUNIT_ASSERT_EQUAL( 1, handler.GetEvents() );
+ CPPUNIT_ASSERT_EQUAL(0, frame->GetEventCount(wxEVT_COMMAND_TEXT_ENTER));
- m_text->Remove(0, 3);
- CPPUNIT_ASSERT_EQUAL( 1, handler.GetEvents() );
+ // we need a text control with wxTE_PROCESS_ENTER for this test
+ delete m_text;
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY, "",
+ wxDefaultPosition, wxDefaultSize,
+ wxTE_PROCESS_ENTER);
+
+ m_text->SetFocus();
- m_text->WriteText("foo");
- CPPUNIT_ASSERT_EQUAL( 1, handler.GetEvents() );
+ sim.Char(WXK_RETURN);
+ wxYield();
- m_text->Clear();
- CPPUNIT_ASSERT_EQUAL( 1, handler.GetEvents() );
+ CPPUNIT_ASSERT_EQUAL(1, frame->GetEventCount(wxEVT_COMMAND_TEXT_ENTER));
+#endif
}
+#endif
-void TextCtrlTestCase::AssertSelection(int from, int to, const char *sel)
+void TextCtrlTestCase::Url()
{
- CPPUNIT_ASSERT( m_text->HasSelection() );
+#if wxUSE_UIACTIONSIMULATOR && defined(__WXMSW__)
+ delete m_text;
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY, "",
+ wxDefaultPosition, wxDefaultSize,
+ wxTE_MULTILINE | wxTE_RICH | wxTE_AUTO_URL);
+
+ wxTestableFrame* frame = wxStaticCast(wxTheApp->GetTopWindow(),
+ wxTestableFrame);
- long fromReal,
- toReal;
- m_text->GetSelection(&fromReal, &toReal);
- CPPUNIT_ASSERT_EQUAL( from, fromReal );
- CPPUNIT_ASSERT_EQUAL( to, toReal );
- CPPUNIT_ASSERT_EQUAL( sel, m_text->GetStringSelection() );
+ EventCounter count(m_text, wxEVT_COMMAND_TEXT_URL);
- CPPUNIT_ASSERT_EQUAL( from, m_text->GetInsertionPoint() );
+ m_text->AppendText("http://www.wxwidgets.org");
+
+ wxUIActionSimulator sim;
+ sim.MouseMove(m_text->ClientToScreen(wxPoint(5, 5)));
+ sim.MouseClick();
+ wxYield();
+
+ CPPUNIT_ASSERT_EQUAL(1, frame->GetEventCount());
+#endif
}
-void TextCtrlTestCase::Selection()
+void TextCtrlTestCase::Style()
{
- m_text->SetValue("0123456789");
+#ifndef __WXOSX__
+ delete m_text;
+ // We need wxTE_RICH under windows for style support
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY, "",
+ wxDefaultPosition, wxDefaultSize, wxTE_RICH);
+
+ // Red text on a white background
+ m_text->SetDefaultStyle(wxTextAttr(*wxRED, *wxWHITE));
+
+ CPPUNIT_ASSERT_EQUAL(m_text->GetDefaultStyle().GetTextColour(), *wxRED);
+ CPPUNIT_ASSERT_EQUAL(m_text->GetDefaultStyle().GetBackgroundColour(),
+ *wxWHITE);
- m_text->SetSelection(2, 4);
- AssertSelection(2, 4, "23"); // not "234"!
+ m_text->AppendText("red on white ");
- m_text->SetSelection(3, -1);
- AssertSelection(3, 10, "3456789");
+ // Red text on a grey background
+ m_text->SetDefaultStyle(wxTextAttr(wxNullColour, *wxLIGHT_GREY));
- m_text->SelectAll();
- AssertSelection(0, 10, "0123456789");
+ CPPUNIT_ASSERT_EQUAL(m_text->GetDefaultStyle().GetTextColour(), *wxRED);
+ CPPUNIT_ASSERT_EQUAL(m_text->GetDefaultStyle().GetBackgroundColour(),
+ *wxLIGHT_GREY);
- m_text->SetSelection(0, 0);
- CPPUNIT_ASSERT( !m_text->HasSelection() );
+ m_text->AppendText("red on grey ");
+
+ // Blue text on a grey background
+ m_text->SetDefaultStyle(wxTextAttr(*wxBLUE));
+
+
+ CPPUNIT_ASSERT_EQUAL(m_text->GetDefaultStyle().GetTextColour(), *wxBLUE);
+ CPPUNIT_ASSERT_EQUAL(m_text->GetDefaultStyle().GetBackgroundColour(),
+ *wxLIGHT_GREY);
+
+ m_text->AppendText("blue on grey");
+
+ // Get getting the style at a specific location
+ wxTextAttr style;
+
+ // We have to check that styles are supported
+ if(m_text->GetStyle(3, style))
+ {
+ CPPUNIT_ASSERT_EQUAL(style.GetTextColour(), *wxRED);
+ CPPUNIT_ASSERT_EQUAL(style.GetBackgroundColour(), *wxWHITE);
+ }
+
+ // And then setting the style
+ if(m_text->SetStyle(15, 18, style))
+ {
+ m_text->GetStyle(17, style);
+
+ CPPUNIT_ASSERT_EQUAL(style.GetTextColour(), *wxRED);
+ CPPUNIT_ASSERT_EQUAL(style.GetBackgroundColour(), *wxWHITE);
+ }
+#endif
}
-void TextCtrlTestCase::InsertionPoint()
+void TextCtrlTestCase::Lines()
{
- CPPUNIT_ASSERT_EQUAL( 0, m_text->GetLastPosition() );
- CPPUNIT_ASSERT_EQUAL( 0, m_text->GetInsertionPoint() );
+#ifndef __WXOSX__
+ delete m_text;
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY, "",
+ wxDefaultPosition, wxSize(400, 200), wxTE_MULTILINE | wxTE_DONTWRAP);
- m_text->SetValue("0"); // should put the insertion point in front
- CPPUNIT_ASSERT_EQUAL( 1, m_text->GetLastPosition() );
- CPPUNIT_ASSERT_EQUAL( 0, m_text->GetInsertionPoint() );
+ m_text->SetValue("line1\nline2\nlong long line 3");
+ m_text->Refresh();
+ m_text->Update();
- m_text->AppendText("12"); // should update the insertion point position
- CPPUNIT_ASSERT_EQUAL( 3, m_text->GetLastPosition() );
- CPPUNIT_ASSERT_EQUAL( 3, m_text->GetInsertionPoint() );
+ CPPUNIT_ASSERT_EQUAL(3, m_text->GetNumberOfLines());
+ CPPUNIT_ASSERT_EQUAL(5, m_text->GetLineLength(0));
+ CPPUNIT_ASSERT_EQUAL("line2", m_text->GetLineText(1));
+ CPPUNIT_ASSERT_EQUAL(16, m_text->GetLineLength(2));
- m_text->SetInsertionPoint(1);
- CPPUNIT_ASSERT_EQUAL( 3, m_text->GetLastPosition() );
- CPPUNIT_ASSERT_EQUAL( 1, m_text->GetInsertionPoint() );
+ m_text->AppendText("\n\nMore text on line 5");
- m_text->SetInsertionPointEnd();
- CPPUNIT_ASSERT_EQUAL( 3, m_text->GetInsertionPoint() );
+ CPPUNIT_ASSERT_EQUAL(5, m_text->GetNumberOfLines());
+ CPPUNIT_ASSERT_EQUAL(0, m_text->GetLineLength(3));
+ CPPUNIT_ASSERT_EQUAL("", m_text->GetLineText(3));
+#endif
+}
- m_text->SetInsertionPoint(0);
- m_text->WriteText("-"); // should move it after the written text
- CPPUNIT_ASSERT_EQUAL( 4, m_text->GetLastPosition() );
- CPPUNIT_ASSERT_EQUAL( 1, m_text->GetInsertionPoint() );
+void TextCtrlTestCase::LogTextCtrl()
+{
+ delete m_text;
+ m_text = new wxTextCtrl(wxTheApp->GetTopWindow(), wxID_ANY, "",
+ wxDefaultPosition, wxSize(400, 200),
+ wxTE_MULTILINE);
+
+ CPPUNIT_ASSERT(m_text->IsEmpty());
+
+ wxLogTextCtrl* logtext = new wxLogTextCtrl(m_text);
+
+ wxLog* old = wxLog::SetActiveTarget(logtext);
+
+ logtext->LogText("text");
+
+ delete wxLog::SetActiveTarget(old);
+
+ CPPUNIT_ASSERT(!m_text->IsEmpty());
}
+#endif //wxUSE_TEXTCTRL