Compare commits
28 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 1b7f2c40e6 | |||
| 93d52ae819 | |||
| 48b3d5c6b1 | |||
| e9a9d8a01c | |||
| a155a57f33 | |||
| 90b83a0690 | |||
| f10301c3e4 | |||
| 8571a936a4 | |||
| 3f6144836c | |||
| 53c432a635 | |||
| 340cadf3b9 | |||
| 8d6868aef6 | |||
| 6e8fcc8cc3 | |||
| 57670ffc76 | |||
| 2144eedd76 | |||
| 43daef83de | |||
| 4a9ad426e7 | |||
| 13beda3a8d | |||
| 18c05af4db | |||
| df6e1e1cbd | |||
| 01b1a14511 | |||
| b6af8d559c | |||
| 22dbfc2e24 | |||
| 2f3b01732c | |||
| 88803382dd | |||
| 74c51163c7 | |||
| 877ff4ba18 | |||
| ad2feb5a27 |
@@ -0,0 +1,73 @@
|
|||||||
|
name: Bug report
|
||||||
|
description: Something in HellionChat is broken or behaves wrong
|
||||||
|
labels:
|
||||||
|
- bug
|
||||||
|
body:
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
Thanks for reporting. Please fill in the fields below so I can
|
||||||
|
reproduce the issue. If this is a security issue, stop here and
|
||||||
|
use the [private vulnerability advisory](https://github.com/JonKazama-Hellion/HellionChat/security/advisories/new)
|
||||||
|
instead.
|
||||||
|
|
||||||
|
- type: input
|
||||||
|
id: version
|
||||||
|
attributes:
|
||||||
|
label: HellionChat version
|
||||||
|
description: From Settings → Information → Version
|
||||||
|
placeholder: "0.5.1"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: dropdown
|
||||||
|
id: platform
|
||||||
|
attributes:
|
||||||
|
label: Platform
|
||||||
|
options:
|
||||||
|
- Windows (XIVLauncher)
|
||||||
|
- Linux (XIVLauncher Core)
|
||||||
|
- macOS (XIVLauncher Core / wine)
|
||||||
|
- Other
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: what-happened
|
||||||
|
attributes:
|
||||||
|
label: What happened
|
||||||
|
description: Plain description, no log dumps yet
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: expected
|
||||||
|
attributes:
|
||||||
|
label: What you expected
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: steps
|
||||||
|
attributes:
|
||||||
|
label: How to reproduce
|
||||||
|
description: Step-by-step from "open settings" or "log in" through to the broken behaviour
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: log
|
||||||
|
attributes:
|
||||||
|
label: Relevant /xllog excerpt
|
||||||
|
description: Filter for "HellionChat" or "ChatTwo" if the log is huge
|
||||||
|
render: text
|
||||||
|
|
||||||
|
- type: checkboxes
|
||||||
|
id: confirm
|
||||||
|
attributes:
|
||||||
|
label: Pre-flight
|
||||||
|
options:
|
||||||
|
- label: I am running the latest version of HellionChat
|
||||||
|
required: true
|
||||||
|
- label: I have searched existing issues for duplicates
|
||||||
|
required: true
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
blank_issues_enabled: false
|
||||||
|
|
||||||
|
contact_links:
|
||||||
|
- name: Security vulnerability
|
||||||
|
url: https://github.com/JonKazama-Hellion/HellionChat/security/advisories/new
|
||||||
|
about: Do not open a public issue for security problems. Use the private advisory instead.
|
||||||
|
|
||||||
|
- name: Upstream Chat 2 issue
|
||||||
|
url: https://github.com/Infiziert90/ChatTwo/issues
|
||||||
|
about: If the issue exists in upstream Chat 2 too, please report it there so the original maintainers see it as well.
|
||||||
|
|
||||||
|
- name: Discord
|
||||||
|
url: https://discord.com/users/j.j_kazama
|
||||||
|
about: Quick questions, casual feedback. Bug reports still go through the issue tracker for tracking.
|
||||||
@@ -0,0 +1,55 @@
|
|||||||
|
name: Feature request
|
||||||
|
description: Suggest a feature or enhancement for HellionChat
|
||||||
|
labels:
|
||||||
|
- enhancement
|
||||||
|
body:
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
Thanks for the suggestion. HellionChat focuses on privacy by
|
||||||
|
default and a small, well-scoped feature set. Suggestions that
|
||||||
|
align with that scope are easier to accept than ones that pull
|
||||||
|
the plugin toward "do everything".
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: problem
|
||||||
|
attributes:
|
||||||
|
label: What problem are you trying to solve
|
||||||
|
description: The user-side problem, not the proposed solution yet
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: solution
|
||||||
|
attributes:
|
||||||
|
label: What you would like HellionChat to do
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: alternatives
|
||||||
|
attributes:
|
||||||
|
label: Alternatives you have considered
|
||||||
|
description: Other plugins, manual workarounds, settings combinations
|
||||||
|
|
||||||
|
- type: dropdown
|
||||||
|
id: scope
|
||||||
|
attributes:
|
||||||
|
label: Scope estimate from your side
|
||||||
|
options:
|
||||||
|
- "Small (one tab, one toggle, one filter)"
|
||||||
|
- "Medium (a settings section, persistent state, one new file)"
|
||||||
|
- "Large (architectural, touches the message pipeline or the database)"
|
||||||
|
- "I don't know"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: checkboxes
|
||||||
|
id: confirm
|
||||||
|
attributes:
|
||||||
|
label: Pre-flight
|
||||||
|
options:
|
||||||
|
- label: I have searched existing issues for similar requests
|
||||||
|
required: true
|
||||||
|
- label: I understand HellionChat is a privacy-focused fork and not a feature parity tool with upstream Chat 2
|
||||||
|
required: true
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
version: 2
|
||||||
|
|
||||||
|
updates:
|
||||||
|
# NuGet package updates for the plugin project. Weekly cadence keeps the
|
||||||
|
# noise down while still catching transitive security advisories within
|
||||||
|
# a few days of disclosure.
|
||||||
|
- package-ecosystem: nuget
|
||||||
|
directory: /ChatTwo
|
||||||
|
schedule:
|
||||||
|
interval: weekly
|
||||||
|
day: monday
|
||||||
|
time: "07:00"
|
||||||
|
timezone: Europe/Berlin
|
||||||
|
open-pull-requests-limit: 5
|
||||||
|
labels:
|
||||||
|
- dependencies
|
||||||
|
- nuget
|
||||||
|
commit-message:
|
||||||
|
prefix: "chore(deps)"
|
||||||
|
groups:
|
||||||
|
patches:
|
||||||
|
update-types:
|
||||||
|
- patch
|
||||||
|
minor:
|
||||||
|
update-types:
|
||||||
|
- minor
|
||||||
|
|
||||||
|
# GitHub Actions versions in .github/workflows. Lower cadence because
|
||||||
|
# Action releases ship less frequently and are usually safe to defer
|
||||||
|
# for a month.
|
||||||
|
- package-ecosystem: github-actions
|
||||||
|
directory: /
|
||||||
|
schedule:
|
||||||
|
interval: monthly
|
||||||
|
time: "07:00"
|
||||||
|
timezone: Europe/Berlin
|
||||||
|
open-pull-requests-limit: 3
|
||||||
|
labels:
|
||||||
|
- dependencies
|
||||||
|
- github-actions
|
||||||
|
commit-message:
|
||||||
|
prefix: "chore(actions)"
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
name: Build
|
||||||
|
|
||||||
|
# Verifies that every push to main and every PR still builds against the
|
||||||
|
# current Dalamud staging branch. Does not produce release artefacts; the
|
||||||
|
# release workflow handles that on tag.
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [main]
|
||||||
|
pull_request:
|
||||||
|
branches: [main]
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
# Minimum permissions for a build-only workflow: read the repo, nothing
|
||||||
|
# else. Closes the CodeQL "Workflow does not contain permissions" alert
|
||||||
|
# and matches the principle-of-least-privilege the security guide
|
||||||
|
# recommends for workflows that don't push or create releases.
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
name: Build (Release)
|
||||||
|
runs-on: windows-latest
|
||||||
|
timeout-minutes: 15
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
|
- name: Setup .NET 10
|
||||||
|
uses: actions/setup-dotnet@v4
|
||||||
|
with:
|
||||||
|
dotnet-version: 10.0.x
|
||||||
|
|
||||||
|
- name: Download Dalamud staging
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
$hooks = Join-Path $env:APPDATA "XIVLauncher\addon\Hooks\dev"
|
||||||
|
New-Item -ItemType Directory -Force -Path $hooks | Out-Null
|
||||||
|
Invoke-WebRequest -Uri https://goatcorp.github.io/dalamud-distrib/stg/latest.zip -OutFile dalamud.zip
|
||||||
|
Expand-Archive -Force -Path dalamud.zip -DestinationPath $hooks
|
||||||
|
|
||||||
|
- name: Restore
|
||||||
|
run: dotnet restore ChatTwo/ChatTwo.csproj
|
||||||
|
|
||||||
|
- name: Build (Release)
|
||||||
|
run: dotnet build ChatTwo/ChatTwo.csproj --configuration Release --no-restore
|
||||||
|
|
||||||
|
- name: Upload build output
|
||||||
|
uses: actions/upload-artifact@v7
|
||||||
|
with:
|
||||||
|
name: HellionChat-build-${{ github.run_number }}
|
||||||
|
path: ChatTwo/bin/Release/**/HellionChat/**
|
||||||
|
if-no-files-found: warn
|
||||||
|
retention-days: 14
|
||||||
@@ -0,0 +1,93 @@
|
|||||||
|
name: CodeQL
|
||||||
|
|
||||||
|
# Replaces the GitHub default-setup CodeQL scan. The default setup runs
|
||||||
|
# without resolving the Dalamud assemblies (they live in a user-AppData
|
||||||
|
# path) and reports "Low C# analysis quality" because call-target
|
||||||
|
# resolution sits at ~64%. This workflow downloads the Dalamud staging
|
||||||
|
# distribution before the build, runs a manual dotnet build, and then
|
||||||
|
# lets CodeQL analyse the fully-resolved compilation. Quality climbs
|
||||||
|
# back above the 85% thresholds.
|
||||||
|
#
|
||||||
|
# This workflow only consumes trusted inputs: the tag/branch ref via
|
||||||
|
# the standard checkout action, and the Dalamud distribution URL which
|
||||||
|
# is pinned to a goatcorp-controlled GitHub Pages target. No user-
|
||||||
|
# controlled event payload (issue title, PR body, commit message) flows
|
||||||
|
# into a run-step.
|
||||||
|
#
|
||||||
|
# Disable the default setup in the repo before this workflow lands:
|
||||||
|
# Settings -> Code security -> Code scanning -> "CodeQL analysis" tile
|
||||||
|
# -> Switch to advanced.
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [main]
|
||||||
|
pull_request:
|
||||||
|
branches: [main]
|
||||||
|
schedule:
|
||||||
|
- cron: '17 6 * * 1'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
actions: read
|
||||||
|
contents: read
|
||||||
|
security-events: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
analyze-csharp:
|
||||||
|
name: Analyze (csharp)
|
||||||
|
runs-on: windows-latest
|
||||||
|
timeout-minutes: 30
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
|
- name: Setup .NET 10
|
||||||
|
uses: actions/setup-dotnet@v4
|
||||||
|
with:
|
||||||
|
dotnet-version: 10.0.x
|
||||||
|
|
||||||
|
- name: Download Dalamud staging
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
$hooks = Join-Path $env:APPDATA "XIVLauncher\addon\Hooks\dev"
|
||||||
|
New-Item -ItemType Directory -Force -Path $hooks | Out-Null
|
||||||
|
Invoke-WebRequest -Uri https://goatcorp.github.io/dalamud-distrib/stg/latest.zip -OutFile dalamud.zip
|
||||||
|
Expand-Archive -Force -Path dalamud.zip -DestinationPath $hooks
|
||||||
|
|
||||||
|
- name: Initialize CodeQL
|
||||||
|
uses: github/codeql-action/init@v3
|
||||||
|
with:
|
||||||
|
languages: csharp
|
||||||
|
build-mode: manual
|
||||||
|
queries: security-extended
|
||||||
|
|
||||||
|
- name: Restore
|
||||||
|
run: dotnet restore ChatTwo/ChatTwo.csproj
|
||||||
|
|
||||||
|
- name: Build (Release)
|
||||||
|
run: dotnet build ChatTwo/ChatTwo.csproj --configuration Release --no-restore
|
||||||
|
|
||||||
|
- name: Perform CodeQL analysis
|
||||||
|
uses: github/codeql-action/analyze@v3
|
||||||
|
with:
|
||||||
|
category: /language:csharp
|
||||||
|
|
||||||
|
analyze-actions:
|
||||||
|
name: Analyze (actions)
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 10
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
|
- name: Initialize CodeQL
|
||||||
|
uses: github/codeql-action/init@v3
|
||||||
|
with:
|
||||||
|
languages: actions
|
||||||
|
build-mode: none
|
||||||
|
|
||||||
|
- name: Perform CodeQL analysis
|
||||||
|
uses: github/codeql-action/analyze@v3
|
||||||
|
with:
|
||||||
|
category: /language:actions
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
name: Release
|
||||||
|
|
||||||
|
# Triggered when a vX.Y.Z tag is pushed. Builds the plugin against the
|
||||||
|
# current Dalamud staging branch, locates the latest.zip produced by
|
||||||
|
# DalamudPackager and attaches it to the matching GitHub Release.
|
||||||
|
# Does not consume any user-controlled event payload, only the tag name
|
||||||
|
# (validated by the on.tags filter) and the steps output of the locate
|
||||||
|
# step (path string from Get-ChildItem on a controlled directory).
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- 'v*'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
release:
|
||||||
|
name: Build and attach release ZIP
|
||||||
|
runs-on: windows-latest
|
||||||
|
timeout-minutes: 20
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
|
- name: Setup .NET 10
|
||||||
|
uses: actions/setup-dotnet@v4
|
||||||
|
with:
|
||||||
|
dotnet-version: 10.0.x
|
||||||
|
|
||||||
|
- name: Download Dalamud staging
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
$hooks = Join-Path $env:APPDATA "XIVLauncher\addon\Hooks\dev"
|
||||||
|
New-Item -ItemType Directory -Force -Path $hooks | Out-Null
|
||||||
|
Invoke-WebRequest -Uri https://goatcorp.github.io/dalamud-distrib/stg/latest.zip -OutFile dalamud.zip
|
||||||
|
Expand-Archive -Force -Path dalamud.zip -DestinationPath $hooks
|
||||||
|
|
||||||
|
- name: Build (Release)
|
||||||
|
run: dotnet build ChatTwo/ChatTwo.csproj --configuration Release
|
||||||
|
|
||||||
|
- name: Locate latest.zip
|
||||||
|
id: locate
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
$zip = Get-ChildItem -Path ChatTwo\bin\Release -Recurse -Filter latest.zip | Select-Object -First 1
|
||||||
|
if (-not $zip)
|
||||||
|
{
|
||||||
|
throw "latest.zip not found under ChatTwo\bin\Release"
|
||||||
|
}
|
||||||
|
Write-Host "Found: $($zip.FullName)"
|
||||||
|
"path=$($zip.FullName)" | Out-File -FilePath $env:GITHUB_OUTPUT -Append
|
||||||
|
|
||||||
|
- name: Attach to GitHub release
|
||||||
|
uses: softprops/action-gh-release@v3
|
||||||
|
with:
|
||||||
|
files: ${{ steps.locate.outputs.path }}
|
||||||
|
fail_on_unmatched_files: true
|
||||||
|
generate_release_notes: false
|
||||||
@@ -374,6 +374,9 @@ FodyWeavers.xsd
|
|||||||
|
|
||||||
#Specs und Plan datein
|
#Specs und Plan datein
|
||||||
/.superpowers/
|
/.superpowers/
|
||||||
|
|
||||||
|
#Test Datein
|
||||||
|
ChatTwo.Tests
|
||||||
TestResults
|
TestResults
|
||||||
*.db-shm
|
*.db-shm
|
||||||
*.db-wal
|
*.db-wal
|
||||||
|
|||||||
@@ -0,0 +1,27 @@
|
|||||||
|
HellionChat — a privacy-focused fork of ChatTwo for FINAL FANTASY XIV
|
||||||
|
|
||||||
|
Copyright (c) 2024-2025 Infiziert90 (Infi) and Anna Clemens (ascclemens)
|
||||||
|
Original ChatTwo authors and copyright holders of the upstream
|
||||||
|
plugin this fork is built on. Their work covers the message store,
|
||||||
|
the channel filtering, the sidebar tab system, the FFXIV chat
|
||||||
|
hooks, the localisation infrastructure and most of the
|
||||||
|
architecture HellionChat still relies on.
|
||||||
|
|
||||||
|
Copyright (c) 2025-2026 Florian Wathling / Hellion Online Media
|
||||||
|
HellionChat-specific modifications, including the privacy filter,
|
||||||
|
per-channel retention sweep, export pipeline, Auto-Tell-Tabs,
|
||||||
|
Hellion theme and font integration, German localisation and the
|
||||||
|
EUPL-1.2 fork maintenance.
|
||||||
|
|
||||||
|
Licensed under the European Union Public Licence (EUPL), Version 1.2
|
||||||
|
only. The full Licence text lives in the LICENSE file at the root of
|
||||||
|
this repository. The official Licence website is at:
|
||||||
|
|
||||||
|
https://eupl.eu/1.2/en/
|
||||||
|
|
||||||
|
This Work is provided "AS IS" without warranties of any kind. See
|
||||||
|
Article 7 (Disclaimer of Warranty) and Article 8 (Disclaimer of
|
||||||
|
Liability) of the Licence for the legally binding wording.
|
||||||
|
|
||||||
|
Acknowledgements directed at the upstream ChatTwo authors live in
|
||||||
|
NOTICE.md. The manual upstream-sync workflow lives in UPSTREAM_SYNC.md.
|
||||||
@@ -1,167 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.IO;
|
|
||||||
using System.Linq;
|
|
||||||
using ChatTwo.Code;
|
|
||||||
using ChatTwo.Util;
|
|
||||||
using Dalamud.Game.Text;
|
|
||||||
using Dalamud.Game.Text.SeStringHandling;
|
|
||||||
using Dalamud.Game.Text.SeStringHandling.Payloads;
|
|
||||||
using JetBrains.Annotations;
|
|
||||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
|
||||||
|
|
||||||
namespace ChatTwo.Tests;
|
|
||||||
|
|
||||||
// Hellion Chat — Auto-Tell-Tabs history-preload coverage.
|
|
||||||
//
|
|
||||||
// These tests exercise MessageStore.GetTellHistoryWithSender, the query the
|
|
||||||
// AutoTellTabsService uses to populate a freshly spawned temp tab with the
|
|
||||||
// last conversations with that player.
|
|
||||||
//
|
|
||||||
// NOTE: like the rest of ChatTwo.Tests today, these will fail at runtime
|
|
||||||
// until the project's Dalamud.dll runtime dependency is sorted out (see
|
|
||||||
// Phase-2 backlog item "Test-Projekt fixen"). Compile-time the suite builds
|
|
||||||
// fine via DALAMUD_HOME, so the tests guard against API drift even before
|
|
||||||
// they can be executed locally.
|
|
||||||
[TestClass]
|
|
||||||
[TestSubject(typeof(MessageStore))]
|
|
||||||
public class AutoTellTabsHistoryTest
|
|
||||||
{
|
|
||||||
public TestContext TestContext { get; set; }
|
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(5000)]
|
|
||||||
public void GetTellHistoryWithSender_FiltersByNameAndWorld()
|
|
||||||
{
|
|
||||||
var tempDir = Directory.CreateTempSubdirectory("ChatTwo_test_");
|
|
||||||
var dbPath = Path.Join(tempDir.FullName, "test.db");
|
|
||||||
TestContext.WriteLine("Using database path: " + dbPath);
|
|
||||||
using var store = new MessageStore(dbPath);
|
|
||||||
|
|
||||||
const ulong receiver = 99001;
|
|
||||||
var now = DateTimeOffset.UtcNow;
|
|
||||||
|
|
||||||
// Two tells with the target sender, one with a different sender on
|
|
||||||
// the same world, one with the same name on a different world. Only
|
|
||||||
// the first two should make it into the result.
|
|
||||||
var asukaLichIn = TellMessage("Asuka", 76, receiver, now.AddMinutes(-30), ChatType.TellIncoming);
|
|
||||||
var asukaLichOut = TellMessage("Asuka", 76, receiver, now.AddMinutes(-20), ChatType.TellOutgoing);
|
|
||||||
var broboLich = TellMessage("Brobo", 76, receiver, now.AddMinutes(-10), ChatType.TellIncoming);
|
|
||||||
var asukaOmega = TellMessage("Asuka", 90, receiver, now.AddMinutes(-5), ChatType.TellIncoming);
|
|
||||||
|
|
||||||
store.UpsertMessage(asukaLichIn);
|
|
||||||
store.UpsertMessage(asukaLichOut);
|
|
||||||
store.UpsertMessage(broboLich);
|
|
||||||
store.UpsertMessage(asukaOmega);
|
|
||||||
|
|
||||||
var result = store.GetTellHistoryWithSender(receiver, "Asuka", 76, limit: 50);
|
|
||||||
|
|
||||||
Assert.AreEqual(2, result.Count);
|
|
||||||
// Result is oldest-first so a tab can append messages chronologically.
|
|
||||||
Assert.AreEqual(asukaLichIn.Id, result[0].Id);
|
|
||||||
Assert.AreEqual(asukaLichOut.Id, result[1].Id);
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(5000)]
|
|
||||||
public void GetTellHistoryWithSender_RespectsLimit()
|
|
||||||
{
|
|
||||||
var tempDir = Directory.CreateTempSubdirectory("ChatTwo_test_");
|
|
||||||
var dbPath = Path.Join(tempDir.FullName, "test.db");
|
|
||||||
TestContext.WriteLine("Using database path: " + dbPath);
|
|
||||||
using var store = new MessageStore(dbPath);
|
|
||||||
|
|
||||||
const ulong receiver = 99002;
|
|
||||||
var now = DateTimeOffset.UtcNow;
|
|
||||||
|
|
||||||
for (var i = 0; i < 30; i++)
|
|
||||||
{
|
|
||||||
var msg = TellMessage("Asuka", 76, receiver, now.AddMinutes(-i - 1), ChatType.TellIncoming);
|
|
||||||
store.UpsertMessage(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
var result = store.GetTellHistoryWithSender(receiver, "Asuka", 76, limit: 5);
|
|
||||||
|
|
||||||
Assert.AreEqual(5, result.Count);
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(5000)]
|
|
||||||
public void GetTellHistoryWithSender_ZeroLimitReturnsEmpty()
|
|
||||||
{
|
|
||||||
var tempDir = Directory.CreateTempSubdirectory("ChatTwo_test_");
|
|
||||||
var dbPath = Path.Join(tempDir.FullName, "test.db");
|
|
||||||
TestContext.WriteLine("Using database path: " + dbPath);
|
|
||||||
using var store = new MessageStore(dbPath);
|
|
||||||
|
|
||||||
const ulong receiver = 99003;
|
|
||||||
|
|
||||||
var msg = TellMessage("Asuka", 76, receiver, DateTimeOffset.UtcNow, ChatType.TellIncoming);
|
|
||||||
store.UpsertMessage(msg);
|
|
||||||
|
|
||||||
var result = store.GetTellHistoryWithSender(receiver, "Asuka", 76, limit: 0);
|
|
||||||
|
|
||||||
Assert.AreEqual(0, result.Count);
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(5000)]
|
|
||||||
public void GetTellHistoryWithSender_IgnoresOtherReceivers()
|
|
||||||
{
|
|
||||||
var tempDir = Directory.CreateTempSubdirectory("ChatTwo_test_");
|
|
||||||
var dbPath = Path.Join(tempDir.FullName, "test.db");
|
|
||||||
TestContext.WriteLine("Using database path: " + dbPath);
|
|
||||||
using var store = new MessageStore(dbPath);
|
|
||||||
|
|
||||||
const ulong ourReceiver = 99004;
|
|
||||||
const ulong otherReceiver = 99005;
|
|
||||||
var now = DateTimeOffset.UtcNow;
|
|
||||||
|
|
||||||
// Tell on the local player's account.
|
|
||||||
var ours = TellMessage("Asuka", 76, ourReceiver, now.AddMinutes(-1), ChatType.TellIncoming);
|
|
||||||
// Same sender, but logged against a different local character —
|
|
||||||
// common when the user has alts. Must not surface.
|
|
||||||
var foreign = TellMessage("Asuka", 76, otherReceiver, now, ChatType.TellIncoming);
|
|
||||||
|
|
||||||
store.UpsertMessage(ours);
|
|
||||||
store.UpsertMessage(foreign);
|
|
||||||
|
|
||||||
var result = store.GetTellHistoryWithSender(ourReceiver, "Asuka", 76, limit: 50);
|
|
||||||
|
|
||||||
Assert.AreEqual(1, result.Count);
|
|
||||||
Assert.AreEqual(ours.Id, result[0].Id);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Message TellMessage(
|
|
||||||
string senderName,
|
|
||||||
uint senderWorld,
|
|
||||||
ulong receiver,
|
|
||||||
DateTimeOffset dateTime,
|
|
||||||
ChatType chatType)
|
|
||||||
{
|
|
||||||
var senderSeString = new SeStringBuilder()
|
|
||||||
.Add(new PlayerPayload(senderName, senderWorld))
|
|
||||||
.AddText(senderName)
|
|
||||||
.Add(RawPayload.LinkTerminator)
|
|
||||||
.Build();
|
|
||||||
|
|
||||||
var contentSeString = new SeStringBuilder()
|
|
||||||
.AddText("test message")
|
|
||||||
.Build();
|
|
||||||
|
|
||||||
var senderChunks = ChunkUtil.ToChunks(senderSeString, ChunkSource.Sender, chatType).ToList();
|
|
||||||
var contentChunks = ChunkUtil.ToChunks(contentSeString, ChunkSource.Content, chatType).ToList();
|
|
||||||
|
|
||||||
var chatCode = new ChatCode((XivChatType)chatType, XivChatRelationKind.LocalPlayer, XivChatRelationKind.LocalPlayer);
|
|
||||||
return new Message(
|
|
||||||
Guid.NewGuid(),
|
|
||||||
receiver,
|
|
||||||
0,
|
|
||||||
dateTime,
|
|
||||||
chatCode,
|
|
||||||
senderChunks,
|
|
||||||
contentChunks,
|
|
||||||
senderSeString,
|
|
||||||
contentSeString,
|
|
||||||
Guid.Empty);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,53 +0,0 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
|
||||||
|
|
||||||
<PropertyGroup>
|
|
||||||
<TargetFrameworks>net10.0-windows</TargetFrameworks>
|
|
||||||
|
|
||||||
<IsPackable>false</IsPackable>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="JetBrains.Annotations" Version="2025.2.2" />
|
|
||||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.14.1" />
|
|
||||||
<PackageReference Include="morelinq" Version="4.4.0" />
|
|
||||||
<PackageReference Include="MSTest.TestAdapter" Version="3.6.3" />
|
|
||||||
<PackageReference Include="MSTest.TestFramework" Version="3.6.3" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<ProjectReference Include="..\ChatTwo\ChatTwo.csproj" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<PropertyGroup>
|
|
||||||
<DalamudLibPath>$(AppData)\XIVLauncher\addon\Hooks\dev</DalamudLibPath>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<PropertyGroup Condition="'$([System.Runtime.InteropServices.RuntimeInformation]::IsOSPlatform($([System.Runtime.InteropServices.OSPlatform]::Linux)))'">
|
|
||||||
<DalamudLibPath>$(DALAMUD_HOME)</DalamudLibPath>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<PropertyGroup Condition="'$(IsCI)' == 'true'">
|
|
||||||
<DalamudLibPath>$(HOME)/dalamud</DalamudLibPath>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<Reference Include="Dalamud">
|
|
||||||
<HintPath>$(DalamudLibPath)\Dalamud.dll</HintPath>
|
|
||||||
<Private>false</Private>
|
|
||||||
</Reference>
|
|
||||||
<Reference Include="FFXIVClientStructs">
|
|
||||||
<HintPath>$(DalamudLibPath)\FFXIVClientStructs.dll</HintPath>
|
|
||||||
<Private>false</Private>
|
|
||||||
</Reference>
|
|
||||||
<Reference Include="Lumina">
|
|
||||||
<HintPath>$(DalamudLibPath)\Lumina.dll</HintPath>
|
|
||||||
<Private>false</Private>
|
|
||||||
</Reference>
|
|
||||||
<Reference Include="Lumina.Excel">
|
|
||||||
<HintPath>$(DalamudLibPath)\Lumina.Excel.dll</HintPath>
|
|
||||||
<Private>false</Private>
|
|
||||||
</Reference>
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
|
|
||||||
</Project>
|
|
||||||
@@ -1,293 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.IO;
|
|
||||||
using System.Linq;
|
|
||||||
using ChatTwo.Code;
|
|
||||||
using ChatTwo.Util;
|
|
||||||
using Dalamud.Game.Text;
|
|
||||||
using Dalamud.Game.Text.SeStringHandling;
|
|
||||||
using Dalamud.Game.Text.SeStringHandling.Payloads;
|
|
||||||
using JetBrains.Annotations;
|
|
||||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
|
||||||
using Chat2PartyFinderPayload = ChatTwo.Util.PartyFinderPayload;
|
|
||||||
|
|
||||||
namespace ChatTwo.Tests;
|
|
||||||
|
|
||||||
[TestClass]
|
|
||||||
[TestSubject(typeof(MessageStore))]
|
|
||||||
public class MessageStoreTest {
|
|
||||||
// From Message.cs
|
|
||||||
private static readonly byte[] ExtraChatChannelPayloadBytes = [0, 0x27, 18, 0x20];
|
|
||||||
|
|
||||||
public TestContext TestContext { get; set; }
|
|
||||||
|
|
||||||
public static string GetImportPath() {
|
|
||||||
string[] importPaths = [
|
|
||||||
@".\TestData",
|
|
||||||
@"..\TestData",
|
|
||||||
@"..\..\TestData",
|
|
||||||
@"..\..\..\TestData",
|
|
||||||
];
|
|
||||||
var importPath = importPaths.FirstOrDefault(Directory.Exists);
|
|
||||||
if (string.IsNullOrEmpty(importPath)) {
|
|
||||||
throw new DirectoryNotFoundException("Could not find the import path");
|
|
||||||
}
|
|
||||||
return importPath;
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(5000)]
|
|
||||||
public void StoreAndRetrieve() {
|
|
||||||
var tempDir = Directory.CreateTempSubdirectory("ChatTwo_test_");
|
|
||||||
var dbPath = Path.Join(tempDir.FullName, "test.db");
|
|
||||||
TestContext.WriteLine("Using database path: " + dbPath);
|
|
||||||
using var store = new MessageStore(dbPath);
|
|
||||||
|
|
||||||
// Write the message.
|
|
||||||
var input = BigMessage();
|
|
||||||
store.UpsertMessage(input);
|
|
||||||
|
|
||||||
// Read the message back.
|
|
||||||
using var messageEnumerator = store.GetMostRecentMessages();
|
|
||||||
var messages = messageEnumerator.ToList();
|
|
||||||
Assert.AreEqual(1, messages.Count);
|
|
||||||
AssertMessagesEqual(input, messages.First());
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(5000)]
|
|
||||||
public void RetrieveMultiple() {
|
|
||||||
var tempDir = Directory.CreateTempSubdirectory("ChatTwo_test_");
|
|
||||||
var dbPath = Path.Join(tempDir.FullName, "test.db");
|
|
||||||
TestContext.WriteLine("Using database path: " + dbPath);
|
|
||||||
using var store = new MessageStore(dbPath);
|
|
||||||
|
|
||||||
// Insert 10 messages in the wrong order of date.
|
|
||||||
var messages = new List<Message>();
|
|
||||||
const uint receiver = 12345;
|
|
||||||
var now = DateTimeOffset.UtcNow;
|
|
||||||
for (var i = 0; i < 10; i++) {
|
|
||||||
var message = BigMessage(true, receiver, now.AddSeconds(-i));
|
|
||||||
TestContext.WriteLine($"Inserting message {i}: {message.Id}");
|
|
||||||
store.UpsertMessage(message);
|
|
||||||
messages.Add(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Insert a message for a different receiver. This shouldn't be returned
|
|
||||||
// because of the receiver filtering.
|
|
||||||
var otherReceiverMsg = BigMessage(receiver: receiver + 1, dateTime: now.AddSeconds(1));
|
|
||||||
TestContext.WriteLine($"Inserting other receiver message: {otherReceiverMsg.Id}");
|
|
||||||
store.UpsertMessage(otherReceiverMsg);
|
|
||||||
|
|
||||||
// Query the most recent 5 messages. Should return the 4 newest messages
|
|
||||||
// from the list, as well as the different receiver message because we
|
|
||||||
// aren't filtering.
|
|
||||||
using var unfilteredMessageEnumerator = store.GetMostRecentMessages(count: 5);
|
|
||||||
var outputMessages = unfilteredMessageEnumerator.ToList();
|
|
||||||
var gotIds = outputMessages.Select(m => m.Id).ToList();
|
|
||||||
TestContext.WriteLine($"Query 1 got IDs: {string.Join(", ", gotIds)}");
|
|
||||||
AssertGuidsEqual(new List<Guid> {
|
|
||||||
messages[3].Id,
|
|
||||||
messages[2].Id,
|
|
||||||
messages[1].Id,
|
|
||||||
messages[0].Id,
|
|
||||||
otherReceiverMsg.Id
|
|
||||||
}, gotIds);
|
|
||||||
|
|
||||||
// Query the most recent 5 messages but filter by receiver ID.
|
|
||||||
using var filteredByReceiverMessageEnumerator = store.GetMostRecentMessages(receiver: receiver, count: 5);
|
|
||||||
outputMessages = filteredByReceiverMessageEnumerator.ToList();
|
|
||||||
gotIds = outputMessages.Select(m => m.Id).ToList();
|
|
||||||
TestContext.WriteLine($"Query 2 got IDs: {string.Join(", ", gotIds)}");
|
|
||||||
AssertGuidsEqual(new List<Guid> {
|
|
||||||
messages[4].Id,
|
|
||||||
messages[3].Id,
|
|
||||||
messages[2].Id,
|
|
||||||
messages[1].Id,
|
|
||||||
messages[0].Id,
|
|
||||||
}, gotIds);
|
|
||||||
|
|
||||||
// Query the most recent 5 messages but only since a specific date.
|
|
||||||
using var filteredByReceiverAndDateMessageEnumerator = store.GetMostRecentMessages(receiver, since: messages[1].Date, count: 5);
|
|
||||||
outputMessages = filteredByReceiverAndDateMessageEnumerator.ToList();
|
|
||||||
gotIds = outputMessages.Select(m => m.Id).ToList();
|
|
||||||
TestContext.WriteLine($"Query 3 got IDs: {string.Join(", ", gotIds)}");
|
|
||||||
AssertGuidsEqual(new List<Guid> {
|
|
||||||
messages[1].Id,
|
|
||||||
messages[0].Id,
|
|
||||||
}, gotIds);
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(5000)]
|
|
||||||
// This test guards against the data format changing in an incompatible way.
|
|
||||||
public void RetrieveExisting() {
|
|
||||||
var input = BigMessage(uniqId: false);
|
|
||||||
|
|
||||||
var dbPath = Path.Join(GetImportPath(), "existing.db");
|
|
||||||
TestContext.WriteLine($"Using existing database: {dbPath}");
|
|
||||||
Assert.IsTrue(File.Exists(dbPath));
|
|
||||||
|
|
||||||
// Uncomment this section to regenerate the existing database.
|
|
||||||
/*
|
|
||||||
File.Delete(dbPath);
|
|
||||||
using (var newStore = new MessageStore(dbPath)) {
|
|
||||||
newStore.UpsertMessage(input);
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
using var store = new MessageStore(dbPath);
|
|
||||||
using var existingMessageEnumerator = store.GetMostRecentMessages();
|
|
||||||
var output = existingMessageEnumerator.ToList();
|
|
||||||
Assert.AreEqual(1, output.Count);
|
|
||||||
AssertMessagesEqual(input, output[0]);
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(30_000)]
|
|
||||||
public void ProfileMany() {
|
|
||||||
const int count = 20_000;
|
|
||||||
|
|
||||||
var tempDir = Directory.CreateTempSubdirectory("ChatTwo_test_");
|
|
||||||
var dbPath = Path.Join(tempDir.FullName, "test.db");
|
|
||||||
TestContext.WriteLine("Using database path: " + dbPath);
|
|
||||||
using var store = new MessageStore(dbPath);
|
|
||||||
|
|
||||||
for (var i = 0; i < count; i++) {
|
|
||||||
var message = BigMessage(uniqId: true);
|
|
||||||
store.UpsertMessage(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
using var messageEnumerator = store.GetMostRecentMessages(count: count);
|
|
||||||
var messages = messageEnumerator.ToList();
|
|
||||||
Assert.AreEqual(count, messages.Count);
|
|
||||||
foreach (var message in messages) {
|
|
||||||
// Load the message because they are lazily parsed.
|
|
||||||
Assert.IsTrue(message.Id != Guid.Empty);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static Message BigMessage(bool uniqId = true, uint receiver = 12345, DateTimeOffset? dateTime = null) {
|
|
||||||
// NOTE: These values aren't valid in the game.
|
|
||||||
// NOTE: we can't test UiForeground, UiGlow, or AutoTranslatePayload
|
|
||||||
// because they load data from the game.
|
|
||||||
var senderSeString = new SeStringBuilder()
|
|
||||||
.AddText("<")
|
|
||||||
.Add(new PlayerPayload("Player Name", 12345))
|
|
||||||
.AddItalics("Player Name")
|
|
||||||
.Add(RawPayload.LinkTerminator)
|
|
||||||
.AddText(">: ")
|
|
||||||
.Build();
|
|
||||||
var extraChatId = Guid.Parse("03d9e6d4-dc1a-4005-bbe7-66b8c3529277");
|
|
||||||
var contentSeString = new SeStringBuilder()
|
|
||||||
.Add(new RawPayload(ExtraChatChannelPayloadBytes.Concat(extraChatId.ToByteArray()).ToArray()))
|
|
||||||
.AddIcon(BitmapFontIcon.IslandSanctuary)
|
|
||||||
.AddMapLink(1, 2, 3, 4)
|
|
||||||
.AddText("map")
|
|
||||||
.Add(RawPayload.LinkTerminator)
|
|
||||||
.AddQuestLink(12345)
|
|
||||||
.AddText("quest")
|
|
||||||
.Add(RawPayload.LinkTerminator)
|
|
||||||
.Add(new DalamudLinkPayload())
|
|
||||||
.AddText("dalamud")
|
|
||||||
.Add(RawPayload.LinkTerminator)
|
|
||||||
.AddStatusLink(12345)
|
|
||||||
.AddText("status")
|
|
||||||
.Add(RawPayload.LinkTerminator)
|
|
||||||
.AddPartyFinderLink(12345)
|
|
||||||
.AddText("party finder")
|
|
||||||
.Add(RawPayload.LinkTerminator)
|
|
||||||
.Build();
|
|
||||||
|
|
||||||
// Add Chat 2 specific payloads (that can't be serialized into the
|
|
||||||
// SeString).
|
|
||||||
var contentChunks = ChunkUtil.ToChunks(contentSeString, ChunkSource.Content, ChatType.Say).ToList();
|
|
||||||
contentChunks = contentChunks.Concat([
|
|
||||||
new TextChunk(ChunkSource.Content, new Chat2PartyFinderPayload(12345), "chat 2 party finder"),
|
|
||||||
new TextChunk(ChunkSource.Content, new AchievementPayload(12345), "chat 2 achievement"),
|
|
||||||
new TextChunk(ChunkSource.Content, new UriPayload(new Uri("https://dalamud.dev")), "chat 2 uri"),
|
|
||||||
]).ToList();
|
|
||||||
|
|
||||||
var chatCode = new ChatCode((XivChatType)46, XivChatRelationKind.LocalPlayer, XivChatRelationKind.EngagedEnemy);
|
|
||||||
return new Message(
|
|
||||||
uniqId ? Guid.NewGuid() : Guid.Parse("f011343e-6a21-49e5-a6f9-238f0f1f8c2c"),
|
|
||||||
receiver,
|
|
||||||
54321,
|
|
||||||
dateTime ?? DateTimeOffset.FromUnixTimeMilliseconds(1713520182440),
|
|
||||||
chatCode,
|
|
||||||
ChunkUtil.ToChunks(senderSeString, ChunkSource.Sender, ChatType.Debug).ToList(),
|
|
||||||
contentChunks,
|
|
||||||
senderSeString,
|
|
||||||
contentSeString,
|
|
||||||
extraChatId
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static void AssertMessagesEqual(Message input, Message output) {
|
|
||||||
// Check basic fields.
|
|
||||||
Assert.AreEqual(input.Id, output.Id);
|
|
||||||
Assert.AreEqual(input.Receiver, output.Receiver);
|
|
||||||
Assert.AreEqual(input.ContentId, output.ContentId);
|
|
||||||
// Assert time is within 1 second
|
|
||||||
var timeDifference = Math.Abs(input.Date.ToUniversalTime().Subtract(output.Date.ToUniversalTime()).TotalSeconds);
|
|
||||||
Assert.IsTrue(timeDifference < 1);
|
|
||||||
Assert.AreEqual(input.Code, output.Code);
|
|
||||||
Assert.AreEqual($"{input.SenderSource.Encode():X}", $"{output.SenderSource.Encode():X}");
|
|
||||||
Assert.AreEqual($"{input.ContentSource.Encode():X}", $"{output.ContentSource.Encode():X}");
|
|
||||||
Assert.AreEqual(input.SortCodeV2, output.SortCodeV2);
|
|
||||||
Assert.AreEqual(input.ExtraChatChannel, output.ExtraChatChannel);
|
|
||||||
|
|
||||||
// Check chunks.
|
|
||||||
AssertChunksEqual(input.Sender, output.Sender);
|
|
||||||
AssertChunksEqual(input.Content, output.Content);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void AssertChunksEqual(IReadOnlyList<Chunk> inputChunks, IReadOnlyList<Chunk> outputChunks) {
|
|
||||||
Assert.AreEqual(inputChunks.Count, outputChunks.Count);
|
|
||||||
for (var i = 0; i < inputChunks.Count; i++) {
|
|
||||||
var inputChunk = inputChunks[i];
|
|
||||||
var outputChunk = outputChunks[i];
|
|
||||||
Assert.AreEqual(inputChunk.Source, outputChunk.Source);
|
|
||||||
switch (inputChunk.Link) {
|
|
||||||
case AchievementPayload inputAchievementPayload:
|
|
||||||
Assert.AreEqual(inputAchievementPayload.Id, ((AchievementPayload) outputChunk.Link)!.Id);
|
|
||||||
break;
|
|
||||||
case Chat2PartyFinderPayload inputPartyFinderPayload:
|
|
||||||
Assert.AreEqual(inputPartyFinderPayload.Id, ((Chat2PartyFinderPayload) outputChunk.Link)!.Id);
|
|
||||||
break;
|
|
||||||
case UriPayload inputUriPayload:
|
|
||||||
Assert.AreEqual(inputUriPayload.Uri, ((UriPayload) outputChunk.Link)!.Uri);
|
|
||||||
break;
|
|
||||||
case null:
|
|
||||||
Assert.IsTrue(outputChunk.Link == null);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
Assert.AreEqual($"{inputChunk.Link.Encode():X}", $"{outputChunk.Link!.Encode():X}");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (inputChunk) {
|
|
||||||
case TextChunk inputTextChunk:
|
|
||||||
var outputTextChunk = (TextChunk)outputChunk;
|
|
||||||
Assert.AreEqual(inputTextChunk.FallbackColour, outputTextChunk.FallbackColour);
|
|
||||||
Assert.AreEqual(inputTextChunk.Foreground, outputTextChunk.Foreground);
|
|
||||||
Assert.AreEqual(inputTextChunk.Glow, outputTextChunk.Glow);
|
|
||||||
Assert.AreEqual(inputTextChunk.Italic, outputTextChunk.Italic);
|
|
||||||
Assert.AreEqual(inputTextChunk.Content, outputTextChunk.Content);
|
|
||||||
break;
|
|
||||||
case IconChunk inputIconChunk:
|
|
||||||
Assert.AreEqual(inputIconChunk.Icon, ((IconChunk) outputChunk).Icon);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new Exception("Unknown chunk type");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void AssertGuidsEqual(IReadOnlyList<Guid> expected, IReadOnlyList<Guid> got) {
|
|
||||||
Assert.AreEqual(expected.Count, got.Count);
|
|
||||||
for (var i = 0; i < expected.Count; i++) {
|
|
||||||
Assert.AreEqual(expected[i].ToString(), got[i].ToString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Vendored
BIN
Binary file not shown.
@@ -200,7 +200,11 @@ internal sealed class AutoTellTabsService : IDisposable
|
|||||||
// Preload first so the tab opens with chronological history above
|
// Preload first so the tab opens with chronological history above
|
||||||
// the current message — and so a slow DB query never causes a
|
// the current message — and so a slow DB query never causes a
|
||||||
// visible "empty tab, then history pops in" effect on screen.
|
// visible "empty tab, then history pops in" effect on screen.
|
||||||
PreloadHistory(tab, partner.Name, partner.World);
|
// The current message is already persisted in the store by the
|
||||||
|
// time MessageProcessed fires (see MessageManager.cs: UpsertMessage
|
||||||
|
// runs before the event), so we have to exclude it explicitly to
|
||||||
|
// avoid the separator landing below the live tell.
|
||||||
|
PreloadHistory(tab, partner.Name, partner.World, currentMessage.Id);
|
||||||
|
|
||||||
tab.AddMessage(currentMessage, unread: true);
|
tab.AddMessage(currentMessage, unread: true);
|
||||||
Plugin.Config.Tabs.Add(tab);
|
Plugin.Config.Tabs.Add(tab);
|
||||||
@@ -238,7 +242,7 @@ internal sealed class AutoTellTabsService : IDisposable
|
|||||||
return $"{playerName}@World{worldRowId}";
|
return $"{playerName}@World{worldRowId}";
|
||||||
}
|
}
|
||||||
|
|
||||||
private void PreloadHistory(Tab tab, string senderName, uint senderWorld)
|
private void PreloadHistory(Tab tab, string senderName, uint senderWorld, Guid currentMessageId)
|
||||||
{
|
{
|
||||||
var preloadCount = Plugin.Config.AutoTellTabsHistoryPreload;
|
var preloadCount = Plugin.Config.AutoTellTabsHistoryPreload;
|
||||||
if (preloadCount <= 0)
|
if (preloadCount <= 0)
|
||||||
@@ -248,13 +252,21 @@ internal sealed class AutoTellTabsService : IDisposable
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
// Pull one extra row because the live tell that triggered this
|
||||||
|
// spawn is already in the store and would otherwise eat one of
|
||||||
|
// the user's preload-budget slots.
|
||||||
var history = _store.GetTellHistoryWithSender(
|
var history = _store.GetTellHistoryWithSender(
|
||||||
_messageManager.CurrentContentId,
|
_messageManager.CurrentContentId,
|
||||||
senderName,
|
senderName,
|
||||||
senderWorld,
|
senderWorld,
|
||||||
preloadCount);
|
preloadCount + 1);
|
||||||
|
|
||||||
if (history.Count == 0)
|
var historicMessages = history
|
||||||
|
.Where(m => m.Id != currentMessageId)
|
||||||
|
.Take(preloadCount)
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
if (historicMessages.Count == 0)
|
||||||
{
|
{
|
||||||
// No prior tells with this player — leave the tab to start
|
// No prior tells with this player — leave the tab to start
|
||||||
// empty so the user does not see a "history loaded" marker
|
// empty so the user does not see a "history loaded" marker
|
||||||
@@ -265,7 +277,7 @@ internal sealed class AutoTellTabsService : IDisposable
|
|||||||
// The history list is already oldest-first, so a plain AddPrune
|
// The history list is already oldest-first, so a plain AddPrune
|
||||||
// loop produces the chronological order the user expects to see
|
// loop produces the chronological order the user expects to see
|
||||||
// when the tab opens.
|
// when the tab opens.
|
||||||
foreach (var message in history)
|
foreach (var message in historicMessages)
|
||||||
{
|
{
|
||||||
tab.Messages.AddPrune(message, MessageManager.MessageDisplayLimit);
|
tab.Messages.AddPrune(message, MessageManager.MessageDisplayLimit);
|
||||||
}
|
}
|
||||||
|
|||||||
Executable → Regular
+11
-12
@@ -4,7 +4,7 @@
|
|||||||
0.1.0 is our bootstrap release; the underlying Chat 2 base is
|
0.1.0 is our bootstrap release; the underlying Chat 2 base is
|
||||||
called out in the yaml changelog so users can see what it
|
called out in the yaml changelog so users can see what it
|
||||||
derives from. -->
|
derives from. -->
|
||||||
<Version>0.5.1</Version>
|
<Version>0.5.4</Version>
|
||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
<!-- HellionChat fork: assembly is renamed so Dalamud uses
|
<!-- HellionChat fork: assembly is renamed so Dalamud uses
|
||||||
pluginConfigs/HellionChat instead of pluginConfigs/ChatTwo,
|
pluginConfigs/HellionChat instead of pluginConfigs/ChatTwo,
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="MessagePack" Version="3.1.4" />
|
<PackageReference Include="MessagePack" Version="3.1.4" />
|
||||||
<PackageReference Include="Microsoft.Data.Sqlite" Version="9.0.0" />
|
<PackageReference Include="Microsoft.Data.Sqlite" Version="10.0.7" />
|
||||||
<PackageReference Include="morelinq" Version="4.4.0" />
|
<PackageReference Include="morelinq" Version="4.4.0" />
|
||||||
<PackageReference Include="Pidgin" Version="3.3.0" />
|
<PackageReference Include="Pidgin" Version="3.3.0" />
|
||||||
<PackageReference Include="SixLabors.ImageSharp" Version="3.1.12" />
|
<PackageReference Include="SixLabors.ImageSharp" Version="3.1.12" />
|
||||||
@@ -57,17 +57,16 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
|
|
||||||
|
<!-- Plugin icon. Copy images/* into the build output so Dalamud
|
||||||
|
finds the icon next to the DLL, and let the SDK default
|
||||||
|
DalamudPackager pipeline include the same path in the
|
||||||
|
release ZIP. Earlier we shipped a custom DalamudPackager
|
||||||
|
targets override that explicitly set HandleImages and
|
||||||
|
ImagesPath; that override conflicted with the SDK 15
|
||||||
|
default and the resulting manifest carried no IconUrl.
|
||||||
|
Removed in v0.5.2. -->
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Folder Include="images\" />
|
<None Include="images\**">
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<!-- Copy images/icon.png next to the built DLL so Dalamud's local
|
|
||||||
plugin loader finds it at <plugindir>/images/icon.png. The
|
|
||||||
DalamudPackager.targets file in this directory then includes
|
|
||||||
the same path inside the release ZIP — see that file for the
|
|
||||||
full packaging override. -->
|
|
||||||
<ItemGroup>
|
|
||||||
<None Include="images\icon.png">
|
|
||||||
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
||||||
</None>
|
</None>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|||||||
@@ -73,8 +73,10 @@ public class Configuration : IPluginConfiguration
|
|||||||
public bool HellionThemeEnabled = true;
|
public bool HellionThemeEnabled = true;
|
||||||
|
|
||||||
// Window background opacity, 0.5–1.0. Lower values make the plugin
|
// Window background opacity, 0.5–1.0. Lower values make the plugin
|
||||||
// panes more glass-like so the game shines through. Default ~92%.
|
// panes more glass-like so the game shines through. Default 0.5
|
||||||
public float HellionThemeWindowOpacity = 0.92f;
|
// matches the maintainer's daily-driver preference; users who want
|
||||||
|
// a less translucent look bump it up in Aussehen → Theme.
|
||||||
|
public float HellionThemeWindowOpacity = 0.5f;
|
||||||
|
|
||||||
// Use the bundled Exo 2 font (OFL-1.1) for the regular plugin font
|
// Use the bundled Exo 2 font (OFL-1.1) for the regular plugin font
|
||||||
// instead of whatever GlobalFontV2.FontId points at. Default ON so a
|
// instead of whatever GlobalFontV2.FontId points at. Default ON so a
|
||||||
@@ -158,7 +160,11 @@ public class Configuration : IPluginConfiguration
|
|||||||
public bool PlaySounds = true;
|
public bool PlaySounds = true;
|
||||||
public bool KeepInputFocus = true;
|
public bool KeepInputFocus = true;
|
||||||
public int MaxLinesToRender = 5_000; // 1-10000
|
public int MaxLinesToRender = 5_000; // 1-10000
|
||||||
public bool Use24HourClock;
|
// Default ON to match a German / European 24h locale. The
|
||||||
|
// ChatLogWindow.cs format-flip in v0.5.1 honours this strictly via
|
||||||
|
// CultureInfo.InvariantCulture so the result is consistent across
|
||||||
|
// host locales.
|
||||||
|
public bool Use24HourClock = true;
|
||||||
|
|
||||||
public bool ShowEmotes = true;
|
public bool ShowEmotes = true;
|
||||||
public HashSet<string> BlockedEmotes = [];
|
public HashSet<string> BlockedEmotes = [];
|
||||||
|
|||||||
@@ -1,76 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<!--
|
|
||||||
HellionChat — DalamudPackager override.
|
|
||||||
|
|
||||||
The default DalamudPackager.targets shipped by the SDK does not set
|
|
||||||
HandleImages / ImagesPath, so the images/ directory is silently
|
|
||||||
excluded from the release ZIP. The presence of this file at
|
|
||||||
$(ProjectDir)DalamudPackager.targets disables the SDK's default
|
|
||||||
target (it guards on `!Exists('$(PackagerTargetFile)')`) and lets
|
|
||||||
us call the packager task ourselves with the image fields wired in.
|
|
||||||
|
|
||||||
Apart from HandleImages + ImagesPath the property list mirrors the
|
|
||||||
SDK default verbatim so we don't lose any other manifest field as
|
|
||||||
the upstream SDK evolves.
|
|
||||||
-->
|
|
||||||
<Project>
|
|
||||||
<Target Name="HellionDalamudPackagerDebug"
|
|
||||||
AfterTargets="Build"
|
|
||||||
Condition="'$(Configuration)' == 'Debug'">
|
|
||||||
<DalamudPackager ProjectDir="$(ProjectDir)"
|
|
||||||
OutputPath="$(OutputPath)"
|
|
||||||
AssemblyName="$(AssemblyName)"
|
|
||||||
MakeZip="false"
|
|
||||||
Author="$(Author)"
|
|
||||||
Name="$(Name)"
|
|
||||||
MinimumDalamudVersion="$(MinimumDalamudVersion)"
|
|
||||||
Punchline="$(Punchline)"
|
|
||||||
Description="$(Description)"
|
|
||||||
ApplicableVersion="$(ApplicableVersion)"
|
|
||||||
RepoUrl="$(RepoUrl)"
|
|
||||||
Tags="$(Tags)"
|
|
||||||
CategoryTags="$(CategoryTags)"
|
|
||||||
DalamudApiLevel="$(DalamudApiLevel)"
|
|
||||||
LoadRequiredState="$(LoadRequiredState)"
|
|
||||||
LoadSync="$(LoadSync)"
|
|
||||||
CanUnloadAsync="$(CanUnloadAsync)"
|
|
||||||
LoadPriority="$(LoadPriority)"
|
|
||||||
ImageUrls="$(ImageUrls)"
|
|
||||||
IconUrl="$(IconUrl)"
|
|
||||||
Changelog="$(Changelog)"
|
|
||||||
AcceptsFeedback="$(AcceptsFeedback)"
|
|
||||||
FeedbackMessage="$(FeedbackMessage)"
|
|
||||||
HandleImages="true"
|
|
||||||
ImagesPath="$(ProjectDir)images" />
|
|
||||||
</Target>
|
|
||||||
|
|
||||||
<Target Name="HellionDalamudPackagerRelease"
|
|
||||||
AfterTargets="Build"
|
|
||||||
Condition="'$(Configuration)' == 'Release'">
|
|
||||||
<DalamudPackager ProjectDir="$(ProjectDir)"
|
|
||||||
OutputPath="$(OutputPath)"
|
|
||||||
AssemblyName="$(AssemblyName)"
|
|
||||||
MakeZip="true"
|
|
||||||
Author="$(Author)"
|
|
||||||
Name="$(Name)"
|
|
||||||
MinimumDalamudVersion="$(MinimumDalamudVersion)"
|
|
||||||
Punchline="$(Punchline)"
|
|
||||||
Description="$(Description)"
|
|
||||||
ApplicableVersion="$(ApplicableVersion)"
|
|
||||||
RepoUrl="$(RepoUrl)"
|
|
||||||
Tags="$(Tags)"
|
|
||||||
CategoryTags="$(CategoryTags)"
|
|
||||||
DalamudApiLevel="$(DalamudApiLevel)"
|
|
||||||
LoadRequiredState="$(LoadRequiredState)"
|
|
||||||
LoadSync="$(LoadSync)"
|
|
||||||
CanUnloadAsync="$(CanUnloadAsync)"
|
|
||||||
LoadPriority="$(LoadPriority)"
|
|
||||||
ImageUrls="$(ImageUrls)"
|
|
||||||
IconUrl="$(IconUrl)"
|
|
||||||
Changelog="$(Changelog)"
|
|
||||||
AcceptsFeedback="$(AcceptsFeedback)"
|
|
||||||
FeedbackMessage="$(FeedbackMessage)"
|
|
||||||
HandleImages="true"
|
|
||||||
ImagesPath="$(ProjectDir)images" />
|
|
||||||
</Target>
|
|
||||||
</Project>
|
|
||||||
+54
-271
@@ -33,6 +33,10 @@ description: |-
|
|||||||
Based on Chat 2 by Infi and Anna, licensed under EUPL-1.2.
|
Based on Chat 2 by Infi and Anna, licensed under EUPL-1.2.
|
||||||
repo_url: https://github.com/JonKazama-Hellion/HellionChat
|
repo_url: https://github.com/JonKazama-Hellion/HellionChat
|
||||||
accepts_feedback: true
|
accepts_feedback: true
|
||||||
|
icon_url: https://raw.githubusercontent.com/JonKazama-Hellion/HellionChat/main/ChatTwo/images/icon.png
|
||||||
|
image_urls:
|
||||||
|
- https://raw.githubusercontent.com/JonKazama-Hellion/HellionChat/main/ChatTwo/images/chatWindow.png
|
||||||
|
- https://raw.githubusercontent.com/JonKazama-Hellion/HellionChat/main/ChatTwo/images/withSimpleTweaks.png
|
||||||
tags:
|
tags:
|
||||||
- Social
|
- Social
|
||||||
- UI
|
- UI
|
||||||
@@ -40,278 +44,57 @@ tags:
|
|||||||
- Replacement
|
- Replacement
|
||||||
- Privacy
|
- Privacy
|
||||||
changelog: |-
|
changelog: |-
|
||||||
|
**Hellion Chat 0.5.4 — WrapText hardening**
|
||||||
|
|
||||||
|
Replaces the unsafe pointer-arithmetic in ImGuiUtil.WrapText with
|
||||||
|
Span- and index-based control flow. Closes the persistent CodeQL
|
||||||
|
Critical alert "unvalidated local pointer arithmetic" that kept
|
||||||
|
re-firing on every shape of the previous fix.
|
||||||
|
|
||||||
|
Hardening:
|
||||||
|
|
||||||
|
- WrapText now allocates a buffer sized by Encoding.UTF8.GetMaxByteCount
|
||||||
|
via ArrayPool, validates the actual encoded length against that
|
||||||
|
ceiling, and threads the rest of the algorithm through int offsets
|
||||||
|
instead of raw byte pointers
|
||||||
|
- Pointer arithmetic only happens inside two small private helpers
|
||||||
|
(CalcWordWrap and DrawText) that take the pinned base pointer plus
|
||||||
|
int offsets sourced from the plugin's own logic, not from any
|
||||||
|
virtual-method return
|
||||||
|
- Added a 16 KiB upper bound on the buffer rent to prevent a
|
||||||
|
pathological input from triggering an unbounded ArrayPool allocation
|
||||||
|
|
||||||
|
No user-visible behaviour change. Word-wrap output is byte-identical
|
||||||
|
to v0.5.3.
|
||||||
|
|
||||||
|
Based on Chat 2 1.35.3 (upstream Infiziert90/ChatTwo, EUPL-1.2).
|
||||||
|
|
||||||
|
**Hellion Chat 0.5.3 — Pointer arithmetic hardening**
|
||||||
|
|
||||||
|
Closed CodeQL Critical alert in ImGuiUtil.WrapText by validating the
|
||||||
|
encoded byte buffer length via GetByteCount before pointer
|
||||||
|
arithmetic. Single-fix patch on top of v0.5.2.
|
||||||
|
|
||||||
|
**Hellion Chat 0.5.2 — Bugfix patch**
|
||||||
|
|
||||||
|
Auto-Tell-Tabs history-separator landed below the live tell instead
|
||||||
|
of above (preload now excludes the trigger message). Plugin icon
|
||||||
|
packaging fixed by removing a stale DalamudPackager.targets override
|
||||||
|
that conflicted with the SDK 15 default. Default config aligned to
|
||||||
|
the maintainer's daily driver: HellionThemeWindowOpacity 0.5,
|
||||||
|
Use24HourClock true, Gruppe tab no longer auto-routes /party. Two
|
||||||
|
earlier CodeQL findings closed (workflow permissions, empty-input
|
||||||
|
pointer arithmetic).
|
||||||
|
|
||||||
**Hellion Chat 0.5.1 — Backlog Sweep**
|
**Hellion Chat 0.5.1 — Backlog Sweep**
|
||||||
|
|
||||||
Pure hardening and polish. No new features. Eight backlog items
|
Pure hardening and polish. Eight backlog items from the v0.5.0
|
||||||
from the v0.5.0 codebase review collected into one patch:
|
codebase review collected into one patch: cleanup-preview-stale
|
||||||
|
detection, greeted-tab dim background, Performance HelpMarker
|
||||||
|
consistency, Tabs/Database tab names from HellionStrings,
|
||||||
|
FontChooser framework-thread marshalling, async-void on
|
||||||
|
EmoteCache.LoadData, parameterised SQL via BindIntList helper.
|
||||||
|
|
||||||
- Cleanup preview now flags itself as out-of-date when the user
|
---
|
||||||
edits the whitelist after the last refresh, and the refresh
|
|
||||||
button is visually emphasised in that state
|
|
||||||
- Greeted Auto-Tell-Tabs now also dim their selection and hover
|
|
||||||
backgrounds in the sidebar, not just the text
|
|
||||||
- Performance section in the General tab moves to the standard
|
|
||||||
HelpMarker tooltip pattern instead of a wall-of-text description
|
|
||||||
- Tabs and Database settings tabs pull their display name from
|
|
||||||
HellionStrings instead of the upstream Language bundle, so all
|
|
||||||
eight tabs share one i18n source
|
|
||||||
- FontChooser results are now marshalled onto the framework thread
|
|
||||||
via Plugin.Framework.Run instead of being written to settings
|
|
||||||
state directly from the threadpool
|
|
||||||
- EmoteCache.LoadData drops async void and the four CS8618 build
|
|
||||||
warnings the build has been carrying since v0.4.0
|
|
||||||
- All MessageStore SQL paths that fed dynamic value lists into
|
|
||||||
interpolated SQL now use named parameter bindings via a new
|
|
||||||
BindIntList helper. Same behaviour, defence against future
|
|
||||||
user-input regressions
|
|
||||||
|
|
||||||
Configuration version is unchanged at 10. No migration. Existing
|
Earlier history at https://github.com/JonKazama-Hellion/HellionChat/releases.
|
||||||
installs upgrade silently.
|
|
||||||
|
|
||||||
Based on Chat 2 1.35.3 (upstream Infiziert90/ChatTwo, EUPL-1.2).
|
|
||||||
|
|
||||||
**Hellion Chat 0.5.0 — Settings UX polish**
|
|
||||||
|
|
||||||
The settings window has been pulled apart and rebuilt around eight
|
|
||||||
themed tabs instead of the twelve organic ones it grew into.
|
|
||||||
Settings now sit where they belong and the wall-of-text descriptions
|
|
||||||
have been replaced with hover help markers across every section.
|
|
||||||
|
|
||||||
What changed in this release:
|
|
||||||
|
|
||||||
- Twelve tabs collapsed into eight: General, Appearance, Window,
|
|
||||||
Chat, Tabs, Privacy, Database and Information
|
|
||||||
- Theme and font controls moved out of the Privacy tab into
|
|
||||||
Appearance where they belong
|
|
||||||
- Auto-Tell-Tabs settings, message preview and emote controls now
|
|
||||||
live under one Chat tab with collapsible sections
|
|
||||||
- About and Changelog merged into a single Information tab
|
|
||||||
- Disabled settings keep their tooltip help marker visible so you
|
|
||||||
can still read why an option is greyed out
|
|
||||||
- Section headings start collapsed by default, the same pattern
|
|
||||||
used for the Auto-Tell-Tabs preload section in 0.4.0
|
|
||||||
|
|
||||||
Configuration version bumps from 9 to 10 as a wipe migration. The
|
|
||||||
old config file is copied to HellionChat.json.pre-v10-backup before
|
|
||||||
the new defaults are written, so you can restore your previous
|
|
||||||
setup by hand if anything looks off. A one-shot notification on
|
|
||||||
first start explains the reset.
|
|
||||||
|
|
||||||
No changes to message storage, retention sweep, the privacy filter
|
|
||||||
or the export pipeline. Tabs and chat history are untouched by the
|
|
||||||
migration.
|
|
||||||
|
|
||||||
Based on Chat 2 1.35.3 (upstream Infiziert90/ChatTwo, EUPL-1.2).
|
|
||||||
|
|
||||||
**Hellion Chat 0.4.0 — Auto-Tell-Tabs**
|
|
||||||
|
|
||||||
Auto-Tell-Tabs lets you turn each /tell into a session-only tab
|
|
||||||
dedicated to that conversation partner. The original use case is
|
|
||||||
the FFXIV club greeter who has to track 5–15 parallel "hi, welcome"
|
|
||||||
exchanges; everyone else can disable the feature in one click and
|
|
||||||
go back to a single Tell Exclusive tab.
|
|
||||||
|
|
||||||
What lands in this release:
|
|
||||||
|
|
||||||
- Auto-spawn temp tab "Name@World" on /tell (incoming and outgoing)
|
|
||||||
- Tab limit (default 15, range 1–50) with LRU drop that prefers
|
|
||||||
greeted tabs first, then sorts by last activity
|
|
||||||
- History preload from the local message store (default 20 tells,
|
|
||||||
range 0–100) with a "— Earlier conversations —" separator above
|
|
||||||
the live tell that triggered the spawn
|
|
||||||
- Optional "mark as greeted" toggle button (off by default,
|
|
||||||
greeter-specific) that dims the tab name and lets you flip the
|
|
||||||
status
|
|
||||||
- Section header "Active Tells (n)" or compact-mode separator in
|
|
||||||
the sidebar between persistent tabs and the temp tabs
|
|
||||||
- Settings UI under Chat (toggle / limit / compact / greeted-toggle)
|
|
||||||
and Privacy (history preload count), with hover-tooltip help
|
|
||||||
markers replacing the previous wall-of-text descriptions for the
|
|
||||||
new sections
|
|
||||||
- Save and load filters strip temp tabs from the on-disk config so
|
|
||||||
a crash or a sidebar-mode toggle never persists or wipes them
|
|
||||||
|
|
||||||
Compatibility note: if XIV Messanger or another plugin is
|
|
||||||
suppressing direct messages, disable its "Suppress DMs" option so
|
|
||||||
Hellion Chat can receive tells and open the auto tabs.
|
|
||||||
|
|
||||||
Configuration version bumps from 8 to 9. Existing users get a one-
|
|
||||||
shot notification on the first start, defaults are seeded by
|
|
||||||
property initializers, persistent tabs are untouched.
|
|
||||||
|
|
||||||
The vertical sidebar tab view becomes the default for fresh
|
|
||||||
installs; existing users keep their saved preference.
|
|
||||||
|
|
||||||
Inspired by the per-sender tab pattern in XIV InstantMessenger
|
|
||||||
(Limiana, AGPL-3.0). No code was ported across the licence
|
|
||||||
boundary; only the architectural concept influenced this design.
|
|
||||||
|
|
||||||
Based on Chat 2 1.35.3 (upstream Infiziert90/ChatTwo, EUPL-1.2).
|
|
||||||
|
|
||||||
**Hellion Chat 0.3.1 — Upstream emote regression fix**
|
|
||||||
|
|
||||||
Cherry-picks Infi's upstream commit ff899ff "Fix a regression
|
|
||||||
from API 15 updates" which changes the BetterTTV emote DTOs
|
|
||||||
(Emote and Top100) from public fields to public properties.
|
|
||||||
System.Text.Json under the API 15 toolchain only honours the
|
|
||||||
[JsonPropertyName] attribute on properties, so the previous
|
|
||||||
field-based version deserialised every fetched emote into empty
|
|
||||||
default values. Result: BetterTTV emotes were silently broken
|
|
||||||
on fresh installs. The fix is six lines and applies cleanly on
|
|
||||||
top of our defensive null-check from earlier; the EmoteCache
|
|
||||||
path-traversal hardening from 0.3.0 stays as it is.
|
|
||||||
|
|
||||||
Authorship of the fix is preserved with git cherry-pick -x, so
|
|
||||||
Infi shows up as the author on the commit. Thanks to him for
|
|
||||||
catching it in the upstream codebase.
|
|
||||||
|
|
||||||
**Hellion Chat 0.3.0 — Audit hardening, brand sweep and rebrand of slash commands**
|
|
||||||
|
|
||||||
This release closes the remaining audit follow-ups from the
|
|
||||||
0.2.0 cleanup and finishes turning Hellion Chat into a properly
|
|
||||||
branded fork rather than a Chat 2 with a different name.
|
|
||||||
|
|
||||||
Slash commands have been renamed across the board so they no
|
|
||||||
longer collide with the upstream plugin and tell you which
|
|
||||||
plugin owns them at a glance:
|
|
||||||
|
|
||||||
- /chat2 becomes /hellion
|
|
||||||
- /chat2Viewer becomes /hellionView
|
|
||||||
- /clearlog2 becomes /clearhellion
|
|
||||||
- /chat2Debugger becomes /hellionDebugger (internal)
|
|
||||||
- /chat2SeString becomes /hellionSeString (internal)
|
|
||||||
|
|
||||||
This is a breaking change for anyone with macros bound to the
|
|
||||||
old command names. The upstream Chat 2 commands keep working
|
|
||||||
if you also have that plugin installed.
|
|
||||||
|
|
||||||
Privacy and storage hardening based on the post-0.2.0 audit:
|
|
||||||
|
|
||||||
- Privacy filter master switch now states explicitly that the
|
|
||||||
filter only governs storage, not the live chat log
|
|
||||||
- Emote cache refuses to write outside its own directory if a
|
|
||||||
third-party API ever returns a path that escapes
|
|
||||||
- Retention sweep is serialised so the 24h auto-sweep and the
|
|
||||||
manual button cannot launch in parallel and race for the
|
|
||||||
SQLite connection
|
|
||||||
- DbViewer paging uses an int constant and the matching SQL
|
|
||||||
parameter name (the upstream code passed a float and a name
|
|
||||||
without the parameter prefix; both worked in practice but
|
|
||||||
were inconsistent)
|
|
||||||
|
|
||||||
Visual identity now matches the Hellion Online Media website:
|
|
||||||
|
|
||||||
- Theme palette switched to Arctic Cyan plus Ember Orange,
|
|
||||||
matching the website's BRANDING.md tokens
|
|
||||||
- Active tabs and window title bars use a brand-color-dark teal
|
|
||||||
variation as identity colour, replacing the previous slate
|
|
||||||
violet that did not appear in the brand
|
|
||||||
- Resize grips and scrollbar grabs picked up Ember Orange
|
|
||||||
instead of industrial amber on hover and active states
|
|
||||||
|
|
||||||
About tab rewritten and properly localised:
|
|
||||||
|
|
||||||
- New "Why this fork exists" block sets out the mission in
|
|
||||||
neutral terms, framing Chat 2's full-history default as the
|
|
||||||
right one for most users while explaining the narrower
|
|
||||||
default footprint this fork chose
|
|
||||||
- All Hellion-specific About copy now lives in HellionStrings
|
|
||||||
in EN and DE, so German users see the Hellion sections in
|
|
||||||
German rather than the upstream English fallback
|
|
||||||
- Webinterface absence is described as a focus mismatch
|
|
||||||
(different use case, substantial rebuild) rather than as
|
|
||||||
a security issue with the upstream code
|
|
||||||
- Translator list at the bottom of the About tab is reachable
|
|
||||||
again on smaller settings windows
|
|
||||||
|
|
||||||
Based on Chat 2 1.35.3 (upstream Infiziert90/ChatTwo, EUPL-1.2).
|
|
||||||
|
|
||||||
**Hellion Chat 0.2.0 — Webinterface removed**
|
|
||||||
|
|
||||||
The upstream webinterface has been removed in its entirety. It
|
|
||||||
serves a different use case from the smaller default footprint
|
|
||||||
this fork is built around, namely remote access to chat from a
|
|
||||||
second device. Aligning it with the data minimisation defaults
|
|
||||||
Hellion Chat ships with would have meant a substantial rebuild.
|
|
||||||
Removing it was the cleaner path for this particular fork.
|
|
||||||
|
|
||||||
What changed in this release:
|
|
||||||
|
|
||||||
- Settings tab "Webinterface" is gone, the corresponding
|
|
||||||
Configuration fields (WebinterfaceEnabled / AutoStart / Password /
|
|
||||||
Port / AuthStore / MaxLinesToSend) are dropped and stale entries
|
|
||||||
fall out of the JSON on the next save automatically
|
|
||||||
- The whole ChatTwo/Http tree, the bundled Svelte frontend in
|
|
||||||
websiteBuild.zip and the WebinterfaceUtil helper are deleted
|
|
||||||
- Watson.Lite (the HTTP server) and Newtonsoft.Json (only used by
|
|
||||||
the webinterface JSON wire format) are removed from the
|
|
||||||
package references
|
|
||||||
- DbViewer's "Chat2 JSON Export" button is dropped because it
|
|
||||||
serialised the database into the webinterface message protocol;
|
|
||||||
the Privacy tab's MessageExporter (Markdown, JSON, CSV with
|
|
||||||
channel and date filters) covers the same ground without the
|
|
||||||
proprietary shape
|
|
||||||
- About tab notes the absence so users coming from Chat 2 do not
|
|
||||||
look for it
|
|
||||||
- Configuration version bumps from 7 to 8 with a one-shot
|
|
||||||
notification (EN + DE)
|
|
||||||
|
|
||||||
No changes to the privacy filter, retention sweep, first-run wizard
|
|
||||||
or export pipeline. Existing chat history is preserved.
|
|
||||||
|
|
||||||
Based on Chat 2 1.35.3 (upstream Infiziert90/ChatTwo, EUPL-1.2).
|
|
||||||
|
|
||||||
**Hellion Chat 0.1.2 — About tab rebrand, DBViewer polish**
|
|
||||||
|
|
||||||
- About tab now shows Hellion-specific maintainer, license, EU/US/JP
|
|
||||||
disclaimer and SQUARE ENIX disclaimer instead of the inherited
|
|
||||||
Chat 2 contact info; original ChatTwo translator credits stay
|
|
||||||
visible under a clearly labelled upstream tree node
|
|
||||||
- Localization clarified: Hellion-specific German strings are
|
|
||||||
maintained by the fork maintainer, the Crowdin contributor list
|
|
||||||
only covers the inherited upstream strings
|
|
||||||
- Cherry-picked DBViewer UI improvements from upstream Chat 2
|
|
||||||
(auto-scroll-reset on page change, tooltips on date reset,
|
|
||||||
folder export, page arrows, localized export-running messages)
|
|
||||||
- README rewritten in the Hellion project style with a tech-stack
|
|
||||||
table, architecture tree, database column list, install guide,
|
|
||||||
upstream-sync workflow notes and project-status checklist
|
|
||||||
|
|
||||||
**Hellion Chat 0.1.1 — Packaging and migration fixes**
|
|
||||||
|
|
||||||
- Plugin icon now ships inside the bundle, so the Hellion logo
|
|
||||||
renders locally in the Dalamud plugin list once installed (the
|
|
||||||
previous release relied only on the remote IconUrl)
|
|
||||||
- Plugin icon downsampled from 1024×1024 to 256×256 to match the
|
|
||||||
rendered size; loads faster and caches better
|
|
||||||
- Migration from upstream Chat 2 is more robust: each file move is
|
|
||||||
wrapped individually, a locked SQLite database no longer aborts
|
|
||||||
the rest of the migration, and a warning notification fires when
|
|
||||||
any file is held open (with a hint to disable Chat 2 and restart
|
|
||||||
the game)
|
|
||||||
- README ships a step-by-step migration guide (fresh install versus
|
|
||||||
coming from Chat 2) and a troubleshooting section with manual
|
|
||||||
recovery commands for Linux and Windows
|
|
||||||
|
|
||||||
**Hellion Chat 0.1.0 — Initial fork release**
|
|
||||||
|
|
||||||
Privacy
|
|
||||||
- Channel whitelist filter in MessageStore.UpsertMessage with a
|
|
||||||
Privacy-First default (own conversations only)
|
|
||||||
- Per-channel retention with a 24-hour idempotent background sweep
|
|
||||||
- Retroactive cleanup with a Ctrl+Shift confirm and VACUUM
|
|
||||||
- Export to Markdown / JSON / CSV via Dalamud's file dialog
|
|
||||||
|
|
||||||
Onboarding
|
|
||||||
- First-run wizard with three profiles: Privacy-First / Casual /
|
|
||||||
Full History
|
|
||||||
- Configuration migration that seeds defaults on update
|
|
||||||
- One-shot migration from upstream Chat 2's pluginConfigs layout
|
|
||||||
- Migrate3 idempotency recovery for half-migrated databases
|
|
||||||
|
|
||||||
Look & feel
|
|
||||||
- Localized UI (English and German) with live language switching
|
|
||||||
- Industrial HUD theme with cyan-teal action accents, slate-violet
|
|
||||||
tabs, amber active highlights and a window-opacity slider
|
|
||||||
|
|
||||||
Based on Chat 2 1.35.3 (upstream Infiziert90/ChatTwo, EUPL-1.2).
|
|
||||||
|
|||||||
+143
-90
@@ -1,3 +1,4 @@
|
|||||||
|
using System.Buffers;
|
||||||
using System.Numerics;
|
using System.Numerics;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using ChatTwo.Code;
|
using ChatTwo.Code;
|
||||||
@@ -58,123 +59,175 @@ internal static class ImGuiUtil
|
|||||||
handler.Click(chunk, payload, button);
|
handler.Click(chunk, payload, button);
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static unsafe void WrapText(string csText, Chunk chunk, PayloadHandler? handler, Vector4 defaultText, float lineWidth)
|
// Ceiling on the byte buffer for a single rendered line. UTF-8 takes at
|
||||||
|
// most 4 bytes per char; ImGui's internal ImString limit is well below
|
||||||
|
// this and FFXIV's chat lines top out around a few hundred chars in
|
||||||
|
// practice. The cap prevents an unbounded ArrayPool rent if a caller
|
||||||
|
// ever feeds in a degenerate input.
|
||||||
|
private const int MaxLineByteCount = 16 * 1024;
|
||||||
|
|
||||||
|
internal static void WrapText(string csText, Chunk chunk, PayloadHandler? handler, Vector4 defaultText, float lineWidth)
|
||||||
{
|
{
|
||||||
void Text(byte* text, byte* textEnd)
|
|
||||||
{
|
|
||||||
var oldPos = ImGui.GetCursorScreenPos();
|
|
||||||
|
|
||||||
ImGuiNative.TextUnformatted(text, textEnd);
|
|
||||||
PostPayload(chunk, handler);
|
|
||||||
|
|
||||||
if (!ReferenceEquals(LastLink, chunk.Link))
|
|
||||||
PayloadBounds.Clear();
|
|
||||||
|
|
||||||
LastLink = chunk.Link;
|
|
||||||
|
|
||||||
if (Hovered != null && ReferenceEquals(Hovered, chunk.Link))
|
|
||||||
{
|
|
||||||
defaultText.W = 0.25f;
|
|
||||||
var actualCol = ColourUtil.Vector4ToAbgr(defaultText);
|
|
||||||
ImGui.GetWindowDrawList().AddRectFilled(oldPos, oldPos + ImGui.GetItemRectSize(), actualCol);
|
|
||||||
|
|
||||||
foreach (var (start, size) in PayloadBounds)
|
|
||||||
ImGui.GetWindowDrawList().AddRectFilled(start, start + size, actualCol);
|
|
||||||
|
|
||||||
PayloadBounds.Clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Hovered == null && chunk.Link != null)
|
|
||||||
PayloadBounds.Add((oldPos, ImGui.GetItemRectSize()));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (csText.Length == 0)
|
if (csText.Length == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
foreach (var part in csText.Split(["\r\n", "\r", "\n"], StringSplitOptions.None))
|
foreach (var part in csText.Split(["\r\n", "\r", "\n"], StringSplitOptions.None))
|
||||||
{
|
{
|
||||||
var bytes = Encoding.UTF8.GetBytes(part);
|
if (part.Length == 0)
|
||||||
fixed (byte* rawText = bytes)
|
|
||||||
{
|
{
|
||||||
var text = rawText;
|
ImGui.TextUnformatted("");
|
||||||
var textEnd = text + bytes.Length;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// empty string
|
// Allocate against the encoder's own MaxByteCount so the buffer
|
||||||
if (text == null)
|
// we hand to ImGui is sized by us. The actual byte count
|
||||||
|
// returned by GetBytes is then validated against that ceiling
|
||||||
|
// before any pointer arithmetic touches it; CodeQL recognises
|
||||||
|
// that comparison as a sanitiser for the
|
||||||
|
// cs/unvalidated-local-pointer-arithmetic taint flow.
|
||||||
|
var maxBytes = Encoding.UTF8.GetMaxByteCount(part.Length);
|
||||||
|
if (maxBytes <= 0 || maxBytes > MaxLineByteCount)
|
||||||
|
{
|
||||||
|
ImGui.TextUnformatted("");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var buffer = ArrayPool<byte>.Shared.Rent(maxBytes);
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var written = Encoding.UTF8.GetBytes(part, 0, part.Length, buffer, 0);
|
||||||
|
if (written <= 0 || written > maxBytes)
|
||||||
{
|
{
|
||||||
ImGui.TextUnformatted("");
|
ImGui.TextUnformatted("");
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
var widthLeft = ImGui.GetContentRegionAvail().X;
|
WrapEncodedLine(buffer.AsSpan(0, written), chunk, handler, defaultText, lineWidth);
|
||||||
var endPrevLine = ImGuiNative.CalcWordWrapPositionA(ImGui.GetFont().Handle, ImGuiHelpers.GlobalScale, text, textEnd, widthLeft);
|
}
|
||||||
if (endPrevLine == null)
|
finally
|
||||||
continue;
|
{
|
||||||
|
ArrayPool<byte>.Shared.Return(buffer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var firstSpace = FindFirstSpace(text, textEnd);
|
private static unsafe void WrapEncodedLine(ReadOnlySpan<byte> bytes, Chunk chunk, PayloadHandler? handler, Vector4 defaultText, float lineWidth)
|
||||||
var properBreak = firstSpace <= endPrevLine;
|
{
|
||||||
|
var byteCount = bytes.Length;
|
||||||
|
if (byteCount == 0)
|
||||||
|
{
|
||||||
|
ImGui.TextUnformatted("");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fixed (byte* basePtr = bytes)
|
||||||
|
{
|
||||||
|
var widthLeft = ImGui.GetContentRegionAvail().X;
|
||||||
|
var endPrev = CalcWordWrap(basePtr, 0, byteCount, widthLeft);
|
||||||
|
if (endPrev < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var firstSpace = FindFirstSpace(bytes, 0, byteCount);
|
||||||
|
var properBreak = firstSpace <= endPrev;
|
||||||
|
if (properBreak)
|
||||||
|
{
|
||||||
|
DrawText(basePtr, 0, endPrev, chunk, handler, defaultText);
|
||||||
|
}
|
||||||
|
else if (lineWidth == 0f)
|
||||||
|
{
|
||||||
|
ImGui.TextUnformatted("");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Check whether the next chunk would wrap at or past the
|
||||||
|
// first space. If yes, force a line break.
|
||||||
|
var wrapPos = CalcWordWrap(basePtr, 0, firstSpace, lineWidth);
|
||||||
|
if (wrapPos >= firstSpace)
|
||||||
|
ImGui.TextUnformatted("");
|
||||||
|
}
|
||||||
|
|
||||||
|
widthLeft = ImGui.GetContentRegionAvail().X;
|
||||||
|
var lineStart = 0;
|
||||||
|
while (endPrev < byteCount)
|
||||||
|
{
|
||||||
if (properBreak)
|
if (properBreak)
|
||||||
{
|
lineStart = endPrev;
|
||||||
Text(text, endPrevLine);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (lineWidth == 0f)
|
|
||||||
{
|
|
||||||
ImGui.TextUnformatted("");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// check if the next bit is longer than the entire line width
|
|
||||||
var wrapPos = ImGuiNative.CalcWordWrapPositionA(ImGui.GetFont().Handle, ImGuiHelpers.GlobalScale, text, firstSpace, lineWidth);
|
|
||||||
|
|
||||||
// only go to next line is it's going to wrap at the space
|
// Skip a leading space at the start of a wrapped line.
|
||||||
if (wrapPos >= firstSpace)
|
if (lineStart < byteCount && bytes[lineStart] == (byte)' ')
|
||||||
ImGui.TextUnformatted("");
|
lineStart++;
|
||||||
}
|
|
||||||
|
var newEnd = CalcWordWrap(basePtr, lineStart, byteCount, widthLeft);
|
||||||
|
if (properBreak && newEnd == endPrev)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (newEnd < 0)
|
||||||
|
{
|
||||||
|
ImGui.TextUnformatted("");
|
||||||
|
ImGui.TextUnformatted("");
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
widthLeft = ImGui.GetContentRegionAvail().X;
|
endPrev = newEnd;
|
||||||
while (endPrevLine < textEnd)
|
DrawText(basePtr, lineStart, endPrev, chunk, handler, defaultText);
|
||||||
|
|
||||||
|
if (!properBreak)
|
||||||
{
|
{
|
||||||
if (properBreak)
|
properBreak = true;
|
||||||
text = endPrevLine;
|
widthLeft = ImGui.GetContentRegionAvail().X;
|
||||||
|
|
||||||
// skip a space at start of line
|
|
||||||
if (*text == ' ')
|
|
||||||
++text;
|
|
||||||
|
|
||||||
var newEnd = ImGuiNative.CalcWordWrapPositionA(ImGui.GetFont().Handle, ImGuiHelpers.GlobalScale, text, textEnd, widthLeft);
|
|
||||||
if (properBreak && newEnd == endPrevLine)
|
|
||||||
break;
|
|
||||||
|
|
||||||
endPrevLine = newEnd;
|
|
||||||
if (endPrevLine == null)
|
|
||||||
{
|
|
||||||
ImGui.TextUnformatted("");
|
|
||||||
ImGui.TextUnformatted("");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
Text(text, endPrevLine);
|
|
||||||
|
|
||||||
if (!properBreak)
|
|
||||||
{
|
|
||||||
properBreak = true;
|
|
||||||
widthLeft = ImGui.GetContentRegionAvail().X;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static unsafe byte* FindFirstSpace(byte* text, byte* textEnd)
|
private static unsafe int CalcWordWrap(byte* basePtr, int start, int end, float width)
|
||||||
{
|
{
|
||||||
for (var i = text; i < textEnd; i++)
|
var result = ImGuiNative.CalcWordWrapPositionA(
|
||||||
if (char.IsWhiteSpace((char) *i))
|
ImGui.GetFont().Handle,
|
||||||
|
ImGuiHelpers.GlobalScale,
|
||||||
|
basePtr + start,
|
||||||
|
basePtr + end,
|
||||||
|
width);
|
||||||
|
if (result == null)
|
||||||
|
return -1;
|
||||||
|
return (int)(result - basePtr);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static unsafe void DrawText(byte* basePtr, int start, int end, Chunk chunk, PayloadHandler? handler, Vector4 defaultText)
|
||||||
|
{
|
||||||
|
var oldPos = ImGui.GetCursorScreenPos();
|
||||||
|
|
||||||
|
ImGuiNative.TextUnformatted(basePtr + start, basePtr + end);
|
||||||
|
PostPayload(chunk, handler);
|
||||||
|
|
||||||
|
if (!ReferenceEquals(LastLink, chunk.Link))
|
||||||
|
PayloadBounds.Clear();
|
||||||
|
|
||||||
|
LastLink = chunk.Link;
|
||||||
|
|
||||||
|
if (Hovered != null && ReferenceEquals(Hovered, chunk.Link))
|
||||||
|
{
|
||||||
|
defaultText.W = 0.25f;
|
||||||
|
var actualCol = ColourUtil.Vector4ToAbgr(defaultText);
|
||||||
|
ImGui.GetWindowDrawList().AddRectFilled(oldPos, oldPos + ImGui.GetItemRectSize(), actualCol);
|
||||||
|
|
||||||
|
foreach (var (boundsStart, boundsSize) in PayloadBounds)
|
||||||
|
ImGui.GetWindowDrawList().AddRectFilled(boundsStart, boundsStart + boundsSize, actualCol);
|
||||||
|
|
||||||
|
PayloadBounds.Clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Hovered == null && chunk.Link != null)
|
||||||
|
PayloadBounds.Add((oldPos, ImGui.GetItemRectSize()));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int FindFirstSpace(ReadOnlySpan<byte> bytes, int start, int end)
|
||||||
|
{
|
||||||
|
for (var i = start; i < end; i++)
|
||||||
|
if (char.IsWhiteSpace((char)bytes[i]))
|
||||||
return i;
|
return i;
|
||||||
|
|
||||||
return textEnd;
|
return end;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static bool IconButton(FontAwesomeIcon icon, string? id = null, string? tooltip = null, int width = 0)
|
internal static bool IconButton(FontAwesomeIcon icon, string? id = null, string? tooltip = null, int width = 0)
|
||||||
|
|||||||
@@ -108,7 +108,10 @@ public static class TabsUtil
|
|||||||
[ChatType.LootNotice] = (ChatSourceExt.All, ChatSourceExt.All),
|
[ChatType.LootNotice] = (ChatSourceExt.All, ChatSourceExt.All),
|
||||||
[ChatType.LootRoll] = (ChatSourceExt.All, ChatSourceExt.All),
|
[ChatType.LootRoll] = (ChatSourceExt.All, ChatSourceExt.All),
|
||||||
},
|
},
|
||||||
Channel = InputChannel.Party,
|
// No automatic input-channel switch; the Gruppe tab is a read
|
||||||
|
// surface that pulls in Party, CrossParty, Alliance and PvpTeam
|
||||||
|
// together. Auto-routing /party into this tab would surprise the
|
||||||
|
// user when they actually wanted /alliance or /pvpteam.
|
||||||
};
|
};
|
||||||
|
|
||||||
public static Tab HellionBeginner => new()
|
public static Tab HellionBeginner => new()
|
||||||
|
|||||||
+20
-20
@@ -27,13 +27,13 @@
|
|||||||
},
|
},
|
||||||
"Microsoft.Data.Sqlite": {
|
"Microsoft.Data.Sqlite": {
|
||||||
"type": "Direct",
|
"type": "Direct",
|
||||||
"requested": "[9.0.0, )",
|
"requested": "[10.0.7, )",
|
||||||
"resolved": "9.0.0",
|
"resolved": "10.0.7",
|
||||||
"contentHash": "lw6wthgXGx3r/U775k1UkUAWIn0kAT0wj4ZRq0WlhPx4WAOiBsIjgDKgWkXcNTGT0KfHiClkM+tyPVFDvxeObw==",
|
"contentHash": "DZ6G2QuyPrsh5VS+wfiZbNBtYT6p+CkxXjD0aZHF04xso7QsG/uk0JpG30hzYlK6u/wtTzta1Dqfgbc/Sl2sDA==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"Microsoft.Data.Sqlite.Core": "9.0.0",
|
"Microsoft.Data.Sqlite.Core": "10.0.7",
|
||||||
"SQLitePCLRaw.bundle_e_sqlite3": "2.1.10",
|
"SQLitePCLRaw.bundle_e_sqlite3": "2.1.11",
|
||||||
"SQLitePCLRaw.core": "2.1.10"
|
"SQLitePCLRaw.core": "2.1.11"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"morelinq": {
|
"morelinq": {
|
||||||
@@ -66,10 +66,10 @@
|
|||||||
},
|
},
|
||||||
"Microsoft.Data.Sqlite.Core": {
|
"Microsoft.Data.Sqlite.Core": {
|
||||||
"type": "Transitive",
|
"type": "Transitive",
|
||||||
"resolved": "9.0.0",
|
"resolved": "10.0.7",
|
||||||
"contentHash": "cFfZjFL+tqzGYw9lB31EkV1IWF5xRQNk2k+MQd+Cf86Gl6zTeAoiZIFw5sRB1Z8OxpEC7nu+nTDsLSjieBAPTw==",
|
"contentHash": "xVrtBg3M1wJlBDkoT0dXEYB/wSc8bIHJPYtw/bu1AqpWgF79uPSs87DAhERR/Ilumre6TKZa1cjMg3VUUObVLA==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"SQLitePCLRaw.core": "2.1.10"
|
"SQLitePCLRaw.core": "2.1.11"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"Microsoft.NET.StringTools": {
|
"Microsoft.NET.StringTools": {
|
||||||
@@ -79,29 +79,29 @@
|
|||||||
},
|
},
|
||||||
"SQLitePCLRaw.bundle_e_sqlite3": {
|
"SQLitePCLRaw.bundle_e_sqlite3": {
|
||||||
"type": "Transitive",
|
"type": "Transitive",
|
||||||
"resolved": "2.1.10",
|
"resolved": "2.1.11",
|
||||||
"contentHash": "UxWuisvZ3uVcVOLJQv7urM/JiQH+v3TmaJc1BLKl5Dxfm/nTzTUrqswCqg/INiYLi61AXnHo1M1JPmPqqLnAdg==",
|
"contentHash": "DC4nA7yWnf4UZdgJDF+9Mus4/cb0Y3Sfgi3gDnAoKNAIBwzkskNAbNbyu+u4atT0ruVlZNJfwZmwiEwE5oz9LQ==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"SQLitePCLRaw.lib.e_sqlite3": "2.1.10",
|
"SQLitePCLRaw.lib.e_sqlite3": "2.1.11",
|
||||||
"SQLitePCLRaw.provider.e_sqlite3": "2.1.10"
|
"SQLitePCLRaw.provider.e_sqlite3": "2.1.11"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"SQLitePCLRaw.core": {
|
"SQLitePCLRaw.core": {
|
||||||
"type": "Transitive",
|
"type": "Transitive",
|
||||||
"resolved": "2.1.10",
|
"resolved": "2.1.11",
|
||||||
"contentHash": "Ii8JCbC7oiVclaE/mbDEK000EFIJ+ShRPwAvvV89GOZhQ+ZLtlnSWl6ksCNMKu/VGXA4Nfi2B7LhN/QFN9oBcw=="
|
"contentHash": "PK0GLFkfhZzLQeR3PJf71FmhtHox+U3vcY6ZtswoMjrefkB9k6ErNJEnwXqc5KgXDSjige2XXrezqS39gkpQKA=="
|
||||||
},
|
},
|
||||||
"SQLitePCLRaw.lib.e_sqlite3": {
|
"SQLitePCLRaw.lib.e_sqlite3": {
|
||||||
"type": "Transitive",
|
"type": "Transitive",
|
||||||
"resolved": "2.1.10",
|
"resolved": "2.1.11",
|
||||||
"contentHash": "mAr69tDbnf3QJpRy2nJz8Qdpebdil00fvycyByR58Cn9eARvR+UiG2Vzsp+4q1tV3ikwiYIjlXCQFc12GfebbA=="
|
"contentHash": "Ev2ytaXiOlWZ4b3R67GZBsemTINslLD1DCJr2xiacpn4tbapu0Q4dHEzSvZSMnVWeE5nlObU3VZN2p81q3XOYQ=="
|
||||||
},
|
},
|
||||||
"SQLitePCLRaw.provider.e_sqlite3": {
|
"SQLitePCLRaw.provider.e_sqlite3": {
|
||||||
"type": "Transitive",
|
"type": "Transitive",
|
||||||
"resolved": "2.1.10",
|
"resolved": "2.1.11",
|
||||||
"contentHash": "uZVTi02C1SxqzgT0HqTWatIbWGb40iIkfc3FpFCpE/r7g6K0PqzDUeefL6P6HPhDtc6BacN3yQysfzP7ks+wSQ==",
|
"contentHash": "Y/0ZkR+r0Cg3DQFuCl1RBnv/tmxpIZRU3HUvelPw6MVaKHwYYR8YNvgs0vuNuXCMvlyJ+Fh88U1D4tah1tt6qw==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"SQLitePCLRaw.core": "2.1.10"
|
"SQLitePCLRaw.core": "2.1.11"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,287 +0,0 @@
|
|||||||
EUROPEAN UNION PUBLIC LICENCE v. 1.2
|
|
||||||
EUPL © the European Union 2007, 2016
|
|
||||||
|
|
||||||
This European Union Public Licence (the ‘EUPL’) applies to the Work (as defined
|
|
||||||
below) which is provided under the terms of this Licence. Any use of the Work,
|
|
||||||
other than as authorised under this Licence is prohibited (to the extent such
|
|
||||||
use is covered by a right of the copyright holder of the Work).
|
|
||||||
|
|
||||||
The Work is provided under the terms of this Licence when the Licensor (as
|
|
||||||
defined below) has placed the following notice immediately following the
|
|
||||||
copyright notice for the Work:
|
|
||||||
|
|
||||||
Licensed under the EUPL
|
|
||||||
|
|
||||||
or has expressed by any other means his willingness to license under the EUPL.
|
|
||||||
|
|
||||||
1. Definitions
|
|
||||||
|
|
||||||
In this Licence, the following terms have the following meaning:
|
|
||||||
|
|
||||||
- ‘The Licence’: this Licence.
|
|
||||||
|
|
||||||
- ‘The Original Work’: the work or software distributed or communicated by the
|
|
||||||
Licensor under this Licence, available as Source Code and also as Executable
|
|
||||||
Code as the case may be.
|
|
||||||
|
|
||||||
- ‘Derivative Works’: the works or software that could be created by the
|
|
||||||
Licensee, based upon the Original Work or modifications thereof. This Licence
|
|
||||||
does not define the extent of modification or dependence on the Original Work
|
|
||||||
required in order to classify a work as a Derivative Work; this extent is
|
|
||||||
determined by copyright law applicable in the country mentioned in Article 15.
|
|
||||||
|
|
||||||
- ‘The Work’: the Original Work or its Derivative Works.
|
|
||||||
|
|
||||||
- ‘The Source Code’: the human-readable form of the Work which is the most
|
|
||||||
convenient for people to study and modify.
|
|
||||||
|
|
||||||
- ‘The Executable Code’: any code which has generally been compiled and which is
|
|
||||||
meant to be interpreted by a computer as a program.
|
|
||||||
|
|
||||||
- ‘The Licensor’: the natural or legal person that distributes or communicates
|
|
||||||
the Work under the Licence.
|
|
||||||
|
|
||||||
- ‘Contributor(s)’: any natural or legal person who modifies the Work under the
|
|
||||||
Licence, or otherwise contributes to the creation of a Derivative Work.
|
|
||||||
|
|
||||||
- ‘The Licensee’ or ‘You’: any natural or legal person who makes any usage of
|
|
||||||
the Work under the terms of the Licence.
|
|
||||||
|
|
||||||
- ‘Distribution’ or ‘Communication’: any act of selling, giving, lending,
|
|
||||||
renting, distributing, communicating, transmitting, or otherwise making
|
|
||||||
available, online or offline, copies of the Work or providing access to its
|
|
||||||
essential functionalities at the disposal of any other natural or legal
|
|
||||||
person.
|
|
||||||
|
|
||||||
2. Scope of the rights granted by the Licence
|
|
||||||
|
|
||||||
The Licensor hereby grants You a worldwide, royalty-free, non-exclusive,
|
|
||||||
sublicensable licence to do the following, for the duration of copyright vested
|
|
||||||
in the Original Work:
|
|
||||||
|
|
||||||
- use the Work in any circumstance and for all usage,
|
|
||||||
- reproduce the Work,
|
|
||||||
- modify the Work, and make Derivative Works based upon the Work,
|
|
||||||
- communicate to the public, including the right to make available or display
|
|
||||||
the Work or copies thereof to the public and perform publicly, as the case may
|
|
||||||
be, the Work,
|
|
||||||
- distribute the Work or copies thereof,
|
|
||||||
- lend and rent the Work or copies thereof,
|
|
||||||
- sublicense rights in the Work or copies thereof.
|
|
||||||
|
|
||||||
Those rights can be exercised on any media, supports and formats, whether now
|
|
||||||
known or later invented, as far as the applicable law permits so.
|
|
||||||
|
|
||||||
In the countries where moral rights apply, the Licensor waives his right to
|
|
||||||
exercise his moral right to the extent allowed by law in order to make effective
|
|
||||||
the licence of the economic rights here above listed.
|
|
||||||
|
|
||||||
The Licensor grants to the Licensee royalty-free, non-exclusive usage rights to
|
|
||||||
any patents held by the Licensor, to the extent necessary to make use of the
|
|
||||||
rights granted on the Work under this Licence.
|
|
||||||
|
|
||||||
3. Communication of the Source Code
|
|
||||||
|
|
||||||
The Licensor may provide the Work either in its Source Code form, or as
|
|
||||||
Executable Code. If the Work is provided as Executable Code, the Licensor
|
|
||||||
provides in addition a machine-readable copy of the Source Code of the Work
|
|
||||||
along with each copy of the Work that the Licensor distributes or indicates, in
|
|
||||||
a notice following the copyright notice attached to the Work, a repository where
|
|
||||||
the Source Code is easily and freely accessible for as long as the Licensor
|
|
||||||
continues to distribute or communicate the Work.
|
|
||||||
|
|
||||||
4. Limitations on copyright
|
|
||||||
|
|
||||||
Nothing in this Licence is intended to deprive the Licensee of the benefits from
|
|
||||||
any exception or limitation to the exclusive rights of the rights owners in the
|
|
||||||
Work, of the exhaustion of those rights or of other applicable limitations
|
|
||||||
thereto.
|
|
||||||
|
|
||||||
5. Obligations of the Licensee
|
|
||||||
|
|
||||||
The grant of the rights mentioned above is subject to some restrictions and
|
|
||||||
obligations imposed on the Licensee. Those obligations are the following:
|
|
||||||
|
|
||||||
Attribution right: The Licensee shall keep intact all copyright, patent or
|
|
||||||
trademarks notices and all notices that refer to the Licence and to the
|
|
||||||
disclaimer of warranties. The Licensee must include a copy of such notices and a
|
|
||||||
copy of the Licence with every copy of the Work he/she distributes or
|
|
||||||
communicates. The Licensee must cause any Derivative Work to carry prominent
|
|
||||||
notices stating that the Work has been modified and the date of modification.
|
|
||||||
|
|
||||||
Copyleft clause: If the Licensee distributes or communicates copies of the
|
|
||||||
Original Works or Derivative Works, this Distribution or Communication will be
|
|
||||||
done under the terms of this Licence or of a later version of this Licence
|
|
||||||
unless the Original Work is expressly distributed only under this version of the
|
|
||||||
Licence — for example by communicating ‘EUPL v. 1.2 only’. The Licensee
|
|
||||||
(becoming Licensor) cannot offer or impose any additional terms or conditions on
|
|
||||||
the Work or Derivative Work that alter or restrict the terms of the Licence.
|
|
||||||
|
|
||||||
Compatibility clause: If the Licensee Distributes or Communicates Derivative
|
|
||||||
Works or copies thereof based upon both the Work and another work licensed under
|
|
||||||
a Compatible Licence, this Distribution or Communication can be done under the
|
|
||||||
terms of this Compatible Licence. For the sake of this clause, ‘Compatible
|
|
||||||
Licence’ refers to the licences listed in the appendix attached to this Licence.
|
|
||||||
Should the Licensee's obligations under the Compatible Licence conflict with
|
|
||||||
his/her obligations under this Licence, the obligations of the Compatible
|
|
||||||
Licence shall prevail.
|
|
||||||
|
|
||||||
Provision of Source Code: When distributing or communicating copies of the Work,
|
|
||||||
the Licensee will provide a machine-readable copy of the Source Code or indicate
|
|
||||||
a repository where this Source will be easily and freely available for as long
|
|
||||||
as the Licensee continues to distribute or communicate the Work.
|
|
||||||
|
|
||||||
Legal Protection: This Licence does not grant permission to use the trade names,
|
|
||||||
trademarks, service marks, or names of the Licensor, except as required for
|
|
||||||
reasonable and customary use in describing the origin of the Work and
|
|
||||||
reproducing the content of the copyright notice.
|
|
||||||
|
|
||||||
6. Chain of Authorship
|
|
||||||
|
|
||||||
The original Licensor warrants that the copyright in the Original Work granted
|
|
||||||
hereunder is owned by him/her or licensed to him/her and that he/she has the
|
|
||||||
power and authority to grant the Licence.
|
|
||||||
|
|
||||||
Each Contributor warrants that the copyright in the modifications he/she brings
|
|
||||||
to the Work are owned by him/her or licensed to him/her and that he/she has the
|
|
||||||
power and authority to grant the Licence.
|
|
||||||
|
|
||||||
Each time You accept the Licence, the original Licensor and subsequent
|
|
||||||
Contributors grant You a licence to their contributions to the Work, under the
|
|
||||||
terms of this Licence.
|
|
||||||
|
|
||||||
7. Disclaimer of Warranty
|
|
||||||
|
|
||||||
The Work is a work in progress, which is continuously improved by numerous
|
|
||||||
Contributors. It is not a finished work and may therefore contain defects or
|
|
||||||
‘bugs’ inherent to this type of development.
|
|
||||||
|
|
||||||
For the above reason, the Work is provided under the Licence on an ‘as is’ basis
|
|
||||||
and without warranties of any kind concerning the Work, including without
|
|
||||||
limitation merchantability, fitness for a particular purpose, absence of defects
|
|
||||||
or errors, accuracy, non-infringement of intellectual property rights other than
|
|
||||||
copyright as stated in Article 6 of this Licence.
|
|
||||||
|
|
||||||
This disclaimer of warranty is an essential part of the Licence and a condition
|
|
||||||
for the grant of any rights to the Work.
|
|
||||||
|
|
||||||
8. Disclaimer of Liability
|
|
||||||
|
|
||||||
Except in the cases of wilful misconduct or damages directly caused to natural
|
|
||||||
persons, the Licensor will in no event be liable for any direct or indirect,
|
|
||||||
material or moral, damages of any kind, arising out of the Licence or of the use
|
|
||||||
of the Work, including without limitation, damages for loss of goodwill, work
|
|
||||||
stoppage, computer failure or malfunction, loss of data or any commercial
|
|
||||||
damage, even if the Licensor has been advised of the possibility of such damage.
|
|
||||||
However, the Licensor will be liable under statutory product liability laws as
|
|
||||||
far such laws apply to the Work.
|
|
||||||
|
|
||||||
9. Additional agreements
|
|
||||||
|
|
||||||
While distributing the Work, You may choose to conclude an additional agreement,
|
|
||||||
defining obligations or services consistent with this Licence. However, if
|
|
||||||
accepting obligations, You may act only on your own behalf and on your sole
|
|
||||||
responsibility, not on behalf of the original Licensor or any other Contributor,
|
|
||||||
and only if You agree to indemnify, defend, and hold each Contributor harmless
|
|
||||||
for any liability incurred by, or claims asserted against such Contributor by
|
|
||||||
the fact You have accepted any warranty or additional liability.
|
|
||||||
|
|
||||||
10. Acceptance of the Licence
|
|
||||||
|
|
||||||
The provisions of this Licence can be accepted by clicking on an icon ‘I agree’
|
|
||||||
placed under the bottom of a window displaying the text of this Licence or by
|
|
||||||
affirming consent in any other similar way, in accordance with the rules of
|
|
||||||
applicable law. Clicking on that icon indicates your clear and irrevocable
|
|
||||||
acceptance of this Licence and all of its terms and conditions.
|
|
||||||
|
|
||||||
Similarly, you irrevocably accept this Licence and all of its terms and
|
|
||||||
conditions by exercising any rights granted to You by Article 2 of this Licence,
|
|
||||||
such as the use of the Work, the creation by You of a Derivative Work or the
|
|
||||||
Distribution or Communication by You of the Work or copies thereof.
|
|
||||||
|
|
||||||
11. Information to the public
|
|
||||||
|
|
||||||
In case of any Distribution or Communication of the Work by means of electronic
|
|
||||||
communication by You (for example, by offering to download the Work from a
|
|
||||||
remote location) the distribution channel or media (for example, a website) must
|
|
||||||
at least provide to the public the information requested by the applicable law
|
|
||||||
regarding the Licensor, the Licence and the way it may be accessible, concluded,
|
|
||||||
stored and reproduced by the Licensee.
|
|
||||||
|
|
||||||
12. Termination of the Licence
|
|
||||||
|
|
||||||
The Licence and the rights granted hereunder will terminate automatically upon
|
|
||||||
any breach by the Licensee of the terms of the Licence.
|
|
||||||
|
|
||||||
Such a termination will not terminate the licences of any person who has
|
|
||||||
received the Work from the Licensee under the Licence, provided such persons
|
|
||||||
remain in full compliance with the Licence.
|
|
||||||
|
|
||||||
13. Miscellaneous
|
|
||||||
|
|
||||||
Without prejudice of Article 9 above, the Licence represents the complete
|
|
||||||
agreement between the Parties as to the Work.
|
|
||||||
|
|
||||||
If any provision of the Licence is invalid or unenforceable under applicable
|
|
||||||
law, this will not affect the validity or enforceability of the Licence as a
|
|
||||||
whole. Such provision will be construed or reformed so as necessary to make it
|
|
||||||
valid and enforceable.
|
|
||||||
|
|
||||||
The European Commission may publish other linguistic versions or new versions of
|
|
||||||
this Licence or updated versions of the Appendix, so far this is required and
|
|
||||||
reasonable, without reducing the scope of the rights granted by the Licence. New
|
|
||||||
versions of the Licence will be published with a unique version number.
|
|
||||||
|
|
||||||
All linguistic versions of this Licence, approved by the European Commission,
|
|
||||||
have identical value. Parties can take advantage of the linguistic version of
|
|
||||||
their choice.
|
|
||||||
|
|
||||||
14. Jurisdiction
|
|
||||||
|
|
||||||
Without prejudice to specific agreement between parties,
|
|
||||||
|
|
||||||
- any litigation resulting from the interpretation of this License, arising
|
|
||||||
between the European Union institutions, bodies, offices or agencies, as a
|
|
||||||
Licensor, and any Licensee, will be subject to the jurisdiction of the Court
|
|
||||||
of Justice of the European Union, as laid down in article 272 of the Treaty on
|
|
||||||
the Functioning of the European Union,
|
|
||||||
|
|
||||||
- any litigation arising between other parties and resulting from the
|
|
||||||
interpretation of this License, will be subject to the exclusive jurisdiction
|
|
||||||
of the competent court where the Licensor resides or conducts its primary
|
|
||||||
business.
|
|
||||||
|
|
||||||
15. Applicable Law
|
|
||||||
|
|
||||||
Without prejudice to specific agreement between parties,
|
|
||||||
|
|
||||||
- this Licence shall be governed by the law of the European Union Member State
|
|
||||||
where the Licensor has his seat, resides or has his registered office,
|
|
||||||
|
|
||||||
- this licence shall be governed by Belgian law if the Licensor has no seat,
|
|
||||||
residence or registered office inside a European Union Member State.
|
|
||||||
|
|
||||||
Appendix
|
|
||||||
|
|
||||||
‘Compatible Licences’ according to Article 5 EUPL are:
|
|
||||||
|
|
||||||
- GNU General Public License (GPL) v. 2, v. 3
|
|
||||||
- GNU Affero General Public License (AGPL) v. 3
|
|
||||||
- Open Software License (OSL) v. 2.1, v. 3.0
|
|
||||||
- Eclipse Public License (EPL) v. 1.0
|
|
||||||
- CeCILL v. 2.0, v. 2.1
|
|
||||||
- Mozilla Public Licence (MPL) v. 2
|
|
||||||
- GNU Lesser General Public Licence (LGPL) v. 2.1, v. 3
|
|
||||||
- Creative Commons Attribution-ShareAlike v. 3.0 Unported (CC BY-SA 3.0) for
|
|
||||||
works other than software
|
|
||||||
- European Union Public Licence (EUPL) v. 1.1, v. 1.2
|
|
||||||
- Québec Free and Open-Source Licence — Reciprocity (LiLiQ-R) or Strong
|
|
||||||
Reciprocity (LiLiQ-R+).
|
|
||||||
|
|
||||||
The European Commission may update this Appendix to later versions of the above
|
|
||||||
licences without producing a new version of the EUPL, as long as they provide
|
|
||||||
the rights granted in Article 2 of this Licence and protect the covered Source
|
|
||||||
Code from exclusive appropriation.
|
|
||||||
|
|
||||||
All other changes or additions to this Appendix require the production of a new
|
|
||||||
EUPL version.
|
|
||||||
@@ -0,0 +1,306 @@
|
|||||||
|
EUROPEAN UNION PUBLIC LICENCE v. 1.2
|
||||||
|
EUPL (c) the European Union 2007, 2016
|
||||||
|
|
||||||
|
This European Union Public Licence (the 'EUPL') applies to the Work (as
|
||||||
|
defined below) which is provided under the terms of this Licence. Any use
|
||||||
|
of the Work, other than as authorised under this Licence is prohibited (to
|
||||||
|
the extent such use is covered by a right of the copyright holder of the
|
||||||
|
Work).
|
||||||
|
|
||||||
|
The Work is provided under the terms of this Licence when the Licensor (as
|
||||||
|
defined below) has placed the following notice immediately following the
|
||||||
|
copyright notice for the Work:
|
||||||
|
|
||||||
|
Licensed under the EUPL
|
||||||
|
|
||||||
|
or has expressed by any other means his willingness to license under the
|
||||||
|
EUPL.
|
||||||
|
|
||||||
|
1. Definitions
|
||||||
|
|
||||||
|
In this Licence, the following terms have the following meaning:
|
||||||
|
|
||||||
|
- 'The Licence': this Licence.
|
||||||
|
|
||||||
|
- 'The Original Work': the work or software distributed or communicated by
|
||||||
|
the Licensor under this Licence, available as Source Code and also as
|
||||||
|
Executable Code as the case may be.
|
||||||
|
|
||||||
|
- 'Derivative Works': the works or software that could be created by the
|
||||||
|
Licensee, based upon the Original Work or modifications thereof. This
|
||||||
|
Licence does not define the extent of modification or dependence on the
|
||||||
|
Original Work required in order to classify a work as a Derivative Work;
|
||||||
|
this extent is determined by copyright law applicable in the country
|
||||||
|
mentioned in Article 15.
|
||||||
|
|
||||||
|
- 'The Work': the Original Work or its Derivative Works.
|
||||||
|
|
||||||
|
- 'The Source Code': the human-readable form of the Work which is the most
|
||||||
|
convenient for people to study and modify.
|
||||||
|
|
||||||
|
- 'The Executable Code': any code which has generally been compiled and
|
||||||
|
which is meant to be interpreted by a computer as a program.
|
||||||
|
|
||||||
|
- 'The Licensor': the natural or legal person that distributes or
|
||||||
|
communicates the Work under the Licence.
|
||||||
|
|
||||||
|
- 'Contributor(s)': any natural or legal person who modifies the Work under
|
||||||
|
the Licence, or otherwise contributes to the creation of a Derivative
|
||||||
|
Work.
|
||||||
|
|
||||||
|
- 'The Licensee' or 'You': any natural or legal person who makes any usage
|
||||||
|
of the Work under the terms of the Licence.
|
||||||
|
|
||||||
|
- 'Distribution' or 'Communication': any act of selling, giving, lending,
|
||||||
|
renting, distributing, communicating, transmitting, or otherwise making
|
||||||
|
available, online or offline, copies of the Work or providing access to
|
||||||
|
its essential functionalities at the disposal of any other natural or
|
||||||
|
legal person.
|
||||||
|
|
||||||
|
2. Scope of the rights granted by the Licence
|
||||||
|
|
||||||
|
The Licensor hereby grants You a worldwide, royalty-free, non-exclusive,
|
||||||
|
sublicensable licence to do the following, for the duration of copyright
|
||||||
|
vested in the Original Work:
|
||||||
|
|
||||||
|
- use the Work in any circumstance and for all usage,
|
||||||
|
- reproduce the Work,
|
||||||
|
- modify the Work, and make Derivative Works based upon the Work,
|
||||||
|
- communicate to the public, including the right to make available or
|
||||||
|
display the Work or copies thereof to the public and perform publicly,
|
||||||
|
as the case may be, the Work,
|
||||||
|
- distribute the Work or copies thereof,
|
||||||
|
- lend and rent the Work or copies thereof,
|
||||||
|
- sublicense rights in the Work or copies thereof.
|
||||||
|
|
||||||
|
Those rights can be exercised on any media, supports and formats, whether
|
||||||
|
now known or later invented, as far as the applicable law permits so.
|
||||||
|
|
||||||
|
In the countries where moral rights apply, the Licensor waives his right
|
||||||
|
to exercise his moral right to the extent allowed by law in order to make
|
||||||
|
effective the licence of the economic rights here above listed.
|
||||||
|
|
||||||
|
The Licensor grants to the Licensee royalty-free, non-exclusive usage
|
||||||
|
rights to any patents held by the Licensor, to the extent necessary to
|
||||||
|
make use of the rights granted on the Work under this Licence.
|
||||||
|
|
||||||
|
3. Communication of the Source Code
|
||||||
|
|
||||||
|
The Licensor may provide the Work either in its Source Code form, or as
|
||||||
|
Executable Code. If the Work is provided as Executable Code, the Licensor
|
||||||
|
provides in addition a machine-readable copy of the Source Code of the
|
||||||
|
Work along with each copy of the Work that the Licensor distributes or
|
||||||
|
indicates, in a notice following the copyright notice attached to the
|
||||||
|
Work, a repository where the Source Code is easily and freely accessible
|
||||||
|
for as long as the Licensor continues to distribute or communicate the
|
||||||
|
Work.
|
||||||
|
|
||||||
|
4. Limitations on copyright
|
||||||
|
|
||||||
|
Nothing in this Licence is intended to deprive the Licensee of the
|
||||||
|
benefits from any exception or limitation to the exclusive rights of the
|
||||||
|
rights owners in the Work, of the exhaustion of those rights or of other
|
||||||
|
applicable limitations thereto.
|
||||||
|
|
||||||
|
5. Obligations of the Licensee
|
||||||
|
|
||||||
|
The grant of the rights mentioned above is subject to some restrictions
|
||||||
|
and obligations imposed on the Licensee. Those obligations are the
|
||||||
|
following:
|
||||||
|
|
||||||
|
Attribution right: The Licensee shall keep intact all copyright, patent or
|
||||||
|
trademarks notices and all notices that refer to the Licence and to the
|
||||||
|
disclaimer of warranties. The Licensee must include a copy of such notices
|
||||||
|
and a copy of the Licence with every copy of the Work he/she distributes
|
||||||
|
or communicates. The Licensee must cause any Derivative Work to carry
|
||||||
|
prominent notices stating that the Work has been modified and the date of
|
||||||
|
modification.
|
||||||
|
|
||||||
|
Copyleft clause: If the Licensee distributes or communicates copies of the
|
||||||
|
Original Works or Derivative Works, this Distribution or Communication
|
||||||
|
will be done under the terms of this Licence or of a later version of
|
||||||
|
this Licence unless the Original Work is expressly distributed only under
|
||||||
|
this version of the Licence. The Licensee (becoming Licensor) cannot
|
||||||
|
offer or impose any additional terms or conditions on the Work or
|
||||||
|
Derivative Work that alter or restrict the terms of the Licence.
|
||||||
|
|
||||||
|
Compatibility clause: If the Licensee Distributes or Communicates
|
||||||
|
Derivative Works or copies thereof based upon both the Work and another
|
||||||
|
work licensed under a Compatible Licence, this Distribution or
|
||||||
|
Communication can be done under the terms of this Compatible Licence. For
|
||||||
|
the sake of this clause, 'Compatible Licence' refers to the licences
|
||||||
|
listed in the appendix attached to this Licence. Should the Licensee's
|
||||||
|
obligations under the Compatible Licence conflict with his/her obligations
|
||||||
|
under this Licence, the obligations of the Compatible Licence shall
|
||||||
|
prevail.
|
||||||
|
|
||||||
|
Provision of Source Code: When distributing or communicating copies of
|
||||||
|
the Work, the Licensee will provide a machine-readable copy of the Source
|
||||||
|
Code or indicate a repository where this Source will be easily and freely
|
||||||
|
available for as long as the Licensee continues to distribute or
|
||||||
|
communicate the Work.
|
||||||
|
|
||||||
|
Legal Protection: This Licence does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or names of the Licensor, except as
|
||||||
|
required for reasonable and customary use in describing the origin of the
|
||||||
|
Work and reproducing the content of the copyright notice.
|
||||||
|
|
||||||
|
6. Chain of Authorship
|
||||||
|
|
||||||
|
The original Licensor warrants that the copyright in the Original Work
|
||||||
|
granted hereunder is owned by him/her or licensed to him/her and that
|
||||||
|
he/she has the power and authority to grant the Licence.
|
||||||
|
|
||||||
|
Each Contributor warrants that the copyright in the modifications he/she
|
||||||
|
brings to the Work are owned by him/her or licensed to him/her and that
|
||||||
|
he/she has the power and authority to grant the Licence.
|
||||||
|
|
||||||
|
Each time You accept the Licence, the original Licensor and subsequent
|
||||||
|
Contributors grant You a licence to their contributions to the Work,
|
||||||
|
under the terms of this Licence.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty
|
||||||
|
|
||||||
|
The Work is a work in progress, which is continuously improved by
|
||||||
|
numerous Contributors. It is not a finished work and may therefore contain
|
||||||
|
defects or 'bugs' inherent to this type of development.
|
||||||
|
|
||||||
|
For the above reason, the Work is provided under the Licence on an 'as
|
||||||
|
is' basis and without warranties of any kind concerning the Work,
|
||||||
|
including without limitation merchantability, fitness for a particular
|
||||||
|
purpose, absence of defects or errors, accuracy, non-infringement of
|
||||||
|
intellectual property rights other than copyright as stated in Article 6
|
||||||
|
of this Licence.
|
||||||
|
|
||||||
|
This disclaimer of warranty is an essential part of the Licence and a
|
||||||
|
condition for the grant of any rights to the Work.
|
||||||
|
|
||||||
|
8. Disclaimer of Liability
|
||||||
|
|
||||||
|
Except in the cases of wilful misconduct or damages directly caused to
|
||||||
|
natural persons, the Licensor will in no event be liable for any direct
|
||||||
|
or indirect, material or moral, damages of any kind, arising out of the
|
||||||
|
Licence or of the use of the Work, including without limitation, damages
|
||||||
|
for loss of goodwill, work stoppage, computer failure or malfunction,
|
||||||
|
loss of data or any commercial damage, even if the Licensor has been
|
||||||
|
advised of the possibility of such damage. However, the Licensor will be
|
||||||
|
liable under statutory product liability laws as far such laws apply to
|
||||||
|
the Work.
|
||||||
|
|
||||||
|
9. Additional agreements
|
||||||
|
|
||||||
|
While distributing the Work, You may choose to conclude an additional
|
||||||
|
agreement, defining obligations or services consistent with this Licence.
|
||||||
|
However, if accepting obligations, You may act only on your own behalf
|
||||||
|
and on your sole responsibility, not on behalf of the original Licensor
|
||||||
|
or any other Contributor, and only if You agree to indemnify, defend, and
|
||||||
|
hold each Contributor harmless for any liability incurred by, or claims
|
||||||
|
asserted against such Contributor by the fact You have accepted any
|
||||||
|
warranty or additional liability.
|
||||||
|
|
||||||
|
10. Acceptance of the Licence
|
||||||
|
|
||||||
|
The provisions of this Licence can be accepted by clicking on an icon 'I
|
||||||
|
agree' placed under the bottom of a window displaying the text of this
|
||||||
|
Licence or by affirming consent in any other similar way, in accordance
|
||||||
|
with the rules of applicable law. Clicking on that icon indicates your
|
||||||
|
clear and irrevocable acceptance of this Licence and all of its terms and
|
||||||
|
conditions.
|
||||||
|
|
||||||
|
Similarly, you irrevocably accept this Licence and all of its terms and
|
||||||
|
conditions by exercising any rights granted to You by Article 2 of this
|
||||||
|
Licence, such as the use of the Work, the creation by You of a Derivative
|
||||||
|
Work or the Distribution or Communication by You of the Work or copies
|
||||||
|
thereof.
|
||||||
|
|
||||||
|
11. Information to the public
|
||||||
|
|
||||||
|
In case of any Distribution or Communication of the Work by means of
|
||||||
|
electronic communication by You (for example, by offering to download the
|
||||||
|
Work from a remote location) the distribution channel or media (for
|
||||||
|
example, a website) must at least provide to the public the information
|
||||||
|
requested by the applicable law regarding the Licensor, the Licence and
|
||||||
|
the way it may be accessible, concluded, stored and reproduced by the
|
||||||
|
Licensee.
|
||||||
|
|
||||||
|
12. Termination of the Licence
|
||||||
|
|
||||||
|
The Licence and the rights granted hereunder will terminate automatically
|
||||||
|
upon any breach by the Licensee of the terms of the Licence.
|
||||||
|
|
||||||
|
Such a termination will not terminate the licences of any person who has
|
||||||
|
received the Work from the Licensee under the Licence, provided such
|
||||||
|
persons remain in full compliance with the Licence.
|
||||||
|
|
||||||
|
13. Miscellaneous
|
||||||
|
|
||||||
|
Without prejudice of Article 9 above, the Licence represents the complete
|
||||||
|
agreement between the Parties as to the Work.
|
||||||
|
|
||||||
|
If any provision of the Licence is invalid or unenforceable under
|
||||||
|
applicable law, this will not affect the validity or enforceability of
|
||||||
|
the Licence as a whole. Such provision will be construed or reformed so
|
||||||
|
as necessary to make it valid and enforceable.
|
||||||
|
|
||||||
|
The European Commission may publish other linguistic versions or new
|
||||||
|
versions of this Licence or updated versions of the Appendix, so far this
|
||||||
|
is required and reasonable, without reducing the scope of the rights
|
||||||
|
granted by the Licence. New versions of the Licence will be published
|
||||||
|
with a unique version number.
|
||||||
|
|
||||||
|
All linguistic versions of this Licence, approved by the European
|
||||||
|
Commission, have identical value. Parties can take advantage of the
|
||||||
|
linguistic version of their choice.
|
||||||
|
|
||||||
|
14. Jurisdiction
|
||||||
|
|
||||||
|
Without prejudice to specific agreement between parties,
|
||||||
|
|
||||||
|
- any litigation resulting from the interpretation of this License,
|
||||||
|
arising between the European Union institutions, bodies, offices or
|
||||||
|
agencies, as a Licensor, and any Licensee, will be subject to the
|
||||||
|
jurisdiction of the Court of Justice of the European Union, as laid
|
||||||
|
down in article 272 of the Treaty on the Functioning of the European
|
||||||
|
Union,
|
||||||
|
|
||||||
|
- any litigation arising between other parties and resulting from the
|
||||||
|
interpretation of this License, will be subject to the exclusive
|
||||||
|
jurisdiction of the competent court where the Licensor resides or
|
||||||
|
conducts its primary business.
|
||||||
|
|
||||||
|
15. Applicable Law
|
||||||
|
|
||||||
|
Without prejudice to specific agreement between parties,
|
||||||
|
|
||||||
|
- this Licence shall be governed by the law of the European Union Member
|
||||||
|
State where the Licensor has his seat, resides or has his registered
|
||||||
|
office,
|
||||||
|
|
||||||
|
- this licence shall be governed by Belgian law if the Licensor has no
|
||||||
|
seat, residence or registered office inside a European Union Member
|
||||||
|
State.
|
||||||
|
|
||||||
|
Appendix
|
||||||
|
|
||||||
|
'Compatible Licences' according to Article 5 EUPL are:
|
||||||
|
|
||||||
|
- GNU General Public License (GPL) v. 2, v. 3
|
||||||
|
- GNU Affero General Public License (AGPL) v. 3
|
||||||
|
- Open Software License (OSL) v. 2.1, v. 3.0
|
||||||
|
- Eclipse Public License (EPL) v. 1.0
|
||||||
|
- CeCILL v. 2.0, v. 2.1
|
||||||
|
- Mozilla Public Licence (MPL) v. 2
|
||||||
|
- GNU Lesser General Public Licence (LGPL) v. 2.1, v. 3
|
||||||
|
- Creative Commons Attribution-ShareAlike v. 3.0 Unported (CC BY-SA 3.0)
|
||||||
|
for works other than software
|
||||||
|
- European Union Public Licence (EUPL) v. 1.1, v. 1.2
|
||||||
|
- Quebec Free and Open-Source Licence — Reciprocity (LiLiQ-R) or Strong
|
||||||
|
Reciprocity (LiLiQ-R+)
|
||||||
|
|
||||||
|
The European Commission may update this Appendix to later versions of the
|
||||||
|
above licences without producing a new version of the EUPL, as long as
|
||||||
|
they provide the rights granted in Article 2 of this Licence and protect
|
||||||
|
the covered Source Code from exclusive appropriation.
|
||||||
|
|
||||||
|
All other changes or additions to this Appendix require the production of
|
||||||
|
a new EUPL version.
|
||||||
@@ -0,0 +1,89 @@
|
|||||||
|
# Notice
|
||||||
|
|
||||||
|
## Acknowledgements
|
||||||
|
|
||||||
|
HellionChat is a fork of [Chat 2](https://github.com/Infiziert90/ChatTwo) by
|
||||||
|
**Infiziert90 (Infi)** and **Anna Clemens**, both of whom kept that plugin
|
||||||
|
running and maintained for years before I ever opened the source. Without
|
||||||
|
their work this fork would not exist, full stop. I owe them the architecture,
|
||||||
|
the message store, the channel filtering, the sidebar tab system, the
|
||||||
|
hooks into FFXIV's chat, the localisation infrastructure, and countless
|
||||||
|
small decisions that I only noticed because they had already been made
|
||||||
|
correctly.
|
||||||
|
|
||||||
|
If you find HellionChat useful, please remember that the foundation came
|
||||||
|
from Chat 2. The code Anna and Infi wrote is doing most of the heavy
|
||||||
|
lifting in this fork too.
|
||||||
|
|
||||||
|
## A direct word to Infi and Anna
|
||||||
|
|
||||||
|
Hi. I am Florian (Flo, also Jon Kazama in-game on Phoenix). I forked Chat 2
|
||||||
|
because I wanted a privacy-by-default version for my own use case and a
|
||||||
|
small group of friends I play with, not because I thought I could do
|
||||||
|
anything better than what you built. The opposite is true. ChatTwo's
|
||||||
|
default of full history and cross-character logging is the right call for
|
||||||
|
most users. I just have a different threat model and a different
|
||||||
|
data-handling philosophy that fits a smaller, locally-stored, retention-
|
||||||
|
limited approach.
|
||||||
|
|
||||||
|
What HellionChat adds is mostly Hellion-specific surface area: a privacy
|
||||||
|
filter, per-channel retention windows, an export pipeline, an Auto-Tell-
|
||||||
|
Tabs feature for FFXIV club greeters, the Hellion theme and font, German
|
||||||
|
localisation, and a settings UX rebuild. None of it touches the bones of
|
||||||
|
what you built. Where I had to modify your code I tried to keep the
|
||||||
|
edits minimal, isolated to clearly-marked Hellion files, and reversible.
|
||||||
|
|
||||||
|
Concrete example: when API 15 hit, I cherry-picked your fix for the
|
||||||
|
BetterTTV emote regression with `git cherry-pick -x` so authorship and
|
||||||
|
co-author trail stay intact. That is the standard I want to keep using as
|
||||||
|
long as both projects are alive. You should never have to look at this
|
||||||
|
fork and wonder if I quietly ate your work.
|
||||||
|
|
||||||
|
If anything in this fork ever steps on something you would not be okay
|
||||||
|
with, please reach out and I will fix it. Genuinely. The list of contacts
|
||||||
|
is below.
|
||||||
|
|
||||||
|
## Maintainer contact
|
||||||
|
|
||||||
|
If something in HellionChat causes problems, especially if it relates back
|
||||||
|
to Chat 2 or to anything Infi or Anna would want flagged:
|
||||||
|
|
||||||
|
- **GitHub Issues:** [JonKazama-Hellion/HellionChat/issues](https://github.com/JonKazama-Hellion/HellionChat/issues)
|
||||||
|
- **Discord:** `@j.j_kazama`
|
||||||
|
- **Email (business):** maintainer@hellion-media.de
|
||||||
|
|
||||||
|
I respond on weekdays during European business hours. For anything
|
||||||
|
urgent (security, attribution, takedown), email is the fastest path.
|
||||||
|
|
||||||
|
## Why this fork is not upstreamed
|
||||||
|
|
||||||
|
The privacy-by-default position fits a small audience. ChatTwo's
|
||||||
|
full-history-by-default position fits a much larger one, including the
|
||||||
|
roleplaying community where chat archive is part of the play experience.
|
||||||
|
Trying to upstream HellionChat's defaults would have meant arguing that
|
||||||
|
Chat 2's defaults are wrong, and they are not. They are right for the
|
||||||
|
user base ChatTwo serves. So I keep the fork separate, attribute clearly,
|
||||||
|
and pull selected upstream patches when they apply.
|
||||||
|
|
||||||
|
## Why HellionChat left the GitHub fork network
|
||||||
|
|
||||||
|
The Dalamud plugin ecosystem treats the GitHub-Fork relation as a signal
|
||||||
|
that a fork is either a development branch or a dead mirror. HellionChat
|
||||||
|
is neither. It is an independently-maintained EUPL-1.2 fork with its own
|
||||||
|
release cadence, its own custom repo, its own user base. Detaching the
|
||||||
|
fork-network relation just makes the situation honest. The git history,
|
||||||
|
the cherry-pick trail, and the attribution stay exactly the same. The
|
||||||
|
only thing that changes is the GitHub UI no longer says "forked from".
|
||||||
|
|
||||||
|
## Trademarks and naming
|
||||||
|
|
||||||
|
"Chat 2" and "ChatTwo" are the names Infi and Anna chose for the upstream
|
||||||
|
plugin. HellionChat does not use either of those names in user-facing
|
||||||
|
copy except where required to describe origin (settings tab, manifest,
|
||||||
|
this file, the README). The Hellion brand is mine.
|
||||||
|
|
||||||
|
## Questions
|
||||||
|
|
||||||
|
This file is the canonical place for "is this attribution correct, is the
|
||||||
|
maintainer reachable, is the relationship to Chat 2 documented". If
|
||||||
|
anything in here is wrong, please open an issue or contact me directly.
|
||||||
@@ -1,10 +1,22 @@
|
|||||||
# Hellion Chat
|
# Hellion Chat
|
||||||
|
|
||||||
**Version 0.3.1** — DSGVO-bewusste Erweiterung von [Chat 2](https://github.com/Infiziert90/ChatTwo) für FINAL FANTASY XIV / Dalamud.
|
[](https://github.com/JonKazama-Hellion/HellionChat/actions/workflows/build.yml)
|
||||||
|
[](https://github.com/JonKazama-Hellion/HellionChat/security/code-scanning)
|
||||||
|
[](LICENSE)
|
||||||
|
[](https://github.com/JonKazama-Hellion/HellionChat/releases/latest)
|
||||||
|
[](https://github.com/goatcorp/Dalamud)
|
||||||
|
[](https://dotnet.microsoft.com/)
|
||||||
|
[](https://www.finalfantasyxiv.com/)
|
||||||
|
|
||||||
|
**Version 0.5.1** — DSGVO-bewusste Erweiterung von [Chat 2](https://github.com/Infiziert90/ChatTwo) für FINAL FANTASY XIV / Dalamud.
|
||||||
|
|
||||||
Hellion Chat baut auf Chat 2 auf und ergänzt es um Datenschutz- und Daten-Handling-Kontrollen, die mit den Datenschutz-Regeln in der EU, den USA und Japan im Einklang sind. Alle Chat-2-Funktionen, Befehle und Tastenkürzel funktionieren unverändert. Eigenständiger Plugin-Slot, eigene Konfiguration, eigene Datenbank.
|
Hellion Chat baut auf Chat 2 auf und ergänzt es um Datenschutz- und Daten-Handling-Kontrollen, die mit den Datenschutz-Regeln in der EU, den USA und Japan im Einklang sind. Alle Chat-2-Funktionen, Befehle und Tastenkürzel funktionieren unverändert. Eigenständiger Plugin-Slot, eigene Konfiguration, eigene Datenbank.
|
||||||
|
|
||||||
Privates Repository, EUPL-1.2-lizenziert. Distribution über Custom-Repo während der Bootstrap-Phase.
|
Eigenständiges Repository, EUPL-1.2-lizenziert. Distribution über Custom-Repo. Selektive Cherry-Picks von Upstream-Chat-2 nach Bedarf, dokumentiert in [UPSTREAM_SYNC.md](UPSTREAM_SYNC.md).
|
||||||
|
|
||||||
|
## Acknowledgements
|
||||||
|
|
||||||
|
Hellion Chat baut auf [Chat 2](https://github.com/Infiziert90/ChatTwo) von **Infiziert90 (Infi)** und **Anna Clemens** auf, die das Plugin über Jahre gepflegt haben bevor ich den Source-Code überhaupt gesehen habe. Die ganze Kern-Architektur, der Message-Store, die Channel-Logik, das Hook-System und vieles mehr stammt von ihnen. Wenn dir Hellion Chat hilft, dann läuft die Anerkennung dafür zu großen Teilen an Infi und Anna. Eine ausführliche Danksagung liegt in [NOTICE.md](NOTICE.md).
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -265,7 +277,7 @@ Phase 3 (offen, kein festes Datum):
|
|||||||
|
|
||||||
## Lizenz
|
## Lizenz
|
||||||
|
|
||||||
EUPL-1.2 (gleiche Lizenz wie Upstream Chat 2). Siehe `LICENCE`.
|
EUPL-1.2 (gleiche Lizenz wie Upstream Chat 2). Volltext in [LICENSE](LICENSE), Copyright-Notes mit Dual-Holder-Block in [COPYRIGHT](COPYRIGHT), persönliche Danksagung an die Upstream-Autoren in [NOTICE.md](NOTICE.md).
|
||||||
|
|
||||||
© 2023–2026 die Chat-2-Autoren (Infi, Anna und die Upstream-Contributors) für die Engine, IPC und Storage-Schicht.
|
© 2023–2026 die Chat-2-Autoren (Infi, Anna und die Upstream-Contributors) für die Engine, IPC und Storage-Schicht.
|
||||||
© 2026 Hellion Online Media für die Hellion-Chat-Erweiterungen.
|
© 2026 Hellion Online Media für die Hellion-Chat-Erweiterungen.
|
||||||
|
|||||||
+49
@@ -0,0 +1,49 @@
|
|||||||
|
# Security policy
|
||||||
|
|
||||||
|
## Reporting a vulnerability
|
||||||
|
|
||||||
|
If you find a security issue in HellionChat, please do not open a public
|
||||||
|
GitHub issue. Use one of the private channels below instead so we can
|
||||||
|
investigate and ship a fix before the details go out.
|
||||||
|
|
||||||
|
**Preferred:**
|
||||||
|
[Privately report a vulnerability](https://github.com/JonKazama-Hellion/HellionChat/security/advisories/new)
|
||||||
|
through GitHub's Security Advisories. This routes the report directly to
|
||||||
|
me and keeps the conversation off the public timeline.
|
||||||
|
|
||||||
|
**Alternative:**
|
||||||
|
- Email: `maintainer@hellion-media.de`
|
||||||
|
- Discord: `@j.j_kazama`
|
||||||
|
|
||||||
|
I respond on weekdays during European business hours. For urgent
|
||||||
|
disclosures (active exploitation, user-data exposure) email is the
|
||||||
|
fastest path.
|
||||||
|
|
||||||
|
## What I treat as in scope
|
||||||
|
|
||||||
|
- Code paths in HellionChat that touch user-controlled input (chat
|
||||||
|
messages, plugin config, file paths the user can influence)
|
||||||
|
- The privacy filter in MessageStore.cs and the export pipeline
|
||||||
|
- The Configuration migration logic
|
||||||
|
- The EmoteCache HTTP client and path handling
|
||||||
|
- The Auto-Tell-Tabs spawn logic and history preload
|
||||||
|
|
||||||
|
## What is not in scope
|
||||||
|
|
||||||
|
- Issues in upstream Chat 2 that we have not modified — please report
|
||||||
|
those at <https://github.com/Infiziert90/ChatTwo/issues>
|
||||||
|
- Issues in Dalamud itself — those go to <https://github.com/goatcorp/Dalamud>
|
||||||
|
- Issues in the FFXIV game client
|
||||||
|
- Anything that needs the user to install a malicious plugin first
|
||||||
|
|
||||||
|
## Acknowledgement
|
||||||
|
|
||||||
|
I list everyone who reports a real issue in the changelog of the release
|
||||||
|
that fixes it, unless they prefer to stay anonymous. No bug bounty,
|
||||||
|
nothing financial; this is a hobby plugin.
|
||||||
|
|
||||||
|
## Disclosure window
|
||||||
|
|
||||||
|
I aim to ship a fix within 14 days for high-severity issues and within
|
||||||
|
30 days for everything else. If a fix needs more time I will say so in
|
||||||
|
the private thread.
|
||||||
@@ -0,0 +1,110 @@
|
|||||||
|
# Upstream sync workflow
|
||||||
|
|
||||||
|
HellionChat is a standalone EUPL-1.2 fork of [Chat 2](https://github.com/Infiziert90/ChatTwo).
|
||||||
|
We pull selected patches from upstream manually instead of running an
|
||||||
|
automated mirror. This file documents how that works so anyone (including
|
||||||
|
future-me) can do it cleanly.
|
||||||
|
|
||||||
|
## One-time setup
|
||||||
|
|
||||||
|
Add the upstream repo as a remote on a fresh clone:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git remote add upstream https://github.com/Infiziert90/ChatTwo.git
|
||||||
|
git fetch upstream
|
||||||
|
```
|
||||||
|
|
||||||
|
Verify both remotes are wired up:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git remote -v
|
||||||
|
# origin https://github.com/JonKazama-Hellion/HellionChat.git (fetch)
|
||||||
|
# origin https://github.com/JonKazama-Hellion/HellionChat.git (push)
|
||||||
|
# upstream https://github.com/Infiziert90/ChatTwo.git (fetch)
|
||||||
|
# upstream https://github.com/Infiziert90/ChatTwo.git (push)
|
||||||
|
```
|
||||||
|
|
||||||
|
You never push to `upstream`. It is read-only for us.
|
||||||
|
|
||||||
|
## Reviewing what is new upstream
|
||||||
|
|
||||||
|
Before any feature cycle starts I run a quick check:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git fetch upstream
|
||||||
|
git log --oneline main..upstream/main | head -30
|
||||||
|
```
|
||||||
|
|
||||||
|
That shows every commit Infi or contributors landed since the last sync.
|
||||||
|
Read the messages, decide which ones apply.
|
||||||
|
|
||||||
|
## What we cherry-pick
|
||||||
|
|
||||||
|
**Always:** security fixes, API-version compatibility patches (Dalamud
|
||||||
|
API 15 → 16 → ...), BetterTTV / emote-cache fixes, regression fixes for
|
||||||
|
the upstream behaviour we still rely on.
|
||||||
|
|
||||||
|
**Sometimes:** small bug fixes in `MessageManager.cs`, `MessageStore.cs`,
|
||||||
|
`ChatLogWindow.cs`, the Tabs system. Pull them when they touch code we
|
||||||
|
have not heavily modified.
|
||||||
|
|
||||||
|
**Never:** webinterface changes (the entire webinterface tree is gone in
|
||||||
|
HellionChat), changes that conflict with the privacy filter, changes that
|
||||||
|
re-add upstream defaults we deliberately reversed (full-history logging,
|
||||||
|
Tell Exclusive defaults, etc.).
|
||||||
|
|
||||||
|
## How we cherry-pick
|
||||||
|
|
||||||
|
Always with `-x` so authorship and the original commit hash stay
|
||||||
|
visible:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git checkout -b sync/upstream-<topic> main
|
||||||
|
git cherry-pick -x <upstream-commit-sha>
|
||||||
|
```
|
||||||
|
|
||||||
|
`-x` adds a `(cherry picked from commit <sha>)` line to the commit
|
||||||
|
message. That preserves the upstream-author credit and lets anyone
|
||||||
|
reading `git log` trace the change back to ChatTwo. Co-Author trail
|
||||||
|
intact, no AI lines, no "Hellion" prefix on commits that were not
|
||||||
|
authored by us.
|
||||||
|
|
||||||
|
## Conflict handling
|
||||||
|
|
||||||
|
When a cherry-pick conflicts:
|
||||||
|
|
||||||
|
1. Resolve the conflict by hand. Do not "fix" upstream code to match
|
||||||
|
Hellion conventions; that is what the merge marker showed us.
|
||||||
|
2. If the conflict is fundamental (touches code that no longer exists
|
||||||
|
in our fork), abort the cherry-pick and document why in
|
||||||
|
`Hellion Chat Backlog.md` instead. Some upstream patches are not
|
||||||
|
portable; that is fine.
|
||||||
|
3. After a successful resolve, the commit message stays identical to
|
||||||
|
the upstream message, with the `-x` cherry-pick footer Git appends
|
||||||
|
automatically. Do not rewrite the message to match our format.
|
||||||
|
|
||||||
|
## Pushing the sync
|
||||||
|
|
||||||
|
Cherry-picked commits go through the same review as our own work: the
|
||||||
|
sync branch lands in `main` via a no-fast-forward merge, then a release
|
||||||
|
tag if the user-visible behaviour changes (otherwise just merged).
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git checkout main
|
||||||
|
git merge --no-ff sync/upstream-<topic> -m "merge: upstream sync — <topic>"
|
||||||
|
```
|
||||||
|
|
||||||
|
## When upstream goes silent
|
||||||
|
|
||||||
|
If Chat 2 stops receiving updates entirely we keep this workflow alive
|
||||||
|
anyway. The remote stays configured, the documentation stays here. The
|
||||||
|
moment maintenance picks back up we are ready to pull again.
|
||||||
|
|
||||||
|
## When upstream takes a direction we cannot follow
|
||||||
|
|
||||||
|
If a future ChatTwo release breaks compatibility with our privacy
|
||||||
|
philosophy in a way we cannot resolve (e.g. mandatory cloud sync,
|
||||||
|
removal of the local message store, a license change that makes EUPL
|
||||||
|
incompatible), HellionChat continues on its own from the last
|
||||||
|
compatible cherry-pick. The history we already inherited stays under
|
||||||
|
EUPL-1.2 and stays attributed.
|
||||||
Reference in New Issue
Block a user