feat:add Popover demo

This commit is contained in:
pengfei.zhou 2019-11-28 21:04:27 +08:00
parent 38a06c9658
commit a79e288679
9 changed files with 132 additions and 22 deletions

View File

@ -14,4 +14,5 @@ export default [
'src/NavbarDemo', 'src/NavbarDemo',
'src/RefreshableDemo', 'src/RefreshableDemo',
'src/FlowLayoutDemo', 'src/FlowLayoutDemo',
'src/PopoverDemo'
] ]

46
demo/src/PopoverDemo.ts Normal file
View File

@ -0,0 +1,46 @@
import { Group, Panel, popover, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, Text, scroller, layoutConfig, image, IView, IVLayout, ScaleType, modal, IText, network } from "doric";
import { title, label, colors } from "./utils";
@Entry
class PopoverDemo extends Panel {
build(rootView: Group): void {
scroller(vlayout([
title("Popover Demo"),
label('Popover').apply({
width: 200,
height: 50,
bgColor: colors[0],
textSize: 30,
textColor: Color.WHITE,
layoutConfig: layoutConfig().exactly(),
onClick: () => {
popover(context).show(text({
width: 200,
height: 50,
bgColor: colors[0],
textColor: Color.WHITE,
layoutConfig: layoutConfig().exactly().a(Gravity.Center),
text: "This is PopOver Window",
onClick: () => {
modal(context).toast('Dismissed after 3 seconds')
setTimeout(() => {
popover(context).dismiss()
}, 3000)
},
}).also(v => {
let idx = 0
v.onClick = () => {
v.bgColor = colors[(++idx) % colors.length]
}
}))
}
} as IText),
]).apply({
layoutConfig: layoutConfig().atmost().h(LayoutSpec.WRAP_CONTENT),
gravity: gravity().center(),
space: 10,
} as IVLayout)).apply({
layoutConfig: layoutConfig().atmost(),
}).in(rootView)
}
}

View File

@ -7,7 +7,7 @@
#import "Doric.h" #import "Doric.h"
@interface DoricPopoverPlugin () @interface DoricPopoverPlugin ()
@property(nonatomic, strong) DoricRootNode *popoverNode; @property(nonatomic, strong) DoricViewNode *popoverNode;
@property(nonatomic, strong) UIView *fullScreenView; @property(nonatomic, strong) UIView *fullScreenView;
@end @end
@ -20,19 +20,26 @@ - (void)show:(NSDictionary *)params withPromise:(DoricPromise *)promise {
it.width = superView.width; it.width = superView.width;
it.height = superView.height; it.height = superView.height;
it.top = it.left = 0; it.top = it.left = 0;
[superView addSubview:self.fullScreenView]; [superView addSubview:it];
UITapGestureRecognizer *gestureRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(dismissPopover)]; UITapGestureRecognizer *gestureRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(dismissPopover)];
[it addGestureRecognizer:gestureRecognizer]; [it addGestureRecognizer:gestureRecognizer];
}]; }];
} }
[superView bringSubviewToFront:self.fullScreenView]; [superView bringSubviewToFront:self.fullScreenView];
self.fullScreenView.hidden = NO; if (self.popoverNode) {
if (!self.popoverNode) { [self dismissPopover];
self.popoverNode = [[DoricRootNode alloc] initWithContext:self.doricContext];
DoricStackView *view = [[DoricStackView alloc] initWithFrame:self.fullScreenView.frame];
[self.popoverNode setupRootView:view];
} }
[self.popoverNode render:params[@"props"]]; self.fullScreenView.hidden = NO;
NSString *viewId = params[@"id"];
NSString *type = params[@"type"];
self.popoverNode = [[DoricViewNode create:self.doricContext withType:type] also:^(DoricViewNode *it) {
it.viewId = viewId;
[it initWithSuperNode:nil];
it.view.layoutConfig = [DoricLayoutConfig new];
[it blend:params[@"props"]];
[self.fullScreenView addSubview:it.view];
[self.doricContext.headNodes addObject:it];
}];
[promise resolve:nil]; [promise resolve:nil];
}); });
} }
@ -45,10 +52,12 @@ - (void)dismiss:(NSDictionary *)params withPromise:(DoricPromise *)promise {
} }
- (void)dismissPopover { - (void)dismissPopover {
[self.doricContext.headNodes removeObject:self.popoverNode];
self.popoverNode.view.hidden = YES; self.popoverNode.view.hidden = YES;
self.fullScreenView.hidden = YES; self.fullScreenView.hidden = YES;
[self.popoverNode.view.subviews forEach:^(__kindof UIView *obj) { [self.popoverNode.view.subviews forEach:^(__kindof UIView *obj) {
[obj removeFromSuperview]; [obj removeFromSuperview];
}]; }];
self.popoverNode = nil;
} }
@end @end

View File

@ -36,10 +36,15 @@ - (void)render:(NSDictionary *)argument {
__weak typeof(self) _self = self; __weak typeof(self) _self = self;
dispatch_async(dispatch_get_main_queue(), ^{ dispatch_async(dispatch_get_main_queue(), ^{
__strong typeof(_self) self = _self; __strong typeof(_self) self = _self;
[self.doricContext.rootNode also:^(DoricRootNode *it) {
it.viewId = argument[@"id"]; NSString *viewId = argument[@"id"];
[it render:argument[@"props"]]; if ([self.doricContext.rootNode.viewId isEqualToString:viewId]) {
}]; [self.doricContext.rootNode render:argument[@"props"]];
} else {
DoricViewNode *viewNode = [self headViewNodeByViewId:viewId];
[viewNode blend:argument[@"props"]];
}
}); });
} }
@ -49,6 +54,7 @@ - (DoricViewNode *)headViewNodeByViewId:(NSString *)viewId {
return node; return node;
} }
} }
self.doricContext.rootNode.viewId = viewId;
return self.doricContext.rootNode; return self.doricContext.rootNode;
} }

View File

@ -24,7 +24,6 @@
#import "DoricLayouts.h" #import "DoricLayouts.h"
#import "UIView+Doric.h" #import "UIView+Doric.h"
NS_ASSUME_NONNULL_BEGIN
@class DoricSuperNode; @class DoricSuperNode;
@interface DoricViewNode <V:UIView *> : DoricContextHolder @interface DoricViewNode <V:UIView *> : DoricContextHolder
@ -55,5 +54,3 @@ NS_ASSUME_NONNULL_BEGIN
- (void)requestLayout; - (void)requestLayout;
@end @end
NS_ASSUME_NONNULL_END

View File

@ -126,6 +126,8 @@ - (void)blendView:(UIView *)view forPropName:(NSString *)name propValue:(id)prop
} else if ([name isEqualToString:@"layoutConfig"]) { } else if ([name isEqualToString:@"layoutConfig"]) {
if (self.superNode && [prop isKindOfClass:[NSDictionary class]]) { if (self.superNode && [prop isKindOfClass:[NSDictionary class]]) {
[self.superNode blendSubNode:self layoutConfig:prop]; [self.superNode blendSubNode:self layoutConfig:prop];
} else {
[self blendLayoutConfig:prop];
} }
} else if ([name isEqualToString:@"onClick"]) { } else if ([name isEqualToString:@"onClick"]) {
self.callbackIds[@"onClick"] = prop; self.callbackIds[@"onClick"] = prop;
@ -246,4 +248,36 @@ - (NSNumber *)getRotation {
return @(degree); return @(degree);
} }
- (void)blendLayoutConfig:(NSDictionary *)params {
[params[@"widthSpec"] also:^(NSNumber *it) {
if (it) {
self.layoutConfig.widthSpec = (DoricLayoutSpec) [it integerValue];
}
}];
[params[@"heightSpec"] also:^(NSNumber *it) {
if (it) {
self.layoutConfig.heightSpec = (DoricLayoutSpec) [it integerValue];
}
}];
NSDictionary *margin = params[@"margin"];
if (margin) {
self.layoutConfig.margin = DoricMarginMake(
[(NSNumber *) margin[@"left"] floatValue],
[(NSNumber *) margin[@"top"] floatValue],
[(NSNumber *) margin[@"right"] floatValue],
[(NSNumber *) margin[@"bottom"] floatValue]);
}
NSNumber *alignment = params[@"alignment"];
if (alignment) {
self.layoutConfig.alignment = (DoricGravity) [alignment integerValue];
}
NSNumber *weight = params[@"weight"];
if (weight) {
self.layoutConfig.weight = (DoricGravity) [weight integerValue];
}
}
@end @end

View File

@ -18,3 +18,4 @@ export * from './navbar'
export * from './navigator' export * from './navigator'
export * from './network' export * from './network'
export * from './storage' export * from './storage'
export * from './popover'

View File

@ -15,13 +15,25 @@
*/ */
import { BridgeContext } from "../runtime/global" import { BridgeContext } from "../runtime/global"
import { View } from "../ui/view" import { View } from "../ui/view"
import { Panel } from "../ui/panel"
export function popover(context: BridgeContext) { export function popover(context: BridgeContext) {
const entity = context.entity
let panel: Panel | undefined = undefined
if (entity instanceof Panel) {
panel = entity
}
return { return {
show: (view: View) => { show: (view: View) => {
return context.popover.show(view) if (panel) {
panel.addHeadView(view)
}
return context.popover.show(view.toModel())
}, },
dismiss: () => { dismiss: (view: View | undefined = undefined) => {
if (panel && view) {
panel.removeHeadView(view)
}
return context.popover.dismiss() return context.popover.dismiss()
}, },
} }

View File

@ -137,16 +137,20 @@ export abstract class Panel {
} }
private hookBeforeNativeCall() { private hookBeforeNativeCall() {
this.__root__.clean() for (let v of this.headviews.values()) {
v.clean()
}
} }
private hookAfterNativeCall() { private hookAfterNativeCall() {
//Here insert a native call to ensure the promise is resolved done. //Here insert a native call to ensure the promise is resolved done.
nativeEmpty() nativeEmpty()
if (this.__root__.isDirty()) { for (let v of this.headviews.values()) {
const model = this.__root__.toModel() if (v.isDirty()) {
const model = v.toModel()
this.nativeRender(model) this.nativeRender(model)
} }
} }
}
} }