- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我的 UI 有一个 UITableView 和另一个 View 。 UITableView 具有可折叠部分,如 sample code 所示。 。加载后看起来像这样:
用户可以使用滑动手势来展开右侧的灰色 View 。我只是改变灰色 View 的原点并使用动画来实现这一点。那么它看起来像这样:
但是,当我关闭 UITableView 的一部分时,UI 似乎会“重置”并恢复到其原始布局:
当我打开或关闭 UITableView 的一部分时,我希望灰色 View 保持在原来的位置。我怎样才能做到这一点?
我的 ViewController 的代码如下:
#import "BooklistMaster.h"
#import "BooklistDetailViewController.h"
#import "Booklist.h"
#import "BooklistTitleCell.h"
#import "BooklistTitle.h"
#import "BooklistDetailHeaderView.h"
#import "BooklistDetailSectionInfo.h"
#import "BooklistDetailHandler.h"
static NSString *SectionHeaderViewIdentifier = @"SectionHeaderViewIdentifier";
@interface BooklistDetailViewController ()
@property (nonatomic) NSMutableArray *sectionInfoArray;
@property (nonatomic) NSIndexPath *pinchedIndexPath;
@property (nonatomic) NSInteger openSectionIndex;
@property (nonatomic) CGFloat initialPinchHeight;
@property (nonatomic) IBOutlet BooklistDetailHeaderView *sectionHeaderView;
// use the uniformRowHeight property if the pinch gesture should change all row heights simultaneously
@property (nonatomic) NSInteger uniformRowHeight;
@end
//#define DEFAULT_ROW_HEIGHT 88
#define HEADER_HEIGHT 48
@implementation BooklistDetailViewController
@synthesize listTitleLabel;
@synthesize backButton;
@synthesize booklistMaster;
@synthesize titleBarView;
@synthesize searchBar;
@synthesize tap;
@synthesize booklistId;
@synthesize booklistName;
@synthesize spinner;
@synthesize detailPaneFocused;
@synthesize detailView;
@synthesize swipeLeft;
@synthesize swipeRight;
@synthesize tableView;
- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
if (self) {
// Custom initialization
}
return self;
}
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view.
[self.view sendSubviewToBack:tableView];
detailPaneFocused=NO;
listTitleLabel.text = booklistName;
[spinner startAnimating];
[spinner setHidden:NO];
[titleBarView setBackgroundColor:[UIColor colorWithRed:.77647 green:.77647 blue:.79607 alpha:1]];
tap = [[UITapGestureRecognizer alloc]
initWithTarget:self
action:@selector(dismissKeyboard)];
[tap setCancelsTouchesInView:YES];
swipeLeft = [[UISwipeGestureRecognizer alloc]initWithTarget:self action:@selector(didSwipeLeft:)];
[swipeLeft setDirection:UISwipeGestureRecognizerDirectionLeft];
[detailView addGestureRecognizer:swipeLeft];
swipeRight = [[UISwipeGestureRecognizer alloc]initWithTarget:self action:@selector(didSwipeRight:)];
[swipeRight setDirection:UISwipeGestureRecognizerDirectionRight];
[detailView addGestureRecognizer:swipeRight];
// Set up default values.
self.tableView.sectionHeaderHeight = HEADER_HEIGHT;
/*
The section info array is thrown away in viewWillUnload, so it's OK to set the default values here. If you keep the section information etc. then set the default values in the designated initializer.
*/
// self.uniformRowHeight = DEFAULT_ROW_HEIGHT;
self.openSectionIndex = NSNotFound;
UINib *sectionHeaderNib = [UINib nibWithNibName:@"BooklistDetailSectionHeaderView" bundle:nil];
[self.tableView registerNib:sectionHeaderNib forHeaderFooterViewReuseIdentifier:SectionHeaderViewIdentifier];
self.tableView.tableFooterView = [[UIView alloc] initWithFrame:CGRectZero];
BooklistDetailHandler *handler = [[BooklistDetailHandler alloc] initForController:self];
[handler getBooklistForId:booklistId];
}
-(void)didSwipeRight:(UISwipeGestureRecognizer *)recognizer{
if (detailPaneFocused==YES) {
detailPaneFocused=NO;
[UIView animateWithDuration:.3
animations:^ {
CGRect newBounds = self.detailView.frame;
newBounds.origin.x += 200;
self.detailView.frame = newBounds;
[self.detailView layoutSubviews];
}];
}
}
-(void)didSwipeLeft:(UISwipeGestureRecognizer *)recognizer{
if (detailPaneFocused==NO) {
detailPaneFocused=YES;
[UIView animateWithDuration:.3
animations:^ {
CGRect newBounds = self.detailView.frame;
newBounds.origin.x -= 200;
self.detailView.frame = newBounds;
[self.detailView layoutSubviews];
}];
}
}
- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
- (void)viewWillAppear:(BOOL)animated {
[super viewWillAppear:animated];
}
- (IBAction)buttonPressed:(id)sender {
[[self navigationController] popViewControllerAnimated:YES];
}
-(void)dismissKeyboard {
[searchBar resignFirstResponder];
}
- (void)searchBarSearchButtonClicked:(UISearchBar *)searchBar
{
[self dismissKeyboard];
}
-(void)searchBarTextDidBeginEditing:(UISearchBar *)searchBar{
[self.view addGestureRecognizer:tap];
[UIView animateWithDuration:.3
animations:^ {
CGRect newBounds = self.searchBar.frame;
newBounds.size.width += 200; //newBounds.size.width -= 215; to contract
newBounds.origin.x -= 200;
self.searchBar.frame = newBounds;
[self.searchBar layoutSubviews];
}];
}
-(void)searchBarTextDidEndEditing:(UISearchBar *)searchBar{
[self.view removeGestureRecognizer:tap];
[UIView animateWithDuration:.3
animations:^ {
CGRect newBounds = self.searchBar.frame;
newBounds.size.width -= 200; //newBounds.size.width -= 215; to contract
newBounds.origin.x += 200;
self.searchBar.frame = newBounds;
[self.searchBar layoutSubviews];
}];
}
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView{
return booklistMaster.subLists.count;
}
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{
// Booklist *list = [booklistMaster.subLists objectAtIndex:section];
// return list.booklistTitles.count;
BooklistDetailSectionInfo *sectionInfo = (self.sectionInfoArray)[section];
NSInteger numStoriesInSection = sectionInfo.booklist.booklistTitles.count;
return sectionInfo.open ? numStoriesInSection : 0;
}
//- (NSArray *)sectionIndexTitlesForTableView:(UITableView *)tableView{
//
// NSMutableArray *array = [[NSMutableArray alloc] init];
// for (Booklist *list in booklistMaster.subLists) {
// [array addObject:@"a"];
// }
//
// return array;
//
//}
- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section{
BooklistDetailHeaderView *sectionHeaderView = [self.tableView dequeueReusableHeaderFooterViewWithIdentifier:SectionHeaderViewIdentifier];
BooklistDetailSectionInfo *sectionInfo = (self.sectionInfoArray)[section];
sectionInfo.headerView = sectionHeaderView;
sectionHeaderView.titleLabel.text = sectionInfo.booklist.listName;
sectionHeaderView.section = section;
sectionHeaderView.delegate = self;
return sectionHeaderView;
}
//- (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section{
//
//}
//- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
//
// BooklistDetailSectionInfo *sectionInfo = (self.sectionInfoArray)[indexPath.section];
// return [[sectionInfo objectInRowHeightsAtIndex:indexPath.row] floatValue];
// // Alternatively, return rowHeight.
//}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
BooklistTitleCell *cell = [tableView dequeueReusableCellWithIdentifier:@"BooklistTitleCell" forIndexPath:indexPath];
Booklist *booklist = [booklistMaster.subLists objectAtIndex:indexPath.section];
BooklistTitle *booklistTitle = [booklist.booklistTitles objectAtIndex:indexPath.row];
cell.titleLabel.text=booklistTitle.title;
return cell;
}
- (void)sectionHeaderView:(BooklistDetailHeaderView *)sectionHeaderView sectionOpened:(NSInteger)sectionOpened {
BooklistDetailSectionInfo *sectionInfo = (self.sectionInfoArray)[sectionOpened];
sectionInfo.open = YES;
/*
Create an array containing the index paths of the rows to insert: These correspond to the rows for each quotation in the current section.
*/
NSInteger countOfRowsToInsert = sectionInfo.booklist.booklistTitles.count;
NSMutableArray *indexPathsToInsert = [[NSMutableArray alloc] init];
for (NSInteger i = 0; i < countOfRowsToInsert; i++) {
[indexPathsToInsert addObject:[NSIndexPath indexPathForRow:i inSection:sectionOpened]];
}
/*
Create an array containing the index paths of the rows to delete: These correspond to the rows for each quotation in the previously-open section, if there was one.
*/
NSMutableArray *indexPathsToDelete = [[NSMutableArray alloc] init];
NSInteger previousOpenSectionIndex = self.openSectionIndex;
if (previousOpenSectionIndex != NSNotFound) {
BooklistDetailSectionInfo *previousOpenSection = (self.sectionInfoArray)[previousOpenSectionIndex];
previousOpenSection.open = NO;
[previousOpenSection.headerView toggleOpenWithUserAction:NO];
NSInteger countOfRowsToDelete = previousOpenSection.booklist.booklistTitles.count;
for (NSInteger i = 0; i < countOfRowsToDelete; i++) {
[indexPathsToDelete addObject:[NSIndexPath indexPathForRow:i inSection:previousOpenSectionIndex]];
}
}
// style the animation so that there's a smooth flow in either direction
UITableViewRowAnimation insertAnimation;
UITableViewRowAnimation deleteAnimation;
if (previousOpenSectionIndex == NSNotFound || sectionOpened < previousOpenSectionIndex) {
insertAnimation = UITableViewRowAnimationTop;
deleteAnimation = UITableViewRowAnimationBottom;
}
else {
insertAnimation = UITableViewRowAnimationBottom;
deleteAnimation = UITableViewRowAnimationTop;
}
// apply the updates
[self.tableView beginUpdates];
[self.tableView insertRowsAtIndexPaths:indexPathsToInsert withRowAnimation:insertAnimation];
[self.tableView deleteRowsAtIndexPaths:indexPathsToDelete withRowAnimation:deleteAnimation];
[self.tableView endUpdates];
self.openSectionIndex = sectionOpened;
}
- (void)sectionHeaderView:(BooklistDetailHeaderView *)sectionHeaderView sectionClosed:(NSInteger)sectionClosed {
/*
Create an array of the index paths of the rows in the section that was closed, then delete those rows from the table view.
*/
BooklistDetailSectionInfo *sectionInfo = (self.sectionInfoArray)[sectionClosed];
sectionInfo.open = NO;
NSInteger countOfRowsToDelete = [self.tableView numberOfRowsInSection:sectionClosed];
if (countOfRowsToDelete > 0) {
NSMutableArray *indexPathsToDelete = [[NSMutableArray alloc] init];
for (NSInteger i = 0; i < countOfRowsToDelete; i++) {
[indexPathsToDelete addObject:[NSIndexPath indexPathForRow:i inSection:sectionClosed]];
}
[self.tableView deleteRowsAtIndexPaths:indexPathsToDelete withRowAnimation:UITableViewRowAnimationTop];
}
self.openSectionIndex = NSNotFound;
}
-(void)setMasterBooklist:(BooklistMaster *)list{
booklistMaster=list;
/*
Check whether the section info array has been created, and if so whether the section count still matches the current section count. In general, you need to keep the section info synchronized with the rows and section. If you support editing in the table view, you need to appropriately update the section info during editing operations.
*/
if ((self.sectionInfoArray == nil) ||
([self.sectionInfoArray count] != [self numberOfSectionsInTableView:self.tableView])) {
// For each play, set up a corresponding SectionInfo object to contain the default height for each row.
NSMutableArray *infoArray = [[NSMutableArray alloc] init];
for (Booklist *list in self.booklistMaster.subLists) {
BooklistDetailSectionInfo *sectionInfo = [[BooklistDetailSectionInfo alloc] init];
sectionInfo.booklist = list;
sectionInfo.open = NO;
// NSNumber *defaultRowHeight = @(DEFAULT_ROW_HEIGHT);
NSInteger countOfTitles = sectionInfo.booklist.booklistTitles.count;
for (NSInteger i = 0; i < countOfTitles; i++) {
//[sectionInfo insertObject:defaultRowHeight inRowHeightsAtIndex:i];
}
[infoArray addObject:sectionInfo];
}
self.sectionInfoArray = infoArray;
}
[spinner stopAnimating];
[spinner setHidden:YES];
[self.tableView reloadData];
}
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
// if (detailPaneFocused==NO){
// [self focusDetailPane];
// }
}
- (NSIndexPath *)tableView:(UITableView *)tableView willSelectRowAtIndexPath:(NSIndexPath *)indexPath{
if (detailPaneFocused==NO&&![indexPath isEqual:[tableView indexPathForSelectedRow]]){
[self focusDetailPane];
}
return indexPath;
}
- (void)focusDetailPane{
detailPaneFocused=YES;
[UIView animateWithDuration:.3
animations:^ {
CGRect newBounds = self.detailView.frame;
newBounds.origin.x -= 200;
self.detailView.frame = newBounds;
[self.detailView layoutSubviews];
}];
}
@end
最佳答案
这是由于自动布局造成的。如果在打开自动布局的情况下直接更改任何框架,则一旦 View 必须更新自身, View 将恢复到由其约束定义的位置。您应该通过为您想要更改的任何约束创建 IBOutlet 来完成动画,并在代码中调整它们的常量值(或者关闭自动布局)。
关于iOS:表格重新加载改变布局,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20253012/
这个问题在这里已经有了答案: How does Scala's apply() method magic work? (3 个回答) 9年前关闭。 假设我在 scala 中有一个 MyList 类,其
这个问题在这里已经有了答案: What is a non-capturing group in regular expressions? (18 个回答) Reference - What does
这个问题是针对嵌入式系统的! 我有以下选项来初始化一个对象: Object* o = new Object(arg); 这会将对象放入堆中并返回指向它的指针。我不喜欢在嵌入式软件中使用动态分配。 Ob
我自己搜索过,没能成功的正则表达式。 我有一个 html 文件,其中包含 [] 之间的变量我想把每一个字都写进去。 [client_name][client_company] [cl
我是 Python 新手。我不明白为什么这段代码不起作用: reOptions = re.search( "[\s+@twitter\s+(?P\w+):(?P.*?)\s+]", d
在过去 7 个月左右的时间里,我几乎一直在使用 .NET C# 进行编程。在那之前,我的大部分编程都是用 C++(从学校里学的)。在工作中,我可能需要在接下来的几个月里做一大堆 C 语言。我对 C 的
我是 RE 的新手,我正在尝试获取歌词并分离出歌词标题、和声和主唱: 下面是一些歌词的例子: [Intro] D.A. got that dope! [Chorus: Travis Scott] Ic
这可能是不可能的,但我想检查是否可以用一种简单的方式表达这样的事情: // obviously doesn't work class Foo : IFoo where T: Bar {
我们的应用程序中有“user”和“study”实体,存储在它们各自的表中。一项研究代表一种研究和已收集的数据。它们是多对多的关系,所以我们需要一个链接表:studies_users。 我们为用户分配角
将测试条件添加到 Visual Studio 2010 数据库单元测试(对于 SQL Server 2008)时,这些条件称为例如rowCountCondition1、rowCountConditio
在模拟器上,我可以从设置中卸载 SD 卡。 然后我可以将它安装到我的操作系统上,然后正常卸载它。 我一直无法弄清楚如何在模拟器上重新安装它(无需重新启动)。 提示: adb 命令 remount 是无
假设在一个分支上执行了一系列提交,但该分支尚未与主干重新同步。是否可以从提交中生成全局补丁?是否可以从一系列提交中生成“分组”补丁?如果是,如何? 最佳答案 svn diff -rXXX:YYY UR
在某些情况下,我想在我的应用程序中锁定调整大小功能,为此我尝试对属性进行数据绑定(bind),并且不允许在某些情况下更改它,但没有成功。 有没有办法这样做? 这是我不成功的尝试: XAML: Vie
当我的计算机连接多个显示器时,我可以检测它们,并根据从获取的值设置位置来向它们绘制图形 get(0, 'MonitorPositions') 但是,当我在 MATLAB 运行时断开监视器时,此属性不会
我们有一个grails应用程序,该应用程序在grails数据库中存储了各种域对象。该应用程序连接到第二个数据库,运行一些原始sql,并在表中显示结果。它基本上是一个报告服务器。 我们通过在DataSo
无法比较来自不同容器的迭代器(参见这里的示例: https://stackoverflow.com/a/4664519/225186 )(或者从技术上讲,它不需要有意义。) 这就提出了另一个问题,来自
我有以下情况: 家长 Activity : ParentActivityClass { private Intent intent; @Override public void onCreate(Bu
我经常将元素与附加功能 Hook ,例如: $('.myfav').autocomplete(); $('.myfav').datepicker(); $('.myfav').click(somefu
因此,我将 tooltipster.js 库用于工具提示,并尝试更改工具提示在不同屏幕尺寸上的默认距离。 所以这是默认的 init 的样子: $(inputTooltipTrigger).tool
我在 ARM7 嵌入式环境中工作。我使用的编译器不支持完整的 C++ 功能。它不支持的一项功能是动态类型转换。 有没有办法实现dynamic_cast<>() ? 我使用 Google 寻找代码,但到
我是一名优秀的程序员,十分优秀!