BsScriptProjectLibrary.cpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. #include "BsScriptProjectLibrary.h"
  2. #include "BsScriptMeta.h"
  3. #include "BsMonoField.h"
  4. #include "BsMonoClass.h"
  5. #include "BsMonoMethod.h"
  6. #include "BsMonoManager.h"
  7. #include "BsMonoUtil.h"
  8. #include "BsScriptResource.h"
  9. #include "BsResources.h"
  10. #include "BsResource.h"
  11. #include "BsProjectResourceMeta.h"
  12. #include "BsScriptResourceManager.h"
  13. #include "BsScriptTexture2D.h"
  14. #include "BsScriptSpriteTexture.h"
  15. #include "BsScriptFont.h"
  16. #include "BsScriptImportOptions.h"
  17. #include "BsEditorApplication.h"
  18. #include "BsPath.h"
  19. using namespace std::placeholders;
  20. namespace BansheeEngine
  21. {
  22. ScriptProjectLibrary::OnEntryChangedThunkDef ScriptProjectLibrary::OnEntryAddedThunk;
  23. ScriptProjectLibrary::OnEntryChangedThunkDef ScriptProjectLibrary::OnEntryRemovedThunk;
  24. ScriptProjectLibrary::OnEntryChangedThunkDef ScriptProjectLibrary::OnEntryImportedThunk;
  25. HEvent ScriptProjectLibrary::mOnEntryAddedConn;
  26. HEvent ScriptProjectLibrary::mOnEntryRemovedConn;
  27. HEvent ScriptProjectLibrary::mOnEntryImportedConn;
  28. ScriptProjectLibrary::ScriptProjectLibrary(MonoObject* instance)
  29. :ScriptObject(instance)
  30. { }
  31. void ScriptProjectLibrary::initRuntimeData()
  32. {
  33. metaData.scriptClass->addInternalCall("Internal_Refresh", &ScriptProjectLibrary::internal_Refresh);
  34. metaData.scriptClass->addInternalCall("Internal_Create", &ScriptProjectLibrary::internal_Create);
  35. metaData.scriptClass->addInternalCall("Internal_Load", &ScriptProjectLibrary::internal_Load);
  36. metaData.scriptClass->addInternalCall("Internal_Save", &ScriptProjectLibrary::internal_Save);
  37. metaData.scriptClass->addInternalCall("Internal_GetRoot", &ScriptProjectLibrary::internal_GetRoot);
  38. metaData.scriptClass->addInternalCall("Internal_Reimport", &ScriptProjectLibrary::internal_Reimport);
  39. metaData.scriptClass->addInternalCall("Internal_GetEntry", &ScriptProjectLibrary::internal_GetEntry);
  40. metaData.scriptClass->addInternalCall("Internal_GetPath", &ScriptProjectLibrary::internal_GetPath);
  41. metaData.scriptClass->addInternalCall("Internal_GetPathFromUUID", &ScriptProjectLibrary::internal_GetPathFromUUID);
  42. metaData.scriptClass->addInternalCall("Internal_Search", &ScriptProjectLibrary::internal_Search);
  43. metaData.scriptClass->addInternalCall("Internal_Delete", &ScriptProjectLibrary::internal_Delete);
  44. metaData.scriptClass->addInternalCall("Internal_CreateFolder", &ScriptProjectLibrary::internal_CreateFolder);
  45. metaData.scriptClass->addInternalCall("Internal_Rename", &ScriptProjectLibrary::internal_Rename);
  46. metaData.scriptClass->addInternalCall("Internal_Move", &ScriptProjectLibrary::internal_Move);
  47. metaData.scriptClass->addInternalCall("Internal_Copy", &ScriptProjectLibrary::internal_Copy);
  48. metaData.scriptClass->addInternalCall("Internal_GetResourceFolder", &ScriptProjectLibrary::internal_GetResourceFolder);
  49. metaData.scriptClass->addInternalCall("Internal_SetIncludeInBuild", &ScriptProjectLibrary::internal_SetIncludeInBuild);
  50. OnEntryAddedThunk = (OnEntryChangedThunkDef)metaData.scriptClass->getMethod("Internal_DoOnEntryAdded", 1)->getThunk();
  51. OnEntryRemovedThunk = (OnEntryChangedThunkDef)metaData.scriptClass->getMethod("Internal_DoOnEntryRemoved", 1)->getThunk();
  52. OnEntryImportedThunk = (OnEntryChangedThunkDef)metaData.scriptClass->getMethod("Internal_DoOnEntryImported", 1)->getThunk();
  53. }
  54. MonoArray* ScriptProjectLibrary::internal_Refresh(MonoString* path, bool import)
  55. {
  56. Path nativePath = MonoUtil::monoToWString(path);
  57. if (!nativePath.isAbsolute())
  58. nativePath.makeAbsolute(gProjectLibrary().getResourcesFolder());
  59. Vector<Path> dirtyResources;
  60. gProjectLibrary().checkForModifications(nativePath, import, dirtyResources);
  61. ScriptArray output = ScriptArray::create<WString>((UINT32)dirtyResources.size());
  62. for (UINT32 i = 0; i < (UINT32)dirtyResources.size(); i++)
  63. {
  64. output.set(i, dirtyResources[i].toWString());
  65. }
  66. return output.getInternal();
  67. }
  68. void ScriptProjectLibrary::internal_Create(MonoObject* resource, MonoString* path)
  69. {
  70. ScriptResource* scrResource = ScriptResource::toNative(resource);
  71. Path resourcePath = MonoUtil::monoToWString(path);
  72. gProjectLibrary().createEntry(scrResource->getGenericHandle(), resourcePath);
  73. }
  74. MonoObject* ScriptProjectLibrary::internal_Load(MonoString* path)
  75. {
  76. Path resourcePath = MonoUtil::monoToWString(path);
  77. HResource resource;
  78. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(resourcePath);
  79. if (entry != nullptr && entry->type == ProjectLibrary::LibraryEntryType::File)
  80. {
  81. ProjectLibrary::ResourceEntry* resEntry = static_cast <ProjectLibrary::ResourceEntry*>(entry);
  82. if (resEntry->meta != nullptr)
  83. resource = gResources().loadFromUUID(resEntry->meta->getUUID());
  84. }
  85. if (!resource)
  86. return nullptr;
  87. ScriptResourceBase* scriptResource;
  88. ScriptResourceManager::instance().getScriptResource(resource, &scriptResource, true);
  89. return scriptResource->getManagedInstance();
  90. }
  91. void ScriptProjectLibrary::internal_Save(MonoObject* resource)
  92. {
  93. ScriptResource* srcResource = ScriptResource::toNative(resource);
  94. if (srcResource != nullptr)
  95. gProjectLibrary().saveEntry(srcResource->getGenericHandle());
  96. }
  97. MonoObject* ScriptProjectLibrary::internal_GetRoot()
  98. {
  99. return ScriptDirectoryEntry::create(static_cast<const ProjectLibrary::DirectoryEntry*>(gProjectLibrary().getRootEntry()));
  100. }
  101. void ScriptProjectLibrary::internal_Reimport(MonoString* path, MonoObject* options, bool force)
  102. {
  103. Path assetPath = MonoUtil::monoToWString(path);
  104. SPtr<ImportOptions> nativeOptions;
  105. if (options != nullptr)
  106. {
  107. ScriptImportOptions* scriptOptions = ScriptImportOptions::toNative(options);
  108. nativeOptions = scriptOptions->getImportOptions();
  109. }
  110. gProjectLibrary().reimport(assetPath, nativeOptions, force);
  111. }
  112. MonoObject* ScriptProjectLibrary::internal_GetEntry(MonoString* path)
  113. {
  114. Path assetPath = MonoUtil::monoToWString(path);
  115. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(assetPath);
  116. if (entry == nullptr)
  117. return nullptr;
  118. if (entry->type == ProjectLibrary::LibraryEntryType::File)
  119. return ScriptFileEntry::create(static_cast<ProjectLibrary::ResourceEntry*>(entry));
  120. else
  121. return ScriptDirectoryEntry::create(static_cast<ProjectLibrary::DirectoryEntry*>(entry));
  122. }
  123. MonoString* ScriptProjectLibrary::internal_GetPathFromUUID(MonoString* uuid)
  124. {
  125. String nativeUUID = MonoUtil::monoToString(uuid);
  126. Path nativePath = gProjectLibrary().uuidToPath(nativeUUID);
  127. return MonoUtil::wstringToMono(MonoManager::instance().getDomain(), nativePath.toWString());
  128. }
  129. MonoString* ScriptProjectLibrary::internal_GetPath(MonoObject* resource)
  130. {
  131. ScriptResource* srcResource = ScriptResource::toNative(resource);
  132. if (srcResource != nullptr)
  133. {
  134. Path nativePath = gProjectLibrary().uuidToPath(srcResource->getGenericHandle().getUUID());
  135. nativePath.getRelative(gProjectLibrary().getResourcesFolder());
  136. return MonoUtil::wstringToMono(MonoManager::instance().getDomain(), nativePath.toWString());
  137. }
  138. return nullptr;
  139. }
  140. MonoArray* ScriptProjectLibrary::internal_Search(MonoString* pattern, MonoArray* types)
  141. {
  142. WString strPattern = MonoUtil::monoToWString(pattern);
  143. Vector<UINT32> typeIds;
  144. if (types != nullptr)
  145. {
  146. ScriptArray typeArray(types);
  147. for (UINT32 i = 0; i < typeArray.size(); i++)
  148. {
  149. UINT32 typeId = ScriptResource::getTypeIdFromType((ScriptResourceType)typeArray.get<UINT32>(i));
  150. typeIds.push_back(typeId);
  151. }
  152. }
  153. Vector<ProjectLibrary::LibraryEntry*> foundEntries = gProjectLibrary().search(strPattern, typeIds);
  154. UINT32 idx = 0;
  155. ScriptArray outArray = ScriptArray::create<ScriptLibraryEntry>((UINT32)foundEntries.size());
  156. for (auto& entry : foundEntries)
  157. {
  158. MonoObject* managedEntry = nullptr;
  159. if (entry->type == ProjectLibrary::LibraryEntryType::File)
  160. managedEntry = ScriptFileEntry::create(static_cast<ProjectLibrary::ResourceEntry*>(entry));
  161. else
  162. managedEntry = ScriptDirectoryEntry::create(static_cast<ProjectLibrary::DirectoryEntry*>(entry));
  163. outArray.set(idx, managedEntry);
  164. idx++;
  165. }
  166. return outArray.getInternal();
  167. }
  168. void ScriptProjectLibrary::internal_Delete(MonoString* path)
  169. {
  170. Path pathToDelete = MonoUtil::monoToWString(path);
  171. gProjectLibrary().deleteEntry(pathToDelete);
  172. }
  173. void ScriptProjectLibrary::internal_CreateFolder(MonoString* path)
  174. {
  175. Path folderToCreate = MonoUtil::monoToWString(path);
  176. gProjectLibrary().createFolderEntry(folderToCreate);
  177. }
  178. void ScriptProjectLibrary::internal_Rename(MonoString* path, MonoString* name, bool overwrite)
  179. {
  180. Path oldPath = MonoUtil::monoToWString(path);
  181. Path newPath = oldPath.getParent().append(MonoUtil::monoToWString(name));
  182. gProjectLibrary().moveEntry(oldPath, newPath, overwrite);
  183. }
  184. void ScriptProjectLibrary::internal_Move(MonoString* oldPath, MonoString* newPath, bool overwrite)
  185. {
  186. Path oldPathNative = MonoUtil::monoToWString(oldPath);
  187. Path newPathNative = MonoUtil::monoToWString(newPath);
  188. gProjectLibrary().moveEntry(oldPathNative, newPathNative, overwrite);
  189. }
  190. void ScriptProjectLibrary::internal_Copy(MonoString* source, MonoString* destination, bool overwrite)
  191. {
  192. Path oldPathNative = MonoUtil::monoToWString(source);
  193. Path newPathNative = MonoUtil::monoToWString(destination);
  194. gProjectLibrary().copyEntry(oldPathNative, newPathNative, overwrite);
  195. }
  196. MonoString* ScriptProjectLibrary::internal_GetResourceFolder()
  197. {
  198. return MonoUtil::wstringToMono(MonoManager::instance().getDomain(), gProjectLibrary().getResourcesFolder().toWString());
  199. }
  200. void ScriptProjectLibrary::internal_SetIncludeInBuild(MonoString* path, bool include)
  201. {
  202. Path pathNative = MonoUtil::monoToWString(path);
  203. gProjectLibrary().setIncludeInBuild(pathNative, include);
  204. }
  205. void ScriptProjectLibrary::startUp()
  206. {
  207. mOnEntryAddedConn = gProjectLibrary().onEntryAdded.connect(std::bind(&ScriptProjectLibrary::onEntryAdded, _1));
  208. mOnEntryRemovedConn = gProjectLibrary().onEntryRemoved.connect(std::bind(&ScriptProjectLibrary::onEntryRemoved, _1));
  209. mOnEntryImportedConn = gProjectLibrary().onEntryImported.connect(std::bind(&ScriptProjectLibrary::onEntryImported, _1));
  210. }
  211. void ScriptProjectLibrary::shutDown()
  212. {
  213. mOnEntryAddedConn.disconnect();
  214. mOnEntryRemovedConn.disconnect();
  215. mOnEntryImportedConn.disconnect();
  216. }
  217. void ScriptProjectLibrary::onEntryAdded(const Path& path)
  218. {
  219. Path relativePath = path;
  220. if (relativePath.isAbsolute())
  221. relativePath.makeRelative(gProjectLibrary().getResourcesFolder());
  222. MonoString* pathStr = MonoUtil::wstringToMono(MonoManager::instance().getDomain(), relativePath.toWString());
  223. MonoUtil::invokeThunk(OnEntryAddedThunk, pathStr);
  224. }
  225. void ScriptProjectLibrary::onEntryRemoved(const Path& path)
  226. {
  227. Path relativePath = path;
  228. if (relativePath.isAbsolute())
  229. relativePath.makeRelative(gProjectLibrary().getResourcesFolder());
  230. MonoString* pathStr = MonoUtil::wstringToMono(MonoManager::instance().getDomain(), relativePath.toWString());
  231. MonoUtil::invokeThunk(OnEntryRemovedThunk, pathStr);
  232. }
  233. void ScriptProjectLibrary::onEntryImported(const Path& path)
  234. {
  235. Path relativePath = path;
  236. if (relativePath.isAbsolute())
  237. relativePath.makeRelative(gProjectLibrary().getResourcesFolder());
  238. MonoString* pathStr = MonoUtil::wstringToMono(MonoManager::instance().getDomain(), relativePath.toWString());
  239. MonoUtil::invokeThunk(OnEntryImportedThunk, pathStr);
  240. }
  241. ScriptLibraryEntryBase::ScriptLibraryEntryBase(MonoObject* instance)
  242. :ScriptObjectBase(instance)
  243. { }
  244. void ScriptLibraryEntry::initRuntimeData()
  245. {
  246. metaData.scriptClass->addInternalCall("Internal_GetPath", &ScriptLibraryEntry::internal_GetPath);
  247. metaData.scriptClass->addInternalCall("Internal_GetName", &ScriptLibraryEntry::internal_GetName);
  248. metaData.scriptClass->addInternalCall("Internal_GetType", &ScriptLibraryEntry::internal_GetType);
  249. metaData.scriptClass->addInternalCall("Internal_GetParent", &ScriptLibraryEntry::internal_GetParent);
  250. }
  251. MonoString* ScriptLibraryEntry::internal_GetPath(ScriptLibraryEntryBase* thisPtr)
  252. {
  253. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  254. if (entry == nullptr)
  255. return nullptr;
  256. Path relativePath = entry->path;
  257. relativePath.makeRelative(gProjectLibrary().getResourcesFolder());
  258. return MonoUtil::wstringToMono(MonoManager::instance().getDomain(), relativePath.toWString());
  259. }
  260. MonoString* ScriptLibraryEntry::internal_GetName(ScriptLibraryEntryBase* thisPtr)
  261. {
  262. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  263. if (entry == nullptr)
  264. return nullptr;
  265. return MonoUtil::wstringToMono(MonoManager::instance().getDomain(), entry->elementName);
  266. }
  267. ProjectLibrary::LibraryEntryType ScriptLibraryEntry::internal_GetType(ScriptLibraryEntryBase* thisPtr)
  268. {
  269. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  270. if (entry == nullptr)
  271. return ProjectLibrary::LibraryEntryType::File; // Note: We don't actually know what this entry is, because it doesn't exist anymore
  272. return entry->type;
  273. }
  274. MonoObject* ScriptLibraryEntry::internal_GetParent(ScriptLibraryEntryBase* thisPtr)
  275. {
  276. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  277. if (entry == nullptr || entry->parent == nullptr)
  278. return nullptr;
  279. return ScriptDirectoryEntry::create(entry->parent);
  280. }
  281. ScriptDirectoryEntry::ScriptDirectoryEntry(MonoObject* instance, const Path& assetPath)
  282. :ScriptObject(instance)
  283. {
  284. mAssetPath = assetPath;
  285. }
  286. MonoObject* ScriptDirectoryEntry::create(const ProjectLibrary::DirectoryEntry* entry)
  287. {
  288. MonoObject* managedInstance = metaData.scriptClass->createInstance();
  289. bs_new<ScriptDirectoryEntry>(managedInstance, entry->path);
  290. return managedInstance;
  291. }
  292. void ScriptDirectoryEntry::initRuntimeData()
  293. {
  294. metaData.scriptClass->addInternalCall("Internal_GetChildren", &ScriptDirectoryEntry::internal_GetChildren);
  295. }
  296. MonoArray* ScriptDirectoryEntry::internal_GetChildren(ScriptDirectoryEntry* thisPtr)
  297. {
  298. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  299. if (entry == nullptr || entry->type != ProjectLibrary::LibraryEntryType::Directory)
  300. return ScriptArray::create<ScriptLibraryEntry>(0).getInternal();
  301. ProjectLibrary::DirectoryEntry* dirEntry = static_cast<ProjectLibrary::DirectoryEntry*>(entry);
  302. ScriptArray outArray = ScriptArray::create<ScriptLibraryEntry>((UINT32)dirEntry->mChildren.size());
  303. for (UINT32 i = 0; i < (UINT32)dirEntry->mChildren.size(); i++)
  304. {
  305. ProjectLibrary::LibraryEntry* childEntry = dirEntry->mChildren[i];
  306. MonoObject* managedChildEntry = nullptr;
  307. if (childEntry->type == ProjectLibrary::LibraryEntryType::File)
  308. managedChildEntry = ScriptFileEntry::create(static_cast<ProjectLibrary::ResourceEntry*>(childEntry));
  309. else
  310. managedChildEntry = ScriptDirectoryEntry::create(static_cast<ProjectLibrary::DirectoryEntry*>(childEntry));
  311. outArray.set(i, managedChildEntry);
  312. }
  313. return outArray.getInternal();
  314. }
  315. ScriptFileEntry::ScriptFileEntry(MonoObject* instance, const Path& assetPath)
  316. :ScriptObject(instance)
  317. {
  318. mAssetPath = assetPath;
  319. }
  320. MonoObject* ScriptFileEntry::create(const ProjectLibrary::ResourceEntry* entry)
  321. {
  322. MonoObject* managedInstance = metaData.scriptClass->createInstance();
  323. bs_new<ScriptFileEntry>(managedInstance, entry->path);
  324. return managedInstance;
  325. }
  326. void ScriptFileEntry::initRuntimeData()
  327. {
  328. metaData.scriptClass->addInternalCall("Internal_GetImportOptions", &ScriptFileEntry::internal_GetImportOptions);
  329. metaData.scriptClass->addInternalCall("Internal_GetUUID", &ScriptFileEntry::internal_GetUUID);
  330. metaData.scriptClass->addInternalCall("Internal_GetIcon", &ScriptFileEntry::internal_GetIcon);
  331. metaData.scriptClass->addInternalCall("Internal_GetResourceType", &ScriptFileEntry::internal_GetResourceType);
  332. metaData.scriptClass->addInternalCall("Internal_GetIncludeInBuild", &ScriptFileEntry::internal_GetIncludeInBuild);
  333. }
  334. MonoObject* ScriptFileEntry::internal_GetImportOptions(ScriptFileEntry* thisPtr)
  335. {
  336. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  337. if (entry == nullptr || entry->type != ProjectLibrary::LibraryEntryType::File)
  338. return nullptr;
  339. ProjectLibrary::ResourceEntry* fileEntry = static_cast<ProjectLibrary::ResourceEntry*>(entry);
  340. if (fileEntry->meta != nullptr)
  341. return ScriptImportOptions::create(fileEntry->meta->getImportOptions());
  342. else
  343. return nullptr;
  344. }
  345. MonoString* ScriptFileEntry::internal_GetUUID(ScriptFileEntry* thisPtr)
  346. {
  347. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  348. if (entry == nullptr || entry->type != ProjectLibrary::LibraryEntryType::File)
  349. return nullptr;
  350. ProjectLibrary::ResourceEntry* fileEntry = static_cast<ProjectLibrary::ResourceEntry*>(entry);
  351. if (fileEntry->meta != nullptr)
  352. return MonoUtil::stringToMono(MonoManager::instance().getDomain(), fileEntry->meta->getUUID());
  353. else
  354. return nullptr;
  355. }
  356. MonoObject* ScriptFileEntry::internal_GetIcon(ScriptFileEntry* thisPtr)
  357. {
  358. // TODO - Icons not supported yet
  359. return nullptr;
  360. }
  361. ScriptResourceType ScriptFileEntry::internal_GetResourceType(ScriptFileEntry* thisPtr)
  362. {
  363. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  364. if (entry == nullptr || entry->type != ProjectLibrary::LibraryEntryType::File)
  365. return ScriptResourceType::Undefined;
  366. ProjectLibrary::ResourceEntry* fileEntry = static_cast<ProjectLibrary::ResourceEntry*>(entry);
  367. if (fileEntry->meta != nullptr)
  368. return ScriptResource::getTypeFromTypeId(fileEntry->meta->getTypeID());
  369. return ScriptResourceType::Undefined;
  370. }
  371. bool ScriptFileEntry::internal_GetIncludeInBuild(ScriptFileEntry* thisPtr)
  372. {
  373. ProjectLibrary::LibraryEntry* entry = gProjectLibrary().findEntry(thisPtr->getAssetPath());
  374. if (entry == nullptr || entry->type != ProjectLibrary::LibraryEntryType::File)
  375. return false;
  376. ProjectLibrary::ResourceEntry* fileEntry = static_cast<ProjectLibrary::ResourceEntry*>(entry);
  377. if (fileEntry->meta != nullptr)
  378. return fileEntry->meta->getIncludeInBuild();
  379. return false;
  380. }
  381. }