refactor text_editor
fix issues that caret works incorrectly in line-wrapped mode.
This commit is contained in:
		
							parent
							
								
									df6c707356
								
							
						
					
					
						commit
						a4f15f7bb0
					
				| @ -85,6 +85,8 @@ namespace nana{	namespace widgets | |||||||
| 			text_editor(window, graph_reference, const text_editor_scheme*); | 			text_editor(window, graph_reference, const text_editor_scheme*); | ||||||
| 			~text_editor(); | 			~text_editor(); | ||||||
| 
 | 
 | ||||||
|  | 			size caret_size() const; | ||||||
|  | 
 | ||||||
| 			void set_highlight(const ::std::string& name, const ::nana::color&, const ::nana::color&); | 			void set_highlight(const ::std::string& name, const ::nana::color&, const ::nana::color&); | ||||||
| 			void erase_highlight(const ::std::string& name); | 			void erase_highlight(const ::std::string& name); | ||||||
| 			void set_keyword(const ::std::wstring& kw, const std::string& name, bool case_sensitive, bool whole_word_matched); | 			void set_keyword(const ::std::wstring& kw, const std::string& name, bool case_sensitive, bool whole_word_matched); | ||||||
| @ -217,8 +219,10 @@ namespace nana{	namespace widgets | |||||||
| 			std::vector<upoint> _m_render_text(const ::nana::color& text_color); | 			std::vector<upoint> _m_render_text(const ::nana::color& text_color); | ||||||
| 			void _m_pre_calc_lines(std::size_t line_off, std::size_t lines); | 			void _m_pre_calc_lines(std::size_t line_off, std::size_t lines); | ||||||
| 
 | 
 | ||||||
| 			::nana::point	_m_caret_to_screen(::nana::upoint pos) const; | 			//Caret to screen coordinate or context coordiate(in pixels)
 | ||||||
| 			::nana::upoint	_m_screen_to_caret(::nana::point pos) const; | 			::nana::point	_m_caret_to_coordinate(::nana::upoint pos, bool to_screen_coordinate = true) const; | ||||||
|  | 			//Screen coordinate or context coordinate(in pixels) to caret,
 | ||||||
|  | 			::nana::upoint	_m_coordinate_to_caret(::nana::point pos, bool from_screen_coordinate = true) const; | ||||||
| 
 | 
 | ||||||
| 			bool _m_pos_from_secondary(std::size_t textline, const nana::upoint& secondary, unsigned & pos); | 			bool _m_pos_from_secondary(std::size_t textline, const nana::upoint& secondary, unsigned & pos); | ||||||
| 			bool _m_pos_secondary(const nana::upoint& charpos, nana::upoint& secondary_pos) const; | 			bool _m_pos_secondary(const nana::upoint& charpos, nana::upoint& secondary_pos) const; | ||||||
| @ -243,8 +247,9 @@ namespace nana{	namespace widgets | |||||||
| 			unsigned _m_tabs_pixels(size_type tabs) const; | 			unsigned _m_tabs_pixels(size_type tabs) const; | ||||||
| 			nana::size _m_text_extent_size(const char_type*, size_type n) const; | 			nana::size _m_text_extent_size(const char_type*, size_type n) const; | ||||||
| 
 | 
 | ||||||
| 			/// Moves the view of window.
 | 			/// Adjust position of view to make caret stay in screen
 | ||||||
| 			bool _m_move_offset_x_while_over_border(int many); | 			bool _m_adjust_view(); | ||||||
|  | 
 | ||||||
| 			bool _m_move_select(bool record_undo); | 			bool _m_move_select(bool record_undo); | ||||||
| 
 | 
 | ||||||
| 			int _m_text_top_base() const; | 			int _m_text_top_base() const; | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| /*
 | /*
 | ||||||
|  *	A textbase class implementation |  *	A textbase class implementation | ||||||
|  *	Nana C++ Library(http://www.nanapro.org)
 |  *	Nana C++ Library(http://www.nanapro.org)
 | ||||||
|  *	Copyright(C) 2003-2016 Jinhao(cnjinhao@hotmail.com) |  *	Copyright(C) 2003-2017 Jinhao(cnjinhao@hotmail.com) | ||||||
|  * |  * | ||||||
|  *	Distributed under the Boost Software License, Version 1.0. |  *	Distributed under the Boost Software License, Version 1.0. | ||||||
|  *	(See accompanying file LICENSE_1_0.txt or copy at |  *	(See accompanying file LICENSE_1_0.txt or copy at | ||||||
| @ -44,7 +44,7 @@ namespace skeletons | |||||||
| 		{ | 		{ | ||||||
| 			attr_max_.reset(); | 			attr_max_.reset(); | ||||||
| 			//Insert an empty string for the first line of empty text.
 | 			//Insert an empty string for the first line of empty text.
 | ||||||
| 			text_cont_.emplace_back(); | 			text_cont_.emplace_back(new string_type); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		void set_event_agent(textbase_event_agent_interface * evt) | 		void set_event_agent(textbase_event_agent_interface * evt) | ||||||
| @ -55,7 +55,7 @@ namespace skeletons | |||||||
| 		bool empty() const | 		bool empty() const | ||||||
| 		{ | 		{ | ||||||
| 			return (text_cont_.empty() || | 			return (text_cont_.empty() || | ||||||
| 					((text_cont_.size() == 1) && (text_cont_[0].empty()))); | 					((text_cont_.size() == 1) && (text_cont_.front()->empty()))); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		bool load(const char* file_utf8) | 		bool load(const char* file_utf8) | ||||||
| @ -135,10 +135,10 @@ namespace skeletons | |||||||
| 			while(ifs.good()) | 			while(ifs.good()) | ||||||
| 			{ | 			{ | ||||||
| 				std::getline(ifs, str_mbs); | 				std::getline(ifs, str_mbs); | ||||||
| 				text_cont_.emplace_back(static_cast<string_type&&>(nana::charset{ str_mbs })); | 				text_cont_.emplace_back(new string_type(static_cast<string_type&&>(nana::charset{ str_mbs }))); | ||||||
| 				if(text_cont_.back().size() > attr_max_.size) | 				if(text_cont_.back()->size() > attr_max_.size) | ||||||
| 				{ | 				{ | ||||||
| 					attr_max_.size = text_cont_.back().size(); | 					attr_max_.size = text_cont_.back()->size(); | ||||||
| 					attr_max_.line = text_cont_.size() - 1; | 					attr_max_.line = text_cont_.size() - 1; | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| @ -218,9 +218,9 @@ namespace skeletons | |||||||
| 						byte_order_translate_4bytes(str); | 						byte_order_translate_4bytes(str); | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				text_cont_.emplace_back(static_cast<string_type&&>(nana::charset{ str, encoding })); | 				text_cont_.emplace_back(new string_type(static_cast<string_type&&>(nana::charset{ str, encoding }))); | ||||||
| 
 | 
 | ||||||
| 				attr_max_.size = text_cont_.back().size(); | 				attr_max_.size = text_cont_.back()->size(); | ||||||
| 				attr_max_.line = 0; | 				attr_max_.line = 0; | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| @ -236,10 +236,10 @@ namespace skeletons | |||||||
| 						byte_order_translate_4bytes(str); | 						byte_order_translate_4bytes(str); | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				text_cont_.emplace_back(static_cast<string_type&&>(nana::charset{ str, encoding })); | 				text_cont_.emplace_back(new string_type(static_cast<string_type&&>(nana::charset{ str, encoding }))); | ||||||
| 				if(text_cont_.back().size() > attr_max_.size) | 				if(text_cont_.back()->size() > attr_max_.size) | ||||||
| 				{ | 				{ | ||||||
| 					attr_max_.size = text_cont_.back().size(); | 					attr_max_.size = text_cont_.back()->size(); | ||||||
| 					attr_max_.line = text_cont_.size() - 1; | 					attr_max_.line = text_cont_.size() - 1; | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| @ -253,6 +253,9 @@ namespace skeletons | |||||||
| 			std::ofstream ofs(to_osmbstr(fs), std::ios::binary); | 			std::ofstream ofs(to_osmbstr(fs), std::ios::binary); | ||||||
| 			if(ofs && text_cont_.size()) | 			if(ofs && text_cont_.size()) | ||||||
| 			{ | 			{ | ||||||
|  | 				auto i = text_cont_.cbegin(); | ||||||
|  | 				auto const count = text_cont_.size() - 1; | ||||||
|  | 
 | ||||||
| 				std::string last_mbs; | 				std::string last_mbs; | ||||||
| 
 | 
 | ||||||
| 				if (is_unicode) | 				if (is_unicode) | ||||||
| @ -272,32 +275,27 @@ namespace skeletons | |||||||
| 					if (bytes) | 					if (bytes) | ||||||
| 						ofs.write(le_boms[static_cast<int>(encoding)], bytes); | 						ofs.write(le_boms[static_cast<int>(encoding)], bytes); | ||||||
| 
 | 
 | ||||||
| 					if (text_cont_.size() > 1) | 					for (std::size_t pos = 0; pos < count; ++pos) | ||||||
| 					{ | 					{ | ||||||
| 						std::string mbs; | 						auto mbs = nana::charset(**(i++)).to_bytes(encoding); | ||||||
| 						for (auto i = text_cont_.cbegin(), end = text_cont_.cend() - 1; i != end; ++i) |  | ||||||
| 						{ |  | ||||||
| 							std::string(nana::charset(*i).to_bytes(encoding)).swap(mbs); |  | ||||||
| 							mbs += "\r\n"; |  | ||||||
| 						ofs.write(mbs.c_str(), static_cast<std::streamsize>(mbs.size())); | 						ofs.write(mbs.c_str(), static_cast<std::streamsize>(mbs.size())); | ||||||
| 						} | 						ofs.write("\r\n", 2); | ||||||
| 					} | 					} | ||||||
| 
 | 
 | ||||||
| 					last_mbs = nana::charset(text_cont_.back()).to_bytes(encoding); | 					last_mbs = nana::charset(*text_cont_.back()).to_bytes(encoding); | ||||||
| 				} | 				} | ||||||
| 				else | 				else | ||||||
| 				{ | 				{ | ||||||
| 					if (text_cont_.size() > 1) | 					for (std::size_t pos = 0; pos < count; ++pos) | ||||||
| 					{ | 					{ | ||||||
| 						for (auto i = text_cont_.cbegin(), end = text_cont_.cend() - 1; i != end; ++i) | 						std::string mbs = nana::charset(**(i++)); | ||||||
| 						{ |  | ||||||
| 							std::string mbs = nana::charset(*i); |  | ||||||
| 						ofs.write(mbs.c_str(), mbs.size()); | 						ofs.write(mbs.c_str(), mbs.size()); | ||||||
| 						ofs.write("\r\n", 2); | 						ofs.write("\r\n", 2); | ||||||
| 					} | 					} | ||||||
|  | 
 | ||||||
|  | 					last_mbs = nana::charset(*text_cont_.back()); | ||||||
| 				} | 				} | ||||||
| 					last_mbs = nana::charset(text_cont_.back()); | 
 | ||||||
| 				} |  | ||||||
| 				ofs.write(last_mbs.c_str(), static_cast<std::streamsize>(last_mbs.size())); | 				ofs.write(last_mbs.c_str(), static_cast<std::streamsize>(last_mbs.size())); | ||||||
| 				_m_saved(std::move(fs)); | 				_m_saved(std::move(fs)); | ||||||
| 			} | 			} | ||||||
| @ -310,8 +308,8 @@ namespace skeletons | |||||||
| 
 | 
 | ||||||
| 		const string_type& getline(size_type pos) const | 		const string_type& getline(size_type pos) const | ||||||
| 		{ | 		{ | ||||||
| 			if(pos < text_cont_.size()) | 			if (pos < text_cont_.size()) | ||||||
| 				return text_cont_[pos]; | 				return *text_cont_[pos]; | ||||||
| 
 | 
 | ||||||
| 			return nullstr_; | 			return nullstr_; | ||||||
| 		} | 		} | ||||||
| @ -323,13 +321,13 @@ namespace skeletons | |||||||
| 	public: | 	public: | ||||||
| 		void replace(size_type pos, string_type && text) | 		void replace(size_type pos, string_type && text) | ||||||
| 		{ | 		{ | ||||||
| 			if(text_cont_.size() <= pos) | 			if (text_cont_.size() <= pos) | ||||||
| 			{ | 			{ | ||||||
| 				text_cont_.emplace_back(std::move(text)); | 				text_cont_.emplace_back(new string_type(std::move(text))); | ||||||
| 				pos = text_cont_.size() - 1; | 				pos = text_cont_.size() - 1; | ||||||
| 			} | 			} | ||||||
| 			else | 			else | ||||||
| 				text_cont_[pos].swap(text); | 				_m_at(pos).swap(text); | ||||||
| 
 | 
 | ||||||
| 			_m_make_max(pos); | 			_m_make_max(pos); | ||||||
| 			_m_edited(); | 			_m_edited(); | ||||||
| @ -339,7 +337,7 @@ namespace skeletons | |||||||
| 		{ | 		{ | ||||||
| 			if(pos.y < text_cont_.size()) | 			if(pos.y < text_cont_.size()) | ||||||
| 			{ | 			{ | ||||||
| 				string_type& lnstr = text_cont_[pos.y]; | 				string_type& lnstr = _m_at(pos.y); | ||||||
| 
 | 
 | ||||||
| 				if(pos.x < lnstr.size()) | 				if(pos.x < lnstr.size()) | ||||||
| 					lnstr.insert(pos.x, str); | 					lnstr.insert(pos.x, str); | ||||||
| @ -348,7 +346,7 @@ namespace skeletons | |||||||
| 			} | 			} | ||||||
| 			else | 			else | ||||||
| 			{ | 			{ | ||||||
| 				text_cont_.emplace_back(std::move(str)); | 				text_cont_.emplace_back(new string_type(std::move(str))); | ||||||
| 				pos.y = static_cast<unsigned>(text_cont_.size() - 1); | 				pos.y = static_cast<unsigned>(text_cont_.size() - 1); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| @ -358,10 +356,10 @@ namespace skeletons | |||||||
| 
 | 
 | ||||||
| 		void insertln(size_type pos, string_type&& str) | 		void insertln(size_type pos, string_type&& str) | ||||||
| 		{ | 		{ | ||||||
| 			if(pos < text_cont_.size()) | 			if (pos < text_cont_.size()) | ||||||
| 				text_cont_.emplace(text_cont_.begin() + pos, std::move(str)); | 				text_cont_.emplace(_m_iat(pos), new string_type(std::move(str))); | ||||||
| 			else | 			else | ||||||
| 				text_cont_.emplace_back(std::move(str)); | 				text_cont_.emplace_back(new string_type(std::move(str))); | ||||||
| 
 | 
 | ||||||
| 			_m_make_max(pos); | 			_m_make_max(pos); | ||||||
| 			_m_edited(); | 			_m_edited(); | ||||||
| @ -371,7 +369,7 @@ namespace skeletons | |||||||
| 		{ | 		{ | ||||||
| 			if (line < text_cont_.size()) | 			if (line < text_cont_.size()) | ||||||
| 			{ | 			{ | ||||||
| 				string_type& lnstr = text_cont_[line]; | 				string_type& lnstr = _m_at(line); | ||||||
| 				if ((pos == 0) && (count >= lnstr.size())) | 				if ((pos == 0) && (count >= lnstr.size())) | ||||||
| 					lnstr.clear(); | 					lnstr.clear(); | ||||||
| 				else | 				else | ||||||
| @ -393,7 +391,7 @@ namespace skeletons | |||||||
| 			if (pos + n > text_cont_.size()) | 			if (pos + n > text_cont_.size()) | ||||||
| 				n = text_cont_.size() - pos; | 				n = text_cont_.size() - pos; | ||||||
| 
 | 
 | ||||||
| 			text_cont_.erase(text_cont_.begin() + pos, text_cont_.begin() + (pos + n)); | 			text_cont_.erase(_m_iat(pos), _m_iat(pos + n)); | ||||||
| 
 | 
 | ||||||
| 			if (pos <= attr_max_.line && attr_max_.line < pos + n) | 			if (pos <= attr_max_.line && attr_max_.line < pos + n) | ||||||
| 				_m_scan_for_max(); | 				_m_scan_for_max(); | ||||||
| @ -408,7 +406,7 @@ namespace skeletons | |||||||
| 		{ | 		{ | ||||||
| 			text_cont_.clear(); | 			text_cont_.clear(); | ||||||
| 			attr_max_.reset(); | 			attr_max_.reset(); | ||||||
| 			text_cont_.emplace_back();	//text_cont_ must not be empty
 | 			text_cont_.emplace_back(new string_type);	//text_cont_ must not be empty
 | ||||||
| 
 | 
 | ||||||
| 			_m_saved(std::string()); | 			_m_saved(std::string()); | ||||||
| 		} | 		} | ||||||
| @ -417,9 +415,14 @@ namespace skeletons | |||||||
| 		{ | 		{ | ||||||
| 			if(pos + 1 < text_cont_.size()) | 			if(pos + 1 < text_cont_.size()) | ||||||
| 			{ | 			{ | ||||||
| 				text_cont_[pos] += text_cont_[pos + 1]; | 				auto i = _m_iat(pos + 1); | ||||||
| 				text_cont_.erase(text_cont_.begin() + (pos + 1)); | 				_m_at(pos) += **i; | ||||||
|  | 				text_cont_.erase(i); | ||||||
| 				_m_make_max(pos); | 				_m_make_max(pos); | ||||||
|  | 
 | ||||||
|  | 				//If the maxline is behind the pos line,
 | ||||||
|  | 				//decrease the maxline. Because a line between maxline and pos line
 | ||||||
|  | 				//has been deleted.
 | ||||||
| 				if(pos < attr_max_.line) | 				if(pos < attr_max_.line) | ||||||
| 					--attr_max_.line; | 					--attr_max_.line; | ||||||
| 
 | 
 | ||||||
| @ -458,9 +461,19 @@ namespace skeletons | |||||||
| 			return edited() || filename_.empty(); | 			return edited() || filename_.empty(); | ||||||
| 		} | 		} | ||||||
| 	private: | 	private: | ||||||
|  | 		string_type& _m_at(size_type pos) | ||||||
|  | 		{ | ||||||
|  | 			return **_m_iat(pos); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		typename std::deque<std::unique_ptr<string_type>>::iterator _m_iat(size_type pos) | ||||||
|  | 		{ | ||||||
|  | 			return text_cont_.begin() + pos; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		void _m_make_max(std::size_t pos) | 		void _m_make_max(std::size_t pos) | ||||||
| 		{ | 		{ | ||||||
| 			const string_type& str = text_cont_[pos]; | 			const string_type& str = _m_at(pos); | ||||||
| 			if(str.size() > attr_max_.size) | 			if(str.size() > attr_max_.size) | ||||||
| 			{ | 			{ | ||||||
| 				attr_max_.size = str.size(); | 				attr_max_.size = str.size(); | ||||||
| @ -472,11 +485,11 @@ namespace skeletons | |||||||
| 		{ | 		{ | ||||||
| 			attr_max_.size = 0; | 			attr_max_.size = 0; | ||||||
| 			std::size_t n = 0; | 			std::size_t n = 0; | ||||||
| 			for(auto & s : text_cont_) | 			for(auto & p : text_cont_) | ||||||
| 			{ | 			{ | ||||||
| 				if(s.size() > attr_max_.size) | 				if(p->size() > attr_max_.size) | ||||||
| 				{ | 				{ | ||||||
| 					attr_max_.size = s.size(); | 					attr_max_.size = p->size(); | ||||||
| 					attr_max_.line = n; | 					attr_max_.line = n; | ||||||
| 				} | 				} | ||||||
| 				++n; | 				++n; | ||||||
| @ -514,7 +527,7 @@ namespace skeletons | |||||||
| 				evt_agent_->text_changed(); | 				evt_agent_->text_changed(); | ||||||
| 		} | 		} | ||||||
| 	private: | 	private: | ||||||
| 		std::deque<string_type>	text_cont_; | 		std::deque<std::unique_ptr<string_type>>	text_cont_; | ||||||
| 		textbase_event_agent_interface* evt_agent_{ nullptr }; | 		textbase_event_agent_interface* evt_agent_{ nullptr }; | ||||||
| 
 | 
 | ||||||
| 		mutable bool			changed_{ false }; | 		mutable bool			changed_{ false }; | ||||||
|  | |||||||
| @ -319,31 +319,31 @@ namespace nana { | |||||||
| 
 | 
 | ||||||
| 			void content_view::content_size(const size& sz, bool try_update) | 			void content_view::content_size(const size& sz, bool try_update) | ||||||
| 			{ | 			{ | ||||||
|  | 				auto const view_sz = this->view_area(sz); | ||||||
|  | 
 | ||||||
| 				if (sz.height < impl_->content_size.height) | 				if (sz.height < impl_->content_size.height) | ||||||
| 				{ | 				{ | ||||||
| 					if (impl_->origin.y + impl_->disp_area.height > sz.height) | 					if (impl_->origin.y + view_sz.height > sz.height) | ||||||
| 					{ | 					{ | ||||||
| 						if (impl_->disp_area.height > sz.height) | 						if (view_sz.height > sz.height) | ||||||
| 							impl_->origin.y = 0; | 							impl_->origin.y = 0; | ||||||
| 						else | 						else | ||||||
| 							impl_->origin.y = sz.height - impl_->disp_area.height; | 							impl_->origin.y = sz.height - view_sz.height; | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				if (sz.width < impl_->content_size.width) | 				if (sz.width < impl_->content_size.width) | ||||||
| 				{ | 				{ | ||||||
| 					if (impl_->origin.x + impl_->disp_area.width > sz.width) | 					if (impl_->origin.x + view_sz.width > sz.width) | ||||||
| 					{ | 					{ | ||||||
| 						if (impl_->disp_area.width > sz.width) | 						if (view_sz.width > sz.width) | ||||||
| 							impl_->origin.x = 0; | 							impl_->origin.x = 0; | ||||||
| 						else | 						else | ||||||
| 							impl_->origin.x = sz.width - impl_->disp_area.width; | 							impl_->origin.x = sz.width - view_sz.width; | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 				impl_->content_size = sz; | 				impl_->content_size = sz; | ||||||
| 
 |  | ||||||
| 				impl_->size_changed(try_update); | 				impl_->size_changed(try_update); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| @ -388,11 +388,16 @@ namespace nana { | |||||||
| 
 | 
 | ||||||
| 			rectangle content_view::view_area() const | 			rectangle content_view::view_area() const | ||||||
| 			{ | 			{ | ||||||
| 				unsigned extra_horz = (impl_->disp_area.width < impl_->content_size.width ? space() : 0); | 				return view_area(impl_->content_size); | ||||||
| 				unsigned extra_vert = (impl_->disp_area.height < impl_->content_size.height + extra_horz ? space() : 0); | 			} | ||||||
|  | 
 | ||||||
|  | 			rectangle content_view::view_area(const size& alt_content_size) const | ||||||
|  | 			{ | ||||||
|  | 				unsigned extra_horz = (impl_->disp_area.width < alt_content_size.width ? space() : 0); | ||||||
|  | 				unsigned extra_vert = (impl_->disp_area.height < alt_content_size.height + extra_horz ? space() : 0); | ||||||
| 
 | 
 | ||||||
| 				if ((0 == extra_horz) && extra_vert) | 				if ((0 == extra_horz) && extra_vert) | ||||||
| 					extra_horz = (impl_->disp_area.width < impl_->content_size.width + extra_vert ? space() : 0); | 					extra_horz = (impl_->disp_area.width < alt_content_size.width + extra_vert ? space() : 0); | ||||||
| 
 | 
 | ||||||
| 				return rectangle{ | 				return rectangle{ | ||||||
| 					impl_->disp_area.position(), | 					impl_->disp_area.position(), | ||||||
|  | |||||||
| @ -65,6 +65,7 @@ namespace skeletons | |||||||
| 		void draw_corner(graph_reference); | 		void draw_corner(graph_reference); | ||||||
| 
 | 
 | ||||||
| 		rectangle view_area() const; | 		rectangle view_area() const; | ||||||
|  | 		rectangle view_area(const size& alt_content_size) const; | ||||||
| 
 | 
 | ||||||
| 		unsigned extra_space(bool horz) const; | 		unsigned extra_space(bool horz) const; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ | |||||||
| #include <nana/gui/widgets/widget.hpp> | #include <nana/gui/widgets/widget.hpp> | ||||||
| #include "content_view.hpp" | #include "content_view.hpp" | ||||||
| 
 | 
 | ||||||
|  | #include <deque> | ||||||
| #include <numeric> | #include <numeric> | ||||||
| #include <cwctype> | #include <cwctype> | ||||||
| #include <cstring> | #include <cstring> | ||||||
| @ -45,7 +46,7 @@ namespace nana{	namespace widgets | |||||||
| 		{ | 		{ | ||||||
| 		public: | 		public: | ||||||
| 			using command = EnumCommand; | 			using command = EnumCommand; | ||||||
| 			using container = std::deque < std::unique_ptr<undoable_command_interface<command>> >; | 			using container = std::deque<std::unique_ptr<undoable_command_interface<command>>>; | ||||||
| 
 | 
 | ||||||
| 			void clear() | 			void clear() | ||||||
| 			{ | 			{ | ||||||
| @ -555,7 +556,6 @@ namespace nana{	namespace widgets | |||||||
| 			virtual std::size_t take_lines() const = 0; | 			virtual std::size_t take_lines() const = 0; | ||||||
| 			/// Returns the number of lines that the line of text specified by pos takes.
 | 			/// Returns the number of lines that the line of text specified by pos takes.
 | ||||||
| 			virtual std::size_t take_lines(std::size_t pos) const = 0; | 			virtual std::size_t take_lines(std::size_t pos) const = 0; | ||||||
| 			virtual bool adjust_caret_into_screen() = 0; |  | ||||||
| 		}; | 		}; | ||||||
| 
 | 
 | ||||||
| 		inline bool is_right_text(const unicode_bidi::entity& e) | 		inline bool is_right_text(const unicode_bidi::entity& e) | ||||||
| @ -685,62 +685,6 @@ namespace nana{	namespace widgets | |||||||
| 			{ | 			{ | ||||||
| 				return 1; | 				return 1; | ||||||
| 			} | 			} | ||||||
| 
 |  | ||||||
| 			//adjust_caret_into_screen
 |  | ||||||
| 			//@brief:	Adjust the text offset in order to moving caret into visible area if it is out of the visible area
 |  | ||||||
| 			//@note:	the function assumes the points_.caret is correct
 |  | ||||||
| 			bool adjust_caret_into_screen() override |  | ||||||
| 			{ |  | ||||||
| 				const auto scrlines = editor_.screen_lines(); |  | ||||||
| 				if (0 == scrlines) |  | ||||||
| 					return false; |  | ||||||
| 
 |  | ||||||
| 				auto const pre_origin = editor_.impl_->cview->origin(); |  | ||||||
| 				auto origin = pre_origin; |  | ||||||
| 
 |  | ||||||
| 				auto& points = editor_.points_; |  | ||||||
| 				auto& textbase = editor_.textbase(); |  | ||||||
| 
 |  | ||||||
| 				auto&	lnstr = textbase.getline(points.caret.y); |  | ||||||
| 				const auto x = (std::min)(points.caret.x, static_cast<decltype(points.caret.x)>(lnstr.size())); |  | ||||||
| 				auto const text_w = editor_._m_pixels_by_char(lnstr, x); |  | ||||||
| 
 |  | ||||||
| 				auto area_w = editor_.impl_->cview->view_area().width; |  | ||||||
| 
 |  | ||||||
| 				if (static_cast<int>(text_w) < origin.x) |  | ||||||
| 				{ |  | ||||||
| 					auto delta_pixels = editor_._m_text_extent_size(L"    ", 4).width; |  | ||||||
| 					origin.x = (text_w > delta_pixels ? text_w - delta_pixels : 0); |  | ||||||
| 				} |  | ||||||
| 				else if (area_w && (text_w >= origin.x + area_w)) |  | ||||||
| 					origin.x = text_w - area_w + 2; |  | ||||||
| 
 |  | ||||||
| 				int row = origin.y / static_cast<int>(editor_.line_height()); |  | ||||||
| 				if (points.caret.y >= row + scrlines)	//implicit condition scrlines > 0
 |  | ||||||
| 				{ |  | ||||||
| 					row = static_cast<int>(points.caret.y - scrlines) + 1; |  | ||||||
| 				} |  | ||||||
| 				else if (static_cast<int>(points.caret.y) < row) |  | ||||||
| 				{ |  | ||||||
| 					if (scrlines >= static_cast<unsigned>(row)) |  | ||||||
| 						row = 0; |  | ||||||
| 					else |  | ||||||
| 						row = static_cast<int>(row - scrlines); |  | ||||||
| 				} |  | ||||||
| 				else if (row && (textbase.lines() <= scrlines)) |  | ||||||
| 					row = 0; |  | ||||||
| 
 |  | ||||||
| 				if(row != origin.y / static_cast<int>(editor_.line_height())) |  | ||||||
| 					origin.y = row * editor_.line_height(); |  | ||||||
| 
 |  | ||||||
| 				if (pre_origin != origin) |  | ||||||
| 				{ |  | ||||||
| 					editor_.impl_->cview->move_origin(origin - pre_origin); |  | ||||||
| 					editor_.impl_->cview->sync(true); |  | ||||||
| 					return true; |  | ||||||
| 				} |  | ||||||
| 				return false; |  | ||||||
| 			} |  | ||||||
| 		private: | 		private: | ||||||
| 			text_editor& editor_; | 			text_editor& editor_; | ||||||
| 			std::vector<text_section> sections_; | 			std::vector<text_section> sections_; | ||||||
| @ -773,9 +717,9 @@ namespace nana{	namespace widgets | |||||||
| 				if ((0 == editor_.textbase().lines()) || (0 == line_px)) | 				if ((0 == editor_.textbase().lines()) || (0 == line_px)) | ||||||
| 					return coord; | 					return coord; | ||||||
| 
 | 
 | ||||||
| 				auto screen_rows = (top - editor_.text_area_.area.y + editor_.impl_->cview->origin().y) / line_px; | 				auto text_row = (std::max)(0, (top - editor_.text_area_.area.y + editor_.impl_->cview->origin().y) / line_px); | ||||||
| 
 | 
 | ||||||
| 				coord = _m_textline(static_cast<std::size_t>(screen_rows)); | 				coord = _m_textline(static_cast<std::size_t>(text_row)); | ||||||
| 				if (linemtr_.size() <= coord.first) | 				if (linemtr_.size() <= coord.first) | ||||||
| 				{ | 				{ | ||||||
| 					coord.first = linemtr_.size() - 1; | 					coord.first = linemtr_.size() - 1; | ||||||
| @ -795,22 +739,26 @@ namespace nana{	namespace widgets | |||||||
| 					std::swap(first, second); | 					std::swap(first, second); | ||||||
| 
 | 
 | ||||||
| 				if (second < linemtr_.size()) | 				if (second < linemtr_.size()) | ||||||
| 					linemtr_.erase(linemtr_.begin() + first + 1, linemtr_.begin() + second); | 					linemtr_.erase(linemtr_.begin() + first + 1, linemtr_.begin() + second + 1); | ||||||
| 
 | 
 | ||||||
| 				pre_calc_line(first, editor_.width_pixels()); | 				auto const width_px = editor_.width_pixels(); | ||||||
| 
 | 
 | ||||||
|  | 				pre_calc_line(first, width_px); | ||||||
|  | 
 | ||||||
|  | 				/*
 | ||||||
| 				//textbase is implement by using deque, and the linemtr holds the text pointers
 | 				//textbase is implement by using deque, and the linemtr holds the text pointers
 | ||||||
| 				//If the textbase is changed, it will check the text pointers.
 | 				//If the textbase is changed, it will check the text pointers.
 | ||||||
| 				std::size_t line = 0; | 				std::size_t line = 0; | ||||||
| 				for (auto & mtr: linemtr_) | 				for (auto & mtr: linemtr_)	//deprecated
 | ||||||
| 				{ | 				{ | ||||||
| 					auto& linestr = editor_.textbase().getline(line); | 					auto& linestr = editor_.textbase().getline(line); | ||||||
| 					auto p = mtr.line_sections.front().begin; | 					auto p = mtr.line_sections.front().begin; | ||||||
| 					if (p < linestr.c_str() || (linestr.c_str() + linestr.size() < p)) | 					if (p < linestr.c_str() || (linestr.c_str() + linestr.size() < p)) | ||||||
| 						pre_calc_line(line, editor_.width_pixels()); | 						pre_calc_line(line, width_px); | ||||||
| 
 | 
 | ||||||
| 					++line; | 					++line; | ||||||
| 				} | 				} | ||||||
|  | 				*/ | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			void add_lines(std::size_t pos, std::size_t lines) override | 			void add_lines(std::size_t pos, std::size_t lines) override | ||||||
| @ -942,42 +890,6 @@ namespace nana{	namespace widgets | |||||||
| 			{ | 			{ | ||||||
| 				return (pos < linemtr_.size() ? linemtr_[pos].take_lines : 0); | 				return (pos < linemtr_.size() ? linemtr_[pos].take_lines : 0); | ||||||
| 			} | 			} | ||||||
| 
 |  | ||||||
| 			bool adjust_caret_into_screen() override |  | ||||||
| 			{ |  | ||||||
| 				const auto scrlines = editor_.screen_lines(); |  | ||||||
| 				if (0 == scrlines) |  | ||||||
| 					return false; |  | ||||||
| 
 |  | ||||||
| 				const auto & points = editor_.points_; |  | ||||||
| 				auto off_coord = _m_textline(static_cast<std::size_t>(editor_._m_text_topline())); |  | ||||||
| 
 |  | ||||||
| 				nana::upoint caret_secondary; |  | ||||||
| 				editor_._m_pos_secondary(points.caret, caret_secondary); |  | ||||||
| 
 |  | ||||||
| 				//Use the caret line for the offset line when caret is in front of current offset line.
 |  | ||||||
| 				if (off_coord.first > points.caret.y || ((off_coord.first == points.caret.y) && (off_coord.second > caret_secondary.y))) |  | ||||||
| 				{ |  | ||||||
| 					//Use the line which was specified by points.caret for the first line.
 |  | ||||||
| 					_m_set_offset_by_secondary(row_coordinate(points.caret.y, caret_secondary.y)); |  | ||||||
| 					return true; |  | ||||||
| 				} |  | ||||||
| 
 |  | ||||||
| 				//Find the last screen line. If the text does not reach the bottom of screen,
 |  | ||||||
| 				//do not adjust the offset line.
 |  | ||||||
| 				row_coordinate bottom; |  | ||||||
| 				if (false == _m_advance_secondary(off_coord, static_cast<int>(scrlines - 1), bottom)) |  | ||||||
| 					return false; |  | ||||||
| 
 |  | ||||||
| 				//Do not adjust the offset line if the caret line does not reach the bottom line.
 |  | ||||||
| 				if (points.caret.y < bottom.first || ((points.caret.y == bottom.first) && (caret_secondary.y <= bottom.second))) |  | ||||||
| 					return false; |  | ||||||
| 
 |  | ||||||
| 				_m_advance_secondary(row_coordinate(points.caret.y, caret_secondary.y), -static_cast<int>(scrlines - 1), bottom); |  | ||||||
| 
 |  | ||||||
| 				_m_set_offset_by_secondary(bottom); |  | ||||||
| 				return true; |  | ||||||
| 			} |  | ||||||
| 		private: | 		private: | ||||||
| 			void _m_text_section(const std::wstring& str, std::vector<text_section>& tsec) | 			void _m_text_section(const std::wstring& str, std::vector<text_section>& tsec) | ||||||
| 			{ | 			{ | ||||||
| @ -1014,90 +926,6 @@ namespace nana{	namespace widgets | |||||||
| 					tsec.emplace_back(word, end, unsigned{}); | 					tsec.emplace_back(word, end, unsigned{}); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			void _m_set_offset_by_secondary(row_coordinate row) |  | ||||||
| 			{ |  | ||||||
| 				for (auto i = linemtr_.begin(), end = linemtr_.begin() + row.first; i != end; ++i) |  | ||||||
| 					row.second += i->take_lines; |  | ||||||
| 
 |  | ||||||
| 				auto origin = editor_.impl_->cview->origin(); |  | ||||||
| 				origin.y = static_cast<int>(row.second * editor_.line_height()); |  | ||||||
| 				editor_.impl_->cview->move_origin(origin - editor_.impl_->cview->origin()); |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			bool _m_advance_secondary(row_coordinate row, int distance, row_coordinate& new_row) |  | ||||||
| 			{ |  | ||||||
| 				if ((row.first >= linemtr_.size()) || (row.second >= linemtr_[row.first].take_lines)) |  | ||||||
| 					return false; |  | ||||||
| 
 |  | ||||||
| 				if (0 == distance) |  | ||||||
| 				{ |  | ||||||
| 					new_row = row; |  | ||||||
| 					return true; |  | ||||||
| 				} |  | ||||||
| 
 |  | ||||||
| 				if (distance < 0) |  | ||||||
| 				{ |  | ||||||
| 					std::size_t n = static_cast<std::size_t>(-distance); |  | ||||||
| 
 |  | ||||||
| 					if (row.second > n) |  | ||||||
| 					{ |  | ||||||
| 						new_row.first = row.first; |  | ||||||
| 						new_row.second = row.second - n; |  | ||||||
| 						return true; |  | ||||||
| 					} |  | ||||||
| 
 |  | ||||||
| 					if (0 == row.first) |  | ||||||
| 						return false; |  | ||||||
| 
 |  | ||||||
| 					--row.first; |  | ||||||
| 					n -= (row.second + 1); |  | ||||||
| 
 |  | ||||||
| 					while (true) |  | ||||||
| 					{ |  | ||||||
| 						auto lines = linemtr_[row.first].take_lines; |  | ||||||
| 						if (lines >= n) |  | ||||||
| 						{ |  | ||||||
| 							new_row.first = row.first; |  | ||||||
| 							new_row.second = lines - n; |  | ||||||
| 							return true; |  | ||||||
| 						} |  | ||||||
| 						if (0 == row.first) |  | ||||||
| 							return false; |  | ||||||
| 
 |  | ||||||
| 						--row.first; |  | ||||||
| 						n -= lines; |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				else |  | ||||||
| 				{ |  | ||||||
| 					std::size_t n = static_cast<std::size_t>(distance); |  | ||||||
| 
 |  | ||||||
| 					auto delta_lines = linemtr_[row.first].take_lines - (row.second + 1); |  | ||||||
| 
 |  | ||||||
| 					if (delta_lines >= n) |  | ||||||
| 					{ |  | ||||||
| 						new_row.first = row.first; |  | ||||||
| 						new_row.second = row.second + n; |  | ||||||
| 						return true; |  | ||||||
| 					} |  | ||||||
| 
 |  | ||||||
| 					n -= delta_lines; |  | ||||||
| 
 |  | ||||||
| 					while (++row.first < linemtr_.size()) |  | ||||||
| 					{ |  | ||||||
| 						auto & mtr = linemtr_[row.first]; |  | ||||||
| 						if (mtr.take_lines >= n) |  | ||||||
| 						{ |  | ||||||
| 							new_row.first = row.first; |  | ||||||
| 							new_row.second = n - 1; |  | ||||||
| 							return true; |  | ||||||
| 						} |  | ||||||
| 						n -= mtr.take_lines; |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				return false; |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			row_coordinate _m_textline(std::size_t scrline) const | 			row_coordinate _m_textline(std::size_t scrline) const | ||||||
| 			{ | 			{ | ||||||
| 				row_coordinate coord; | 				row_coordinate coord; | ||||||
| @ -1236,6 +1064,12 @@ namespace nana{	namespace widgets | |||||||
| 				this->reset_caret(); | 				this->reset_caret(); | ||||||
| 			}; | 			}; | ||||||
| 
 | 
 | ||||||
|  | 			impl_->cview->events().hover_outside = [this](const point& pos) { | ||||||
|  | 				mouse_caret(pos, false); | ||||||
|  | 				if (selection::mode::mouse_selected == select_.mode_selection || selection::mode::method_selected == select_.mode_selection) | ||||||
|  | 					set_end_caret(false); | ||||||
|  | 			}; | ||||||
|  | 
 | ||||||
| 			API::create_caret(wd, { 1, line_height() }); | 			API::create_caret(wd, { 1, line_height() }); | ||||||
| 			API::bgcolor(wd, colors::white); | 			API::bgcolor(wd, colors::white); | ||||||
| 			API::fgcolor(wd, colors::black); | 			API::fgcolor(wd, colors::black); | ||||||
| @ -1249,6 +1083,11 @@ namespace nana{	namespace widgets | |||||||
| 			delete impl_; | 			delete impl_; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | 		size text_editor::caret_size() const | ||||||
|  | 		{ | ||||||
|  | 			return { 1, line_height() }; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		void text_editor::set_highlight(const std::string& name, const ::nana::color& fgcolor, const ::nana::color& bgcolor) | 		void text_editor::set_highlight(const std::string& name, const ::nana::color& fgcolor, const ::nana::color& bgcolor) | ||||||
| 		{ | 		{ | ||||||
| 			if (fgcolor.invisible() && bgcolor.invisible()) | 			if (fgcolor.invisible() && bgcolor.invisible()) | ||||||
| @ -1680,7 +1519,7 @@ namespace nana{	namespace widgets | |||||||
| 					API::set_capture(window_, true); | 					API::set_capture(window_, true); | ||||||
| 					text_area_.captured = true; | 					text_area_.captured = true; | ||||||
| 
 | 
 | ||||||
| 					if (this->hit_select_area(_m_screen_to_caret(arg.pos), true)) | 					if (this->hit_select_area(_m_coordinate_to_caret(arg.pos), true)) | ||||||
| 					{ | 					{ | ||||||
| 						//The selected of text can be moved only if it is editable
 | 						//The selected of text can be moved only if it is editable
 | ||||||
| 						if (attributes_.editable) | 						if (attributes_.editable) | ||||||
| @ -1732,7 +1571,7 @@ namespace nana{	namespace widgets | |||||||
| 						//no move occurs
 | 						//no move occurs
 | ||||||
| 						select(false); | 						select(false); | ||||||
| 
 | 
 | ||||||
| 						move_caret(_m_screen_to_caret(arg.pos)); | 						move_caret(_m_coordinate_to_caret(arg.pos)); | ||||||
| 					} | 					} | ||||||
| 
 | 
 | ||||||
| 					select_.mode_selection = selection::mode::no_selected; | 					select_.mode_selection = selection::mode::no_selected; | ||||||
| @ -1798,7 +1637,8 @@ namespace nana{	namespace widgets | |||||||
| 
 | 
 | ||||||
| 				if (graph_) | 				if (graph_) | ||||||
| 				{ | 				{ | ||||||
| 					impl_->capacities.behavior->adjust_caret_into_screen(); | 					this->_m_adjust_view(); | ||||||
|  | 
 | ||||||
| 					reset_caret(); | 					reset_caret(); | ||||||
| 					impl_->try_refresh = sync_graph::refresh; | 					impl_->try_refresh = sync_graph::refresh; | ||||||
| 					points_.xpos = 0; | 					points_.xpos = 0; | ||||||
| @ -1832,7 +1672,7 @@ namespace nana{	namespace widgets | |||||||
| 			const unsigned line_pixels = line_height(); | 			const unsigned line_pixels = line_height(); | ||||||
| 
 | 
 | ||||||
| 			//The coordinate of caret
 | 			//The coordinate of caret
 | ||||||
| 			auto coord = _m_caret_to_screen(crtpos); | 			auto coord = _m_caret_to_coordinate(crtpos); | ||||||
| 
 | 
 | ||||||
| 			const int line_bottom = coord.y + static_cast<int>(line_pixels); | 			const int line_bottom = coord.y + static_cast<int>(line_pixels); | ||||||
| 
 | 
 | ||||||
| @ -1863,7 +1703,7 @@ namespace nana{	namespace widgets | |||||||
| 			//Adjust the caret into screen when the caret position is modified by this function
 | 			//Adjust the caret into screen when the caret position is modified by this function
 | ||||||
| 			if (reset_caret && (!hit_text_area(coord))) | 			if (reset_caret && (!hit_text_area(coord))) | ||||||
| 			{ | 			{ | ||||||
| 				impl_->capacities.behavior->adjust_caret_into_screen(); | 				this->_m_adjust_view(); | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
| 				caret->visible(true); | 				caret->visible(true); | ||||||
| 				return true; | 				return true; | ||||||
| @ -1949,7 +1789,7 @@ namespace nana{	namespace widgets | |||||||
| 			select_.b = points_.caret; | 			select_.b = points_.caret; | ||||||
| 			points_.xpos = points_.caret.x; | 			points_.xpos = points_.caret.x; | ||||||
| 
 | 
 | ||||||
| 			if(new_sel_end || (stay_in_view && impl_->capacities.behavior->adjust_caret_into_screen())) | 			if (new_sel_end || (stay_in_view && this->_m_adjust_view())) | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -1985,7 +1825,7 @@ namespace nana{	namespace widgets | |||||||
| 			{ | 			{ | ||||||
| 				points_.caret = select_.b; | 				points_.caret = select_.b; | ||||||
| 
 | 
 | ||||||
| 				if (impl_->capacities.behavior->adjust_caret_into_screen()) | 				if (this->_m_adjust_view()) | ||||||
| 					impl_->try_refresh = sync_graph::refresh; | 					impl_->try_refresh = sync_graph::refresh; | ||||||
| 
 | 
 | ||||||
| 				reset_caret(); | 				reset_caret(); | ||||||
| @ -1994,7 +1834,7 @@ namespace nana{	namespace widgets | |||||||
| 
 | 
 | ||||||
| 			if (_m_move_select(true)) | 			if (_m_move_select(true)) | ||||||
| 			{ | 			{ | ||||||
| 				impl_->capacities.behavior->adjust_caret_into_screen(); | 				this->_m_adjust_view(); | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
| 				return true; | 				return true; | ||||||
| 			} | 			} | ||||||
| @ -2117,7 +1957,8 @@ namespace nana{	namespace widgets | |||||||
| 
 | 
 | ||||||
| 			if(graph_) | 			if(graph_) | ||||||
| 			{ | 			{ | ||||||
| 				impl_->capacities.behavior->adjust_caret_into_screen(); | 				this->_m_adjust_view(); | ||||||
|  | 
 | ||||||
| 				reset_caret(); | 				reset_caret(); | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
| 				_m_reset_content_size(true); | 				_m_reset_content_size(true); | ||||||
| @ -2269,7 +2110,8 @@ namespace nana{	namespace widgets | |||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			impl_->cview->move_origin(origin - impl_->cview->origin()); | 			impl_->cview->move_origin(origin - impl_->cview->origin()); | ||||||
| 			if (impl_->capacities.behavior->adjust_caret_into_screen() || need_refresh) | 
 | ||||||
|  | 			if (this->_m_adjust_view() || need_refresh) | ||||||
| 				impl_->cview->sync(true); | 				impl_->cview->sync(true); | ||||||
| 
 | 
 | ||||||
| 			_m_reset_content_size(); | 			_m_reset_content_size(); | ||||||
| @ -2320,7 +2162,7 @@ namespace nana{	namespace widgets | |||||||
| 					textbase.erase(points_.caret.y, points_.caret.x, erase_number); | 					textbase.erase(points_.caret.y, points_.caret.x, erase_number); | ||||||
| 					_m_pre_calc_lines(points_.caret.y, 1); | 					_m_pre_calc_lines(points_.caret.y, 1); | ||||||
| 
 | 
 | ||||||
| 					if(_m_move_offset_x_while_over_border(-2) == false) | 					if (!this->_m_adjust_view()) | ||||||
| 					{ | 					{ | ||||||
| 						_m_update_line(points_.caret.y, secondary); | 						_m_update_line(points_.caret.y, secondary); | ||||||
| 						has_to_redraw = false; | 						has_to_redraw = false; | ||||||
| @ -2349,11 +2191,11 @@ namespace nana{	namespace widgets | |||||||
| 			if (record_undo) | 			if (record_undo) | ||||||
| 				impl_->undo.push(std::move(undo_ptr)); | 				impl_->undo.push(std::move(undo_ptr)); | ||||||
| 
 | 
 | ||||||
| 			_m_reset_content_size(has_to_redraw); | 			_m_reset_content_size(false); | ||||||
| 
 | 
 | ||||||
| 			if(has_to_redraw) | 			if(has_to_redraw) | ||||||
| 			{ | 			{ | ||||||
| 				impl_->capacities.behavior->adjust_caret_into_screen(); | 				this->_m_adjust_view(); | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| @ -2367,7 +2209,7 @@ namespace nana{	namespace widgets | |||||||
| 
 | 
 | ||||||
| 			_m_reset_content_size(true); | 			_m_reset_content_size(true); | ||||||
| 
 | 
 | ||||||
| 			impl_->capacities.behavior->adjust_caret_into_screen(); | 			this->_m_adjust_view(); | ||||||
| 			impl_->try_refresh = sync_graph::refresh; | 			impl_->try_refresh = sync_graph::refresh; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -2391,10 +2233,8 @@ namespace nana{	namespace widgets | |||||||
| 				if(points_.caret.x) | 				if(points_.caret.x) | ||||||
| 				{ | 				{ | ||||||
| 					--points_.caret.x; | 					--points_.caret.x; | ||||||
| 
 |  | ||||||
| 					pending = false; | 					pending = false; | ||||||
| 					bool adjust_y = (attributes_.line_wrapped && impl_->capacities.behavior->adjust_caret_into_screen()); | 					if (this->_m_adjust_view()) | ||||||
| 					if (_m_move_offset_x_while_over_border(-2) || adjust_y) |  | ||||||
| 						impl_->try_refresh = sync_graph::refresh; | 						impl_->try_refresh = sync_graph::refresh; | ||||||
| 				} | 				} | ||||||
| 				else if (points_.caret.y) //Move to previous line
 | 				else if (points_.caret.y) //Move to previous line
 | ||||||
| @ -2403,7 +2243,7 @@ namespace nana{	namespace widgets | |||||||
| 					pending = false; | 					pending = false; | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			if (pending && impl_->capacities.behavior->adjust_caret_into_screen()) | 			if (pending && this->_m_adjust_view()) | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
| 
 | 
 | ||||||
| 			points_.xpos = points_.caret.x; | 			points_.xpos = points_.caret.x; | ||||||
| @ -2412,25 +2252,23 @@ namespace nana{	namespace widgets | |||||||
| 		void text_editor::move_right() | 		void text_editor::move_right() | ||||||
| 		{ | 		{ | ||||||
| 			bool do_render = false; | 			bool do_render = false; | ||||||
| 			if(_m_cancel_select(2) == false) | 			if (_m_cancel_select(2) == false) | ||||||
| 			{ | 			{ | ||||||
| 				auto lnstr = textbase().getline(points_.caret.y); | 				auto lnstr = textbase().getline(points_.caret.y); | ||||||
| 				if(lnstr.size() > points_.caret.x) | 				if (lnstr.size() > points_.caret.x) | ||||||
| 				{ | 				{ | ||||||
| 					++points_.caret.x; | 					++points_.caret.x; | ||||||
| 
 | 					do_render = this->_m_adjust_view(); | ||||||
| 					bool adjust_y = (attributes_.line_wrapped && impl_->capacities.behavior->adjust_caret_into_screen()); |  | ||||||
| 					do_render = (_m_move_offset_x_while_over_border(2) || adjust_y); |  | ||||||
| 				} | 				} | ||||||
| 				else if(points_.caret.y + 1 < textbase().lines()) | 				else if (points_.caret.y + 1 < textbase().lines()) | ||||||
| 				{	//Move to next line
 | 				{	//Move to next line
 | ||||||
| 					points_.caret.x = 0; | 					points_.caret.x = 0; | ||||||
| 					++ points_.caret.y; | 					++points_.caret.y; | ||||||
| 					do_render = impl_->capacities.behavior->adjust_caret_into_screen(); | 					do_render = this->_m_adjust_view(); | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			else | 			else | ||||||
| 				do_render = impl_->capacities.behavior->adjust_caret_into_screen(); | 				do_render = this->_m_adjust_view(); | ||||||
| 
 | 
 | ||||||
| 			if (do_render) | 			if (do_render) | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
| @ -2444,155 +2282,123 @@ namespace nana{	namespace widgets | |||||||
| 				select_.a = select_.b = points_.caret; | 				select_.a = select_.b = points_.caret; | ||||||
| 
 | 
 | ||||||
| 			auto origin = impl_->cview->origin(); | 			auto origin = impl_->cview->origin(); | ||||||
| 			origin.y = _m_text_topline(); | 			auto pos = points_.caret; | ||||||
|  | 			auto coord = _m_caret_to_coordinate(points_.caret, false); | ||||||
| 
 | 
 | ||||||
| 			bool changed = false; |  | ||||||
| 			nana::upoint caret = points_.caret; |  | ||||||
| 			wchar_t key = arg.key; | 			wchar_t key = arg.key; | ||||||
| 			size_t nlines = textbase().lines(); | 
 | ||||||
| 			if (arg.ctrl) { | 			auto const line_px = this->line_height(); | ||||||
| 				switch (key) { | 
 | ||||||
| 				case keyboard::os_arrow_left: | 			//The number of text lines
 | ||||||
| 				case keyboard::os_arrow_right: | 			auto const line_count = textbase().lines(); | ||||||
| 					// TODO: move the caret word by word
 | 
 | ||||||
| 					break; | 			//The number of charecters in the line of caret
 | ||||||
| 				case keyboard::os_home: | 			auto const text_length = textbase().getline(points_.caret.y).size(); | ||||||
| 					if (caret.y != 0) { | 
 | ||||||
| 						caret.y = 0; | 			 | ||||||
| 						origin.y = 0; |  | ||||||
| 						changed = true; |  | ||||||
| 					} |  | ||||||
| 					break; |  | ||||||
| 				case keyboard::os_end: |  | ||||||
| 					if (caret.y != nlines - 1) { |  | ||||||
| 						caret.y = static_cast<decltype(caret.y)>(nlines - 1); |  | ||||||
| 						changed = true; |  | ||||||
| 					} |  | ||||||
| 					break; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			size_t lnsz = textbase().getline(caret.y).size(); |  | ||||||
| 			switch (key) { | 			switch (key) { | ||||||
| 			case keyboard::os_arrow_left: | 			case keyboard::os_arrow_left: | ||||||
| 				if (select_.move_to_end && (select_.a != select_.b) && (!arg.shift)) | 				if (select_.move_to_end && (select_.a != select_.b) && (!arg.shift)) | ||||||
| 				{ | 				{ | ||||||
| 					caret = select_.a; | 					pos = select_.a; | ||||||
| 					changed = true; |  | ||||||
| 				} | 				} | ||||||
| 				else | 				else if (pos.x != 0) | ||||||
| 				{ | 				{ | ||||||
| 					if (caret.x != 0) { | 					--pos.x; | ||||||
| 						--caret.x; |  | ||||||
| 						changed = true; |  | ||||||
| 					} |  | ||||||
| 					else { |  | ||||||
| 						if (caret.y != 0) { |  | ||||||
| 							--caret.y; |  | ||||||
| 							caret.x = static_cast<decltype(caret.x)>(textbase().getline(caret.y).size()); |  | ||||||
| 							changed = true; |  | ||||||
| 						} |  | ||||||
| 				} | 				} | ||||||
|  | 				else if (pos.y != 0) { | ||||||
|  | 					--pos.y; | ||||||
|  | 					pos.x = static_cast<decltype(pos.x)>(textbase().getline(pos.y).size()); | ||||||
| 				} | 				} | ||||||
| 				break; | 				break; | ||||||
| 			case keyboard::os_arrow_right: | 			case keyboard::os_arrow_right: | ||||||
| 				if (select_.move_to_end && (select_.a != select_.b) && (!arg.shift)) | 				if (select_.move_to_end && (select_.a != select_.b) && (!arg.shift)) | ||||||
| 				{ | 				{ | ||||||
| 					caret = select_.b; | 					pos = select_.b; | ||||||
| 					changed = true; |  | ||||||
| 				} | 				} | ||||||
| 				else | 				else if (pos.x < text_length) | ||||||
| 				{ | 				{ | ||||||
| 					if (caret.x < lnsz) { | 					++pos.x; | ||||||
| 						++caret.x; |  | ||||||
| 						changed = true; |  | ||||||
| 					} |  | ||||||
| 					else { |  | ||||||
| 						if (caret.y != nlines - 1) { |  | ||||||
| 							++caret.y; |  | ||||||
| 							caret.x = 0; |  | ||||||
| 							changed = true; |  | ||||||
| 						} |  | ||||||
| 				} | 				} | ||||||
|  | 				else if (pos.y != line_count - 1) | ||||||
|  | 				{ | ||||||
|  | 					++pos.y; | ||||||
|  | 					pos.x = 0; | ||||||
| 				} | 				} | ||||||
| 				break; | 				break; | ||||||
| 			case keyboard::os_arrow_up: | 			case keyboard::os_arrow_up: | ||||||
|  | 				coord.y -= static_cast<int>(line_px); | ||||||
|  | 				break; | ||||||
| 			case keyboard::os_arrow_down: | 			case keyboard::os_arrow_down: | ||||||
| 				{ | 				coord.y += static_cast<int>(line_px); | ||||||
| 					auto screen_pt = _m_caret_to_screen(caret); |  | ||||||
| 					int offset = line_height(); |  | ||||||
| 					if (key == keyboard::os_arrow_up) { |  | ||||||
| 						offset = -offset; |  | ||||||
| 					} |  | ||||||
| 					screen_pt.y += offset; |  | ||||||
| 
 |  | ||||||
| 					auto const new_caret = _m_screen_to_caret(screen_pt); |  | ||||||
| 					if (new_caret != caret) { |  | ||||||
| 						caret = new_caret; |  | ||||||
| 						if (screen_pt.y < 0) { |  | ||||||
| 							scroll(true, true); |  | ||||||
| 						} |  | ||||||
| 						changed = true; |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				break; | 				break; | ||||||
| 			case keyboard::os_home: | 			case keyboard::os_home: | ||||||
| 				if (caret.x != 0) { | 				//move the caret to the begining of the line
 | ||||||
| 					caret.x = 0; | 				pos.x = 0; | ||||||
| 					changed = true; | 
 | ||||||
| 				} | 				//move the caret to the begining of the text if Ctrl is pressed
 | ||||||
|  | 				if (arg.ctrl) | ||||||
|  | 					pos.y = 0; | ||||||
| 				break; | 				break; | ||||||
| 			case keyboard::os_end: | 			case keyboard::os_end: | ||||||
| 				if (caret.x < lnsz) { | 				//move the caret to the end of the line
 | ||||||
| 					caret.x = static_cast<decltype(caret.x)>(lnsz); | 				pos.x = static_cast<decltype(pos.x)>(text_length); | ||||||
| 					changed = true; | 
 | ||||||
| 				} | 				//move the caret to the end of the text if Ctrl is pressed
 | ||||||
|  | 				if(arg.ctrl) | ||||||
|  | 					pos.y = (line_count - 1) * line_px; | ||||||
| 				break; | 				break; | ||||||
| 			case keyboard::os_pageup: | 			case keyboard::os_pageup: | ||||||
| 				if (caret.y >= screen_lines() && origin.y >= static_cast<int>(screen_lines())) { | 				if(origin.y > 0) | ||||||
| 					origin.y -= screen_lines(); | 				{ | ||||||
| 					caret.y -= screen_lines(); | 					auto off = coord - origin; | ||||||
| 					changed = true; | 					origin.y -= (std::min)(origin.y, static_cast<int>(impl_->cview->view_area().height)); | ||||||
|  | 					coord = off + origin; | ||||||
| 				} | 				} | ||||||
| 				break; | 				break; | ||||||
| 			case keyboard::os_pagedown: | 			case keyboard::os_pagedown: | ||||||
| 				if (caret.y + screen_lines() <= impl_->capacities.behavior->take_lines()) { | 				if (impl_->cview->content_size().height > impl_->cview->view_area().height) | ||||||
| 					origin.y += static_cast<int>(screen_lines()); | 				{ | ||||||
| 					caret.y += screen_lines(); | 					auto off = coord - origin; | ||||||
| 					changed = true; | 					origin.y = (std::min)(origin.y + static_cast<int>(impl_->cview->view_area().height), static_cast<int>(impl_->cview->content_size().height - impl_->cview->view_area().height)); | ||||||
|  | 					coord = off + origin; | ||||||
| 				} | 				} | ||||||
| 				break; | 				break; | ||||||
| 			} | 			} | ||||||
| 			if (select_.a != caret || select_.b != caret) { | 
 | ||||||
| 				changed = true; | 			if (pos == points_.caret) | ||||||
|  | 			{ | ||||||
|  | 				impl_->cview->move_origin(origin - impl_->cview->origin()); | ||||||
|  | 				pos = _m_coordinate_to_caret(coord, false); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			if (changed) { | 			if (pos != points_.caret) { | ||||||
| 				if (arg.shift) { | 				if (arg.shift) { | ||||||
| 					switch (key) { | 					switch (key) { | ||||||
| 					case keyboard::os_arrow_left: | 					case keyboard::os_arrow_left: | ||||||
| 					case keyboard::os_arrow_up: | 					case keyboard::os_arrow_up: | ||||||
| 					case keyboard::os_home: | 					case keyboard::os_home: | ||||||
| 					case keyboard::os_pageup: | 					case keyboard::os_pageup: | ||||||
| 						select_.b = caret; | 						select_.b = pos; | ||||||
| 						break; | 						break; | ||||||
| 					case keyboard::os_arrow_right: | 					case keyboard::os_arrow_right: | ||||||
| 					case keyboard::os_arrow_down: | 					case keyboard::os_arrow_down: | ||||||
| 					case keyboard::os_end: | 					case keyboard::os_end: | ||||||
| 					case keyboard::os_pagedown: | 					case keyboard::os_pagedown: | ||||||
| 						select_.b = caret; | 						select_.b = pos; | ||||||
| 						break; | 						break; | ||||||
| 					} | 					} | ||||||
| 				}else { |  | ||||||
| 					select_.b = caret; |  | ||||||
| 					select_.a = caret; |  | ||||||
| 				} | 				} | ||||||
| 				points_.caret = caret; | 				else { | ||||||
| 
 | 					select_.b = pos; | ||||||
| 				origin.y *= line_height(); | 					select_.a = pos; | ||||||
| 				impl_->cview->move_origin(origin - impl_->cview->origin()); | 				} | ||||||
| 				impl_->cview->sync(true); | 				points_.caret = pos; | ||||||
| 				points_.xpos = points_.caret.x; | 				points_.xpos = points_.caret.x; | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
|  | 				this->_m_adjust_view(); | ||||||
|  | 				impl_->cview->sync(true); | ||||||
|  | 				this->reset_caret(); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -2633,9 +2439,9 @@ namespace nana{	namespace widgets | |||||||
| 
 | 
 | ||||||
| 		const upoint& text_editor::mouse_caret(const point& scrpos, bool stay_in_view)	//From screen position
 | 		const upoint& text_editor::mouse_caret(const point& scrpos, bool stay_in_view)	//From screen position
 | ||||||
| 		{ | 		{ | ||||||
| 			points_.caret = _m_screen_to_caret(scrpos); | 			points_.caret = _m_coordinate_to_caret(scrpos); | ||||||
| 
 | 
 | ||||||
| 			if (stay_in_view && impl_->capacities.behavior->adjust_caret_into_screen()) | 			if (stay_in_view && this->_m_adjust_view()) | ||||||
| 				impl_->try_refresh = sync_graph::refresh; | 				impl_->try_refresh = sync_graph::refresh; | ||||||
| 
 | 
 | ||||||
| 			move_caret(points_.caret); | 			move_caret(points_.caret); | ||||||
| @ -2649,7 +2455,7 @@ namespace nana{	namespace widgets | |||||||
| 
 | 
 | ||||||
| 		point text_editor::caret_screen_pos() const | 		point text_editor::caret_screen_pos() const | ||||||
| 		{ | 		{ | ||||||
| 			return _m_caret_to_screen(points_.caret); | 			return _m_caret_to_coordinate(points_.caret); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		bool text_editor::scroll(bool upwards, bool vert) | 		bool text_editor::scroll(bool upwards, bool vert) | ||||||
| @ -2667,7 +2473,7 @@ namespace nana{	namespace widgets | |||||||
| 				{ | 				{ | ||||||
| 					auto const height = line_height(); | 					auto const height = line_height(); | ||||||
| 
 | 
 | ||||||
| 					auto top = _m_caret_to_screen(upoint{ 0, static_cast<unsigned>(row.first) }).y; | 					auto top = _m_caret_to_coordinate(upoint{ 0, static_cast<unsigned>(row.first) }).y; | ||||||
| 					std::size_t lines = 1; | 					std::size_t lines = 1; | ||||||
| 
 | 
 | ||||||
| 					if (whole_line) | 					if (whole_line) | ||||||
| @ -2751,7 +2557,7 @@ namespace nana{	namespace widgets | |||||||
| 				this->impl_->capacities.behavior->pre_calc_line(pos, width_px); | 				this->impl_->capacities.behavior->pre_calc_line(pos, width_px); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		nana::point	text_editor::_m_caret_to_screen(nana::upoint pos) const | 		nana::point	text_editor::_m_caret_to_coordinate(nana::upoint pos, bool to_screen_coordinate) const | ||||||
| 		{ | 		{ | ||||||
| 			auto const behavior = impl_->capacities.behavior; | 			auto const behavior = impl_->capacities.behavior; | ||||||
| 			auto const sections = behavior->line(pos.y); | 			auto const sections = behavior->line(pos.y); | ||||||
| @ -2759,7 +2565,7 @@ namespace nana{	namespace widgets | |||||||
| 			std::size_t lines = 0;	//lines before the caret line;
 | 			std::size_t lines = 0;	//lines before the caret line;
 | ||||||
| 			for (std::size_t i = 0; i < pos.y; ++i) | 			for (std::size_t i = 0; i < pos.y; ++i) | ||||||
| 			{ | 			{ | ||||||
| 				lines += behavior->line(i).size(); | 				lines += behavior->take_lines(i); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			const text_section * sct_ptr = nullptr; | 			const text_section * sct_ptr = nullptr; | ||||||
| @ -2767,6 +2573,8 @@ namespace nana{	namespace widgets | |||||||
| 			if (0 != pos.x) | 			if (0 != pos.x) | ||||||
| 			{ | 			{ | ||||||
| 				std::wstring str; | 				std::wstring str; | ||||||
|  | 
 | ||||||
|  | 				std::size_t sct_pos = 0; | ||||||
| 				for (auto & sct : sections) | 				for (auto & sct : sections) | ||||||
| 				{ | 				{ | ||||||
| 					std::size_t chsize = sct.end - sct.begin; | 					std::size_t chsize = sct.end - sct.begin; | ||||||
| @ -2776,7 +2584,9 @@ namespace nana{	namespace widgets | |||||||
| 					else | 					else | ||||||
| 						str.append(sct.begin, sct.end); | 						str.append(sct.begin, sct.end); | ||||||
| 
 | 
 | ||||||
| 					if (pos.x <= chsize) | 					//In line-wrapped mode. If the caret is at the end of a line which is not the last section,
 | ||||||
|  | 					//the caret should be moved to the beginning of next section line.
 | ||||||
|  | 					if ((sct_pos + 1 < sections.size()) ? (pos.x < chsize) : (pos.x <= chsize)) | ||||||
| 					{ | 					{ | ||||||
| 						sct_ptr = &sct; | 						sct_ptr = &sct; | ||||||
| 						if (pos.x == chsize) | 						if (pos.x == chsize) | ||||||
| @ -2790,6 +2600,8 @@ namespace nana{	namespace widgets | |||||||
| 						pos.x -= static_cast<unsigned>(chsize); | 						pos.x -= static_cast<unsigned>(chsize); | ||||||
| 						++lines; | 						++lines; | ||||||
| 					} | 					} | ||||||
|  | 
 | ||||||
|  | 					++sct_pos; | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| @ -2803,13 +2615,26 @@ namespace nana{	namespace widgets | |||||||
| 			else | 			else | ||||||
| 				scrpos.x += _m_text_x(*sct_ptr); | 				scrpos.x += _m_text_x(*sct_ptr); | ||||||
| 
 | 
 | ||||||
|  | 			if (!to_screen_coordinate) | ||||||
|  | 			{ | ||||||
|  | 				scrpos.y = static_cast<int>(lines * line_height()); | ||||||
|  | 				//_m_text_x includes origin x and text_area x. remove these factors
 | ||||||
|  | 				scrpos.x += (impl_->cview->origin().x - text_area_.area.x); | ||||||
|  | 
 | ||||||
|  | 			} | ||||||
|  | 			else | ||||||
| 				scrpos.y = static_cast<int>(lines * line_height()) - impl_->cview->origin().y + this->_m_text_top_base(); | 				scrpos.y = static_cast<int>(lines * line_height()) - impl_->cview->origin().y + this->_m_text_top_base(); | ||||||
|  | 
 | ||||||
| 			return scrpos; | 			return scrpos; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		upoint text_editor::_m_screen_to_caret(point scrpos) const | 		upoint text_editor::_m_coordinate_to_caret(point scrpos, bool from_screen_coordinate) const | ||||||
| 		{ | 		{ | ||||||
|  | 			if (!from_screen_coordinate) | ||||||
|  | 				scrpos -= (impl_->cview->origin() - point{ text_area_.area.x, this->_m_text_top_base() }); | ||||||
|  | 
 | ||||||
| 			auto const behavior = impl_->capacities.behavior; | 			auto const behavior = impl_->capacities.behavior; | ||||||
|  | 
 | ||||||
| 			auto const row = behavior->text_position_from_screen(scrpos.y); | 			auto const row = behavior->text_position_from_screen(scrpos.y); | ||||||
| 
 | 
 | ||||||
| 			auto sections = behavior->line(row.first); | 			auto sections = behavior->line(row.first); | ||||||
| @ -2833,10 +2658,8 @@ namespace nana{	namespace widgets | |||||||
| 			unicode_bidi{}.linestr(text_ptr, text_size, reordered); | 			unicode_bidi{}.linestr(text_ptr, text_size, reordered); | ||||||
| 
 | 
 | ||||||
| 			nana::upoint res(static_cast<unsigned>(real_str.begin - sections.front().begin), static_cast<unsigned>(row.first)); | 			nana::upoint res(static_cast<unsigned>(real_str.begin - sections.front().begin), static_cast<unsigned>(row.first)); | ||||||
| 			scrpos.x -= _m_text_x(sections[row.second]); |  | ||||||
| 
 | 
 | ||||||
| 			if (scrpos.x < 0) | 			scrpos.x = (std::max)(0, (scrpos.x - _m_text_x(sections[row.second]))); | ||||||
| 				scrpos.x = 0; |  | ||||||
| 
 | 
 | ||||||
| 			for (auto & ent : reordered) | 			for (auto & ent : reordered) | ||||||
| 			{ | 			{ | ||||||
| @ -2849,7 +2672,12 @@ namespace nana{	namespace widgets | |||||||
| 				} | 				} | ||||||
| 				scrpos.x -= str_px; | 				scrpos.x -= str_px; | ||||||
| 			} | 			} | ||||||
| 			res.x = static_cast<unsigned>(textbase().getline(res.y).size()); | 
 | ||||||
|  | 			//move the caret to the end of this section.
 | ||||||
|  | 			res.x = text_size; | ||||||
|  | 			for (std::size_t i = 0; i < row.second; ++i) | ||||||
|  | 				res.x += static_cast<int>(sections[i].end - sections[i].begin); | ||||||
|  | 
 | ||||||
| 			return res; | 			return res; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -2939,7 +2767,7 @@ namespace nana{	namespace widgets | |||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			_m_pos_from_secondary(points_.caret.y, secondary_pos, points_.caret.x); | 			_m_pos_from_secondary(points_.caret.y, secondary_pos, points_.caret.x); | ||||||
| 			return behavior->adjust_caret_into_screen(); | 			return this->_m_adjust_view(); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		void text_editor::_m_update_line(std::size_t pos, std::size_t secondary_count_before) | 		void text_editor::_m_update_line(std::size_t pos, std::size_t secondary_count_before) | ||||||
| @ -2947,7 +2775,7 @@ namespace nana{	namespace widgets | |||||||
| 			auto behavior = impl_->capacities.behavior; | 			auto behavior = impl_->capacities.behavior; | ||||||
| 			if (behavior->take_lines(pos) == secondary_count_before) | 			if (behavior->take_lines(pos) == secondary_count_before) | ||||||
| 			{ | 			{ | ||||||
| 				auto top = _m_caret_to_screen(upoint{ 0, static_cast<unsigned>(pos) }).y; | 				auto top = _m_caret_to_coordinate(upoint{ 0, static_cast<unsigned>(pos) }).y; | ||||||
| 
 | 
 | ||||||
| 				const unsigned pixels = line_height(); | 				const unsigned pixels = line_height(); | ||||||
| 				const rectangle update_area = { text_area_.area.x, top, width_pixels(), static_cast<unsigned>(pixels * secondary_count_before) }; | 				const rectangle update_area = { text_area_.area.x, top, width_pixels(), static_cast<unsigned>(pixels * secondary_count_before) }; | ||||||
| @ -3009,13 +2837,13 @@ namespace nana{	namespace widgets | |||||||
| 			size csize; | 			size csize; | ||||||
| 
 | 
 | ||||||
| 			if (this->attributes_.line_wrapped) | 			if (this->attributes_.line_wrapped) | ||||||
| 			{ |  | ||||||
| 				if (calc_lines) |  | ||||||
| 			{ | 			{ | ||||||
| 				//detect if vertical scrollbar is required
 | 				//detect if vertical scrollbar is required
 | ||||||
| 				auto const max_lines = screen_lines(true); | 				auto const max_lines = screen_lines(true); | ||||||
| 					auto text_lines = textbase().lines(); |  | ||||||
| 
 | 
 | ||||||
|  | 				if (calc_lines) | ||||||
|  | 				{ | ||||||
|  | 					auto text_lines = textbase().lines(); | ||||||
| 					if (text_lines <= max_lines) | 					if (text_lines <= max_lines) | ||||||
| 					{ | 					{ | ||||||
| 						std::size_t lines = 0; | 						std::size_t lines = 0; | ||||||
| @ -3034,19 +2862,20 @@ namespace nana{	namespace widgets | |||||||
| 
 | 
 | ||||||
| 					//enable vertical scrollbar when text_lines > max_lines
 | 					//enable vertical scrollbar when text_lines > max_lines
 | ||||||
| 					csize.width = _m_width_px(text_lines <= max_lines); | 					csize.width = _m_width_px(text_lines <= max_lines); | ||||||
| 
 |  | ||||||
| 					impl_->capacities.behavior->pre_calc_lines(csize.width); | 					impl_->capacities.behavior->pre_calc_lines(csize.width); | ||||||
| 				} | 				} | ||||||
| 				else | 				else | ||||||
|  | 				{ | ||||||
| 					csize.width = impl_->cview->content_size().width; | 					csize.width = impl_->cview->content_size().width; | ||||||
| 				} | 				} | ||||||
|  | 			} | ||||||
| 			else | 			else | ||||||
| 			{ | 			{ | ||||||
| 				if (calc_lines) | 				if (calc_lines) | ||||||
| 					impl_->capacities.behavior->pre_calc_lines(0); | 					impl_->capacities.behavior->pre_calc_lines(0); | ||||||
| 
 | 
 | ||||||
| 				auto maxline = textbase().max_line(); | 				auto maxline = textbase().max_line(); | ||||||
| 				csize.width = _m_text_extent_size(textbase().getline(maxline.first).c_str(), maxline.second).width; | 				csize.width = _m_text_extent_size(textbase().getline(maxline.first).c_str(), maxline.second).width + caret_size().width; | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			csize.height = static_cast<unsigned>(impl_->capacities.behavior->take_lines() * line_height()); | 			csize.height = static_cast<unsigned>(impl_->capacities.behavior->take_lines() * line_height()); | ||||||
| @ -3237,11 +3066,13 @@ namespace nana{	namespace widgets | |||||||
| 				{ | 				{ | ||||||
| 				case 1: | 				case 1: | ||||||
| 					points_.caret = a; | 					points_.caret = a; | ||||||
| 					_m_move_offset_x_while_over_border(-2); | 					//_m_move_offset_x_while_over_border(-2);	//deprecated
 | ||||||
|  | 					this->_m_adjust_view(); | ||||||
| 					break; | 					break; | ||||||
| 				case 2: | 				case 2: | ||||||
| 					points_.caret = b; | 					points_.caret = b; | ||||||
| 					_m_move_offset_x_while_over_border(2); | 					//_m_move_offset_x_while_over_border(2);	//deprecated
 | ||||||
|  | 					this->_m_adjust_view(); | ||||||
| 					break; | 					break; | ||||||
| 				} | 				} | ||||||
| 				select_.a = select_.b = points_.caret; | 				select_.a = select_.b = points_.caret; | ||||||
| @ -3271,50 +3102,63 @@ namespace nana{	namespace widgets | |||||||
| 			return graph_.text_extent_size(str, static_cast<unsigned>(n)); | 			return graph_.text_extent_size(str, static_cast<unsigned>(n)); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		//_m_move_offset_x_while_over_border
 | 		bool text_editor::_m_adjust_view() | ||||||
| 		//@brief: Move the view window
 |  | ||||||
| 		bool text_editor::_m_move_offset_x_while_over_border(int many) |  | ||||||
| 		{ | 		{ | ||||||
| 			//x never beyonds border in line-wrapped mode.
 | 			auto const view_area = impl_->cview->view_area(); | ||||||
| 			if (attributes_.line_wrapped || (0 == many)) | 
 | ||||||
|  | 			auto const line_px = static_cast<int>(this->line_height()); | ||||||
|  | 			auto coord = _m_caret_to_coordinate(points_.caret, true); | ||||||
|  | 
 | ||||||
|  | 			if (view_area.is_hit(coord) && view_area.is_hit({coord.x, coord.y + line_px})) | ||||||
| 				return false; | 				return false; | ||||||
| 
 | 
 | ||||||
| 			const string_type& lnstr = textbase().getline(points_.caret.y); | 			unsigned extra_count_horz = 4; | ||||||
| 			unsigned width = _m_text_extent_size(lnstr.c_str(), points_.caret.x).width; | 			unsigned extra_count_vert = 0; | ||||||
| 
 | 
 | ||||||
| 			const auto count = static_cast<unsigned>(std::abs(many)); | 			auto const origin = impl_->cview->origin(); | ||||||
| 			if(many < 0) | 			coord = _m_caret_to_coordinate(points_.caret, false); | ||||||
|  | 
 | ||||||
|  | 			point moved_origin; | ||||||
|  | 
 | ||||||
|  | 			//adjust x-axis if it isn't line-wrapped mode
 | ||||||
|  | 			if (!attributes_.line_wrapped) | ||||||
| 			{ | 			{ | ||||||
| 				auto origin = impl_->cview->origin(); | 				auto extra = points_.caret; | ||||||
| 
 | 
 | ||||||
| 				if (origin.x && (origin.x >= static_cast<int>(width))) | 				if (coord.x < origin.x) | ||||||
| 				{	//Out of screen text area
 |  | ||||||
| 					if (points_.caret.x > count) |  | ||||||
| 						origin.x = static_cast<int>(width - _m_text_extent_size(lnstr.c_str() + points_.caret.x - count, count).width); |  | ||||||
| 					else |  | ||||||
| 						origin.x = 0; |  | ||||||
| 
 |  | ||||||
| 					impl_->cview->move_origin(origin - impl_->cview->origin()); |  | ||||||
| 					return true; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			else |  | ||||||
| 				{ | 				{ | ||||||
| 				auto const right_pos = impl_->cview->view_area().right(); | 					extra.x -= (std::min)(extra_count_horz, points_.caret.x); | ||||||
| 				width += text_area_.area.x; | 					moved_origin.x = _m_caret_to_coordinate(extra, false).x - origin.x; | ||||||
| 
 | 				} | ||||||
| 				auto origin = impl_->cview->origin(); | 				else if (coord.x + static_cast<int>(caret_size().width) >= origin.x + static_cast<int>(view_area.width)) | ||||||
| 				if (static_cast<int>(width) - origin.x >= right_pos) | 				{ | ||||||
| 				{	//Out of screen text area
 | 					extra.x = (std::min)(textbase().getline(points_.caret.y).size(), points_.caret.x + extra_count_horz); | ||||||
| 					origin.x = static_cast<int>(width) - right_pos + 1; | 					auto new_origin = _m_caret_to_coordinate(extra, false).x + static_cast<int>(caret_size().width) - static_cast<int>(view_area.width); | ||||||
| 					auto rest_size = lnstr.size() - points_.caret.x; | 					moved_origin.x = new_origin - origin.x; | ||||||
| 					origin.x += static_cast<int>(_m_text_extent_size(lnstr.c_str() + points_.caret.x, (rest_size >= static_cast<unsigned>(many) ? static_cast<unsigned>(many) : rest_size)).width); |  | ||||||
| 
 |  | ||||||
| 					impl_->cview->move_origin(origin - impl_->cview->origin()); |  | ||||||
| 					return true; |  | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			return false; | 
 | ||||||
|  | 			//upoint pos2nd;
 | ||||||
|  | 			//this->_m_pos_secondary(points_.caret, pos2nd);	//deprecated
 | ||||||
|  | 
 | ||||||
|  | 			auto extra_px = static_cast<int>(line_px * extra_count_vert); | ||||||
|  | 
 | ||||||
|  | 			if (coord.y < origin.y) | ||||||
|  | 			{ | ||||||
|  | 				//Top of caret is less than the top of view
 | ||||||
|  | 
 | ||||||
|  | 				moved_origin.y = (std::max)(0, coord.y - extra_px) - origin.y; | ||||||
|  | 			} | ||||||
|  | 			else if (coord.y + line_px >= origin.y + static_cast<int>(view_area.height)) | ||||||
|  | 			{ | ||||||
|  | 				//Bottom of caret is greater than the bottom of view
 | ||||||
|  | 
 | ||||||
|  | 				auto const bottom = static_cast<int>(impl_->capacities.behavior->take_lines() * line_px); | ||||||
|  | 				auto new_origin = (std::min)(coord.y + line_px + extra_px, bottom) - static_cast<int>(view_area.height); | ||||||
|  | 				moved_origin.y = new_origin - origin.y; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			return impl_->cview->move_origin(moved_origin); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		bool text_editor::_m_move_select(bool record_undo) | 		bool text_editor::_m_move_select(bool record_undo) | ||||||
| @ -3696,9 +3540,9 @@ namespace nana{	namespace widgets | |||||||
| 
 | 
 | ||||||
| 		bool text_editor::_m_update_caret_line(std::size_t secondary_before) | 		bool text_editor::_m_update_caret_line(std::size_t secondary_before) | ||||||
| 		{ | 		{ | ||||||
| 			if (false == impl_->capacities.behavior->adjust_caret_into_screen()) | 			if(false == this->_m_adjust_view()) | ||||||
| 			{ | 			{ | ||||||
| 				if (_m_caret_to_screen(points_.caret).x < impl_->cview->view_area().right()) | 				if (_m_caret_to_coordinate(points_.caret).x < impl_->cview->view_area().right()) | ||||||
| 				{ | 				{ | ||||||
| 					_m_update_line(points_.caret.y, secondary_before); | 					_m_update_line(points_.caret.y, secondary_before); | ||||||
| 					return false; | 					return false; | ||||||
| @ -3789,4 +3633,3 @@ namespace nana{	namespace widgets | |||||||
| 	}//end namespace skeletons
 | 	}//end namespace skeletons
 | ||||||
| }//end namespace widgets
 | }//end namespace widgets
 | ||||||
| }//end namespace nana
 | }//end namespace nana
 | ||||||
| 
 |  | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Jinhao
						Jinhao