百度AI开放平台接入

作者:追风剑情 发布于:2021-8-9 17:39 分类:Unity3d

  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.Net.Http;
  5. using UnityEngine;
  6. using UnityEngine.Networking;
  7. /// <summary>
  8. /// 百度AI开放平台
  9. /// 人像分割 https://ai.baidu.com/tech/body/seg
  10. /// 在线文档 https://ai.baidu.com/ai-doc/BODY/Fk3cpyxua
  11. /// 错误码查询 https://ai.baidu.com/ai-doc/IMAGERECOGNITION/Rk3bcxhio
  12. ///
  13. /// 人像动漫化 https://ai.baidu.com/tech/imageprocess/selfie_anime
  14. /// 在线文档 https://ai.baidu.com/ai-doc/IMAGEPROCESS/Mk4i6olx5
  15. /// </summary>
  16. public class BaiduAISDK : MonoBehaviour
  17. {
  18. //人像分割请求地址
  19. public const string BODY_SEG_URL = "https://aip.baidubce.com/rest/2.0/image-classify/v1/body_seg";
  20. public const string BODY_SEG_APP_ID = "";
  21. public const string BODY_SEG_APP_KEY = "";
  22. public const string BODY_SEG_APP_SECRET = "";
  23. public AccessTokenResult bodySegAccessTokenResult;
  24. //人像分割完成
  25. public Action<Texture2D, Texture2D, Texture2D> BodySegCompleted;
  26. //未检测到人像
  27. public Action OnNotFoundPersionError;
  28. //人像分割失败
  29. public Action BodySegError;
  30. //是否从人像分割图中裁剪出人物图片
  31. public bool IsClipPersion = false;
  32. //是否将人像上下居中显示
  33. public bool IsVeticallyMiddle = false;
  34.  
  35. //人像动漫化请求地址
  36. public const string SELFIE_ANIME_URL = "https://aip.baidubce.com/rest/2.0/image-process/v1/selfie_anime";
  37. public const string SELFIE_ANIME_APP_ID = "";
  38. public const string SELFIE_ANIME_APP_KEY = "";
  39. public const string SELFIE_ANIME_APP_SECRET = "";
  40. public AccessTokenResult selfieAnimeAccessTokenResult;
  41. //人像动漫化完成
  42. public Action<Texture2D> SelfieAnimeCompleted;
  43. //人像动漫化失败
  44. public Action SelfieAnimeError;
  45.  
  46. //证书验证
  47. public class AcceptCertificateHandler : CertificateHandler
  48. {
  49. //负责拒绝或接受在 https 请求上接收到的证书
  50. protected override bool ValidateCertificate(byte[] certificateData)
  51. {
  52. return true;
  53. }
  54. }
  55.  
  56. private void Start()
  57. {
  58. Debug.Log("*** request body_seg access token...");
  59. bodySegAccessTokenResult = GetAccessToken(BODY_SEG_APP_KEY, BODY_SEG_APP_SECRET);
  60. if (string.IsNullOrEmpty(bodySegAccessTokenResult.access_token))
  61. Debug.LogError("request body_seg access token failure");
  62.  
  63. Debug.Log("*** request selfie_anime access token...");
  64. selfieAnimeAccessTokenResult = GetAccessToken(SELFIE_ANIME_APP_KEY, SELFIE_ANIME_APP_SECRET);
  65. if (string.IsNullOrEmpty(selfieAnimeAccessTokenResult.access_token))
  66. Debug.LogError("request selfie_anime access token failure");
  67. }
  68.  
  69. //base64转Texture2D
  70. private Texture2D Base64ToTexture(string base64, int width, int height)
  71. {
  72. byte[] imgBytes = Convert.FromBase64String(base64);
  73. Texture2D new_tex = new Texture2D(width, height, TextureFormat.RGBA32, false);
  74. new_tex.LoadImage(imgBytes);
  75. new_tex.Apply();
  76. return new_tex;
  77. }
  78.  
  79. //去掉背景RGB值
  80. private void StripBackground(Texture2D tex)
  81. {
  82. Color32[] colors = tex.GetPixels32();
  83. for (int i = 0; i < colors.Length; i++)
  84. {
  85. Color32 c = colors[i];
  86. //人像边缘附近的alpha值介于0到1
  87. //根据文档上的说明,前景的alpha值大于0.5
  88. if (c.a <= 0.5f)
  89. {
  90. c.a = c.r = c.g = c.b = 0;
  91. colors[i] = c;
  92. }
  93. }
  94. tex.SetPixels32(colors);
  95. tex.Apply();
  96. }
  97.  
  98. //生成一张Alpha图
  99. private Texture2D GenerateAlphaTexture(Texture2D tex)
  100. {
  101. Color32[] colors = tex.GetPixels32();
  102. for (int i = 0; i < colors.Length; i++)
  103. {
  104. Color32 c = colors[i];
  105. c.r = c.g = c.b = c.a;
  106. colors[i] = c;
  107. }
  108.  
  109. Texture2D alpha_tex = new Texture2D(tex.width, tex.height, TextureFormat.RGBA32, false);
  110. alpha_tex.SetPixels32(colors);
  111. alpha_tex.Apply();
  112.  
  113. return alpha_tex;
  114. }
  115.  
  116. //生成裁剪后的人像图
  117. private Texture2D ClipPersionTexture(Texture2D tex, BodySegResult result)
  118. {
  119. if (result.person_info == null || result.person_info.Length == 0)
  120. return null;
  121. var rect = result.person_info[0];
  122. Debug.LogFormat("person_num={0}", result.person_num);
  123. Debug.LogFormat("(left={0}, top={1}, width={2}, height={3})",
  124. (int)rect.left, (int)rect.top, (int)rect.width, (int)rect.height);
  125.  
  126. //默认返回的人体框top不准确,这里进行重新定位
  127. rect.top = CalculatePersionTop(tex, result);
  128.  
  129. //Texture的原点坐标在左下角
  130. int x = (int)rect.left;
  131. int y = tex.height - (int)rect.top - (int)rect.height;
  132. y = Mathf.Clamp(y, 0, tex.height - 1);
  133. int width = (int)rect.width;
  134. int height = (int)rect.height;
  135.  
  136. Color[] colors = tex.GetPixels(x, y, width, height);
  137. Texture2D new_tex = new Texture2D(width, height, TextureFormat.RGBA32, false);
  138. new_tex.SetPixels(colors);
  139. new_tex.Apply();
  140. return new_tex;
  141. }
  142.  
  143. //计算人像top
  144. private int CalculatePersionTop(Texture2D tex, BodySegResult result)
  145. {
  146. int top = 0;
  147. if (result.person_info == null || result.person_info.Length == 0)
  148. return top;
  149. var rect = result.person_info[0];
  150. //从上到下扫描图片
  151. int y = tex.height - (int)rect.top;
  152. for (; y > 0; y--)
  153. {
  154. for (int x = 0; x < tex.width; x++)
  155. {
  156. Color c = tex.GetPixel(x, y);
  157. //根据百度返回的人体区域Alpha通道概率分数重新定位人体区域top
  158. if (c.a >= rect.score)
  159. {
  160. top = tex.height - y;
  161. goto FOUND;
  162. }
  163. }
  164. }
  165. FOUND:
  166. return top;
  167. }
  168.  
  169. //计算人像bottom
  170. private int CalculatePersionBottom(Texture2D tex, BodySegResult result)
  171. {
  172. int bottom = 0;
  173. if (result.person_info == null || result.person_info.Length == 0)
  174. return bottom;
  175. var rect = result.person_info[0];
  176. //从下到上扫描图片
  177. int topY = tex.height - (int)rect.top;
  178. int y = tex.height - (int)rect.top - (int)rect.height;
  179. y = Mathf.Clamp(y, 0, tex.height);
  180. for (; y < topY; y++)
  181. {
  182. for (int x = 0; x < tex.width; x++)
  183. {
  184. Color c = tex.GetPixel(x, y);
  185. if (c.a >= rect.score)
  186. {
  187. bottom = y;
  188. goto FOUND;
  189. }
  190. }
  191. }
  192. FOUND:
  193. return bottom;
  194. }
  195.  
  196. //将人像上下居中
  197. private Texture2D VeticallyMiddleTexture(Texture2D tex, BodySegResult result)
  198. {
  199. int top = CalculatePersionTop(tex, result);
  200. int bottom = CalculatePersionBottom(tex, result);
  201. int middleX = 0;
  202. int middleY = (top + bottom) / 2;
  203.  
  204. //提取人像区域color
  205. int x = 0;
  206. int y = bottom;
  207. int width = tex.width;
  208. int height = tex.height - top - bottom;
  209. Color[] colors = tex.GetPixels(x, y, width, height);
  210.  
  211. Debug.LogFormat("top={0}, bottom={1}, middleX={2}, middleY={3}, width={4}, height={5}",
  212. top, bottom, middleX, middleY, width, height);
  213.  
  214. //将人像上下居中
  215. Texture2D new_tex = new Texture2D(tex.width, tex.height, TextureFormat.RGBA32, false);
  216. new_tex.SetPixels(middleX, middleY, width, height, colors);
  217. Color transparent = new Color(0, 0, 0, 0);
  218. SetPixelRows(new_tex, middleY + height, tex.height, transparent);
  219. SetPixelRows(new_tex, 0, middleY, transparent);
  220. new_tex.Apply();
  221. return new_tex;
  222. }
  223.  
  224. private void SetPixelRows(Texture2D tex, int startY, int endY, Color color)
  225. {
  226. for (int i = startY; i <= endY; i++)
  227. SetPixelRow(tex, i, color);
  228. }
  229.  
  230. private void SetPixelRow(Texture2D tex, int y, Color color)
  231. {
  232. for (int x=0; x < tex.width; x++)
  233. {
  234. if (y < tex.height)
  235. tex.SetPixel(x, y, color);
  236. }
  237. }
  238.  
  239. // 合并Alpha通道
  240. public Texture2D CombineAlphaTexture(Texture2D rgbTex, Texture2D alphaTex)
  241. {
  242. Texture2D newTex = new Texture2D(rgbTex.width, rgbTex.height, TextureFormat.RGBA32, false);
  243. for (int y=0; y<rgbTex.height; y++)
  244. {
  245. for (int x = 0; x < rgbTex.width; x++)
  246. {
  247. Color c = rgbTex.GetPixel(x, y);
  248. Color alpha = alphaTex.GetPixel(x, y);
  249. c.a = alpha.a;
  250. newTex.SetPixel(x, y, c);
  251. }
  252. }
  253. newTex.Apply();
  254. return newTex;
  255. }
  256.  
  257. //获取 Access Token
  258. public AccessTokenResult GetAccessToken(string appKey, string secretKey)
  259. {
  260. String authHost = "https://aip.baidubce.com/oauth/2.0/token";
  261. HttpClient client = new HttpClient();
  262. List<KeyValuePair<String, String>> paraList = new List<KeyValuePair<string, string>>();
  263. paraList.Add(new KeyValuePair<string, string>("grant_type", "client_credentials"));
  264. paraList.Add(new KeyValuePair<string, string>("client_id", appKey));
  265. paraList.Add(new KeyValuePair<string, string>("client_secret", secretKey));
  266.  
  267. HttpResponseMessage response = client.PostAsync(authHost, new FormUrlEncodedContent(paraList)).Result;
  268. String result = response.Content.ReadAsStringAsync().Result;
  269. Debug.Log(result);
  270. var token = JsonUtility.FromJson<AccessTokenResult>(result);
  271. return token;
  272. }
  273.  
  274. //人像分割
  275. public void AsyncBodySeg(Texture2D tex)
  276. {
  277. if (tex == null)
  278. return;
  279. StartCoroutine(Post(BODY_SEG_URL, bodySegAccessTokenResult.access_token,
  280. "foreground", tex,
  281. //完成
  282. (json) => {
  283. var result = JsonUtility.FromJson<BodySegResult>(json);
  284. if (!string.IsNullOrEmpty(result.error_msg))
  285. {
  286. BodySegError?.Invoke();
  287. return;
  288. }
  289. if (result.person_num == 0)
  290. {
  291. OnNotFoundPersionError?.Invoke();
  292. return;
  293. }
  294. Texture2D new_tex = Base64ToTexture(result.foreground, tex.width, tex.height);
  295. Debug.LogFormat("Respose image size: width={0}, height={1}", new_tex.width, new_tex.height);
  296. Texture2D persion_tex;
  297. if (IsClipPersion)
  298. {
  299. new_tex = ClipPersionTexture(new_tex, result);
  300. persion_tex = new_tex;
  301. }
  302. else
  303. {
  304. persion_tex = ClipPersionTexture(new_tex, result);
  305. }
  306.  
  307. if (IsVeticallyMiddle)
  308. {
  309. new_tex = VeticallyMiddleTexture(new_tex, result);
  310. }
  311. Texture2D alpha_tex = GenerateAlphaTexture(new_tex);
  312. BodySegCompleted?.Invoke(new_tex, alpha_tex, persion_tex);
  313. },
  314. //失败
  315. () => {
  316. BodySegError?.Invoke();
  317. }));
  318. //注意:分割后的图片会将背景区的alpha设为0,同时会保留背景区的RGB值。
  319. }
  320.  
  321. //人像动漫化
  322. public void AsyncSelfieAnime(Texture2D tex)
  323. {
  324. if (tex == null)
  325. return;
  326.  
  327. StartCoroutine(Post(SELFIE_ANIME_URL, selfieAnimeAccessTokenResult.access_token,
  328. "anime", tex,
  329. //完成
  330. (json) => {
  331. var result = JsonUtility.FromJson<SelfieAnimeResult>(json);
  332. if (!string.IsNullOrEmpty(result.error_msg))
  333. {
  334. SelfieAnimeError?.Invoke();
  335. return;
  336. }
  337. Texture2D new_tex = Base64ToTexture(result.image, tex.width, tex.height);
  338. SelfieAnimeCompleted?.Invoke(new_tex);
  339. },
  340. //失败
  341. () => {
  342. SelfieAnimeError?.Invoke();
  343. }));
  344. //注意:返回的图片没有alpha通道
  345. }
  346.  
  347. //POST 请求
  348. private IEnumerator Post(string url, string accessToken, string type, Texture2D tex, Action<string> completedCallback, Action errorCallback)
  349. {
  350. yield return null;
  351. byte[] bytes = tex.EncodeToPNG();
  352. //byte[] bytes = tex.EncodeToJPG();
  353. string base64 = Convert.ToBase64String(bytes);
  354.  
  355. //创建表单
  356. WWWForm form = new WWWForm();
  357. form.AddField("type", type);
  358. form.AddField("image", base64);
  359.  
  360. url = string.Format("{0}?access_token={1}", url, accessToken);
  361. //执行请求
  362. UnityWebRequest request = UnityWebRequest.Post(url, form);
  363. //使用https需要设置certificateHandler
  364. request.certificateHandler = new AcceptCertificateHandler();
  365. request.useHttpContinue = false;
  366. request.chunkedTransfer = false;
  367. request.timeout = 10;
  368. request.downloadHandler = (DownloadHandler)new DownloadHandlerBuffer();
  369. request.SetRequestHeader("Content-Type", "application/x-www-form-urlencoded");
  370. request.SendWebRequest();
  371.  
  372. while (!request.isDone)
  373. {
  374. Debug.LogFormat("上传进度: {0}", request.uploadProgress);
  375. yield return new WaitForSeconds(0.1f);
  376. }
  377.  
  378. if (request.isHttpError || request.isNetworkError || request.responseCode != 200)
  379. {
  380. Debug.LogErrorFormat("上传出错\n{0}", request.error);
  381. errorCallback?.Invoke();
  382. yield break; //跳出协程
  383. }
  384.  
  385. Debug.Log("上传成功!");
  386.  
  387. string json = request.downloadHandler.text;
  388. //-------常见错误-------
  389. //{"error_code":18,"error_msg":"Open api qps request limit reached"}
  390. //原因: 请求次数已达上限
  391. //解决方法: 控制台领取免费额度,或者充值。
  392.  
  393. //{"error_code":6,"error_msg":"No permission to access data"}
  394. //没有请求权限
  395. //解决方法: 控制台领取免费额度,或者充值。
  396. //--------End-----------
  397. Debug.Log(json);
  398.  
  399. completedCallback?.Invoke(json);
  400. }
  401.  
  402. //Access Token 返回结果
  403. public class AccessTokenResult
  404. {
  405. public string refresh_token;
  406. public string expires_in;
  407. public string scope;
  408. public string session_key;
  409. public string access_token;
  410. public string session_secret;
  411.  
  412. public string error = string.Empty;
  413. public string error_description = string.Empty;
  414. }
  415.  
  416. //人像分割返回结果
  417. public class BodySegResult
  418. {
  419. //分割结果图片,base64编码之后的二值图像,需二次处理方能查看分割效果
  420. public string labelmap;
  421. /*
  422. 分割后人像前景的scoremap,归一到0-255,不用进行二次处理,直接解码保存图片即可。
  423. Base64编码后的灰度图文件,图片中每个像素点的灰度值 = 置信度 * 255,
  424. 置信度为原图对应像素点位于人体轮廓内的置信度,取值范围[0, 1]
  425. */
  426. public string scoremap;
  427. /*
  428. 分割后的人像前景抠图,透明背景,Base64编码后的png格式图片,不用进行二次处理,
  429. 直接解码保存图片即可。将置信度大于0.5的像素抠出来,
  430. 并通过image matting技术消除锯齿
  431. */
  432. public string foreground;
  433. //检测到的人体框数目
  434. public int person_num;
  435. //人体框信息
  436. public PersonRect[] person_info;
  437.  
  438. [Serializable]
  439. public class PersonRect
  440. {
  441. //人体区域的高度,注意当值为0时 数据类型为int
  442. public float height;
  443. //人体区域离左边界的距离,注意当值为0时 数据类型为int
  444. public float left;
  445. //人体区域离上边界的距离,注意当值为0时 数据类型为int
  446. public float top;
  447. //人体区域的宽度,注意当值为0时 数据类型为int
  448. public float width;
  449. //人体框的概率分数,取值0-1,,注意当值为0时 数据类型为int
  450. //alpha值>=score的区域即为人像区域
  451. public float score;
  452. }
  453.  
  454. public int error_code;
  455. public string error_msg;
  456. }
  457.  
  458. //人像动漫化返回
  459. public class SelfieAnimeResult
  460. {
  461. //唯一的log id,用于问题定位
  462. public UInt64 log_id;
  463. //处理后图片的Base64编码
  464. public string image;
  465.  
  466. public int error_code;
  467. public string error_msg;
  468. }
  469. }

原图
4444.png

分割出来的人像以及动漫化后的人像
3333.png

标签: Unity3d

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号