- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我创建了一个自定义 collectionViewLayout
,它根据设备屏幕大小按比例缩放单元格。包含在单元格中的所有图像似乎都被拉伸(stretch)了。我试过使用 aspectFit 和 aspectFill,但它们只会进一步扭曲照片。
下面是 CollectionView 布局代码:
import UIKit
import Photos
class ViewController: UIViewController,UICollectionViewDelegate,UICollectionViewDelegateFlowLayout,UICollectionViewDataSource {
var imageArray:[UIImage] = [UIImage]()
var collectionView:UICollectionView! = {
let layout = UICollectionViewFlowLayout()
layout.minimumInteritemSpacing = 1
layout.minimumLineSpacing = 1
layout.scrollDirection = .vertical
let CV = UICollectionView(frame: UIScreen.main.bounds, collectionViewLayout: layout)
CV.translatesAutoresizingMaskIntoConstraints = false
return CV
}()
//Add to the official project
var PhotoAlbumTitleLabel:UILabel = {
let lbl = UILabel()
lbl.translatesAutoresizingMaskIntoConstraints = false
lbl.textColor = .white
lbl.textAlignment = .center
lbl.font = lbl.font.withSize(36)
lbl.text = "Photo Album"
return lbl
}()
var fetchResult: PHFetchResult<PHAsset> = PHFetchResult()
override func viewDidLoad() {
super.viewDidLoad()
let layout = customCollectionViewLayout()
// layout.minimumInteritemSpacing = 1
// layout.minimumLineSpacing = 1
// layout.scrollDirection = .vertical
collectionView = UICollectionView(frame: self.view.frame, collectionViewLayout: layout)
// layout.itemSize = CGSize(width: UIScreen.main.bounds.width/3 - 1, height: UIScreen.main.bounds.width/3 - 1)
self.collectionView.delegate = self
self.collectionView.dataSource = self
collectionView.register(customPhotoCell.self, forCellWithReuseIdentifier: "Cell")
self.view.addSubview(collectionView)
//Add to the official project
self.view.addSubview(PhotoAlbumTitleLabel)
setUpLayout()
fetchAssets()
let photos = PHPhotoLibrary.authorizationStatus()
if photos == .notDetermined {
PHPhotoLibrary.requestAuthorization({status in
if status == .authorized{
print("permission granted")
DispatchQueue.main.async {
self.fetchAssets()
self.collectionView.reloadData()
}
} else {
print("permission not granted")
}
})
}
}
func setUpLayout(){
collectionView.topAnchor.constraint(equalTo: self.view.topAnchor, constant: 0).isActive = true
collectionView.bottomAnchor.constraint(equalTo: self.view.bottomAnchor, constant: 0).isActive = true
collectionView.trailingAnchor.constraint(equalTo: self.view.trailingAnchor, constant: 0).isActive = true
collectionView.leadingAnchor.constraint(equalTo: self.view.leadingAnchor, constant: 0).isActive = true
PhotoAlbumTitleLabel.topAnchor.constraint(equalTo: self.view.topAnchor, constant: -45).isActive = true
PhotoAlbumTitleLabel.centerXAnchor.constraint(equalTo: self.view.centerXAnchor, constant: 0).isActive = true
}
func fetchAssets(){
let fetchOptions = PHFetchOptions()
fetchOptions.sortDescriptors = [NSSortDescriptor(key: "creationDate", ascending: false)]
fetchResult = PHAsset.fetchAssets(with: .image, options: fetchOptions)
}
func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
return fetchResult.count
}
func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "Cell", for: indexPath) as! customPhotoCell
// cell.backgroundColor = .blue
// cell.imgView.contentMode = .scaleToFill
// cell.imgView.image = #imageLiteral(resourceName: "ManOnTheMoon")
let imageView = cell.imgView
let requestOptions = PHImageRequestOptions()
requestOptions.isSynchronous = true
requestOptions.deliveryMode = .highQualityFormat
PHImageManager.default().requestImage(for: fetchResult.object(at: indexPath.item) , targetSize: CGSize(width: 200,height: 200), contentMode: .aspectFill, options: requestOptions, resultHandler: {
image,error in
if let error = error{
print(error)
}
self.imageArray.append(image!)
imageView.image = image!
imageView.contentMode = .scaleToFill
})
return cell
}
func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, minimumLineSpacingForSectionAt section: Int) -> CGFloat {
return 1
}
func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, minimumInteritemSpacingForSectionAt section: Int) -> CGFloat {
return 1
}
func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
let width = self.view.bounds.width/3 - 1
return CGSize(width: width, height: width)
}
}
class customPhotoCell:UICollectionViewCell{
var imgView:UIImageView = {
let imgV = UIImageView()
imgV.backgroundColor = .orange
imgV.translatesAutoresizingMaskIntoConstraints = false
return imgV
}()
override init(frame: CGRect) {
super.init(frame: frame)
print("Inside of init")
self.addSubview(imgView)
imgView.topAnchor.constraint(equalTo: self.topAnchor, constant: 0).isActive = true
imgView.bottomAnchor.constraint(equalTo: self.bottomAnchor, constant: 0).isActive = true
imgView.leadingAnchor.constraint(equalTo: self.leadingAnchor, constant: 0).isActive = true
imgView.trailingAnchor.constraint(equalTo: self.trailingAnchor, constant: 0).isActive = true
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}
extension UIImageView{
func fetchImage(asset: PHAsset, contentMode: PHImageContentMode, targetSize: CGSize) {
let options = PHImageRequestOptions()
options.version = .original
PHImageManager.default().requestImage(for: asset, targetSize: targetSize, contentMode: contentMode, options: options) { image, _ in
guard let image = image else { return }
switch contentMode {
case .aspectFill:
self.contentMode = .scaleAspectFill
case .aspectFit:
self.contentMode = .scaleAspectFit
}
self.image = image
}
}
}
extension UIImage{
func resizeImageWith() -> UIImage {
let aspectRatio = CGFloat(self.size.width / self.size.height)
let newWidth = UIScreen.main.bounds.width
let newSize = CGSize(width: newWidth, height: newWidth/aspectRatio)
UIGraphicsBeginImageContextWithOptions(newSize, true, 0)
self.draw(in: CGRect(origin: CGPoint(x: 0, y: 0), size: newSize))
let newImage = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()
return newImage!
}
}
下面是 CustomCollectionViewLayout 代码:
import UIKit
class gridLayouts{
static var screenHeight = UIScreen.main.bounds.height
static var screenWidth = UIScreen.main.bounds.width
var xOffsets:[CGFloat]
var yOffsets:[CGFloat]
var itemWidths:[CGFloat]
var itemHeights:[CGFloat]
var blockHeight:CGFloat?
init(xOffsets:[CGFloat],yOffsets:[CGFloat],itemWidths:[CGFloat], itemHeights:[CGFloat],blockHeight:CGFloat) {
self.xOffsets = xOffsets
self.yOffsets = yOffsets
self.itemWidths = itemWidths
self.itemHeights = itemHeights
self.blockHeight = blockHeight
}
func calcBlockHeight()->CGFloat{
//var totalWidth:CGFloat = CGFloat()
var blockHeight:CGFloat = 0
// print("itemWidths.count: \(itemWidths.count)")
for item in 0...itemWidths.count-1{
//totalWidth += self.itemWidths[item]
if self.xOffsets[item] + self.itemWidths[item] == UIScreen.main.bounds.width{
//var heightsInRow = itemHeights[0...item]
// print("smallest row height: \(heightsInRow.min()!)")
// print("final item in row: \(item)")
// blockHeight += heightsInRow.min()!
// print("item number: \(item)")
// print("item width: \((self.itemWidths[item]+xOffsets[item])/UIScreen.main.bounds.width)")
blockHeight += self.itemHeights[item]
// heightsInRow.removeAll()
//totalWidth = 0
}
}
return blockHeight
}
func updateYoffSets(previousBlockHeight:CGFloat)->[CGFloat]{
var updatedYoffsets = self.yOffsets
for item in 0...self.yOffsets.count - 1{
updatedYoffsets[item] += previousBlockHeight
}
return updatedYoffsets
}
}
//protocol customCollectionViewLayoutDelegate: class {
// func collectionView(_ collectionView:UICollectionView, heightForPhotoAtIndexPath indexPath:IndexPath) -> CGFloat
// func collectionView(_ collectionView:UICollectionView, widthForPhotoAtIndexPath indexPath:IndexPath) -> CGFloat
//}
class customCollectionViewLayout:UICollectionViewLayout{
// weak var delegate: customCollectionViewLayoutDelegate!
// 2
fileprivate var numberOfColumns = 2
fileprivate var cellPadding: CGFloat = 0
// 3
fileprivate var cache = [UICollectionViewLayoutAttributes]()
// 4
fileprivate var contentHeight: CGFloat = 0
fileprivate var contentWidth: CGFloat {
guard let collectionView = collectionView else {
return 0
}
let insets = collectionView.contentInset
return collectionView.bounds.width - (insets.left + insets.right)
}
// 5
override var collectionViewContentSize: CGSize {
return CGSize(width: contentWidth, height: contentHeight)
}
override func prepare() {
var testArr:[CGFloat] = [0,0,0,0,0,0]
// print("testArr first element before: \(testArr[0])")
testArr.incrementBy(5)
// print("testArr first element after: \(testArr[0])")
// 1
guard cache.isEmpty == true, let collectionView = collectionView else {
return
}
setUpLayout()
let screenWidth = UIScreen.main.bounds.width
let screenHeight = UIScreen.main.bounds.height
// 3
var totalBlockHeight:CGFloat = 0
var currentgridLayout:gridLayouts = appleLayout
var updatedYoffSets:[CGFloat] = [CGFloat]()
for item in 0 ..< collectionView.numberOfItems(inSection: 0) {
let indexPath = IndexPath(item: item, section: 0)
if item%8 == 0{
print("\(item): Im divisible by 8")
//currentgridLayout = NextLayout()
print("Incremented by \(totalBlockHeight/screenHeight)")
//make sure to move this down
// updatedYoffSets = currentgridLayout.updateYoffSets(previousBlockHeight: totalBlockHeight)
// totalBlockHeight += currentgridLayout.calcBlockHeight()
///totalBlockHeight = currentgridLayout.calcBlockHeight()
print("Total block Height: \(totalBlockHeight)")
print("Total block Height: \(totalBlockHeight/screenHeight)")
if (collectionView.numberOfItems(inSection: 0) - item < 8){
print("Im inside the if \(item)")
switch (collectionView.numberOfItems(inSection: 0) - item )%8{
case 0:
print("Im in case 0")
currentgridLayout = NextLayout()
case 1:
print("Im in case 1")
currentgridLayout = gridLayouts(xOffsets: [0], yOffsets: [0], itemWidths: [screenWidth], itemHeights: [screenHeight/3], blockHeight: screenHeight/3)
case 2:
print("Im in case 2")
currentgridLayout = gridLayouts(xOffsets: [0,screenWidth/2], yOffsets: [0,0], itemWidths: [screenWidth/2,screenWidth/2], itemHeights: [screenHeight/3], blockHeight: screenHeight/3)
case 3:
print("Im in case 3")
currentgridLayout = gridLayouts(xOffsets: [0,screenWidth/3,2*screenWidth/3], yOffsets: [0,0,0], itemWidths: [screenWidth/3,screenWidth/3,screenWidth/3], itemHeights: [screenHeight/3,screenHeight/3,screenHeight/3], blockHeight: screenHeight/3)
case 4:
print("Im in case 4")
currentgridLayout = gridLayouts(xOffsets: [0,2*screenWidth/3,0,screenWidth/3], yOffsets: [0,0,screenHeight/3,screenHeight/3], itemWidths: [2*screenWidth/3,screenWidth/3,screenWidth/3,2*screenWidth/3], itemHeights: [screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3], blockHeight: 2*screenHeight/3)
case 5:
print("Im in case 5")
currentgridLayout = gridLayouts(xOffsets: Array(appleLayout.xOffsets.prefix(through: 4)), yOffsets: Array(appleLayout.yOffsets.prefix(through: 4)), itemWidths: Array(appleLayout.itemWidths.prefix(through: 4)), itemHeights: Array(appleLayout.itemHeights.prefix(through: 4)), blockHeight: appleLayout.blockHeight!)
print("currentgridLayout.xOffsets.count: \(currentgridLayout.xOffsets.count)")
case 6:
print("Im in case 6")
currentgridLayout = gridLayouts(xOffsets: [0,2*screenWidth/3,2*screenWidth/3,0,screenWidth/3,2*screenWidth/3], yOffsets: [0,0,0,2*screenHeight/3,2*screenHeight/3,2*screenHeight/3], itemWidths: [2*screenWidth/3,screenWidth/3,screenWidth/3,screenWidth/3,screenWidth/3,screenWidth/3], itemHeights: [2*screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3], blockHeight: screenHeight)
case 7:
print("Im in case 7")
currentgridLayout = gridLayouts(xOffsets: [0,2*screenWidth/3,2*screenWidth/3,0,screenWidth/3,2*screenWidth/3,0], yOffsets: [0,0,0,2*screenHeight/3,2*screenHeight/3,2*screenHeight/3,screenHeight], itemWidths: [2*screenWidth/3,screenWidth/3,screenWidth/3,screenWidth/3,screenWidth/3,screenWidth/3,screenWidth], itemHeights: [2*screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3,2*screenHeight/3], blockHeight: screenHeight+2*screenHeight/3)
default:
print("Im in the default case")
currentgridLayout = NextLayout()
}
}
updatedYoffSets = currentgridLayout.updateYoffSets(previousBlockHeight: totalBlockHeight)
totalBlockHeight += currentgridLayout.calcBlockHeight()
}
let frame = CGRect(x: currentgridLayout.xOffsets[item%8], y: updatedYoffSets[item%8], width: currentgridLayout.itemWidths[item%8], height: currentgridLayout.itemHeights[item%8])
//
var insetFrame = frame.insetBy(dx: cellPadding, dy: cellPadding)
//changed
insetFrame = frame.insetBy(dx: 2, dy: 2)
// 5
let attributes = UICollectionViewLayoutAttributes(forCellWith: indexPath)
attributes.frame = insetFrame
cache.append(attributes)
// 6
contentHeight = max(contentHeight, frame.maxY)
//yOffset[column] = yOffset[column] + blockHeight
//push the y insets down
}
print("The block height is \(currentgridLayout.calcBlockHeight())")
print("The block height is \(currentgridLayout.calcBlockHeight()/UIScreen.main.bounds.height) screens")
}
func NextYOffSet( yOffset:[CGFloat],previousBlockHeight:CGFloat)->[CGFloat]{
var offSet = yOffset
for item in 0...offSet.count{
offSet[item] += previousBlockHeight
}
return offSet
}
func NextLayout()->gridLayouts{
let rand = Int(arc4random_uniform(UInt32(2)))
print("rand num \(rand)")
switch rand {
case 0:
print("Apple Layout Returned")
return appleLayout
default:
print("My Layout Returned")
return myLayout
}
}
var myLayout:gridLayouts!
var appleLayout:gridLayouts!
func setUpLayout(){
let screenWidth = gridLayouts.screenWidth
let screenHeight = gridLayouts.screenHeight
myLayout = gridLayouts(
xOffsets: [0,screenWidth/4,screenWidth/2,0,screenWidth/4,0,screenWidth/2,0],
yOffsets: [0,0,0,screenHeight/4,screenHeight/4,screenHeight/2,screenHeight/2,screenHeight],
itemWidths: [screenWidth/4,screenWidth/4,screenWidth/2,screenWidth/4,screenWidth/4,screenWidth/2,screenWidth/2,screenWidth],
itemHeights: [screenHeight/4,screenHeight/4,screenHeight/2,screenHeight/4,screenHeight/4,screenHeight/2,screenHeight/2,screenHeight],
blockHeight: 2*screenHeight)
appleLayout = gridLayouts(xOffsets: [0,screenWidth/3,0,screenWidth/3,2*screenWidth/3,0,2*screenWidth/3,2*screenWidth/3], yOffsets: [0,0,screenHeight/3,screenHeight/3,screenHeight/3,2*screenHeight/3,2*screenHeight/3,screenHeight], itemWidths: [screenWidth/3,2*screenWidth/3,screenWidth/3,screenWidth/3,screenWidth/3,2*screenWidth/3,screenWidth/3,screenWidth/3], itemHeights: [screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3,screenHeight/3,2*screenHeight/3,screenHeight/3,screenHeight/3], blockHeight: 4*screenHeight/3)
print("myLayout blockHeight = \(myLayout.calcBlockHeight()/screenHeight)")
print("appleLayout blockHeight = \(appleLayout.calcBlockHeight()/screenHeight)")
}
override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? {
var visibleLayoutAttributes = [UICollectionViewLayoutAttributes]()
// Loop through the cache and look for items in the rect
for attributes in cache {
if attributes.frame.intersects(rect) {
visibleLayoutAttributes.append(attributes)
}
}
return visibleLayoutAttributes
}
override func layoutAttributesForItem(at indexPath: IndexPath) -> UICollectionViewLayoutAttributes? {
return cache[indexPath.item]
}
}
extension Array where Iterator.Element == CGFloat{
mutating func incrementBy(_ amount:CGFloat){
for element in 0...self.count-1{
self[element] += amount
}
}
func incrementedArray(_ amount:CGFloat)->[CGFloat]{
var incrementedArray = self
for element in 0...incrementedArray.count-1{
incrementedArray[element] += amount
}
return incrementedArray
}
}
最佳答案
将 imageView 的内容模式设置为 .scaleAspectFill 并将 clipsToBounds 设置为 true
imageView.contentMode = .scaleAspectFill
imageView.clipsToBounds = true
关于ios - 在自定义 collectionViewLayout 中,所有图像看起来都被拉伸(stretch)了,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49990311/
IO 设备如何知道属于它的内存中的值在memory mapped IO 中发生了变化? ? 例如,假设内存地址 0 专用于保存 VGA 设备的背景颜色。当我们更改 memory[0] 中的值时,VGA
我目前正在开发一个使用Facebook sdk登录(通过FBLoginView)的iOS应用。 一切正常,除了那些拥有较旧版本的facebook的人。 当他们按下“使用Facebook登录”按钮时,他
假设我有: this - is an - example - with some - dashesNSRange将使用`rangeOfString:@“-”拾取“-”的第一个实例,但是如果我只想要最后
Card.io SDK提供以下详细信息: 卡号,有效期,月份,年份,CVV和邮政编码。 如何从此SDK获取国家名称。 - (void)userDidProvideCreditCardInfo:(Car
iOS 应用程序如何从网络服务下载图片并在安装过程中将它们安装到用户的 iOS 设备上?可能吗? 最佳答案 您无法控制应用在用户设备上的安装,因此无法在安装过程中下载其他数据。 只需在安装后首次启动应
我曾经开发过一款企业版 iOS 产品,我们公司曾将其出售给大型企业,供他们的员工使用。 该应用程序通过 AppStore 提供,企业用户获得了公司特定的配置文件(包含应用程序配置文件)以启用他们有权使
我正在尝试将 Card.io SDK 集成到我的 iOS 应用程序中。我想为 CardIO ui 做一个简单的本地化,如更改取消按钮标题或“在此保留信用卡”提示文本。 我在 github 上找到了这个
我正在使用 CardIOView 和 CardIOViewDelegate 类,没有可以设置为 YES 的 BOOL 来扫描 collectCardholderName。我可以看到它在 CardIOP
我有一个集成了通话工具包的 voip 应用程序。每次我从我的 voip 应用程序调用时,都会在 native 电话应用程序中创建一个新的最近通话记录。我在 voip 应用程序中也有自定义联系人(电话应
iOS 应用程序如何知道应用程序打开时屏幕上是否已经有键盘?应用程序运行后,它可以接收键盘显示/隐藏通知。但是,如果应用程序在分屏模式下作为辅助应用程序打开,而主应用程序已经显示键盘,则辅助应用程序不
我在模拟器中收到以下错误: ImageIO: CGImageReadSessionGetCachedImageBlockData *** CGImageReadSessionGetCachedIm
如 Apple 文档所示,可以通过 EAAccessory Framework 与经过认证的配件(由 Apple 认证)进行通信。但是我有点困惑,因为一些帖子告诉我它也可以通过 CoreBluetoo
尽管现在的调试器已经很不错了,但有时找出应用程序中正在发生的事情的最好方法仍然是古老的 NSLog。当您连接到计算机时,这样做很容易; Xcode 会帮助弹出日志查看器面板,然后就可以了。当您不在办公
在我的 iOS 应用程序中,我定义了一些兴趣点。其中一些有一个 Kontakt.io 信标的名称,它绑定(bind)到一个特定的 PoI(我的意思是通常贴在信标标签上的名称)。现在我想在附近发现信标,
我正在为警报提示创建一个 trigger.io 插件。尝试从警报提示返回数据。这是我的代码: // Prompt + (void)show_prompt:(ForgeTask*)task{
您好,我是 Apple iOS 的新手。我阅读并搜索了很多关于推送通知的文章,但我没有发现任何关于 APNS 从 io4 到 ios 6 的新更新的信息。任何人都可以向我提供 APNS 如何在 ios
UITabBar 的高度似乎在 iOS 7 和 8/9/10/11 之间发生了变化。我发布这个问题是为了让其他人轻松找到答案。 那么:在 iPhone 和 iPad 上的 iOS 8/9/10/11
我想我可以针对不同的 iOS 版本使用不同的 Storyboard。 由于 UI 的差异,我将创建下一个 Storyboard: Main_iPhone.storyboard Main_iPad.st
我正在写一些东西,我将使用设备的 iTunes 库中的一部分音轨来覆盖 2 个视频的组合,例如: AVMutableComposition* mixComposition = [[AVMutableC
我创建了一个简单的 iOS 程序,可以顺利编译并在 iPad 模拟器上运行良好。当我告诉 XCode 4 使用我连接的 iPad 设备时,无法编译相同的程序。问题似乎是当我尝试使用附加的 iPad 时
我是一名优秀的程序员,十分优秀!