Browse Source

Merge branch 'master' of http://1.94.207.143:3000/chongqing/store-project into lyz_dev

Mcal 3 weeks ago
parent
commit
2e42d965bf
47 changed files with 1645 additions and 849 deletions
  1. 1 5
      .vscode/extensions.json
  2. 25 4
      README.md
  3. 0 5
      package.json
  4. 0 49
      pnpm-lock.yaml
  5. 423 168
      src-tauri/Cargo.lock
  6. 6 6
      src-tauri/Cargo.toml
  7. 0 10
      src-tauri/capabilities/default.json
  8. 0 78
      src-tauri/resources/init.sql
  9. 222 0
      src-tauri/resources/table_struct.sql
  10. 15 0
      src-tauri/src/commands/category.rs
  11. 15 0
      src-tauri/src/commands/commodity.rs
  12. 3 0
      src-tauri/src/commands/mod.rs
  13. 14 0
      src-tauri/src/commands/table.rs
  14. 44 16
      src-tauri/src/lib.rs
  15. 37 0
      src-tauri/src/models/menu_cate.rs
  16. 66 0
      src-tauri/src/models/menu_commodity.rs
  17. 57 0
      src-tauri/src/models/menu_sku.rs
  18. 47 0
      src-tauri/src/models/menu_sku_spec.rs
  19. 54 0
      src-tauri/src/models/menu_spec.rs
  20. 13 0
      src-tauri/src/models/mod.rs
  21. 11 0
      src-tauri/src/models/prelude.rs
  22. 34 0
      src-tauri/src/models/store_area.rs
  23. 36 0
      src-tauri/src/models/store_material.rs
  24. 40 0
      src-tauri/src/models/store_order.rs
  25. 40 0
      src-tauri/src/models/store_table.rs
  26. 39 0
      src-tauri/src/service/category.rs
  27. 2 0
      src-tauri/src/service/mod.rs
  28. 63 0
      src-tauri/src/service/table.rs
  29. 51 0
      src-tauri/src/utils.rs
  30. 21 17
      src/components/noProduct.vue
  31. 0 11
      src/config/db.ts
  32. 0 126
      src/config/kysely.ts
  33. 13 2
      src/hook/i18n.ts
  34. 4 7
      src/interface/MenuCate.ts
  35. 6 7
      src/interface/MenuProduct.ts
  36. 15 0
      src/interface/ProductSku.ts
  37. 8 0
      src/interface/ProductSpec.ts
  38. 12 0
      src/interface/StoreTable.ts
  39. 0 12
      src/model/index.ts
  40. 0 18
      src/model/menu_sku.ts
  41. 0 24
      src/model/menu_spec.ts
  42. 10 0
      src/service/CategoryService.ts
  43. 8 0
      src/service/TableService.ts
  44. 2 9
      src/store/user.ts
  45. 0 66
      src/utils/request.ts
  46. 184 161
      src/views/menu/index.vue
  47. 4 48
      src/views/test/index.vue

+ 1 - 5
.vscode/extensions.json

@@ -1,7 +1,3 @@
 {
-  "recommendations": [
-    "Vue.volar",
-    "tauri-apps.tauri-vscode",
-    "rust-lang.rust-analyzer"
-  ]
+  "recommendations": ["Vue.volar", "rust-lang.rust-analyzer"]
 }

+ 25 - 4
README.md

@@ -1,7 +1,28 @@
-# Tauri + Vue 3
+# 盼达点餐系统
 
-This template should help get you started developing with Tauri + Vue 3 in Vite. The template uses Vue 3 `<script setup>` SFCs, check out the [script setup docs](https://v3.vuejs.org/api/sfc-script-setup.html#sfc-script-setup) to learn more.
+## 数据库操作
 
-## Recommended IDE Setup
+1. 菜品信息
+   1. 菜品分类
+   2. 菜品粗略信息
+   3. sku 信息
+   4. 规格信息
+2. 桌子管理
+   1. 区域管理
+   2. 桌号管理
+3. 订单管理
+   1. 创建订单
+   2. 订单同步
+4. 材料管理
 
-- [VS Code](https://code.visualstudio.com/) + [Volar](https://marketplace.visualstudio.com/items?itemName=Vue.volar) + [Tauri](https://marketplace.visualstudio.com/items?itemName=tauri-apps.tauri-vscode) + [rust-analyzer](https://marketplace.visualstudio.com/items?itemName=rust-lang.rust-analyzer)
+## 数据同步
+
+1. 二维码展示&扫码
+2. 服务端 API
+   1. 获取分类
+   2. 获取商品
+   3. 订单创建
+   4. 桌子状态
+      1. 添加挂单
+      2. 追加菜品
+      3. 取消挂单/结账

+ 0 - 5
package.json

@@ -18,12 +18,8 @@
     "@element-plus/icons-vue": "^2.3.1",
     "@tailwindcss/vite": "^4.0.8",
     "@tauri-apps/api": "^2",
-    "@tauri-apps/plugin-http": "~2",
-    "@tauri-apps/plugin-opener": "^2",
     "@tauri-apps/plugin-os": "~2",
-    "@tauri-apps/plugin-sql": "^2.2.0",
     "@tauri-apps/plugin-store": "~2",
-    "@tauri-apps/plugin-upload": "~2",
     "element-plus": "^2.9.5",
     "pinia": "^3.0.1",
     "tailwindcss": "^4.0.8",
@@ -34,7 +30,6 @@
   "devDependencies": {
     "@tauri-apps/cli": "^2",
     "@vitejs/plugin-vue": "^5.2.1",
-    "kysely": "^0.27.6",
     "sass-embedded": "^1.85.0",
     "typescript": "^5.7.3",
     "unplugin-element-plus": "^0.9.1",

+ 0 - 49
pnpm-lock.yaml

@@ -17,24 +17,12 @@ importers:
       '@tauri-apps/api':
         specifier: ^2
         version: 2.2.0
-      '@tauri-apps/plugin-http':
-        specifier: ~2
-        version: 2.3.0
-      '@tauri-apps/plugin-opener':
-        specifier: ^2
-        version: 2.2.5
       '@tauri-apps/plugin-os':
         specifier: ~2
         version: 2.2.0
-      '@tauri-apps/plugin-sql':
-        specifier: ^2.2.0
-        version: 2.2.0
       '@tauri-apps/plugin-store':
         specifier: ~2
         version: 2.2.0
-      '@tauri-apps/plugin-upload':
-        specifier: ~2
-        version: 2.2.1
       element-plus:
         specifier: ^2.9.5
         version: 2.9.5(vue@3.5.13(typescript@5.7.3))
@@ -60,9 +48,6 @@ importers:
       '@vitejs/plugin-vue':
         specifier: ^5.2.1
         version: 5.2.1(vite@6.1.1(jiti@2.4.2)(lightningcss@1.29.1)(sass-embedded@1.85.0))(vue@3.5.13(typescript@5.7.3))
-      kysely:
-        specifier: ^0.27.6
-        version: 0.27.6
       sass-embedded:
         specifier: ^1.85.0
         version: 1.85.0
@@ -543,24 +528,12 @@ packages:
     engines: {node: '>= 10'}
     hasBin: true
 
-  '@tauri-apps/plugin-http@2.3.0':
-    resolution: {integrity: sha512-pigTvz+zzAqbIhCzRiR1GE98Jw7A03j2V+Eiexr9thBI8VfMiwFQMcbgON51xlwnVaI72LdbYKNajU84im8tlg==}
-
-  '@tauri-apps/plugin-opener@2.2.5':
-    resolution: {integrity: sha512-hHsJ9RPWpZvZEPVFaL+d25gABMUMOf/A6ESXnvf/ii9guTukj58WXsAE/SOysXRIhej7kseRCxnOnIMpSCdUsQ==}
-
   '@tauri-apps/plugin-os@2.2.0':
     resolution: {integrity: sha512-HszbCdbisMlu5QhCNAN8YIWyz2v33abAWha6+uvV2CKX8P5VSct/y+kEe22JeyqrxCnWlQ3DRx7s49Byg7/0EA==}
 
-  '@tauri-apps/plugin-sql@2.2.0':
-    resolution: {integrity: sha512-yGdybpaMENe/p6lTXslvDHYNNvD9qB7palaBBF5fJHdYSkwd3vrLiYU9dFfLwUAwnsBylND55EiivWsjhazejA==}
-
   '@tauri-apps/plugin-store@2.2.0':
     resolution: {integrity: sha512-hJTRtuJis4w5fW1dkcgftsYxKXK0+DbAqurZ3CURHG5WkAyyZgbxpeYctw12bbzF9ZbZREXZklPq8mocCC3Sgg==}
 
-  '@tauri-apps/plugin-upload@2.2.1':
-    resolution: {integrity: sha512-2EyVhJYLAp2mJH0UzO3QGU0vPk/YWvAfdI2wXbczyzEZY/AZVa9wConyB1TV/NGhyJRim4LwWgkmnEvcKLkECw==}
-
   '@types/estree@1.0.6':
     resolution: {integrity: sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==}
 
@@ -715,10 +688,6 @@ packages:
     resolution: {integrity: sha512-rg9zJN+G4n2nfJl5MW3BMygZX56zKPNVEYYqq7adpmMh4Jn2QNEwhvQlFy6jPVdcod7txZtKHWnyZiA3a0zP7A==}
     hasBin: true
 
-  kysely@0.27.6:
-    resolution: {integrity: sha512-FIyV/64EkKhJmjgC0g2hygpBv5RNWVPyNCqSAD7eTCv6eFWNIi4PN1UvdSJGicN/o35bnevgis4Y0UDC0qi8jQ==}
-    engines: {node: '>=14.0.0'}
-
   lightningcss-darwin-arm64@1.29.1:
     resolution: {integrity: sha512-HtR5XJ5A0lvCqYAoSv2QdZZyoHNttBpa5EP9aNuzBQeKGfbyH5+UipLWvVzpP4Uml5ej4BYs5I9Lco9u1fECqw==}
     engines: {node: '>= 12.0.0'}
@@ -1394,30 +1363,14 @@ snapshots:
       '@tauri-apps/cli-win32-ia32-msvc': 2.2.7
       '@tauri-apps/cli-win32-x64-msvc': 2.2.7
 
-  '@tauri-apps/plugin-http@2.3.0':
-    dependencies:
-      '@tauri-apps/api': 2.2.0
-
-  '@tauri-apps/plugin-opener@2.2.5':
-    dependencies:
-      '@tauri-apps/api': 2.2.0
-
   '@tauri-apps/plugin-os@2.2.0':
     dependencies:
       '@tauri-apps/api': 2.2.0
 
-  '@tauri-apps/plugin-sql@2.2.0':
-    dependencies:
-      '@tauri-apps/api': 2.2.0
-
   '@tauri-apps/plugin-store@2.2.0':
     dependencies:
       '@tauri-apps/api': 2.2.0
 
-  '@tauri-apps/plugin-upload@2.2.1':
-    dependencies:
-      '@tauri-apps/api': 2.2.0
-
   '@types/estree@1.0.6': {}
 
   '@types/lodash-es@4.17.12':
@@ -1623,8 +1576,6 @@ snapshots:
 
   jiti@2.4.2: {}
 
-  kysely@0.27.6: {}
-
   lightningcss-darwin-arm64@1.29.1:
     optional: true
 

File diff suppressed because it is too large
+ 423 - 168
src-tauri/Cargo.lock


+ 6 - 6
src-tauri/Cargo.toml

@@ -18,17 +18,17 @@ crate-type = ["staticlib", "cdylib", "rlib"]
 tauri-build = { version = "2", features = [] }
 
 [dependencies]
+sea-orm = { version = "1.1.0", features = [
+    "sqlx-sqlite",
+    "runtime-tokio-rustls",
+    "macros",
+] }
 tauri = { version = "2", features = [] }
-tauri-plugin-opener = "2"
 serde = { version = "1", features = ["derive"] }
 serde_json = "1"
 tauri-plugin-store = "2"
-tauri-plugin-http = "2"
-tauri-plugin-upload = "2"
 tauri-plugin-os = "2"
-[dependencies.tauri-plugin-sql]
-features = ["sqlite"] # or "postgres", or "mysql"
-version = "2"
+anyhow = "1.0.97"
 
 [target.'cfg(not(any(target_os = "android", target_os = "ios")))'.dependencies]
 tauri-plugin-single-instance = "2"

+ 0 - 10
src-tauri/capabilities/default.json

@@ -7,17 +7,7 @@
   ],
   "permissions": [
     "core:default",
-    "opener:default",
-    "sql:default",
-    "sql:allow-execute",
     "store:default",
-    {
-      "identifier": "http:default",
-      "allow": [
-        "http://192.168.1.110:8080"
-      ]
-    },
-    "upload:default",
     "os:default"
   ]
 }

+ 0 - 78
src-tauri/resources/init.sql

@@ -1,78 +0,0 @@
--- 创建菜单分类表
-CREATE TABLE
-    menu_cate (
-        id INTEGER PRIMARY KEY AUTOINCREMENT, -- 主键
-        name TEXT DEFAULT '', -- 分类名称
-        logo TEXT DEFAULT '', -- 分类Logo
-        local_logo TEXT DEFAULT '', -- 本地Logo路径
-        description TEXT, -- 分类描述
-        create_time DATETIME DEFAULT NULL, -- 创建时间
-        update_time DATETIME DEFAULT NULL, -- 更新时间
-        sort INTEGER DEFAULT 0, -- 排序
-        status TEXT DEFAULT 'A' -- 状态 (A-Active, D-Deleted)
-    );
-
--- 创建商品表
-CREATE TABLE
-    menu_commodity (
-        id INTEGER PRIMARY KEY AUTOINCREMENT, -- 主键
-        type TEXT DEFAULT 'product', -- 商品类型
-        merchant_id INTEGER DEFAULT 0, -- 商户ID
-        store_id INTEGER DEFAULT 0, -- 店铺ID
-        name TEXT DEFAULT '', -- 商品名称
-        cate_id INTEGER DEFAULT 0, -- 分类ID
-        goods_no TEXT DEFAULT '', -- 商品编号
-        is_single_spec TEXT DEFAULT 'Y', -- 是否单规格 (Y-Yes, N-No)
-        logo TEXT DEFAULT '', -- 商品Logo
-        images TEXT DEFAULT '', -- 商品图片
-        price REAL DEFAULT 0.00, -- 价格
-        line_price REAL DEFAULT 0.00, -- 划线价
-        stock INTEGER DEFAULT 0, -- 库存
-        weight REAL DEFAULT 0.00, -- 重量
-        coupon_ids TEXT DEFAULT '', -- 优惠券ID
-        service_time INTEGER DEFAULT 0, -- 服务时间
-        init_sale INTEGER DEFAULT 0, -- 初始销量
-        sale_point TEXT DEFAULT '', -- 卖点
-        can_use_point TEXT DEFAULT 'N', -- 是否可用积分 (Y-Yes, N-No)
-        is_member_discount TEXT DEFAULT 'Y', -- 是否会员折扣 (Y-Yes, N-No)
-        sort INTEGER DEFAULT 0, -- 排序
-        description TEXT, -- 商品描述
-        create_time DATETIME DEFAULT NULL, -- 创建时间
-        update_time DATETIME DEFAULT NULL, -- 更新时间
-        operator TEXT DEFAULT NULL, -- 操作员
-        status TEXT DEFAULT 'A' -- 状态 (A-Active, D-Deleted)
-    );
-
--- 创建SKU表
-CREATE TABLE
-    menu_sku (
-        id INTEGER PRIMARY KEY AUTOINCREMENT, -- 主键
-        sku_no TEXT DEFAULT '', -- SKU编号
-        logo TEXT DEFAULT '', -- SKU Logo
-        goods_id INTEGER NOT NULL DEFAULT 0, -- 商品ID
-        stock INTEGER NOT NULL DEFAULT 0, -- 库存
-        price REAL NOT NULL DEFAULT 0.00, -- 价格
-        line_price REAL NOT NULL DEFAULT 0.00, -- 划线价
-        weight REAL DEFAULT 0.00, -- 重量
-        status TEXT NOT NULL DEFAULT 'A' -- 状态 (A-Active, D-Deleted)
-    );
-
--- 创建规格表
-CREATE TABLE
-    menu_spec (
-        id INTEGER PRIMARY KEY AUTOINCREMENT, -- 主键
-        goods_id INTEGER NOT NULL DEFAULT 0, -- 商品ID
-        name TEXT NOT NULL DEFAULT '', -- 规格名称
-        value TEXT NOT NULL DEFAULT '', -- 规格值
-        status TEXT DEFAULT 'A' -- 状态 (A-Active, D-Deleted)
-    );
-
--- 创建SKU和规格的中间表
-CREATE TABLE
-    menu_sku_spec (
-        sku_id INTEGER NOT NULL, -- SKU ID
-        spec_id INTEGER NOT NULL, -- 规格ID
-        PRIMARY KEY (sku_id, spec_id),
-        FOREIGN KEY (sku_id) REFERENCES menu_sku (id),
-        FOREIGN KEY (spec_id) REFERENCES menu_spec (id)
-    );

+ 222 - 0
src-tauri/resources/table_struct.sql

@@ -0,0 +1,222 @@
+-- 创建菜单分类表
+CREATE TABLE
+    menu_cate (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 主键
+        name TEXT NOT NULL DEFAULT '', -- 分类名称
+        logo TEXT NOT NULL DEFAULT '', -- 分类Logo
+        description TEXT, -- 分类描述
+        create_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 创建时间
+        update_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 更新时间
+        sort INTEGER NOT NULL DEFAULT 0, -- 排序
+        status INTEGER NOT NULL DEFAULT 1 -- 状态 (1-Active, 0-Inactive)
+    );
+
+-- 新增菜单分类表的更新时间触发器
+CREATE TRIGGER trg_menu_cate_update_time AFTER
+UPDATE ON menu_cate FOR EACH ROW WHEN NEW.update_time = OLD.update_time BEGIN
+UPDATE menu_cate
+SET
+    update_time = CURRENT_TIMESTAMP
+WHERE
+    id = OLD.id;
+
+END;
+
+-- 创建商品表
+CREATE TABLE
+    menu_commodity (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 主键
+        name TEXT NOT NULL DEFAULT '', -- 商品名称
+        cate_id INTEGER NOT NULL DEFAULT 0, -- 分类ID
+        goods_no TEXT NOT NULL DEFAULT '', -- 商品编号
+        is_single_spec INTEGER NOT NULL DEFAULT 1, -- 是否单规格 (1-单规格, 0-多规格)
+        logo TEXT NOT NULL DEFAULT '', -- 商品Logo
+        price REAL NOT NULL DEFAULT 0.00, -- 价格
+        line_price REAL NOT NULL DEFAULT 0.00, -- 商品原价
+        stock INTEGER NOT NULL DEFAULT 0, -- 库存
+        sort INTEGER NOT NULL DEFAULT 0, -- 排序
+        description TEXT NOT NULL DEFAULT '', -- 商品描述
+        create_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 创建时间
+        update_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 更新时间
+        status INTEGER NOT NULL DEFAULT 1, -- 状态 (1-Active, 0-Inactive),
+        FOREIGN KEY (cate_id) REFERENCES menu_cate (id)
+    );
+
+-- 新增订单退款表的更新时间触发器
+CREATE TRIGGER trg_menu_commodity_update_time AFTER
+UPDATE ON menu_commodity FOR EACH ROW WHEN NEW.update_time = OLD.update_time BEGIN
+UPDATE menu_commodity
+SET
+    update_time = CURRENT_TIMESTAMP
+WHERE
+    id = OLD.id;
+
+END;
+
+-- 创建SKU表
+CREATE TABLE
+    menu_sku (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 主键
+        sku_no TEXT NOT NULL DEFAULT '', -- SKU编号
+        logo TEXT NOT NULL DEFAULT '', -- SKU Logo
+        goods_id INTEGER NOT NULL DEFAULT 0, -- 商品ID
+        stock INTEGER NOT NULL DEFAULT 0, -- 库存
+        price REAL NOT NULL DEFAULT 0.00, -- 价格
+        line_price REAL NOT NULL DEFAULT 0.00, -- 商品原价
+        status INTEGER NOT NULL DEFAULT 1, -- 状态 (1-Active, 0-Inactive),
+        FOREIGN KEY (goods_id) REFERENCES menu_commodity (id)
+    );
+
+-- 创建规格表
+CREATE TABLE
+    menu_spec (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 主键
+        goods_id INTEGER NOT NULL DEFAULT 0, -- 商品ID
+        name TEXT NOT NULL DEFAULT '', -- 规格名称
+        value TEXT NOT NULL DEFAULT '', -- 规格值
+        status INTEGER NOT NULL DEFAULT 1, -- 状态 (1-Active, 0-Inactive),
+        FOREIGN KEY (goods_id) REFERENCES menu_commodity (id)
+    );
+
+-- 创建SKU和规格的中间表
+CREATE TABLE
+    menu_sku_spec (
+        sku_id INTEGER NOT NULL, -- SKU ID
+        spec_id INTEGER NOT NULL, -- 规格ID
+        PRIMARY KEY (sku_id, spec_id),
+        FOREIGN KEY (sku_id) REFERENCES menu_sku (id),
+        FOREIGN KEY (spec_id) REFERENCES menu_spec (id)
+    );
+
+-- 创建店铺订单表信息
+CREATE TABLE
+    store_order (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 自增ID
+        order_sn TEXT NOT NULL DEFAULT '', -- 订单号
+        table_id INTEGER NOT NULL, -- 所属桌码ID
+        pay_type INTEGER NOT NULL DEFAULT 0, -- 支付方式 (1-现金, 0-Paypal)
+        amount REAL NOT NULL DEFAULT 0.00, -- 订单金额
+        pay_amount REAL NOT NULL DEFAULT 0.00, -- 支付金额
+        discount REAL NOT NULL DEFAULT 0.00, -- 折扣金额
+        param TEXT NOT NULL DEFAULT '', -- 订单参数
+        service_fee REAL DEFAULT NULL, -- 服务费
+        remark TEXT NOT NULL DEFAULT '', -- 用户备注
+        create_time DATETIME DEFAULT NULL, -- 创建时间
+        update_time DATETIME DEFAULT NULL, -- 更新时间
+        status TEXT NOT NULL DEFAULT 'A', -- 订单状态
+        pay_time DATETIME DEFAULT NULL, -- 支付时间
+        pay_status INTEGER NOT NULL DEFAULT 1, -- 支付状态
+        settle_status INTEGER NOT NULL DEFAULT 1, -- 结算状态
+        goods TEXT NOT NULL DEFAULT '{}' -- 订单商品信息
+    );
+
+-- 新增订单表的更新时间触发器
+CREATE TRIGGER trg_order_update_time AFTER
+UPDATE ON store_order FOR EACH ROW WHEN NEW.update_time = OLD.update_time BEGIN
+UPDATE store_order
+SET
+    update_time = CURRENT_TIMESTAMP
+WHERE
+    id = OLD.id;
+
+END;
+
+-- 创建店铺材料表信息
+CREATE TABLE
+    store_material (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 材料表id
+        material_no TEXT NOT NULL, -- 材料编号
+        material_type TEXT NOT NULL, -- 材料类别
+        name TEXT NOT NULL, -- 材料名称
+        specification TEXT NOT NULL, -- 规格
+        inventory INTEGER NOT NULL DEFAULT 0, -- 库存数量
+        shelf_life INTEGER NOT NULL DEFAULT 0, -- 保质期
+        unit TEXT NOT NULL, -- 单位
+        img TEXT NOT NULL, -- 材料文件/图片id
+        purchase_period INTEGER NOT NULL DEFAULT 0, -- 采购期数
+        create_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 创建时间
+        update_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 更新时间
+        status INTEGER NOT NULL DEFAULT 1 -- 状态0下架1上架
+    );
+
+-- 新增材料表的更新时间触发器
+CREATE TRIGGER trg_material_update_time AFTER
+UPDATE ON store_material FOR EACH ROW WHEN NEW.update_time = OLD.update_time BEGIN
+UPDATE store_material
+SET
+    update_time = CURRENT_TIMESTAMP
+WHERE
+    id = OLD.id;
+
+END;
+
+-- 店铺区域表
+CREATE TABLE
+    store_area (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 区域id
+        name TEXT NOT NULL, -- 区域名称
+        description TEXT NOT NULL DEFAULT '', -- 区域描述
+        create_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 创建时间
+        update_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 更新时间
+        status INTEGER NOT NULL DEFAULT 1 -- 状态 (1-有效, 0-无效)
+    );
+
+-- 新增区域表的更新时间触发器
+CREATE TRIGGER trg_area_update_time AFTER
+UPDATE ON store_area FOR EACH ROW WHEN NEW.update_time = OLD.update_time BEGIN
+UPDATE store_area
+SET
+    update_time = CURRENT_TIMESTAMP
+WHERE
+    id = OLD.id;
+
+END;
+
+-- 店铺座位表
+CREATE TABLE
+    store_table (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 桌子id
+        area_id INTEGER NOT NULL, -- 所属区域id
+        table_no TEXT NOT NULL, -- 桌号
+        capacity INTEGER NOT NULL DEFAULT 0, -- 座位数
+        create_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 创建时间
+        update_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 更新时间
+        status INTEGER NOT NULL DEFAULT 1, -- 状态 (1-有效, 0-无效)
+        FOREIGN KEY (area_id) REFERENCES store_area (id)
+    );
+
+-- 新增桌子表的更新时间触发器
+CREATE TRIGGER trg_table_update_time AFTER
+UPDATE ON store_table FOR EACH ROW WHEN NEW.update_time = OLD.update_time BEGIN
+UPDATE store_table
+SET
+    update_time = CURRENT_TIMESTAMP
+WHERE
+    id = OLD.id;
+
+END;
+
+-- 创建订单退款表
+CREATE TABLE
+    store_order_refund (
+        id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, -- 主键
+        order_id INTEGER NOT NULL, -- 关联订单ID
+        refund_amount REAL NOT NULL DEFAULT 0.00, -- 退款金额
+        refund_reason TEXT NOT NULL DEFAULT '', -- 退款原因
+        refund_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 退款时间
+        create_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 创建时间
+        update_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, -- 更新时间
+        status INTEGER NOT NULL DEFAULT 1, -- 状态 (1-有效, 0-无效)
+        FOREIGN KEY (order_id) REFERENCES store_order (id)
+    );
+
+-- 新增订单退款表的更新时间触发器
+CREATE TRIGGER trg_order_refund_update_time AFTER
+UPDATE ON store_order_refund FOR EACH ROW WHEN NEW.update_time = OLD.update_time BEGIN
+UPDATE store_order_refund
+SET
+    update_time = CURRENT_TIMESTAMP
+WHERE
+    id = OLD.id;
+
+END;

+ 15 - 0
src-tauri/src/commands/category.rs

@@ -0,0 +1,15 @@
+use crate::{
+    CmdResult, CmdState,
+    service::category::{Category, CategoryWithSpu},
+};
+
+#[tauri::command]
+pub async fn get_category(app_state: CmdState<'_>) -> CmdResult<Vec<CategoryWithSpu>> {
+    let category = Category::new(app_state.conn.clone());
+    let categories = category.get_category_commodity().await;
+    match categories {
+        Ok(categories) => Ok(categories),
+        Err(e) => Err(format!("请求分类失败{:?}", e)),
+    }
+    // Ok("111".into())
+}

+ 15 - 0
src-tauri/src/commands/commodity.rs

@@ -0,0 +1,15 @@
+use crate::{
+    CmdResult, CmdState,
+    service::category::{Category, CategoryWithSpu},
+};
+
+#[tauri::command]
+pub async fn get_commodity(app_state: CmdState<'_>) -> CmdResult<Vec<CategoryWithSpu>> {
+    let category = Category::new(app_state.conn.clone());
+    let categories = category.get_category_commodity().await;
+    match categories {
+        Ok(categories) => Ok(categories),
+        Err(e) => Err(format!("请求分类失败{:?}", e)),
+    }
+    // Ok("111".into())
+}

+ 3 - 0
src-tauri/src/commands/mod.rs

@@ -0,0 +1,3 @@
+pub mod category;
+pub mod commodity;
+pub mod table;

+ 14 - 0
src-tauri/src/commands/table.rs

@@ -0,0 +1,14 @@
+use crate::{
+    CmdResult, CmdState,
+    service::table::{AreaWithTable, Table},
+};
+
+#[tauri::command]
+pub async fn get_table_area(app_state: CmdState<'_>) -> CmdResult<Vec<AreaWithTable>> {
+    let table_service = Table::new(app_state.conn.clone());
+    let area_table = table_service.get_table_area().await;
+    match area_table {
+        Ok(areas) => Ok(areas),
+        Err(e) => Err(format!("获取区域错误{:?}", e)),
+    }
+}

+ 44 - 16
src-tauri/src/lib.rs

@@ -1,30 +1,58 @@
-use std::vec;
+mod commands;
+mod models;
+mod service;
+mod utils;
 
+use commands::{category, commodity, table};
+use sea_orm::{ConnectionTrait, DatabaseConnection};
+use std::fs::File;
+use std::sync::Arc;
+use tauri::State;
 use tauri::{AppHandle, Manager};
-use tauri_plugin_sql::{Migration, MigrationKind};
+
+type CmdState<'a> = State<'a, AppState>;
+type CmdResult<T> = Result<T, String>;
+
+#[derive(Default)]
+pub struct AppState {
+    pub conn: Arc<DatabaseConnection>,
+}
 
 #[cfg_attr(mobile, tauri::mobile_entry_point)]
 pub fn run() {
-    let migrations = vec![Migration {
-        version: 1,
-        description: "create_initial_tables",
-        sql: include_str!(concat!(env!("CARGO_MANIFEST_DIR"), "/resources/init.sql")),
-        kind: MigrationKind::Up,
-    }];
     tauri::Builder::default()
+        .setup(|app| {
+            tauri::async_runtime::block_on(async {
+                let app_dir = app.handle().path().app_data_dir()?;
+                std::fs::create_dir_all(&app_dir)?;
+                let db_path = app_dir.join("store.sqlite");
+                if !db_path.exists() {
+                    File::create(&db_path)?;
+                }
+                let db_url = format!("sqlite:{}", db_path.display());
+                let conn = sea_orm::Database::connect(&db_url)
+                    .await
+                    .map_err(|e| anyhow::anyhow!("Database connection error: {}", e))?;
+                conn.execute_unprepared("PRAGMA journal_mode=WAL;")
+                    .await
+                    .map_err(|e| anyhow::anyhow!("Failed to set WAL mode: {}", e))?;
+                utils::migration_db(app, &conn).await?;
+                app.manage(AppState {
+                    conn: Arc::new(conn),
+                });
+                Ok(())
+            })
+        })
+        .invoke_handler(tauri::generate_handler![
+            table::get_table_area,
+            category::get_category,
+            commodity::get_commodity
+        ])
         .plugin(tauri_plugin_os::init())
         .plugin(tauri_plugin_single_instance::init(|app, _args, _cwd| {
             let _ = show_window(app);
         }))
-        .plugin(tauri_plugin_upload::init())
-        .plugin(tauri_plugin_http::init())
         .plugin(tauri_plugin_store::Builder::new().build())
-        .plugin(
-            tauri_plugin_sql::Builder::default()
-                .add_migrations("sqlite:store.db", migrations)
-                .build(),
-        )
-        .plugin(tauri_plugin_opener::init())
         .run(tauri::generate_context!())
         .expect("error while running tauri application");
 }

+ 37 - 0
src-tauri/src/models/menu_cate.rs

@@ -0,0 +1,37 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::{Deserialize, Serialize};
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize, Deserialize)]
+#[sea_orm(table_name = "menu_cate")]
+pub struct Model {
+    #[sea_orm(primary_key)]
+    pub id: i32,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub name: Option<String>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub logo: Option<String>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub description: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub create_time: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub update_time: Option<String>,
+    pub sort: Option<i32>,
+    pub status: Option<i32>,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {
+    #[sea_orm(has_many = "super::menu_commodity::Entity")]
+    MenuCommodity,
+}
+
+impl Related<super::menu_commodity::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuCommodity.def()
+    }
+}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 66 - 0
src-tauri/src/models/menu_commodity.rs

@@ -0,0 +1,66 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::{Deserialize, Serialize};
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize, Deserialize)]
+#[sea_orm(table_name = "menu_commodity")]
+pub struct Model {
+    #[sea_orm(primary_key)]
+    pub id: i32,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub name: Option<String>,
+    pub cate_id: Option<i32>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub goods_no: Option<String>,
+    pub is_single_spec: Option<i32>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub logo: Option<String>,
+    pub price: Option<Decimal>,
+    pub line_price: Option<Decimal>,
+    pub stock: Option<i32>,
+    pub sort: Option<i32>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub description: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub create_time: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub update_time: Option<String>,
+    pub status: Option<i32>,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {
+    #[sea_orm(
+        belongs_to = "super::menu_cate::Entity",
+        from = "Column::CateId",
+        to = "super::menu_cate::Column::Id",
+        on_update = "NoAction",
+        on_delete = "NoAction"
+    )]
+    MenuCate,
+    #[sea_orm(has_many = "super::menu_sku::Entity")]
+    MenuSku,
+    #[sea_orm(has_many = "super::menu_spec::Entity")]
+    MenuSpec,
+}
+
+impl Related<super::menu_cate::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuCate.def()
+    }
+}
+
+impl Related<super::menu_sku::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuSku.def()
+    }
+}
+
+impl Related<super::menu_spec::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuSpec.def()
+    }
+}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 57 - 0
src-tauri/src/models/menu_sku.rs

@@ -0,0 +1,57 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::{Deserialize, Serialize};
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Deserialize, Serialize)]
+#[sea_orm(table_name = "menu_sku")]
+pub struct Model {
+    #[sea_orm(primary_key)]
+    pub id: i32,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub sku_no: Option<String>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub logo: Option<String>,
+    pub goods_id: i32,
+    pub stock: i32,
+    pub price: Decimal,
+    pub line_price: Decimal,
+    pub status: Option<i32>,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {
+    #[sea_orm(
+        belongs_to = "super::menu_commodity::Entity",
+        from = "Column::GoodsId",
+        to = "super::menu_commodity::Column::Id",
+        on_update = "NoAction",
+        on_delete = "NoAction"
+    )]
+    MenuCommodity,
+    #[sea_orm(has_many = "super::menu_sku_spec::Entity")]
+    MenuSkuSpec,
+}
+
+impl Related<super::menu_commodity::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuCommodity.def()
+    }
+}
+
+impl Related<super::menu_sku_spec::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuSkuSpec.def()
+    }
+}
+
+impl Related<super::menu_spec::Entity> for Entity {
+    fn to() -> RelationDef {
+        super::menu_sku_spec::Relation::MenuSpec.def()
+    }
+    fn via() -> Option<RelationDef> {
+        Some(super::menu_sku_spec::Relation::MenuSku.def().rev())
+    }
+}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 47 - 0
src-tauri/src/models/menu_sku_spec.rs

@@ -0,0 +1,47 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::{Deserialize, Serialize};
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize, Deserialize)]
+#[sea_orm(table_name = "menu_sku_spec")]
+pub struct Model {
+    #[sea_orm(primary_key, auto_increment = false)]
+    pub sku_id: i32,
+    #[sea_orm(primary_key, auto_increment = false)]
+    pub spec_id: i32,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {
+    #[sea_orm(
+        belongs_to = "super::menu_sku::Entity",
+        from = "Column::SkuId",
+        to = "super::menu_sku::Column::Id",
+        on_update = "NoAction",
+        on_delete = "NoAction"
+    )]
+    MenuSku,
+    #[sea_orm(
+        belongs_to = "super::menu_spec::Entity",
+        from = "Column::SpecId",
+        to = "super::menu_spec::Column::Id",
+        on_update = "NoAction",
+        on_delete = "NoAction"
+    )]
+    MenuSpec,
+}
+
+impl Related<super::menu_sku::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuSku.def()
+    }
+}
+
+impl Related<super::menu_spec::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuSpec.def()
+    }
+}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 54 - 0
src-tauri/src/models/menu_spec.rs

@@ -0,0 +1,54 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::Serialize;
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize)]
+#[sea_orm(table_name = "menu_spec")]
+pub struct Model {
+    #[sea_orm(primary_key)]
+    pub id: i32,
+    pub goods_id: i32,
+    #[sea_orm(column_type = "Text")]
+    pub name: String,
+    #[sea_orm(column_type = "Text")]
+    pub value: String,
+    pub status: Option<i32>,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {
+    #[sea_orm(
+        belongs_to = "super::menu_commodity::Entity",
+        from = "Column::GoodsId",
+        to = "super::menu_commodity::Column::Id",
+        on_update = "NoAction",
+        on_delete = "NoAction"
+    )]
+    MenuCommodity,
+    #[sea_orm(has_many = "super::menu_sku_spec::Entity")]
+    MenuSkuSpec,
+}
+
+impl Related<super::menu_commodity::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuCommodity.def()
+    }
+}
+
+impl Related<super::menu_sku_spec::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::MenuSkuSpec.def()
+    }
+}
+
+impl Related<super::menu_sku::Entity> for Entity {
+    fn to() -> RelationDef {
+        super::menu_sku_spec::Relation::MenuSku.def()
+    }
+    fn via() -> Option<RelationDef> {
+        Some(super::menu_sku_spec::Relation::MenuSpec.def().rev())
+    }
+}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 13 - 0
src-tauri/src/models/mod.rs

@@ -0,0 +1,13 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+pub mod prelude;
+
+pub mod menu_cate;
+pub mod menu_commodity;
+pub mod menu_sku;
+pub mod menu_sku_spec;
+pub mod menu_spec;
+pub mod store_area;
+pub mod store_material;
+pub mod store_order;
+pub mod store_table;

+ 11 - 0
src-tauri/src/models/prelude.rs

@@ -0,0 +1,11 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+pub use super::menu_cate::Entity as MenuCate;
+pub use super::menu_commodity::Entity as MenuCommodity;
+// pub use super::menu_sku::Entity as MenuSku;
+// pub use super::menu_sku_spec::Entity as MenuSkuSpec;
+// pub use super::menu_spec::Entity as MenuSpec;
+pub use super::store_area::Entity as StoreArea;
+// pub use super::store_material::Entity as StoreMaterial;
+// pub use super::store_order::Entity as StoreOrder;
+pub use super::store_table::Entity as StoreTable;

+ 34 - 0
src-tauri/src/models/store_area.rs

@@ -0,0 +1,34 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::{Deserialize, Serialize};
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize, Deserialize)]
+#[sea_orm(table_name = "store_area")]
+pub struct Model {
+    #[sea_orm(primary_key)]
+    pub id: i32,
+    #[sea_orm(column_type = "Text")]
+    pub name: String,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub description: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub create_time: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub update_time: Option<String>,
+    pub status: Option<i32>,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {
+    #[sea_orm(has_many = "super::store_table::Entity")]
+    StoreTable,
+}
+
+impl Related<super::store_table::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::StoreTable.def()
+    }
+}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 36 - 0
src-tauri/src/models/store_material.rs

@@ -0,0 +1,36 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::{Deserialize, Serialize};
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize, Deserialize)]
+#[sea_orm(table_name = "store_material")]
+pub struct Model {
+    #[sea_orm(primary_key)]
+    pub id: i32,
+    #[sea_orm(column_type = "Text")]
+    pub material_no: String,
+    #[sea_orm(column_type = "Text")]
+    pub material_type: String,
+    #[sea_orm(column_type = "Text")]
+    pub name: String,
+    #[sea_orm(column_type = "Text")]
+    pub specification: String,
+    pub inventory: i32,
+    pub shelf_life: i32,
+    #[sea_orm(column_type = "Text")]
+    pub unit: String,
+    #[sea_orm(column_type = "Text")]
+    pub img: String,
+    pub purchase_period: Option<i32>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub create_time: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub update_time: Option<String>,
+    pub status: Option<i32>,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 40 - 0
src-tauri/src/models/store_order.rs

@@ -0,0 +1,40 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::{Deserialize, Serialize};
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize, Deserialize)]
+#[sea_orm(table_name = "store_order")]
+pub struct Model {
+    #[sea_orm(primary_key)]
+    pub id: i32,
+    #[sea_orm(column_type = "Text")]
+    pub order_sn: String,
+    pub table_id: i32,
+    pub pay_type: i32,
+    pub amount: Option<Decimal>,
+    pub pay_amount: Option<Decimal>,
+    pub discount: Option<Decimal>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub param: Option<String>,
+    pub service_fee: Option<Decimal>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub remark: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub create_time: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub update_time: Option<String>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub status: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub pay_time: Option<String>,
+    pub pay_status: Option<i32>,
+    pub settle_status: Option<i32>,
+    #[sea_orm(column_type = "Text", nullable)]
+    pub goods: Option<String>,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 40 - 0
src-tauri/src/models/store_table.rs

@@ -0,0 +1,40 @@
+//! `SeaORM` Entity, @generated by sea-orm-codegen 1.1.0
+
+use sea_orm::entity::prelude::*;
+use serde::{Deserialize, Serialize};
+
+#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize, Deserialize)]
+#[sea_orm(table_name = "store_table")]
+pub struct Model {
+    #[sea_orm(primary_key)]
+    pub id: i32,
+    pub area_id: i32,
+    #[sea_orm(column_type = "Text")]
+    pub table_no: String,
+    pub capacity: Option<i32>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub create_time: Option<String>,
+    #[sea_orm(column_type = "custom(\"DATETIME\")", nullable)]
+    pub update_time: Option<String>,
+    pub status: Option<i32>,
+}
+
+#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
+pub enum Relation {
+    #[sea_orm(
+        belongs_to = "super::store_area::Entity",
+        from = "Column::AreaId",
+        to = "super::store_area::Column::Id",
+        on_update = "NoAction",
+        on_delete = "NoAction"
+    )]
+    StoreArea,
+}
+
+impl Related<super::store_area::Entity> for Entity {
+    fn to() -> RelationDef {
+        Relation::StoreArea.def()
+    }
+}
+
+impl ActiveModelBehavior for ActiveModel {}

+ 39 - 0
src-tauri/src/service/category.rs

@@ -0,0 +1,39 @@
+use std::sync::Arc;
+
+use sea_orm::{DatabaseConnection, EntityTrait};
+use serde::Serialize;
+
+use crate::models::{
+    menu_cate,
+    menu_commodity::{self},
+    prelude::{MenuCate, MenuCommodity},
+};
+
+pub struct Category {
+    conn: Arc<DatabaseConnection>,
+}
+
+#[derive(Debug, Serialize)]
+pub struct CategoryWithSpu {
+    #[serde(flatten)]
+    pub category: menu_cate::Model,
+    pub goods: Vec<menu_commodity::Model>,
+}
+
+impl Category {
+    pub fn new(conn: Arc<DatabaseConnection>) -> Self {
+        return Self { conn };
+    }
+    pub async fn get_category_commodity(&self) -> anyhow::Result<Vec<CategoryWithSpu>> {
+        let categories_with_goods: Vec<(menu_cate::Model, Vec<menu_commodity::Model>)> =
+            MenuCate::find()
+                .find_with_related(MenuCommodity)
+                .all(&*self.conn)
+                .await?;
+
+        Ok(categories_with_goods
+            .into_iter()
+            .map(|(category, goods)| CategoryWithSpu { category, goods })
+            .collect())
+    }
+}

+ 2 - 0
src-tauri/src/service/mod.rs

@@ -0,0 +1,2 @@
+pub mod category;
+pub mod table;

+ 63 - 0
src-tauri/src/service/table.rs

@@ -0,0 +1,63 @@
+use std::sync::Arc;
+
+use anyhow::Ok;
+use sea_orm::{DatabaseConnection, EntityTrait, QuerySelect};
+use serde::{Deserialize, Serialize};
+
+use crate::models::{
+    prelude::{StoreArea, StoreTable},
+    store_area, store_table,
+};
+
+pub struct Table {
+    conn: Arc<DatabaseConnection>,
+}
+
+#[derive(Deserialize, Serialize)]
+struct QueryTable {
+    id: i32,
+    table_no: String,
+    capacity: Option<i32>,
+}
+
+#[derive(Deserialize, Serialize)]
+pub struct AreaWithTable {
+    id: i32,
+    name: String,
+    description: Option<String>,
+    tables: Vec<QueryTable>,
+}
+
+impl Table {
+    pub fn new(conn: Arc<DatabaseConnection>) -> Self {
+        return Self { conn };
+    }
+    pub async fn get_table_area(&self) -> anyhow::Result<Vec<AreaWithTable>> {
+        let area_with_table = StoreArea::find()
+            .find_with_related(StoreTable)
+            .column(store_area::Column::Id)
+            .column(store_area::Column::Name)
+            .column(store_area::Column::Description)
+            .column(store_table::Column::Id)
+            .column(store_table::Column::TableNo)
+            .column(store_table::Column::Capacity)
+            .all(&*self.conn)
+            .await?;
+        Ok(area_with_table
+            .into_iter()
+            .map(|(area, tables)| AreaWithTable {
+                id: area.id,
+                name: area.name,
+                description: area.description,
+                tables: tables
+                    .iter()
+                    .map(|table| QueryTable {
+                        id: table.id,
+                        table_no: table.table_no.clone(),
+                        capacity: table.capacity,
+                    })
+                    .collect(),
+            })
+            .collect())
+    }
+}

+ 51 - 0
src-tauri/src/utils.rs

@@ -0,0 +1,51 @@
+use std::fs;
+
+use sea_orm::{ConnectionTrait, DatabaseConnection, Statement, TransactionTrait};
+use tauri::{Manager, path::BaseDirectory};
+
+pub async fn migration_db(app: &tauri::App, conn: &DatabaseConnection) -> anyhow::Result<()> {
+    let sql_path = app
+        .path()
+        .resolve("resources/table_struct.sql", BaseDirectory::Resource)?;
+    conn.execute(Statement::from_string(
+        conn.get_database_backend(),
+        "CREATE TABLE IF NOT EXISTS seaql_migrations (
+            version VARCHAR(255) PRIMARY KEY,
+            applied_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
+        )"
+        .to_owned(),
+    ))
+    .await?;
+    let applied: Vec<String> = conn
+        .query_all(Statement::from_string(
+            conn.get_database_backend(),
+            "SELECT version FROM seaql_migrations".to_owned(),
+        ))
+        .await?
+        .into_iter()
+        .filter_map(|row| row.try_get("", "version").ok())
+        .collect();
+
+    if let Some(sql_path_stem) = sql_path.file_stem() {
+        if let Some(version) = sql_path_stem.to_str() {
+            if !applied.contains(&version.to_string()) {
+                let sql_str = fs::read_to_string(&sql_path)?;
+                let tx = conn.begin().await?;
+                tx.execute(Statement::from_string(conn.get_database_backend(), sql_str))
+                    .await?;
+                tx.execute(Statement::from_string(
+                    conn.get_database_backend(),
+                    format!(
+                        "INSERT INTO seaql_migrations (version) VALUES ('{}')",
+                        version
+                    ),
+                ))
+                .await?;
+                tx.commit().await?;
+                println!("✅ Applied migration: {}", version);
+            }
+        }
+    }
+
+    Ok(())
+}

+ 21 - 17
src/components/noProduct.vue

@@ -5,21 +5,25 @@
   </div>
 </template>
 <script setup>
-import { useI18n } from 'vue-i18n'
-import { defineProps, watch } from 'vue'
-const { t } = useI18n()
-const props = defineProps({
-  img: {
-    type: String,
-    default: '@/assets/imgs/noFoods.svg',
-  },
-  text: {
-    type: String,
-    default: '还没添加菜品哦',
-  },
-})
-console.log(props)
-watch(()=>props.text, (newVal, oldVal) => {
-  console.log('text changed', newVal, oldVal)
-})
+  import { useI18n } from 'vue-i18n'
+  import { watch } from 'vue'
+
+  const { t } = useI18n()
+  const props = defineProps({
+    img: {
+      type: String,
+      default: '@/assets/imgs/noFoods.svg',
+    },
+    text: {
+      type: String,
+      default: '还没添加菜品哦',
+    },
+  })
+  console.log(props)
+  watch(
+    () => props.text,
+    (newVal, oldVal) => {
+      console.log('text changed', newVal, oldVal)
+    },
+  )
 </script>

+ 0 - 11
src/config/db.ts

@@ -1,11 +0,0 @@
-import TauriDatabase from '@tauri-apps/plugin-sql'
-import { Kysely, CamelCasePlugin } from 'kysely'
-import { TauriDialect } from './kysely'
-import { Database } from '@/model'
-
-export const DB = new Kysely<Database>({
-  dialect: new TauriDialect({
-    database: await TauriDatabase.load('sqlite:store.db'),
-  }),
-  plugins: [new CamelCasePlugin()],
-})

+ 0 - 126
src/config/kysely.ts

@@ -1,126 +0,0 @@
-import {
-  CompiledQuery,
-  DatabaseConnection,
-  DatabaseIntrospector,
-  Dialect,
-  Driver,
-  Kysely,
-  SqliteAdapter,
-  SqliteIntrospector,
-  SqliteQueryCompiler,
-  QueryCompiler,
-  QueryResult,
-} from 'kysely'
-import Database from '@tauri-apps/plugin-sql'
-
-export interface TauriDialectConfig {
-  database: Database
-}
-
-export class TauriDialect implements Dialect {
-  #config: TauriDialectConfig
-
-  constructor(config: TauriDialectConfig) {
-    this.#config = config
-  }
-
-  createAdapter() {
-    return new SqliteAdapter()
-  }
-
-  createDriver(): Driver {
-    return new D1Driver(this.#config)
-  }
-
-  createQueryCompiler(): QueryCompiler {
-    return new SqliteQueryCompiler()
-  }
-
-  createIntrospector(db: Kysely<any>): DatabaseIntrospector {
-    return new SqliteIntrospector(db)
-  }
-}
-
-class D1Driver implements Driver {
-  #config: TauriDialectConfig
-
-  constructor(config: TauriDialectConfig) {
-    this.#config = config
-  }
-
-  async init(): Promise<void> {}
-
-  async acquireConnection(): Promise<DatabaseConnection> {
-    return new TauriConnection(this.#config)
-  }
-
-  async beginTransaction(conn: TauriConnection): Promise<void> {
-    return await conn.beginTransaction()
-  }
-
-  async commitTransaction(conn: TauriConnection): Promise<void> {
-    return await conn.commitTransaction()
-  }
-
-  async rollbackTransaction(conn: TauriConnection): Promise<void> {
-    return await conn.rollbackTransaction()
-  }
-
-  async releaseConnection(_conn: TauriConnection): Promise<void> {}
-
-  async destroy(): Promise<void> {}
-}
-
-class TauriConnection implements DatabaseConnection {
-  #config: TauriDialectConfig
-  //   #transactionClient?: D1Connection
-
-  constructor(config: TauriDialectConfig) {
-    this.#config = config
-  }
-
-  async executeQuery<O>(compiledQuery: CompiledQuery): Promise<QueryResult<O>> {
-    if (compiledQuery.sql.startsWith('select')) {
-      const results = await this.#config.database.select<O[]>(compiledQuery.sql, [
-        ...compiledQuery.parameters,
-      ])
-      return {
-        rows: results,
-      }
-    }
-    const results = await this.#config.database.execute(compiledQuery.sql, [
-      ...compiledQuery.parameters,
-    ])
-
-    const numAffectedRows = results.rowsAffected > 0 ? BigInt(results.rowsAffected) : undefined
-
-    return {
-      insertId:
-        results.lastInsertId === undefined || results.lastInsertId === null
-          ? undefined
-          : BigInt(results.lastInsertId),
-      rows: [],
-      numAffectedRows,
-      numUpdatedOrDeletedRows: numAffectedRows,
-    }
-  }
-
-  async beginTransaction() {
-    throw new Error('Transactions are not supported yet.')
-  }
-
-  async commitTransaction() {
-    throw new Error('Transactions are not supported yet.')
-  }
-
-  async rollbackTransaction() {
-    throw new Error('Transactions are not supported yet.')
-  }
-
-  async *streamQuery<O>(
-    _compiledQuery: CompiledQuery,
-    _chunkSize: number,
-  ): AsyncIterableIterator<QueryResult<O>> {
-    throw new Error('D1 Driver does not support streaming')
-  }
-}

+ 13 - 2
src/hook/i18n.ts

@@ -1,12 +1,23 @@
 import { reactive, ref, toRefs, watchEffect } from 'vue'
 import { useI18n, UseI18nOptions } from 'vue-i18n'
 
+type OptionalLastN<
+  T extends any[],
+  N extends number,
+  Acc extends any[] = [],
+> = T['length'] extends N
+  ? [...T, ...Acc]
+  : T extends [...infer Rest, infer Last]
+  ? OptionalLastN<Rest, N, [Last?, ...Acc]>
+  : Acc
+
 export function useStoreI18n(options?: UseI18nOptions) {
   const { t, locale } = useI18n(options)
+
   const keyMap = reactive<Record<string, string>>({})
-  function $t(key: string) {
+  function $t(...[key, ...options]: OptionalLastN<Parameters<typeof t>, 1>) {
     watchEffect(() => {
-      if (locale.value) keyMap[key] = t(key)
+      if (locale.value) keyMap[key] = t(key, ...options)
     })
     return toRefs(keyMap)[key]
   }

+ 4 - 7
src/model/menu_cate.ts → src/interface/MenuCate.ts

@@ -1,8 +1,8 @@
-import { Generated, Insertable, Selectable, Updateable } from 'kysely'
+import { MenuProductInfo } from './MenuProduct'
 
 // 菜单分类表
-export interface MenuCateTable {
-  id: Generated<number> // 主键
+export interface MenuCategory {
+  id: number // 主键
   name: string // 分类名称
   logo: string // 分类Logo
   localLogo: string // 本地Logo路径
@@ -11,8 +11,5 @@ export interface MenuCateTable {
   updateTime?: string // 更新时间
   sort: number // 排序
   status: 'A' | 'D' // 状态 (A-Active, D-Deleted)
+  products: MenuProductInfo[]
 }
-
-export type MenuCate = Selectable<MenuCateTable>
-export type NewMenuCate = Insertable<MenuCateTable>
-export type MenuCateUpdate = Updateable<MenuCateTable>

+ 6 - 7
src/model/menu_commodity.ts → src/interface/MenuProduct.ts

@@ -1,8 +1,9 @@
-import { Generated, Insertable, Selectable, Updateable } from 'kysely'
+import { ProductSku } from './ProductSku'
+import { ProductSpecification } from './ProductSpec'
 
 // 商品表
-export interface MenuCommodityTable {
-  id: Generated<number> // 主键
+export interface MenuProductInfo {
+  id: number // 主键
   type: string // 商品类型
   merchantId: number // 商户ID
   storeId: number // 店铺ID
@@ -28,8 +29,6 @@ export interface MenuCommodityTable {
   updateTime?: string // 更新时间
   operator?: string // 操作员
   status: 'A' | 'D' // 状态 (A-Active, D-Deleted)
+  skus: ProductSku[]
+  specs: ProductSpecification[]
 }
-
-export type MenuCommodity = Selectable<MenuCommodityTable>
-export type NewMenuCommodity = Insertable<MenuCommodityTable>
-export type MenuCommodityUpdate = Updateable<MenuCommodityTable>

+ 15 - 0
src/interface/ProductSku.ts

@@ -0,0 +1,15 @@
+import { ProductSpecification } from './ProductSpec'
+
+// SKU表
+export interface ProductSku {
+  id: number // 主键
+  skuNo: string // SKU编号
+  logo: string // SKU Logo
+  goodsId: number // 商品ID
+  stock: number // 库存
+  price: number // 价格
+  linePrice: number // 划线价
+  weight: number // 重量
+  status: 'A' | 'D' // 状态 (A-Active, D-Deleted)
+  specs: ProductSpecification[]
+}

+ 8 - 0
src/interface/ProductSpec.ts

@@ -0,0 +1,8 @@
+// 规格表
+export interface ProductSpecification {
+  id: number // 主键
+  goodsId: number // 商品ID
+  name: string // 规格名称
+  value: string // 规格值
+  status: 'A' | 'D' // 状态 (A-Active, D-Deleted)
+}

+ 12 - 0
src/interface/StoreTable.ts

@@ -0,0 +1,12 @@
+export interface Area {
+  id: number
+  name: string
+  description: string
+  tables: Table[]
+}
+
+export interface Table {
+  id: number
+  tableNo: string
+  capacity: number
+}

+ 0 - 12
src/model/index.ts

@@ -1,12 +0,0 @@
-import { MenuCateTable } from './menu_cate'
-import { MenuCommodityTable } from './menu_commodity'
-import { MenuSkuTable } from './menu_sku'
-import { MenuSkuSpec, MenuSpecTable } from './menu_spec'
-
-export interface Database {
-  menuCate: MenuCateTable
-  menuCommodity: MenuCommodityTable
-  menuSku: MenuSkuTable
-  menuSpec: MenuSpecTable
-  menuSkuSpec: MenuSkuSpec
-}

+ 0 - 18
src/model/menu_sku.ts

@@ -1,18 +0,0 @@
-import { Generated, Insertable, Selectable, Updateable } from 'kysely'
-
-// SKU表
-export interface MenuSkuTable {
-  id: Generated<number> // 主键
-  skuNo: string // SKU编号
-  logo: string // SKU Logo
-  goodsId: number // 商品ID
-  stock: number // 库存
-  price: number // 价格
-  linePrice: number // 划线价
-  weight: number // 重量
-  status: 'A' | 'D' // 状态 (A-Active, D-Deleted)
-}
-
-export type MenuSku = Selectable<MenuSkuTable>
-export type NewMenuSku = Insertable<MenuSkuTable>
-export type MenuSkuUpdate = Updateable<MenuSkuTable>

+ 0 - 24
src/model/menu_spec.ts

@@ -1,24 +0,0 @@
-import { Generated, Insertable, Selectable, Updateable } from 'kysely'
-
-// 规格表
-export interface MenuSpecTable {
-  id: Generated<number> // 主键
-  goodsId: number // 商品ID
-  name: string // 规格名称
-  value: string // 规格值
-  status: 'A' | 'D' // 状态 (A-Active, D-Deleted)
-}
-
-export type MenuSpec = Selectable<MenuSpecTable>
-export type NewMenuSpec = Insertable<MenuSpecTable>
-export type MenuSpecUpdate = Updateable<MenuSpecTable>
-
-// SKU和规格的中间表
-export interface MenuSkuSpecTable {
-  skuId: number // SKU ID
-  specId: number // 规格ID
-}
-
-export type MenuSkuSpec = Selectable<MenuSkuSpecTable>
-export type NewMenuSkuSpec = Insertable<MenuSkuSpecTable>
-export type MenuSkuSpecUpdate = Updateable<MenuSkuSpecTable>

+ 10 - 0
src/service/CategoryService.ts

@@ -0,0 +1,10 @@
+import { MenuCategory } from '@/interface/MenuCate'
+import { invoke } from '@tauri-apps/api/core'
+
+export class CategoryService {
+  static async getAll() {
+    const categories = await invoke<MenuCategory>('get_category')
+    console.log('categories: ', categories)
+    return categories
+  }
+}

+ 8 - 0
src/service/TableService.ts

@@ -0,0 +1,8 @@
+import { Area } from '@/interface/StoreTable'
+import { invoke } from '@tauri-apps/api/core'
+
+export class TableService {
+  static getAreaTables() {
+    return invoke<Area>('get_table_area')
+  }
+}

+ 2 - 9
src/store/user.ts

@@ -1,6 +1,4 @@
 import { LoginForm } from '@/interface/user'
-import router from '@/router'
-import { request } from '@/utils/request'
 import { defineStore } from 'pinia'
 
 export const useUserStore = defineStore('user', {
@@ -9,13 +7,8 @@ export const useUserStore = defineStore('user', {
   }),
   actions: {
     async login(data: LoginForm) {
-      const { token } = await request<{ token: string }>({
-        url: '/backendApi/login/cashRegisterDoLogin',
-        method: 'POST',
-        data,
-        noAuth: true,
-      })
-      this.token = token
+      console.log('data: ', data)
+      this.token = 'token'
     },
   },
   getters: {

+ 0 - 66
src/utils/request.ts

@@ -1,66 +0,0 @@
-import { useUserStore } from '@/store/user'
-import { fetch } from '@tauri-apps/plugin-http'
-
-type CommonOption = {
-  url: string
-  query?: Record<string, string>
-  noAuth?: boolean
-  headers?: Record<string, string>
-}
-
-type GetOption = {
-  method: 'GET' | 'DELETE'
-}
-
-type PostOption = {
-  method: 'POST' | 'PUT'
-  data?: unknown
-}
-
-type RequestOptions = CommonOption & (GetOption | PostOption)
-
-type Response<T> = {
-  code: number
-  data: T
-  msg: string
-}
-
-export const request = async <T = void>(config: RequestOptions) => {
-  const { noAuth, url, headers, method, query } = config
-  if (!noAuth) {
-    const { getToken } = useUserStore()
-    headers['Access-Token'] = getToken
-  }
-  const reqPath = new URL(url, import.meta.env.VITE_BASE_URL || '')
-  if (query) Object.keys(query).forEach((key) => reqPath.searchParams.append(key, query[key]))
-
-  try {
-    console.log('reqPath.href: ', reqPath.href)
-    let body = null
-    if (config.method === 'POST' || (config.method === 'PUT' && config.data)) {
-      body = JSON.stringify(config.data)
-    }
-
-    const response = await fetch(reqPath.href, {
-      method,
-      headers: {
-        'Content-Type': 'application/json',
-        ...headers,
-      },
-      body,
-    })
-
-    if (!response.ok) {
-      throw new Error(`HTTP error! status: ${response.status}`)
-    }
-
-    const { code, data, msg } = (await response.json()) as Response<T>
-    if (code !== 200) {
-      throw new Error(msg)
-    }
-    return data
-  } catch (error) {
-    console.error('Request failed', error)
-    throw error
-  }
-}

+ 184 - 161
src/views/menu/index.vue

@@ -4,24 +4,37 @@
       <div class="w-full h-full overflow-y-auto relative hide-bar">
         <div class="sticky top-0 w-full bg-[#fff] p-[10px]">
           <div class="h-[60px] flex justify-between items-center">
-            <div class="flex-1 h-full pr-[10px] flex items-center justify-between cursor-pointer overflow-hidden">
+            <div
+              class="flex-1 h-full pr-[10px] flex items-center justify-between cursor-pointer overflow-hidden"
+            >
               <div
-              @click="handleArrow(-1)"
-                class="text-[#999] shrink-0 flex items-center justify-center h-[34px] w-[44px] border border-[#e6e6e6] rounded-[12px]">
+                @click="handleArrow(-1)"
+                class="text-[#999] shrink-0 flex items-center justify-center h-[34px] w-[44px] border border-[#e6e6e6] rounded-[12px]"
+              >
                 <el-icon>
                   <ArrowLeft />
                 </el-icon>
               </div>
-              <div ref="scrollRef" class="overflow-x-auto bg-[#fff]" style="width: calc(100% - 108px)">
+              <div
+                ref="scrollRef"
+                class="overflow-x-auto bg-[#fff]"
+                style="width: calc(100% - 108px)"
+              >
                 <div ref="" class="flex flex-nowrap items-center">
-                  <div @click="selectMenu(index)" v-for="(item,index) in products" class="flex items-center p-2 shrink-0 mr-[20px] h-[34px]" :class="[currentIndex == index ? 'active' : '' ]">
+                  <div
+                    @click="selectMenu(index)"
+                    v-for="(item, index) in products"
+                    class="flex items-center p-2 shrink-0 mr-[20px] h-[34px]"
+                    :class="[currentIndex == index ? 'active' : '']"
+                  >
                     {{ item.name }}
                   </div>
                 </div>
               </div>
               <div
-              @click="handleArrow(1)"
-                class="text-[#999] shrink-0 flex items-center justify-center h-[34px] w-[44px] border border-[#e6e6e6] rounded-[12px]">
+                @click="handleArrow(1)"
+                class="text-[#999] shrink-0 flex items-center justify-center h-[34px] w-[44px] border border-[#e6e6e6] rounded-[12px]"
+              >
                 <el-icon>
                   <ArrowRight />
                 </el-icon>
@@ -29,14 +42,25 @@
             </div>
           </div>
           <div class="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-6">
-            <div v-for="(product,index) in products" :key="product.id" @click="show = true"
-              class="bg-white rounded-xl shadow-sm transition-transform" >
-              <div class="relative flex items-center bg-[#fef8f4] justify-center mb-4 rounded overflow-hidden">
-                <img :src="product.image" :alt="product.name" class="w-25 h-25 my-1 object-cover rounded-full" />
+            <div
+              v-for="(product, index) in products"
+              :key="product.id"
+              @click="show = true"
+              class="bg-white rounded-xl shadow-sm transition-transform"
+            >
+              <div
+                class="relative flex items-center bg-[#fef8f4] justify-center mb-4 rounded overflow-hidden"
+              >
+                <img
+                  :src="product.image"
+                  :alt="product.name"
+                  class="w-25 h-25 my-1 object-cover rounded-full"
+                />
               </div>
               <div class="text-left px-2">
                 <h3
-                  class="text-gray-800 font-medium text-sm mb-2 text-container w-full text-nowrap line-clamp-2 sm:line-clamp-1 md:line-clamp-2 lg:line-clamp-3">
+                  class="text-gray-800 font-medium text-sm mb-2 text-container w-full text-nowrap line-clamp-2 sm:line-clamp-1 md:line-clamp-2 lg:line-clamp-3"
+                >
                   {{ product.name }}
                 </h3>
                 <p class="text-orange-500 font-semibold">${{ product.price.toFixed(2) }}</p>
@@ -52,162 +76,161 @@
     <OrderDrawer v-model:show="show" />
     <deskDrawer v-model:show="deskShow" />
     <payDrawer v-model:show="payShow" />
-
   </div>
 </template>
-<script  setup>
-import { ref, onMounted, nextTick } from 'vue'
-import { useI18n } from 'vue-i18n'
-import { ArrowRight, ArrowLeft } from '@element-plus/icons-vue'
-import rightOrder from './components/rightOrder/rightOrder.vue'
-import OrderDrawer from './components/order-drawer.vue'
-import deskDrawer from './components/desk-drawer.vue'
-import payDrawer from './components/pay-drawer.vue'
-const { t } = useI18n()
-
-const show = ref(false)
-
-onMounted(() => { })
-
-const products = ref([
-  {
-    id: 1,
-    name: 'Spicy Seasoned Seafood Noodles',
-    price: 2.29,
-    image: 'https://ai-public.mastergo.com/ai/img_res/e53b11a05b7778db2bc50338e47047c4.jpg',
-  },
-  {
-    id: 2,
-    name: 'Grilled Salmon Bowl',
-    price: 3.49,
-    image: 'https://ai-public.mastergo.com/ai/img_res/07e4619fc2f26fe9f6a0afaebc47dceb.jpg',
-  },
-  {
-    id: 3,
-    name: 'Teriyaki Chicken Rice',
-    price: 2.99,
-    image: 'https://ai-public.mastergo.com/ai/img_res/1709d2e0e3495f8089da24be4ad2d26a.jpg',
-  },
-  {
-    id: 4,
-    name: 'Vegetable Stir Fry Noodles',
-    price: 2.49,
-    image: 'https://ai-public.mastergo.com/ai/img_res/043e50f136a6ec7f5f37eb448cf298c7.jpg',
-  },
-  {
-    id: 5,
-    name: 'Spicy Tuna Poke Bowl',
-    price: 3.99,
-    image: 'https://ai-public.mastergo.com/ai/img_res/7f8ad399ad651c8de6896bc611be1936.jpg',
-  },
-  {
-    id: 6,
-    name: 'Beef Bulgogi Bowl',
-    price: 3.79,
-    image: 'https://ai-public.mastergo.com/ai/img_res/49b0d0fbf6654dcef422a7e3334c8b6a.jpg',
-  },
-  {
-    id: 7,
-    name: 'Shrimp Pad Thai',
-    price: 3.29,
-    image: 'https://ai-public.mastergo.com/ai/img_res/ca1fef96dffcfa401768314721606c8f.jpg',
-  },
-  {
-    id: 8,
-    name: 'Miso Ramen Bowl',
-    price: 2.89,
-    image: 'https://ai-public.mastergo.com/ai/img_res/210debe79bb22fec1944cab575f2759f.jpg',
-  },
-])
-
-// 挂单桌子
-const deskShow = ref(false)
-const catgroyClick = () => {
-  deskShow.value = true
-}
-
-const scrollRef = ref(null);
-const currentIndex = ref(0);
-let positions = [];
-
-const recalculatePositions = () => {
-  const flexContainer = scrollRef.value?.firstElementChild;
-  if (!flexContainer) return [];
-  
-  const items = Array.from(flexContainer.children);
-  let total = 0;
-  
-  return items.map((item, index) => {
-    const style = window.getComputedStyle(item);
-    const margin = index < items.length - 1 ? parseFloat(style.marginRight) : 0;
-    const start = total;
-    total += item.offsetWidth + margin;
-    return { start, end: total - margin };
-  });
-};
-
-const scrollToIndex = (index) => {
-  if (!scrollRef.value) return;
-  
-  const containerWidth = scrollRef.value.offsetWidth;
-  const target = positions[index];
-  
-  if (!target) return;
-  
-  const scrollPos = Math.max(0, Math.min(
-    target.start - (containerWidth - (target.end - target.start)) / 2,
-    scrollRef.value.scrollWidth - containerWidth
-  ));
-  
-  scrollRef.value.scrollTo({
-    left: scrollPos,
-    behavior: 'smooth'
-  });
-};
-
-const handleArrow = (direction) => {
-  positions = recalculatePositions();
-  currentIndex.value = Math.max(0, Math.min(
-    currentIndex.value + direction,
-    positions.length - 1
-  ));
-  scrollToIndex(currentIndex.value);
-};
-
-const selectMenu = (index) => {
-  currentIndex.value = index;
-  scrollToIndex(index);
-};
-
-//pay
-const payShow = ref(false)
-const onPay = ()=>{
-  payShow.value = true
-}
+<script setup>
+  import { ref, onMounted, nextTick } from 'vue'
+  import { useI18n } from 'vue-i18n'
+  import { ArrowRight, ArrowLeft } from '@element-plus/icons-vue'
+  import rightOrder from './components/rightOrder/rightOrder.vue'
+  import OrderDrawer from './components/order-drawer.vue'
+  import deskDrawer from './components/desk-drawer.vue'
+  import payDrawer from './components/pay-drawer.vue'
+  const { t } = useI18n()
+
+  const show = ref(false)
+
+  onMounted(() => {})
+
+  const products = ref([
+    {
+      id: 1,
+      name: 'Spicy Seasoned Seafood Noodles',
+      price: 2.29,
+      image: 'https://ai-public.mastergo.com/ai/img_res/e53b11a05b7778db2bc50338e47047c4.jpg',
+    },
+    {
+      id: 2,
+      name: 'Grilled Salmon Bowl',
+      price: 3.49,
+      image: 'https://ai-public.mastergo.com/ai/img_res/07e4619fc2f26fe9f6a0afaebc47dceb.jpg',
+    },
+    {
+      id: 3,
+      name: 'Teriyaki Chicken Rice',
+      price: 2.99,
+      image: 'https://ai-public.mastergo.com/ai/img_res/1709d2e0e3495f8089da24be4ad2d26a.jpg',
+    },
+    {
+      id: 4,
+      name: 'Vegetable Stir Fry Noodles',
+      price: 2.49,
+      image: 'https://ai-public.mastergo.com/ai/img_res/043e50f136a6ec7f5f37eb448cf298c7.jpg',
+    },
+    {
+      id: 5,
+      name: 'Spicy Tuna Poke Bowl',
+      price: 3.99,
+      image: 'https://ai-public.mastergo.com/ai/img_res/7f8ad399ad651c8de6896bc611be1936.jpg',
+    },
+    {
+      id: 6,
+      name: 'Beef Bulgogi Bowl',
+      price: 3.79,
+      image: 'https://ai-public.mastergo.com/ai/img_res/49b0d0fbf6654dcef422a7e3334c8b6a.jpg',
+    },
+    {
+      id: 7,
+      name: 'Shrimp Pad Thai',
+      price: 3.29,
+      image: 'https://ai-public.mastergo.com/ai/img_res/ca1fef96dffcfa401768314721606c8f.jpg',
+    },
+    {
+      id: 8,
+      name: 'Miso Ramen Bowl',
+      price: 2.89,
+      image: 'https://ai-public.mastergo.com/ai/img_res/210debe79bb22fec1944cab575f2759f.jpg',
+    },
+  ])
+
+  // 挂单桌子
+  const deskShow = ref(false)
+  const catgroyClick = () => {
+    deskShow.value = true
+  }
+
+  const scrollRef = ref(null)
+  const currentIndex = ref(0)
+  let positions = []
+
+  const recalculatePositions = () => {
+    const flexContainer = scrollRef.value?.firstElementChild
+    if (!flexContainer) return []
+
+    const items = Array.from(flexContainer.children)
+    let total = 0
+
+    return items.map((item, index) => {
+      const style = window.getComputedStyle(item)
+      const margin = index < items.length - 1 ? parseFloat(style.marginRight) : 0
+      const start = total
+      total += item.offsetWidth + margin
+      return { start, end: total - margin }
+    })
+  }
+
+  const scrollToIndex = (index) => {
+    if (!scrollRef.value) return
+
+    const containerWidth = scrollRef.value.offsetWidth
+    const target = positions[index]
+
+    if (!target) return
+
+    const scrollPos = Math.max(
+      0,
+      Math.min(
+        target.start - (containerWidth - (target.end - target.start)) / 2,
+        scrollRef.value.scrollWidth - containerWidth,
+      ),
+    )
+
+    scrollRef.value.scrollTo({
+      left: scrollPos,
+      behavior: 'smooth',
+    })
+  }
+
+  const handleArrow = (direction) => {
+    positions = recalculatePositions()
+    currentIndex.value = Math.max(0, Math.min(currentIndex.value + direction, positions.length - 1))
+    scrollToIndex(currentIndex.value)
+  }
+
+  const selectMenu = (index) => {
+    currentIndex.value = index
+    scrollToIndex(index)
+  }
+
+  //pay
+  const payShow = ref(false)
+  const onPay = () => {
+    payShow.value = true
+  }
 </script>
 <style scoped>
-.active {
-  background-color: #f67f20;
-  color:white;
-  border-radius: 5px;
-}
-::-webkit-scrollbar-thumb {
-  background-color: red;
-}
-
-/* 隐藏整个滚动条 */
-::-webkit-scrollbar {
-  display: none;
-}
+  .active {
+    background-color: #f67f20;
+    color: white;
+    border-radius: 5px;
+  }
+  ::-webkit-scrollbar-thumb {
+    background-color: red;
+  }
+
+  /* 隐藏整个滚动条 */
+  ::-webkit-scrollbar {
+    display: none;
+  }
 
-/* 或者仅隐藏滚动条的轨道,保留滑块可操作性,但视觉上不显示滚动条 */
-/* ::-webkit-scrollbar-track {
+  /* 或者仅隐藏滚动条的轨道,保留滑块可操作性,但视觉上不显示滚动条 */
+  /* ::-webkit-scrollbar-track {
   display: none;
 } */
 
-.text-container {
-  width: 100%;
-  overflow: hidden;
-  text-overflow: ellipsis;
+  .text-container {
+    width: 100%;
+    overflow: hidden;
+    text-overflow: ellipsis;
   }
 </style>

+ 4 - 48
src/views/test/index.vue

@@ -64,7 +64,6 @@
 
 <script setup lang="ts">
   import { ref } from 'vue'
-  import { DB } from '@/config/db'
 
   const jsonData = ref([])
   const newData = ref({
@@ -86,34 +85,9 @@
   })
   const showModal = ref(false)
 
-  const fetchData = () => {
-    // 查询数据
-    DB.selectFrom('menuCate')
-      .select(['id', 'name', 'logo', 'localLogo', 'sort', 'description', 'status'])
-      .execute()
-      .then((data) => {
-        console.log('data: ', data)
-        jsonData.value = data
-      })
-  }
+  const fetchData = () => {}
 
-  const addData = () => {
-    // 添加数据
-    DB.insertInto('menuCate')
-      .values(newData.value)
-      .execute()
-      .then(() => {
-        fetchData()
-        newData.value = {
-          name: '',
-          logo: '',
-          localLogo: '',
-          sort: 0,
-          description: '',
-          status: 'A',
-        }
-      })
-  }
+  const addData = () => {}
 
   const openEditModal = (item: any) => {
     editId.value = item.id
@@ -134,27 +108,9 @@
     }
   }
 
-  const updateData = () => {
-    // 更新数据
-    DB.updateTable('menuCate')
-      .set(editData.value)
-      .where('id', '=', editId.value)
-      .execute()
-      .then(() => {
-        fetchData()
-        closeModal()
-      })
-  }
+  const updateData = () => {}
 
-  const deleteData = (id: number) => {
-    // 删除数据
-    DB.deleteFrom('menuCate')
-      .where('id', '=', id)
-      .execute()
-      .then(() => {
-        fetchData()
-      })
-  }
+  const deleteData = (id: number) => {}
 
   fetchData()
 </script>

Some files were not shown because too many files changed in this diff