Merge remote-tracking branch 'origin/master'

This commit is contained in:
liujiahua123123 2019-09-11 20:33:40 +08:00
commit 72e4208df9
6 changed files with 105 additions and 50 deletions

View File

@ -28,7 +28,7 @@ import java.util.concurrent.ExecutionException;
* *
* @author NaturalHG * @author NaturalHG
*/ */
public class MiraiServer { public final class MiraiServer {
private static MiraiServer instance; private static MiraiServer instance;
public static MiraiServer getInstance() { public static MiraiServer getInstance() {

View File

@ -2,6 +2,8 @@ package net.mamoe.mirai.message.defaults
import net.mamoe.mirai.message.Message import net.mamoe.mirai.message.Message
import net.mamoe.mirai.message.MessageId import net.mamoe.mirai.message.MessageId
import net.mamoe.mirai.network.packet.cutTail
import net.mamoe.mirai.network.packet.md5
import java.awt.image.BufferedImage import java.awt.image.BufferedImage
import java.io.* import java.io.*
import java.net.URL import java.net.URL
@ -54,4 +56,12 @@ class Image : Message {
} }
return this.imageID == another.imageID return this.imageID == another.imageID
} }
companion object {
fun getImageID(filename: String): ByteArray = md5(filename).cutTail(1)
}
}
fun main() {
println(0xB0)
} }

View File

@ -245,3 +245,13 @@ fun DataOutputStream.writeVarByteArray(byteArray: ByteArray) {
fun DataOutputStream.writeVarString(str: String) { fun DataOutputStream.writeVarString(str: String) {
this.writeVarByteArray(str.toByteArray()) this.writeVarByteArray(str.toByteArray())
} }
fun DataOutputStream.writeVarShort(short: Int) {
this.writeByte(0x02)
this.writeShort(short)
}
fun DataOutputStream.writeVarInt(int: Int) {
this.writeByte(0x04)
this.writeInt(int)
}

View File

@ -0,0 +1,35 @@
package net.mamoe.mirai.network.packet.image
import net.mamoe.mirai.network.packet.*
import net.mamoe.mirai.utils.writeProtoInt
import java.awt.image.BufferedImage
/**
* 查询群消息的 image id.
* That is, 查询服务器上是否有这个图片, 有就返回 id, 没有就需要上传
*
* @author Him188moe
*/
@PacketId("03 88")
@ExperimentalUnsignedTypes
class ClientGetGroupImageIDPacket(
val bot: Long,
val sessionKey: ByteArray,
val group: Long,
val image: BufferedImage
) : ClientPacket() {
override fun encode() {
this.writeRandom(2)
this.writeQQ(bot)
this.writeHex("04 00 00 00 01 01 01 00 00 68 20 00 00 00 00 00 00 00 00")
this.encryptAndWrite(sessionKey) {
it.writeHex("00 00 00 07 00 00 00 5E 08 01 12 03 98 01 01 10 01 1A")
it.writeHex("5A")
it.writeHex("08")
it.writeProtoInt(group)
it.writeHex("08")
it.writeProtoInt(image.height)
}
}
}

View File

@ -8,9 +8,10 @@ import java.util.concurrent.Callable;
/** /**
* Convert IMAGE into Chars that could shows in terminal * Convert IMAGE into Chars that could shows in terminal
*
* @author NaturalHG * @author NaturalHG
*/ */
public class CharImageConverter implements Callable<String> { public final class CharImageConverter implements Callable<String> {
/** /**
* width should depends on the width of the terminal * width should depends on the width of the terminal
@ -19,74 +20,73 @@ public class CharImageConverter implements Callable<String> {
private int width; private int width;
private double ignoreRate; private double ignoreRate;
public CharImageConverter(BufferedImage image, int width){ public CharImageConverter(BufferedImage image, int width) {
this(image,width,0.95); this(image, width, 0.95);
} }
public CharImageConverter(BufferedImage image, int width, double ignoreRate){ public CharImageConverter(BufferedImage image, int width, double ignoreRate) {
this.image = image; this.image = image;
this.width = width; this.width = width;
this.ignoreRate = ignoreRate; this.ignoreRate = ignoreRate;
} }
@Override @Override
public String call(){ public String call() {
/* /*
* resize Image * resize Image
* */ * */
int newHeight = (int)(this.image.getHeight() * (((double)width) /this.image.getWidth())); int newHeight = (int) (this.image.getHeight() * (((double) width) / this.image.getWidth()));
Image tmp = image.getScaledInstance(width, newHeight, Image.SCALE_SMOOTH); Image tmp = image.getScaledInstance(width, newHeight, Image.SCALE_SMOOTH);
BufferedImage dimg = new BufferedImage(width, newHeight, BufferedImage.TYPE_INT_ARGB); BufferedImage dimg = new BufferedImage(width, newHeight, BufferedImage.TYPE_INT_ARGB);
Graphics2D g2d = dimg.createGraphics(); Graphics2D g2d = dimg.createGraphics();
g2d.drawImage(tmp, 0, 0, null); g2d.drawImage(tmp, 0, 0, null);
this.image = dimg; this.image = dimg;
int background = this.gray(image.getRGB(0,0)); int background = gray(image.getRGB(0, 0));
StringBuilder builder = new StringBuilder(); StringBuilder builder = new StringBuilder();
List<StringBuilder> lines = new ArrayList<>(this.image.getHeight()); List<StringBuilder> lines = new ArrayList<>(this.image.getHeight());
int minXPos = this.width; int minXPos = this.width;
int maxXPos = 0; int maxXPos = 0;
for (int y = 0; y < image.getHeight(); y++) { for (int y = 0; y < image.getHeight(); y++) {
StringBuilder builderLine = new StringBuilder(); StringBuilder builderLine = new StringBuilder();
for (int x = 0; x < image.getWidth(); x++) { for (int x = 0; x < image.getWidth(); x++) {
int gray = this.gray(image.getRGB(x, y)); int gray = gray(image.getRGB(x, y));
if (grayCompare(gray, background)) { if (grayCompare(gray, background)) {
builderLine.append(" "); builderLine.append(" ");
} else { } else {
builderLine.append("#"); builderLine.append("#");
if (x < minXPos) { if (x < minXPos) {
minXPos = x; minXPos = x;
} }
if (x > maxXPos) { if (x > maxXPos) {
maxXPos = x; maxXPos = x;
} }
} }
} }
if (builderLine.toString().isBlank()) { if (builderLine.toString().isBlank()) {
continue; continue;
} }
lines.add(builderLine); lines.add(builderLine);
}
for (int i=0;i<lines.size();++i) {
builder.append(lines.get(i).substring(minXPos, maxXPos)).append("\n");
} }
return builder.toString(); for (StringBuilder line : lines) {
builder.append(line.substring(minXPos, maxXPos)).append("\n");
}
return builder.toString();
} }
private int gray(int rgb) { private static int gray(int rgb) {
int R = (rgb & 0xff0000) >> 16; int R = (rgb & 0xff0000) >> 16;
int G = (rgb & 0x00ff00) >> 8; int G = (rgb & 0x00ff00) >> 8;
int B = rgb & 0x0000ff; int B = rgb & 0x0000ff;
int gray = (R * 30 + G * 59 + B * 11 + 50) / 100;
return (R * 30 + G * 59 + B * 11 + 50) / 100; return (R * 30 + G * 59 + B * 11 + 50) / 100;
} }
public boolean grayCompare(int g1, int g2){ public boolean grayCompare(int g1, int g2) {
return ((double)Math.min(g1,g2)/Math.max(g1,g2)) >= ignoreRate; return ((double) Math.min(g1, g2) / Math.max(g1, g2)) >= ignoreRate;
} }
} }

View File

@ -15,13 +15,13 @@ import java.io.DataOutputStream
* TODO improve * TODO improve
*/ */
@ExperimentalUnsignedTypes @ExperimentalUnsignedTypes
fun DataOutputStream.writeProtoFixedInt(int: Int) { fun DataOutputStream.writeProtoFixedInt(int: Long) {
if (int == 128) { if (int == 0xFFL) {
this.writeShort(0x80_01)//unsigned//1000000010000001 this.writeShort(0x80_01)//unsigned//1000000010000001
return return
} }
this.writeByte(int.rem(128) + 128) this.writeByte((int.rem(0xFF) + 0xFF).toInt())
this.writeByte(int / 128) this.writeByte((int / 0xFF).toInt())
} }
/** /**
@ -30,9 +30,9 @@ fun DataOutputStream.writeProtoFixedInt(int: Int) {
* TODO improve * TODO improve
*/ */
@ExperimentalUnsignedTypes @ExperimentalUnsignedTypes
fun DataOutputStream.writeProtoInt(int: Int) { fun DataOutputStream.writeProtoInt(int: Long) {
if (int < 128) { if (int < 0xFF) {
this.writeByte(int and 0xFF)//10000000 this.writeByte((int and 0xFF).toInt())//10000000
return return
} }
this.writeProtoFixedInt(int) this.writeProtoFixedInt(int)