/* * ConfigPlist.h * * Created on: Oct 9, 2020 * Author: jief */ #ifndef _CONFIGPLISTCLASS_GUI_H_ #define _CONFIGPLISTCLASS_GUI_H_ // -------------------------------------------------- GUI class class GUI_Class : public XmlDict { using super = XmlDict; public: // -------------------------------------------------- Mouse class class GUI_Mouse_Class : public XmlDict { using super = XmlDict; protected: XmlInt64 Speed = XmlInt64(); XmlBool Enabled = XmlBool(); XmlBool Mirror = XmlBool(); XmlUInt64 DoubleClickTime = XmlUInt64(); XmlDictField m_fields[4] = { {"Speed", Speed}, {"Enabled", Enabled}, {"Mirror", Mirror}, {"DoubleClickTime", DoubleClickTime}, }; public: virtual void getFields(XmlDictField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; int64_t dgetPointerSpeed() const { return Speed.isDefined() ? Speed.value() : 0; }; bool dgetPointerEnabled() const { return Enabled.isDefined() ? Enabled.value() ? dgetPointerSpeed() != 0 : false : dgetPointerSpeed() != 0 ; }; bool dgetPointerMirror() const { return Mirror.isDefined() ? Mirror.value() : false; }; uint64_t dgetDoubleClickTime() const { return DoubleClickTime.isDefined() ? DoubleClickTime.value() : 0; }; }; // -------------------------------------------------- Mouse class end // -------------------------------------------------- Scan class // bool or dict class GUI_Scan_Class : public XmlUnion { public: // -------------------------------------------------- Scan as dict class class GUI_ScanAsDict_Class : public XmlDict { using super = XmlDict; protected: XmlBool Entries = XmlBool(); XmlBool Tool = XmlBool(); XmlBool Linux = XmlBool(); XmlBoolOrString Legacy = XmlBoolOrString(); XmlBoolOrString Kernel = XmlBoolOrString(); XmlDictField m_fields[5] = { {"Entries", Entries}, {"Tool", Tool}, {"Linux", Linux}, {"Legacy", Legacy}, {"Kernel", Kernel}, }; public: virtual void getFields(XmlDictField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; bool dgetDisableEntryScan() const { return Entries.isDefined() && Entries.value() == false; }; bool dgetDisableToolScan() const { return Tool.isDefined() && Tool.value() == false; }; bool dgetLinux() const { return !(Linux.isDefined() && Linux.value() == false); }; bool dgetNoLegacy() const { if ( !Legacy.isDefined() ) return false; if ( Legacy.xmlBool.isDefined() && Legacy.xmlBool.value() == false ) return true; return false; } bool dgetLegacyFirst() const { return Legacy.isDefined() && Legacy.xmlString8.isDefined() && Legacy.xmlString8.value().startWithIC("F"); }; UINT8 dgetKernel() const { if ( !Kernel.isDefined() ) return 0; if ( Kernel.xmlBool.isDefined() ) { if ( Kernel.xmlBool.value() == false ) return KERNEL_SCAN_NONE; return 0; } if ( !Kernel.xmlString8.isDefined() || Kernel.xmlString8.value().isEmpty() ) return 0; if ( Kernel.xmlString8.value().startWithOrEqualToIC("ne") ) return KERNEL_SCAN_NEWEST; if ( Kernel.xmlString8.value().startWithOrEqualToIC("n") ) return KERNEL_SCAN_NONE; if ( Kernel.xmlString8.value().startWithOrEqualToIC("o") ) return KERNEL_SCAN_OLDEST; if ( Kernel.xmlString8.value().startWithOrEqualToIC("f") ) return KERNEL_SCAN_FIRST; if ( Kernel.xmlString8.value().startWithOrEqualToIC("l") ) return KERNEL_SCAN_LAST; if ( Kernel.xmlString8.value().startWithOrEqualToIC("m") ) return KERNEL_SCAN_MOSTRECENT; if ( Kernel.xmlString8.value().startWithOrEqualToIC("e") ) return KERNEL_SCAN_EARLIEST; return 0; } }; // -------------------------------------------------- Scan as dict class end const GUI_Class& parent; public: XmlBool ScanAsBool = XmlBool(); GUI_ScanAsDict_Class ScanAsAsDict = GUI_ScanAsDict_Class(); GUI_Scan_Class(const GUI_Class& _parent) : parent(_parent) {} XmlUnionField m_fields[2] = { ScanAsBool, ScanAsAsDict}; virtual void getFields(XmlUnionField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; bool dgetDisableEntryScan() const { if ( ScanAsBool.isDefined() && ScanAsBool.value() == false ) return true; return ScanAsAsDict.dgetDisableEntryScan(); } bool dgetDisableToolScan() const { if ( ScanAsBool.isDefined() && ScanAsBool.value() == false ) return true; return ScanAsAsDict.dgetDisableToolScan(); } bool dgetLinuxScan() const { if ( !ScanAsAsDict.isDefined() ) return parent.isDefined(); // TODO: different default value if section is not defined return ScanAsAsDict.dgetLinux(); } bool dgetNoLegacy() const { if ( ScanAsBool.isDefined() && ScanAsBool.value() == false ) return true; return ScanAsAsDict.dgetNoLegacy(); } bool dgetLegacyFirst() const { if ( !ScanAsAsDict.isDefined() ) return false; return ScanAsAsDict.dgetLegacyFirst(); } UINT8 dgetKernelScan() const { if ( !ScanAsAsDict.isDefined() ) return 0; return ScanAsAsDict.dgetKernel(); } }; // -------------------------------------------------- Scan class end // -------------------------------------------------- Custom_Class class end class GUI_Custom_Class : public XmlDict { using super = XmlDict; public: class GUI_Custom_Entry_VolumeType_Class : public XmlUnion { using super = XmlDict; protected: XmlString8AllowEmpty VolumeTypeAsString = XmlString8AllowEmpty(); XmlArray VolumeTypeAsArray = XmlArray(); XmlUnionField m_fields[2] = { VolumeTypeAsString, VolumeTypeAsArray}; virtual void getFields(XmlUnionField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; static UINT8 CheckVolumeType(UINT8 VolumeType, const XString8& VolumeTypeAsString) { UINT8 VolumeTypeTmp = VolumeType; if (VolumeTypeAsString.isEqualIC("Internal")) { VolumeTypeTmp |= VOLTYPE_INTERNAL; } else if (VolumeTypeAsString.isEqualIC("External")) { VolumeTypeTmp |= VOLTYPE_EXTERNAL; } else if (VolumeTypeAsString.isEqualIC("Optical")) { VolumeTypeTmp |= VOLTYPE_OPTICAL; } else if (VolumeTypeAsString.isEqualIC("FireWire")) { VolumeTypeTmp |= VOLTYPE_FIREWIRE; } return VolumeTypeTmp; } public: UINT8 dgetVolumeType() const { if ( VolumeTypeAsString.isDefined() ) return CheckVolumeType(0, VolumeTypeAsString.value()); if ( VolumeTypeAsArray.isDefined() ) { UINT8 VolumeType = 0; for (size_t idx = 0; idx < VolumeTypeAsArray.size() ; ++idx) { VolumeType = CheckVolumeType(VolumeType, VolumeTypeAsArray[idx].value()); } return VolumeType; } return 0; } }; class GUI_Custom_SubEntry_Class; // // //class CUSTOM_LOADER_ENTRY_SETTINGS; //class GUI_Custom_Entry_Class; //template class XmlArray; //void CompareCustomEntries(const XString8& label, const XObjArray& olDCustomEntries, const XmlArray& newCustomEntries); class GUI_Custom_Entry_Class : public XmlDict { using super = XmlDict; public: // const GUI_Custom_Entry_Class* Parent = NULL; class HiddenClass: public XmlUnion { using super = XmlUnion; public: XmlBool xmlBool = XmlBool(); XmlString8AllowEmpty xmlString8 = XmlString8AllowEmpty(); XmlUnionField m_fields[2] = { xmlBool, xmlString8}; virtual void getFields(XmlUnionField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; virtual bool validate(XmlLiteParser* xmlLiteParser, const XString8& xmlPath, const XmlParserPosition& keyPos, bool generateErrors) override { RETURN_IF_FALSE( super::validate(xmlLiteParser, xmlPath, keyPos, generateErrors) ); if ( !xmlString8.isDefined() ) return true; if ( xmlString8.value().isEqualIC("Always") ) return true; xmlLiteParser->addWarning(generateErrors, S8Printf("Expecting a boolean or \"Always\" for tag '%s:%d'.", xmlPath.c_str(), keyPos.getLine())); return false; // parsing can continue. } }; protected: XmlBool Disabled = XmlBool(); XmlStringW Volume = XmlStringW(); XmlStringW Path = XmlStringW(); XmlStringW Settings = XmlStringW(); XmlBool CommonSettings = XmlBool(); XmlString8AllowEmpty AddArguments = XmlString8AllowEmpty(); XmlString8AllowEmpty Arguments = XmlString8AllowEmpty(); XmlString8AllowEmpty Title = XmlString8AllowEmpty(); XmlString8AllowEmpty FullTitle = XmlString8AllowEmpty(); XmlStringW ImagePath = XmlStringW(); XmlData ImageData = XmlData(); XmlStringW DriveImagePath = XmlStringW(); XmlData DriveImageData = XmlData(); XmlString8AllowEmpty Hotkey = XmlString8AllowEmpty(); // todo check it's only one UTF16 char (or maybe we should switch to char32_t) XmlBoolOrStringOrData CustomLogo = XmlBoolOrStringOrData(); // Todo validate XmlString8AllowEmpty BootBgColor = XmlString8AllowEmpty(); HiddenClass Hidden = HiddenClass(); // Type XmlString8AllowEmpty Type = XmlString8AllowEmpty(); GUI_Custom_Entry_VolumeType_Class VolumeType = GUI_Custom_Entry_VolumeType_Class(); XmlBoolOrString InjectKexts = XmlBoolOrString(); XmlBool NoCaches = XmlBool(); XmlString8AllowEmpty Kernel = XmlString8AllowEmpty(); XmlBool ForceTextMode = XmlBool(); // 2021-04-22 public: XmlArray SubEntries = XmlArray(); protected: XmlDictField m_fields[24] = { {"Disabled", Disabled}, {"Volume", Volume}, {"Path", Path}, {"Settings", Settings}, {"CommonSettings", CommonSettings}, {"AddArguments", AddArguments}, {"Arguments", Arguments}, {"Title", Title}, {"FullTitle", FullTitle}, {"Image", ImagePath}, {"ImageData", ImageData}, {"DriveImage", DriveImagePath}, {"DriveImageData", DriveImageData}, {"Hotkey", Hotkey}, {"CustomLogo", CustomLogo}, {"BootBgColor", BootBgColor}, {"Hidden", Hidden}, {"Type", Type}, {"VolumeType", VolumeType}, {"InjectKexts", InjectKexts}, {"NoCaches", NoCaches}, {"Kernel", Kernel}, {"SubEntries", SubEntries}, {"ForceTextMode", ForceTextMode}, }; public: virtual void getFields(XmlDictField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; virtual bool validate(XmlLiteParser* xmlLiteParser, const XString8& xmlPath, const XmlParserPosition& keyPos, bool generateErrors) override; bool dgetDisabled() const { return Disabled.isDefined() ? Disabled.value() : false; }; const XStringW& dgetVolume() const { return Volume.isDefined() ? Volume.value() : NullXStringW; }; const XStringW& dgetPath() const { return Path.isDefined() ? Path.value() : NullXStringW; }; const XStringW& dgetSettings() const { return Settings.isDefined() ? Settings.value() : NullXStringW; }; BOOLEAN dgetCommonSettings() const { return CommonSettings.isDefined() ? CommonSettings.value() : false; }; const XString8& dgetAddArguments() const { return AddArguments.isDefined() ? AddArguments.value() : NullXString8; }; const undefinable_XString8 dgetArguments() const { return Arguments.isDefined() ? undefinable_XString8(Arguments.value()) : undefinable_XString8(); }; const XString8& dgetm_Title() const { return Title.isDefined() ? Title.value() : NullXString8; }; const XString8& dgetFullTitle() const { return FullTitle.isDefined() ? FullTitle.value() : NullXString8; }; const XStringW& dgetm_ImagePath() const { return ImagePath.isDefined() ? ImagePath.value() : NullXStringW; }; const XBuffer& dgetImageData() const { return ImageData.isDefined() ? ImageData.value() : XBuffer::NullXBuffer; }; const XStringW& dgetm_DriveImagePath() const { return DriveImagePath.isDefined() ? DriveImagePath.value() : NullXStringW; }; const XBuffer& dgetDriveImageData() const { return DriveImageData.isDefined() ? DriveImageData.value() : XBuffer::NullXBuffer; }; CHAR16 dgetHotkey() const { return Hotkey.isDefined() && Hotkey.value().notEmpty() ? Hotkey.value()[0] : 0; }; const XString8& dgetCustomLogoAsXString8() const { return CustomLogo.xmlString8.isDefined() ? CustomLogo.xmlString8.value() : NullXString8; }; const XBuffer& dgetCustomLogoAsData() const { return CustomLogo.xmlData.isDefined() ? CustomLogo.xmlData.value() : XBuffer::NullXBuffer; }; UINT8 dgetCustomLogoTypeSettings() const { if ( CustomLogo.xmlBool.isDefined() && CustomLogo.xmlBool.value() ) return CUSTOM_BOOT_APPLE; if ( CustomLogo.xmlString8.isDefined() ) { if ( CustomLogo.xmlString8.value() == "Apple"_XS8 ) return CUSTOM_BOOT_APPLE; if ( CustomLogo.xmlString8.value() == "Alternate"_XS8 ) return CUSTOM_BOOT_ALT_APPLE; if ( CustomLogo.xmlString8.value() == "Theme"_XS8 ) return CUSTOM_BOOT_THEME; return CUSTOM_BOOT_USER; } if ( CustomLogo.xmlData.isDefined() ) { return CUSTOM_BOOT_USER; } return CUSTOM_BOOT_DISABLED; } EFI_GRAPHICS_OUTPUT_BLT_PIXEL dgetBootBgColor() const { if ( !BootBgColor.isDefined() ) return EFI_GRAPHICS_OUTPUT_BLT_PIXEL({0,0,0,0}); UINTN Color = AsciiStrHexToUintn(BootBgColor.value()); return EFI_GRAPHICS_OUTPUT_BLT_PIXEL({uint8_t((Color >> 8) & 0xFF),uint8_t((Color >> 16) & 0xFF),uint8_t((Color >> 24) & 0xFF),uint8_t((Color >> 0) & 0xFF)}); } bool dgetHidden() const { return Hidden.isDefined() && Hidden.xmlBool.isDefined() ? Hidden.xmlBool.value() : false; }; bool dgetAlwaysHidden() const { return Hidden.isDefined() && Hidden.xmlString8.isDefined() ? Hidden.xmlString8.value().isEqualIC("Always") : false; }; UINT8 dgetType() const { if ( Type.isDefined() ) { if ((Type.value().isEqualIC("OSX")) || (Type.value() .isEqualIC("macOS"))) { return OSTYPE_OSX; } else if (Type.value() .isEqualIC("OSXInstaller")) { return OSTYPE_OSX_INSTALLER; } else if (Type.value() .isEqualIC("OSXRecovery")) { return OSTYPE_RECOVERY; } else if (Type.value() .isEqualIC("Windows")) { return OSTYPE_WINEFI; } else if (Type.value() .isEqualIC("Linux")) { return OSTYPE_LIN; } else if (Type.value() .isEqualIC("LinuxKernel")) { return OSTYPE_LINEFI; } else { return OSTYPE_OTHER; } } else { if ( dgetPath().notEmpty() ) { // Try to set Entry->type from Entry->Path return GetOSTypeFromPath(dgetPath()); }else{ return 0; } } } UINT8 dgetVolumeType() const { return VolumeType.isDefined() ? VolumeType.dgetVolumeType() : /*Parent ? Parent->dgetVolumeType() :*/ 0; } // VolumeType is duplicated in DuplicateCustomEntry(), but unconditionnally assigned in FillinCustomEntry(). So no "inheritance" from parent. INT8 dgetInjectKexts() const { if ( OSTYPE_IS_OSX(dgetType()) || OSTYPE_IS_OSX_RECOVERY(dgetType()) || OSTYPE_IS_OSX_INSTALLER(dgetType()) ) { if ( InjectKexts.xmlBool.isDefined() ) return InjectKexts.xmlBool.value(); if ( InjectKexts.xmlString8.isDefined() && InjectKexts.xmlString8.value().isEqualIC("Detect") ) return 2; } return -1; } undefinable_bool dgetNoCaches() const { return NoCaches.isDefined() && NoCaches.value() ? undefinable_bool(true) : undefinable_bool(); }; UINT8 dgetKernelScan() const { if ( !Kernel.isDefined() ) return KERNEL_SCAN_ALL; if ((Kernel.value()[0] == 'N') || (Kernel.value()[0] == 'n')) { return KERNEL_SCAN_NEWEST; } else if ((Kernel.value()[0] == 'O') || (Kernel.value()[0] == 'o')) { return KERNEL_SCAN_OLDEST; } else if ((Kernel.value()[0] == 'F') || (Kernel.value()[0] == 'f')) { return KERNEL_SCAN_FIRST; } else if ((Kernel.value()[0] == 'L') || (Kernel.value()[0] == 'l')) { return KERNEL_SCAN_LAST; } else if ((Kernel.value()[0] == 'M') || (Kernel.value()[0] == 'm')) { return KERNEL_SCAN_MOSTRECENT; } else if ((Kernel.value()[0] == 'E') || (Kernel.value()[0] == 'e')) { return KERNEL_SCAN_EARLIEST; } return KERNEL_SCAN_ALL; } // 2021-04-22 decltype(ForceTextMode)::ValueType dgetForceTextMode() const { return ForceTextMode.isDefined() ? ForceTextMode.value() : false; }; // /* calculated values */ // UINT8 getFlags(bool NoCachesDefault) const { // UINT8 Flags = 0; // if ( Arguments.isDefined() ) Flags = OSFLAG_SET(Flags, OSFLAG_NODEFAULTARGS); // if ( dgetAlwaysHidden() ) Flags = OSFLAG_SET(Flags, OSFLAG_DISABLED); // if ( dgetType() == OSTYPE_LIN ) Flags = OSFLAG_SET(Flags, OSFLAG_NODEFAULTARGS); // if (OSTYPE_IS_OSX(dgetType()) || OSTYPE_IS_OSX_RECOVERY(dgetType()) || OSTYPE_IS_OSX_INSTALLER(dgetType())) { // Flags = OSFLAG_UNSET(Flags, OSFLAG_NOCACHES); // } // if ( NoCaches.isDefined() ) { // if ( NoCaches ) Flags = OSFLAG_SET(Flags, OSFLAG_NOCACHES); // }else{ // if (NoCachesDefault) { // Flags = OSFLAG_SET(Flags, OSFLAG_NOCACHES); // } // } // if ( SubEntries.notEmpty() ) Flags = OSFLAG_SET(Flags, OSFLAG_NODEFAULTMENU); // return Flags; // } }; //class CUSTOM_LOADER_SUBENTRY_SETTINGS; //class GUI_Custom_SubEntry_Class; //template class XmlArray; //void CompareCustomSubEntries(const XString8& label, const XObjArray& olDCustomEntries, const XmlArray& newCustomEntries); class GUI_Custom_SubEntry_Class : public XmlDict { using super = XmlDict; public: const GUI_Custom_Entry_Class* Parent = NULL; protected: XmlBool Disabled = XmlBool(); XmlString8AllowEmpty AddArguments = XmlString8AllowEmpty(); XmlString8AllowEmpty Arguments = XmlString8AllowEmpty(); XmlString8AllowEmpty Title = XmlString8AllowEmpty(); XmlString8AllowEmpty FullTitle = XmlString8AllowEmpty(); XmlBool NoCaches = XmlBool(); public: protected: XmlDictField m_fields[6] = { {"Disabled", Disabled}, {"AddArguments", AddArguments}, {"Arguments", Arguments}, {"Title", Title}, {"FullTitle", FullTitle}, {"NoCaches", NoCaches}, }; public: virtual void getFields(XmlDictField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; virtual bool validate(XmlLiteParser* xmlLiteParser, const XString8& xmlPath, const XmlParserPosition& keyPos, bool generateErrors) override { if ( !super::validate(xmlLiteParser, xmlPath, keyPos, generateErrors) ) return false; return true; } bool dgetDisabled() const { return Disabled.isDefined() ? Disabled.value() : false; }; const XString8& dget_AddArguments() const { return AddArguments.isDefined() ? AddArguments.value() : NullXString8; }; const undefinable_XString8 dget_Arguments() const { return Arguments.isDefined() ? undefinable_XString8(Arguments.value()) : undefinable_XString8(); }; undefinable_XString8 dget_Title() const { return Title.isDefined() ? undefinable_XString8(Title.value()) : undefinable_XString8(); }; undefinable_XString8 dget_FullTitle() const { return FullTitle.isDefined() ? undefinable_XString8(FullTitle.value()) : undefinable_XString8(); }; undefinable_bool dget_NoCaches() const { return NoCaches.isDefined() && NoCaches.value() ? undefinable_bool(true) : undefinable_bool(); }; // XString8 dget_Title() const { // if ( Title.isDefined() && Title.xstring8.notEmpty() ) return Title; // if ( OSTYPE_IS_OSX_RECOVERY(Parent->dgetType()) ) { // return "Recovery"_XS8; // TODO use a shared static // } else if ( OSTYPE_IS_OSX_INSTALLER(Parent->dgetType()) ) { // return "Install macOS"_XS8; // TODO use a shared static // } // return NullXStringW; // // if ( Title.isDefined() ) return Title.xstring8; // if ( FullTitle.isDefined() ) return NullXString8; // if ( Parent ) return Parent->dgetTitle(); // return NullXString8; // }; // const XString8& dget_FullTitle() const { // if ( FullTitle.isDefined() ) return FullTitle.xstring8; // if ( Title.isDefined() ) return NullXString8; // if ( Parent ) return Parent->dgetFullTitle(); // return NullXString8; // }; // class CUSTOM_LOADER_SUBENTRY_SETTINGS; // friend void ::CompareCustomSubEntries(const XString8& label, const XObjArray<::CUSTOM_LOADER_SUBENTRY_SETTINGS>& olDCustomEntries, const XmlArray& newCustomEntries); }; class GUI_Custom_Legacy_Class : public XmlDict { using super = XmlDict; protected: XmlBool Disabled = XmlBool(); XmlString8AllowEmpty Volume = XmlString8AllowEmpty(); XmlString8AllowEmpty Title = XmlString8AllowEmpty(); XmlString8AllowEmpty FullTitle = XmlString8AllowEmpty(); XmlString8AllowEmpty ImagePath = XmlString8AllowEmpty(); XmlData ImageData = XmlData(); XmlString8AllowEmpty DriveImagePath = XmlString8AllowEmpty(); XmlData DriveImageData = XmlData(); class HotKeyClass : public XmlString8AllowEmpty { using super = XmlString8AllowEmpty; virtual bool validate(XmlLiteParser* xmlLiteParser, const XString8& xmlPath, const XmlParserPosition& keyPos, bool generateErrors) override { RETURN_IF_FALSE( super::validate(xmlLiteParser, xmlPath, keyPos, generateErrors) ); if ( !isDefined() ) return true; if ( xstring8.length() != 1 ) { xmlLiteParser->addError(generateErrors, S8Printf("HotKey must a string of only one UTF-16 char at line %d.", keyPos.getLine())); return false; } if ( xstring8[0] >= __WCHAR_MAX__ ) { xmlLiteParser->addError(generateErrors, S8Printf("HotKey must be a UTF-16 char at line %d.", keyPos.getLine())); return false; } return true; } } Hotkey = HotKeyClass(); XmlBoolOrString Hidden = XmlBoolOrString(); XmlString8AllowEmpty Type = XmlString8AllowEmpty(); GUI_Custom_Entry_VolumeType_Class VolumeType = GUI_Custom_Entry_VolumeType_Class(); XmlDictField m_fields[12] = { {"Disabled", Disabled}, {"Volume", Volume}, {"Title", Title}, {"FullTitle", FullTitle}, {"Image", ImagePath}, {"ImageData", ImageData}, {"DriveImage", DriveImagePath}, {"DriveImageData", DriveImageData}, {"Hotkey", Hotkey}, {"Hidden", Hidden}, {"Type", Type}, {"VolumeType", VolumeType}, }; public: virtual void getFields(XmlDictField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; bool dgetDisabled() const { return Disabled.isDefined() ? Disabled.value() : false; }; const XString8& dgetVolume() const { return Volume.isDefined() ? Volume.value() : NullXString8; }; const XString8& dgetTitle() const { return Title.isDefined() ? Title.value() : NullXString8; }; const XString8& dgetFullTitle() const { return FullTitle.isDefined() ? FullTitle.value() : NullXString8; }; const XString8& dgetImagePath() const { return ImagePath.isDefined() ? ImagePath.value() : NullXString8; }; const XBuffer& dgetImageData() const { return ImageData.isDefined() ? ImageData.value() : XBuffer::NullXBuffer; }; const XString8& dgetDriveImagePath() const { return DriveImagePath.isDefined() ? DriveImagePath.value() : NullXString8; }; const XBuffer& dgetDriveImageData() const { return DriveImageData.isDefined() ? DriveImageData.value() : XBuffer::NullXBuffer; }; CHAR16 dgetHotkey() const { return Hotkey.isDefined() && Hotkey.value().notEmpty() ? Hotkey.value()[0] : 0; }; bool dgetHidden() const { return Hidden.isDefined() && Hidden.xmlBool.isDefined() ? Hidden.xmlBool.value() : false; }; bool dgetAlwaysHidden() const { return Hidden.isDefined() && Hidden.xmlString8.isDefined() ? Hidden.xmlString8.value().isEqualIC("Always") : false; }; UINT8 dgetType() const { if ( Type.isDefined() ) { if (Type.value().isEqualIC("Windows")) { return OSTYPE_WIN; } else if (Type.value().isEqualIC("Linux")) { return OSTYPE_LIN; } else { return OSTYPE_OTHER; } } return 0; } UINT8 dgetVolumeType() const { return VolumeType.isDefined() ? VolumeType.dgetVolumeType() : /*Parent ? Parent->dgetVolumeType() :*/ 0; } // VolumeType is duplicated in DuplicateCustomEntry(), but unconditionnally assigned in FillinCustomEntry(). So no "inheritance" from parent. }; class GUI_Custom_Tool_Class : public XmlDict { using super = XmlDict; protected: XmlBool Disabled = XmlBool(); XmlString8AllowEmpty Volume = XmlString8AllowEmpty(); XmlString8AllowEmpty Path = XmlString8AllowEmpty(); XmlString8AllowEmpty Arguments = XmlString8AllowEmpty(); XmlString8AllowEmpty Title = XmlString8AllowEmpty(); XmlString8AllowEmpty FullTitle = XmlString8AllowEmpty(); XmlString8AllowEmpty ImagePath = XmlString8AllowEmpty(); XmlData ImageData = XmlData(); XmlString8AllowEmpty Hotkey = XmlString8AllowEmpty(); XmlBoolOrString Hidden = XmlBoolOrString(); GUI_Custom_Entry_VolumeType_Class VolumeType = GUI_Custom_Entry_VolumeType_Class(); XmlDictField m_fields[11] = { {"Disabled", Disabled}, {"Volume", Volume}, {"Path", Path}, {"Arguments", Arguments}, {"Title", Title}, {"FullTitle", FullTitle}, {"Image", ImagePath}, {"ImageData", ImageData}, {"Hotkey", Hotkey}, {"Hidden", Hidden}, {"VolumeType", VolumeType}, }; public: virtual void getFields(XmlDictField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; bool dgetDisabled() const { return Disabled.isDefined() ? Disabled.value() : false; }; const XString8& dgetVolume() const { return Volume.isDefined() ? Volume.value() : NullXString8; }; const XString8& dgetPath() const { return Path.isDefined() ? Path.value() : NullXString8; }; const XString8& dgetArguments() const { return Arguments.isDefined() ? Arguments.value() : NullXString8; }; const XString8& dgetTitle() const { return Title.isDefined() ? Title.value() : NullXString8; }; const XString8& dgetFullTitle() const { return FullTitle.isDefined() ? FullTitle.value() : NullXString8; }; const XString8& dgetImagePath() const { return ImagePath.isDefined() ? ImagePath.value() : NullXString8; }; const XBuffer& dgetImageData() const { return ImageData.isDefined() ? ImageData.value() : XBuffer::NullXBuffer; }; CHAR16 dgetHotkey() const { return Hotkey.isDefined() && Hotkey.value().notEmpty() ? Hotkey.value()[0] : 0; }; bool dgetHidden() const { return Hidden.isDefined() && Hidden.xmlBool.isDefined() ? Hidden.xmlBool.value() : false; }; bool dgetAlwaysHidden() const { return Hidden.isDefined() && Hidden.xmlString8.isDefined() ? Hidden.xmlString8.value().isEqualIC("Always") : false; }; UINT8 dgetVolumeType() const { return VolumeType.isDefined() ? VolumeType.dgetVolumeType() : /*Parent ? Parent->dgetVolumeType() :*/ 0; } // VolumeType is duplicated in DuplicateCustomEntry(), but unconditionnally assigned in FillinCustomEntry(). So no "inheritance" from parent. }; public: XmlArray Entries = XmlArray(); XmlArray Legacy = XmlArray(); XmlArray Tool = XmlArray(); protected: XmlDictField m_fields[3] = { {"Entries", Entries}, {"Legacy", Legacy}, {"Tool", Tool}, }; public: virtual void getFields(XmlDictField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; }; // -------------------------------------------------- GUI class protected: // Timezone XmlInt32 Timezone = XmlInt32(); // TODO : checkeck Timezone validity // Theme XmlString8AllowEmpty Theme = XmlString8AllowEmpty(); // EmbeddedThemeType class EmbeddedThemeTypeClass: public XmlString8AllowEmpty { using super = XmlString8AllowEmpty; virtual bool validate(XmlLiteParser* xmlLiteParser, const XString8& xmlPath, const XmlParserPosition& keyPos, bool generateErrors) override { RETURN_IF_FALSE( super::validate(xmlLiteParser, xmlPath, keyPos, generateErrors) ); if ( xstring8.isEqualIC("Dark") ) return true; if ( xstring8.isEqualIC("Light") ) return true; if ( xstring8.isEqualIC("Daytime") ) return true; xmlLiteParser->addError(generateErrors, S8Printf("EmbeddedThemeType must \"Dark\" \"Light\" or \"Daytime\" at line %d.", keyPos.getLine())); return false; } } EmbeddedThemeType = EmbeddedThemeTypeClass(); // PlayAsync XmlBool PlayAsync = XmlBool(); // CustomIcons XmlBool CustomIcons = XmlBool(); // TextOnly XmlBool TextOnly = XmlBool(); // ShowOptimus XmlBool ShowOptimus = XmlBool(); // ScreenResolution class ScreenResolutionClass: public XmlString8AllowEmpty { using super = XmlString8AllowEmpty; virtual bool validate(XmlLiteParser* xmlLiteParser, const XString8& xmlPath, const XmlParserPosition& keyPos, bool generateErrors) override { RETURN_IF_FALSE( super::validate(xmlLiteParser, xmlPath, keyPos, generateErrors) ); // TODO Check that resolution is Integer x Integer return true; xmlLiteParser->addError(generateErrors, S8Printf("ScreenResolution must be {width}x{height} at line %d.", keyPos.getLine())); return false; } } ScreenResolution = ScreenResolutionClass(); // ProvideConsoleGop XmlBool ProvideConsoleGop = XmlBool(); // ConsoleMode class ConsoleModeClass : public XmlInt64OrString { using super = XmlInt64OrString; public: // ConsoleMode_Class() : XmlInt64OrString(false) {}; virtual bool validate(XmlLiteParser* xmlLiteParser, const XString8& xmlPath, const XmlParserPosition& keyPos, bool generateErrors) override { RETURN_IF_FALSE( super::validate(xmlLiteParser, xmlPath, keyPos, generateErrors) ); if ( !xmlString8.isDefined() ) return true; if ( xmlString8.value().isEqualIC("Max") ) return true; if ( xmlString8.value().isEqualIC("Min") ) return true; xmlLiteParser->addWarning(generateErrors, S8Printf("Expecting an integer or \"Min\" or \"Max\" for tag '%s:%d'.", xmlPath.c_str(), keyPos.getLine())); return false; } } ConsoleMode = ConsoleModeClass(); // Language class LanguageClass: public XmlString8AllowEmpty { using super = XmlString8AllowEmpty; virtual bool validate(XmlLiteParser* xmlLiteParser, const XString8& xmlPath, const XmlParserPosition& keyPos, bool generateErrors) override { RETURN_IF_FALSE( super::validate(xmlLiteParser, xmlPath, keyPos, generateErrors) ); if ( xstring8.containsIC("en") ) return true; if ( xstring8.containsIC("ru") ) return true; if ( xstring8.containsIC("ua") ) return true; if ( xstring8.containsIC("fr") ) return true; if ( xstring8.containsIC("it") ) return true; if ( xstring8.containsIC("es") ) return true; if ( xstring8.containsIC("pt") ) return true; if ( xstring8.containsIC("br") ) return true; if ( xstring8.containsIC("de") ) return true; if ( xstring8.containsIC("nl") ) return true; if ( xstring8.containsIC("pl") ) return true; if ( xstring8.containsIC("cz") ) return true; if ( xstring8.containsIC("hr") ) return true; if ( xstring8.containsIC("id") ) return true; if ( xstring8.containsIC("zh_CN") ) return true; if ( xstring8.containsIC("ro") ) return true; if ( xstring8.containsIC("ko") ) return true; xmlLiteParser->addError(generateErrors, S8Printf("Language '%s' is unknown at line %d. Ignored.", xstring8.c_str(), keyPos.getLine())); return false; } } Language = LanguageClass(); // KbdPrevLang XmlBool KbdPrevLang = XmlBool(); public: // Mouse GUI_Mouse_Class Mouse = GUI_Mouse_Class(); // Hide protected: XmlString8Array Hide = XmlString8Array(); public: // Scan GUI_Scan_Class Scan = GUI_Scan_Class(*this); // Custom GUI_Custom_Class Custom = GUI_Custom_Class(); protected: XmlDictField m_fields[16] = { {"Timezone", Timezone}, {"Theme", Theme}, {"EmbeddedThemeType", EmbeddedThemeType}, {"PlayAsync", PlayAsync}, {"CustomIcons", CustomIcons}, {"TextOnly", TextOnly}, {"ShowOptimus", ShowOptimus}, {"ScreenResolution", ScreenResolution}, {"ProvideConsoleGop", ProvideConsoleGop}, {"ConsoleMode", ConsoleMode}, {"Language", Language}, {"KbdPrevLang", KbdPrevLang}, {"Mouse", Mouse}, {"Hide", Hide}, {"Scan", Scan}, {"Custom", Custom}, }; public: virtual void getFields(XmlDictField** fields, size_t* nb) override { *fields = m_fields; *nb = sizeof(m_fields)/sizeof(m_fields[0]); }; // virtual bool validate(XmlLiteParser* xmlLiteParser, const char* name, XmlAbstractType* xmlTyp, const XString8& xmlPath, const XmlParserPosition& pos, bool generateErrors) override; int32_t dgetTimezone() const { return Timezone.isDefined() ? Timezone.value() : 0xFF; }; const XString8& dgetTheme() const { return Theme.isDefined() ? Theme.value() : NullXString8; }; const XString8& dgetEmbeddedThemeType() const { return EmbeddedThemeType.isDefined() ? EmbeddedThemeType.value() : NullXString8; }; bool dgetPlayAsync() const { return PlayAsync.isDefined() ? PlayAsync.value() : false; }; bool dgetCustomIcons() const { return CustomIcons.isDefined() ? CustomIcons.value() : false; }; bool dgetTextOnly() const { return TextOnly.isDefined() ? TextOnly.value() : false; }; bool dgetShowOptimus() const { return ShowOptimus.isDefined() ? ShowOptimus.value() : false; }; const XString8& dgetScreenResolution() const { return ScreenResolution.isDefined() ? ScreenResolution.value() : NullXString8; }; bool dgetProvideConsoleGop() const { return isDefined() ? ProvideConsoleGop.isDefined() ? ProvideConsoleGop.value() : true : false; }; // TODO: different default value if section is not defined int64_t dgetConsoleMode() const { if ( ConsoleMode.xmlInt64.isDefined() ) { return ConsoleMode.xmlInt64.value(); } else if ( ConsoleMode.xmlString8.isDefined() ) { if ( ConsoleMode.xmlString8.value().contains("Max") ) { return -1; } else if ( ConsoleMode.xmlString8.value().contains("Min") ) { return -2; } else { return (INT32)AsciiStrDecimalToUintn(ConsoleMode.xmlString8.value()); } } return 0; } const decltype(Language)::ValueType& dgetLanguage() const { return Language.isDefined() ? Language.value() : Language.nullValue; }; LanguageCode dgetlanguageCode() const { if ( !Language.isDefined() ) return english; if ( Language.value().contains("en") ) { return english; } else if ( Language.value().contains("ru")) { return russian; } else if ( Language.value().contains("ua")) { return ukrainian; } else if ( Language.value().contains("fr")) { return french; //default is extended latin } else if ( Language.value().contains("it")) { return italian; } else if ( Language.value().contains("es")) { return spanish; } else if ( Language.value().contains("pt")) { return portuguese; } else if ( Language.value().contains("br")) { return brasil; } else if ( Language.value().contains("de")) { return german; } else if ( Language.value().contains("nl")) { return dutch; } else if ( Language.value().contains("pl")) { return polish; } else if ( Language.value().contains("cz")) { return czech; } else if ( Language.value().contains("hr")) { return croatian; } else if ( Language.value().contains("id")) { return indonesian; } else if ( Language.value().contains("zh_CN")) { return chinese; } else if ( Language.value().contains("ro")) { return romanian; } else if ( Language.value().contains("ko")) { return korean; } return english; } bool dgetKbdPrevLang() const { return KbdPrevLang.isDefined() ? KbdPrevLang.value() : false; }; /* calculated value */ bool getDarkEmbedded(bool isDaylight) const { if ( !EmbeddedThemeType.isDefined() ) return false; if ( EmbeddedThemeType.value().isEqualIC("Dark") ) return true; if ( EmbeddedThemeType.value().isEqualIC("Daytime") ) return !isDaylight; return false; } const decltype(Hide)::ValueType& dgetHVHideStrings() const { return Hide.isDefined() ? Hide.value() : Hide.nullValue; }; }; #endif /* _CONFIGPLISTCLASS_GUI_H_ */