dbweb-common
Version:
用`yarn add dbweb-common`安装,不要忘记修改`angular.json`里的 `architect\build\options\assets`,加上
303 lines • 29 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
import { Component, ElementRef, EventEmitter, HostBinding, HostListener, Output, ViewChild } from '@angular/core';
import { distance, getInsertPointBetweenCircleAndLine, pointInCicle } from './geometric';
/**
* @record
*/
export function MoveEvent() { }
if (false) {
/** @type {?} */
MoveEvent.prototype.startDragPoint;
}
export class PortraitMaskComponent {
/**
* @param {?} elRef
*/
constructor(elRef) {
this.elRef = elRef;
this.hostClass = true;
this.cx = 100;
this.cy = 100;
this.r = 50;
this.handleX = 150;
this.handleY = 100;
this.handleR = 6;
this.cursor = 'grab';
this.startDrag = new EventEmitter();
this.dragging = new EventEmitter();
this.maskChange = new EventEmitter();
this.mouseDown = this.mouseDown.bind(this);
this.mouseUp = this.mouseUp.bind(this);
this.move = this.move.bind(this);
}
/**
* @param {?} x
* @param {?} y
* @return {?}
*/
toScreenPoint(x, y) {
/** @type {?} */
const pt = this.svg.nativeElement.createSVGPoint();
pt.x = x;
pt.y = y;
return pt.matrixTransform(this.svg.nativeElement.getScreenCTM());
}
// 将一个距离转换成屏幕尺寸
/**
* @param {?} val
* @return {?}
*/
toScreenDistance(val) {
/** @type {?} */
const p1 = this.toScreenPoint(0, 0);
/** @type {?} */
const p2 = this.toScreenPoint(val, 0);
return p2.x - p1.x;
}
/**
* @param {?} x
* @param {?} y
* @return {?}
*/
toSVGPoint(x, y) {
/** @type {?} */
const pt = this.svg.nativeElement.createSVGPoint();
pt.x = x;
pt.y = y;
return pt.matrixTransform(this.svg.nativeElement.getScreenCTM().inverse());
}
// 获取handle的受限的圆心坐标 svg
/**
* @param {?} mousePoint
* @return {?}
*/
getLimitHandleCenterPostion(mousePoint) {
// 根据mouse point拿到对应的圆心位置
/** @type {?} */
const point = { x: mousePoint.x - this.handleDistanceOffset.x, y: mousePoint.y - this.handleDistanceOffset.y };
// Create an SVGPoint for future math
/** @type {?} */
let r = distance(point.x, point.y, this.cx, this.cy);
if (r <= this.maxR) {
return point;
}
r = this.maxR;
/** @type {?} */
const arr = getInsertPointBetweenCircleAndLine(this.cx, this.cy, point.x, point.y, this.cx, this.cy, r);
/** @type {?} */
const d1 = distance(point.x, point.y, arr[0].x, arr[0].y);
/** @type {?} */
const d2 = distance(point.x, point.y, arr[1].x, arr[1].y);
if (d1 < d2) {
return arr[0];
}
else {
return arr[1];
}
}
// tslint:disable-next-line: quotemark
/**
* @param {?} event
* @return {?}
*/
mouseDown(event) {
if (event.button !== 0) {
return;
}
/** @type {?} */
const p = this.toSVGPoint(event.x, event.y);
this.handleDistanceOffset = { x: p.x - this.handleX, y: p.y - this.handleY };
this.maskDistanceOffset = { x: p.x - this.cx, y: p.y - this.cy };
this.startDragPoint = { x: event.x, y: event.y };
if (pointInCicle(p.x, p.y, this.handleX, this.handleY, this.handleR)) {
this.dragType = 'handle';
}
else if (pointInCicle(p.x, p.y, this.cx, this.cy, this.r)) {
this.dragType = 'mask';
}
else {
this.dragType = 'outside';
this.startDrag.emit(event);
}
window.addEventListener('mousemove', this.move, false);
window.addEventListener('mouseup', this.mouseUp, false);
event.stopPropagation();
}
/**
* @return {?}
*/
ngOnInit() {
this.calcSize();
}
/**
* @return {?}
*/
emitMaskRect() {
/** @type {?} */
const p1 = this.toScreenPoint(this.cx - this.r, this.cy - this.r);
/** @type {?} */
const p2 = this.toScreenPoint(this.cx + this.r, this.cy + this.r);
this.maskChange.emit({
left: p1.x,
top: p1.y,
right: p2.x,
bottom: p2.y,
width: p2.x - p1.x,
height: p2.y - p1.y
});
}
/**
* @return {?}
*/
ngAfterViewInit() {
this.emitMaskRect();
}
/**
* @param {?} event
* @return {?}
*/
move(event) {
switch (this.dragType) {
case 'handle': {
/** @type {?} */
const p = this.getLimitHandleCenterPostion(this.toSVGPoint(event.x, event.y));
this.handleX = p.x;
this.handleY = p.y;
this.r = distance(this.handleX, this.handleY, this.cx, this.cy);
this.emitMaskRect();
break;
}
case 'mask':
{
/** @type {?} */
const p = this.toSVGPoint(event.x, event.y);
p.x = Math.max(0, Math.min(this.svgWidth, p.x));
p.y = Math.max(0, Math.min(this.svgHeight, p.y));
this.cx = p.x - this.maskDistanceOffset.x;
this.cy = p.y - this.maskDistanceOffset.y;
this.handleX = p.x - this.handleDistanceOffset.x;
this.handleY = p.y - this.handleDistanceOffset.y;
this.emitMaskRect();
}
break;
case 'outside':
this.dragging.emit({ x: event.x, y: event.y });
break;
default:
break;
}
event.stopPropagation();
event.preventDefault();
}
/**
* @param {?} event
* @return {?}
*/
mouseUp(event) {
window.removeEventListener('mousemove', this.move, false);
window.removeEventListener('mouseup', this.mouseUp, false);
}
// 重置到中心点,半径为最大
/**
* @return {?}
*/
reset() {
this.calcSize();
this.emitMaskRect();
}
/**
* @private
* @return {?}
*/
calcSize() {
/** @type {?} */
const rect = this.elRef.nativeElement.getBoundingClientRect();
/** @type {?} */
const p = this.toSVGPoint(rect.left + rect.width / 2, rect.top + rect.height / 2);
/** @type {?} */
const p1 = this.toSVGPoint(rect.left, rect.top);
/** @type {?} */
const p2 = this.toSVGPoint(rect.right, rect.bottom);
this.cx = p.x;
this.cy = p.y;
this.maxR = this.r = (p2.x - p1.x) / 2;
this.svgWidth = p2.x - p1.x;
this.svgHeight = p2.y - p1.y;
this.handleX = this.cx + this.r;
this.handleY = this.cy;
}
}
PortraitMaskComponent.decorators = [
{ type: Component, args: [{
selector: 'common-portrait-mask',
template: "<svg version=\"1.1\" width=\"100%\" height=\"100%\" xmlns=\"http://www.w3.org/2000/svg\"\r\n\txmlns:xlink=\"http://www.w3.org/1999/xlink\" #svg>\r\n\t<style>\r\n\t\t.handle {\r\n\t\t\tcursor: grab;\r\n\t\t}\r\n\t</style>\r\n\t<defs>\r\n\t\t<mask id=\"Mask\">\r\n\t\t\t<rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" fill=\"white\" />\r\n\t\t\t<circle [attr.cx]=\"cx\" [attr.cy]=\"cy\" [attr.r]=\"r\" />\r\n\t\t</mask>\r\n\t</defs>\r\n\r\n\t<rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" opacity=\"0.4\"\r\n\t\tmask=\"url(#Mask)\" />\r\n\t<circle [attr.cx]=\"cx\" [attr.cy]=\"cy\" [attr.r]=\"r\" stroke-width=\"2\" #maskCicle\r\n\t\tclass=\"border-cicle\"/>\r\n\t<circle [attr.cx]=\"handleX\" [attr.cy]=\"handleY\" [attr.r]=\"handleR\" class=\"handle\"/>\r\n</svg>",
styles: [":host{width:100%;height:100%;display:block;position:relative}.handle{cursor:hand}"]
}] }
];
/** @nocollapse */
PortraitMaskComponent.ctorParameters = () => [
{ type: ElementRef }
];
PortraitMaskComponent.propDecorators = {
hostClass: [{ type: HostBinding, args: ['class.portrait-mask',] }],
startDrag: [{ type: Output }],
dragging: [{ type: Output }],
maskChange: [{ type: Output }],
svg: [{ type: ViewChild, args: ['svg', { static: true },] }],
handle: [{ type: ViewChild, args: ['handle', { static: true },] }],
maskCicle: [{ type: ViewChild, args: ['maskCicle', { static: true },] }],
mouseDown: [{ type: HostListener, args: ['mousedown', ['$event'],] }]
};
if (false) {
/** @type {?} */
PortraitMaskComponent.prototype.hostClass;
/** @type {?} */
PortraitMaskComponent.prototype.cx;
/** @type {?} */
PortraitMaskComponent.prototype.cy;
/** @type {?} */
PortraitMaskComponent.prototype.r;
/** @type {?} */
PortraitMaskComponent.prototype.maxR;
/** @type {?} */
PortraitMaskComponent.prototype.svgWidth;
/** @type {?} */
PortraitMaskComponent.prototype.svgHeight;
/** @type {?} */
PortraitMaskComponent.prototype.handleX;
/** @type {?} */
PortraitMaskComponent.prototype.handleY;
/** @type {?} */
PortraitMaskComponent.prototype.handleR;
/** @type {?} */
PortraitMaskComponent.prototype.cursor;
/** @type {?} */
PortraitMaskComponent.prototype.maskDistanceOffset;
/** @type {?} */
PortraitMaskComponent.prototype.handleDistanceOffset;
/** @type {?} */
PortraitMaskComponent.prototype.startDragPoint;
/** @type {?} */
PortraitMaskComponent.prototype.dragType;
/** @type {?} */
PortraitMaskComponent.prototype.startDrag;
/** @type {?} */
PortraitMaskComponent.prototype.dragging;
/** @type {?} */
PortraitMaskComponent.prototype.maskChange;
/** @type {?} */
PortraitMaskComponent.prototype.svg;
/** @type {?} */
PortraitMaskComponent.prototype.handle;
/** @type {?} */
PortraitMaskComponent.prototype.maskCicle;
/**
* @type {?}
* @private
*/
PortraitMaskComponent.prototype.elRef;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"portrait-mask.component.js","sourceRoot":"ng://dbweb-common/","sources":["lib/userinfo/portrait-edit/portrait-mask/portrait-mask.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Point } from '@angular/cdk/drag-drop/typings/drag-ref';\nimport {\n\tAfterViewInit,\n\tComponent,\n\tElementRef,\n\tEventEmitter,\n\tHostBinding,\n\tHostListener,\n\tOnInit,\n\tOutput,\n\tViewChild\n} from '@angular/core';\nimport { distance, getInsertPointBetweenCircleAndLine, pointInCicle } from './geometric';\n\ntype DargType = 'handle' | 'mask' | 'outside';\nexport interface MoveEvent {\n\tstartDragPoint: Point;\n}\n@Component({\n\tselector: 'common-portrait-mask',\n\ttemplateUrl: './portrait-mask.component.html',\n\tstyleUrls: ['./portrait-mask.component.scss']\n})\nexport class PortraitMaskComponent implements OnInit, AfterViewInit {\n\t@HostBinding('class.portrait-mask') hostClass = true;\n\tcx = 100;\n\tcy = 100;\n\tr = 50;\n\tmaxR: number;\n\tsvgWidth: number; // svg 坐标的总宽度\n\tsvgHeight: number;\n\thandleX = 150;\n\thandleY = 100;\n\thandleR = 6;\n\tcursor = 'grab';\n\tmaskDistanceOffset: Point; // 鼠标点击位置与圆心的偏移\n\thandleDistanceOffset: Point;\n\tstartDragPoint: Point; // 开始拖动的点，屏幕坐标\n\tdragType: DargType;\n\t@Output() startDrag = new EventEmitter<MouseEvent>();\n\t@Output() dragging = new EventEmitter<Point>();\n\t@Output() maskChange = new EventEmitter<ClientRect>();\n\t@ViewChild('svg', { static: true }) svg: ElementRef;\n\t@ViewChild('handle', { static: true }) handle: ElementRef;\n\t@ViewChild('maskCicle', { static: true }) maskCicle: ElementRef;\n\tconstructor(private elRef: ElementRef<HTMLElement>) {\n\t\tthis.mouseDown = this.mouseDown.bind(this);\n\t\tthis.mouseUp = this.mouseUp.bind(this);\n\t\tthis.move = this.move.bind(this);\n\t}\n\ttoScreenPoint(x, y): Point {\n\t\tconst pt = this.svg.nativeElement.createSVGPoint();\n\t\tpt.x = x;\n\t\tpt.y = y;\n\t\treturn pt.matrixTransform(this.svg.nativeElement.getScreenCTM());\n\t}\n\t// 将一个距离转换成屏幕尺寸\n\ttoScreenDistance(val: number): number {\n\t\tconst p1 = this.toScreenPoint(0, 0);\n\t\tconst p2 = this.toScreenPoint(val, 0);\n\t\treturn p2.x - p1.x;\n\t}\n\ttoSVGPoint(x, y): Point {\n\t\tconst pt = this.svg.nativeElement.createSVGPoint();\n\t\tpt.x = x;\n\t\tpt.y = y;\n\t\treturn pt.matrixTransform(this.svg.nativeElement.getScreenCTM().inverse());\n\t}\n\t// 获取handle的受限的圆心坐标 svg\n\tgetLimitHandleCenterPostion(mousePoint: Point): Point {\n\t\t// 根据mouse point拿到对应的圆心位置\n\t\tconst point = { x: mousePoint.x - this.handleDistanceOffset.x, y: mousePoint.y - this.handleDistanceOffset.y };\n\t\t// Create an SVGPoint for future math\n\t\tlet r = distance(point.x, point.y, this.cx, this.cy);\n\t\tif (r <= this.maxR) {\n\t\t\treturn point;\n\t\t}\n\t\tr = this.maxR;\n\t\tconst arr = getInsertPointBetweenCircleAndLine(this.cx, this.cy, point.x, point.y, this.cx, this.cy, r);\n\n\t\tconst d1 = distance(point.x, point.y, arr[0].x, arr[0].y);\n\t\tconst d2 = distance(point.x, point.y, arr[1].x, arr[1].y);\n\t\tif (d1 < d2) {\n\t\t\treturn arr[0];\n\t\t} else {\n\t\t\treturn arr[1];\n\t\t}\n\t}\n\n\t// tslint:disable-next-line: quotemark\n\t@HostListener('mousedown', ['$event'])\n\tmouseDown(event: MouseEvent) {\n\t\tif (event.button !== 0) {\n\t\t\treturn;\n\t\t}\n\t\tconst p = this.toSVGPoint(event.x, event.y);\n\t\tthis.handleDistanceOffset = { x: p.x - this.handleX, y: p.y - this.handleY };\n\t\tthis.maskDistanceOffset = { x: p.x - this.cx, y: p.y - this.cy };\n\t\tthis.startDragPoint = { x: event.x, y: event.y };\n\n\t\tif (pointInCicle(p.x, p.y, this.handleX, this.handleY, this.handleR)) {\n\t\t\tthis.dragType = 'handle';\n\t\t} else if (pointInCicle(p.x, p.y, this.cx, this.cy, this.r)) {\n\t\t\tthis.dragType = 'mask';\n\t\t} else {\n\t\t\tthis.dragType = 'outside';\n\t\t\tthis.startDrag.emit(event);\n\t\t}\n\t\twindow.addEventListener('mousemove', this.move, false);\n\t\twindow.addEventListener('mouseup', this.mouseUp, false);\n\t\tevent.stopPropagation();\n\t}\n\tngOnInit() {\n\t\tthis.calcSize();\n\t}\n\temitMaskRect() {\n\t\tconst p1 = this.toScreenPoint(this.cx - this.r, this.cy - this.r);\n\t\tconst p2 = this.toScreenPoint(this.cx + this.r, this.cy + this.r);\n\t\tthis.maskChange.emit({\n\t\t\tleft: p1.x,\n\t\t\ttop: p1.y,\n\t\t\tright: p2.x,\n\t\t\tbottom: p2.y,\n\t\t\twidth: p2.x - p1.x,\n\t\t\theight: p2.y - p1.y\n\t\t});\n\t}\n\tngAfterViewInit() {\n\t\tthis.emitMaskRect();\n\t}\n\tmove(event: MouseEvent) {\n\t\tswitch (this.dragType) {\n\t\t\tcase 'handle': {\n\t\t\t\tconst p = this.getLimitHandleCenterPostion(this.toSVGPoint(event.x, event.y));\n\t\t\t\tthis.handleX = p.x;\n\t\t\t\tthis.handleY = p.y;\n\t\t\t\tthis.r = distance(this.handleX, this.handleY, this.cx, this.cy);\n\t\t\t\tthis.emitMaskRect();\n\t\t\t\tbreak;\n\t\t\t}\n\t\t\tcase 'mask':\n\t\t\t\t{\n\t\t\t\t\tconst p = this.toSVGPoint(event.x, event.y);\n\t\t\t\t\tp.x = Math.max(0, Math.min(this.svgWidth, p.x));\n\t\t\t\t\tp.y = Math.max(0, Math.min(this.svgHeight, p.y));\n\t\t\t\t\tthis.cx = p.x - this.maskDistanceOffset.x;\n\t\t\t\t\tthis.cy = p.y - this.maskDistanceOffset.y;\n\t\t\t\t\tthis.handleX = p.x - this.handleDistanceOffset.x;\n\t\t\t\t\tthis.handleY = p.y - this.handleDistanceOffset.y;\n\t\t\t\t\tthis.emitMaskRect();\n\t\t\t\t}\n\t\t\t\tbreak;\n\t\t\tcase 'outside':\n\t\t\t\tthis.dragging.emit({ x: event.x, y: event.y });\n\n\t\t\t\tbreak;\n\t\t\tdefault:\n\t\t\t\tbreak;\n\t\t}\n\t\tevent.stopPropagation();\n\t\tevent.preventDefault();\n\t}\n\tmouseUp(event: MouseEvent) {\n\t\twindow.removeEventListener('mousemove', this.move, false);\n\t\twindow.removeEventListener('mouseup', this.mouseUp, false);\n\t}\n\t// 重置到中心点，半径为最大\n\treset() {\n\t\tthis.calcSize();\n\n\t\tthis.emitMaskRect();\n\t}\n\n\tprivate calcSize() {\n\t\tconst rect = this.elRef.nativeElement.getBoundingClientRect();\n\t\tconst p = this.toSVGPoint(rect.left + rect.width / 2, rect.top + rect.height / 2);\n\t\tconst p1 = this.toSVGPoint(rect.left, rect.top);\n\t\tconst p2 = this.toSVGPoint(rect.right, rect.bottom);\n\t\tthis.cx = p.x;\n\t\tthis.cy = p.y;\n\t\tthis.maxR = this.r = (p2.x - p1.x) / 2;\n\t\tthis.svgWidth = p2.x - p1.x;\n\t\tthis.svgHeight = p2.y - p1.y;\n\t\tthis.handleX = this.cx + this.r;\n\t\tthis.handleY = this.cy;\n\t}\n}\n"]}