Create Project
- Blazor Server App
- [Auth]-[Individual User Account]-[Save in User Account App]
data:image/s3,"s3://crabby-images/0f396/0f396cf5d1eb09d3dd234ee8bcf8e0fe33950ea1" alt=""
Active Database
- Startup.cs
public void ConfigureServices(IServiceCollection services)
{
services.AddDbContext<ApplicationDbContext>(options =>
options.UseSqlServer(
Configuration.GetConnectionString("DefaultConnection")));
...
}
Create a new Database
- RankingDB
- Create in Local DB
- Click [Property]
- Copy context in [Link String]
data:image/s3,"s3://crabby-images/1a5c5/1a5c5fd67d8c93ca8f7651ea553612ec951103b2" alt=""
data:image/s3,"s3://crabby-images/3fdf9/3fdf9abb536eb0289b5612526b3384fed93b2d38" alt=""
data:image/s3,"s3://crabby-images/8a2df/8a2dfb922d545cdc705b669cea05da92337fc6ab" alt=""
- Paste in
appsetting.json
{
"ConnectionStrings": {
"DefaultConnection": "Data Source=(localdb)\\MSSQLLocalDB;Initial Catalog=RankingDB;Integrated Security=True;Connect Timeout=30;Encrypt=False;TrustServerCertificate=False;ApplicationIntent=ReadWrite;MultiSubnetFailover=False"
},
...
Register User
- Register User in Web
- Execute the solution
- Click [Register]
- Fill forms and Click [Register] button
- If you register first user, then click [Migrations Apply] button then the button will be change to [Migrations Applied]
- Press [F5] on Keyborad or refresh button in web
data:image/s3,"s3://crabby-images/30104/301046438a4d8fd4df3b3c94cbf29f6606a801f5" alt=""
data:image/s3,"s3://crabby-images/c03ef/c03efbe75d889f69e445fed9f7871abbf2c397fb" alt=""
data:image/s3,"s3://crabby-images/86563/86563ec42a6032a7aa9d82836b666433777b04b9" alt=""
- Click [Click here to comfirm your account], then your email will be comfirmed
- Login with this account
data:image/s3,"s3://crabby-images/ceee7/ceee78a78355c67cfda74749f25a32e9e6ad1bf8" alt=""
data:image/s3,"s3://crabby-images/7a5b5/7a5b53f40b3759b49533261fda09aef7640d4a14" alt=""
data:image/s3,"s3://crabby-images/2a0e8/2a0e8a7e243e2cbeacb0f65d6bcaa58d09e9accd" alt=""
- You can see your log in data in topbar
data:image/s3,"s3://crabby-images/8c96f/8c96fe87ab77cbd6ab4bc1b82930ed7ae0ed042a" alt=""
Check User
- Check User in Database
- Open
dbo.AspNetUsers
Data - You can check your registed account
- Open
data:image/s3,"s3://crabby-images/7fb4a/7fb4a080acff69e7d7201a9f1c7c1f95f0a844ac" alt=""
data:image/s3,"s3://crabby-images/30671/30671a7486db94b495f9331d07425bac0090abb0" alt=""
data:image/s3,"s3://crabby-images/ebb7b/ebb7b97c254e954f84ad1bf5c1f9e577d89c0cce" alt=""
Migration
- Models
- Create
Data\Models
folder
- Create
data:image/s3,"s3://crabby-images/031cd/031cd19535711df2caf85b7c19661573072b0b4a" alt=""
- Data\Models\GameResult.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
namespace RankingApp.Data.Models
{
public class GameResult
{
public int Id { get; set; }
public int UserId { get; set; }
public string UserName { get; set; }
public int Score { get; set; }
public DateTime Date { get; set; }
}
}
- Data\ApplicationDbContext.cs
public DbSet<GameResult> GameResults { get; set; }
data:image/s3,"s3://crabby-images/156e1/156e1aeee988332c4670e52d065271b9d06b6f96" alt=""
data:image/s3,"s3://crabby-images/82d9b/82d9b758056b95ed4ca8e228c35d2f6b6fdc71dc" alt=""
- Input Data in Database directly
data:image/s3,"s3://crabby-images/1beba/1bebae86c4de335f1889a7b21104b0fc5c63085d" alt=""
data:image/s3,"s3://crabby-images/3ca1e/3ca1e59ce4b5eff297e61c2f12e13a748d32a3e3" alt=""
data:image/s3,"s3://crabby-images/4ae60/4ae609dcdc8cb1e6565c096c36623a2184874e49" alt=""
Database Version Management
- Up()
- Builds operations that will migrate the database
up
- That is, builds the operations that will take the database from the state left in by the previous migration so that it is up-to-date with regard to migartion
- This method must to be overridden in each class the inherits from
Migration
- Builds operations that will migrate the database
protected override void Up(MigrationBuilder migrationBuilder)
{
...
}
- Down()
- Builds operations that will migrate the database
down
- That is, builds the operations that will take the database from the state left in by the this migration so that it returns to the state that it was in before this migration was applied.
- This method must to be overridden in each class the inherits from
Migration
if bothup
anddown
migrations are to be supported. If it is not overridden, then calling it will throw and it will not to be possible to migrate in thedown
direction.
- Builds operations that will migrate the database
protected override void Down(MigrationBuilder migrationBuilder)
{
...
}
Authorization
- AuthorizeView
- Displays differing content depending on the user’s authorization status
- Authorized
- The content that will be displayed if the user is authorized.
- NotAuthorized
- The content that will be displayed if the user is not authorized.
<AuthorizeView>
<Authorized>
...
</Authorized>
<NotAuthorized>
...
</NotAuthorized>
</AuthorizeView>
data:image/s3,"s3://crabby-images/c446e/c446ead166aebb368101741efa564469155d164e" alt=""
data:image/s3,"s3://crabby-images/fea0e/fea0e3ed956d7cbbe0c13a8827f76406414d99c1" alt=""
CRUD
- FormResult Method
- Creates a System.Threading.Tasks.Task`1 that’s completed successfully with the specified result.
CREATE
public Task<GameResult> AddGameResult(GameResult gameResult)
{
_context.GameResults.Add(gameResult);
_context.SaveChanges();
return Task.FromResult(gameResult);
}
- SaveChanges Method
- Saves all changes made in this context to the database.
- This method will automatically call Microsoft.EntityFrameworkCore.ChangeTracking.ChangeTracker.DetectChanges to discover any changes to entity instances before saving to the underlying database.
- This can be disabled via Microsoft.EntityFrameworkCore.ChangeTracking.ChangeTracker.AutoDetectChangesEnabled.
void AddGameResult()
{
_showPopup = true;
_gameResult = new GameResult() { id = 0 };
}
async Task SaveGameResult()
{
if (_gameResult.id == 0)
{
_gameResult.Date = DateTime.Now;
var result = RankingService.AddGameResult(_gameResult);
}
...
_gameResults = await RankingService.GetGameResultsAsync();
_showPopup = false;
}
data:image/s3,"s3://crabby-images/dd62f/dd62f47cb530b1c88667579d54c875a1744454b2" alt=""
READ
public Task<List<GameResult>> GetGameResultsAsync()
{
List<GameResult> results = _context.GameResults
.OrderByDescending(item => item.Score)
.ToList();
return Task.FromResult(results);
}
protected override async Task OnInitializedAsync()
{
_gameResults = await RankingService.GetGameResultsAsync();
}
UPDATE
public Task<bool> UpdateGameResult(GameResult gameResult)
{
var findResult = _context.GameResults
.Where(x => x.Id == gameResult.Id)
.FirstOrDefault();
if (findResult == null)
return Task.FromResult(false) ;
findResult.UserName = gameResult.UserName;
findResult.Score = gameResult.Score;
_context.SaveChanges();
return Task.FromResult(true);
}
void UpdateGameResult(GameResult gameResult)
{
_showPopup = true;
_gameResult = gameResult;
}
async Task SaveGameResult()
{
if (_gameResult.Id == 0)
...
else
{
var result = RankingService.UpdateGameResult(_gameResult);
}
_gameResults = await RankingService.GetGameResultsAsync();
_showPopup = false;
}
DELETE
public Task<bool> DeleteGameResult(GameResult gameResult)
{
var findResult = _context.GameResults
.Where(x => x.Id == gameResult.Id)
.FirstOrDefault();
if (findResult == null)
return Task.FromResult(false);
_context.GameResults.Remove(gameResult);
_context.SaveChanges();
return Task.FromResult(true);
}
async Task DeleteGameResult(GameResult gameResult)
{
var result = RankingService.DeleteGameResult(gameResult);
_gameResults = await RankingService.GetGameResultsAsync();
}
SharedData
Add a new project
- SharedData
- Class Library(.Net Core)
- Data\GameResult.cs
using System;
using System.Collections.Generic;
using System.Text;
namespace SharedData.Models
{
public class GameResult
{
public int Id { get; set; }
public int UserId { get; set; }
public string UserName { get; set; }
public int Score { get; set; }
public DateTime Date { get; set; }
}
}
data:image/s3,"s3://crabby-images/2cb1c/2cb1c564117c96a553157d1b1e628e6ac50ede15" alt=""
PostMan
Download
- Postman
- Download Postman
- deactivate [Settings]-[Enable SSL certificate verification]
data:image/s3,"s3://crabby-images/8fd33/8fd33e43f1010e407adfa41b7ba4654e8a57b8f9" alt=""
WebApi
Add a new project
- WebApi
- ASP.NET Core Web Application
- template: API
- package: Microsoft.EntityFrameworkCore, Microsoft.EntityFrameworkCore.Design, Microsoft.EntityFrameworkCore.SqlServer
data:image/s3,"s3://crabby-images/530ef/530efccea13e694bdbddc8a5b1bb123a14ef1ad4" alt=""
Reference
- Dependency
- references
SharedData
- references
Database Connect
- Data\ApplicationDbContext.cs
using Microsoft.EntityFrameworkCore;
using SharedData.Models;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
namespace WebApi.Data
{
public class ApplicationDbContext: DbContext
{
public DbSet<GameResult>GameResults { get; set; }
public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options)
: base(options){ }
}
}
- Startup.cs
public void ConfigureServices(IServiceCollection services)
{
...
services.AddDbContext<ApplicationDbContext>(options =>
options.UseSqlServer(
Configuration.GetConnectionString("DefaultConnection")));
}
- appsettings.json
{
"ConnectionStrings": {
"DefaultConnection": "Data Source=(localdb)\\MSSQLLocalDB;Initial Catalog=RankingApiDB;Integrated Security=True;Connect Timeout=30;Encrypt=False;TrustServerCertificate=False;ApplicationIntent=ReadWrite;MultiSubnetFailover=False"
},
...
}
data:image/s3,"s3://crabby-images/73548/7354809ae6756f2f0f04936bfa4911f5f8470f2d" alt=""
Controllers\RankingController.cs
- REST (Representational State Transfer)
- Reusing features from the original HTTP communication to create a data transmission rule
- ApiController feature
- can return C# object
- if return null, then 204 Response(No Content) in cli
- string → text/plain
- rest(int, bool) → application/json
- Read
- ex: GET/ api/ ranking → get all items
- ex: GET/ api/ ranking/1 → get item which is id=1
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using SharedData.Models;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using WebApi.Data;
namespace WebApi.Controllers
{
[Route("api/[controller]")]
[ApiController]
public class RankingController : ControllerBase
{
ApplicationDbContext _context;
public RankingController(ApplicationDbContext context)
{
_context = context;
}
// Read
[HttpGet]
public List<GameResult> GetGameResults()
{
List<GameResult> results = _context.GameResults
.OrderByDescending(item => item.Score)
.ToList();
return results;
}
[HttpGet("{id}")]
public GameResult GetGameResults(int id)
{
GameResult result = _context.GameResults
.Where(item => item.Id == id)
.FirstOrDefault();
return result;
}
...
- Add data in
dbo.GameResults
data:image/s3,"s3://crabby-images/34ea6/34ea6e2579e619aeef01b587123c485481029810" alt=""
data:image/s3,"s3://crabby-images/f1fc3/f1fc38b6a289d117c3e0882665e05d6f44ca0bb1" alt=""
data:image/s3,"s3://crabby-images/e9bb7/e9bb7bccbefca9b3014e2287553f17a3d0708ddc" alt=""
data:image/s3,"s3://crabby-images/28862/28862a1a8d7b744a14d8b08635a745ab3c5a7c3a" alt=""
data:image/s3,"s3://crabby-images/3786a/3786af4f67b4150cdf45198b7be0393d105a646b" alt=""
- Create
- ex: POST/ api/ ranking → Create item(real data in Body)
...
[HttpPost]
public GameResult AddGameResult([FromBody]GameResult gameResult)
{
_context.GameResults.Add(gameResult);
_context.SaveChanges();
return gameResult;
}
...
data:image/s3,"s3://crabby-images/9eacd/9eacd34041123463ad124b62ad9fa3a8403c2b6e" alt=""
- Update
- ex: PUT /api/ ranking(not used in web because of PUT authority problem) → request update item(real data in Body)
...
[HttpPut]
public bool UpdateGameResult([FromBody] GameResult gameResult)
{
var findResult = _context.GameResults
.Where(x => x.Id == gameResult.Id)
.FirstOrDefault();
if (findResult == null)
return false;
findResult.UserName = gameResult.UserName;
findResult.Score = gameResult.Score;
_context.SaveChanges();
return true;
}
...
data:image/s3,"s3://crabby-images/a3cae/a3cae3f06e022c18336de6abfe7fcdf77f9ba1e4" alt=""
data:image/s3,"s3://crabby-images/bbf41/bbf410ec224b865ed0b11751d4975f2079dfba3a" alt=""
- Delete
- DELETE/ api/ ranking/ 1(not used in web because of DELETE authority problem) → delete item which is id=1
...
[HttpDelete("{id}")]
public bool DeleteGameResult(int id)
{
var findResult = _context.GameResults
.Where(x => x.Id == id)
.FirstOrDefault();
if (findResult == null)
return false;
_context.GameResults.Remove(findResult);
_context.SaveChanges();
return true;
}
...
data:image/s3,"s3://crabby-images/16cca/16ccac27d45b5d4b3624d498e66d5c2b41582074" alt=""
data:image/s3,"s3://crabby-images/93693/9369347df67f5748c10bb81c6abc35dca25ceae2" alt=""
Interlock RankingApp and WebApi
RankingApp
- Reference
- reference
SharedData
- reference
- Folder Cleanup
- delete
Data\Models
- delete
data:image/s3,"s3://crabby-images/1ecb6/1ecb66b4538763084cb9ed272b8463d2fa23405b" alt=""
- Data\ApplicationDbContext.cs
using Microsoft.AspNetCore.Identity.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore;
using System;
using System.Collections.Generic;
using System.Text;
namespace RankingApp.Data
{
public class ApplicationDbContext : IdentityDbContext
{
public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options)
: base(options)
{ }
}
}
- Data\Services\RankingService.cs
using Newtonsoft.Json;
using SharedData.Models;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;
namespace RankingApp.Data.Services
{
public class RankingService
{
HttpClient _httpClient;
public RankingService(HttpClient client)
{
_httpClient = client;
}
// Create
public async Task<GameResult> AddGameResult(GameResult gameResult)
{
string jsonStr = JsonConvert.SerializeObject(gameResult);
var content = new StringContent(jsonStr, Encoding.UTF8, "application/json");
var result = await _httpClient.PostAsync("api/ranking", content);
if (result.IsSuccessStatusCode == false)
throw new Exception("AddGameResult Failed");
var resultContent = await result.Content.ReadAsStringAsync();
GameResult resGameResult = JsonConvert.DeserializeObject<GameResult>(resultContent);
return resGameResult;
}
// Read
public async Task<List<GameResult>> GetGameResultsAsync()
{
var result = await _httpClient.GetAsync("api/ranking");
var resultContent = await result.Content.ReadAsStringAsync();
List<GameResult> resGameResults = JsonConvert.DeserializeObject<List<GameResult>>(resultContent);
return resGameResults;
}
// Update
public async Task<bool> UpdateGameResult(GameResult gameResult)
{
string jsonStr = JsonConvert.SerializeObject(gameResult);
var content = new StringContent(jsonStr, Encoding.UTF8, "application/json");
var result = await _httpClient.PutAsync("api/ranking", content);
if (result.IsSuccessStatusCode == false)
throw new Exception("AddGameResult Failed");
return true;
}
// Delete
public async Task<bool> DeleteGameResult(GameResult gameResult)
{
var result = await _httpClient.DeleteAsync($"api/ranking/{gameResult.Id}");
if (result.IsSuccessStatusCode == false)
throw new Exception("DeleteGameResult Faild");
return true;
}
}
}
- Razor.razor
...
async Task SaveGameResult()
{
if (_gameResult.Id == 0)
{
_gameResult.Date = DateTime.Now;
var result = await RankingService.AddGameResult(_gameResult);
}
else
{
var result = await RankingService.UpdateGameResult(_gameResult);
}
_gameResults = await RankingService.GetGameResultsAsync();
_showPopup = false;
}
- Startup.cs
- use number in
Properties\launchSetting.json
- use number in
public void ConfigureServices(IServiceCollection services)
{
...
services.AddScoped<AuthenticationStateProvider, RevalidatingIdentityAuthenticationStateProvider<IdentityUser>>();
services.AddHttpClient<RankingService>(c =>
{
c.BaseAddress = new Uri("https://localhost:44313");
});
}
data:image/s3,"s3://crabby-images/f739c/f739cd2f3e65802c43474dc3b431c91fa5447caa" alt=""
Build
Start several project at once
- Solution
- [Property] - [Several start project]
data:image/s3,"s3://crabby-images/9f73f/9f73f708d4020302d261c90b9b8d0d2322a59cb8" alt=""
Result(CRUD)
CREATE
READ
data:image/s3,"s3://crabby-images/b98bd/b98bd3ffa999f5e31acd514cceb83c96278f332e" alt=""
UPDATE
DELETE
Interlock RankingApp, WebApi and Unity
WebTest
- WebManager.cs
- Put your WebApi Server Port number in
_baseUrl
- Put your WebApi Server Port number in
using System;
using System.Collections;
using System.Collections.Generic;
using System.Text;
using UnityEngine;
using UnityEngine.Networking;
public class GameResult
{
public string userName;
public int score;
}
public class WebManager : MonoBehaviour
{
string _baseUrl = "https://localhost:44358/api";
void Start()
{
GameResult res = new GameResult()
{
userName = "Hanna",
score = 999
};
SendPostRequest("ranking", res, (uwr) =>
{
Debug.Log("Post Success!");
});
SendGetAllRequest("ranking", (uwr) =>
{
Debug.Log("Get All Success!");
});
}
public void SendPostRequest(String url, object obj, Action<UnityWebRequest> callback)
{
StartCoroutine(CoSendWebRequest(url, "POST", obj, callback));
}
public void SendGetAllRequest(String url, Action<UnityWebRequest> callback)
{
StartCoroutine(CoSendWebRequest(url, "GET", null, callback));
}
IEnumerator CoSendWebRequest(string url, string method, object obj, Action<UnityWebRequest> callback)
{
yield return null;
string sendUrl = $"{_baseUrl}/{url}/";
byte[] jsonBytes = null;
if(obj != null)
{
string jsonStr = JsonUtility.ToJson(obj);
jsonBytes = Encoding.UTF8.GetBytes(jsonStr);
}
var uwr = new UnityWebRequest(sendUrl, method);
uwr.uploadHandler = new UploadHandlerRaw(jsonBytes);
uwr.downloadHandler = new DownloadHandlerBuffer();
uwr.SetRequestHeader("Content-Type", "application/json");
yield return uwr.SendWebRequest();
if(uwr.isNetworkError || uwr.isHttpError)
{
Debug.Log(uwr.error);
}
else
{
Debug.Log("Recy " + uwr.downloadHandler.text);
callback.Invoke(uwr);
}
}
}
- GameObject
data:image/s3,"s3://crabby-images/4f2a8/4f2a83431d8518887837e31c38810a77b028e58a" alt=""
Result
data:image/s3,"s3://crabby-images/1e3a4/1e3a4ca438e54ecc64581913d8eb5a39c26f6f0f" alt=""