Browse Source

feat: 增加scribe文档,注意检查:Providers目录下的文件有无必要修改

Yin Bin 4 months ago
parent
commit
6f38743a07
53 changed files with 8630 additions and 89 deletions
  1. 4 0
      .scribe/.filehashes
  2. 7 0
      .scribe/auth.md
  3. 490 0
      .scribe/endpoints.cache/00.yaml
  4. 488 0
      .scribe/endpoints/00.yaml
  5. 53 0
      .scribe/endpoints/custom.0.yaml
  6. 12 0
      .scribe/intro.md
  7. 69 7
      ansible.cfg
  8. 139 13
      app/Http/Controllers/Client/AgentController.php
  9. 114 0
      app/Http/Controllers/Client/WalletController.php
  10. 1 3
      app/Models/MemberAddress.php
  11. 26 0
      app/Models/Order.php
  12. 38 0
      app/Providers/EventServiceProvider.php
  13. 40 0
      app/Providers/RouteServiceProvider.php
  14. 1 0
      app/Services/Client/AccountService.php
  15. 665 65
      composer.lock
  16. 99 0
      config/app.php
  17. 148 0
      config/scribe.php
  18. 77 0
      database/migrations/0001_01_01_000000_create_sessions_table.php
  19. 329 0
      public/docs/collection.json
  20. 393 0
      public/docs/css/theme-default.print.css
  21. 1090 0
      public/docs/css/theme-default.style.css
  22. BIN
      public/docs/images/navbar.png
  23. 1490 0
      public/docs/index.html
  24. 149 0
      public/docs/js/theme-default-4.38.0.js
  25. 289 0
      public/docs/js/tryitout-4.38.0.js
  26. 408 0
      public/docs/openapi.yaml
  27. 3 0
      resources/views/vendor/scribe/components/badges/auth.blade.php
  28. 1 0
      resources/views/vendor/scribe/components/badges/base.blade.php
  29. 5 0
      resources/views/vendor/scribe/components/badges/http-method.blade.php
  30. 75 0
      resources/views/vendor/scribe/components/field-details.blade.php
  31. 101 0
      resources/views/vendor/scribe/components/nested-fields.blade.php
  32. 34 0
      resources/views/vendor/scribe/external/elements.blade.php
  33. 23 0
      resources/views/vendor/scribe/external/rapidoc.blade.php
  34. 27 0
      resources/views/vendor/scribe/external/scalar.blade.php
  35. 12 0
      resources/views/vendor/scribe/markdown/auth.blade.php
  36. 13 0
      resources/views/vendor/scribe/markdown/intro.blade.php
  37. 34 0
      resources/views/vendor/scribe/partials/example-requests/bash.md.blade.php
  38. 62 0
      resources/views/vendor/scribe/partials/example-requests/javascript.md.blade.php
  39. 51 0
      resources/views/vendor/scribe/partials/example-requests/php.md.blade.php
  40. 52 0
      resources/views/vendor/scribe/partials/example-requests/python.md.blade.php
  41. 185 0
      resources/views/vendor/scribe/themes/default/endpoint.blade.php
  42. 21 0
      resources/views/vendor/scribe/themes/default/groups.blade.php
  43. 74 0
      resources/views/vendor/scribe/themes/default/index.blade.php
  44. 69 0
      resources/views/vendor/scribe/themes/default/sidebar.blade.php
  45. 63 0
      resources/views/vendor/scribe/themes/elements/components/field-details.blade.php
  46. 37 0
      resources/views/vendor/scribe/themes/elements/components/nested-fields.blade.php
  47. 263 0
      resources/views/vendor/scribe/themes/elements/endpoint.blade.php
  48. 28 0
      resources/views/vendor/scribe/themes/elements/groups.blade.php
  49. 360 0
      resources/views/vendor/scribe/themes/elements/index.blade.php
  50. 92 0
      resources/views/vendor/scribe/themes/elements/sidebar.blade.php
  51. 322 0
      resources/views/vendor/scribe/themes/elements/try_it_out.blade.php
  52. 4 1
      routes/api.php
  53. 0 0
      script/ansible/README.md

+ 4 - 0
.scribe/.filehashes

@@ -0,0 +1,4 @@
+# GENERATED. YOU SHOULDN'T MODIFY OR DELETE THIS FILE.
+# Scribe uses this file to know when you change something manually in your docs.
+.scribe/intro.md=1acf2f617f526e06067ddf1283e86f54
+.scribe/auth.md=281aef6ee09869b8a198579af7b0de3f

+ 7 - 0
.scribe/auth.md

@@ -0,0 +1,7 @@
+# Authenticating requests
+
+To authenticate requests, include an **`Authorization`** header with the value **`"Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"`**.
+
+All authenticated endpoints are marked with a `requires authentication` badge in the documentation below.
+
+通过登录接口获取 token,格式为 Bearer {token}

+ 490 - 0
.scribe/endpoints.cache/00.yaml

@@ -0,0 +1,490 @@
+## Autogenerated by Scribe. DO NOT MODIFY.
+
+name: Endpoints
+description: ''
+endpoints:
+  -
+    httpMethods:
+      - GET
+    uri: 'api/scribe/update-token/{mobile}'
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: 更新API文档的认证Token
+      description: 通过手机号获取验证码并登录,然后更新API文档的认证Token
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+    urlParameters:
+      mobile:
+        name: mobile
+        description: 手机号码.
+        required: true
+        example: '13800138000'
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanUrlParameters:
+      mobile: '13800138000'
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters: []
+    cleanBodyParameters: []
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "Token更新成功",
+            "data": {
+              "token": "your_new_token_here"
+            }
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/account/send-code
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 发送验证码'
+      description: 向指定手机号发送验证码
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      mobile:
+        name: mobile
+        description: 手机号码.
+        required: true
+        example: '13800138000'
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      mobile: '13800138000'
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "验证码发送成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/account/login
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 用户登录'
+      description: 使用手机号和验证码登录账户
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      mobile:
+        name: mobile
+        description: 手机号码.
+        required: true
+        example: '13800138000'
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+      code:
+        name: code
+        description: 验证码.
+        required: true
+        example: '123456'
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      mobile: '13800138000'
+      code: '123456'
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "登录成功",
+            "data": {
+              "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...",
+              "user": {
+                "id": 1,
+                "mobile": "13800138000",
+                "nickname": "用户昵称"
+              }
+            }
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/account/wx-login
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 微信登录'
+      description: 使用微信openid登录账户
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      openid:
+        name: openid
+        description: 微信openid.
+        required: true
+        example: wx_123456789
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      openid: wx_123456789
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "登录成功",
+            "data": {
+              "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...",
+              "user": {
+                "id": 1,
+                "openid": "wx_123456789",
+                "nickname": "微信昵称"
+              }
+            }
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/account/logout
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 用户退出'
+      description: 退出当前账户登录状态
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters: []
+    cleanBodyParameters: []
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "退出成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - DELETE
+    uri: api/account
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 用户注销'
+      description: 永久注销当前账户
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters: []
+    cleanBodyParameters: []
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "注销成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/user/withdraw
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[用户管理] 用户提现'
+      description: 提现用户的余额
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      amount:
+        name: amount
+        description: 提现金额.
+        required: false
+        example: '100.00'
+        type: decimal
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      amount: '100.00'
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "提现成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/user/feedback
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[用户管理] 用户反馈'
+      description: 提交用户的反馈信息
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      content:
+        name: content
+        description: 反馈内容.
+        required: false
+        example: 这是一个反馈信息
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      content: 这是一个反馈信息
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "提交成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/user/apply-coach
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[用户管理] 申请成为技师'
+      description: 申请成为技师
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters: []
+    cleanBodyParameters: []
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "申请成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []

+ 488 - 0
.scribe/endpoints/00.yaml

@@ -0,0 +1,488 @@
+name: Endpoints
+description: ''
+endpoints:
+  -
+    httpMethods:
+      - GET
+    uri: 'api/scribe/update-token/{mobile}'
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: 更新API文档的认证Token
+      description: 通过手机号获取验证码并登录,然后更新API文档的认证Token
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+    urlParameters:
+      mobile:
+        name: mobile
+        description: 手机号码.
+        required: true
+        example: '13800138000'
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanUrlParameters:
+      mobile: '13800138000'
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters: []
+    cleanBodyParameters: []
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "Token更新成功",
+            "data": {
+              "token": "your_new_token_here"
+            }
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/account/send-code
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 发送验证码'
+      description: 向指定手机号发送验证码
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      mobile:
+        name: mobile
+        description: 手机号码.
+        required: true
+        example: '13800138000'
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      mobile: '13800138000'
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "验证码发送成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/account/login
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 用户登录'
+      description: 使用手机号和验证码登录账户
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      mobile:
+        name: mobile
+        description: 手机号码.
+        required: true
+        example: '13800138000'
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+      code:
+        name: code
+        description: 验证码.
+        required: true
+        example: '123456'
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      mobile: '13800138000'
+      code: '123456'
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "登录成功",
+            "data": {
+              "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...",
+              "user": {
+                "id": 1,
+                "mobile": "13800138000",
+                "nickname": "用户昵称"
+              }
+            }
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/account/wx-login
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 微信登录'
+      description: 使用微信openid登录账户
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      openid:
+        name: openid
+        description: 微信openid.
+        required: true
+        example: wx_123456789
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      openid: wx_123456789
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "登录成功",
+            "data": {
+              "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...",
+              "user": {
+                "id": 1,
+                "openid": "wx_123456789",
+                "nickname": "微信昵称"
+              }
+            }
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/account/logout
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 用户退出'
+      description: 退出当前账户登录状态
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters: []
+    cleanBodyParameters: []
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "退出成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - DELETE
+    uri: api/account
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[账户管理] 用户注销'
+      description: 永久注销当前账户
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters: []
+    cleanBodyParameters: []
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "注销成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/user/withdraw
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[用户管理] 用户提现'
+      description: 提现用户的余额
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      amount:
+        name: amount
+        description: 提现金额.
+        required: false
+        example: '100.00'
+        type: decimal
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      amount: '100.00'
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "提现成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/user/feedback
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[用户管理] 用户反馈'
+      description: 提交用户的反馈信息
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+      Content-Type: application/json
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters:
+      content:
+        name: content
+        description: 反馈内容.
+        required: false
+        example: 这是一个反馈信息
+        type: string
+        enumValues: []
+        exampleWasSpecified: true
+        nullable: false
+        custom: []
+    cleanBodyParameters:
+      content: 这是一个反馈信息
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "提交成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []
+  -
+    httpMethods:
+      - POST
+    uri: api/user/apply-coach
+    metadata:
+      groupName: Endpoints
+      groupDescription: ''
+      subgroup: ''
+      subgroupDescription: ''
+      title: '[用户管理] 申请成为技师'
+      description: 申请成为技师
+      authenticated: true
+      custom: []
+    headers:
+      Authorization: 'Bearer 68|ynrtTr0JPkmVmV80Cd0o2PIapOZk8mA3xDWnpqt1b3045d50'
+    urlParameters: []
+    cleanUrlParameters: []
+    queryParameters: []
+    cleanQueryParameters: []
+    bodyParameters: []
+    cleanBodyParameters: []
+    fileParameters: []
+    responses:
+      -
+        status: 200
+        content: |-
+          {
+            "code": 200,
+            "message": "申请成功",
+            "data": null
+          }
+        headers: []
+        description: ''
+        custom: []
+    responseFields: []
+    auth:
+      - headers
+      - Authorization
+      - 'Bearer Bearer {YOUR_AUTH_KEY}'
+    controller: null
+    method: null
+    route: null
+    custom: []

+ 53 - 0
.scribe/endpoints/custom.0.yaml

@@ -0,0 +1,53 @@
+# To include an endpoint that isn't a part of your Laravel app (or belongs to a vendor package),
+# you can define it in a custom.*.yaml file, like this one.
+# Each custom file should contain an array of endpoints. Here's an example:
+# See https://scribe.knuckles.wtf/laravel/documenting/custom-endpoints#extra-sorting-groups-in-custom-endpoint-files for more options
+
+#- httpMethods:
+#    - POST
+#  uri: api/doSomething/{param}
+#  metadata:
+#    groupName: The group the endpoint belongs to. Can be a new group or an existing group.
+#    groupDescription: A description for the group. You don't need to set this for every endpoint; once is enough.
+#    subgroup: You can add a subgroup, too.
+#    title: Do something
+#    description: 'This endpoint allows you to do something.'
+#    authenticated: false
+#  headers:
+#    Content-Type: application/json
+#    Accept: application/json
+#  urlParameters:
+#    param:
+#      name: param
+#      description: A URL param for no reason.
+#      required: true
+#      example: 2
+#      type: integer
+#  queryParameters:
+#    speed:
+#      name: speed
+#      description: How fast the thing should be done. Can be `slow` or `fast`.
+#      required: false
+#      example: fast
+#      type: string
+#  bodyParameters:
+#    something:
+#      name: something
+#      description: The things we should do.
+#      required: true
+#      example:
+#        - string 1
+#        - string 2
+#      type: 'string[]'
+#  responses:
+#    - status: 200
+#      description: 'When the thing was done smoothly.'
+#      content: # Your response content can be an object, an array, a string or empty.
+#         {
+#           "hey": "ho ho ho"
+#         }
+#  responseFields:
+#    hey:
+#      name: hey
+#      description: Who knows?
+#      type: string # This is optional

+ 12 - 0
.scribe/intro.md

@@ -0,0 +1,12 @@
+# Introduction
+
+Owl Admin API 接口文档
+
+<aside>
+    <strong>Base URL</strong>: <code>http://192.168.110.10</code>
+</aside>
+
+欢迎使用 Owl Admin API 文档。
+
+本文档提供了所有 API 接口的详细信息,包括请求参数、响应格式等。
+

+ 69 - 7
ansible.cfg

@@ -6,6 +6,9 @@ use App\Http\Controllers\Controller;
 use App\Services\Client\AccountService;
 use Illuminate\Http\Request;
 
+/**
+ * 账户相关的API接口
+ */
 class AccountController extends Controller
 {
     protected AccountService $service;
@@ -16,7 +19,16 @@ class AccountController extends Controller
     }
 
     /**
-     * 发送验证码
+     * [账户管理] 发送验证码
+     * 
+     * 向指定手机号发送验证码
+     * 
+     * @bodyParam mobile string required 手机号码. Example: 13800138000
+     * @response {
+     *   "code": 200,
+     *   "message": "验证码发送成功",
+     *   "data": null
+     * }
      */
     public function sendVerifyCode(Request $request)
     {
@@ -25,7 +37,24 @@ class AccountController extends Controller
     }
 
     /**
-     * 用户登录
+     * [账户管理] 用户登录
+     * 
+     * 使用手机号和验证码登录账户
+     * 
+     * @bodyParam mobile string required 手机号码. Example: 13800138000
+     * @bodyParam code string required 验证码. Example: 123456
+     * @response {
+     *   "code": 200,
+     *   "message": "登录成功",
+     *   "data": {
+     *     "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...",
+     *     "user": {
+     *       "id": 1,
+     *       "mobile": "13800138000",
+     *       "nickname": "用户昵称"
+     *     }
+     *   }
+     * }
      */
     public function login(Request $request)
     {
@@ -35,7 +64,23 @@ class AccountController extends Controller
     }
 
     /**
-     * 微信登录
+     * [账户管理] 微信登录
+     * 
+     * 使用微信openid登录账户
+     * 
+     * @bodyParam openid string required 微信openid. Example: wx_123456789
+     * @response {
+     *   "code": 200,
+     *   "message": "登录成功",
+     *   "data": {
+     *     "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...",
+     *     "user": {
+     *       "id": 1,
+     *       "openid": "wx_123456789",
+     *       "nickname": "微信昵称"
+     *     }
+     *   }
+     * }
      */
     public function wxLogin(Request $request)
     {
@@ -44,7 +89,16 @@ class AccountController extends Controller
     }
 
     /**
-     * 用户退出
+     * [账户管理] 用户退出
+     * 
+     * 退出当前账户登录状态
+     * 
+     * @authenticated
+     * @response {
+     *   "code": 200,
+     *   "message": "退出成功",
+     *   "data": null
+     * }
      */
     public function logout()
     {
@@ -52,11 +106,19 @@ class AccountController extends Controller
     }
 
     /**
-     * 用户注销
+     * [账户管理] 用户注销
+     * 
+     * 永久注销当前账户
+     * 
+     * @authenticated
+     * @response {
+     *   "code": 200,
+     *   "message": "注销成功",
+     *   "data": null
+     * }
      */
     public function destroy()
     {
         return $this->service->deleteAccount();
     }
-
-} 
+}

+ 139 - 13
app/Http/Controllers/Client/AgentController.php

@@ -6,6 +6,9 @@ use App\Http\Controllers\Controller;
 use App\Services\Client\UserService;
 use Illuminate\Http\Request;
 
+/**
+ * 用户相关的API接口
+ */
 class UserController extends Controller
 {
     protected UserService $service;
@@ -16,7 +19,16 @@ class UserController extends Controller
     }
 
     /**
-     * 发送验证码
+     * [用户管理] 发送验证码
+     * 
+     * 向指定手机号发送验证码
+     * 
+     * @bodyParam mobile string required 手机号码. Example: 13800138000
+     * @response {
+     *   "code": 200,
+     *   "message": "验证码发送成功",
+     *   "data": null
+     * }
      */
     public function sendVerifyCode(Request $request)
     {
@@ -25,7 +37,24 @@ class UserController extends Controller
     }
 
     /**
-     * 用户登录
+     * [用户管理] 用户登录
+     * 
+     * 使用手机号和验证码登录
+     * 
+     * @bodyParam mobile string required 手机号码. Example: 13800138000
+     * @bodyParam code string required 验证码. Example: 123456
+     * @response {
+     *   "code": 200,
+     *   "message": "登录成功",
+     *   "data": {
+     *     "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...",
+     *     "user": {
+     *       "id": 1,
+     *       "mobile": "13800138000",
+     *       "nickname": "用户昵称"
+     *     }
+     *   }
+     * }
      */
     public function login(Request $request)
     {
@@ -35,7 +64,23 @@ class UserController extends Controller
     }
 
     /**
-     * 微信登录
+     * [用户管理] 微信登录
+     * 
+     * 使用微信openid登录
+     * 
+     * @bodyParam openid string required 微信openid. Example: wx_123456789
+     * @response {
+     *   "code": 200,
+     *   "message": "登录成功",
+     *   "data": {
+     *     "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...",
+     *     "user": {
+     *       "id": 1,
+     *       "openid": "wx_123456789",
+     *       "nickname": "微信昵称"
+     *     }
+     *   }
+     * }
      */
     public function wxLogin(Request $request)
     {
@@ -44,7 +89,16 @@ class UserController extends Controller
     }
 
     /**
-     * 用户退出
+     * [用户管理] 用户退出
+     * 
+     * 退出登录,清除token
+     * 
+     * @authenticated
+     * @response {
+     *   "code": 200,
+     *   "message": "退出成功",
+     *   "data": null
+     * }
      */
     public function logout()
     {
@@ -52,7 +106,20 @@ class UserController extends Controller
     }
 
     /**
-     * 获取用户信息
+     * [用户管理] 获取用户信息
+     * 
+     * 获取当前用户的信息
+     * 
+     * @authenticated
+     * @response {
+     *   "code": 200,
+     *   "message": "获取成功",
+     *   "data": {
+     *     "id": 1,
+     *     "mobile": "13800138000",
+     *     "nickname": "用户昵称"
+     *   }
+     * }
      */
     public function info()
     {
@@ -60,7 +127,18 @@ class UserController extends Controller
     }
 
     /**
-     * 修改用户信息
+     * [用户管理] 修改用户信息
+     * 
+     * 修改当前用户的信息
+     * 
+     * @authenticated
+     * @bodyParam nickname string 用户昵称. Example: 用户昵称
+     * @bodyParam avatar string 用户头像. Example: https://example.com/avatar.jpg
+     * @response {
+     *   "code": 200,
+     *   "message": "修改成功",
+     *   "data": null
+     * }
      */
     public function updateInfo(Request $request)
     {
@@ -69,7 +147,19 @@ class UserController extends Controller
     }
 
     /**
-     * 获取用户钱包
+     * [用户管理] 获取用户钱包
+     * 
+     * 获取当前用户的钱包信息
+     * 
+     * @authenticated
+     * @response {
+     *   "code": 200,
+     *   "message": "获取成功",
+     *   "data": {
+     *     "balance": 100.00,
+     *     "freeze": 0.00
+     *   }
+     * }
      */
     public function wallet()
     {
@@ -77,7 +167,17 @@ class UserController extends Controller
     }
 
     /**
-     * 用户提现
+     * [用户管理] 用户提现
+     * 
+     * 提现用户的余额
+     * 
+     * @authenticated
+     * @bodyParam amount decimal 提现金额. Example: 100.00
+     * @response {
+     *   "code": 200,
+     *   "message": "提现成功",
+     *   "data": null
+     * }
      */
     public function withdraw(Request $request)
     {
@@ -86,7 +186,16 @@ class UserController extends Controller
     }
 
     /**
-     * 用户注销
+     * [用户管理] 用户注销
+     * 
+     * 注销当前用户的账号
+     * 
+     * @authenticated
+     * @response {
+     *   "code": 200,
+     *   "message": "注销成功",
+     *   "data": null
+     * }
      */
     public function delete()
     {
@@ -94,7 +203,17 @@ class UserController extends Controller
     }
 
     /**
-     * 用户反馈
+     * [用户管理] 用户反馈
+     * 
+     * 提交用户的反馈信息
+     * 
+     * @authenticated
+     * @bodyParam content string 反馈内容. Example: 这是一个反馈信息
+     * @response {
+     *   "code": 200,
+     *   "message": "提交成功",
+     *   "data": null
+     * }
      */
     public function feedback(Request $request)
     {
@@ -103,12 +222,19 @@ class UserController extends Controller
     }
 
     /**
+     * [用户管理] 申请成为技师
+     * 
      * 申请成为技师
+     * 
+     * @authenticated
+     * @response {
+     *   "code": 200,
+     *   "message": "申请成功",
+     *   "data": null
+     * }
      */
     public function applyCoach()
     {
         return $this->service->applyCoach();
     }
-
-
-} 
+}

+ 114 - 0
app/Http/Controllers/Client/WalletController.php

@@ -0,0 +1,114 @@
+<?php
+
+namespace App\Http\Controllers;
+
+use App\Services\Client\AccountService;
+use Illuminate\Http\Request;
+use Illuminate\Support\Facades\File;
+use Illuminate\Support\Facades\Artisan;
+
+class ScribeController extends Controller
+{
+    protected AccountService $accountService;
+
+    public function __construct(AccountService $accountService)
+    {
+        $this->accountService = $accountService;
+    }
+
+    /**
+     * 更新API文档的认证Token
+     * 
+     * 通过手机号获取验证码并登录,然后更新API文档的认证Token
+     * 
+     * @urlParam mobile string required 手机号码. Example: 13800138000
+     * @response {
+     *   "code": 200,
+     *   "message": "Token更新成功",
+     *   "data": {
+     *     "token": "your_new_token_here"
+     *   }
+     * }
+     */
+    public function updateAuthToken(Request $request, $mobile)
+    {
+        try {
+            // 1. 发送验证码
+            $result = $this->accountService->sendVerifyCode($mobile);
+            $code = $result['code'];
+
+            // 2. 登录获取token
+            $loginResult = $this->accountService->login($mobile, $code);
+            $token = $loginResult['token'];
+
+            // 3. 更新.env文件
+            $envPath = base_path('.env');
+            $envContent = File::get($envPath);
+            
+            // 更新或添加YOUR_AUTH_KEY
+            if (preg_match('/^YOUR_AUTH_KEY=.*$/m', $envContent)) {
+                $envContent = preg_replace('/^YOUR_AUTH_KEY=.*$/m', 'YOUR_AUTH_KEY=' . $token, $envContent);
+            } else {
+                $envContent .= "\nYOUR_AUTH_KEY=" . $token;
+            }
+            
+            File::put($envPath, $envContent);
+
+            // 4. 清除所有缓存并生成文档
+            $commands = [
+                'cd ' . base_path(),
+                'php artisan config:clear',
+                'php artisan cache:clear',
+                'php artisan route:clear',
+                'rm -rf .scribe/endpoints.cache',
+                'rm -rf public/docs',  // 删除旧的静态文件
+                'php artisan scribe:generate'
+            ];
+
+            $output = [];
+            $returnVar = 0;
+            exec(implode(' && ', $commands) . ' 2>&1', $output, $returnVar);
+
+            if ($returnVar !== 0) {
+                throw new \RuntimeException('Failed to generate documentation: ' . implode("\n", $output));
+            }
+
+            // 5. 直接更新 auth.md 文件
+            $authPath = base_path('.scribe/auth.md');
+            if (File::exists($authPath)) {
+                $authContent = File::get($authPath);
+                $authContent = preg_replace(
+                    '/\*\*`"Bearer [^"`]*"`\*\*/',
+                    '**`"Bearer ' . $token . '"`**',
+                    $authContent
+                );
+                File::put($authPath, $authContent);
+            }
+
+            // 6. 更新静态 HTML 文件
+            $docsPath = public_path('docs/index.html');
+            if (File::exists($docsPath)) {
+                $docsContent = File::get($docsPath);
+                $docsContent = preg_replace(
+                    '/Bearer [a-zA-Z0-9|]*/',
+                    'Bearer ' . $token,
+                    $docsContent
+                );
+                File::put($docsPath, $docsContent);
+            }
+
+            return response()->json([
+                'code' => 200,
+                'message' => 'Token更新成功',
+                'data' => [
+                    'token' => $token
+                ]
+            ]);
+        } catch (\Exception $e) {
+            return response()->json([
+                'code' => 500,
+                'message' => '更新失败: ' . $e->getMessage()
+            ], 500);
+        }
+    }
+}

+ 1 - 3
app/Models/MemberAddress.php

@@ -15,6 +15,7 @@ class MemberUser extends Model
 
 	protected $table = 'member_users';
 	protected $hidden = ['password'];
+	protected $fillable = ['mobile', 'state', 'register_area'];
 
 	protected static function booted()
 	{
@@ -25,7 +26,4 @@ class MemberUser extends Model
 			]);
 		});
 	}
-
-
-
 }

+ 26 - 0
app/Models/Order.php

@@ -0,0 +1,26 @@
+<?php
+
+namespace App\Providers;
+
+use Illuminate\Foundation\Support\Providers\AuthServiceProvider as ServiceProvider;
+use Illuminate\Support\Facades\Gate;
+
+class AuthServiceProvider extends ServiceProvider
+{
+    /**
+     * The model to policy mappings for the application.
+     *
+     * @var array<class-string, class-string>
+     */
+    protected $policies = [
+        //
+    ];
+
+    /**
+     * Register any authentication / authorization services.
+     */
+    public function boot(): void
+    {
+        $this->registerPolicies();
+    }
+}

+ 38 - 0
app/Providers/EventServiceProvider.php

@@ -0,0 +1,38 @@
+<?php
+
+namespace App\Providers;
+
+use Illuminate\Auth\Events\Registered;
+use Illuminate\Auth\Listeners\SendEmailVerificationNotification;
+use Illuminate\Foundation\Support\Providers\EventServiceProvider as ServiceProvider;
+use Illuminate\Support\Facades\Event;
+
+class EventServiceProvider extends ServiceProvider
+{
+    /**
+     * The event to listener mappings for the application.
+     *
+     * @var array<class-string, array<int, class-string>>
+     */
+    protected $listen = [
+        Registered::class => [
+            SendEmailVerificationNotification::class,
+        ],
+    ];
+
+    /**
+     * Register any events for your application.
+     */
+    public function boot(): void
+    {
+        //
+    }
+
+    /**
+     * Determine if events and listeners should be automatically discovered.
+     */
+    public function shouldDiscoverEvents(): bool
+    {
+        return false;
+    }
+}

+ 40 - 0
app/Providers/RouteServiceProvider.php

@@ -0,0 +1,40 @@
+<?php
+
+namespace App\Providers;
+
+use Illuminate\Cache\RateLimiting\Limit;
+use Illuminate\Foundation\Support\Providers\RouteServiceProvider as ServiceProvider;
+use Illuminate\Http\Request;
+use Illuminate\Support\Facades\RateLimiter;
+use Illuminate\Support\Facades\Route;
+
+class RouteServiceProvider extends ServiceProvider
+{
+    /**
+     * The path to your application's "home" route.
+     *
+     * Typically, users are redirected here after authentication.
+     *
+     * @var string
+     */
+    public const HOME = '/home';
+
+    /**
+     * Define your route model bindings, pattern filters, and other route configuration.
+     */
+    public function boot(): void
+    {
+        RateLimiter::for('api', function (Request $request) {
+            return Limit::perMinute(60)->by($request->user()?->id ?: $request->ip());
+        });
+
+        $this->routes(function () {
+            Route::middleware('api')
+                ->prefix('api')
+                ->group(base_path('routes/api.php'));
+
+            Route::middleware('web')
+                ->group(base_path('routes/web.php'));
+        });
+    }
+}

+ 1 - 0
app/Services/Client/AccountService.php

@@ -7,6 +7,7 @@
     "require": {
         "php": "^8.2",
         "iwzh/owl-scheduling": "^1.0",
+        "knuckleswtf/scribe": "^4.38",
         "laravel/framework": "v11.30.0",
         "laravel/sanctum": "^4.0",
         "laravel/tinker": "^2.9",

File diff suppressed because it is too large
+ 665 - 65
composer.lock


+ 99 - 0
config/app.php

@@ -123,4 +123,103 @@ return [
         'store' => env('APP_MAINTENANCE_STORE', 'database'),
     ],
 
+    /*
+    |--------------------------------------------------------------------------
+    | Autoloaded Service Providers
+    |--------------------------------------------------------------------------
+    |
+    | The service providers listed here will be automatically loaded on the
+    | request to your application. Feel free to add your own services to
+    | this array to grant expanded functionality to your applications.
+    |
+    */
+
+    'providers' => [
+        // Laravel Framework Service Providers...
+        Illuminate\Auth\AuthServiceProvider::class,
+        Illuminate\Broadcasting\BroadcastServiceProvider::class,
+        Illuminate\Bus\BusServiceProvider::class,
+        Illuminate\Cache\CacheServiceProvider::class,
+        Illuminate\Foundation\Providers\ConsoleSupportServiceProvider::class,
+        Illuminate\Cookie\CookieServiceProvider::class,
+        Illuminate\Database\DatabaseServiceProvider::class,
+        Illuminate\Encryption\EncryptionServiceProvider::class,
+        Illuminate\Filesystem\FilesystemServiceProvider::class,
+        Illuminate\Foundation\Providers\FoundationServiceProvider::class,
+        Illuminate\Hashing\HashServiceProvider::class,
+        Illuminate\Mail\MailServiceProvider::class,
+        Illuminate\Notifications\NotificationServiceProvider::class,
+        Illuminate\Pagination\PaginationServiceProvider::class,
+        Illuminate\Pipeline\PipelineServiceProvider::class,
+        Illuminate\Queue\QueueServiceProvider::class,
+        Illuminate\Redis\RedisServiceProvider::class,
+        Illuminate\Auth\Passwords\PasswordResetServiceProvider::class,
+        Illuminate\Session\SessionServiceProvider::class,
+        Illuminate\Translation\TranslationServiceProvider::class,
+        Illuminate\Validation\ValidationServiceProvider::class,
+        Illuminate\View\ViewServiceProvider::class,
+
+        // Package Service Providers...
+        Knuckles\Scribe\ScribeServiceProvider::class,
+
+        // Application Service Providers...
+        App\Providers\AppServiceProvider::class,
+        App\Providers\AuthServiceProvider::class,
+        App\Providers\EventServiceProvider::class,
+        App\Providers\RouteServiceProvider::class,
+    ],
+
+    /*
+    |--------------------------------------------------------------------------
+    | Class Aliases
+    |--------------------------------------------------------------------------
+    |
+    | This array of class aliases will be registered when this application
+    | is started. However, feel free to register as many as you wish as
+    | the aliases are "lazy" loaded so they don't hinder performance.
+    |
+    */
+
+    'aliases' => [
+        'App' => Illuminate\Support\Facades\App::class,
+        'Arr' => Illuminate\Support\Arr::class,
+        'Artisan' => Illuminate\Support\Facades\Artisan::class,
+        'Auth' => Illuminate\Support\Facades\Auth::class,
+        'Blade' => Illuminate\Support\Facades\Blade::class,
+        'Broadcast' => Illuminate\Support\Facades\Broadcast::class,
+        'Bus' => Illuminate\Support\Facades\Bus::class,
+        'Cache' => Illuminate\Support\Facades\Cache::class,
+        'Config' => Illuminate\Support\Facades\Config::class,
+        'Cookie' => Illuminate\Support\Facades\Cookie::class,
+        'Crypt' => Illuminate\Support\Facades\Crypt::class,
+        'Date' => Illuminate\Support\Carbon::class,
+        'DB' => Illuminate\Support\Facades\DB::class,
+        'Eloquent' => Illuminate\Database\Eloquent\Model::class,
+        'Event' => Illuminate\Support\Facades\Event::class,
+        'File' => Illuminate\Support\Facades\File::class,
+        'Gate' => Illuminate\Support\Facades\Gate::class,
+        'Hash' => Illuminate\Support\Facades\Hash::class,
+        'Http' => Illuminate\Support\Facades\Http::class,
+        'Js' => Illuminate\Support\Js::class,
+        'Lang' => Illuminate\Support\Facades\Lang::class,
+        'Log' => Illuminate\Support\Facades\Log::class,
+        'Mail' => Illuminate\Support\Facades\Mail::class,
+        'Notification' => Illuminate\Support\Facades\Notification::class,
+        'Password' => Illuminate\Support\Facades\Password::class,
+        'Process' => Illuminate\Support\Facades\Process::class,
+        'Queue' => Illuminate\Support\Facades\Queue::class,
+        'RateLimiter' => Illuminate\Support\Facades\RateLimiter::class,
+        'Redirect' => Illuminate\Support\Facades\Redirect::class,
+        'Request' => Illuminate\Support\Facades\Request::class,
+        'Response' => Illuminate\Support\Facades\Response::class,
+        'Route' => Illuminate\Support\Facades\Route::class,
+        'Schema' => Illuminate\Support\Facades\Schema::class,
+        'Session' => Illuminate\Support\Facades\Session::class,
+        'Storage' => Illuminate\Support\Facades\Storage::class,
+        'Str' => Illuminate\Support\Str::class,
+        'URL' => Illuminate\Support\Facades\URL::class,
+        'Validator' => Illuminate\Support\Facades\Validator::class,
+        'View' => Illuminate\Support\Facades\View::class,
+    ],
+
 ];

+ 148 - 0
config/scribe.php

@@ -0,0 +1,148 @@
+<?php
+
+use Knuckles\Scribe\Extracting\Strategies;
+
+return [
+    'title' => 'Owl Admin API Documentation',
+    
+    'description' => 'Owl Admin API 接口文档',
+
+    'base_url' => null,
+
+    'routes' => [
+        [
+            'match' => [
+                'prefixes' => ['api/*'],
+                'domains' => ['*'],
+                'versions' => ['v1'],
+            ],
+            'include' => [],
+            'exclude' => [],
+        ],
+    ],
+
+    'type' => 'static',
+    'theme' => 'default',
+
+    'static' => [
+        'output_path' => 'public/docs',
+    ],
+
+    'laravel' => [
+        'add_routes' => true,
+        'docs_url' => '/docs',
+        'middleware' => [],
+        // Directory within `public` in which to store CSS and JS assets.
+        // By default, assets are stored in `public/vendor/scribe`.
+        // If set, assets will be stored in `public/{{assets_directory}}`
+        'assets_directory' => null,
+    ],
+
+    'try_it_out' => [
+        'enabled' => true,
+        'base_url' => null,
+        'use_csrf' => false,
+        'csrf_url' => '/sanctum/csrf-cookie',
+    ],
+
+    'auth' => [
+        'enabled' => true,
+        'default' => true,
+        'in' => 'bearer',
+        'name' => 'Authorization',
+        'use_value' => 'Bearer {YOUR_AUTH_KEY}',
+        'placeholder' => env('YOUR_AUTH_KEY'),
+        'extra_info' => '通过登录接口获取 token,格式为 Bearer {token}',
+    ],
+
+    'intro_text' => <<<INTRO
+    欢迎使用 Owl Admin API 文档。
+
+    本文档提供了所有 API 接口的详细信息,包括请求参数、响应格式等。
+    INTRO,
+
+    'example_languages' => [
+        'bash',
+        'javascript',
+        'php'
+    ],
+
+    'examples' => [
+        'faker_seed' => null,
+        'models_source' => ['factoryCreate', 'factoryMake', 'databaseFirst'],
+    ],
+
+    'strategies' => [
+        'metadata' => [
+            Strategies\Metadata\GetFromDocBlocks::class,
+            Strategies\Metadata\GetFromMetadataAttributes::class,
+        ],
+        'urlParameters' => [
+            Strategies\UrlParameters\GetFromLaravelAPI::class,
+            Strategies\UrlParameters\GetFromUrlParamAttribute::class,
+            Strategies\UrlParameters\GetFromUrlParamTag::class,
+        ],
+        'queryParameters' => [
+            Strategies\QueryParameters\GetFromFormRequest::class,
+            Strategies\QueryParameters\GetFromInlineValidator::class,
+            Strategies\QueryParameters\GetFromQueryParamAttribute::class,
+            Strategies\QueryParameters\GetFromQueryParamTag::class,
+        ],
+        'headers' => [
+            Strategies\Headers\GetFromRouteRules::class,
+            Strategies\Headers\GetFromHeaderAttribute::class,
+            Strategies\Headers\GetFromHeaderTag::class,
+        ],
+        'bodyParameters' => [
+            Strategies\BodyParameters\GetFromFormRequest::class,
+            Strategies\BodyParameters\GetFromInlineValidator::class,
+            Strategies\BodyParameters\GetFromBodyParamAttribute::class,
+            Strategies\BodyParameters\GetFromBodyParamTag::class,
+        ],
+        'responses' => [
+            Strategies\Responses\UseResponseAttributes::class,
+            Strategies\Responses\UseTransformerTags::class,
+            Strategies\Responses\UseResponseTag::class,
+            Strategies\Responses\UseResponseFileTag::class,
+            Strategies\Responses\UseApiResourceTags::class,
+            Strategies\Responses\ResponseCalls::class,
+        ],
+        'responseFields' => [
+            Strategies\ResponseFields\GetFromResponseFieldAttribute::class,
+            Strategies\ResponseFields\GetFromResponseFieldTag::class,
+        ],
+    ],
+
+    'routeMatcher' => \Knuckles\Scribe\Matching\RouteMatcher::class,
+
+    'database_connections_to_transact' => [config('database.default')],
+    'external' => ['html_attributes' => []],
+    // Generate a Postman collection (v2.1.0) in addition to HTML docs.
+    // For 'static' docs, the collection will be generated to public/docs/collection.json.
+    // For 'laravel' docs, it will be generated to storage/app/scribe/collection.json.
+    // Setting `laravel.add_routes` to true (above) will also add a route for the collection.
+    'postman' => ['enabled' => true, 'overrides' => []],
+    // Generate an OpenAPI spec (v3.0.1) in addition to docs webpage.
+    // For 'static' docs, the collection will be generated to public/docs/openapi.yaml.
+    // For 'laravel' docs, it will be generated to storage/app/scribe/openapi.yaml.
+    // Setting `laravel.add_routes` to true (above) will also add a route for the spec.
+    'openapi' => ['enabled' => true, 'overrides' => []],
+    // Custom logo path. This will be used as the value of the src attribute for the <img> tag,
+    // so make sure it points to an accessible URL or path. Set to false to not use a logo.
+    // For example, if your logo is in public/img:
+    // - 'logo' => '../img/logo.png' // for `static` type (output folder is public/docs)
+    // - 'logo' => 'img/logo.png' // for `laravel` type
+    'logo' => false,
+    // Customize the "Last updated" value displayed in the docs by specifying tokens and formats.
+    // Examples:
+    // - {date:F j Y} => March 28, 2022
+    // - {git:short} => Short hash of the last Git commit
+    // Available tokens are `{date:<format>}` and `{git:<format>}`.
+    // The format you pass to `date` will be passed to PHP's `date()` function.
+    // The format you pass to `git` can be either "short" or "long".
+    'last_updated' => 'Last updated: {date:F j, Y}',
+    'fractal' => [
+        // If you are using a custom serializer with league/fractal, you can specify it here.
+        'serializer' => null,
+    ],
+];

+ 77 - 0
database/migrations/0001_01_01_000000_create_sessions_table.php

@@ -0,0 +1,77 @@
+<?php
+
+return [
+    "labels" => [
+        "search" => "Search",
+        "base_url" => "Base URL",
+    ],
+
+    "auth" => [
+        "none" => "This API is not authenticated.",
+        "instruction" => [
+            "query" => <<<TEXT
+                To authenticate requests, include a query parameter **`:parameterName`** in the request.
+                TEXT,
+            "body" => <<<TEXT
+                To authenticate requests, include a parameter **`:parameterName`** in the body of the request.
+                TEXT,
+            "query_or_body" => <<<TEXT
+                To authenticate requests, include a parameter **`:parameterName`** either in the query string or in the request body.
+                TEXT,
+            "bearer" => <<<TEXT
+                To authenticate requests, include an **`Authorization`** header with the value **`"Bearer :placeholder"`**.
+                TEXT,
+            "basic" => <<<TEXT
+                To authenticate requests, include an **`Authorization`** header in the form **`"Basic {credentials}"`**. 
+                The value of `{credentials}` should be your username/id and your password, joined with a colon (:), 
+                and then base64-encoded.
+                TEXT,
+            "header" => <<<TEXT
+                To authenticate requests, include a **`:parameterName`** header with the value **`":placeholder"`**.
+                TEXT,
+        ],
+        "details" => <<<TEXT
+            All authenticated endpoints are marked with a `requires authentication` badge in the documentation below.
+            TEXT,
+    ],
+
+    "headings" => [
+        "introduction" => "Introduction",
+        "auth" => "Authenticating requests",
+    ],
+
+    "endpoint" => [
+        "request" => "Request",
+        "headers" => "Headers",
+        "url_parameters" => "URL Parameters",
+        "body_parameters" => "Body Parameters",
+        "query_parameters" => "Query Parameters",
+        "response" => "Response",
+        "response_fields" => "Response Fields",
+        "example_request" => "Example request",
+        "example_response" => "Example response",
+        "responses" => [
+            "binary" => "Binary data",
+            "empty" => "Empty response",
+        ],
+    ],
+
+    "try_it_out" => [
+        "open" => "Try it out ⚡",
+        "cancel" => "Cancel 🛑",
+        "send" => "Send Request 💥",
+        "loading" => "⏱ Sending...",
+        "received_response" => "Received response",
+        "request_failed" => "Request failed with error",
+        "error_help" => <<<TEXT
+            Tip: Check that you're properly connected to the network.
+            If you're a maintainer of ths API, verify that your API is running and you've enabled CORS.
+            You can check the Dev Tools console for debugging information.
+            TEXT,
+    ],
+
+    "links" => [
+        "postman" => "View Postman collection",
+        "openapi" => "View OpenAPI spec",
+    ],
+];

+ 329 - 0
public/docs/collection.json

@@ -0,0 +1,329 @@
+{
+    "variable": [
+        {
+            "id": "baseUrl",
+            "key": "baseUrl",
+            "type": "string",
+            "name": "string",
+            "value": "http:\/\/192.168.110.10"
+        }
+    ],
+    "info": {
+        "name": "Owl Admin API Documentation",
+        "_postman_id": "ca5ccc9c-07c7-408d-aabc-bdd7bb5d79e3",
+        "description": "Owl Admin API \u63a5\u53e3\u6587\u6863",
+        "schema": "https:\/\/schema.getpostman.com\/json\/collection\/v2.1.0\/collection.json"
+    },
+    "item": [
+        {
+            "name": "Endpoints",
+            "description": "",
+            "item": [
+                {
+                    "name": "\u66f4\u65b0API\u6587\u6863\u7684\u8ba4\u8bc1Token",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/scribe\/update-token\/:mobile",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/scribe\/update-token\/:mobile",
+                            "variable": [
+                                {
+                                    "id": "mobile",
+                                    "key": "mobile",
+                                    "value": "13800138000",
+                                    "description": "\u624b\u673a\u53f7\u7801."
+                                }
+                            ]
+                        },
+                        "method": "GET",
+                        "header": [
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": null,
+                        "description": "\u901a\u8fc7\u624b\u673a\u53f7\u83b7\u53d6\u9a8c\u8bc1\u7801\u5e76\u767b\u5f55\uff0c\u7136\u540e\u66f4\u65b0API\u6587\u6863\u7684\u8ba4\u8bc1Token"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"Token\u66f4\u65b0\u6210\u529f\",\n  \"data\": {\n    \"token\": \"your_new_token_here\"\n  }\n}",
+                            "name": ""
+                        }
+                    ]
+                },
+                {
+                    "name": "[\u8d26\u6237\u7ba1\u7406] \u53d1\u9001\u9a8c\u8bc1\u7801",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/account\/send-code",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/account\/send-code"
+                        },
+                        "method": "POST",
+                        "header": [
+                            {
+                                "key": "Content-Type",
+                                "value": "application\/json"
+                            },
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": {
+                            "mode": "raw",
+                            "raw": "{\"mobile\":\"13800138000\"}"
+                        },
+                        "description": "\u5411\u6307\u5b9a\u624b\u673a\u53f7\u53d1\u9001\u9a8c\u8bc1\u7801"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"\u9a8c\u8bc1\u7801\u53d1\u9001\u6210\u529f\",\n  \"data\": null\n}",
+                            "name": ""
+                        }
+                    ]
+                },
+                {
+                    "name": "[\u8d26\u6237\u7ba1\u7406] \u7528\u6237\u767b\u5f55",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/account\/login",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/account\/login"
+                        },
+                        "method": "POST",
+                        "header": [
+                            {
+                                "key": "Content-Type",
+                                "value": "application\/json"
+                            },
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": {
+                            "mode": "raw",
+                            "raw": "{\"mobile\":\"13800138000\",\"code\":\"123456\"}"
+                        },
+                        "description": "\u4f7f\u7528\u624b\u673a\u53f7\u548c\u9a8c\u8bc1\u7801\u767b\u5f55\u8d26\u6237"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"\u767b\u5f55\u6210\u529f\",\n  \"data\": {\n    \"token\": \"eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...\",\n    \"user\": {\n      \"id\": 1,\n      \"mobile\": \"13800138000\",\n      \"nickname\": \"\u7528\u6237\u6635\u79f0\"\n    }\n  }\n}",
+                            "name": ""
+                        }
+                    ]
+                },
+                {
+                    "name": "[\u8d26\u6237\u7ba1\u7406] \u5fae\u4fe1\u767b\u5f55",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/account\/wx-login",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/account\/wx-login"
+                        },
+                        "method": "POST",
+                        "header": [
+                            {
+                                "key": "Content-Type",
+                                "value": "application\/json"
+                            },
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": {
+                            "mode": "raw",
+                            "raw": "{\"openid\":\"wx_123456789\"}"
+                        },
+                        "description": "\u4f7f\u7528\u5fae\u4fe1openid\u767b\u5f55\u8d26\u6237"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"\u767b\u5f55\u6210\u529f\",\n  \"data\": {\n    \"token\": \"eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...\",\n    \"user\": {\n      \"id\": 1,\n      \"openid\": \"wx_123456789\",\n      \"nickname\": \"\u5fae\u4fe1\u6635\u79f0\"\n    }\n  }\n}",
+                            "name": ""
+                        }
+                    ]
+                },
+                {
+                    "name": "[\u8d26\u6237\u7ba1\u7406] \u7528\u6237\u9000\u51fa",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/account\/logout",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/account\/logout"
+                        },
+                        "method": "POST",
+                        "header": [
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": null,
+                        "description": "\u9000\u51fa\u5f53\u524d\u8d26\u6237\u767b\u5f55\u72b6\u6001"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"\u9000\u51fa\u6210\u529f\",\n  \"data\": null\n}",
+                            "name": ""
+                        }
+                    ]
+                },
+                {
+                    "name": "[\u8d26\u6237\u7ba1\u7406] \u7528\u6237\u6ce8\u9500",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/account",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/account"
+                        },
+                        "method": "DELETE",
+                        "header": [
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": null,
+                        "description": "\u6c38\u4e45\u6ce8\u9500\u5f53\u524d\u8d26\u6237"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"\u6ce8\u9500\u6210\u529f\",\n  \"data\": null\n}",
+                            "name": ""
+                        }
+                    ]
+                },
+                {
+                    "name": "[\u7528\u6237\u7ba1\u7406] \u7528\u6237\u63d0\u73b0",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/user\/withdraw",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/user\/withdraw"
+                        },
+                        "method": "POST",
+                        "header": [
+                            {
+                                "key": "Content-Type",
+                                "value": "application\/json"
+                            },
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": {
+                            "mode": "raw",
+                            "raw": "{\"amount\":\"100.00\"}"
+                        },
+                        "description": "\u63d0\u73b0\u7528\u6237\u7684\u4f59\u989d"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"\u63d0\u73b0\u6210\u529f\",\n  \"data\": null\n}",
+                            "name": ""
+                        }
+                    ]
+                },
+                {
+                    "name": "[\u7528\u6237\u7ba1\u7406] \u7528\u6237\u53cd\u9988",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/user\/feedback",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/user\/feedback"
+                        },
+                        "method": "POST",
+                        "header": [
+                            {
+                                "key": "Content-Type",
+                                "value": "application\/json"
+                            },
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": {
+                            "mode": "raw",
+                            "raw": "{\"content\":\"\u8fd9\u662f\u4e00\u4e2a\u53cd\u9988\u4fe1\u606f\"}"
+                        },
+                        "description": "\u63d0\u4ea4\u7528\u6237\u7684\u53cd\u9988\u4fe1\u606f"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"\u63d0\u4ea4\u6210\u529f\",\n  \"data\": null\n}",
+                            "name": ""
+                        }
+                    ]
+                },
+                {
+                    "name": "[\u7528\u6237\u7ba1\u7406] \u7533\u8bf7\u6210\u4e3a\u6280\u5e08",
+                    "request": {
+                        "url": {
+                            "host": "{{baseUrl}}",
+                            "path": "api\/user\/apply-coach",
+                            "query": [],
+                            "raw": "{{baseUrl}}\/api\/user\/apply-coach"
+                        },
+                        "method": "POST",
+                        "header": [
+                            {
+                                "key": "Accept",
+                                "value": "application\/json"
+                            }
+                        ],
+                        "body": null,
+                        "description": "\u7533\u8bf7\u6210\u4e3a\u6280\u5e08"
+                    },
+                    "response": [
+                        {
+                            "header": [],
+                            "code": 200,
+                            "body": "{\n  \"code\": 200,\n  \"message\": \"\u7533\u8bf7\u6210\u529f\",\n  \"data\": null\n}",
+                            "name": ""
+                        }
+                    ]
+                }
+            ]
+        }
+    ],
+    "auth": {
+        "type": "bearer",
+        "bearer": [
+            {
+                "key": "Authorization",
+                "type": "string"
+            }
+        ]
+    }
+}

+ 393 - 0
public/docs/css/theme-default.print.css

@@ -0,0 +1,393 @@
+/* Copied from https://github.com/slatedocs/slate/blob/c4b4c0b8f83e891ca9fab6bbe9a1a88d5fe41292/stylesheets/print.css and unminified */
+/*! normalize.css v3.0.2 | MIT License | git.io/normalize */
+
+html {
+    font-family: sans-serif;
+    -ms-text-size-adjust: 100%;
+    -webkit-text-size-adjust: 100%
+}
+
+body {
+    margin: 0
+}
+
+article,
+aside,
+details,
+figcaption,
+figure,
+footer,
+header,
+hgroup,
+main,
+menu,
+nav,
+section,
+summary {
+    display: block
+}
+
+audio,
+canvas,
+progress,
+video {
+    display: inline-block;
+    vertical-align: baseline
+}
+
+audio:not([controls]) {
+    display: none;
+    height: 0
+}
+
+[hidden],
+template {
+    display: none
+}
+
+a {
+    background-color: transparent
+}
+
+a:active,
+a:hover {
+    outline: 0
+}
+
+abbr[title] {
+    border-bottom: 1px dotted
+}
+
+b,
+strong {
+    font-weight: bold
+}
+
+dfn {
+    font-style: italic
+}
+
+h1 {
+    font-size: 2em;
+    margin: 0.67em 0
+}
+
+mark {
+    background: #ff0;
+    color: #000
+}
+
+small {
+    font-size: 80%
+}
+
+sub,
+sup {
+    font-size: 75%;
+    line-height: 0;
+    position: relative;
+    vertical-align: baseline
+}
+
+sup {
+    top: -0.5em
+}
+
+sub {
+    bottom: -0.25em
+}
+
+img {
+    border: 0
+}
+
+svg:not(:root) {
+    overflow: hidden
+}
+
+figure {
+    margin: 1em 40px
+}
+
+hr {
+    box-sizing: content-box;
+    height: 0
+}
+
+pre {
+    overflow: auto
+}
+
+code,
+kbd,
+pre,
+samp {
+    font-family: monospace, monospace;
+    font-size: 1em
+}
+
+button,
+input,
+optgroup,
+select,
+textarea {
+    color: inherit;
+    font: inherit;
+    margin: 0
+}
+
+button {
+    overflow: visible
+}
+
+button,
+select {
+    text-transform: none
+}
+
+button,
+html input[type="button"],
+input[type="reset"],
+input[type="submit"] {
+    -webkit-appearance: button;
+    cursor: pointer
+}
+
+button[disabled],
+html input[disabled] {
+    cursor: default
+}
+
+button::-moz-focus-inner,
+input::-moz-focus-inner {
+    border: 0;
+    padding: 0
+}
+
+input {
+    line-height: normal
+}
+
+input[type="checkbox"],
+input[type="radio"] {
+    box-sizing: border-box;
+    padding: 0
+}
+
+input[type="number"]::-webkit-inner-spin-button,
+input[type="number"]::-webkit-outer-spin-button {
+    height: auto
+}
+
+input[type="search"] {
+    -webkit-appearance: textfield;
+    box-sizing: content-box
+}
+
+input[type="search"]::-webkit-search-cancel-button,
+input[type="search"]::-webkit-search-decoration {
+    -webkit-appearance: none
+}
+
+fieldset {
+    border: 1px solid #c0c0c0;
+    margin: 0 2px;
+    padding: 0.35em 0.625em 0.75em
+}
+
+legend {
+    border: 0;
+    padding: 0
+}
+
+textarea {
+    overflow: auto
+}
+
+optgroup {
+    font-weight: bold
+}
+
+table {
+    border-collapse: collapse;
+    border-spacing: 0
+}
+
+td,
+th {
+    padding: 0
+}
+
+.content h1,
+.content h2,
+.content h3,
+.content h4,
+body {
+    font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
+    font-size: 14px
+}
+
+.content h1,
+.content h2,
+.content h3,
+.content h4 {
+    font-weight: bold
+}
+
+.content pre,
+.content code {
+    font-family: Consolas, Menlo, Monaco, "Lucida Console", "Liberation Mono", "DejaVu Sans Mono", "Bitstream Vera Sans Mono", "Courier New", monospace, serif;
+    font-size: 12px;
+    line-height: 1.5
+}
+
+.content pre,
+.content code {
+    word-break: break-all;
+    -webkit-hyphens: auto;
+    -ms-hyphens: auto;
+    hyphens: auto
+}
+
+@font-face {
+    font-family: 'slate';
+    src: url(../fonts/slate.eot?-syv14m);
+    src: url(../fonts/slate.eot?#iefix-syv14m) format("embedded-opentype"), url(../fonts/slate.woff2?-syv14m) format("woff2"), url(../fonts/slate.woff?-syv14m) format("woff"), url(../fonts/slate.ttf?-syv14m) format("truetype"), url(../fonts/slate.svg?-syv14m#slate) format("svg");
+    font-weight: normal;
+    font-style: normal
+}
+
+.content aside.warning:before,
+.content aside.notice:before,
+.content aside.success:before {
+    font-family: 'slate';
+    speak: none;
+    font-style: normal;
+    font-weight: normal;
+    font-variant: normal;
+    text-transform: none;
+    line-height: 1
+}
+
+.content aside.warning:before {
+    content: "\e600"
+}
+
+.content aside.notice:before {
+    content: "\e602"
+}
+
+.content aside.success:before {
+    content: "\e606"
+}
+
+.tocify,
+.toc-footer,
+.lang-selector,
+.search,
+#nav-button {
+    display: none
+}
+
+.tocify-wrapper>img {
+    margin: 0 auto;
+    display: block
+}
+
+.content {
+    font-size: 12px
+}
+
+.content pre,
+.content code {
+    border: 1px solid #999;
+    border-radius: 5px;
+    font-size: 0.8em
+}
+
+.content pre code {
+    border: 0
+}
+
+.content pre {
+    padding: 1.3em
+}
+
+.content code {
+    padding: 0.2em
+}
+
+.content table {
+    border: 1px solid #999
+}
+
+.content table tr {
+    border-bottom: 1px solid #999
+}
+
+.content table td,
+.content table th {
+    padding: 0.7em
+}
+
+.content p {
+    line-height: 1.5
+}
+
+.content a {
+    text-decoration: none;
+    color: #000
+}
+
+.content h1 {
+    font-size: 2.5em;
+    padding-top: 0.5em;
+    padding-bottom: 0.5em;
+    margin-top: 1em;
+    margin-bottom: 21px;
+    border: 2px solid #ccc;
+    border-width: 2px 0;
+    text-align: center
+}
+
+.content h2 {
+    font-size: 1.8em;
+    margin-top: 2em;
+    border-top: 2px solid #ccc;
+    padding-top: 0.8em
+}
+
+.content h1+h2,
+.content h1+div+h2 {
+    border-top: none;
+    padding-top: 0;
+    margin-top: 0
+}
+
+.content h3,
+.content h4 {
+    font-size: 0.8em;
+    margin-top: 1.5em;
+    margin-bottom: 0.8em;
+    text-transform: uppercase
+}
+
+.content h5,
+.content h6 {
+    text-transform: uppercase
+}
+
+.content aside {
+    padding: 1em;
+    border: 1px solid #ccc;
+    border-radius: 5px;
+    margin-top: 1.5em;
+    margin-bottom: 1.5em;
+    line-height: 1.6
+}
+
+.content aside:before {
+    vertical-align: middle;
+    padding-right: 0.5em;
+    font-size: 14px
+}

+ 1090 - 0
public/docs/css/theme-default.style.css

@@ -0,0 +1,1090 @@
+/*! normalize.css v3.0.2 | MIT License | git.io/normalize */
+
+html {
+    font-family: 'Open Sans', sans-serif;
+    font-size: 1.2em;
+    -ms-text-size-adjust: 100%;
+    -webkit-text-size-adjust: 100%
+}
+
+body {
+    margin: 0
+}
+
+article,
+aside,
+details,
+figcaption,
+figure,
+footer,
+header,
+hgroup,
+main,
+menu,
+nav,
+section {
+    display: block
+}
+
+summary {
+    cursor: pointer;
+}
+
+audio,
+canvas,
+progress,
+video {
+    display: inline-block;
+    vertical-align: baseline
+}
+
+audio:not([controls]) {
+    display: none;
+    height: 0
+}
+
+[hidden],
+template {
+    display: none
+}
+
+a {
+    background-color: transparent
+}
+
+a:active,
+a:hover {
+    outline: 0
+}
+
+abbr[title] {
+    border-bottom: 1px dotted
+}
+
+b,
+strong {
+    font-weight: 700
+}
+
+dfn {
+    font-style: italic
+}
+
+h1 {
+    font-size: 2em;
+    margin: .67em 0
+}
+
+mark {
+    background: #ff0;
+    color: #000
+}
+
+small {
+    font-size: 80%
+}
+
+sub,
+sup {
+    font-size: 75%;
+    line-height: 0;
+    position: relative;
+    vertical-align: baseline
+}
+
+sup {
+    top: -.5em
+}
+
+sub {
+    bottom: -.25em
+}
+
+img {
+    border: 0
+}
+
+svg:not(:root) {
+    overflow: hidden
+}
+
+figure {
+    margin: 1em 40px
+}
+
+hr {
+    box-sizing: content-box;
+    height: 0
+}
+
+pre {
+    overflow: auto
+}
+
+code,
+kbd,
+pre,
+samp {
+    font-family: monospace, monospace;
+    font-size: 1em
+}
+
+button,
+input,
+optgroup,
+select,
+textarea {
+    color: inherit;
+    font: inherit;
+    margin: 0
+}
+
+button {
+    overflow: visible
+}
+
+button,
+select {
+    text-transform: none
+}
+
+button,
+html input[type=button],
+input[type=reset],
+input[type=submit] {
+    -webkit-appearance: button;
+    cursor: pointer
+}
+
+button[disabled],
+html input[disabled] {
+    cursor: default
+}
+
+button::-moz-focus-inner,
+input::-moz-focus-inner {
+    border: 0;
+    padding: 0
+}
+
+input {
+    line-height: normal
+}
+
+input[type=checkbox],
+input[type=radio] {
+    box-sizing: border-box;
+    padding: 0
+}
+
+input[type=number]::-webkit-inner-spin-button,
+input[type=number]::-webkit-outer-spin-button {
+    height: auto
+}
+
+input[type=search] {
+    -webkit-appearance: textfield;
+    box-sizing: content-box
+}
+
+input[type=search]::-webkit-search-cancel-button,
+input[type=search]::-webkit-search-decoration {
+    -webkit-appearance: none
+}
+
+fieldset {
+    border: 1px solid silver;
+    margin: 0 2px;
+    padding: .35em .625em .75em
+}
+
+legend {
+    border: 0;
+    padding: 0
+}
+
+textarea {
+    overflow: auto
+}
+
+optgroup {
+    font-weight: 700
+}
+
+table {
+    border-collapse: collapse;
+    border-spacing: 0
+}
+
+td,
+th {
+    padding: 0
+}
+
+body,
+html {
+    font-family: 'Open Sans', Helvetica Neue, Helvetica, Arial, Microsoft Yahei, 微软雅黑, STXihei, 华文细黑, sans-serif;
+    font-size: 16px;
+}
+
+.content h1,
+.content h2,
+.content h3,
+.content h4,
+.content h5,
+.content h6 {
+    font-family: 'Open Sans', Helvetica Neue, Helvetica, Arial, Microsoft Yahei, 微软雅黑, STXihei, 华文细黑, sans-serif;
+}
+
+.content h1,
+.content h2,
+.content h3,
+.content h4,
+.content h5,
+.content h6 {
+    font-weight: 700
+}
+
+.content code,
+.content pre {
+    font-family: Consolas, Menlo, Monaco, Lucida Console, Liberation Mono, DejaVu Sans Mono, Bitstream Vera Sans Mono, Courier New, monospace, serif;
+    font-size: 14px;
+    line-height: 1.5
+}
+
+.content code {
+    word-break: break-all;
+    word-break: break-word;
+    -webkit-hyphens: auto;
+    -ms-hyphens: auto;
+    hyphens: auto
+}
+
+.content aside.notice:before,
+.content aside.success:before,
+.content aside.warning:before,
+.tocify-wrapper>.search:before {
+    font-family: 'Open Sans', sans-serif;
+    speak: none;
+    font-style: normal;
+    font-variant: normal;
+    text-transform: none;
+    line-height: 1
+}
+
+.content aside.warning:before {
+    content: "✋"
+}
+
+.content aside.notice:before {
+    content: "ℹ"
+}
+
+.content aside.success:before {
+    content: "✅"
+}
+
+.tocify-wrapper>.search:before {
+    content: "🔎"
+}
+
+.highlight .c,
+.highlight .c1,
+.highlight .cm,
+.highlight .cs {
+    color: #909090
+}
+
+.highlight,
+.highlight .w {
+    background-color: #292929
+}
+
+.hljs {
+    display: block;
+    overflow-x: auto;
+    padding: .5em;
+    background: #23241f
+}
+
+.hljs,
+.hljs-subst,
+.hljs-tag {
+    color: #f8f8f2
+}
+
+.hljs-emphasis,
+.hljs-strong {
+    color: #a8a8a2
+}
+
+.hljs-bullet,
+.hljs-link,
+.hljs-literal,
+.hljs-number,
+.hljs-quote,
+.hljs-regexp {
+    color: #ae81ff
+}
+
+.hljs-code,
+.hljs-section,
+.hljs-selector-class,
+.hljs-title {
+    color: #a6e22e
+}
+
+.hljs-strong {
+    font-weight: 700
+}
+
+.hljs-emphasis {
+    font-style: italic
+}
+
+.hljs-attr,
+.hljs-keyword,
+.hljs-name,
+.hljs-selector-tag {
+    color: #f92672
+}
+
+.hljs-attribute,
+.hljs-symbol {
+    color: #66d9ef
+}
+
+.hljs-class .hljs-title,
+.hljs-params {
+    color: #f8f8f2
+}
+
+.hljs-addition,
+.hljs-built_in,
+.hljs-builtin-name,
+.hljs-selector-attr,
+.hljs-selector-id,
+.hljs-selector-pseudo,
+.hljs-string,
+.hljs-template-variable,
+.hljs-type,
+.hljs-variable {
+    color: #e6db74
+}
+
+.hljs-comment,
+.hljs-deletion,
+.hljs-meta {
+    color: #75715e
+}
+
+body,
+html {
+    color: #333;
+    padding: 0;
+    margin: 0;
+    -webkit-font-smoothing: antialiased;
+    -moz-osx-font-smoothing: grayscale;
+    background-color: whitesmoke;
+    height: 100%;
+    -webkit-text-size-adjust: none
+}
+
+#toc>ul>li>a>span {
+    float: right;
+    background-color: #2484ff;
+    border-radius: 40px;
+    width: 20px
+}
+
+.tocify-wrapper {
+    transition: left .3s ease-in-out;
+    overflow-y: auto;
+    overflow-x: hidden;
+    position: fixed;
+    z-index: 30;
+    top: 0;
+    left: 0;
+    bottom: 0;
+    width: 230px;
+    background-color: #393939;
+    font-size: 13px;
+    font-weight: 700
+}
+
+.tocify-wrapper .lang-selector {
+    display: none
+}
+
+.tocify-wrapper .lang-selector a {
+    padding-top: .5em;
+    padding-bottom: .5em
+}
+
+.tocify-wrapper>img {
+    display: block
+}
+
+.tocify-wrapper>.search {
+    position: relative
+}
+
+.tocify-wrapper>.search input {
+    background: #393939;
+    border-width: 0 0 1px;
+    border-color: #666;
+    padding: 6px 0 6px 20px;
+    box-sizing: border-box;
+    margin: 10px 15px;
+    width: 200px;
+    outline: none;
+    color: #fff;
+    border-radius: 0
+}
+
+.tocify-wrapper>.search:before {
+    position: absolute;
+    top: 17px;
+    left: 15px;
+    color: #fff
+}
+
+.tocify-wrapper img+.tocify {
+    margin-top: 20px
+}
+
+.tocify-wrapper .search-results {
+    margin-top: 0;
+    box-sizing: border-box;
+    height: 0;
+    overflow-y: auto;
+    overflow-x: hidden;
+    transition-property: height, margin;
+    transition-duration: .18s;
+    transition-timing-function: ease-in-out;
+    background: linear-gradient(180deg, rgba(0, 0, 0, .2), transparent 8px), linear-gradient(0deg, rgba(0, 0, 0, .2), transparent 8px), linear-gradient(180deg, #000, transparent 1.5px), linear-gradient(0deg, #939393, hsla(0, 0%, 58%, 0) 1.5px), #262626
+}
+
+.tocify-wrapper .search-results.visible {
+    height: 30%;
+    margin-bottom: 1em
+}
+
+.tocify-wrapper .search-results li {
+    margin: 1em 15px;
+    line-height: 1
+}
+
+.tocify-wrapper a {
+    color: #fff;
+    text-decoration: none
+}
+
+.tocify-wrapper .search-results a:hover {
+    text-decoration: underline
+}
+
+.tocify-wrapper .toc-footer li,
+.tocify-wrapper .tocify-item>a {
+    padding: 0 15px;
+    display: block;
+    overflow-x: hidden;
+    white-space: nowrap;
+    text-overflow: ellipsis
+}
+.tocify-wrapper .tocify-item.level-3>a {
+    padding: 0 25px;
+}
+
+.tocify-wrapper li,
+.tocify-wrapper ul {
+    list-style: none;
+    margin: 0;
+    padding: 0;
+    line-height: 28px
+}
+
+.tocify-wrapper li {
+    color: #fff;
+    transition-property: background;
+    transition-timing-function: linear;
+    transition-duration: .23s
+}
+
+.tocify-wrapper .tocify-focus {
+    box-shadow: 0 1px 0 #000;
+    background-color: #2467af;
+    color: #fff;
+    font-weight: bold;
+}
+
+.tocify-wrapper .tocify-subheader {
+    display: none;
+    background-color: #262626;
+    font-weight: 500;
+    background: linear-gradient(180deg, rgba(0, 0, 0, .2), transparent 8px), linear-gradient(0deg, rgba(0, 0, 0, .2), transparent 8px), linear-gradient(180deg, #000, transparent 1.5px), linear-gradient(0deg, #939393, hsla(0, 0%, 58%, 0) 1.5px), #262626
+}
+
+.tocify-wrapper .jets-searching .tocify-subheader,
+.tocify-wrapper .tocify-subheader.visible {
+    display: block;
+}
+
+.tocify-wrapper .tocify-subheader .tocify-item>a {
+    padding-left: 25px;
+    font-size: 12px
+}
+
+.tocify-wrapper .tocify-subheader .tocify-item.level-3>a {
+    padding-left: 35px;
+}
+
+.tocify-wrapper .tocify-subheader>li:last-child {
+    box-shadow: none
+}
+
+.tocify-wrapper .toc-footer {
+    padding: 1em 0;
+    margin-top: 1em;
+    border-top: 1px dashed #666
+}
+
+.tocify-wrapper .toc-footer a,
+.tocify-wrapper .toc-footer li {
+    color: #fff;
+    text-decoration: none
+}
+
+.tocify-wrapper .toc-footer a:hover {
+    text-decoration: underline
+}
+
+.tocify-wrapper .toc-footer li {
+    font-size: .8em;
+    line-height: 1.7;
+    text-decoration: none
+}
+
+#nav-button {
+    padding: 0 1.5em 5em 0;
+    display: none;
+    position: fixed;
+    top: 0;
+    left: 0;
+    z-index: 100;
+    color: #000;
+    text-decoration: none;
+    font-weight: 700;
+    opacity: .7;
+    line-height: 16px;
+    transition: left .3s ease-in-out
+}
+
+#nav-button span {
+    display: block;
+    padding: 6px;
+    background-color: rgba(234, 242, 246, .7);
+    -webkit-transform-origin: 0 0;
+    transform-origin: 0 0;
+    -webkit-transform: rotate(-90deg) translate(-100%);
+    transform: rotate(-90deg) translate(-100%);
+    border-radius: 0 0 0 5px
+}
+
+#nav-button img {
+    height: 16px;
+    vertical-align: bottom
+}
+
+#nav-button:hover {
+    opacity: 1
+}
+
+#nav-button.open {
+    left: 230px
+}
+
+.page-wrapper {
+    margin-left: 230px;
+    position: relative;
+    z-index: 10;
+    background-color: #eaf2f6;
+    min-height: 100%;
+    padding-bottom: 1px
+}
+
+.page-wrapper .dark-box {
+    width: 50%;
+    background-color: #393939;
+    position: absolute;
+    right: 0;
+    top: 0;
+    bottom: 0
+}
+
+.page-wrapper .lang-selector {
+    position: fixed;
+    z-index: 50;
+    border-bottom: 5px solid #393939
+}
+
+.lang-selector {
+    background-color: #222;
+    width: 100%;
+    font-weight: 700
+}
+
+.lang-selector button {
+    display: block;
+    float: left;
+    color: #fff;
+    text-decoration: none;
+    padding: 0 10px;
+    line-height: 30px;
+    outline: 0;
+    background: transparent;
+    border: none;
+}
+
+.lang-selector button:active,
+.lang-selector button:hover,
+.lang-selector button:focus {
+    background-color: #111;
+    color: #fff
+}
+
+.lang-selector button.active {
+    background-color: #393939;
+    color: #fff
+}
+
+.lang-selector:after {
+    content: '';
+    clear: both;
+    display: block
+}
+
+.content {
+    position: relative;
+    z-index: 30
+}
+
+.content:after {
+    content: '';
+    display: block;
+    clear: both
+}
+
+.content>aside,
+.content>details,
+.content>dl,
+.content>h1,
+.content>h2,
+.content>h3,
+.content>h4,
+.content>h5,
+.content>h6,
+.content>ol,
+.content>p,
+.content>table,
+.content>ul,
+.content>div,
+.content>form>aside,
+.content>form>details,
+.content>form>h1,
+.content>form>h2,
+.content>form>h3,
+.content>form>h4,
+.content>form>h5,
+.content>form>h6,
+.content>form>p,
+.content>form>table,
+.content>form>ul,
+.content>form>div {
+    margin-right: 50%;
+    padding: 0 28px;
+    box-sizing: border-box;
+    display: block;
+    text-shadow: 0 1px 0 #fff
+}
+
+.content>ol,
+.content>ul {
+    padding-left: 43px
+}
+
+.content>div,
+.content>h1,
+.content>h2 {
+    clear: both
+}
+
+.content h1 {
+    font-size: 30px;
+    padding-top: .5em;
+    padding-bottom: .5em;
+    border-bottom: 1px solid #ccc;
+    margin-bottom: 21px;
+    margin-top: 2em;
+    border-top: 1px solid #ddd;
+    background-image: linear-gradient(180deg, #fff, #f9f9f9)
+}
+
+.content div:first-child+h1,
+.content h1:first-child {
+    border-top-width: 0;
+    margin-top: 0
+}
+
+.content h2 {
+    font-size: 20px;
+    margin-top: 4em;
+    margin-bottom: 0;
+    border-top: 1px solid #ccc;
+    padding-top: 1.2em;
+    padding-bottom: 1.2em;
+    background-image: linear-gradient(180deg, hsla(0, 0%, 100%, .4), hsla(0, 0%, 100%, 0))
+}
+
+.content h1+div+h2,
+.content h1+h2 {
+    margin-top: -21px;
+    border-top: none
+}
+
+.content h3,
+.content h4,
+.content h5,
+.content h6 {
+    font-size: 15px;
+    margin-top: 2.5em;
+    margin-bottom: .8em
+}
+
+.content h4,
+.content h5,
+.content h6 {
+    font-size: 10px
+}
+
+.content hr {
+    margin: 2em 0;
+    border-top: 2px solid #393939;
+    border-bottom: 2px solid #eaf2f6
+}
+
+.content table {
+    margin-bottom: 1em;
+    overflow: auto
+}
+
+.content table td,
+.content table th {
+    text-align: left;
+    vertical-align: top;
+    line-height: 1.6
+}
+
+.content table th {
+    padding: 5px 10px;
+    border-bottom: 1px solid #ccc;
+    vertical-align: bottom
+}
+
+.content table td {
+    padding: 10px
+}
+
+.content table tr:last-child {
+    border-bottom: 1px solid #ccc
+}
+
+.content table tr:nth-child(odd)>td {
+    background-color: #ebf3f6
+}
+
+.content table tr:nth-child(even)>td {
+    background-color: #ebf2f6
+}
+
+.content dt {
+    font-weight: 700
+}
+
+.content dd {
+    margin-left: 15px
+}
+
+.content dd,
+.content dt,
+.content li,
+.content p {
+    line-height: 1.6;
+    margin-top: 0
+}
+
+.content img {
+    max-width: 100%
+}
+
+.content code {
+    padding: 3px;
+    border-radius: 3px
+}
+
+.content pre>code {
+    background-color: transparent;
+    padding: 0
+}
+
+.content aside {
+    padding-top: 1em;
+    padding-bottom: 1em;
+    margin-top: 1.5em;
+    margin-bottom: 1.5em;
+    background: #292929;
+    line-height: 1.6;
+    color: #c8c8c8;
+    text-shadow: none;
+}
+
+.content aside.info {
+    background: #8fbcd4;
+    text-shadow: 0 1px 0 #a0c6da;
+    color: initial;
+}
+
+.content aside.warning {
+    background-color: #c97a7e;
+    text-shadow: 0 1px 0 #d18e91;
+    color: initial;
+}
+
+.content aside.success {
+    background-color: #6ac174;
+    text-shadow: 0 1px 0 #80ca89;
+    color: initial;
+}
+
+.content aside:before {
+    vertical-align: middle;
+    padding-right: .5em;
+    font-size: 14px
+}
+
+.content .search-highlight {
+    padding: 2px;
+    margin: -2px;
+    border-radius: 4px;
+    border: 1px solid #f7e633;
+    text-shadow: 1px 1px 0 #666;
+    background: linear-gradient(to top left, #f7e633, #f1d32f)
+}
+
+.content blockquote,
+.content pre {
+    background-color: #292929;
+    color: #fff;
+    padding: 1.5em 28px;
+    margin: 0;
+    width: 50%;
+    float: right;
+    clear: right;
+    box-sizing: border-box;
+    text-shadow: 0 1px 2px rgba(0, 0, 0, .4)
+}
+
+.content blockquote pre.sf-dump,
+.content pre pre.sf-dump {
+    width: 100%;
+}
+
+.content .annotation {
+    background-color: #292929;
+    color: #fff;
+    padding: 0 28px;
+    margin: 0;
+    width: 50%;
+    float: right;
+    clear: right;
+    box-sizing: border-box;
+    text-shadow: 0 1px 2px rgba(0, 0, 0, .4)
+}
+
+.content .annotation pre {
+    padding: 0 0;
+    width: 100%;
+    float: none;
+}
+
+.content blockquote>p,
+.content pre>p {
+    margin: 0
+}
+
+.content blockquote a,
+.content pre a {
+    color: #fff;
+    text-decoration: none;
+    border-bottom: 1px dashed #ccc
+}
+
+.content blockquote>p {
+    background-color: #1c1c1c;
+    border-radius: 5px;
+    padding: 13px;
+    color: #ccc;
+    border-top: 1px solid #000;
+    border-bottom: 1px solid #404040
+}
+
+@media (max-width:930px) {
+    .tocify-wrapper {
+        left: -230px
+    }
+    .tocify-wrapper.open {
+        left: 0
+    }
+    .page-wrapper {
+        margin-left: 0
+    }
+    #nav-button {
+        display: block
+    }
+    .tocify-wrapper .tocify-item>a {
+        padding-top: .3em;
+        padding-bottom: .3em
+    }
+}
+
+@media (max-width:700px) {
+    .dark-box {
+        display: none
+    }
+    .tocify-wrapper .lang-selector {
+        display: block
+    }
+    .page-wrapper .lang-selector {
+        display: none
+    }
+    .content>aside,
+    .content>details,
+    .content>dl,
+    .content>h1,
+    .content>h2,
+    .content>h3,
+    .content>h4,
+    .content>h5,
+    .content>h6,
+    .content>ol,
+    .content>p,
+    .content>table,
+    .content>ul,
+    .content>div,
+    .content>form>aside,
+    .content>form>details,
+    .content>form>h1,
+    .content>form>h2,
+    .content>form>h3,
+    .content>form>h4,
+    .content>form>h5,
+    .content>form>h6,
+    .content>form>p,
+    .content>form>table,
+    .content>form>ul,
+    .content>form>div {
+        margin-right: 0;
+    }
+    .content blockquote,
+    .content pre {
+        float: none;
+        width: auto
+    }
+    .content .annotation {
+        float: none;
+        width: auto
+    }
+}
+
+.badge {
+    padding: 1px 9px 2px;
+    white-space: nowrap;
+    -webkit-border-radius: 9px;
+    -moz-border-radius: 9px;
+    border-radius: 9px;
+    color: #ffffff;
+    text-shadow: none !important;
+    font-weight: bold;
+}
+
+.badge.badge-darkred {
+    background-color: darkred;
+}
+
+.badge.badge-red {
+    background-color: red;
+}
+
+.badge.badge-blue {
+    background-color: blue;
+}
+
+.badge.badge-darkblue {
+    background-color: darkblue;
+}
+
+.badge.badge-green {
+    background-color: green;
+}
+
+.badge.badge-darkgreen {
+    background-color: darkgreen;
+}
+
+.badge.badge-purple {
+    background-color: purple;
+}
+
+.badge.badge-black {
+    background-color: black;
+}
+
+.badge.badge-grey {
+    background-color: grey;
+}
+
+.fancy-heading-panel {
+    background-color: lightgrey;
+    border-radius: 5px;
+    padding-left: 5px !important;
+    padding-top: 5px !important;
+    padding-bottom: 5px !important;
+    margin-left: 25px;
+    margin-right: 10px;
+    width: 47%;
+}
+
+@media screen and (max-width: 700px) {
+    .fancy-heading-panel {
+        width: 95%;
+    }
+
+}
+
+button {
+    border: none;
+}
+
+* {
+    /* Foreground, Background */
+    scrollbar-color: #3c4c67 transparent;
+}
+*::-webkit-scrollbar { /* Background */
+    width: 10px;
+    height: 10px;
+    background: transparent;
+}
+
+*::-webkit-scrollbar-thumb { /* Foreground */
+    background: #626161;
+}

BIN
public/docs/images/navbar.png


+ 1490 - 0
public/docs/index.html

@@ -0,0 +1,1490 @@
+<!doctype html>
+<html lang="en">
+<head>
+    <meta charset="utf-8">
+    <meta content="IE=edge,chrome=1" http-equiv="X-UA-Compatible">
+    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
+    <title>Owl Admin API Documentation</title>
+
+    <link href="https://fonts.googleapis.com/css?family=Open+Sans&display=swap" rel="stylesheet">
+
+    <link rel="stylesheet" href="../docs/css/theme-default.style.css" media="screen">
+    <link rel="stylesheet" href="../docs/css/theme-default.print.css" media="print">
+
+    <script src="https://cdn.jsdelivr.net/npm/lodash@4.17.10/lodash.min.js"></script>
+
+    <link rel="stylesheet"
+          href="https://unpkg.com/@highlightjs/cdn-assets@11.6.0/styles/obsidian.min.css">
+    <script src="https://unpkg.com/@highlightjs/cdn-assets@11.6.0/highlight.min.js"></script>
+
+    <script src="https://cdnjs.cloudflare.com/ajax/libs/jets/0.14.1/jets.min.js"></script>
+
+    <style id="language-style">
+        /* starts out as display none and is replaced with js later  */
+                    body .content .bash-example code { display: none; }
+                    body .content .javascript-example code { display: none; }
+                    body .content .php-example code { display: none; }
+            </style>
+
+    <script>
+        var tryItOutBaseUrl = "http://192.168.110.10";
+        var useCsrf = Boolean();
+        var csrfUrl = "/sanctum/csrf-cookie";
+    </script>
+    <script src="../docs/js/tryitout-4.38.0.js"></script>
+
+    <script src="../docs/js/theme-default-4.38.0.js"></script>
+
+</head>
+
+<body data-languages="[&quot;bash&quot;,&quot;javascript&quot;,&quot;php&quot;]">
+
+<a href="#" id="nav-button">
+    <span>
+        MENU
+        <img src="../docs/images/navbar.png" alt="navbar-image"/>
+    </span>
+</a>
+<div class="tocify-wrapper">
+    
+            <div class="lang-selector">
+                                            <button type="button" class="lang-button" data-language-name="bash">bash</button>
+                                            <button type="button" class="lang-button" data-language-name="javascript">javascript</button>
+                                            <button type="button" class="lang-button" data-language-name="php">php</button>
+                    </div>
+    
+    <div class="search">
+        <input type="text" class="search" id="input-search" placeholder="Search">
+    </div>
+
+    <div id="toc">
+                    <ul id="tocify-header-introduction" class="tocify-header">
+                <li class="tocify-item level-1" data-unique="introduction">
+                    <a href="#introduction">Introduction</a>
+                </li>
+                            </ul>
+                    <ul id="tocify-header-authenticating-requests" class="tocify-header">
+                <li class="tocify-item level-1" data-unique="authenticating-requests">
+                    <a href="#authenticating-requests">Authenticating requests</a>
+                </li>
+                            </ul>
+                    <ul id="tocify-header-endpoints" class="tocify-header">
+                <li class="tocify-item level-1" data-unique="endpoints">
+                    <a href="#endpoints">Endpoints</a>
+                </li>
+                                    <ul id="tocify-subheader-endpoints" class="tocify-subheader">
+                                                    <li class="tocify-item level-2" data-unique="endpoints-GETapi-scribe-update-token--mobile-">
+                                <a href="#endpoints-GETapi-scribe-update-token--mobile-">更新API文档的认证Token</a>
+                            </li>
+                                                                                <li class="tocify-item level-2" data-unique="endpoints-POSTapi-account-send-code">
+                                <a href="#endpoints-POSTapi-account-send-code">[账户管理] 发送验证码</a>
+                            </li>
+                                                                                <li class="tocify-item level-2" data-unique="endpoints-POSTapi-account-login">
+                                <a href="#endpoints-POSTapi-account-login">[账户管理] 用户登录</a>
+                            </li>
+                                                                                <li class="tocify-item level-2" data-unique="endpoints-POSTapi-account-wx-login">
+                                <a href="#endpoints-POSTapi-account-wx-login">[账户管理] 微信登录</a>
+                            </li>
+                                                                                <li class="tocify-item level-2" data-unique="endpoints-POSTapi-account-logout">
+                                <a href="#endpoints-POSTapi-account-logout">[账户管理] 用户退出</a>
+                            </li>
+                                                                                <li class="tocify-item level-2" data-unique="endpoints-DELETEapi-account">
+                                <a href="#endpoints-DELETEapi-account">[账户管理] 用户注销</a>
+                            </li>
+                                                                                <li class="tocify-item level-2" data-unique="endpoints-POSTapi-user-withdraw">
+                                <a href="#endpoints-POSTapi-user-withdraw">[用户管理] 用户提现</a>
+                            </li>
+                                                                                <li class="tocify-item level-2" data-unique="endpoints-POSTapi-user-feedback">
+                                <a href="#endpoints-POSTapi-user-feedback">[用户管理] 用户反馈</a>
+                            </li>
+                                                                                <li class="tocify-item level-2" data-unique="endpoints-POSTapi-user-apply-coach">
+                                <a href="#endpoints-POSTapi-user-apply-coach">[用户管理] 申请成为技师</a>
+                            </li>
+                                                                        </ul>
+                            </ul>
+            </div>
+
+    <ul class="toc-footer" id="toc-footer">
+                    <li style="padding-bottom: 5px;"><a href="../docs/collection.json">View Postman collection</a></li>
+                            <li style="padding-bottom: 5px;"><a href="../docs/openapi.yaml">View OpenAPI spec</a></li>
+                <li><a href="http://github.com/knuckleswtf/scribe">Documentation powered by Scribe ✍</a></li>
+    </ul>
+
+    <ul class="toc-footer" id="last-updated">
+        <li>Last updated: November 18, 2024</li>
+    </ul>
+</div>
+
+<div class="page-wrapper">
+    <div class="dark-box"></div>
+    <div class="content">
+        <h1 id="introduction">Introduction</h1>
+<p>Owl Admin API 接口文档</p>
+<aside>
+    <strong>Base URL</strong>: <code>http://192.168.110.10</code>
+</aside>
+<p>欢迎使用 Owl Admin API 文档。</p>
+<p>本文档提供了所有 API 接口的详细信息,包括请求参数、响应格式等。</p>
+
+        <h1 id="authenticating-requests">Authenticating requests</h1>
+<p>To authenticate requests, include an <strong><code>Authorization</code></strong> header with the value <strong><code>"Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"</code></strong>.</p>
+<p>All authenticated endpoints are marked with a <code>requires authentication</code> badge in the documentation below.</p>
+<p>通过登录接口获取 token,格式为 Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743{token}</p>
+
+        <h1 id="endpoints">Endpoints</h1>
+
+    
+
+                                <h2 id="endpoints-GETapi-scribe-update-token--mobile-">更新API文档的认证Token</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>通过手机号获取验证码并登录,然后更新API文档的认证Token</p>
+
+<span id="example-requests-GETapi-scribe-update-token--mobile-">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request GET \
+    --get "http://192.168.110.10/api/scribe/update-token/13800138000" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/scribe/update-token/13800138000"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Accept": "application/json",
+};
+
+fetch(url, {
+    method: "GET",
+    headers,
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/scribe/update-token/13800138000';
+$response = $client-&gt;get(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-GETapi-scribe-update-token--mobile-">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;Token更新成功&quot;,
+    &quot;data&quot;: {
+        &quot;token&quot;: &quot;your_new_token_here&quot;
+    }
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-GETapi-scribe-update-token--mobile-" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-GETapi-scribe-update-token--mobile-"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-GETapi-scribe-update-token--mobile-"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-GETapi-scribe-update-token--mobile-" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-GETapi-scribe-update-token--mobile-">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-GETapi-scribe-update-token--mobile-" data-method="GET"
+      data-path="api/scribe/update-token/{mobile}"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('GETapi-scribe-update-token--mobile-', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-GETapi-scribe-update-token--mobile-"
+                    onclick="tryItOut('GETapi-scribe-update-token--mobile-');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-GETapi-scribe-update-token--mobile-"
+                    onclick="cancelTryOut('GETapi-scribe-update-token--mobile-');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-GETapi-scribe-update-token--mobile-"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-green">GET</small>
+            <b><code>api/scribe/update-token/{mobile}</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="GETapi-scribe-update-token--mobile-"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                        <h4 class="fancy-heading-panel"><b>URL Parameters</b></h4>
+                    <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>mobile</code></b>&nbsp;&nbsp;
+<small>string</small>&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="mobile"                data-endpoint="GETapi-scribe-update-token--mobile-"
+               value="13800138000"
+               data-component="url">
+    <br>
+<p>手机号码. Example: <code>13800138000</code></p>
+            </div>
+                    </form>
+
+                    <h2 id="endpoints-POSTapi-account-send-code">[账户管理] 发送验证码</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>向指定手机号发送验证码</p>
+
+<span id="example-requests-POSTapi-account-send-code">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request POST \
+    "http://192.168.110.10/api/account/send-code" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743" \
+    --header "Content-Type: application/json" \
+    --data "{
+    \"mobile\": \"13800138000\"
+}"
+</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/account/send-code"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Content-Type": "application/json",
+    "Accept": "application/json",
+};
+
+let body = {
+    "mobile": "13800138000"
+};
+
+fetch(url, {
+    method: "POST",
+    headers,
+    body: JSON.stringify(body),
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/account/send-code';
+$response = $client-&gt;post(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+            'Content-Type' =&gt; 'application/json',
+        ],
+        'json' =&gt; [
+            'mobile' =&gt; '13800138000',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-POSTapi-account-send-code">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;验证码发送成功&quot;,
+    &quot;data&quot;: null
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-POSTapi-account-send-code" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-POSTapi-account-send-code"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-POSTapi-account-send-code"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-POSTapi-account-send-code" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-POSTapi-account-send-code">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-POSTapi-account-send-code" data-method="POST"
+      data-path="api/account/send-code"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('POSTapi-account-send-code', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-POSTapi-account-send-code"
+                    onclick="tryItOut('POSTapi-account-send-code');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-POSTapi-account-send-code"
+                    onclick="cancelTryOut('POSTapi-account-send-code');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-POSTapi-account-send-code"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-black">POST</small>
+            <b><code>api/account/send-code</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="POSTapi-account-send-code"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Content-Type</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Content-Type"                data-endpoint="POSTapi-account-send-code"
+               value="application/json"
+               data-component="header">
+    <br>
+<p>Example: <code>application/json</code></p>
+            </div>
+                                <h4 class="fancy-heading-panel"><b>Body Parameters</b></h4>
+        <div style=" padding-left: 28px;  clear: unset;">
+            <b style="line-height: 2;"><code>mobile</code></b>&nbsp;&nbsp;
+<small>string</small>&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="mobile"                data-endpoint="POSTapi-account-send-code"
+               value="13800138000"
+               data-component="body">
+    <br>
+<p>手机号码. Example: <code>13800138000</code></p>
+        </div>
+        </form>
+
+                    <h2 id="endpoints-POSTapi-account-login">[账户管理] 用户登录</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>使用手机号和验证码登录账户</p>
+
+<span id="example-requests-POSTapi-account-login">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request POST \
+    "http://192.168.110.10/api/account/login" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743" \
+    --header "Content-Type: application/json" \
+    --data "{
+    \"mobile\": \"13800138000\",
+    \"code\": \"123456\"
+}"
+</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/account/login"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Content-Type": "application/json",
+    "Accept": "application/json",
+};
+
+let body = {
+    "mobile": "13800138000",
+    "code": "123456"
+};
+
+fetch(url, {
+    method: "POST",
+    headers,
+    body: JSON.stringify(body),
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/account/login';
+$response = $client-&gt;post(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+            'Content-Type' =&gt; 'application/json',
+        ],
+        'json' =&gt; [
+            'mobile' =&gt; '13800138000',
+            'code' =&gt; '123456',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-POSTapi-account-login">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;登录成功&quot;,
+    &quot;data&quot;: {
+        &quot;token&quot;: &quot;eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...&quot;,
+        &quot;user&quot;: {
+            &quot;id&quot;: 1,
+            &quot;mobile&quot;: &quot;13800138000&quot;,
+            &quot;nickname&quot;: &quot;用户昵称&quot;
+        }
+    }
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-POSTapi-account-login" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-POSTapi-account-login"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-POSTapi-account-login"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-POSTapi-account-login" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-POSTapi-account-login">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-POSTapi-account-login" data-method="POST"
+      data-path="api/account/login"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('POSTapi-account-login', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-POSTapi-account-login"
+                    onclick="tryItOut('POSTapi-account-login');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-POSTapi-account-login"
+                    onclick="cancelTryOut('POSTapi-account-login');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-POSTapi-account-login"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-black">POST</small>
+            <b><code>api/account/login</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="POSTapi-account-login"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Content-Type</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Content-Type"                data-endpoint="POSTapi-account-login"
+               value="application/json"
+               data-component="header">
+    <br>
+<p>Example: <code>application/json</code></p>
+            </div>
+                                <h4 class="fancy-heading-panel"><b>Body Parameters</b></h4>
+        <div style=" padding-left: 28px;  clear: unset;">
+            <b style="line-height: 2;"><code>mobile</code></b>&nbsp;&nbsp;
+<small>string</small>&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="mobile"                data-endpoint="POSTapi-account-login"
+               value="13800138000"
+               data-component="body">
+    <br>
+<p>手机号码. Example: <code>13800138000</code></p>
+        </div>
+                <div style=" padding-left: 28px;  clear: unset;">
+            <b style="line-height: 2;"><code>code</code></b>&nbsp;&nbsp;
+<small>string</small>&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="code"                data-endpoint="POSTapi-account-login"
+               value="123456"
+               data-component="body">
+    <br>
+<p>验证码. Example: <code>123456</code></p>
+        </div>
+        </form>
+
+                    <h2 id="endpoints-POSTapi-account-wx-login">[账户管理] 微信登录</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>使用微信openid登录账户</p>
+
+<span id="example-requests-POSTapi-account-wx-login">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request POST \
+    "http://192.168.110.10/api/account/wx-login" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743" \
+    --header "Content-Type: application/json" \
+    --data "{
+    \"openid\": \"wx_123456789\"
+}"
+</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/account/wx-login"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Content-Type": "application/json",
+    "Accept": "application/json",
+};
+
+let body = {
+    "openid": "wx_123456789"
+};
+
+fetch(url, {
+    method: "POST",
+    headers,
+    body: JSON.stringify(body),
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/account/wx-login';
+$response = $client-&gt;post(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+            'Content-Type' =&gt; 'application/json',
+        ],
+        'json' =&gt; [
+            'openid' =&gt; 'wx_123456789',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-POSTapi-account-wx-login">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;登录成功&quot;,
+    &quot;data&quot;: {
+        &quot;token&quot;: &quot;eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...&quot;,
+        &quot;user&quot;: {
+            &quot;id&quot;: 1,
+            &quot;openid&quot;: &quot;wx_123456789&quot;,
+            &quot;nickname&quot;: &quot;微信昵称&quot;
+        }
+    }
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-POSTapi-account-wx-login" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-POSTapi-account-wx-login"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-POSTapi-account-wx-login"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-POSTapi-account-wx-login" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-POSTapi-account-wx-login">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-POSTapi-account-wx-login" data-method="POST"
+      data-path="api/account/wx-login"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('POSTapi-account-wx-login', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-POSTapi-account-wx-login"
+                    onclick="tryItOut('POSTapi-account-wx-login');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-POSTapi-account-wx-login"
+                    onclick="cancelTryOut('POSTapi-account-wx-login');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-POSTapi-account-wx-login"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-black">POST</small>
+            <b><code>api/account/wx-login</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="POSTapi-account-wx-login"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Content-Type</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Content-Type"                data-endpoint="POSTapi-account-wx-login"
+               value="application/json"
+               data-component="header">
+    <br>
+<p>Example: <code>application/json</code></p>
+            </div>
+                                <h4 class="fancy-heading-panel"><b>Body Parameters</b></h4>
+        <div style=" padding-left: 28px;  clear: unset;">
+            <b style="line-height: 2;"><code>openid</code></b>&nbsp;&nbsp;
+<small>string</small>&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="openid"                data-endpoint="POSTapi-account-wx-login"
+               value="wx_123456789"
+               data-component="body">
+    <br>
+<p>微信openid. Example: <code>wx_123456789</code></p>
+        </div>
+        </form>
+
+                    <h2 id="endpoints-POSTapi-account-logout">[账户管理] 用户退出</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>退出当前账户登录状态</p>
+
+<span id="example-requests-POSTapi-account-logout">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request POST \
+    "http://192.168.110.10/api/account/logout" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/account/logout"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Accept": "application/json",
+};
+
+fetch(url, {
+    method: "POST",
+    headers,
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/account/logout';
+$response = $client-&gt;post(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-POSTapi-account-logout">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;退出成功&quot;,
+    &quot;data&quot;: null
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-POSTapi-account-logout" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-POSTapi-account-logout"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-POSTapi-account-logout"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-POSTapi-account-logout" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-POSTapi-account-logout">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-POSTapi-account-logout" data-method="POST"
+      data-path="api/account/logout"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('POSTapi-account-logout', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-POSTapi-account-logout"
+                    onclick="tryItOut('POSTapi-account-logout');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-POSTapi-account-logout"
+                    onclick="cancelTryOut('POSTapi-account-logout');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-POSTapi-account-logout"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-black">POST</small>
+            <b><code>api/account/logout</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="POSTapi-account-logout"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                        </form>
+
+                    <h2 id="endpoints-DELETEapi-account">[账户管理] 用户注销</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>永久注销当前账户</p>
+
+<span id="example-requests-DELETEapi-account">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request DELETE \
+    "http://192.168.110.10/api/account" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/account"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Accept": "application/json",
+};
+
+fetch(url, {
+    method: "DELETE",
+    headers,
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/account';
+$response = $client-&gt;delete(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-DELETEapi-account">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;注销成功&quot;,
+    &quot;data&quot;: null
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-DELETEapi-account" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-DELETEapi-account"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-DELETEapi-account"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-DELETEapi-account" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-DELETEapi-account">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-DELETEapi-account" data-method="DELETE"
+      data-path="api/account"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('DELETEapi-account', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-DELETEapi-account"
+                    onclick="tryItOut('DELETEapi-account');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-DELETEapi-account"
+                    onclick="cancelTryOut('DELETEapi-account');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-DELETEapi-account"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-red">DELETE</small>
+            <b><code>api/account</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="DELETEapi-account"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                        </form>
+
+                    <h2 id="endpoints-POSTapi-user-withdraw">[用户管理] 用户提现</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>提现用户的余额</p>
+
+<span id="example-requests-POSTapi-user-withdraw">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request POST \
+    "http://192.168.110.10/api/user/withdraw" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743" \
+    --header "Content-Type: application/json" \
+    --data "{
+    \"amount\": \"100.00\"
+}"
+</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/user/withdraw"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Content-Type": "application/json",
+    "Accept": "application/json",
+};
+
+let body = {
+    "amount": "100.00"
+};
+
+fetch(url, {
+    method: "POST",
+    headers,
+    body: JSON.stringify(body),
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/user/withdraw';
+$response = $client-&gt;post(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+            'Content-Type' =&gt; 'application/json',
+        ],
+        'json' =&gt; [
+            'amount' =&gt; '100.00',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-POSTapi-user-withdraw">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;提现成功&quot;,
+    &quot;data&quot;: null
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-POSTapi-user-withdraw" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-POSTapi-user-withdraw"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-POSTapi-user-withdraw"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-POSTapi-user-withdraw" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-POSTapi-user-withdraw">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-POSTapi-user-withdraw" data-method="POST"
+      data-path="api/user/withdraw"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('POSTapi-user-withdraw', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-POSTapi-user-withdraw"
+                    onclick="tryItOut('POSTapi-user-withdraw');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-POSTapi-user-withdraw"
+                    onclick="cancelTryOut('POSTapi-user-withdraw');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-POSTapi-user-withdraw"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-black">POST</small>
+            <b><code>api/user/withdraw</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="POSTapi-user-withdraw"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Content-Type</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Content-Type"                data-endpoint="POSTapi-user-withdraw"
+               value="application/json"
+               data-component="header">
+    <br>
+<p>Example: <code>application/json</code></p>
+            </div>
+                                <h4 class="fancy-heading-panel"><b>Body Parameters</b></h4>
+        <div style=" padding-left: 28px;  clear: unset;">
+            <b style="line-height: 2;"><code>amount</code></b>&nbsp;&nbsp;
+<small>decimal</small>&nbsp;
+<i>optional</i> &nbsp;
+                <input type="text" style="display: none"
+                              name="amount"                data-endpoint="POSTapi-user-withdraw"
+               value="100.00"
+               data-component="body">
+    <br>
+<p>提现金额. Example: <code>100.00</code></p>
+        </div>
+        </form>
+
+                    <h2 id="endpoints-POSTapi-user-feedback">[用户管理] 用户反馈</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>提交用户的反馈信息</p>
+
+<span id="example-requests-POSTapi-user-feedback">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request POST \
+    "http://192.168.110.10/api/user/feedback" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743" \
+    --header "Content-Type: application/json" \
+    --data "{
+    \"content\": \"这是一个反馈信息\"
+}"
+</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/user/feedback"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Content-Type": "application/json",
+    "Accept": "application/json",
+};
+
+let body = {
+    "content": "这是一个反馈信息"
+};
+
+fetch(url, {
+    method: "POST",
+    headers,
+    body: JSON.stringify(body),
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/user/feedback';
+$response = $client-&gt;post(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+            'Content-Type' =&gt; 'application/json',
+        ],
+        'json' =&gt; [
+            'content' =&gt; '这是一个反馈信息',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-POSTapi-user-feedback">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;提交成功&quot;,
+    &quot;data&quot;: null
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-POSTapi-user-feedback" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-POSTapi-user-feedback"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-POSTapi-user-feedback"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-POSTapi-user-feedback" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-POSTapi-user-feedback">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-POSTapi-user-feedback" data-method="POST"
+      data-path="api/user/feedback"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('POSTapi-user-feedback', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-POSTapi-user-feedback"
+                    onclick="tryItOut('POSTapi-user-feedback');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-POSTapi-user-feedback"
+                    onclick="cancelTryOut('POSTapi-user-feedback');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-POSTapi-user-feedback"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-black">POST</small>
+            <b><code>api/user/feedback</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="POSTapi-user-feedback"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Content-Type</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Content-Type"                data-endpoint="POSTapi-user-feedback"
+               value="application/json"
+               data-component="header">
+    <br>
+<p>Example: <code>application/json</code></p>
+            </div>
+                                <h4 class="fancy-heading-panel"><b>Body Parameters</b></h4>
+        <div style=" padding-left: 28px;  clear: unset;">
+            <b style="line-height: 2;"><code>content</code></b>&nbsp;&nbsp;
+<small>string</small>&nbsp;
+<i>optional</i> &nbsp;
+                <input type="text" style="display: none"
+                              name="content"                data-endpoint="POSTapi-user-feedback"
+               value="这是一个反馈信息"
+               data-component="body">
+    <br>
+<p>反馈内容. Example: <code>这是一个反馈信息</code></p>
+        </div>
+        </form>
+
+                    <h2 id="endpoints-POSTapi-user-apply-coach">[用户管理] 申请成为技师</h2>
+
+<p>
+<small class="badge badge-darkred">requires authentication</small>
+</p>
+
+<p>申请成为技师</p>
+
+<span id="example-requests-POSTapi-user-apply-coach">
+<blockquote>Example request:</blockquote>
+
+
+<div class="bash-example">
+    <pre><code class="language-bash">curl --request POST \
+    "http://192.168.110.10/api/user/apply-coach" \
+    --header "Authorization: Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"</code></pre></div>
+
+
+<div class="javascript-example">
+    <pre><code class="language-javascript">const url = new URL(
+    "http://192.168.110.10/api/user/apply-coach"
+);
+
+const headers = {
+    "Authorization": "Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743",
+    "Accept": "application/json",
+};
+
+fetch(url, {
+    method: "POST",
+    headers,
+}).then(response =&gt; response.json());</code></pre></div>
+
+
+<div class="php-example">
+    <pre><code class="language-php">$client = new \GuzzleHttp\Client();
+$url = 'http://192.168.110.10/api/user/apply-coach';
+$response = $client-&gt;post(
+    $url,
+    [
+        'headers' =&gt; [
+            'Authorization' =&gt; 'Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743',
+        ],
+    ]
+);
+$body = $response-&gt;getBody();
+print_r(json_decode((string) $body));</code></pre></div>
+
+</span>
+
+<span id="example-responses-POSTapi-user-apply-coach">
+            <blockquote>
+            <p>Example response (200):</p>
+        </blockquote>
+                <pre>
+
+<code class="language-json" style="max-height: 300px;">{
+    &quot;code&quot;: 200,
+    &quot;message&quot;: &quot;申请成功&quot;,
+    &quot;data&quot;: null
+}</code>
+ </pre>
+    </span>
+<span id="execution-results-POSTapi-user-apply-coach" hidden>
+    <blockquote>Received response<span
+                id="execution-response-status-POSTapi-user-apply-coach"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-POSTapi-user-apply-coach"
+      data-empty-response-text="<Empty response>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-POSTapi-user-apply-coach" hidden>
+    <blockquote>Request failed with error:</blockquote>
+    <pre><code id="execution-error-message-POSTapi-user-apply-coach">
+
+Tip: Check that you&#039;re properly connected to the network.
+If you&#039;re a maintainer of ths API, verify that your API is running and you&#039;ve enabled CORS.
+You can check the Dev Tools console for debugging information.</code></pre>
+</span>
+<form id="form-POSTapi-user-apply-coach" data-method="POST"
+      data-path="api/user/apply-coach"
+      data-authed="1"
+      data-hasfiles="0"
+      data-isarraybody="0"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('POSTapi-user-apply-coach', this);">
+    <h3>
+        Request&nbsp;&nbsp;&nbsp;
+                    <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-POSTapi-user-apply-coach"
+                    onclick="tryItOut('POSTapi-user-apply-coach');">Try it out ⚡
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-POSTapi-user-apply-coach"
+                    onclick="cancelTryOut('POSTapi-user-apply-coach');" hidden>Cancel 🛑
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-POSTapi-user-apply-coach"
+                    data-initial-text="Send Request 💥"
+                    data-loading-text="⏱ Sending..."
+                    hidden>Send Request 💥
+            </button>
+            </h3>
+            <p>
+            <small class="badge badge-black">POST</small>
+            <b><code>api/user/apply-coach</code></b>
+        </p>
+                <h4 class="fancy-heading-panel"><b>Headers</b></h4>
+                                <div style="padding-left: 28px; clear: unset;">
+                <b style="line-height: 2;"><code>Authorization</code></b>&nbsp;&nbsp;
+&nbsp;
+ &nbsp;
+                <input type="text" style="display: none"
+                              name="Authorization" class="auth-value"               data-endpoint="POSTapi-user-apply-coach"
+               value="Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743"
+               data-component="header">
+    <br>
+<p>Example: <code>Bearer 69|KO7sJGJlD0AxUDOxnKf5uEs4AnvEJKeRQG527cns36375743</code></p>
+            </div>
+                        </form>
+
+            
+
+        
+    </div>
+    <div class="dark-box">
+                    <div class="lang-selector">
+                                                        <button type="button" class="lang-button" data-language-name="bash">bash</button>
+                                                        <button type="button" class="lang-button" data-language-name="javascript">javascript</button>
+                                                        <button type="button" class="lang-button" data-language-name="php">php</button>
+                            </div>
+            </div>
+</div>
+</body>
+</html>

+ 149 - 0
public/docs/js/theme-default-4.38.0.js

@@ -0,0 +1,149 @@
+document.addEventListener('DOMContentLoaded', function() {
+    const updateHash = function (id) {
+        window.location.hash = `#${id}`;
+    };
+
+    const navButton = document.getElementById('nav-button');
+    const menuWrapper = document.querySelector('.tocify-wrapper');
+    function toggleSidebar(event) {
+        event.preventDefault();
+        if (menuWrapper) {
+            menuWrapper.classList.toggle('open');
+            navButton.classList.toggle('open');
+        }
+    }
+    function closeSidebar() {
+        if (menuWrapper) {
+            menuWrapper.classList.remove('open');
+            navButton.classList.remove('open');
+        }
+    }
+    navButton.addEventListener('click', toggleSidebar);
+
+    window.hljs.highlightAll();
+
+    const wrapper = document.getElementById('toc');
+    // https://jets.js.org/
+    window.jets = new window.Jets({
+        // *OR - Selects elements whose values contains at least one part of search substring
+        searchSelector: '*OR',
+        searchTag: '#input-search',
+        contentTag: '#toc li',
+        didSearch: function(term) {
+            wrapper.classList.toggle('jets-searching', String(term).length > 0)
+        },
+        // map these accent keys to plain values
+        diacriticsMap: {
+            a: 'ÀÁÂÃÄÅàáâãäåĀāąĄ',
+            c: 'ÇçćĆčČ',
+            d: 'đĐďĎ',
+            e: 'ÈÉÊËèéêëěĚĒēęĘ',
+            i: 'ÌÍÎÏìíîïĪī',
+            l: 'łŁ',
+            n: 'ÑñňŇńŃ',
+            o: 'ÒÓÔÕÕÖØòóôõöøŌō',
+            r: 'řŘ',
+            s: 'ŠšśŚ',
+            t: 'ťŤ',
+            u: 'ÙÚÛÜùúûüůŮŪū',
+            y: 'ŸÿýÝ',
+            z: 'ŽžżŻźŹ'
+        }
+    });
+
+    function hashChange() {
+        const currentItems = document.querySelectorAll('.tocify-subheader.visible, .tocify-item.tocify-focus');
+        Array.from(currentItems).forEach((elem) => {
+            elem.classList.remove('visible', 'tocify-focus');
+        });
+
+        const currentTag = document.querySelector(`a[href="${window.location.hash}"]`);
+        if (currentTag) {
+            const parent = currentTag.closest('.tocify-subheader');
+            if (parent) {
+                parent.classList.add('visible');
+            }
+
+            const siblings = currentTag.closest('.tocify-header');
+            if (siblings) {
+                Array.from(siblings.querySelectorAll('.tocify-subheader')).forEach((elem) => {
+                    elem.classList.add('visible');
+                });
+            }
+
+            currentTag.parentElement.classList.add('tocify-focus');
+
+            // wait for dom changes to be done
+            setTimeout(() => {
+                currentTag.scrollIntoView({ behavior: 'smooth', block: 'center', inline: 'center' });
+                // only close the sidebar on level-2 events
+                if (currentTag.parentElement.classList.contains('level-2')) {
+                    closeSidebar();
+                }
+            }, 1500);
+        }
+    }
+
+    let languages = JSON.parse(document.body.getAttribute('data-languages'));
+    // Support a key => value object where the key is the name, or an array of strings where the value is the name
+    if (!Array.isArray(languages)) {
+        languages = Object.values(languages);
+    }
+    // if there is no language use the first one
+    const currentLanguage = window.localStorage.getItem('language') || languages[0];
+    const languageStyle = document.getElementById('language-style');
+    const langSelector = document.querySelectorAll('.lang-selector button.lang-button');
+
+    function setActiveLanguage(newLanguage) {
+        window.localStorage.setItem('language', newLanguage);
+        if (!languageStyle) {
+            return;
+        }
+
+        const newStyle = languages.map((language) => {
+            return language === newLanguage
+                // the current one should be visible
+                ? `body .content .${language}-example pre { display: block; }`
+                // the inactive one should be hidden
+                : `body .content .${language}-example pre { display: none; }`;
+        }).join(`\n`);
+
+        Array.from(langSelector).forEach((elem) => {
+            elem.classList.toggle('active', elem.getAttribute('data-language-name') === newLanguage);
+        });
+
+        const activeHash = window.location.hash.slice(1);
+
+        languageStyle.innerHTML = newStyle;
+
+        setTimeout(() => {
+            updateHash(activeHash);
+        }, 200);
+    }
+
+    setActiveLanguage(currentLanguage);
+
+    Array.from(langSelector).forEach((elem) => {
+        elem.addEventListener('click', () => {
+            const newLanguage = elem.getAttribute('data-language-name');
+            setActiveLanguage(newLanguage);
+        });
+    });
+
+    window.addEventListener('hashchange', hashChange, false);
+
+    const divs = document.querySelectorAll('.content h1[id], .content h2[id]');
+
+    document.addEventListener('scroll', () => {
+        divs.forEach(item => {
+            const rect = item.getBoundingClientRect();
+            if (rect.top > 0 && rect.top < 150) {
+                const location = window.location.toString().split('#')[0];
+                history.replaceState(null, null, location + '#' + item.id);
+                hashChange();
+            }
+        });
+    });
+
+    hashChange();
+});

+ 289 - 0
public/docs/js/tryitout-4.38.0.js

@@ -0,0 +1,289 @@
+window.abortControllers = {};
+
+function cacheAuthValue() {
+    // Whenever the auth header is set for one endpoint, cache it for the others
+    window.lastAuthValue = '';
+    let authInputs = document.querySelectorAll(`.auth-value`)
+    authInputs.forEach(el => {
+        el.addEventListener('input', (event) => {
+            window.lastAuthValue = event.target.value;
+            authInputs.forEach(otherInput => {
+                if (otherInput === el) return;
+                // Don't block the main thread
+                setTimeout(() => {
+                    otherInput.value = window.lastAuthValue;
+                }, 0);
+            });
+        });
+    });
+}
+
+window.addEventListener('DOMContentLoaded', cacheAuthValue);
+
+function getCookie(name) {
+    if (!document.cookie) {
+        return null;
+    }
+
+    const cookies = document.cookie.split(';')
+        .map(c => c.trim())
+        .filter(c => c.startsWith(name + '='));
+
+    if (cookies.length === 0) {
+        return null;
+    }
+
+    return decodeURIComponent(cookies[0].split('=')[1]);
+}
+
+function tryItOut(endpointId) {
+    document.querySelector(`#btn-tryout-${endpointId}`).hidden = true;
+    document.querySelector(`#btn-canceltryout-${endpointId}`).hidden = false;
+    const executeBtn = document.querySelector(`#btn-executetryout-${endpointId}`).hidden = false;
+    executeBtn.disabled = false;
+
+    // Show all input fields
+    document.querySelectorAll(`input[data-endpoint=${endpointId}],label[data-endpoint=${endpointId}]`)
+        .forEach(el => el.style.display = 'block');
+
+    if (document.querySelector(`#form-${endpointId}`).dataset.authed === "1") {
+        const authElement = document.querySelector(`#auth-${endpointId}`);
+        authElement && (authElement.hidden = false);
+    }
+    // Expand all nested fields
+    document.querySelectorAll(`#form-${endpointId} details`)
+        .forEach(el => el.open = true);
+}
+
+function cancelTryOut(endpointId) {
+    if (window.abortControllers[endpointId]) {
+        window.abortControllers[endpointId].abort();
+        delete window.abortControllers[endpointId];
+    }
+
+    document.querySelector(`#btn-tryout-${endpointId}`).hidden = false;
+    const executeBtn = document.querySelector(`#btn-executetryout-${endpointId}`);
+    executeBtn.hidden = true;
+    executeBtn.textContent = executeBtn.dataset.initialText;
+    document.querySelector(`#btn-canceltryout-${endpointId}`).hidden = true;
+    // Hide inputs
+    document.querySelectorAll(`input[data-endpoint=${endpointId}],label[data-endpoint=${endpointId}]`)
+        .forEach(el => el.style.display = 'none');
+    document.querySelectorAll(`#form-${endpointId} details`)
+        .forEach(el => el.open = false);
+    const authElement = document.querySelector(`#auth-${endpointId}`);
+    authElement && (authElement.hidden = true);
+
+    document.querySelector('#execution-results-' + endpointId).hidden = true;
+    document.querySelector('#execution-error-' + endpointId).hidden = true;
+
+    // Revert to sample code blocks
+    document.querySelector('#example-requests-' + endpointId).hidden = false;
+    document.querySelector('#example-responses-' + endpointId).hidden = false;
+}
+
+function makeAPICall(method, path, body = {}, query = {}, headers = {}, endpointId = null) {
+    console.log({endpointId, path, body, query, headers});
+
+    if (!(body instanceof FormData) && typeof body !== "string") {
+        body = JSON.stringify(body)
+    }
+
+    const url = new URL(window.tryItOutBaseUrl + '/' + path.replace(/^\//, ''));
+
+    // We need this function because if you try to set an array or object directly to a URLSearchParams object,
+    // you'll get [object Object] or the array.toString()
+    function addItemToSearchParamsObject(key, value, searchParams) {
+            if (Array.isArray(value)) {
+                value.forEach((v, i) => {
+                    // Append {filters: [first, second]} as filters[0]=first&filters[1]second
+                    addItemToSearchParamsObject(key + '[' + i + ']', v, searchParams);
+                })
+            } else if (typeof value === 'object' && value !== null) {
+                Object.keys(value).forEach((i) => {
+                    // Append {filters: {name: first}} as filters[name]=first
+                    addItemToSearchParamsObject(key + '[' + i + ']', value[i], searchParams);
+                });
+            } else {
+                searchParams.append(key, value);
+            }
+    }
+
+    Object.keys(query)
+        .forEach(key => addItemToSearchParamsObject(key, query[key], url.searchParams));
+
+    window.abortControllers[endpointId] = new AbortController();
+
+    return fetch(url, {
+        method,
+        headers,
+        body: method === 'GET' ? undefined : body,
+        signal: window.abortControllers[endpointId].signal,
+        referrer: window.tryItOutBaseUrl,
+        mode: 'cors',
+        credentials: 'same-origin',
+    })
+        .then(response => Promise.all([response.status, response.statusText, response.text(), response.headers]));
+}
+
+function hideCodeSamples(endpointId) {
+    document.querySelector('#example-requests-' + endpointId).hidden = true;
+    document.querySelector('#example-responses-' + endpointId).hidden = true;
+}
+
+function handleResponse(endpointId, response, status, headers) {
+    hideCodeSamples(endpointId);
+
+    // Hide error views
+    document.querySelector('#execution-error-' + endpointId).hidden = true;
+
+    const responseContentEl = document.querySelector('#execution-response-content-' + endpointId);
+
+    // Check if the response contains Laravel's  dd() default dump output
+    const isLaravelDump = response.includes('Sfdump');
+
+    // If it's a Laravel dd() dump, use innerHTML to render it safely
+    if (isLaravelDump) {
+        responseContentEl.innerHTML = response === '' ? responseContentEl.dataset.emptyResponseText : response;
+    } else {
+        // Otherwise, stick to textContent for regular responses
+        responseContentEl.textContent = response === '' ? responseContentEl.dataset.emptyResponseText : response;
+    }
+
+    // Prettify it if it's JSON
+    let isJson = false;
+    try {
+        const jsonParsed = JSON.parse(response);
+        if (jsonParsed !== null) {
+            isJson = true;
+            response = JSON.stringify(jsonParsed, null, 4);
+            responseContentEl.textContent = response;
+        }
+    } catch (e) {
+
+    }
+
+    isJson && window.hljs.highlightElement(responseContentEl);
+    const statusEl = document.querySelector('#execution-response-status-' + endpointId);
+    statusEl.textContent = ` (${status})`;
+    document.querySelector('#execution-results-' + endpointId).hidden = false;
+    statusEl.scrollIntoView({behavior: "smooth", block: "center"});
+}
+
+function handleError(endpointId, err) {
+    hideCodeSamples(endpointId);
+    // Hide response views
+    document.querySelector('#execution-results-' + endpointId).hidden = true;
+
+    // Show error views
+    let errorMessage = err.message || err;
+    const $errorMessageEl = document.querySelector('#execution-error-message-' + endpointId);
+    $errorMessageEl.textContent = errorMessage + $errorMessageEl.textContent;
+    const errorEl = document.querySelector('#execution-error-' + endpointId);
+    errorEl.hidden = false;
+    errorEl.scrollIntoView({behavior: "smooth", block: "center"});
+
+}
+
+async function executeTryOut(endpointId, form) {
+    const executeBtn = document.querySelector(`#btn-executetryout-${endpointId}`);
+    executeBtn.textContent = executeBtn.dataset.loadingText;
+    executeBtn.disabled = true;
+    executeBtn.scrollIntoView({behavior: "smooth", block: "center"});
+
+    let body;
+    let setter;
+    if (form.dataset.hasfiles === "1") {
+        body = new FormData();
+        setter = (name, value) => body.append(name, value);
+    } else if (form.dataset.isarraybody === "1") {
+        body = [];
+        setter = (name, value) => _.set(body, name, value);
+    } else {
+        body = {};
+        setter = (name, value) => _.set(body, name, value);
+    }
+    const bodyParameters = form.querySelectorAll('input[data-component=body]');
+    bodyParameters.forEach(el => {
+        let value = el.value;
+
+        if (el.type === 'number' && typeof value === 'string') {
+            value = parseFloat(value);
+        }
+
+        if (el.type === 'file' && el.files[0]) {
+            setter(el.name, el.files[0]);
+            return;
+        }
+
+        if (el.type !== 'radio') {
+            if (value === "" && el.required === false) {
+                // Don't include empty optional values in the request
+                return;
+            }
+            setter(el.name, value);
+            return;
+        }
+
+        if (el.checked) {
+            value = (value === 'false') ? false : true;
+            setter(el.name, value);
+        }
+    });
+
+    const query = {};
+    const queryParameters = form.querySelectorAll('input[data-component=query]');
+    queryParameters.forEach(el => {
+        if (el.type !== 'radio' || (el.type === 'radio' && el.checked)) {
+            if (el.value === '') {
+                // Don't include empty values in the request
+                return;
+            }
+
+            _.set(query, el.name, el.value);
+        }
+    });
+
+    let path = form.dataset.path;
+    const urlParameters = form.querySelectorAll('input[data-component=url]');
+    urlParameters.forEach(el => (path = path.replace(new RegExp(`\\{${el.name}\\??}`), el.value)));
+
+    const headers = Object.fromEntries(Array.from(form.querySelectorAll('input[data-component=header]'))
+        .map(el => [el.name, el.value]));
+
+    // When using FormData, the browser sets the correct content-type + boundary
+    let method = form.dataset.method;
+    if (body instanceof FormData) {
+        delete headers['Content-Type'];
+
+        // When using FormData with PUT or PATCH, use method spoofing so PHP can access the post body
+        if (['PUT', 'PATCH'].includes(form.dataset.method)) {
+            method = 'POST';
+            setter('_method', form.dataset.method);
+        }
+    }
+
+    let preflightPromise = Promise.resolve();
+    if (window.useCsrf && window.csrfUrl) {
+        preflightPromise = makeAPICall('GET', window.csrfUrl).then(() => {
+            headers['X-XSRF-TOKEN'] = getCookie('XSRF-TOKEN');
+        });
+    }
+
+    return preflightPromise.then(() => makeAPICall(method, path, body, query, headers, endpointId))
+        .then(([responseStatus, statusText, responseContent, responseHeaders]) => {
+            handleResponse(endpointId, responseContent, responseStatus, responseHeaders)
+        })
+        .catch(err => {
+            if (err.name === "AbortError") {
+                console.log("Request cancelled");
+                return;
+            }
+            console.log("Error while making request: ", err);
+            handleError(endpointId, err);
+        })
+        .finally(() => {
+            executeBtn.disabled = false;
+            executeBtn.textContent = executeBtn.dataset.initialText;
+        });
+}

+ 408 - 0
public/docs/openapi.yaml

@@ -0,0 +1,408 @@
+openapi: 3.0.3
+info:
+  title: 'Owl Admin API Documentation'
+  description: 'Owl Admin API 接口文档'
+  version: 1.0.0
+servers:
+  -
+    url: 'http://192.168.110.10'
+paths:
+  '/api/scribe/update-token/{mobile}':
+    get:
+      summary: 更新API文档的认证Token
+      operationId: APIToken
+      description: 通过手机号获取验证码并登录,然后更新API文档的认证Token
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: Token更新成功
+                  data:
+                    token: your_new_token_here
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: Token更新成功
+                  data:
+                    type: object
+                    properties:
+                      token:
+                        type: string
+                        example: your_new_token_here
+      tags:
+        - Endpoints
+    parameters:
+      -
+        in: path
+        name: mobile
+        description: 手机号码.
+        example: '13800138000'
+        required: true
+        schema:
+          type: string
+  /api/account/send-code:
+    post:
+      summary: '[账户管理] 发送验证码'
+      operationId: ''
+      description: 向指定手机号发送验证码
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: 验证码发送成功
+                  data: null
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: 验证码发送成功
+                  data:
+                    type: string
+                    example: null
+      tags:
+        - Endpoints
+      requestBody:
+        required: true
+        content:
+          application/json:
+            schema:
+              type: object
+              properties:
+                mobile:
+                  type: string
+                  description: 手机号码.
+                  example: '13800138000'
+                  nullable: false
+              required:
+                - mobile
+  /api/account/login:
+    post:
+      summary: '[账户管理] 用户登录'
+      operationId: ''
+      description: 使用手机号和验证码登录账户
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: 登录成功
+                  data:
+                    token: eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...
+                    user:
+                      id: 1
+                      mobile: '13800138000'
+                      nickname: 用户昵称
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: 登录成功
+                  data:
+                    type: object
+                    properties:
+                      token:
+                        type: string
+                        example: eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...
+                      user:
+                        type: object
+                        properties:
+                          id:
+                            type: integer
+                            example: 1
+                          mobile:
+                            type: string
+                            example: '13800138000'
+                          nickname:
+                            type: string
+                            example: 用户昵称
+      tags:
+        - Endpoints
+      requestBody:
+        required: true
+        content:
+          application/json:
+            schema:
+              type: object
+              properties:
+                mobile:
+                  type: string
+                  description: 手机号码.
+                  example: '13800138000'
+                  nullable: false
+                code:
+                  type: string
+                  description: 验证码.
+                  example: '123456'
+                  nullable: false
+              required:
+                - mobile
+                - code
+  /api/account/wx-login:
+    post:
+      summary: '[账户管理] 微信登录'
+      operationId: ''
+      description: 使用微信openid登录账户
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: 登录成功
+                  data:
+                    token: eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...
+                    user:
+                      id: 1
+                      openid: wx_123456789
+                      nickname: 微信昵称
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: 登录成功
+                  data:
+                    type: object
+                    properties:
+                      token:
+                        type: string
+                        example: eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...
+                      user:
+                        type: object
+                        properties:
+                          id:
+                            type: integer
+                            example: 1
+                          openid:
+                            type: string
+                            example: wx_123456789
+                          nickname:
+                            type: string
+                            example: 微信昵称
+      tags:
+        - Endpoints
+      requestBody:
+        required: true
+        content:
+          application/json:
+            schema:
+              type: object
+              properties:
+                openid:
+                  type: string
+                  description: 微信openid.
+                  example: wx_123456789
+                  nullable: false
+              required:
+                - openid
+  /api/account/logout:
+    post:
+      summary: '[账户管理] 用户退出'
+      operationId: ''
+      description: 退出当前账户登录状态
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: 退出成功
+                  data: null
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: 退出成功
+                  data:
+                    type: string
+                    example: null
+      tags:
+        - Endpoints
+  /api/account:
+    delete:
+      summary: '[账户管理] 用户注销'
+      operationId: ''
+      description: 永久注销当前账户
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: 注销成功
+                  data: null
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: 注销成功
+                  data:
+                    type: string
+                    example: null
+      tags:
+        - Endpoints
+  /api/user/withdraw:
+    post:
+      summary: '[用户管理] 用户提现'
+      operationId: ''
+      description: 提现用户的余额
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: 提现成功
+                  data: null
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: 提现成功
+                  data:
+                    type: string
+                    example: null
+      tags:
+        - Endpoints
+      requestBody:
+        required: false
+        content:
+          application/json:
+            schema:
+              type: object
+              properties:
+                amount:
+                  type: decimal
+                  description: 提现金额.
+                  example: '100.00'
+                  nullable: false
+  /api/user/feedback:
+    post:
+      summary: '[用户管理] 用户反馈'
+      operationId: ''
+      description: 提交用户的反馈信息
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: 提交成功
+                  data: null
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: 提交成功
+                  data:
+                    type: string
+                    example: null
+      tags:
+        - Endpoints
+      requestBody:
+        required: false
+        content:
+          application/json:
+            schema:
+              type: object
+              properties:
+                content:
+                  type: string
+                  description: 反馈内容.
+                  example: 这是一个反馈信息
+                  nullable: false
+  /api/user/apply-coach:
+    post:
+      summary: '[用户管理] 申请成为技师'
+      operationId: ''
+      description: 申请成为技师
+      parameters: []
+      responses:
+        200:
+          description: ''
+          content:
+            application/json:
+              schema:
+                type: object
+                example:
+                  code: 200
+                  message: 申请成功
+                  data: null
+                properties:
+                  code:
+                    type: integer
+                    example: 200
+                  message:
+                    type: string
+                    example: 申请成功
+                  data:
+                    type: string
+                    example: null
+      tags:
+        - Endpoints
+tags:
+  -
+    name: Endpoints
+    description: ''
+components:
+  securitySchemes:
+    default:
+      type: http
+      scheme: bearer
+      description: '通过登录接口获取 token,格式为 Bearer {token}'
+security:
+  -
+    default: []

+ 3 - 0
resources/views/vendor/scribe/components/badges/auth.blade.php

@@ -0,0 +1,3 @@
+@if($authenticated)@component('scribe::components.badges.base', ['colour' => "darkred", 'text' => 'requires authentication'])
+@endcomponent
+@endif

+ 1 - 0
resources/views/vendor/scribe/components/badges/base.blade.php

@@ -0,0 +1 @@
+<small class="badge badge-{{ $colour }}">{{ $text }}</small>

+ 5 - 0
resources/views/vendor/scribe/components/badges/http-method.blade.php

@@ -0,0 +1,5 @@
+@component('scribe::components.badges.base', [
+    'colour' => \Knuckles\Scribe\Tools\WritingUtils::$httpMethodToCssColour[$method],
+    'text' => $method,
+    ])
+@endcomponent

+ 75 - 0
resources/views/vendor/scribe/components/field-details.blade.php

@@ -0,0 +1,75 @@
+@php
+    $html ??= []; $class = $html['class'] ?? null;
+@endphp
+<b style="line-height: 2;"><code>{{ $name }}</code></b>&nbsp;&nbsp;
+@if($type)<small>{{ $type }}</small>@endif&nbsp;
+@if($isInput && !$required)<i>optional</i>@endif &nbsp;
+@if($isInput && empty($hasChildren))
+    @php
+        $isList = Str::endsWith($type, '[]');
+        $fullName = str_replace('[]', '.0', $fullName ?? $name);
+        $baseType = $isList ? substr($type, 0, -2) : $type;
+        // Ignore the first '[]': the frontend will take care of it
+        while (\Str::endsWith($baseType, '[]')) {
+            $fullName .= '.0';
+            $baseType = substr($baseType, 0, -2);
+        }
+        // When the body is an array, the item names will be ".0.thing"
+        $fullName = ltrim($fullName, '.');
+        $inputType = match($baseType) {
+            'number', 'integer' => 'number',
+            'file' => 'file',
+            default => 'text',
+        };
+    @endphp
+    @if($type === 'boolean')
+        <label data-endpoint="{{ $endpointId }}" style="display: none">
+            <input type="radio" name="{{ $fullName }}"
+                   value="{{$component === 'body' ? 'true' : 1}}"
+                   data-endpoint="{{ $endpointId }}"
+                   data-component="{{ $component }}" @if($class)class="{{ $class }}"@endif
+            >
+            <code>true</code>
+        </label>
+        <label data-endpoint="{{ $endpointId }}" style="display: none">
+            <input type="radio" name="{{ $fullName }}"
+                   value="{{$component === 'body' ? 'false' : 0}}"
+                   data-endpoint="{{ $endpointId }}"
+                   data-component="{{ $component }}" @if($class)class="{{ $class }}"@endif
+            >
+            <code>false</code>
+        </label>
+    @elseif($isList)
+        <input type="{{ $inputType }}" style="display: none"
+               @if($inputType === 'number')step="any"@endif
+               name="{{ $fullName."[0]" }}" @if($class)class="{{ $class }}"@endif
+               data-endpoint="{{ $endpointId }}"
+               data-component="{{ $component }}">
+        <input type="{{ $inputType }}" style="display: none"
+               name="{{ $fullName."[1]" }}" @if($class)class="{{ $class }}"@endif
+               data-endpoint="{{ $endpointId }}"
+               data-component="{{ $component }}">
+    @else
+        <input type="{{ $inputType }}" style="display: none"
+               @if($inputType === 'number')step="any"@endif
+               name="{{ $fullName }}" @if($class)class="{{ $class }}"@endif
+               data-endpoint="{{ $endpointId }}"
+               value="{!! (isset($example) && (is_string($example) || is_numeric($example))) ? $example : '' !!}"
+               data-component="{{ $component }}">
+    @endif
+@endif
+<br>
+@php
+    if($example !== null && $example !== '' && !is_array($example)) {
+        $exampleAsString = $example;
+        if (is_bool($example)) {
+            $exampleAsString = $example ? "true" : "false";
+        }
+        $description .= " Example: `$exampleAsString`";
+    }
+@endphp
+{!! Parsedown::instance()->text(trim($description)) !!}
+@if(!empty($enumValues))
+Must be one of:
+<ul style="list-style-type: square;">{!! implode(" ", array_map(fn($val) => "<li><code>$val</code></li>", $enumValues)) !!}</ul>
+@endif

+ 101 - 0
resources/views/vendor/scribe/components/nested-fields.blade.php

@@ -0,0 +1,101 @@
+@php
+    $isInput ??= true;
+    $level ??= 0;
+@endphp
+@foreach($fields as $name => $field)
+    @if($name === '[]')
+        @php
+            $description = "The request body is an array (<code>{$field['type']}</code>`)";
+            $description .= !empty($field['description']) ? ", representing ".lcfirst($field['description'])."." : '.';
+            if(count($field['__fields'])) $description .= " Each item has the following properties:";
+        @endphp
+        {!! Parsedown::instance()->text($description) !!}
+
+        @foreach($field['__fields'] as $subfieldName => $subfield)
+                @if(!empty($subfield['__fields']))
+                    <x-scribe::nested-fields
+                            :fields="[$subfieldName => $subfield]" :endpointId="$endpointId" :isInput="$isInput" :level="$level + 2"
+                    />
+                @else
+                    <div style="margin-left: {{ ($level + 2) * 14 }}px; clear: unset;">
+                        @component('scribe::components.field-details', [
+                          'name' => $subfieldName,
+                          'fullName' => $subfield['name'],
+                          'type' => $subfield['type'] ?? 'string',
+                          'required' => $subfield['required'] ?? false,
+                          'description' => $subfield['description'] ?? '',
+                          'example' => $subfield['example'] ?? '',
+                          'enumValues' => $subfield['enumValues'] ?? null,
+                          'endpointId' => $endpointId,
+                          'hasChildren' => false,
+                          'component' => 'body',
+                          'isInput' => $isInput,
+                        ])
+                        @endcomponent
+                    </div>
+                @endif
+            @endforeach
+    @elseif(!empty($field['__fields']))
+        <div style="@if($level) margin-left: {{ $level * 14 }}px;@else padding-left: 28px; @endif clear: unset;">
+        <details>
+            <summary style="padding-bottom: 10px;">
+                @component('scribe::components.field-details', [
+                  'name' => $name,
+                  'fullName' => $field['name'],
+                  'type' => $field['type'] ?? 'string',
+                  'required' => $field['required'] ?? false,
+                  'description' => $field['description'] ?? '',
+                  'example' => $field['example'] ?? '',
+                  'enumValues' => $field['enumValues'] ?? null,
+                  'endpointId' => $endpointId,
+                  'hasChildren' => true,
+                  'component' => 'body',
+                  'isInput' => $isInput,
+                ])
+                @endcomponent
+            </summary>
+            @foreach($field['__fields'] as $subfieldName => $subfield)
+                @if(!empty($subfield['__fields']))
+                    <x-scribe::nested-fields
+                            :fields="[$subfieldName => $subfield]" :endpointId="$endpointId" :isInput="$isInput" :level="$level + 1"
+                    />
+                @else
+                    <div style="margin-left: {{ ($level + 1) * 14 }}px; clear: unset;">
+                        @component('scribe::components.field-details', [
+                          'name' => $subfieldName,
+                          'fullName' => $subfield['name'],
+                          'type' => $subfield['type'] ?? 'string',
+                          'required' => $subfield['required'] ?? false,
+                          'description' => $subfield['description'] ?? '',
+                          'example' => $subfield['example'] ?? '',
+                          'enumValues' => $subfield['enumValues'] ?? null,
+                          'endpointId' => $endpointId,
+                          'hasChildren' => false,
+                          'component' => 'body',
+                          'isInput' => $isInput,
+                        ])
+                        @endcomponent
+                    </div>
+                @endif
+            @endforeach
+        </details>
+        </div>
+    @else
+        <div style="@if($level) margin-left: {{ ($level + 1) * 14 }}px;@else padding-left: 28px; @endif clear: unset;">
+            @component('scribe::components.field-details', [
+              'name' => $name,
+              'fullName' => $field['name'],
+              'type' => $field['type'] ?? 'string',
+              'required' => $field['required'] ?? false,
+              'description' => $field['description'] ?? '',
+              'example' => $field['example'] ?? '',
+              'enumValues' => $field['enumValues'] ?? null,
+              'endpointId' => $endpointId,
+              'hasChildren' => false,
+              'component' => 'body',
+              'isInput' => $isInput,
+            ])
+            @endcomponent
+        </div>
+    @endif
+@endforeach

+ 34 - 0
resources/views/vendor/scribe/external/elements.blade.php

@@ -0,0 +1,34 @@
+<!-- See https://github.com/stoplightio/elements/blob/main/docs/getting-started/elements/elements-options.md for config -->
+<!doctype html>
+<html lang="en">
+<head>
+    <meta charset="utf-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
+    <title>{!! $metadata['title'] !!}</title>
+    <!-- Embed elements Elements via Web Component -->
+    <script src="https://unpkg.com/@stoplight/elements/web-components.min.js"></script>
+    <link rel="stylesheet" href="https://unpkg.com/@stoplight/elements/styles.min.css">
+    <style>
+        body {
+            height: 100vh;
+        }
+    </style>
+</head>
+<body>
+
+<elements-api
+@foreach($htmlAttributes as $attribute => $value)
+    {{-- Attributes specified first override later ones --}}
+    {!! $attribute !!}="{!! $value !!}"
+@endforeach
+    apiDescriptionUrl="{!! $metadata['openapi_spec_url'] !!}"
+    router="hash"
+    layout="sidebar"
+    hideTryIt="{!! ($tryItOut['enabled'] ?? true) ? '' : 'true'!!}"
+@if(!empty($metadata['logo']))
+    logo="{!! $metadata['logo'] !!}"
+@endif
+/>
+
+</body>
+</html>

+ 23 - 0
resources/views/vendor/scribe/external/rapidoc.blade.php

@@ -0,0 +1,23 @@
+<!-- See https://rapidocweb.com/api.html for options -->
+<!doctype html> <!-- Important: must specify -->
+<html>
+<head>
+    <meta charset="utf-8"> <!-- Important: rapi-doc uses utf8 characters -->
+    <script type="module" src="https://unpkg.com/rapidoc/dist/rapidoc-min.js"></script>
+</head>
+<body>
+<rapi-doc
+@foreach($htmlAttributes as $attribute => $value)
+    {{-- Attributes specified first override later ones --}}
+    {!! $attribute !!}="{!! $value !!}"
+@endforeach
+    spec-url="{!! $metadata['openapi_spec_url'] !!}"
+    render-style="read"
+    allow-try="{!! ($tryItOut['enabled'] ?? true) ? 'true' : 'false'!!}"
+>
+    @if($metadata['logo'])
+        <img slot="logo" src="{!! $metadata['logo'] !!}"/>
+    @endif
+</rapi-doc>
+</body>
+</html>

+ 27 - 0
resources/views/vendor/scribe/external/scalar.blade.php

@@ -0,0 +1,27 @@
+<!doctype html>
+<html>
+<head>
+    <title>{!! $metadata['title'] !!}</title>
+    <meta charset="utf-8"/>
+    <meta
+        name="viewport"
+        content="width=device-width, initial-scale=1"/>
+    <style>
+        body {
+            margin: 0;
+        }
+    </style>
+</head>
+<body>
+
+<script
+    id="api-reference"
+@foreach($htmlAttributes as $attribute => $value)
+    {{-- Attributes specified first override later ones --}}
+    {!! $attribute !!}="{!! $value !!}"
+@endforeach
+    data-url="{!! $metadata['openapi_spec_url'] !!}">
+</script>
+<script src="https://cdn.jsdelivr.net/npm/@scalar/api-reference"></script>
+</body>
+</html>

+ 12 - 0
resources/views/vendor/scribe/markdown/auth.blade.php

@@ -0,0 +1,12 @@
+@php
+    use Knuckles\Scribe\Tools\Utils as u;
+@endphp
+# {{ u::trans("scribe::headings.auth") }}
+
+@if(!$isAuthed)
+{!! u::trans("scribe::auth.none") !!}
+@else
+{!! $authDescription !!}
+
+{!! $extraAuthInfo !!}
+@endif

+ 13 - 0
resources/views/vendor/scribe/markdown/intro.blade.php

@@ -0,0 +1,13 @@
+@php
+    use Knuckles\Scribe\Tools\Utils as u;
+@endphp
+# {{ u::trans("scribe::headings.introduction") }}
+
+{!! $description !!}
+
+<aside>
+    <strong>{{ u::trans("scribe::labels.base_url") }}</strong>: <code>{!! $baseUrl !!}</code>
+</aside>
+
+{!! $introText !!}
+

+ 34 - 0
resources/views/vendor/scribe/partials/example-requests/bash.md.blade.php

@@ -0,0 +1,34 @@
+@php
+    use Knuckles\Scribe\Tools\WritingUtils as u;
+    /** @var  Knuckles\Camel\Output\OutputEndpointData $endpoint */
+@endphp
+```bash
+curl --request {{$endpoint->httpMethods[0]}} \
+    {{$endpoint->httpMethods[0] == 'GET' ? '--get ' : ''}}"{!! rtrim($baseUrl, '/') !!}/{{ ltrim($endpoint->boundUri, '/') }}@if(count($endpoint->cleanQueryParameters))?{!! u::printQueryParamsAsString($endpoint->cleanQueryParameters) !!}@endif"@if(count($endpoint->headers)) \
+@foreach($endpoint->headers as $header => $value)
+    --header "{{$header}}: {{ addslashes($value) }}"@if(! ($loop->last) || ($loop->last && count($endpoint->bodyParameters))) \
+@endif
+@endforeach
+@endif
+@if($endpoint->hasFiles() || (isset($endpoint->headers['Content-Type']) && $endpoint->headers['Content-Type'] == 'multipart/form-data' && count($endpoint->cleanBodyParameters)))
+@foreach($endpoint->cleanBodyParameters as $parameter => $value)
+@foreach(u::getParameterNamesAndValuesForFormData($parameter, $value) as $key => $actualValue)
+    --form "{!! "$key=".$actualValue !!}"@if(!($loop->parent->last) || count($endpoint->fileParameters))\
+@endif
+@endforeach
+@endforeach
+@foreach($endpoint->fileParameters as $parameter => $value)
+@foreach(u::getParameterNamesAndValuesForFormData($parameter, $value) as $key => $file)
+    --form "{!! "$key=@".$file->path() !!}" @if(!($loop->parent->last))\
+@endif
+@endforeach
+@endforeach
+@elseif(count($endpoint->cleanBodyParameters))
+@if ($endpoint->headers['Content-Type'] == 'application/x-www-form-urlencoded')
+    --data "{!! http_build_query($endpoint->cleanBodyParameters, '', '&') !!}"
+@else
+    --data "{!! addslashes(json_encode($endpoint->cleanBodyParameters, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT)) !!}"
+@endif
+@endif
+
+```

+ 62 - 0
resources/views/vendor/scribe/partials/example-requests/javascript.md.blade.php

@@ -0,0 +1,62 @@
+@php
+    use Knuckles\Scribe\Tools\WritingUtils as u;
+    /** @var  Knuckles\Camel\Output\OutputEndpointData $endpoint */
+@endphp
+```javascript
+const url = new URL(
+    "{!! rtrim($baseUrl, '/') !!}/{{ ltrim($endpoint->boundUri, '/') }}"
+);
+@if(count($endpoint->cleanQueryParameters))
+
+const params = {!! u::printQueryParamsAsKeyValue($endpoint->cleanQueryParameters, "\"", ":", 4, "{}") !!};
+Object.keys(params)
+    .forEach(key => url.searchParams.append(key, params[key]));
+@endif
+
+@if(!empty($endpoint->headers))
+const headers = {
+@foreach($endpoint->headers as $header => $value)
+    "{{$header}}": "{{$value}}",
+@endforeach
+@empty($endpoint->headers['Accept'])
+    "Accept": "application/json",
+@endempty
+};
+@endif
+
+@if($endpoint->hasFiles() || (isset($endpoint->headers['Content-Type']) && $endpoint->headers['Content-Type'] == 'multipart/form-data' && count($endpoint->cleanBodyParameters)))
+const body = new FormData();
+@foreach($endpoint->cleanBodyParameters as $parameter => $value)
+@foreach( u::getParameterNamesAndValuesForFormData($parameter, $value) as $key => $actualValue)
+body.append('{!! $key !!}', '{!! $actualValue !!}');
+@endforeach
+@endforeach
+@foreach($endpoint->fileParameters as $parameter => $value)
+@foreach( u::getParameterNamesAndValuesForFormData($parameter, $value) as $key => $file)
+body.append('{!! $key !!}', document.querySelector('input[name="{!! $key !!}"]').files[0]);
+@endforeach
+@endforeach
+@elseif(count($endpoint->cleanBodyParameters))
+@if ($endpoint->headers['Content-Type'] == 'application/x-www-form-urlencoded')
+let body = "{!! http_build_query($endpoint->cleanBodyParameters, '', '&') !!}";
+@else
+let body = {!! json_encode($endpoint->cleanBodyParameters, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE) !!};
+@endif
+@endif
+
+fetch(url, {
+    method: "{{$endpoint->httpMethods[0]}}",
+@if(count($endpoint->headers))
+    headers,
+@endif
+@if($endpoint->hasFiles() || (isset($endpoint->headers['Content-Type']) && $endpoint->headers['Content-Type'] == 'multipart/form-data' && count($endpoint->cleanBodyParameters)))
+    body,
+@elseif(count($endpoint->cleanBodyParameters))
+@if ($endpoint->headers['Content-Type'] == 'application/x-www-form-urlencoded')
+    body,
+@else
+    body: JSON.stringify(body),
+@endif
+@endif
+}).then(response => response.json());
+```

+ 51 - 0
resources/views/vendor/scribe/partials/example-requests/php.md.blade.php

@@ -0,0 +1,51 @@
+@php
+    use Knuckles\Scribe\Tools\WritingUtils as u;
+    /** @var  Knuckles\Camel\Output\OutputEndpointData $endpoint */
+@endphp
+```php
+$client = new \GuzzleHttp\Client();
+$url = '{!! rtrim($baseUrl, '/') . '/' . ltrim($endpoint->boundUri, '/') !!}';
+@if($endpoint->hasHeadersOrQueryOrBodyParams())
+$response = $client->{{ strtolower($endpoint->httpMethods[0]) }}(
+    $url,
+    [
+@if(!empty($endpoint->headers))
+        'headers' => {!! u::printPhpValue($endpoint->headers, 8) !!},
+@endif
+@if(!empty($endpoint->cleanQueryParameters))
+        'query' => {!! u::printQueryParamsAsKeyValue($endpoint->cleanQueryParameters, "'", " =>", 12, "[]", 8) !!},
+@endif
+@if($endpoint->hasFiles() || (isset($endpoint->headers['Content-Type']) && $endpoint->headers['Content-Type'] == 'multipart/form-data' && !empty($endpoint->cleanBodyParameters)))
+        'multipart' => [
+@foreach($endpoint->cleanBodyParameters as $parameter => $value)
+@foreach(u::getParameterNamesAndValuesForFormData($parameter, $value) as $key => $actualValue)
+            [
+                'name' => '{!! $key !!}',
+                'contents' => '{!! $actualValue !!}'
+            ],
+@endforeach
+@endforeach
+@foreach($endpoint->fileParameters as $parameter => $value)
+@foreach(u::getParameterNamesAndValuesForFormData($parameter, $value) as $key => $file)
+            [
+                'name' => '{!!  $key !!}',
+                'contents' => fopen('{!! $file->path() !!}', 'r')
+            ],
+@endforeach
+@endforeach
+        ],
+@elseif(count($endpoint->cleanBodyParameters))
+@if ($endpoint->headers['Content-Type'] == 'application/x-www-form-urlencoded')
+        'form_params' => {!! u::printPhpValue($endpoint->cleanBodyParameters, 8) !!},
+@else
+        'json' => {!! u::printPhpValue($endpoint->cleanBodyParameters, 8) !!},
+@endif
+@endif
+    ]
+);
+@else
+$response = $client->{{ strtolower($endpoint->httpMethods[0]) }}($url);
+@endif
+$body = $response->getBody();
+print_r(json_decode((string) $body));
+```

+ 52 - 0
resources/views/vendor/scribe/partials/example-requests/python.md.blade.php

@@ -0,0 +1,52 @@
+@php
+  use Knuckles\Scribe\Tools\WritingUtils as u;
+  /** @var  Knuckles\Camel\Output\OutputEndpointData $endpoint */
+@endphp
+```python
+import requests
+import json
+
+url = '{!! rtrim($baseUrl, '/') !!}/{{ $endpoint->boundUri }}'
+@if($endpoint->hasFiles() || (isset($endpoint->headers['Content-Type']) && $endpoint->headers['Content-Type'] == 'multipart/form-data' && count($endpoint->cleanBodyParameters)))
+files = {
+@foreach($endpoint->cleanBodyParameters as $parameter => $value)
+@foreach(u::getParameterNamesAndValuesForFormData($parameter, $value) as $key => $actualValue)
+  '{!! $key !!}': (None, '{!! $actualValue !!}')@if(!($loop->parent->last) || count($endpoint->fileParameters)),
+@endif
+@endforeach
+@endforeach
+@foreach($endpoint->fileParameters as $parameter => $value)
+@foreach(u::getParameterNamesAndValuesForFormData($parameter, $value) as $key => $file)
+  '{!! $key !!}': open('{!! $file->path() !!}', 'rb')@if(!($loop->parent->last)),
+@endif
+@endforeach
+@endforeach
+}
+@endif
+@if(count($endpoint->cleanBodyParameters))
+payload = {!! json_encode($endpoint->cleanBodyParameters, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE) !!}
+@endif
+@if(count($endpoint->cleanQueryParameters))
+params = {!! u::printQueryParamsAsKeyValue($endpoint->cleanQueryParameters, "'", ":", 2, "{}") !!}
+@endif
+@if(!empty($endpoint->headers))
+headers = {
+@foreach($endpoint->headers as $header => $value)
+  '{{$header}}': '{{$value}}'@if(!($loop->last)),
+@endif
+@endforeach
+
+}
+
+@endif
+@php
+$optionalArguments = [];
+if (count($endpoint->headers)) $optionalArguments[] = "headers=headers";
+if (count($endpoint->fileParameters)) $optionalArguments[] = "files=files";
+if (count($endpoint->cleanBodyParameters) && $endpoint->headers['Content-Type'] != 'multipart/form-data') $optionalArguments[] = (count($endpoint->fileParameters) || $endpoint->headers['Content-Type'] == 'application/x-www-form-urlencoded' ? "data=payload" : "json=payload");
+if (count($endpoint->cleanQueryParameters)) $optionalArguments[] = "params=params";
+$optionalArguments = implode(', ',$optionalArguments);
+@endphp
+response = requests.request('{{$endpoint->httpMethods[0]}}', url, {{ $optionalArguments }})
+response.json()
+```

+ 185 - 0
resources/views/vendor/scribe/themes/default/endpoint.blade.php

@@ -0,0 +1,185 @@
+@php
+    use Knuckles\Scribe\Tools\Utils as u;
+    /** @var  Knuckles\Camel\Output\OutputEndpointData $endpoint */
+@endphp
+
+<h2 id="{!! $endpoint->fullSlug() !!}">{{ $endpoint->name() }}</h2>
+
+<p>
+@component('scribe::components.badges.auth', ['authenticated' => $endpoint->isAuthed()])
+@endcomponent
+</p>
+
+{!! Parsedown::instance()->text($endpoint->metadata->description ?: '') !!}
+
+<span id="example-requests-{!! $endpoint->endpointId() !!}">
+<blockquote>{{ u::trans("scribe::endpoint.example_request") }}:</blockquote>
+
+@foreach($metadata['example_languages'] as $language)
+
+<div class="{{ $language }}-example">
+    @include("scribe::partials.example-requests.$language")
+</div>
+
+@endforeach
+</span>
+
+<span id="example-responses-{!! $endpoint->endpointId() !!}">
+@if($endpoint->isGet() || $endpoint->hasResponses())
+    @foreach($endpoint->responses as $response)
+        <blockquote>
+            <p>{{ u::trans("scribe::endpoint.example_response") }} ({{ $response->fullDescription() }}):</p>
+        </blockquote>
+        @if(count($response->headers))
+        <details class="annotation">
+            <summary style="cursor: pointer;">
+                <small onclick="textContent = parentElement.parentElement.open ? 'Show headers' : 'Hide headers'">Show headers</small>
+            </summary>
+            <pre><code class="language-http">@foreach($response->headers as $header => $value)
+{{ $header }}: {{ is_array($value) ? implode('; ', $value) : $value }}
+@endforeach </code></pre></details> @endif
+        <pre>
+@if(is_string($response->content) && Str::startsWith($response->content, "<<binary>>"))
+<code>{!! u::trans("scribe::endpoint.responses.binary") !!} - {{ htmlentities(str_replace("<<binary>>", "", $response->content)) }}</code>
+@elseif($response->status == 204)
+<code>{!! u::trans("scribe::endpoint.responses.empty") !!}</code>
+@else
+@php($parsed = json_decode($response->content))
+{{-- If response is a JSON string, prettify it. Otherwise, just print it --}}
+<code class="language-json" style="max-height: 300px;">{!! htmlentities($parsed != null ? json_encode($parsed, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES) : $response->content) !!}</code>
+@endif </pre>
+    @endforeach
+@endif
+</span>
+<span id="execution-results-{{ $endpoint->endpointId() }}" hidden>
+    <blockquote>{{ u::trans("scribe::try_it_out.received_response") }}<span
+                id="execution-response-status-{{ $endpoint->endpointId() }}"></span>:
+    </blockquote>
+    <pre class="json"><code id="execution-response-content-{{ $endpoint->endpointId() }}"
+      data-empty-response-text="<{{ u::trans("scribe::endpoint.responses.empty") }}>" style="max-height: 400px;"></code></pre>
+</span>
+<span id="execution-error-{{ $endpoint->endpointId() }}" hidden>
+    <blockquote>{{ u::trans("scribe::try_it_out.request_failed") }}:</blockquote>
+    <pre><code id="execution-error-message-{{ $endpoint->endpointId() }}">{{ "\n\n".u::trans("scribe::try_it_out.error_help") }}</code></pre>
+</span>
+<form id="form-{{ $endpoint->endpointId() }}" data-method="{{ $endpoint->httpMethods[0] }}"
+      data-path="{{ $endpoint->uri }}"
+      data-authed="{{ $endpoint->isAuthed() ? 1 : 0 }}"
+      data-hasfiles="{{ $endpoint->hasFiles() ? 1 : 0 }}"
+      data-isarraybody="{{ $endpoint->isArrayBody() ? 1 : 0 }}"
+      autocomplete="off"
+      onsubmit="event.preventDefault(); executeTryOut('{{ $endpoint->endpointId() }}', this);">
+    <h3>
+        {{ u::trans("scribe::endpoint.request") }}&nbsp;&nbsp;&nbsp;
+        @if($metadata['try_it_out']['enabled'] ?? false)
+            <button type="button"
+                    style="background-color: #8fbcd4; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-tryout-{{ $endpoint->endpointId() }}"
+                    onclick="tryItOut('{{ $endpoint->endpointId() }}');">{{ u::trans("scribe::try_it_out.open") }}
+            </button>
+            <button type="button"
+                    style="background-color: #c97a7e; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-canceltryout-{{ $endpoint->endpointId() }}"
+                    onclick="cancelTryOut('{{ $endpoint->endpointId() }}');" hidden>{{ u::trans("scribe::try_it_out.cancel") }}
+            </button>&nbsp;&nbsp;
+            <button type="submit"
+                    style="background-color: #6ac174; padding: 5px 10px; border-radius: 5px; border-width: thin;"
+                    id="btn-executetryout-{{ $endpoint->endpointId() }}"
+                    data-initial-text="{{ u::trans("scribe::try_it_out.send") }}"
+                    data-loading-text="{{ u::trans("scribe::try_it_out.loading") }}"
+                    hidden>{{ u::trans("scribe::try_it_out.send") }}
+            </button>
+        @endif
+    </h3>
+    @foreach($endpoint->httpMethods as $method)
+        <p>
+            @component('scribe::components.badges.http-method', ['method' => $method])@endcomponent
+            <b><code>{{$endpoint->uri}}</code></b>
+        </p>
+    @endforeach
+    @if(count($endpoint->headers))
+        <h4 class="fancy-heading-panel"><b>{{ u::trans("scribe::endpoint.headers") }}</b></h4>
+        @foreach($endpoint->headers as $name => $example)
+            <?php
+                $htmlOptions = [];
+                if ($endpoint->isAuthed() && $metadata['auth']['location'] == 'header' && $metadata['auth']['name'] == $name) {
+                  $htmlOptions = [ 'class' => 'auth-value', ];
+                  }
+            ?>
+            <div style="padding-left: 28px; clear: unset;">
+                @component('scribe::components.field-details', [
+                  'name' => $name,
+                  'type' => null,
+                  'required' => true,
+                  'description' => null,
+                  'example' => $example,
+                  'endpointId' => $endpoint->endpointId(),
+                  'component' => 'header',
+                  'isInput' => true,
+                  'html' => $htmlOptions,
+                ])
+                @endcomponent
+            </div>
+        @endforeach
+    @endif
+    @if(count($endpoint->urlParameters))
+        <h4 class="fancy-heading-panel"><b>{{ u::trans("scribe::endpoint.url_parameters") }}</b></h4>
+        @foreach($endpoint->urlParameters as $attribute => $parameter)
+            <div style="padding-left: 28px; clear: unset;">
+                @component('scribe::components.field-details', [
+                  'name' => $parameter->name,
+                  'type' => $parameter->type ?? 'string',
+                  'required' => $parameter->required,
+                  'description' => $parameter->description,
+                  'example' => $parameter->example ?? '',
+                  'enumValues' => $parameter->enumValues,
+                  'endpointId' => $endpoint->endpointId(),
+                  'component' => 'url',
+                  'isInput' => true,
+                ])
+                @endcomponent
+            </div>
+        @endforeach
+    @endif
+    @if(count($endpoint->queryParameters))
+        <h4 class="fancy-heading-panel"><b>{{ u::trans("scribe::endpoint.query_parameters") }}</b></h4>
+        @foreach($endpoint->queryParameters as $attribute => $parameter)
+                <?php
+                $htmlOptions = [];
+                if ($endpoint->isAuthed() && $metadata['auth']['location'] == 'query' && $metadata['auth']['name'] == $attribute) {
+                    $htmlOptions = [ 'class' => 'auth-value', ];
+                }
+                ?>
+            <div style="padding-left: 28px; clear: unset;">
+                @component('scribe::components.field-details', [
+                  'name' => $parameter->name,
+                  'type' => $parameter->type,
+                  'required' => $parameter->required,
+                  'description' => $parameter->description,
+                  'example' => $parameter->example ?? '',
+                  'enumValues' => $parameter->enumValues,
+                  'endpointId' => $endpoint->endpointId(),
+                  'component' => 'query',
+                  'isInput' => true,
+                  'html' => $htmlOptions,
+                ])
+                @endcomponent
+            </div>
+        @endforeach
+    @endif
+    @if(count($endpoint->nestedBodyParameters))
+        <h4 class="fancy-heading-panel"><b>{{ u::trans("scribe::endpoint.body_parameters") }}</b></h4>
+        <x-scribe::nested-fields
+                :fields="$endpoint->nestedBodyParameters" :endpointId="$endpoint->endpointId()"
+        />
+    @endif
+</form>
+
+@if(count($endpoint->responseFields))
+    <h3>{{ u::trans("scribe::endpoint.response") }}</h3>
+    <h4 class="fancy-heading-panel"><b>{{ u::trans("scribe::endpoint.response_fields") }}</b></h4>
+    <x-scribe::nested-fields
+            :fields="$endpoint->nestedResponseFields" :endpointId="$endpoint->endpointId()"
+            :isInput="false"
+    />
+@endif

+ 21 - 0
resources/views/vendor/scribe/themes/default/groups.blade.php

@@ -0,0 +1,21 @@
+@foreach($groupedEndpoints as $group)
+    <h1 id="{!! Str::slug($group['name']) !!}">{!! $group['name'] !!}</h1>
+
+    {!! Parsedown::instance()->text($group['description']) !!}
+
+    @foreach($group['subgroups'] as $subgroupName => $subgroup)
+        @if($subgroupName !== "")
+            <h2 id="{!! Str::slug($group['name']) !!}-{!! Str::slug($subgroupName) !!}">{{ $subgroupName }}</h2>
+            @php($subgroupDescription = collect($subgroup)->first(fn ($e) => $e->metadata->subgroupDescription)?->metadata?->subgroupDescription)
+            @if($subgroupDescription)
+                <p>
+                    {!! Parsedown::instance()->text($subgroupDescription) !!}
+                </p>
+            @endif
+        @endif
+        @foreach($subgroup as $endpoint)
+            @include("scribe::themes.default.endpoint")
+        @endforeach
+    @endforeach
+@endforeach
+

+ 74 - 0
resources/views/vendor/scribe/themes/default/index.blade.php

@@ -0,0 +1,74 @@
+@php
+    use Knuckles\Scribe\Tools\WritingUtils as u;
+@endphp
+<!doctype html>
+<html lang="en">
+<head>
+    <meta charset="utf-8">
+    <meta content="IE=edge,chrome=1" http-equiv="X-UA-Compatible">
+    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
+    <title>{!! $metadata['title'] !!}</title>
+
+    <link href="https://fonts.googleapis.com/css?family=Open+Sans&display=swap" rel="stylesheet">
+
+    <link rel="stylesheet" href="{!! $assetPathPrefix !!}css/theme-default.style.css" media="screen">
+    <link rel="stylesheet" href="{!! $assetPathPrefix !!}css/theme-default.print.css" media="print">
+
+    <script src="https://cdn.jsdelivr.net/npm/lodash@4.17.10/lodash.min.js"></script>
+
+    <link rel="stylesheet"
+          href="https://unpkg.com/@highlightjs/cdn-assets@11.6.0/styles/obsidian.min.css">
+    <script src="https://unpkg.com/@highlightjs/cdn-assets@11.6.0/highlight.min.js"></script>
+
+    <script src="https://cdnjs.cloudflare.com/ajax/libs/jets/0.14.1/jets.min.js"></script>
+
+@if(isset($metadata['example_languages']))
+    <style id="language-style">
+        /* starts out as display none and is replaced with js later  */
+        @foreach($metadata['example_languages'] as $lang)
+            body .content .{{ $lang }}-example code { display: none; }
+        @endforeach
+    </style>
+@endif
+
+@if($tryItOut['enabled'] ?? true)
+    <script>
+        var tryItOutBaseUrl = "{!! $tryItOut['base_url'] ?? config('app.url') !!}";
+        var useCsrf = Boolean({!! $tryItOut['use_csrf'] ?? null !!});
+        var csrfUrl = "{!! $tryItOut['csrf_url'] ?? null !!}";
+    </script>
+    <script src="{{ u::getVersionedAsset($assetPathPrefix.'js/tryitout.js') }}"></script>
+@endif
+
+    <script src="{{ u::getVersionedAsset($assetPathPrefix.'js/theme-default.js') }}"></script>
+
+</head>
+
+<body data-languages="{{ json_encode($metadata['example_languages'] ?? []) }}">
+
+@include("scribe::themes.default.sidebar")
+
+<div class="page-wrapper">
+    <div class="dark-box"></div>
+    <div class="content">
+        {!! $intro !!}
+
+        {!! $auth !!}
+
+        @include("scribe::themes.default.groups")
+
+        {!! $append !!}
+    </div>
+    <div class="dark-box">
+        @if(isset($metadata['example_languages']))
+            <div class="lang-selector">
+                @foreach($metadata['example_languages'] as $name => $lang)
+                    @php if (is_numeric($name)) $name = $lang; @endphp
+                    <button type="button" class="lang-button" data-language-name="{{$lang}}">{{$name}}</button>
+                @endforeach
+            </div>
+        @endif
+    </div>
+</div>
+</body>
+</html>

+ 69 - 0
resources/views/vendor/scribe/themes/default/sidebar.blade.php

@@ -0,0 +1,69 @@
+@php
+    use Knuckles\Scribe\Tools\Utils as u;
+@endphp
+<a href="#" id="nav-button">
+    <span>
+        MENU
+        <img src="{!! $assetPathPrefix !!}images/navbar.png" alt="navbar-image"/>
+    </span>
+</a>
+<div class="tocify-wrapper">
+    @if($metadata['logo'] != false)
+        <img src="{{ $metadata['logo'] }}" alt="logo" class="logo" style="padding-top: 10px;" width="100%"/>
+    @endif
+
+    @isset($metadata['example_languages'])
+        <div class="lang-selector">
+            @foreach($metadata['example_languages'] as $name => $lang)
+                @php if (is_numeric($name)) $name = $lang; @endphp
+                <button type="button" class="lang-button" data-language-name="{{ $lang }}">{{ $name }}</button>
+            @endforeach
+        </div>
+    @endisset
+
+    <div class="search">
+        <input type="text" class="search" id="input-search" placeholder="{{ u::trans("scribe::labels.search") }}">
+    </div>
+
+    <div id="toc">
+        @foreach($headings as $h1)
+            <ul id="tocify-header-{{ $h1['slug'] }}" class="tocify-header">
+                <li class="tocify-item level-1" data-unique="{!! $h1['slug'] !!}">
+                    <a href="#{!! $h1['slug'] !!}">{!! $h1['name'] !!}</a>
+                </li>
+                @if(count($h1['subheadings']) > 0)
+                    <ul id="tocify-subheader-{!! $h1['slug'] !!}" class="tocify-subheader">
+                        @foreach($h1['subheadings'] as $h2)
+                            <li class="tocify-item level-2" data-unique="{!! $h2['slug'] !!}">
+                                <a href="#{!! $h2['slug'] !!}">{!! $h2['name'] !!}</a>
+                            </li>
+                            @if(count($h2['subheadings']) > 0)
+                                <ul id="tocify-subheader-{!! $h2['slug'] !!}" class="tocify-subheader">
+                                    @foreach($h2['subheadings'] as $h3)
+                                        <li class="tocify-item level-3" data-unique="{!! $h3['slug'] !!}">
+                                            <a href="#{!! $h3['slug'] !!}">{!! $h3['name'] !!}</a>
+                                        </li>
+                                    @endforeach
+                                </ul>
+                            @endif
+                        @endforeach
+                    </ul>
+                @endif
+            </ul>
+        @endforeach
+    </div>
+
+    <ul class="toc-footer" id="toc-footer">
+        @if($metadata['postman_collection_url'])
+            <li style="padding-bottom: 5px;"><a href="{!! $metadata['postman_collection_url'] !!}">{!! u::trans("scribe::links.postman") !!}</a></li>
+        @endif
+        @if($metadata['openapi_spec_url'])
+            <li style="padding-bottom: 5px;"><a href="{!! $metadata['openapi_spec_url'] !!}">{!! u::trans("scribe::links.openapi") !!}</a></li>
+        @endif
+        <li><a href="http://github.com/knuckleswtf/scribe">Documentation powered by Scribe ✍</a></li>
+    </ul>
+
+    <ul class="toc-footer" id="last-updated">
+        <li>{{ $metadata['last_updated'] }}</li>
+    </ul>
+</div>

+ 63 - 0
resources/views/vendor/scribe/themes/elements/components/field-details.blade.php

@@ -0,0 +1,63 @@
+@php
+    $hasChildren ??= false;
+    $isArrayBody = $name == "[]";
+    $expandable = $hasChildren && !$isArrayBody;
+@endphp
+
+<div class="sl-flex sl-relative sl-max-w-full sl-py-2 sl-pl-3">
+    <div class="sl-w-1 sl-mt-2 sl-mr-3 sl--ml-3 sl-border-t"></div>
+    <div class="sl-stack sl-stack--vertical sl-stack--1 sl-flex sl-flex-1 sl-flex-col sl-items-stretch sl-max-w-full sl-ml-2 @if($expandable) sl-cursor-pointer @endif">
+        <div class="sl-flex sl-items-center sl-max-w-full">
+            @if($expandable)
+                <div class="sl-flex sl-justify-center sl-w-8 sl--ml-8 sl-pl-3 sl-text-muted expansion-chevrons" role="button">
+                    <svg aria-hidden="true" focusable="false" data-prefix="fas" data-icon="chevron-right"
+                         class="svg-inline--fa fa-chevron-right fa-fw fa-sm sl-icon" role="img"
+                         xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                        <path fill="currentColor"
+                              d="M96 480c-8.188 0-16.38-3.125-22.62-9.375c-12.5-12.5-12.5-32.75 0-45.25L242.8 256L73.38 86.63c-12.5-12.5-12.5-32.75 0-45.25s32.75-12.5 45.25 0l192 192c12.5 12.5 12.5 32.75 0 45.25l-192 192C112.4 476.9 104.2 480 96 480z"></path>
+                    </svg>
+                </div>
+            @endif
+            @unless($isArrayBody)
+                <div class="sl-flex sl-items-baseline sl-text-base">
+                    <div class="sl-font-mono sl-font-semibold sl-mr-2">{{ $name }}</div>
+                    @if($type)
+                        <span class="sl-truncate sl-text-muted">{{ $type }}</span>
+                    @endif
+                </div>
+                @if($required)
+                    <div class="sl-flex-1 sl-h-px sl-mx-3"></div>
+                    <span class="sl-ml-2 sl-text-warning">required</span>
+                @endif
+            @endunless
+        </div>
+        @if($description)
+        <div class="sl-prose sl-markdown-viewer" style="font-size: 12px;">
+            {!! Parsedown::instance()->text($description) !!}
+        </div>
+        @endif
+        @if(!empty($enumValues))
+            Must be one of:
+            <ul style="list-style-position: inside; list-style-type: square;">{!! implode(" ", array_map(fn($val) => "<li><code>$val</code></li>", $enumValues)) !!}</ul>
+        @endif
+        @if($isArrayBody)
+            <div class="sl-flex sl-items-baseline sl-text-base">
+                <div class="sl-font-mono sl-font-semibold sl-mr-2">array of:</div>
+                @if($required)
+                    <div class="sl-flex-1 sl-h-px sl-mx-3"></div>
+                    <span class="sl-ml-2 sl-text-warning">required</span>
+                @endif
+            </div>
+        @endif
+        @if(!$hasChildren && !is_null($example) && $example !== '')
+            <div class="sl-stack sl-stack--horizontal sl-stack--2 sl-flex sl-flex-row sl-items-baseline sl-text-muted">
+                <span>Example:</span> <!-- <span> important for spacing -->
+                <div class="sl-flex sl-flex-1 sl-flex-wrap" style="gap: 4px;">
+                    <div class="sl-max-w-full sl-break-all sl-px-1 sl-bg-canvas-tint sl-text-muted sl-rounded sl-border">
+                        {{ is_array($example) || is_bool($example) ? json_encode($example) : $example }}
+                    </div>
+                </div>
+            </div>
+        @endif
+    </div>
+</div>

+ 37 - 0
resources/views/vendor/scribe/themes/elements/components/nested-fields.blade.php

@@ -0,0 +1,37 @@
+@php
+    $level ??= 0;
+    $levelNestingClass = match($level) {
+        0 => "sl-ml-px",
+        default => "sl-ml-7"
+    };
+    $expandable ??= !isset($fields["[]"]);
+@endphp
+
+@foreach($fields as $name => $field)
+    <div class="{{ $expandable ? 'expandable' : '' }} sl-text-sm sl-border-l {{ $levelNestingClass }}">
+        @component('scribe::themes.elements.components.field-details', [
+          'name' => $name,
+          'type' => $field['type'] ?? 'string',
+          'required' => $field['required'] ?? false,
+          'description' => $field['description'] ?? '',
+          'example' => $field['example'] ?? '',
+          'enumValues' => $field['enumValues'] ?? null,
+          'endpointId' => $endpointId,
+          'hasChildren' => !empty($field['__fields']),
+          'component' => 'body',
+        ])
+        @endcomponent
+
+        @if(!empty($field['__fields']))
+            <div class="children" style="{{ $expandable ? 'display: none;' : '' }}">
+                @component('scribe::themes.elements.components.nested-fields', [
+                  'fields' => $field['__fields'],
+                  'endpointId' => $endpointId,
+                  'level' => $level + 1,
+                  'expandable'=> $expandable,
+                ])
+                @endcomponent
+            </div>
+        @endif
+    </div>
+@endforeach

+ 263 - 0
resources/views/vendor/scribe/themes/elements/endpoint.blade.php

@@ -0,0 +1,263 @@
+@php
+    use Knuckles\Scribe\Tools\Utils as u;
+    /** @var  Knuckles\Camel\Output\OutputEndpointData $endpoint */
+@endphp
+
+<div class="sl-stack sl-stack--vertical sl-stack--8 HttpOperation sl-flex sl-flex-col sl-items-stretch sl-w-full">
+    <div class="sl-stack sl-stack--vertical sl-stack--5 sl-flex sl-flex-col sl-items-stretch">
+        <div class="sl-relative">
+            <div class="sl-stack sl-stack--horizontal sl-stack--5 sl-flex sl-flex-row sl-items-center">
+                <h2 class="sl-text-3xl sl-leading-tight sl-font-prose sl-text-heading sl-mt-5 sl-mb-1"
+                    id="{!! $endpoint->fullSlug() !!}">
+                    {{ $endpoint->name() }}
+                </h2>
+            </div>
+        </div>
+
+        <div class="sl-relative">
+            <div title="{{ rtrim($baseUrl, '/') . '/'. ltrim($endpoint->uri, '/') }}"
+                     class="sl-stack sl-stack--horizontal sl-stack--3 sl-inline-flex sl-flex-row sl-items-center sl-max-w-full sl-font-mono sl-py-2 sl-pr-4 sl-bg-canvas-50 sl-rounded-lg"
+                >
+                    @foreach($endpoint->httpMethods as $method)
+                        <div class="sl-text-lg sl-font-semibold sl-px-2.5 sl-py-1 sl-text-on-primary sl-rounded-lg"
+                             style="background-color: {{ \Knuckles\Scribe\Tools\WritingUtils::$httpMethodToCssColour[$method] }};"
+                        >
+                            {{ $method }}
+                        </div>
+                    @endforeach
+                    <div class="sl-flex sl-overflow-x-hidden sl-text-lg sl-select-all">
+                        <div dir="rtl"
+                             class="sl-overflow-x-hidden sl-truncate sl-text-muted">{{ rtrim($baseUrl, '/') }}</div>
+                        <div class="sl-flex-1 sl-font-semibold">/{{ ltrim($endpoint->uri, '/') }}</div>
+                    </div>
+
+                        @if($endpoint->metadata->authenticated)
+                            <div class="sl-font-prose sl-font-semibold sl-px-1.5 sl-py-0.5 sl-text-on-primary sl-rounded-lg"
+                                 style="background-color: darkred"
+                            >requires authentication
+                            </div>
+                        @endif
+            </div>
+        </div>
+
+        {!! Parsedown::instance()->text($endpoint->metadata->description ?: '') !!}
+    </div>
+    <div class="sl-flex">
+        <div data-testid="two-column-left" class="sl-flex-1 sl-w-0">
+            <div class="sl-stack sl-stack--vertical sl-stack--10 sl-flex sl-flex-col sl-items-stretch">
+                <div class="sl-stack sl-stack--vertical sl-stack--8 sl-flex sl-flex-col sl-items-stretch">
+                    @if(count($endpoint->headers))
+                        <div class="sl-stack sl-stack--vertical sl-stack--5 sl-flex sl-flex-col sl-items-stretch">
+                            <h3 class="sl-text-2xl sl-leading-snug sl-font-prose">
+                                {{ u::trans("scribe::endpoint.headers") }}
+                            </h3>
+                            <div class="sl-text-sm">
+                                @foreach($endpoint->headers as $header => $value)
+                                    @component('scribe::themes.elements.components.field-details', [
+                                      'name' => $header,
+                                      'type' => null,
+                                      'required' => false,
+                                      'description' => null,
+                                      'example' => $value,
+                                      'endpointId' => $endpoint->endpointId(),
+                                      'component' => 'header',
+                                      'isInput' => true,
+                                    ])
+                                    @endcomponent
+                                @endforeach
+                            </div>
+                        </div>
+                    @endif
+
+                    @if(count($endpoint->urlParameters))
+                        <div class="sl-stack sl-stack--vertical sl-stack--6 sl-flex sl-flex-col sl-items-stretch">
+                            <h3 class="sl-text-2xl sl-leading-snug sl-font-prose">{{ u::trans("scribe::endpoint.url_parameters") }}</h3>
+
+                            <div class="sl-text-sm">
+                                @foreach($endpoint->urlParameters as $attribute => $parameter)
+                                    @component('scribe::themes.elements.components.field-details', [
+                                      'name' => $parameter->name,
+                                      'type' => $parameter->type ?? 'string',
+                                      'required' => $parameter->required,
+                                      'description' => $parameter->description,
+                                      'example' => $parameter->example ?? '',
+                                      'enumValues' => $parameter->enumValues,
+                                      'endpointId' => $endpoint->endpointId(),
+                                      'component' => 'url',
+                                      'isInput' => true,
+                                    ])
+                                    @endcomponent
+                                @endforeach
+                            </div>
+                        </div>
+                    @endif
+
+
+                    @if(count($endpoint->queryParameters))
+                            <div class="sl-stack sl-stack--vertical sl-stack--6 sl-flex sl-flex-col sl-items-stretch">
+                                <h3 class="sl-text-2xl sl-leading-snug sl-font-prose">{{ u::trans("scribe::endpoint.query_parameters") }}</h3>
+
+                                <div class="sl-text-sm">
+                                    @foreach($endpoint->queryParameters as $attribute => $parameter)
+                                        @component('scribe::themes.elements.components.field-details', [
+                                          'name' => $parameter->name,
+                                          'type' => $parameter->type,
+                                          'required' => $parameter->required,
+                                          'description' => $parameter->description,
+                                          'example' => $parameter->example ?? '',
+                                          'enumValues' => $parameter->enumValues,
+                                          'endpointId' => $endpoint->endpointId(),
+                                          'component' => 'query',
+                                          'isInput' => true,
+                                        ])
+                                        @endcomponent
+                                    @endforeach
+                            </div>
+                        </div>
+                    @endif
+
+                    @if(count($endpoint->nestedBodyParameters))
+                        <div class="sl-stack sl-stack--vertical sl-stack--6 sl-flex sl-flex-col sl-items-stretch">
+                            <h3 class="sl-text-2xl sl-leading-snug sl-font-prose">{{ u::trans("scribe::endpoint.body_parameters") }}</h3>
+
+                                <div class="sl-text-sm">
+                                    @component('scribe::themes.elements.components.nested-fields', [
+                                      'fields' => $endpoint->nestedBodyParameters,
+                                      'endpointId' => $endpoint->endpointId(),
+                                    ])
+                                    @endcomponent
+                            </div>
+                        </div>
+                    @endif
+
+                    @if(count($endpoint->responseFields))
+                            <div class="sl-stack sl-stack--vertical sl-stack--6 sl-flex sl-flex-col sl-items-stretch">
+                                <h3 class="sl-text-2xl sl-leading-snug sl-font-prose">{{ u::trans("scribe::endpoint.response_fields") }}</h3>
+
+                                <div class="sl-text-sm">
+                                    @component('scribe::themes.elements.components.nested-fields', [
+                                      'fields' => $endpoint->nestedResponseFields,
+                                      'endpointId' => $endpoint->endpointId(),
+                                      'isInput' => false,
+                                    ])
+                                    @endcomponent
+                                </div>
+                            </div>
+                        @endif
+                </div>
+            </div>
+        </div>
+
+        <div data-testid="two-column-right" class="sl-relative sl-w-2/5 sl-ml-16" style="max-width: 500px;">
+            <div class="sl-stack sl-stack--vertical sl-stack--6 sl-flex sl-flex-col sl-items-stretch">
+
+                @if($metadata['try_it_out']['enabled'] ?? false)
+                    @include("scribe::themes.elements.try_it_out")
+                @endif
+
+                    @if($metadata['example_languages'])
+                        <div class="sl-panel sl-outline-none sl-w-full sl-rounded-lg">
+                            <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-3 sl-pl-4 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-select-none">
+                                <div class="sl-flex sl-flex-1 sl-items-center sl-h-lg">
+                                    <div class="sl--ml-2">
+                                        {{ u::trans("scribe::endpoint.example_request") }}:
+                                        <select class="example-request-lang-toggle sl-text-base"
+                                                aria-label="Request Sample Language"
+                                                onchange="switchExampleLanguage(event.target.value);">
+                                            @foreach($metadata['example_languages'] as $language)
+                                                <option>{{ $language }}</option>
+                                            @endforeach
+                                        </select>
+                                    </div>
+                                </div>
+                            </div>
+                            @foreach($metadata['example_languages'] as $index => $language)
+                                <div class="sl-bg-canvas-100 example-request example-request-{{ $language }}"
+                                     style="{{ $index == 0 ? '' : 'display: none;' }}">
+                                    <div class="sl-px-0 sl-py-1">
+                                        <div style="max-height: 400px;" class="sl-overflow-y-auto sl-rounded">
+                                            @include("scribe::partials.example-requests.$language")
+                                        </div>
+                                    </div>
+                                </div>
+                            @endforeach
+                        </div>
+                    @endif
+
+                    @if($endpoint->isGet() || $endpoint->hasResponses())
+                        <div class="sl-panel sl-outline-none sl-w-full sl-rounded-lg">
+                            <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-3 sl-pl-4 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-select-none">
+                                <div class="sl-flex sl-flex-1 sl-items-center sl-py-2">
+                                    <div class="sl--ml-2">
+                                        <div class="sl-h-sm sl-text-base sl-font-medium sl-px-1.5 sl-text-muted sl-rounded sl-border-transparent sl-border">
+                                            <div class="sl-mb-2 sl-inline-block">{{ u::trans("scribe::endpoint.example_response") }}:</div>
+                                            <div class="sl-mb-2 sl-inline-block">
+                                                <select
+                                                        class="example-response-{{ $endpoint->endpointId() }}-toggle sl-text-base"
+                                                        aria-label="Response sample"
+                                                        onchange="switchExampleResponse('{{ $endpoint->endpointId() }}', event.target.value);">
+                                                    @foreach($endpoint->responses as $index => $response)
+                                                        <option value="{{ $index }}">{{ $response->fullDescription() }}</option>
+                                                    @endforeach
+                                                </select></div>
+                                        </div>
+                                    </div>
+                                </div>
+                                <button type="button"
+                                        class="sl-button sl-h-sm sl-text-base sl-font-medium sl-px-1.5 hover:sl-bg-canvas-50 active:sl-bg-canvas-100 sl-text-muted hover:sl-text-body focus:sl-text-body sl-rounded sl-border-transparent sl-border disabled:sl-opacity-70">
+                                    <div class="sl-mx-0">
+                                        <svg aria-hidden="true" focusable="false" data-prefix="fas" data-icon="copy"
+                                             class="svg-inline--fa fa-copy fa-fw fa-sm sl-icon" role="img"
+                                             xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
+                                            <path fill="currentColor"
+                                                  d="M384 96L384 0h-112c-26.51 0-48 21.49-48 48v288c0 26.51 21.49 48 48 48H464c26.51 0 48-21.49 48-48V128h-95.1C398.4 128 384 113.6 384 96zM416 0v96h96L416 0zM192 352V128h-144c-26.51 0-48 21.49-48 48v288c0 26.51 21.49 48 48 48h192c26.51 0 48-21.49 48-48L288 416h-32C220.7 416 192 387.3 192 352z"></path>
+                                        </svg>
+                                    </div>
+                                </button>
+                            </div>
+                            @foreach($endpoint->responses as $index => $response)
+                                <div class="sl-panel__content-wrapper sl-bg-canvas-100 example-response-{{ $endpoint->endpointId() }} example-response-{{ $endpoint->endpointId() }}-{{ $index }}"
+                                     style=" {{ $index == 0 ? '' : 'display: none;' }}"
+                                >
+                                    <div class="sl-panel__content sl-p-0">@if(count($response->headers))
+                                            <details class="sl-pl-2">
+                                                <summary style="cursor: pointer; list-style: none;">
+                                                    <small>
+                                                        <span class="expansion-chevrons">
+
+    <svg aria-hidden="true" focusable="false" data-prefix="fas"
+         data-icon="chevron-right"
+         class="svg-inline--fa fa-chevron-right fa-fw sl-icon sl-text-muted"
+         xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+        <path fill="currentColor"
+              d="M96 480c-8.188 0-16.38-3.125-22.62-9.375c-12.5-12.5-12.5-32.75 0-45.25L242.8 256L73.38 86.63c-12.5-12.5-12.5-32.75 0-45.25s32.75-12.5 45.25 0l192 192c12.5 12.5 12.5 32.75 0 45.25l-192 192C112.4 476.9 104.2 480 96 480z"></path>
+    </svg>
+                                                            </span>
+                                                        Headers
+                                                    </small>
+                                                </summary>
+                                                <pre><code class="language-http">@foreach($response->headers as $header => $value)
+                                                            {{ $header }}
+                                                            : {{ is_array($value) ? implode('; ', $value) : $value }}
+                                                        @endforeach </code></pre>
+                                            </details>
+                                        @endif
+                                        @if(is_string($response->content) && Str::startsWith($response->content, "<<binary>>"))
+                                            <pre><code>[{{ u::trans("scribe::endpoint.responses.binary") }}] - {{ htmlentities(str_replace("<<binary>>", "", $response->content)) }}</code></pre>
+                                        @elseif($response->status == 204)
+                                            <pre><code>[{{ u::trans("scribe::endpoint.responses.empty") }}]</code></pre>
+                                        @else
+                                            @php($parsed = json_decode($response->content))
+                                            {{-- If response is a JSON string, prettify it. Otherwise, just print it --}}
+                                            <pre><code style="max-height: 300px;"
+                                                       class="language-json sl-overflow-x-auto sl-overflow-y-auto">{!! htmlentities($parsed != null ? json_encode($parsed, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES) : $response->content) !!}</code></pre>
+                                        @endif
+                                    </div>
+                                </div>
+                            @endforeach
+                        </div>
+                    @endif
+        </div>
+    </div>
+</div>
+

+ 28 - 0
resources/views/vendor/scribe/themes/elements/groups.blade.php

@@ -0,0 +1,28 @@
+@foreach($groupedEndpoints as $group)
+    <h1 id="{!! Str::slug($group['name']) !!}"
+        class="sl-text-5xl sl-leading-tight sl-font-prose sl-text-heading"
+    >
+        {!! $group['name'] !!}
+    </h1>
+
+    {!! Parsedown::instance()->text($group['description']) !!}
+
+    @foreach($group['subgroups'] as $subgroupName => $subgroup)
+        @if($subgroupName !== "")
+            <h2 id="{!! Str::slug($group['name']) !!}-{!! Str::slug($subgroupName) !!}"
+                class="sl-text-3xl sl-leading-tight sl-font-prose sl-text-heading sl-mt-5 sl-mb-3"
+            >
+                {{ $subgroupName }}
+            </h2>
+            @php($subgroupDescription = collect($subgroup)->first(fn ($e) => $e->metadata->subgroupDescription)?->metadata?->subgroupDescription)
+            @if($subgroupDescription)
+                {!! Parsedown::instance()->text($subgroupDescription) !!}
+            @endif
+            <br>
+        @endif
+        @foreach($subgroup as $endpoint)
+            @include("scribe::themes.elements.endpoint")
+        @endforeach
+    @endforeach
+@endforeach
+

+ 360 - 0
resources/views/vendor/scribe/themes/elements/index.blade.php

@@ -0,0 +1,360 @@
+@php
+    use Knuckles\Scribe\Tools\WritingUtils as u;
+@endphp
+
+        <!doctype html>
+<html lang="en">
+<head>
+    <meta charset="utf-8">
+    <meta content="IE=edge,chrome=1" http-equiv="X-UA-Compatible">
+    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
+    <title>{!! $metadata['title'] !!}</title>
+
+    <link href="https://fonts.googleapis.com/css?family=PT+Sans&display=swap" rel="stylesheet">
+
+    <link rel="stylesheet" href="{!! $assetPathPrefix !!}css/theme-elements.style.css" media="screen">
+
+    <script src="https://cdn.jsdelivr.net/npm/lodash@4.17.10/lodash.min.js"></script>
+
+    <link rel="stylesheet"
+          href="https://unpkg.com/@highlightjs/cdn-assets@11.6.0/styles/docco.min.css">
+    <script src="https://unpkg.com/@highlightjs/cdn-assets@11.6.0/highlight.min.js"></script>
+    <script>hljs.highlightAll();</script>
+    <script type="module">
+        import {CodeJar} from 'https://medv.io/codejar/codejar.js'
+        window.CodeJar = CodeJar;
+    </script>
+
+    @if($tryItOut['enabled'] ?? true)
+        <script>
+            var tryItOutBaseUrl = "{{ $tryItOut['base_url'] ?? config('app.url') }}";
+            var useCsrf = Boolean({{ $tryItOut['use_csrf'] ?? null }});
+            var csrfUrl = "{{ $tryItOut['csrf_url'] ?? null }}";
+        </script>
+        <script src="{{ u::getVersionedAsset($assetPathPrefix.'js/tryitout.js') }}"></script>
+        <style>
+            .code-editor, .response-content {
+                color: whitesmoke;
+                background-color: transparent;
+            }
+            /*
+             Problem: we want syntax highlighting for the Try It Out JSON body code editor
+             However, the Try It Out area uses a dark background, while request and response samples
+             (which are already highlighted) use a light background. HighlightJS can only use one theme per document.
+             Our options:
+             1. Change the bg of one. => No, it looks out of place on the page.
+             2. Use the same highlighting for both. => Nope, one would be unreadable.
+             3. Copy styles for a dark-bg h1js theme and prefix them for the CodeEditor, which is what we're doing.
+             Since it's only JSON, we only need a few styles anyway.
+             Styles taken from the Nord theme: https://github.com/highlightjs/highlight.js/blob/3997c9b430a568d5ad46d96693b90a74fc01ea7f/src/styles/nord.css#L2
+             */
+            .code-editor > .hljs-attr {
+                color: #8FBCBB;
+            }
+            .code-editor > .hljs-string {
+                color: #A3BE8C;
+            }
+            .code-editor > .hljs-number {
+                color: #B48EAD;
+            }
+            .code-editor > .hljs-literal{
+                color: #81A1C1;
+            }
+
+        </style>
+
+        <script>
+            function tryItOut(btnElement) {
+                btnElement.disabled = true;
+
+                let endpointId = btnElement.dataset.endpoint;
+
+                let errorPanel = document.querySelector(`.tryItOut-error[data-endpoint=${endpointId}]`);
+                errorPanel.hidden = true;
+                let responsePanel = document.querySelector(`.tryItOut-response[data-endpoint=${endpointId}]`);
+                responsePanel.hidden = true;
+
+                let form = btnElement.form;
+                let { method, path, hasjsonbody: hasJsonBody} = form.dataset;
+                let body = {};
+                if (hasJsonBody === "1") {
+                    body = form.querySelector('.code-editor').textContent;
+                } else if (form.dataset.hasfiles === "1") {
+                    body = new FormData();
+                    form.querySelectorAll('input[data-component=body]')
+                        .forEach(el => {
+                            if (el.type === 'file') {
+                                if (el.files[0]) body.append(el.name, el.files[0])
+                            } else body.append(el.name, el.value);
+                        });
+                } else {
+                    form.querySelectorAll('input[data-component=body]').forEach(el => {
+                        _.set(body, el.name, el.value);
+                    });
+                }
+
+                const urlParameters = form.querySelectorAll('input[data-component=url]');
+                urlParameters.forEach(el => (path = path.replace(new RegExp(`\\{${el.name}\\??}`), el.value)));
+
+                const headers = Object.fromEntries(Array.from(form.querySelectorAll('input[data-component=header]'))
+                    .map(el => [el.name, (el.dataset.prefix || '') + el.value]));
+
+                const query = {}
+                form.querySelectorAll('input[data-component=query]').forEach(el => {
+                    _.set(query, el.name, el.value);
+                });
+
+                let preflightPromise = Promise.resolve();
+                if (window.useCsrf && window.csrfUrl) {
+                    preflightPromise = makeAPICall('GET', window.csrfUrl).then(() => {
+                        headers['X-XSRF-TOKEN'] = getCookie('XSRF-TOKEN');
+                    });
+                }
+
+                // content type has to be unset otherwise file upload won't work
+                if (form.dataset.hasfiles === "1") {
+                    delete headers['Content-Type'];
+                }
+
+                return preflightPromise.then(() => makeAPICall(method, path, body, query, headers, endpointId))
+                    .then(([responseStatus, statusText, responseContent, responseHeaders]) => {
+                        responsePanel.hidden = false;
+                        responsePanel.querySelector(`.response-status`).textContent = responseStatus + " " + statusText ;
+
+                        let contentEl = responsePanel.querySelector(`.response-content`);
+                        if (responseContent === '') {
+                            contentEl.textContent = contentEl.dataset.emptyResponseText;
+                            return;
+                        }
+
+                        // Prettify it if it's JSON
+                        let isJson = false;
+                        try {
+                            const jsonParsed = JSON.parse(responseContent);
+                            if (jsonParsed !== null) {
+                                isJson = true;
+                                responseContent = JSON.stringify(jsonParsed, null, 4);
+                            }
+                        } catch (e) {}
+
+                        // Replace HTML entities
+                        responseContent = responseContent.replace(/[<>&]/g, (i) => '&#' + i.charCodeAt(0) + ';');
+
+                        contentEl.innerHTML = responseContent;
+                        isJson && window.hljs.highlightElement(contentEl);
+                    })
+                    .catch(err => {
+                        console.log(err);
+                        let errorMessage = err.message || err;
+                        errorPanel.hidden = false;
+                        errorPanel.querySelector(`.error-message`).textContent = errorMessage;
+                    })
+                    .finally(() => { btnElement.disabled = false } );
+            }
+
+            window.addEventListener('DOMContentLoaded', () => {
+                document.querySelectorAll('.tryItOut-btn').forEach(el => {
+                    el.addEventListener('click', () => tryItOut(el));
+                });
+            })
+        </script>
+    @endif
+
+</head>
+
+<body>
+
+@if($metadata['example_languages'])
+    <script>
+        function switchExampleLanguage(lang) {
+            document.querySelectorAll(`.example-request`).forEach(el => el.style.display = 'none');
+            document.querySelectorAll(`.example-request-${lang}`).forEach(el => el.style.display = 'initial');
+            document.querySelectorAll(`.example-request-lang-toggle`).forEach(el => el.value = lang);
+        }
+    </script>
+@endif
+
+<script>
+    function switchExampleResponse(endpointId, index) {
+        document.querySelectorAll(`.example-response-${endpointId}`).forEach(el => el.style.display = 'none');
+        document.querySelectorAll(`.example-response-${endpointId}-${index}`).forEach(el => el.style.display = 'initial');
+        document.querySelectorAll(`.example-response-${endpointId}-toggle`).forEach(el => el.value = index);
+    }
+
+
+    /*
+     * Requirement: a div with class `expansion-chevrons`
+     *   (or `expansion-chevrons-solid` to use the solid version).
+     * Also add the `expanded` class if your div is expanded by default.
+     */
+    function toggleExpansionChevrons(evt) {
+        let elem = evt.currentTarget;
+
+        let chevronsArea = elem.querySelector('.expansion-chevrons');
+        const solid = chevronsArea.classList.contains('expansion-chevrons-solid');
+        const newState = chevronsArea.classList.contains('expanded') ? 'expand' : 'expanded';
+        if (newState === 'expanded') {
+            const selector = solid ? '#expanded-chevron-solid' : '#expanded-chevron';
+            const template = document.querySelector(selector);
+            const chevron = template.content.cloneNode(true);
+            chevronsArea.replaceChildren(chevron);
+            chevronsArea.classList.add('expanded');
+        } else {
+            const selector = solid ? '#expand-chevron-solid' : '#expand-chevron';
+            const template = document.querySelector(selector);
+            const chevron = template.content.cloneNode(true);
+            chevronsArea.replaceChildren(chevron);
+            chevronsArea.classList.remove('expanded');
+        }
+
+    }
+
+    /**
+     * 1. Make sure the children are inside the parent element
+     * 2. Add `expandable` class to the parent
+     * 3. Add `children` class to the children.
+     * 4. Wrap the default chevron SVG in a div with class `expansion-chevrons`
+     *   (or `expansion-chevrons-solid` to use the solid version).
+     *   Also add the `expanded` class if your div is expanded by default.
+     */
+    function toggleElementChildren(evt) {
+        let elem = evt.currentTarget;
+        let children = elem.querySelector(`.children`);
+        if (!children) return;
+
+        if (children.contains(event.target)) return;
+
+        let oldState = children.style.display
+        if (oldState === 'none') {
+            children.style.removeProperty('display');
+            toggleExpansionChevrons(evt);
+        } else {
+            children.style.display = 'none';
+            toggleExpansionChevrons(evt);
+        }
+
+        evt.stopPropagation();
+    }
+
+    function highlightSidebarItem(evt = null) {
+        if (evt && evt.oldURL) {
+            let oldHash = new URL(evt.oldURL).hash.slice(1);
+            if (oldHash) {
+                let previousItem = window['sidebar'].querySelector(`#toc-item-${oldHash}`);
+                previousItem.classList.remove('sl-bg-primary-tint');
+                previousItem.classList.add('sl-bg-canvas-100');
+            }
+        }
+
+        let newHash = location.hash.slice(1);
+        if (newHash) {
+            let item = window['sidebar'].querySelector(`#toc-item-${newHash}`);
+            item.classList.remove('sl-bg-canvas-100');
+            item.classList.add('sl-bg-primary-tint');
+        }
+    }
+
+    addEventListener('DOMContentLoaded', () => {
+        highlightSidebarItem();
+
+        document.querySelectorAll('.code-editor').forEach(elem => CodeJar(elem, (editor) => {
+            // highlight.js does not trim old tags,
+            // which means highlighting doesn't update on type (only on paste)
+            // See https://github.com/antonmedv/codejar/issues/18
+            editor.textContent = editor.textContent
+            return hljs.highlightElement(editor)
+        }));
+
+        document.querySelectorAll('.expandable').forEach(el => {
+            el.addEventListener('click', toggleElementChildren);
+        });
+
+        document.querySelectorAll('details').forEach(el => {
+            el.addEventListener('toggle', toggleExpansionChevrons);
+        });
+    });
+
+    addEventListener('hashchange', highlightSidebarItem);
+</script>
+
+<div class="sl-elements sl-antialiased sl-h-full sl-text-base sl-font-ui sl-text-body sl-flex sl-inset-0">
+
+    @include("scribe::themes.elements.sidebar")
+
+    <div class="sl-overflow-y-auto sl-flex-1 sl-w-full sl-px-16 sl-bg-canvas sl-py-16" style="max-width: 1500px;">
+
+        <div class="sl-mb-10">
+            <div class="sl-mb-4">
+                <h1 class="sl-text-5xl sl-leading-tight sl-font-prose sl-font-semibold sl-text-heading">
+                    {!! $metadata['title'] !!}
+                </h1>
+                @if($metadata['postman_collection_url'])
+                    <a title="Download Postman collection" class="sl-mx-1"
+                       href="{!! $metadata['postman_collection_url'] !!}" target="_blank">
+                        <small>Postman collection →</small>
+                    </a>
+                @endif
+                @if($metadata['openapi_spec_url'])
+                    <a title="Download OpenAPI spec" class="sl-mx-1"
+                       href="{!! $metadata['openapi_spec_url'] !!}" target="_blank">
+                        <small>OpenAPI spec →</small>
+                    </a>
+                @endif
+            </div>
+
+            <div class="sl-prose sl-markdown-viewer sl-my-4">
+                {!! $intro !!}
+
+                {!! $auth !!}
+            </div>
+        </div>
+
+        @include("scribe::themes.elements.groups")
+
+        <div class="sl-prose sl-markdown-viewer sl-my-5">
+            {!! $append !!}
+        </div>
+    </div>
+
+</div>
+
+<template id="expand-chevron">
+    <svg aria-hidden="true" focusable="false" data-prefix="fas"
+         data-icon="chevron-right"
+         class="svg-inline--fa fa-chevron-right fa-fw sl-icon sl-text-muted"
+         xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+        <path fill="currentColor"
+              d="M96 480c-8.188 0-16.38-3.125-22.62-9.375c-12.5-12.5-12.5-32.75 0-45.25L242.8 256L73.38 86.63c-12.5-12.5-12.5-32.75 0-45.25s32.75-12.5 45.25 0l192 192c12.5 12.5 12.5 32.75 0 45.25l-192 192C112.4 476.9 104.2 480 96 480z"></path>
+    </svg>
+</template>
+
+<template id="expanded-chevron">
+    <svg aria-hidden="true" focusable="false" data-prefix="fas"
+         data-icon="chevron-down"
+         class="svg-inline--fa fa-chevron-down fa-fw sl-icon sl-text-muted"
+         xmlns="http://www.w3.org/2000/svg" viewBox="0 0 448 512">
+        <path fill="currentColor"
+              d="M224 416c-8.188 0-16.38-3.125-22.62-9.375l-192-192c-12.5-12.5-12.5-32.75 0-45.25s32.75-12.5 45.25 0L224 338.8l169.4-169.4c12.5-12.5 32.75-12.5 45.25 0s12.5 32.75 0 45.25l-192 192C240.4 412.9 232.2 416 224 416z"></path>
+    </svg>
+</template>
+
+<template id="expand-chevron-solid">
+    <svg aria-hidden="true" focusable="false" data-prefix="fas" data-icon="caret-right"
+         class="svg-inline--fa fa-caret-right fa-fw sl-icon" role="img" xmlns="http://www.w3.org/2000/svg"
+         viewBox="0 0 256 512">
+        <path fill="currentColor"
+              d="M118.6 105.4l128 127.1C252.9 239.6 256 247.8 256 255.1s-3.125 16.38-9.375 22.63l-128 127.1c-9.156 9.156-22.91 11.9-34.88 6.943S64 396.9 64 383.1V128c0-12.94 7.781-24.62 19.75-29.58S109.5 96.23 118.6 105.4z"></path>
+    </svg>
+</template>
+
+<template id="expanded-chevron-solid">
+    <svg aria-hidden="true" focusable="false" data-prefix="fas"
+         data-icon="caret-down"
+         class="svg-inline--fa fa-caret-down fa-fw sl-icon" role="img"
+         xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+        <path fill="currentColor"
+              d="M310.6 246.6l-127.1 128C176.4 380.9 168.2 384 160 384s-16.38-3.125-22.63-9.375l-127.1-128C.2244 237.5-2.516 223.7 2.438 211.8S19.07 192 32 192h255.1c12.94 0 24.62 7.781 29.58 19.75S319.8 237.5 310.6 246.6z"></path>
+    </svg>
+</template>
+</body>
+</html>

+ 92 - 0
resources/views/vendor/scribe/themes/elements/sidebar.blade.php

@@ -0,0 +1,92 @@
+<div id="sidebar" class="sl-flex sl-overflow-y-auto sl-flex-col sl-sticky sl-inset-y-0 sl-pt-8 sl-bg-canvas-100 sl-border-r"
+     style="width: calc((100% - 1800px) / 2 + 300px); padding-left: calc((100% - 1800px) / 2); min-width: 300px; max-height: 100vh">
+    <div class="sl-flex sl-items-center sl-mb-5 sl-ml-4">
+        @if($metadata['logo'] != false)
+            <div class="sl-inline sl-overflow-x-hidden sl-overflow-y-hidden sl-mr-3 sl-rounded-lg"
+                 style="background-color: transparent;">
+                <img src="{{ $metadata['logo'] }}" height="30px" width="30px" alt="logo">
+            </div>
+        @endif
+        <h4 class="sl-text-paragraph sl-leading-snug sl-font-prose sl-font-semibold sl-text-heading">
+            {{ $metadata['title'] }}
+        </h4>
+    </div>
+
+    <div class="sl-flex sl-overflow-y-auto sl-flex-col sl-flex-grow sl-flex-shrink">
+        <div class="sl-overflow-y-auto sl-w-full sl-bg-canvas-100">
+            <div class="sl-my-3">
+                @foreach($headings as $h1)
+                    <div class="expandable">
+                        <div title="{!! $h1['name'] !!}" id="toc-item-{!! $h1['slug'] !!}"
+                             class="sl-flex sl-items-center sl-h-md sl-pr-4 sl-pl-4 sl-bg-canvas-100 hover:sl-bg-canvas-200 sl-cursor-pointer sl-select-none">
+                            <a href="#{!! $h1['slug'] !!}"
+                               class="sl-flex-1 sl-items-center sl-truncate sl-mr-1.5 sl-p-0">{!! $h1['name'] !!}</a>
+                            @if(count($h1['subheadings']) > 0)
+                                <div class="sl-flex sl-items-center sl-text-xs expansion-chevrons">
+                                    <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                         data-icon="chevron-right"
+                                         class="svg-inline--fa fa-chevron-right fa-fw sl-icon sl-text-muted"
+                                         xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                        <path fill="currentColor"
+                                              d="M96 480c-8.188 0-16.38-3.125-22.62-9.375c-12.5-12.5-12.5-32.75 0-45.25L242.8 256L73.38 86.63c-12.5-12.5-12.5-32.75 0-45.25s32.75-12.5 45.25 0l192 192c12.5 12.5 12.5 32.75 0 45.25l-192 192C112.4 476.9 104.2 480 96 480z"></path>
+                                    </svg>
+                                </div>
+                            @endif
+                        </div>
+
+                        @if(count($h1['subheadings']) > 0)
+                            <div class="children" style="display: none;">
+                                @foreach($h1['subheadings'] as $h2)
+                                    <div class="expandable">
+                                        <div class="sl-flex sl-items-center sl-h-md sl-pr-4 sl-pl-8 sl-bg-canvas-100 hover:sl-bg-canvas-200 sl-cursor-pointer sl-select-none"
+                                             id="toc-item-{!! $h2['slug'] !!}">
+                                            <div class="sl-flex-1 sl-items-center sl-truncate sl-mr-1.5 sl-p-0" title="{!! $h2['name'] !!}">
+                                                <a class="ElementsTableOfContentsItem sl-block sl-no-underline"
+                                                   href="#{!! $h2['slug'] !!}">
+                                                    {!! $h2['name'] !!}
+                                                </a>
+                                            </div>
+                                            @if(count($h2['subheadings']) > 0)
+                                                <div class="sl-flex sl-items-center sl-text-xs expansion-chevrons">
+                                                    <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                                         data-icon="chevron-right"
+                                                         class="svg-inline--fa fa-chevron-right fa-fw sl-icon sl-text-muted"
+                                                         xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                                        <path fill="currentColor"
+                                                              d="M96 480c-8.188 0-16.38-3.125-22.62-9.375c-12.5-12.5-12.5-32.75 0-45.25L242.8 256L73.38 86.63c-12.5-12.5-12.5-32.75 0-45.25s32.75-12.5 45.25 0l192 192c12.5 12.5 12.5 32.75 0 45.25l-192 192C112.4 476.9 104.2 480 96 480z"></path>
+                                                    </svg>
+                                                </div>
+                                            @endif
+                                        </div>
+
+                                        @if(count($h2['subheadings']) > 0)
+                                            <div class="children" style="display: none;">
+                                                @foreach($h2['subheadings'] as $h3)
+                                                    <a class="ElementsTableOfContentsItem sl-block sl-no-underline"
+                                                       href="#{!! $h3['slug'] !!}">
+                                                        <div title="{!! $h3['name'] !!}" id="toc-item-{!! $h3['slug'] !!}"
+                                                             class="sl-flex sl-items-center sl-h-md sl-pr-4 sl-pl-12 sl-bg-canvas-100 hover:sl-bg-canvas-200 sl-cursor-pointer sl-select-none">
+                                                            {!! $h3['name'] !!}
+                                                        </div>
+                                                    </a>
+                                                @endforeach
+                                            </div>
+                                        @endif
+                                    </div>
+                                @endforeach
+                            </div>
+                        @endif
+                    </div>
+                @endforeach
+            </div>
+
+        </div>
+        <div class="sl-flex sl-items-center sl-px-4 sl-py-3 sl-border-t">
+            {{ $metadata['last_updated'] }}
+        </div>
+
+        <div class="sl-flex sl-items-center sl-px-4 sl-py-3 sl-border-t">
+            <a href="http://github.com/knuckleswtf/scribe">Documentation powered by Scribe ✍</a>
+        </div>
+    </div>
+</div>

+ 322 - 0
resources/views/vendor/scribe/themes/elements/try_it_out.blade.php

@@ -0,0 +1,322 @@
+@php
+    use Knuckles\Scribe\Tools\Utils as u;
+    /** @var \Knuckles\Camel\Output\OutputEndpointData $endpoint */
+@endphp
+
+<div class="sl-inverted">
+    <div class="sl-overflow-y-hidden sl-rounded-lg">
+        <form class="TryItPanel sl-bg-canvas-100 sl-rounded-lg"
+              data-method="{{ $endpoint->httpMethods[0] }}"
+              data-path="{{ $endpoint->uri }}"
+              data-hasfiles="{{ $endpoint->hasFiles() ? 1 : 0 }}"
+              data-hasjsonbody="{{ $endpoint->hasJsonBody() ? 1 : 0 }}">
+            @if($endpoint->isAuthed() && $metadata['auth']['location'] !== 'body')
+                <div class="sl-panel sl-outline-none sl-w-full expandable">
+                    <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-4 sl-pl-3 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-cursor-pointer sl-select-none"
+                         role="button">
+                        <div class="sl-flex sl-flex-1 sl-items-center sl-h-lg">
+                            <div class="sl-flex sl-items-center sl-mr-1.5 expansion-chevrons expansion-chevrons-solid expanded">
+                                <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                     data-icon="caret-down"
+                                     class="svg-inline--fa fa-caret-down fa-fw sl-icon" role="img"
+                                     xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                    <path fill="currentColor"
+                                          d="M310.6 246.6l-127.1 128C176.4 380.9 168.2 384 160 384s-16.38-3.125-22.63-9.375l-127.1-128C.2244 237.5-2.516 223.7 2.438 211.8S19.07 192 32 192h255.1c12.94 0 24.62 7.781 29.58 19.75S319.8 237.5 310.6 246.6z"></path>
+                                </svg>
+                            </div>
+                            Auth
+                        </div>
+                    </div>
+                    <div class="sl-panel__content-wrapper sl-bg-canvas-100 children" role="region">
+                        <div class="ParameterGrid sl-p-4">
+                            <label aria-hidden="true"
+                                   for="auth-{{ $endpoint->endpointId() }}">{{ $metadata['auth']['name'] }}</label>
+                            <span class="sl-mx-3">:</span>
+                            <div class="sl-flex sl-flex-1">
+                                <div class="sl-input sl-flex-1 sl-relative">
+                                    <code>{{ $metadata['auth']['prefix'] }}</code>
+                                    <input aria-label="{{ $metadata['auth']['name'] }}"
+                                           id="auth-{{ $endpoint->endpointId() }}"
+                                           data-component="{{ $metadata['auth']['location'] }}"
+                                           data-prefix="{{ $metadata['auth']['prefix'] }}"
+                                           name="{{ $metadata['auth']['name'] }}"
+                                           placeholder="{{ $metadata['auth']['placeholder'] }}"
+                                           class="auth-value sl-relative {{ $metadata['auth']['prefix'] ? 'sl-w-3/5' : 'sl-w-full sl-pr-2.5 sl-pl-2.5' }} sl-h-md sl-text-base sl-rounded sl-border-transparent hover:sl-border-input focus:sl-border-primary sl-border">
+                                </div>
+                            </div>
+                        </div>
+                    </div>
+                </div>
+            @endif
+
+            @if(count($endpoint->headers))
+                <div class="sl-panel sl-outline-none sl-w-full expandable">
+                    <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-4 sl-pl-3 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-cursor-pointer sl-select-none"
+                         role="button">
+                        <div class="sl-flex sl-flex-1 sl-items-center sl-h-lg">
+                            <div class="sl-flex sl-items-center sl-mr-1.5 expansion-chevrons expansion-chevrons-solid expanded">
+                                <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                     data-icon="caret-down"
+                                     class="svg-inline--fa fa-caret-down fa-fw sl-icon" role="img"
+                                     xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                    <path fill="currentColor"
+                                          d="M310.6 246.6l-127.1 128C176.4 380.9 168.2 384 160 384s-16.38-3.125-22.63-9.375l-127.1-128C.2244 237.5-2.516 223.7 2.438 211.8S19.07 192 32 192h255.1c12.94 0 24.62 7.781 29.58 19.75S319.8 237.5 310.6 246.6z"></path>
+                                </svg>
+                            </div>
+                            Headers
+                        </div>
+                    </div>
+                    <div class="sl-panel__content-wrapper sl-bg-canvas-100 children" role="region">
+                        <div class="ParameterGrid sl-p-4">
+                            @foreach($endpoint->headers as $name => $example)
+                                @php
+                                    if($endpoint->isAuthed() && $metadata['auth']['location'] === 'header' && $name === $metadata['auth']['name']) continue;
+                                @endphp
+                                <label aria-hidden="true"
+                                       for="header-{{ $endpoint->endpointId() }}-{{ $name }}">{{ $name }}</label>
+                                <span class="sl-mx-3">:</span>
+                                <div class="sl-flex sl-flex-1">
+                                    <div class="sl-input sl-flex-1 sl-relative">
+                                        <input aria-label="{{ $name }}" name="{{ $name }}"
+                                               id="header-{{ $endpoint->endpointId() }}-{{ $name }}"
+                                               value="{{ $example }}" data-component="header"
+                                               class="sl-relative sl-w-full sl-h-md sl-text-base sl-pr-2.5 sl-pl-2.5 sl-rounded sl-border-transparent hover:sl-border-input focus:sl-border-primary sl-border">
+                                    </div>
+                                </div>
+                            @endforeach
+                        </div>
+                    </div>
+                </div>
+            @endif
+
+            @if(count($endpoint->urlParameters))
+                <div class="sl-panel sl-outline-none sl-w-full expandable">
+                    <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-4 sl-pl-3 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-cursor-pointer sl-select-none"
+                         role="button">
+                        <div class="sl-flex sl-flex-1 sl-items-center sl-h-lg">
+                            <div class="sl-flex sl-items-center sl-mr-1.5 expansion-chevrons expansion-chevrons-solid expanded">
+                                <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                     data-icon="caret-down"
+                                     class="svg-inline--fa fa-caret-down fa-fw sl-icon" role="img"
+                                     xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                    <path fill="currentColor"
+                                          d="M310.6 246.6l-127.1 128C176.4 380.9 168.2 384 160 384s-16.38-3.125-22.63-9.375l-127.1-128C.2244 237.5-2.516 223.7 2.438 211.8S19.07 192 32 192h255.1c12.94 0 24.62 7.781 29.58 19.75S319.8 237.5 310.6 246.6z"></path>
+                                </svg>
+                            </div>
+                            URL Parameters
+                        </div>
+                    </div>
+                    <div class="sl-panel__content-wrapper sl-bg-canvas-100 children" role="region">
+                        <div class="ParameterGrid sl-p-4">
+                            @foreach($endpoint->urlParameters as $name => $parameter)
+                                <label aria-hidden="true"
+                                       for="urlparam-{{ $endpoint->endpointId() }}-{{ $name }}">{{ $name }}</label>
+                                <span class="sl-mx-3">:</span>
+                                <div class="sl-flex sl-flex-1">
+                                    <div class="sl-input sl-flex-1 sl-relative">
+                                        <input aria-label="{{ $name }}" name="{{ $name }}"
+                                               id="urlparam-{{ $endpoint->endpointId() }}-{{ $name }}"
+                                               placeholder="{{ $parameter->description }}"
+                                               value="{{ $parameter->example }}" data-component="url"
+                                               class="sl-relative sl-w-full sl-h-md sl-text-base sl-pr-2.5 sl-pl-2.5 sl-rounded sl-border-transparent hover:sl-border-input focus:sl-border-primary sl-border">
+                                    </div>
+                                </div>
+                            @endforeach
+                        </div>
+                    </div>
+                </div>
+            @endif
+
+            @if(count($endpoint->queryParameters))
+                <div class="sl-panel sl-outline-none sl-w-full expandable">
+                    <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-4 sl-pl-3 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-cursor-pointer sl-select-none"
+                         role="button">
+                        <div class="sl-flex sl-flex-1 sl-items-center sl-h-lg">
+                            <div class="sl-flex sl-items-center sl-mr-1.5 expansion-chevrons expansion-chevrons-solid expanded">
+                                <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                     data-icon="caret-down"
+                                     class="svg-inline--fa fa-caret-down fa-fw sl-icon" role="img"
+                                     xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                    <path fill="currentColor"
+                                          d="M310.6 246.6l-127.1 128C176.4 380.9 168.2 384 160 384s-16.38-3.125-22.63-9.375l-127.1-128C.2244 237.5-2.516 223.7 2.438 211.8S19.07 192 32 192h255.1c12.94 0 24.62 7.781 29.58 19.75S319.8 237.5 310.6 246.6z"></path>
+                                </svg>
+                            </div>
+                            Query Parameters
+                        </div>
+                    </div>
+                    <div class="sl-panel__content-wrapper sl-bg-canvas-100 children" role="region">
+                        <div class="ParameterGrid sl-p-4">
+                            @foreach($endpoint->queryParameters as $name => $parameter)
+                                @php
+                                    /** @var \Knuckles\Camel\Output\Parameter $parameter */
+                                    if ($parameter->type == 'object') // Skip; individual object children are listed
+                                        continue;
+                                    if (str_contains($name, "[]"))
+                                        // This likely belongs to an obj-array (eg objs[].a); we only show the parent (objs[]), so skip
+                                        continue;
+                                    if($endpoint->isAuthed() && $metadata['auth']['location'] === 'query'
+                                    && $name === $metadata['auth']['name']) continue;
+                                @endphp
+                                <label aria-hidden="true"
+                                       for="queryparam-{{ $endpoint->endpointId() }}-{{ $name }}">{{ $name }}</label>
+                                <span class="sl-mx-3">:</span>
+                                <div class="sl-flex sl-flex-1">
+                                    <div class="sl-input sl-flex-1 sl-relative">
+                                        @if(str_ends_with($parameter->type, '[]'))
+                                            <input aria-label="{{ $name }}" name="{{ $name }}"
+                                                   id="queryparam-{{ $endpoint->endpointId() }}-{{ $name }}"
+                                                   placeholder="{{ $parameter->description }}"
+                                                   value="{{ json_encode($parameter->example) }}" data-component="query"
+                                                   class="sl-relative sl-w-full sl-h-md sl-text-base sl-pr-2.5 sl-pl-2.5 sl-rounded sl-border-transparent hover:sl-border-input focus:sl-border-primary sl-border"
+                                            >
+                                        @else
+                                            <input aria-label="{{ $name }}" name="{{ $name }}"
+                                                   id="queryparam-{{ $endpoint->endpointId() }}-{{ $name }}"
+                                                   placeholder="{{ $parameter->description }}"
+                                                   value="{{ $parameter->example }}" data-component="query"
+                                                   class="sl-relative sl-w-full sl-h-md sl-text-base sl-pr-2.5 sl-pl-2.5 sl-rounded sl-border-transparent hover:sl-border-input focus:sl-border-primary sl-border"
+                                            >
+                                        @endif
+                                    </div>
+                                </div>
+                            @endforeach
+                        </div>
+                    </div>
+                </div>
+            @endif
+
+            @if(count($endpoint->bodyParameters))
+                <div class="sl-panel sl-outline-none sl-w-full expandable">
+                    <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-4 sl-pl-3 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-cursor-pointer sl-select-none"
+                         role="button">
+                        <div class="sl-flex sl-flex-1 sl-items-center sl-h-lg">
+                            <div class="sl-flex sl-items-center sl-mr-1.5 expansion-chevrons expansion-chevrons-solid expanded">
+                                <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                     data-icon="caret-down"
+                                     class="svg-inline--fa fa-caret-down fa-fw sl-icon" role="img"
+                                     xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                    <path fill="currentColor"
+                                          d="M310.6 246.6l-127.1 128C176.4 380.9 168.2 384 160 384s-16.38-3.125-22.63-9.375l-127.1-128C.2244 237.5-2.516 223.7 2.438 211.8S19.07 192 32 192h255.1c12.94 0 24.62 7.781 29.58 19.75S319.8 237.5 310.6 246.6z"></path>
+                                </svg>
+                            </div>
+                            Body
+                        </div>
+                    </div>
+                    <div class="sl-panel__content-wrapper sl-bg-canvas-100 children" role="region">
+                        @if($endpoint->hasJsonBody())
+                            <div class="TextRequestBody sl-p-4">
+                                <div class="code-editor language-json"
+                                     id="json-body-{{ $endpoint->endpointId() }}"
+                                     style="font-family: var(--font-code); font-size: 12px; line-height: var(--lh-code);"
+                                >{!! json_encode($endpoint->getSampleBody(), JSON_PRETTY_PRINT) !!}</div>
+                            </div>
+                        @else
+                            <div class="ParameterGrid sl-p-4">
+                                @foreach($endpoint->bodyParameters as $name => $parameter)
+                                    @php
+                                        /** @var \Knuckles\Camel\Output\Parameter $parameter */
+                                        if ($parameter->type == 'object') // Skip; individual object children are listed
+                                            continue;
+                                        if (str_contains($name, "[]"))
+                                            // This likely belongs to an obj-array (eg objs[].a); we only show the parent (objs[]), so skip
+                                            continue;
+                                    @endphp
+                                    <label aria-hidden="true"
+                                           for="bodyparam-{{ $endpoint->endpointId() }}-{{ $name }}">{{ $name }}</label>
+                                    <span class="sl-mx-3">:</span>
+                                    <div class="sl-flex sl-flex-1">
+                                        <div class="sl-input sl-flex-1 sl-relative">
+                                            @if($parameter->type == 'file')
+                                                <input aria-label="{{ $name }}" name="{{ $name }}"
+                                                       id="bodyparam-{{ $endpoint->endpointId() }}-{{ $name }}"
+                                                       type="file" data-component="body"
+                                                       class="sl-relative sl-w-full sl-h-md sl-text-base sl-pr-2.5 sl-pl-2.5 sl-rounded sl-border-transparent hover:sl-border-input focus:sl-border-primary sl-border"
+                                                >
+                                            @elseif(str_ends_with($parameter->type, '[]'))
+                                                <input aria-label="{{ $name }}" name="{{ $name }}"
+                                                       id="bodyparam-{{ $endpoint->endpointId() }}-{{ $name }}"
+                                                       placeholder="{{ $parameter->description }}"
+                                                       value="{{ json_encode($parameter->example) }}" data-component="body"
+                                                       class="sl-relative sl-w-full sl-h-md sl-text-base sl-pr-2.5 sl-pl-2.5 sl-rounded sl-border-transparent hover:sl-border-input focus:sl-border-primary sl-border"
+                                                >
+                                            @else
+                                                <input aria-label="{{ $name }}" name="{{ $name }}"
+                                                       id="bodyparam-{{ $endpoint->endpointId() }}-{{ $name }}"
+                                                       placeholder="{{ $parameter->description }}"
+                                                       value="{{ $parameter->example }}" data-component="body"
+                                                       class="sl-relative sl-w-full sl-h-md sl-text-base sl-pr-2.5 sl-pl-2.5 sl-rounded sl-border-transparent hover:sl-border-input focus:sl-border-primary sl-border"
+                                                >
+                                            @endif
+                                        </div>
+                                    </div>
+                                @endforeach
+                            </div>
+                        @endif
+                    </div>
+                </div>
+            @endif
+
+            <div class="SendButtonHolder sl-mt-4 sl-p-4 sl-pt-0">
+                <div class="sl-stack sl-stack--horizontal sl-stack--2 sl-flex sl-flex-row sl-items-center">
+                    <button type="button" data-endpoint="{{ $endpoint->endpointId() }}"
+                            class="tryItOut-btn sl-button sl-h-sm sl-text-base sl-font-medium sl-px-1.5 sl-bg-primary hover:sl-bg-primary-dark active:sl-bg-primary-darker disabled:sl-bg-canvas-100 sl-text-on-primary disabled:sl-text-body sl-rounded sl-border-transparent sl-border disabled:sl-opacity-70"
+                    >
+                        {{ u::trans("scribe::try_it_out.send") }}
+                    </button>
+                </div>
+            </div>
+
+            <div data-endpoint="{{ $endpoint->endpointId() }}"
+                 class="tryItOut-error expandable sl-panel sl-outline-none sl-w-full" hidden>
+                <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-4 sl-pl-3 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-cursor-pointer sl-select-none"
+                     role="button">
+                    <div class="sl-flex sl-flex-1 sl-items-center sl-h-lg">
+                        <div class="sl-flex sl-items-center sl-mr-1.5 expansion-chevrons expansion-chevrons-solid expanded">
+                            <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                 data-icon="caret-down"
+                                 class="svg-inline--fa fa-caret-down fa-fw sl-icon" role="img"
+                                 xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                <path fill="currentColor"
+                                      d="M310.6 246.6l-127.1 128C176.4 380.9 168.2 384 160 384s-16.38-3.125-22.63-9.375l-127.1-128C.2244 237.5-2.516 223.7 2.438 211.8S19.07 192 32 192h255.1c12.94 0 24.62 7.781 29.58 19.75S319.8 237.5 310.6 246.6z"></path>
+                            </svg>
+                        </div>
+                        {{ u::trans("scribe::try_it_out.request_failed") }}
+                    </div>
+                </div>
+                <div class="sl-panel__content-wrapper sl-bg-canvas-100 children" role="region">
+                    <div class="sl-panel__content sl-p-4">
+                        <p class="sl-pb-2"><strong class="error-message"></strong></p>
+                        <p class="sl-pb-2">{{ u::trans("scribe::try_it_out.error_help") }}</p>
+                    </div>
+                </div>
+            </div>
+
+                <div data-endpoint="{{ $endpoint->endpointId() }}"
+                     class="tryItOut-response expandable sl-panel sl-outline-none sl-w-full" hidden>
+                    <div class="sl-panel__titlebar sl-flex sl-items-center sl-relative focus:sl-z-10 sl-text-base sl-leading-none sl-pr-4 sl-pl-3 sl-bg-canvas-200 sl-text-body sl-border-input focus:sl-border-primary sl-cursor-pointer sl-select-none"
+                         role="button">
+                        <div class="sl-flex sl-flex-1 sl-items-center sl-h-lg">
+                            <div class="sl-flex sl-items-center sl-mr-1.5 expansion-chevrons expansion-chevrons-solid expanded">
+                                <svg aria-hidden="true" focusable="false" data-prefix="fas"
+                                     data-icon="caret-down"
+                                     class="svg-inline--fa fa-caret-down fa-fw sl-icon" role="img"
+                                     xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
+                                    <path fill="currentColor"
+                                          d="M310.6 246.6l-127.1 128C176.4 380.9 168.2 384 160 384s-16.38-3.125-22.63-9.375l-127.1-128C.2244 237.5-2.516 223.7 2.438 211.8S19.07 192 32 192h255.1c12.94 0 24.62 7.781 29.58 19.75S319.8 237.5 310.6 246.6z"></path>
+                                </svg>
+                            </div>
+                            {{ u::trans("scribe::try_it_out.received_response") }}
+                        </div>
+                    </div>
+                    <div class="sl-panel__content-wrapper sl-bg-canvas-100 children" role="region">
+                        <div class="sl-panel__content sl-p-4">
+                            <p class="sl-pb-2 response-status"></p>
+                            <pre><code class="sl-pb-2 response-content language-json"
+                                       data-empty-response-text="<{{ u::trans("scribe::endpoint.responses.empty") }}>"
+                                       style="max-height: 300px;"></code></pre>
+                        </div>
+                    </div>
+                </div>
+        </form>
+    </div>
+</div>

+ 4 - 1
routes/api.php

@@ -2,8 +2,12 @@
 
 use App\Http\Controllers\Client\AccountController;
 use App\Http\Controllers\Client\UserController;
+use App\Http\Controllers\ScribeController;
 use Illuminate\Support\Facades\Route;
 
+// API文档相关
+Route::get('scribe/update-token/{mobile}', [ScribeController::class, 'updateAuthToken']);
+
 // 无需认证的公开路由
 Route::prefix('account')->group(function () {
     // 发送验证码
@@ -41,4 +45,3 @@ Route::middleware('auth:sanctum')->group(function () {
         Route::post('apply-coach', [UserController::class, 'applyCoach']);
     });
 });
-

+ 0 - 0
script/ansible/README.md


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