// Copyright 2014-2017 ClassicalSharp | Licensed under BSD-3
using System;
using System.Drawing;
using OpenTK.Input;
#if ANDROID
using Android.Graphics;
#endif
namespace ClassicalSharp.Gui.Widgets {
	public abstract class InputWidget : Widget {
		
		public InputWidget(Game game, Font font) : base(game) {
			Text = new WrappableStringBuffer(Utils.StringLength * MaxLines);
			lines = new string[MaxLines];
			lineSizes = new Size[MaxLines];
			
			DrawTextArgs args = new DrawTextArgs("_", font, true);
			caretTex = game.Drawer2D.MakeChatTextTexture(ref args, 0, 0);
			caretTex.Width = (short)((caretTex.Width * 3) / 4);
			caretWidth = caretTex.Width; caretHeight = caretTex.Height;
			this.font = font;
			
			if (Prefix == null) return;
			args = new DrawTextArgs(Prefix, font, true);
			Size size = game.Drawer2D.MeasureChatSize(ref args);
			prefixWidth = Width = size.Width;
			prefixHeight = Height = size.Height;
		}
		
		public InputWidget SetLocation(Anchor horAnchor, Anchor verAnchor, int xOffset, int yOffset) {
			HorizontalAnchor = horAnchor; VerticalAnchor = verAnchor;
			XOffset = xOffset; YOffset = yOffset;
			CalculatePosition();
			return this;
		}
		
		protected int caret = -1;
		protected Texture inputTex, caretTex, prefixTex;
		protected readonly Font font;
		protected int caretWidth, caretHeight, prefixWidth, prefixHeight;
		protected FastColour caretColour;
		
		///  The raw text entered. 
		///  You should Append() to add more text, as that also updates the caret position and texture. 
		public WrappableStringBuffer Text;
		
		///  The maximum number of lines that may be entered. 
		public abstract int MaxLines { get; }
		
		///  The maximum number of characters that can fit on one line. 
		public abstract int MaxCharsPerLine { get; }
		
		///  The prefix string that is always shown before the input text. Can be null. 
		public abstract string Prefix { get; }
		
		///  The horizontal offset (in pixels) from the start of the box background
		/// to the beginning of the input texture. 
		public abstract int Padding { get; }
		
		///  Whether a caret should be drawn at the position characters 
		/// are inserted/deleted from the input text. 
		public bool ShowCaret;
		
		protected string[] lines; // raw text of each line
		protected Size[] lineSizes; // size of each line in pixels
		protected int caretCol, caretRow; // coordinates of caret
		protected double caretAccumulator;
		
		public override void Init() {
			if (lines.Length > 1) {
				Text.WordWrap(game.Drawer2D, lines, MaxCharsPerLine);
			} else {
				lines[0] = Text.ToString();
			}
			
			CalculateLineSizes();
			RemakeTexture();
			UpdateCaret();
		}
		
		public override void Dispose() {
			gfx.DeleteTexture(ref inputTex);
		}
		
		public void DisposeFully() {
			Dispose();
			gfx.DeleteTexture(ref caretTex);
			gfx.DeleteTexture(ref prefixTex);
		}
		public override void CalculatePosition() {
			int oldX = X, oldY = Y;
			base.CalculatePosition();
			
			caretTex.X1 += X - oldX; caretTex.Y1 += Y - oldY;
			inputTex.X1 += X - oldX; inputTex.Y1 += Y - oldY;
		}
		
		
		///  Calculates the sizes of each line in the text buffer. 
		public void CalculateLineSizes() {
			for (int y = 0; y < lineSizes.Length; y++)
				lineSizes[y] = Size.Empty;
			lineSizes[0].Width = prefixWidth;
			
			DrawTextArgs args = new DrawTextArgs(null, font, true);
			for (int y = 0; y < MaxLines; y++) {
				args.Text = lines[y];
				lineSizes[y] += game.Drawer2D.MeasureChatSize(ref args);
			}
			if (lineSizes[0].Height == 0) lineSizes[0].Height = prefixHeight;
		}
		
		///  Calculates the location and size of the caret character 
		public void UpdateCaret() {
			if (caret >= Text.Length) caret = -1;
			Text.GetCoords(caret, lines, out caretCol, out caretRow);
			DrawTextArgs args = new DrawTextArgs(null, font, true);
			IDrawer2D drawer = game.Drawer2D;
			caretAccumulator = 0;
			if (caretCol == MaxCharsPerLine) {
				caretTex.X1 = X + Padding + lineSizes[caretRow].Width;
				caretColour = FastColour.Yellow;
				caretTex.Width = (short)caretWidth;
			} else {
				args.Text = lines[caretRow].Substring(0, caretCol);
				Size trimmedSize = drawer.MeasureChatSize(ref args);
				if (caretRow == 0) trimmedSize.Width += prefixWidth;
				caretTex.X1 = X + Padding + trimmedSize.Width;
				caretColour = FastColour.Scale(FastColour.White, 0.8f);
				
				string line = lines[caretRow];
				if (caretCol < line.Length) {
					args.Text = new String(line[caretCol], 1);
					caretTex.Width = (short)drawer.MeasureChatSize(ref args).Width;
				} else {
					caretTex.Width = (short)caretWidth;
				}
			}
			caretTex.Y1 = lineSizes[0].Height * caretRow + inputTex.Y1 + 2;
			
			// Update the colour of the caret
			char code = GetLastColour(caretCol, caretRow);
			if (code != '\0') caretColour = drawer.Colours[code];
		}
		
		protected void RenderCaret(double delta) {
			if (!ShowCaret) return;
			
			caretAccumulator += delta;
			if ((caretAccumulator % 1) < 0.5)
				caretTex.Render(gfx, caretColour);
		}
		
		///  Remakes the raw texture containg all the chat lines. 
		///  Also updates the dimensions of the widget. 
		public virtual void RemakeTexture() {
			int totalHeight = 0, maxWidth = 0;
			for (int i = 0; i < MaxLines; i++) {
				totalHeight += lineSizes[i].Height;
				maxWidth = Math.Max(maxWidth, lineSizes[i].Width);
			}
			Size size = new Size(maxWidth, totalHeight);
			caretAccumulator = 0;
			
			int realHeight = 0;
			using (Bitmap bmp = IDrawer2D.CreatePow2Bitmap(size))
				using (IDrawer2D drawer = game.Drawer2D)
			{
				drawer.SetBitmap(bmp);
				DrawTextArgs args = new DrawTextArgs(null, font, true);
				if (Prefix != null) {
					args.Text = Prefix;
					drawer.DrawChatText(ref args, 0, 0);
				}
				
				for (int i = 0; i < lines.Length; i++) {
					if (lines[i] == null) break;
					args.Text = lines[i];
					char lastCol = GetLastColour(0, i);
					if (!IDrawer2D.IsWhiteColour(lastCol))
						args.Text = "&" + lastCol + args.Text;
					
					int offset = i == 0 ? prefixWidth : 0;
					drawer.DrawChatText(ref args, offset, realHeight);
					realHeight += lineSizes[i].Height;
				}
				inputTex = drawer.Make2DTexture(bmp, size, 0, 0);
			}
			
			Width = size.Width;
			Height = realHeight == 0 ? prefixHeight : realHeight;
			CalculatePosition();
			inputTex.X1 = X + Padding; inputTex.Y1 = Y;
		}
		
		protected char GetLastColour(int indexX, int indexY) {
			int x = indexX;
			IDrawer2D drawer = game.Drawer2D;
			for (int y = indexY; y >= 0; y--) {
				string part = lines[y];
				char code = drawer.LastColour(part, x);
				if (code != '\0') return code;
				if (y > 0) x = lines[y - 1].Length;
			}
			return '\0';
		}
		
		///  Invoked when the user presses enter. 
		public virtual void EnterInput() {
			Clear();
			Height = prefixHeight;
		}
		
		
		///  Clears all the characters from the text buffer 
		///  Deletes the native texture. 
		public void Clear() {
			Text.Clear();
			for (int i = 0; i < lines.Length; i++)
				lines[i] = null;
			
			caret = -1;
			Dispose();
		}
		///  Appends a sequence of characters to current text buffer. 
		///  Potentially recreates the native texture. 
		public void Append(string text) {
			int appended = 0;
			foreach (char c in text) {
				if (TryAppendChar(c)) appended++;
			}
			
			if (appended == 0) return;
			Recreate();
		}
		
		///  Appends a single character to current text buffer. 
		///  Potentially recreates the native texture. 
		public void Append(char c) {
			if (!TryAppendChar(c)) return;
			Recreate();
		}
		
		
		bool TryAppendChar(char c) {
			int totalChars = MaxCharsPerLine * lines.Length;
			if (Text.Length == totalChars) return false;
			if (!AllowedChar(c)) return false;
			
			AppendChar(c);
			return true;
		}
		
		protected virtual bool AllowedChar(char c) {
			return Utils.IsValidInputChar(c, game);
		}
		
		protected void AppendChar(char c) {
			if (caret == -1) {
				Text.InsertAt(Text.Length, c);
			} else {
				Text.InsertAt(caret, c);
				caret++;
				if (caret >= Text.Length) caret = -1;
			}
		}
		
		protected void DeleteChar() {
			if (Text.Length == 0) return;
			
			if (caret == -1) {
				Text.DeleteAt(Text.Length - 1);
			} else if (caret > 0) {
				caret--;
				Text.DeleteAt(caret);
			}
		}
		
		
		#region Input handling
		
		protected bool ControlDown() {
			return OpenTK.Configuration.RunningOnMacOS ?
				(game.IsKeyDown(Key.WinLeft) || game.IsKeyDown(Key.WinRight))
				: (game.IsKeyDown(Key.ControlLeft) || game.IsKeyDown(Key.ControlRight));
		}
		
		public override bool HandlesKeyPress(char key) {
			if (!game.HideGui) Append(key);
			return true;
		}
		
		public override bool HandlesKeyDown(Key key) {
			if (game.HideGui) return key < Key.F1 || key > Key.F35;
			bool clipboardDown = ControlDown();
			
			if (key == Key.Left) LeftKey(clipboardDown);
			else if (key == Key.Right) RightKey(clipboardDown);
			else if (key == Key.BackSpace) BackspaceKey(clipboardDown);
			else if (key == Key.Delete) DeleteKey();
			else if (key == Key.Home) HomeKey();
			else if (key == Key.End) EndKey();
			else if (clipboardDown && !OtherKey(key)) return false;
			
			return true;
		}
		
		public override bool HandlesKeyUp(Key key) { return true; }
		
		public override bool HandlesMouseClick(int mouseX, int mouseY, MouseButton button) {
			if (button == MouseButton.Left)
				SetCaretToCursor(mouseX, mouseY);
			return true;
		}
		
		
		void BackspaceKey(bool controlDown) {
			if (controlDown) {
				if (caret == -1) caret = Text.Length - 1;
				int len = Text.GetBackLength(caret);
				if (len == 0) return;
				
				caret -= len;
				if (caret < 0) caret = 0;
				for (int i = 0; i <= len; i++)
					Text.DeleteAt(caret);
				
				if (caret >= Text.Length) caret = -1;
				if (caret == -1 &&  Text.Length > 0) {
					Text.value[Text.Length] = ' ';
				} else if (caret >= 0 && Text.value[caret] != ' ') {
					Text.InsertAt(caret, ' ');
				}
				Recreate();
			} else if (!Text.Empty && caret != 0) {
				int index = caret == -1 ? Text.Length - 1 : caret;
				if (CheckColour(index - 1)) {
					DeleteChar(); // backspace XYZ%e to XYZ
				} else if (CheckColour(index - 2)) {
					DeleteChar(); DeleteChar(); // backspace XYZ%eH to XYZ
				}
				
				DeleteChar();
				Recreate();
			}
		}
		bool CheckColour(int index) {
			if (index < 0) return false;
			char code = Text.value[index], col = Text.value[index + 1];
			return (code == '%' || code == '&') && game.Drawer2D.ValidColour(col);
		}
		
		void DeleteKey() {
			if (!Text.Empty && caret != -1) {
				Text.DeleteAt(caret);
				if (caret >= Text.Length) caret = -1;
				Recreate();
			}
		}
		
		void LeftKey(bool controlDown) {
			if (controlDown) {
				if (caret == -1)
					caret = Text.Length - 1;
				caret -= Text.GetBackLength(caret);
				UpdateCaret();
				return;
			}
			
			if (!Text.Empty) {
				if (caret == -1) caret = Text.Length;
				caret--;
				if (caret < 0) caret = 0;
				UpdateCaret();
			}
		}
		
		void RightKey(bool controlDown) {
			if (controlDown) {
				caret += Text.GetForwardLength(caret);
				if (caret >= Text.Length) caret = -1;
				UpdateCaret();
				return;
			}
			
			if (!Text.Empty && caret != -1) {
				caret++;
				if (caret >= Text.Length) caret = -1;
				UpdateCaret();
			}
		}
		
		void HomeKey() {
			if (Text.Empty) return;
			caret = 0;
			UpdateCaret();
		}
		
		void EndKey() {
			caret = -1;
			UpdateCaret();
		}
		
		static char[] trimChars = {'\r', '\n', '\v', '\f', ' ', '\t', '\0'};
		bool OtherKey(Key key) {
			int totalChars = MaxCharsPerLine * lines.Length;
			if (key == Key.V && Text.Length < totalChars) {
				string text = null;
				try {
					text = game.window.ClipboardText.Trim(trimChars);
				} catch (Exception ex) {
					ErrorHandler.LogError("Paste from clipboard", ex);
					const string warning = "&cError while trying to paste from clipboard.";
					game.Chat.Add(warning, MessageType.ClientStatus4);
					return true;
				}
				if (String.IsNullOrEmpty(text)) return true;
				Append(text);
				return true;
			} else if (key == Key.C) {
				if (Text.Empty) return true;
				try {
					game.window.ClipboardText = Text.ToString();
				} catch (Exception ex) {
					ErrorHandler.LogError("Copy to clipboard", ex);
					const string warning = "&cError while trying to copy to clipboard.";
					game.Chat.Add(warning, MessageType.ClientStatus4);
				}
				return true;
			}
			return false;
		}
		
		
		protected unsafe void SetCaretToCursor(int mouseX, int mouseY) {
			mouseX -= inputTex.X1; mouseY -= inputTex.Y1;
			DrawTextArgs args = new DrawTextArgs(null, font, true);
			IDrawer2D drawer = game.Drawer2D;
			int offset = 0, elemHeight = caretHeight;
			string oneChar = new String('A', 1);
			
			for (int y = 0; y < lines.Length; y++) {
				string line = lines[y];
				int xOffset = y == 0 ? prefixWidth : 0;
				if (line == null) continue;
				
				for (int x = 0; x < line.Length; x++) {
					args.Text = line.Substring(0, x);
					int trimmedWidth = drawer.MeasureChatSize(ref args).Width + xOffset;
					// avoid allocating an unnecessary string
					fixed(char* ptr = oneChar)
						ptr[0] = line[x];
					
					args.Text = oneChar;
					int elemWidth = drawer.MeasureChatSize(ref args).Width;
					
					if (GuiElement.Contains(trimmedWidth, y * elemHeight, elemWidth, elemHeight, mouseX, mouseY)) {
						caret = offset + x;
						UpdateCaret(); return;
					}
				}
				offset += line.Length;
			}
			caret = -1;
			UpdateCaret();
		}
		
		#endregion
	}
}