`

rc-align

 
阅读更多

rc-align模块基于dom-align模块实现,用于以参照节点调整事件源节点的位置。

 

Align.js

'use strict';

Object.defineProperty(exports, "__esModule", {
  value: true
});

var _react = require('react');
var _react2 = _interopRequireDefault(_react);

var _reactDom = require('react-dom');
var _reactDom2 = _interopRequireDefault(_reactDom);

// domAlign(sourceNode,refNode,options)根据参照节点refNode和配置options调整事件源节点的位置
var _domAlign = require('dom-align');
var _domAlign2 = _interopRequireDefault(_domAlign);

var _addEventListener = require('rc-util/lib/Dom/addEventListener');
var _addEventListener2 = _interopRequireDefault(_addEventListener);

var _isWindow = require('./isWindow');
var _isWindow2 = _interopRequireDefault(_isWindow);

function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }

// 设置定时器,执行fn;调用返回值启动执行定时器;调用返回值的clear方法,清除定时器
function buffer(fn, ms) {
  var timer = void 0;

  function clear() {
    if (timer) {
      clearTimeout(timer);
      timer = null;
    }
  }

  function bufferFn() {
    clear();
    timer = setTimeout(fn, ms);
  }

  bufferFn.clear = clear;

  return bufferFn;
}

var Align = _react2["default"].createClass({
  displayName: 'Align',

  propTypes: {
    childrenProps: _react.PropTypes.object,// 为props.children提供props的映射,属性值为this.props的属性名
    align: _react.PropTypes.object.isRequired,// 位置调整配置项
    target: _react.PropTypes.func,// 获取参照节点,同时用于判断组件重绘时是否调整事件源节点的位置
    onAlign: _react.PropTypes.func,// 事件源节点位置调整后执行的回调函数
    monitorBufferTime: _react.PropTypes.number,// 监听resize事件的定时事件
    monitorWindowResize: _react.PropTypes.bool,// 浏览器缩放时是否调整事件源节点的位置
    disabled: _react.PropTypes.bool,// 是否启用位置调整
    children: _react.PropTypes.any// 实际显示用的待渲染节点
  },

  getDefaultProps: function getDefaultProps() {
    return {
      target: function target() {
        return window;
      },
      onAlign: function onAlign(sourceNode,offsetInfo) {},
      monitorBufferTime: 50,
      monitorWindowResize: false,
      disabled: false
    };
  },
  componentDidMount: function componentDidMount() {
    var props = this.props;
    // 调整事件源节点的位置
    this.forceAlign();

    // 监听窗口缩放resize事件
    if (!props.disabled && props.monitorWindowResize) {
      this.startMonitorWindowResize();
    }
  },
  componentDidUpdate: function componentDidUpdate(prevProps) {
    var reAlign = false;
    var props = this.props;

    // props.align或props.target返回值变更,重新调整事件源节点的位置
    if (!props.disabled) {
      if (prevProps.disabled || prevProps.align !== props.align) {
        reAlign = true;
      } else {
        var lastTarget = prevProps.target();
        var currentTarget = props.target();
        if ((0, _isWindow2["default"])(lastTarget) && (0, _isWindow2["default"])(currentTarget)) {
          reAlign = false;
        } else if (lastTarget !== currentTarget) {
          reAlign = true;
        }
      }
    }

    if (reAlign) {
      this.forceAlign();
    }

    if (props.monitorWindowResize && !props.disabled) {
      this.startMonitorWindowResize();
    } else {
      this.stopMonitorWindowResize();
    }
  },

  // 组件卸载时,解绑resize事件及清除定时器
  componentWillUnmount: function componentWillUnmount() {
    this.stopMonitorWindowResize();
  },

  // 当浏览器窗口缩放、即'resize'事件发生时,调整事件源节点的位置
  startMonitorWindowResize: function startMonitorWindowResize() {
    if (!this.resizeHandler) {
      this.bufferMonitor = buffer(this.forceAlign, this.props.monitorBufferTime);
      this.resizeHandler = (0, _addEventListener2["default"])(window, 'resize', this.bufferMonitor);
    }
  },

  // 解绑'resize'事件的监听函数、及清除调整位置的定时器
  stopMonitorWindowResize: function stopMonitorWindowResize() {
    if (this.resizeHandler) {
      this.bufferMonitor.clear();
      this.resizeHandler.remove();
      this.resizeHandler = null;
    }
  },

  // 调用domAlign调整事件源节点的位置,并执行this.props.onAlign方法
  forceAlign: function forceAlign() {
    var props = this.props;
    if (!props.disabled) {
      // 获取事件源节点,即this.props.children渲染后的节点元素
      var source = _reactDom2["default"].findDOMNode(this);

      // 调用domAlign调整事件源节点的位置后,再调用this.props.onAlign(source,offsetInfo)
      // 参数offsetInfo是事件源节点实际位置调整数据
      props.onAlign(source, (0, _domAlign2["default"])(source, props.target(), props.align));
    }
  },

  // rc-align只为props.children提供位置调整的实现,渲染依旧是props.children的固有机制,除去props变更
  render: function render() {
    var _props = this.props,
        childrenProps = _props.childrenProps,
        children = _props.children;

    // this.props.children只允许为单个reactElement
    var child = _react2["default"].Children.only(children);

    // this.props.children映射this.props中将要置入子元素child的属性
    if (childrenProps) {
      var newProps = {};
      for (var prop in childrenProps) {
        if (childrenProps.hasOwnProperty(prop)) {
          newProps[prop] = this.props[childrenProps[prop]];
        }
      }

      // 使用react.cloneElement方法为child提供新的props
      return _react2["default"].cloneElement(child, newProps);
    }
    return child;
  }
});

exports["default"] = Align;
module.exports = exports['default'];

 

dom-align:index.js

'use strict';

Object.defineProperty(exports, "__esModule", {
  value: true
});

var _utils = require('./utils');
var _utils2 = _interopRequireDefault(_utils);

var _getOffsetParent = require('./getOffsetParent');
var _getOffsetParent2 = _interopRequireDefault(_getOffsetParent);

var _getVisibleRectForElement = require('./getVisibleRectForElement');
var _getVisibleRectForElement2 = _interopRequireDefault(_getVisibleRectForElement);

var _adjustForViewport = require('./adjustForViewport');
var _adjustForViewport2 = _interopRequireDefault(_adjustForViewport);

var _getRegion = require('./getRegion');
var _getRegion2 = _interopRequireDefault(_getRegion);

var _getElFuturePos = require('./getElFuturePos');
var _getElFuturePos2 = _interopRequireDefault(_getElFuturePos);

function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }

// http://yiminghe.iteye.com/blog/1124720

/**
 * align dom node flexibly
 * @author yiminghe@gmail.com
 */

function isFailX(elFuturePos, elRegion, visibleRect) {
  return elFuturePos.left < visibleRect.left || elFuturePos.left + elRegion.width > visibleRect.right;
}

function isFailY(elFuturePos, elRegion, visibleRect) {
  return elFuturePos.top < visibleRect.top || elFuturePos.top + elRegion.height > visibleRect.bottom;
}

function isCompleteFailX(elFuturePos, elRegion, visibleRect) {
  return elFuturePos.left > visibleRect.right || elFuturePos.left + elRegion.width < visibleRect.left;
}

function isCompleteFailY(elFuturePos, elRegion, visibleRect) {
  return elFuturePos.top > visibleRect.bottom || elFuturePos.top + elRegion.height < visibleRect.top;
}

function flip(points, reg, map) {
  var ret = [];
  _utils2["default"].each(points, function (p) {
    ret.push(p.replace(reg, function (m) {
      return map[m];
    }));
  });
  return ret;
}

function flipOffset(offset, index) {
  offset[index] = -offset[index];
  return offset;
}

function convertOffset(str, offsetLen) {
  var n = void 0;
  if (/%$/.test(str)) {
    n = parseInt(str.substring(0, str.length - 1), 10) / 100 * offsetLen;
  } else {
    n = parseInt(str, 10);
  }
  return n || 0;
}

function normalizeOffset(offset, el) {
  offset[0] = convertOffset(offset[0], el.width);
  offset[1] = convertOffset(offset[1], el.height);
}

// 参数el待调整位置的事件源节点
// 参数refNode调整位置时的参照节点
// 参数align配置项{points,offset,targetOffset,overflow,target,source}
// align.points设置参照节点和目标节点对齐时的参照点,如["tl","cr"],取el的左上位置对齐relNode的右中位置
// align.offset设置为百分比时,根据事件源节点的宽高执行偏移;数值时,即正向以该数值执行偏移[10,"20%"]
// align.offset设置为百分比时,根据参照节点的宽高反向执行偏移;数值时,即反向执行偏移
// align.adjustX、adjustY可放置区域无法容纳事件源节点时,反向进行调整;若反向调整失败,削减事件源节点的宽高进行调整
// align.useCssRight、useCssBottom、align.useCssTransform设置调整事件源节点的模式
function domAlign(el, refNode, align) {
  var points = align.points;
  var offset = align.offset || [0, 0];
  var targetOffset = align.targetOffset || [0, 0];
  var overflow = align.overflow;
  var target = align.target || refNode;
  var source = align.source || el;
  offset = [].concat(offset);
  targetOffset = [].concat(targetOffset);
  overflow = overflow || {};
  var newOverflowCfg = {};

  var fail = 0;

  // 获取事件源节点position="static"的祖先节点,作为可放置范围,计算其top、left、bottom、right
  var visibleRect = (0, _getVisibleRectForElement2["default"])(source);

  // 事件源节点所占的区域, 含left、top、width、height属性
  var elRegion = (0, _getRegion2["default"])(source);

  // 目标节点所占的区域, 含left、top、width、height属性
  var refNodeRegion = (0, _getRegion2["default"])(target);

  // 将偏移量offset转换成数值
  normalizeOffset(offset, elRegion);
  normalizeOffset(targetOffset, refNodeRegion);

  // 根据对齐点points及偏移量offset、targetOffset计算事件源节点最终的x、y坐标{left,top}
  var elFuturePos = (0, _getElFuturePos2["default"])(elRegion, refNodeRegion, points, offset, targetOffset);
  
  // 获取事件源节点对齐后的left、top属性,返回{left,top,width,height}
  var newElRegion = _utils2["default"].merge(elRegion, elFuturePos);

  // 可放置区域不能完全容纳事件源节点,作反向调整,如左对齐的变成右对齐、相应偏移量也反向
  if (visibleRect && (overflow.adjustX || overflow.adjustY)) {
    if (overflow.adjustX) {
      if (isFailX(elFuturePos, elRegion, visibleRect)) {
        var newPoints = flip(points, /[lr]/ig, {
          l: 'r',
          r: 'l'
        });
        var newOffset = flipOffset(offset, 0);
        var newTargetOffset = flipOffset(targetOffset, 0);
        var newElFuturePos = (0, _getElFuturePos2["default"])(elRegion, refNodeRegion, newPoints, newOffset, newTargetOffset);
        if (!isCompleteFailX(newElFuturePos, elRegion, visibleRect)) {
          fail = 1;
          points = newPoints;
          offset = newOffset;
          targetOffset = newTargetOffset;
        }
      }
    }

    if (overflow.adjustY) {
      if (isFailY(elFuturePos, elRegion, visibleRect)) {
        var _newPoints = flip(points, /[tb]/ig, {
          t: 'b',
          b: 't'
        });
        var _newOffset = flipOffset(offset, 1);
        var _newTargetOffset = flipOffset(targetOffset, 1);
        var _newElFuturePos = (0, _getElFuturePos2["default"])(elRegion, refNodeRegion, _newPoints, _newOffset, _newTargetOffset);
        if (!isCompleteFailY(_newElFuturePos, elRegion, visibleRect)) {
          fail = 1;
          points = _newPoints;
          offset = _newOffset;
          targetOffset = _newTargetOffset;
        }
      }
    }

    if (fail) {
      elFuturePos = (0, _getElFuturePos2["default"])(elRegion, refNodeRegion, points, offset, targetOffset);
      _utils2["default"].mix(newElRegion, elFuturePos);
    }

    newOverflowCfg.adjustX = overflow.adjustX && isFailX(elFuturePos, elRegion, visibleRect);
    newOverflowCfg.adjustY = overflow.adjustY && isFailY(elFuturePos, elRegion, visibleRect);

    // 配置项要求调整,同时反向调整失败,削减事件源节点的宽高进行调整;newElRegion计算事件源节点允许的宽高
    if (newOverflowCfg.adjustX || newOverflowCfg.adjustY) {
      newElRegion = (0, _adjustForViewport2["default"])(elFuturePos, elRegion, visibleRect, newOverflowCfg);
    }
  }

  // 削减宽高
  if (newElRegion.width !== elRegion.width) {
    _utils2["default"].css(source, 'width', _utils2["default"].width(source) + newElRegion.width - elRegion.width);
  }
  if (newElRegion.height !== elRegion.height) {
    _utils2["default"].css(source, 'height', _utils2["default"].height(source) + newElRegion.height - elRegion.height);
  }

  // 调整事件源节点的位置,尾参option约定调整的方式
  _utils2["default"].offset(source, {
    left: newElRegion.left,
    top: newElRegion.top
  }, {
    useCssRight: align.useCssRight,
    useCssBottom: align.useCssBottom,
    useCssTransform: align.useCssTransform
  });

  // 返回事件源节点实际的调整数据
  return {
    points: points,
    offset: offset,
    targetOffset: targetOffset,
    overflow: newOverflowCfg
  };
}

domAlign.__getOffsetParent = _getOffsetParent2["default"];

domAlign.__getVisibleRectForElement = _getVisibleRectForElement2["default"];

exports["default"] = domAlign;

module.exports = exports['default'];

 

 

0
0
分享到:
评论

相关推荐

    Python库 | subaligner-0.0.6rc1-py3.7.egg

    aligned_subtitles = subaligner.align(video_path, subtitle_path) # 输出对齐结果 print(aligned_subtitles) ``` `subaligner` 库可能还支持其他高级功能,如批量处理、自定义模型训练以及结果可视化等。为了...

    PyPI 官网下载 | fpdf2-2.1.0rc1-py2.py3-none-any.whl

    ", ln=1, align=C) ``` 这段代码创建了一个新的PDF,添加了一页,设置字体和大小,然后在页面上写入了"Hello World!"。 此外,fpdf2还支持添加图片、创建多列布局、插入超链接、定义页面模板等功能,使得开发者...

    BIMRevit软件常用快捷键.pdf

    * RC - 或 Enter 恢复上一次选择集 Navigation命令: * Ctrl + 左方向键 - 建模与绘图工具常用快捷键 Edit Modify命令: * 编辑修改工具常用快捷键 View Control命令: * SR - 区域放大 * ZR - 象限点 * SQ...

    第四章 网页边框和网页文字阴影.docx

    * InternetExplorer(IE7/IE8RC1) * Opera(9.6) * Safari(3.2.1,Windows) 六、文字阴影效果 文字阴影效果可以使用以下代码实现: ``` .textShadowSingle{ font-size:3.2em; color:#F5F5F5; text-shadow:3px 3px ...

    第六章 创建网站多列.docx

    需要注意的是,虽然CSS3的多列布局在Firefox、Safari、Chrome等现代浏览器中得到了较好的支持,但在Internet Explorer 7和8RC1这样的旧版本浏览器中,需要使用 `-moz-` 和 `-webkit-` 前缀来实现兼容。然而,对于不...

    Rust container cheat sheet.pdf

    - tagstrongweak:可能与Rc或Arc的弱引用有关,弱引用不增加引用计数,但可以用来检查引用是否还有效。 - C: Copy + Clone:指出了Rust中可以被复制(Copy)和克隆(Clone)的类型。 - Option&lt;T&gt;:表示一个值可能...

    Excel导入,将数据导入到数据库.pdf

    在这个例子中,需要的jar包包括:poi-3.0-rc4-20070503.jar、poi-contrib-3.0-rc4-20070503.jar以及poi-scratchpad-3.0-rc4-20070503.jar。这些库提供了读取Excel文件(.xls格式,不支持.xlsx)的能力。 接着,我们...

    jfreechart-1.0.4.jar jfreechart-1.0.4-swt.jar

    &lt;div align="cente"&gt;&lt;P&gt;&lt;font size="4" face="楷体_GB2312"&gt;&lt;strong&gt;&lt;font color="#0099ff"&gt;招生统计信息&lt;/font&gt;&lt;/strong&gt;&lt;/font&gt;&lt; /p&gt; //设置生成饼状图 int i=0; String[] area={"南京市","镇江市","常州市",...

    MFC工具栏ToolBar按钮添加下拉菜单

    MFC 工具栏 ToolBar 按钮添加... pPopup-&gt;TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, rc.left, rc.bottom, this); } ``` 通过以上步骤,可以在 MFC 工具栏 ToolBar 按钮添加下拉菜单,以提供更多的功能选项。

    uboott移植实验手册及技术文档

    .align 2 DW_STACK_START: .word STACK_BASE+STACK_SIZE-4 三、创建 board/fs2410/nand_read.c 文件,加入读 NAND Flash 的操作。 #include &lt;config.h&gt; #define __REGb(x) (*(volatile unsigned char *)(x)) #...

    embedalign:通过词法边缘化来深层生成单词表示的模式

    tensorflow r1.4 pip install --upgrade https://storage.googleapis.com/tensorflow/linux/gpu/tensorflow_gpu-1.4.0rc1-cp35-cp35m-linux_x86_64.whl 标签embedalign.naacl2018.submission 例子 查看

    学习Excel导入将数据导入到数据库.pdf

    &lt;tr align="center"&gt; ;"&gt; 上传" class="buttonC4"/&gt; ``` 当用户提交文件后,服务器端的Java代码开始执行。这里,我们需要处理文件上传并读取Excel数据。注意,确保Excel文件中的数据是文本格式,因为非...

    CSS3教程(4):网页边框和网页文字阴影

    需要注意的是,虽然大多数现代浏览器(如Firefox, Chrome, Opera, Safari)已经支持这些CSS3特性,但Internet Explorer 7和8(不包括RC1版本)并不支持`box-shadow`和`text-shadow`。因此,在实际应用中,可能需要...

    angularjs表格ng-table使用备忘录

    &lt;script data-require="angular.js@*" data-semver="1.2.0-rc3-nonmin" src="angular.js"&gt; &lt;script data-require="ng-table@*" data-semver="0.3.0" src="ngTable.js"&gt; &lt;link data-require="ng-table@*" data-semver...

    VC 在CStatic控件绘图

    需要注意的是,如果使用`SetBitmap()`,则`CStatic`控件的“Align Text”属性不应设置为居中,否则可能导致图片无法正常显示。 ### 方法二:使用设备上下文(DC)绘图 第二种方法涉及更底层的绘图操作,通过修改`...

    mfc可视化计算器编程

    主要使用到Layout菜单中的Align功能对各个按钮进行对其,使界面更加整洁。拖出的控件有上面的一个Edit控件用于显示数字,Button控件用于处理鼠标的消息。 3. 建立的变量,控件的命名,对应的消息处理函数对应表 ...

    c++面向对象程序设计MFC简单计算器实验报告.docx

    - 设置控件属性,如ID、大小、位置等,使用Layout菜单的Align功能优化布局。 - 编写事件处理函数,响应用户的鼠标操作。 ### 3. 程序详细设计 #### 4.1 主调模块 - 创建窗口类,注册窗口类,并定义消息循环,处理...

    Raize Components 6.0 BUG PATCH

    Build_RC6.cmd文件进行修改,英文好一点的就不要往下看了,自己按照提示就可以修改好了,英文不好的,我会提示你更改哪几项,具体要改的值得根据你自己电脑实际更改。 ①找到 rem goto InitComplete 行,吧前面的...

Global site tag (gtag.js) - Google Analytics