Add impl for WeChat on HarmonyOS (#153)

Signed-off-by: yeliulee <yeliuleet@gmail.com>
This commit is contained in:
Sherman Chu
2024-11-21 23:07:38 +08:00
committed by GitHub
parent ecdec656b2
commit a37c68aa21
44 changed files with 1129 additions and 5 deletions

View File

@ -1,3 +1,7 @@
## 6.1.0
* OpenHarmony
## 6.0.2
* Android

View File

@ -106,6 +106,22 @@ https://${your applinks domain}/universal_link/${example_app}/wechat/
> SDK universal_link SDK path
### OpenHarmony / HarmonyOS
> OpenHarmony SDK API, 使
module.json5 scheme
```json5
{
"module": {
"querySchemes": [
"weixin"
],
}
}
```
### Flutter
*

19
example/ohos/.gitignore vendored Normal file
View File

@ -0,0 +1,19 @@
/node_modules
/oh_modules
/local.properties
/.idea
**/build
/.hvigor
.cxx
/.clangd
/.clang-format
/.clang-tidy
**/.test
*.har
**/BuildProfile.ets
**/oh-package-lock.json5
**/src/main/resources/rawfile/flutter_assets/
**/libs/arm64-v8a/libapp.so
**/libs/arm64-v8a/libflutter.so
**/libs/arm64-v8a/libvmservice_snapshot.so

View File

@ -0,0 +1,10 @@
{
"app": {
"bundleName": "io.github.v7lin.wechat_kit_example",
"vendor": "example",
"versionCode": 1000000,
"versionName": "1.0.0",
"icon": "$media:app_icon",
"label": "$string:app_name"
}
}

View File

@ -0,0 +1,8 @@
{
"string": [
{
"name": "app_name",
"value": "wechat_kit_example"
}
]
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.6 KiB

View File

@ -0,0 +1,54 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
{
"app": {
"signingConfigs": [],
"products": [
{
"name": "default",
"signingConfig": "default",
"compatibleSdkVersion": "5.0.0(12)",
"runtimeOS": "HarmonyOS",
}
]
},
"modules": [
{
"name": "entry",
"srcPath": "./entry",
"targets": [
{
"name": "default",
"applyToProducts": [
"default"
]
}
]
},
{
"name": "wechat_kit",
"srcPath": "../../ohos",
"targets": [
{
"name": "default",
"applyToProducts": [
"default"
]
}
]
}
]
}

7
example/ohos/entry/.gitignore vendored Normal file
View File

@ -0,0 +1,7 @@
/node_modules
/oh_modules
/.preview
/build
/.cxx
/.test

View File

@ -0,0 +1,29 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
{
"apiType": 'stageMode',
"buildOption": {
},
"targets": [
{
"name": "default",
"runtimeOS": "HarmonyOS"
},
{
"name": "ohosTest",
}
]
}

View File

@ -0,0 +1,17 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
// Script for compiling build behavior. It is built in the build plug-in and cannot be modified currently.
export { hapTasks } from '@ohos/hvigor-ohos-plugin';

View File

@ -0,0 +1,27 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
{
"name": "entry",
"version": "1.0.0",
"description": "Please describe the basic information.",
"main": "",
"author": "",
"license": "",
"dependencies": {
"wechat_kit": "file:../../../ohos"
},
}

View File

@ -0,0 +1,24 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { FlutterAbility, FlutterEngine } from '@ohos/flutter_ohos';
import { GeneratedPluginRegistrant } from '../plugins/GeneratedPluginRegistrant';
export default class EntryAbility extends FlutterAbility {
configureFlutterEngine(flutterEngine: FlutterEngine) {
super.configureFlutterEngine(flutterEngine)
GeneratedPluginRegistrant.registerWith(flutterEngine)
}
}

View File

@ -0,0 +1,38 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import common from '@ohos.app.ability.common';
import { FlutterPage } from '@ohos/flutter_ohos'
let storage = LocalStorage.getShared()
const EVENT_BACK_PRESS = 'EVENT_BACK_PRESS'
@Entry(storage)
@Component
struct Index {
private context = getContext(this) as common.UIAbilityContext
@LocalStorageLink('viewId') viewId: string = "";
build() {
Column() {
FlutterPage({ viewId: this.viewId })
}
}
onBackPress(): boolean {
this.context.eventHub.emit(EVENT_BACK_PRESS)
return true
}
}

View File

@ -0,0 +1,53 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
{
"module": {
"name": "entry",
"type": "entry",
"description": "$string:module_desc",
"mainElement": "EntryAbility",
"deviceTypes": [
"phone"
],
"deliveryWithInstall": true,
"installationFree": false,
"pages": "$profile:main_pages",
"abilities": [
{
"name": "EntryAbility",
"srcEntry": "./ets/entryability/EntryAbility.ets",
"description": "$string:EntryAbility_desc",
"icon": "$media:icon",
"label": "$string:EntryAbility_label",
"startWindowIcon": "$media:icon",
"startWindowBackground": "$color:start_window_background",
"exported": true,
"skills": [
{
"entities": [
"entity.system.home"
],
"actions": [
"action.system.home"
]
}
]
}
],
"requestPermissions": [
{"name" : "ohos.permission.INTERNET"},
]
}
}

View File

@ -0,0 +1,8 @@
{
"color": [
{
"name": "start_window_background",
"value": "#FFFFFF"
}
]
}

View File

@ -0,0 +1,16 @@
{
"string": [
{
"name": "module_desc",
"value": "module description"
},
{
"name": "EntryAbility_desc",
"value": "description"
},
{
"name": "EntryAbility_label",
"value": "wechat_kit_example"
}
]
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.6 KiB

View File

@ -0,0 +1,8 @@
{
"string": [
{
"name": "enable_impeller",
"value": "true"
}
]
}

View File

@ -0,0 +1,5 @@
{
"src": [
"pages/Index"
]
}

View File

@ -0,0 +1,16 @@
{
"string": [
{
"name": "module_desc",
"value": "module description"
},
{
"name": "EntryAbility_desc",
"value": "description"
},
{
"name": "EntryAbility_label",
"value": "wechat_kit_example"
}
]
}

View File

@ -0,0 +1,16 @@
{
"string": [
{
"name": "module_desc",
"value": "模块描述"
},
{
"name": "EntryAbility_desc",
"value": "description"
},
{
"name": "EntryAbility_label",
"value": "wechat_kit_example"
}
]
}

View File

@ -0,0 +1,50 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import hilog from '@ohos.hilog';
import { describe, beforeAll, beforeEach, afterEach, afterAll, it, expect } from '@ohos/hypium'
export default function abilityTest() {
describe('ActsAbilityTest', function () {
// Defines a test suite. Two parameters are supported: test suite name and test suite function.
beforeAll(function () {
// Presets an action, which is performed only once before all test cases of the test suite start.
// This API supports only one parameter: preset action function.
})
beforeEach(function () {
// Presets an action, which is performed before each unit test case starts.
// The number of execution times is the same as the number of test cases defined by **it**.
// This API supports only one parameter: preset action function.
})
afterEach(function () {
// Presets a clear action, which is performed after each unit test case ends.
// The number of execution times is the same as the number of test cases defined by **it**.
// This API supports only one parameter: clear action function.
})
afterAll(function () {
// Presets a clear action, which is performed after all test cases of the test suite end.
// This API supports only one parameter: clear action function.
})
it('assertContain',0, function () {
// Defines a test case. This API supports three parameters: test case name, filter parameter, and test case function.
hilog.info(0x0000, 'testTag', '%{public}s', 'it begin');
let a = 'abc'
let b = 'b'
// Defines a variety of assertion methods, which are used to declare expected boolean conditions.
expect(a).assertContain(b)
expect(a).assertEqual(a)
})
})
}

View File

@ -0,0 +1,20 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import abilityTest from './Ability.test'
export default function testsuite() {
abilityTest()
}

View File

@ -0,0 +1,63 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import UIAbility from '@ohos.app.ability.UIAbility';
import AbilityDelegatorRegistry from '@ohos.app.ability.abilityDelegatorRegistry';
import hilog from '@ohos.hilog';
import { Hypium } from '@ohos/hypium';
import testsuite from '../test/List.test';
import window from '@ohos.window';
export default class TestAbility extends UIAbility {
onCreate(want, launchParam) {
hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onCreate');
hilog.info(0x0000, 'testTag', '%{public}s', 'want param:' + JSON.stringify(want) ?? '');
hilog.info(0x0000, 'testTag', '%{public}s', 'launchParam:'+ JSON.stringify(launchParam) ?? '');
var abilityDelegator: any
abilityDelegator = AbilityDelegatorRegistry.getAbilityDelegator()
var abilityDelegatorArguments: any
abilityDelegatorArguments = AbilityDelegatorRegistry.getArguments()
hilog.info(0x0000, 'testTag', '%{public}s', 'start run testcase!!!');
Hypium.hypiumTest(abilityDelegator, abilityDelegatorArguments, testsuite)
}
onDestroy() {
hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onDestroy');
}
onWindowStageCreate(windowStage: window.WindowStage) {
hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onWindowStageCreate');
windowStage.loadContent('testability/pages/Index', (err, data) => {
if (err.code) {
hilog.error(0x0000, 'testTag', 'Failed to load the content. Cause: %{public}s', JSON.stringify(err) ?? '');
return;
}
hilog.info(0x0000, 'testTag', 'Succeeded in loading the content. Data: %{public}s',
JSON.stringify(data) ?? '');
});
}
onWindowStageDestroy() {
hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onWindowStageDestroy');
}
onForeground() {
hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onForeground');
}
onBackground() {
hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onBackground');
}
}

View File

@ -0,0 +1,49 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import hilog from '@ohos.hilog';
@Entry
@Component
struct Index {
aboutToAppear() {
hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility index aboutToAppear');
}
@State message: string = 'Hello World'
build() {
Row() {
Column() {
Text(this.message)
.fontSize(50)
.fontWeight(FontWeight.Bold)
Button() {
Text('next page')
.fontSize(20)
.fontWeight(FontWeight.Bold)
}.type(ButtonType.Capsule)
.margin({
top: 20
})
.backgroundColor('#0D9FFB')
.width('35%')
.height('5%')
.onClick(()=>{
})
}
.width('100%')
}
.height('100%')
}
}

View File

@ -0,0 +1,64 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import hilog from '@ohos.hilog';
import TestRunner from '@ohos.application.testRunner';
import AbilityDelegatorRegistry from '@ohos.app.ability.abilityDelegatorRegistry';
var abilityDelegator = undefined
var abilityDelegatorArguments = undefined
async function onAbilityCreateCallback() {
hilog.info(0x0000, 'testTag', '%{public}s', 'onAbilityCreateCallback');
}
async function addAbilityMonitorCallback(err: any) {
hilog.info(0x0000, 'testTag', 'addAbilityMonitorCallback : %{public}s', JSON.stringify(err) ?? '');
}
export default class OpenHarmonyTestRunner implements TestRunner {
constructor() {
}
onPrepare() {
hilog.info(0x0000, 'testTag', '%{public}s', 'OpenHarmonyTestRunner OnPrepare ');
}
async onRun() {
hilog.info(0x0000, 'testTag', '%{public}s', 'OpenHarmonyTestRunner onRun run');
abilityDelegatorArguments = AbilityDelegatorRegistry.getArguments()
abilityDelegator = AbilityDelegatorRegistry.getAbilityDelegator()
var testAbilityName = abilityDelegatorArguments.bundleName + '.TestAbility'
let lMonitor = {
abilityName: testAbilityName,
onAbilityCreate: onAbilityCreateCallback,
};
abilityDelegator.addAbilityMonitor(lMonitor, addAbilityMonitorCallback)
var cmd = 'aa start -d 0 -a TestAbility' + ' -b ' + abilityDelegatorArguments.bundleName
var debug = abilityDelegatorArguments.parameters['-D']
if (debug == 'true')
{
cmd += ' -D'
}
hilog.info(0x0000, 'testTag', 'cmd : %{public}s', cmd);
abilityDelegator.executeShellCommand(cmd,
(err: any, d: any) => {
hilog.info(0x0000, 'testTag', 'executeShellCommand : err : %{public}s', JSON.stringify(err) ?? '');
hilog.info(0x0000, 'testTag', 'executeShellCommand : data : %{public}s', d.stdResult ?? '');
hilog.info(0x0000, 'testTag', 'executeShellCommand : data : %{public}s', d.exitCode ?? '');
})
hilog.info(0x0000, 'testTag', '%{public}s', 'OpenHarmonyTestRunner onRun end');
}
}

View File

@ -0,0 +1,51 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
{
"module": {
"name": "entry_test",
"type": "feature",
"description": "$string:module_test_desc",
"mainElement": "TestAbility",
"deviceTypes": [
"phone"
],
"deliveryWithInstall": true,
"installationFree": false,
"pages": "$profile:test_pages",
"abilities": [
{
"name": "TestAbility",
"srcEntry": "./ets/testability/TestAbility.ets",
"description": "$string:TestAbility_desc",
"icon": "$media:icon",
"label": "$string:TestAbility_label",
"exported": true,
"startWindowIcon": "$media:icon",
"startWindowBackground": "$color:start_window_background",
"skills": [
{
"actions": [
"action.system.home"
],
"entities": [
"entity.system.home"
]
}
]
}
]
}
}

View File

@ -0,0 +1,8 @@
{
"color": [
{
"name": "start_window_background",
"value": "#FFFFFF"
}
]
}

View File

@ -0,0 +1,16 @@
{
"string": [
{
"name": "module_test_desc",
"value": "test ability description"
},
{
"name": "TestAbility_desc",
"value": "the test ability"
},
{
"name": "TestAbility_label",
"value": "test label"
}
]
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.6 KiB

View File

@ -0,0 +1,5 @@
{
"src": [
"testability/pages/Index"
]
}

View File

@ -0,0 +1,20 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
{
"modelVersion": "5.0.0",
"dependencies": {
}
}

View File

@ -0,0 +1,21 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { appTasks } from '@ohos/hvigor-ohos-plugin';
export default {
system: appTasks, /* Built-in plugin of Hvigor. It cannot be modified. */
plugins:[] /* Custom plugin to extend the functionality of Hvigor. */
}

View File

@ -0,0 +1,34 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
{
"modelVersion": "5.0.0",
"name": "wechat_kit_example",
"version": "1.0.0",
"description": "Please describe the basic information.",
"main": "",
"author": "",
"license": "",
"dependencies": {
"@ohos/flutter_ohos": "file:./har/flutter.har"
},
"devDependencies": {
"@ohos/hypium": "1.0.6"
},
"overrides": {
"@ohos/flutter_ohos": "file:./har/flutter.har",
"wechat_kit": "file:../../ohos"
}
}

View File

@ -136,7 +136,7 @@ class MethodChannelWechatKit extends WechatKitPlatform {
String? state,
int type = WechatAuthType.kNormal,
}) {
assert((Platform.isAndroid && type == WechatAuthType.kNormal) ||
assert((!Platform.isIOS && type == WechatAuthType.kNormal) ||
(Platform.isIOS &&
<int>[WechatAuthType.kNormal, WechatAuthType.kWeb].contains(type)));
return methodChannel.invokeMethod<void>('auth', <String, dynamic>{
@ -235,14 +235,14 @@ class MethodChannelWechatKit extends WechatKitPlatform {
/// https://developers.weixin.qq.com/doc/oplatform/Mobile_App/Share_and_Favorites/iOS.html
assert(
(imageData != null &&
((Platform.isAndroid &&
((!Platform.isIOS &&
imageData.lengthInBytes <= 1 * 1024 * 1024) ||
(Platform.isIOS &&
imageData.lengthInBytes <= 10 * 1024 * 1024))) ||
(imageUri != null &&
imageUri.isScheme('file') &&
imageUri.toFilePath().length <= 10 * 1024 &&
((Platform.isAndroid &&
((!Platform.isIOS &&
File.fromUri(imageUri).lengthSync() <=
25 * 1024 * 1024) ||
(Platform.isIOS &&
@ -282,7 +282,7 @@ class MethodChannelWechatKit extends WechatKitPlatform {
fileUri.toFilePath().length <= 10 * 1024 &&
File.fromUri(fileUri).lengthSync() <= 10 * 1024 * 1024),
);
assert(Platform.isAndroid || (fileExtension?.isNotEmpty ?? false));
assert(!Platform.isIOS || (fileExtension?.isNotEmpty ?? false));
return methodChannel.invokeMethod<void>(
'shareFile',
<String, dynamic>{

10
ohos/.gitignore vendored Normal file
View File

@ -0,0 +1,10 @@
/node_modules
/oh_modules
/.preview
/.idea
/build
/.cxx
/.test
/BuildProfile.ets
/oh-package-lock.json5
/local.properties

10
ohos/build-profile.json5 Normal file
View File

@ -0,0 +1,10 @@
{
"apiType": "stageMode",
"buildOption": {
},
"targets": [
{
"name": "default"
}
]
}

BIN
ohos/har/flutter.har Normal file

Binary file not shown.

2
ohos/hvigorfile.ts Normal file
View File

@ -0,0 +1,2 @@
// Script for compiling build behavior. It is built in the build plug-in and cannot be modified currently.
export { harTasks } from '@ohos/hvigor-ohos-plugin';

17
ohos/index.ets Normal file
View File

@ -0,0 +1,17 @@
/*
* Copyright (c) 2023 Hunan OpenValley Digital Industry Development Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import WechatKitPlugin from './src/main/ets/components/plugin/WechatKitPlugin';
export default WechatKitPlugin;

12
ohos/oh-package.json5 Normal file
View File

@ -0,0 +1,12 @@
{
"name": "wechat_kit",
"version": "1.0.0",
"description": "Please describe the basic information.",
"main": "index.ets",
"author": "",
"license": "Apache-2.0",
"dependencies": {
"@ohos/flutter_ohos": "file:./har/flutter.har",
"@tencent/wechat_open_sdk": "1.0.3"
}
}

View File

@ -0,0 +1,285 @@
import {
AbilityAware,
AbilityPluginBinding,
Any,
FlutterPlugin,
FlutterPluginBinding,
MethodCall,
MethodCallHandler,
MethodChannel,
MethodResult,
NewWantListener,
} from '@ohos/flutter_ohos'
import { AbilityConstant, bundleManager, Want } from '@kit.AbilityKit'
import * as WeChatSDK from "@tencent/wechat_open_sdk"
import { buffer, util } from '@kit.ArkTS'
/** WechatKitPlugin **/
export default class WechatKitPlugin implements FlutterPlugin, AbilityAware, NewWantListener, MethodCallHandler {
private channel: MethodChannel | null = null
private abilityBinding: AbilityPluginBinding | null = null
private wxApi: WeChatSDK.WXApi | null = null
private diffDevOauth: WeChatSDK.IDiffDevOAuth | null = null
getUniqueClassName(): string {
return "WechatKitPlugin"
}
onAttachedToEngine(binding: FlutterPluginBinding): void {
this.channel = new MethodChannel(binding.getBinaryMessenger(), "v7lin.github.io/wechat_kit");
this.channel.setMethodCallHandler(this)
}
onDetachedFromEngine(binding: FlutterPluginBinding): void {
if (this.channel != null) {
this.channel.setMethodCallHandler(null)
}
}
onAttachedToAbility(binding: AbilityPluginBinding): void {
this.abilityBinding = binding
this.abilityBinding.addOnNewWantListener(this)
}
onDetachedFromAbility(): void {
if (this.abilityBinding != null) {
this.abilityBinding.removeOnNewWantListener(this)
}
this.abilityBinding = null
}
onNewWant(want: Want, launchParams: AbilityConstant.LaunchParam): void {
this.wxApi?.handleWant(want, this.wxApiEventHandler)
}
wxApiEventHandler: WeChatSDK.WXApiEventHandler = {
onReq: (req: WeChatSDK.BaseReq): void => {
// TODO: 受限于 SDK, 当前无需实现
},
onResp: (resp: WeChatSDK.BaseResp): void => {
const map: Map<string, Any> = new Map()
map.set("errorCode", resp.errCode)
map.set("errorMsg", resp.errStr)
if (resp instanceof WeChatSDK.SendAuthResp) {
if(resp.errCode == WeChatSDK.ErrCode.ERR_OK) {
map.set("code", resp.code)
map.set("state", resp.state)
map.set("lang", resp.lang)
map.set("country", resp.country)
}
this.channel?.invokeMethod("onAuthResp", map)
} else if (resp instanceof WeChatSDK.SendMessageToWXResp) {
this.channel?.invokeMethod("onShareMsgResp", map)
} else if (resp instanceof WeChatSDK.PayResp) {
if(resp.errCode == WeChatSDK.ErrCode.ERR_OK) {
map.set("returnKey", resp.returnKey)
}
this.channel?.invokeMethod("onPayResp", map)
} else if (resp instanceof WeChatSDK.LaunchMiniProgramResp) {
if(resp.errCode == WeChatSDK.ErrCode.ERR_OK) {
map.set("extMsg", resp.extMsg)
}
this.channel?.invokeMethod("onLaunchMiniProgramResp", map)
}
}
}
onMethodCall(call: MethodCall, result: MethodResult): void {
switch (call.method) {
case "registerApp":
this.registerApp(call, result)
break;
case "handleInitialWXReq":
this.handleInitialWXReq(call, result)
break;
case "isInstalled":
const installed = bundleManager.canOpenLink("weixin://")
result.success(installed)
break;
case "isSupportApi":
// 受限于 SDK, 当前直接返回 false
result.success(false)
break;
case "isSupportStateApi":
// 受限于 SDK, 当前直接返回 false
result.success(false)
break;
case "openWechat":
this.handleOpenWechatCall(call, result)
break;
case "auth":
this.handleAuthCall(call, result)
break;
case "startQrauth":
case "stopQrauth":
this.handleQRAuthCall(call, result)
break;
case "openUrl":
// 受限于 SDK, 当前直接返回 null
result.success(null)
break;
case "openRankList":
// 受限于 SDK, 当前直接返回 null
result.success(null)
break;
case "shareText":
this.handleShareTextCall(call, result)
break;
case "shareImage":
case "shareFile":
case "shareEmoji":
case "shareMusic":
case "shareVideo":
case "shareWebpage":
case "shareMiniProgram":
this.handleShareMediaCall(call, result)
break;
case "subscribeMsg":
// 受限于 SDK, 当前直接返回 null
result.success(null)
break;
case "launchMiniProgram":
this.handleLaunchMiniProgramCall(call, result)
break;
case "openCustomerServiceChat":
// 受限于 SDK, 当前直接返回 null
break;
case "openBusinessView":
// 受限于 SDK, 当前直接返回 null
break;
case "openBusinessWebview":
// 受限于 SDK, 当前直接返回 null
break;
case "pay":
this.handlePayCall(call, result)
break;
default:
result.notImplemented()
break;
}
}
registerApp(call: MethodCall, result: MethodResult): void {
const appId: string = call.argument("appId")
this.wxApi = WeChatSDK.WXAPIFactory.createWXAPI(appId)
this.diffDevOauth = WeChatSDK.DiffDevOAuthFactory.getDiffDevOAuth()
result.success(null)
}
handleInitialWXReq(call: MethodCall, result: MethodResult): void {
if(this.abilityBinding) {
this.wxApi?.handleWant(this.abilityBinding.getAbility().launchWant, this.wxApiEventHandler)
}
result.success(null)
}
async handleOpenWechatCall(call: MethodCall, result: MethodResult): Promise<void> {
const didOpen = await this.wxApi?.openWechat(this.abilityBinding?.getAbility().context)
result.success(didOpen)
}
handleAuthCall(call: MethodCall, result: MethodResult): void {
const request = new WeChatSDK.SendAuthReq()
request.scope = call.argument("scope")
request.state = call.argument("state")
this.wxApi?.sendReq(this.abilityBinding?.getAbility().context, request)
result.success(null)
}
handleQRAuthCall(call: MethodCall, result: MethodResult): void {
if (call.method === "startQrauth") {
const appId: string = call.argument("appId")
const scope: string = call.argument("scope")
const noncestr: string = call.argument("noncestr")
const timestamp: string = call.argument("timestamp")
const signature: string = call.argument("signature")
this.diffDevOauth?.startOAuth(appId, scope, noncestr, timestamp, signature, this.qrAuthListener)
} else if (call.method === "stopQrauth") {
this.diffDevOauth?.stopOAuth()
}
result.success(null)
}
qrAuthListener: WeChatSDK.OAuthCallback = {
onGotQRCode: (base64JpegImageBuffer: string): void => {
const base64Codec = new util.Base64Helper()
const bytes = base64Codec.decodeSync(base64JpegImageBuffer)
const resp: Map<string, Any> = new Map()
resp.set("imageData", bytes)
this.channel?.invokeMethod("onAuthGotQrcode", resp)
},
onQRCodeScanned: (): void => {
this.channel?.invokeMethod("onAuthQrcodeScanned", null)
},
onAuthFinish: (authCode: string): void => {
const resp: Map<string, Any> = new Map()
resp.set("errorCode", 0)
resp.set("authCode", authCode)
this.channel?.invokeMethod("onAuthFinish", resp)
},
onAuthError: (errCode: WeChatSDK.OAuthErrCode, errMsg: string): void => {
const resp: Map<string, Any> = new Map()
resp.set("errorCode", errCode)
resp.set("errorMsg", errMsg)
this.channel?.invokeMethod("onAuthError", resp)
}
}
handleLaunchMiniProgramCall(call: MethodCall, result: MethodResult): void {
const request = new WeChatSDK.LaunchMiniProgramReq()
request.userName = call.argument("username")
request.path = call.argument("path")
request.miniprogramType = call.argument("type")
this.wxApi?.sendReq(this.abilityBinding?.getAbility().context, request)
result.success(null)
}
handlePayCall(call: MethodCall, result: MethodResult): void {
const request = new WeChatSDK.PayReq()
request.appId = call.argument("appId")
request.partnerId = call.argument("partnerId")
request.prepayId = call.argument("prepayId")
request.nonceStr = call.argument("noncestr")
request.timeStamp = call.argument("timestamp")
request.packageValue = call.argument("package")
request.sign = call.argument("sign")
this.wxApi?.sendReq(this.abilityBinding?.getAbility().context, request)
result.success(null)
}
handleShareTextCall(call: MethodCall, result: MethodResult): void {
const textObj = new WeChatSDK.WXTextObject()
textObj.text = call.argument("text")
const mediaMsg = new WeChatSDK.WXMediaMessage()
mediaMsg.mediaObject = textObj
const req = new WeChatSDK.SendMessageToWXReq()
req.scene = call.argument("scene")
req.message = mediaMsg
this.wxApi?.sendReq(this.abilityBinding?.getAbility().context, req)
result.success(null)
}
handleShareMediaCall(call: MethodCall, result: MethodResult): void {
const req = new WeChatSDK.SendMessageToWXReq()
const mediaMsg = new WeChatSDK.WXMediaMessage()
if(call.method == "shareImage") {
const imageObj = new WeChatSDK.WXImageObject()
if(call.hasArgument("imageData")) {
const buff: buffer.Buffer = buffer.from(call.argument("imageData"))
imageObj.imageData = buff.toString("base64", 0, buff.length)
} else if (call.hasArgument("imageUri")) {
imageObj.uri = call.argument("imageUri")
}
mediaMsg.mediaObject = imageObj
} else {
// 受限于 SDK, 其他分享类型暂时不支持
}
this.wxApi?.sendReq(this.abilityBinding?.getAbility().context, req)
result.success(null)
}
}

View File

@ -0,0 +1,10 @@
{
"module": {
"name": "wechat_kit",
"type": "har",
"deviceTypes": [
"default",
"tablet"
]
}
}

View File

@ -1,6 +1,6 @@
name: wechat_kit
description: A powerful Flutter plugin allowing developers to auth/share/pay with natvie Android & iOS Wechat SDKs.
version: 6.0.2
version: 6.1.0
homepage: https://github.com/RxReader/wechat_kit
repository: https://github.com/RxReader/wechat_kit.git
issue_tracker: https://github.com/RxReader/wechat_kit/issues
@ -49,6 +49,8 @@ flutter:
pluginClass: WechatKitPlugin
ios:
pluginClass: WechatKitPlugin
ohos:
pluginClass: WechatKitPlugin
# To add assets to your plugin package, add an assets section, like this:
# assets: