123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406 |
- //---------------------------------------------------------------------------
- #include <CorePCH.h>
- #pragma hdrstop
- #include "Option.h"
- //---------------------------------------------------------------------------
- #pragma package(smart_init)
- //---------------------------------------------------------------------------
- const wchar_t ArrayValueDelimiter = L'[';
- const wchar_t ArrayValueEnd = L']';
- //---------------------------------------------------------------------------
- __fastcall TOptions::TOptions()
- {
- FSwitchMarks = L"/-";
- FSwitchValueDelimiters = UnicodeString(L"=:") + ArrayValueDelimiter;
- FNoMoreSwitches = false;
- FParamCount = 0;
- }
- //---------------------------------------------------------------------------
- __fastcall TOptions::TOptions(const TOptions & Source)
- {
- FSwitchMarks = Source.FSwitchMarks;
- FSwitchValueDelimiters = Source.FSwitchValueDelimiters;
- FOptions = Source.FOptions;
- FOriginalOptions = Source.FOriginalOptions;
- FNoMoreSwitches = Source.FNoMoreSwitches;
- FParamCount = Source.FParamCount;
- }
- //---------------------------------------------------------------------------
- void __fastcall TOptions::Parse(const UnicodeString & CmdLine)
- {
- UnicodeString ACmdLine = CmdLine;
- UnicodeString Param;
- while (CutToken(ACmdLine, Param))
- {
- Add(Param);
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TOptions::Add(UnicodeString Value)
- {
- if (!FNoMoreSwitches &&
- (Value.Length() == 2) &&
- (Value[1] == Value[2]) &&
- (FSwitchMarks.Pos(Value[1]) > 0))
- {
- FNoMoreSwitches = true;
- }
- else
- {
- bool Switch = false;
- int Index = 0; // shut up
- wchar_t SwitchMark = L'\0';
- wchar_t ValueDelimiter = L'\0';
- if (!FNoMoreSwitches &&
- (Value.Length() >= 2) &&
- (FSwitchMarks.Pos(Value[1]) > 0))
- {
- Index = 2;
- Switch = true;
- SwitchMark = Value[1];
- while (Switch && (Index <= Value.Length()))
- {
- if (Value.IsDelimiter(FSwitchValueDelimiters, Index))
- {
- ValueDelimiter = Value[Index];
- break;
- }
- // This is to treat /home/martin as parameter, not as switch.
- else if ((Value[Index] == L'?') ||
- IsLetter(Value[Index]) ||
- ((Value[Index] == L'-') && (SwitchMark == L'-') && (Value[2] == L'-'))) // allow --puttygen-switches
- {
- // noop
- }
- else
- {
- Switch = false;
- break;
- }
- ++Index;
- }
- }
- TOption Option;
- if (Switch)
- {
- Option.Type = otSwitch;
- Option.Name = Value.SubString(2, Index - 2);
- Option.Value = Value.SubString(Index + 1, Value.Length());
- if ((ValueDelimiter == ArrayValueDelimiter) && EndsStr(ArrayValueEnd, Option.Value))
- {
- Option.Value.SetLength(Option.Value.Length() - 1);
- }
- Option.ValueSet = (Index <= Value.Length());
- }
- else
- {
- Option.Type = otParam;
- Option.Value = Value;
- Option.ValueSet = false; // unused
- ++FParamCount;
- }
- Option.Used = false;
- Option.SwitchMark = SwitchMark;
- FOptions.push_back(Option);
- }
- FOriginalOptions = FOptions;
- }
- //---------------------------------------------------------------------------
- UnicodeString __fastcall TOptions::GetParam(int Index)
- {
- DebugAssert((Index >= 1) && (Index <= FParamCount));
- UnicodeString Result;
- size_t I = 0;
- while ((I < FOptions.size()) && (Index > 0))
- {
- if (FOptions[I].Type == otParam)
- {
- --Index;
- if (Index == 0)
- {
- Result = FOptions[I].Value;
- FOptions[I].Used = true;
- }
- }
- ++I;
- }
- return Result;
- }
- //---------------------------------------------------------------------------
- UnicodeString TOptions::ConsumeParam()
- {
- UnicodeString Result = Param[1];
- ParamsProcessed(1, 1);
- return Result;
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::GetEmpty()
- {
- return FOptions.empty();
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::FindSwitch(const UnicodeString Switch,
- UnicodeString & Value, int & ParamsStart, int & ParamsCount, bool CaseSensitive, bool & ValueSet)
- {
- ParamsStart = 0;
- ValueSet = false;
- int Index = 0;
- bool Found = false;
- while ((Index < int(FOptions.size())) && !Found)
- {
- if (FOptions[Index].Type == otParam)
- {
- ParamsStart++;
- }
- else if (FOptions[Index].Type == otSwitch)
- {
- if ((!CaseSensitive && SameText(FOptions[Index].Name, Switch)) ||
- (CaseSensitive && SameStr(FOptions[Index].Name, Switch)))
- {
- Found = true;
- Value = FOptions[Index].Value;
- ValueSet = FOptions[Index].ValueSet;
- FOptions[Index].Used = true;
- }
- }
- Index++;
- }
- ParamsCount = 0;
- if (Found)
- {
- ParamsStart++;
- while ((Index + ParamsCount < int(FOptions.size())) &&
- (FOptions[Index + ParamsCount].Type == otParam))
- {
- ParamsCount++;
- }
- }
- else
- {
- ParamsStart = 0;
- }
- return Found;
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::FindSwitch(const UnicodeString Switch, UnicodeString & Value)
- {
- bool ValueSet;
- return FindSwitch(Switch, Value, ValueSet);
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::FindSwitch(const UnicodeString Switch, UnicodeString & Value, bool & ValueSet)
- {
- int ParamsStart;
- int ParamsCount;
- return FindSwitch(Switch, Value, ParamsStart, ParamsCount, false, ValueSet);
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::FindSwitch(const UnicodeString Switch)
- {
- UnicodeString Value;
- int ParamsStart;
- int ParamsCount;
- bool ValueSet;
- return FindSwitch(Switch, Value, ParamsStart, ParamsCount, false, ValueSet);
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::FindSwitchCaseSensitive(const UnicodeString Switch)
- {
- UnicodeString Value;
- int ParamsStart;
- int ParamsCount;
- bool ValueSet;
- return FindSwitch(Switch, Value, ParamsStart, ParamsCount, true, ValueSet);
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::FindSwitch(const UnicodeString Switch,
- TStrings * Params, int ParamsMax)
- {
- return DoFindSwitch(Switch, Params, ParamsMax, false);
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::FindSwitchCaseSensitive(const UnicodeString Switch,
- TStrings * Params, int ParamsMax)
- {
- return DoFindSwitch(Switch, Params, ParamsMax, true);
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::DoFindSwitch(const UnicodeString Switch,
- TStrings * Params, int ParamsMax, bool CaseSensitive)
- {
- UnicodeString Value;
- int ParamsStart;
- int ParamsCount;
- bool ValueSet;
- bool Result = FindSwitch(Switch, Value, ParamsStart, ParamsCount, CaseSensitive, ValueSet);
- if (Result)
- {
- int AParamsCount;
- if (TryStrToInt(Value, AParamsCount) && (AParamsCount < ParamsCount))
- {
- ParamsCount = AParamsCount;
- }
- if ((ParamsMax >= 0) && (ParamsCount > ParamsMax))
- {
- ParamsCount = ParamsMax;
- }
- int Index = 0;
- while (Index < ParamsCount)
- {
- Params->Add(Param[ParamsStart + Index]);
- Index++;
- }
- ParamsProcessed(ParamsStart, ParamsCount);
- }
- return Result;
- }
- //---------------------------------------------------------------------------
- UnicodeString __fastcall TOptions::SwitchValue(const UnicodeString Switch,
- const UnicodeString Default)
- {
- UnicodeString Value;
- FindSwitch(Switch, Value);
- if (Value.IsEmpty())
- {
- Value = Default;
- }
- return Value;
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::SwitchValue(const UnicodeString Switch, bool Default, bool DefaultOnNonExistence)
- {
- bool Result;
- int IntValue;
- UnicodeString Value;
- if (!FindSwitch(Switch, Value))
- {
- Result = DefaultOnNonExistence;
- }
- else if (Value.IsEmpty())
- {
- Result = Default;
- }
- else if (SameText(Value, "on"))
- {
- Result = true;
- }
- else if (SameText(Value, "off"))
- {
- Result = false;
- }
- else if (TryStrToInt(Value, IntValue))
- {
- Result = (IntValue != 0);
- }
- else
- {
- throw Exception(FMTLOAD(URL_OPTION_BOOL_VALUE_ERROR, (Value)));
- }
- return Result;
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::SwitchValue(const UnicodeString Switch, bool Default)
- {
- return SwitchValue(Switch, Default, Default);
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::UnusedSwitch(UnicodeString & Switch)
- {
- bool Result = false;
- size_t Index = 0;
- while (!Result && (Index < FOptions.size()))
- {
- if ((FOptions[Index].Type == otSwitch) &&
- !FOptions[Index].Used)
- {
- Switch = FOptions[Index].Name;
- Result = true;
- }
- ++Index;
- }
- return Result;
- }
- //---------------------------------------------------------------------------
- bool __fastcall TOptions::WasSwitchAdded(UnicodeString & Switch, UnicodeString & Value, wchar_t & SwitchMark)
- {
- bool Result =
- DebugAlwaysTrue(FOptions.size() > 0) &&
- (FOptions.back().Type == otSwitch);
- if (Result)
- {
- TOption & Option = FOptions.back();
- Switch = Option.Name;
- Value = Option.Value;
- SwitchMark = Option.SwitchMark;
- }
- return Result;
- }
- //---------------------------------------------------------------------------
- void __fastcall TOptions::ParamsProcessed(int ParamsStart, int ParamsCount)
- {
- if (ParamsCount > 0)
- {
- DebugAssert((ParamsStart >= 0) && ((ParamsStart - ParamsCount + 1) <= FParamCount));
- size_t Index = 0;
- while ((Index < FOptions.size()) && (ParamsStart > 0))
- {
- if (FOptions[Index].Type == otParam)
- {
- --ParamsStart;
- if (ParamsStart == 0)
- {
- while (ParamsCount > 0)
- {
- DebugAssert(Index < FOptions.size());
- DebugAssert(FOptions[Index].Type == otParam);
- FOptions.erase(FOptions.begin() + Index);
- --FParamCount;
- --ParamsCount;
- }
- }
- }
- Index++;
- }
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TOptions::LogOptions(TLogOptionEvent OnLogOption)
- {
- for (size_t Index = 0; Index < FOriginalOptions.size(); Index++)
- {
- const TOption & Option = FOriginalOptions[Index];
- UnicodeString LogStr;
- switch (Option.Type)
- {
- case otParam:
- LogStr = FORMAT(L"Parameter: %s", (Option.Value));
- DebugAssert(Option.Name.IsEmpty());
- break;
- case otSwitch:
- LogStr =
- FORMAT(L"Switch: %s%s%s%s",
- (FSwitchMarks[1], Option.Name, (Option.Value.IsEmpty() ? UnicodeString() : FSwitchValueDelimiters.SubString(1, 1)), Option.Value));
- break;
- default:
- DebugFail();
- break;
- }
- OnLogOption(LogStr);
- }
- }
|