我正在编写一个需要同时评估和选择网格上的复选框元素的Typescript / Protractor测试。这些复选框缺少ID。由于我可以处理与这些复选框位于同一行的输入元素,因此我尝试通过元素查找器(by.id,by.xpath和by.css)的组合来获取它们。但我的尝试失败了。
我想写的是一个小函数,它将根据一个遥远的兄弟"返回行中的目标复选框元素。行上的元素和行上的复选框索引。我已经尝试了几件事,我认为最有希望的尝试发布在这里。
这是我定位的网格标记部分。为清楚起见,省略了几列信息。这就是我所争辩的。我无权更改标记,我只被要求编写量角器测试。
<div ng-repeat="(rowRenderIndex, row) in rowContainer.renderedRows track by $index" class="ui-grid-row ng-scope" ng-style="Viewport.rowStyle(rowRenderIndex)">
<div ui-grid-row="row" row-render-index="rowRenderIndex" class="ng-isolate-scope">
<div ng-repeat="(colRenderIndex, col) in colContainer.renderedColumns track by col.uid" class="ui-grid-cell ng-scope ui-grid-coluiGrid-09T" ng-class="{ 'ui-grid-row-header-cell': col.isRowHeader }" ui-grid-cell="">
<div ng-click="grid.appScope.rowColumnSelect(row, 'rating')" ng-class="{'px-grid-row-selected': grid.appScope.highlightCell(row, 'rating'), 'px-can-edit': grid.appScope.clickableCell(row.entity, 'rating'), 'px-grid-total-row': row.entity.isTotal, 'px-grid-notable-row': row.entity.isNotable}" class="ng-scope">
<div class="ui-grid-cell-contents ng-hide" ng-hide="grid.appScope.showCell(row, 'rating')">
</div>
<div ng-show="grid.appScope.showCell(row, 'rating')" name="loanLossAssumption3" class="ui-grid-cell-contents">
<div ng-if="grid.appScope.haveData(row)" class="text-right ng-scope">
<div error-display-strategy="grid" px-input-field="" id="loanLossAssumptionrating3" name="loanLossAssumptionrating3" ng-if="grid.appScope.inEditMode(row, 'rating')" input-type="number" px-input-required="true" px-round-value="0" ng-model="row.entity.rating" class="ng-pristine ng-untouched ng-valid ng-scope ng-isolate-scope">
<div ng-class="{'row': !vm.isGridErrorDisplay && !vm.autoWidth}">
<div>
<div tooltip="" tooltip-placement="bottom" tooltip-append-to-body="true" tooltip-trigger="open" class="ng-scope">
<div px-validation="loanLossAssumptionrating3InputField" ng-class="{'col-xs-12': !vm.isGridErrorDisplay && !vm.autoWidth}" min="" max="" input-type="number" maxlength="" minlength="" error-display-strategy="grid" px-tooltip="" class="ng-isolate-scope">
<div class="row">
<div ng-class="vm.inputFieldClass()" class="col-xs-12">
<div class="input-group px-input-group-overrides px-no-grid-padding" ng-hide="vm.pxInputDisabled">
<span id="loanLossAssumptionrating3Prefix" name="loanLossAssumptionrating3Prefix" class="input-group-addon input-left-addon ng-binding ng-hide" ng-show="vm.prefix" ng-class="{'px-auto-width': vm.autoWidth}">
</span>
<div class="inner-addon">
<input id="loanLossAssumptionrating3InputField" name="loanLossAssumptionrating3InputField" class="form-control ng-pristine ng-untouched ng-valid ng-valid-px-email ng-valid-required ng-valid-unique" ng-class="{'px-input-with-tooltip': !!vm.pxTooltip, 'px-auto-width': vm.autoWidth}" ng-required="vm.pxInputRequired" px-round-value="0" px-edit-rate="false" type="text" ng-model="vm.ngModel" px-email-validator="" ng-disabled="vm.pxControlDisabled" placeholder="" ng-change="vm.onChange()" ng-focus="vm.showTooltip(true)" ng-blur="vm.showTooltip(false)" px-unique-field="" autocomplete="off" required="required">
</div>
<span class="fa fa-exclamation-circle inline-error form-control-feedback" ng-style="{left:vm.feedbackIconLeft()}" style="left: 0px;">
</span>
<span class="input-group-addon input-empty-addon" ng-hide="vm.postfix" ng-class="{'px-auto-width': vm.autoWidth}">
</span>
</div>
</div>
<span ng-transclude="" class="px-vertical-middle">
</span>
</div>
</div>
</div>
</div>
<div px-tooltip-view="" px-tooltip="vm.pxTooltip" ng-click="vm.applyAdjustment()" class="ng-isolate-scope">
</div>
</div>
</div>
<span ng-hide="grid.appScope.inEditMode(row, 'rating')" ng-class="{'px-font-highlighted': grid.appScope.showHighlighted(row, 'rating'), 'px-font-enabled': grid.appScope.showEnabled(row, 'rating'), 'px-font-disabled': grid.appScope.showDisabled(row, 'rating'), 'px-cannot-edit': !grid.appScope.clickableCell(row.entity, 'rating')}" class="ng-binding px-font-enabled px-cannot-edit ng-hide">3</span>
</div>
</div>
</div>
</div>
<div ng-repeat="(colRenderIndex, col) in colContainer.renderedColumns track by col.uid"
class="ui-grid-cell ng-scope ui-grid-coluiGrid-09Y" ng-class="{ 'ui-grid-row-header-cell': col.isRowHeader }" ui-grid-cell="">
<div ng-click="grid.appScope.rowColumnSelect(row, 'isDefault')"
ng-class="{'px-grid-row-selected': grid.appScope.highlightCell(row, 'isDefault'), 'px-can-edit': grid.appScope.clickableCell(row.entity, 'isDefault'), 'px-grid-total-row': row.entity.isTotal, 'px-grid-notable-row': row.entity.isNotable}"
class="ng-scope">
<div class="ui-grid-cell-contents ng-hide" ng-hide="grid.appScope.showCell(row, 'isDefault')">
</div>
<div ng-show="grid.appScope.showCell(row, 'isDefault')" name="loanLossAssumptionfalse" class="ui-grid-cell-contents"> <!-- Changes to loanLossAssumptiontrue-->
<div ng-if="grid.appScope.haveData(row)" class="text-center ng-scope">
<input ng-disabled="!grid.appScope.inEditMode(row, 'isDefault')" type="checkbox" ng-input="row.entity.isDefault" ng-model="row.entity.isDefault"
class="ng-pristine ng-valid ng-touched">
</div>
</div>
</div>
</div>
</div>
</div>
在这种情况下,我在输入字段上有一个句柄,其ID为loanLossAssumptionrating3InputField。从这里开始,我需要从这个ui-grid-row中获取其他复选框。在此示例中,在6个结束div标签之前,标记末尾附近会出现一个复选框。
随着时间的推移,我们构建了一个函数库,因此可以参考&#34; lib。&#34;在这段代码中。这只是我们图书馆的工作参考。这段代码位于pageObject中。
lib.getCheckboxInRow(fields.ratingFieldId, 0).then(function (checkboxElement) {
// The fields.ratingFieldId value = "loanLossAssumptionrating3InputField"
checkboxElement.click();
});
这只是一个&#34;点击&#34;尝试,但我还需要获取该元素并确定是否单击了该复选框。在我的情况下,通过标记执行此操作的唯一方法是读取父元素中的属性。我知道该怎么做,但是提供这个细节来说明为什么我试图返回元素本身。
这是我的目标功能。下面显示了两次尝试。这两个都失败了。这是我的图书馆。
// Attempt A
public getCheckboxInRow(siblingElementId: string, checkboxIndex: number): webdriver.promise.Promise<protractor.ElementFinder> {
var xPathToRow : string = "./ancestor::div[contains(concat(' ', @class, ' '), ' ui-grid-row ')][1]";
return element(by.id(siblingElementId)).element(by.xpath(xPathToRow)).then(function (row) {
return row.all(by.css("input[@type='checkbox']")).filter.then(function (checkboxArray) {
return checkboxArray[checkboxIndex];
});
});
}
// Attempt B
public getCheckboxInRow(siblingElementId: string, checkboxIndex: number): webdriver.promise.Promise<protractor.ElementFinder> {
var xPathToRow : string = "./ancestor::div[contains(concat(' ', @class, ' '), ' ui-grid-row ')][1]";
var cssToCheckbox : string = "input[@type='checkbox']";
return element(by.id(siblingElementId)).element(by.xpath(xPathToRow)).all(by.css(cssToCheckbox)).filter(function(checkbox) {
return true;
}).then(function (checkboxArray) {
return checkboxArray[checkboxIndex];
});
}
用&#34;尝试A&#34;发生以下错误。
TypeError:element(...)。element(...)。then不是函数
错误更具戏剧性,&#34;尝试B&#34;。
InvalidSelectorError:无效的选择器:无效或非法的选择器 已指定(会话信息:chrome = 47.0.2526.111)(驱动程序信息: chromedriver = 2.19.346078 (6f1f0cde889532d48ce8242342d0b84f94b114a1),platform = Windows NT 6.3 x86_64)(警告:服务器未提供任何堆栈跟踪 信息)命令持续时间或超时:30毫秒For 有关此错误的文档,请访问: http://seleniumhq.org/exceptions/invalid_selector_exception.html建立 info:version:&#39; 2.47.1&#39;,revision:&#39; 411b314&#39;,time:&#39; 2015-07-30 03:03:16&#39;系统信息:主持人:&#39; PSIDEVKPALM&#39;,ip:&#39; 10.226.128.27&#39;, os.name:&#39; Windows Server 2012 R2&#39;,os.arch:&#39; x86&#39;,os.version:&#39; 6.3&#39;, java.version:&#39; 1.8.0_66&#39;司机信息: org.openqa.selenium.chrome.ChromeDriver功能 [{applicationCacheEnabled = false,rotate = false, mobileEmulationEnabled =假, 铬= {userDataDir = C:\用户\ kpalmer \应用程序数据\本地\温度\ 2 \ scoped_dir7352_8503}, takesHeapSnapshot = true,databaseEnabled = false,handlesAlerts = true, hasTouchScreen = false,版本= 47.0.2526.111,platform = WIN8_1, browserConnectionEnabled = false,nativeEvents = true, acceptSslCerts = true,locationContextEnabled = true, webStorageEnabled = true,browserName = chrome,takesScreenshot = true, javascriptEnabled = true,cssSelectorsEnabled = true}]会话ID: 12959162beeed374185734ef29b03b57 ***元素信息:{Using = css selector,value = input [@type =&#39; checkbox&#39;]}
花了很多时间研究这个问题,我不知道为什么会失败。
最终我想要的是一个函数,它将返回一个checkbox元素,该元素与该行中的另一个元素位于同一行,基于复选框索引,这样我们可以单击复选框或确定该元素是否为选中或取消选中。
感谢您的帮助。
===编辑1/2/2016下午===
谢谢你@alecxe,你是一个巨大的帮助。以下是我运行的最终实现。这两种方法都有效。
其中任何一个都可以在我的pageObject中使用。
lib.getCheckboxInRow(fields.ratingFieldId, 0).click();
lib.getCheckboxArrayInRow(fields.ratingFieldId).filter(function (checkboxArray, index) {
return (index === 0);
}).first().click();
这些是图书馆的目标功能。我必须调整的一件事是返回类型,从承诺到元素查找器。这些工作就像一个冠军。
public getCheckboxInRow(siblingElementId: string, checkboxIndex: number): protractor.ElementFinder {
var xPathToRow : string = "./ancestor::div[contains(concat(' ', @class, ' '), ' ui-grid-row ')][1]";
return element(by.id(siblingElementId)).element(by.xpath(xPathToRow)).all(by.css("input[type=checkbox]")).filter(function (checkboxArray, index) {
return index === checkboxIndex;
}).first();
};
public getCheckboxArrayInRow(siblingElementId: string): protractor.ElementArrayFinder {
var xPathToRow : string = "./ancestor::div[contains(concat(' ', @class, ' '), ' ui-grid-row ')][1]";
return element(by.id(siblingElementId)).element(by.xpath(xPathToRow)).all(by.css("input[type=checkbox]"));
};
答案 0 :(得分:2)
让我们回顾一下您的尝试。
Attempt A
您收到TypeError: element(...).element(...).then is not a function
错误,因为自Protractor 2.0.0(changelog)以来您无法使用ElementFinder
解析then()
。
在这种情况下,您可以继续链接element
和all()
:
public getCheckboxInRow(siblingElementId: string, checkboxIndex: number): webdriver.promise.Promise<protractor.ElementFinder> {
var xPathToRow : string = "./ancestor::div[contains(concat(' ', @class, ' '), ' ui-grid-row ')][1]";
return element(by.id(siblingElementId)).element(by.xpath(xPathToRow)).all(by.css("input[type=checkbox]")).filter(function (checkboxArray, index) {
return index === checkboxIndex;
}).first();
}
我还修改了应用filter()
功能的方式。并注意first()
调用 - 这将有助于我们获得单个过滤元素而不是数组。
Attempt B
您收到An invalid or illegal selector was specified
错误,因为您实际上有一个无效的CSS选择器input[@type='checkbox']
- 您不需要在CSS中添加@
属性名称 - 将其替换为{{ 1}}(也不需要单引号)。
并且,您也不会在此尝试中正确应用input[type=checkbox]
功能。修正版:
filter()
基本上与尝试A相同。