diff --git a/android/script/replace_links.bat b/android/script/replace_links.bat index df4a2fd991..1f67247bc5 100644 --- a/android/script/replace_links.bat +++ b/android/script/replace_links.bat @@ -40,7 +40,9 @@ cp -r ../data/sound-strings/ assets/ cp -r ../data/countries-strings/ assets/ cp -r ../data/cuisine-strings/ assets/ -cp -r ../data/icudt57l.dat/ assets/ +cp -r ../data/icudt57l.dat assets/ + +cp -r ../data/local_ads_symbols.txt assets/ rm -rf flavors/mwm-ttf-assets mkdir flavors\\mwm-ttf-assets diff --git a/drape_frontend/watch/cpu_drawer.cpp b/drape_frontend/watch/cpu_drawer.cpp index 499582ea32..61dabb5163 100644 --- a/drape_frontend/watch/cpu_drawer.cpp +++ b/drape_frontend/watch/cpu_drawer.cpp @@ -202,12 +202,12 @@ void CPUDrawer::Flush() void CPUDrawer::DrawMyPosition(m2::PointD const & myPxPotision) { - m_renderer->DrawSymbol(myPxPotision, dp::Center, IconInfo("watch-my-position")); + m_renderer->DrawSymbol(myPxPotision, dp::Center, IconInfo("current-position")); } void CPUDrawer::DrawSearchResult(m2::PointD const & pxPosition) { - m_renderer->DrawSymbol(pxPosition, dp::Top, IconInfo("watch-search-result")); + m_renderer->DrawSymbol(pxPosition, dp::Top, IconInfo("search-result")); } void CPUDrawer::DrawSearchArrow(double azimut) diff --git a/iphone/Maps/Maps.xcodeproj/project.pbxproj b/iphone/Maps/Maps.xcodeproj/project.pbxproj index b75087ccaa..1656962e7e 100644 --- a/iphone/Maps/Maps.xcodeproj/project.pbxproj +++ b/iphone/Maps/Maps.xcodeproj/project.pbxproj @@ -507,6 +507,7 @@ 34F742321E0834F400AC1FD6 /* UIViewController+Navigation.mm in Sources */ = {isa = PBXBuildFile; fileRef = 34F742301E0834F400AC1FD6 /* UIViewController+Navigation.mm */; }; 34F742331E0834F400AC1FD6 /* UIViewController+Navigation.mm in Sources */ = {isa = PBXBuildFile; fileRef = 34F742301E0834F400AC1FD6 /* UIViewController+Navigation.mm */; }; 34FE4C451BCC013500066718 /* MWMMapWidgets.mm in Sources */ = {isa = PBXBuildFile; fileRef = 34FE4C441BCC013500066718 /* MWMMapWidgets.mm */; }; + 450703091E9E6CF000E8C029 /* local_ads_symbols.txt in Resources */ = {isa = PBXBuildFile; fileRef = 450703081E9E6CF000E8C029 /* local_ads_symbols.txt */; }; 4519503A1B7A3E070085DA05 /* patterns.txt in Resources */ = {isa = PBXBuildFile; fileRef = 451950391B7A3E070085DA05 /* patterns.txt */; }; 452FCA3B1B6A3DF7007019AB /* colors.txt in Resources */ = {isa = PBXBuildFile; fileRef = 452FCA3A1B6A3DF7007019AB /* colors.txt */; }; 4554B6EA1E55F02B0084017F /* drules_proto_vehicle_clear.bin in Resources */ = {isa = PBXBuildFile; fileRef = 4554B6E81E55F02B0084017F /* drules_proto_vehicle_clear.bin */; }; @@ -1865,6 +1866,7 @@ 34FE4C431BCC013500066718 /* MWMMapWidgets.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = MWMMapWidgets.h; sourceTree = ""; }; 34FE4C441BCC013500066718 /* MWMMapWidgets.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = MWMMapWidgets.mm; sourceTree = ""; }; 3DDB4BC31DAB98F000F4D021 /* libpartners_api.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = libpartners_api.a; path = "../../../omim-xcode-build/Debug-iphonesimulator/libpartners_api.a"; sourceTree = ""; }; + 450703081E9E6CF000E8C029 /* local_ads_symbols.txt */ = {isa = PBXFileReference; lastKnownFileType = text; name = local_ads_symbols.txt; path = ../../data/local_ads_symbols.txt; sourceTree = ""; }; 451950391B7A3E070085DA05 /* patterns.txt */ = {isa = PBXFileReference; lastKnownFileType = text; name = patterns.txt; path = ../../data/patterns.txt; sourceTree = ""; }; 452FCA3A1B6A3DF7007019AB /* colors.txt */ = {isa = PBXFileReference; lastKnownFileType = text; name = colors.txt; path = ../../data/colors.txt; sourceTree = ""; }; 4554B6E81E55F02B0084017F /* drules_proto_vehicle_clear.bin */ = {isa = PBXFileReference; lastKnownFileType = archive.macbinary; name = drules_proto_vehicle_clear.bin; path = ../../data/drules_proto_vehicle_clear.bin; sourceTree = ""; }; @@ -4080,6 +4082,7 @@ FA065FC61286143F00FEA989 /* External Resources */ = { isa = PBXGroup; children = ( + 450703081E9E6CF000E8C029 /* local_ads_symbols.txt */, BB7626B41E8559980031D71C /* icudt57l.dat */, 4554B6E81E55F02B0084017F /* drules_proto_vehicle_clear.bin */, 4554B6E91E55F02B0084017F /* drules_proto_vehicle_dark.bin */, @@ -4324,6 +4327,7 @@ isa = PBXResourcesBuildPhase; buildActionMask = 2147483647; files = ( + 450703091E9E6CF000E8C029 /* local_ads_symbols.txt in Resources */, 4554B6EA1E55F02B0084017F /* drules_proto_vehicle_clear.bin in Resources */, 4554B6EB1E55F02B0084017F /* drules_proto_vehicle_dark.bin in Resources */, EEA61601134C496A003A9827 /* 01_dejavusans.ttf in Resources */, diff --git a/local_ads/CMakeLists.txt b/local_ads/CMakeLists.txt index 21abef6020..4b54b099a1 100644 --- a/local_ads/CMakeLists.txt +++ b/local_ads/CMakeLists.txt @@ -9,6 +9,8 @@ set( event.hpp file_helpers.cpp file_helpers.hpp + icons_info.cpp + icons_info.hpp statistics.cpp statistics.hpp ) diff --git a/local_ads/campaign.hpp b/local_ads/campaign.hpp index 2dca4461b1..172e4f114f 100644 --- a/local_ads/campaign.hpp +++ b/local_ads/campaign.hpp @@ -1,5 +1,7 @@ #pragma once +#include "local_ads/icons_info.hpp" + #include #include @@ -18,9 +20,7 @@ struct Campaign { } - // TODO(mgsergio): Provide a working imlpementation. - std::string GetIconName() const { return "test-l"; } - + std::string GetIconName() const { return IconsInfo::Instance().GetIcon(m_iconId); } uint32_t m_featureId; uint16_t m_iconId; uint8_t m_daysBeforeExpired; diff --git a/local_ads/icons_info.cpp b/local_ads/icons_info.cpp new file mode 100644 index 0000000000..846ff0dcc0 --- /dev/null +++ b/local_ads/icons_info.cpp @@ -0,0 +1,72 @@ +#include "local_ads/icons_info.hpp" + +#include "coding/reader.hpp" + +#include "platform/platform.hpp" + +#include "base/assert.hpp" +#include "base/logging.hpp" +#include "base/string_utils.hpp" + +#include + +namespace +{ +char const kDelimiter = '_'; + +void ParseIconsFile(std::string const & iconsFile, + std::function const & handler) +{ + ASSERT(handler != nullptr, ()); + try + { + std::string fileData; + ReaderPtr(GetPlatform().GetReader(iconsFile)).ReadAsString(fileData); + strings::Tokenize(fileData, "\n", [&handler](std::string const & str) { + if (!str.empty()) + handler(str); + }); + } + catch (Reader::Exception const & e) + { + LOG(LWARNING, ("Error reading file ", iconsFile, " : ", e.what())); + } +} +} // namespace + +namespace local_ads +{ +IconsInfo & IconsInfo::Instance() +{ + static IconsInfo iconsInfo; + return iconsInfo; +} + +void IconsInfo::SetSourceFile(std::string const & fileName) +{ + std::map icons; + ParseIconsFile(fileName, [&icons](std::string const & icon) { + auto const pos = icon.find(kDelimiter); + if (pos == std::string::npos) + return; + uint32_t index = 0; + strings::to_uint(icon.substr(0, pos), index); + icons[static_cast(index)] = icon; + }); + + { + std::lock_guard lock(m_mutex); + m_fileName = fileName; + std::swap(m_icons, icons); + } +} + +std::string IconsInfo::GetIcon(uint16_t index) const +{ + std::lock_guard lock(m_mutex); + auto const it = m_icons.find(index); + if (it == m_icons.end()) + return {}; + return it->second; +} +} // namespace local_ads diff --git a/local_ads/icons_info.hpp b/local_ads/icons_info.hpp new file mode 100644 index 0000000000..0bcf71d5c8 --- /dev/null +++ b/local_ads/icons_info.hpp @@ -0,0 +1,30 @@ +#pragma once + +#include "base/macros.hpp" + +#include +#include +#include +#include + +namespace local_ads +{ +class IconsInfo +{ +public: + static IconsInfo & Instance(); + + void SetSourceFile(std::string const & fileName); + std::string GetIcon(uint16_t index) const; + +private: + IconsInfo() = default; + ~IconsInfo() {} + + std::string m_fileName; + std::map m_icons; + mutable std::mutex m_mutex; + + DISALLOW_COPY_AND_MOVE(IconsInfo); +}; +} // namespace local_ads diff --git a/local_ads/local_ads.pro b/local_ads/local_ads.pro index 2f3a9e6c00..bb685f2cba 100644 --- a/local_ads/local_ads.pro +++ b/local_ads/local_ads.pro @@ -10,6 +10,7 @@ SOURCES += \ campaign_serialization.cpp \ event.cpp \ file_helpers.cpp \ + icons_info.cpp \ statistics.cpp \ @@ -18,4 +19,5 @@ HEADERS += \ campaign_serialization.hpp \ event.hpp \ file_helpers.hpp \ + icons_info.hpp \ statistics.hpp \ diff --git a/local_ads/local_ads_tests/local_ads_tests.pro b/local_ads/local_ads_tests/local_ads_tests.pro index 20a8e5f018..dc2bcf7a69 100644 --- a/local_ads/local_ads_tests/local_ads_tests.pro +++ b/local_ads/local_ads_tests/local_ads_tests.pro @@ -4,7 +4,7 @@ CONFIG -= app_bundle TEMPLATE = app ROOT_DIR = ../.. -DEPENDENCIES = local_ads platform_tests_support platform coding geometry base stats_client +DEPENDENCIES = platform_tests_support local_ads platform coding geometry base stats_client include($$ROOT_DIR/common.pri) diff --git a/map/local_ads_manager.cpp b/map/local_ads_manager.cpp index 8b0bae5c0c..8f37fa9910 100644 --- a/map/local_ads_manager.cpp +++ b/map/local_ads_manager.cpp @@ -2,6 +2,7 @@ #include "local_ads/campaign_serialization.hpp" #include "local_ads/file_helpers.hpp" +#include "local_ads/icons_info.hpp" #include "drape_frontend/drape_engine.hpp" #include "drape_frontend/visual_params.hpp" @@ -16,6 +17,7 @@ namespace { std::string const kCampaignFile = "local_ads_campaigns.dat"; +std::string const kLocalAdsSymbolsFile = "local_ads_symbols.txt"; auto constexpr kWWanUpdateTimeout = std::chrono::hours(12); void SerializeCampaign(FileWriter & writer, std::string const & countryName, @@ -154,6 +156,9 @@ void LocalAdsManager::UpdateViewport(ScreenBase const & screen) void LocalAdsManager::ThreadRoutine() { + local_ads::IconsInfo::Instance().SetSourceFile( + my::JoinFoldersToPath(GetPlatform().ResourcesDir(), kLocalAdsSymbolsFile)); + std::string const campaignFile = GetPath(kCampaignFile); // Read persistence data. diff --git a/omim.pro b/omim.pro index 319a588b47..3bf4ef0808 100644 --- a/omim.pro +++ b/omim.pro @@ -254,7 +254,7 @@ SUBDIRS = 3party base coding geometry editor indexer routing routing_common sear SUBDIRS *= partners_api_tests local_ads_tests.subdir = local_ads/local_ads_tests - local_ads_tests.depends = base coding platform local_ads + local_ads_tests.depends = base local_ads platform_tests_support platform coding SUBDIRS *= local_ads_tests tracking_tests.subdir = tracking/tracking_tests diff --git a/qt/CMakeLists.txt b/qt/CMakeLists.txt index 87fef21c9a..10d9056c9e 100644 --- a/qt/CMakeLists.txt +++ b/qt/CMakeLists.txt @@ -152,6 +152,7 @@ copy_resources( WorldCoasts.mwm WorldCoasts_obsolete.mwm icudt57l.dat + local_ads_symbols.txt 01_dejavusans.ttf 02_droidsans-fallback.ttf diff --git a/qt/qt.pro b/qt/qt.pro index c372eec28a..061cb0d1fc 100644 --- a/qt/qt.pro +++ b/qt/qt.pro @@ -63,6 +63,7 @@ OTHER_RES.files = ../data/copyright.html ../data/eula.html ../data/welcome.html ../data/languages.txt ../data/categories.txt \ ../data/packed_polygons.bin res/logo.png \ ../data/editor.config \ + ../data/local_ads_symbols.txt \ CLASSIFICATOR_RES.path = $$DATADIR CLASSIFICATOR_RES.files = ../data/classificator.txt \ diff --git a/tools/python/generate_local_ads_symbols.py b/tools/python/generate_local_ads_symbols.py new file mode 100755 index 0000000000..1aab9048dc --- /dev/null +++ b/tools/python/generate_local_ads_symbols.py @@ -0,0 +1,62 @@ +#!/usr/bin/env python + +import os +import sys + +PREFIX_DELIMITER = '_' + + +def enumerate_symbols(symbols_folder_path): + symbols = [] + for filename in os.listdir(symbols_folder_path): + parts = os.path.splitext(filename) + if parts[1] == ".svg": + symbols.append(parts[0]) + return symbols + + +def check_symbols(symbols): + numbers = set() + for s in symbols: + pos = s.find(PREFIX_DELIMITER) + n = s[:pos] + if pos < 0 or not n.isdigit(): + raise ValueError('Symbol ' + s + ' must have a numeric prefix') + elif int(n) in numbers: + raise ValueError('Symbol ' + s + ' has duplicated numeric prefix') + else: + numbers.add(int(n)) + + +if __name__ == '__main__': + if len(sys.argv) < 2: + print('Usage: {0} []'.format(sys.argv[0])) + sys.exit(-1) + + path_to_styles = os.path.join(sys.argv[1], 'clear') + if not os.path.isdir(path_to_styles): + print('Invalid path to styles folder') + sys.exit(-1) + + target_path = '' + if len(sys.argv) >= 3: + target_path = sys.argv[2] + output_name = os.path.join(target_path, 'local_ads_symbols.txt'); + if os.path.exists(output_name): + os.remove(output_name) + + paths = ['style-clear', 'style-night'] + symbols = [] + for folder_path in paths: + s = enumerate_symbols(os.path.join(path_to_styles, folder_path, 'symbols-ad')) + if len(symbols) != 0: + symbols.sort() + s.sort() + if symbols != s: + raise ValueError('Different symbols set in folders' + str(paths)) + else: + symbols = s + check_symbols(symbols) + with open(output_name, "w") as text_file: + for symbol in symbols: + text_file.write(symbol) diff --git a/tools/unix/generate_symbols.sh b/tools/unix/generate_symbols.sh index f1c499e5bc..70a333b16c 100755 --- a/tools/unix/generate_symbols.sh +++ b/tools/unix/generate_symbols.sh @@ -1,9 +1,13 @@ #!/bin/bash set -e -u +# Prevent python from generating compiled *.pyc files +export PYTHONDONTWRITEBYTECODE=1 + OMIM_PATH="${OMIM_PATH:-$(cd "$(dirname "$0")/../.."; pwd)}" SKIN_GENERATOR="$OMIM_PATH/out/release/skin_generator" DATA_PATH="$OMIM_PATH/data" +LOCAL_ADS_SYMBOLS_GENERATOR="$OMIM_PATH/tools/python/generate_local_ads_symbols.py" # If skin_generator does not exist then build it if [ ! -f $SKIN_GENERATOR ]; @@ -103,3 +107,6 @@ BuildSkin clear clear hdpi 27 false _clear symbols-ad -ad BuildSkin clear clear xhdpi 36 false _clear symbols-ad -ad BuildSkin clear clear xxhdpi 54 false _clear symbols-ad -ad BuildSkin clear clear 6plus 54 false _clear symbols-ad -ad + +echo "Generate local ads symbols" +python "$LOCAL_ADS_SYMBOLS_GENERATOR" "$DATA_PATH/styles" "$DATA_PATH" diff --git a/xcode/local_ads/local_ads.xcodeproj/project.pbxproj b/xcode/local_ads/local_ads.xcodeproj/project.pbxproj index 00c49cb0f7..dce6a2ef9e 100644 --- a/xcode/local_ads/local_ads.xcodeproj/project.pbxproj +++ b/xcode/local_ads/local_ads.xcodeproj/project.pbxproj @@ -7,6 +7,8 @@ objects = { /* Begin PBXBuildFile section */ + 450703041E9E6C3400E8C029 /* icons_info.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 450703021E9E6C3400E8C029 /* icons_info.cpp */; }; + 450703051E9E6C3400E8C029 /* icons_info.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 450703031E9E6C3400E8C029 /* icons_info.hpp */; }; 455C5DA51E97EBAC00DBFE48 /* file_helpers_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 455C5DA11E97EBA200DBFE48 /* file_helpers_tests.cpp */; }; 455C5DA61E97EBAF00DBFE48 /* statistics_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 455C5DA21E97EBA200DBFE48 /* statistics_tests.cpp */; }; 455C5DAD1E97EBC300DBFE48 /* event.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 455C5DA71E97EBC300DBFE48 /* event.cpp */; }; @@ -31,6 +33,8 @@ /* End PBXBuildFile section */ /* Begin PBXFileReference section */ + 450703021E9E6C3400E8C029 /* icons_info.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = icons_info.cpp; sourceTree = ""; }; + 450703031E9E6C3400E8C029 /* icons_info.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = icons_info.hpp; sourceTree = ""; }; 455C5DA11E97EBA200DBFE48 /* file_helpers_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = file_helpers_tests.cpp; sourceTree = ""; }; 455C5DA21E97EBA200DBFE48 /* statistics_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = statistics_tests.cpp; sourceTree = ""; }; 455C5DA71E97EBC300DBFE48 /* event.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = event.cpp; sourceTree = ""; }; @@ -114,6 +118,8 @@ 455C5DA81E97EBC300DBFE48 /* event.hpp */, 455C5DA91E97EBC300DBFE48 /* file_helpers.cpp */, 455C5DAA1E97EBC300DBFE48 /* file_helpers.hpp */, + 450703021E9E6C3400E8C029 /* icons_info.cpp */, + 450703031E9E6C3400E8C029 /* icons_info.hpp */, 455C5DAB1E97EBC300DBFE48 /* statistics.cpp */, 455C5DAC1E97EBC300DBFE48 /* statistics.hpp */, ); @@ -155,6 +161,7 @@ buildActionMask = 2147483647; files = ( 455C5DB01E97EBC300DBFE48 /* file_helpers.hpp in Headers */, + 450703051E9E6C3400E8C029 /* icons_info.hpp in Headers */, 45FFD6591E965E0600DB854E /* campaign.hpp in Headers */, 455C5DAE1E97EBC300DBFE48 /* event.hpp in Headers */, 45FFD6581E965E0600DB854E /* campaign_serialization.hpp in Headers */, @@ -254,6 +261,7 @@ files = ( 455C5DAD1E97EBC300DBFE48 /* event.cpp in Sources */, 45FFD6571E965E0600DB854E /* campaign_serialization.cpp in Sources */, + 450703041E9E6C3400E8C029 /* icons_info.cpp in Sources */, 455C5DB11E97EBC300DBFE48 /* statistics.cpp in Sources */, 455C5DAF1E97EBC300DBFE48 /* file_helpers.cpp in Sources */, ); diff --git a/xcode/qtMapsMe/qtMapsMe.xcodeproj/project.pbxproj b/xcode/qtMapsMe/qtMapsMe.xcodeproj/project.pbxproj index ed878a70d7..af24a53b84 100644 --- a/xcode/qtMapsMe/qtMapsMe.xcodeproj/project.pbxproj +++ b/xcode/qtMapsMe/qtMapsMe.xcodeproj/project.pbxproj @@ -25,6 +25,7 @@ 34FFD30E1E9CEF010010AD12 /* liblocal_ads.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 34FFD30C1E9CEF010010AD12 /* liblocal_ads.a */; }; 34FFD30F1E9CEF490010AD12 /* map_widget.hpp in Sources */ = {isa = PBXBuildFile; fileRef = 34FFD2F21E9CEEA20010AD12 /* map_widget.hpp */; }; 34FFD3101E9CEF490010AD12 /* resources_common.qrc in Sources */ = {isa = PBXBuildFile; fileRef = 34FFD2FD1E9CEEA20010AD12 /* resources_common.qrc */; }; + 450703071E9E6C7100E8C029 /* local_ads_symbols.txt in Resources */ = {isa = PBXBuildFile; fileRef = 450703061E9E6C7100E8C029 /* local_ads_symbols.txt */; }; 45B5B58A1CA4216B00D93E36 /* create_feature_dialog.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 45B5B5861CA4216B00D93E36 /* create_feature_dialog.cpp */; }; 45B5B58B1CA4216B00D93E36 /* place_page_dialog.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 45B5B5881CA4216B00D93E36 /* place_page_dialog.cpp */; }; 45B5B58C1CA4219C00D93E36 /* create_feature_dialog.hpp in Sources */ = {isa = PBXBuildFile; fileRef = 45B5B5871CA4216B00D93E36 /* create_feature_dialog.hpp */; }; @@ -235,6 +236,7 @@ 34FFD2FF1E9CEEA20010AD12 /* scale_slider.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = scale_slider.hpp; sourceTree = ""; }; 34FFD30B1E9CEF010010AD12 /* libicu.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = libicu.a; path = "/Users/igrechuhin/Repo/omim/xcode/icu/../../../omim-build/xcode/Debug/libicu.a"; sourceTree = ""; }; 34FFD30C1E9CEF010010AD12 /* liblocal_ads.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = liblocal_ads.a; path = "/Users/igrechuhin/Repo/omim/xcode/local_ads/../../../omim-build/xcode/Debug/liblocal_ads.a"; sourceTree = ""; }; + 450703061E9E6C7100E8C029 /* local_ads_symbols.txt */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; path = local_ads_symbols.txt; sourceTree = ""; }; 45B5B5861CA4216B00D93E36 /* create_feature_dialog.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = create_feature_dialog.cpp; sourceTree = ""; }; 45B5B5871CA4216B00D93E36 /* create_feature_dialog.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = create_feature_dialog.hpp; sourceTree = ""; }; 45B5B5881CA4216B00D93E36 /* place_page_dialog.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = place_page_dialog.cpp; sourceTree = ""; }; @@ -537,6 +539,7 @@ 6714E6001BD14016008AB603 /* resources-mdpi_clear */, 6714E6011BD14016008AB603 /* resources-mdpi_dark */, 6729A5BB1A693013007D5872 /* countries.txt */, + 450703061E9E6C7100E8C029 /* local_ads_symbols.txt */, ); name = Resources; path = ../../data; @@ -694,6 +697,7 @@ 45B5B59B1CA422C800D93E36 /* resources-xhdpi_clear in Resources */, 671182DC1C7F0D8C00CB8177 /* packed_polygons_obsolete.bin in Resources */, 45B5B5981CA422C800D93E36 /* resources-6plus_clear in Resources */, + 450703071E9E6C7100E8C029 /* local_ads_symbols.txt in Resources */, 671182DB1C7F0D8C00CB8177 /* countries_obsolete.txt in Resources */, 67F4D3F51CC5400900C5D9BB /* cuisine-strings in Resources */, );