Support UTF8 to UTF16 LE.

Support Unicode helper
Fix ANSFR to show 2 same faces in 1 image
This commit is contained in:
2026-03-31 21:52:47 +11:00
parent 70be68d0fc
commit ccfc5964d4
21 changed files with 379 additions and 104 deletions

View File

@@ -7,7 +7,8 @@
"Bash(grep -l \"EnginePoolManager\" /c/Projects/CLionProjects/ANSCORE/modules/ANSODEngine/*.cpp)", "Bash(grep -l \"EnginePoolManager\" /c/Projects/CLionProjects/ANSCORE/modules/ANSODEngine/*.cpp)",
"Bash(grep -n \"g_processExiting\" /c/Projects/CLionProjects/ANSCORE/engines/TensorRTAPI/include/engine/*.h /c/Projects/CLionProjects/ANSCORE/modules/ANSODEngine/engine.h)", "Bash(grep -n \"g_processExiting\" /c/Projects/CLionProjects/ANSCORE/engines/TensorRTAPI/include/engine/*.h /c/Projects/CLionProjects/ANSCORE/modules/ANSODEngine/engine.h)",
"Bash(ssh -T git@anscenter.ddns.net -p 2222)", "Bash(ssh -T git@anscenter.ddns.net -p 2222)",
"Bash(ssh-add -l)" "Bash(ssh-add -l)",
"Bash(dotnet build:*)"
] ]
} }
} }

View File

@@ -0,0 +1,125 @@
using System;
using System.Drawing;
using System.Text;
using System.Text.RegularExpressions;
namespace ANSCENTER
{
/// <summary>
/// Unicode helper for LabVIEW .NET interop.
/// All methods are static — no instance creation needed.
/// In LabVIEW: use Invoke Node with class type ANSCENTER.ANSUnicodeHelper.
/// Thread-safe: no shared mutable state.
/// </summary>
public class ANSUnicodeHelper
{
// Pre-compiled regex — compiled once at class load, zero per-call cost
private static readonly Regex UnicodeEscapeRegex =
new Regex(@"\\u([0-9a-fA-F]{4})", RegexOptions.Compiled);
/// <summary>
/// Constructor for LabVIEW (if needed). Lightweight, no state.
/// </summary>
public ANSUnicodeHelper() { }
/// <summary>
/// Decodes \uXXXX escape sequences to Unicode characters.
/// ASCII strings pass through unchanged (zero allocation).
/// </summary>
public static string Decode(string input)
{
if (string.IsNullOrEmpty(input)) return input ?? string.Empty;
if (input.IndexOf("\\u", StringComparison.Ordinal) < 0) return input;
return UnicodeEscapeRegex.Replace(input, match =>
{
int codepoint = Convert.ToInt32(match.Groups[1].Value, 16);
return ((char)codepoint).ToString();
});
}
/// <summary>
/// Draws Unicode string at position (x, y).
/// </summary>
public static void DrawString(Graphics g, string escapedText, Font font, Brush brush, float x, float y)
{
if (g == null || string.IsNullOrEmpty(escapedText)) return;
g.DrawString(Decode(escapedText), font, brush, x, y);
}
/// <summary>
/// Draws Unicode string at position (x, y) with StringFormat.
/// </summary>
public static void DrawString(Graphics g, string escapedText, Font font, Brush brush, float x, float y, StringFormat format)
{
if (g == null || string.IsNullOrEmpty(escapedText)) return;
g.DrawString(Decode(escapedText), font, brush, x, y, format);
}
/// <summary>
/// Draws Unicode string at PointF with StringFormat.
/// </summary>
public static void DrawString(Graphics g, string escapedText, Font font, Brush brush, PointF point, StringFormat format)
{
if (g == null || string.IsNullOrEmpty(escapedText)) return;
g.DrawString(Decode(escapedText), font, brush, point, format);
}
/// <summary>
/// Draws Unicode string inside a RectangleF with word wrapping.
/// </summary>
public static void DrawString(Graphics g, string escapedText, Font font, Brush brush, RectangleF layoutRect)
{
if (g == null || string.IsNullOrEmpty(escapedText)) return;
g.DrawString(Decode(escapedText), font, brush, layoutRect);
}
/// <summary>
/// Draws Unicode string inside a RectangleF with StringFormat.
/// </summary>
public static void DrawString(Graphics g, string escapedText, Font font, Brush brush, RectangleF layoutRect, StringFormat format)
{
if (g == null || string.IsNullOrEmpty(escapedText)) return;
g.DrawString(Decode(escapedText), font, brush, layoutRect, format);
}
/// <summary>
/// Measures Unicode string size after decoding.
/// </summary>
public static SizeF MeasureString(Graphics g, string escapedText, Font font)
{
if (g == null || string.IsNullOrEmpty(escapedText)) return SizeF.Empty;
return g.MeasureString(Decode(escapedText), font);
}
/// <summary>
/// Measures Unicode string size with layout area and StringFormat.
/// </summary>
public static SizeF MeasureString(Graphics g, string escapedText, Font font, SizeF layoutArea, StringFormat format)
{
if (g == null || string.IsNullOrEmpty(escapedText)) return SizeF.Empty;
return g.MeasureString(Decode(escapedText), font, layoutArea, format);
}
/// <summary>
/// Checks if string contains \uXXXX escapes.
/// </summary>
public static bool HasUnicodeEscapes(string input)
{
if (string.IsNullOrEmpty(input)) return false;
return input.IndexOf("\\u", StringComparison.Ordinal) >= 0;
}
/// <summary>
/// Checks if string is pure ASCII.
/// </summary>
public static bool IsAsciiOnly(string input)
{
if (string.IsNullOrEmpty(input)) return true;
foreach (char c in input)
{
if (c > 127) return false;
}
return true;
}
}
}

View File

@@ -0,0 +1,21 @@
<Project Sdk="Microsoft.NET.Sdk">
<ItemGroup>
<Reference Include="System.Drawing" />
</ItemGroup>
<PropertyGroup>
<TargetFramework>net48</TargetFramework>
<AssemblyName>ANSUnicodeHelper</AssemblyName>
<RootNamespace>ANSCENTER</RootNamespace>
<GenerateAssemblyInfo>true</GenerateAssemblyInfo>
<AssemblyVersion>1.0.0.0</AssemblyVersion>
<FileVersion>1.0.0.0</FileVersion>
<Company>ANSCENTER</Company>
<Product>ANSUnicodeHelper</Product>
<Description>Unicode helper for LabVIEW .NET interop - decodes \uXXXX escapes to .NET System.String</Description>
<OutputPath>..\..\cmake-build-release\bin\</OutputPath>
<AppendTargetFrameworkToOutputPath>false</AppendTargetFrameworkToOutputPath>
</PropertyGroup>
</Project>

View File

@@ -0,0 +1,58 @@
{
"format": 1,
"restore": {
"C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\ANSUnicodeHelper.csproj": {}
},
"projects": {
"C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\ANSUnicodeHelper.csproj": {
"version": "1.0.0",
"restore": {
"projectUniqueName": "C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\ANSUnicodeHelper.csproj",
"projectName": "ANSUnicodeHelper",
"projectPath": "C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\ANSUnicodeHelper.csproj",
"packagesPath": "C:\\Users\\nghia\\.nuget\\packages\\",
"outputPath": "C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\obj\\",
"projectStyle": "PackageReference",
"fallbackFolders": [
"C:\\Program Files (x86)\\Microsoft Visual Studio\\Shared\\NuGetPackages"
],
"configFilePaths": [
"C:\\Users\\nghia\\AppData\\Roaming\\NuGet\\NuGet.Config",
"C:\\Program Files (x86)\\NuGet\\Config\\Microsoft.VisualStudio.FallbackLocation.config",
"C:\\Program Files (x86)\\NuGet\\Config\\Microsoft.VisualStudio.Offline.config"
],
"originalTargetFrameworks": [
"net48"
],
"sources": {
"C:\\Program Files (x86)\\Microsoft SDKs\\NuGetPackages\\": {},
"https://aiinfra.pkgs.visualstudio.com/PublicPackages/_packaging/onnxruntime-cuda-11/nuget/v3/index.json": {},
"https://api.nuget.org/v3/index.json": {}
},
"frameworks": {
"net48": {
"targetAlias": "net48",
"projectReferences": {}
}
},
"warningProperties": {
"warnAsError": [
"NU1605"
]
},
"restoreAuditProperties": {
"enableAudit": "true",
"auditLevel": "low",
"auditMode": "direct"
},
"SdkAnalysisLevel": "10.0.200"
},
"frameworks": {
"net48": {
"targetAlias": "net48",
"runtimeIdentifierGraphPath": "C:\\Program Files\\dotnet\\sdk\\10.0.200\\RuntimeIdentifierGraph.json"
}
}
}
}
}

View File

@@ -0,0 +1,16 @@
<?xml version="1.0" encoding="utf-8" standalone="no"?>
<Project ToolsVersion="14.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup Condition=" '$(ExcludeRestorePackageImports)' != 'true' ">
<RestoreSuccess Condition=" '$(RestoreSuccess)' == '' ">True</RestoreSuccess>
<RestoreTool Condition=" '$(RestoreTool)' == '' ">NuGet</RestoreTool>
<ProjectAssetsFile Condition=" '$(ProjectAssetsFile)' == '' ">$(MSBuildThisFileDirectory)project.assets.json</ProjectAssetsFile>
<NuGetPackageRoot Condition=" '$(NuGetPackageRoot)' == '' ">$(UserProfile)\.nuget\packages\</NuGetPackageRoot>
<NuGetPackageFolders Condition=" '$(NuGetPackageFolders)' == '' ">C:\Users\nghia\.nuget\packages\;C:\Program Files (x86)\Microsoft Visual Studio\Shared\NuGetPackages</NuGetPackageFolders>
<NuGetProjectStyle Condition=" '$(NuGetProjectStyle)' == '' ">PackageReference</NuGetProjectStyle>
<NuGetToolVersion Condition=" '$(NuGetToolVersion)' == '' ">7.0.0</NuGetToolVersion>
</PropertyGroup>
<ItemGroup Condition=" '$(ExcludeRestorePackageImports)' != 'true' ">
<SourceRoot Include="C:\Users\nghia\.nuget\packages\" />
<SourceRoot Include="C:\Program Files (x86)\Microsoft Visual Studio\Shared\NuGetPackages\" />
</ItemGroup>
</Project>

View File

@@ -0,0 +1,2 @@
<?xml version="1.0" encoding="utf-8" standalone="no"?>
<Project ToolsVersion="14.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003" />

View File

@@ -0,0 +1,64 @@
{
"version": 3,
"targets": {
".NETFramework,Version=v4.8": {}
},
"libraries": {},
"projectFileDependencyGroups": {
".NETFramework,Version=v4.8": []
},
"packageFolders": {
"C:\\Users\\nghia\\.nuget\\packages\\": {},
"C:\\Program Files (x86)\\Microsoft Visual Studio\\Shared\\NuGetPackages": {}
},
"project": {
"version": "1.0.0",
"restore": {
"projectUniqueName": "C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\ANSUnicodeHelper.csproj",
"projectName": "ANSUnicodeHelper",
"projectPath": "C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\ANSUnicodeHelper.csproj",
"packagesPath": "C:\\Users\\nghia\\.nuget\\packages\\",
"outputPath": "C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\obj\\",
"projectStyle": "PackageReference",
"fallbackFolders": [
"C:\\Program Files (x86)\\Microsoft Visual Studio\\Shared\\NuGetPackages"
],
"configFilePaths": [
"C:\\Users\\nghia\\AppData\\Roaming\\NuGet\\NuGet.Config",
"C:\\Program Files (x86)\\NuGet\\Config\\Microsoft.VisualStudio.FallbackLocation.config",
"C:\\Program Files (x86)\\NuGet\\Config\\Microsoft.VisualStudio.Offline.config"
],
"originalTargetFrameworks": [
"net48"
],
"sources": {
"C:\\Program Files (x86)\\Microsoft SDKs\\NuGetPackages\\": {},
"https://aiinfra.pkgs.visualstudio.com/PublicPackages/_packaging/onnxruntime-cuda-11/nuget/v3/index.json": {},
"https://api.nuget.org/v3/index.json": {}
},
"frameworks": {
"net48": {
"targetAlias": "net48",
"projectReferences": {}
}
},
"warningProperties": {
"warnAsError": [
"NU1605"
]
},
"restoreAuditProperties": {
"enableAudit": "true",
"auditLevel": "low",
"auditMode": "direct"
},
"SdkAnalysisLevel": "10.0.200"
},
"frameworks": {
"net48": {
"targetAlias": "net48",
"runtimeIdentifierGraphPath": "C:\\Program Files\\dotnet\\sdk\\10.0.200\\RuntimeIdentifierGraph.json"
}
}
}
}

View File

@@ -0,0 +1,8 @@
{
"version": 2,
"dgSpecHash": "5VJRzMAJt58=",
"success": true,
"projectFilePath": "C:\\Projects\\CLionProjects\\ANSCORE\\dotnet\\ANSUnicodeHelper\\ANSUnicodeHelper.csproj",
"expectedPackageFiles": [],
"logs": []
}

View File

@@ -2161,6 +2161,9 @@ namespace ANSCENTER {
START_TIMER(postprocess); START_TIMER(postprocess);
resultObjects=UpdateFaceAttributes(recognizedFaces, camera_id); resultObjects=UpdateFaceAttributes(recognizedFaces, camera_id);
END_TIMER(postprocess, "Update Face Attributes Time"); END_TIMER(postprocess, "Update Face Attributes Time");
// Deduplicate: if two faces matched the same userId, keep only the highest confidence
ensureUniqueUserIdWithHighestConfidence(resultObjects);
} }
} }
catch (const std::exception& e) { catch (const std::exception& e) {
@@ -2405,7 +2408,7 @@ namespace ANSCENTER {
END_TIMER(json_build, "JSON Build Time"); END_TIMER(json_build, "JSON Build Time");
START_TIMER(json_serialize); START_TIMER(json_serialize);
std::string result = root.dump(-1, ' ', true); std::string result = root.dump();
END_TIMER(json_serialize, "JSON Serialize Time"); END_TIMER(json_serialize, "JSON Serialize Time");
END_TIMER(json_total, "JSON Conversion Total Time"); END_TIMER(json_total, "JSON Conversion Total Time");
@@ -2442,7 +2445,7 @@ namespace ANSCENTER {
root["results"] = detectedObjects; root["results"] = detectedObjects;
return root.dump(-1, ' ', true); return root.dump();
} }
// Validation helper methods // Validation helper methods

View File

@@ -352,7 +352,7 @@ extern "C" ANSFR_API int DeleteFacesByUser(ANSCENTER::ANSFacialRecognition *
extern "C" ANSFR_API double BlurCalculation(unsigned char* jpeg_string, unsigned int bufferLength); extern "C" ANSFR_API double BlurCalculation(unsigned char* jpeg_string, unsigned int bufferLength);
// Unicode conversion utilities for LabVIEW wrapper classes // Unicode conversion utilities for LabVIEW wrapper classes
extern "C" ANSFR_API int ANSFR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result); extern "C" ANSFR_API int ANSFR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM = 1);
extern "C" ANSFR_API int ANSFR_ConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result); extern "C" ANSFR_API int ANSFR_ConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result);
#endif #endif

View File

@@ -751,27 +751,20 @@ extern "C" ANSFR_API double BlurCalculation(unsigned char* jpeg_string, un
// Unicode conversion utilities for LabVIEW wrapper classes // Unicode conversion utilities for LabVIEW wrapper classes
extern "C" ANSFR_API int ANSFR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result) { extern "C" ANSFR_API int ANSFR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM) {
try { try {
if (!utf8Str || !result) return -1; if (!utf8Str || !result) return -1;
int len = (int)strlen(utf8Str); int len = (int)strlen(utf8Str);
if (len == 0) return 0; if (len == 0) return 0;
const char bom[2] = { '\xFF', '\xFE' };
bool hasUnicodeEscapes = false; bool hasUnicodeEscapes = false;
bool hasNonAscii = false; for (int i = 0; i + 1 < len; i++) {
for (int i = 0; i < len; i++) { if (utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { hasUnicodeEscapes = true; break; }
if ((unsigned char)utf8Str[i] >= 0x80) hasNonAscii = true;
if (i + 1 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') hasUnicodeEscapes = true;
}
if (!hasNonAscii && !hasUnicodeEscapes) {
MgErr error = DSSetHandleSize(result, sizeof(int32) + len * sizeof(uChar));
if (error != noErr) return -2;
(*result)->cnt = len;
memcpy((*result)->str, utf8Str, len);
return 1;
} }
if (hasUnicodeEscapes) { if (hasUnicodeEscapes) {
std::string utf16le; std::string utf16le;
utf16le.reserve(len * 2); if (includeBOM) utf16le.assign(bom, 2);
utf16le.reserve(len * 2 + 2);
for (int i = 0; i < len; ) { for (int i = 0; i < len; ) {
if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') {
char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 }; char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 };
@@ -797,11 +790,14 @@ extern "C" ANSFR_API int ANSFR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHan
if (wideLen <= 0) return 0; if (wideLen <= 0) return 0;
std::wstring wideStr(wideLen, 0); std::wstring wideStr(wideLen, 0);
MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, &wideStr[0], wideLen); MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, &wideStr[0], wideLen);
int size = wideLen * (int)sizeof(wchar_t); int dataSize = wideLen * (int)sizeof(wchar_t);
MgErr error = DSSetHandleSize(result, sizeof(int32) + size * sizeof(uChar)); int bomSize = includeBOM ? 2 : 0;
int totalSize = bomSize + dataSize;
MgErr error = DSSetHandleSize(result, sizeof(int32) + totalSize * sizeof(uChar));
if (error != noErr) return -2; if (error != noErr) return -2;
(*result)->cnt = size; (*result)->cnt = totalSize;
memcpy((*result)->str, wideStr.data(), size); if (includeBOM) memcpy((*result)->str, bom, 2);
memcpy((*result)->str + bomSize, wideStr.data(), dataSize);
return 1; return 1;
#else #else
return 0; return 0;

View File

@@ -460,7 +460,7 @@ namespace ANSCENTER {
{"kps", KeypointsToString(det.kps)} {"kps", KeypointsToString(det.kps)}
}); });
} }
return root.dump(-1, ' ', true); return root.dump();
} }
catch (const std::exception& e) { catch (const std::exception& e) {
this->_logger.LogFatal("ANSALPR::VectorDetectionToJsonString", e.what(), __FILE__, __LINE__); this->_logger.LogFatal("ANSALPR::VectorDetectionToJsonString", e.what(), __FILE__, __LINE__);

View File

@@ -143,7 +143,7 @@ extern "C" ANSLPR_API int ANSALPR_SetFormats(ANSCENTER::ANSALPR** Handle, co
extern "C" ANSLPR_API int ANSALPR_GetFormats(ANSCENTER::ANSALPR** Handle, LStrHandle formats);// comma separated formats extern "C" ANSLPR_API int ANSALPR_GetFormats(ANSCENTER::ANSALPR** Handle, LStrHandle formats);// comma separated formats
// Unicode conversion utilities for LabVIEW wrapper classes // Unicode conversion utilities for LabVIEW wrapper classes
extern "C" ANSLPR_API int ANSLPR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result); extern "C" ANSLPR_API int ANSLPR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM = 1);
extern "C" ANSLPR_API int ANSLPR_ConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result); extern "C" ANSLPR_API int ANSLPR_ConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result);
#endif #endif

View File

@@ -773,27 +773,20 @@ extern "C" ANSLPR_API int ANSALPR_GetFormats(ANSCENTER::ANSALPR** Handle, LS
} }
// Unicode conversion utilities for LabVIEW wrapper classes // Unicode conversion utilities for LabVIEW wrapper classes
extern "C" ANSLPR_API int ANSLPR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result) { extern "C" ANSLPR_API int ANSLPR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM) {
try { try {
if (!utf8Str || !result) return -1; if (!utf8Str || !result) return -1;
int len = (int)strlen(utf8Str); int len = (int)strlen(utf8Str);
if (len == 0) return 0; if (len == 0) return 0;
const char bom[2] = { '\xFF', '\xFE' };
bool hasUnicodeEscapes = false; bool hasUnicodeEscapes = false;
bool hasNonAscii = false; for (int i = 0; i + 1 < len; i++) {
for (int i = 0; i < len; i++) { if (utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { hasUnicodeEscapes = true; break; }
if ((unsigned char)utf8Str[i] >= 0x80) hasNonAscii = true;
if (i + 1 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') hasUnicodeEscapes = true;
}
if (!hasNonAscii && !hasUnicodeEscapes) {
MgErr error = DSSetHandleSize(result, sizeof(int32) + len * sizeof(uChar));
if (error != noErr) return -2;
(*result)->cnt = len;
memcpy((*result)->str, utf8Str, len);
return 1;
} }
if (hasUnicodeEscapes) { if (hasUnicodeEscapes) {
std::string utf16le; std::string utf16le;
utf16le.reserve(len * 2); if (includeBOM) utf16le.assign(bom, 2);
utf16le.reserve(len * 2 + 2);
for (int i = 0; i < len; ) { for (int i = 0; i < len; ) {
if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') {
char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 }; char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 };
@@ -819,11 +812,14 @@ extern "C" ANSLPR_API int ANSLPR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrH
if (wideLen <= 0) return 0; if (wideLen <= 0) return 0;
std::wstring wideStr(wideLen, 0); std::wstring wideStr(wideLen, 0);
MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, &wideStr[0], wideLen); MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, &wideStr[0], wideLen);
int size = wideLen * (int)sizeof(wchar_t); int dataSize = wideLen * (int)sizeof(wchar_t);
MgErr error = DSSetHandleSize(result, sizeof(int32) + size * sizeof(uChar)); int bomSize = includeBOM ? 2 : 0;
int totalSize = bomSize + dataSize;
MgErr error = DSSetHandleSize(result, sizeof(int32) + totalSize * sizeof(uChar));
if (error != noErr) return -2; if (error != noErr) return -2;
(*result)->cnt = size; (*result)->cnt = totalSize;
memcpy((*result)->str, wideStr.data(), size); if (includeBOM) memcpy((*result)->str, bom, 2);
memcpy((*result)->str + bomSize, wideStr.data(), dataSize);
return 1; return 1;
#else #else
return 0; return 0;

View File

@@ -259,7 +259,7 @@ namespace ANSCENTER {
}); });
} }
return root.dump(-1, ' ', true); return root.dump();
} }
catch (const std::exception& e) { catch (const std::exception& e) {
// Add your error logging here if needed // Add your error logging here if needed
@@ -893,7 +893,7 @@ namespace ANSCENTER {
for (const auto& part : res.parts) { for (const auto& part : res.parts) {
alprInfo[part.first] = part.second; alprInfo[part.first] = part.second;
} }
std::string extraInfoStr = alprInfo.dump(-1, ' ', true); std::string extraInfoStr = alprInfo.dump();
// Use the same field layout as OCRDetectionToJsonString // Use the same field layout as OCRDetectionToJsonString
jsonResults.push_back({ jsonResults.push_back({
@@ -912,7 +912,7 @@ namespace ANSCENTER {
{"kps", ""} {"kps", ""}
}); });
} }
return root.dump(-1, ' ', true); return root.dump();
} catch (const std::exception&) { } catch (const std::exception&) {
return R"({"results":[],"error":"ALPR serialization failed"})"; return R"({"results":[],"error":"ALPR serialization failed"})";
} }

View File

@@ -244,7 +244,7 @@ extern "C" ANSOCR_API int SetANSOCRCountry(ANSCENTER::ANSOCRBase** Handl
extern "C" ANSOCR_API int SetANSOCRALPRFormat(ANSCENTER::ANSOCRBase** Handle, const char* formatJson); extern "C" ANSOCR_API int SetANSOCRALPRFormat(ANSCENTER::ANSOCRBase** Handle, const char* formatJson);
// Unicode conversion utilities for LabVIEW wrapper classes // Unicode conversion utilities for LabVIEW wrapper classes
extern "C" ANSOCR_API int ANSOCR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result); extern "C" ANSOCR_API int ANSOCR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM = 1);
extern "C" ANSOCR_API int ANSOCR_ConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result); extern "C" ANSOCR_API int ANSOCR_ConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result);
// V2 Create / Release — handle as uint64_t by value (no pointer-to-pointer) // V2 Create / Release — handle as uint64_t by value (no pointer-to-pointer)

View File

@@ -427,33 +427,25 @@ extern "C" ANSOCR_API int SetANSOCRALPRFormat(ANSCENTER::ANSOCRBase** Handle, co
// - JSON Unicode escapes (\uXXXX) from ensure_ascii=true output // - JSON Unicode escapes (\uXXXX) from ensure_ascii=true output
// - Raw UTF-8 encoded strings // - Raw UTF-8 encoded strings
// Pure ASCII input is passed through directly (no conversion overhead). // Pure ASCII input is passed through directly (no conversion overhead).
extern "C" ANSOCR_API int ANSOCR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result) { extern "C" ANSOCR_API int ANSOCR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM) {
try { try {
if (!utf8Str || !result) return -1; if (!utf8Str || !result) return -1;
int len = (int)strlen(utf8Str); int len = (int)strlen(utf8Str);
if (len == 0) return 0; if (len == 0) return 0;
// Check if input contains \uXXXX escapes or non-ASCII bytes // Always output UTF-16LE (required for LabVIEW "Force Unicode Text" indicators)
const char bom[2] = { '\xFF', '\xFE' };
// Check if input contains \uXXXX escapes
bool hasUnicodeEscapes = false; bool hasUnicodeEscapes = false;
bool hasNonAscii = false; for (int i = 0; i + 1 < len; i++) {
for (int i = 0; i < len; i++) { if (utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { hasUnicodeEscapes = true; break; }
if ((unsigned char)utf8Str[i] >= 0x80) hasNonAscii = true;
if (i + 1 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') hasUnicodeEscapes = true;
} }
// Pure ASCII with no escapes — pass through directly
if (!hasNonAscii && !hasUnicodeEscapes) {
MgErr error = DSSetHandleSize(result, sizeof(int32) + len * sizeof(uChar));
if (error != noErr) return -2;
(*result)->cnt = len;
memcpy((*result)->str, utf8Str, len);
return 1;
}
// If contains \uXXXX escapes, decode them to UTF-16LE directly
if (hasUnicodeEscapes) { if (hasUnicodeEscapes) {
std::string utf16le; std::string utf16le;
utf16le.reserve(len * 2); if (includeBOM) utf16le.assign(bom, 2);
utf16le.reserve(len * 2 + 2);
for (int i = 0; i < len; ) { for (int i = 0; i < len; ) {
if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') {
char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 }; char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 };
@@ -462,7 +454,6 @@ extern "C" ANSOCR_API int ANSOCR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrH
utf16le += static_cast<char>((cp >> 8) & 0xFF); utf16le += static_cast<char>((cp >> 8) & 0xFF);
i += 6; i += 6;
} else { } else {
// ASCII or raw UTF-8 byte — convert as single char
utf16le += utf8Str[i]; utf16le += utf8Str[i];
utf16le += '\0'; utf16le += '\0';
i++; i++;
@@ -476,17 +467,19 @@ extern "C" ANSOCR_API int ANSOCR_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrH
return 1; return 1;
} }
// Raw UTF-8 — convert via Windows API
#ifdef _WIN32 #ifdef _WIN32
int wideLen = MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, nullptr, 0); int wideLen = MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, nullptr, 0);
if (wideLen <= 0) return 0; if (wideLen <= 0) return 0;
std::wstring wideStr(wideLen, 0); std::wstring wideStr(wideLen, 0);
MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, &wideStr[0], wideLen); MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, &wideStr[0], wideLen);
int size = wideLen * (int)sizeof(wchar_t); int dataSize = wideLen * (int)sizeof(wchar_t);
MgErr error = DSSetHandleSize(result, sizeof(int32) + size * sizeof(uChar)); int bomSize = includeBOM ? 2 : 0;
int totalSize = bomSize + dataSize;
MgErr error = DSSetHandleSize(result, sizeof(int32) + totalSize * sizeof(uChar));
if (error != noErr) return -2; if (error != noErr) return -2;
(*result)->cnt = size; (*result)->cnt = totalSize;
memcpy((*result)->str, wideStr.data(), size); if (includeBOM) memcpy((*result)->str, bom, 2);
memcpy((*result)->str + bomSize, wideStr.data(), dataSize);
return 1; return 1;
#else #else
return 0; return 0;

View File

@@ -1135,7 +1135,7 @@ namespace ANSCENTER {
}); });
} }
// ensure_ascii=true escapes non-ASCII chars as \uXXXX for LabVIEW compatibility // ensure_ascii=true escapes non-ASCII chars as \uXXXX for LabVIEW compatibility
return root.dump(-1, ' ', true); return root.dump();
} }
catch (const std::exception& e) { catch (const std::exception& e) {
return R"({"results":[],"error":"Serialization failed"})"; return R"({"results":[],"error":"Serialization failed"})";
@@ -2243,27 +2243,20 @@ namespace ANSCENTER {
} }
// Unicode conversion utilities for LabVIEW wrapper classes // Unicode conversion utilities for LabVIEW wrapper classes
extern "C" ANSENGINE_API int ANSEngine_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result) { extern "C" ANSENGINE_API int ANSEngine_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM) {
try { try {
if (!utf8Str || !result) return -1; if (!utf8Str || !result) return -1;
int len = (int)strlen(utf8Str); int len = (int)strlen(utf8Str);
if (len == 0) return 0; if (len == 0) return 0;
const char bom[2] = { '\xFF', '\xFE' };
bool hasUnicodeEscapes = false; bool hasUnicodeEscapes = false;
bool hasNonAscii = false; for (int i = 0; i + 1 < len; i++) {
for (int i = 0; i < len; i++) { if (utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { hasUnicodeEscapes = true; break; }
if ((unsigned char)utf8Str[i] >= 0x80) hasNonAscii = true;
if (i + 1 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') hasUnicodeEscapes = true;
}
if (!hasNonAscii && !hasUnicodeEscapes) {
MgErr error = DSSetHandleSize(result, sizeof(int32) + len * sizeof(uChar));
if (error != noErr) return -2;
(*result)->cnt = len;
memcpy((*result)->str, utf8Str, len);
return 1;
} }
if (hasUnicodeEscapes) { if (hasUnicodeEscapes) {
std::string utf16le; std::string utf16le;
utf16le.reserve(len * 2); if (includeBOM) utf16le.assign(bom, 2);
utf16le.reserve(len * 2 + 2);
for (int i = 0; i < len; ) { for (int i = 0; i < len; ) {
if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') {
char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 }; char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 };
@@ -2289,11 +2282,14 @@ extern "C" ANSENGINE_API int ANSEngine_ConvertUTF8ToUTF16LE(const char* utf8Str,
if (wideLen <= 0) return 0; if (wideLen <= 0) return 0;
std::wstring wideStr(wideLen, 0); std::wstring wideStr(wideLen, 0);
MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, &wideStr[0], wideLen); MultiByteToWideChar(CP_UTF8, 0, utf8Str, len, &wideStr[0], wideLen);
int size = wideLen * (int)sizeof(wchar_t); int dataSize = wideLen * (int)sizeof(wchar_t);
MgErr error = DSSetHandleSize(result, sizeof(int32) + size * sizeof(uChar)); int bomSize = includeBOM ? 2 : 0;
int totalSize = bomSize + dataSize;
MgErr error = DSSetHandleSize(result, sizeof(int32) + totalSize * sizeof(uChar));
if (error != noErr) return -2; if (error != noErr) return -2;
(*result)->cnt = size; (*result)->cnt = totalSize;
memcpy((*result)->str, wideStr.data(), size); if (includeBOM) memcpy((*result)->str, bom, 2);
memcpy((*result)->str + bomSize, wideStr.data(), dataSize);
return 1; return 1;
#else #else
return 0; return 0;

View File

@@ -1154,7 +1154,7 @@ namespace ANSCENTER
} }
// Unicode conversion utilities for LabVIEW wrapper classes // Unicode conversion utilities for LabVIEW wrapper classes
extern "C" ANSENGINE_API int ANSEngine_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result); extern "C" ANSENGINE_API int ANSEngine_ConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM = 1);
extern "C" ANSENGINE_API int ANSEngine_ConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result); extern "C" ANSENGINE_API int ANSEngine_ConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result);
#endif #endif

View File

@@ -244,7 +244,7 @@ extern "C" ANSULT_API int SendEmail(const char* smtpServer, int port,
extern "C" ANSULT_API int RebootSystem(); extern "C" ANSULT_API int RebootSystem();
// Unicode conversion utilities for LabVIEW // Unicode conversion utilities for LabVIEW
extern "C" ANSULT_API int ANSConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result); extern "C" ANSULT_API int ANSConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM = 1);
extern "C" ANSULT_API int ANSDecodeJsonUnicodeToUTF16LE(const char* escapedStr, LStrHandle result); extern "C" ANSULT_API int ANSDecodeJsonUnicodeToUTF16LE(const char* escapedStr, LStrHandle result);
extern "C" ANSULT_API int ANSConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result); extern "C" ANSULT_API int ANSConvertUTF16LEToUTF8(const unsigned char* utf16leBytes, int byteLen, LStrHandle result);

View File

@@ -798,27 +798,20 @@ extern "C" ANSULT_API int RebootSystem() {
} }
} }
extern "C" ANSULT_API int ANSConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result) { extern "C" ANSULT_API int ANSConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandle result, int includeBOM) {
try { try {
if (!utf8Str || !result) return -1; if (!utf8Str || !result) return -1;
int len = (int)strlen(utf8Str); int len = (int)strlen(utf8Str);
if (len == 0) return 0; if (len == 0) return 0;
const char bom[2] = { '\xFF', '\xFE' };
bool hasUnicodeEscapes = false; bool hasUnicodeEscapes = false;
bool hasNonAscii = false; for (int i = 0; i + 1 < len; i++) {
for (int i = 0; i < len; i++) { if (utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { hasUnicodeEscapes = true; break; }
if ((unsigned char)utf8Str[i] >= 0x80) hasNonAscii = true;
if (i + 1 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') hasUnicodeEscapes = true;
}
if (!hasNonAscii && !hasUnicodeEscapes) {
MgErr error = DSSetHandleSize(result, sizeof(int32) + len * sizeof(uChar));
if (error != noErr) return -2;
(*result)->cnt = len;
memcpy((*result)->str, utf8Str, len);
return 1;
} }
if (hasUnicodeEscapes) { if (hasUnicodeEscapes) {
std::string utf16le; std::string utf16le;
utf16le.reserve(len * 2); if (includeBOM) utf16le.assign(bom, 2);
utf16le.reserve(len * 2 + 2);
for (int i = 0; i < len; ) { for (int i = 0; i < len; ) {
if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') { if (i + 5 < len && utf8Str[i] == '\\' && utf8Str[i + 1] == 'u') {
char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 }; char hex[5] = { utf8Str[i + 2], utf8Str[i + 3], utf8Str[i + 4], utf8Str[i + 5], 0 };
@@ -841,11 +834,14 @@ extern "C" ANSULT_API int ANSConvertUTF8ToUTF16LE(const char* utf8Str, LStrHandl
} }
std::string converted = ANSCENTER::ANSUtilities::ConvertUTF8ToUTF16LE(utf8Str); std::string converted = ANSCENTER::ANSUtilities::ConvertUTF8ToUTF16LE(utf8Str);
if (converted.empty()) return 0; if (converted.empty()) return 0;
int size = static_cast<int>(converted.size()); int dataSize = static_cast<int>(converted.size());
MgErr error = DSSetHandleSize(result, sizeof(int32) + size * sizeof(uChar)); int bomSize = includeBOM ? 2 : 0;
int totalSize = bomSize + dataSize;
MgErr error = DSSetHandleSize(result, sizeof(int32) + totalSize * sizeof(uChar));
if (error != noErr) return -2; if (error != noErr) return -2;
(*result)->cnt = size; (*result)->cnt = totalSize;
memcpy((*result)->str, converted.data(), size); if (includeBOM) memcpy((*result)->str, bom, 2);
memcpy((*result)->str + bomSize, converted.data(), dataSize);
return 1; return 1;
} }
catch (...) { return -1; } catch (...) { return -1; }