Known框架實戰(zhàn)演練——進(jìn)銷存財務(wù)管理

本文介紹如何實現(xiàn)進(jìn)銷存管理系統(tǒng)的財務(wù)對賬模塊,財務(wù)對賬模塊包括供應(yīng)商對賬和客戶對賬2個菜單頁面荷辕。供應(yīng)商和客戶對賬字段相同,因此可共用一個頁面組件類控嗜。

1. 配置模塊

運行項目疆栏,在【系統(tǒng)管理-模塊管理】中配置如下模塊菜單承边,配置教程參考之前的教程石挂。

一級模塊 二級模塊 代碼 圖標(biāo) Url 描述
財務(wù)管理 Finance property-safety
客戶對賬單 CustomerAccount unordered-list /fms/CustomerAccount 查詢和維護(hù)客戶對賬單信息富岳。
供應(yīng)商對賬單 SupplierAccount unordered-list /fms/SupplierAccount 查詢和維護(hù)供應(yīng)商對賬單信息拯腮。

2. 實體類

JxcLite項目Entities文件夾下面添加JxAccountHead.csJxAccountList.cs兩個實體類文件,實體類代碼可以直接復(fù)制模塊管理中由模型設(shè)置生成的代碼萝喘。文章中只簡單描述一下實體類的定義,具體代碼參見開源启妹,代碼定義如下:

namespace JxcLite.Entities;

/// <summary>
/// 對賬單表頭信息類饶米。
/// </summary>
public class JxAccountHead : EntityBase { }

/// <summary>
/// 對賬單表體信息類。
/// </summary>
public class JxAccountList : EntityBase { }

3. 建表腳本

打開JxcLite.Web項目Resources文件夾下的Tables.sql資源文件车胡,復(fù)制粘貼由【模塊管理-模型設(shè)置】中生成的建表腳本檬输。文章中只簡單描述一下建表腳本,具體腳本參見開源匈棘,內(nèi)容如下:

CREATE TABLE [JxAccountHead] (
    [Id]         varchar(50)      NOT NULL PRIMARY KEY,
    ...
    [Files]      nvarchar(500)    NULL
);

CREATE TABLE [JxAccountList] (
    [Id]         varchar(50)      NOT NULL PRIMARY KEY,
    ...
    [BillId]     varchar(50)      NOT NULL
);

4. 服務(wù)接口

JxcLite項目Services文件夾下面添加財務(wù)管理模塊服務(wù)接口丧慈,文件名定義為IFinanceService.cs,該接口定義前后端交互的Api訪問方法羹饰,包括分頁查詢伊滋、批量刪除實體、保存實體队秩。具體方法定義如下:

namespace JxcLite.Services;

public interface IFinanceService : IService
{
    //分頁查詢客戶或供應(yīng)商對賬單笑旺,通過查詢條件Type字段篩選
    Task<PagingResult<JxAccountHead>> QueryAccountsAsync(PagingCriteria criteria);
    //根據(jù)賬單類型獲取默認(rèn)對賬單信息
    Task<JxAccountHead> GetDefaultAccountAsync(string type);
    //批量刪除對賬單表頭及表體信息
    Task<Result> DeleteAccountsAsync(List<JxAccountHead> models);
    //保存對賬單表頭信息
    Task<Result> SaveAccountAsync(UploadInfo<JxAccountHead> info);
}

5. 服務(wù)實現(xiàn)

JxcLite.Web項目Services文件夾下面添加財務(wù)管理模塊服務(wù)接口的實現(xiàn)類乌妙,文件名定義為FinanceService.cs熊经,文章中只簡單描述一下實現(xiàn)類的定義和繼承槐壳,具體實現(xiàn)參見開源,定義如下:

namespace JxcLite.Web.Services;

class FinanceService(Context context) : ServiceBase(context), IFinanceService
{
    public Task<PagingResult<JxAccountHead>> QueryAccountsAsync(PagingCriteria criteria) { }
    public Task<JxAccountHead> GetDefaultAccountAsync(string type) { }
    public Task<Result> DeleteAccountsAsync(List<JxAccountHead> models) { }
    public Task<Result> SaveAccountAsync(UploadInfo<JxAccountHead> info) { }
}

雙擊打開JxcLite.Web項目中的AppWeb.cs文件崇堰,在AddJxcLiteCore方法中注冊服務(wù)類特幔,前端組件可以通過依賴注入工廠創(chuàng)建服務(wù)的實例。代碼如下:

public static class AppWeb
{
    public static void AddJxcLiteCore(this IServiceCollection services)
    {
        services.AddScoped<IFinanceService, FinanceService>();
    }
}

6. 數(shù)據(jù)依賴

JxcLite.Web項目Repositories文件夾下面添加財務(wù)管理模塊數(shù)據(jù)依賴類,文件名定義為FinanceRepository.cs困肩,文章中只簡單描述一下依賴類的定義,具體實現(xiàn)參見開源,定義如下:

namespace JxcLite.Web.Repositories;

class FinanceRepository
{
    internal static Task<PagingResult<JxAccountHead>> QueryAccountsAsync(Database db, PagingCriteria criteria) { }
    internal static Task<List<JxBillList>> GetBillListsAsync(Database db, string headId) { }
    //根據(jù)前綴獲取最大業(yè)務(wù)單號
    internal static Task<string> GetMaxAccountNoAsync(Database db, string prefix) { }
    internal static Task DeleteAccountListsAsync(Database db, string headId) { }
    internal static Task DeleteAccountListAsync(Database db, string headId, string billId) { }
}

7. 列表頁面

JxcLite.Client項目Pages\Finance文件夾下面添加AccountList.cs單據(jù)列表組件侈百,該組件是客戶和供應(yīng)商對賬單的列表組件共用類锭魔,具體實現(xiàn)參見開源漠秋,部分代碼如下:

namespace JxcLite.Client.Pages.Finance;

public class AccountList : BaseTablePage<JxAccountHead>
{
    private IFinanceService Service;
    //取得對賬類型(客戶、供應(yīng)商)焰雕,由具體對賬單頁面重寫該類型
    protected virtual string Type { get; }
    
    protected override async Task OnPageInitAsync()
    {
        await base.OnPageInitAsync();
        Service = await CreateServiceAsync<IFinanceService>();//創(chuàng)建服務(wù)
        Table.FormType = typeof(AccountForm);//自定義表單類型
        Table.OnQuery = QueryAccountsAsync;  //查詢方法
        //下面是設(shè)置列表欄位顯示的模板
        Table.Column(c => c.Status).Template((b, r) => b.Tag(r.Status));
        Table.Column(c => c.AccountDate).Type(FieldType.Date);
    }
    //新增
    public async void New()
    {
        var row = await Service.GetDefaultBillAsync(Type);
        Table.NewForm(Service.SaveBillAsync, row);
    }
    //編輯
    public void Edit(JxAccountHead row) => Table.EditForm(Service.SaveAccountAsync, row);
    //批量刪除和刪除
    public void DeleteM() => Table.DeleteM(Service.DeleteAccountsAsync);
    public void Delete(JxAccountHead row) => Table.Delete(Service.DeleteAccountsAsync, row);
    //導(dǎo)出
    public async void Export() => await ExportDataAsync();
    
    private Task<PagingResult<JxAccountHead>> QueryAccountsAsync(PagingCriteria criteria)
    {
        //設(shè)置對賬單類型查詢條件
        criteria.SetQuery(nameof(JxAccountHead.Type), QueryType.Equal, Type);
        return Service.QueryAccountsAsync(criteria);
    }
}

8. 表頭組件

首先打開JxcLite.Client項目Shared文件夾下面TypeForms.cs文件,添加對賬單表頭類型表單組件,代碼如下:

namespace JxcLite.Client.Shared;

public class AccountHeadTypeForm : AntForm<JxAccountHead> { }

再在JxcLite.Client項目Pages\Finance文件夾下面添加AccountHead.razor文件己肮,具體實現(xiàn)參見開源,部分代碼如下:

@inherits BaseForm<JxAccountHead>

<AccountHeadTypeForm Form="Model" ShowAction>
    <AntRow>
        <DataItem Span="8" Label="對賬單號" Required>
            <AntInput Disabled @bind-Value="@context.AccountNo" />
        </DataItem>
        <DataItem Span="8" Label="單證狀態(tài)">
            <KTag Text="@context.Status" />
        </DataItem>
        <DataItem Span="8" Label="對賬日期" Required>
            <AntDatePicker @bind-Value="@context.AccountDate" />
        </DataItem>
    </AntRow>
    <AntRow>
        <DataItem Span="8" Label="商業(yè)伙伴" Required>
            <PartnerPicker Value="@context.Partner" AllowClear Type="@context.Type"
                           ValueChanged="e=>context.Partner=e[0].Name" />
        </DataItem>
        <DataItem Span="8" Label="業(yè)務(wù)日期" Required>
            <AntRangePicker @bind-RangeValue="@context.BizDates" />
        </DataItem>
        <DataItem Span="8" Label="總金額">
            <AntDecimal @bind-Value="@context.TotalAmount" /> 元
        </DataItem>
    </AntRow>
    <AntRow>
        <DataItem Span="8" Label="合同號">
            <AntInput @bind-Value="@context.ContractNo" />
        </DataItem>
        <DataItem Span="8" Label="發(fā)票號">
            <AntInput @bind-Value="@context.InvoiceNo" />
        </DataItem>
    </AntRow>
    <AntRow>
        <DataItem Span="24" Label="備注">
            <AntTextArea @bind-Value="@context.Note" />
        </DataItem>
    </AntRow>
    <AntRow>
        <DataItem Span="24" Label="附件">
            <KUpload @ref="upload" ReadOnly="Model.IsView" Value="@context.Files"
                     OpenFile IsButton="!Model.Data.IsNew" OnFilesChanged="OnFilesChanged" />
        </DataItem>
    </AntRow>
</AccountHeadTypeForm>

@code {
    private KUpload upload;

    private async void OnFilesChanged(List<FileDataInfo> files)
    {
        if (Model.Data.IsNew)
        {
            Model.Files[nameof(JxAccountHead.Files)] = files;
        }
        else
        {
            Model.Files[nameof(JxAccountHead.Files)] = files;
            await Model.SaveAsync(d => upload.SetValue(d.Files), false);
        }
    }
}

9. 業(yè)務(wù)單表格組件

再在JxcLite.Client項目Shared文件夾下面添加BillHeadTable.cs文件般甲,該組件為對賬單明細(xì)列表鹅颊,具體實現(xiàn)參見開源觅闽,部分代碼如下:

namespace JxcLite.Client.Shared;

public class BillHeadTable : BaseTable<JxBillHead>
{
    private IBillService Service;

    [Parameter] public JxAccountHead Account { get; set; }

    protected override async Task OnInitAsync()
    {
        await base.OnInitAsync();
        Service = await CreateServiceAsync<IBillService>();
        Table.ShowPager = true;
        Table.OnQuery = QueryBillsAsync;
        if (!ReadOnly)
        {
            Table.Toolbar.AddAction(nameof(New));
            Table.Toolbar.AddAction(nameof(DeleteM));
            Table.SelectType = TableSelectType.Checkbox;
        }
        Table.AddColumn(c => c.BillNo, true).Width(100);
        Table.AddColumn(c => c.Status).Width(100).Template((b, r) => b.Tag(r.Status));
        Table.AddColumn(c => c.BillDate).Width(100).Type(FieldType.Date);
        Table.AddColumn(c => c.Partner).Width(150);
        Table.AddColumn(c => c.ContractNo).Width(100);
        Table.AddColumn(c => c.InvoiceNo).Width(100);
        Table.AddColumn(c => c.TotalAmount).Width(100).Sum();
        Table.AddColumn(c => c.Note).Width(200);
        if (!ReadOnly)
        {
            Table.AddAction(nameof(Delete));
        }
    }

    public void New() { }
    public void DeleteM() { }
    public void Edit(JxBillHead row) { }
    public void Delete(JxBillHead row) { }

    private Task<PagingResult<JxBillHead>> QueryBillsAsync(PagingCriteria criteria)
    {
        criteria.Parameters[nameof(BillQueryType)] = BillQueryType.Account;
        criteria.SetQuery("BizId", QueryType.Equal, Account.Id);
        return Service.QueryBillsAsync(criteria);
    }
}

10. 表單組件

再在JxcLite.Client項目Pages\Finance文件夾下面添加AccountForm.cs文件蛉拙,該組件為對賬單彈窗表單組件,分表頭信息和對賬明細(xì)兩個標(biāo)簽大咱,代碼如下:

namespace JxcLite.Client.Pages.Finance;

class AccountForm : BaseTabForm
{
    [Parameter] public FormModel<JxAccountHead> Model { get; set; }

    protected override async Task OnInitFormAsync()
    {
        await base.OnInitFormAsync();
        Tab.AddTab("表頭信息", BuildHead);
        Tab.AddTab("對賬明細(xì)", BuildList);
        Tab.Right = b => b.Tag(Model.Data.Status);
    }

    private void BuildHead(RenderTreeBuilder builder)
    {
        builder.Component<AccountHead>().Set(c => c.Model, Model).Build();
    }

    private void BuildList(RenderTreeBuilder builder)
    {
        builder.Component<BillHeadTable>()
               .Set(c => c.ReadOnly, Model.IsView)
               .Set(c => c.Account, Model.Data)
               .Build();
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末间景,一起剝皮案震驚了整個濱河市倘要,隨后出現(xiàn)的幾起案子圾亏,更是在濱河造成了極大的恐慌十拣,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,602評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件志鹃,死亡現(xiàn)場離奇詭異夭问,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)曹铃,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,442評論 2 382
  • 文/潘曉璐 我一進(jìn)店門缰趋,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人陕见,你說我怎么就攤上這事秘血。” “怎么了评甜?”我有些...
    開封第一講書人閱讀 152,878評論 0 344
  • 文/不壞的土叔 我叫張陵直撤,是天一觀的道長。 經(jīng)常有香客問我蜕着,道長,這世上最難降的妖魔是什么红柱? 我笑而不...
    開封第一講書人閱讀 55,306評論 1 279
  • 正文 為了忘掉前任承匣,我火速辦了婚禮,結(jié)果婚禮上锤悄,老公的妹妹穿的比我還像新娘韧骗。我一直安慰自己,他們只是感情好零聚,可當(dāng)我...
    茶點故事閱讀 64,330評論 5 373
  • 文/花漫 我一把揭開白布袍暴。 她就那樣靜靜地躺著,像睡著了一般隶症。 火紅的嫁衣襯著肌膚如雪政模。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,071評論 1 285
  • 那天蚂会,我揣著相機(jī)與錄音淋样,去河邊找鬼。 笑死胁住,一個胖子當(dāng)著我的面吹牛趁猴,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播彪见,決...
    沈念sama閱讀 38,382評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼儡司,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了余指?” 一聲冷哼從身側(cè)響起捕犬,我...
    開封第一講書人閱讀 37,006評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后或听,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體探孝,經(jīng)...
    沈念sama閱讀 43,512評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,965評論 2 325
  • 正文 我和宋清朗相戀三年誉裆,在試婚紗的時候發(fā)現(xiàn)自己被綠了顿颅。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,094評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡足丢,死狀恐怖粱腻,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情斩跌,我是刑警寧澤绍些,帶...
    沈念sama閱讀 33,732評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站耀鸦,受9級特大地震影響柬批,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜袖订,卻給世界環(huán)境...
    茶點故事閱讀 39,283評論 3 307
  • 文/蒙蒙 一氮帐、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧洛姑,春花似錦上沐、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,286評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至硫眯,卻和暖如春蕴侧,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背两入。 一陣腳步聲響...
    開封第一講書人閱讀 31,512評論 1 262
  • 我被黑心中介騙來泰國打工戈盈, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人谆刨。 一個月前我還...
    沈念sama閱讀 45,536評論 2 354
  • 正文 我出身青樓塘娶,卻偏偏與公主長得像,于是被迫代替她去往敵國和親痊夭。 傳聞我的和親對象是個殘疾皇子刁岸,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,828評論 2 345

推薦閱讀更多精彩內(nèi)容