223 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			223 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| local DataTableUtils = require("Utils.DataTableUtils")
 | ||
| local ESlateVisibility = import("ESlateVisibility")
 | ||
| local SlateBlueprintLibrary = import("SlateBlueprintLibrary")
 | ||
| local WidgetLayoutLibrary = import("WidgetLayoutLibrary")
 | ||
| local BusyGamePlayLibrary = import("BusyGamePlayLibrary")
 | ||
| 
 | ||
| local CUT_MASK_DISPLAY_TIME = 1.2  -- 刀光显示的时长
 | ||
| local CUT_MASK_FADEOUT_TIME = 0.6  -- 刀光开始渐隐的时间点
 | ||
| 
 | ||
| --- @class PreCookCenterWidget
 | ||
| --- @field ImgContainer table
 | ||
| --- @field ImgCookMaterial table
 | ||
| --- @field BtnMain table
 | ||
| local PreCookCenterWidget = {}
 | ||
| 
 | ||
| function PreCookCenterWidget:ctor()
 | ||
|     self.mouse_tracks = {}      -- 记录当前鼠标的轨迹
 | ||
|     self.rendering_tracks = {}  -- 正在被渲染的刀光
 | ||
|     self.is_pressed = false
 | ||
| end
 | ||
| 
 | ||
| 
 | ||
| function PreCookCenterWidget:OnInitialized()
 | ||
|     self.bHasScriptImplementedTick = true
 | ||
| 
 | ||
|     self:BP_BindLuaEnhancedInput(self.IA_TouchBegin, function()
 | ||
|         self.is_pressed = true
 | ||
|         self.mouse_tracks = {}
 | ||
|         self.rendering_tracks = {self.mouse_tracks}
 | ||
|         print("new track start")
 | ||
|     end)
 | ||
| 
 | ||
|     self:BP_BindLuaEnhancedInput(self.IA_TouchEnd, function()
 | ||
|         self.is_pressed = false
 | ||
|         self.mouse_tracks = {}
 | ||
|         print("track end")
 | ||
|     end)
 | ||
| 
 | ||
|     self.BtnMain:SetVisibility(ESlateVisibility.Collapsed)
 | ||
| end
 | ||
| 
 | ||
| function PreCookCenterWidget:Construct()
 | ||
|     -- self.bHasScriptImplementedTick = true
 | ||
| end
 | ||
| 
 | ||
| function PreCookCenterWidget:Destruct()
 | ||
| 
 | ||
| end
 | ||
| 
 | ||
| function PreCookCenterWidget:SetEmpty()
 | ||
|     self.ImgContainer:SetVisibility(ESlateVisibility.Collapsed)
 | ||
|     self.ImgCookMaterial:SetVisibility(ESlateVisibility.Collapsed)
 | ||
| end
 | ||
| 
 | ||
| function PreCookCenterWidget:AddContainer(pre_cook_contianer_id)
 | ||
|     local row = DataTableUtils.GetDataTableRow("PreCookItemConfig", pre_cook_contianer_id)
 | ||
|     if not row then return end
 | ||
|     self.ImgContainer:SetBrushFromSoftTexture(row.CenterDisplayResource, true)
 | ||
|     self.ImgContainer:SetVisibility(ESlateVisibility.SelfHitTestInvisible)
 | ||
| end
 | ||
| 
 | ||
| function PreCookCenterWidget:RemoveContainer()
 | ||
|     self.ImgContainer:SetVisibility(ESlateVisibility.Collapsed)
 | ||
| end
 | ||
| 
 | ||
| function PreCookCenterWidget:AddCookMaterial(pre_cook_material_id)
 | ||
|     local row = DataTableUtils.GetDataTableRow("PreCookItemConfig", pre_cook_material_id)
 | ||
|     if not row then return end
 | ||
|     self.ImgCookMaterial:SetBrushFromSoftTexture(row.CenterDisplayResource, true)
 | ||
|     self.ImgCookMaterial:SetVisibility(ESlateVisibility.SelfHitTestInvisible)
 | ||
| end
 | ||
| 
 | ||
| --- 从起点到终点画一条线,以这条线为新的x坐标轴,将所有的点坐标映射到新坐标系下
 | ||
| local function TransformCurveToEndpointAxes(points)
 | ||
|     local A = points[1]
 | ||
|     local B = points[#points]
 | ||
| 
 | ||
|     -- 计算向量AB
 | ||
|     local dx = B.X - A.X
 | ||
|     local dy = B.Y - A.Y
 | ||
|     local len = math.sqrt(dx * dx + dy * dy)
 | ||
|     if len == 0 then
 | ||
|         return {}
 | ||
|         -- error("Start and end points are the same, cannot define X-axis.")
 | ||
|     end
 | ||
| 
 | ||
|     -- 计算X轴单位向量
 | ||
|     local ux = dx / len
 | ||
|     local uy = dy / len
 | ||
| 
 | ||
|     -- 计算Y轴单位向量(逆时针旋转90度)
 | ||
|     local vx = -uy
 | ||
|     local vy = ux
 | ||
| 
 | ||
|     -- 映射所有点到新坐标系
 | ||
|     local new_points = {}
 | ||
|     for i, point in ipairs(points) do
 | ||
|         local apx = point.X - A.X
 | ||
|         local apy = point.Y - A.Y
 | ||
|         local new_x = apx * ux + apy * uy  -- 点积与X轴单位向量
 | ||
|         local new_y = apx * vx + apy * vy  -- 点积与Y轴单位向量
 | ||
|         new_points[i] = {X = new_x, Y = new_y}
 | ||
|     end
 | ||
| 
 | ||
|     return new_points
 | ||
| end
 | ||
| 
 | ||
| 
 | ||
| --- 将曲线的Y坐标规范到(-0.5,0.5)的范围内,供材质使用
 | ||
| local function NormalizeCurveYToHalfRange(points)
 | ||
|     if #points < 2 then return end
 | ||
|     local length = points[#points].X - points[1].X
 | ||
|     for _, point in pairs(points) do
 | ||
|         point.Y = - point.Y / length  -- 临时加个取反
 | ||
|         point.X = point.X / length
 | ||
|     end
 | ||
|     return points
 | ||
| end
 | ||
| 
 | ||
| 
 | ||
| local function UpdateCutMaskData(rendering_tracks, delta_time)
 | ||
|     local new_visible_tracks = {}
 | ||
|     for _, track in ipairs(rendering_tracks) do
 | ||
|         local is_visible = false
 | ||
|         for _, point in pairs(track) do
 | ||
|             local remain = math.max(point.remain - delta_time, 0)
 | ||
|             point.remain = remain
 | ||
|             if remain > 0 then is_visible = true end
 | ||
|         end
 | ||
|         if is_visible then
 | ||
|             table.insert(new_visible_tracks, track)
 | ||
|         end
 | ||
|     end
 | ||
|     return new_visible_tracks
 | ||
| end
 | ||
| 
 | ||
| local function DrawCutMaskImage(widget, mouse_tracks)
 | ||
|     local FVector2D = import("Vector2D")
 | ||
|     local FWidgetTransform = import("WidgetTransform")
 | ||
|     -- 设置图片合理的位移、旋转、缩放的参数
 | ||
|     local translation, scale = FVector2D(), FVector2D()
 | ||
|     local render_transform = FWidgetTransform()
 | ||
|     local first_point, last_point = mouse_tracks[1], mouse_tracks[#mouse_tracks]
 | ||
|     local delta_x = last_point.X - first_point.X
 | ||
|     local delta_y = last_point.Y - first_point.Y
 | ||
|     local mask_length = (delta_x^2 + delta_y^2)^0.5  -- 轨迹长度,确定缩放参数
 | ||
|     translation.X, translation.Y = first_point.X, first_point.Y  -- 第一个点确定图片唯一
 | ||
|     scale.X, scale.Y = mask_length / 512, 1
 | ||
|     render_transform.Scale = scale
 | ||
|     render_transform.Translation = translation
 | ||
|     render_transform.Angle = (math.atan(delta_y, delta_x) / (2 * math.pi)) * 360  -- 第一个点与最后一个点连线确定图片旋转角度
 | ||
| 
 | ||
|     widget:SetRenderTransform(render_transform)
 | ||
| end
 | ||
| 
 | ||
| 
 | ||
| -- 更新刀痕的材质
 | ||
| local function UpdateCusMaskMaterial(widget, texture, mouse_track)
 | ||
|     local transformed_tracks = TransformCurveToEndpointAxes(mouse_track)
 | ||
|     local normalize_tracks = NormalizeCurveYToHalfRange(transformed_tracks)
 | ||
| 
 | ||
|     local offsets = {}
 | ||
|     for _, track in ipairs(normalize_tracks) do
 | ||
|         table.insert(offsets, track.Y)
 | ||
|     end
 | ||
| 
 | ||
|     BusyGamePlayLibrary.UpdateTextureBuffer(texture, offsets)
 | ||
| 
 | ||
|     local material = widget:GetDynamicMaterial()
 | ||
|     material:SetTextureParameterValue("Param", texture)
 | ||
|     material:SetScalarParameterValue("VertexCount", #offsets)
 | ||
|     material:SetScalarParameterValue("SourceWidth", 512)
 | ||
| end
 | ||
| 
 | ||
| function PreCookCenterWidget:GetValidCutMaskWidget()
 | ||
|     return self.ImgMask
 | ||
| end
 | ||
| 
 | ||
| function PreCookCenterWidget:ResetAllCutMaskWidget()
 | ||
| end
 | ||
| 
 | ||
| 
 | ||
| function PreCookCenterWidget:Tick(geometry, delta_time)
 | ||
|     -- 计算鼠标点被限定在该区域下的坐标
 | ||
| 
 | ||
|     local size = SlateBlueprintLibrary.GetLocalSize(geometry)
 | ||
|     local cursor_pos = WidgetLayoutLibrary.GetMousePositionOnViewport(self)
 | ||
|     local left_top = SlateBlueprintLibrary.GetLocalTopLeft(geometry)
 | ||
| 
 | ||
|     -- local fixed_x = math.min(math.max(cursor_pos.X - left_top.X, 0), size.X)
 | ||
|     -- local fixed_y = math.min(math.max(cursor_pos.Y - left_top.Y, 0), size.Y)
 | ||
| 
 | ||
|     local fixed_x, fixed_y = cursor_pos.X - left_top.X, cursor_pos.Y - left_top.Y
 | ||
|     if fixed_x < 0 or fixed_x > size.X or fixed_y < 0 or fixed_y > size.Y then return end
 | ||
| 
 | ||
| 
 | ||
|     -- 更新鼠标移动轨迹
 | ||
|     if self.is_pressed then
 | ||
|         local last_point = self.mouse_tracks[#self.mouse_tracks]
 | ||
|         if not last_point or math.abs(last_point.X - fixed_x) > 1 or math.abs(last_point.Y - fixed_y) > 1 then
 | ||
|             table.insert(self.mouse_tracks, {X=fixed_x, Y=fixed_y, remain=0.5})
 | ||
|         end
 | ||
|     end
 | ||
| 
 | ||
|     -- 更新正在渲染的轨迹数据
 | ||
|     local rendering_tracks = UpdateCutMaskData(self.rendering_tracks, delta_time)
 | ||
| 
 | ||
|     -- 绘制刀迹
 | ||
|     for _, track in ipairs(rendering_tracks) do
 | ||
|         if #track > 2 then
 | ||
|             local widget = self:GetValidCutMaskWidget()
 | ||
|             DrawCutMaskImage(widget, track)
 | ||
|             -- UpdateCusMaskMaterial(widget, self.DataTexture, track)
 | ||
|         end
 | ||
|     end
 | ||
| 
 | ||
|     -- print("Ticking", #rendering_tracks)
 | ||
| 
 | ||
|     self.rendering_tracks = rendering_tracks
 | ||
| end
 | ||
| 
 | ||
| 
 | ||
| return Class(nil, nil, PreCookCenterWidget) |