#include "Browser.h"


#define LTIMING(x) // RTIMING(x)

#define QLOG(x)


static const char s_dbver[] = "Assist++ 1.1";

CppBase& BrowserBase()
{
	static CppBase b;
	return b;
}

ArrayMap<String, BrowserFileInfo>& FileSet()
{
	static ArrayMap<String, BrowserFileInfo> x;
	return x;
}

Vector<Browser *>& RBrowser()
{
	static Vector<Browser *> bp;
	return bp;
}

void     Register(Browser *b)
{
	RBrowser().Add(b);
}

void     UnRegister(Browser *b)
{
	Vector<Browser *>& rb = RBrowser();
	int q = FindIndex(rb, b);
	if(q >= 0)
		rb.Remove(q);
}

static bool   s_console;
static String s_file;

void BrowserScanError(int line, const String& text)
{
	if(s_console)
		PutConsole(String().Cat() << s_file << " (" << line << "): " << text);
}

int NoSlash(int c)
{
	return c == '/' || c == '\\' || c == ':' ? '$' : c;
}

void SaveBrowserBase()
{
	CppBase& base = BrowserBase();
	if(base.GetCount() == 0)
		return;
	ArrayMap<String, BrowserFileInfo>& fileset = FileSet();
	ArrayMap<String, StringStream> out;
	String s;
	for(int j = 0; j < base.GetCount(); j++) {
		CppNest& nest = base[j];
		for(int k = 0; k < nest.GetCount(); k++) {
			CppItem& m = nest.item[k];
			for(int p = 0; p < m.pos.GetCount(); p++) {
				StringStream& o = out.GetAdd(m.pos[p].GetFile());
				s = base.GetKey(j);
				o % s;
				s = nest.key[k];
				o % s;
				s = nest.name[k];
				o % s;
				o % nest.namespacel;
				o % m;
				o % m.pos[p].line;
			}
		}
	}

	const Workspace& wspc = GetIdeWorkspace();
	for(int i = 0; i < wspc.GetCount(); i++) {
		String package = wspc[i];
		String cfg = ConfigFile("cfg");
		RealizeDirectory(cfg);
		FileOut fo(AppendFileName(cfg, ForceExt(Filter(Nvl(package, "&"), NoSlash) + '@' + GetVarsName(), ".cdb")));
		s = s_dbver;
		fo % s;
		for(int i = 0; i < fileset.GetCount(); i++)
			if(fileset[i].package == package) {
				String fn = fileset.GetKey(i);
				String q = out.GetAdd(fn).GetResult();
				BrowserFileInfo& f = fileset[i];
				fo % fn;
				fo % f.file;
				fo % f.time;
				fo % q;
			}
	}
}

void LoadBrowserBase(Progress& pi)
{
	CppBase& base = BrowserBase();
	ArrayMap<String, BrowserFileInfo>& fileset = FileSet();
	String s;
	const Workspace& wspc = GetIdeWorkspace();
	pi.SetText("Assist++ loading packages");
	pi.SetTotal(wspc.GetCount());
	pi.SetPos(0);
	for(int i = 0; i < wspc.GetCount(); i++) {
		pi.Step();
		String package = wspc[i];
		String cfg = ConfigFile("cfg");
		FileIn in(AppendFileName(cfg, ForceExt(Filter(package, NoSlash) + '@' + GetVarsName(), ".cdb")));
		in.LoadThrowing();
		if(in) {
			try {
				in % s;
				if(s != s_dbver)
					throw LoadingError();
				while(!in.IsEof()) {
					String fn;
					in % fn;
					BrowserFileInfo& f = fileset.GetAdd(fn);
					in % f.file;
					in % f.time;
					f.package = package;
					String q;
					in % q;
					StringStream ss(q);
					ss.LoadThrowing();
					while(!ss.IsEof()) {
						String nest, key, name;
						ss % nest;
						ss % key;
						ss % name;
						CppNest& n = base.GetAdd(nest);
						ss % n.namespacel;
						CppItem& m = n.GetAdd(key, name);
						ss % m;
						CppPos& p = m.pos.Add();
						p.file = GetCppFileIndex(fn);
						ss % p.line;
					}
				}
			}
			catch(LoadingError) {
				BrowserBase().Clear();
				fileset.Clear();
				return;
			}
		}
	}
}

Vector<String> IgnoreList()
{
	Vector<String> ignore;
	FileIn in(CommonPath("dppignore.txt"));
	while(!in.IsEof()) {
		String s = in.GetLine();
		if(!s.IsEmpty())
			ignore.Add(s);
	}
	return ignore;
}

void FinishBase(const CppWordsHash& w)
{
	TimeStop tm;
	Qualify(BrowserBase(), w);
}

void ReQualifyBrowserBase()
{
	Qualify(BrowserBase(), AllCppWords());
}

Vector<String> SortedNests()
{
	LTIMING("SortedNests()");
	CppBase& base = BrowserBase();
	Vector<String> n;
	for(int i = 0; i < base.GetCount(); i++)
		if(!base.nest.IsUnlinked(i))
			n.Add(base.GetKey(i));
	Sort(n);
	return n;
}

CppWordsHash Difference(const Vector<String>& a)
{
	TimeStop tm;
	Vector<String> b = SortedNests();
	CppWordsHash h;
	int ai = 0;
	int bi = 0;
	while(ai < a.GetCount() && bi < b.GetCount()) {
		int q = SgnCompare(a[ai], b[bi]);
		if(q < 0)
			h.AddWords(a[ai++]);
		else
		if(q > 0)
			h.AddWords(b[bi++]);
		else {
			ai++;
			bi++;
		}
	}
	while(ai < a.GetCount())
		h.AddWords(a[ai++]);
	while(bi < b.GetCount())
		h.AddWords(b[bi++]);
	QLOG("Difference " << tm);
	return h;
}

void UpdateBrowserBase(Progress& pi)
{
	Index<String> fp;
	Vector<String> scan;
	ArrayMap<String, BrowserFileInfo>& set = FileSet();
	const Workspace& wspc = GetIdeWorkspace();
	CppBase& base = BrowserBase();
	pi.SetText("Assist++ checking packages");
	pi.SetTotal(wspc.GetCount());
	pi.SetPos(0);
	for(int i = 0; i < wspc.GetCount(); i++) {
		pi.Step();
		const Package& pk = wspc.GetPackage(i);
		String n = wspc[i];
		for(int i = 0; i < pk.file.GetCount(); i++) {
			String path = SourcePath(n, pk.file[i]);
			String ext = ToUpper(GetFileExt(path));
			if(ext == ".C" || ext == ".CPP" || ext == ".CC" || ext == ".CXX" ||
			   ext == ".H" || ext == ".HPP" || ext == ".HH" || ext == ".HXX" ||
			   ext == ".LAY") {
				fp.Add(path);
				int q = set.Find(path);
				Time tm = FileGetTime(path);
				if(q < 0 || set[q].time != tm)
					scan.Add(path);
				BrowserFileInfo& bf = set.GetAdd(path);
				bf.package = n;
				bf.file = pk.file[i];
				bf.time = tm;
			}
		}
	}

	Vector<String> remove(scan, 1);
	Vector<int> rm;
	for(int i = 0; i < set.GetCount(); i++)
		if(fp.Find(set.GetKey(i)) < 0) {
			remove.Add(set.GetKey(i));
			rm.Add(i);
		}
	if(remove.GetCount() == 0)
		return;
	set.Remove(rm);
	Remove(base, remove);
	if(scan.GetCount()) {
		pi.SetTotal(scan.GetCount());
		pi.SetPos(0);
		pi.AlignText(ALIGN_LEFT);
		Vector<String> ignore = IgnoreList();
		for(int i = 0; i < scan.GetCount(); i++) {
			s_file = scan[i];
			pi.SetText(s_file);
			pi.Step();
			String ext = ToUpper(GetFileExt(s_file));
			if(ext == ".LAY")
				ScanLayFile(s_file);
			else {
				FileIn fi(s_file);
				Parse(fi, ignore, base, s_file, callback(BrowserScanError));
			}
		}
	}
}

void BrowserBaseScan(Stream& s, const String& fn)
{
	LTIMING("BrowserBaseScan");
	TimeStop tm;
	Vector<String> before = SortedNests();
	QLOG("Scan1 " << tm);
	CppBase& base = BrowserBase();
	QLOG("Scan2 " << tm);
	Vector<String> remove;
	remove.Add(fn);
	Remove(base, remove);
	QLOG("Scan3 " << tm);
	Parse(s, IgnoreList(), base, fn, CNULL);
	QLOG("Scan4 " << tm);
	FinishBase(Difference(before));
	QLOG("Scan total " << tm);
	QLOG("---------");
}

void BrowserBaseScanLay(const String& fn)
{
	LTIMING("BrowserBaseScanLay");
	Vector<String> before = SortedNests();
	CppBase& base = BrowserBase();
	Vector<String> remove;
	remove.Add(fn);
	Remove(base, remove);
	ScanLayFile(fn);
	FinishBase(Difference(before));
}

void ClearBrowserBase()
{
	BrowserBase().Clear();
	FileSet().Clear();
}

void StartBrowserBase()
{
	if(BrowserBase().GetCount() == 0) {
		Progress pi;
		pi.Title("Assist++");
		LoadBrowserBase(pi);
		UpdateBrowserBase(pi);
		FinishBase(AllCppWords());
	}
}

void SyncBrowserBase()
{
	if(BrowserBase().GetCount()) {
		Progress pi;
		pi.Title("Assist++");
		UpdateBrowserBase(pi);
		FinishBase(AllCppWords());
	}
}

void RescanBrowserBase()
{
	ClearBrowserBase();
	s_console = true;
	Progress pi;
	pi.Title("Assist++");
	UpdateBrowserBase(pi);
	FinishBase(AllCppWords());
	s_console = false;
}

bool ExistsBrowserItem(const String& item)
{
	String nest, key;
	if(!SplitNestKey(item, nest, key))
		return false;
	CppBase& base = BrowserBase();
	int q = base.Find(nest);
	return q >= 0 ? base[q].key.Find(key) >= 0 : false;
}


syntax highlighted by Code2HTML, v. 0.9.1