commit 0d9423e6e4a54a55fd12ef43f7c3340ea4b689fb Author: xuejiahao Date: Fri Oct 17 14:33:33 2025 +0800 upload code diff --git a/.eslintrc.js b/.eslintrc.js new file mode 100644 index 0000000..636efe9 --- /dev/null +++ b/.eslintrc.js @@ -0,0 +1,13 @@ +module.exports = { + extends: [ + '@cybozu/eslint-config/globals/kintone.js', + '@cybozu/eslint-config/lib/base.js', + '@cybozu/eslint-config/lib/kintone.js', + '@cybozu/eslint-config/lib/prettier.js', + ], + rules: { + 'prettier/prettier': ['error', { singleQuote: true }], + 'space-before-function-paren': 0, + 'object-curly-spacing': 0, + }, +}; diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..7202a8e --- /dev/null +++ b/.gitignore @@ -0,0 +1,29 @@ +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +pnpm-debug.log* +lerna-debug.log* + +node_modules +dist +dist-ssr +*.local + +# Editor directories and files +.vscode/* +!.vscode/extensions.json +.idea +.DS_Store +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? +data + +private.ppk +package-lock.json +yarn.lock \ No newline at end of file diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..a13dd61 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,7 @@ +{ + "singleQuote": true, + "semi": true, + "trailingComma": "all", + "printWidth": 120, + "bracketSpacing": true +} diff --git a/.vscode/extensions.json b/.vscode/extensions.json new file mode 100644 index 0000000..a7cea0b --- /dev/null +++ b/.vscode/extensions.json @@ -0,0 +1,3 @@ +{ + "recommendations": ["Vue.volar"] +} diff --git a/README.md b/README.md new file mode 100644 index 0000000..a48ad49 --- /dev/null +++ b/README.md @@ -0,0 +1,189 @@ + +# kintone-vue-ts-template + +使用 Vue 、ts 和 Vite 创建 kintone plugin 的初始化模板,先由 [create-plugin](https://cybozu.dev/ja/kintone/sdk/development-environment/create-plugin/) 生成之后再手动引入 Vue。 + +> プラグイン開発手順:https://cybozu.dev/ja/kintone/tips/development/plugins/development-plugin/ + +包括了以下 kintone 库: + +- [kintone-ui-component (v1.22.0)](https://cybozu.dev/ja/kintone/sdk/library/kintone-ui-component-v1/)([文档](https://ui-component.kintone.dev/ja/)) +- [@cybozu/eslint-config](https://cybozu.dev/ja/kintone/sdk/development-environment/eslint-config/) +- [@kintone/rest-api-client](https://cybozu.dev/ja/kintone/sdk/rest-api-client/kintone-javascript-client/)([文档](https://github.com/kintone/js-sdk/tree/main/packages/rest-api-client)) +- [kintone/plugin-packer](https://cybozu.dev/ja/kintone/sdk/development-environment/plugin-packer/) +- [@kintone/plugin-uploader](https://cybozu.dev/ja/kintone/sdk/development-environment/plugin-uploader/) +- [@kintone/dts-gen](https://cybozu.dev/ja/kintone/sdk/library/dts-gen/) + +# 使用步骤 + +首先进行安装 + +```bash +npm install # 或 yarn +``` + +随后需要修改项目信息: + +1. `package.json` +```diff +{ ++ "name": "项目名", +- "name": "kintone-vue-template", + "private": true, ++ "version": "版本号", +- "version": "0.0.0", + "type": "module", + "scripts": { + "vite:build": "vite build", + "build": "vite build && npm run pkg", + "build-upload": "npm run build && npm run upload", + "pkg": "kintone-plugin-packer --ppk private.ppk --out dist/plugin.zip dist/src", ++ "upload": "修改 kintone 域名、用户名和密码" +- "upload": "kintone-plugin-uploader --base-url https://alicorn.cybozu.com --username maxz --password 7ld7i8vd dist/plugin.zip " + }, + ... +``` + +2. `src/manifest.json` + +```diff +{ + ... ++ "icon": "icon 路径", +- "icon": "image/icon.png", + "config": { + ... ++ "required_params": [ "/config 页面必填的参数" ] +- "required_params": [ "buttonName" ] + }, ++ "name": "项目名", +- "name": { +- "en": "kintone Vue template", +- "ja": "kintone Vue テンプレート" +- }, ++ "description": "项目描述", +- "description": { +- "en": "kintone Vue template for creating plugin", +- "ja": "kintoneプラグイン作成用 Vue テンプレート" +- }, ++ "description": "项目网站", +- "homepage_url": { +- "en": "https://www.alicorns.co.jp/", +- "ja": "https://www.alicorns.co.jp/" +- }, + ... +} +``` + +# 编译流程 + +- `build` 会生成 `dist` 文件夹,以及 `plugin.zip` 文件 + +> 如果只需要 `dist` 文件夹,可以执行 `vite:build` + +```bash +npm run build # 或 yarn build + +# 仅生成 `dist` 文件夹 +npm run vite:build # 或 yarn vite:build +``` + +- `upload` 会将 `plugin.zip` 上传到 Kintone plugin + +```bash +npm run upload # 或 yarn upload +``` + +- `build-upload` 会顺序执行上面两步 + +```bash +npm run build-upload # 或 yarn build-upload +``` + +# 开发说明 + +## 项目结构 + +``` +├── src +│ ├── components +│ │ ├── basic +│ │ │ ├── PluginInput.vue +│ │ │ └── PluginLabel.vue +│ │ └── Config.vue +│ ├── css +│ │ ├── 51-modern-default.css +│ │ └── config.css +│ ├── i18n +│ │ ├── lang +│ │ │ ├── en.ts +│ │ │ └── ja.ts +│ │ └── index.ts +│ ├── js +│ │ ├── desktop.ts +│ │ └── mobile.ts +│ ├── types +│ │ ├── index.d.ts +│ │ ├── model.d.ts +│ │ └── my-kintone.d.ts +│ ├── main.ts +│ └── manifest.json +├── README.md +├── components.d.ts +├── env.d.ts +├── index.html +├── package.json +├── tsconfig.json +└── vite.config.ts +``` + +当前模板中已经有一个案例,可以参考并且修改。 + +1. 对于 Config 页面,请修改 `components/Config.vue` +2. 对于 PC 的 App 页面,请修改 `js/desktop.ts` +3. 对于 Mobile 的 App 页面,请修改 `js/mobile.ts` + +这三个文件是强制指定的,请不要修改。 + +### 关于 Config 页面 + +只有 **Config 页面** 的开发可以使用 Vue Template,其他页面请使用 `js` 文件。 + +基本上和 Vue 开发没有区别,但是有以下一些限制: +1. CSS 样式只能使用 `css` 文件 +2. `kintone-ui-component` 库有一些限制 + + +### 关于 CSS + +1. 暂时不支持在 Vue Template 中创建,**只能在 `css` 文件夹下新建 css 文件** +2. 目前在 `config` 页面引入了 `51-modern-default.css`,里面存放了 kintone 的一些样式 + - 如果不需要使用就在 `manifest.json` 中删除 + ```diff + "config": { + "html": "html/config.html", + "js": [ + "js/config.js" + ], + "css": [ + - "css/51-modern-default.css", + "css/config.css" + ], + "required_params": [ + "buttonName" + ] + }, + ``` + + + +### + + +**支持 Vue 页面**, +不支持 Vue 页面, + + +## 对于 `kintone-ui-component` 库的限制 + +v-model 语法糖无法使用 \ No newline at end of file diff --git a/components.d.ts b/components.d.ts new file mode 100644 index 0000000..f0e951b --- /dev/null +++ b/components.d.ts @@ -0,0 +1,14 @@ +/* eslint-disable */ +// @ts-nocheck +// Generated by unplugin-vue-components +// Read more: https://github.com/vuejs/core/pull/3399 +export {} + +/* prettier-ignore */ +declare module 'vue' { + export interface GlobalComponents { + Config: typeof import('./src/components/Config.vue')['default'] + PluginInput: typeof import('./src/components/basic/PluginInput.vue')['default'] + PluginLabel: typeof import('./src/components/basic/PluginLabel.vue')['default'] + } +} diff --git a/env.d.ts b/env.d.ts new file mode 100644 index 0000000..528bb9b --- /dev/null +++ b/env.d.ts @@ -0,0 +1,8 @@ +/// +/// + +declare module '*.vue' { + import { DefineComponent } from 'vue'; + const component: DefineComponent<{}, {}, any>; + export default component; +} diff --git a/index.html b/index.html new file mode 100644 index 0000000..81b9372 --- /dev/null +++ b/index.html @@ -0,0 +1,4 @@ +
+
+
+ \ No newline at end of file diff --git a/package.json b/package.json new file mode 100644 index 0000000..8981220 --- /dev/null +++ b/package.json @@ -0,0 +1,36 @@ +{ + "name": "kintone-vue-template", + "private": true, + "version": "0.0.0", + "type": "module", + "scripts": { + "vite:build": "vite build", + "build": "vite build && npm run pkg", + "build-upload": "npm run build && npm run upload", + "pkg": "kintone-plugin-packer --ppk private.ppk --out dist/plugin.zip dist/src", + "upload": "kintone-plugin-uploader --base-url https://alicorn.cybozu.com --username maxz --password 7ld7i8vd dist/plugin.zip " + }, + "dependencies": { + "@kintone/rest-api-client": "^5.7.5", + "kintone-ui-component": "1.22.0", + "rollup-plugin-css-only": "^4.5.2", + "vue": "^3.5.13", + "vue-i18n": "^11.0.1" + }, + "devDependencies": { + "@cybozu/eslint-config": "^23.0.0", + "@kintone/dts-gen": "^8.1.1", + "@kintone/plugin-packer": "^8.1.3", + "@kintone/plugin-uploader": "^9.1.2", + "@types/node-rsa": "^1.1.4", + "@vitejs/plugin-vue": "^5.2.1", + "@vue/tsconfig": "^0.7.0", + "eslint": "^8.57.0", + "node-rsa": "^1.1.1", + "rollup-plugin-copy": "^3.5.0", + "typescript": "^5.7.3", + "unplugin-vue-components": "^28.0.0", + "vite": "^6.0.1", + "vue-tsc": "^2.1.10" + } +} diff --git a/src/assets/icon.png b/src/assets/icon.png new file mode 100644 index 0000000..aed0c9f Binary files /dev/null and b/src/assets/icon.png differ diff --git a/src/components/Config.vue b/src/components/Config.vue new file mode 100644 index 0000000..e00154e --- /dev/null +++ b/src/components/Config.vue @@ -0,0 +1,81 @@ + + diff --git a/src/components/basic/PluginInput.vue b/src/components/basic/PluginInput.vue new file mode 100644 index 0000000..651c9e8 --- /dev/null +++ b/src/components/basic/PluginInput.vue @@ -0,0 +1,43 @@ + + + diff --git a/src/components/basic/PluginLabel.vue b/src/components/basic/PluginLabel.vue new file mode 100644 index 0000000..32691ed --- /dev/null +++ b/src/components/basic/PluginLabel.vue @@ -0,0 +1,15 @@ + + + \ No newline at end of file diff --git a/src/css/51-modern-default.css b/src/css/51-modern-default.css new file mode 100644 index 0000000..0b338ee --- /dev/null +++ b/src/css/51-modern-default.css @@ -0,0 +1,655 @@ +@charset "UTF-8"; +/* + * Copyright (c) 2014 Cybozu + * + * Licensed under the MIT License +*/ + +*[class|="kintoneplugin"] { + color: #333; + word-wrap: break-word; + font-size: 16px; + line-height: 1.5; +} + +:lang(us) *[class|="kintoneplugin"] { + font-family: 'HelveticaNeueW02-45Ligh', Arial, 'Hiragino Kaku Gothic ProN', Meiryo, sans-serif; +} + +:lang(ja) *[class|="kintoneplugin"] { + font-family: 'メイリオ', 'Hiragino Kaku Gothic ProN', Meiryo, sans-serif; +} + +:lang(zh) *[class|="kintoneplugin"] { + font-family: '微软雅黑', 'Microsoft YaHei', '新宋体', NSimSun, STHeiti, Hei, 'Heiti SC', sans-serif; +} + +/*Alert message*/ +/* +
+

Alert message、
Alert message

+

Alert message

+
+*/ +.kintoneplugin-alert { + position: relative; + display: block; + margin: 8px 0; + padding: 4px 18px; + background: #e74c3c; + color: #f6f6f6; +} + +/* Row for the settings */ +/* +
Settings 1
+
Settings 2
+*/ +.kintoneplugin-row { + margin-bottom: 24px; +} + +/* Heading for each settings*/ +/* Heading */ +/* +
Heading for the setting
+*/ +.kintoneplugin-label { + margin-bottom: 8px; + font-weight: bold; +} + +/* Title for each settings */ +/* Title */ +/* +
Title of the setting
+*/ +.kintoneplugin-title { + margin-bottom: 8px; +} + +/* Description for each settings */ +/* Description */ +/* +
Description of the setting
+*/ +.kintoneplugin-desc { + margin-bottom: 8px; + font-size: 14px; + color: #888; +} + +/* Required settings*/ +/* +
Title*
+*/ +.kintoneplugin-require { + color: #c09853; +} + +/* Input (text box) */ +/* +
+ +
+*/ +.kintoneplugin-input-outer { + position: relative; + display: inline-block; + vertical-align: top; +} + +.kintoneplugin-input-text { + display: inline-block; + box-sizing: border-box; + margin: 0; + padding: 0 8px; + height: 48px; + outline: none; + border: 1px solid #e3e7e8; + background-color: #fff; + box-shadow: 4px 4px 12px #f5f5f5 inset, -4px -4px 12px #f5f5f5 inset; + color: #a7a7a7; + font-size: 14px; +} + +.kintoneplugin-input-text:focus { + background-color: #e2f2fe; + box-shadow: none; + color: #000; +} + +/* Check box */ +/* For IE8, specify .lt-ie9 for the parent. */ +/* +
+
+
+
+*/ +.kintoneplugin-input-checkbox-item { + display: block; + margin-right: 16px; + margin-bottom: 8px; + padding-left: 1px; + max-width: 98%; +} + +.kintoneplugin-input-checkbox-item-block { + display: block; +} +.kintoneplugin-input-checkbox-item-inline { + display: inline-block; +} + +.kintoneplugin-input-checkbox-item:hover + label { + color: #666; +} + +.kintoneplugin-input-checkbox-item input[type="checkbox"] { + display: none; + cursor: pointer; +} + +.kintoneplugin-input-checkbox-item input[type="checkbox"] + label { + position: relative; + display: inline-block; + margin-left: 32px; + vertical-align: middle; + font-size: 14px; + cursor: pointer; +} + +.kintoneplugin-input-checkbox-item input[type="checkbox"][disabled] + label { + color: #bababa; + cursor: not-allowed; +} + +.kintoneplugin-input-checkbox-item input[type="checkbox"] + label:before { + position: absolute; + top: 50%; + left: -30px; + display: inline-block; + box-sizing: border-box; + margin-top: -11px; + width: 21px; + height: 21px; + background: #fff; + background: url() no-repeat center center #fff; + content: ""; +} + +.kintoneplugin-input-checkbox-item input[type="checkbox"]:checked + label:after { + position: absolute; + top: 50%; + left: -30px; + box-sizing: border-box; + margin-top: -11px; + width: 21px; + height: 21px; + background: url() no-repeat center center #fff; + content: ""; +} + +.kintoneplugin-input-checkbox-item input[type="checkbox"][disabled]:checked + label:after { + background-image: url(); +} + +.lt-ie9 .kintoneplugin-input-checkbox-item input[type="checkbox"] { + display: inline-block; + vertical-align: middle; +} + +.lt-ie9 .kintoneplugin-input-checkbox-item input[type="checkbox"] + label { + margin-left: 4px; +} + +.lt-ie9 .kintoneplugin-input-checkbox-item input[type="checkbox"] + label:before { + display: none; +} + +.lt-ie9 .kintoneplugin-input-checkbox-item input[type="checkbox"]:checked + label:after { + display: none; +} + +/* Dropdown */ +/* +
+
+
Dropdown
+
+
+
+
Option A
+
Option B
+
Option C
+
+*/ +.kintoneplugin-dropdown-outer { + display: inline-block; +} + +.kintoneplugin-dropdown { + position: relative; + display: inline-block; + overflow: hidden; + box-sizing: border-box; + margin-right: 8px; + margin-bottom: 0; + padding: 0 16px; + min-width: 80px; + max-width: 280px; + height: 48px; + border: 1px solid #e3e7e8; + background-color: #f7f9fa; + box-shadow: 1px 1px 1px #fff inset; + color: #3498db; + text-overflow: ellipsis; +} + +.kintoneplugin-dropdown:hover { + background-color: #f4f7f8; + cursor: pointer; +} + +.kintoneplugin-dropdown-selected { + padding-right: 48px; + background: url() no-repeat right center; +} + +.kintoneplugin-dropdown-selected-name { + color: #3498db; + font-size: 14px; + line-height: 48px; +} + +.kintoneplugin-dropdown-list { + padding: 12px 0 0 0; + min-width: 280px; + border: 1px solid #e3e7e8; + background-color: #fff; + box-shadow: 0 5px 10px rgba(0, 0, 0, 0.1); + line-height: 1; +} + +.kintoneplugin-dropdown-list-item { + padding: 1px 16px 8px 25px; + line-height: 1; + cursor: pointer; +} + +.kintoneplugin-dropdown-list-item-name { + font: normal 13px Arial, sans-serif; + line-height: 1; +} + +.kintoneplugin-dropdown-list-item-selected { + background: url() no-repeat 7px 4px; +} + +.kintoneplugin-dropdown-list-item-selected .kintoneplugin-dropdown-list-item-name { + color: #3498db; +} + +/* Dropdown (Simple) */ +/* +
+
+ +
+
+*/ +.kintoneplugin-select-outer { + display: inline-block; +} + +.kintoneplugin-select { + position: relative; + display: inline-block; + overflow: hidden; + box-sizing: border-box; + margin-right: 8px; + padding: 0 32px 0 8px; + min-width: 80px; + max-width: 280px; + height: 48px; + border: 1px solid #e3e7e8; + background-color: #f7f9fa; + box-shadow: 1px 1px 1px #fff inset; + text-overflow: ellipsis; +} + +.kintoneplugin-select:after { + position: absolute; + top: 0; + right: 0; + bottom: 0; + display: block; + width: 32px; + background: url() no-repeat center center transparent; + content: ""; + pointer-events: none; +} + +.kintoneplugin-select:hover { + background-color: #f4f7f8; + cursor: pointer; +} + +.lt-ie9 .kintoneplugin-select:hover { + background-color: transparent; +} + +.kintoneplugin-select select { + margin: 0; + min-width: 140px; + width: 144%; + height: 48px; + outline: none; + border: 0 none; + background-color: transparent; + color: #3498db; + font-size: 13px; + cursor: pointer; + -moz-appearance: none; + -webkit-appearance: none; + -ms-appearance: none; + appearance: none; +} + +.kintoneplugin-select option { + background-color: #fff; + color: #333; +} + +/* Radio Button */ +/* For IE8, specify .lt-ie9 for the parent. */ +/* +
+*/ +.kintoneplugin-input-radio-item { + display: inline-block; + margin-right: 16px; + margin-bottom: 8px; + padding-left: 1px; + max-width: 98%; + font-size: 14px; +} + +.kintoneplugin-input-radio-item:hover + label { + color: #666; +} + +.kintoneplugin-input-radio-item input[type="radio"] { + display: none; + cursor: pointer; +} + +.kintoneplugin-input-radio-item input[type="radio"] + label { + position: relative; + display: inline-block; + margin-left: 32px; + vertical-align: middle; + cursor: pointer; +} + +.kintoneplugin-input-radio-item input[type="checkbox"][disabled] + label { + color: #bababa; + cursor: not-allowed; +} + +.kintoneplugin-input-radio-item input[type="radio"] + label:before { + position: absolute; + top: 50%; + left: -30px; + box-sizing: border-box; + margin-top: -11px; + width: 21px; + height: 21px; + border: 1px solid #e3e7e8; + border-radius: 50%; + background: #fff; + box-shadow: 1px 1px 3px #f5f5f5 inset, -1px -1px 3px #f5f5f5 inset; + content: ""; + font-size: 14px; +} + +.kintoneplugin-input-radio-item input[type="radio"]:checked + label:after { + position: absolute; + top: 50%; + left: -26px; + margin-top: -7px; + width: 13px; + height: 13px; + border-radius: 50%; + background-color: #3498db; + content: ""; +} + +.lt-ie9 .kintoneplugin-input-radio-item input[type="radio"] { + display: inline-block; + vertical-align: middle; +} + +.lt-ie9 .kintoneplugin-input-radio-item input[type="radio"] + label { + margin-left: 4px; +} + +.lt-ie9 .kintoneplugin-input-radio-item input[type="radio"] + label:before { + display: none; +} + +.lt-ie9 .kintoneplugin-input-radio-item input[type="radio"]:checked + label:after { + display: none; +} + +/* Button */ +/* + +*/ +.kintoneplugin-button-normal { + display: inline-block; + box-sizing: border-box; + padding: 0 16px; + min-width: 163px; + height: 48px; + outline: none; + border: 1px solid #e3e7e8; + background-color: #f7f9fa; + box-shadow: 1px 1px 1px #fff inset; + color: #3498db; + text-align: center; + line-height: 48px; +} + +.kintoneplugin-button-normal:hover { + background-color: #c8d6dd; + box-shadow: none; + cursor: pointer; +} + +/* Disabled button */ +/* + +*/ +.kintoneplugin-button-disabled { + display: inline-block; + box-sizing: border-box; + padding: 0 16px; + min-width: 163px; + height: 48px; + outline: none; + border: 1px solid #e3e7e8; + background-color: #dbdcdd; + box-shadow: none; + color: #bababa; + text-align: center; + line-height: 48px; +} + +/* Dialog OK button */ +/* + +*/ +.kintoneplugin-button-dialog-ok { + display: inline-block; + box-sizing: border-box; + padding: 0 16px; + min-width: 163px; + height: 48px; + outline: none; + border: 1px solid #e3e7e8; + background-color: #3498db; + box-shadow: 1px 1px 1px #8ccbee inset; + color: #fff; + text-align: center; + line-height: 48px; +} + +.kintoneplugin-button-dialog-ok:hover { + background-color: #1d6fa5; + cursor: pointer; +} + +/* Dialog Cancel button */ +/* + +*/ +.kintoneplugin-button-dialog-cancel { + display: inline-block; + box-sizing: border-box; + padding: 0 16px; + min-width: 163px; + height: 48px; + outline: none; + border: 1px solid #e3e7e8; + background-color: #f7f9fa; + box-shadow: 1px 1px 1px #fff inset; + color: #3498db; + text-align: center; + line-height: 48px; +} + +.kintoneplugin-button-dialog-cancel:hover { + background-color: #c8d6dd; + box-shadow: none; + cursor: pointer; +} + + +/* Table */ +/* + + + + + + + + + + + + + +
Title1
+
+
+
+ +
+
+
+
+ + +
+*/ +.kintoneplugin-table { + border-collapse: collapse; + border-spacing: 0; + margin-left: 18px; + margin-bottom: 16px; +} + +.kintoneplugin-table-th { + border-color: #3498db; + height: 40px; + color: #fff; + box-sizing: border-box; + text-align: left; + font-weight: 400; + font-size: 12px; + white-space: nowrap; + border-width: 2px; + background-color: #3498db; +} + +.kintoneplugin-table-th .title { + display: inline-block; + padding: 4px 8px; + box-sizing: border-box; + min-width: 204px; +} + +.kintoneplugin-table-th-blankspace { + background-color: transparent; +} + +.kintoneplugin-table td { + border-color: #e3e7e8; + border-style: solid; + border-width: 0 1px 1px 0; + vertical-align: top; + padding: 4px 0; +} + +.kintoneplugin-table td:first-child { + border-left-width: 1px; +} + +.kintoneplugin-table td.kintoneplugin-table-td-operation { + border-right: 0; + border-bottom: 0; +} + +.kintoneplugin-table-td-control { + padding: 0 8px; +} + +.kintoneplugin-table-td-control-value { + overflow: hidden; + padding: 4px 0; + min-height: 21px; + color: #333; + background-color: transparent; +} + +.kintoneplugin-table-td-operation .kintoneplugin-button-add-row-image, .kintoneplugin-table-td-operation .kintoneplugin-button-remove-row-image { + display: inline-block; + width: 24px; + height: 24px; +} + +.kintoneplugin-table-td-operation .kintoneplugin-button-add-row-image { + margin-left: 12px; + background: url() center center no-repeat; + border: 1px solid transparent; +} +.kintoneplugin-table-td-operation .kintoneplugin-button-add-row-image:hover { + cursor: pointer; +} +.kintoneplugin-table-td-operation .kintoneplugin-button-remove-row-image { + margin-left: 4px; + background: url() center center no-repeat; + border: 1px solid transparent; +} + +.kintoneplugin-table-td-operation .kintoneplugin-button-remove-row-image:hover { + cursor: pointer; + background: url() center center no-repeat; +} diff --git a/src/css/config.css b/src/css/config.css new file mode 100644 index 0000000..449d29d --- /dev/null +++ b/src/css/config.css @@ -0,0 +1,73 @@ +/* 辅助类 */ +.flex-row { + display: flex; +} +.hidden { + visibility: hidden; +} + +/* config 页面 */ +#app { + width: 60vw; + min-width: 1030px; +} + +/* 最上面的说明 */ +.settings-heading { + padding: 1em 0; + margin-bottom: 8px; +} + +/* label 样式 */ +.kintoneplugin-label { + padding-left: 20px; + line-height: 40px; +} + +.kintoneplugin-label-required-icon { + font-size: 20px; + vertical-align: -3px; + color: #e74c3c; + margin-left: 4px; + line-height: 1; +} + +/* label input 单行的情况 */ +.flex-row .kintoneplugin-label { + margin: 0; + width: 8.5em; +} + +/* 底部按钮空间 */ +.action-area { + padding: 24px 0; + margin-bottom: 32px; + text-align: right; + border-top: none; +} +.save-btn { + margin-left: 16px; + margin-right: 24px; +} + +/* loading 遮罩 */ +#main-area { + position: relative; +} +#main-area [class^="kuc-spinner"][class$="__mask"] { + position: absolute; + background-color: white; +} +#main-area [class^="kuc-spinner"][class$="__spinner"] { + position: absolute; +} +#main-area [class^="kuc-spinner"][class$="__spinner__loader"] { + fill: #3498db; +} + +/* 输入框宽度 */ +.kuc-text-input { + --kuc-text-input-width: max(16vw, 200px); + --kuc-dropdown-toggle-width: max(16vw, 200px); + --kuc-combobox-toggle-width: max(16vw, 200px); +} \ No newline at end of file diff --git a/src/css/desktop.css b/src/css/desktop.css new file mode 100644 index 0000000..e69de29 diff --git a/src/css/mobile.css b/src/css/mobile.css new file mode 100644 index 0000000..e69de29 diff --git a/src/i18n/index.ts b/src/i18n/index.ts new file mode 100644 index 0000000..834fcea --- /dev/null +++ b/src/i18n/index.ts @@ -0,0 +1,13 @@ +import { createI18n } from 'vue-i18n'; +import ja from './lang/ja'; +import en from './lang/en'; + +const i18n = createI18n({ + legacy: false, + locale: kintone.getLoginUser().language || 'ja', + messages: { + ja, + en, + }, +}); +export default i18n; diff --git a/src/i18n/lang/en.ts b/src/i18n/lang/en.ts new file mode 100644 index 0000000..e54a4d3 --- /dev/null +++ b/src/i18n/lang/en.ts @@ -0,0 +1,14 @@ +export default { + config: { + title: 'kintone Vue template', + desc: 'kintone Vue template for creating plugin', + button: { + label: 'button name', + default: 'button' + }, + message: { + label: 'message', + default: '' + }, + }, +}; diff --git a/src/i18n/lang/ja.ts b/src/i18n/lang/ja.ts new file mode 100644 index 0000000..3cae0bf --- /dev/null +++ b/src/i18n/lang/ja.ts @@ -0,0 +1,14 @@ +export default { + config: { + title: 'kintone Vue テンプレート', + desc: 'kintoneプラグイン作成用 Vue テンプレート', + button: { + label: 'ボタン名', + default: 'ボタン' + }, + message: { + label: 'メッセージ', + default: '' + }, + }, +}; diff --git a/src/js/desktop.ts b/src/js/desktop.ts new file mode 100644 index 0000000..52a7fb1 --- /dev/null +++ b/src/js/desktop.ts @@ -0,0 +1,36 @@ +import type { Setting } from '@/types/model'; +import { KintoneRestAPIClient } from '@kintone/rest-api-client'; + +(function (PLUGIN_ID) { + kintone.events.on('app.record.index.show', () => { + // App id + const appId = kintone.app.getId()?.toString(); + if (!appId) return; + + // get setting + const setting: Setting = kintone.plugin.app.getConfig(PLUGIN_ID); + + // header space + const headerSpace = kintone.app.getHeaderMenuSpaceElement(); + if (!headerSpace) { + throw new Error('このページではヘッダー要素が利用できません。'); + } + + const btnId = 'template-btn-id'; + // ボタン追加 + if (document.getElementById(btnId)) return; + const button = new Kucs[KUC_VERSION].Button({ + text: setting.buttonName, + type: 'submit', + id: btnId, + }); + const client = new KintoneRestAPIClient(); + button.addEventListener('click', async () => { + const { plugins } = await client.app.getPlugins({ + app: appId + }) + alert(setting.message + "\n--------\n【Plugins】 " + plugins.map(p => p.name).join('、')); + }); + headerSpace.appendChild(button); + }); +})(kintone.$PLUGIN_ID); diff --git a/src/js/mobile.ts b/src/js/mobile.ts new file mode 100644 index 0000000..8bba15a --- /dev/null +++ b/src/js/mobile.ts @@ -0,0 +1,36 @@ +import type { Setting } from '@/types/model'; +import { KintoneRestAPIClient } from '@kintone/rest-api-client'; + +(function (PLUGIN_ID) { + kintone.events.on('mobile.app.record.index.show', () => { + // App id + const appId = kintone.app.getId()?.toString(); + if (!appId) return; + + // get setting + const setting: Setting = kintone.plugin.app.getConfig(PLUGIN_ID); + + // header space + const headerSpace = kintone.mobile.app.getHeaderSpaceElement(); + if (!headerSpace) { + throw new Error('このページではヘッダー要素が利用できません。'); + } + + const btnId = 'template-btn-id'; + // ボタン追加 + if (document.getElementById(btnId)) return; + const button = new Kucs[KUC_VERSION].MobileButton({ + text: setting.buttonName, + type: 'submit', + id: btnId, + }); + const client = new KintoneRestAPIClient(); + button.addEventListener('click', async () => { + const { plugins } = await client.app.getPlugins({ + app: appId + }) + alert(setting.message + "\n--------\n【Plugins】 " + plugins.map(p => p.name).join('、')); + }); + headerSpace.appendChild(button); + }); +})(kintone.$PLUGIN_ID); diff --git a/src/main.ts b/src/main.ts new file mode 100644 index 0000000..30e3c7c --- /dev/null +++ b/src/main.ts @@ -0,0 +1,5 @@ +import { createApp } from 'vue'; +import Config from './components/Config.vue'; +import i18n from './i18n'; + +createApp(Config, { pluginId: kintone.$PLUGIN_ID }).use(i18n).mount('#app'); diff --git a/src/manifest.json b/src/manifest.json new file mode 100644 index 0000000..1bd389a --- /dev/null +++ b/src/manifest.json @@ -0,0 +1,50 @@ +{ + "$schema": "https://raw.githubusercontent.com/kintone/js-sdk/%40kintone/plugin-manifest-validator%4010.2.0/packages/plugin-manifest-validator/manifest-schema.json", + "manifest_version": 1, + "version": 1, + "type": "APP", + "desktop": { + "js": [ + "js/kuc.min.js", + "js/desktop.js" + ], + "css": [ + "css/desktop.css" + ] + }, + "icon": "image/icon.png", + "config": { + "html": "html/config.html", + "js": [ + "js/config.js" + ], + "css": [ + "css/51-modern-default.css", + "css/config.css" + ], + "required_params": [ + "buttonName" + ] + }, + "name": { + "en": "kintone Vue template", + "ja": "kintone Vue テンプレート" + }, + "description": { + "en": "kintone Vue template for creating plugin", + "ja": "kintoneプラグイン作成用 Vue テンプレート" + }, + "homepage_url": { + "en": "https://www.alicorns.co.jp/", + "ja": "https://www.alicorns.co.jp/" + }, + "mobile": { + "js": [ + "js/kuc.min.js", + "js/mobile.js" + ], + "css": [ + "css/mobile.css" + ] + } +} \ No newline at end of file diff --git a/src/types/index.d.ts b/src/types/index.d.ts new file mode 100644 index 0000000..8a48749 --- /dev/null +++ b/src/types/index.d.ts @@ -0,0 +1,8 @@ +declare global { + const Kucs: { + [version: string]: any; + }; + const KUC_VERSION: string; + const KUC_VERSION_DASHED: string; +} +export {}; diff --git a/src/types/model.d.ts b/src/types/model.d.ts new file mode 100644 index 0000000..795a732 --- /dev/null +++ b/src/types/model.d.ts @@ -0,0 +1,4 @@ +export interface Setting { + buttonName: string; + message: string; +} \ No newline at end of file diff --git a/src/types/my-kintone.d.ts b/src/types/my-kintone.d.ts new file mode 100644 index 0000000..a91cc9c --- /dev/null +++ b/src/types/my-kintone.d.ts @@ -0,0 +1,11 @@ +import { KintoneFormFieldProperty } from '@kintone/rest-api-client'; + +export interface KucEvent { + detail: T; +} + +export type SelectType = + | KintoneFormFieldProperty.CheckBox + | KintoneFormFieldProperty.RadioButton + | KintoneFormFieldProperty.Dropdown + | KintoneFormFieldProperty.MultiSelect; diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..87293ab --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,15 @@ +{ + "extends": "@vue/tsconfig/tsconfig.dom.json", + "include": ["env.d.ts", "src/**/*", "src/**/*.vue", "components.d.ts"], + "exclude": ["src/**/__tests__/*"], + "files" : [ + "./node_modules/@kintone/dts-gen/kintone.d.ts", + ], + "compilerOptions": { + "esModuleInterop": true, + "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo", + "paths": { + "@/*": ["./src/*"] + } + } +} \ No newline at end of file diff --git a/vite.config.ts b/vite.config.ts new file mode 100644 index 0000000..725599e --- /dev/null +++ b/vite.config.ts @@ -0,0 +1,150 @@ +import { defineConfig } from 'vite' +import vue from '@vitejs/plugin-vue' +import copy from "rollup-plugin-copy"; +import * as path from 'path'; +import * as fs from 'fs'; +import Components from 'unplugin-vue-components/vite'; +import RSA from 'node-rsa'; + +// Read kintone-ui-component version from package.json +const packageJson = JSON.parse(fs.readFileSync(path.resolve(__dirname, 'package.json'), 'utf-8')); +const kucVersion = packageJson.dependencies['kintone-ui-component']; +const formattedVersion = kucVersion.replace(/\./g, '-'); // e.g., 1.22.0 -> 1-22-0 +const dottedVersion = kucVersion; // e.g., 1.22.0 -> 1.22.0 + +/** + * Create a private key for a kintone plugin + */ +export const generatePrivateKey = () => { + const key = new RSA({ b: 1024 }); + return key.exportKey("pkcs1-private"); +}; + +// Check if private.ppk exists, if not, generate it +const privateKeyPath = path.resolve(__dirname, 'private.ppk'); +if (!fs.existsSync(privateKeyPath)) { + const privateKey = generatePrivateKey(); + fs.writeFileSync(privateKeyPath, privateKey); +} + +function replaceKucTagsPlugin() { + return { + name: 'vite-plugin-replace-tags', + async load(id) { + if (id.endsWith('.vue')) { + const content = await fs.promises.readFile(id, 'utf-8'); + + const usedComponent = {} + + let res = content + .replace(new RegExp(``, 'g'), (match, p1) => ``) + .replace(new RegExp(`]*)>`, 'g'), (match, p1, p2) => { + usedComponent[p1] = true; + return `` + }); + if (Object.keys(usedComponent).length) { + let importScript = ''; + res = importScript + res; + } + + return res; + } + } + }; +} + +// https://vite.dev/config/ +export default defineConfig({ + plugins: [ + vue({ + template: { + compilerOptions: { + isCustomElement: (tag) => tag.startsWith("kuc-"), + } + } + }), + Components(), + copy({ + targets: [ + { src: 'dist/index.html', dest: 'dist/src/html', rename: 'config.html' }, + { src: 'src/assets/*.js', dest: 'dist/src/js' }, + { src: 'src/assets/*.png', dest: 'dist/src/image' }, + { src: 'src/css/*', dest: 'dist/src/css' }, + { src: 'node_modules/kintone-ui-component/umd/kuc.min.js', dest: 'dist/src/js' }, + ], + hook: 'writeBundle' // 指定在何时复制文件 + }), + { + name: 'manifest-updater', + writeBundle() { + try { + const manifestPath = path.resolve(__dirname, 'dist/src/manifest.json'); + const destDir = path.dirname(manifestPath); + fs.mkdirSync(destDir, { recursive: true }); + fs.copyFileSync(path.resolve(__dirname, 'src/manifest.json'), manifestPath); + const manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf-8')); + const viteManifestPath = path.resolve(__dirname, 'dist/.vite/manifest.json'); + if (!fs.existsSync(viteManifestPath)) return; + + const viteManifest = JSON.parse(fs.readFileSync(viteManifestPath, 'utf-8')); + + // 收集所有生成的 chunk 文件 + const chunkFiles: string[] = []; + Object.values(viteManifest).forEach((value: any) => { + if (value.file && value.file.endsWith('.chunk.js')) { + chunkFiles.push('js/' + path.basename(value.file)); + } + }); + + // 更新 desktop 和 mobile 的 js 数组,包含 chunk 文件 + if (chunkFiles.length > 0) { + manifest.desktop.js = ['js/kuc.min.js', 'js/desktop.js', ...chunkFiles]; + manifest.mobile.js = ['js/kuc.min.js', 'js/mobile.js', ...chunkFiles]; + } + + fs.writeFileSync(manifestPath, JSON.stringify(manifest, null, 2)); + } catch (error) { + console.error('Error updating manifest:', error); + } + } + }, + replaceKucTagsPlugin() + ], + resolve: { + alias: { + '@': path.resolve(__dirname, 'src'), + }, + }, + build: { + rollupOptions: { + input: { + config: path.resolve(__dirname, 'index.html'), + desktop: path.resolve(__dirname, 'src/js/desktop.ts'), + mobile: path.resolve(__dirname, 'src/js/mobile.ts'), + }, + output: { + entryFileNames: (chunkInfo) => { + return 'src/js/[name].js'; // 默认处理为 JS 文件 + }, + chunkFileNames: 'src/js/[name]-[hash].chunk.js', + assetFileNames: 'src/[ext]/[name].[ext]', + }, + }, + sourcemap: 'inline', + manifest: true, + }, + define: { + KUC_VERSION: JSON.stringify(dottedVersion), + KUC_VERSION_DASHED: JSON.stringify(formattedVersion) + } +})