Поднимать загрузку изображений, изменять их размер, сохранять в базе данных, отображать

StackOverflow https://stackoverflow.com/questions/1404814

Вопрос

Есть ли краткий пример того, как загрузить изображение, изменить его размер, сохранить в базе данных, а затем отправить изображение с помощью Lift?

Я уверен, что смог бы собрать это воедино с помощью API загрузки файлов, Java 2D API, Lift Mapper и Response.Но есть ли какой-нибудь пример кода, которому я могу следовать, чтобы сделать это "правильным" или рекомендуемым способом?

Это было полезно?

Решение

Я сделал это для поля Mapper, связанного с s3, создав новый MappedField. У меня также есть код для изменения размера, но я не тестировал и не развертывал (поэтому используйте его с осторожностью).

class MappedS3Image[T<:Mapper[T]](owner: T, val path:String, maxWidth: String, maxHeight:String) extends MappedString[T](owner, 36) {

  def url:String = MappedS3Image.fullImgPath(path, is)

  def setFromUpload(fileHolder: Box[FileParamHolder]) = {
      S3Sender.uploadImageToS3(path, fileHolder).map(this.set(_))
  }

  override def asHtml:Node = <img src={url} style={"max-width:" + maxWidth + ";max-height:"+maxHeight} />
  override def _toForm: Box[Elem] = Full(SHtml.fileUpload(fu=>setFromUpload(Full(fu))))

}


import java.awt.Image 
import java.awt.image.BufferedImage
import javax.imageio.ImageIO
import java.awt.Graphics2D
import java.awt.AlphaComposite

object ImageResizer {

    def resize(is:java.io.InputStream, maxWidth:Int, maxHeight:Int):BufferedImage = {
        val originalImage:BufferedImage = ImageIO.read(is)

        val height = originalImage.getHeight
        val width = originalImage.getWidth

        if (width <= maxWidth && height <= maxHeight)
            originalImage
        else {
            var scaledWidth:Int = width
            var scaledHeight:Int = height
            val ratio:Double = width/height
            if (scaledWidth > maxWidth){
                scaledWidth = maxWidth
                scaledHeight = (scaledWidth.doubleValue/ratio).intValue
            }
            if (scaledHeight > maxHeight){
                scaledHeight = maxHeight
                scaledWidth = (scaledHeight.doubleValue*ratio).intValue
            }
            val scaledBI = new BufferedImage(scaledWidth, scaledHeight,  BufferedImage.TYPE_INT_RGB)
            val g = scaledBI.createGraphics
            g.setComposite(AlphaComposite.Src)
            g.drawImage(originalImage, 0, 0, scaledWidth, scaledHeight, null);
            g.dispose
            scaledBI
        }
    }
}

Другие советы

Другой ответ прекрасно описывает, как изменить размер изображения и сохранить ссылку на файл в файловой системе.

Если вы хотите использовать lift mapper для хранения фактического содержимого файла, вам необходимо создать свой пользовательский объект модели и определить для него двоичное поле.Попробуйте что-то вроде этого:

package code {
package model {


import _root_.net.liftweb.mapper._
import _root_.net.liftweb.util._
import _root_.net.liftweb.common._


// singleton object which manipulates storing of Document instances
object Document extends Document with KeyedMetaMapper[Long, Document] {
}



class Document extends KeyedMapper[Long, Document] {
  def getSingleton = Document
  def primaryKeyField = id

  object id extends MappedLongIndex(this)

  object name extends MappedString(this, 20) {
    override def displayName = "Name"
    override def writePermission_? = true
  }

  object content extends MappedBinary(this) {
    override def displayName = "Content"
    override def writePermission_? = true
  }
}



}
}

Затем, в классе bootstrap, добавьте это Document в конце:

Schemifier.schemify(true, Schemifier.infoF _, User, Document)

Вуаля.Используя Document save (new Document) сохраняет его в базе данных.A new Documentполя могут быть заданы с помощью set способ.Попробуйте поиграть с delete_!, find, findAll методы проведения Document синглтон, чтобы удалить или найти его в базе данных.С этого момента все должно быть просто.

Наконец, чтобы отобразить изображение, вы можете переопределить правила диспетчеризации Lift (в классе bootstrap, Boot.scala).Попробуйте поиграть с этим примером, который переопределяет правила для запросов PDF:

def getFile(filename: String): Option[Document] = {
  val alldocs = Document.findAll()
  alldocs.find(_.name.get == filename)
}

LiftRules.statelessDispatchTable.append {
  case Req("file" :: name :: Nil, "pdf", GetRequest) =>
    () =>
    println("Got request for: " + name + ".pdf")
    for {
      stream <- tryo(
        getFile(name + ".pdf") map { 
          doc => new java.io.ByteArrayInputStream(doc.content.get)
        } getOrElse null
      )
      if null ne stream
    } yield StreamingResponse(stream,
                              () => stream.close,
                              stream.available,
                              List("Content-Type" -> "application/pdf"),
                              Nil,
                              200)
}

Основываясь на принятом ответе Джона Хоффмана, я исправил ошибки. Его версия портит соотношение сторон (оно всегда становится 1: 1), потому что математика была отключена в нескольких местах. Эта версия изменяет размеры больших изображений до тех пор, пока они не поместятся, и учитывает соотношение сторон.

def resize(is:java.io.InputStream, maxWidth:Int, maxHeight:Int):BufferedImage = {
    require (maxWidth > 0)
    require (maxHeight > 0)
    val originalImage:BufferedImage = ImageIO.read(is)

    var height = originalImage.getHeight
    var width = originalImage.getWidth

    // Shortcut to save a pointless reprocessing in case the image is small enough already
    if (width <= maxWidth && height <= maxHeight)
        originalImage
    else {          
        // If the picture was too big, it will either fit by width or height.
        // This essentially resizes the dimensions twice, until it fits
        if (width > maxWidth){
          height = (height.doubleValue() * (maxWidth.doubleValue() / width.doubleValue())).intValue
          width = maxWidth
        }
        if (height > maxHeight){
          width = (width.doubleValue() * (maxHeight.doubleValue() / height.doubleValue())).intValue
          height = maxHeight
        }
        val scaledBI = new BufferedImage(width, height,  BufferedImage.TYPE_INT_RGB)
        val g = scaledBI.createGraphics
        g.setComposite(AlphaComposite.Src)
        g.drawImage(originalImage, 0, 0, width, height, null);
        g.dispose
        scaledBI
    }
}
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top